xref: /aosp_15_r20/external/fec/init_rs.h (revision 638691a093b4f9473cd6ee8f3e0139deef159a86)
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