From 52a7ce08cf478ee16b79e7f8c5b953b5da6f1388 Mon Sep 17 00:00:00 2001 From: Mirek Kratochvil Date: Tue, 25 Dec 2012 18:17:00 +0100 Subject: [PATCH] fmtseq: primitive key generation --- src/fmtseq.cpp | 96 ++++++++++++++++++++++++++++++++++++++++++++++++++ src/fmtseq.h | 15 ++++++-- 2 files changed, 108 insertions(+), 3 deletions(-) diff --git a/src/fmtseq.cpp b/src/fmtseq.cpp index 925dec3..7285618 100644 --- a/src/fmtseq.cpp +++ b/src/fmtseq.cpp @@ -17,5 +17,101 @@ */ #include "fmtseq.h" +#include "arcfour.h" +using namespace fmtseq; + +void prepare_keygen (arcfour& kg, const std::vector&SK, uint idx) +{ + kg.clear(); + kg.init (8); + kg.load_key (SK); + std::vectortmp; + while (idx) { + tmp.push_back (idx & 0xff); + idx >>= 8; + } + tmp.resize (16, 0); //prevent chaining to other numbers + kg.load_key (tmp); +} + +//don't feed zero +static uint log2 (uint x) +{ + uint r = 0; + while (x) { + ++r; + x >>= 1; + } + return r - 1; +} + +int fmtseq::generate (pubkey&pub, privkey&priv, + prng&rng, hash_func&hf, + uint h, uint l) +{ + + uint i, j; + + //first off, generate a secret key for commitment generator. + priv.SK.resize (1 << 8); + for (i = 0; i < (1 << 8); ++i) { + priv.SK[i] = rng.random (1 << 8); + } + + priv.h = h; + priv.l = l; + + std::vector stk; + stk.reserve (h * l + 1); + + uint sigs = 1 << (h * l); + + //number of commitments needed for signature (bits+log2(bits)) + uint commitments = 8 * hf.size(); + commitments += log2 (commitments); + + arcfour generator; + std::vector x, y, Y; + + x.resize (hf.size() ); + y.resize (hf.size() ); + + for (i = 0; i < sigs; ++i) { + //generate commitments and concat publics into Y + Y.clear(); + Y.reserve (commitments * hf.size() ); + prepare_keygen (generator, priv.SK, i); + for (j = 0; j < commitments; ++j) { + generator.gen (hf.size(), x); + y = hf (x); + Y.insert (Y.end(), y.begin(), y.end() ); + } + + stk.push_back (privkey::tree_stk_item (0, hf (Y) ) ); + + for (;;) { + if (stk.size() < 2) break; + if ( (stk.end() - 1)->level != (stk.end() - 2)->level) break; + + Y.clear(); + Y.insert (Y.end(), + (stk.end() - 2)->item.begin(), + (stk.end() - 2)->item.end() ); + Y.insert (Y.end(), + (stk.end() - 1)->item.begin(), + (stk.end() - 1)->item.end() ); + uint l = stk.back().level + 1; + stk.pop_back(); + stk.pop_back(); + stk.push_back (privkey::tree_stk_item (l, hf (Y) ) ); + } + } + + //now there's the public verification key available in the stack. + pub.check = stk.back().item; + pub.H = h * l; + + return 0; +} diff --git a/src/fmtseq.h b/src/fmtseq.h index 2c03201..6e9579b 100644 --- a/src/fmtseq.h +++ b/src/fmtseq.h @@ -20,7 +20,7 @@ #define _fmtseq_h_ #include -#include +#include #include "types.h" #include "bvector.h" #include "sencode.h" @@ -45,12 +45,21 @@ public: //FMT caches std::vector > exist; std::vector > desired; - std::vector > desired_stack; + + struct tree_stk_item { + uint level; + std::vector item; + tree_stk_item() {} + tree_stk_item (uint L, std::vector i) + : level (L), item (i) {} + }; + + std::vector > desired_stack; int sign (const bvector&, bvector&, hash_func&); uint sigs_remaining() { - return (1 << () ) - sigs_used; + return (1 << (h * l) ) - sigs_used; } uint hash_size (hash_func&hf) {