811 lines
29 KiB
811 lines
29 KiB
|
|
/*
|
|
* This file is Copyright Daniel Silverstone <dsilvers@digital-scurf.org> 2006,2015
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person
|
|
* obtaining a copy of this software and associated documentation
|
|
* files (the "Software"), to deal in the Software without
|
|
* restriction, including without limitation the rights to use, copy,
|
|
* modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
* of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
#include "iguana777.h"
|
|
#include "secp256k1/include/secp256k1.h"
|
|
#include "secp256k1/include/secp256k1_schnorr.h"
|
|
#include "secp256k1/include/secp256k1_rangeproof.h"
|
|
|
|
// ------------------------------------------------------[ Preparation ]----
|
|
|
|
static gfshare_ctx *_gfshare_ctx_init_core(uint8_t *sharenrs,uint32_t sharecount,uint8_t threshold,uint32_t size,void *space,int32_t spacesize)
|
|
{
|
|
gfshare_ctx *ctx; int32_t allocsize;
|
|
allocsize = (int32_t)(sizeof(struct _gfshare_ctx) + threshold * size);
|
|
if ( allocsize > spacesize )
|
|
{
|
|
printf("malloc allocsize %d vs spacesize.%d\n",allocsize,spacesize);
|
|
ctx = malloc(allocsize);
|
|
if( ctx == NULL )
|
|
return NULL; // errno should still be set from XMALLOC()
|
|
ctx->allocsize = allocsize;
|
|
} else ctx = space;
|
|
memset(ctx,0,allocsize);
|
|
ctx->sharecount = sharecount;
|
|
ctx->threshold = threshold;
|
|
ctx->size = size;
|
|
memcpy(ctx->sharenrs,sharenrs,sharecount);
|
|
ctx->buffersize = threshold * size;
|
|
return(ctx);
|
|
}
|
|
|
|
// Initialise a gfshare context for producing shares
|
|
gfshare_ctx *gfshare_ctx_initenc(uint8_t *sharenrs,uint32_t sharecount,uint8_t threshold,uint32_t size,void *space,int32_t spacesize)
|
|
{
|
|
uint32_t i;
|
|
for (i=0; i<sharecount; i++)
|
|
{
|
|
if ( sharenrs[i] == 0 )
|
|
{
|
|
// can't have x[i] = 0 - that would just be a copy of the secret, in
|
|
// theory (in fact, due to the way we use exp/log for multiplication and
|
|
// treat log(0) as 0, it ends up as a copy of x[i] = 1)
|
|
errno = EINVAL;
|
|
return NULL;
|
|
}
|
|
}
|
|
return(_gfshare_ctx_init_core(sharenrs,sharecount,threshold,size,space,spacesize));
|
|
}
|
|
|
|
// Initialise a gfshare context for recombining shares
|
|
gfshare_ctx *gfshare_ctx_initdec(uint8_t *sharenrs,uint32_t sharecount,uint32_t size,void *space,int32_t spacesize)
|
|
{
|
|
gfshare_ctx *ctx = _gfshare_ctx_init_core(sharenrs,sharecount,sharecount,size,space,spacesize);
|
|
if ( ctx != NULL )
|
|
ctx->threshold = 0;
|
|
return(ctx);
|
|
}
|
|
|
|
// Free a share context's memory
|
|
void gfshare_ctx_free(gfshare_ctx *ctx)
|
|
{
|
|
OS_randombytes(ctx->buffer,ctx->buffersize);
|
|
OS_randombytes(ctx->sharenrs,ctx->sharecount);
|
|
if ( ctx->allocsize != 0 )
|
|
{
|
|
OS_randombytes((uint8_t *)ctx,sizeof(struct _gfshare_ctx));
|
|
free(ctx);
|
|
}
|
|
OS_randombytes((uint8_t *)ctx,sizeof(struct _gfshare_ctx));
|
|
}
|
|
|
|
// --------------------------------------------------------[ Splitting ]----
|
|
|
|
// Provide a secret to the encoder. (this re-scrambles the coefficients)
|
|
void gfshare_ctx_enc_setsecret(gfshare_ctx *ctx,uint8_t *secret)
|
|
{
|
|
memcpy(ctx->buffer + ((ctx->threshold-1) * ctx->size),secret,ctx->size);
|
|
OS_randombytes(ctx->buffer,(ctx->threshold-1) * ctx->size);
|
|
}
|
|
|
|
// Extract a share from the context. 'share' must be preallocated and at least 'size' bytes long. 'sharenr' is the index into the 'sharenrs' array of the share you want.
|
|
void gfshare_ctx_encgetshare(uint8_t *_logs,uint8_t *_exps,gfshare_ctx *ctx,uint8_t sharenr,uint8_t *share)
|
|
{
|
|
uint32_t pos,coefficient,ilog = _logs[ctx->sharenrs[sharenr]];
|
|
uint8_t *share_ptr,*coefficient_ptr = ctx->buffer;
|
|
for (pos=0; pos<ctx->size; pos++)
|
|
share[pos] = *(coefficient_ptr++);
|
|
for (coefficient=1; coefficient<ctx->threshold; coefficient++)
|
|
{
|
|
share_ptr = share;
|
|
for (pos=0; pos<ctx->size; pos++)
|
|
{
|
|
uint8_t share_byte = *share_ptr;
|
|
if ( share_byte != 0 )
|
|
share_byte = _exps[ilog + _logs[share_byte]];
|
|
*share_ptr++ = share_byte ^ *coefficient_ptr++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------[ Recombination ]----
|
|
|
|
// Inform a recombination context of a change in share indexes
|
|
void gfshare_ctx_dec_newshares(gfshare_ctx *ctx,uint8_t *sharenrs)
|
|
{
|
|
memcpy(ctx->sharenrs,sharenrs,ctx->sharecount);
|
|
}
|
|
|
|
// Provide a share context with one of the shares. The 'sharenr' is the index into the 'sharenrs' array
|
|
void gfshare_ctx_dec_giveshare(gfshare_ctx *ctx,uint8_t sharenr,uint8_t *share)
|
|
{
|
|
memcpy(ctx->buffer + (sharenr * ctx->size),share,ctx->size);
|
|
}
|
|
|
|
// Extract the secret by interpolating the shares. secretbuf must be allocated and at least 'size' bytes
|
|
void gfshare_ctx_decextract(uint8_t *_logs,uint8_t *_exps,gfshare_ctx *ctx,uint8_t *secretbuf)
|
|
{
|
|
uint32_t i,j; uint8_t *secret_ptr,*share_ptr,sharei,sharej;
|
|
for (i=0; i<ctx->size; i++)
|
|
secretbuf[i] = 0;
|
|
for (i=0; i<ctx->sharecount; i++)
|
|
{
|
|
// Compute L(i) as per Lagrange Interpolation
|
|
unsigned Li_top = 0, Li_bottom = 0;
|
|
if ( (sharei= ctx->sharenrs[i]) != 0 )
|
|
{
|
|
for (j=0; j<ctx->sharecount; j++)
|
|
{
|
|
if ( i != j && sharei != (sharej= ctx->sharenrs[j]) )
|
|
{
|
|
if ( sharej == 0 )
|
|
continue; // skip empty share
|
|
Li_top += _logs[sharej];
|
|
if ( Li_top >= 0xff )
|
|
Li_top -= 0xff;
|
|
Li_bottom += _logs[sharei ^ sharej];
|
|
if ( Li_bottom >= 0xff )
|
|
Li_bottom -= 0xff;
|
|
}
|
|
}
|
|
if ( Li_bottom > Li_top )
|
|
Li_top += 0xff;
|
|
Li_top -= Li_bottom; // Li_top is now log(L(i))
|
|
secret_ptr = secretbuf, share_ptr = ctx->buffer + (ctx->size * i);
|
|
for (j=0; j<ctx->size; j++)
|
|
{
|
|
if ( *share_ptr != 0 )
|
|
*secret_ptr ^= _exps[Li_top + _logs[*share_ptr]];
|
|
share_ptr++, secret_ptr++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
int32_t gfshare_test(struct supernet_info *myinfo,int32_t M,int32_t N,int32_t datasize)
|
|
{
|
|
int ok = 1, i,k;
|
|
uint8_t * secret = malloc(datasize);
|
|
uint8_t *shares[255];
|
|
uint8_t *recomb = malloc(datasize);
|
|
uint8_t space[8192],sharenrs[255],newsharenrs[255];// = (uint8_t *)strdup("0124z89abehtr");
|
|
gfshare_ctx *G;
|
|
for (i=0; i<N; i++)
|
|
{
|
|
sharenrs[i] = i+1;
|
|
shares[i] = malloc(datasize);
|
|
}
|
|
init_sharenrs(sharenrs,0,N,N);
|
|
/* Stage 1, make a secret */
|
|
for( i = 0; i < datasize; ++i )
|
|
secret[i] = (rand() & 0xff00) >> 8;
|
|
/* Stage 2, split it N ways with a threshold of M */
|
|
G = gfshare_ctx_initenc( sharenrs, N, M, datasize,space,sizeof(space) );
|
|
gfshare_ctx_enc_setsecret( G, secret );
|
|
for (i=0; i<N; i++)
|
|
gfshare_ctx_encgetshare(myinfo->logs,myinfo->exps, G, i, shares[i] );
|
|
gfshare_ctx_free( G );
|
|
/* Prep the decode shape */
|
|
uint8_t save[255];
|
|
memcpy(save,sharenrs,sizeof(sharenrs));
|
|
G = gfshare_ctx_initdec( sharenrs, N, datasize,space,sizeof(space) );
|
|
for (k=0; k<10; k++)
|
|
{
|
|
memcpy(sharenrs,save,sizeof(sharenrs));
|
|
memset(newsharenrs,0,N);
|
|
int32_t j,r,m;
|
|
m = M + (rand() % (N-M+1));
|
|
for (i=0; i<m && i<N; i++)
|
|
{
|
|
r = rand() % N;
|
|
while ( (j= sharenrs[r]) == 0 || newsharenrs[r] != 0 )
|
|
r = rand() % N;
|
|
newsharenrs[r] = j;
|
|
sharenrs[r] = 0;
|
|
}
|
|
for (i=0; i<N; i++)
|
|
{
|
|
if ( newsharenrs[i] != 0 )
|
|
{
|
|
fprintf(stderr,"%d ",newsharenrs[i]);
|
|
gfshare_ctx_dec_giveshare( G, i, shares[i] );
|
|
}
|
|
//newsharenrs[i] = sharenrs[i];
|
|
}
|
|
gfshare_ctx_dec_newshares( G, newsharenrs );
|
|
gfshare_ctx_decextract(myinfo->logs,myinfo->exps, G, recomb );
|
|
for (i=0; i<datasize; i++)
|
|
if ( secret[i] != recomb[i] )
|
|
ok = 0;
|
|
printf("m.%d M.%-3d N.%-3d ok.%d datalen.%d\n",m,M,N,ok,datasize);
|
|
}
|
|
free(recomb), free(secret);
|
|
for (i=0; i<N; i++)
|
|
free(shares[i]);
|
|
return ok!=1;
|
|
}
|
|
|
|
void *bitcoin_ctx()
|
|
{
|
|
void *ptr;
|
|
ptr = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY);
|
|
secp256k1_pedersen_context_initialize(ptr);
|
|
secp256k1_rangeproof_context_initialize(ptr);
|
|
return(ptr);
|
|
}
|
|
|
|
void iguana_fixsecp(struct supernet_info *myinfo)
|
|
{
|
|
myinfo->ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY);
|
|
secp256k1_pedersen_context_initialize(myinfo->ctx);
|
|
secp256k1_rangeproof_context_initialize(myinfo->ctx);
|
|
}
|
|
|
|
void libgfshare_init(struct supernet_info *myinfo,uint8_t _logs[256],uint8_t _exps[510])
|
|
{
|
|
uint32_t i,x = 1;
|
|
myinfo->ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY);
|
|
secp256k1_pedersen_context_initialize(myinfo->ctx);
|
|
secp256k1_rangeproof_context_initialize(myinfo->ctx);
|
|
for (i=0; i<255; i++)
|
|
{
|
|
_exps[i] = x;
|
|
_logs[x] = i;
|
|
x <<= 1;
|
|
if ( x & 0x100 )
|
|
x ^= 0x11d; // Unset the 8th bit and mix in 0x1d
|
|
}
|
|
for (i=255; i<510; i++)
|
|
_exps[i] = _exps[i % 255];
|
|
_logs[0] = 0; // can't log(0) so just set it neatly to 0
|
|
if ( (0) )
|
|
{
|
|
void test_mofn(struct supernet_info *myinfo);
|
|
gfshare_test(myinfo,6,11,32);
|
|
test_mofn(myinfo);
|
|
getchar();
|
|
}
|
|
}
|
|
|
|
// Construct and write out the tables for the gfshare code
|
|
int maingen(int argc,char **argv)
|
|
{
|
|
uint8_t logs[256],exps[255]; uint32_t i;
|
|
libgfshare_init(0,logs,exps);
|
|
// The above generation algorithm clearly demonstrates that
|
|
// logs[exps[i]] == i for 0 <= i <= 254
|
|
// exps[logs[i]] == i for 1 <= i <= 255
|
|
// Spew out the tables
|
|
fprintf(stdout, "\
|
|
/*\n\
|
|
* This file is autogenerated by gfshare_maketable.\n\
|
|
*/\n\
|
|
\n\
|
|
static uint8_t logs[256] = {\n ");
|
|
for ( i = 0; i < 256; ++i )
|
|
{
|
|
fprintf(stdout, "0x%02x", logs[i]);
|
|
if( i == 255 )
|
|
fprintf(stdout, " };\n");
|
|
else if( (i % 8) == 7 )
|
|
fprintf(stdout, ",\n ");
|
|
else
|
|
fprintf(stdout, ", ");
|
|
}
|
|
// The exp table we output from 0 to 509 because that way when we
|
|
// do the lagrange interpolation we don't have to be quite so strict
|
|
// with staying inside the field which makes it quicker
|
|
fprintf(stdout, "\
|
|
\n\
|
|
static uint8_t exps[510] = {\n ");
|
|
for ( i = 0; i < 510; ++i )
|
|
{
|
|
fprintf(stdout, "0x%02x", exps[i % 255]); /* exps[255]==exps[0] */
|
|
if ( i == 509 )
|
|
fprintf(stdout, " };\n");
|
|
else if( (i % 8) == 7)
|
|
fprintf(stdout, ",\n ");
|
|
else
|
|
fprintf(stdout, ", ");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* Copyright © 2014-2017 The SuperNET Developers. *
|
|
* *
|
|
* See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at *
|
|
* the top-level directory of this distribution for the individual copyright *
|
|
* holder information and the developer policies on copyright and licensing. *
|
|
* *
|
|
* Unless otherwise agreed in a custom licensing agreement, no part of the *
|
|
* SuperNET software, including this file may be copied, modified, propagated *
|
|
* or distributed except according to the terms contained in the LICENSE file *
|
|
* *
|
|
* Removal or modification of this copyright notice is prohibited. *
|
|
* *
|
|
******************************************************************************/
|
|
|
|
int32_t init_sharenrs(uint8_t sharenrs[255],uint8_t *orig,int32_t m,int32_t n)
|
|
{
|
|
uint8_t *randvals,valid[255];
|
|
int32_t i,j,r,remains,orign;
|
|
if ( m > n || n >= 0xff ) // reserve 255 for illegal sharei
|
|
{
|
|
printf("illegal M.%d of N.%d\n",m,n);
|
|
return(-1);
|
|
}
|
|
randvals = calloc(1,65536);
|
|
OS_randombytes(randvals,65536);
|
|
if ( orig == 0 && n == m )
|
|
{
|
|
memset(sharenrs,0,n);
|
|
for (i=0; i<255; i++)
|
|
valid[i] = (i + 1);
|
|
remains = orign = 255;
|
|
for (i=0; i<n; i++)
|
|
{
|
|
r = (randvals[i] % remains);
|
|
sharenrs[i] = valid[r];
|
|
printf("%d ",sharenrs[i]);
|
|
valid[r] = valid[--remains];
|
|
}
|
|
printf("FULL SET\n");
|
|
}
|
|
else
|
|
{
|
|
memcpy(valid,orig,n);
|
|
memset(sharenrs,0,n);
|
|
for (i=0; i<n; i++)
|
|
printf("%d ",valid[i]);
|
|
printf("valid\n");
|
|
for (i=0; i<m; i++)
|
|
{
|
|
r = rand() % n;
|
|
while ( (j= valid[r]) == 0 )
|
|
{
|
|
//printf("i.%d j.%d m.%d n.%d r.%d\n",i,j,m,n,r);
|
|
r = rand() % n;
|
|
}
|
|
sharenrs[i] = j;
|
|
valid[r] = 0;
|
|
}
|
|
for (i=0; i<n; i++)
|
|
printf("%d ",valid[i]);
|
|
printf("valid\n");
|
|
for (i=0; i<m; i++)
|
|
printf("%d ",sharenrs[i]);
|
|
printf("sharenrs vals m.%d of n.%d\n",m,n);
|
|
//getchar();
|
|
}
|
|
free(randvals);
|
|
for (i=0; i<m; i++)
|
|
{
|
|
for (j=0; j<m; j++)
|
|
{
|
|
if ( i == j )
|
|
continue;
|
|
if ( sharenrs[i] != 0 && sharenrs[i] == sharenrs[j] )
|
|
{
|
|
printf("FATAL: duplicate entry sharenrs[%d] %d vs %d sharenrs[%d]\n",i,sharenrs[i],sharenrs[j],j);
|
|
return(-1);
|
|
}
|
|
}
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
uint8_t *recoverdata(struct supernet_info *myinfo,uint8_t *shares[],uint8_t *sharenrs,int32_t M,uint8_t *recover,int32_t datasize,int32_t N)
|
|
{
|
|
void *G; int32_t i,m=0; uint8_t recovernrs[255],space[8192];
|
|
memset(recovernrs,0,sizeof(recovernrs));
|
|
for (i=0; i<N; i++)
|
|
if ( shares[i] != 0 )
|
|
recovernrs[i] = sharenrs[i], m++;
|
|
if ( m >= M )
|
|
{
|
|
G = gfshare_ctx_initdec(recovernrs,N,datasize,space,sizeof(space));
|
|
for (i=0; i<N; i++)
|
|
if ( shares[i] != 0 )
|
|
gfshare_ctx_dec_giveshare(G,i,shares[i]);
|
|
gfshare_ctx_dec_newshares(G,recovernrs);
|
|
gfshare_ctx_decextract(myinfo->logs,myinfo->exps,G,recover);
|
|
gfshare_ctx_free(G);
|
|
return(recover);
|
|
} else return(0);
|
|
}
|
|
|
|
void calc_share(struct supernet_info *myinfo,uint8_t *buffer,int32_t size,int32_t M,uint32_t ilog,uint8_t *share)
|
|
{
|
|
uint32_t pos,coefficient; uint8_t *share_ptr,share_byte;
|
|
for (pos=0; pos<size; pos++)
|
|
share[pos] = *(buffer++);
|
|
for (coefficient=1; coefficient<M; coefficient++)
|
|
{
|
|
share_ptr = share;
|
|
for (pos=0; pos<size; pos++)
|
|
{
|
|
share_byte = *share_ptr;
|
|
if ( share_byte != 0 )
|
|
share_byte = myinfo->exps[ilog + myinfo->logs[share_byte]];
|
|
*share_ptr++ = (share_byte ^ *buffer++);
|
|
}
|
|
}
|
|
}
|
|
|
|
void calc_shares(struct supernet_info *myinfo,uint8_t *shares,uint8_t *secret,int32_t size,int32_t width,int32_t M,int32_t N,uint8_t *sharenrs,uint8_t *space,int32_t spacesize)
|
|
{
|
|
int32_t i; uint8_t *buffer;
|
|
if ( M*width > spacesize )
|
|
{
|
|
buffer = calloc(M,width);
|
|
printf("calloc M.%d width.%d\n",M,width);
|
|
} else buffer = space;
|
|
memset(shares,0,N * width);
|
|
memcpy(buffer + ((M - 1) * size),secret,size);
|
|
OS_randombytes(buffer,(M - 1) * size);
|
|
for (i=0; i<N; i++)
|
|
{
|
|
calc_share(myinfo,buffer,size,M,myinfo->logs[sharenrs[i]],&shares[i * width]);
|
|
//printf("(%03d %08x) ",sharenrs[i],calc_crc32(0,&shares[i * width],size));
|
|
}
|
|
if ( buffer != space )
|
|
free(buffer);
|
|
}
|
|
|
|
int32_t calc_sharenrs(uint8_t *sharenrs,int32_t N,uint8_t *data,int32_t datasize)
|
|
{
|
|
bits256 hash,hash2; uint8_t r; int32_t i,j,n = sizeof(hash);
|
|
vcalc_sha256(0,hash.bytes,data,datasize);
|
|
vcalc_sha256(0,hash2.bytes,hash.bytes,sizeof(hash));
|
|
for (i=0; i<N; i++)
|
|
{
|
|
while ( 1 )
|
|
{
|
|
if ( n >= sizeof(hash) )
|
|
{
|
|
vcalc_sha256(0,hash.bytes,hash2.bytes,sizeof(hash2));
|
|
vcalc_sha256(0,hash2.bytes,hash.bytes,sizeof(hash));
|
|
n = 0;
|
|
}
|
|
r = hash2.bytes[n++];
|
|
if ( (sharenrs[i]= r) == 0 || sharenrs[i] == 0xff )
|
|
continue;
|
|
for (j=0; j<i; j++)
|
|
if ( sharenrs[j] == sharenrs[i] )
|
|
break;
|
|
if ( j == i )
|
|
break;
|
|
}
|
|
//printf("%3d ",sharenrs[i]);
|
|
}
|
|
return(N);
|
|
}
|
|
|
|
int32_t calcmofn(struct supernet_info *myinfo,uint8_t *allshares,uint8_t *sharenrs,int32_t M,uint8_t *data,int32_t datasize,int32_t N)
|
|
{
|
|
uint8_t space[8192];
|
|
calc_sharenrs(sharenrs,N,data,datasize);
|
|
calc_shares(myinfo,allshares,(void *)data,datasize,datasize,M,N,sharenrs,space,sizeof(space));
|
|
return(datasize);
|
|
}
|
|
|
|
struct mofn256_info
|
|
{
|
|
bits256 secret;
|
|
uint8_t *sharenrs;
|
|
int32_t M,N,allocsize;
|
|
bits256 allshares[];
|
|
};
|
|
|
|
int32_t mofn256_size(uint8_t M,uint8_t N)
|
|
{
|
|
int32_t allocsize;
|
|
allocsize = ((int32_t)(sizeof(struct mofn256_info) + sizeof(bits256) * N + N));
|
|
if ( (allocsize & 0xf) != 0 )
|
|
allocsize += 0x10 - (allocsize & 0xf);
|
|
return(allocsize);
|
|
}
|
|
|
|
struct mofn256_info *mofn256_init(struct supernet_info *myinfo,bits256 secret,uint8_t M,uint8_t N,int32_t calcflag,uint8_t *space,int32_t spacesize)
|
|
{
|
|
int32_t allocsize; struct mofn256_info *mofn = 0;
|
|
if ( M > N || N == 0 || N == 0xff )
|
|
return(0);
|
|
allocsize = mofn256_size(M,N);
|
|
if ( allocsize > spacesize )
|
|
{
|
|
mofn = calloc(1,allocsize);
|
|
mofn->allocsize = allocsize;
|
|
}
|
|
else
|
|
{
|
|
mofn = (void *)space;
|
|
memset(mofn,0,allocsize);
|
|
}
|
|
mofn->M = M;
|
|
mofn->N = N;
|
|
mofn->sharenrs = (void *)&mofn->allshares[N];
|
|
if ( calcflag != 0 )
|
|
{
|
|
mofn->secret = secret;
|
|
calcmofn(myinfo,mofn->allshares[0].bytes,mofn->sharenrs,M,secret.bytes,sizeof(secret),N);
|
|
}
|
|
return(mofn);
|
|
}
|
|
|
|
bits256 mofn256_recover(struct supernet_info *myinfo,struct mofn256_info *mofn)
|
|
{
|
|
uint8_t *shares[255]; bits256 recover; int32_t i;
|
|
for (i=0; i<mofn->N; i++)
|
|
{
|
|
if ( bits256_nonz(mofn->allshares[i]) != 0 )
|
|
shares[i] = mofn->allshares[i].bytes;
|
|
else shares[i] = 0;
|
|
}
|
|
if ( recoverdata(myinfo,shares,mofn->sharenrs,mofn->M,recover.bytes,sizeof(recover),mofn->N) == 0 )
|
|
memset(recover.bytes,0,sizeof(recover));
|
|
return(recover);
|
|
}
|
|
|
|
int32_t test_mofn256(struct supernet_info *myinfo,int32_t M,int32_t N)
|
|
{
|
|
uint8_t space[8192]; char str[65],str2[65]; struct mofn256_info *mofn,*cmp; bits256 secret,recover; int32_t i,allocsize,retval,m = 0;
|
|
allocsize = mofn256_size(M,N);
|
|
cmp = mofn256_init(myinfo,GENESIS_PUBKEY,M,N,0,space,sizeof(space));
|
|
secret = rand256(0);
|
|
mofn = mofn256_init(myinfo,secret,M,N,1,&space[allocsize],sizeof(space) - allocsize);
|
|
memcpy(cmp->sharenrs,mofn->sharenrs,mofn->N);
|
|
for (i=0; i<N; i++)
|
|
if ( (rand() % 100) < 50 )
|
|
cmp->allshares[i] = mofn->allshares[i], m++;
|
|
recover = mofn256_recover(myinfo,cmp);
|
|
retval = -1 * (bits256_cmp(recover,mofn->secret) != 0);
|
|
if ( bits256_cmp(recover,mofn->secret) != 0 )
|
|
{
|
|
if ( m >= M )
|
|
printf("%s %s error m.%d vs M.%d N.%d\n",bits256_str(str,secret),bits256_str(str2,recover),m,mofn->M,mofn->N);
|
|
}
|
|
#if defined(_M_X64)
|
|
if (((uint64_t)mofn - (uint64_t)space) >= sizeof(space) || ((uint64_t)mofn - (uint64_t)space) < 0)
|
|
free(mofn);
|
|
if (((uint64_t)cmp - (uint64_t)space) >= sizeof(space) || ((uint64_t)cmp - (uint64_t)space) < 0)
|
|
free(cmp);
|
|
#else
|
|
if ( ((long)mofn - (long)space) >= sizeof(space) || ((long)mofn - (long)space) < 0 )
|
|
free(mofn);
|
|
if ( ((long)cmp - (long)space) >= sizeof(space) || ((long)cmp - (long)space) < 0 )
|
|
free(cmp);
|
|
#endif
|
|
return(retval);
|
|
}
|
|
|
|
#define N 11
|
|
#define M 6
|
|
void test_mofn(struct supernet_info *myinfo)
|
|
{
|
|
bits256 allshares[N],secret,recover; uint8_t *shares[N],sharenrs[N]; int32_t i,j,m;
|
|
secret = rand256(0);
|
|
srand(secret.uints[0]);
|
|
calcmofn(myinfo,allshares[0].bytes,sharenrs,M,secret.bytes,sizeof(secret),N);
|
|
for (i=0; i<10000; i++)
|
|
{
|
|
memset(shares,0,sizeof(shares));
|
|
for (j=m=0; j<N; j++)
|
|
if ( (rand() % 100) < 55 )
|
|
shares[j] = allshares[j].bytes, m++;
|
|
if ( recoverdata(myinfo,shares,sharenrs,M,recover.bytes,sizeof(secret),N) != 0 )
|
|
{
|
|
if ( memcmp(secret.bytes,recover.bytes,sizeof(secret)) != 0 )
|
|
printf("FAILED m.%d M.%d N.%d\n",m,M,N);
|
|
else if ( (0) )
|
|
{
|
|
char str[65];
|
|
printf("%s PASSED m.%d M.%d N.%d\n",bits256_str(str,recover),m,M,N);
|
|
}
|
|
} //else printf("not enough shares m.%d M.%d N.%d\n",m,M,N);
|
|
}
|
|
printf("finished %d tests\n",i);
|
|
for (i=0; i<10000; i++)
|
|
test_mofn256(myinfo,M,N);
|
|
printf("finished %d tests256\n",i);
|
|
}
|
|
#undef M
|
|
#undef N
|
|
|
|
#define SECP_ENSURE_CTX int32_t flag = 0; if ( ctx == 0 ) { ctx = secp256k1_context_create(SECP256K1_CONTEXT_SIGN | SECP256K1_CONTEXT_VERIFY); secp256k1_pedersen_context_initialize(ctx); secp256k1_rangeproof_context_initialize(ctx); flag++; } else flag = 0; if ( ctx != 0 )
|
|
#define ENDSECP_ENSURE_CTX if ( flag != 0 ) secp256k1_context_destroy(ctx);
|
|
|
|
int32_t iguana_schnorr_peersign(void *ctx,uint8_t *allpub33,uint8_t *partialsig64,int32_t peeri,bits256 mypriv,bits256 privnonce,bits256 *nonces,int32_t n,bits256 msg256)
|
|
{
|
|
secp256k1_pubkey Rall,ALL,PUBS[256],*PUBptrs[256]; int32_t i,num,retval = -1; size_t plen; uint8_t pubkey[33];
|
|
pubkey[0] = 2;
|
|
SECP_ENSURE_CTX
|
|
{
|
|
for (i=num=0; i<n; i++)
|
|
{
|
|
plen = 33;
|
|
memcpy(pubkey+1,nonces[i].bytes,32);
|
|
if ( secp256k1_ec_pubkey_parse(ctx,&PUBS[i],pubkey,plen) == 0 )
|
|
printf("error extracting pubkey.%d of %d\n",i,n);
|
|
if ( i != peeri )
|
|
PUBptrs[num++] = &PUBS[i];
|
|
}
|
|
PUBptrs[num] = &PUBS[peeri];
|
|
if ( secp256k1_ec_pubkey_combine(ctx,&ALL,(void *)PUBptrs,num+1) != 0 )
|
|
{
|
|
plen = 33;
|
|
secp256k1_ec_pubkey_serialize(ctx,allpub33,&plen,&ALL,SECP256K1_EC_COMPRESSED);
|
|
//for (i=0; i<33; i++)
|
|
// printf("%02x",allpub33[i]);
|
|
//printf("\n");
|
|
} else printf("error combining ALL\n");
|
|
if ( secp256k1_ec_pubkey_combine(ctx,&Rall,(void *)PUBptrs,num) != 0 )
|
|
{
|
|
if ( secp256k1_schnorr_partial_sign(ctx,partialsig64,msg256.bytes,mypriv.bytes,&Rall,privnonce.bytes) == 0 )
|
|
printf("iguana_schnorr_peersign: err %d of num.%d\n",peeri,n);
|
|
else retval = 0;
|
|
} else printf("error parsing pubkey.%d\n",peeri);
|
|
ENDSECP_ENSURE_CTX
|
|
}
|
|
return(retval);
|
|
}
|
|
|
|
bits256 iguana_schnorr_noncepair(void *ctx,bits256 *pubkey,uint8_t odd_even,bits256 msg256,bits256 privkey,int32_t maxj)
|
|
{
|
|
bits256 privnonce; int32_t j; uint8_t pubkey33[33];
|
|
memset(privnonce.bytes,0,sizeof(privnonce));
|
|
for (j=0; j<maxj; j++)
|
|
{
|
|
privnonce = bitcoin_schnorr_noncepair(ctx,pubkey33,msg256,privkey);
|
|
if ( pubkey33[0] == (odd_even + 2) )
|
|
{
|
|
memcpy(pubkey->bytes,pubkey33+1,32);
|
|
break;
|
|
}
|
|
}
|
|
if ( j == maxj )
|
|
{
|
|
printf("couldnt generate even noncepair\n");
|
|
iguana_exit(0,0);
|
|
}
|
|
return(privnonce);
|
|
}
|
|
|
|
struct schnorr_info
|
|
{
|
|
bits256 msg256,privkey,pubkey,privnonce,pubnonce,serhash2;
|
|
int32_t M,N; uint32_t msgid;
|
|
uint8_t sig64[64],combinedsig64[64],allpub[33],combined_allpub[33];
|
|
uint16_t ind,nonz;
|
|
bits256 *pubkeys,*pubnonces;
|
|
uint8_t serialized[65 + sizeof(uint16_t) + sizeof(uint32_t) + sizeof(bits256)*4];
|
|
uint8_t sigs[];
|
|
};
|
|
|
|
int32_t schnorr_rwinitdata(int32_t rwflag,uint8_t *serialized,uint16_t *indp,uint32_t *msgidp,bits256 hashes[4])
|
|
{
|
|
int32_t i,j,len = 0;
|
|
len += iguana_rwnum(rwflag,&serialized[len],sizeof(*indp),indp);
|
|
len += iguana_rwnum(rwflag,&serialized[len],sizeof(*msgidp),msgidp);
|
|
for (j=0; j<4; j++)
|
|
for (i=0; i<32; i++)
|
|
iguana_rwnum(rwflag,&serialized[len++],1,&hashes[j].bytes[i]);
|
|
return(len);
|
|
}
|
|
|
|
struct schnorr_info *schnorr_init(struct supernet_info *myinfo,uint32_t msgid,uint16_t ind,int32_t M,int32_t N,void *space,int32_t spacesize,bits256 msg256)
|
|
{
|
|
uint8_t *serialized; bits256 hashes[4]; int32_t len,odd_even = 0; struct schnorr_info *si = space;
|
|
si->pubnonces = (void *)&si->sigs[N * 64];
|
|
si->pubkeys = &si->pubnonces[N];
|
|
si->M = M;
|
|
si->N = N;
|
|
si->ind = ind;
|
|
si->msgid = msgid;
|
|
si->msg256 = msg256;
|
|
si->pubkey = bitcoin_pub256(myinfo->ctx,&si->privkey,odd_even);
|
|
si->privnonce = iguana_schnorr_noncepair(myinfo->ctx,&si->pubnonce,odd_even,msg256,si->privkey,100);
|
|
len = 0;
|
|
serialized = &si->serialized[65];
|
|
hashes[0] = myinfo->myaddr.persistent;
|
|
hashes[1] = si->pubkey;
|
|
hashes[2] = si->pubnonce;
|
|
hashes[3] = msg256;
|
|
len = schnorr_rwinitdata(1,serialized,&ind,&msgid,hashes);
|
|
blockhash_sha256(si->serhash2.bytes,serialized,len);
|
|
if ( bitcoin_sign(myinfo->ctx,"SCHNORR",si->serialized,si->serhash2,myinfo->persistent_priv,1) != 65 )
|
|
printf("error signing schnorr initdata\n");
|
|
return(si);
|
|
}
|
|
|
|
int32_t schnorr_update(struct supernet_info *myinfo,struct schnorr_info *si,uint8_t *serialized,int32_t recvlen)
|
|
{
|
|
int32_t len; uint16_t ind; uint32_t msgid; bits256 hashes[4];
|
|
// verify compact sig
|
|
len = schnorr_rwinitdata(0,&serialized[65],&ind,&msgid,hashes);
|
|
// verify persistent pubkey matches ind
|
|
if ( bits256_cmp(hashes[3],si->msg256) == 0 && msgid == si->msgid && bits256_nonz(si->pubkeys[ind]) == 0 && bits256_nonz(si->pubnonces[ind]) == 0 )
|
|
{
|
|
si->pubkeys[ind] = hashes[1];
|
|
si->pubnonces[ind] = hashes[2];
|
|
if ( ++si->nonz >= si->M )
|
|
{
|
|
iguana_schnorr_peersign(myinfo->ctx,si->allpub,si->sig64,si->ind,si->privkey,si->privnonce,si->pubnonces,si->M,si->msg256);
|
|
// broadcast si->sig64 + ind + msgid
|
|
}
|
|
}
|
|
return(si->M);
|
|
}
|
|
|
|
/*{
|
|
if ( bitcoin_schnorr_combine(myinfo->ctx,si->combinedsig64,si->combined_allpub,si->sigs,si->M,si->msg256) < 0 )
|
|
printf("error combining k.%d sig64 iter.%d\n",k,iter);
|
|
if ( bitcoin_schnorr_verify(myinfo->ctx,si->combinedsig64,si->msg256,si->combined_allpub,33) < 0 )
|
|
printf("allpub2 error verifying combined sig k.%d\n",k);
|
|
}*/
|
|
|
|
void iguana_schnorr(struct supernet_info *myinfo)
|
|
{
|
|
uint8_t allpubs[256][33],allpub[33],allpub2[33],sig64s[256][64],sig64[64],*sigs[256]; bits256 msg256,privnonces[256],signers,privkeys[256],pubkeys[256],pubkeysB[256],nonces[256]; int32_t i,iter,n,k,maxj = 100;
|
|
OS_randombytes((void *)&n,sizeof(n));
|
|
srand(n);
|
|
n = 1 + (rand() % 255);
|
|
// generate onetime keypairs
|
|
for (i=0; i<n; i++)
|
|
pubkeys[i] = bitcoin_pub256(myinfo->ctx,&privkeys[i],0);
|
|
msg256 = rand256(0);
|
|
for (i=0; i<n; i++)
|
|
privnonces[i] = iguana_schnorr_noncepair(myinfo->ctx,&nonces[i],0,msg256,privkeys[i],maxj);
|
|
for (i=0; i<n; i++)
|
|
iguana_schnorr_peersign(myinfo->ctx,allpubs[i],sig64s[i],i,privkeys[i],privnonces[i],nonces,n,msg256);
|
|
for (iter=0; iter<1; iter++)
|
|
{
|
|
memset(signers.bytes,0,sizeof(signers));
|
|
for (i=k=0; i<n; i++)
|
|
{
|
|
if ( (rand() % 100) < 50 )
|
|
{
|
|
printf("%2d ",i);
|
|
sigs[k] = sig64s[i];
|
|
pubkeysB[k] = pubkeys[i];
|
|
k++;
|
|
SETBIT(signers.bytes,i);
|
|
}
|
|
}
|
|
if ( bitcoin_schnorr_combine(myinfo->ctx,sig64,allpub2,sigs,k,msg256) < 0 )
|
|
printf("error combining k.%d sig64 iter.%d\n",k,iter);
|
|
else if ( bitcoin_schnorr_verify(myinfo->ctx,sig64,msg256,allpub2,33) < 0 )
|
|
printf("allpub2 error verifying combined sig k.%d\n",k);
|
|
else if ( (0) ) // doesnt replicate with subsets
|
|
{
|
|
if ( bitcoin_pubkey_combine(myinfo->ctx,allpub,0,pubkeys,n,0,0) == 0 )
|
|
{
|
|
if ( memcmp(allpub,allpubs[0],33) != 0 )
|
|
{
|
|
printf("\n");
|
|
for (k=0; k<33; k++)
|
|
printf("%02x",allpubs[0][k]);
|
|
printf(" combined\n");
|
|
for (k=0; k<33; k++)
|
|
printf("%02x",allpub[k]);
|
|
printf(" allpub, ");
|
|
printf("allpub mismatch iter.%d i.%d n.%d\n",iter,i,n);
|
|
} else printf("validated iter.%d k.%d %llx\n",iter,k,(long long)signers.txid);
|
|
} //else printf("error combining\n");
|
|
} else printf("passed n.%d\n",n);
|
|
}
|
|
}
|
|
|