2018-12-31 20:04:05 +01:00
|
|
|
//
|
2024-01-01 01:07:21 +01:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024
|
2018-12-31 20:04:05 +01:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#include "td/utils/tl_parsers.h"
|
|
|
|
|
2024-02-07 15:50:57 +01:00
|
|
|
#include "td/utils/format.h"
|
2024-02-04 20:12:18 +01:00
|
|
|
#include "td/utils/logging.h"
|
2019-11-26 18:53:50 +01:00
|
|
|
#include "td/utils/misc.h"
|
2024-02-04 20:12:18 +01:00
|
|
|
#include "td/utils/utf8.h"
|
2019-11-26 18:53:50 +01:00
|
|
|
|
2018-12-31 20:04:05 +01:00
|
|
|
namespace td {
|
|
|
|
|
2018-02-12 18:21:29 +01:00
|
|
|
alignas(4) const unsigned char TlParser::empty_data[sizeof(UInt256)] = {}; // static zero-initialized
|
2018-12-31 20:04:05 +01:00
|
|
|
|
2019-11-26 18:53:50 +01:00
|
|
|
TlParser::TlParser(Slice slice) {
|
|
|
|
data_len = left_len = slice.size();
|
|
|
|
if (is_aligned_pointer<4>(slice.begin())) {
|
|
|
|
data = slice.ubegin();
|
|
|
|
} else {
|
|
|
|
int32 *buf;
|
|
|
|
if (data_len <= small_data_array.size() * sizeof(int32)) {
|
|
|
|
buf = &small_data_array[0];
|
|
|
|
} else {
|
|
|
|
LOG(ERROR) << "Unexpected big unaligned data pointer of length " << slice.size() << " at " << slice.begin();
|
|
|
|
data_buf = std::make_unique<int32[]>(1 + data_len / sizeof(int32));
|
|
|
|
buf = data_buf.get();
|
|
|
|
}
|
|
|
|
std::memcpy(buf, slice.begin(), slice.size());
|
|
|
|
data = reinterpret_cast<unsigned char *>(buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-31 20:04:05 +01:00
|
|
|
void TlParser::set_error(const string &error_message) {
|
|
|
|
if (error.empty()) {
|
|
|
|
CHECK(!error_message.empty());
|
|
|
|
error = error_message;
|
|
|
|
error_pos = data_len - left_len;
|
|
|
|
data = empty_data;
|
|
|
|
left_len = 0;
|
|
|
|
data_len = 0;
|
|
|
|
} else {
|
2020-03-13 23:54:13 +01:00
|
|
|
LOG_CHECK(error_pos != std::numeric_limits<size_t>::max() && data_len == 0 && left_len == 0)
|
2023-03-08 15:23:27 +01:00
|
|
|
<< data_len << ' ' << left_len << ' ' << data << ' ' << &empty_data[0] << ' ' << error_pos << ' ' << error;
|
2018-12-31 20:04:05 +01:00
|
|
|
data = empty_data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-26 18:53:50 +01:00
|
|
|
BufferSlice TlBufferParser::as_buffer_slice(Slice slice) {
|
2020-02-23 20:44:14 +01:00
|
|
|
if (slice.empty()) {
|
|
|
|
return BufferSlice();
|
|
|
|
}
|
2019-11-26 18:53:50 +01:00
|
|
|
if (is_aligned_pointer<4>(slice.data())) {
|
|
|
|
return parent_->from_slice(slice);
|
|
|
|
}
|
|
|
|
return BufferSlice(slice);
|
|
|
|
}
|
|
|
|
|
2024-02-04 20:12:18 +01:00
|
|
|
bool TlBufferParser::is_valid_utf8(CSlice str) const {
|
|
|
|
if (check_utf8(str)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
LOG(WARNING) << "Wrong UTF-8 string [[" << str << "]] in " << format::as_hex_dump<4>(parent_->as_slice());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t TlBufferParser::last_utf8_character_position(Slice str) {
|
|
|
|
CHECK(!str.empty());
|
|
|
|
size_t position = str.size() - 1;
|
|
|
|
while (position != 0 && !is_utf8_character_first_code_unit(static_cast<unsigned char>(str[position]))) {
|
|
|
|
position--;
|
|
|
|
}
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
2018-12-31 20:04:05 +01:00
|
|
|
} // namespace td
|