198 lines
9.9 KiB
198 lines
9.9 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"
|
|
|
|
int64_t iguana_packetsallocated(struct iguana_info *coin) { return(coin->R.packetsallocated - coin->R.packetsfreed); };
|
|
|
|
uint8_t *iguana_decompress(struct iguana_info *coin,int32_t height,int32_t *datalenp,uint8_t *bits,int32_t numbits,int32_t origdatalen)
|
|
{
|
|
uint32_t hdrlen,checklen;
|
|
memcpy(&hdrlen,bits,sizeof(hdrlen));
|
|
bits = &bits[sizeof(hdrlen)];
|
|
*datalenp = 0;
|
|
if ( (hdrlen & (1 << 31)) != 0 )
|
|
{
|
|
hdrlen ^= (1 << 31);
|
|
if ( (hdrlen >> 3) == origdatalen )
|
|
{
|
|
*datalenp = origdatalen;
|
|
return(bits);
|
|
} else printf("\n>>>>>>>>> iguana_decompress.%d numbits.%d %d != origlen.%d\n",height,hdrlen,hdrlen>>3,origdatalen), getchar();
|
|
}
|
|
else if ( hconv_bitlen(hdrlen) == hconv_bitlen(numbits) )
|
|
{
|
|
if ( (checklen= ramcoder_decompress(coin->R.decompressed,sizeof(coin->R.decompressed),bits,hdrlen,bits256_zero)) == origdatalen )
|
|
{
|
|
//printf("DECOMPRESSED %d to %d\n",hconv_bitlen(hdrlen),checklen);
|
|
*datalenp = origdatalen;
|
|
return(coin->R.decompressed);
|
|
}
|
|
else
|
|
{
|
|
printf("\n>>>>>>>>> iguana_decompress.%d hdrlen.%d checklen.%d != origdatalen.%d\n",height,hdrlen,checklen,origdatalen);
|
|
int32_t j;
|
|
for (j=0; j<hconv_bitlen(numbits); j++)
|
|
printf("%02x ",bits[j]);
|
|
printf("compressed.%d\n",numbits/8);
|
|
getchar();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
printf("\n>>>>>>>>>> iguana_decompress.%d hdrlen.%d != numbits.%d\n",height,hdrlen,numbits);
|
|
int32_t j;
|
|
for (j=0; j<=numbits/8; j++)
|
|
printf("%02x ",bits[j]);
|
|
printf("compressed.%d\n",numbits/8);
|
|
getchar();
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/*struct iguana_msgtx *iguana_validpending(struct iguana_info *coin,struct iguana_pending *ptr,struct iguana_block *space)
|
|
{
|
|
struct iguana_block *checkblock; uint8_t *data; int32_t datalen,len; struct iguana_msgtx *tx = 0;
|
|
*space = ptr->block;
|
|
if ( coin->R.recvblocks == 0 || ptr->block.height >= coin->R.numwaitingbits )
|
|
{
|
|
printf("illegal pending height.%d vs %d\n",ptr->block.height,coin->R.numwaitingbits);
|
|
return(0);
|
|
}
|
|
if ( ptr->origdatalen > 0 && ptr->block.height < coin->longestchain && ptr->block.height < coin->blocks.hwmheight )
|
|
{
|
|
if ( (checkblock= iguana_block(coin,space,ptr->block.height)) != 0 )
|
|
{
|
|
if ( iguana_blockcmp(coin,checkblock,space,1) == 0 )
|
|
{
|
|
data = iguana_decompress(coin,ptr->block.height,&datalen,ptr->data,ptr->datalen << 3,ptr->origdatalen);
|
|
//printf("parsed.%d vs max.%d height.%d data.%p\n",coin->blocks.parsedblocks,coin->R.numwaitingbits,ptr->block.height,data);
|
|
if ( data != 0 && iguana_setdependencies(coin,space) == ptr->block.height )
|
|
{
|
|
if ( (tx= iguana_gentxarray(coin,&len,space,data,datalen)) != 0 && len == datalen )
|
|
return(tx);
|
|
} else printf("iguana_validpending: error gentx block.%d\n",coin->blocks.parsedblocks);
|
|
} else printf("iguana_validpending: error setting vars block.%d\n",ptr->block.height);
|
|
if ( tx != 0 )
|
|
iguana_freetx(tx,ptr->numtx);
|
|
} else printf("iguana_validpending cant get checkblock %d vs hwmheight.%d\n",ptr->block.height,coin->blocks.hwmheight);
|
|
}
|
|
return(0);
|
|
}*/
|
|
|
|
/*int32_t iguana_processrecv(struct iguana_info *coin)
|
|
{
|
|
int32_t height; struct iguana_block space; struct iguana_msgtx *tx = 0;
|
|
struct iguana_pending *ptr = 0; int32_t retval = -1;
|
|
height = coin->blocks.parsedblocks;
|
|
if ( coin->R.recvblocks != 0 && height < coin->R.numwaitingbits )
|
|
{
|
|
if ( (ptr= coin->R.recvblocks[height]) != 0 )
|
|
{
|
|
//printf("iguana_processrecv height.%d %p\n",height,ptr);
|
|
coin->R.recvblocks[height] = 0;
|
|
if ( (tx= iguana_validpending(coin,ptr,&space)) != 0 )
|
|
{
|
|
retval = iguana_parseblock(coin,&space,tx,ptr->numtx);
|
|
if ( space.L.numunspents+space.numvouts != coin->latest.dep.numunspents )
|
|
printf("block->firstvout+block->numvouts (%d+%d) != %d coin->latest.deps.numunspentinds\n",space.L.numunspents,space.numvouts,coin->latest.dep.numunspents), getchar();
|
|
if ( retval < 0 )
|
|
printf("iguana_processrecv: error parsing block.%d tx.%p\n",ptr->block.height,tx);
|
|
if ( tx != 0 )
|
|
iguana_freetx(tx,ptr->numtx);
|
|
} else printf("error getting pending %d %p\n",height,ptr);
|
|
if ( coin->R.maprecvdata == 0 )
|
|
{
|
|
coin->R.packetsfreed += ptr->allocsize;
|
|
myfree(ptr,ptr->allocsize);
|
|
}
|
|
}
|
|
else if ( time(NULL) > coin->parsetime+1 )
|
|
{
|
|
coin->parsetime = (uint32_t)time(NULL);
|
|
printf("backstop.%d %s\n",height,bits256_str(iguana_blockhash(coin,height)));
|
|
bits256 hash2 = iguana_blockhash(coin,height);
|
|
iguana_request_data(coin,coin->peers.ranked[0],&hash2,1,MSG_BLOCK,1);
|
|
iguana_waitclear(coin,height);
|
|
iguana_waitstart(coin,height);
|
|
iguana_updatewaiting(coin,height+1,100);
|
|
}
|
|
} else printf("processrecv: no recvbits!\n");
|
|
return(retval);
|
|
}
|
|
|
|
int32_t iguana_recvblock(struct iguana_info *coin,struct iguana_peer *addr,struct iguana_block *block,struct iguana_msgtx *tx,int32_t numtx,uint8_t *data,int32_t origdatalen)
|
|
{
|
|
struct iguana_pending *ptr; int32_t allocsize,checklen,numbits; uint32_t datalen,hdrlen;
|
|
if ( coin->R.recvblocks == 0 || coin->R.recvblocks[block->height] != 0 )
|
|
{
|
|
coin->sleeptime++;
|
|
if ( coin->sleeptime > 10000 )
|
|
coin->sleeptime = 10000;
|
|
if ( 0 && addr != coin->peers.localaddr )
|
|
printf("%s recv duplicate at height.%d sleepmillis %.3f\n",addr->ipaddr,block->height,(double)coin->sleeptime/1000.); // add validation/merging
|
|
}
|
|
else
|
|
{
|
|
coin->sleeptime *= .995;
|
|
if ( coin->sleeptime < 1000 )
|
|
coin->sleeptime = 1000;
|
|
// validate block here
|
|
datalen = origdatalen;
|
|
hdrlen = (1 << 31) | (datalen << 3);
|
|
coin->R.srcdatalen += datalen;
|
|
if ( 0 && (numbits= ramcoder_compress(coin->R.compressed,sizeof(coin->R.compressed),data,datalen,coin->R.histo,bits256_zero)) > 0 )
|
|
{
|
|
memset(coin->R.checkbuf,0,datalen);
|
|
if ( (checklen= ramcoder_decompress(coin->R.checkbuf,sizeof(coin->R.checkbuf),coin->R.compressed,numbits,bits256_zero)) == datalen )
|
|
{
|
|
if ( memcmp(coin->R.checkbuf,data,datalen) == 0 )
|
|
{
|
|
hdrlen = numbits;
|
|
data = coin->R.compressed;
|
|
printf("height.%d datalen.%d -> numbits.%d %d compression ratio %.3f [%.4f]\n",block->height,datalen,numbits,hconv_bitlen(numbits),(double)datalen/hconv_bitlen(numbits),(double)coin->R.srcdatalen/(coin->R.compressedtotal+hconv_bitlen(numbits)+sizeof(hdrlen)));
|
|
datalen = hconv_bitlen(numbits);
|
|
} else printf("ramcoder data datalen.%d compare error\n",datalen), getchar();
|
|
}
|
|
else printf("ramcoder codec error origdatalen.%d numbits.%d datalen. %d -> %d\n",origdatalen,numbits,datalen,checklen), getchar();
|
|
} //else printf("ramcoder compress error %d -> numbits.%d\n",datalen,numbits), getchar();
|
|
coin->R.compressedtotal += (datalen + sizeof(hdrlen));
|
|
allocsize = (int32_t)(sizeof(*ptr) + datalen + sizeof(hdrlen));
|
|
if ( coin->R.maprecvdata != 0 )
|
|
{
|
|
ptr = iguana_tmpalloc(coin,"recv",&coin->R.RSPACE,allocsize);
|
|
if ( block->height > coin->R.RSPACE.maxheight )
|
|
coin->R.RSPACE.maxheight = block->height;
|
|
ptr->next = (int32_t)((long)iguana_tmpalloc(coin,"recv",&coin->R.RSPACE,0) - (long)ptr);
|
|
}
|
|
else
|
|
{
|
|
ptr = mycalloc('P',1,allocsize);
|
|
coin->R.packetsallocated += allocsize;
|
|
}
|
|
ptr->allocsize = allocsize;
|
|
ptr->datalen = datalen;
|
|
memcpy(ptr->data,&hdrlen,sizeof(hdrlen));
|
|
memcpy(&ptr->data[sizeof(hdrlen)],data,datalen);
|
|
ptr->ipbits = addr != 0 ? addr->ipbits : 0;
|
|
ptr->block = *block;
|
|
ptr->numtx = numtx;
|
|
ptr->origdatalen = origdatalen;
|
|
if ( (rand() % 1000) == 0 )
|
|
printf("%s recv.%d ptr.%p datalen.%d orig.%d %.3f | parsed.%d hwm.%d longest.%d | %d/%d elapsed %.2f\n",addr != 0 ? addr->ipaddr : "local",block->height,ptr,datalen,origdatalen,(double)origdatalen/datalen,coin->blocks.parsedblocks,coin->blocks.hwmheight,coin->longestchain,iguana_updatewaiting(coin,coin->blocks.parsedblocks,coin->width*10),coin->width*10,(double)(time(NULL)-coin->starttime)/60.);
|
|
coin->R.recvblocks[block->height] = ptr;
|
|
}
|
|
return(0);
|
|
}*/
|
|
|