ultimatepp/uppsrc/plugin/bz2/bz2upp.cpp
cxl 9544433b9a Core: InFilterStream FilterEof
git-svn-id: svn://ultimatepp.org/upp/trunk@11768 f0d560ea-af0d-0410-9eb7-867de7ffcac7
2018-02-09 08:44:26 +00:00

340 lines
6.4 KiB
C++

#include <Core/Core.h>
#include <plugin/bz2/bz2.h>
#define LDUMP(x) // DDUMP(x)
#define LLOG(x) // DLOG(x)
namespace Upp {
namespace bz2 {
static void* bzalloc_new(void *opaque, int items, int size)
{
return new byte[items * size];
}
static void bzfree_new(void *opaque, void *addr)
{
delete[] (byte *)addr;
}
void Lib::Begin()
{
Free();
SetError(false);
out.Clear();
}
void Lib::Compress()
{
Begin();
if(BZ2_bzCompressInit(&z, compression_level, 0, 30) != BZ_OK)
Panic("BZ2_bzCompressInit failed");
mode = DEFLATE;
}
void Lib::Decompress(bool all)
{
Begin();
rdall = all;
eos = false;
if(BZ2_bzDecompressInit(&z, 0, 0) != BZ_OK)
Panic("BZ2_bzDecompressInit failed");
mode = INFLATE;
}
void Lib::Pump(bool finish)
{
if(error)
return;
ASSERT(mode);
if(!output)
output.Alloc(chunk);
if (mode == INFLATE) {
z.avail_out = chunk;
z.next_out = output;
while (z.avail_in && !IsEOS()) {
const int code = BZ2_bzDecompress(&z);
const int count = chunk - z.avail_out;
if(count) {
WhenOut((const char *)~output, count);
if (error) {
Free();
break;
}
total += count;
z.avail_out = chunk;
z.next_out = output;
}
if(code == BZ_STREAM_END) {
if (rdall) {
BZ2_bzDecompressEnd(&z);
if(BZ2_bzDecompressInit(&z, 0, 0) != BZ_OK)
Panic("BZ2_bzDecompressInit failed");
}
else
eos = true;
}
if(code != BZ_OK && code != BZ_STREAM_END) {
LLOG("BZLIB ERROR " << code);
Free();
SetError(true);
break;
}
}
} else {
const int flush = finish ? BZ_FINISH : BZ_RUN;
z.avail_out = chunk;
z.next_out = output;
while (z.avail_in) {
const int code = BZ2_bzCompress(&z, flush);
const int count = chunk - z.avail_out;
if(count) {
WhenOut((const char *)~output, count);
if (error) {
Free();
break;
}
z.avail_out = chunk;
z.next_out = output;
}
if(code != BZ_RUN_OK && code != BZ_FINISH_OK) {
LLOG("BZLIB ERROR " << code);
Free();
SetError(true);
break;
}
}
}
}
void Lib::Put(const void *ptr, int size)
{
if(error)
return;
// return false;
LLOG("BZLIB Put " << size);
const char *p = reinterpret_cast<const char *>(ptr);
while(size && !IsEOS()) {
int psz = (int) min(size, INT_MAX / 4);
if(mode == DEFLATE)
total += size;
z.next_in = const_cast<char *>(p);
z.avail_in = size;
Pump(false);
size -= psz;
p += psz;
}
// return !IsEOS();
}
void Lib::PutOut(const void *ptr, int size)
{
LLOG("BZLIB PutOut " << out.GetCount());
out.Cat((const char *)ptr, (int)size);
}
void Lib::End()
{
LLOG("BZLIB End");
if(mode != INFLATE)
Pump(true);
Free();
}
void Lib::Free()
{
if(mode == INFLATE)
BZ2_bzDecompressEnd(&z);
else if(mode == DEFLATE)
BZ2_bzCompressEnd(&z);
mode = NONE;
total = 0;
}
void Lib::Init()
{
mode = NONE;
SetError(false);
rdall = true;
eos = false;
Zero(z);
}
void Lib::Clear()
{
Free();
Init();
}
Lib& Lib::ChunkSize(int n)
{
ASSERT(n < INT_MAX / 4);
output.Clear();
chunk = n;
return *this;
}
Lib::Lib()
{
Init();
z.bzalloc = bzalloc_new;
z.bzfree = bzfree_new;
z.opaque = 0;
chunk = 4096;
WhenOut = callback(this, &Lib::PutOut);
compression_level = 9;
}
Lib::~Lib()
{
Free();
}
}
void BZ2Decompress(Stream& out, Stream& in, Gate<int, int> progress)
{
enum { BUF_SIZE = 65536 };
Buffer<char> input(BUF_SIZE), output(BUF_SIZE);
int avail = in.Get(input, BUF_SIZE);
if(avail == 0)
return;
bz_stream z;
Zero(z);
z.bzalloc = bz2::bzalloc_new;
z.bzfree = bz2::bzfree_new;
z.opaque = 0;
if(BZ2_bzDecompressInit(&z, 0, 0) != BZ_OK)
{
out.SetError();
return;
}
z.next_in = input;
z.avail_in = avail;
z.next_out = output;
z.avail_out = BUF_SIZE;
int code;
bool running = true;
int64 total = in.GetLeft();
int done = 0;
do
{
if(z.avail_in == 0 && running)
{
if((z.avail_in = in.Get(z.next_in = input, BUF_SIZE)) == 0)
running = false;
done += z.avail_in;
if(progress(done, (int)total) || in.IsError())
{
BZ2_bzDecompressEnd(&z);
out.SetError();
return;
}
}
code = BZ2_bzDecompress(&z);
if(z.avail_out == 0)
{
out.Put(z.next_out = output, z.avail_out = BUF_SIZE);
if(out.IsError())
{
BZ2_bzDecompressEnd(&z);
return;
}
}
}
while(code == BZ_OK);
if(z.avail_out < BUF_SIZE)
out.Put(output, BUF_SIZE - z.avail_out);
BZ2_bzDecompressEnd(&z);
}
void BZ2Compress(Stream& out, Stream& in, Gate<int, int> progress)
{
enum { BUF_SIZE = 65536 };
Buffer<char> input(BUF_SIZE), output(BUF_SIZE);
bz_stream z;
z.bzalloc = bz2::bzalloc_new;
z.bzfree = bz2::bzfree_new;
z.opaque = 0;
if(BZ2_bzCompressInit(&z, 9, 0, 30) != BZ_OK)
{
out.SetError();
return;
}
z.avail_in = 0;
z.avail_out = BUF_SIZE;
z.next_out = output;
int code;
int flush = BZ_RUN;
int64 total = in.GetLeft();
int done = 0;
do
{
if(z.avail_in == 0 && flush == BZ_RUN)
{
z.next_in = input;
if((z.avail_in = in.Get(z.next_in = input, BUF_SIZE)) == 0)
flush = BZ_FINISH;
done += z.avail_in;
if(progress(done, (int)total) || in.IsError())
{
BZ2_bzCompressEnd(&z);
out.SetError();
return;
}
}
code = BZ2_bzCompress(&z, flush);
if(z.avail_out == 0)
{
out.Put(z.next_out = output, z.avail_out = BUF_SIZE);
if(out.IsError())
{
BZ2_bzCompressEnd(&z);
return;
}
}
}
while(code == BZ_RUN_OK || code == BZ_FINISH_OK);
if(z.avail_out < BUF_SIZE)
out.Put(output, BUF_SIZE - z.avail_out);
BZ2_bzCompressEnd(&z);
if(code != BZ_STREAM_END)
out.SetError();
}
String BZ2Compress(Stream& in, Gate<int, int> progress)
{
StringStream out;
BZ2Compress(out, in, progress);
return out;
}
String BZ2Decompress(Stream& in, Gate<int, int> progress)
{
StringStream out;
BZ2Decompress(out, in, progress);
return out;
}
String BZ2Compress(const void *data, int64 len, Gate<int, int> progress)
{
MemReadStream in(data, len);
return BZ2Compress(in, progress);
}
String BZ2Decompress(const void *data, int64 len, Gate<int, int> progress)
{
MemReadStream in(data, len);
return BZ2Decompress(in, progress);
}
String BZ2Compress(const String& data, Gate<int, int> progress)
{
return BZ2Compress(~data, data.GetLength(), progress);
}
String BZ2Decompress(const String& data, Gate<int, int> progress)
{
return BZ2Decompress(~data, data.GetLength(), progress);
}
}