/* */ #include "GZipDecodingStreamFilter.h" #include #include "fmt.h" #include "DlAbortEx.h" namespace aria2 { const std::string GZipDecodingStreamFilter::NAME("GZipDecodingStreamFilter"); GZipDecodingStreamFilter::GZipDecodingStreamFilter( std::unique_ptr delegate) : StreamFilter{std::move(delegate)}, strm_{nullptr}, 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; // initialize 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_ = nullptr; } } ssize_t GZipDecodingStreamFilter::transform(const std::shared_ptr& out, const std::shared_ptr& 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 && ret != Z_BUF_ERROR) { 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