/****************************************************************************** * 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. * * * ******************************************************************************/ // iguana_OS has functions that invoke system calls. Whenever possible stdio and similar functions are use and most functions are fully portable and in this file. For things that require OS specific, the call is routed to iguana_OS_portable_* Usually, all but one OS can be handled with the same code, so iguana_OS_portable.c has most of this shared logic and an #ifdef iguana_OS_nonportable.c // time functions are treated separately due to the confusions even in unix #include "OS_portable.h" #include #ifndef _WIN32 #include #endif #ifndef MAP_FILE #define MAP_FILE 0 #endif char *OS_mvstr() { #ifdef __WIN32 return("rename"); #else return("mv"); #endif } void OS_randombytes(unsigned char *x,long xlen) { OS_portable_randombytes(x,xlen); } static double _kb(double n) { return(n / 1024.); } static double _mb(double n) { return(n / (1024.*1024.)); } static double _gb(double n) { return(n / (1024.*1024.*1024.)); } char *mbstr(char *str,double n) { if ( n < 1024*1024*10 ) sprintf(str,"%.3fkb",_kb(n)); else if ( n < 1024*1024*1024 ) sprintf(str,"%.1fMB",_mb(n)); else sprintf(str,"%.2fGB",_gb(n)); return(str); } long myallocated(uint8_t type,long change) { static int64_t Total_allocated,HWM_allocated,Type_allocated[256]; int32_t i; int64_t total = 0; char buf[2049],str[65]; buf[0] = 0; if ( type == 0 && change <= 0 ) { for (i=0; i<256; i++) { if ( Type_allocated[i] != 0 ) { total += Type_allocated[i]; if ( change == 0 ) sprintf(buf+strlen(buf),"(%c %s) ",i,mbstr(str,Type_allocated[i])); } } if ( change == 0 ) { sprintf(buf + strlen(buf),"-> total %lld %s",(long long)total,mbstr(str,total)); printf("%s\n",buf); } } else { Type_allocated[type] += change; Total_allocated += change; if ( Total_allocated > HWM_allocated ) { HWM_allocated = Total_allocated * 1.5; printf("HWM allocated %ld %s\n",(long)Total_allocated,mbstr(str,Total_allocated)); } } return(total); } void *mycalloc(uint8_t type,int32_t n,long itemsize) { #ifdef USE_MYCALLOC //static portable_mutex_t MEMmutex; struct allocitem *item; int64_t allocsize = ((uint64_t)n * itemsize); if ( type == 0 && n == 0 && itemsize == 0 ) { //portable_mutex_init(&MEMmutex); myfree(mycalloc('t',1024,1024 * 32),1024*1024*32); return(0); } //portable_mutex_lock(&MEMmutex); myallocated(type,allocsize); while ( (item= calloc(1,sizeof(struct allocitem) + allocsize + 16)) == 0 ) { char str[65]; printf("mycalloc.%c: need to wait for memory.(%d,%ld) %s to be available\n",type,n,itemsize,mbstr(str,allocsize)); sleep(1); } //printf("calloc origptr.%p retptr.%p size.%ld\n",item,(void *)(long)item + sizeof(*item),allocsize); item->allocsize = (uint32_t)allocsize; item->type = type; //portable_mutex_unlock(&MEMmutex); return((void *)((long)item + sizeof(*item))); #else return(calloc(n,itemsize)); #endif } struct queueitem *queueitem(char *str) { /*struct queueitem *item; int32_t n,allocsize; char *data; uint8_t type = 'y'; //portable_mutex_lock(&MEMmutex); n = (uint32_t)strlen(str) + 1; allocsize = (uint32_t)(sizeof(struct queueitem) + n); myallocated(type,allocsize); while ( (item= calloc(1,allocsize)) == 0 ) { char str[65]; printf("queueitem: need to wait for memory.(%d,%ld) %s to be available\n",n,(long)sizeof(*item),mbstr(str,allocsize)); sleep(1); } item->allocsize = (uint32_t)allocsize; item->type = type; data = (void *)(long)((long)item + sizeof(*item)); memcpy(data,str,n); //printf("(%c) queueitem.%p itemdata.%p n.%d allocsize.%d\n",type,item,data,n,allocsize); //portable_mutex_unlock(&MEMmutex); return(data);*/ struct stritem *sitem; int32_t len; len = (int32_t)strlen(str); sitem = calloc(1,sizeof(*sitem) + len + 16); memcpy(sitem->str,str,len); return(&sitem->DL); } #ifdef USE_MYCALLOC void _myfree(uint8_t type,int32_t origallocsize,void *origptr,int32_t allocsize) { //portable_mutex_lock(&MEMmutex); if ( allocsize == origallocsize ) { myallocated(type,-allocsize); // Type_allocated[type & 0xff] -= allocsize; // Total_allocated -= allocsize; //printf("myfree.%p size.%d %d type %x\n",origptr,allocsize,origallocsize,type); free(origptr); } else { printf("myfree size error %d vs %d at %p\n",allocsize,origallocsize,origptr); static int32_t y,z; printf("div by zero! %d\n",y/z); getchar(); } //portable_mutex_unlock(&MEMmutex); } void myfree(void *_ptr,long allocsize) { struct allocitem *item = (void *)((long)_ptr - sizeof(struct allocitem)); if ( allocsize == 0 ) { printf("myfree zero allocsize %p?\n",_ptr); return; } _myfree(item->type,item->allocsize,item,(uint32_t)allocsize); } /*void free_queueitem(void *itemdata) { struct queueitem *item = (void *)((long)itemdata - sizeof(struct queueitem)); //printf("freeq item.%p itemdata.%p size.%d\n",item,itemdata,item->allocsize); _myfree(item->type,item->allocsize,item,item->allocsize); }*/ void *myrealloc(uint8_t type,void *oldptr,long oldsize,long newsize) { void *newptr; newptr = mycalloc(type,1,newsize); //printf("newptr.%p type.%c oldsize.%ld newsize.%ld\n",newptr,type,oldsize,newsize); if ( oldptr != 0 ) { memcpy(newptr,oldptr,oldsize < newsize ? oldsize : newsize); myfree(oldptr,oldsize); } return(newptr); } #else void myfree(void *_ptr,long allocsize) { free(_ptr); } void *myrealloc(uint8_t type,void *oldptr,long oldsize,long newsize) { return(realloc(oldptr,newsize)); } #endif 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; #if defined(_M_X64) realptr = mycalloc('A', 1, (uint64_t)(allocsize + 16 + sizeof(realptr))); #else realptr = mycalloc('A',1,(long)(allocsize + 16 + sizeof(realptr))); #endif tmp = _align16((long)realptr + sizeof(ptr)); memcpy(&ptr,&tmp,sizeof(ptr)); #if defined(_M_X64) memcpy((void *)((unsigned char *)ptr - sizeof(realptr)), &realptr, sizeof(realptr)); #else memcpy((void *)((long)ptr - sizeof(realptr)),&realptr,sizeof(realptr)); #endif //printf("aligned_alloc(%llu) realptr.%p -> ptr.%p, diff.%ld\n",(long long)allocsize,realptr,ptr,((long)ptr - (long)realptr)); return(ptr); } int32_t myaligned_free(void *ptr,long size) { 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); myfree(realptr,size + 16 + sizeof(realptr)); return(0); } 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)//,int32_t offsetflag) { //struct queueitem *item; if ( queue->name[0] == 0 && name != 0 && name[0] != 0 ) strcpy(queue->name,name);//,sizeof(queue->name)); //fprintf(stderr,"enqueue.(%s) %p\n",queue->name,item); lock_queue(queue); //item = (struct queueitem *)((long)origitem - offsetflag*sizeof(struct queueitem)); DL_APPEND(queue->list,item); portable_mutex_unlock(&queue->mutex); //printf("queue_enqueue name.(%s) origitem.%p append.%p list.%p\n",name,origitem,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; //printf("queue_dequeue name.(%s) dequeue.%p list.%p\n",queue->name,item,queue->list); DL_DELETE(queue->list,item); } 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,int32_t freeitem) { struct allocitem *ptr; struct queueitem *tmp,*item = 0; lock_queue(queue); if ( queue->list != 0 ) { DL_FOREACH_SAFE(queue->list,item,tmp) { ptr = (void *)((long)item - sizeof(struct allocitem)); if ( item == copy || (ptr->allocsize == copysize && memcmp((void *)((long)item + sizeof(struct queueitem)),(void *)((long)item + sizeof(struct queueitem)),copysize) == 0) ) { DL_DELETE(queue->list,item); portable_mutex_unlock(&queue->mutex); //printf("name.(%s) deleted item.%p list.%p\n",queue->name,item,queue->list); //if ( freeitem != 0 ) // myfree(item,copysize); return(item); } } } portable_mutex_unlock(&queue->mutex); return(0); } void *queue_free(queue_t *queue) { struct queueitem *tmp,*item = 0; lock_queue(queue); if ( queue->list != 0 ) { DL_FOREACH_SAFE(queue->list,item,tmp) { DL_DELETE(queue->list,item); myfree(item,sizeof(struct queueitem)); } //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,*tmp,*item = 0; lock_queue(queue); if ( queue->list != 0 ) { DL_FOREACH_SAFE(queue->list,item,tmp) { ptr = mycalloc('c',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; } void iguana_memreset(struct OS_memspace *mem) { mem->used = 0; #ifdef IGUANA_PEERALLOC mem->availptrs = mem->outofptrs = mem->numptrs = 0; memset(mem->ptrs,0,sizeof(mem->ptrs)); memset(mem->maxsizes,0,sizeof(mem->maxsizes)); memset(mem->allocsizes,0,sizeof(mem->allocsizes)); #endif if ( mem->threadsafe != 0 ) portable_mutex_init(&mem->mutex); } void iguana_mempurge(struct OS_memspace *mem) { if ( mem->allocated != 0 && mem->ptr != 0 )//&& mem->totalsize > 0 ) { //printf("mempurge.(%s) %ld\n",mem->name,(long)mem->totalsize); myfree(mem->ptr,mem->totalsize), mem->ptr = 0; } iguana_memreset(mem); mem->totalsize = 0; } void *iguana_meminit(struct OS_memspace *mem,char *name,void *ptr,int64_t totalsize,int32_t threadsafe) { strcpy(mem->name,name); if ( ptr == 0 ) { if ( mem->ptr != 0 && mem->totalsize < totalsize ) { iguana_mempurge(mem); mem->totalsize = 0; //printf("memptr.%p totalsize.%ld vs totalsize.%ld\n",mem->ptr,(long)mem->totalsize,(long)totalsize); } //else printf("mem->ptr.%p mem->totalsize %ld\n",mem->ptr,(long)mem->totalsize); if ( mem->ptr == 0 ) { //static long alloc; //alloc += totalsize; //char str[65]; printf("iguana_meminit.(%s) alloc %s\n",name,mbstr(str,totalsize)); if ( (mem->ptr= mycalloc('d',1,totalsize)) == 0 ) { printf("iguana_meminit: cant get %d bytes\n",(int32_t)totalsize); exit(-1); return(0); } mem->totalsize = totalsize; } //else printf("memptr.%p\n",mem->ptr); //printf("meminit.(%s) %d vs %ld\n",mem->name,(int32_t)totalsize,(long)mem->totalsize); mem->allocated = 1; } else { iguana_mempurge(mem); mem->ptr = ptr; mem->totalsize = totalsize; } mem->threadsafe = threadsafe; mem->alignflag = 4; iguana_memreset(mem); if ( mem->totalsize == 0 ) printf("meminit.%s ILLEGAL STATE null size\n",mem->name), getchar(); return(mem->ptr); } int64_t iguana_memallocated(struct OS_memspace *mem) { int64_t avail = (mem->totalsize - mem->used); #ifdef IGUANA_PEERALLOC int32_t i; for (i=0; inumptrs; i++) if ( mem->allocsizes[i] == 0 ) avail += mem->maxsizes[i]; #endif return(avail); } void *iguana_memalloc(struct OS_memspace *mem,long size,int32_t clearflag) { int32_t modval; void *ptr = 0; if ( mem->ptr == 0 ) return(0); //printf("iguana_memalloc.%s size.%ld used.%llu of %llu, numptrs.%d avail.%d %lld\n",mem->name,size,(long long)mem->used,(long long)mem->totalsize,mem->numptrs,mem->availptrs,(long long)iguana_memallocated(mem)); //if ( mem->threadsafe != 0 ) // portable_mutex_lock(&mem->mutex); #ifdef IGUANA_PEERALLOC if ( mem->availptrs == mem->numptrs && mem->used > (mem->totalsize >> 1) ) iguana_memreset(mem); #endif if ( (mem->used + size) <= mem->totalsize ) { /* * solution to calculate memory address in a portable way * in all platform sizeof(char) / sizeof(uchar) == 1 * @author - fadedreamz@gmail.com */ #if defined(_M_X64) ptr = (void *)((unsigned char *)mem->ptr + mem->used); #else ptr = (void *)(long)(((long)mem->ptr + mem->used)); #endif mem->used += size; if ( size*clearflag != 0 ) memset(ptr,0,size); if ( mem->alignflag != 0 ) { if ( (modval= (mem->used % mem->alignflag)) != 0 ) mem->used += mem->alignflag - modval; } #ifdef IGUANA_PEERALLOC if ( mem->numptrs < sizeof(mem->ptrs)/sizeof(*mem->ptrs) ) { mem->allocsizes[mem->numptrs] = mem->maxsizes[mem->numptrs] = (int32_t)size; mem->ptrs[mem->numptrs++] = ptr; } else { mem->outofptrs++; printf("iguana_memalloc: numptrs.%d outofptrs.%d\n",mem->numptrs,mem->outofptrs); } #endif //printf(">>>>>>>>> USED.%s alloc %ld used %ld alloc.%ld -> %s %p\n",mem->name,size,(long)mem->used,(long)mem->totalsize,mem->name,ptr); } else { printf("error memalloc mem.%p (%s) alloc %ld used %ld totalsize.%ld -> %s %p\n",mem,mem->name,size,(long)mem->used,(long)mem->totalsize,mem->name,ptr); ptr = calloc(1,size); } //if ( mem->threadsafe != 0 ) // portable_mutex_unlock(&mem->mutex); return(ptr); } int64_t iguana_memfree(struct OS_memspace *mem,void *ptr,int32_t size) { #ifdef IGUANA_PEERALLOC int32_t i; int64_t avail = -1; if ( mem->threadsafe != 0 ) portable_mutex_lock(&mem->mutex); for (i=0; inumptrs; i++) { if ( ptr == mem->ptrs[i] ) { if ( mem->allocsizes[i] == size ) { mem->availptrs++; mem->allocsizes[i] = 0; avail = (mem->totalsize - mem->used); //printf("avail %llu\n",(long long)avail); } else printf("iguana_memfree.%s: mismatched size %d for ptr.%p %d\n",mem->name,size,ptr,mem->allocsizes[i]); if ( mem->threadsafe != 0 ) portable_mutex_unlock(&mem->mutex); return(avail); } } if ( mem->threadsafe != 0 ) portable_mutex_unlock(&mem->mutex); printf("iguana_memfree: cant find ptr.%p %d\n",ptr,size); return(avail); #else printf("iguana_free not supported without #define IGUANA_PEERALLOC\n"); return(0); #endif } int32_t OS_truncate(char *fname,long filesize) { return(OS_portable_truncate(fname,filesize)); } char *OS_compatible_path(char *str) { return(OS_portable_path(str)); } int32_t OS_renamefile(char *fname,char *newfname) { return(OS_portable_renamefile(fname,newfname)); } int32_t OS_removefile(char *fname,int32_t scrubflag) { FILE *fp; long i,fpos; char tmp[512]; strcpy(tmp,fname); OS_compatible_path(tmp); if ( (fp= fopen(tmp,"rb+")) != 0 ) { //printf("delete(%s)\n",fname); if ( scrubflag != 0 ) { fseek(fp,0,SEEK_END); fpos = ftell(fp); rewind(fp); for (i=0; i 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 OS_copyfile(char *src,char *dest,int32_t cmpflag) { int64_t allocsize,len = -1; char *buf; FILE *srcfp,*destfp; if ( (srcfp= fopen(OS_compatible_path(src),"rb")) != 0 ) { if ( (destfp= fopen(OS_compatible_path(dest),"wb")) != 0 ) { #ifdef WIN32 allocsize = 1024 * 1024 * 8L; #else allocsize = 1024 * 1024 * 128L; #endif 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 && OS_compare_files(src,dest) != 0) ) printf("Error copying files (%s) -> (%s)\n",src,dest), len = -1; return(len); } int32_t OS_releasemap(void *ptr,unsigned long 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 _OS_closemap(struct OS_mappedptr *mp) { if ( mp->actually_allocated != 0 && mp->fileptr != 0 ) myaligned_free(mp->fileptr,mp->allocsize); else if ( mp->fileptr != 0 ) OS_releasemap(mp->fileptr,mp->allocsize); mp->fileptr = 0; mp->closetime = (uint32_t)time(NULL); mp->opentime = 0; } void OS_closemap(struct OS_mappedptr *mp) { struct OS_mappedptr tmp; tmp = *mp; _OS_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); } long OS_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(OS_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(OS_compatible_path(fname),"wb")) != 0 ) fclose(fp); } if ( allocsize < filesize ) { //printf("filesize.%ld is less than %ld\n",filesize,allocsize); if ( (fp= fopen(OS_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 filesize ) { OS_truncate(fname,filesize); return(filesize); } else return(allocsize); } int32_t OS_openmap(struct OS_mappedptr *mp) { unsigned long 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); OS_closemap(mp); } mp->allocsize = allocsize; // printf("calling map_file with expected %ld\n",mp->allocsize); mp->fileptr = OS_mapfile(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 *OS_mappedptr(void **ptrp,struct OS_mappedptr *mp,unsigned long allocsize,int32_t rwflag,char *fname) { unsigned long filesize; mp->actually_allocated = 0;//!os_supports_mappedfiles(); if ( fname != 0 ) { if ( strcmp(mp->fname,fname) == 0 ) { if ( mp->fileptr != 0 ) { OS_releasemap(mp->fileptr,mp->allocsize); mp->fileptr = 0; } OS_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 = OS_ensurefilesize(fname,allocsize,0); if ( OS_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 ) OS_releasemap(mp->fileptr,mp->allocsize); mp->allocsize = allocsize; mp->changedsize = (allocsize - filesize); OS_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); } void *OS_filealloc(struct OS_mappedptr *M,char *fname,struct OS_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 ( OS_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 *OS_loadfile(char *fname,char **bufp,long *lenp,long *allocsizep) { FILE *fp; long filesize,buflen = *allocsizep; char *buf = *bufp; *lenp = 0; if ( (fp= fopen(OS_compatible_path(fname),"rb")) != 0 ) { fseek(fp,0,SEEK_END); filesize = ftell(fp); if ( filesize == 0 ) { fclose(fp); *lenp = 0; printf("OS_loadfile null size.(%s)\n",fname); return(0); } if ( filesize > buflen-1 ) { *allocsizep = filesize+1; *bufp = buf = realloc(buf,(long)*allocsizep); } rewind(fp); if ( buf == 0 ) printf("Null buf ???\n"); else { if ( fread(buf,1,(long)filesize,fp) != (unsigned long)filesize ) printf("error reading filesize.%ld\n",(long)filesize); buf[filesize] = 0; } fclose(fp); *lenp = filesize; //printf("loaded.(%s)\n",buf); } //else printf("OS_loadfile couldnt load.(%s)\n",fname); return(buf); } void *OS_filestr(long *allocsizep,char *fname) { long filesize = 0; char *buf = 0; *allocsizep = 0; return(OS_loadfile(fname,&buf,&filesize,allocsizep)); } // following functions cant be fully implemented in one or more OS void *OS_mapfile(char *fname,long *filesizep,int32_t enablewrite) // win and pnacl dont have mmap64 { return(OS_portable_mapfile(fname,filesizep,enablewrite)); } /*int32_t OS_syncmap(struct OS_mappedptr *mp,long len) // pnacl doesnt implement sync { return(OS_portable_syncmap(mp,len)); } void *OS_tmpalloc(char *dirname,char *name,struct OS_memspace *mem,long origsize) // no syncmap no tmpalloc { return(OS_portable_tmpalloc(dirname,name,mem,origsize)); }*/ void OS_init() { extern bits256 GENESIS_PUBKEY,GENESIS_PRIVKEY; tai_now(); decode_hex(GENESIS_PUBKEY.bytes,sizeof(GENESIS_PUBKEY),GENESIS_PUBKEYSTR); decode_hex(GENESIS_PRIVKEY.bytes,sizeof(GENESIS_PRIVKEY),GENESIS_PRIVKEYSTR); SaM_PrepareIndices(); OS_portable_init(); } int32_t OS_getline(int32_t waitflag,char *line,int32_t max,char *dispstr) { if ( dispstr != 0 && dispstr[0] != 0 ) fprintf(stderr,"%s",dispstr); line[0] = 0; #ifndef _WIN32 if ( waitflag == 0 ) { static char prevline[1024]; struct timeval timeout; fd_set fdset; int32_t s; line[0] = 0; FD_ZERO(&fdset); FD_SET(STDIN_FILENO,&fdset); timeout.tv_sec = 0, timeout.tv_usec = 10000; if ( (s= select(1,&fdset,NULL,NULL,&timeout)) < 0 ) fprintf(stderr,"wait_for_input: error select s.%d\n",s); else { if ( FD_ISSET(STDIN_FILENO,&fdset) > 0 && fgets(line,max,stdin) == line ) { line[strlen(line)-1] = 0; if ( line[0] == 0 || (line[0] == '.' && line[1] == 0) ) strcpy(line,prevline); else strcpy(prevline,line); } } return((int32_t)strlen(line)); } #endif if ( fgets(line,max,stdin) != 0 ) line[strlen(line)-1] = 0; return((int32_t)strlen(line)); } //////////// test suite for: /* int64_t OS_filesize(char *fname); void OS_ensure_directory(char *dirname); long OS_ensurefilesize(char *fname,long filesize,int32_t truncateflag); int32_t OS_truncate(char *fname,long filesize); int32_t OS_renamefile(char *fname,char *newfname); int32_t OS_removefile(char *fname,int32_t scrubflag); void *OS_mapfile(char *fname,long *filesizep,int32_t enablewrite); int32_t OS_releasemap(void *ptr,uint64_t filesize); double OS_milliseconds(); void OS_randombytes(uint8_t *x,long xlen); */ int32_t iguana_OStests() { static uint16_t pairs[0x100][0x100],mappairs[0x100][0x100]; uint8_t buf[4096],*bufptr; int32_t val,min,minij,maxij,max,i,j,histo[0x100],retval = 0,n=0; double startmilli,endmilli; FILE *fp; char *name,*name2,*dirname; long filesize; void *fileptr; startmilli = OS_milliseconds(); printf("\n>>>>>>>>>> starting tests. Please count the seconds (or use stopwatch)\n"); name = "OStests"; name2 = "OStests2"; dirname = "tmp"; fp = fopen(name,"wb"); memset(histo,0,sizeof(histo)); memset(pairs,0,sizeof(pairs)); memset(mappairs,0,sizeof(mappairs)); for (i=0; i<4096; i++) { OS_randombytes(buf,sizeof(buf)); for (j=0; j 0 ) pairs[buf[j-1]][buf[j]]++; } } fclose(fp); printf("\nend of random bytes\n\n"); if ( OS_filesize(name) != n ) printf("FAIL OS_filesize %lld != %d error and if OS_filesize doesnt work, nothing else will work\n",(long long)OS_filesize(name),n), retval--; else { printf("PASS OS_filesize.(%s) matches %d\n",name,n); OS_renamefile(name,name2); if ( OS_filesize(name2) != n ) printf("FAIL OS_renamefile returns filesize %lld != %d\n",(long long)OS_filesize(name2),n), retval--; else printf("PASS OS_renamefile (%s) -> (%s) worked\n",name,name2); if ( (fileptr= OS_mapfile(name2,&filesize,0)) == 0 ) printf("FAIL OS_mapfile.(%s) returns null\n",name2), retval--; else if ( filesize != n ) printf("FAIL OS_mapfile.(%s) returns %ld != %d\n",name2,filesize,n), retval--; else { bufptr = fileptr; for (i=0; i<4096; i++) { memcpy(buf,bufptr,sizeof(buf)); bufptr += sizeof(buf); for (j=1; j max ) max = val; else if ( val < min ) min = val; for (j=0; j<0x100; j++) { if ( (val= pairs[i][j]) > maxij ) maxij = val; else if ( val < minij ) minij = val; } } endmilli = OS_milliseconds(); printf("\n\nDid that take %.3f seconds? If not, there is a problem with OS_milliseconds\n\nMake sure above numbers look random and the min/max are within specified range:\n<3%% %.2f%% min %d max %d | <75%% %.3f%% minij %d maxij %d\n",(endmilli - startmilli)/1000.,100*(double)max/min - 100.,min,max,100*(double)maxij/minij - 100.,minij,maxij); return(retval); }