You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
701 lines
26 KiB
701 lines
26 KiB
/******************************************************************************
|
|
* Copyright © 2014-2015 The SuperNET Developers. *
|
|
* *
|
|
* See the AUTHORS, DEVELOPER-AGREEMENT and LICENSE files at *
|
|
* the top-level directory of this distribution for the individual copyright *
|
|
* holder information and the developer policies on copyright and licensing. *
|
|
* *
|
|
* Unless otherwise agreed in a custom licensing agreement, no part of the *
|
|
* SuperNET software, including this file may be copied, modified, propagated *
|
|
* or distributed except according to the terms contained in the LICENSE file *
|
|
* *
|
|
* Removal or modification of this copyright notice is prohibited. *
|
|
* *
|
|
******************************************************************************/
|
|
#ifdef notyet
|
|
|
|
#ifdef DEFINES_ONLY
|
|
#ifndef crypto777_system777_h
|
|
#define crypto777_system777_h
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <unistd.h>
|
|
#include <curl/curl.h>
|
|
#include <sys/types.h>
|
|
#include <sys/time.h>
|
|
//#include "../includes/miniupnp/miniwget.h"
|
|
//#include "../includes/miniupnp/miniupnpc.h"
|
|
//#include "../includes/miniupnp/upnpcommands.h"
|
|
//#include "../includes/miniupnp/upnperrors.h"
|
|
#include "../includes/uthash.h"
|
|
#include "../includes/cJSON.h"
|
|
#include "../utils/utils777.c"
|
|
#include "../utils/inet.c"
|
|
#include "../includes/utlist.h"
|
|
#include "../includes/nonportable.h"
|
|
#include "../includes/portable777.h"
|
|
|
|
|
|
#define SUPERNET_PORT 7777
|
|
#define LB_OFFSET 1
|
|
#define PUBGLOBALS_OFFSET 2
|
|
#define PUBRELAYS_OFFSET 3
|
|
#define SUPERNET_APIENDPOINT "tcp://127.0.0.1:7776"
|
|
|
|
#define nn_errstr() nn_strerror(nn_errno())
|
|
|
|
extern int32_t Debuglevel;
|
|
|
|
#ifndef MIN
|
|
#define MIN(x,y) (((x)<=(y)) ? (x) : (y))
|
|
#endif
|
|
#ifndef MAX
|
|
#define MAX(x,y) (((x)>=(y)) ? (x) : (y))
|
|
#endif
|
|
typedef int32_t (*ptm)(int32_t,char *args[]);
|
|
// nonportable functions needed in the OS specific directory
|
|
int32_t is_bundled_plugin(char *plugin);
|
|
|
|
struct sendendpoints { int32_t push,rep,pub,survey; };
|
|
struct recvendpoints { int32_t pull,req,sub,respond; };
|
|
struct biendpoints { int32_t bus,pair; };
|
|
struct allendpoints { struct sendendpoints send; struct recvendpoints recv; struct biendpoints both; };
|
|
union endpoints { int32_t all[sizeof(struct allendpoints) / sizeof(int32_t)]; struct allendpoints socks; };
|
|
struct db777_entry { UT_hash_handle hh; uint32_t allocsize,valuelen,valuesize,keylen:30,linked:1,dirty:1; uint8_t value[]; };
|
|
|
|
struct db777
|
|
{
|
|
void *db,*asyncdb;
|
|
portable_mutex_t mutex;
|
|
struct db777_entry *table;
|
|
int32_t reqsock,valuesize,matrixentries;
|
|
uint32_t start_RTblocknum;
|
|
void **matrix; char *dirty;
|
|
char compression[8],dbname[32],name[16],coinstr[16],flags;
|
|
void *ctl,*env; char namestr[32],restoredir[512],argspecialpath[512],argsubdir[512],restorelogdir[512],argname[512],argcompression[512],backupdir[512];
|
|
uint8_t checkbuf[10000000];
|
|
};
|
|
|
|
struct env777
|
|
{
|
|
char coinstr[16],subdir[64];
|
|
void *ctl,*env,*transactions;
|
|
struct db777 dbs[16];
|
|
int32_t numdbs,needbackup,lastbackup,currentbackup,matrixentries;
|
|
uint32_t start_RTblocknum;
|
|
};
|
|
|
|
#define DEFAULT_APISLEEP 100 // milliseconds
|
|
#define NUM_PLUGINTAGS 8192
|
|
struct applicant_info { uint64_t senderbits; uint32_t nonce; char startflag,lbendpoint[128],relayendpoint[128],globalendpoint[128]; };
|
|
|
|
struct SuperNET_info
|
|
{
|
|
char WEBSOCKETD[1024],NXTAPIURL[1024],NXTSERVER[1024],DBPATH[1024],DATADIR[1024],transport[16],BACKUPS[512],SERVICENXT[64];
|
|
char myipaddr[64],myNXTacct[64],myNXTaddr[64],NXTACCT[64],NXTADDR[64],NXTACCTSECRET[8192],SERVICESECRET[8192],userhome[512],hostname[512];
|
|
uint64_t my64bits; uint8_t myprivkey[32],mypubkey[32];
|
|
uint32_t myipbits,nonces[512],numnonces; struct applicant_info *responses; cJSON *peersjson; char lbendpoint[128],relayendpoint[128],globalendpoint[128];
|
|
int32_t usessl,ismainnet,Debuglevel,SuperNET_retval,APISLEEP,gatewayid,numgateways,readyflag,UPNP,iamrelay,disableNXT,NXTconfirms,automatch,PLUGINTIMEOUT,ppid,noncing,pullsock,telepathicdelay,peggy,idlegap,exchangeidle,recvtimeout;
|
|
uint16_t port,serviceport,pangeaport;
|
|
uint64_t tags[NUM_PLUGINTAGS][3];
|
|
struct kv777 *PM,*rawPM,*protocols,*alias,*services,*invoices,*NXTtxids,*NXTaccts;
|
|
struct dKV777 *relays;
|
|
cJSON *argjson;
|
|
//#ifdef INSIDE_MGW
|
|
// struct env777 *DBs;
|
|
//#endif
|
|
}; extern struct SuperNET_info SUPERNET;
|
|
|
|
struct coins_info
|
|
{
|
|
int32_t num,readyflag,slicei;
|
|
cJSON *argjson;
|
|
struct coin777 **LIST;
|
|
// this will be at the end of the plugins structure and will be called with all zeros to _init
|
|
}; extern struct coins_info COINS;
|
|
|
|
struct db777_info
|
|
{
|
|
char PATH[1024],RAMDISK[1024];
|
|
int32_t numdbs,readyflag;
|
|
struct db777 *DBS[1024];
|
|
}; extern struct db777_info SOPHIA;
|
|
|
|
#define MAX_MGWSERVERS 16
|
|
struct MGW_info
|
|
{
|
|
char PATH[1024],serverips[MAX_MGWSERVERS][64],bridgeipaddr[64],bridgeacct[64];
|
|
uint64_t srv64bits[MAX_MGWSERVERS],issuers[64];
|
|
int32_t M,numissuers,readyflag,port;
|
|
union endpoints all;
|
|
uint32_t numrecv,numsent;
|
|
}; extern struct MGW_info MGW;
|
|
|
|
#define MAX_RAMCHAINS 128
|
|
struct ramchain_info
|
|
{
|
|
char PATH[1024],coins[MAX_RAMCHAINS][16],pullnode[64];
|
|
double lastupdate[MAX_RAMCHAINS];
|
|
union endpoints all;
|
|
int32_t num,readyflag,fastmode,verifyspends;
|
|
// this will be at the end of the plugins structure and will be called with all zeros to _init
|
|
}; extern struct ramchain_info RAMCHAINS;
|
|
|
|
struct cashier_info
|
|
{
|
|
int32_t readyflag;
|
|
}; extern struct cashier_info CASHIER;
|
|
|
|
struct prices_info
|
|
{
|
|
int32_t readyflag;
|
|
}; extern struct prices_info PRICES;
|
|
|
|
#define DEFAULT_PEGGYDAYS 21
|
|
#define PEGGY_LOCK 1
|
|
#define PEGGY_REDEEM 2
|
|
struct teleport_info
|
|
{
|
|
uint64_t availablemilli;
|
|
int32_t readyflag;
|
|
}; extern struct teleport_info TELEPORT;
|
|
|
|
struct InstantDEX_info
|
|
{
|
|
int32_t readyflag,numhist;
|
|
struct txinds777_info *history;
|
|
}; extern struct InstantDEX_info INSTANTDEX;
|
|
|
|
#define MAX_SERVERNAME 128
|
|
struct relayargs
|
|
{
|
|
//char *(*commandprocessor)(struct relayargs *args,uint8_t *msg,int32_t len);
|
|
char name[16],endpoint[MAX_SERVERNAME];
|
|
int32_t sock,type,bindflag,sendtimeout,recvtimeout;
|
|
};
|
|
struct _relay_info { int32_t sock,num,mytype,desttype; struct endpoint connections[1 << CONNECTION_NUMBITS]; };
|
|
struct direct_connection { char handler[16]; struct endpoint epbits; int32_t sock; };
|
|
|
|
struct relay_info
|
|
{
|
|
struct _relay_info active;
|
|
struct nn_pollfd pfd[16];
|
|
int32_t readyflag,subclient,lbclient,lbserver,servicesock,pubglobal,pubrelays,numservers;
|
|
}; extern struct relay_info RELAYS;
|
|
|
|
void expand_epbits(char *endpoint,struct endpoint epbits);
|
|
struct endpoint calc_epbits(char *transport,uint32_t ipbits,uint16_t port,int32_t type);
|
|
int upnpredirect(const char* eport, const char* iport, const char* proto, const char* description);
|
|
|
|
void *myaligned_alloc(uint64_t allocsize);
|
|
int32_t aligned_free(void *alignedptr);
|
|
|
|
void *portable_thread_create(void *funcp,void *argp);
|
|
void randombytes(unsigned char *x,long xlen);
|
|
double milliseconds(void);
|
|
void msleep(uint32_t milliseconds);
|
|
#define portable_sleep(n) msleep((n) * 1000)
|
|
|
|
int32_t getline777(char *line,int32_t max);
|
|
char *bitcoind_RPC(char **retstrp,char *debugstr,char *url,char *userpass,char *command,char *args);
|
|
uint16_t wait_for_myipaddr(char *ipaddr);
|
|
void process_userinput(char *line);
|
|
|
|
int32_t init_socket(char *suffix,char *typestr,int32_t type,char *_bindaddr,char *_connectaddr,int32_t timeout);
|
|
int32_t shutdown_plugsocks(union endpoints *socks);
|
|
int32_t nn_local_broadcast(int32_t pushsock,uint64_t instanceid,int32_t flags,uint8_t *retstr,int32_t len);
|
|
//char *poll_local_endpoints(int32_t *lenp,int32_t pullsock);
|
|
struct endpoint nn_directepbits(char *retbuf,char *transport,char *ipaddr,uint16_t port);
|
|
int32_t nn_directsend(struct endpoint epbits,uint8_t *msg,int32_t len);
|
|
|
|
void ensure_directory(char *dirname);
|
|
uint32_t is_ipaddr(char *str);
|
|
|
|
uint64_t calc_ipbits(char *ipaddr);
|
|
void expand_ipbits(char *ipaddr,uint64_t ipbits);
|
|
char *ipbits_str(uint64_t ipbits);
|
|
char *ipbits_str2(uint64_t ipbits);
|
|
struct sockaddr_in conv_ipbits(uint64_t ipbits);
|
|
uint32_t conv_domainname(char *ipaddr,char *domain);
|
|
int32_t ismyaddress(char *server);
|
|
|
|
void set_endpointaddr(char *transport,char *endpoint,char *domain,uint16_t port,int32_t type);
|
|
int32_t nn_portoffset(int32_t type);
|
|
|
|
char *plugin_method(int32_t sock,char **retstrp,int32_t localaccess,char *plugin,char *method,uint64_t daemonid,uint64_t instanceid,char *origargstr,int32_t len,int32_t timeout,char *tokenstr);
|
|
//char *nn_direct(char *ipaddr,uint8_t *data,int32_t len);
|
|
//char *nn_publish(uint8_t *data,int32_t len,int32_t nostr);
|
|
//char *nn_allrelays(uint8_t *data,int32_t len,int32_t timeoutmillis,char *localresult);
|
|
char *nn_loadbalanced(uint8_t *data,int32_t len);
|
|
char *relays_jsonstr(char *jsonstr,cJSON *argjson);
|
|
struct daemon_info *find_daemoninfo(int32_t *indp,char *name,uint64_t daemonid,uint64_t instanceid);
|
|
int32_t init_pingpong_queue(struct pingpong_queue *ppq,char *name,int32_t (*action)(),queue_t *destq,queue_t *errorq);
|
|
int32_t process_pingpong_queue(struct pingpong_queue *ppq,void *argptr);
|
|
uint8_t *replace_forwarder(char *pluginbuf,uint8_t *data,int32_t *datalenp);
|
|
int32_t nn_socket_status(int32_t sock,int32_t timeoutmillis);
|
|
|
|
char *nn_busdata_processor(uint8_t *msg,int32_t len);
|
|
void busdata_init(int32_t sendtimeout,int32_t recvtimeout,int32_t firstiter);
|
|
int32_t busdata_poll();
|
|
char *busdata_sync(uint32_t *noncep,char *jsonstr,char *broadcastmode,char *destNXTaddr);
|
|
int32_t parse_ipaddr(char *ipaddr,char *ip_port);
|
|
int32_t construct_tokenized_req(uint32_t *noncep,char *tokenized,char *cmdjson,char *NXTACCTSECRET,char *broadcastmode);
|
|
int32_t validate_token(struct destbuf *forwarder,struct destbuf *pubkey,struct destbuf *NXTaddr,char *tokenizedtxt,int32_t strictflag);
|
|
uint32_t busdata_nonce(int32_t *leveragep,char *str,char *broadcaststr,int32_t maxmillis,uint32_t nonce);
|
|
int32_t nonce_leverage(char *broadcaststr);
|
|
char *get_broadcastmode(cJSON *json,char *broadcastmode);
|
|
cJSON *serviceprovider_json();
|
|
int32_t nn_createsocket(char *endpoint,int32_t bindflag,char *name,int32_t type,uint16_t port,int32_t sendtimeout,int32_t recvtimeout);
|
|
int32_t nn_lbsocket(int32_t maxmillis,int32_t port,uint16_t globalport,uint16_t relaysport);
|
|
int32_t OS_init();
|
|
int32_t nn_settimeouts(int32_t sock,int32_t sendtimeout,int32_t recvtimeout);
|
|
int32_t is_duplicate_tag(uint64_t tag);
|
|
void portable_OS_init();
|
|
void telepathic_PM(char *destNXT,char *PM);
|
|
extern queue_t TelepathyQ;
|
|
uint16_t parse_endpoint(int32_t *ip6flagp,char *transport,char *ipbuf,char *retbuf,char *endpoint,uint16_t default_port);
|
|
cJSON *protocols_json(char *protocol);
|
|
|
|
uint64_t millistamp();
|
|
uint32_t calc_timeslot(int32_t millidiff,int32_t timeres);
|
|
|
|
int32_t _add_related(uint64_t *assetids,int32_t n,uint64_t refbaseid,uint64_t refrelid,int32_t exchangeid,uint64_t baseid,uint64_t relid);
|
|
int32_t add_related(uint64_t *assetids,int32_t n,uint64_t supported[][2],long num,int32_t exchangeid,uint64_t baseid,uint64_t relid);
|
|
int32_t add_NXT_assetids(uint64_t *assetids,int32_t n,uint64_t assetid);
|
|
int32_t add_exchange_assetid(uint64_t *assetids,int32_t n,uint64_t baseid,uint64_t relid,int32_t exchangeid);
|
|
int32_t add_exchange_assetids(uint64_t *assetids,int32_t n,uint64_t refassetid,uint64_t baseid,uint64_t relid,int32_t exchangeid,char *symbolmap[][8],int32_t numsymbols);
|
|
double prices777_baseprice(uint32_t timestamp,int32_t basenum);
|
|
|
|
|
|
#define MAXTIMEDIFF 60
|
|
|
|
#endif
|
|
#else
|
|
#ifndef crypto777_system777_c
|
|
#define crypto777_system777_c
|
|
|
|
#ifndef crypto777_system777_h
|
|
#define DEFINES_ONLY
|
|
#include "../common/system777.c"
|
|
#undef DEFINES_ONLY
|
|
#endif
|
|
|
|
struct nn_clock
|
|
{
|
|
uint64_t last_tsc;
|
|
uint64_t last_time;
|
|
} Global_timer;
|
|
|
|
void msleep(uint32_t milliseconds)
|
|
{
|
|
void nn_sleep (int milliseconds);
|
|
nn_sleep(milliseconds);
|
|
}
|
|
typedef void (portable_thread_func)(void *);
|
|
/*
|
|
double milliseconds()
|
|
{
|
|
uint64_t nn_clock_now (struct nn_clock *self);
|
|
return(nn_clock_now(&Global_timer));
|
|
}*/
|
|
|
|
|
|
struct nn_thread
|
|
{
|
|
portable_thread_func *routine;
|
|
void *arg;
|
|
void *handle;
|
|
};
|
|
void nn_thread_init (struct nn_thread *self,portable_thread_func *routine,void *arg);
|
|
void nn_thread_term (struct nn_thread *self);
|
|
|
|
/*static uint64_t _align16(uint64_t ptrval) { if ( (ptrval & 15) != 0 ) ptrval += 16 - (ptrval & 15); return(ptrval); }
|
|
|
|
void *myaligned_alloc(uint64_t allocsize)
|
|
{
|
|
void *ptr,*realptr; uint64_t tmp;
|
|
realptr = calloc(1,allocsize + 16 + sizeof(realptr));
|
|
tmp = _align16((long)realptr + sizeof(ptr));
|
|
memcpy(&ptr,&tmp,sizeof(ptr));
|
|
memcpy((void *)((long)ptr - sizeof(realptr)),&realptr,sizeof(realptr));
|
|
printf("aligned_alloc(%llu) realptr.%p -> ptr.%p, diff.%ld\n",(long long)allocsize,realptr,ptr,((long)ptr - (long)realptr));
|
|
return(ptr);
|
|
}
|
|
|
|
int32_t aligned_free(void *ptr)
|
|
{
|
|
void *realptr;
|
|
long diff;
|
|
if ( ((long)ptr & 0xf) != 0 )
|
|
{
|
|
printf("misaligned ptr.%p being aligned_free\n",ptr);
|
|
return(-1);
|
|
}
|
|
memcpy(&realptr,(void *)((long)ptr - sizeof(realptr)),sizeof(realptr));
|
|
diff = ((long)ptr - (long)realptr);
|
|
if ( diff < (long)sizeof(ptr) || diff > 32 )
|
|
{
|
|
printf("ptr %p and realptr %p too far apart %ld\n",ptr,realptr,diff);
|
|
return(-2);
|
|
}
|
|
//printf("aligned_free: ptr %p -> realptr %p %ld\n",ptr,realptr,diff);
|
|
free(realptr);
|
|
return(0);
|
|
}*/
|
|
|
|
void *portable_thread_create(void *funcp,void *argp)
|
|
{
|
|
//void nn_thread_term(struct nn_thread *self);
|
|
void nn_thread_init(struct nn_thread *self,portable_thread_func *routine,void *arg);
|
|
struct nn_thread *ptr;
|
|
ptr = (struct nn_thread *)malloc(sizeof(*ptr));
|
|
nn_thread_init(ptr,(portable_thread_func *)funcp,argp);
|
|
return(ptr);
|
|
}
|
|
/*
|
|
struct queueitem *queueitem(char *str)
|
|
{
|
|
struct queueitem *item = calloc(1,sizeof(struct queueitem) + strlen(str) + 1);
|
|
strcpy((char *)((long)item + sizeof(struct queueitem)),str);
|
|
return(item);
|
|
}
|
|
|
|
struct queueitem *queuedata(void *data,int32_t datalen)
|
|
{
|
|
struct queueitem *item = calloc(1,sizeof(struct queueitem) + datalen);
|
|
memcpy((char *)((long)item + sizeof(struct queueitem)),data,datalen);
|
|
return(item);
|
|
}
|
|
|
|
void free_queueitem(void *itemptr) { free((void *)((long)itemptr - sizeof(struct queueitem))); }
|
|
|
|
void lock_queue(queue_t *queue)
|
|
{
|
|
if ( queue->initflag == 0 )
|
|
{
|
|
portable_mutex_init(&queue->mutex);
|
|
queue->initflag = 1;
|
|
}
|
|
portable_mutex_lock(&queue->mutex);
|
|
}
|
|
|
|
void queue_enqueue(char *name,queue_t *queue,struct queueitem *item)
|
|
{
|
|
if ( queue->list == 0 && name != 0 && name[0] != 0 )
|
|
safecopy(queue->name,name,sizeof(queue->name));
|
|
if ( item == 0 )
|
|
{
|
|
printf("FATAL type error: queueing empty value\n");//, getchar();
|
|
return;
|
|
}
|
|
lock_queue(queue);
|
|
DL_APPEND(queue->list,item);
|
|
portable_mutex_unlock(&queue->mutex);
|
|
//printf("name.(%s) append.%p list.%p\n",name,item,queue->list);
|
|
}
|
|
|
|
void *queue_dequeue(queue_t *queue,int32_t offsetflag)
|
|
{
|
|
struct queueitem *item = 0;
|
|
lock_queue(queue);
|
|
if ( queue->list != 0 )
|
|
{
|
|
item = queue->list;
|
|
DL_DELETE(queue->list,item);
|
|
//printf("name.(%s) dequeue.%p list.%p\n",queue->name,item,queue->list);
|
|
}
|
|
portable_mutex_unlock(&queue->mutex);
|
|
if ( item != 0 && offsetflag != 0 )
|
|
return((void *)((long)item + sizeof(struct queueitem)));
|
|
else return(item);
|
|
}
|
|
|
|
void *queue_delete(queue_t *queue,struct queueitem *copy,int32_t copysize)
|
|
{
|
|
struct queueitem *item = 0;
|
|
lock_queue(queue);
|
|
if ( queue->list != 0 )
|
|
{
|
|
DL_FOREACH(queue->list,item)
|
|
{
|
|
if ( memcmp((void *)((long)item + sizeof(struct queueitem)),(void *)((long)item + sizeof(struct queueitem)),copysize) == 0 )
|
|
{
|
|
DL_DELETE(queue->list,item);
|
|
return(item);
|
|
}
|
|
}
|
|
//printf("name.(%s) dequeue.%p list.%p\n",queue->name,item,queue->list);
|
|
}
|
|
portable_mutex_unlock(&queue->mutex);
|
|
return(0);
|
|
}
|
|
|
|
void *queue_free(queue_t *queue)
|
|
{
|
|
struct queueitem *item = 0;
|
|
lock_queue(queue);
|
|
if ( queue->list != 0 )
|
|
{
|
|
DL_FOREACH(queue->list,item)
|
|
{
|
|
DL_DELETE(queue->list,item);
|
|
free(item);
|
|
}
|
|
//printf("name.(%s) dequeue.%p list.%p\n",queue->name,item,queue->list);
|
|
}
|
|
portable_mutex_unlock(&queue->mutex);
|
|
return(0);
|
|
}
|
|
|
|
void *queue_clone(queue_t *clone,queue_t *queue,int32_t size)
|
|
{
|
|
struct queueitem *ptr,*item = 0;
|
|
lock_queue(queue);
|
|
if ( queue->list != 0 )
|
|
{
|
|
DL_FOREACH(queue->list,item)
|
|
{
|
|
ptr = calloc(1,sizeof(*ptr));
|
|
memcpy(ptr,item,size);
|
|
queue_enqueue(queue->name,clone,ptr);
|
|
}
|
|
//printf("name.(%s) dequeue.%p list.%p\n",queue->name,item,queue->list);
|
|
}
|
|
portable_mutex_unlock(&queue->mutex);
|
|
return(0);
|
|
}
|
|
|
|
int32_t queue_size(queue_t *queue)
|
|
{
|
|
int32_t count = 0;
|
|
struct queueitem *tmp;
|
|
lock_queue(queue);
|
|
DL_COUNT(queue->list,tmp,count);
|
|
portable_mutex_unlock(&queue->mutex);
|
|
return count;
|
|
}*/
|
|
|
|
int32_t init_pingpong_queue(struct pingpong_queue *ppq,char *name,int32_t (*action)(),queue_t *destq,queue_t *errorq)
|
|
{
|
|
ppq->name = name;
|
|
ppq->destqueue = destq;
|
|
ppq->errorqueue = errorq;
|
|
ppq->action = action;
|
|
ppq->offset = 1;
|
|
return(queue_size(&ppq->pingpong[0]) + queue_size(&ppq->pingpong[1])); // init mutex side effect
|
|
}
|
|
|
|
// seems a bit wastefull to do all the two iter queueing/dequeuing with threadlock overhead
|
|
// however, there is assumed to be plenty of CPU time relative to actual blockchain events
|
|
// also this method allows for adding of parallel threads without worry
|
|
int32_t process_pingpong_queue(struct pingpong_queue *ppq,void *argptr)
|
|
{
|
|
int32_t iter,retval,freeflag = 0;
|
|
void *ptr;
|
|
//printf("%p process_pingpong_queue.%s %d %d\n",ppq,ppq->name,queue_size(&ppq->pingpong[0]),queue_size(&ppq->pingpong[1]));
|
|
for (iter=0; iter<2; iter++)
|
|
{
|
|
while ( (ptr= queue_dequeue(&ppq->pingpong[iter],ppq->offset)) != 0 )
|
|
{
|
|
if ( Debuglevel > 2 )
|
|
printf("%s pingpong[%d].%p action.%p\n",ppq->name,iter,ptr,ppq->action);
|
|
retval = (*ppq->action)(&ptr,argptr);
|
|
if ( retval == 0 )
|
|
queue_enqueue(ppq->name,&ppq->pingpong[iter ^ 1],ptr,0);
|
|
else if ( ptr != 0 )
|
|
{
|
|
if ( retval < 0 )
|
|
{
|
|
if ( Debuglevel > 0 )
|
|
printf("%s iter.%d errorqueue %p vs %p\n",ppq->name,iter,ppq->errorqueue,&ppq->pingpong[0]);
|
|
if ( ppq->errorqueue == &ppq->pingpong[0] )
|
|
queue_enqueue(ppq->name,&ppq->pingpong[iter ^ 1],ptr,0);
|
|
else if ( ppq->errorqueue != 0 )
|
|
queue_enqueue(ppq->name,ppq->errorqueue,ptr,0);
|
|
else freeflag = 1;
|
|
}
|
|
else if ( ppq->destqueue != 0 )
|
|
{
|
|
if ( Debuglevel > 0 )
|
|
printf("%s iter.%d destqueue %p vs %p\n",ppq->name,iter,ppq->destqueue,&ppq->pingpong[0]);
|
|
if ( ppq->destqueue == &ppq->pingpong[0] )
|
|
queue_enqueue(ppq->name,&ppq->pingpong[iter ^ 1],ptr,0);
|
|
else if ( ppq->destqueue != 0 )
|
|
queue_enqueue(ppq->name,ppq->destqueue,ptr,0);
|
|
else freeflag = 1;
|
|
}
|
|
if ( freeflag != 0 )
|
|
{
|
|
if ( ppq->offset == 0 )
|
|
free(ptr);
|
|
else free_queueitem(ptr);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return(queue_size(&ppq->pingpong[0]) + queue_size(&ppq->pingpong[0]));
|
|
}
|
|
|
|
uint16_t wait_for_myipaddr(char *ipaddr)
|
|
{
|
|
uint16_t port = 0;
|
|
printf("need a portable way to find IP addr\n");
|
|
//getchar();
|
|
return(port);
|
|
}
|
|
|
|
int32_t ismyaddress(char *server)
|
|
{
|
|
char ipaddr[64]; uint32_t ipbits;
|
|
if ( strncmp(server,"tcp://",6) == 0 )
|
|
server += 6;
|
|
else if ( strncmp(server,"ws://",5) == 0 )
|
|
server += 5;
|
|
if ( (ipbits= is_ipaddr(server)) != 0 )
|
|
{
|
|
if ( strcmp(server,SUPERNET.myipaddr) == 0 || calc_ipbits(SUPERNET.myipaddr) == ipbits )
|
|
{
|
|
printf("(%s) MATCHES me (%s)\n",server,SUPERNET.myipaddr);
|
|
return(1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( SUPERNET.hostname[0] != 0 && strcmp(SUPERNET.hostname,server) == 0 )
|
|
return(1);
|
|
else if ( (ipbits= conv_domainname(ipaddr,server)) != 0 || SUPERNET.my64bits == ipbits )
|
|
return(1);
|
|
else if ( (strcmp(SUPERNET.myipaddr,ipaddr) == 0 || strcmp(SUPERNET.hostname,ipaddr) == 0) )
|
|
return(1);
|
|
}
|
|
//printf("(%s) is not me (%s)\n",server,SUPERNET.myipaddr);
|
|
return(0);
|
|
}
|
|
|
|
int32_t nn_local_broadcast(int32_t sock,uint64_t instanceid,int32_t flags,uint8_t *retstr,int32_t len)
|
|
{
|
|
int32_t i,sendlen,errs = 0;
|
|
if ( sock >= 0 )
|
|
{
|
|
for (i=0; i<10; i++)
|
|
if ( (nn_socket_status(sock,1) & NN_POLLOUT) != 0 )
|
|
break;
|
|
if ( (sendlen= nn_send(sock,(char *)retstr,len,0)) <= 0 )
|
|
errs++, printf("sending to socket.%d sendlen.%d len.%d (%s) [%s]\n",sock,sendlen,len,nn_strerror(nn_errno()),retstr);
|
|
else if ( Debuglevel > 2 )
|
|
printf("nn_local_broadcast SENT.(%s) len.%d sendlen.%d vs strlen.%ld instanceid.%llu -> sock.%d\n",retstr,len,sendlen,(long)strlen((char *)retstr),(long long)instanceid,sock);
|
|
}
|
|
return(errs);
|
|
}
|
|
|
|
int32_t plugin_result(char *retbuf,cJSON *json,uint64_t tag)
|
|
{
|
|
char *error,*result;
|
|
error = cJSON_str(cJSON_GetObjectItem(json,"error"));
|
|
result = cJSON_str(cJSON_GetObjectItem(json,"result"));
|
|
if ( error != 0 || result != 0 )
|
|
{
|
|
sprintf(retbuf,"{\"result\":\"completed\",\"tag\":\"%llu\"}",(long long)tag);
|
|
return(1);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
double estimate_completion(double startmilli,int32_t processed,int32_t numleft)
|
|
{
|
|
double elapsed,rate;
|
|
if ( processed <= 0 )
|
|
return(0.);
|
|
elapsed = (milliseconds() - startmilli);
|
|
rate = (elapsed / processed);
|
|
if ( rate <= 0. )
|
|
return(0.);
|
|
//printf("numleft %d rate %f\n",numleft,rate);
|
|
return(numleft * rate);
|
|
}
|
|
|
|
void clear_alloc_space(struct alloc_space *mem,int32_t alignflag)
|
|
{
|
|
memset(mem->ptr,0,mem->size);
|
|
mem->used = 0;
|
|
mem->alignflag = alignflag;
|
|
}
|
|
|
|
void rewind_alloc_space(struct alloc_space *mem,int32_t flags)
|
|
{
|
|
if ( (flags & 1) != 0 )
|
|
clear_alloc_space(mem,1);
|
|
else mem->used = 0;
|
|
mem->alignflag = ((flags & ~1) != 0);
|
|
}
|
|
|
|
struct alloc_space *init_alloc_space(struct alloc_space *mem,void *ptr,long size,int32_t flags)
|
|
{
|
|
if ( mem == 0 )
|
|
mem = calloc(1,size + sizeof(*mem)), ptr = mem->space;
|
|
mem->size = size;
|
|
mem->ptr = ptr;
|
|
rewind_alloc_space(mem,flags);
|
|
return(mem);
|
|
}
|
|
|
|
void *memalloc(struct alloc_space *mem,long size,int32_t clearflag)
|
|
{
|
|
void *ptr = 0;
|
|
if ( (mem->used + size) > mem->size )
|
|
{
|
|
printf("alloc: (mem->used %ld + %ld size) %ld > %ld mem->size\n",mem->used,size,(mem->used + size),mem->size);
|
|
while ( 1 )
|
|
portable_sleep(1);
|
|
}
|
|
ptr = (void *)((long)mem->ptr + mem->used);
|
|
mem->used += size;
|
|
if ( clearflag != 0 )
|
|
memset(ptr,0,size);
|
|
if ( mem->alignflag != 0 && (mem->used & 0xf) != 0 )
|
|
mem->used += 0x10 - (mem->used & 0xf);
|
|
return(ptr);
|
|
}
|
|
|
|
uint64_t millistamp() { return(time(NULL)*1000 + ((uint64_t)milliseconds() % 1000)); }
|
|
uint32_t calc_timeslot(int32_t millidiff,int32_t timeres) { return(((uint32_t)(millistamp() + (timeres>>1) + millidiff) / timeres)); }
|
|
|
|
#define GENESISACCT "1739068987193023818" // NXT-MRCC-2YLS-8M54-3CMAJ
|
|
#define GENESISPUBKEYSTR "1259ec21d31a30898d7cd1609f80d9668b4778e3d97e941044b39f0c44d2e51b"
|
|
#define GENESISPRIVKEYSTR "88a71671a6edd987ad9e9097428fc3f169decba3ac8f10da7b24e0ca16803b70"
|
|
#define GENESIS_SECRET "It was a bright cold day in April, and the clocks were striking thirteen."
|
|
|
|
void portable_OS_init()
|
|
{
|
|
uint64_t conv_NXTpassword(unsigned char *mysecret,unsigned char *mypublic,uint8_t *pass,int32_t passlen);
|
|
char hexstr[65]; bits256 privkey,pubkey; uint64_t nxt64bits=0; extern bits256 GENESIS_PUBKEY,GENESIS_PRIVKEY;
|
|
void SaM_PrepareIndices();
|
|
decode_hex(GENESIS_PUBKEY.bytes,sizeof(GENESIS_PUBKEY),GENESISPUBKEYSTR);
|
|
decode_hex(GENESIS_PRIVKEY.bytes,sizeof(GENESIS_PRIVKEY),GENESISPRIVKEYSTR);
|
|
printf("decoded genesis 123\n");
|
|
nxt64bits = conv_NXTpassword(privkey.bytes,pubkey.bytes,(void *)GENESIS_SECRET,(int32_t)strlen(GENESIS_SECRET));
|
|
char pubkeystr[67]; uint8_t pub[33];
|
|
hexstr[0]= 0;
|
|
btc_priv2pub(pub,GENESIS_PRIVKEY.bytes);
|
|
init_hexbytes_noT(pubkeystr,pub,33);
|
|
printf("pubkey.%s %llx\n",pubkeystr,*(long long *)pub);
|
|
printf("%s conv_NXTpassword %llu\n",__TIME__,(long long)nxt64bits);
|
|
if ( nxt64bits != calc_nxt64bits(GENESISACCT) )
|
|
printf("GENESIS_ACCT mismatch %llu != %llu\n",(long long)nxt64bits,(long long)calc_nxt64bits(GENESISACCT));
|
|
if ( memcmp(GENESIS_PUBKEY.bytes,pubkey.bytes,sizeof(pubkey)) != 0 )
|
|
printf("GENESIS_PUBKEY mismatch %llu != %llu\n",(long long)GENESIS_PUBKEY.txid,(long long)pubkey.txid);
|
|
if ( memcmp(GENESIS_PRIVKEY.bytes,privkey.bytes,sizeof(privkey)) != 0 )
|
|
{
|
|
init_hexbytes_noT(hexstr,privkey.bytes,sizeof(privkey));
|
|
//printf("%s GENESIS_PRIVKEY mismatch %llu != %llu\n",hexstr,(long long)GENESIS_PRIVKEY.txid,(long long)privkey.txid);
|
|
}
|
|
printf("%s GENESIS_PRIVKEY %llx GENESIS_PUBKEY %llx\n",hexstr,(long long)GENESIS_PRIVKEY.txid,(long long)GENESIS_PUBKEY.txid);
|
|
OS_init();
|
|
curl_global_init(CURL_GLOBAL_ALL); //init the curl session
|
|
SaM_PrepareIndices();
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
#endif
|
|
|