1015 lines
35 KiB
1015 lines
35 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. *
|
|
* *
|
|
******************************************************************************/
|
|
|
|
#include "iguana777.h"
|
|
#include <sys/stat.h>
|
|
#ifndef MAP_FILE
|
|
#define MAP_FILE 0
|
|
#endif
|
|
|
|
int32_t conv_date(int32_t *secondsp,char *buf);
|
|
|
|
uint32_t OS_conv_datenum(int32_t datenum,int32_t hour,int32_t minute,int32_t second) // datenum+H:M:S -> unix time
|
|
{
|
|
#ifdef __PNACL
|
|
PostMessage("timegm is not implemented\n");
|
|
return(0);
|
|
#else
|
|
struct tm t;
|
|
memset(&t,0,sizeof(t));
|
|
t.tm_year = (datenum / 10000) - 1900, t.tm_mon = ((datenum / 100) % 100) - 1, t.tm_mday = (datenum % 100);
|
|
t.tm_hour = hour, t.tm_min = minute, t.tm_sec = second;
|
|
return((uint32_t)timegm(&t));
|
|
#endif
|
|
}
|
|
|
|
int32_t OS_conv_unixtime(int32_t *secondsp,time_t timestamp) // gmtime -> datenum + number of seconds
|
|
{
|
|
struct tm t; int32_t datenum; uint32_t checktime; char buf[64];
|
|
t = *gmtime(×tamp);
|
|
strftime(buf, sizeof(buf), "%Y-%m-%dT%H:%M:%SZ",&t); //printf("%s\n",buf);
|
|
datenum = conv_date(secondsp,buf);
|
|
if ( (checktime= OS_conv_datenum(datenum,*secondsp/3600,(*secondsp%3600)/60,*secondsp%60)) != timestamp )
|
|
{
|
|
printf("error: timestamp.%lu -> (%d + %d) -> %u\n",timestamp,datenum,*secondsp,checktime);
|
|
return(-1);
|
|
}
|
|
return(datenum);
|
|
}
|
|
|
|
char *OS_mvstr()
|
|
{
|
|
#ifdef __WIN32
|
|
return("rename");
|
|
#else
|
|
return("mv");
|
|
#endif
|
|
}
|
|
|
|
char *OS_rmstr() { return("rm"); }
|
|
|
|
int32_t os_supports_mappedfiles() { return(1); }
|
|
|
|
int32_t portable_truncate(char *fname,long filesize) { return(truncate(fname,filesize)); }
|
|
|
|
char *iguana_compatible_path(char *str)
|
|
{
|
|
return(str);
|
|
}
|
|
|
|
void ensure_directory(char *dirname)
|
|
{
|
|
FILE *fp; int32_t retval; char fname[512];
|
|
iguana_removefile(dirname,0);
|
|
sprintf(fname,"%s/.tmpmarker",dirname);
|
|
if ( (fp= fopen(iguana_compatible_path(fname),"rb")) == 0 )
|
|
{
|
|
if ( (fp= fopen(iguana_compatible_path(dirname),"rb")) == 0 )
|
|
{
|
|
retval = mkdir(dirname,511);
|
|
printf("mkdir.(%s) retval.%d errno.%d %s\n",dirname,retval,errno,strerror(errno));
|
|
} else fclose(fp), printf("dirname.(%s) exists\n",dirname);
|
|
if ( (fp= fopen(fname,"wb")) != 0 )
|
|
fclose(fp), printf("created.(%s)\n",fname);
|
|
else printf("cant create.(%s) errno.%d %s\n",fname,errno,strerror(errno));
|
|
} else fclose(fp), printf("%s exists\n",fname);
|
|
}
|
|
|
|
int32_t iguana_renamefile(char *fname,char *newfname)
|
|
{
|
|
char cmd[1024];
|
|
sprintf(cmd,"%s %s %s",OS_mvstr(),fname,newfname);
|
|
return(system(cmd));
|
|
}
|
|
|
|
int32_t iguana_removefile(char *fname,int32_t scrubflag)
|
|
{
|
|
FILE *fp;
|
|
char cmdstr[1024];
|
|
long i,fpos;
|
|
if ( (fp= fopen(iguana_compatible_path(fname),"rb+")) != 0 )
|
|
{
|
|
//printf("delete(%s)\n",fname);
|
|
if ( scrubflag != 0 )
|
|
{
|
|
fseek(fp,0,SEEK_END);
|
|
fpos = ftell(fp);
|
|
rewind(fp);
|
|
for (i=0; i<fpos; i++)
|
|
fputc(0xff,fp);
|
|
fflush(fp);
|
|
}
|
|
fclose(fp);
|
|
sprintf(cmdstr,"%s %s",OS_rmstr(),fname);
|
|
if ( system(iguana_compatible_path(cmdstr)) != 0 )
|
|
printf("error deleting file.(%s)\n",cmdstr);
|
|
else return(1);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
uint64_t iguana_filesize(char *fname)
|
|
{
|
|
FILE *fp; uint64_t fsize = 0;
|
|
if ( (fp= fopen(fname,"rb")) != 0 )
|
|
{
|
|
fseek(fp,0,SEEK_END);
|
|
fsize = ftell(fp);
|
|
fclose(fp);
|
|
}
|
|
return(fsize);
|
|
}
|
|
|
|
int32_t iguana_compare_files(char *fname,char *fname2)
|
|
{
|
|
int32_t offset,errs = 0;
|
|
long len,len2;
|
|
char buf[8192],buf2[8192];
|
|
FILE *fp,*fp2;
|
|
if ( (fp= fopen(iguana_compatible_path(fname),"rb")) != 0 )
|
|
{
|
|
if ( (fp2= fopen(iguana_compatible_path(fname2),"rb")) != 0 )
|
|
{
|
|
while ( (len= fread(buf,1,sizeof(buf),fp)) > 0 && (len2= fread(buf2,1,sizeof(buf2),fp2)) == len )
|
|
if ( (offset= memcmp(buf,buf2,len)) != 0 )
|
|
printf("compare error at offset.%d: (%s) src.%ld vs. (%s) dest.%ld\n",offset,fname,ftell(fp),fname2,ftell(fp2)), errs++;
|
|
fclose(fp2);
|
|
}
|
|
fclose(fp);
|
|
}
|
|
return(errs);
|
|
}
|
|
|
|
int64_t iguana_copyfile(char *src,char *dest,int32_t cmpflag)
|
|
{
|
|
int64_t allocsize,len = -1;
|
|
char *buf;
|
|
FILE *srcfp,*destfp;
|
|
if ( (srcfp= fopen(iguana_compatible_path(src),"rb")) != 0 )
|
|
{
|
|
if ( (destfp= fopen(iguana_compatible_path(dest),"wb")) != 0 )
|
|
{
|
|
allocsize = 1024 * 1024 * 128L;
|
|
buf = mycalloc('F',1,allocsize);
|
|
while ( (len= fread(buf,1,allocsize,srcfp)) > 0 )
|
|
if ( (long)fwrite(buf,1,len,destfp) != len )
|
|
printf("write error at (%s) src.%ld vs. (%s) dest.%ld\n",src,ftell(srcfp),dest,ftell(destfp));
|
|
len = ftell(destfp);
|
|
fclose(destfp);
|
|
myfree(buf,allocsize);
|
|
}
|
|
fclose(srcfp);
|
|
}
|
|
if ( len < 0 || (cmpflag != 0 && iguana_compare_files(src,dest) != 0) )
|
|
printf("Error copying files (%s) -> (%s)\n",src,dest), len = -1;
|
|
return(len);
|
|
}
|
|
|
|
void *map_file(char *fname,long *filesizep,int32_t enablewrite)
|
|
{
|
|
//void *mmap64(void *addr,size_t len,int32_t prot,int32_t flags,int32_t fildes,off_t off);
|
|
int32_t fd,rwflags,flags = MAP_FILE|MAP_SHARED;
|
|
uint64_t filesize;
|
|
void *ptr = 0;
|
|
*filesizep = 0;
|
|
if ( enablewrite != 0 )
|
|
fd = open(fname,O_RDWR);
|
|
else fd = open(fname,O_RDONLY);
|
|
if ( fd < 0 )
|
|
{
|
|
printf("map_file: error opening enablewrite.%d %s\n",enablewrite,fname);
|
|
return(0);
|
|
}
|
|
if ( *filesizep == 0 )
|
|
filesize = (uint64_t)lseek(fd,0,SEEK_END);
|
|
else filesize = *filesizep;
|
|
rwflags = PROT_READ;
|
|
if ( enablewrite != 0 )
|
|
rwflags |= PROT_WRITE;
|
|
//#if __i386__
|
|
ptr = mmap(0,filesize,rwflags,flags,fd,0);
|
|
//#else
|
|
//ptr = mmap64(0,filesize,rwflags,flags,fd,0);
|
|
//#endif
|
|
close(fd);
|
|
if ( ptr == 0 || ptr == MAP_FAILED )
|
|
{
|
|
printf("map_file.write%d: mapping %s failed? mp %p\n",enablewrite,fname,ptr);
|
|
return(0);
|
|
}
|
|
*filesizep = filesize;
|
|
return(ptr);
|
|
}
|
|
|
|
int32_t iguana_releasemap(void *ptr,uint64_t filesize)
|
|
{
|
|
int32_t retval;
|
|
if ( ptr == 0 )
|
|
{
|
|
printf("release_map_file: null ptr\n");
|
|
return(-1);
|
|
}
|
|
retval = munmap(ptr,filesize);
|
|
if ( retval != 0 )
|
|
printf("release_map_file: munmap error %p %llu: err %d\n",ptr,(long long)filesize,retval);
|
|
return(retval);
|
|
}
|
|
|
|
void _iguana_closemap(struct iguana_mappedptr *mp)
|
|
{
|
|
if ( mp->actually_allocated != 0 && mp->fileptr != 0 )
|
|
myaligned_free(mp->fileptr,mp->allocsize);
|
|
else if ( mp->fileptr != 0 )
|
|
iguana_releasemap(mp->fileptr,mp->allocsize);
|
|
mp->fileptr = 0;
|
|
mp->closetime = (uint32_t)time(NULL);
|
|
mp->opentime = 0;
|
|
}
|
|
|
|
void iguana_closemap(struct iguana_mappedptr *mp)
|
|
{
|
|
struct iguana_mappedptr tmp;
|
|
tmp = *mp;
|
|
_iguana_closemap(mp);
|
|
memset(mp,0,sizeof(*mp));
|
|
mp->actually_allocated = tmp.actually_allocated;
|
|
mp->allocsize = tmp.allocsize;
|
|
mp->rwflag = tmp.rwflag;
|
|
strcpy(mp->fname,tmp.fname);
|
|
}
|
|
|
|
int32_t iguana_syncmap(struct iguana_mappedptr *mp,long len)
|
|
{
|
|
//static long Sync_total;
|
|
int32_t err = -1;
|
|
if ( mp->actually_allocated != 0 )
|
|
return(0);
|
|
if ( mp->fileptr != 0 && mp->dirty != 0 )
|
|
{
|
|
if ( len == 0 )
|
|
len = mp->allocsize;
|
|
err = msync(mp->fileptr,len,MS_SYNC);
|
|
if ( err != 0 )
|
|
printf("sync (%s) len %llu, err %d errno.%d\n",mp->fname,(long long)len,err,errno);
|
|
//Sync_total += len;
|
|
mp->dirty = 0;
|
|
}
|
|
return(err);
|
|
}
|
|
|
|
long iguana_ensurefilesize(char *fname,long filesize,int32_t truncateflag)
|
|
{
|
|
FILE *fp;
|
|
char *zeroes;
|
|
long i,n,allocsize = 0;
|
|
//printf("ensure_filesize.(%s) %ld %s | ",fname,filesize,mbstr(filesize));
|
|
if ( (fp= fopen(iguana_compatible_path(fname),"rb")) != 0 )
|
|
{
|
|
fseek(fp,0,SEEK_END);
|
|
allocsize = ftell(fp);
|
|
fclose(fp);
|
|
//printf("(%s) exists size.%ld\n",fname,allocsize);
|
|
}
|
|
else
|
|
{
|
|
//printf("try to create.(%s)\n",fname);
|
|
if ( (fp= fopen(iguana_compatible_path(fname),"wb")) != 0 )
|
|
fclose(fp);
|
|
}
|
|
if ( allocsize < filesize )
|
|
{
|
|
//printf("filesize.%ld is less than %ld\n",filesize,allocsize);
|
|
if ( (fp= fopen(iguana_compatible_path(fname),"ab")) != 0 )
|
|
{
|
|
zeroes = myaligned_alloc(16L*1024*1024);
|
|
memset(zeroes,0,16*1024*1024);
|
|
n = filesize - allocsize;
|
|
while ( n > 16*1024*1024 )
|
|
{
|
|
fwrite(zeroes,1,16*1024*1024,fp);
|
|
n -= 16*1024*1024;
|
|
fprintf(stderr,"+");
|
|
}
|
|
for (i=0; i<n; i++)
|
|
fputc(0,fp);
|
|
fclose(fp);
|
|
myaligned_free(zeroes,16L*1024*1024);
|
|
}
|
|
return(filesize);
|
|
}
|
|
else if ( allocsize*truncateflag > filesize )
|
|
{
|
|
portable_truncate(fname,filesize);
|
|
return(filesize);
|
|
}
|
|
else return(allocsize);
|
|
}
|
|
|
|
int32_t iguana_openmap(struct iguana_mappedptr *mp)
|
|
{
|
|
uint64_t allocsize = mp->allocsize;
|
|
if ( mp->actually_allocated != 0 )
|
|
{
|
|
if ( mp->fileptr == 0 )
|
|
mp->fileptr = myaligned_alloc(mp->allocsize);
|
|
else memset(mp->fileptr,0,mp->allocsize);
|
|
return(0);
|
|
}
|
|
else
|
|
{
|
|
if ( mp->fileptr != 0 )
|
|
{
|
|
//printf("opening already open mappedptr, pending %p\n",mp->pending);
|
|
iguana_closemap(mp);
|
|
}
|
|
mp->allocsize = allocsize;
|
|
// printf("calling map_file with expected %ld\n",mp->allocsize);
|
|
mp->fileptr = map_file(mp->fname,&mp->allocsize,mp->rwflag);
|
|
if ( mp->fileptr == 0 || mp->allocsize != allocsize )
|
|
{
|
|
//printf("error mapping(%s) ptr %p mapped %ld vs allocsize %ld\n",mp->fname,mp->fileptr,mp->allocsize,allocsize);
|
|
return(-1);
|
|
}
|
|
mp->closetime = 0;
|
|
mp->opentime = (uint32_t)time(NULL);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
void *iguana_mappedptr(void **ptrp,struct iguana_mappedptr *mp,uint64_t allocsize,int32_t rwflag,char *fname)
|
|
{
|
|
uint64_t filesize;
|
|
mp->actually_allocated = !os_supports_mappedfiles();
|
|
if ( fname != 0 )
|
|
{
|
|
if ( strcmp(mp->fname,fname) == 0 )
|
|
{
|
|
if ( mp->fileptr != 0 )
|
|
{
|
|
iguana_releasemap(mp->fileptr,mp->allocsize);
|
|
mp->fileptr = 0;
|
|
}
|
|
iguana_openmap(mp);
|
|
if ( ptrp != 0 )
|
|
(*ptrp) = mp->fileptr;
|
|
return(mp->fileptr);
|
|
}
|
|
strcpy(mp->fname,fname);
|
|
}
|
|
else mp->actually_allocated = 1;
|
|
mp->rwflag = rwflag;
|
|
mp->allocsize = allocsize;
|
|
if ( rwflag != 0 && mp->actually_allocated == 0 && allocsize != 0 )
|
|
allocsize = iguana_ensurefilesize(fname,allocsize,0);
|
|
if ( iguana_openmap(mp) != 0 )
|
|
{
|
|
char str[65];
|
|
//printf("init_mappedptr %s.rwflag.%d | ",fname,rwflag);
|
|
if ( allocsize != 0 )
|
|
printf("error mapping(%s) rwflag.%d ptr %p mapped %llu vs allocsize %llu %s\n",fname,rwflag,mp->fileptr,(long long)mp->allocsize,(long long)allocsize,mbstr(str,allocsize));
|
|
else allocsize = mp->allocsize;
|
|
if ( rwflag != 0 && allocsize != mp->allocsize )
|
|
{
|
|
filesize = mp->allocsize;
|
|
if ( mp->fileptr != 0 )
|
|
iguana_releasemap(mp->fileptr,mp->allocsize);
|
|
mp->allocsize = allocsize;
|
|
mp->changedsize = (allocsize - filesize);
|
|
iguana_openmap(mp);
|
|
if ( mp->fileptr == 0 || mp->allocsize != allocsize )
|
|
{
|
|
printf("SECOND error mapping(%s) ptr %p mapped %llu vs allocsize %llu\n",fname,mp->fileptr,(long long)mp->allocsize,(long long)allocsize);
|
|
exit(-1);
|
|
}
|
|
}
|
|
}
|
|
if ( ptrp != 0 )
|
|
(*ptrp) = mp->fileptr;
|
|
return(mp->fileptr);
|
|
}
|
|
|
|
//int64_t iguana_packetsallocated(struct iguana_info *coin) { return(coin->R.packetsallocated - coin->R.packetsfreed); };
|
|
|
|
void *filealloc(struct iguana_mappedptr *M,char *fname,struct iguana_memspace *mem,long size)
|
|
{
|
|
//printf("mem->used %ld size.%ld | size.%ld\n",mem->used,size,mem->size);
|
|
//printf("filemalloc.(%s) new space.%ld %s\n",fname,mem->size,mbstr(size));
|
|
memset(M,0,sizeof(*M));
|
|
mem->totalsize = size;
|
|
if ( iguana_mappedptr(0,M,mem->totalsize,1,fname) == 0 )
|
|
{
|
|
printf("couldnt create mapped file.(%s)\n",fname);
|
|
exit(-1);
|
|
}
|
|
mem->ptr = M->fileptr;
|
|
mem->used = 0;
|
|
return(M->fileptr);
|
|
}
|
|
|
|
void *iguana_tmpalloc(struct iguana_info *coin,char *name,struct iguana_memspace *mem,long origsize)
|
|
{
|
|
char fname[1024]; void *ptr; long size;
|
|
#ifdef __PNACL
|
|
return(mycalloc('T',1,origsize));
|
|
#endif
|
|
//portable_mutex_lock(&mem->mutex);
|
|
if ( origsize != 0 && (mem->M.fileptr == 0 || (mem->used + origsize) > mem->totalsize) )
|
|
{
|
|
coin->TMPallocated += origsize;
|
|
memset(&mem->M,0,sizeof(mem->M));
|
|
sprintf(fname,"tmp/%s/%s.%d",coin->symbol,name,mem->counter), iguana_compatible_path(fname);
|
|
mem->counter++;
|
|
if ( mem->totalsize == 0 )
|
|
{
|
|
//if ( strcmp(name,"recv") == 0 )
|
|
// mem->size = IGUANA_RSPACE_SIZE * ((strcmp(coin->symbol,"BTC") == 0) ? 16 : 1);
|
|
// else
|
|
#ifdef IGUANA_MAPHASHTABLES
|
|
mem->totalsize = (1024 * 1024 * 128);
|
|
#else
|
|
mem->size = (1024 * 1024 * 16);
|
|
#endif
|
|
}
|
|
//if ( coin->R.RSPACE.size == 0 )
|
|
// coin->R.RSPACE.size = mem->size;
|
|
if ( mem->totalsize > origsize )
|
|
size = mem->totalsize;
|
|
else size = origsize;
|
|
fprintf(stderr,"filealloc.(%s) -> ",fname);
|
|
if ( filealloc(&mem->M,fname,mem,size) == 0 )
|
|
{
|
|
printf("couldnt map tmpfile %s\n",fname);
|
|
return(0);
|
|
}
|
|
fprintf(stderr,"created\n");
|
|
}
|
|
ptr = iguana_memalloc(mem,origsize,1);
|
|
//portable_mutex_unlock(&mem->mutex);
|
|
return(ptr);
|
|
}
|
|
|
|
#ifdef oldway
|
|
void *iguana_kvfixiterator(struct iguana_info *coin,struct iguanakv *kv,struct iguana_kvitem *item,uint64_t args,void *key,void *value,int32_t valuesize)
|
|
{
|
|
int64_t offset = (int64_t)args;
|
|
if ( args != 0 && (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
{
|
|
item->hh.key = (void *)((uint64_t)item->hh.key + (int64_t)offset);
|
|
//printf("iguana_kvfixiterator rawind.%-5d: (%s)\n",item->rawind,bits256_str(*(bits256 *)item->hh.key));
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
void *_iguana_kvensure(struct iguana_info *coin,int32_t rwflag,int32_t *maxindp,char *name,char *fname,double mult,int32_t incr,uint32_t ind,struct iguana_mappedptr *M,int32_t HDDvaluesize)
|
|
{
|
|
long needed,prevsize = 0;
|
|
needed = (ind + 2) * HDDvaluesize;
|
|
//printf("ensure.%s ind.%d needed.%ld origptr.%p\n",kv->name,ind,needed,origptr);
|
|
if ( needed > M->allocsize )
|
|
{
|
|
needed = (((needed * mult) + incr * HDDvaluesize) / HDDvaluesize) * HDDvaluesize;
|
|
printf("REMAP.%s %llu -> %ld [%ld] (%s) (ind.%d mult.%f incr.%d size.%d\n",name,(long long)M->allocsize,needed,(long)(needed - M->allocsize)/HDDvaluesize,fname,ind,mult,incr,HDDvaluesize);
|
|
if ( M->fileptr != 0 )
|
|
{
|
|
iguana_syncmap(M,0);
|
|
iguana_releasemap(M->fileptr,M->allocsize);
|
|
M->fileptr = 0, prevsize = M->allocsize;
|
|
M->allocsize = 0;
|
|
}
|
|
needed = iguana_ensurefilesize(fname,needed,0);
|
|
}
|
|
if ( M->fileptr == 0 )
|
|
{
|
|
if ( iguana_mappedptr(0,M,0,rwflag,fname) != 0 )
|
|
{
|
|
if ( 1 && prevsize > M->allocsize )
|
|
memset((void *)((uint64_t)M->fileptr + prevsize),0,(M->allocsize - prevsize));
|
|
printf("%p %s maxitems.%llu (MEMsize.%ld / itemsize.%d) prevsize.%ld needed.%ld\n",M->fileptr,name,(long long)*maxindp,(long)M->allocsize,HDDvaluesize,prevsize,needed);
|
|
}
|
|
}
|
|
return(M->fileptr);
|
|
}
|
|
|
|
void *iguana_kvensure(struct iguana_info *coin,struct iguanakv *kv,uint32_t ind)
|
|
{
|
|
char fname[512]; void *origptr; int32_t maxitemind,n,rwflag = 1;
|
|
if ( (int32_t)ind < 0 )
|
|
ind = 0;
|
|
origptr = kv->HDDitems;
|
|
kv->HDDitems = _iguana_kvensure(coin,rwflag,&kv->maxitemind,kv->name,kv->fname,kv->mult,kv->incr,ind,&kv->M,kv->HDDvaluesize);
|
|
if ( kv->HDDitemsp != 0 )
|
|
*kv->HDDitemsp = kv->HDDitems;
|
|
n = (int32_t)(kv->M.allocsize / kv->HDDvaluesize);
|
|
if ( ind < n )
|
|
ind = n;
|
|
if ( kv->valuesize3 != 0 )
|
|
{
|
|
sprintf(fname,"%s3",kv->fname);
|
|
if ( (kv->HDDitems3= _iguana_kvensure(coin,rwflag,&maxitemind,kv->name,fname,kv->mult,kv->incr,ind,&kv->M3,kv->valuesize3)) == 0 )
|
|
{
|
|
printf("HDDitems3 null ptr for %s\n",fname);
|
|
return(0);
|
|
}
|
|
//printf("third file\n");
|
|
if ( kv->HDDitems3p != 0 )
|
|
*kv->HDDitems3p = kv->HDDitems3;
|
|
}
|
|
if ( kv->valuesize2 != 0 )
|
|
{
|
|
sprintf(fname,"%s2",kv->fname);
|
|
if ( (kv->HDDitems2= _iguana_kvensure(coin,rwflag,&maxitemind,kv->name,fname,kv->mult,kv->incr,ind,&kv->M2,kv->valuesize2)) == 0 )
|
|
{
|
|
printf("HDDitems3 null ptr for %s\n",fname);
|
|
return(0);
|
|
}
|
|
//printf("second file\n");
|
|
if ( kv->HDDitems2p != 0 )
|
|
*kv->HDDitems2p = kv->HDDitems2;
|
|
}
|
|
if ( origptr != 0 && origptr != kv->M.fileptr && (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
{
|
|
if ( iguana_kviterate(coin,kv,(int64_t)((int64_t)kv->HDDitems - (int64_t)origptr),iguana_kvfixiterator) != 0 )
|
|
printf("ERROR relinked pointers\n");
|
|
else printf("hashtable relinked\n");
|
|
kv->incr *= 1.25;
|
|
}
|
|
return(kv->HDDitems);
|
|
}
|
|
|
|
void *iguana_kvsaveiterator(struct iguana_info *coin,struct iguanakv *kv,struct iguana_kvitem *item,uint64_t args,void *key,void *value,int32_t valuesize)
|
|
{
|
|
FILE *fp = (FILE *)args;
|
|
if ( args != 0 )
|
|
{
|
|
if ( fwrite(value,1,kv->RAMvaluesize,fp) != kv->RAMvaluesize )
|
|
{
|
|
printf("Error saving key.[%d]\n",kv->RAMvaluesize);
|
|
return(value);
|
|
}
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
long iguana_kvsave(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
FILE *fp; long retval = -1; char fname[512],oldfname[512],cmd[512];
|
|
sprintf(fname,"%s.tmp",kv->name);
|
|
if ( (fp= fopen(fname,"wb")) != 0 )
|
|
{
|
|
if ( iguana_kviterate(coin,kv,(uint64_t)fp,iguana_kvsaveiterator) == 0 )
|
|
{
|
|
printf("save %ld to HDD\n",ftell(fp));
|
|
retval = ftell(fp);
|
|
}
|
|
else printf("error saving item at %ld\n",ftell(fp));
|
|
fclose(fp);
|
|
} else printf("error creating(%s)\n",fname);
|
|
if ( retval > 0 )
|
|
{
|
|
sprintf(oldfname,"%s.%u",kv->name,(uint32_t)time(NULL));
|
|
sprintf(cmd,"%s %s %s",OS_mvstr(),kv->name,oldfname);
|
|
retval = system(cmd);
|
|
sprintf(cmd,"%s %s %s",OS_mvstr(),fname,kv->name);
|
|
retval = system(cmd);
|
|
}
|
|
return(retval);
|
|
}
|
|
|
|
int32_t iguana_valuesize(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
int32_t valuesize = kv->RAMvaluesize;
|
|
return(valuesize);
|
|
}
|
|
|
|
int32_t iguana_itemsize(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
int32_t itemsize = sizeof(struct iguana_kvitem);
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) == 0 )
|
|
itemsize += iguana_valuesize(coin,kv);
|
|
return(itemsize);
|
|
}
|
|
|
|
void *iguana_itemvalue(struct iguana_info *coin,void **itemkeyp,struct iguanakv *kv,void *ptr,struct iguana_kvitem *item)
|
|
{
|
|
void *itemvalue = 0;
|
|
*itemkeyp = 0;
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
{
|
|
if ( ptr != 0 )
|
|
{
|
|
*itemkeyp = &((uint8_t *)ptr)[kv->keyoffset];
|
|
itemvalue = ptr;
|
|
} else printf("error setting itemvalue\n");
|
|
}
|
|
if ( *itemkeyp == 0 )
|
|
{
|
|
*itemkeyp = item->keyvalue;
|
|
itemvalue = &item->keyvalue[kv->keysize];
|
|
}
|
|
return(itemvalue);
|
|
}
|
|
|
|
void *iguana_itemptr(struct iguana_info *coin,struct iguanakv *kv,uint32_t itemind)
|
|
{
|
|
void *ptr = kv->HDDitems;
|
|
//printf("%s itemind.%d\n",kv->name,itemind);
|
|
if ( ptr == 0 || itemind >= kv->maxitemind )
|
|
{
|
|
kv->HDDitems = iguana_kvensure(coin,kv,itemind+1);
|
|
if ( (ptr= kv->HDDitems) == 0 )
|
|
{
|
|
printf("SECOND ERROR %s overflow? %p itemind.%llu vs max.%llu\n",kv->name,ptr,(long long)itemind,(long long)kv->maxitemind);
|
|
return(0);
|
|
}
|
|
}
|
|
ptr = (void *)((uint64_t)ptr + kv->HDDvaluesize*itemind);
|
|
return(ptr);
|
|
}
|
|
|
|
void iguana_copy(struct iguana_info *coin,struct iguanakv *kv,int32_t rwflag,void *itemvalue,void *value,int32_t valuesize)
|
|
{
|
|
void *src,*dest;
|
|
/*if ( (kv->flags & IGUANA_MAPPED_ITEM) != 0 && rwflag != 0 )
|
|
{
|
|
itemvalue = (void *)(((uint64_t))itemvalue + sizeof(UT_hash_handle));
|
|
value = (void *)(((uint64_t))value + sizeof(UT_hash_handle));
|
|
valuesize -= sizeof(UT_hash_handle);
|
|
//printf("value.%p itemvalue.%p valuesize.%d\n",value,itemvalue,valuesize);
|
|
}*/
|
|
if ( rwflag != 0 )
|
|
src = value, dest = itemvalue;
|
|
else src = itemvalue, dest = value;
|
|
memcpy(dest,src,valuesize);
|
|
}
|
|
|
|
static int32_t iguana_RWmmap(int32_t writeflag,void *value,struct iguana_info *coin,struct iguanakv *kv,uint32_t itemind)
|
|
{
|
|
static const uint8_t zeroes[4096];
|
|
void *ptr,*itemvalue=0; int32_t i,itemsize,valuesize,retval = 0;
|
|
itemsize = iguana_itemsize(coin,kv);
|
|
valuesize = iguana_valuesize(coin,kv);
|
|
if ( (ptr= iguana_itemptr(coin,kv,itemind)) != 0 )
|
|
{
|
|
//itemvalue = iguana_itemvalue(coin,&itemkey,kv,ptr,0);
|
|
if ( writeflag != 0 )
|
|
{
|
|
//struct iguana777_addrinfo *A; struct coin_offsets B,tmpB;
|
|
//itemsize = kv->RAMvaluesize;
|
|
/*if ( strcmp(sp->name,"addrinfos") == 0 )
|
|
{
|
|
A = ptr;
|
|
itemsize = (sizeof(*A) - sizeof(A->coinaddr) + A->addrlen + A->scriptlen);
|
|
}*/
|
|
if ( writeflag == 1 && (kv->flags & IGUANA_VOLATILE) == 0 )
|
|
{
|
|
/*if ( strcmp(sp->name,"blocks") == 0 )
|
|
{
|
|
memcpy(&B,ptr,sizeof(B));
|
|
memcpy(&tmpB,value,sizeof(tmpB));
|
|
if ( memcmp(&B.blockhash.bytes,zeroes,sizeof(B.blockhash)) == 0 && memcmp(&B.merkleroot.bytes,zeroes,sizeof(B.merkleroot)) == 0 )
|
|
B.blockhash = tmpB.blockhash, B.merkleroot = tmpB.merkleroot, ptr = &B;
|
|
}*/
|
|
if ( 0 && memcmp(value,itemvalue,valuesize) != 0 && valuesize <= sizeof(zeroes) )
|
|
{
|
|
if ( memcmp(itemvalue,zeroes,valuesize) != 0 )
|
|
{
|
|
printf("\n");
|
|
for (i=0; i<valuesize; i++)
|
|
printf("%02x ",((uint8_t *)itemvalue)[i]);
|
|
printf("existing.%s %d <-- overwritten\n",kv->name,kv->RAMvaluesize);
|
|
for (i=0; i<valuesize; i++)
|
|
printf("%02x ",((uint8_t *)value)[i]);
|
|
printf("new value.%s %d itemind.%u fileptr.%p ptr.%p\n",kv->name,kv->RAMvaluesize,itemind,kv->M.fileptr,ptr);
|
|
}
|
|
}
|
|
}
|
|
if ( kv->fp == 0 )
|
|
iguana_copy(coin,kv,writeflag,ptr,value,valuesize);
|
|
else // all ready for rb+ fp and readonly mapping, but need to init properly
|
|
{
|
|
/*fseek(sp->fp,(uint64_t)sp->itemsize * itemind,SEEK_SET);
|
|
fwrite(value,1,valuesize,sp->fp);
|
|
if ( memcmp(itemvalue,value,valuesize) != 0 )
|
|
printf("FATAL: write mmap error\n"), getchar();*/
|
|
printf("iguana_RWmmap: need to test sp->fp first\n"), exit(1);
|
|
}
|
|
}
|
|
else iguana_copy(coin,kv,writeflag,ptr,value,valuesize);
|
|
} else retval = -2;
|
|
return(retval);
|
|
}
|
|
|
|
void iguana_kvlock(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
if ( kv->threadsafe != 0 )
|
|
portable_mutex_lock(&kv->KVmutex);
|
|
}
|
|
|
|
void iguana_kvunlock(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
if ( kv->threadsafe != 0 )
|
|
portable_mutex_unlock(&kv->KVmutex);
|
|
}
|
|
|
|
int32_t iguana_kvdelete(struct iguana_info *coin,struct iguanakv *kv,void *key)
|
|
{
|
|
int32_t retval = -1; struct iguana_kvitem *ptr = 0;
|
|
if ( kv == 0 )
|
|
return(-1);
|
|
iguana_kvlock(coin,kv);
|
|
HASH_FIND(hh,kv->hashtables[((uint8_t *)key)[kv->keysize>>1]],key,kv->keysize,ptr);
|
|
if ( ptr != 0 )
|
|
{
|
|
HASH_DELETE(hh,kv->hashtables[((uint8_t *)key)[kv->keysize>>1]],ptr);
|
|
if ( (kv->flags & IGUANA_MAPPED_ITEM) == 0 )
|
|
myfree(ptr,iguana_itemsize(coin,kv));
|
|
retval = 0;
|
|
}
|
|
iguana_kvlock(coin,kv);
|
|
return(retval);
|
|
}
|
|
|
|
void *_iguana_kvread(struct iguana_info *coin,struct iguanakv *kv,void *key,void *value,uint32_t *itemindp)
|
|
{
|
|
void *itemkey,*itemvalue,*ptr=0; int32_t valuesize,itemind = 0; struct iguana_kvitem *item = 0;
|
|
if ( kv == 0 )
|
|
{
|
|
printf("iguana_kvread: null ramkv??\n");
|
|
return(0);
|
|
}
|
|
if ( kv->keysize == 0 )
|
|
{
|
|
printf("kvwrite %s only supports itemind MMap access\n",kv->name);
|
|
return(0);
|
|
}
|
|
valuesize = iguana_valuesize(coin,kv);
|
|
//printf("search for [%llx] keysize.%d\n",*(long long *)key,kv->keysize);
|
|
HASH_FIND(hh,kv->hashtables[((uint8_t *)key)[kv->keysize>>1]],key,kv->keysize,item);
|
|
if ( item != 0 )
|
|
{
|
|
if ( itemindp != 0 )
|
|
*itemindp = itemind = item->hh.itemind;
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
ptr = iguana_itemptr(coin,kv,itemind);
|
|
if ( (itemvalue= iguana_itemvalue(coin,&itemkey,kv,ptr,item)) != 0 )
|
|
{
|
|
//printf("itemind.%d: key.%p value.%p valuesize.%d\n",itemind,itemkey,itemvalue,valuesize);
|
|
iguana_copy(coin,kv,0,itemvalue,value,valuesize);
|
|
}
|
|
else printf("_kvread null itemvalue for itemind.%d\n",itemind);
|
|
return(value);
|
|
}
|
|
//printf("cache miss %s\n",bits256_str(*(bits256 *)key));
|
|
if ( itemindp != 0 )
|
|
*itemindp = 0;
|
|
return(0);
|
|
}
|
|
|
|
void *_iguana_kvwrite(struct iguana_info *coin,struct iguanakv *kv,void *key,void *value,uint32_t *itemindp)
|
|
{
|
|
void *ptr=0,*itemvalue=0,*itemkey=0; struct iguana_kvitem *item = 0; uint32_t itemsize,valuesize,itemind = *itemindp;
|
|
if ( kv == 0 )
|
|
return(0);
|
|
valuesize = iguana_valuesize(coin,kv);
|
|
itemsize = iguana_itemsize(coin,kv);
|
|
if ( kv->keysize == 0 )
|
|
{
|
|
printf("kvwrite %s only supports itemind MMap access\n",kv->name);
|
|
return(0);
|
|
}
|
|
HASH_FIND(hh,kv->hashtables[((uint8_t *)key)[kv->keysize>>1]],key,kv->keysize,item);
|
|
if ( item != 0 )
|
|
{
|
|
if ( 0 && itemind != item->hh.itemind && itemind != (uint32_t)-1 )
|
|
{
|
|
printf("%s override itemind %d -> %d\n",kv->name,item->hh.itemind,itemind);
|
|
item->hh.itemind = itemind;
|
|
} else itemind = item->hh.itemind;
|
|
*itemindp = itemind;
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
ptr = iguana_itemptr(coin,kv,itemind);
|
|
if ( (itemvalue= iguana_itemvalue(coin,&itemkey,kv,ptr,item)) != 0 )
|
|
{
|
|
if ( memcmp(itemvalue,value,valuesize) != 0 )
|
|
{
|
|
//printf("%s: item.%d updating %p\n",kv->name,item->hh.itemind,key);
|
|
iguana_copy(coin,kv,1,itemvalue,value,valuesize);
|
|
kv->updated++;
|
|
}
|
|
} else printf("kvwrite null itemvalue itemind.%d\n",itemind);
|
|
return(item);
|
|
}
|
|
else
|
|
{
|
|
kv->numkeys++;
|
|
if ( itemind == (uint32_t)-1 )
|
|
itemind = kv->numkeys;
|
|
*itemindp = itemind;
|
|
if ( item == 0 )
|
|
{
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
ptr = iguana_itemptr(coin,kv,itemind);
|
|
item = ((kv->flags & IGUANA_MAPPED_ITEM) == 0) ? mycalloc('I',1,itemsize) : iguana_tmpalloc(coin,kv->name,&kv->HASHPTRS,itemsize);
|
|
if ( item == 0 )
|
|
printf("fatal out of mem error\n"), getchar();
|
|
}
|
|
itemvalue = iguana_itemvalue(coin,&itemkey,kv,ptr,item);
|
|
}
|
|
if ( itemvalue != 0 && itemkey != 0 && item != 0 )
|
|
{
|
|
valuesize = iguana_valuesize(coin,kv);
|
|
item->hh.itemind = itemind;
|
|
iguana_copy(coin,kv,1,itemvalue,value,valuesize);
|
|
memcpy(itemkey,key,kv->keysize);
|
|
//if ( strcmp(kv->name,"txids") == 0 )
|
|
//printf("add.(%s) itemind.%d kv->numkeys.%d keysize.%d (%s) valuesize.%d:%d\n",kv->name,itemind,kv->numkeys,kv->keysize,bits256_str(*(bits256 *)key),kv->HDDvaluesize,kv->RAMvaluesize);
|
|
HASH_ADD_KEYPTR(hh,kv->hashtables[((uint8_t *)itemkey)[kv->keysize>>1]],itemkey,kv->keysize,item);
|
|
kv->M.dirty++;
|
|
HASH_FIND(hh,kv->hashtables[((uint8_t *)key)[kv->keysize>>1]],key,kv->keysize,item);
|
|
if ( kv->dispflag != 0 || item == 0 || item->hh.itemind != itemind )
|
|
fprintf(stderr,">> %s found item.%p iguana_kvwrite numkeys.%d kv.(%p) table.%p write kep.%p size.%d, %p value.(%08x) size.%d itemind.%d:%d\n",kv->name,item,kv->numkeys,key,kv->hashtables[((uint8_t *)itemkey)[kv->keysize>>1]],itemkey,kv->keysize,itemvalue,itemvalue!=0?calc_crc32(0,itemvalue,valuesize):0,valuesize,item!=0?item->hh.itemind:0,itemind);
|
|
if ( item != 0 )
|
|
return(value);
|
|
else printf("null item after find kvwrite error\n"), getchar();
|
|
} else printf("kvwrite pointer error %p %p %p\n",itemkey,itemvalue,item), getchar();
|
|
return(0);
|
|
}
|
|
|
|
void *iguana_kvread(struct iguana_info *coin,struct iguanakv *kv,void *key,void *value,uint32_t *itemindp)
|
|
{
|
|
void *retptr = 0;
|
|
// if ( strcmp(kv->name,"txids") == 0 )
|
|
// printf("iguana_kvread.(%s) key.%p keysize.%d flag.%d itemind.%d\n",kv->name,key,kv->keysize,kv->flags,*itemindp);
|
|
portable_mutex_lock(&kv->MMmutex);
|
|
if ( key == 0 || kv->keysize == 0 )
|
|
{
|
|
if ( iguana_RWmmap(0,value,coin,kv,*itemindp) == 0 )
|
|
retptr = value;
|
|
else printf("%s %d vs %d RMmmap.0 error\n",kv->name,*itemindp,(int32_t)(kv->M.allocsize/kv->HDDvaluesize));
|
|
} else retptr = _iguana_kvread(coin,kv,key,value,itemindp);
|
|
portable_mutex_unlock(&kv->MMmutex);
|
|
return(retptr);
|
|
}
|
|
|
|
void *iguana_kvwrite(struct iguana_info *coin,struct iguanakv *kv,void *key,void *value,uint32_t *itemindp)
|
|
{
|
|
void *retptr = 0;
|
|
portable_mutex_lock(&kv->MMmutex);
|
|
if ( key == 0 || kv->keysize == 0 )
|
|
{
|
|
kv->M.dirty++;
|
|
if ( iguana_RWmmap(1,value,coin,kv,*itemindp) == 0 )
|
|
retptr = value;
|
|
else printf("%s %d vs %d RMmmap.1 error\n",kv->name,*itemindp,(int32_t)(kv->M.allocsize/kv->HDDvaluesize));
|
|
} else retptr = _iguana_kvwrite(coin,kv,key,value,itemindp);
|
|
portable_mutex_unlock(&kv->MMmutex);
|
|
return(retptr);
|
|
}
|
|
|
|
int32_t iguana_kvchecktable(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
uint32_t itemind,checkind; int32_t err = 0; uint8_t key[8192];
|
|
for (itemind=1; itemind<=kv->numkeys; itemind++)
|
|
{
|
|
if ( iguana_RWmmap(0,kv->space,coin,kv,itemind) == 0 )
|
|
{
|
|
if ( kv->keysize != 0 && kv->keysize < sizeof(key) )
|
|
{
|
|
memcpy(key,(void *)((long)kv->space + kv->keyoffset),kv->keysize);
|
|
if ( _iguana_kvread(coin,kv,key,kv->space,&checkind) == 0 || checkind != itemind )
|
|
{
|
|
printf("kvread.%s miscompares checkind.%d vs %d\n",kv->name,checkind,itemind);
|
|
err++;
|
|
}
|
|
}
|
|
} else err++, printf("%s itemind.%d doesnt map properly\n",kv->name,itemind);
|
|
}
|
|
return(-err);
|
|
}
|
|
|
|
void *iguana_kviterate(struct iguana_info *coin,struct iguanakv *kv,uint64_t args,void *(*iterator)(struct iguana_info *coin,struct iguanakv *kv,struct iguana_kvitem *item,uint64_t args,void *key,void *value,int32_t valuesize))
|
|
{
|
|
struct iguana_kvitem *item,*tmp; int32_t t; void *ptr=0,*itemvalue,*itemkey=0,*retval = 0;
|
|
if ( kv == 0 )
|
|
return(0);
|
|
for (t=0; t<0x100; t++)
|
|
{
|
|
HASH_ITER(hh,kv->hashtables[t],item,tmp)
|
|
{
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
ptr = iguana_itemptr(coin,kv,item->hh.itemind);
|
|
if ( (itemvalue= iguana_itemvalue(coin,&itemkey,kv,ptr,item)) != 0 && itemkey != 0 )
|
|
{
|
|
if ( (retval= (*iterator)(coin,kv,item,args,itemkey,itemvalue,kv->RAMvaluesize)) != 0 )
|
|
return(retval);
|
|
} else return(retval);
|
|
}
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
int32_t iguana_kvtruncate(struct iguana_info *coin,struct iguanakv *kv,uint32_t maxitemind)
|
|
{
|
|
struct iguana_kvitem *item,*tmp; int32_t t,n = 0;
|
|
if ( kv->numkeys < maxitemind )
|
|
return(-1);
|
|
for (t=0; t<0x100; t++)
|
|
{
|
|
HASH_ITER(hh,kv->hashtables[t],item,tmp)
|
|
{
|
|
if ( item->hh.itemind >= maxitemind )
|
|
{
|
|
HASH_DEL(kv->hashtables[t],item);
|
|
if ( (kv->flags & IGUANA_MAPPED_ITEM) == 0 )
|
|
myfree(item,iguana_itemsize(coin,kv));
|
|
n++;
|
|
}
|
|
}
|
|
}
|
|
printf(">>>>>>>>>> kv.%s truncated.%d to maxitemind.%d\n",kv->name,n,maxitemind);
|
|
kv->numkeys = maxitemind;
|
|
return(iguana_kvchecktable(coin,kv));
|
|
}
|
|
|
|
void iguana_kvfree(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
struct iguana_kvitem *ptr,*tmp; int32_t t;
|
|
if ( kv != 0 )
|
|
{
|
|
iguana_kvlock(coin,kv);
|
|
for (t=0; t<0x100; t++)
|
|
{
|
|
HASH_ITER(hh,kv->hashtables[t],ptr,tmp)
|
|
{
|
|
HASH_DEL(kv->hashtables[t],ptr);
|
|
if ( (kv->flags & IGUANA_MAPPED_ITEM) == 0 )
|
|
myfree(ptr,iguana_itemsize(coin,kv));
|
|
}
|
|
}
|
|
iguana_kvunlock(coin,kv);
|
|
myfree(kv,sizeof(*kv));
|
|
}
|
|
}
|
|
|
|
int32_t iguana_kvclone(struct iguana_info *coin,struct iguanakv *clone,struct iguanakv *kv)
|
|
{
|
|
void *ptr=0,*itemkey,*itemvalue; struct iguana_kvitem *item,*tmp; int32_t t,n = 0;
|
|
printf("need to add support for mapped data\n");
|
|
if ( kv != 0 )
|
|
{
|
|
for (t=0; t<0x100; t++)
|
|
{
|
|
HASH_ITER(hh,kv->hashtables[t],item,tmp)
|
|
{
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
ptr = iguana_itemptr(coin,kv,item->hh.itemind);
|
|
if ( (itemvalue= iguana_itemvalue(coin,&itemkey,kv,ptr,item)) != 0 && itemkey != 0 )
|
|
{
|
|
iguana_kvwrite(coin,clone,itemkey,itemvalue,&item->hh.itemind);
|
|
n++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return(n);
|
|
}
|
|
|
|
int32_t iguana_kvdisp(struct iguana_info *coin,struct iguanakv *kv)
|
|
{
|
|
struct iguana_kvitem *item,*tmp; void *ptr=0,*itemkey,*itemvalue; int32_t t,n = 0; char hexstr[8192];
|
|
printf("iguana_kvdisp.(%s) numkeys.%d\n",kv->name,kv->numkeys);
|
|
if ( kv == 0 )
|
|
return(0);
|
|
for (t=0; t<0x100; t++)
|
|
{
|
|
HASH_ITER(hh,kv->hashtables[t],item,tmp)
|
|
{
|
|
if ( (kv->flags & IGUANA_ITEMIND_DATA) != 0 )
|
|
ptr = iguana_itemptr(coin,kv,item->hh.itemind);
|
|
if ( (itemvalue= iguana_itemvalue(coin,&itemkey,kv,ptr,item)) != 0 )
|
|
{
|
|
init_hexbytes_noT(hexstr,itemvalue,kv->RAMvaluesize);
|
|
char str[65];
|
|
bits256_str(str,*(bits256 *)itemkey);
|
|
printf("itemind.%d %s %s len.%d height.%d\n",item->hh.itemind,str,hexstr,kv->RAMvaluesize,((struct iguana_block *)itemvalue)->height);
|
|
}
|
|
n++;
|
|
}
|
|
}
|
|
printf("iguana_kvdisp.(%s) n.%d items\n",kv->name,n);
|
|
return(n);
|
|
}
|
|
#endif
|
|
|
|
|