/* */ #include "GZipFile.h" #include #include #include "a2io.h" #include "util.h" namespace aria2 { GZipFile::GZipFile(const char* filename, const char* mode) : BufferedFile(0), fp_(0), open_(false), 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__ open_ = fp; if (open_) { int fd = dup(fileno(fp)); if ((open_ = fd) >= 0) { open_ = (fp_ = gzdopen(fd, mode)); if (!open_) { ::close(fd); } } if (open_) { #if HAVE_GZBUFFER gzbuffer(fp_, 1<<17); #endif #if HAVE_GZSETPARAMS gzsetparams(fp_, 2, Z_DEFAULT_STRATEGY); #endif } fclose(fp); } } GZipFile::~GZipFile() { close(); free(buf_); } int GZipFile::close() { if (open_) { open_ = false; return gzclose(fp_); } return 0; } bool GZipFile::isError() const { int rv = 0; const char *e = gzerror(fp_, &rv); return (e != 0 && *e != 0) || rv != 0; } size_t GZipFile::read(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::write(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::gets(char* s, int size) { return gzgets(fp_, s, size); } int GZipFile::flush() { return gzflush(fp_, 0); } int GZipFile::vprintf(const char* format, va_list va) { ssize_t len; #ifdef __MINGW32__ // Windows vsnprintf returns -1 when output is truncated, so we // cannot use same logic in non-MINGW32 code. len = _vscprintf(format, va); if(len == 0) { return 0; } // Include terminate null ++len; if(len > static_cast(buflen_)) { while(static_cast(buflen_) < len) { buflen_ *= 2; } buf_ = reinterpret_cast(realloc(buf_, buflen_)); } len = vsnprintf(buf_, buflen_, format, va); if(len < 0) { return len; } #else // !__MINGW32__ 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; } } #endif // !__MINGW32__ return gzwrite(fp_, buf_, len); } } // namespace aria2