decoding with berlekamp trace
This commit is contained in:
parent
5bc1106063
commit
d24550c126
|
@ -126,6 +126,7 @@ public:
|
||||||
uint add (uint, uint);
|
uint add (uint, uint);
|
||||||
uint mult (uint, uint);
|
uint mult (uint, uint);
|
||||||
uint exp (uint, int);
|
uint exp (uint, int);
|
||||||
|
uint exp (int);
|
||||||
uint inv (uint);
|
uint inv (uint);
|
||||||
uint sq_root (uint);
|
uint sq_root (uint);
|
||||||
};
|
};
|
||||||
|
|
|
@ -65,3 +65,76 @@ bool evaluate_error_locator_dumb (polynomial&a, bvector&ev, gf2m&fld)
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* berlekamp trace algorithm - we puncture roots of incoming polynomial into
|
||||||
|
* the vector of size fld.n
|
||||||
|
*
|
||||||
|
* Inspired by implementation from HyMES.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <set>
|
||||||
|
|
||||||
|
bool evaluate_error_locator_trace (polynomial&sigma, bvector&ev, gf2m&fld)
|
||||||
|
{
|
||||||
|
ev.clear();
|
||||||
|
ev.resize (fld.n, 0);
|
||||||
|
|
||||||
|
std::vector<polynomial> trace_aux, trace; //trace cache
|
||||||
|
trace_aux.resize (fld.m);
|
||||||
|
trace.resize (fld.m);
|
||||||
|
|
||||||
|
trace_aux[0] = polynomial();
|
||||||
|
trace_aux[0].resize (2, 0);
|
||||||
|
trace_aux[0][1] = 1; //trace_aux[0] = x
|
||||||
|
trace[0] = trace_aux[0]; //trace[0] = x
|
||||||
|
|
||||||
|
for (uint i = 1; i < fld.m; ++i) {
|
||||||
|
trace_aux[i] = trace_aux[i-1];
|
||||||
|
trace_aux[i].square (fld);
|
||||||
|
trace_aux[i].mod (sigma, fld);
|
||||||
|
trace[0].add (trace_aux[i], fld);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::set<std::pair<uint, polynomial> > stk; //"stack"
|
||||||
|
|
||||||
|
stk.insert (make_pair (0, sigma) );
|
||||||
|
|
||||||
|
while (!stk.empty() ) {
|
||||||
|
|
||||||
|
uint i = stk.begin()->first;
|
||||||
|
polynomial cur = stk.begin()->second;
|
||||||
|
|
||||||
|
stk.erase (stk.begin() );
|
||||||
|
|
||||||
|
int deg = cur.degree();
|
||||||
|
|
||||||
|
if (deg <= 0) continue;
|
||||||
|
if (deg == 1) { //found a linear factor
|
||||||
|
ev[fld.mult (cur[0], fld.inv (cur[1]) ) ] = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i >= fld.m) return false;
|
||||||
|
|
||||||
|
if (trace[i].zero() ) {
|
||||||
|
//compute the trace if it isn't cached
|
||||||
|
uint a = fld.exp (i);
|
||||||
|
for (uint j = 0; j < fld.m; ++j) {
|
||||||
|
trace[i].add_mult (trace_aux[j], a, fld);
|
||||||
|
a = fld.mult (a, a);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
polynomial t;
|
||||||
|
t = cur.gcd (trace[i], fld);
|
||||||
|
polynomial q, r;
|
||||||
|
cur.divmod (t, q, r, fld);
|
||||||
|
|
||||||
|
stk.insert (make_pair (i + 1, t) );
|
||||||
|
stk.insert (make_pair (i + 1, q) );
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -139,6 +139,12 @@ uint gf2m::exp (uint a, int k)
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint gf2m::exp (int k)
|
||||||
|
{
|
||||||
|
//return x^k
|
||||||
|
return exp (1 << 1, k);
|
||||||
|
}
|
||||||
|
|
||||||
uint gf2m::inv (uint a)
|
uint gf2m::inv (uint a)
|
||||||
{
|
{
|
||||||
if (!a) return 0;
|
if (!a) return 0;
|
||||||
|
|
|
@ -83,7 +83,7 @@ int privkey::decrypt (const bvector&in, bvector&out)
|
||||||
compute_error_locator (syndrome, fld, g, sqInv, loc);
|
compute_error_locator (syndrome, fld, g, sqInv, loc);
|
||||||
|
|
||||||
bvector ev;
|
bvector ev;
|
||||||
if (!evaluate_error_locator_dumb (loc, ev, fld) )
|
if (!evaluate_error_locator_trace (loc, ev, fld) )
|
||||||
return 1; //if decoding somehow failed, fail as well.
|
return 1; //if decoding somehow failed, fail as well.
|
||||||
|
|
||||||
// check the error vector, it should have exactly t == deg (g) errors
|
// check the error vector, it should have exactly t == deg (g) errors
|
||||||
|
@ -151,7 +151,7 @@ int privkey::sign (const bvector&in, bvector&out, uint delta, uint attempts, prn
|
||||||
|
|
||||||
compute_error_locator (synd, fld, g, sqInv, loc);
|
compute_error_locator (synd, fld, g, sqInv, loc);
|
||||||
|
|
||||||
if (evaluate_error_locator_dumb (loc, e2, fld) ) {
|
if (evaluate_error_locator_trace (loc, e2, fld) ) {
|
||||||
|
|
||||||
//create the decodable message
|
//create the decodable message
|
||||||
p.add (e);
|
p.add (e);
|
||||||
|
|
|
@ -62,7 +62,7 @@ int privkey::decrypt (const bvector&in, bvector&out)
|
||||||
compute_error_locator (unsc, fld, g, sqInv, loc);
|
compute_error_locator (unsc, fld, g, sqInv, loc);
|
||||||
|
|
||||||
bvector ev;
|
bvector ev;
|
||||||
if (!evaluate_error_locator_dumb (loc, ev, fld) )
|
if (!evaluate_error_locator_trace (loc, ev, fld) )
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
if ( (int) ev.hamming_weight() != g.degree() )
|
if ( (int) ev.hamming_weight() != g.degree() )
|
||||||
|
@ -95,7 +95,7 @@ int privkey::sign (const bvector&in, bvector&out, uint delta, uint attempts, prn
|
||||||
|
|
||||||
compute_error_locator (synd_unsc, fld, g, sqInv, loc);
|
compute_error_locator (synd_unsc, fld, g, sqInv, loc);
|
||||||
|
|
||||||
if (evaluate_error_locator_dumb (loc, e, fld) ) {
|
if (evaluate_error_locator_trace (loc, e, fld) ) {
|
||||||
|
|
||||||
Pinv.permute (e, out);
|
Pinv.permute (e, out);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -112,6 +112,7 @@ bool polynomial::is_irreducible (gf2m&fld) const
|
||||||
xmodf.mod (*this, fld); //mod f
|
xmodf.mod (*this, fld); //mod f
|
||||||
|
|
||||||
uint d = degree();
|
uint d = degree();
|
||||||
|
if (d < 0) return false;
|
||||||
for (uint i = 1; i <= (d / 2); ++i) {
|
for (uint i = 1; i <= (d / 2); ++i) {
|
||||||
for (uint j = 0; j < fld.m; ++j) {
|
for (uint j = 0; j < fld.m; ++j) {
|
||||||
t = xi;
|
t = xi;
|
||||||
|
|
Loading…
Reference in a new issue