ClickHouse/dbms/src/Compression/CompressionCodecZSTD.cpp

84 lines
2.6 KiB
C++
Raw Normal View History

2018-10-11 02:57:48 +00:00
#include <Compression/CompressionCodecZSTD.h>
#include <IO/CompressedStream.h>
2018-12-14 13:27:35 +00:00
#include <IO/ReadHelpers.h>
2018-10-11 02:57:48 +00:00
#include <Compression/CompressionFactory.h>
#include <zstd.h>
#include <Core/Field.h>
#include <Parsers/IAST.h>
#include <Parsers/ASTLiteral.h>
#include <Common/typeid_cast.h>
namespace DB
{
namespace ErrorCodes
{
extern const int CANNOT_COMPRESS;
extern const int CANNOT_DECOMPRESS;
2018-12-14 13:27:35 +00:00
extern const int ILLEGAL_SYNTAX_FOR_CODEC_TYPE;
2018-10-11 02:57:48 +00:00
}
char CompressionCodecZSTD::getMethodByte()
{
return static_cast<char>(CompressionMethodByte::ZSTD);
}
void CompressionCodecZSTD::getCodecDesc(String & codec_desc)
{
codec_desc = "ZSTD";
}
size_t CompressionCodecZSTD::getCompressedReserveSize(size_t uncompressed_size)
{
return ZSTD_compressBound(uncompressed_size);
}
size_t CompressionCodecZSTD::compress(char * source, size_t source_size, char * dest)
{
size_t compressed_size = ZSTD_compress(dest, ZSTD_compressBound(source_size), source, source_size, level);
if (ZSTD_isError(compressed_size))
throw Exception("Cannot compress block with ZSTD: " + std::string(ZSTD_getErrorName(compressed_size)), ErrorCodes::CANNOT_COMPRESS);
return compressed_size;
}
size_t CompressionCodecZSTD::decompress(char * source, size_t source_size, char * dest, size_t size_decompressed)
{
size_t res = ZSTD_decompress(dest, size_decompressed, source, source_size);
// compressed_buffer + COMPRESSED_BLOCK_HEADER_SIZE, size_compressed_without_checksum - COMPRESSED_BLOCK_HEADER_SIZE);
if (ZSTD_isError(res))
throw Exception("Cannot ZSTD_decompress: " + std::string(ZSTD_getErrorName(res)), ErrorCodes::CANNOT_DECOMPRESS);
return size_decompressed;
}
CompressionCodecZSTD::CompressionCodecZSTD(int level)
:level(level)
{
}
void registerCodecZSTD(CompressionCodecFactory & factory)
{
UInt8 method_code = static_cast<char>(CompressionMethodByte::ZSTD);
factory.registerCompressionCodec("ZSTD", method_code, [&](const ASTPtr & arguments) -> CompressionCodecPtr
{
int level = 0;
if (arguments && !arguments->children.empty())
{
2018-12-14 13:27:35 +00:00
if (arguments->children.size() != 1)
throw Exception("ZSTD codec must have 1 parameter, given " + std::to_string(arguments->children.size()), ErrorCodes::ILLEGAL_SYNTAX_FOR_CODEC_TYPE);
2018-10-11 02:57:48 +00:00
const auto children = arguments->children;
const ASTLiteral * literal = static_cast<const ASTLiteral *>(children[0].get());
level = literal->value.safeGet<UInt64>();
}
return std::make_shared<CompressionCodecZSTD>(level);
});
}
}