2017-01-27 12:05:45 +00:00
|
|
|
// gzip.h - originally written and placed in the public domain by Wei Dai
|
2016-04-06 02:12:56 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \file gzip.h
|
|
|
|
/// \brief GZIP compression and decompression (RFC 1952)
|
2016-04-06 02:12:56 +00:00
|
|
|
|
2015-11-05 06:59:46 +00:00
|
|
|
#ifndef CRYPTOPP_GZIP_H
|
|
|
|
#define CRYPTOPP_GZIP_H
|
|
|
|
|
|
|
|
#include "cryptlib.h"
|
|
|
|
#include "zdeflate.h"
|
|
|
|
#include "zinflate.h"
|
|
|
|
#include "crc.h"
|
|
|
|
|
|
|
|
NAMESPACE_BEGIN(CryptoPP)
|
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief GZIP Compression (RFC 1952)
|
2015-11-05 06:59:46 +00:00
|
|
|
class Gzip : public Deflator
|
|
|
|
{
|
|
|
|
public:
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Construct a Gzip compressor
|
|
|
|
/// \param attachment an attached transformation
|
|
|
|
/// \param deflateLevel the deflate level
|
|
|
|
/// \param log2WindowSize the window size
|
|
|
|
/// \param detectUncompressible flag to detect if data is compressible
|
|
|
|
/// \details detectUncompressible makes it faster to process uncompressible files, but
|
|
|
|
/// if a file has both compressible and uncompressible parts, it may fail to compress
|
|
|
|
/// some of the compressible parts.
|
2017-03-01 11:10:06 +00:00
|
|
|
Gzip(BufferedTransformation *attachment=NULLPTR, unsigned int deflateLevel=DEFAULT_DEFLATE_LEVEL, unsigned int log2WindowSize=DEFAULT_LOG2_WINDOW_SIZE, bool detectUncompressible=true)
|
2017-08-18 08:59:30 +00:00
|
|
|
: Deflator(attachment, deflateLevel, log2WindowSize, detectUncompressible), m_totalLen(0), m_filetime(0) { }
|
2017-05-12 23:52:42 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Construct a Gzip compressor
|
|
|
|
/// \param parameters a set of NameValuePairs to initialize this object
|
|
|
|
/// \param attachment an attached transformation
|
|
|
|
/// \details Possible parameter names: Log2WindowSize, DeflateLevel, DetectUncompressible
|
2017-03-01 11:10:06 +00:00
|
|
|
Gzip(const NameValuePairs ¶meters, BufferedTransformation *attachment=NULLPTR)
|
2017-08-18 08:59:30 +00:00
|
|
|
: Deflator(parameters, attachment), m_totalLen(0), m_filetime(0)
|
2017-05-12 23:52:42 +00:00
|
|
|
{
|
|
|
|
IsolatedInitialize(parameters);
|
|
|
|
}
|
2015-11-05 06:59:46 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \param filetime the filetime to set in the header. The application is responsible for setting it.
|
2017-05-12 19:09:21 +00:00
|
|
|
void SetFiletime(word32 filetime) { m_filetime = filetime; }
|
2017-05-12 20:53:07 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \param filename the original filename to set in the header. The application is responsible for setting it.
|
|
|
|
/// RFC 1952 requires a ISO/IEC 8859-1 encoding.
|
|
|
|
/// \param throwOnEncodingError if throwOnEncodingError is true, then the filename is checked to ensure it is
|
|
|
|
/// ISO/IEC 8859-1 encoded. If the filename does not adhere to ISO/IEC 8859-1, then a InvalidDataFormat
|
|
|
|
/// is thrown. If throwOnEncodingError is false then the filename is not checked.
|
2017-05-12 19:09:21 +00:00
|
|
|
void SetFilename(const std::string& filename, bool throwOnEncodingError = false);
|
2017-05-12 20:53:07 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \param comment the comment to set in the header. The application is responsible for setting it.
|
|
|
|
/// RFC 1952 requires a ISO/IEC 8859-1 encoding.
|
|
|
|
/// \param throwOnEncodingError if throwOnEncodingError is true, then the comment is checked to ensure it is
|
|
|
|
/// ISO/IEC 8859-1 encoded. If the comment does not adhere to ISO/IEC 8859-1, then a InvalidDataFormat
|
|
|
|
/// is thrown. If throwOnEncodingError is false then the comment is not checked.
|
2017-05-12 19:09:21 +00:00
|
|
|
void SetComment(const std::string& comment, bool throwOnEncodingError = false);
|
|
|
|
|
2017-05-12 23:52:42 +00:00
|
|
|
void IsolatedInitialize(const NameValuePairs ¶meters);
|
|
|
|
|
2015-11-05 06:59:46 +00:00
|
|
|
protected:
|
|
|
|
enum {MAGIC1=0x1f, MAGIC2=0x8b, // flags for the header
|
|
|
|
DEFLATED=8, FAST=4, SLOW=2};
|
2017-05-12 20:53:07 +00:00
|
|
|
|
2017-05-12 19:09:21 +00:00
|
|
|
enum FLAG_MASKS {
|
|
|
|
FILENAME=8, COMMENTS=16};
|
2015-11-05 06:59:46 +00:00
|
|
|
|
|
|
|
void WritePrestreamHeader();
|
|
|
|
void ProcessUncompressedData(const byte *string, size_t length);
|
|
|
|
void WritePoststreamTail();
|
|
|
|
|
|
|
|
word32 m_totalLen;
|
|
|
|
CRC32 m_crc;
|
2017-05-12 20:53:07 +00:00
|
|
|
|
2017-05-12 19:09:21 +00:00
|
|
|
word32 m_filetime;
|
|
|
|
std::string m_filename;
|
|
|
|
std::string m_comment;
|
2015-11-05 06:59:46 +00:00
|
|
|
};
|
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief GZIP Decompression (RFC 1952)
|
2015-11-05 06:59:46 +00:00
|
|
|
class Gunzip : public Inflator
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
typedef Inflator::Err Err;
|
2016-04-06 02:12:56 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Exception thrown when a header decoding error occurs
|
2015-11-05 06:59:46 +00:00
|
|
|
class HeaderErr : public Err {public: HeaderErr() : Err(INVALID_DATA_FORMAT, "Gunzip: header decoding error") {}};
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Exception thrown when the tail is too short
|
2015-11-05 06:59:46 +00:00
|
|
|
class TailErr : public Err {public: TailErr() : Err(INVALID_DATA_FORMAT, "Gunzip: tail too short") {}};
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Exception thrown when a CRC error occurs
|
2015-11-05 06:59:46 +00:00
|
|
|
class CrcErr : public Err {public: CrcErr() : Err(DATA_INTEGRITY_CHECK_FAILED, "Gunzip: CRC check error") {}};
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Exception thrown when a length error occurs
|
2015-11-05 06:59:46 +00:00
|
|
|
class LengthErr : public Err {public: LengthErr() : Err(DATA_INTEGRITY_CHECK_FAILED, "Gunzip: length check error") {}};
|
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Construct a Gunzip decompressor
|
|
|
|
/// \param attachment an attached transformation
|
|
|
|
/// \param repeat decompress multiple compressed streams in series
|
|
|
|
/// \param autoSignalPropagation 0 to turn off MessageEnd signal
|
2017-03-01 11:10:06 +00:00
|
|
|
Gunzip(BufferedTransformation *attachment = NULLPTR, bool repeat = false, int autoSignalPropagation = -1);
|
2015-11-05 06:59:46 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \return the filetime of the stream as set in the header. The application is responsible for setting it on the decompressed file.
|
2017-05-12 19:09:21 +00:00
|
|
|
word32 GetFiletime() const { return m_filetime; }
|
2017-05-12 20:53:07 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \return the filename of the stream as set in the header. The application is responsible for setting it on the decompressed file.
|
|
|
|
/// \param throwOnEncodingError if throwOnEncodingError is true, then the filename is checked to ensure it is
|
|
|
|
/// ISO/IEC 8859-1 encoded. If the filename does not adhere to ISO/IEC 8859-1, then a InvalidDataFormat is thrown.
|
|
|
|
/// If throwOnEncodingError is false then the filename is not checked.
|
2017-05-12 19:09:21 +00:00
|
|
|
const std::string& GetFilename(bool throwOnEncodingError = false) const;
|
2017-05-12 20:53:07 +00:00
|
|
|
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \return the comment of the stream as set in the header.
|
|
|
|
/// \param throwOnEncodingError if throwOnEncodingError is true, then the comment is checked to ensure it is
|
|
|
|
/// ISO/IEC 8859-1 encoded. If the comment does not adhere to ISO/IEC 8859-1, then a InvalidDataFormat is thrown.
|
|
|
|
/// If throwOnEncodingError is false then the comment is not checked.
|
2017-05-12 19:09:21 +00:00
|
|
|
const std::string& GetComment(bool throwOnEncodingError = false) const;
|
|
|
|
|
2015-11-05 06:59:46 +00:00
|
|
|
protected:
|
|
|
|
enum {
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief First header magic value
|
2015-11-05 06:59:46 +00:00
|
|
|
MAGIC1=0x1f,
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Second header magic value
|
2015-11-05 06:59:46 +00:00
|
|
|
MAGIC2=0x8b,
|
2017-11-29 15:54:33 +00:00
|
|
|
/// \brief Deflated flag
|
2015-11-05 06:59:46 +00:00
|
|
|
DEFLATED=8
|
|
|
|
};
|
|
|
|
|
|
|
|
enum FLAG_MASKS {
|
|
|
|
CONTINUED=2, EXTRA_FIELDS=4, FILENAME=8, COMMENTS=16, ENCRYPTED=32};
|
|
|
|
|
|
|
|
unsigned int MaxPrestreamHeaderSize() const {return 1024;}
|
|
|
|
void ProcessPrestreamHeader();
|
|
|
|
void ProcessDecompressedData(const byte *string, size_t length);
|
|
|
|
unsigned int MaxPoststreamTailSize() const {return 8;}
|
|
|
|
void ProcessPoststreamTail();
|
|
|
|
|
|
|
|
word32 m_length;
|
|
|
|
CRC32 m_crc;
|
2017-05-12 19:09:21 +00:00
|
|
|
|
|
|
|
word32 m_filetime;
|
|
|
|
std::string m_filename;
|
|
|
|
std::string m_comment;
|
2015-11-05 06:59:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
NAMESPACE_END
|
|
|
|
|
|
|
|
#endif
|