mce_qd complete
(doesn't work though)
This commit is contained in:
parent
aa5b1ac2cc
commit
cd1a321041
|
@ -392,9 +392,10 @@ public:
|
||||||
std::vector<uint> support; //computed goppa support
|
std::vector<uint> support; //computed goppa support
|
||||||
polynomial g; //computed goppa polynomial
|
polynomial g; //computed goppa polynomial
|
||||||
std::vector<polynomial> sqInv;
|
std::vector<polynomial> sqInv;
|
||||||
|
//blocks of signature lines of pre-permuted check matrix
|
||||||
std::vector<bvector> Hc; //signature lines of pre-permuted check matrix
|
std::vector<std::vector<bvector> > Hc;
|
||||||
std::vector<uint> support_pos; //pre-permuted positions of support rows
|
//pre-permuted positions of support rows
|
||||||
|
std::vector<uint> support_pos;
|
||||||
|
|
||||||
int decrypt (const bvector&, bvector&);
|
int decrypt (const bvector&, bvector&);
|
||||||
int prepare();
|
int prepare();
|
||||||
|
|
|
@ -333,16 +333,17 @@ int mce_qd::generate (pubkey&pub, privkey&priv, prng&rng,
|
||||||
|
|
||||||
int privkey::prepare()
|
int privkey::prepare()
|
||||||
{
|
{
|
||||||
|
uint s, i, j;
|
||||||
std::cout << "prepare" << std::endl;
|
std::cout << "prepare" << std::endl;
|
||||||
//compute H signature from essence
|
//compute H signature from essence
|
||||||
Hsig.resize (fld.n / 2);
|
Hsig.resize (fld.n / 2);
|
||||||
Hsig[0] = fld.inv (essence[fld.m - 1]);
|
Hsig[0] = fld.inv (essence[fld.m - 1]);
|
||||||
for (uint s = 0; s < fld.m - 1; ++s) {
|
for (s = 0; s < fld.m - 1; ++s) {
|
||||||
uint i = 1 << s; //i = 2^s
|
i = 1 << s; //i = 2^s
|
||||||
|
|
||||||
Hsig[i] = fld.inv (fld.add (essence[s], essence[fld.m - 1]) );
|
Hsig[i] = fld.inv (fld.add (essence[s], essence[fld.m - 1]) );
|
||||||
|
|
||||||
for (uint j = 1; j < i; ++j)
|
for (j = 1; j < i; ++j)
|
||||||
Hsig[i + j] = fld.inv
|
Hsig[i + j] = fld.inv
|
||||||
(fld.add
|
(fld.add
|
||||||
(fld.inv (Hsig[i]),
|
(fld.inv (Hsig[i]),
|
||||||
|
@ -354,17 +355,76 @@ int privkey::prepare()
|
||||||
|
|
||||||
//compute the support
|
//compute the support
|
||||||
support.resize (fld.n / 2);
|
support.resize (fld.n / 2);
|
||||||
for (uint i = 0; i < fld.n / 2; ++i) {
|
std::set<uint> used_support;
|
||||||
|
for (i = 0; i < fld.n / 2; ++i) {
|
||||||
support[i] = fld.add
|
support[i] = fld.add
|
||||||
(fld.inv (Hsig[i]),
|
(fld.inv (Hsig[i]),
|
||||||
essence[fld.m - 1]);
|
essence[fld.m - 1]);
|
||||||
|
|
||||||
|
//support consistency check
|
||||||
|
if (used_support.count (support[i]) )
|
||||||
|
return 1;
|
||||||
|
used_support.insert (support[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO prepare permuted Hsig (that can be applied to the ciphertext)
|
//prepare permuted Hsig (so that it can be applied directly)
|
||||||
|
//and prepare reverse support position lookup data
|
||||||
|
uint block_size = 1 << T;
|
||||||
|
uint pos, blk_perm;
|
||||||
|
polynomial tmp_blk, tmp_pblk;
|
||||||
|
bvector cotrace, tmp_block;
|
||||||
|
std::vector<uint> sbl1, sbl2, permuted_support;
|
||||||
|
|
||||||
//TODO prepare function that converts a support zero to ciphertext
|
tmp_blk.resize (block_size);
|
||||||
//position
|
tmp_pblk.resize (block_size);
|
||||||
|
|
||||||
|
Hc.resize (fld.m);
|
||||||
|
for (i = 0; i < fld.m; ++i) {
|
||||||
|
Hc[i].resize (block_count);
|
||||||
|
}
|
||||||
|
|
||||||
|
permuted_support.resize (block_size * block_count);
|
||||||
|
sbl1.resize (block_size);
|
||||||
|
sbl2.resize (block_size);
|
||||||
|
|
||||||
|
//go through all the blocks of original H and convert them if they
|
||||||
|
//aren't discarded.
|
||||||
|
for (i = 0; i < (fld.n / 2) / block_size; ++i) {
|
||||||
|
pos = block_perm[i];
|
||||||
|
if (pos >= block_count) continue; //was discarded
|
||||||
|
blk_perm = block_perms[pos];
|
||||||
|
pos = hperm[pos];
|
||||||
|
|
||||||
|
//permute i-th block of H to pos-th block of Hc,
|
||||||
|
//also move the support.
|
||||||
|
for (j = 0; j < block_size; ++j) {
|
||||||
|
tmp_blk[j] = Hsig[j + i * block_size];
|
||||||
|
sbl1[j] = support[j + i * block_size];
|
||||||
|
}
|
||||||
|
permutation::permute_dyadic (blk_perm, tmp_blk, tmp_pblk);
|
||||||
|
permutation::permute_dyadic (blk_perm, sbl1, sbl2);
|
||||||
|
|
||||||
|
//store support to permuted support
|
||||||
|
for (j = 0; j < block_size; ++j)
|
||||||
|
permuted_support[j + pos * block_size] = sbl2[j];
|
||||||
|
|
||||||
|
//cotrace permuted H block to Hc
|
||||||
|
cotrace.from_poly_cotrace (tmp_pblk, fld);
|
||||||
|
|
||||||
|
for (j = 0; j < fld.m; ++j)
|
||||||
|
cotrace.get_block (j * block_size, block_size, Hc[j][pos]);
|
||||||
|
}
|
||||||
|
|
||||||
|
//convert the support result to actual lookup
|
||||||
|
support_pos.clear();
|
||||||
|
//fld.n in support lookup means that it isn't there (we don't have -1)
|
||||||
|
support_pos.resize (fld.n, fld.n);
|
||||||
|
for (i = 0; i < block_size * block_count; ++i)
|
||||||
|
support_pos[permuted_support[i]] = i;
|
||||||
|
|
||||||
|
for (i = 0; i < support_pos.size(); ++i) {
|
||||||
|
std::cout << "support " << i << " has position " << support_pos[i] << std::endl;
|
||||||
|
}
|
||||||
|
|
||||||
//goppa polynomial
|
//goppa polynomial
|
||||||
g.clear();
|
g.clear();
|
||||||
|
@ -372,7 +432,7 @@ int privkey::prepare()
|
||||||
polynomial tmp;
|
polynomial tmp;
|
||||||
tmp.resize (2, 1);
|
tmp.resize (2, 1);
|
||||||
uint t = 1 << T;
|
uint t = 1 << T;
|
||||||
for (uint i = 0; i < t; ++i) {
|
for (i = 0; i < t; ++i) {
|
||||||
tmp[0] = fld.inv (Hsig[i]);
|
tmp[0] = fld.inv (Hsig[i]);
|
||||||
g.mult (tmp, fld);
|
g.mult (tmp, fld);
|
||||||
}
|
}
|
||||||
|
@ -451,15 +511,13 @@ int privkey::decrypt (const bvector & in, bvector & out)
|
||||||
uint i, j, k;
|
uint i, j, k;
|
||||||
|
|
||||||
synd_vec.resize (block_size * fld.m);
|
synd_vec.resize (block_size * fld.m);
|
||||||
hp.resize (block_size);
|
|
||||||
cp.resize (block_size);
|
cp.resize (block_size);
|
||||||
res.resize (block_size);
|
res.resize (block_size);
|
||||||
|
|
||||||
for (i = 0; i < block_count; ++i) {
|
for (i = 0; i < block_count; ++i) {
|
||||||
in.get_block (i * block_size, block_size, cp);
|
in.get_block (i * block_size, block_size, cp);
|
||||||
for (j = 0; j < fld.m; ++j) {
|
for (j = 0; j < fld.m; ++j) {
|
||||||
Hc[j].get_block (i * block_size, block_size, hp);
|
fwht_dyadic_multiply (Hc[j][i], cp, res);
|
||||||
fwht_dyadic_multiply (hp, cp, res);
|
|
||||||
synd_vec.add_offset (res, j * block_size);
|
synd_vec.add_offset (res, j * block_size);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -475,10 +533,12 @@ int privkey::decrypt (const bvector & in, bvector & out)
|
||||||
//TODO evaluator should return error positions, not bvector. fix it everywhere!
|
//TODO evaluator should return error positions, not bvector. fix it everywhere!
|
||||||
|
|
||||||
out = in;
|
out = in;
|
||||||
|
out.resize (plain_size() );
|
||||||
//flip error positions of out.
|
//flip error positions of out.
|
||||||
for (i = 0; i < ev.size(); ++i) if (ev[i]) {
|
for (i = 0; i < ev.size(); ++i) if (ev[i]) {
|
||||||
if (support_pos[i] == -1) return 1; //couldn't decode TODO is it true?
|
if (support_pos[i] == fld.n) return 1; //couldn't decode TODO is it true?
|
||||||
out[i] = !out[i];
|
if (i < plain_size() )
|
||||||
|
out[i] = !out[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
Loading…
Reference in a new issue