1 /******************************************************************************
2 * Copyright © 2014-2017 The SuperNET Developers. *
4 * See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at *
5 * the top-level directory of this distribution for the individual copyright *
6 * holder information and the developer policies on copyright and licensing. *
8 * Unless otherwise agreed in a custom licensing agreement, no part of the *
9 * SuperNET software, including this file may be copied, modified, propagated *
10 * or distributed except according to the terms contained in the LICENSE file *
12 * Removal or modification of this copyright notice is prohibited. *
14 ******************************************************************************/
16 // komodo functions that interact with bitcoind C++
19 #include <curl/curl.h>
20 #include <curl/easy.h>
22 #include <curl/curl.h>
23 #include <curl/easy.h>
26 #include "komodo_defs.h"
28 int32_t komodo_notaries(uint8_t pubkeys[64][33],int32_t height,uint32_t timestamp);
29 int32_t komodo_electednotary(int32_t *numnotariesp,uint8_t *pubkey33,int32_t height,uint32_t timestamp);
31 //#define issue_curl(cmdstr) bitcoind_RPC(0,(char *)"curl",(char *)"http://127.0.0.1:7776",0,0,(char *)(cmdstr))
33 struct MemoryStruct { char *memory; size_t size; };
34 struct return_string { char *ptr; size_t len; };
36 // return data from the server
37 #define CURL_GLOBAL_ALL (CURL_GLOBAL_SSL|CURL_GLOBAL_WIN32)
38 #define CURL_GLOBAL_SSL (1<<0)
39 #define CURL_GLOBAL_WIN32 (1<<1)
42 /************************************************************************
44 * Initialize the string handler so that it is thread safe
46 ************************************************************************/
48 void init_string(struct return_string *s)
51 s->ptr = (char *)calloc(1,s->len+1);
54 fprintf(stderr,"init_string malloc() failed\n");
60 /************************************************************************
62 * Use the "writer" to accumulate text until done
64 ************************************************************************/
66 size_t accumulatebytes(void *ptr,size_t size,size_t nmemb,struct return_string *s)
68 size_t new_len = s->len + size*nmemb;
69 s->ptr = (char *)realloc(s->ptr,new_len+1);
72 fprintf(stderr, "accumulate realloc() failed\n");
75 memcpy(s->ptr+s->len,ptr,size*nmemb);
76 s->ptr[new_len] = '\0';
81 /************************************************************************
83 * return the current system time in milliseconds
85 ************************************************************************/
87 #define EXTRACT_BITCOIND_RESULT // if defined, ensures error is null and returns the "result" field
88 #ifdef EXTRACT_BITCOIND_RESULT
90 /************************************************************************
92 * perform post processing of the results
94 ************************************************************************/
96 char *post_process_bitcoind_RPC(char *debugstr,char *command,char *rpcstr,char *params)
98 long i,j,len; char *retstr = 0; cJSON *json,*result,*error;
99 //printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC.%s.[%s]\n",debugstr,command,rpcstr);
100 if ( command == 0 || rpcstr == 0 || rpcstr[0] == 0 )
102 if ( strcmp(command,"signrawtransaction") != 0 )
103 printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC.%s.[%s]\n",debugstr,command,rpcstr);
106 json = cJSON_Parse(rpcstr);
109 printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC.%s can't parse.(%s) params.(%s)\n",debugstr,command,rpcstr,params);
113 result = cJSON_GetObjectItem(json,"result");
114 error = cJSON_GetObjectItem(json,"error");
115 if ( error != 0 && result != 0 )
117 if ( (error->type&0xff) == cJSON_NULL && (result->type&0xff) != cJSON_NULL )
119 retstr = cJSON_Print(result);
120 len = strlen(retstr);
121 if ( retstr[0] == '"' && retstr[len-1] == '"' )
123 for (i=1,j=0; i<len-1; i++,j++)
124 retstr[j] = retstr[i];
128 else if ( (error->type&0xff) != cJSON_NULL || (result->type&0xff) != cJSON_NULL )
130 if ( strcmp(command,"signrawtransaction") != 0 )
131 printf("<<<<<<<<<<< bitcoind_RPC: %s post_process_bitcoind_RPC (%s) error.%s\n",debugstr,command,rpcstr);
134 } else retstr = rpcstr;
136 //fprintf(stderr,"<<<<<<<<<<< bitcoind_RPC: postprocess returns.(%s)\n",retstr);
141 /************************************************************************
145 ************************************************************************/
147 char *bitcoind_RPC(char **retstrp,char *debugstr,char *url,char *userpass,char *command,char *params)
149 static int didinit,count,count2; static double elapsedsum,elapsedsum2;
150 struct curl_slist *headers = NULL; struct return_string s; CURLcode res; CURL *curl_handle;
151 char *bracket0,*bracket1,*databuf = 0; long len; int32_t specialcase,numretries; double starttime;
155 curl_global_init(CURL_GLOBAL_ALL); //init the curl session
158 if ( debugstr != 0 && strcmp(debugstr,"BTCD") == 0 && command != 0 && strcmp(command,"SuperNET") == 0 )
160 else specialcase = 0;
162 strcpy(url,"http://127.0.0.1:7876/nxt");
163 if ( specialcase != 0 && 0 )
164 printf("<<<<<<<<<<< bitcoind_RPC: debug.(%s) url.(%s) command.(%s) params.(%s)\n",debugstr,url,command,params);
168 starttime = OS_milliseconds();
169 curl_handle = curl_easy_init();
171 headers = curl_slist_append(0,"Expect:");
173 curl_easy_setopt(curl_handle,CURLOPT_USERAGENT,"mozilla/4.0");//"Mozilla/4.0 (compatible; )");
174 curl_easy_setopt(curl_handle,CURLOPT_HTTPHEADER, headers);
175 curl_easy_setopt(curl_handle,CURLOPT_URL, url);
176 curl_easy_setopt(curl_handle,CURLOPT_WRITEFUNCTION, (void *)accumulatebytes); // send all data to this function
177 curl_easy_setopt(curl_handle,CURLOPT_WRITEDATA, &s); // we pass our 's' struct to the callback
178 curl_easy_setopt(curl_handle,CURLOPT_NOSIGNAL, 1L); // supposed to fix "Alarm clock" and long jump crash
179 curl_easy_setopt(curl_handle,CURLOPT_NOPROGRESS, 1L); // no progress callback
180 if ( strncmp(url,"https",5) == 0 )
182 curl_easy_setopt(curl_handle,CURLOPT_SSL_VERIFYPEER,0);
183 curl_easy_setopt(curl_handle,CURLOPT_SSL_VERIFYHOST,0);
186 curl_easy_setopt(curl_handle,CURLOPT_USERPWD, userpass);
190 if ( command != 0 && specialcase == 0 )
192 len = strlen(params);
193 if ( len > 0 && params[0] == '[' && params[len-1] == ']' ) {
194 bracket0 = bracket1 = (char *)"";
198 bracket0 = (char *)"[";
199 bracket1 = (char *)"]";
202 databuf = (char *)malloc(256 + strlen(command) + strlen(params));
203 sprintf(databuf,"{\"id\":\"jl777\",\"method\":\"%s\",\"params\":%s%s%s}",command,bracket0,params,bracket1);
204 //printf("url.(%s) userpass.(%s) databuf.(%s)\n",url,userpass,databuf);
206 } //else if ( specialcase != 0 ) fprintf(stderr,"databuf.(%s)\n",params);
207 curl_easy_setopt(curl_handle,CURLOPT_POST,1L);
209 curl_easy_setopt(curl_handle,CURLOPT_POSTFIELDS,databuf);
210 else curl_easy_setopt(curl_handle,CURLOPT_POSTFIELDS,params);
212 //laststart = milliseconds();
213 res = curl_easy_perform(curl_handle);
214 curl_slist_free_all(headers);
215 curl_easy_cleanup(curl_handle);
216 if ( databuf != 0 ) // clean up temporary buffer
221 if ( res != CURLE_OK )
224 if ( specialcase != 0 )
226 printf("<<<<<<<<<<< bitcoind_RPC.(%s): BTCD.%s timeout params.(%s) s.ptr.(%s) err.%d\n",url,command,params,s.ptr,res);
230 else if ( numretries >= 1 )
232 //printf("Maximum number of retries exceeded!\n");
236 if ( (rand() % 1000) == 0 )
237 printf( "curl_easy_perform() failed: %s %s.(%s %s), retries: %d\n",curl_easy_strerror(res),debugstr,url,command,numretries);
239 sleep((1<<numretries));
245 if ( command != 0 && specialcase == 0 )
248 elapsedsum += (OS_milliseconds() - starttime);
249 if ( (count % 1000000) == 0)
250 printf("%d: ave %9.6f | elapsed %.3f millis | bitcoind_RPC.(%s) url.(%s)\n",count,elapsedsum/count,(OS_milliseconds() - starttime),command,url);
256 return(post_process_bitcoind_RPC(debugstr,command,s.ptr,params));
260 if ( 0 && specialcase != 0 )
261 fprintf(stderr,"<<<<<<<<<<< bitcoind_RPC: BTCD.(%s) -> (%s)\n",params,s.ptr);
263 elapsedsum2 += (OS_milliseconds() - starttime);
264 if ( (count2 % 10000) == 0)
265 printf("%d: ave %9.6f | elapsed %.3f millis | NXT calls.(%s) cmd.(%s)\n",count2,elapsedsum2/count2,(double)(OS_milliseconds() - starttime),url,command);
269 printf("bitcoind_RPC: impossible case\n");
274 static size_t WriteMemoryCallback(void *ptr,size_t size,size_t nmemb,void *data)
276 size_t realsize = (size * nmemb);
277 struct MemoryStruct *mem = (struct MemoryStruct *)data;
278 mem->memory = (char *)((ptr != 0) ? realloc(mem->memory,mem->size + realsize + 1) : malloc(mem->size + realsize + 1));
279 if ( mem->memory != 0 )
282 memcpy(&(mem->memory[mem->size]),ptr,realsize);
283 mem->size += realsize;
284 mem->memory[mem->size] = 0;
286 //printf("got %d bytes\n",(int32_t)(size*nmemb));
290 char *curl_post(CURL **cHandlep,char *url,char *userpass,char *postfields,char *hdr0,char *hdr1,char *hdr2,char *hdr3)
292 struct MemoryStruct chunk; CURL *cHandle; long code; struct curl_slist *headers = 0;
293 if ( (cHandle= *cHandlep) == NULL )
294 *cHandlep = cHandle = curl_easy_init();
295 else curl_easy_reset(cHandle);
297 //curl_easy_setopt(cHandle,CURLOPT_VERBOSE, 1);
299 curl_easy_setopt(cHandle,CURLOPT_USERAGENT,"mozilla/4.0");//"Mozilla/4.0 (compatible; )");
300 curl_easy_setopt(cHandle,CURLOPT_SSL_VERIFYPEER,0);
301 //curl_easy_setopt(cHandle,CURLOPT_SSLVERSION,1);
302 curl_easy_setopt(cHandle,CURLOPT_URL,url);
303 curl_easy_setopt(cHandle,CURLOPT_CONNECTTIMEOUT,10);
304 if ( userpass != 0 && userpass[0] != 0 )
305 curl_easy_setopt(cHandle,CURLOPT_USERPWD,userpass);
306 if ( postfields != 0 && postfields[0] != 0 )
308 curl_easy_setopt(cHandle,CURLOPT_POST,1);
309 curl_easy_setopt(cHandle,CURLOPT_POSTFIELDS,postfields);
311 if ( hdr0 != NULL && hdr0[0] != 0 )
313 //printf("HDR0.(%s) HDR1.(%s) HDR2.(%s) HDR3.(%s)\n",hdr0!=0?hdr0:"",hdr1!=0?hdr1:"",hdr2!=0?hdr2:"",hdr3!=0?hdr3:"");
314 headers = curl_slist_append(headers,hdr0);
315 if ( hdr1 != 0 && hdr1[0] != 0 )
316 headers = curl_slist_append(headers,hdr1);
317 if ( hdr2 != 0 && hdr2[0] != 0 )
318 headers = curl_slist_append(headers,hdr2);
319 if ( hdr3 != 0 && hdr3[0] != 0 )
320 headers = curl_slist_append(headers,hdr3);
321 } //headers = curl_slist_append(0,"Expect:");
323 curl_easy_setopt(cHandle,CURLOPT_HTTPHEADER,headers);
324 //res = curl_easy_perform(cHandle);
325 memset(&chunk,0,sizeof(chunk));
326 curl_easy_setopt(cHandle,CURLOPT_WRITEFUNCTION,WriteMemoryCallback);
327 curl_easy_setopt(cHandle,CURLOPT_WRITEDATA,(void *)&chunk);
328 curl_easy_perform(cHandle);
329 curl_easy_getinfo(cHandle,CURLINFO_RESPONSE_CODE,&code);
331 curl_slist_free_all(headers);
333 printf("(%s) server responded with code %ld (%s)\n",url,code,chunk.memory);
334 return(chunk.memory);
337 char *komodo_issuemethod(char *userpass,char *method,char *params,uint16_t port)
339 //static void *cHandle;
340 char url[512],*retstr=0,*retstr2=0,postdata[8192];
341 if ( params == 0 || params[0] == 0 )
342 params = (char *)"[]";
343 if ( strlen(params) < sizeof(postdata)-128 )
345 sprintf(url,(char *)"http://127.0.0.1:%u",port);
346 sprintf(postdata,"{\"method\":\"%s\",\"params\":%s}",method,params);
347 //printf("[%s] (%s) postdata.(%s) params.(%s) USERPASS.(%s)\n",ASSETCHAINS_SYMBOL,url,postdata,params,KMDUSERPASS);
348 retstr2 = bitcoind_RPC(&retstr,(char *)"debug",url,userpass,method,params);
349 //retstr = curl_post(&cHandle,url,USERPASS,postdata,0,0,0,0);
354 int32_t notarizedtxid_height(char *dest,char *txidstr,int32_t *kmdnotarized_heightp)
356 char *jsonstr,params[256],*userpass; uint16_t port; cJSON *json,*item; int32_t height = 0,txid_height = 0,txid_confirmations = 0;
358 *kmdnotarized_heightp = 0;
359 if ( strcmp(dest,"KMD") == 0 )
362 userpass = KMDUSERPASS;
364 else if ( strcmp(dest,"BTC") == 0 )
367 userpass = BTCUSERPASS;
370 if ( userpass[0] != 0 )
372 if ( (jsonstr= komodo_issuemethod(userpass,(char *)"getinfo",params,port)) != 0 )
374 //printf("(%s)\n",jsonstr);
375 if ( (json= cJSON_Parse(jsonstr)) != 0 )
377 if ( (item= jobj(json,(char *)"result")) != 0 )
379 height = jint(item,(char *)"blocks");
380 *kmdnotarized_heightp = strcmp(dest,"KMD") == 0 ? jint(item,(char *)"notarized") : height;
386 sprintf(params,"[\"%s\", 1]",txidstr);
387 if ( (jsonstr= komodo_issuemethod(userpass,(char *)"getrawtransaction",params,port)) != 0 )
389 //printf("(%s)\n",jsonstr);
390 if ( (json= cJSON_Parse(jsonstr)) != 0 )
392 if ( (item= jobj(json,(char *)"result")) != 0 )
394 txid_confirmations = jint(item,(char *)"confirmations");
395 if ( txid_confirmations > 0 && height > txid_confirmations )
396 txid_height = height - txid_confirmations;
397 else txid_height = height;
398 //printf("height.%d tconfs.%d txid_height.%d\n",height,txid_confirmations,txid_height);
408 int32_t komodo_verifynotarizedscript(int32_t height,uint8_t *script,int32_t len,uint256 NOTARIZED_HASH)
410 int32_t i; uint256 hash; char params[256];
412 ((uint8_t *)&hash)[i] = script[2+i];
413 if ( hash == NOTARIZED_HASH )
416 printf("%02x",((uint8_t *)&NOTARIZED_HASH)[i]);
417 printf(" notarized, ");
419 printf("%02x",((uint8_t *)&hash)[i]);
420 printf(" opreturn from [%s] ht.%d MISMATCHED\n",ASSETCHAINS_SYMBOL,height);
424 int32_t komodo_verifynotarization(char *symbol,char *dest,int32_t height,int32_t NOTARIZED_HEIGHT,uint256 NOTARIZED_HASH,uint256 NOTARIZED_DESTTXID)
426 char params[256],*jsonstr,*hexstr; uint8_t *script,_script[8192]; int32_t n,len,retval = -1; cJSON *json,*txjson,*vouts,*vout,*skey;
431 sprintf(¶ms[i*2 + 2],"%02x",((uint8_t *)&NOTARIZED_DESTTXID)[31-i]);
432 strcat(params,"\", 1]");*/
433 sprintf(params,"[\"%s\", 1]",NOTARIZED_DESTTXID.ToString().c_str());
434 if ( strcmp(symbol,ASSETCHAINS_SYMBOL[0]==0?(char *)"KMD":ASSETCHAINS_SYMBOL) != 0 )
436 if ( 0 && ASSETCHAINS_SYMBOL[0] != 0 )
437 printf("[%s] src.%s dest.%s params.[%s] ht.%d notarized.%d\n",ASSETCHAINS_SYMBOL,symbol,dest,params,height,NOTARIZED_HEIGHT);
438 if ( strcmp(dest,"KMD") == 0 )
440 if ( KMDUSERPASS[0] != 0 )
442 if ( ASSETCHAINS_SYMBOL[0] != 0 )
444 jsonstr = komodo_issuemethod(KMDUSERPASS,(char *)"getrawtransaction",params,KMD_PORT);
445 //printf("userpass.(%s) got (%s)\n",KMDUSERPASS,jsonstr);
447 }//else jsonstr = _dex_getrawtransaction();
448 else return(0); // need universal way to issue DEX* API, since notaries mine most blocks, this ok
450 else if ( strcmp(dest,"BTC") == 0 )
452 if ( BTCUSERPASS[0] != 0 )
454 //printf("BTCUSERPASS.(%s)\n",BTCUSERPASS);
455 jsonstr = komodo_issuemethod(BTCUSERPASS,(char *)"getrawtransaction",params,8332);
457 //else jsonstr = _dex_getrawtransaction();
462 printf("[%s] verifynotarization error unexpected dest.(%s)\n",ASSETCHAINS_SYMBOL,dest);
467 if ( (json= cJSON_Parse(jsonstr)) != 0 )
469 if ( (txjson= jobj(json,(char *)"result")) != 0 && (vouts= jarray(&n,txjson,(char *)"vout")) > 0 )
471 vout = jitem(vouts,n-1);
472 if ( 0 && ASSETCHAINS_SYMBOL[0] != 0 )
473 printf("vout.(%s)\n",jprint(vout,0));
474 if ( (skey= jobj(vout,(char *)"scriptPubKey")) != 0 )
476 if ( (hexstr= jstr(skey,(char *)"hex")) != 0 )
478 //printf("HEX.(%s) vs hash.%s\n",hexstr,NOTARIZED_HASH.ToString().c_str());
479 len = strlen(hexstr) >> 1;
480 decode_hex(script,len,hexstr);
481 if ( script[1] == 0x4c )
486 else if ( script[1] == 0x4d )
491 retval = komodo_verifynotarizedscript(height,script,len,NOTARIZED_HASH);
502 /*uint256 komodo_getblockhash(int32_t height)
504 uint256 hash; char params[128],*hexstr,*jsonstr; cJSON *result; int32_t i; uint8_t revbuf[32];
505 memset(&hash,0,sizeof(hash));
506 sprintf(params,"[%d]",height);
507 if ( (jsonstr= komodo_issuemethod(KMDUSERPASS,(char *)"getblockhash",params,BITCOIND_PORT)) != 0 )
509 if ( (result= cJSON_Parse(jsonstr)) != 0 )
511 if ( (hexstr= jstr(result,(char *)"result")) != 0 )
513 if ( is_hexstr(hexstr,0) == 64 )
515 decode_hex(revbuf,32,hexstr);
517 ((uint8_t *)&hash)[i] = revbuf[31-i];
522 printf("KMD hash.%d (%s) %x\n",height,jsonstr,*(uint32_t *)&hash);
528 uint256 _komodo_getblockhash(int32_t height);*/
530 uint64_t komodo_seed(int32_t height)
533 /*if ( 0 ) // problem during init time, seeds are needed for loading blockindex, so null seeds...
535 uint256 hash,zero; CBlockIndex *pindex;
536 memset(&hash,0,sizeof(hash));
537 memset(&zero,0,sizeof(zero));
540 if ( ASSETCHAINS_SYMBOL[0] == 0 )
541 hash = _komodo_getblockhash(height);
542 if ( memcmp(&hash,&zero,sizeof(hash)) == 0 )
543 hash = komodo_getblockhash(height);
546 printf("%02x",((uint8_t *)&hash)[i]);
547 printf(" seed.%d\n",height);
548 seed = arith_uint256(hash.GetHex()).GetLow64();
552 seed = (height << 13) ^ (height << 2);
554 seed |= (height & 0xffffffff);
555 seed ^= (seed << 17) ^ (seed << 1);
560 uint32_t komodo_txtime(uint256 hash)
564 if (!GetTransaction(hash, tx,
566 Params().GetConsensus(),
570 //printf("null GetTransaction\n");
571 return(tx.nLockTime);
576 void komodo_disconnect(CBlockIndex *pindex,CBlock& block)
578 char symbol[KOMODO_ASSETCHAIN_MAXLEN],dest[KOMODO_ASSETCHAIN_MAXLEN]; struct komodo_state *sp;
579 //fprintf(stderr,"disconnect ht.%d\n",pindex->nHeight);
580 komodo_init(pindex->nHeight);
581 if ( (sp= komodo_stateptr(symbol,dest)) != 0 )
583 //sp->rewinding = pindex->nHeight;
584 //fprintf(stderr,"-%d ",pindex->nHeight);
585 } else printf("komodo_disconnect: ht.%d cant get komodo_state.(%s)\n",pindex->nHeight,ASSETCHAINS_SYMBOL);
589 int32_t komodo_is_notarytx(const CTransaction& tx)
591 uint8_t *ptr; static uint8_t crypto777[33];
592 if ( tx.vout.size() > 0 )
595 ptr = (uint8_t *)tx.vout[0].scriptPubKey.data();
597 ptr = (uint8_t *)&tx.vout[0].scriptPubKey[0];
601 if ( crypto777[0] == 0 )
602 decode_hex(crypto777,33,(char *)CRYPTO777_PUBSECPSTR);
603 if ( memcmp(ptr+1,crypto777,33) == 0 )
605 //printf("found notarytx\n");
613 int32_t komodo_block2height(CBlock *block)
615 int32_t i,n,height = 0; uint8_t *ptr;
616 if ( block->vtx[0].vin.size() > 0 )
619 ptr = (uint8_t *)block->vtx[0].vin[0].scriptSig.data();
621 ptr = (uint8_t *)&block->vtx[0].vin[0].scriptSig[0];
623 if ( ptr != 0 && block->vtx[0].vin[0].scriptSig.size() > 5 )
625 //for (i=0; i<6; i++)
626 // printf("%02x",ptr[i]);
628 for (i=0; i<n; i++) // looks strange but this works
630 //03bb81000101(bb 187) (81 48001) (00 12288256) <- coinbase.6 ht.12288256
631 height += ((uint32_t)ptr[i+1] << (i*8));
632 //printf("(%02x %x %d) ",ptr[i+1],((uint32_t)ptr[i+1] << (i*8)),height);
634 //printf(" <- coinbase.%d ht.%d\n",(int32_t)block->vtx[0].vin[0].scriptSig.size(),height);
636 //komodo_init(height);
641 void komodo_block2pubkey33(uint8_t *pubkey33,CBlock& block)
644 memset(pubkey33,0,33);
645 if ( block.vtx[0].vout.size() > 0 )
648 uint8_t *ptr = (uint8_t *)block.vtx[0].vout[0].scriptPubKey.data();
650 uint8_t *ptr = (uint8_t *)&block.vtx[0].vout[0].scriptPubKey[0];
653 n = block.vtx[0].vout[0].scriptPubKey.size();
655 memcpy(pubkey33,ptr+1,33);
659 int32_t komodo_blockload(CBlock& block,CBlockIndex *pindex)
662 // Open history file to read
663 CAutoFile filein(OpenBlockFile(pindex->GetBlockPos(),true),SER_DISK,CLIENT_VERSION);
667 try { filein >> block; }
668 catch (const std::exception& e)
670 fprintf(stderr,"readblockfromdisk err B\n");
676 uint32_t komodo_chainactive_timestamp()
678 if ( chainActive.Tip() != 0 )
679 return((uint32_t)chainActive.Tip()->GetBlockTime());
683 CBlockIndex *komodo_chainactive(int32_t height)
685 if ( chainActive.Tip() != 0 )
687 if ( height <= chainActive.Tip()->nHeight )
688 return(chainActive[height]);
689 // else fprintf(stderr,"komodo_chainactive height %d > active.%d\n",height,chainActive.Tip()->nHeight);
691 //fprintf(stderr,"komodo_chainactive null chainActive.Tip() height %d\n",height);
695 uint32_t komodo_heightstamp(int32_t height)
698 if ( height > 0 && (ptr= komodo_chainactive(height)) != 0 )
700 //else fprintf(stderr,"komodo_heightstamp null ptr for block.%d\n",height);
704 void komodo_index2pubkey33(uint8_t *pubkey33,CBlockIndex *pindex,int32_t height)
706 CBlock block; int32_t num,i; uint8_t pubkeys[64][33];
707 //komodo_init(height);
708 memset(pubkey33,0,33);
711 if ( pindex->pubkey33[0] == 2 || pindex->pubkey33[0] == 3 )
713 memcpy(pubkey33,pindex->pubkey33,33);
716 if ( komodo_blockload(block,pindex) == 0 )
718 komodo_block2pubkey33(pubkey33,block);
719 if ( pubkey33[0] == 2 || pubkey33[0] == 3 )
721 memcpy(pindex->pubkey33,pubkey33,33);
722 if ( (num= komodo_notaries(pubkeys,(int32_t)pindex->nHeight,(uint32_t)pindex->nTime)) > 0 )
724 pindex->notaryid = -1;
725 for (i=0; i<num; i++)
727 if ( memcmp(pubkeys[i],pubkey33,33) == 0 )
729 pindex->notaryid = i;
734 } else pindex->notaryid = -1;
739 // height -> pubkey33
740 //printf("extending chaintip komodo_index2pubkey33 height.%d need to get pubkey33\n",height);
744 /*void komodo_connectpindex(CBlockIndex *pindex)
747 if ( komodo_blockload(block,pindex) == 0 )
748 komodo_connectblock(pindex,block);
752 int8_t komodo_minerid(int32_t height,uint8_t *pubkey33)
754 int32_t num,i,numnotaries; CBlockIndex *pindex; uint32_t timestamp=0; uint8_t _pubkey33[33],pubkeys[64][33];
755 if ( (pindex= chainActive[height]) != 0 )
757 if ( pindex->pubkey33[0] == 2 || pindex->pubkey33[0] == 3 )
760 memcpy(pubkey33,pindex->pubkey33,33);
761 return(pindex->notaryid);
764 komodo_index2pubkey33(pubkey33,pindex,height);
765 timestamp = pindex->GetBlockTime();
766 if ( (num= komodo_notaries(pubkeys,height,timestamp)) > 0 )
768 for (i=0; i<num; i++)
769 if ( memcmp(pubkeys[i],pubkey33,33) == 0 )
773 return(komodo_electednotary(&numnotaries,pubkey33,height,timestamp));
776 int32_t komodo_eligiblenotary(uint8_t pubkeys[66][33],int32_t *mids,int32_t *nonzpkeysp,int32_t height)
778 int32_t i,j,duplicate; CBlockIndex *pindex; uint8_t pubkey33[33];
779 memset(mids,-1,sizeof(*mids)*66);
780 for (i=duplicate=0; i<66; i++)
782 if ( (pindex= komodo_chainactive(height-i)) != 0 )
784 if ( pindex->notaryid >= 0 && (pindex->pubkey33[0] == 2 || pindex->pubkey33[0] == 3) )
786 memcpy(pubkeys[i],pindex->pubkey33,33);
787 mids[i] = pindex->notaryid;
792 komodo_index2pubkey33(pubkey33,pindex,height-i);
793 memcpy(pubkeys[i],pubkey33,33);
794 if ( (mids[i]= komodo_minerid(height-i,pubkey33)) >= 0 )
796 //mids[i] = *(int32_t *)pubkey33;
800 if ( mids[0] >= 0 && i > 0 && mids[i] == mids[0] )
804 if ( i == 66 && duplicate == 0 && (height > 186233 || *nonzpkeysp > 0) )
809 int32_t komodo_minerids(uint8_t *minerids,int32_t height,int32_t width)
812 for (i=0; i<width; i++,n++)
816 minerids[i] = komodo_minerid(height - i,0);
821 int32_t komodo_is_special(int32_t height,uint8_t pubkey33[33],uint32_t timestamp)
823 int32_t i,notaryid=0,minerid,limit,nid; //uint8_t _pubkey33[33];
824 if ( height >= 225000 )
825 komodo_chosennotary(¬aryid,height,pubkey33,timestamp);
826 if ( height >= 34000 && notaryid >= 0 )
828 if ( height < 79693 )
830 else if ( height < 82000 )
833 for (i=1; i<limit; i++)
835 komodo_chosennotary(&nid,height-i,pubkey33,timestamp);
836 if ( nid == notaryid ) //komodo_minerid(height-i,_pubkey33)
838 if ( (0) && notaryid > 0 )
839 fprintf(stderr,"ht.%d notaryid.%d already mined -i.%d nid.%d\n",height,notaryid,i,nid);
840 if ( height > 225000 )
844 //fprintf(stderr,"special notaryid.%d ht.%d limit.%d\n",notaryid,height,limit);
850 int32_t komodo_MoM(int32_t *notarized_heightp,uint256 *MoMp,uint256 *kmdtxidp,int32_t nHeight)
852 int32_t depth,notarized_ht; uint256 MoM,kmdtxid;
853 depth = komodo_MoMdata(¬arized_ht,&MoM,&kmdtxid,nHeight);
854 memset(MoMp,0,sizeof(*MoMp));
855 memset(kmdtxidp,0,sizeof(*kmdtxidp));
856 *notarized_heightp = 0;
857 if ( depth > 0 && notarized_ht > 0 && nHeight > notarized_ht-depth && nHeight <= notarized_ht )
860 *notarized_heightp = notarized_ht;
866 int32_t komodo_checkpoint(int32_t *notarized_heightp,int32_t nHeight,uint256 hash)
868 int32_t notarized_height,MoMdepth; uint256 MoM,notarized_hash,notarized_desttxid; CBlockIndex *notary; CBlockIndex *pindex;
869 if ( (pindex= chainActive.Tip()) == 0 )
871 notarized_height = komodo_notarizeddata(pindex->nHeight,¬arized_hash,¬arized_desttxid);
872 *notarized_heightp = notarized_height;
873 if ( notarized_height >= 0 && notarized_height <= pindex->nHeight && (notary= mapBlockIndex[notarized_hash]) != 0 )
875 //printf("nHeight.%d -> (%d %s)\n",pindex->Tip()->nHeight,notarized_height,notarized_hash.ToString().c_str());
876 if ( notary->nHeight == notarized_height ) // if notarized_hash not in chain, reorg
878 if ( nHeight < notarized_height )
880 //fprintf(stderr,"[%s] nHeight.%d < NOTARIZED_HEIGHT.%d\n",ASSETCHAINS_SYMBOL,nHeight,notarized_height);
883 else if ( nHeight == notarized_height && memcmp(&hash,¬arized_hash,sizeof(hash)) != 0 )
885 fprintf(stderr,"[%s] nHeight.%d == NOTARIZED_HEIGHT.%d, diff hash\n",ASSETCHAINS_SYMBOL,nHeight,notarized_height);
888 } else fprintf(stderr,"[%s] unexpected error notary_hash %s ht.%d at ht.%d\n",ASSETCHAINS_SYMBOL,notarized_hash.ToString().c_str(),notarized_height,notary->nHeight);
890 else if ( notarized_height > 0 && notarized_height != 73880 && notarized_height >= 170000 )
891 fprintf(stderr,"[%s] couldnt find notarized.(%s %d) ht.%d\n",ASSETCHAINS_SYMBOL,notarized_hash.ToString().c_str(),notarized_height,pindex->nHeight);
895 uint32_t komodo_interest_args(uint32_t *txheighttimep,int32_t *txheightp,uint32_t *tiptimep,uint64_t *valuep,uint256 hash,int32_t n)
898 CTransaction tx; uint256 hashBlock; CBlockIndex *pindex,*tipindex;
899 *txheighttimep = *txheightp = *tiptimep = 0;
901 if ( !GetTransaction(hash,tx,hashBlock,true) )
903 uint32_t locktime = 0;
904 if ( n < tx.vout.size() )
906 if ( (pindex= mapBlockIndex[hashBlock]) != 0 && (tipindex= chainActive.Tip()) != 0 )
908 *valuep = tx.vout[n].nValue;
909 *txheightp = pindex->nHeight;
910 *txheighttimep = pindex->nTime;
911 *tiptimep = tipindex->nTime;
912 locktime = tx.nLockTime;
913 //fprintf(stderr,"tx locktime.%u %.8f height.%d | tiptime.%u\n",locktime,(double)*valuep/COIN,*txheightp,*tiptimep);
919 uint64_t komodo_interest(int32_t txheight,uint64_t nValue,uint32_t nLockTime,uint32_t tiptime);
920 uint64_t komodo_accrued_interest(int32_t *txheightp,uint32_t *locktimep,uint256 hash,int32_t n,int32_t checkheight,uint64_t checkvalue)
922 uint64_t value; uint32_t tiptime,txheighttimep;
923 if ( (*locktimep= komodo_interest_args(&txheighttimep,txheightp,&tiptime,&value,hash,n)) != 0 )
925 if ( (checkvalue == 0 || value == checkvalue) && (checkheight == 0 || *txheightp == checkheight) )
926 return(komodo_interest(*txheightp,value,*locktimep,tiptime));
927 //fprintf(stderr,"nValue %llu lock.%u:%u nTime.%u -> %llu\n",(long long)coins.vout[n].nValue,coins.nLockTime,timestamp,pindex->nTime,(long long)interest);
928 else fprintf(stderr,"komodo_accrued_interest value mismatch %llu vs %llu or height mismatch %d vs %d\n",(long long)value,(long long)checkvalue,*txheightp,checkheight);
933 int32_t komodo_isrealtime(int32_t *kmdheightp)
935 struct komodo_state *sp; CBlockIndex *pindex;
936 if ( (sp= komodo_stateptrget((char *)"KMD")) != 0 )
937 *kmdheightp = sp->CURRENT_HEIGHT;
938 else *kmdheightp = 0;
939 if ( (pindex= chainActive.Tip()) != 0 && pindex->nHeight >= (int32_t)komodo_longestchain() )
944 int32_t komodo_validate_interest(const CTransaction &tx,int32_t txheight,uint32_t nTime,int32_t dispflag)
946 uint32_t cmptime = nTime;
947 if ( KOMODO_REWIND == 0 && ASSETCHAINS_SYMBOL[0] == 0 && (int64_t)tx.nLockTime >= LOCKTIME_THRESHOLD ) //1473793441 )
949 if ( txheight > 246748 )
951 if ( txheight < 247205 )
953 if ( (int64_t)tx.nLockTime < cmptime-3600 )
955 if ( tx.nLockTime != 1477258935 || dispflag != 0 )
957 fprintf(stderr,"komodo_validate_interest.%d reject.%d [%d] locktime %u cmp2.%u\n",dispflag,txheight,(int32_t)(tx.nLockTime - (cmptime-3600)),(uint32_t)tx.nLockTime,cmptime);
961 if ( 0 && dispflag != 0 )
962 fprintf(stderr,"validateinterest.%d accept.%d [%d] locktime %u cmp2.%u\n",dispflag,(int32_t)txheight,(int32_t)(tx.nLockTime - (cmptime-3600)),(int32_t)tx.nLockTime,cmptime);