123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- //
- // 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
|