/******************************************************************************
* Copyright © 2014 - 2016 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 . *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# include "iguana777.h"
# include "SuperNET.h"
//struct iguana_txid { bits256 txid; uint32_t txidind,firstvout,firstvin,locktime,version,timestamp; uint16_t numvouts,numvins; } __attribute__((packed));
//struct iguana_msgvin { bits256 prev_hash; uint8_t *script; uint32_t prev_vout,scriptlen,sequence; } __attribute__((packed));
//struct iguana_spend { uint32_t spendtxidind; int16_t prevout; uint16_t tbd:14,external:1,diffsequence:1; } __attribute__((packed));
void iguana_vinset ( struct iguana_info * coin , int32_t height , struct iguana_msgvin * vin , struct iguana_txid * tx , int32_t i )
{
struct iguana_spend * s , * S ; uint32_t spendind ; struct iguana_bundle * bp ;
struct iguana_ramchaindata * rdata ; struct iguana_txid * T ; bits256 * X ;
memset ( vin , 0 , sizeof ( * vin ) ) ;
if ( height > = 0 & & height < coin - > chain - > bundlesize * coin - > bundlescount & & ( bp = coin - > bundles [ height / coin - > chain - > bundlesize ] ) ! = 0 & & ( rdata = bp - > ramchain . H . data ) ! = 0 )
{
S = ( void * ) ( long ) ( ( long ) rdata + rdata - > Soffset ) ;
X = ( void * ) ( long ) ( ( long ) rdata + rdata - > Xoffset ) ;
T = ( void * ) ( long ) ( ( long ) rdata + rdata - > Toffset ) ;
spendind = ( tx - > firstvin + i ) ;
s = & S [ spendind ] ;
if ( s - > diffsequence = = 0 )
vin - > sequence = 0xffffffff ;
vin - > prev_vout = s - > prevout ;
iguana_ramchain_spendtxid ( coin , & vin - > prev_hash , T , rdata - > numtxids , X , rdata - > numexternaltxids , s ) ;
}
}
int32_t iguana_voutset ( struct iguana_info * coin , uint8_t * scriptspace , char * asmstr , int32_t height , struct iguana_msgvout * vout , struct iguana_txid * tx , int32_t i )
{
struct iguana_ramchaindata * rdata ; uint32_t unspentind , scriptlen = 0 ; struct iguana_bundle * bp ;
struct iguana_unspent * u , * U ; char coinaddr [ 65 ] ; struct iguana_pkhash * P , * p ; struct vin_info V ;
memset ( vout , 0 , sizeof ( * vout ) ) ;
if ( height > = 0 & & height < coin - > chain - > bundlesize * coin - > bundlescount & & ( bp = coin - > bundles [ height / coin - > chain - > bundlesize ] ) ! = 0 & & ( rdata = bp - > ramchain . H . data ) ! = 0 & & i < tx - > numvouts )
{
U = ( void * ) ( long ) ( ( long ) rdata + rdata - > Uoffset ) ;
P = ( void * ) ( long ) ( ( long ) rdata + rdata - > Poffset ) ;
unspentind = ( tx - > firstvout + i ) ;
u = & U [ unspentind ] ;
if ( u - > txidind ! = tx - > txidind | | u - > vout ! = i | | u - > hdrsi ! = height / coin - > chain - > bundlesize )
printf ( " iguana_voutset: txidind mismatch %d vs %d || %d vs %d || (%d vs %d) \n " , u - > txidind , u - > txidind , u - > vout , i , u - > hdrsi , height / coin - > chain - > bundlesize ) ;
p = & P [ u - > pkind ] ;
vout - > value = u - > value ;
vout - > pk_script = scriptspace ;
memset ( & V , 0 , sizeof ( V ) ) ;
scriptlen = iguana_scriptgen ( coin , & V . M , & V . N , coinaddr , scriptspace , asmstr , p - > rmd160 , u - > type , ( const struct vin_info * ) & V , i ) ;
}
vout - > pk_scriptlen = scriptlen ;
return ( scriptlen ) ;
}
struct iguana_txid * iguana_blocktx ( struct iguana_info * coin , struct iguana_txid * tx , struct iguana_block * block , int32_t i )
{
struct iguana_bundle * bp ; uint32_t txidind ;
if ( i > = 0 & & i < block - > RO . txn_count )
{
if ( block - > height > = 0 ) //
{
if ( ( bp = coin - > bundles [ block - > hdrsi ] ) ! = 0 )
{
if ( ( txidind = block - > RO . firsttxidind ) > 0 ) //bp->firsttxidinds[block->bundlei]) > 0 )
{
if ( iguana_bundletx ( coin , bp , block - > bundlei , tx , txidind + i ) = = tx )
return ( tx ) ;
printf ( " error getting txidind.%d + i.%d from hdrsi.%d \n " , txidind , i , block - > hdrsi ) ;
return ( 0 ) ;
} else printf ( " iguana_blocktx null txidind \n " ) ;
} else printf ( " iguana_blocktx no bp \n " ) ;
}
} else printf ( " i.%d vs txn_count.%d \n " , i , block - > RO . txn_count ) ;
return ( 0 ) ;
}
cJSON * iguana_blockjson ( struct iguana_info * coin , struct iguana_block * block , int32_t txidsflag )
{
char str [ 65 ] , hexstr [ 1024 ] ; int32_t i , len ; struct iguana_txid * tx , T ; struct iguana_msgblock msg ;
bits256 hash2 ; uint8_t serialized [ 1024 ] ; cJSON * array , * json = cJSON_CreateObject ( ) ;
jaddstr ( json , " result " , " success " ) ;
jaddstr ( json , " blockhash " , bits256_str ( str , block - > RO . hash2 ) ) ;
jaddnum ( json , " height " , block - > height ) ;
//jaddnum(json,"ipbits",block->fpipbits);
jaddstr ( json , " merkle_root " , bits256_str ( str , block - > RO . merkle_root ) ) ;
jaddstr ( json , " prev_block " , bits256_str ( str , block - > RO . prev_block ) ) ;
jaddnum ( json , " timestamp " , block - > RO . timestamp ) ;
jaddstr ( json , " utc " , utc_str ( str , block - > RO . timestamp ) ) ;
jaddnum ( json , " nonce " , block - > RO . nonce ) ;
jaddnum ( json , " version " , block - > RO . version ) ;
jaddnum ( json , " numvouts " , block - > RO . numvouts ) ;
jaddnum ( json , " numvins " , block - > RO . numvins ) ;
jaddnum ( json , " recvlen " , block - > RO . recvlen ) ;
jaddnum ( json , " hdrsi " , block - > hdrsi ) ;
jaddnum ( json , " PoW " , block - > PoW ) ;
jaddnum ( json , " bundlei " , block - > bundlei ) ;
jaddnum ( json , " mainchain " , block - > mainchain ) ;
jaddnum ( json , " valid " , block - > valid ) ;
jaddnum ( json , " txn_count " , block - > RO . txn_count ) ;
jaddnum ( json , " nBits " , block - > RO . bits ) ;
serialized [ 0 ] = ( ( uint8_t * ) & block - > RO . bits ) [ 3 ] ;
serialized [ 1 ] = ( ( uint8_t * ) & block - > RO . bits ) [ 2 ] ;
serialized [ 2 ] = ( ( uint8_t * ) & block - > RO . bits ) [ 1 ] ;
serialized [ 3 ] = ( ( uint8_t * ) & block - > RO . bits ) [ 0 ] ;
init_hexbytes_noT ( hexstr , serialized , sizeof ( uint32_t ) ) ;
jaddstr ( json , " nBitshex " , hexstr ) ;
memset ( & msg , 0 , sizeof ( msg ) ) ;
msg . H . version = block - > RO . version ;
msg . H . merkle_root = block - > RO . merkle_root ;
msg . H . timestamp = block - > RO . timestamp ;
msg . H . bits = block - > RO . bits ;
msg . H . nonce = block - > RO . nonce ;
msg . txn_count = 0 ; //block->RO.txn_count;
len = iguana_rwblock ( 1 , & hash2 , serialized , & msg ) ;
init_hexbytes_noT ( hexstr , serialized , len ) ;
jaddstr ( json , " blockheader " , hexstr ) ;
if ( txidsflag ! = 0 )
{
array = cJSON_CreateArray ( ) ;
for ( i = 0 ; i < block - > RO . txn_count ; i + + )
{
if ( ( tx = iguana_blocktx ( coin , & T , block , i ) ) ! = 0 )
jaddistr ( array , bits256_str ( str , tx - > txid ) ) ;
}
jadd ( json , " txids " , array ) ;
//printf("add txids[%d]\n",block->txn_count);
}
return ( json ) ;
}
/*
//char *hashstr,*txidstr,*coinaddr,*txbytes,rmd160str[41],str[65]; int32_t len,height,i,n,valid = 0;
//cJSON *addrs,*retjson,*retitem; uint8_t rmd160[20],addrtype; bits256 hash2,checktxid;
//memset(&hash2,0,sizeof(hash2)); struct iguana_txid *tx,T; struct iguana_block *block = 0;
if ( ( coinaddr = jstr ( json , " address " ) ) ! = 0 )
{
if ( btc_addr2univ ( & addrtype , rmd160 , coinaddr ) = = 0 )
{
if ( addrtype = = coin - > chain - > pubval | | addrtype = = coin - > chain - > p2shval )
valid = 1 ;
else return ( clonestr ( " { \" error \" : \" invalid addrtype \" } " ) ) ;
} else return ( clonestr ( " { \" error \" : \" cant convert address to rmd160 \" } " ) ) ;
}
if ( strcmp ( method , " block " ) = = 0 )
{
height = - 1 ;
if ( ( ( hashstr = jstr ( json , " blockhash " ) ) ! = 0 | | ( hashstr = jstr ( json , " hash " ) ) ! = 0 ) & & strlen ( hashstr ) = = sizeof ( bits256 ) * 2 )
decode_hex ( hash2 . bytes , sizeof ( hash2 ) , hashstr ) ;
else
{
height = juint ( json , " height " ) ;
hash2 = iguana_blockhash ( coin , height ) ;
}
retitem = cJSON_CreateObject ( ) ;
if ( ( block = iguana_blockfind ( coin , hash2 ) ) ! = 0 )
{
if ( ( height > = 0 & & block - > height = = height ) | | memcmp ( hash2 . bytes , block - > RO . hash2 . bytes , sizeof ( hash2 ) ) = = 0 )
{
char str [ 65 ] , str2 [ 65 ] ; printf ( " hash2.(%s) -> %s \n " , bits256_str ( str , hash2 ) , bits256_str ( str2 , block - > RO . hash2 ) ) ;
return ( jprint ( iguana_blockjson ( coin , block , juint ( json , " txids " ) ) , 1 ) ) ;
}
}
else return ( clonestr ( " { \" error \" : \" cant find block \" } " ) ) ;
}
else if ( strcmp ( method , " tx " ) = = 0 )
{
if ( ( ( txidstr = jstr ( json , " txid " ) ) ! = 0 | | ( txidstr = jstr ( json , " hash " ) ) ! = 0 ) & & strlen ( txidstr ) = = sizeof ( bits256 ) * 2 )
{
retitem = cJSON_CreateObject ( ) ;
decode_hex ( hash2 . bytes , sizeof ( hash2 ) , txidstr ) ;
if ( ( tx = iguana_txidfind ( coin , & height , & T , hash2 ) ) ! = 0 )
{
jadd ( retitem , " tx " , iguana_txjson ( coin , tx , height ) ) ;
return ( jprint ( retitem , 1 ) ) ;
}
return ( clonestr ( " { \" error \" : \" cant find txid \" } " ) ) ;
}
else return ( clonestr ( " { \" error \" : \" invalid txid \" } " ) ) ;
}
else if ( strcmp ( method , " rawtx " ) = = 0 )
{
if ( ( ( txidstr = jstr ( json , " txid " ) ) ! = 0 | | ( txidstr = jstr ( json , " hash " ) ) ! = 0 ) & & strlen ( txidstr ) = = sizeof ( bits256 ) * 2 )
{
decode_hex ( hash2 . bytes , sizeof ( hash2 ) , txidstr ) ;
if ( ( tx = iguana_txidfind ( coin , & height , & T , hash2 ) ) ! = 0 )
{
if ( ( len = iguana_txbytes ( coin , coin - > blockspace , sizeof ( coin - > blockspace ) , & checktxid , tx , height , 0 , 0 ) ) > 0 )
{
txbytes = mycalloc ( ' x ' , 1 , len * 2 + 1 ) ;
init_hexbytes_noT ( txbytes , coin - > blockspace , len * 2 + 1 ) ;
retitem = cJSON_CreateObject ( ) ;
jaddstr ( retitem , " txid " , bits256_str ( str , hash2 ) ) ;
jaddnum ( retitem , " height " , height ) ;
jaddstr ( retitem , " rawtx " , txbytes ) ;
myfree ( txbytes , len * 2 + 1 ) ;
return ( jprint ( retitem , 1 ) ) ;
} else return ( clonestr ( " { \" error \" : \" couldnt generate txbytes \" } " ) ) ;
}
return ( clonestr ( " { \" error \" : \" cant find txid \" } " ) ) ;
}
else return ( clonestr ( " { \" error \" : \" invalid txid \" } " ) ) ;
}
else if ( strcmp ( method , " txs " ) = = 0 )
{
if ( ( ( hashstr = jstr ( json , " block " ) ) ! = 0 | | ( hashstr = jstr ( json , " blockhash " ) ) ! = 0 ) & & strlen ( hashstr ) = = sizeof ( bits256 ) * 2 )
{
decode_hex ( hash2 . bytes , sizeof ( hash2 ) , hashstr ) ;
if ( ( block = iguana_blockfind ( coin , hash2 ) ) = = 0 )
return ( clonestr ( " { \" error \" : \" cant find blockhash \" } " ) ) ;
}
else if ( jobj ( json , " height " ) ! = 0 )
{
height = juint ( json , " height " ) ;
hash2 = iguana_blockhash ( coin , height ) ;
if ( ( block = iguana_blockfind ( coin , hash2 ) ) = = 0 )
return ( clonestr ( " { \" error \" : \" cant find block at height \" } " ) ) ;
}
else if ( valid = = 0 )
return ( clonestr ( " { \" error \" : \" txs needs blockhash or height or address \" } " ) ) ;
retitem = cJSON_CreateArray ( ) ;
if ( block ! = 0 )
{
for ( i = 0 ; i < block - > RO . txn_count ; i + + )
{
if ( ( tx = iguana_blocktx ( coin , & T , block , i ) ) ! = 0 )
jaddi ( retitem , iguana_txjson ( coin , tx , - 1 ) ) ;
}
}
else
{
init_hexbytes_noT ( rmd160str , rmd160 , 20 ) ;
jaddnum ( retitem , " addrtype " , addrtype ) ;
jaddstr ( retitem , " rmd160 " , rmd160str ) ;
jaddstr ( retitem , " txlist " , " get list of all tx for this address " ) ;
}
return ( jprint ( retitem , 1 ) ) ;
}
else
{
n = 0 ;
if ( valid = = 0 )
{
if ( ( addrs = jarray ( & n , json , " addrs " ) ) = = 0 )
return ( clonestr ( " { \" error \" : \" need address or addrs \" } " ) ) ;
}
for ( i = 0 ; i < = n ; i + + )
{
retitem = cJSON_CreateObject ( ) ;
if ( i > 0 )
retjson = cJSON_CreateArray ( ) ;
if ( i > 0 )
{
if ( ( coinaddr = jstr ( jitem ( addrs , i - 1 ) , 0 ) ) = = 0 )
return ( clonestr ( " { \" error \" : \" missing address in addrs \" } " ) ) ;
if ( btc_addr2univ ( & addrtype , rmd160 , coinaddr ) < 0 )
{
free_json ( retjson ) ;
return ( clonestr ( " { \" error \" : \" illegal address in addrs \" } " ) ) ;
}
if ( addrtype ! = coin - > chain - > pubval & & addrtype ! = coin - > chain - > p2shval )
return ( clonestr ( " { \" error \" : \" invalid addrtype in addrs \" } " ) ) ;
}
if ( strcmp ( method , " utxo " ) = = 0 )
{
jaddstr ( retitem , " utxo " , " utxo entry " ) ;
}
else if ( strcmp ( method , " unconfirmed " ) = = 0 )
{
jaddstr ( retitem , " unconfirmed " , " unconfirmed entry " ) ;
}
else if ( strcmp ( method , " balance " ) = = 0 )
{
jaddstr ( retitem , " balance " , " balance entry " ) ;
}
else if ( strcmp ( method , " totalreceived " ) = = 0 )
{
jaddstr ( retitem , " totalreceived " , " totalreceived entry " ) ;
}
else if ( strcmp ( method , " totalsent " ) = = 0 )
{
jaddstr ( retitem , " totalsent " , " totalsent entry " ) ;
}
else if ( strcmp ( method , " validateaddress " ) = = 0 )
{
jaddstr ( retitem , " validate " , coinaddr ) ;
}
if ( n = = 0 )
return ( jprint ( retitem , 1 ) ) ;
else jaddi ( retjson , retitem ) ;
}
return ( jprint ( retjson , 1 ) ) ;
}
*/
/*
char * iguana_listsinceblock ( struct supernet_info * myinfo , struct iguana_info * coin , bits256 blockhash , int32_t target )
{
cJSON * retitem = cJSON_CreateObject ( ) ;
return ( jprint ( retitem , 1 ) ) ;
}
char * iguana_getinfo ( struct supernet_info * myinfo , struct iguana_info * coin )
{
cJSON * retitem = cJSON_CreateObject ( ) ;
jaddstr ( retitem , " result " , coin - > statusstr ) ;
return ( jprint ( retitem , 1 ) ) ;
}
char * iguana_getbestblockhash ( struct supernet_info * myinfo , struct iguana_info * coin )
{
cJSON * retitem = cJSON_CreateObject ( ) ;
char str [ 65 ] ; jaddstr ( retitem , " result " , bits256_str ( str , coin - > blocks . hwmchain . RO . hash2 ) ) ;
return ( jprint ( retitem , 1 ) ) ;
}
char * iguana_getblockcount ( struct supernet_info * myinfo , struct iguana_info * coin )
{
cJSON * retitem = cJSON_CreateObject ( ) ;
jaddnum ( retitem , " result " , coin - > blocks . hwmchain . height ) ;
return ( jprint ( retitem , 1 ) ) ;
} */
struct bitcoin_unspent * iguana_bestfit ( struct iguana_info * coin , struct bitcoin_unspent * unspents , int32_t numunspents , uint64_t value , int32_t mode )
{
int32_t i ; uint64_t above , below , gap , atx_value ; struct bitcoin_unspent * vin , * abovevin , * belowvin ;
abovevin = belowvin = 0 ;
for ( above = below = i = 0 ; i < numunspents ; i + + )
{
vin = & unspents [ i ] ;
atx_value = vin - > value ;
//printf("(%.8f vs %.8f)\n",dstr(atx_value),dstr(value));
if ( atx_value = = value )
return ( vin ) ;
else if ( atx_value > value )
{
gap = ( atx_value - value ) ;
if ( above = = 0 | | gap < above )
{
above = gap ;
abovevin = vin ;
}
}
else if ( mode = = 0 )
{
gap = ( value - atx_value ) ;
if ( below = = 0 | | gap < below )
{
below = gap ;
belowvin = vin ;
}
}
}
if ( ( vin = ( abovevin ! = 0 ) ? abovevin : belowvin ) = = 0 & & mode = = 1 )
vin = unspents ;
return ( vin ) ;
}
struct bitcoin_unspent * iguana_unspentsget ( struct supernet_info * myinfo , struct iguana_info * coin , int32_t * numunspentsp )
{
struct bitcoin_unspent * ups = calloc ( 1 , sizeof ( * ups ) ) ; //uint8_t addrtype;
// struct bitcoin_unspent { bits256 txid,privkey; uint64_t value; int32_t vout; };
* numunspentsp = 0 ;
return ( ups ) ;
}
struct bitcoin_spend * iguana_spendset ( struct supernet_info * myinfo , struct iguana_info * coin , int64_t amount , int64_t txfee )
{
int32_t i , mode , numunspents , maxinputs = 1024 ; int64_t remains ; struct bitcoin_unspent * ptr , * up ;
struct bitcoin_unspent * ups ; struct bitcoin_spend * spend ;
if ( ( ups = iguana_unspentsget ( myinfo , coin , & numunspents ) ) = = 0 )
return ( 0 ) ;
spend = calloc ( 1 , sizeof ( * spend ) + sizeof ( * spend - > inputs ) * maxinputs ) ;
spend - > txfee = txfee ;
remains = txfee + amount ;
spend - > satoshis = remains ;
ptr = spend - > inputs ;
for ( i = 0 ; i < maxinputs ; i + + , ptr + + )
{
for ( mode = 1 ; mode > = 0 ; mode - - )
if ( ( up = iguana_bestfit ( coin , ups , numunspents , remains , mode ) ) ! = 0 )
break ;
if ( up ! = 0 )
{
spend - > input_satoshis + = up - > value ;
spend - > inputs [ spend - > numinputs + + ] = * up ;
if ( spend - > input_satoshis > = spend - > satoshis )
{
spend - > netamount = ( spend - > input_satoshis - spend - > satoshis ) ;
spend - > change = ( spend - > input_satoshis - spend - > netamount ) ;
printf ( " numinputs %d sum %.8f vs amount %.8f change %.8f -> txfee %.8f \n " , spend - > numinputs , dstr ( spend - > input_satoshis ) , dstr ( amount ) , dstr ( spend - > change ) , dstr ( spend - > input_satoshis - spend - > change - spend - > netamount ) ) ;
break ;
}
remains - = up - > value ;
} else break ;
}
if ( spend - > input_satoshis > = spend - > satoshis )
{
spend = realloc ( spend , sizeof ( * spend ) + sizeof ( * spend - > inputs ) * spend - > numinputs ) ;
return ( spend ) ;
}
else
{
free ( spend ) ;
return ( 0 ) ;
}
}
cJSON * iguana_signtx ( struct iguana_info * coin , bits256 * txidp , char * * signedtxp , struct bitcoin_spend * spend , cJSON * txobj )
{
int32_t i , j ; char * rawtxstr ; struct vin_info V ; bits256 txid ;
for ( i = 0 ; i < spend - > numinputs ; i + + )
bitcoin_addinput ( coin , txobj , spend - > inputs [ i ] . txid , spend - > inputs [ i ] . vout , 0xffffffff ) ;
for ( i = 0 ; i < spend - > numinputs ; i + + ) // N times less efficient, but for small number of inputs ok
{
if ( * signedtxp ! = 0 )
{
free_json ( txobj ) ;
txobj = bitcoin_hex2json ( coin , & txid , 0 , * signedtxp ) ;
free ( * signedtxp ) ;
}
if ( ( rawtxstr = bitcoin_json2hex ( coin , & txid , txobj ) ) ! = 0 )
{
memset ( & V , 0 , sizeof ( V ) ) ;
for ( j = 0 ; j < sizeof ( spend - > inputs [ i ] . privkeys ) / sizeof ( * spend - > inputs [ i ] . privkeys ) ; j + + )
if ( bits256_nonz ( spend - > inputs [ i ] . privkeys [ j ] ) > 0 )
V . signers [ j ] . privkey = spend - > inputs [ i ] . privkeys [ j ] ;
bitcoin_verifytx ( coin , txidp , signedtxp , rawtxstr , & V ) ;
free ( rawtxstr ) ;
} else break ;
}
if ( * signedtxp ! = 0 & & i ! = spend - > numinputs )
free ( * signedtxp ) , * signedtxp = 0 ;
return ( txobj ) ;
}