/* */ #include "GZipDecodingStreamFilter.h" #include #include "fmt.h" #include "DlAbortEx.h" namespace aria2 { const std::string GZipDecodingStreamFilter::NAME("GZipDecodingStreamFilter"); GZipDecodingStreamFilter::GZipDecodingStreamFilter (const SharedHandle& delegate): StreamFilter(delegate), strm_(0), finished_(false), bytesProcessed_(0) {} GZipDecodingStreamFilter::~GZipDecodingStreamFilter() { release(); } void GZipDecodingStreamFilter::init() { finished_ = false; release(); strm_ = new z_stream(); strm_->zalloc = Z_NULL; strm_->zfree = Z_NULL; strm_->opaque = Z_NULL; strm_->avail_in = 0; strm_->next_in = Z_NULL; // initalize z_stream with gzip/zlib format auto detection enabled. if(Z_OK != inflateInit2(strm_, 47)) { throw DL_ABORT_EX("Initializing z_stream failed."); } } void GZipDecodingStreamFilter::release() { if(strm_) { inflateEnd(strm_); delete strm_; strm_ = 0; } } ssize_t GZipDecodingStreamFilter::transform (const SharedHandle& out, const SharedHandle& segment, const unsigned char* inbuf, size_t inlen) { bytesProcessed_ = 0; ssize_t outlen = 0; if(inlen == 0) { return outlen; } strm_->avail_in = inlen; strm_->next_in = const_cast(inbuf); unsigned char outbuf[OUTBUF_LENGTH]; while(1) { strm_->avail_out = OUTBUF_LENGTH; strm_->next_out = outbuf; int ret = ::inflate(strm_, Z_NO_FLUSH); if(ret == Z_STREAM_END) { finished_ = true; } else if(ret != Z_OK) { throw DL_ABORT_EX(fmt("libz::inflate() failed. cause:%s", strm_->msg)); } size_t produced = OUTBUF_LENGTH-strm_->avail_out; outlen += getDelegate()->transform(out, segment, outbuf, produced); if(strm_->avail_out > 0) { break; } } assert(inlen >= strm_->avail_in); bytesProcessed_ = inlen-strm_->avail_in; return outlen; } bool GZipDecodingStreamFilter::finished() { return finished_ && getDelegate()->finished(); } const std::string& GZipDecodingStreamFilter::getName() const { return NAME; } } // namespace aria2