gnss-sim/3rdparty/boost/mysql/impl/row_impl.ipp

199 lines
5.6 KiB
C++

//
// Copyright (c) 2019-2024 Ruben Perez Hidalgo (rubenperez038 at gmail dot com)
//
// 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)
//
#ifndef BOOST_MYSQL_IMPL_ROW_IMPL_IPP
#define BOOST_MYSQL_IMPL_ROW_IMPL_IPP
#pragma once
#include <boost/mysql/detail/row_impl.hpp>
namespace boost {
namespace mysql {
namespace detail {
inline std::size_t get_string_size(field_view f)
{
switch (f.kind())
{
case field_kind::string: return f.get_string().size();
case field_kind::blob: return f.get_blob().size();
default: return 0;
}
}
inline unsigned char* copy_string(unsigned char* buffer_it, field_view& f)
{
auto str = f.get_string();
if (!str.empty())
{
std::memcpy(buffer_it, str.data(), str.size());
f = field_view(string_view(reinterpret_cast<const char*>(buffer_it), str.size()));
buffer_it += str.size();
}
return buffer_it;
}
inline unsigned char* copy_blob(unsigned char* buffer_it, field_view& f)
{
auto b = f.get_blob();
if (!b.empty())
{
std::memcpy(buffer_it, b.data(), b.size());
f = field_view(blob_view(buffer_it, b.size()));
buffer_it += b.size();
}
return buffer_it;
}
inline std::size_t copy_string_as_offset(unsigned char* buffer_first, std::size_t offset, field_view& f)
{
auto str = f.get_string();
if (!str.empty())
{
std::memcpy(buffer_first + offset, str.data(), str.size());
f = detail::access::construct<field_view>(detail::string_view_offset{offset, str.size()}, false);
return str.size();
}
return 0;
}
inline std::size_t copy_blob_as_offset(unsigned char* buffer_first, std::size_t offset, field_view& f)
{
auto str = f.get_blob();
if (!str.empty())
{
std::memcpy(buffer_first + offset, str.data(), str.size());
f = detail::access::construct<field_view>(detail::string_view_offset{offset, str.size()}, true);
return str.size();
}
return 0;
}
inline void copy_strings(std::vector<field_view>& fields, std::vector<unsigned char>& string_buffer)
{
// Calculate the required size for the new strings
std::size_t size = 0;
for (auto f : fields)
{
size += get_string_size(f);
}
// Make space. The previous fields should be in offset form
string_buffer.resize(string_buffer.size() + size);
// Copy strings and blobs
unsigned char* buffer_it = string_buffer.data();
for (auto& f : fields)
{
switch (f.kind())
{
case field_kind::string: buffer_it = copy_string(buffer_it, f); break;
case field_kind::blob: buffer_it = copy_blob(buffer_it, f); break;
default: break;
}
}
BOOST_ASSERT(buffer_it == string_buffer.data() + size);
}
inline field_view offset_to_string_view(field_view fv, const std::uint8_t* buffer_first)
{
auto& impl = detail::access::get_impl(fv);
if (impl.is_string_offset())
{
return field_view(string_view(
reinterpret_cast<const char*>(buffer_first) + impl.repr.sv_offset_.offset,
impl.repr.sv_offset_.size
));
}
else if (impl.is_blob_offset())
{
return field_view(blob_view(buffer_first + impl.repr.sv_offset_.offset, impl.repr.sv_offset_.size));
}
else
{
return fv;
}
}
} // namespace detail
} // namespace mysql
} // namespace boost
boost::mysql::detail::row_impl::row_impl(const field_view* fields, std::size_t size)
: fields_(fields, fields + size)
{
copy_strings(fields_, string_buffer_);
}
boost::mysql::detail::row_impl::row_impl(const row_impl& rhs) : fields_(rhs.fields_)
{
copy_strings(fields_, string_buffer_);
}
boost::mysql::detail::row_impl& boost::mysql::detail::row_impl::operator=(const row_impl& rhs)
{
assign(rhs.fields_.data(), rhs.fields_.size());
return *this;
}
void boost::mysql::detail::row_impl::assign(const field_view* fields, std::size_t size)
{
// Protect against self-assignment. This is valid as long as we
// don't implement sub-range operators (e.g. row_view[2:4])
if (fields_.data() == fields)
{
BOOST_ASSERT(fields_.size() == size);
}
else
{
fields_.assign(fields, fields + size);
string_buffer_.clear();
copy_strings(fields_, string_buffer_);
}
}
void boost::mysql::detail::row_impl::copy_strings_as_offsets(std::size_t first, std::size_t num_fields)
{
// Preconditions
BOOST_ASSERT(first <= fields_.size());
BOOST_ASSERT(first + num_fields <= fields_.size());
// Calculate the required size for the new strings
std::size_t size = 0;
for (std::size_t i = first; i < first + num_fields; ++i)
{
size += get_string_size(fields_[i]);
}
// Make space. The previous fields should be in offset form
std::size_t old_string_buffer_size = string_buffer_.size();
string_buffer_.resize(old_string_buffer_size + size);
// Copy strings and blobs
std::size_t offset = old_string_buffer_size;
for (std::size_t i = first; i < first + num_fields; ++i)
{
auto& f = fields_[i];
switch (f.kind())
{
case field_kind::string: offset += copy_string_as_offset(string_buffer_.data(), offset, f); break;
case field_kind::blob: offset += copy_blob_as_offset(string_buffer_.data(), offset, f); break;
default: break;
}
}
BOOST_ASSERT(offset == string_buffer_.size());
}
void boost::mysql::detail::row_impl::offsets_to_string_views()
{
for (auto& f : fields_)
f = offset_to_string_view(f, string_buffer_.data());
}
#endif