mirror of
https://github.com/ton-blockchain/ton
synced 2025-03-09 15:40:10 +00:00
initial commit
This commit is contained in:
commit
c2da007f40
1610 changed files with 398047 additions and 0 deletions
220
tdutils/td/utils/Gzip.cpp
Normal file
220
tdutils/td/utils/Gzip.cpp
Normal file
|
@ -0,0 +1,220 @@
|
|||
/*
|
||||
This file is part of TON Blockchain Library.
|
||||
|
||||
TON Blockchain Library is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU Lesser General Public License as published by
|
||||
the Free Software Foundation, either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
TON Blockchain Library is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public License
|
||||
along with TON Blockchain Library. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Copyright 2017-2019 Telegram Systems LLP
|
||||
*/
|
||||
#include "td/utils/Gzip.h"
|
||||
|
||||
char disable_linker_warning_about_empty_file_gzip_cpp TD_UNUSED;
|
||||
|
||||
#if TD_HAVE_ZLIB
|
||||
#include "td/utils/logging.h"
|
||||
|
||||
#include <cstring>
|
||||
#include <limits>
|
||||
#include <utility>
|
||||
|
||||
#include <zlib.h>
|
||||
|
||||
namespace td {
|
||||
|
||||
class Gzip::Impl {
|
||||
public:
|
||||
z_stream stream_;
|
||||
|
||||
// z_stream is not copyable nor movable
|
||||
Impl() = default;
|
||||
Impl(const Impl &other) = delete;
|
||||
Impl &operator=(const Impl &other) = delete;
|
||||
Impl(Impl &&other) = delete;
|
||||
Impl &operator=(Impl &&other) = delete;
|
||||
~Impl() = default;
|
||||
};
|
||||
|
||||
Status Gzip::init_encode() {
|
||||
CHECK(mode_ == Empty);
|
||||
init_common();
|
||||
mode_ = Encode;
|
||||
int ret = deflateInit2(&impl_->stream_, 6, Z_DEFLATED, 15, MAX_MEM_LEVEL, Z_DEFAULT_STRATEGY);
|
||||
if (ret != Z_OK) {
|
||||
return Status::Error(PSLICE() << "zlib deflate init failed: " << ret);
|
||||
}
|
||||
return Status::OK();
|
||||
}
|
||||
|
||||
Status Gzip::init_decode() {
|
||||
CHECK(mode_ == Empty);
|
||||
init_common();
|
||||
mode_ = Decode;
|
||||
int ret = inflateInit2(&impl_->stream_, MAX_WBITS + 32);
|
||||
if (ret != Z_OK) {
|
||||
return Status::Error(PSLICE() << "zlib inflate init failed: " << ret);
|
||||
}
|
||||
return Status::OK();
|
||||
}
|
||||
|
||||
void Gzip::set_input(Slice input) {
|
||||
CHECK(input_size_ == 0);
|
||||
CHECK(!close_input_flag_);
|
||||
CHECK(input.size() <= std::numeric_limits<uInt>::max());
|
||||
CHECK(impl_->stream_.avail_in == 0);
|
||||
input_size_ = input.size();
|
||||
impl_->stream_.avail_in = static_cast<uInt>(input.size());
|
||||
impl_->stream_.next_in = reinterpret_cast<Bytef *>(const_cast<char *>(input.data()));
|
||||
}
|
||||
|
||||
void Gzip::set_output(MutableSlice output) {
|
||||
CHECK(output_size_ == 0);
|
||||
CHECK(output.size() <= std::numeric_limits<uInt>::max());
|
||||
CHECK(impl_->stream_.avail_out == 0);
|
||||
output_size_ = output.size();
|
||||
impl_->stream_.avail_out = static_cast<uInt>(output.size());
|
||||
impl_->stream_.next_out = reinterpret_cast<Bytef *>(output.data());
|
||||
}
|
||||
|
||||
Result<Gzip::State> Gzip::run() {
|
||||
while (true) {
|
||||
int ret;
|
||||
if (mode_ == Decode) {
|
||||
ret = inflate(&impl_->stream_, Z_NO_FLUSH);
|
||||
} else {
|
||||
ret = deflate(&impl_->stream_, close_input_flag_ ? Z_FINISH : Z_NO_FLUSH);
|
||||
}
|
||||
|
||||
if (ret == Z_OK) {
|
||||
return Running;
|
||||
}
|
||||
if (ret == Z_STREAM_END) {
|
||||
// TODO(now): fail if input is not empty;
|
||||
clear();
|
||||
return Done;
|
||||
}
|
||||
clear();
|
||||
return Status::Error(PSLICE() << "zlib error " << ret);
|
||||
}
|
||||
}
|
||||
|
||||
size_t Gzip::left_input() const {
|
||||
return impl_->stream_.avail_in;
|
||||
}
|
||||
size_t Gzip::left_output() const {
|
||||
return impl_->stream_.avail_out;
|
||||
}
|
||||
|
||||
void Gzip::init_common() {
|
||||
std::memset(&impl_->stream_, 0, sizeof(impl_->stream_));
|
||||
impl_->stream_.zalloc = Z_NULL;
|
||||
impl_->stream_.zfree = Z_NULL;
|
||||
impl_->stream_.opaque = Z_NULL;
|
||||
impl_->stream_.avail_in = 0;
|
||||
impl_->stream_.next_in = nullptr;
|
||||
impl_->stream_.avail_out = 0;
|
||||
impl_->stream_.next_out = nullptr;
|
||||
|
||||
input_size_ = 0;
|
||||
output_size_ = 0;
|
||||
|
||||
close_input_flag_ = false;
|
||||
}
|
||||
|
||||
void Gzip::clear() {
|
||||
if (mode_ == Decode) {
|
||||
inflateEnd(&impl_->stream_);
|
||||
} else if (mode_ == Encode) {
|
||||
deflateEnd(&impl_->stream_);
|
||||
}
|
||||
mode_ = Empty;
|
||||
}
|
||||
|
||||
Gzip::Gzip() : impl_(make_unique<Impl>()) {
|
||||
}
|
||||
|
||||
Gzip::Gzip(Gzip &&other) : Gzip() {
|
||||
swap(other);
|
||||
}
|
||||
|
||||
Gzip &Gzip::operator=(Gzip &&other) {
|
||||
CHECK(this != &other);
|
||||
clear();
|
||||
swap(other);
|
||||
return *this;
|
||||
}
|
||||
|
||||
void Gzip::swap(Gzip &other) {
|
||||
using std::swap;
|
||||
swap(impl_, other.impl_);
|
||||
swap(input_size_, other.input_size_);
|
||||
swap(output_size_, other.output_size_);
|
||||
swap(close_input_flag_, other.close_input_flag_);
|
||||
swap(mode_, other.mode_);
|
||||
}
|
||||
|
||||
Gzip::~Gzip() {
|
||||
clear();
|
||||
}
|
||||
|
||||
BufferSlice gzdecode(Slice s) {
|
||||
Gzip gzip;
|
||||
gzip.init_decode().ensure();
|
||||
ChainBufferWriter message;
|
||||
gzip.set_input(s);
|
||||
gzip.close_input();
|
||||
double k = 2;
|
||||
gzip.set_output(message.prepare_append(static_cast<size_t>(static_cast<double>(s.size()) * k)));
|
||||
while (true) {
|
||||
auto r_state = gzip.run();
|
||||
if (r_state.is_error()) {
|
||||
return BufferSlice();
|
||||
}
|
||||
auto state = r_state.ok();
|
||||
if (state == Gzip::Done) {
|
||||
message.confirm_append(gzip.flush_output());
|
||||
break;
|
||||
}
|
||||
if (gzip.need_input()) {
|
||||
return BufferSlice();
|
||||
}
|
||||
if (gzip.need_output()) {
|
||||
message.confirm_append(gzip.flush_output());
|
||||
k *= 1.5;
|
||||
gzip.set_output(message.prepare_append(static_cast<size_t>(static_cast<double>(gzip.left_input()) * k)));
|
||||
}
|
||||
}
|
||||
return message.extract_reader().move_as_buffer_slice();
|
||||
}
|
||||
|
||||
BufferSlice gzencode(Slice s, double k) {
|
||||
Gzip gzip;
|
||||
gzip.init_encode().ensure();
|
||||
gzip.set_input(s);
|
||||
gzip.close_input();
|
||||
size_t max_size = static_cast<size_t>(static_cast<double>(s.size()) * k);
|
||||
BufferWriter message{max_size};
|
||||
gzip.set_output(message.prepare_append());
|
||||
auto r_state = gzip.run();
|
||||
if (r_state.is_error()) {
|
||||
return BufferSlice();
|
||||
}
|
||||
auto state = r_state.ok();
|
||||
if (state != Gzip::Done) {
|
||||
return BufferSlice();
|
||||
}
|
||||
message.confirm_append(gzip.flush_output());
|
||||
return message.as_buffer_slice();
|
||||
}
|
||||
|
||||
} // namespace td
|
||||
#endif
|
Loading…
Add table
Add a link
Reference in a new issue