1*638691a0SAndroid Build Coastguard Worker /* Common code for intializing a Reed-Solomon control block (char or int symbols) 2*638691a0SAndroid Build Coastguard Worker * Copyright 2004 Phil Karn, KA9Q 3*638691a0SAndroid Build Coastguard Worker * May be used under the terms of the GNU Lesser General Public License (LGPL) 4*638691a0SAndroid Build Coastguard Worker */ 5*638691a0SAndroid Build Coastguard Worker #undef NULL 6*638691a0SAndroid Build Coastguard Worker #define NULL ((void *)0) 7*638691a0SAndroid Build Coastguard Worker 8*638691a0SAndroid Build Coastguard Worker { 9*638691a0SAndroid Build Coastguard Worker int i, j, sr,root,iprim; 10*638691a0SAndroid Build Coastguard Worker 11*638691a0SAndroid Build Coastguard Worker rs = NULL; 12*638691a0SAndroid Build Coastguard Worker /* Check parameter ranges */ 13*638691a0SAndroid Build Coastguard Worker if(symsize < 0 || symsize > 8*(int)sizeof(data_t)){ 14*638691a0SAndroid Build Coastguard Worker goto done; 15*638691a0SAndroid Build Coastguard Worker } 16*638691a0SAndroid Build Coastguard Worker 17*638691a0SAndroid Build Coastguard Worker if(fcr < 0 || fcr >= (1<<symsize)) 18*638691a0SAndroid Build Coastguard Worker goto done; 19*638691a0SAndroid Build Coastguard Worker if(prim <= 0 || prim >= (1<<symsize)) 20*638691a0SAndroid Build Coastguard Worker goto done; 21*638691a0SAndroid Build Coastguard Worker if(nroots < 0 || nroots >= (1<<symsize)) 22*638691a0SAndroid Build Coastguard Worker goto done; /* Can't have more roots than symbol values! */ 23*638691a0SAndroid Build Coastguard Worker if(pad < 0 || pad >= ((1<<symsize) -1 - nroots)) 24*638691a0SAndroid Build Coastguard Worker goto done; /* Too much padding */ 25*638691a0SAndroid Build Coastguard Worker 26*638691a0SAndroid Build Coastguard Worker rs = (struct rs *)calloc(1,sizeof(struct rs)); 27*638691a0SAndroid Build Coastguard Worker if(rs == NULL) 28*638691a0SAndroid Build Coastguard Worker goto done; 29*638691a0SAndroid Build Coastguard Worker 30*638691a0SAndroid Build Coastguard Worker rs->mm = symsize; 31*638691a0SAndroid Build Coastguard Worker rs->nn = (1<<symsize)-1; 32*638691a0SAndroid Build Coastguard Worker rs->pad = pad; 33*638691a0SAndroid Build Coastguard Worker 34*638691a0SAndroid Build Coastguard Worker rs->alpha_to = (data_t *)malloc(sizeof(data_t)*(rs->nn+1)); 35*638691a0SAndroid Build Coastguard Worker if(rs->alpha_to == NULL){ 36*638691a0SAndroid Build Coastguard Worker free(rs); 37*638691a0SAndroid Build Coastguard Worker rs = NULL; 38*638691a0SAndroid Build Coastguard Worker goto done; 39*638691a0SAndroid Build Coastguard Worker } 40*638691a0SAndroid Build Coastguard Worker rs->index_of = (data_t *)malloc(sizeof(data_t)*(rs->nn+1)); 41*638691a0SAndroid Build Coastguard Worker if(rs->index_of == NULL){ 42*638691a0SAndroid Build Coastguard Worker free(rs->alpha_to); 43*638691a0SAndroid Build Coastguard Worker free(rs); 44*638691a0SAndroid Build Coastguard Worker rs = NULL; 45*638691a0SAndroid Build Coastguard Worker goto done; 46*638691a0SAndroid Build Coastguard Worker } 47*638691a0SAndroid Build Coastguard Worker 48*638691a0SAndroid Build Coastguard Worker /* Generate Galois field lookup tables */ 49*638691a0SAndroid Build Coastguard Worker rs->index_of[0] = A0; /* log(zero) = -inf */ 50*638691a0SAndroid Build Coastguard Worker rs->alpha_to[A0] = 0; /* alpha**-inf = 0 */ 51*638691a0SAndroid Build Coastguard Worker sr = 1; 52*638691a0SAndroid Build Coastguard Worker for(i=0;i<rs->nn;i++){ 53*638691a0SAndroid Build Coastguard Worker rs->index_of[sr] = i; 54*638691a0SAndroid Build Coastguard Worker rs->alpha_to[i] = sr; 55*638691a0SAndroid Build Coastguard Worker sr <<= 1; 56*638691a0SAndroid Build Coastguard Worker if(sr & (1<<symsize)) 57*638691a0SAndroid Build Coastguard Worker sr ^= gfpoly; 58*638691a0SAndroid Build Coastguard Worker sr &= rs->nn; 59*638691a0SAndroid Build Coastguard Worker } 60*638691a0SAndroid Build Coastguard Worker if(sr != 1){ 61*638691a0SAndroid Build Coastguard Worker /* field generator polynomial is not primitive! */ 62*638691a0SAndroid Build Coastguard Worker free(rs->alpha_to); 63*638691a0SAndroid Build Coastguard Worker free(rs->index_of); 64*638691a0SAndroid Build Coastguard Worker free(rs); 65*638691a0SAndroid Build Coastguard Worker rs = NULL; 66*638691a0SAndroid Build Coastguard Worker goto done; 67*638691a0SAndroid Build Coastguard Worker } 68*638691a0SAndroid Build Coastguard Worker 69*638691a0SAndroid Build Coastguard Worker /* Form RS code generator polynomial from its roots */ 70*638691a0SAndroid Build Coastguard Worker rs->genpoly = (data_t *)malloc(sizeof(data_t)*(nroots+1)); 71*638691a0SAndroid Build Coastguard Worker if(rs->genpoly == NULL){ 72*638691a0SAndroid Build Coastguard Worker free(rs->alpha_to); 73*638691a0SAndroid Build Coastguard Worker free(rs->index_of); 74*638691a0SAndroid Build Coastguard Worker free(rs); 75*638691a0SAndroid Build Coastguard Worker rs = NULL; 76*638691a0SAndroid Build Coastguard Worker goto done; 77*638691a0SAndroid Build Coastguard Worker } 78*638691a0SAndroid Build Coastguard Worker rs->fcr = fcr; 79*638691a0SAndroid Build Coastguard Worker rs->prim = prim; 80*638691a0SAndroid Build Coastguard Worker rs->nroots = nroots; 81*638691a0SAndroid Build Coastguard Worker 82*638691a0SAndroid Build Coastguard Worker /* Find prim-th root of 1, used in decoding */ 83*638691a0SAndroid Build Coastguard Worker for(iprim=1;(iprim % prim) != 0;iprim += rs->nn) 84*638691a0SAndroid Build Coastguard Worker ; 85*638691a0SAndroid Build Coastguard Worker rs->iprim = iprim / prim; 86*638691a0SAndroid Build Coastguard Worker 87*638691a0SAndroid Build Coastguard Worker rs->genpoly[0] = 1; 88*638691a0SAndroid Build Coastguard Worker for (i = 0,root=fcr*prim; i < nroots; i++,root += prim) { 89*638691a0SAndroid Build Coastguard Worker rs->genpoly[i+1] = 1; 90*638691a0SAndroid Build Coastguard Worker 91*638691a0SAndroid Build Coastguard Worker /* Multiply rs->genpoly[] by @**(root + x) */ 92*638691a0SAndroid Build Coastguard Worker for (j = i; j > 0; j--){ 93*638691a0SAndroid Build Coastguard Worker if (rs->genpoly[j] != 0) 94*638691a0SAndroid Build Coastguard Worker rs->genpoly[j] = rs->genpoly[j-1] ^ rs->alpha_to[modnn(rs,rs->index_of[rs->genpoly[j]] + root)]; 95*638691a0SAndroid Build Coastguard Worker else 96*638691a0SAndroid Build Coastguard Worker rs->genpoly[j] = rs->genpoly[j-1]; 97*638691a0SAndroid Build Coastguard Worker } 98*638691a0SAndroid Build Coastguard Worker /* rs->genpoly[0] can never be zero */ 99*638691a0SAndroid Build Coastguard Worker rs->genpoly[0] = rs->alpha_to[modnn(rs,rs->index_of[rs->genpoly[0]] + root)]; 100*638691a0SAndroid Build Coastguard Worker } 101*638691a0SAndroid Build Coastguard Worker /* convert rs->genpoly[] to index form for quicker encoding */ 102*638691a0SAndroid Build Coastguard Worker for (i = 0; i <= nroots; i++) 103*638691a0SAndroid Build Coastguard Worker rs->genpoly[i] = rs->index_of[rs->genpoly[i]]; 104*638691a0SAndroid Build Coastguard Worker done:; 105*638691a0SAndroid Build Coastguard Worker 106*638691a0SAndroid Build Coastguard Worker } 107