From ec81c53463affc51b9e239bda22f441cd6b6975e Mon Sep 17 00:00:00 2001 From: Mirek Kratochvil Date: Sat, 29 Sep 2012 19:02:05 +0200 Subject: [PATCH] mce_qd: code generator working --- include/codecrypt.h | 16 ++++++----- lib/mce_qd.cpp | 65 ++++++++++++++++++++++++++++++++++++++------- 2 files changed, 66 insertions(+), 15 deletions(-) diff --git a/include/codecrypt.h b/include/codecrypt.h index 15fa4f7..a8462e9 100644 --- a/include/codecrypt.h +++ b/include/codecrypt.h @@ -116,6 +116,9 @@ public: void generate_random_with_inversion (uint, matrix&, prng&); bool create_goppa_generator (matrix&, permutation&, prng&); bool create_goppa_generator (matrix&, const permutation&); + + bool create_goppa_generator_dyadic (matrix&, uint&, prng&); + bool create_goppa_generator_dyadic (matrix&, uint); }; /* @@ -131,7 +134,7 @@ public: void generate_random (uint n, prng&); - template void permute (const V&a, V&r) const { + template void permute (const A&a, R&r) const { r.resize (a.size() ); for (uint i = 0; i < size(); ++i) r[item (i) ] = a[i]; } @@ -139,8 +142,8 @@ public: void permute_rows (const matrix&, matrix&) const; //work-alike for dyadic permutations. - template static bool permute_dyadic - (uint sig, const V&a, V&r) { + template static bool permute_dyadic + (uint sig, const A&a, R&r) { //check if the thing has size 2^n uint s = a.size(); @@ -365,7 +368,8 @@ int generate (pubkey&, privkey&, prng&, uint m, uint t); * * Good security, extremely good speed with extremely reduced key size. * Recommended for encryption, but needs some plaintext conversion -- either - * Fujisaki-Okamoto or Kobara-Imai are known to work good. + * Fujisaki-Okamoto or Kobara-Imai are known to work good. Without the + * conversion, the encryption itself is extremely weak. */ namespace mce_qd { @@ -375,9 +379,9 @@ public: std::vector essence; gf2m fld; //we fix q=2^fld.m=fld.n, n=q/2 uint T; //the QD's t parameter is 2^T. - uint hperm; //dyadic permutation of H to G permutation block_perm; //order of blocks uint block_count; //blocks >= block_count are shortened-out + permutation hperm; //block permutation of H block used to get G std::vector block_perms; //dyadic permutations of blocks //derivable stuff @@ -400,7 +404,7 @@ class pubkey { public: uint T; - matrix M; + std::vector qd_sigs; int encrypt (const bvector&, bvector&, prng&); diff --git a/lib/mce_qd.cpp b/lib/mce_qd.cpp index 9662ac4..de1203f 100644 --- a/lib/mce_qd.cpp +++ b/lib/mce_qd.cpp @@ -128,19 +128,63 @@ int mce_qd::generate (pubkey&pub, privkey&priv, prng&rng, Hsig.begin() + (i + 1) * block_size); //permute them - permutation bp; - bp.generate_random (h_block_count, rng); - bp.permute (bl, blp); + priv.block_perm.generate_random (h_block_count, rng); + priv.block_perm.permute (bl, blp); //discard blocks blp.resize (block_count); - //TODO permute individual blocks + //permute individual blocks + priv.block_count = block_count; + priv.block_perms.resize (block_count); + bl.resize (blp.size() ); + for (uint i = 0; i < block_count; ++i) { + priv.block_perms[i] = rng.random (block_size); + permutation::permute_dyadic (priv.block_perms[i], + blp[i], bl[i]); + } - //TODO co-trace to binary H^ - //TODO systematic H - //TODO systematic G - //TODO signature of G + //co-trace blocks to binary H^, retry creating G using hperm. + matrix Hc; + polynomial col; + Hc.resize (block_count * block_size); + + matrix r, ri, l; + + for (;;) { + priv.hperm.generate_random (block_count, rng); + + for (uint i = 0; i < block_count; ++i) + for (uint j = 0; j < block_size; ++j) { + permutation::permute_dyadic + (j, bl[priv.hperm[i]], col); + Hc[i*block_size + j].from_poly_cotrace + (col, fld); + } + + /* + * try computing the redundancy block of G + * + * Since H*G^T = [L | R] * [I | X] = L + R*X = 0 + * we have the solution: X = R^1 * L + */ + + Hc.get_right_square (r); + if (!r.compute_inversion (ri) ) + continue; //retry with other hperm + Hc.strip_right_square (l); + ri.mult (l); + } + + /* + * Redundancy-checking part of G is now (transposed) in ri. + * Get QD signatures by getting every t'th row (transposed). + */ + + pub.T = T; + pub.qd_sigs.resize (ri.width() / t); + for (uint i = 0; i < ri.width(); i += t) + pub.qd_sigs[i/t] = ri[i]; return 0; } @@ -148,17 +192,20 @@ int mce_qd::generate (pubkey&pub, privkey&priv, prng&rng, int privkey::prepare() { - + //TODO compute H signature from essence + //TODO compute goppa code support return 0; } int pubkey::encrypt (const bvector& in, bvector&out, prng&rng) { + //TODO FWHT return 0; } int privkey::decrypt (const bvector&in, bvector&out) { + //TODO decoding return 0; }