/* */ #include "GZipFile.h" #include #include #include "a2io.h" #include "util.h" namespace aria2 { GZipFile::GZipFile(const char* filename, const char* mode) : fp_(nullptr), buflen_(1024), buf_(reinterpret_cast(malloc(buflen_))) { FILE* fp = #ifdef __MINGW32__ a2fopen(utf8ToWChar(filename).c_str(), utf8ToWChar(mode).c_str()); #else // !__MINGW32__ a2fopen(filename, mode); #endif // !__MINGW32__ if (fp) { int fd = dup(fileno(fp)); if (fd != -1) { fp_ = gzdopen(fd, mode); if (fp_) { // fp_ retains fd and gzclose() will close fd as well. #if HAVE_GZBUFFER gzbuffer(fp_, 1<<17); #endif #if HAVE_GZSETPARAMS gzsetparams(fp_, 2, Z_DEFAULT_STRATEGY); #endif } else { ::close(fd); } } fclose(fp); } } GZipFile::~GZipFile() { close(); free(buf_); } int GZipFile::onClose() { int rv = 0; if (fp_) { rv = gzclose(fp_); fp_ = nullptr; } return rv; } bool GZipFile::onSupportsColor() { return false; } bool GZipFile::isError() const { int rv = 0; const char *e = gzerror(fp_, &rv); return (e != nullptr && *e != 0) || rv != 0; } bool GZipFile::isEOF() const { return gzeof(fp_); } bool GZipFile::isOpen() const { return fp_; } size_t GZipFile::onRead(void* ptr, size_t count) { char *data = reinterpret_cast(ptr); size_t read = 0; while (count) { size_t len = std::min(count, (size_t)std::numeric_limits::max()); int rv = gzread(fp_, data, len); if (rv <= 0) { break; } count -= rv; read += rv; data += rv; } return read; } size_t GZipFile::onWrite(const void* ptr, size_t count) { const char *data = reinterpret_cast(ptr); size_t written = 0; while (count) { size_t len = std::min(count, (size_t)std::numeric_limits::max()); int rv = gzwrite(fp_, data, len); if (rv <= 0) { break; } count -= rv; written += rv; data += rv; } return written; } char* GZipFile::onGets(char* s, int size) { return gzgets(fp_, s, size); } int GZipFile::onFlush() { return gzflush(fp_, 0); } int GZipFile::onVprintf(const char* format, va_list va) { ssize_t len; for(;;) { len = vsnprintf(buf_, buflen_, format, va); // len does not include terminating null if(len >= static_cast(buflen_)) { // Include terminate null ++len; // truncated; reallocate buf and try again while(static_cast(buflen_) < len) { buflen_ *= 2; } buf_ = reinterpret_cast(realloc(buf_, buflen_)); } else if(len < 0) { return len; } else { break; } } return gzwrite(fp_, buf_, len); } } // namespace aria2