2018-02-03 19:49:39 +03:00
|
|
|
//
|
2019-01-01 01:02:34 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2019
|
2018-02-03 19:49:39 +03: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/StringBuilder.h"
|
|
|
|
|
2018-12-11 23:18:58 +03:00
|
|
|
#include "td/utils/logging.h"
|
2018-02-03 22:40:06 +03:00
|
|
|
#include "td/utils/misc.h"
|
2018-02-11 18:58:18 +03:00
|
|
|
#include "td/utils/port/thread_local.h"
|
2018-12-11 23:18:58 +03:00
|
|
|
#include "td/utils/Slice.h"
|
2018-02-03 22:40:06 +03:00
|
|
|
|
2018-02-03 19:49:39 +03:00
|
|
|
#include <cstdio>
|
2018-12-11 23:18:58 +03:00
|
|
|
#include <cstring>
|
2018-06-05 02:34:51 +03:00
|
|
|
#include <limits>
|
2018-02-03 22:40:06 +03:00
|
|
|
#include <locale>
|
2018-12-11 23:18:58 +03:00
|
|
|
#include <memory>
|
2018-02-03 22:40:06 +03:00
|
|
|
#include <sstream>
|
2018-06-05 02:34:51 +03:00
|
|
|
#include <utility>
|
2018-02-03 19:49:39 +03:00
|
|
|
|
|
|
|
namespace td {
|
|
|
|
|
2018-12-11 23:18:58 +03:00
|
|
|
StringBuilder::StringBuilder(MutableSlice slice, bool use_buffer)
|
|
|
|
: begin_ptr_(slice.begin()), current_ptr_(begin_ptr_), use_buffer_(use_buffer) {
|
|
|
|
if (slice.size() <= reserved_size) {
|
|
|
|
auto buffer_size = reserved_size + 100;
|
|
|
|
buffer_ = std::make_unique<char[]>(buffer_size);
|
|
|
|
begin_ptr_ = buffer_.get();
|
|
|
|
current_ptr_ = begin_ptr_;
|
|
|
|
end_ptr_ = begin_ptr_ + buffer_size - reserved_size;
|
|
|
|
} else {
|
|
|
|
end_ptr_ = slice.end() - reserved_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(Slice slice) {
|
|
|
|
size_t size = slice.size();
|
|
|
|
if (unlikely(!reserve(size))) {
|
|
|
|
if (end_ptr_ < current_ptr_) {
|
|
|
|
return on_error();
|
|
|
|
}
|
|
|
|
auto available_size = static_cast<size_t>(end_ptr_ + reserved_size - 1 - current_ptr_);
|
|
|
|
if (size > available_size) {
|
|
|
|
error_flag_ = true;
|
|
|
|
size = available_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::memcpy(current_ptr_, slice.begin(), size);
|
|
|
|
current_ptr_ += size;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2018-06-05 02:34:51 +03:00
|
|
|
template <class T>
|
|
|
|
static char *print_uint(char *current_ptr, T x) {
|
|
|
|
if (x < 100) {
|
|
|
|
if (x < 10) {
|
|
|
|
*current_ptr++ = static_cast<char>('0' + x);
|
|
|
|
} else {
|
|
|
|
*current_ptr++ = static_cast<char>('0' + x / 10);
|
|
|
|
*current_ptr++ = static_cast<char>('0' + x % 10);
|
|
|
|
}
|
|
|
|
return current_ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto begin_ptr = current_ptr;
|
|
|
|
do {
|
|
|
|
*current_ptr++ = static_cast<char>('0' + x % 10);
|
|
|
|
x /= 10;
|
|
|
|
} while (x > 0);
|
|
|
|
|
|
|
|
auto end_ptr = current_ptr - 1;
|
|
|
|
while (begin_ptr < end_ptr) {
|
|
|
|
std::swap(*begin_ptr++, *end_ptr--);
|
|
|
|
}
|
|
|
|
|
|
|
|
return current_ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
static char *print_int(char *current_ptr, T x) {
|
|
|
|
if (x < 0) {
|
|
|
|
if (x == std::numeric_limits<T>::min()) {
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << x;
|
|
|
|
auto len = narrow_cast<int>(static_cast<std::streamoff>(ss.tellp()));
|
|
|
|
ss.read(current_ptr, len);
|
|
|
|
return current_ptr + len;
|
|
|
|
}
|
|
|
|
|
|
|
|
*current_ptr++ = '-';
|
|
|
|
x = -x;
|
|
|
|
}
|
|
|
|
|
|
|
|
return print_uint(current_ptr, x);
|
|
|
|
}
|
|
|
|
|
2018-10-30 14:07:28 +03:00
|
|
|
bool StringBuilder::reserve_inner(size_t size) {
|
|
|
|
if (!use_buffer_) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-30 17:22:44 +03:00
|
|
|
|
2018-10-30 14:07:28 +03:00
|
|
|
size_t old_data_size = current_ptr_ - begin_ptr_;
|
2018-10-30 17:22:44 +03:00
|
|
|
if (size >= std::numeric_limits<size_t>::max() - reserved_size - old_data_size - 1) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-30 14:07:28 +03:00
|
|
|
size_t need_data_size = old_data_size + size;
|
|
|
|
size_t old_buffer_size = end_ptr_ - begin_ptr_;
|
2018-10-30 17:22:44 +03:00
|
|
|
if (old_buffer_size >= (std::numeric_limits<size_t>::max() - reserved_size) / 2 - 2) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-10-30 14:07:28 +03:00
|
|
|
size_t new_buffer_size = (old_buffer_size + 1) * 2;
|
|
|
|
if (new_buffer_size < need_data_size) {
|
|
|
|
new_buffer_size = need_data_size;
|
|
|
|
}
|
|
|
|
if (new_buffer_size < 100) {
|
|
|
|
new_buffer_size = 100;
|
|
|
|
}
|
|
|
|
new_buffer_size += reserved_size;
|
2018-10-30 17:22:44 +03:00
|
|
|
auto new_buffer = std::make_unique<char[]>(new_buffer_size);
|
|
|
|
std::memcpy(new_buffer.get(), begin_ptr_, old_data_size);
|
|
|
|
buffer_ = std::move(new_buffer);
|
2018-10-30 14:07:28 +03:00
|
|
|
begin_ptr_ = buffer_.get();
|
|
|
|
current_ptr_ = begin_ptr_ + old_data_size;
|
|
|
|
end_ptr_ = begin_ptr_ + new_buffer_size - reserved_size;
|
2018-10-30 17:22:44 +03:00
|
|
|
CHECK(end_ptr_ > current_ptr_);
|
|
|
|
CHECK(static_cast<size_t>(end_ptr_ - current_ptr_) >= size);
|
2018-10-30 14:07:28 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-03 19:49:39 +03:00
|
|
|
StringBuilder &StringBuilder::operator<<(int x) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-06-05 02:34:51 +03:00
|
|
|
current_ptr_ = print_int(current_ptr_, x);
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(unsigned int x) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-06-05 02:34:51 +03:00
|
|
|
current_ptr_ = print_uint(current_ptr_, x);
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(long int x) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-06-05 02:34:51 +03:00
|
|
|
current_ptr_ = print_int(current_ptr_, x);
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(long unsigned int x) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-06-05 02:34:51 +03:00
|
|
|
current_ptr_ = print_uint(current_ptr_, x);
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(long long int x) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-06-05 02:34:51 +03:00
|
|
|
current_ptr_ = print_int(current_ptr_, x);
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(long long unsigned int x) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-06-05 02:34:51 +03:00
|
|
|
current_ptr_ = print_uint(current_ptr_, x);
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2018-02-11 14:38:08 +03:00
|
|
|
StringBuilder &StringBuilder::operator<<(FixedDouble x) {
|
2018-10-30 17:22:44 +03:00
|
|
|
if (unlikely(!reserve(std::numeric_limits<double>::max_exponent10 + x.precision + 4))) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
2018-02-03 22:40:06 +03:00
|
|
|
|
|
|
|
static TD_THREAD_LOCAL std::stringstream *ss;
|
|
|
|
if (init_thread_local<std::stringstream>(ss)) {
|
2018-10-26 17:11:20 +03:00
|
|
|
auto previous_locale = ss->imbue(std::locale::classic());
|
2018-02-11 12:47:11 +03:00
|
|
|
ss->setf(std::ios_base::fixed, std::ios_base::floatfield);
|
2018-02-03 22:40:06 +03:00
|
|
|
} else {
|
|
|
|
ss->str(std::string());
|
|
|
|
ss->clear();
|
|
|
|
}
|
2018-02-11 14:38:08 +03:00
|
|
|
ss->precision(x.precision);
|
|
|
|
*ss << x.d;
|
2018-02-03 22:40:06 +03:00
|
|
|
|
|
|
|
int len = narrow_cast<int>(static_cast<std::streamoff>(ss->tellp()));
|
2018-02-03 19:49:39 +03:00
|
|
|
auto left = end_ptr_ + reserved_size - current_ptr_;
|
|
|
|
if (unlikely(len >= left)) {
|
|
|
|
error_flag_ = true;
|
2018-02-03 23:13:05 +03:00
|
|
|
len = left ? narrow_cast<int>(left - 1) : 0;
|
2018-02-03 19:49:39 +03:00
|
|
|
}
|
2018-02-03 22:40:06 +03:00
|
|
|
ss->read(current_ptr_, len);
|
|
|
|
current_ptr_ += len;
|
2018-02-03 19:49:39 +03:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &StringBuilder::operator<<(const void *ptr) {
|
2018-10-30 14:07:28 +03:00
|
|
|
if (unlikely(!reserve())) {
|
2018-02-03 19:49:39 +03:00
|
|
|
return on_error();
|
|
|
|
}
|
|
|
|
current_ptr_ += std::snprintf(current_ptr_, reserved_size, "%p", ptr);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace td
|