/****************************************************************************** * Copyright © 2014-2015 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. * * * ******************************************************************************/ #ifdef later #ifdef DEFINES_ONLY #ifndef serdes777_h #define serdes777_h #include <stdio.h> #include <stdint.h> #include <memory.h> #include "../iguana777.h" #define MAX_OPRETURNSIZE 4096 #define PEGGY_MAXLOCKDAYS 180 #define PEGGY_PASTSTAMP 3600 #define PEGGY_FUTURESTAMP 60 #define PRICE_RESOLUTION_ROOT ((int64_t)3163) #define PRICE_RESOLUTION (PRICE_RESOLUTION_ROOT * PRICE_RESOLUTION_ROOT) // 10004569 #define PRICE_RESOLUTION2 (PRICE_RESOLUTION * PRICE_RESOLUTION) // 100091400875761 #define PRICE_RESOLUTION_MAXPVAL ((int64_t)3037000500u) // 303.5613528178975 vs 64 bits: 4294967295 429.30058206405493, #define PRICE_RESOLUTION_MAXUNITS ((int16_t)((int64_t)0x7fffffffffffffffLLu / (SATOSHIDEN * PRICE_RESOLUTION))) // 9219 #define SCALED_PRICE(val,scale) (((scale) * (val)) / PRICE_RESOLUTION) #define Pval(r) ((double)(r)->Pval / PRICE_RESOLUTION) // for display only! #define PERCENTAGE(perc) (((perc) * PRICE_RESOLUTION) / 100) #define PEGGY_RSTATUS_REDEEMED 0 #define PEGGY_RSTATUS_AUTOPURGED 1 #define PEGGY_RSTATUS_MARGINCALL 2 #define PEGGY_FLAGS_HASFUNDING 1 #define PEGGY_FLAGS_PEGGYBASE 2 #define PEGGY_ADDRBTCD 0 #define PEGGY_ADDRCREATE 1 #define PEGGY_ADDRNXT 2 #define PEGGY_ADDRUNIT 3 #define PEGGY_ADDRPUBKEY 4 #define PEGGY_ADDR777 5 #define PEGGY_ADDRFUNDING 6 struct price_resolution { int64_t Pval; }; struct peggy_lock { int16_t peg,denom; uint16_t minlockdays,maxlockdays,clonesmear,mixrange,redemptiongapdays; uint8_t extralockdays,margin; }; struct peggy_newunit { bits256 sha256; struct peggy_lock newlock; }; struct peggy_univaddr { uint8_t addrtype,rmd160[20]; char coin[7]; }; union peggy_addr { struct peggy_univaddr coinaddr; struct peggy_newunit newunit; bits256 sha256; bits384 SaMbits; uint64_t nxt64bits; }; struct peggy_input { uint8_t type,chainlen; union peggy_addr src; uint64_t amount; }; struct peggy_output { uint8_t type,vin; union peggy_addr dest; uint32_t ratio; }; struct peggy_txprices { uint16_t num,maxlockdays; uint32_t timestamp; uint32_t feed[256]; }; struct peggy_txbet { struct price_resolution prediction; char peg[15],binary; }; struct peggy_txmicropay { bits256 claimhash,refundhash; uint32_t expiration; uint8_t chainlen,vin,vout; }; #define PEGGY_MAXSIGS 16 #define PEGGY_MAXINPUTS 15 #define PEGGY_MAXOUTPUTS 16 #define PEGGY_TXNORMAL 0 #define PEGGY_TXBET 1 #define PEGGY_TXPRICES 2 #define PEGGY_TXTUNE 3 #define PEGGY_TXMICROPAY 4 union peggy_bytes8 { uint8_t bytes[8]; uint64_t val; }; struct peggy_txtune { char type,peg[15]; uint64_t val; union peggy_bytes8 B; }; union peggy_txtype { struct peggy_txprices price; struct peggy_txbet bets[64]; struct peggy_txtune tune[64]; struct peggy_txmicropay micropays[16]; }; struct peggy_tx { uint16_t datalen; uint8_t numinputs,numoutputs,txtype,flags,msglen,numdetails; uint32_t timestamp,activation,expiration; struct peggy_input inputs[PEGGY_MAXINPUTS],funding; struct peggy_output outputs[PEGGY_MAXOUTPUTS]; union peggy_txtype details; char hexstr[512]; uint8_t data[4096]; struct acct777_sig sigs[PEGGY_MAXSIGS]; //uint64_t required; }; ///////////////// struct peggy_unit { int64_t estimated_interest,costbasis,amount,marginamount; uint32_t timestamp; int16_t dailyrate; uint8_t baseid,relid; struct peggy_lock lock; bits256 lockhash; }; struct acct777 { uint32_t firstblocknum,firsttimestamp; int64_t balance; }; struct opreturn_buffers { int32_t blocktimestamp,blocknum,num; struct opreturn_entry *ptrs[]; }; char *peggy_tx(char *jsonstr); int32_t serdes777_deserialize(int32_t *signedcountp,struct peggy_tx *Ptx,uint32_t blocktimestamp,uint8_t *data,int32_t totallen); int32_t serdes777_serialize(struct peggy_tx *Ptx,uint32_t blocktimestamp,bits256 privkey,uint32_t timestamp); int32_t peggy_univ2addr(char *coinaddr,struct peggy_univaddr *ua); int32_t peggy_addr2univ(struct peggy_univaddr *ua,char *coinaddr,char *coin); #endif #else #ifndef serdes777_c #define serdes777_c #ifndef serdes777_h #define DEFINES_ONLY #include "serdes777.c" #undef DEFINES_ONLY #endif int32_t serdes777_rwbits(int32_t rwflag,void *ptr,int32_t len,HUFF *hp) { int32_t i,bit; if ( rwflag == 0 ) { for (i=0; i<len; i++) { if ( (bit= hgetbit(hp)) < 0 ) return(-1); if ( bit != 0 ) SETBIT(ptr,i); else CLEARBIT(ptr,i); } //printf("rbits.%d (%02x)\n",len,*(uint8_t *)ptr); } else { //printf("wbits.%d (%02x)\n",len,*(uint8_t *)ptr); for (i=0; i<len; i++) if ( hputbit(hp,GETBIT(ptr,i) != 0) < 0 ) return(-100*i-100); } //printf("rwbits len.%d (%02x)\n",len,*(uint8_t *)dest); //printf("(%d) ",*(uint8_t *)ptr); return(len); } int32_t serdes777_rwsizebits(int32_t rwflag,void *xptr,uint64_t xval,int32_t sizebits,HUFF *hp) { int32_t numbits = 0; if ( 1 && sizebits == 3 ) { if ( rwflag != 0 ) { if ( xval == 0 ) { hputbit(hp,0); return(1); } else { hputbit(hp,1); if ( xval < 3 ) { hputbit(hp,(xval&1) != 0); hputbit(hp,(xval&2) != 0); //hputbit(hp,(xval&4) != 0); return(3); } else { hputbit(hp,1), hputbit(hp,1);//, hputbit(hp,1); if ( serdes777_rwbits(1,xptr,8,hp) != 8 ) return(-30); return(11); } } } else { if ( hgetbit(hp) == 0 ) { *(uint8_t *)xptr = 0; return(1); } else { numbits = hgetbit(hp) + (hgetbit(hp) << 1);// + (hgetbit(hp) << 2); if ( numbits < 3 ) { *(uint8_t *)xptr = numbits; return(3); } else { if ( serdes777_rwbits(0,xptr,8,hp) != 8 ) return(-4); return(11); } } } } else { if ( rwflag != 0 ) { numbits = hcalc_bitsize(xval) - 1; //printf("rwsizebits.%d numbits.%d %02x %llu\n",sizebits,numbits,*(uint8_t *)xptr,(long long)xval); if ( numbits >= (1 << sizebits) ) { printf("numbits overflow.%d doesnt fit into sizebits.%d\n",numbits,sizebits); return(-10); } } if ( serdes777_rwbits(rwflag,&numbits,sizebits,hp) != sizebits ) return(-20); if ( serdes777_rwbits(rwflag,xptr,numbits+1,hp) != numbits+1 ) return(-30); //printf("return.(%02x) numbits.%d + 1 + sizebits.%d\n",*(uint8_t *)xptr,numbits,sizebits); //printf("(%02x) ",*(uint8_t *)xptr); return(numbits + 1 + sizebits); } } #define serdes777_rwchar(rwflag,x,hp) serdes777_rwsizebits(rwflag,xptr,*(uint8_t *)xptr,3,hp) #define serdes777_rwshort(rwflag,x,hp) serdes777_rwsizebits(rwflag,xptr,*(uint16_t *)xptr,4,hp) #define serdes777_rwint(rwflag,x,hp) serdes777_rwsizebits(rwflag,xptr,*(uint32_t *)xptr,5,hp) #define serdes777_rwlong(rwflag,x,hp) serdes777_rwsizebits(rwflag,xptr,*(uint64_t *)xptr,6,hp) int32_t serdes777_convstr(int32_t encoding,void *dest,void *src,int32_t len) { if ( encoding == 16 ) { len >>= 1; decode_hex(dest,len,src); return(8 * len); } else if ( encoding == -16 ) { init_hexbytes_noT(dest,src,len); return(8 * ((len << 1) + 1)); } else if ( encoding == 32 ) return(decode_base32(dest,src,len)); else if ( encoding == -32 ) return(init_base32(dest,src,len)); return(-1); } int32_t serdes777_rwvarstr(int32_t rwflag,uint8_t *data,int32_t *lenp,HUFF *hp) { int32_t i,n,numbits,total; void *xptr = lenp; if ( (total= serdes777_rwchar(rwflag,lenp,hp)) < 0 ) return(total); n = *lenp; for (i=0; i<n; i++) { xptr = &data[i]; if ( (numbits= serdes777_rwchar(rwflag,&data[i],hp)) < 0 ) return(-1); total += numbits; } return(total); } int32_t serdes777_rwstr(int32_t rwflag,uint8_t *data,HUFF *hp,int32_t encoding) { int32_t total,len; uint8_t *ptr,buf[MAX_OPRETURNSIZE*2 + 1]; if ( rwflag == 0 ) { ptr = (encoding != 0) ? buf : data; if ( (total= serdes777_rwvarstr(rwflag,ptr,&len,hp)) < 0 ) return(-1); if ( encoding != 0 ) total = serdes777_convstr(-encoding,data,ptr,total); } else { len = (int32_t)strlen((void *)data); if ( encoding != 0 ) { if ( (len= serdes777_convstr(encoding,buf,data,len)) < 0 ) { printf("serdes777_writestr error doing serdes777_convstr encoding.%d len.%d\n",encoding,len); return(-1); } if ( (len & 7) != 0 ) { printf("serdes777_writestr error doing serdes777_convstr encoding.%d len.%d unaligned return\n",encoding,len); return(-1); } data = (void *)buf; } total = serdes777_rwvarstr(rwflag,data,&len,hp); } return(total); } int32_t serdes777_rw(int32_t rwflag,void *xptr,int32_t size,HUFF *hp) { if ( size < 0 ) // high entropy return(serdes777_rwbits(rwflag,xptr,-size*8,hp)); //else return(serdes777_rwbits(rwflag,xptr,size*8,hp)); switch ( size ) { case 1: return(serdes777_rwchar(rwflag,xptr,hp)); break; case 2: return(serdes777_rwshort(rwflag,xptr,hp)); break; case 4: return(serdes777_rwint(rwflag,xptr,hp)); break; case 8: return(serdes777_rwlong(rwflag,xptr,hp)); break; default: return(serdes777_rwstr(rwflag,xptr,hp,size)); break; } return(-1); } int32_t serdes777_codec(int32_t rwflag,void *ptr,int32_t maxlen,long tokens[][2],long numtokens,HUFF *hp) { int32_t i,numbits,totalbits = 0; // printf("peggy_rwtx.%d %p max.%d %p numtokens.%ld %p\n",rwflag,tx,maxlen,txtokens,numtokens,hp); for (i=0; i<numtokens; i++) { //printf("i.%d peggy_rwtx.%d %p max.%d %p numtokens.%ld %p (%ld %ld)\n",i,rwflag,ptr,maxlen,tokens,numtokens,hp,tokens[i][0],tokens[i][1]); if ( (numbits= serdes777_rw(rwflag,(void *)((long)ptr + tokens[i][0]),(int32_t)tokens[i][1],hp)) < 0 ) return(-i-1); totalbits += numbits; } return(totalbits); } int32_t serdes777_rwlock(int32_t rwflag,struct peggy_lock *lock,HUFF *TX) { long serdes777_locktokens[][2] = { { 0, sizeof(lock->peg) }, { (long)&lock->denom - (long)lock, sizeof(lock->denom) }, { (long)&lock->minlockdays - (long)lock, sizeof(lock->minlockdays) }, { (long)&lock->maxlockdays - (long)lock, sizeof(lock->maxlockdays) }, { (long)&lock->clonesmear - (long)lock, sizeof(lock->clonesmear) }, { (long)&lock->mixrange - (long)lock, sizeof(lock->mixrange) }, { (long)&lock->margin - (long)lock, sizeof(lock->margin) }, }; return(serdes777_codec(rwflag,lock,sizeof(*lock),serdes777_locktokens,sizeof(serdes777_locktokens)/sizeof(*serdes777_locktokens),TX)); } int32_t serdes777_rwaddr(int32_t rwflag,union peggy_addr *addr,int32_t type,HUFF *TX) { int32_t n,numbits = 0; switch ( type ) { case PEGGY_ADDRBTCD: numbits = serdes777_rw(rwflag,&addr->coinaddr,0,TX); break; case PEGGY_ADDRNXT: numbits = serdes777_rw(rwflag,&addr->nxt64bits,(int32_t)-sizeof(addr->nxt64bits),TX); break; case PEGGY_ADDRCREATE: if ( (numbits= serdes777_rw(rwflag,&addr->newunit.sha256.bytes,(int32_t)-sizeof(addr->newunit.sha256),TX)) < 0 ) return(-3); if ( (n= serdes777_rwlock(rwflag,&addr->newunit.newlock,TX)) < 0 ) return(-2); numbits += n; break; case PEGGY_ADDRUNIT: case PEGGY_ADDRPUBKEY: if ( (numbits= serdes777_rw(rwflag,&addr->newunit.sha256.bytes,(int32_t)-sizeof(addr->newunit.sha256),TX)) < 0 ) return(-3); if ( (n= serdes777_rw(rwflag,&addr->newunit.newlock.peg,sizeof(addr->newunit.newlock.peg),TX)) < 0 ) return(-2); if ( (n= serdes777_rw(rwflag,&addr->newunit.newlock.minlockdays,sizeof(addr->newunit.newlock.minlockdays),TX)) < 0 ) return(-2); case PEGGY_ADDR777: numbits = serdes777_rw(rwflag,&addr->SaMbits.bytes,(int32_t)-sizeof(addr->SaMbits),TX); break; default: numbits = -1; break; } return(numbits); } int32_t serdes777_rwbets(int32_t rwflag,struct peggy_txbet *bets,int32_t numbets,HUFF *TX) { long serdes777_bettokens[][2] = { { 0, sizeof(bets[0].prediction) }, { (long)&bets[0].peg - (long)bets, 0 }, { (long)&bets[0].binary - (long)bets, sizeof(bets[0].binary) } }; int32_t i,n,numbits = 0; if ( numbets <= 0 ) return(0); for (i=0; i<numbets; i++) { if ( (n= serdes777_codec(rwflag,&bets[i],sizeof(bets[i]),serdes777_bettokens,sizeof(serdes777_bettokens)/sizeof(*serdes777_bettokens),TX)) < 0 ) return(-1); numbits += n; } return(numbits); } int32_t serdes777_rwmicropay(int32_t rwflag,struct peggy_txmicropay *micropay,int32_t num,HUFF *TX) { long serdes777_mptokens[][2] = { { 0, sizeof(micropay[0].claimhash) }, { (long)µpay[0].refundhash - (long)micropay, sizeof(micropay[0].refundhash) }, { (long)µpay[0].expiration - (long)micropay, sizeof(micropay[0].expiration) }, { (long)µpay[0].chainlen - (long)micropay, sizeof(micropay[0].chainlen) }, { (long)µpay[0].vin - (long)micropay, sizeof(micropay[0].vin) }, { (long)µpay[0].vout - (long)micropay, sizeof(micropay[0].vout) }, }; int32_t i,n,numbits = 0; if ( num <= 0 ) return(0); for (i=0; i<num; i++) { if ( (n= serdes777_codec(rwflag,µpay[i],sizeof(micropay[i]),serdes777_mptokens,sizeof(serdes777_mptokens)/sizeof(*serdes777_mptokens),TX)) < 0 ) return(-1); numbits += n; } return(numbits); } int32_t serdes777_rwinout(int32_t rwflag,int32_t inoutflag,void *inout,HUFF *TX) { int32_t type,a,b,c,d; struct peggy_input *in; struct peggy_output *out; if ( inoutflag == 0 ) { in = inout; a = serdes777_rw(rwflag,&in->type,sizeof(in->type),TX); b = serdes777_rw(rwflag,&in->chainlen,sizeof(in->chainlen),TX); c = serdes777_rwaddr(rwflag,&in->src,in->type,TX); d = serdes777_rw(rwflag,&in->amount,sizeof(in->amount),TX); type = in->type; } else { out = inout; a = serdes777_rw(rwflag,&out->type,sizeof(out->type),TX); b = serdes777_rw(rwflag,&out->vin,sizeof(out->vin),TX); c = serdes777_rwaddr(rwflag,&out->dest,out->type,TX); d = serdes777_rw(rwflag,&out->ratio,sizeof(out->ratio),TX); type = out->type; } if ( a < 0 || b < 0 || c < 0 || d < 0 ) { printf("serdes777_rwinput.%d error encoding type.%d %d %d %d %d\n",rwflag,type,a,b,c,d); //getchar(); return(-1); } return(a + b + c + d); } int32_t serdes777_rwprices(int32_t rwflag,struct peggy_txprices *price,HUFF *TX) { long serdes777_pricetokens[][2] = { { 0, sizeof(price->num) }, //{ (long)&prices->btcusd - (long)prices, sizeof(prices->btcusd) }, { (long)&price->timestamp - (long)price, sizeof(price->timestamp) }, { (long)&price->maxlockdays - (long)price, sizeof(price->maxlockdays) } }; long serdes777_pricetoken2[][2] = { { 0, sizeof(price->feed[0]) } }; int32_t i,n,numbits = -1; if ( (numbits= serdes777_codec(rwflag,&price->num,sizeof(price->num),serdes777_pricetokens,sizeof(serdes777_pricetokens)/sizeof(*serdes777_pricetokens),TX)) < 0 ) return(-1); //printf("numprices.%u btcusd %.6f: ",prices->num,(double)1000.*prices->btcusd); for (i=0; i<price->num; i++) { if ( (n= serdes777_codec(rwflag,&price->feed[i],sizeof(price->feed[i]),serdes777_pricetoken2,sizeof(serdes777_pricetoken2)/sizeof(*serdes777_pricetoken2),TX)) < 0 ) return(-1); //printf("(%d of %d %u) ",i,prices->num,prices->prices[i]); numbits += n; } return(numbits); } int32_t serdes777_rwtune(int32_t rwflag,struct peggy_txtune *tune,int32_t numtunes,HUFF *TX) { long serdes777_tunetokens[][2] = { { 0, sizeof(tune[0].type) }, { (long)&tune[0].peg - (long)tune, 0 }, { (long)&tune[0].val - (long)tune, sizeof(tune[0].val) }, { (long)&tune[0].B - (long)tune, sizeof(tune[0].B) }, }; int32_t i,n,numbits = -1; if ( numtunes <= 0 ) return(0); for (i=0; i<numtunes; i++) { if ( (n= serdes777_codec(rwflag,&tune[i],sizeof(tune[i]),serdes777_tunetokens,sizeof(serdes777_tunetokens)/sizeof(*serdes777_tunetokens),TX)) < 0 ) return(-1); numbits += n; } return(numbits); } int32_t serdes777_rwdetails(int32_t rwflag,struct peggy_tx *Ptx,HUFF *TX) { int32_t n,numbits = 0; //printf("rwdetails.%d offset.%d txtype.%d\n",rwflag,TX->bitoffset,Ptx->txtype); switch ( Ptx->txtype ) { case PEGGY_TXBET: numbits = serdes777_rwbets(rwflag,Ptx->details.bets,Ptx->numdetails,TX); break; case PEGGY_TXPRICES: numbits = serdes777_rwprices(rwflag,&Ptx->details.price,TX); break; case PEGGY_TXTUNE: break; numbits = serdes777_rwtune(rwflag,Ptx->details.tune,Ptx->numdetails,TX); break; case PEGGY_TXMICROPAY: numbits = serdes777_rwmicropay(rwflag,Ptx->details.micropays,Ptx->numdetails,TX); break; } if ( numbits < 0 ) return(-1); if ( Ptx->msglen != 0 ) { if ( (n = serdes777_rwstr(rwflag,(uint8_t *)Ptx->hexstr,TX,16)) < 0 ) return(-1); numbits += n; } return(numbits); } int32_t serdes777_sethdrtokens(long serdes777_hdrtokens[][2],struct peggy_tx *Ptx) { long hdrtokens[][2] = { { 0, -sizeof(Ptx->datalen) }, { (long)&Ptx->numinputs - (long)Ptx, sizeof(Ptx->numinputs) }, { (long)&Ptx->numoutputs - (long)Ptx, sizeof(Ptx->numoutputs) }, { (long)&Ptx->txtype - (long)Ptx, sizeof(Ptx->txtype) }, { (long)&Ptx->flags - (long)Ptx, sizeof(Ptx->flags) }, { (long)&Ptx->msglen - (long)Ptx, sizeof(Ptx->msglen) }, { (long)&Ptx->numdetails - (long)Ptx, sizeof(Ptx->numdetails) }, { (long)&Ptx->timestamp - (long)Ptx, sizeof(Ptx->timestamp) }, { (long)&Ptx->activation - (long)Ptx, sizeof(Ptx->activation) }, { (long)&Ptx->expiration - (long)Ptx, sizeof(Ptx->expiration) }, }; memcpy(serdes777_hdrtokens,hdrtokens,sizeof(hdrtokens)); return((int32_t)(sizeof(hdrtokens)/sizeof(*hdrtokens))); } int32_t serdes777_rwtx(int32_t rwflag,struct peggy_tx *Ptx,HUFF *TX) { int32_t i,n,iter,numbits,totalbits = 0; void *ptr; long serdes777_hdrtokens[16][2]; //printf("peggy_rwtx.%d %p %p size.%d\n",rwflag,Ptx,TX,TX->endpos); if ( (totalbits= serdes777_codec(rwflag,Ptx,sizeof(*Ptx),serdes777_hdrtokens,serdes777_sethdrtokens(serdes777_hdrtokens,Ptx),TX)) < 0 ) { printf("serdes777_process error decoding opreturn datalen.%d hdrtokens\n",totalbits); return(-1); } if ( (Ptx->flags & PEGGY_FLAGS_HASFUNDING) != 0 ) { if ( (numbits= serdes777_rwinout(rwflag,0,&Ptx->funding,TX)) < 0 ) { printf("serdes777_process error serdes.%d funding %.8f %p\n",rwflag,dstr(Ptx->funding.amount),&Ptx->funding.src.coinaddr); return(-1); } totalbits += numbits; } n = Ptx->numinputs; for (iter=0; iter<2; iter++) { for (i=0; i<n; i++) { if ( iter == 0 ) ptr = &Ptx->inputs[i]; else ptr = &Ptx->outputs[i]; if ( (numbits= serdes777_rwinout(rwflag,iter,ptr,TX)) < 0 ) { printf("serdes777_process error serdes.%d input.%d\n",rwflag,i); return(-1); } totalbits += numbits; printf("%d iter.%d .%d\n",i,iter,totalbits/8); } n = Ptx->numoutputs; } if ( (numbits= serdes777_rwdetails(rwflag,Ptx,TX)) < 0 ) { printf("serdes777_process error serdes777_rwdetails.%d datalen.%d\n",i,totalbits); return(-1); } totalbits += numbits; return(totalbits); } int serdes777_checktimestamp(uint32_t blocktimestamp,uint32_t timestamp) { if ( blocktimestamp == 0 || timestamp == 0 ) return(0); else if ( timestamp < ((int64_t)blocktimestamp - PEGGY_PASTSTAMP) || timestamp > ((int64_t)blocktimestamp + PEGGY_FUTURESTAMP) ) return(-1); return(0); } int32_t serdes777_deserialize(int32_t *signedcountp,struct peggy_tx *Ptx,uint32_t blocktimestamp,uint8_t *data,int32_t totallen) { int32_t i,n,len,totalbits,remains; HUFF TX; memset(Ptx,0,sizeof(*Ptx)); memcpy(Ptx->data,data,totallen); memset(&TX,0,sizeof(TX)), _init_HUFF(&TX,totallen,data), TX.endpos = (totallen * 8); if ( (totalbits= serdes777_rwtx(0,Ptx,&TX)) < 0 ) return(-1); if ( serdes777_checktimestamp(blocktimestamp,Ptx->timestamp) < 0 ) { printf("serdes777_deserialize: timestamp.%u too different from Ptx %u %d\n",blocktimestamp,Ptx->timestamp,blocktimestamp - Ptx->timestamp); return(-1); } if ( (len= hconv_bitlen(totalbits)) > totallen ) { printf("serdes777_process error totalbits.%d len.%d exceeded totallen.%d\n",totalbits,len,totallen); return(-1); } Ptx->datalen = len; //for (i=0; i<len; i++) // printf("%02x",Ptx->data[i]); //printf(" crc.%08x datalen.%d vs totallen.%d\n",_crc32(0,Ptx->data,len),len,totallen); remains = (totallen - len); if ( (remains % sizeof(bits256)) != 0 ) { printf("serdes777_process error totalbits.%d remains.%d nonzero modval.%d\n",totalbits,remains,(int32_t)(remains % sizeof(bits256))); return(-1); } n = (int32_t)(remains / sizeof(bits256)); //printf("n.%d remains.%d\n",n,remains); if ( n > 0 ) { if ( Ptx->numinputs == 1 && Ptx->inputs[0].type == PEGGY_ADDRPUBKEY && n == 1 ) { memcpy(Ptx->sigs[0].sigbits.bytes,&data[len],sizeof(bits256)), len += sizeof(bits256); Ptx->sigs[0].pubkey = Ptx->inputs[0].src.sha256; } else { for (i=0; i<n; i++,len+=sizeof(bits256)) memcpy((i & 1) == 0 ? Ptx->sigs[i/2].sigbits.bytes : Ptx->sigs[i/2].pubkey.bytes,&data[len],sizeof(bits256)); } for (i=0; i<n; i++) { if ( Ptx->sigs[i].sigbits.txid != 0 ) { if ( (Ptx->sigs[i].signer64bits= acct777_validate(&Ptx->sigs[i],Ptx->timestamp,Ptx->data,Ptx->datalen)) == 0 ) { printf("Tx validation error at sig.%d\n",i); return(-1); } (*signedcountp)++; printf("len.%d verify.%d t%u %llx %llu\n",len,i,Ptx->timestamp,(long long)Ptx->sigs[i].sigbits.txid,(long long)Ptx->sigs[i].signer64bits); } else break; } } return(len); } int32_t serdes777_serialize(struct peggy_tx *Ptx,uint32_t blocktimestamp,bits256 privkey,uint32_t timestamp) { int32_t i,len,numbits; HUFF TX; memset(Ptx->data,0,sizeof(Ptx->data)); memset(&TX,0,sizeof(TX)), _init_HUFF(&TX,sizeof(Ptx->data),Ptx->data); if ( Ptx->timestamp == 0 ) Ptx->timestamp = timestamp; else if ( timestamp != 0 && serdes777_checktimestamp(blocktimestamp,timestamp) < 0 ) { printf("serdes777_serialize: timestamp.%u too different from Ptx %u %d\n",timestamp,Ptx->timestamp,timestamp - Ptx->timestamp); return(-1); } if ( (numbits= serdes777_rwtx(1,Ptx,&TX)) < 0 ) return(-1); //printf("TX.bitoffset.%d\n",TX.bitoffset); len = hconv_bitlen(TX.bitoffset); Ptx->datalen = len; hseek(&TX,0,SEEK_SET); if ( serdes777_rw(1,&Ptx->datalen,-(int32_t)sizeof(Ptx->datalen),&TX) < 0 ) return(-1); //for (i=0; i<len; i++) // printf("%02x",Ptx->data[i]); //printf(" crc.%08x datalen.%d\n",_crc32(0,Ptx->data,len),len); for (i=0; i<sizeof(Ptx->sigs)/sizeof(*Ptx->sigs); i++) { //printf("scan sig.%d\n",i); if ( Ptx->sigs[i].sigbits.txid == 0 ) break; if ( acct777_validate(&Ptx->sigs[i],Ptx->timestamp,Ptx->data,len) != Ptx->sigs[i].signer64bits ) { printf("Tx validation error at sig.%d\n",i); return(-1); } memcpy(&Ptx->data[len],Ptx->sigs[i].sigbits.bytes,sizeof(Ptx->sigs[i].sigbits)), len += sizeof(Ptx->sigs[i].sigbits); memcpy(&Ptx->data[len],Ptx->sigs[i].pubkey.bytes,sizeof(Ptx->sigs[i].pubkey)), len += sizeof(Ptx->sigs[i].pubkey); } if ( i < sizeof(Ptx->sigs)/sizeof(*Ptx->sigs)-1 && Ptx->timestamp != 0 && privkey.txid != 0 ) { Ptx->sigs[i].signer64bits = acct777_signtx(&Ptx->sigs[i],privkey,Ptx->timestamp,Ptx->data,len); printf("len.%d sign sig.%d %llx %llu t%u\n",len,i,(long long)Ptx->sigs[i].sigbits.txid,(long long)Ptx->sigs[i].signer64bits,Ptx->timestamp); memcpy(&Ptx->data[len],Ptx->sigs[i].sigbits.bytes,sizeof(Ptx->sigs[i].sigbits)), len += sizeof(Ptx->sigs[i].sigbits); memcpy(&Ptx->data[len],Ptx->sigs[i].pubkey.bytes,sizeof(Ptx->sigs[i].pubkey)), len += sizeof(Ptx->sigs[i].pubkey); if ( Ptx->sigs[i].signer64bits != acct777_validate(&Ptx->sigs[i],Ptx->timestamp,Ptx->data,Ptx->datalen) ) { printf("Tx validation error at sig.%d\n",i); //return(-1); } printf("len.%d verify.%d t%u %llx %llu\n",len,i,Ptx->timestamp,(long long)Ptx->sigs[i].sigbits.txid,(long long)Ptx->sigs[i].signer64bits); } //else printf("skip signing t%u %llu\n",timestamp,(long long)privkey.txid); return(len); } int32_t accts777_parse(union peggy_addr *addr,cJSON *item,int32_t type) { int32_t peggy_setname(char *buf,char *name); char name[16],*coinaddr,*hashstr; struct peggy_lock *lock; if ( item == 0 ) return(-1); switch ( type ) { case PEGGY_ADDRBTCD: if ( (coinaddr= jstr(item,"BTCD")) != 0 && peggy_addr2univ(&addr->coinaddr,coinaddr,"BTCD") < 0 ) { printf("illegal coinaddr.(%s)\n",coinaddr); return(-1); } break; case PEGGY_ADDRCREATE: if ( (hashstr= jstr(item,"lockhash")) != 0 && strlen(hashstr) == 64 ) decode_hex(addr->newunit.sha256.bytes,sizeof(bits256),hashstr); lock = &addr->newunit.newlock; lock->peg = peggy_setname(name,jstr(item,"peg")); lock->denom = jint(item,"denom"); lock->minlockdays = jint(item,"minlockdays"); lock->maxlockdays = jint(item,"maxlockdays"); lock->clonesmear = jint(item,"clonesmear"); lock->mixrange = jint(item,"mixrange"); lock->margin = jint(item,"margin"); break; case PEGGY_ADDRNXT: addr->nxt64bits = j64bits(item,"NXT"); break; case PEGGY_ADDRUNIT: if ( (hashstr= jstr(item,"unithash")) != 0 && strlen(hashstr) == 64 ) decode_hex(addr->sha256.bytes,sizeof(bits256),hashstr); break; case PEGGY_ADDR777: if ( (hashstr= jstr(item,"SaMbits")) != 0 && strlen(hashstr) == 96 ) decode_hex(addr->SaMbits.bytes,sizeof(bits384),hashstr); break; case PEGGY_ADDRPUBKEY: if ( (hashstr= jstr(item,"unlockpubkey")) != 0 && strlen(hashstr) == 64 ) decode_hex(addr->newunit.sha256.bytes,sizeof(bits256),hashstr); lock = &addr->newunit.newlock; lock->peg = peggy_setname(name,jstr(item,"peg")); lock->minlockdays = lock->maxlockdays = jint(item,"lockdays"); if ( jint(item,"denom") < 0 ) lock->peg = -lock->peg; break; } return(0); } int32_t peggy_inputs(struct peggy_tx *Ptx,cJSON *array) { int32_t i,n = 0; cJSON *item; if ( array != 0 && (n= cJSON_GetArraySize(array)) > 0 ) { for (i=0; i<n&&i<PEGGY_MAXINPUTS; i++) { item = jitem(array,i); Ptx->inputs[i].type = juint(item,"type"); Ptx->inputs[i].chainlen = juint(item,"chainlen"); Ptx->inputs[i].amount = juint(item,"amount"); accts777_parse(&Ptx->inputs[i].src,item,Ptx->inputs[i].type); } } return(n); } int32_t peggy_outputs(struct peggy_tx *Ptx,cJSON *array) { int32_t i,n=0; cJSON *item; if ( array != 0 && (n= cJSON_GetArraySize(array)) > 0 ) { for (i=0; i<n&&i<PEGGY_MAXOUTPUTS; i++) { item = jitem(array,i); Ptx->outputs[i].type = juint(item,"type"); Ptx->outputs[i].vin = juint(item,"vin"); Ptx->outputs[i].ratio = juint(item,"ratio"); accts777_parse(&Ptx->outputs[i].dest,item,Ptx->outputs[i].type); } } return(n); } int32_t peggy_details(struct peggy_tx *Ptx,cJSON *json,int32_t txtype,uint32_t btcusd) { struct destbuf name; int32_t i,n=0; cJSON *item,*array; char *hashstr,*refundstr; struct peggy_txtune *tune; struct peggy_txmicropay *mp; struct peggy_txbet *bet; Ptx->details.price.timestamp = juint(json,"genesistime"); Ptx->details.price.maxlockdays = juint(json,"maxlockdays"); //printf("BTCUSD.%d vs %d\n",Ptx->details.prices.btcusd,btcusd); if ( (array= jarray(&n,json,"details")) != 0 && (n= cJSON_GetArraySize(array)) > 0 ) { if ( txtype == PEGGY_TXPRICES ) Ptx->details.price.num = n; for (i=0; i<n; i++) { item = jitem(array,i); switch ( txtype ) { case PEGGY_TXNORMAL: break; case PEGGY_TXBET: bet = &Ptx->details.bets[i]; copy_cJSON(&name,jobj(item,"peg")); safecopy(bet->peg,name.buf,sizeof(bet->peg)); bet->binary = juint(item,"binary"); bet->prediction.Pval = juint(item,"prediction"); break; case PEGGY_TXPRICES: Ptx->details.price.feed[i] = juint(item,0); break; case PEGGY_TXTUNE: tune = &Ptx->details.tune[i]; copy_cJSON(&name,jobj(item,"peg")); safecopy(tune->peg,name.buf,sizeof(bet->peg)); tune->type = juint(item,"type"); tune->val = j64bits(item,"val"); tune->B.val = j64bits(item,"valB"); tune->B.bytes[0] = j64bits(item,"interesttenths"); tune->B.bytes[1] = j64bits(item,"posboost"); tune->B.bytes[2] = j64bits(item,"negpenalty"); tune->B.bytes[3] = j64bits(item,"feediv"); tune->B.bytes[4] = j64bits(item,"feemult"); break; case PEGGY_TXMICROPAY: mp = &Ptx->details.micropays[i]; if ( (hashstr= jstr(item,"claimhash")) != 0 && strlen(hashstr) == 64 ) decode_hex(mp->claimhash.bytes,sizeof(bits256),hashstr); if ( (refundstr= jstr(item,"refundhash")) != 0 && strlen(refundstr) == 64 ) decode_hex(mp->refundhash.bytes,sizeof(bits256),refundstr); mp->expiration = juint(item,"expiration"); mp->chainlen = juint(item,"chainlen"); mp->vin = juint(item,"vin"); mp->vout = juint(item,"vout"); break; } } } return(0); } char *peggy_tx(char *jsonstr) { cJSON *json; int32_t i,n,len,signedcount; char *hexstr,opreturnstr[8192],retbuf[4096],retbufstr[8192],checkstr[8192]; struct peggy_tx Ptx,checkPtx; bits256 privkey; struct destbuf tmp; memset(&Ptx,0,sizeof(Ptx)); opreturnstr[0] = 0; memset(privkey.bytes,0,sizeof(privkey)); if ( (json= cJSON_Parse(jsonstr)) != 0 ) { printf("(%s)\n",jsonstr); Ptx.flags = juint(json,"flags"); Ptx.txtype = juint(json,"txtype"); Ptx.numinputs = peggy_inputs(&Ptx,jarray(&n,json,"inputs")); Ptx.numoutputs = peggy_outputs(&Ptx,jarray(&n,json,"outputs")); if ( Ptx.txtype != PEGGY_TXNORMAL ) Ptx.numdetails = peggy_details(&Ptx,json,Ptx.txtype,juint(json,"btcusd")); if ( (hexstr= jstr(json,"privkey")) != 0 && strlen(hexstr) == 64 ) decode_hex(privkey.bytes,sizeof(privkey),hexstr); //else printf("no privkey.%p (%s)\n",hexstr,jsonstr); if ( (hexstr= jstr(json,"hexstr")) != 0 && strlen(hexstr) < sizeof(Ptx.hexstr) ) strcpy(Ptx.hexstr,hexstr), Ptx.msglen = (int32_t)strlen(hexstr)/2; Ptx.activation = juint(json,"activation"); Ptx.expiration = juint(json,"expiration"); if ( (Ptx.funding.amount= juint(json,"funds")) != 0 ) { copy_cJSON(&tmp,jobj(json,"fundsaddr")); //safecopy(Ptx.funding.src.coinaddr,tmp.buf,sizeof(Ptx.funding.src.coinaddr)); if ( peggy_addr2univ(&Ptx.funding.src.coinaddr,tmp.buf,"BTCD") < 0 ) { printf("warning: illegal funding address.(%s)\n",tmp.buf); } Ptx.flags |= PEGGY_FLAGS_HASFUNDING; Ptx.funding.type = PEGGY_ADDRFUNDING; } Ptx.timestamp = (uint32_t)time(NULL); len = serdes777_serialize(&Ptx,Ptx.timestamp,privkey,Ptx.timestamp); init_hexbytes_noT(opreturnstr,Ptx.data,len); //printf("datalen.%d (%s).%ld\n",Ptx.datalen,opreturnstr,strlen(opreturnstr)); /*if ( PEGS != 0 && peggy_txind(&tipvalue,PEGS->accts,0,Ptx.timestamp,0,&Ptx,fundsvalue,fundsaddr) < 0 ) { printf("%s\nerror validating tx\n",opreturnstr); //return(clonestr("\"error\":\"error validating tx\"}")); }*/ len = serdes777_deserialize(&signedcount,&checkPtx,Ptx.timestamp,Ptx.data,len); retbuf[0] = OP_RETURN_OPCODE; if ( len+3 < 0xfe ) { retbuf[1] = len+3; strcpy(retbuf+2,"PAX"); memcpy(retbuf+5,Ptx.data,len); init_hexbytes_noT(retbufstr,(void *)retbuf,len+5); } else { retbuf[1] = 0xfe; retbuf[2] = (len+3) & 0xff; retbuf[3] = ((len+3) >> 8) & 0xff; strcpy(retbuf+4,"PAX"); memcpy(retbuf+7,Ptx.data,len); init_hexbytes_noT(retbufstr,(void *)retbuf,len+7); } init_hexbytes_noT(checkstr,checkPtx.data,len); if ( strcmp(checkstr,opreturnstr) != 0 ) { for (i=0; i<Ptx.datalen; i++) if ( checkstr[i] != opreturnstr[i] ) printf("(%02x != %02x).%d ",checkstr[i],opreturnstr[i],i); printf("%s Ptx\n%s check\npeggy_tx ser/deser error datalen.%d\n",opreturnstr,checkstr,checkPtx.datalen); return(clonestr("\"error\":\"ser/deser error\"}")); } //else printf("peggy_tx success\n%s\n%s\n",opreturnstr,checkstr); free_json(json); //printf("%s\n",retbufstr); return(clonestr(retbufstr)); } return(clonestr("\"error\":\"couldnt create opreturn\"}")); } #endif #endif #endif