/******************************************************************************
* Copyright © 2014 - 2018 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 . *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
//
// LP_utxos.c
// marketmaker
//
int32_t LP_privkey_init ( int32_t mypubsock , struct iguana_info * coin , bits256 myprivkey , bits256 mypub )
{
int32_t enable_utxos = 0 ;
char * script , destaddr [ 64 ] ; cJSON * array , * item ; bits256 txid , deposittxid , zero ; int32_t used , i , flag = 0 , height , n , cmpflag , iambob , vout , depositvout ; uint64_t * values = 0 , satoshis , txfee , biggerval , value , total = 0 ; int64_t targetval ; //struct LP_utxoinfo *utxo;
if ( coin = = 0 | | ( IAMLP = = 0 & & coin - > inactive ! = 0 ) )
{
//printf("coin not active\n");
return ( 0 ) ;
}
if ( coin - > privkeydepth > 0 )
return ( 0 ) ;
coin - > privkeydepth + + ;
LP_address ( coin , coin - > smartaddr ) ;
//if ( coin->inactive == 0 )
// LP_listunspent_issue(coin->symbol,coin->smartaddr,0);
memset ( zero . bytes , 0 , sizeof ( zero ) ) ;
array = LP_listunspent ( coin - > symbol , coin - > smartaddr , zero , zero ) ;
if ( array ! = 0 )
{
txfee = LP_txfeecalc ( coin , 0 , 0 ) ;
if ( is_cJSON_Array ( array ) ! = 0 & & ( n = cJSON_GetArraySize ( array ) ) > 0 )
{
coin - > numutxos = n ;
//printf("LP_privkey_init %s %d\n",coin->symbol,n);
for ( iambob = 0 ; iambob < = 1 ; iambob + + )
{
if ( iambob = = 0 )
values = calloc ( n , sizeof ( * values ) ) ;
else memset ( values , 0 , n * sizeof ( * values ) ) ;
used = 0 ;
for ( i = 0 ; i < n ; i + + )
{
item = jitem ( array , i ) ;
value = LP_listunspent_parseitem ( coin , & txid , & vout , & height , item ) ;
satoshis = LP_txvalue ( destaddr , coin - > symbol , txid , vout ) ;
if ( satoshis ! = 0 & & satoshis ! = value )
printf ( " %s %s privkey_init value %.8f vs %.8f (%s) %.8f %.8f \n " , coin - > symbol , coin - > smartaddr , dstr ( satoshis ) , dstr ( value ) , jprint ( item , 0 ) , jdouble ( item , " amount " ) , jdouble ( item , " interest " ) ) ;
if ( coin - > electrum ! = 0 | | LP_inventory_prevent ( iambob , coin - > symbol , txid , vout ) = = 0 ) //&& height > 0 )
{
values [ i ] = satoshis ;
//flag += LP_address_utxoadd(coin,destaddr,txid,vout,satoshis,height,-1);
} else used + + ;
}
//printf("array.%d\n",n);
while ( used < n - 1 )
{
//for (i=0; i<n; i++)
// printf("%.8f ",dstr(values[i]));
//printf("used.%d of n.%d\n",used,n);
if ( ( i = LP_maxvalue ( values , n ) ) > = 0 )
{
item = jitem ( array , i ) ;
if ( coin - > electrum = = 0 )
{
deposittxid = jbits256 ( item , " txid " ) ;
depositvout = juint ( item , " vout " ) ;
script = jstr ( item , " scriptPubKey " ) ;
}
else
{
deposittxid = jbits256 ( item , " tx_hash " ) ;
depositvout = juint ( item , " tx_pos " ) ;
script = coin - > smartaddr ;
}
biggerval = values [ i ] ;
values [ i ] = 0 , used + + ;
if ( iambob = = 0 )
targetval = ( biggerval / 776 ) + txfee ;
else targetval = ( biggerval / 9 ) * 8 + 2 * txfee ;
if ( targetval < txfee * 2 )
targetval = txfee * 2 ;
//printf("iambob.%d i.%d deposit %.8f min %.8f target %.8f\n",iambob,i,dstr(biggerval),dstr((1+LP_MINSIZE_TXFEEMULT)*txfee),dstr(targetval));
if ( biggerval < ( 1 + LP_MINSIZE_TXFEEMULT ) * txfee )
continue ;
i = - 1 ;
if ( iambob ! = 0 )
{
if ( ( i = LP_nearestvalue ( iambob , values , n , targetval ) ) < 0 )
targetval / = 4 ;
if ( targetval < txfee * ( 1 + LP_MINSIZE_TXFEEMULT ) )
continue ;
}
if ( i > = 0 | | ( i = LP_nearestvalue ( iambob , values , n , targetval ) ) > = 0 )
{
//printf("iambob.%d i.%d %.8f target %.8f\n",iambob,i,dstr(biggerval),dstr(targetval));
item = jitem ( array , i ) ;
cmpflag = 0 ;
if ( coin - > electrum = = 0 )
{
txid = jbits256 ( item , " txid " ) ;
vout = juint ( item , " vout " ) ;
if ( jstr ( item , " scriptPubKey " ) ! = 0 & & strcmp ( script , jstr ( item , " scriptPubKey " ) ) = = 0 )
cmpflag = 1 ;
}
else
{
txid = jbits256 ( item , " tx_hash " ) ;
vout = juint ( item , " tx_pos " ) ;
cmpflag = 1 ;
}
if ( cmpflag ! = 0 )
{
value = values [ i ] ;
values [ i ] = 0 , used + + ;
/*portable_mutex_lock(&LP_UTXOmutex);
if ( iambob ! = 0 )
{
if ( ( utxo = LP_utxoadd ( 1 , coin - > symbol , txid , vout , value , deposittxid , depositvout , biggerval , coin - > smartaddr , mypub , LP_gui , G . LP_sessionid , value ) ) ! = 0 )
{
}
}
else
{
//printf("call utxoadd\n");
if ( ( utxo = LP_utxoadd ( 0 , coin - > symbol , deposittxid , depositvout , biggerval , txid , vout , value , coin - > smartaddr , mypub , LP_gui , G . LP_sessionid , biggerval ) ) ! = 0 )
{
}
}
portable_mutex_unlock ( & LP_UTXOmutex ) ; */
total + = value ;
} // else printf("scriptmismatch.(%s) vs %s\n",script,jprint(item,0));
} //else printf("nothing near i.%d\n",i);
} else break ;
}
if ( enable_utxos = = 0 )
break ;
}
}
free_json ( array ) ;
if ( 0 & & flag ! = 0 )
LP_postutxos ( coin - > symbol , coin - > smartaddr ) ;
}
if ( values ! = 0 )
free ( values ) ;
if ( coin - > privkeydepth > 0 )
coin - > privkeydepth - - ;
//printf("privkey.%s %.8f\n",symbol,dstr(total));
return ( flag ) ;
}
char * LP_secretaddresses ( void * ctx , char * prefix , char * passphrase , int32_t n , uint8_t taddr , uint8_t pubtype )
{
int32_t i ; uint8_t tmptype , pubkey33 [ 33 ] , rmd160 [ 20 ] ; char output [ 777 * 45 ] , str [ 65 ] , str2 [ 65 ] , buf [ 8192 ] , wifstr [ 128 ] , coinaddr [ 64 ] ; bits256 checkprivkey , privkey , pubkey ; cJSON * retjson ;
retjson = cJSON_CreateObject ( ) ;
if ( prefix = = 0 | | prefix [ 0 ] = = 0 )
prefix = " secretaddress " ;
if ( passphrase = = 0 | | passphrase [ 0 ] = = 0 )
passphrase = " password " ;
if ( n < = 0 )
n = 16 ;
else if ( n > 777 )
n = 777 ;
conv_NXTpassword ( privkey . bytes , pubkey . bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
bitcoin_priv2pub ( ctx , " KMD " , pubkey33 , coinaddr , privkey , taddr , pubtype ) ;
printf ( " generator (%s) secrets.[%d] <%s> t.%u p.%u \n " , coinaddr , n , passphrase , taddr , pubtype ) ;
sprintf ( output , " \" addresses \" :[ " ) ;
for ( i = 0 ; i < n ; i + + )
{
sprintf ( buf , " %s %s %03d " , prefix , passphrase , i ) ;
conv_NXTpassword ( privkey . bytes , pubkey . bytes , ( uint8_t * ) buf , ( int32_t ) strlen ( buf ) ) ;
bitcoin_priv2pub ( ctx , " KMD " , pubkey33 , coinaddr , privkey , taddr , pubtype ) ;
bitcoin_priv2wif ( " KMD " , 0 , wifstr , privkey , 188 ) ;
bitcoin_wif2priv ( " KMD " , 0 , & tmptype , & checkprivkey , wifstr ) ;
bitcoin_addr2rmd160 ( " KMD " , taddr , & tmptype , rmd160 , coinaddr ) ;
if ( bits256_cmp ( checkprivkey , privkey ) ! = 0 )
{
printf ( " WIF.(%s) error -> %s vs %s? \n " , wifstr , bits256_str ( str , privkey ) , bits256_str ( str2 , checkprivkey ) ) ;
free_json ( retjson ) ;
return ( clonestr ( " { \" error \" : \" couldnt validate wifstr \" } " ) ) ;
}
else if ( tmptype ! = pubtype )
{
printf ( " checktype.%d != pubtype.%d \n " , tmptype , pubtype ) ;
free_json ( retjson ) ;
return ( clonestr ( " { \" error \" : \" couldnt validate pubtype \" } " ) ) ;
}
jaddstr ( retjson , coinaddr , wifstr ) ;
sprintf ( output + strlen ( output ) , " \\ \" %s \\ \" %c " , coinaddr , i < n - 1 ? ' , ' : ' ' ) ;
printf ( " ./komodo-cli jumblr_secret %s \n " , coinaddr ) ;
}
printf ( " %s] \n " , output ) ;
return ( jprint ( retjson , 1 ) ) ;
}
uint32_t komodo_segid32 ( char * coinaddr )
{
bits256 addrhash ;
vcalc_sha256 ( 0 , ( uint8_t * ) & addrhash , ( uint8_t * ) coinaddr , ( int32_t ) strlen ( coinaddr ) ) ;
return ( addrhash . uints [ 0 ] ) ;
}
char * LP_gen64addrs ( void * ctx , char * passphrase , uint8_t taddr , uint8_t pubtype )
{
int32_t i , segid , n = 64 ; uint8_t tmptype , pubkey33 [ 33 ] , rmd160 [ 20 ] ; char str [ 65 ] , str2 [ 65 ] , buf [ 8192 ] , wifstr [ 64 ] , coinaddr [ 64 ] , coinaddrs [ 64 ] [ 64 ] , wifstrs [ 64 ] [ 64 ] ; uint64_t mask = 0 ; bits256 checkprivkey , privkey , pubkey ; cJSON * retjson , * addrs , * array ;
if ( passphrase = = 0 | | passphrase [ 0 ] = = 0 )
passphrase = " password " ;
memset ( coinaddrs , 0 , sizeof ( coinaddrs ) ) ;
memset ( wifstrs , 0 , sizeof ( wifstrs ) ) ;
conv_NXTpassword ( privkey . bytes , pubkey . bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
bitcoin_priv2pub ( ctx , " KMD " , pubkey33 , coinaddr , privkey , taddr , pubtype ) ;
//sprintf(output,"\"addresses\":[");
for ( i = 0 ; bitweight ( mask ) < 64 ; i + + )
{
sprintf ( buf , " %s %03d " , passphrase , i ) ;
conv_NXTpassword ( privkey . bytes , pubkey . bytes , ( uint8_t * ) buf , ( int32_t ) strlen ( buf ) ) ;
bitcoin_priv2pub ( ctx , " KMD " , pubkey33 , coinaddr , privkey , taddr , pubtype ) ;
bitcoin_priv2wif ( " KMD " , 0 , wifstr , privkey , 188 ) ;
bitcoin_wif2priv ( " KMD " , 0 , & tmptype , & checkprivkey , wifstr ) ;
bitcoin_addr2rmd160 ( " KMD " , taddr , & tmptype , rmd160 , coinaddr ) ;
if ( bits256_cmp ( checkprivkey , privkey ) ! = 0 )
{
printf ( " WIF.(%s) error -> %s vs %s? \n " , wifstr , bits256_str ( str , privkey ) , bits256_str ( str2 , checkprivkey ) ) ;
free_json ( retjson ) ;
return ( clonestr ( " { \" error \" : \" couldnt validate wifstr \" } " ) ) ;
}
else if ( tmptype ! = pubtype )
{
printf ( " checktype.%d != pubtype.%d \n " , tmptype , pubtype ) ;
free_json ( retjson ) ;
return ( clonestr ( " { \" error \" : \" couldnt validate pubtype \" } " ) ) ;
}
segid = komodo_segid32 ( coinaddr ) & 0x3f ;
if ( ( mask & ( 1LL < < segid ) ) = = 0 )
{
mask | = ( 1LL < < segid ) ;
strcpy ( coinaddrs [ segid ] , coinaddr ) ;
strcpy ( wifstrs [ segid ] , wifstr ) ;
printf ( " ./komodo-cli -ac_name=POSTEST64 importprivkey %s " " %s \n " , wifstr , bitweight ( mask ) < 64 ? " false " : " true " ) ;
}
//sprintf(output+strlen(output),"\\\"%s\\\"%c ",coinaddr,i<n-1?',':' ');
}
retjson = cJSON_CreateObject ( ) ;
addrs = cJSON_CreateObject ( ) ;
array = cJSON_CreateArray ( ) ;
for ( i = 0 ; i < n ; i + + )
jaddstr ( addrs , coinaddrs [ i ] , wifstrs [ i ] ) ;
jadd ( retjson , " addrpairs " , addrs ) ;
for ( i = 0 ; i < n ; i + + )
jaddistr ( array , coinaddrs [ i ] ) ;
jadd ( retjson , " addresses " , array ) ;
//printf("%s]\n",output);
return ( jprint ( retjson , 1 ) ) ;
}
static const char base58_chars [ ] = " 123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz " ;
int32_t LP_wifstr_valid ( char * symbol , char * wifstr )
{
bits256 privkey , cmpkey ; uint8_t wiftype ; char cmpstr [ 128 ] , cmpstr2 [ 128 ] ; int32_t i , len , n , a , A ;
if ( ( len = ( int32_t ) strlen ( wifstr ) ) < 50 | | len > 54 )
{
//printf("len.%d is wrong for wif %s\n",len,wifstr);
return ( 0 ) ;
}
memset ( privkey . bytes , 0 , sizeof ( privkey ) ) ;
memset ( cmpkey . bytes , 0 , sizeof ( cmpkey ) ) ;
for ( i = n = a = A = 0 ; wifstr [ i ] ! = 0 ; i + + )
{
if ( strchr ( base58_chars , wifstr [ i ] ) = = 0 )
return ( 0 ) ;
if ( wifstr [ i ] > = ' 1 ' & & wifstr [ i ] < = ' 9 ' )
n + + ;
else if ( wifstr [ i ] > = ' A ' & & wifstr [ i ] < = ' Z ' )
A + + ;
else if ( wifstr [ i ] > = ' a ' & & wifstr [ i ] < = ' z ' )
a + + ;
}
if ( n = = 0 | | A = = 0 | | a = = 0 )
return ( 0 ) ;
if ( A > 5 * a | | a > 5 * A | | a > n * 20 | | A > n * 20 ) // unlikely it is a real wif
{
printf ( " reject wif %s due to n.%d a.%d A.%d (%d %d %d %d) \n " , wifstr , n , a , A , A > 5 * a , a < 5 * A , a > n * 20 , A > n * 20 ) ;
return ( 0 ) ;
}
bitcoin_wif2priv ( symbol , 0 , & wiftype , & privkey , wifstr ) ;
bitcoin_priv2wif ( symbol , 0 , cmpstr , privkey , wiftype ) ;
if ( strcmp ( cmpstr , wifstr ) = = 0 )
{
//printf("%s is valid wif\n",wifstr);
return ( 1 ) ;
}
else if ( bits256_nonz ( privkey ) ! = 0 )
{
bitcoin_wif2priv ( symbol , 0 , & wiftype , & cmpkey , cmpstr ) ;
bitcoin_priv2wiflong ( symbol , 0 , cmpstr2 , privkey , wiftype ) ;
if ( bits256_cmp ( privkey , cmpkey ) = = 0 )
return ( 1 ) ;
char str [ 65 ] , str2 [ 65 ] ; printf ( " %s mismatched wifstr %s -> %s -> %s %s %s \n " , symbol , wifstr , bits256_str ( str , privkey ) , cmpstr , bits256_str ( str2 , cmpkey ) , cmpstr2 ) ;
}
char str [ 65 ] ; printf ( " %s is not a wif, privkey.%s \n " , wifstr , bits256_str ( str , privkey ) ) ;
return ( 0 ) ;
}
char * LP_convaddress ( char * symbol , char * address , char * dest )
{
struct iguana_info * coin , * destcoin ; cJSON * retjson ; char destaddress [ 64 ] , coinaddr2 [ 64 ] ; uint8_t addrtype , rmd160 [ 20 ] , rmd160b [ 20 ] ;
if ( ( coin = LP_coinfind ( symbol ) ) = = 0 | | ( destcoin = LP_coinfind ( dest ) ) = = 0 )
return ( clonestr ( " { \" error \" : \" both coins must be present \" } " ) ) ;
retjson = cJSON_CreateObject ( ) ;
jaddstr ( retjson , " result " , " success " ) ;
jaddstr ( retjson , " coin " , symbol ) ;
jaddstr ( retjson , " address " , address ) ;
jaddstr ( retjson , " destcoin " , dest ) ;
bitcoin_addr2rmd160 ( symbol , coin - > taddr , & addrtype , rmd160 , address ) ;
if ( addrtype = = coin - > pubtype )
{
bitcoin_address ( destcoin - > symbol , destaddress , destcoin - > taddr , destcoin - > pubtype , rmd160 , 20 ) ;
bitcoin_addr2rmd160 ( destcoin - > symbol , destcoin - > taddr , & addrtype , rmd160b , destaddress ) ;
bitcoin_address ( coin - > symbol , coinaddr2 , coin - > taddr , coin - > pubtype , rmd160b , 20 ) ;
}
else if ( addrtype = = coin - > p2shtype )
{
bitcoin_address ( destcoin - > symbol , destaddress , destcoin - > taddr , destcoin - > p2shtype , rmd160 , 20 ) ;
bitcoin_addr2rmd160 ( symbol , coin - > taddr , & addrtype , rmd160b , destaddress ) ;
bitcoin_address ( destcoin - > symbol , coinaddr2 , coin - > taddr , coin - > p2shtype , rmd160b , 20 ) ;
}
else
{
jaddstr ( retjson , " error " , " invalid base58 prefix " ) ;
jaddnum ( retjson , " invalid " , addrtype ) ;
}
if ( strcmp ( address , coinaddr2 ) ! = 0 )
{
jaddstr ( retjson , " error " , " checkaddress mismatch " ) ;
jaddstr ( retjson , " checkaddress " , coinaddr2 ) ;
}
jaddstr ( retjson , " destaddress " , destaddress ) ;
return ( jprint ( retjson , 1 ) ) ;
}
bits256 LP_privkeycalc ( void * ctx , uint8_t * pubkey33 , bits256 * pubkeyp , struct iguana_info * coin , char * passphrase , char * wifstr )
{
//static uint32_t counter;
bits256 privkey , userpub , zero , userpass , checkkey , tmpkey ; char str [ 65 ] , str2 [ 65 ] , tmpstr [ 128 ] ; cJSON * retjson ; uint8_t tmptype , sig [ 128 ] ; int32_t notarized , siglen ; uint64_t nxtaddr ;
if ( ( wifstr = = 0 | | wifstr [ 0 ] = = 0 ) & & LP_wifstr_valid ( coin - > symbol , passphrase ) > 0 )
{
wifstr = passphrase ;
passphrase = 0 ;
}
if ( passphrase ! = 0 & & passphrase [ 0 ] ! = 0 )
{
if ( strlen ( passphrase ) = = 66 & & passphrase [ 0 ] = = ' 0 ' & & passphrase [ 1 ] = = ' x ' & & is_hexstr ( passphrase + 2 , 0 ) = = 64 )
{
decode_hex ( privkey . bytes , 32 , passphrase + 2 ) ;
//printf("ETH style privkey.(%s)\n",passphrase);
}
else
{
calc_NXTaddr ( G . LP_NXTaddr , userpub . bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
conv_NXTpassword ( privkey . bytes , pubkeyp - > bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
privkey . bytes [ 0 ] & = 248 , privkey . bytes [ 31 ] & = 127 , privkey . bytes [ 31 ] | = 64 ;
}
bitcoin_priv2wif ( coin - > symbol , coin - > wiftaddr , tmpstr , privkey , coin - > wiftype ) ;
bitcoin_wif2priv ( coin - > symbol , coin - > wiftaddr , & tmptype , & checkkey , tmpstr ) ;
if ( bits256_cmp ( privkey , checkkey ) ! = 0 )
{
char str [ 65 ] , str2 [ 65 ] ; printf ( " mismatched privkeys from wif conversion: %s -> %s -> %s \n " , bits256_str ( str , privkey ) , tmpstr , bits256_str ( str2 , checkkey ) ) ;
exit ( 1 ) ;
}
}
else
{
bitcoin_wif2priv ( coin - > symbol , coin - > wiftaddr , & tmptype , & privkey , wifstr ) ;
bitcoin_priv2wif ( coin - > symbol , coin - > wiftaddr , tmpstr , privkey , tmptype ) ;
if ( strcmp ( tmpstr , wifstr ) ! = 0 )
{
bitcoin_wif2priv ( coin - > symbol , coin - > wiftaddr , & tmptype , & tmpkey , tmpstr ) ;
if ( bits256_cmp ( tmpkey , privkey ) ! = 0 )
{
char str [ 65 ] ; printf ( " %s error reproducing the wifstr, likely edge case like non-supported uncompressed pubkey privkey.%s \n " , coin - > symbol , bits256_str ( str , privkey ) ) ;
exit ( 1 ) ;
}
}
tmpkey = privkey ;
nxtaddr = conv_NXTpassword ( tmpkey . bytes , pubkeyp - > bytes , 0 , 0 ) ;
RS_encode ( G . LP_NXTaddr , nxtaddr ) ;
}
bitcoin_priv2pub ( ctx , coin - > symbol , coin - > pubkey33 , coin - > smartaddr , privkey , coin - > taddr , coin - > pubtype ) ;
# ifndef NOTETOMIC
if ( coin - > etomic [ 0 ] ! = 0 )
{
uint8_t check64 [ 64 ] , checktype , checkrmd160 [ 20 ] , rmd160 [ 20 ] ; char checkaddr [ 64 ] , checkaddr2 [ 64 ] ;
if ( LP_etomic_priv2pub ( check64 , privkey ) = = 0 )
{
if ( memcmp ( check64 , coin - > pubkey33 + 1 , 32 ) = = 0 )
{
if ( LP_etomic_priv2addr ( checkaddr , privkey ) = = 0 & & LP_etomic_pub2addr ( checkaddr2 , check64 ) = = 0 & & strcmp ( checkaddr , checkaddr2 ) = = 0 )
{
//printf("addr is (%s)\n",checkaddr);
strcpy ( coin - > smartaddr , checkaddr ) ;
decode_hex ( checkrmd160 , 20 , checkaddr + 2 ) ;
bitcoin_addr2rmd160 ( coin - > symbol , coin - > taddr , & checktype , rmd160 , checkaddr ) ;
if ( memcmp ( rmd160 , checkrmd160 , 20 ) ! = 0 )
printf ( " rmd160 doesnt match \n " ) ;
} else printf ( " error getting addr (%s) != (%s) \n " , checkaddr , checkaddr2 ) ;
} else printf ( " pubkey 64 mismatch \n " ) ;
} else printf ( " error creating pubkey \n " ) ;
}
# endif
OS_randombytes ( tmpkey . bytes , sizeof ( tmpkey ) ) ;
siglen = 0 ;
if ( bits256_nonz ( privkey ) = = 0 | | ( siglen = bitcoin_sign ( ctx , coin - > symbol , sig , tmpkey , privkey , 0 ) ) < = 0 )
{
printf ( " illegal privkey %s \n " , bits256_str ( str , privkey ) ) ;
exit ( 0 ) ;
}
if ( bits256_nonz ( privkey ) ! = 0 & & bitcoin_verify ( ctx , sig , siglen , tmpkey , coin - > pubkey33 , 33 ) ! = 0 )
{
printf ( " signature.[%d] for %s by %s didnt verify \n " , siglen , bits256_str ( str , tmpkey ) , bits256_str ( str2 , privkey ) ) ;
exit ( 0 ) ;
}
if ( coin - > counter = = 0 )
{
coin - > counter + + ;
memcpy ( G . LP_pubsecp , coin - > pubkey33 , 33 ) ;
bitcoin_priv2wif ( coin - > symbol , coin - > wiftaddr , tmpstr , privkey , coin - > wiftype ) ;
bitcoin_addr2rmd160 ( coin - > symbol , coin - > taddr , & tmptype , G . LP_myrmd160 , coin - > smartaddr ) ;
LP_privkeyadd ( privkey , G . LP_myrmd160 ) ;
G . LP_privkey = privkey ;
if ( G . counter + + = = 0 )
{
bitcoin_priv2wif ( coin - > symbol , coin - > wiftaddr , G . USERPASS_WIFSTR , privkey , 188 ) ;
bitcoin_wif2priv ( coin - > symbol , coin - > wiftaddr , & tmptype , & checkkey , G . USERPASS_WIFSTR ) ;
if ( bits256_cmp ( checkkey , privkey ) ! = 0 )
{
char str [ 65 ] , str2 [ 65 ] ;
printf ( " FATAL ERROR converting USERPASS_WIFSTR %s -> %s != %s \n " , G . USERPASS_WIFSTR , bits256_str ( str , checkkey ) , bits256_str ( str2 , privkey ) ) ;
exit ( - 1 ) ;
}
conv_NXTpassword ( userpass . bytes , pubkeyp - > bytes , ( uint8_t * ) G . USERPASS_WIFSTR , ( int32_t ) strlen ( G . USERPASS_WIFSTR ) ) ;
userpub = curve25519 ( userpass , curve25519_basepoint9 ( ) ) ;
printf ( " userpass.(%s) \n " , bits256_str ( G . USERPASS , userpub ) ) ;
}
}
if ( strcmp ( coin - > smartaddr , " RPZVpjptzfZnFZZoLnuSbfLexjtkhe6uvn " ) ! = 0 & & coin - > importedprivkey = = 0 & & coin - > electrum = = 0 & & coin - > userpass [ 0 ] ! = 0 & & LP_getheight ( & notarized , coin ) > 0 )
{
memset ( zero . bytes , 0 , sizeof ( zero ) ) ;
LP_listunspent_issue ( coin - > symbol , coin - > smartaddr , 0 , zero , zero ) ;
if ( ( retjson = LP_importprivkey ( coin - > symbol , tmpstr , coin - > smartaddr , - 1 ) ) ! = 0 )
{
if ( jobj ( retjson , " error " ) ! = 0 )
{
printf ( " cant importprivkey.%s %s -> (%s), abort session \n " , coin - > symbol , coin - > smartaddr , jprint ( retjson , 1 ) ) ;
exit ( - 1 ) ;
}
free_json ( retjson ) ;
}
coin - > importedprivkey = ( uint32_t ) time ( NULL ) ;
}
vcalc_sha256 ( 0 , checkkey . bytes , privkey . bytes , sizeof ( privkey ) ) ;
checkkey . bytes [ 0 ] & = 248 , checkkey . bytes [ 31 ] & = 127 , checkkey . bytes [ 31 ] | = 64 ;
G . LP_mypub25519 = * pubkeyp = curve25519 ( checkkey , curve25519_basepoint9 ( ) ) ;
G . LP_mypriv25519 = checkkey ;
LP_pubkeyadd ( G . LP_mypub25519 ) ;
return ( privkey ) ;
}
void verus_utxos ( struct iguana_info * coin , char * coinaddr )
{
cJSON * array , * item ; char buf [ 64 ] , str [ 65 ] ; int32_t i , m , vout , n = 0 ; bits256 txid ;
sprintf ( buf , " [%d, 99999999, [ \" %s \" ]] " , 1 , coinaddr ) ;
array = bitcoin_json ( coin , " listunspent " , buf ) ;
if ( array ! = 0 )
{
if ( ( n = cJSON_GetArraySize ( array ) ) > 0 )
{
for ( i = m = 0 ; i < n ; i + + )
{
item = jitem ( array , i ) ;
if ( fabs ( jdouble ( item , " amount " ) - 64. ) < 0.00011 )
{
txid = jbits256 ( item , " txid " ) ;
vout = jint ( item , " vout " ) ;
printf ( " %d: %s/v%d \n " , m , bits256_str ( str , txid ) , vout ) ;
m + + ;
}
}
}
free_json ( array ) ;
}
printf ( " scanned %d utxos m.%d \n " , n , m ) ;
}
char * verusblocks ( )
{
bits256 hash , txid ; uint8_t script [ 44 ] ; double value , avestakedsize , stakedval , RTu3sum , powsum , supply , possum , histo [ 1280 ] , myhisto [ 1280 ] ; int32_t num10 , num17 , num20 , num16 , num23000 , numpow , numpos , num , locked , height , i , m , n , z , numstaked , posflag , npos , npow ; char hashstr [ 64 ] , firstaddr [ 64 ] , stakingaddr [ 64 ] , * addr0 , * lastaddr , * hexstr ; cJSON * blockjson , * txobj , * vouts , * vout , * vout1 , * sobj , * addresses , * txs ;
struct iguana_info * coin = LP_coinfind ( " VRSC " ) ;
if ( coin = = 0 )
return ( clonestr ( " { \" error \" : \" VRSC not active \" } " ) ) ;
char * coinaddr = " RHV2As4rox97BuE3LK96vMeNY8VsGRTmBj " ;
if ( strcmp ( coinaddr , coin - > smartaddr ) ! = 0 )
return ( clonestr ( " { \" error \" : \" mismatched smartaddr \" } " ) ) ;
//verus_utxos(coin,coin->smartaddr);
hash = LP_getbestblockhash ( coin ) ;
memset ( histo , 0 , sizeof ( histo ) ) ;
memset ( myhisto , 0 , sizeof ( myhisto ) ) ;
num23000 = num16 = num17 = num10 = num20 = numstaked = 0 ;
avestakedsize = possum = powsum = supply = RTu3sum = 0. ;
numpow = numpos = num = npos = npow = 0 ;
if ( bits256_nonz ( hash ) ! = 0 )
{
bits256_str ( hashstr , hash ) ;
height = - 1 ;
while ( ( blockjson = LP_blockjson ( & height , coin - > symbol , hashstr , 0 ) ) ! = 0 )
{
num + + ;
stakedval = 0. ;
height = juint ( blockjson , " height " ) ;
if ( ( txs = jarray ( & n , blockjson , " tx " ) ) ! = 0 )
{
txid = jbits256i ( txs , 0 ) ;
value = 0 ;
posflag = 0 ;
locked = 0 ;
lastaddr = addr0 = " " ;
memset ( script , 0 , sizeof ( script ) ) ;
memset ( firstaddr , 0 , sizeof ( firstaddr ) ) ;
memset ( stakingaddr , 0 , sizeof ( stakingaddr ) ) ;
if ( ( txobj = LP_gettx ( " verus " , coin - > symbol , txid , 0 ) ) ! = 0 )
{
//printf("TX.(%s)\n",jprint(txobj,0));
if ( ( vouts = jarray ( & m , txobj , " vout " ) ) ! = 0 )
{
if ( ( vout = jitem ( vouts , 0 ) ) ! = 0 )
{
value = jdouble ( vout , " value " ) ;
supply + = value ;
hexstr = 0 ;
if ( m = = 2 & & ( vout1 = jitem ( vouts , 1 ) ) ! = 0 )
{
// 6a2001039bbc0bb17576a9149a3af738444dd86b55c86752247aec2e7deb842688ac
if ( jdouble ( vout1 , " value " ) = = 0. & & ( sobj = jobj ( vout1 , " scriptPubKey " ) ) ! = 0 & & ( hexstr = jstr ( sobj , " hex " ) ) ! = 0 & & strlen ( hexstr ) < = 88 )
{
if ( strlen ( hexstr ) = = 68 )
{
decode_hex ( script , 34 , hexstr ) ;
bitcoin_address ( coin - > symbol , firstaddr , coin - > taddr , coin - > pubtype , & script [ 12 ] , 20 ) ;
//printf("%s\n",&hexstr[24]);
}
else
{
decode_hex ( script , 44 , hexstr ) ;
bitcoin_address ( coin - > symbol , firstaddr , coin - > taddr , coin - > pubtype , & script [ 10 ] , 33 ) ;
}
locked = ( ( int32_t ) script [ 6 ] < < 16 ) + ( ( int32_t ) script [ 5 ] < < 8 ) + script [ 4 ] ;
addr0 = firstaddr ;
} else printf ( " unexpected vout1.(%s) (%s).%d %.8f \n " , jprint ( vout1 , 0 ) , hexstr ! = 0 ? hexstr : " " , ( int32_t ) strlen ( hexstr ) , jdouble ( vout1 , " value " ) ) ;
} else printf ( " coinbase without opret (%s) \n " , jprint ( vouts , 0 ) ) ;
}
}
free_json ( txobj ) ;
}
if ( n > 1 & & ( txobj = LP_gettx ( " verus " , coin - > symbol , jbits256i ( txs , n - 1 ) , 0 ) ) ! = 0 )
{
if ( ( vouts = jarray ( & m , txobj , " vout " ) ) ! = 0 )
{
if ( ( vout = jitem ( vouts , 0 ) ) ! = 0 & & m = = 1 )
{
if ( ( sobj = jobj ( vout , " scriptPubKey " ) ) ! = 0 & & ( addresses = jarray ( & z , sobj , " addresses " ) ) ! = 0 )
{
lastaddr = jstri ( addresses , 0 ) ;
if ( lastaddr = = 0 )
lastaddr = " " ;
else
{
strcpy ( stakingaddr , lastaddr ) ;
stakedval = jdouble ( vout , " value " ) ;
avestakedsize + = stakedval ;
numstaked + + ;
//printf("stakedval %f\n",stakedval);
posflag = 1 ;
//printf("ht.%d found staking address.(%s) %.8f (%s)\n",height,stakingaddr,stakedval,jprint(vout,0));
}
} else printf ( " no addresses[0] in (%s) %s \n " , jprint ( vout , 0 ) , sobj ! = 0 ? jprint ( sobj , 0 ) : " " ) ;
} //else printf("n.%d m.%d no first out in lastvout.(%s)\n",n,m,jprint(txobj,0));
} // else printf("cant find vout.(%s)\n",jprint(txobj,0));
free_json ( txobj ) ;
}
if ( posflag ! = 0 )
{
numpos + + ;
if ( strcmp ( coinaddr , stakingaddr ) = = 0 | | strcmp ( " RTu3JZZKLJTcfNwBa19dWRagEfQq49STqC " , stakingaddr ) = = 0 )
{
if ( strcmp ( " RTu3JZZKLJTcfNwBa19dWRagEfQq49STqC " , stakingaddr ) = = 0 )
RTu3sum + = value ;
possum + = value , npos + + ;
if ( num < 1500 )
printf ( " ht.%-5d lock.%-7d PoS cb.(%s) stake.(%s) %.8f %.8f \n " , height , locked , addr0 , stakingaddr , value , stakedval ) ;
if ( height > 23000 )
{
char strbuf [ 64 ] ;
sprintf ( strbuf , " %.0f " , stakedval ) ;
if ( strcmp ( strbuf , " 20 " ) = = 0 )
num20 + + ;
else if ( strcmp ( strbuf , " 17 " ) = = 0 )
num17 + + ;
else if ( strcmp ( strbuf , " 16 " ) = = 0 )
num16 + + ;
else if ( strcmp ( strbuf , " 10 " ) = = 0 )
num10 + + ;
else printf ( " got strbuf.(%s) \ n " ,strbuf) ;
}
}
else if ( 0 & & num < 100 )
printf ( " ht.%-5d lock.%-7d PoS cb.(%s) stake.(%s) %.8f %.8f \n " , height , locked , addr0 , stakingaddr , value , stakedval ) ;
}
else
{
numpow + + ;
if ( num < 100 & & strcmp ( coinaddr , addr0 ) = = 0 )
printf ( " ht.%-5d lock.%-7d PoW coinbase.(%s) %.8f \n " , height , locked , addr0 , value ) ;
if ( strcmp ( coinaddr , addr0 ) = = 0 )
powsum + = value , npow + + ;
}
histo [ locked / 1000 ] + = value ;
if ( strcmp ( coinaddr , addr0 ) = = 0 | | strcmp ( " RTu3JZZKLJTcfNwBa19dWRagEfQq49STqC " , addr0 ) = = 0 )
myhisto [ locked / 1000 ] + = value ;
}
bits256_str ( hashstr , jbits256 ( blockjson , " previousblockhash " ) ) ;
free_json ( blockjson ) ;
if ( height = = 5040 )
break ;
else if ( height = = 23000 )
{
num23000 = num ;
printf ( " num10.%d num16.%d num17.%d num20.%d / num23000.%d -> %.2f%% %.2f%% %.2f%% %.2f%% [%.3f %.3f %.3f %.3f] %.3f ave %.8f \n " , num10 , num16 , num17 , num20 , num23000 , 100. * ( double ) num10 / num23000 , 100. * ( double ) num16 / num23000 , 100. * ( double ) num17 / num23000 , 100. * ( double ) num20 / num23000 , ( 100. * ( double ) num10 / num23000 ) / 2.87 , ( 100. * ( double ) num16 / num23000 ) / 10.5 , ( 100. * ( double ) num17 / num23000 ) / 4.88 , ( 100. * ( double ) num20 / num23000 ) / 5.74 , ( 100. * ( double ) ( num10 + num16 + num17 + num20 ) / num23000 ) / 24 , avestakedsize / numstaked ) ;
}
else if ( ( num % 1000 ) = = 0 | | ( num < 1000 & & ( num % 100 ) = = 0 ) )
{
printf ( " num.%d PoW %.2f%% %.0f %d v %d PoS %.2f%% %.0f -> %.0f supply %.0f PoW %.1f%% PoS %.1f%% both %.1f%% RTu3 %.8f %.1f%% \n " , num , 100. * ( double ) numpow / num , powsum , npow , npos , 100. * ( double ) numpos / num , possum , powsum + possum , supply , 100. * powsum / supply , 100. * possum / supply , 100. * ( powsum + possum ) / supply , RTu3sum , 100. * RTu3sum / supply ) ;
}
}
}
if ( num > 0 )
{
if ( 0 )
{
for ( i = 0 ; i < sizeof ( histo ) / sizeof ( * histo ) ; i + + )
if ( histo [ i ] ! = 0 )
printf ( " %d %.8f, " , i * 1000 , histo [ i ] ) ;
printf ( " timelocked \n " ) ;
for ( i = 0 ; i < sizeof ( myhisto ) / sizeof ( * myhisto ) ; i + + )
if ( myhisto [ i ] ! = 0 )
printf ( " %d %.8f, " , i * 1000 , myhisto [ i ] ) ;
printf ( " mytimelocked \n " ) ;
}
printf ( " num.%d PoW %.2f%% %.8f %d v %d PoS %.2f%% %.8f -> %.8f supply %.8f PoW %.1f%% PoS %.1f%% both %.1f%% RTu3sum %.8f %.1f%% \n " , num , 100. * ( double ) numpow / num , powsum , npow , npos , 100. * ( double ) numpos / num , possum , powsum + possum , supply , 100. * powsum / supply , 100. * possum / supply , 100. * ( powsum + possum ) / supply , RTu3sum , 100. * RTu3sum / supply ) ;
printf ( " num10.%d num16.%d num17.%d num20.%d / num23000.%d -> %.2f%% %.2f%% %.2f%% %.2f%% [%.3f %.3f %.3f %.3f] %.3f ave %.8f \n " , num10 , num16 , num17 , num20 , num23000 , 100. * ( double ) num10 / num23000 , 100. * ( double ) num16 / num23000 , 100. * ( double ) num17 / num23000 , 100. * ( double ) num20 / num23000 , ( 100. * ( double ) num10 / num23000 ) / 2.87 , ( 100. * ( double ) num16 / num23000 ) / 10.5 , ( 100. * ( double ) num17 / num23000 ) / 4.88 , ( 100. * ( double ) num20 / num23000 ) / 5.74 , ( 100. * ( double ) ( num10 + num16 + num17 + num20 ) / num23000 ) / 24 , avestakedsize / numstaked ) ;
}
return ( clonestr ( " { \" result \" : \" success \" } " ) ) ;
}
void LP_privkey_updates ( void * ctx , int32_t pubsock , char * passphrase )
{
struct iguana_info * coin , * tmp ; bits256 pubkey , privkey ; uint8_t pubkey33 [ 33 ] ; int32_t initonly ;
initonly = ( passphrase ! = 0 ) ;
memset ( privkey . bytes , 0 , sizeof ( privkey ) ) ;
memset ( pubkey . bytes , 0 , sizeof ( pubkey ) ) ;
//printf("Total coins: %d\n", HASH_COUNT(LP_coins));
//int num_iter = 0;
HASH_ITER ( hh , LP_coins , coin , tmp )
{
//printf("LP_privkey_updates [%02d / %02d]\n", num_iter++, HASH_COUNT(LP_coins));
if ( initonly ! = 0 )
{
coin - > counter = 0 ;
memset ( coin - > smartaddr , 0 , sizeof ( coin - > smartaddr ) ) ;
if ( bits256_nonz ( privkey ) = = 0 | | coin - > smartaddr [ 0 ] = = 0 )
{
privkey = LP_privkeycalc ( ctx , pubkey33 , & pubkey , coin , passphrase , " " ) ;
}
}
//printf("i.%d of %d\n",i,LP_numcoins);
else if ( IAMLP = = 0 | | coin - > inactive = = 0 )
{
//printf("from updates %s\n",coin->symbol);
if ( 0 & & LP_privkey_init ( pubsock , coin , G . LP_privkey , G . LP_mypub25519 ) = = 0 & & ( LP_rand ( ) % 10 ) = = 0 )
{
//LP_postutxos(coin->symbol,coin->smartaddr);
}
}
}
}
int32_t LP_passphrase_init ( char * passphrase , char * gui , uint16_t netid , char * seednode )
{
static void * ctx ; struct iguana_info * coin , * tmp ; int32_t counter ;
if ( ctx = = 0 )
ctx = bitcoin_ctx ( ) ;
if ( G . LP_pendingswaps ! = 0 )
return ( - 1 ) ;
if ( netid ! = G . netid )
{
if ( IAMLP ! = 0 )
{
printf ( " sorry, LP nodes can only set netid during startup \n " ) ;
return ( - 1 ) ;
}
else
{
printf ( " >>>>>>>>>>>>> netid.%d vs G.netid %d \n " , netid , G . netid ) ;
LP_closepeers ( ) ;
LP_initpeers ( LP_mypubsock , LP_mypeer , LP_myipaddr , RPC_port , netid , seednode ) ;
}
}
G . initializing = 1 ;
if ( gui = = 0 )
gui = " cli " ;
counter = G . USERPASS_COUNTER ;
HASH_ITER ( hh , LP_coins , coin , tmp )
{
coin - > importedprivkey = 0 ;
}
while ( G . waiting = = 0 )
{
printf ( " waiting for G.waiting \n " ) ;
sleep ( 5 ) ;
}
memset ( & G , 0 , sizeof ( G ) ) ;
G . netid = netid ;
safecopy ( G . seednode , seednode , sizeof ( G . seednode ) ) ;
vcalc_sha256 ( 0 , G . LP_passhash . bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
LP_privkey_updates ( ctx , LP_mypubsock , passphrase ) ;
init_hexbytes_noT ( G . LP_myrmd160str , G . LP_myrmd160 , 20 ) ;
G . LP_sessionid = ( uint32_t ) time ( NULL ) ;
safecopy ( G . gui , gui , sizeof ( G . gui ) ) ;
LP_tradebot_pauseall ( ) ;
LP_portfolio_reset ( ) ;
LP_priceinfos_clear ( ) ;
G . USERPASS_COUNTER = counter ;
G . initializing = 0 ;
//LP_cmdchannels();
return ( 0 ) ;
}
void LP_privkey_tests ( )
{
char wifstr [ 64 ] , str [ 65 ] , str2 [ 65 ] ; bits256 privkey , checkkey ; int32_t i ; uint8_t tmptype ;
for ( i = 0 ; i < 200000000 ; i + + )
{
privkey = rand256 ( 0 ) ;
bitcoin_priv2wif ( " KMD " , 0 , wifstr , privkey , 0xff ) ;
bitcoin_wif2priv ( " KMD " , 0 , & tmptype , & checkkey , wifstr ) ;
if ( bits256_cmp ( privkey , checkkey ) ! = 0 )
{
printf ( " i.%d: %s vs %s \n " , i , bits256_str ( str , privkey ) , bits256_str ( str2 , checkkey ) ) ;
exit ( - 1 ) ;
}
if ( ( i % 1000000 ) = = 0 )
fprintf ( stderr , " %.1f%% " , 100. * ( double ) i / 200000000 ) ;
}
printf ( " %d privkeys checked \n " , i ) ;
}
# define JPG_ENCRYPTED_MAXSIZE 32768
int32_t JPG_encrypt ( uint16_t ind , uint8_t encoded [ JPG_ENCRYPTED_MAXSIZE ] , uint8_t * msg , int32_t msglen , bits256 privkey )
{
bits256 pubkey ; int32_t len = 2 ; uint8_t space [ JPG_ENCRYPTED_MAXSIZE ] , * nonce , * cipher ;
pubkey = acct777_pubkey ( privkey ) ;
encoded [ len + + ] = ind & 0xff ;
encoded [ len + + ] = ( ind > > 8 ) & 0xff ;
nonce = & encoded [ len ] ;
OS_randombytes ( nonce , crypto_box_NONCEBYTES ) ;
cipher = & encoded [ len + crypto_box_NONCEBYTES ] ;
msglen = _SuperNET_cipher ( nonce , & encoded [ len + crypto_box_NONCEBYTES ] , msg , msglen , pubkey , privkey , space ) ;
msglen + = crypto_box_NONCEBYTES ;
msg = encoded ;
msglen + = len ;
encoded [ 0 ] = msglen & 0xff ;
encoded [ 1 ] = ( msglen > > 8 ) & 0xff ;
int32_t i ; for ( i = 0 ; i < msglen ; i + + )
printf ( " %02x " , encoded [ i ] ) ;
printf ( " encoded.%d \n " , msglen ) ;
return ( msglen ) ;
}
uint8_t * JPG_decrypt ( uint16_t * indp , int32_t * recvlenp , uint8_t space [ JPG_ENCRYPTED_MAXSIZE + crypto_box_ZEROBYTES ] , uint8_t * encoded , bits256 privkey )
{
bits256 pubkey ; uint8_t * extracted = 0 , * nonce , * cipher ; uint16_t msglen , ind ; int32_t cipherlen , len = 4 ;
* recvlenp = 0 ;
* indp = - 1 ;
pubkey = acct777_pubkey ( privkey ) ;
msglen = ( ( int32_t ) encoded [ 1 ] < < 8 ) | encoded [ 0 ] ;
ind = ( ( int32_t ) encoded [ 3 ] < < 8 ) | encoded [ 2 ] ;
nonce = & encoded [ len ] ;
cipher = & encoded [ len + crypto_box_NONCEBYTES ] ;
cipherlen = msglen - ( len + crypto_box_NONCEBYTES ) ;
if ( cipherlen > 0 & & cipherlen < = JPG_ENCRYPTED_MAXSIZE + crypto_box_ZEROBYTES )
{
//int32_t i; for (i=0; i<cipherlen; i++)
// printf("%02x",cipher[i]);
//printf(" cipherlen\n");
if ( ( extracted = _SuperNET_decipher ( nonce , cipher , space , cipherlen , pubkey , privkey ) ) ! = 0 )
{
//int32_t i; for (i=0; i<msglen; i++)
// printf("%02x",encoded[i]);
//printf(" restored\n");
msglen = ( cipherlen - crypto_box_ZEROBYTES ) ;
* recvlenp = msglen ;
* indp = ind ;
}
} //else printf("cipher.%d too big for %d\n",cipherlen,JPG_ENCRYPTED_MAXSIZE + crypto_box_ZEROBYTES);
return ( extracted ) ;
}
int32_t LP_opreturn_decrypt ( uint16_t * ind16p , uint8_t * decoded , uint8_t * encoded , int32_t encodedlen , char * passphrase )
{
bits256 privkey ; int32_t msglen ; uint8_t * extracted , space [ JPG_ENCRYPTED_MAXSIZE + crypto_box_ZEROBYTES ] ;
if ( passphrase ! = 0 & & passphrase [ 0 ] ! = 0 )
{
vcalc_sha256 ( 0 , privkey . bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
msglen = ( ( int32_t ) encoded [ 1 ] < < 8 ) | encoded [ 0 ] ;
* ind16p = ( ( int32_t ) encoded [ 3 ] < < 8 ) | encoded [ 2 ] ;
if ( msglen = = encodedlen & & ( extracted = JPG_decrypt ( ind16p , & msglen , space , encoded , privkey ) ) ! = 0 )
{
memcpy ( decoded , extracted , msglen ) ;
return ( msglen ) ;
} else return ( - 1 ) ;
}
else
{
* ind16p = calc_crc32 ( 0 , extracted , encodedlen ) ;
memcpy ( decoded , extracted , encodedlen ) ;
return ( encodedlen ) ;
}
}
int32_t LP_opreturn_encrypt ( uint8_t * dest , int32_t maxsize , uint8_t * data , int32_t datalen , char * passphrase , uint16_t ind16 )
{
bits256 privkey ; int32_t len ; uint8_t encoded [ JPG_ENCRYPTED_MAXSIZE ] ;
vcalc_sha256 ( 0 , privkey . bytes , ( uint8_t * ) passphrase , ( int32_t ) strlen ( passphrase ) ) ;
if ( ( len = JPG_encrypt ( ind16 , encoded , data , datalen , privkey ) ) > 0 )
{
//printf("datalen.%d -> len.%d max.%d\n",datalen,len,maxsize);
if ( len < = maxsize )
{
memcpy ( dest , encoded , len ) ;
return ( len ) ;
}
}
return ( - 1 ) ;
}
// from https://github.com/owencm/C-Steganography-Framework
# include "../../crypto777/jpeg/cdjpeg.h" // Common decls for compressing and decompressing jpegs
int32_t LP_jpg_process ( int32_t * capacityp , char * inputfname , char * outputfname , uint8_t * decoded , uint8_t * origdata , int32_t origrequired , int32_t power2 , char * password , uint16_t * indp )
{
struct jpeg_decompress_struct inputinfo ;
struct jpeg_compress_struct outputinfo ;
struct jpeg_error_mgr jerr ;
jvirt_barray_ptr * coef_arrays ;
JDIMENSION i , compnum , rownum , blocknum ;
JBLOCKARRAY coef_buffers [ MAX_COMPONENTS ] ;
JBLOCKARRAY row_ptrs [ MAX_COMPONENTS ] ;
bits256 privkey ; FILE * input_file , * output_file ; int32_t recvlen , msglen , val , modified , emit , totalrows , limit , required ; uint16_t checkind ; uint8_t * decrypted , * space , * data = 0 ;
if ( ( input_file = fopen ( inputfname , READ_BINARY ) ) = = NULL )
{
fprintf ( stderr , " Can't open %s \n " , inputfname ) ;
//exit(EXIT_FAILURE);
return ( - 1 ) ;
}
required = origrequired ;
memset ( privkey . bytes , 0 , sizeof ( privkey ) ) ;
if ( password ! = 0 & & password [ 0 ] ! = 0 )
{
if ( required / 8 > JPG_ENCRYPTED_MAXSIZE - 60 )
return ( - 1 ) ;
data = calloc ( 1 , required / 8 + 512 ) ;
vcalc_sha256 ( 0 , privkey . bytes , ( uint8_t * ) password , ( int32_t ) strlen ( password ) ) ;
if ( origdata ! = 0 )
{
msglen = JPG_encrypt ( * indp , data , origdata , required / 8 , privkey ) ;
required = msglen * 8 ;
{
space = calloc ( 1 , JPG_ENCRYPTED_MAXSIZE ) ;
if ( ( decrypted = JPG_decrypt ( & checkind , & recvlen , space , data , privkey ) ) = = 0 | | recvlen ! = origrequired / 8 | | checkind ! = * indp | | memcmp ( decrypted , origdata , origrequired / 8 ) ! = 0 )
printf ( " A decryption error: checkind.%d vs %d, recvlen.%d vs %d, decrypted.%p \n " , checkind , * indp , recvlen , origrequired / 8 , decrypted ) ;
else
{
for ( i = 0 ; i < recvlen ; i + + )
printf ( " %02x " , decrypted [ i ] ) ;
printf ( " VERIFIED decryption.%d ind.%d msglen.%d required.%d \n " , recvlen , * indp , msglen , required ) ;
}
free ( space ) ;
}
} else required + = 60 * 8 ;
} else data = origdata ;
if ( power2 < 0 | | power2 > 30 )
power2 = 7 ;
limit = 1 ;
while ( power2 > 0 )
{
limit < < = 1 ;
power2 - - ;
}
// Initialize the JPEG compression and decompression objects with default error handling
inputinfo . err = jpeg_std_error ( & jerr ) ;
jpeg_create_decompress ( & inputinfo ) ;
// Specify data source for decompression and recompression
jpeg_stdio_src ( & inputinfo , input_file ) ;
( void ) jpeg_read_header ( & inputinfo , TRUE ) ;
for ( compnum = 0 ; compnum < inputinfo . num_components ; compnum + + )
coef_buffers [ compnum ] = ( ( & inputinfo ) - > mem - > alloc_barray ) ( ( j_common_ptr ) & inputinfo , JPOOL_IMAGE , inputinfo . comp_info [ compnum ] . width_in_blocks , inputinfo . comp_info [ compnum ] . height_in_blocks ) ;
coef_arrays = jpeg_read_coefficients ( & inputinfo ) ;
// Copy DCT coeffs to a new array
int num_components = inputinfo . num_components ;
size_t * block_row_size ; //[num_components];
int * width_in_blocks ; //[num_components];
int * height_in_blocks ; //[num_components];
block_row_size = calloc ( sizeof ( * block_row_size ) , num_components ) ;
width_in_blocks = calloc ( sizeof ( * width_in_blocks ) , num_components ) ;
height_in_blocks = calloc ( sizeof ( * height_in_blocks ) , num_components ) ;
* capacityp = modified = emit = totalrows = 0 ;
if ( decoded ! = 0 )
memset ( decoded , 0 , required / 8 + 1 ) ;
for ( compnum = 0 ; compnum < num_components ; compnum + + )
{
height_in_blocks [ compnum ] = inputinfo . comp_info [ compnum ] . height_in_blocks ;
width_in_blocks [ compnum ] = inputinfo . comp_info [ compnum ] . width_in_blocks ;
block_row_size [ compnum ] = ( size_t ) SIZEOF ( JCOEF ) * DCTSIZE2 * width_in_blocks [ compnum ] ;
for ( rownum = 0 ; rownum < height_in_blocks [ compnum ] ; rownum + + )
{
row_ptrs [ compnum ] = ( ( & inputinfo ) - > mem - > access_virt_barray ) ( ( j_common_ptr ) & inputinfo , coef_arrays [ compnum ] , rownum , ( JDIMENSION ) 1 , FALSE ) ;
for ( blocknum = 0 ; blocknum < width_in_blocks [ compnum ] ; blocknum + + )
{
for ( i = 0 ; i < DCTSIZE2 ; i + + )
{
val = row_ptrs [ compnum ] [ 0 ] [ blocknum ] [ i ] ;
if ( val < - limit | | val > = limit )
{
if ( ( * capacityp ) < required )
{
if ( ( val & 1 ) ! = 0 )
SETBIT ( decoded , ( * capacityp ) ) ;
//printf("%c",(val&1)!=0?'1':'0');
}
( * capacityp ) + + ;
}
coef_buffers [ compnum ] [ rownum ] [ blocknum ] [ i ] = val ;
}
}
}
}
if ( password ! = 0 & & password [ 0 ] ! = 0 )
{
space = calloc ( 1 , JPG_ENCRYPTED_MAXSIZE ) ;
if ( ( decrypted = JPG_decrypt ( indp , & recvlen , space , decoded , privkey ) ) ! = 0 & & recvlen = = origrequired / 8 )
{
for ( i = 0 ; i < recvlen ; i + + )
{
//printf("%02x",decrypted[i]);
decoded [ i ] = decrypted [ i ] ;
}
//printf(" decrypted.%d ind.%d\n",recvlen,*indp);
}
free ( space ) ;
}
//printf(" capacity %d required.%d power2.%d limit.%d\n",*capacityp,required,power2,limit);
if ( * capacityp > required & & outputfname ! = 0 & & outputfname [ 0 ] ! = 0 )
{
if ( ( output_file = fopen ( outputfname , WRITE_BINARY ) ) = = NULL ) {
fprintf ( stderr , " Can't open %s \n " , outputfname ) ;
if ( data ! = origdata )
free ( data ) ;
return ( - 1 ) ;
}
outputinfo . err = jpeg_std_error ( & jerr ) ;
jpeg_create_compress ( & outputinfo ) ;
jpeg_stdio_dest ( & outputinfo , output_file ) ;
jpeg_copy_critical_parameters ( & inputinfo , & outputinfo ) ;
// Print out or modify DCT coefficients
for ( compnum = 0 ; compnum < num_components ; compnum + + )
{
for ( rownum = 0 ; rownum < height_in_blocks [ compnum ] ; rownum + + )
{
for ( blocknum = 0 ; blocknum < width_in_blocks [ compnum ] ; blocknum + + )
{
//printf("\n\nComponent: %i, Row:%i, Column: %i\n", compnum, rownum, blocknum);
for ( i = 0 ; i < DCTSIZE2 & & emit < required ; i + + )
{
val = coef_buffers [ compnum ] [ rownum ] [ blocknum ] [ i ] ;
if ( val < - limit | | val > = limit )
{
val & = ~ 1 ;
if ( GETBIT ( data , emit ) ! = 0 ) //|| (emit >= required && (rand() & 1) != 0) )
val | = 1 ;
//printf("%c",(val&1)!=0?'1':'0');
coef_buffers [ compnum ] [ rownum ] [ blocknum ] [ i ] = val ;
emit + + ;
}
//printf("%i,", coef_buffers[compnum][rownum][blocknum][i]);
}
}
}
}
//printf(" emit.%d\n",emit);
// Output the new DCT coeffs to a JPEG file
modified = 0 ;
for ( compnum = 0 ; compnum < num_components ; compnum + + )
{
for ( rownum = 0 ; rownum < height_in_blocks [ compnum ] ; rownum + + )
{
row_ptrs [ compnum ] = ( ( & outputinfo ) - > mem - > access_virt_barray ) ( ( j_common_ptr ) & outputinfo , coef_arrays [ compnum ] , rownum , ( JDIMENSION ) 1 , TRUE ) ;
if ( memcmp ( row_ptrs [ compnum ] [ 0 ] [ 0 ] , coef_buffers [ compnum ] [ rownum ] [ 0 ] , block_row_size [ compnum ] ) ! = 0 )
{
memcpy ( row_ptrs [ compnum ] [ 0 ] [ 0 ] , coef_buffers [ compnum ] [ rownum ] [ 0 ] , block_row_size [ compnum ] ) ;
modified + + ;
}
totalrows + + ;
}
}
// Write to the output file
jpeg_write_coefficients ( & outputinfo , coef_arrays ) ;
// Finish compression and release memory
jpeg_finish_compress ( & outputinfo ) ;
jpeg_destroy_compress ( & outputinfo ) ;
fclose ( output_file ) ;
}
jpeg_finish_decompress ( & inputinfo ) ;
jpeg_destroy_decompress ( & inputinfo ) ;
fclose ( input_file ) ;
if ( modified ! = 0 )
{
printf ( " New DCT coefficients successfully written to %s, capacity %d modifiedrows.%d/%d emit.%d \n " , outputfname , * capacityp , modified , totalrows , emit ) ;
}
free ( block_row_size ) ;
free ( width_in_blocks ) ;
free ( height_in_blocks ) ;
if ( data ! = origdata )
free ( data ) ;
return ( modified ) ;
}
char * LP_jpg ( char * srcfile , char * destfile , int32_t power2 , char * passphrase , char * datastr , int32_t required , uint16_t ind )
{
cJSON * retjson ; int32_t len = 0 , modified , capacity ; char * decodedstr ; uint8_t * data = 0 , * decoded = 0 ;
if ( srcfile ! = 0 & & srcfile [ 0 ] ! = 0 )
{
retjson = cJSON_CreateObject ( ) ;
if ( datastr ! = 0 & & datastr [ 0 ] ! = 0 )
{
if ( ( len = is_hexstr ( datastr , 0 ) ) > 0 )
{
len > > = 1 ;
data = calloc ( 1 , len ) ;
decode_hex ( data , len , datastr ) ;
required = len * 8 ;
//int32_t i; for (i=0; i<required; i++)
// printf("%c",'0'+(GETBIT(data,i)!=0));
//printf(" datastr.%d %s\n",required,datastr);
}
}
if ( required > 0 )
decoded = calloc ( 1 , len + required ) ;
if ( ( modified = LP_jpg_process ( & capacity , srcfile , destfile , decoded , data , required , power2 , passphrase , & ind ) ) < 0 )
jaddstr ( retjson , " error " , " file not found " ) ;
else
{
jaddstr ( retjson , " result " , " success " ) ;
jaddnum ( retjson , " modifiedrows " , modified ) ;
if ( modified ! = 0 )
jaddstr ( retjson , " outputfile " , destfile ) ;
jaddnum ( retjson , " power2 " , power2 ) ;
jaddnum ( retjson , " capacity " , capacity ) ;
jaddnum ( retjson , " required " , required ) ;
jaddnum ( retjson , " ind " , ind ) ;
}
if ( decoded ! = 0 )
{
if ( capacity > 0 )
{
//printf("len.%d required.%d capacity.%d\n",len,required,capacity);
decodedstr = calloc ( 1 , ( len + required ) * 2 + 1 ) ;
init_hexbytes_noT ( decodedstr , decoded , required / 8 ) ;
jaddstr ( retjson , " decoded " , decodedstr ) ;
free ( decodedstr ) ;
}
free ( decoded ) ;
}
if ( data ! = 0 )
free ( data ) ;
return ( jprint ( retjson , 1 ) ) ;
} else return ( clonestr ( " { \" error \" : \" no source file error \" } " ) ) ;
}