/****************************************************************************** * Copyright © 2014-2017 The SuperNET Developers. * * * * See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at * * the top-level directory of this distribution for the individual copyright * * holder information and the developer policies on copyright and licensing. * * * * Unless otherwise agreed in a custom licensing agreement, no part of the * * SuperNET software, including this file may be copied, modified, propagated * * or distributed except according to the terms contained in the LICENSE file * * * * Removal or modification of this copyright notice is prohibited. * * * ******************************************************************************/ // // 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; isymbol,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= 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,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 %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 %s -> %s %s %s\n",wifstr,bits256_str(str,privkey),cmpstr,bits256_str(str2,cmpkey),cmpstr2); if ( bits256_cmp(privkey,cmpkey) == 0 ) return(1); } return(0); } 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 tmpstr[128]; cJSON *retjson; uint8_t tmptype; int32_t notarized; uint64_t nxtaddr; if ( (wifstr == 0 || wifstr[0] == 0) && LP_wifstr_valid(passphrase) > 0 ) { wifstr = passphrase; passphrase = 0; } if ( passphrase != 0 && passphrase[0] != 0 ) { 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; //vcalc_sha256(0,checkkey.bytes,(uint8_t *)passphrase,(int32_t)strlen(passphrase)); //printf("SHA256.(%s) ",bits256_str(pstr,checkkey)); //printf("privkey.(%s)\n",bits256_str(pstr,privkey)); bitcoin_priv2wif(coin->wiftaddr,tmpstr,privkey,coin->wiftype); bitcoin_wif2priv(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->wiftaddr,&tmptype,&privkey,wifstr); bitcoin_priv2wif(coin->wiftaddr,tmpstr,privkey,tmptype); if ( strcmp(tmpstr,wifstr) != 0 ) { printf("error reproducing the wifstr, likely edge case like non-supported uncompressed pubkey\n"); exit(1); } tmpkey = privkey; nxtaddr = conv_NXTpassword(tmpkey.bytes,pubkeyp->bytes,0,0); RS_encode(G.LP_NXTaddr,nxtaddr); } bitcoin_priv2pub(ctx,coin->pubkey33,coin->smartaddr,privkey,coin->taddr,coin->pubtype); if ( coin->counter == 0 ) { coin->counter++; memcpy(G.LP_pubsecp,coin->pubkey33,33); bitcoin_priv2wif(coin->wiftaddr,tmpstr,privkey,coin->wiftype); bitcoin_addr2rmd160(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->wiftaddr,G.USERPASS_WIFSTR,privkey,188); bitcoin_wif2priv(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 ( coin->importedprivkey == 0 && coin->electrum == 0 && coin->userpass[0] != 0 && LP_getheight(¬arized,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 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) { static void *ctx; int32_t counter; //iambob,; struct LP_utxoinfo *utxo,*tmp; if ( ctx == 0 ) ctx = bitcoin_ctx(); if ( G.LP_pendingswaps != 0 ) return(-1); G.initializing = 1; if ( gui == 0 ) gui = "cli"; counter = G.USERPASS_COUNTER; while ( G.waiting == 0 ) { printf("waiting for G.waiting\n"); sleep(5); } memset(&G,0,sizeof(G)); 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; 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(0,wifstr,privkey,0xff); bitcoin_wif2priv(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); } // 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 *data,int32_t required,int32_t power2,char *passphrase) { 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]; FILE *input_file,*output_file; int32_t val,modified,emit,totalrows,limit; if ( power2 < 0 || power2 > 16 ) power2 = 4; limit = 1; while ( power2 > 0 ) { limit <<= 1; power2--; } if ((input_file = fopen(inputfname, READ_BINARY)) == NULL) { fprintf(stderr, "Can't open %s\n", inputfname); exit(EXIT_FAILURE); } // 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; compnummem->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]; *capacityp = modified = emit = totalrows = 0; if ( decoded != 0 ) memset(decoded,0,required/8+1); for (compnum=0; compnummem->access_virt_barray)((j_common_ptr)&inputinfo,coef_arrays[compnum],rownum,(JDIMENSION)1,FALSE); for (blocknum=0; blocknum power2 ) { if ( decoded != 0 && (val & 1) != 0 && *capacityp < required ) decoded[*capacityp >> 3] |= (1 << (*capacityp & 7)); (*capacityp)++; } coef_buffers[compnum][rownum][blocknum][i] = val; } } } } printf("capacity %d required.%d\n",*capacityp,required); if ( *capacityp > required && outputfname != 0 && outputfname[0] != 0 ) { if ((output_file = fopen(outputfname, WRITE_BINARY)) == NULL) { fprintf(stderr, "Can't open %s\n", outputfname); exit(EXIT_FAILURE); } 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 power2 ) { val &= ~1; if ( (data[emit >> 3] & (1 << (emit&7))) != 0 ) val |= 1; emit++; } coef_buffers[compnum][rownum][blocknum][i] = val; //printf("%i,", coef_buffers[compnum][rownum][blocknum][i]); } } } } //printf("\n\n"); /* Output the new DCT coeffs to a JPEG file */ modified = 0; for (compnum=0; compnummem->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); } return(modified); } char *LP_jpg(char *srcfile,char *destfile,int32_t power2,char *passphrase,char *datastr,int32_t required) { 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(); jaddstr(retjson,"result","success"); 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; } } if ( required > 0 ) decoded = calloc(1,len); modified = LP_jpg_process(&capacity,srcfile,destfile,decoded,data,required,power2,passphrase); jaddnum(retjson,"modifiedrows",modified); if ( modified != 0 ) jaddstr(retjson,"outputfile",destfile); jaddnum(retjson,"power2",power2); jaddnum(retjson,"capacity",capacity); jaddnum(retjson,"required",required); if ( decoded != 0 ) { decodedstr = calloc(1,len*2+1); init_hexbytes_noT(decodedstr,decoded,len); 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\"}")); }