|
|
@ -28,63 +28,129 @@
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "lte/utils/pack.h"
|
|
|
|
|
|
|
|
|
|
|
|
unsigned int cword;
|
|
|
|
#include "lte/fec/crc.h"
|
|
|
|
|
|
|
|
|
|
|
|
unsigned int icrc1(unsigned int crc, unsigned short onech,int long_crc,
|
|
|
|
|
|
|
|
int left_shift,unsigned int poly)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
unsigned int tmp=(unsigned int) (crc ^ (onech << (long_crc >> 1) ));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (i=0;i<left_shift;i++) {
|
|
|
|
void gen_crc_table(crc_t *crc_params) {
|
|
|
|
if (tmp & (0x1<<(long_crc-1)))
|
|
|
|
|
|
|
|
tmp=(tmp<<1)^poly;
|
|
|
|
int i, j, ord=(crc_params->order-8);
|
|
|
|
else
|
|
|
|
unsigned long bit, crc;
|
|
|
|
tmp <<= 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
for (i=0; i<256; i++) {
|
|
|
|
|
|
|
|
crc = ((unsigned long)i)<<ord;
|
|
|
|
|
|
|
|
for (j=0; j<8; j++) {
|
|
|
|
|
|
|
|
bit = crc & crc_params->crchighbit;
|
|
|
|
|
|
|
|
crc<<= 1;
|
|
|
|
|
|
|
|
if (bit) crc^= crc_params->polynom;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
crc_params->table[i]=crc & crc_params->crcmask;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned int crc(unsigned int crc, char *bufptr, int len,
|
|
|
|
|
|
|
|
int long_crc,unsigned int poly, int paste_word) {
|
|
|
|
unsigned long crctable (unsigned long length, crc_t *crc_params) {
|
|
|
|
|
|
|
|
|
|
|
|
int i,k;
|
|
|
|
// Polynom order 8, 16, 24 or 32 only.
|
|
|
|
unsigned int data;
|
|
|
|
int ord=crc_params->order-8;
|
|
|
|
int stop;
|
|
|
|
unsigned long crc = crc_params->crcinit;
|
|
|
|
unsigned int ret;
|
|
|
|
unsigned char* data = crc_params->data0;
|
|
|
|
|
|
|
|
|
|
|
|
cword=crc;
|
|
|
|
while (length--){
|
|
|
|
|
|
|
|
crc = (crc << 8) ^ crc_params->table[ ((crc >> (ord)) & 0xff) ^ *data++];
|
|
|
|
k=0;
|
|
|
|
|
|
|
|
stop=0;
|
|
|
|
|
|
|
|
while(!stop) {
|
|
|
|
|
|
|
|
data=0;
|
|
|
|
|
|
|
|
for (i=0;i<long_crc/2;i++) {
|
|
|
|
|
|
|
|
if (bufptr[k] && k<len)
|
|
|
|
|
|
|
|
data|=(0x1<<(long_crc/2-1-i));
|
|
|
|
|
|
|
|
k++;
|
|
|
|
|
|
|
|
if (k==len) {
|
|
|
|
|
|
|
|
stop=1;
|
|
|
|
|
|
|
|
i++;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return((crc ^ crc_params->crcxor) & crc_params->crcmask);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unsigned long reversecrcbit(unsigned int crc, int nbits, crc_t *crc_params) {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unsigned long m, rmask=0x1;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for(m=0; m<nbits; m++){
|
|
|
|
|
|
|
|
if((rmask & crc) == 0x01 )crc = (crc ^ crc_params->polynom)>>1;
|
|
|
|
|
|
|
|
else crc = crc >> 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return((crc ^ crc_params->crcxor) & crc_params->crcmask);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int crc_init(crc_t *crc_par){
|
|
|
|
|
|
|
|
|
|
|
|
cword=(unsigned int) (icrc1((unsigned int) (cword<<long_crc>>long_crc),
|
|
|
|
// Compute bit masks for whole CRC and CRC high bit
|
|
|
|
data,long_crc,i,poly)<<long_crc)>>long_crc;
|
|
|
|
crc_par->crcmask = ((((unsigned long)1<<(crc_par->order-1))-1)<<1)|1;
|
|
|
|
|
|
|
|
crc_par->crchighbit = (unsigned long)1<<(crc_par->order-1);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
printf("crcmask=%x, crchightbit=%x\n",
|
|
|
|
|
|
|
|
(unsigned int)crc_par->crcmask, (unsigned int)crc_par->crchighbit);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// check parameters
|
|
|
|
|
|
|
|
if (crc_par->order%8 != 0) {
|
|
|
|
|
|
|
|
printf("ERROR, invalid order=%d, it must be 8, 16, 24 or 32.\n", crc_par->order);
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ret=cword;
|
|
|
|
if (crc_par->crcinit != (crc_par->crcinit & crc_par->crcmask)) {
|
|
|
|
if (paste_word) {
|
|
|
|
printf("ERROR, invalid crcinit.\n");
|
|
|
|
cword<<=32-long_crc;
|
|
|
|
return(0);
|
|
|
|
for (i=0;i<long_crc;i++) {
|
|
|
|
|
|
|
|
bufptr[i+len]=((cword&(0x1<<31))>>31);
|
|
|
|
|
|
|
|
cword<<=1;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (crc_par->crcxor != (crc_par->crcxor & crc_par->crcmask)) {
|
|
|
|
|
|
|
|
printf("ERROR, invalid crcxor.\n");
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (ret);
|
|
|
|
// generate lookup table
|
|
|
|
|
|
|
|
gen_crc_table(crc_par);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unsigned int crc_attach(char *bufptr, int len, crc_t *crc_params) {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int i, len8, res8, a;
|
|
|
|
|
|
|
|
unsigned int crc;
|
|
|
|
|
|
|
|
char *pter;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef _WITHMALLOC
|
|
|
|
|
|
|
|
crc_params->data0 = (unsigned char *)malloc(sizeof(*crc_params->data0) * (len+crc_params->order)*2);
|
|
|
|
|
|
|
|
if (!crc_params->data0) {
|
|
|
|
|
|
|
|
perror("malloc ERROR: Allocating memory for data pointer in crc() function");
|
|
|
|
|
|
|
|
return(-1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
if((((len+crc_params->order)>>3) + 1) > MAX_LENGTH){
|
|
|
|
|
|
|
|
printf("ERROR: Not enough memory allocated\n");
|
|
|
|
|
|
|
|
return(-1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//# Pack bits into bytes
|
|
|
|
|
|
|
|
len8=(len>>3);
|
|
|
|
|
|
|
|
res8=8-(len - (len8<<3));
|
|
|
|
|
|
|
|
if(res8>0)a=1;
|
|
|
|
|
|
|
|
else a=0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Zeroed additional bits
|
|
|
|
|
|
|
|
memset((char *)(bufptr+len),0,(32)*sizeof(char));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for(i=0; i<len8+a; i++){
|
|
|
|
|
|
|
|
pter=(char *)(bufptr+8*i);
|
|
|
|
|
|
|
|
crc_params->data0[i]=(unsigned char)(unpack_bits(&pter, 8)&0xFF);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate CRC
|
|
|
|
|
|
|
|
crc=crctable(len8+a, crc_params);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Reverse CRC res8 positions
|
|
|
|
|
|
|
|
if(a==1)crc=reversecrcbit(crc, res8, crc_params);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Add CRC
|
|
|
|
|
|
|
|
pter=(char *)(bufptr+len);
|
|
|
|
|
|
|
|
pack_bits(crc, &pter, crc_params->order);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef _WITHMALLOC
|
|
|
|
|
|
|
|
free(crc_params->data0);
|
|
|
|
|
|
|
|
crc_params->data0=NULL;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//Return CRC value
|
|
|
|
|
|
|
|
return crc;
|
|
|
|
|
|
|
|
}
|
|
|
|