| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 | #include "StdInc.h"#include "CCompressedStream.h"#include <zlib.h>static const int inflateBlockSize = 10000;CCompressedStream::CCompressedStream(std::unique_ptr<CInputStream> stream, bool gzip, size_t decompressedSize):	gzipStream(std::move(stream)),	buffer(decompressedSize),	decompressedSize(0),	compressedBuffer(inflateBlockSize),	position(0){	assert(gzipStream);	// Allocate inflate state	inflateState = new z_stream;	inflateState->zalloc = Z_NULL;	inflateState->zfree = Z_NULL;	inflateState->opaque = Z_NULL;	inflateState->avail_in = 0;	inflateState->next_in = Z_NULL;	int wbits = 15;	if (gzip)		wbits += 16;	int ret = inflateInit2(inflateState, wbits);	if (ret != Z_OK)		throw std::runtime_error("Failed to initialize inflate!\n");}CCompressedStream::~CCompressedStream(){	if (inflateState)	{		inflateEnd(inflateState);		delete inflateState;	}}si64 CCompressedStream::read(ui8 * data, si64 size){	decompressTill(position + size);	auto start = buffer.begin() + position;	si64 toRead = std::min<si64>(size, decompressedSize - position);	std::copy(start, start + toRead, data);	position += toRead;	return size;}si64 CCompressedStream::seek(si64 position){	decompressTill(position);	this->position = std::min<si64>(position, decompressedSize);	return this->position;}si64 CCompressedStream::tell(){	return position;}si64 CCompressedStream::skip(si64 delta){	decompressTill(position + delta);	si64 oldPosition = position;	position = std::min<si64>(position + delta , decompressedSize);	return position - oldPosition;}si64 CCompressedStream::getSize(){	decompressTill(-1);	return decompressedSize;}void CCompressedStream::decompressTill(si64 newSize){	assert(newSize < 100 * 1024 * 1024); //just in case	if (inflateState == nullptr)		return; //file already decompressed	if (newSize >= 0 && newSize <= inflateState->total_out)		return; //no need to decompress anything	bool toEnd = newSize < 0; //if true - we've got request to read whole file	if (toEnd && buffer.empty())		buffer.resize(16 * 1024); //some space for initial decompression	if (!toEnd && buffer.size() < newSize)		buffer.resize(newSize);	bool fileEnded = false; //end of file reached	int endLoop = false;	do	{		if (inflateState->avail_in == 0)		{			//inflate ran out of available data or was not initialized yet			// get new input data and update state accordingly			si64 size = gzipStream->read(compressedBuffer.data(), compressedBuffer.size());			if (size != compressedBuffer.size())				fileEnded = true; //end of file reached			inflateState->avail_in = size;			inflateState->next_in  = compressedBuffer.data();		}		inflateState->avail_out = buffer.size() - inflateState->total_out;		inflateState->next_out = buffer.data() + inflateState->total_out;		int ret = inflate(inflateState, Z_NO_FLUSH);		switch (ret)		{		case Z_STREAM_END: //end decompression			endLoop = true;			break;		case Z_OK: //decompress next chunk			endLoop = false;			break;		case Z_BUF_ERROR:			{				if (toEnd)				{					//not enough memory. Allocate bigger buffer and try again					buffer.resize(buffer.size() * 2);					endLoop = false;				}				else				{					//specific amount of data was requested. inflate extracted all requested data					// but returned "error". Ensure that enough data was extracted and return					assert(inflateState->total_out == newSize);					endLoop = true;				}			}			break;		default:			throw std::runtime_error("Decompression error!\n");		}	}	while (!endLoop);	// Clean up and return	if (fileEnded)	{		inflateEnd(inflateState);		buffer.resize(inflateState->total_out);		decompressedSize = inflateState->total_out;		vstd::clear_pointer(inflateState);	}	else		decompressedSize = inflateState->total_out;}bool CCompressedStream::getNextBlock(){	if (!inflateState)		return false;	if (inflateReset(inflateState) < 0)		return false;	decompressedSize = 0;	position = 0;	buffer.clear();	return true;}
 |