A massively spiffy yet delicately unobtrusive compression library. (grpc依赖)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

467 lines
12 KiB

14 years ago
/*
* A C++ I/O streams interface to the zlib gz* functions
14 years ago
*
14 years ago
* by Ludwig Schwardt <schwardt@sun.ac.za>
* original version by Kevin Ruland <kevin@rodin.wustl.edu>
14 years ago
*
14 years ago
* This version is standard-compliant and compatible with gcc 3.x.
*/
#ifndef ZFSTREAM_H
#define ZFSTREAM_H
#include <istream> // not iostream, since we don't need cin/cout
#include <ostream>
#include "zlib.h"
/*****************************************************************************/
/**
* @brief Gzipped file stream buffer class.
*
* This class implements basic_filebuf for gzipped files. It doesn't yet support
14 years ago
* seeking (allowed by zlib but slow/limited), putback and read/write access
* (tricky). Otherwise, it attempts to be a drop-in replacement for the standard
14 years ago
* file streambuf.
*/
14 years ago
class gzfilebuf : public std::streambuf
14 years ago
{
14 years ago
public:
14 years ago
// Default constructor.
gzfilebuf();
14 years ago
14 years ago
// Destructor.
14 years ago
virtual
14 years ago
~gzfilebuf();
/**
* @brief Set compression level and strategy on the fly.
14 years ago
* @param comp_level Compression level (see zlib.h for allowed values)
* @param comp_strategy Compression strategy (see zlib.h for allowed values)
14 years ago
* @return Z_OK on success, Z_STREAM_ERROR otherwise.
14 years ago
*
14 years ago
* Unfortunately, these parameters cannot be modified separately, as the
* previous zfstream version assumed. Since the strategy is seldom changed,
* it can default and setcompression(level) then becomes like the old
* setcompressionlevel(level).
*/
14 years ago
int
setcompression(int comp_level,
int comp_strategy = Z_DEFAULT_STRATEGY);
14 years ago
/**
* @brief Check if file is open.
* @return True if file is open.
*/
14 years ago
bool
14 years ago
is_open() const { return (file != NULL); }
14 years ago
14 years ago
/**
* @brief Open gzipped file.
* @param name File name.
* @param mode Open mode flags.
* @return @c this on success, NULL on failure.
*/
14 years ago
gzfilebuf*
open(const char* name,
14 years ago
std::ios_base::openmode mode);
14 years ago
14 years ago
/**
* @brief Attach to already open gzipped file.
* @param fd File descriptor.
* @param mode Open mode flags.
* @return @c this on success, NULL on failure.
*/
14 years ago
gzfilebuf*
attach(int fd,
std::ios_base::openmode mode);
14 years ago
/**
* @brief Close gzipped file.
* @return @c this on success, NULL on failure.
*/
14 years ago
gzfilebuf*
14 years ago
close();
14 years ago
14 years ago
protected:
/**
* @brief Convert ios open mode int to mode string used by zlib.
* @return True if valid mode flag combination.
*/
14 years ago
bool
open_mode(std::ios_base::openmode mode,
char* c_mode) const;
14 years ago
/**
* @brief Number of characters available in stream buffer.
* @return Number of characters.
14 years ago
*
14 years ago
* This indicates number of characters in get area of stream buffer.
* These characters can be read without accessing the gzipped file.
*/
virtual std::streamsize
showmanyc();
14 years ago
14 years ago
/**
* @brief Fill get area from gzipped file.
* @return First character in get area on success, EOF on error.
14 years ago
*
14 years ago
* This actually reads characters from gzipped file to stream
* buffer. Always buffered.
*/
virtual int_type
underflow();
14 years ago
14 years ago
/**
* @brief Write put area to gzipped file.
* @param c Extra character to add to buffer contents.
* @return Non-EOF on success, EOF on error.
14 years ago
*
* This actually writes characters in stream buffer to
* gzipped file. With unbuffered output this is done one
14 years ago
* character at a time.
14 years ago
*/
virtual int_type
14 years ago
overflow(int_type c = traits_type::eof());
14 years ago
14 years ago
/**
* @brief Installs external stream buffer.
* @param p Pointer to char buffer.
* @param n Size of external buffer.
* @return @c this on success, NULL on failure.
14 years ago
*
14 years ago
* Call setbuf(0,0) to enable unbuffered output.
*/
14 years ago
virtual std::streambuf*
setbuf(char_type* p,
std::streamsize n);
14 years ago
/**
* @brief Flush stream buffer to file.
* @return 0 on success, -1 on error.
14 years ago
*
14 years ago
* This calls underflow(EOF) to do the job.
*/
14 years ago
virtual int
14 years ago
sync();
14 years ago
14 years ago
//
// Some future enhancements
14 years ago
//
14 years ago
// virtual int_type uflow();
// virtual int_type pbackfail(int_type c = traits_type::eof());
14 years ago
// virtual pos_type
// seekoff(off_type off,
// std::ios_base::seekdir way,
// std::ios_base::openmode mode = std::ios_base::in|std::ios_base::out);
// virtual pos_type
// seekpos(pos_type sp,
// std::ios_base::openmode mode = std::ios_base::in|std::ios_base::out);
14 years ago
private:
/**
* @brief Allocate internal buffer.
14 years ago
*
14 years ago
* This function is safe to call multiple times. It will ensure
* that a proper internal buffer exists if it is required. If the
* buffer already exists or is external, the buffer pointers will be
* reset to their original state.
*/
14 years ago
void
14 years ago
enable_buffer();
14 years ago
14 years ago
/**
* @brief Destroy internal buffer.
14 years ago
*
14 years ago
* This function is safe to call multiple times. It will ensure
* that the internal buffer is deallocated if it exists. In any
* case, it will also reset the buffer pointers.
*/
14 years ago
void
14 years ago
disable_buffer();
14 years ago
14 years ago
/**
* Underlying file pointer.
*/
gzFile file;
14 years ago
14 years ago
/**
* Mode in which file was opened.
*/
std::ios_base::openmode io_mode;
14 years ago
14 years ago
/**
* @brief True if this object owns file descriptor.
*
14 years ago
* This makes the class responsible for closing the file
14 years ago
* upon destruction.
*/
bool own_fd;
14 years ago
14 years ago
/**
* @brief Stream buffer.
14 years ago
*
* For simplicity this remains allocated on the free store for the
14 years ago
* entire life span of the gzfilebuf object, unless replaced by setbuf.
*/
char_type* buffer;
14 years ago
14 years ago
/**
* @brief Stream buffer size.
14 years ago
*
14 years ago
* Defaults to system default buffer size (typically 8192 bytes).
* Modified by setbuf.
*/
std::streamsize buffer_size;
14 years ago
14 years ago
/**
* @brief True if this object owns stream buffer.
*
14 years ago
* This makes the class responsible for deleting the buffer
14 years ago
* upon destruction.
*/
bool own_buffer;
};
/*****************************************************************************/
/**
* @brief Gzipped file input stream class.
*
* This class implements ifstream for gzipped files. Seeking and putback
* is not supported yet.
*/
14 years ago
class gzifstream : public std::istream
14 years ago
{
public:
// Default constructor
gzifstream();
14 years ago
14 years ago
/**
* @brief Construct stream on gzipped file to be opened.
* @param name File name.
* @param mode Open mode flags (forced to contain ios::in).
*/
explicit
14 years ago
gzifstream(const char* name,
std::ios_base::openmode mode = std::ios_base::in);
14 years ago
/**
* @brief Construct stream on already open gzipped file.
* @param fd File descriptor.
* @param mode Open mode flags (forced to contain ios::in).
*/
14 years ago
explicit
gzifstream(int fd,
std::ios_base::openmode mode = std::ios_base::in);
14 years ago
/**
* Obtain underlying stream buffer.
14 years ago
*/
gzfilebuf*
14 years ago
rdbuf() const
14 years ago
{ return const_cast<gzfilebuf*>(&sb); }
14 years ago
/**
* @brief Check if file is open.
* @return True if file is open.
*/
14 years ago
bool
14 years ago
is_open() { return sb.is_open(); }
14 years ago
14 years ago
/**
* @brief Open gzipped file.
* @param name File name.
* @param mode Open mode flags (forced to contain ios::in).
14 years ago
*
14 years ago
* Stream will be in state good() if file opens successfully;
* otherwise in state fail(). This differs from the behavior of
* ifstream, which never sets the state to good() and therefore
* won't allow you to reuse the stream for a second file unless
* you manually clear() the state. The choice is a matter of
* convenience.
*/
14 years ago
void
open(const char* name,
14 years ago
std::ios_base::openmode mode = std::ios_base::in);
/**
* @brief Attach to already open gzipped file.
* @param fd File descriptor.
* @param mode Open mode flags (forced to contain ios::in).
14 years ago
*
14 years ago
* Stream will be in state good() if attach succeeded; otherwise
* in state fail().
*/
14 years ago
void
attach(int fd,
std::ios_base::openmode mode = std::ios_base::in);
14 years ago
/**
* @brief Close gzipped file.
14 years ago
*
14 years ago
* Stream will be in state fail() if close failed.
*/
14 years ago
void
14 years ago
close();
14 years ago
14 years ago
private:
/**
* Underlying stream buffer.
14 years ago
*/
14 years ago
gzfilebuf sb;
};
/*****************************************************************************/
/**
* @brief Gzipped file output stream class.
*
* This class implements ofstream for gzipped files. Seeking and putback
* is not supported yet.
*/
class gzofstream : public std::ostream
{
public:
// Default constructor
gzofstream();
14 years ago
14 years ago
/**
* @brief Construct stream on gzipped file to be opened.
* @param name File name.
* @param mode Open mode flags (forced to contain ios::out).
*/
explicit
14 years ago
gzofstream(const char* name,
std::ios_base::openmode mode = std::ios_base::out);
14 years ago
/**
* @brief Construct stream on already open gzipped file.
* @param fd File descriptor.
* @param mode Open mode flags (forced to contain ios::out).
*/
14 years ago
explicit
gzofstream(int fd,
std::ios_base::openmode mode = std::ios_base::out);
14 years ago
/**
* Obtain underlying stream buffer.
14 years ago
*/
gzfilebuf*
14 years ago
rdbuf() const
14 years ago
{ return const_cast<gzfilebuf*>(&sb); }
14 years ago
/**
* @brief Check if file is open.
* @return True if file is open.
*/
14 years ago
bool
14 years ago
is_open() { return sb.is_open(); }
14 years ago
14 years ago
/**
* @brief Open gzipped file.
* @param name File name.
* @param mode Open mode flags (forced to contain ios::out).
14 years ago
*
14 years ago
* Stream will be in state good() if file opens successfully;
* otherwise in state fail(). This differs from the behavior of
* ofstream, which never sets the state to good() and therefore
* won't allow you to reuse the stream for a second file unless
* you manually clear() the state. The choice is a matter of
* convenience.
*/
14 years ago
void
open(const char* name,
14 years ago
std::ios_base::openmode mode = std::ios_base::out);
/**
* @brief Attach to already open gzipped file.
* @param fd File descriptor.
* @param mode Open mode flags (forced to contain ios::out).
14 years ago
*
14 years ago
* Stream will be in state good() if attach succeeded; otherwise
* in state fail().
*/
14 years ago
void
attach(int fd,
std::ios_base::openmode mode = std::ios_base::out);
14 years ago
/**
* @brief Close gzipped file.
14 years ago
*
14 years ago
* Stream will be in state fail() if close failed.
*/
14 years ago
void
14 years ago
close();
14 years ago
14 years ago
private:
/**
* Underlying stream buffer.
14 years ago
*/
14 years ago
gzfilebuf sb;
};
/*****************************************************************************/
14 years ago
14 years ago
/**
* @brief Gzipped file output stream manipulator class.
*
* This class defines a two-argument manipulator for gzofstream. It is used
* as base for the setcompression(int,int) manipulator.
*/
template<typename T1, typename T2>
class gzomanip2
{
public:
// Allows insertor to peek at internals
template <typename Ta, typename Tb>
14 years ago
friend gzofstream&
operator<<(gzofstream&,
const gzomanip2<Ta,Tb>&);
14 years ago
// Constructor
gzomanip2(gzofstream& (*f)(gzofstream&, T1, T2),
14 years ago
T1 v1,
T2 v2);
14 years ago
private:
// Underlying manipulator function
gzofstream&
(*func)(gzofstream&, T1, T2);
14 years ago
// Arguments for manipulator function
14 years ago
T1 val1;
T2 val2;
};
/*****************************************************************************/
14 years ago
14 years ago
// Manipulator function thunks through to stream buffer
14 years ago
inline gzofstream&
14 years ago
setcompression(gzofstream &gzs, int l, int s = Z_DEFAULT_STRATEGY)
{
(gzs.rdbuf())->setcompression(l, s);
return gzs;
}
// Manipulator constructor stores arguments
template<typename T1, typename T2>
14 years ago
inline
14 years ago
gzomanip2<T1,T2>::gzomanip2(gzofstream &(*f)(gzofstream &, T1, T2),
14 years ago
T1 v1,
T2 v2)
14 years ago
: func(f), val1(v1), val2(v2)
{ }
// Insertor applies underlying manipulator function to stream
template<typename T1, typename T2>
14 years ago
inline gzofstream&
operator<<(gzofstream& s, const gzomanip2<T1,T2>& m)
14 years ago
{ return (*m.func)(s, m.val1, m.val2); }
// Insert this onto stream to simplify setting of compression level
14 years ago
inline gzomanip2<int,int>
setcompression(int l, int s = Z_DEFAULT_STRATEGY)
14 years ago
{ return gzomanip2<int,int>(&setcompression, l, s); }
#endif // ZFSTREAM_H