mirror of
https://github.com/uroni/urbackup_backend.git
synced 2025-10-26 11:36:50 +00:00
732 lines
18 KiB
C++
732 lines
18 KiB
C++
/*************************************************************************
|
|
* UrBackup - Client/Server backup system
|
|
* Copyright (C) 2011-2016 Martin Raiber
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
**************************************************************************/
|
|
|
|
#include "CompressedFile.h"
|
|
#include "../stringtools.h"
|
|
#include <assert.h>
|
|
#include <memory>
|
|
#include <algorithm>
|
|
#include <memory.h>
|
|
#ifndef NO_ZSTD_COMPRESSION
|
|
#include <zstd.h>
|
|
#endif
|
|
#include "LRUMemCache.h"
|
|
|
|
#define MINIZ_NO_ZLIB_COMPATIBLE_NAMES
|
|
#include "../common/miniz.h"
|
|
|
|
const size_t c_cacheBuffersize = 2*1024*1024;
|
|
const size_t c_ncacheItems = 5;
|
|
const size_t c_blockbufHeadersize = 2 * sizeof(_u32);
|
|
const char headerMagic[] = "URBACKUP COMPRESSED FILE#";
|
|
const char headerVersionV1_0[] = "1.0";
|
|
const char headerVersionV1_1[] = "1.1";
|
|
const _u32 mode_none = 0;
|
|
const _u32 mode_zlib = 1;
|
|
const _u32 mode_zstd = 2;
|
|
const size_t c_header_size = sizeof(headerMagic) + sizeof(headerVersionV1_0) + sizeof(__int64) + sizeof(__int64) + sizeof(_u32);
|
|
|
|
|
|
CompressedFile::CompressedFile( std::string pFilename, int pMode, size_t n_threads)
|
|
: error(false), currentPosition(0),
|
|
finished(false), filesize(0), noMagic(false),
|
|
mutex(Server->createMutex()), n_threads(n_threads), numBlockOffsets(0)
|
|
{
|
|
uncompressedFile = Server->openFile(pFilename, pMode);
|
|
|
|
if(uncompressedFile==NULL)
|
|
{
|
|
Server->Log("Could not open compressed file \""+pFilename+"\"", LL_ERROR);
|
|
error=true;
|
|
return;
|
|
}
|
|
|
|
if(pMode == MODE_READ ||
|
|
pMode == MODE_RW )
|
|
{
|
|
readOnly=true;
|
|
readHeader(&error);
|
|
}
|
|
else
|
|
{
|
|
readOnly=false;
|
|
blocksize = c_cacheBuffersize;
|
|
writeHeader();
|
|
hotCache.reset(new LRUMemCache(blocksize, c_ncacheItems, n_threads));
|
|
initCompressedBuffers(n_threads);
|
|
}
|
|
|
|
if(hotCache.get())
|
|
{
|
|
hotCache->setCacheEvictionCallback(this);
|
|
}
|
|
|
|
uncompressedFileSize = uncompressedFile->Size();
|
|
}
|
|
|
|
CompressedFile::CompressedFile(IFile* file, bool openExisting, bool readOnly, size_t n_threads)
|
|
: error(false), currentPosition(0),
|
|
finished(false), uncompressedFile(file), filesize(0), readOnly(readOnly),
|
|
noMagic(false), n_threads(n_threads), numBlockOffsets(0)
|
|
{
|
|
if(openExisting)
|
|
{
|
|
readHeader(&error);
|
|
}
|
|
else
|
|
{
|
|
blocksize = c_cacheBuffersize;
|
|
writeHeader();
|
|
hotCache.reset(new LRUMemCache(blocksize, c_ncacheItems, n_threads));
|
|
initCompressedBuffers(n_threads);
|
|
}
|
|
if(hotCache.get()!=NULL)
|
|
{
|
|
hotCache->setCacheEvictionCallback(this);
|
|
}
|
|
|
|
uncompressedFileSize = uncompressedFile->Size();
|
|
}
|
|
|
|
CompressedFile::~CompressedFile()
|
|
{
|
|
hotCache.reset();
|
|
|
|
if(!finished)
|
|
{
|
|
finish();
|
|
}
|
|
|
|
delete uncompressedFile;
|
|
|
|
IScopedLock lock(mutex.get());
|
|
for (size_t i = 0; i < compressedBuffers.size(); ++i)
|
|
{
|
|
assert(compressedBuffers[i] != NULL);
|
|
delete[] compressedBuffers[i];
|
|
}
|
|
}
|
|
|
|
bool CompressedFile::hasError()
|
|
{
|
|
return error;
|
|
}
|
|
|
|
void CompressedFile::readHeader(bool *has_error)
|
|
{
|
|
std::string header;
|
|
header.resize(c_header_size);
|
|
if(readFromFile(0, &header[0], c_header_size, has_error)!=c_header_size)
|
|
{
|
|
Server->Log("Error while reading compressed file header", LL_ERROR);
|
|
error=true;
|
|
return;
|
|
}
|
|
|
|
if(!next(header, 0, headerMagic))
|
|
{
|
|
Server->Log("Magic in header not found for compressed file", LL_ERROR);
|
|
error=true;
|
|
noMagic=true;
|
|
return;
|
|
}
|
|
|
|
std::string headerVersion = header.substr(sizeof(headerMagic)-1, sizeof(headerVersionV1_0)-1);
|
|
|
|
if (headerVersion != headerVersionV1_0
|
|
&& headerVersion != headerVersionV1_1)
|
|
{
|
|
Server->Log("Unknown compressed file version: " + headerVersion, LL_ERROR);
|
|
error = true;
|
|
return;
|
|
}
|
|
|
|
memcpy(&index_offset, header.data()+sizeof(headerMagic) - 1 + sizeof(headerVersionV1_0), sizeof(index_offset));
|
|
memcpy(&filesize, header.data()+sizeof(headerMagic) - 1 + sizeof(headerVersionV1_0) +sizeof(index_offset), sizeof(filesize));
|
|
memcpy(&blocksize, header.data()+sizeof(headerMagic) - 1 + sizeof(headerVersionV1_0) +sizeof(index_offset)+sizeof(filesize), sizeof(blocksize));
|
|
|
|
index_offset = little_endian(index_offset);
|
|
filesize = little_endian(filesize);
|
|
blocksize = little_endian(blocksize);
|
|
|
|
hotCache.reset(new LRUMemCache(blocksize, c_ncacheItems, n_threads));
|
|
|
|
readIndex(has_error);
|
|
}
|
|
|
|
void CompressedFile::readIndex(bool *has_error)
|
|
{
|
|
size_t nOffsetItems = static_cast<size_t>(filesize/blocksize + ((filesize%blocksize!=0)?1:0));
|
|
|
|
if(nOffsetItems==0)
|
|
{
|
|
Server->Log("Compressed file contains nothing", LL_ERROR);
|
|
error=true;
|
|
return;
|
|
}
|
|
|
|
blockOffsets.resize(nOffsetItems);
|
|
numBlockOffsets = nOffsetItems;
|
|
|
|
if(readFromFile(index_offset, reinterpret_cast<char*>(&blockOffsets[0]), static_cast<_u32>(sizeof(__int64)*nOffsetItems), has_error)
|
|
!=sizeof(__int64)*nOffsetItems)
|
|
{
|
|
Server->Log("Error while reading block offsets", LL_ERROR);
|
|
error=true;
|
|
return;
|
|
}
|
|
|
|
if(is_big_endian())
|
|
{
|
|
for(size_t i=0;i<blockOffsets.size();++i)
|
|
{
|
|
blockOffsets[i] = little_endian(blockOffsets[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool CompressedFile::Seek( _i64 spos )
|
|
{
|
|
assert(!finished);
|
|
currentPosition = spos;
|
|
return true;
|
|
}
|
|
|
|
_u32 CompressedFile::Read(_i64 spos, char* buffer, _u32 bsize, bool *has_error)
|
|
{
|
|
if(!Seek(spos))
|
|
{
|
|
if (has_error) *has_error = true;
|
|
return 0;
|
|
}
|
|
|
|
return Read(buffer, bsize, has_error);
|
|
}
|
|
|
|
_u32 CompressedFile::Read( char* buffer, _u32 bsize, bool *has_error)
|
|
{
|
|
assert(!finished);
|
|
|
|
size_t cacheSize;
|
|
char* cachePtr = hotCache->get(currentPosition, cacheSize);
|
|
|
|
if(cachePtr == NULL)
|
|
{
|
|
if(!fillCache(currentPosition, !readOnly, has_error))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
cachePtr = hotCache->get(currentPosition, cacheSize);
|
|
|
|
if(cachePtr==NULL)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
size_t canRead = bsize;
|
|
if(cacheSize<canRead)
|
|
canRead = cacheSize;
|
|
if(currentPosition+static_cast<int64>(canRead)>filesize)
|
|
canRead = filesize-currentPosition;
|
|
|
|
if(canRead==0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
memcpy(buffer, cachePtr, canRead);
|
|
|
|
currentPosition+=canRead;
|
|
|
|
if(canRead<bsize)
|
|
{
|
|
return static_cast<_u32>(canRead) + Read(buffer + canRead, bsize-static_cast<_u32>(canRead));
|
|
}
|
|
|
|
return static_cast<_u32>(canRead);
|
|
}
|
|
|
|
std::string CompressedFile::Read( _u32 tr, bool *has_error)
|
|
{
|
|
assert(!finished);
|
|
|
|
if(tr==0)
|
|
return std::string();
|
|
|
|
std::string ret;
|
|
ret.resize(tr);
|
|
|
|
if(Read(&ret[0], static_cast<_u32>(ret.size()), has_error)!=tr)
|
|
{
|
|
return std::string();
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
std::string CompressedFile::Read(_i64 spos, _u32 tr, bool *has_error)
|
|
{
|
|
if(!Seek(spos))
|
|
{
|
|
if (has_error) *has_error = true;
|
|
return std::string();
|
|
}
|
|
|
|
return Read(tr, has_error);
|
|
}
|
|
|
|
bool CompressedFile::fillCache( __int64 offset, bool errorMsg, bool *has_error)
|
|
{
|
|
size_t block = static_cast<size_t>(offset/blocksize);
|
|
|
|
if(block>=blockOffsets.size())
|
|
{
|
|
if(errorMsg)
|
|
{
|
|
Server->Log("Block "+convert(block)+" to read not found in block index", LL_ERROR);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
char* buf = hotCache->create(offset);
|
|
|
|
if(error)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if(blockOffsets[block]==-1)
|
|
{
|
|
memset(buf, 0, blocksize);
|
|
return true;
|
|
}
|
|
|
|
const __int64 blockDataOffset = blockOffsets[block];
|
|
|
|
char blockheaderBuf[2*sizeof(_u32)];
|
|
if(readFromFile(blockDataOffset, blockheaderBuf, sizeof(blockheaderBuf), has_error)!=sizeof(blockheaderBuf))
|
|
{
|
|
Server->Log("Error while reading block header", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
_u32 compressedSize;
|
|
memcpy(&compressedSize, blockheaderBuf, sizeof(compressedSize));
|
|
compressedSize = little_endian(compressedSize);
|
|
_u32 mode;
|
|
memcpy(&mode, blockheaderBuf + sizeof(compressedSize), sizeof(mode));
|
|
|
|
if(mode==mode_none)
|
|
{
|
|
if(compressedSize>blocksize)
|
|
{
|
|
Server->Log("Blocksize too large at offset "+convert(blockDataOffset)+" ("+convert(compressedSize)+" bytes)", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
if(readFromFile(blockDataOffset + c_blockbufHeadersize, buf, compressedSize, has_error)!=compressedSize)
|
|
{
|
|
Server->Log("Error while reading uncompressed data from "+convert(blockDataOffset)+" ("+convert(compressedSize)+" bytes)", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
if(compressedBuffer.size()<compressedSize)
|
|
{
|
|
compressedBuffer.resize(compressedSize);
|
|
}
|
|
|
|
if(readFromFile(blockDataOffset + c_blockbufHeadersize, &compressedBuffer[0], compressedSize, has_error)!=compressedSize)
|
|
{
|
|
Server->Log("Error while reading compressed data from "+convert(blockDataOffset)+" ("+convert(compressedSize)+" bytes)", LL_ERROR);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
mz_ulong rdecomp=0;
|
|
if(mode==mode_zlib)
|
|
{
|
|
rdecomp = blocksize;
|
|
int rc = mz_uncompress(reinterpret_cast<unsigned char*>(buf), &rdecomp,
|
|
reinterpret_cast<const unsigned char*>(compressedBuffer.data()), static_cast<mz_ulong>(compressedSize));
|
|
|
|
if(rc != MZ_OK)
|
|
{
|
|
Server->Log("Error while decompressing file (miniz). Error code "+convert(rc), LL_ERROR);
|
|
return false;
|
|
}
|
|
}
|
|
#ifndef NO_ZSTD_COMPRESSION
|
|
else if (mode == mode_zstd)
|
|
{
|
|
rdecomp = blocksize;
|
|
const size_t rc = ZSTD_decompress(buf, blocksize,
|
|
compressedBuffer.data(), compressedSize);
|
|
|
|
if (ZSTD_isError(rc))
|
|
{
|
|
Server->Log(std::string("Error while decompressing file (zstd). Error code ") + ZSTD_getErrorName(rc), LL_ERROR);
|
|
return false;
|
|
}
|
|
}
|
|
#endif
|
|
else
|
|
{
|
|
Server->Log("Unknown compression method " + convert(mode)+" in CompressedFile", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
|
|
if(rdecomp!=blocksize && offset+blocksize<filesize)
|
|
{
|
|
Server->Log("Did not receive enough bytes from compressed stream. Expected "+convert(blocksize)+" received "+convert((size_t)rdecomp), LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
_u32 CompressedFile::Write( const char* buffer, _u32 bsize, bool *has_error)
|
|
{
|
|
assert(!finished);
|
|
|
|
_u32 maxWrite = static_cast<_u32>(((currentPosition/blocksize)+1)*blocksize - currentPosition);
|
|
_u32 write = bsize;
|
|
if(maxWrite<bsize)
|
|
{
|
|
write=maxWrite;
|
|
}
|
|
|
|
size_t cacheSize;
|
|
char* cachePtr = hotCache->get(currentPosition, cacheSize);
|
|
|
|
if(cachePtr==NULL)
|
|
{
|
|
fillCache(currentPosition, false, has_error);
|
|
}
|
|
|
|
if(error)
|
|
{
|
|
error=false;
|
|
return 0;
|
|
}
|
|
|
|
hotCache->put(currentPosition, buffer, write);
|
|
currentPosition += write;
|
|
|
|
if(error)
|
|
{
|
|
error=false;
|
|
return 0;
|
|
}
|
|
|
|
if(currentPosition>filesize)
|
|
{
|
|
filesize=currentPosition;
|
|
}
|
|
|
|
if(maxWrite<bsize)
|
|
{
|
|
return write + Write(buffer + write, bsize-write);
|
|
}
|
|
|
|
return write;
|
|
}
|
|
|
|
_u32 CompressedFile::Write(_i64 spos, const char* buffer, _u32 bsize, bool *has_error)
|
|
{
|
|
if(!Seek(spos))
|
|
{
|
|
if (has_error) *has_error = true;
|
|
return 0;
|
|
}
|
|
|
|
return Write(buffer, bsize, has_error);
|
|
}
|
|
|
|
_u32 CompressedFile::Write( const std::string &tw, bool *has_error)
|
|
{
|
|
return Write(tw.data(), static_cast<_u32>(tw.size()), has_error);
|
|
}
|
|
|
|
_u32 CompressedFile::Write(_i64 spos, const std::string &tw, bool *has_error)
|
|
{
|
|
return Write(spos, tw.data(), static_cast<_u32>(tw.size()), has_error);
|
|
}
|
|
|
|
void CompressedFile::evictFromLruCache( const SCacheItem& item )
|
|
{
|
|
if(readOnly)
|
|
return;
|
|
|
|
char* compBuffer;
|
|
size_t compBufferIdx;
|
|
|
|
{
|
|
IScopedLock lock(mutex.get());
|
|
compBuffer = getCompressedBuffer(compBufferIdx);
|
|
}
|
|
|
|
#ifdef NO_ZSTD_COMPRESSION
|
|
const _u32 mode = mode_zlib;
|
|
mz_ulong compBytes = static_cast<mz_ulong>(compressedBufferSize - c_blockbufHeadersize);
|
|
const int rc = mz_compress(reinterpret_cast<unsigned char*>(compBuffer)+ c_blockbufHeadersize, &compBytes,
|
|
reinterpret_cast<const unsigned char*>(item.buffer), blocksize);
|
|
|
|
if(rc!=MZ_OK)
|
|
{
|
|
error=true;
|
|
Server->Log("Error while compressing data. Error code: "+convert(rc), LL_ERROR);
|
|
return;
|
|
}
|
|
#else
|
|
const _u32 mode = mode_zstd;
|
|
const size_t compBytes = ZSTD_compress(compBuffer+ c_blockbufHeadersize, compressedBufferSize - c_blockbufHeadersize, item.buffer, blocksize,
|
|
7);
|
|
if (ZSTD_isError(compBytes))
|
|
{
|
|
error = true;
|
|
Server->Log(std::string("Error while compressing data (ZSTD). Error code: ") + ZSTD_getErrorName(compBytes), LL_ERROR);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
int64 blockOffset;
|
|
{
|
|
IScopedLock lock(mutex.get());
|
|
blockOffset = uncompressedFileSize;
|
|
uncompressedFileSize += c_blockbufHeadersize + compBytes;
|
|
}
|
|
|
|
const _u32 compBytesEndian = little_endian(static_cast<_u32>(compBytes));
|
|
const _u32 modeEndian = little_endian(mode);
|
|
|
|
memcpy(compBuffer, &compBytesEndian, sizeof(compBytesEndian));
|
|
memcpy(compBuffer + sizeof(compBytesEndian), &modeEndian, sizeof(modeEndian));
|
|
|
|
if(writeToFile(blockOffset, compBuffer, static_cast<_u32>(c_blockbufHeadersize + compBytes))!=static_cast<_u32>(c_blockbufHeadersize + compBytes))
|
|
{
|
|
error=true;
|
|
Server->Log("Error while writing compressed data to file", LL_ERROR);
|
|
return;
|
|
}
|
|
|
|
size_t blockIdx = static_cast<size_t>(item.offset/blocksize);
|
|
|
|
IScopedLock lock(mutex.get());
|
|
returnCompressedBuffer(compBuffer, compBufferIdx);
|
|
const size_t currNumBlockOffsets = blockOffsets.size();
|
|
if(blockOffsets.size()<=blockIdx)
|
|
{
|
|
size_t new_size = (blockIdx + 1) * 2;
|
|
blockOffsets.resize(new_size);
|
|
for (size_t i = currNumBlockOffsets; i < new_size; ++i)
|
|
{
|
|
blockOffsets[i] = -1;
|
|
}
|
|
}
|
|
|
|
numBlockOffsets = (std::max)(numBlockOffsets, blockIdx + 1);
|
|
blockOffsets[blockIdx] = blockOffset;
|
|
}
|
|
|
|
void CompressedFile::writeHeader()
|
|
{
|
|
char header[c_header_size];
|
|
char* cptr = header;
|
|
memcpy(cptr, headerMagic, sizeof(headerMagic)-1);
|
|
cptr+=sizeof(headerMagic)-1;
|
|
memcpy(cptr, headerVersionV1_1, sizeof(headerVersionV1_1));
|
|
cptr += sizeof(headerVersionV1_1);
|
|
__int64 indexOffsetEndian = little_endian(index_offset);
|
|
memcpy(cptr, &indexOffsetEndian, sizeof(indexOffsetEndian));
|
|
cptr+=sizeof(indexOffsetEndian);
|
|
__int64 filesizeEndian = little_endian(filesize);
|
|
memcpy(cptr, &filesizeEndian, sizeof(filesizeEndian));
|
|
cptr+=sizeof(filesizeEndian);
|
|
_u32 blocksizeEndian = little_endian(blocksize);
|
|
memcpy(cptr, &blocksize, sizeof(blocksizeEndian));
|
|
|
|
if(writeToFile(0 ,header, c_header_size)!=c_header_size)
|
|
{
|
|
Server->Log("Error writing header to compressed file");
|
|
error=true;
|
|
}
|
|
}
|
|
|
|
void CompressedFile::writeIndex()
|
|
{
|
|
index_offset = uncompressedFile->Size();
|
|
|
|
if (blockOffsets.empty())
|
|
{
|
|
return;
|
|
}
|
|
|
|
assert(numBlockOffsets <= blockOffsets.size());
|
|
_u32 nOffsetBytes = static_cast<_u32>(sizeof(__int64)*numBlockOffsets);
|
|
if(writeToFile(index_offset, reinterpret_cast<char*>(&blockOffsets[0]), nOffsetBytes)!=nOffsetBytes)
|
|
{
|
|
error=true;
|
|
Server->Log("Error while writing compressed file index", LL_ERROR);
|
|
return;
|
|
}
|
|
}
|
|
|
|
void CompressedFile::initCompressedBuffers(size_t n_init)
|
|
{
|
|
compressedBufferSize = mz_compressBound(static_cast<mz_ulong>(blocksize))+ c_blockbufHeadersize;
|
|
#ifndef NO_ZSTD_COMPRESSION
|
|
const size_t csize = ZSTD_compressBound(blocksize)*2;
|
|
if (csize + c_blockbufHeadersize > compressedBufferSize)
|
|
{
|
|
compressedBufferSize = csize + c_blockbufHeadersize;
|
|
}
|
|
#endif
|
|
|
|
for (size_t i = 0; i < n_init; ++i)
|
|
{
|
|
compressedBuffers.push_back(new char[compressedBufferSize]);
|
|
}
|
|
}
|
|
|
|
char* CompressedFile::getCompressedBuffer(size_t& compressed_buffer_idx)
|
|
{
|
|
for (size_t i = 0; i < compressedBuffers.size(); ++i)
|
|
{
|
|
if (compressedBuffers[i] != NULL)
|
|
{
|
|
compressed_buffer_idx = i;
|
|
char* ret = compressedBuffers[i];
|
|
compressedBuffers[i] = NULL;
|
|
return ret;
|
|
}
|
|
}
|
|
assert(false);
|
|
return NULL;
|
|
}
|
|
|
|
void CompressedFile::returnCompressedBuffer(char* buf, size_t compressed_buffer_idx)
|
|
{
|
|
assert(compressedBuffers[compressed_buffer_idx] == NULL);
|
|
compressedBuffers[compressed_buffer_idx] = buf;
|
|
}
|
|
|
|
bool CompressedFile::finish()
|
|
{
|
|
assert(!finished);
|
|
|
|
if(hotCache.get())
|
|
{
|
|
hotCache->clear();
|
|
}
|
|
|
|
if(!readOnly)
|
|
{
|
|
writeIndex();
|
|
writeHeader();
|
|
|
|
if (!uncompressedFile->Sync())
|
|
{
|
|
error = true;
|
|
Server->Log("Error syncing uncompressed file to disk", LL_ERROR);
|
|
}
|
|
}
|
|
|
|
if(!error)
|
|
{
|
|
finished = true;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
error=false;
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
_i64 CompressedFile::Size( void )
|
|
{
|
|
return filesize;
|
|
}
|
|
|
|
_i64 CompressedFile::RealSize()
|
|
{
|
|
return uncompressedFile->Size();
|
|
}
|
|
|
|
std::string CompressedFile::getFilename( void )
|
|
{
|
|
return uncompressedFile->getFilename();
|
|
}
|
|
|
|
_u32 CompressedFile::readFromFile(int64 offset, char* buffer, _u32 bsize, bool *has_error)
|
|
{
|
|
_u32 read = 0;
|
|
do
|
|
{
|
|
_u32 rc = uncompressedFile->Read(offset, buffer+read, bsize-read, has_error);
|
|
if(rc<=0)
|
|
{
|
|
return read;
|
|
}
|
|
read+=rc;
|
|
offset += rc;
|
|
} while (read<bsize);
|
|
|
|
return read;
|
|
}
|
|
|
|
_u32 CompressedFile::writeToFile(int64 offset, const char* buffer, _u32 bsize)
|
|
{
|
|
_u32 written = 0;
|
|
do
|
|
{
|
|
_u32 w = uncompressedFile->Write(offset, buffer + written, bsize-written);
|
|
if(w<=0)
|
|
{
|
|
return written;
|
|
}
|
|
written += w;
|
|
offset += w;
|
|
} while (written<bsize);
|
|
|
|
return written;
|
|
}
|
|
|
|
bool CompressedFile::hasNoMagic()
|
|
{
|
|
return noMagic;
|
|
}
|
|
|
|
bool CompressedFile::PunchHole( _i64 spos, _i64 size )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool CompressedFile::Sync()
|
|
{
|
|
return finish();
|
|
}
|