#include #include namespace DB { namespace ErrorCodes { extern const int LZMA_STREAM_DECODER_FAILED; } LZMAInflatingReadBuffer::LZMAInflatingReadBuffer(std::unique_ptr in_, size_t buf_size, char * existing_memory, size_t alignment) : CompressedReadBufferWrapper(std::move(in_), buf_size, existing_memory, alignment), eof_flag(false) { lstr = LZMA_STREAM_INIT; lstr.allocator = nullptr; lstr.next_in = nullptr; lstr.avail_in = 0; lstr.next_out = nullptr; lstr.avail_out = 0; // 500 mb uint64_t memlimit = 500ULL << 20; lzma_ret ret = lzma_stream_decoder(&lstr, memlimit, LZMA_CONCATENATED); // lzma does not provide api for converting error code to string unlike zlib if (ret != LZMA_OK) throw Exception( ErrorCodes::LZMA_STREAM_DECODER_FAILED, "lzma_stream_decoder initialization failed: error code: {}; lzma version: {}", ret, LZMA_VERSION_STRING); } LZMAInflatingReadBuffer::~LZMAInflatingReadBuffer() { lzma_end(&lstr); } bool LZMAInflatingReadBuffer::nextImpl() { if (eof_flag) return false; lzma_action action = LZMA_RUN; lzma_ret ret; do { if (!lstr.avail_in) { in->nextIfAtEnd(); lstr.next_in = reinterpret_cast(in->position()); lstr.avail_in = in->buffer().end() - in->position(); } if (in->eof()) { action = LZMA_FINISH; } lstr.next_out = reinterpret_cast(internal_buffer.begin()); lstr.avail_out = internal_buffer.size(); ret = lzma_code(&lstr, action); in->position() = in->buffer().end() - lstr.avail_in; } while (ret == LZMA_OK && lstr.avail_out == internal_buffer.size()); working_buffer.resize(internal_buffer.size() - lstr.avail_out); if (ret == LZMA_STREAM_END) { if (in->eof()) { eof_flag = true; return !working_buffer.empty(); } else { throw Exception( ErrorCodes::LZMA_STREAM_DECODER_FAILED, "lzma decoder finished, but input stream has not exceeded: error code: {}; lzma version: {}{}", ret, LZMA_VERSION_STRING, getExceptionEntryWithFileName(*in)); } } if (ret != LZMA_OK) throw Exception( ErrorCodes::LZMA_STREAM_DECODER_FAILED, "lzma_stream_decoder failed: error code: error code {}; lzma version: {}{}", ret, LZMA_VERSION_STRING, getExceptionEntryWithFileName(*in)); return true; } }