LCOV - code coverage report
Current view: top level - capy - buffers.hpp (source / functions) Coverage Total Hit Missed
Test: coverage_remapped.info Lines: 96.9 % 65 63 2
Test Date: 2026-05-14 20:50:55 Functions: 100.0 % 145 145

           TLA  Line data    Source code
       1                 : //
       2                 : // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com)
       3                 : //
       4                 : // Distributed under the Boost Software License, Version 1.0. (See accompanying
       5                 : // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
       6                 : //
       7                 : // Official repository: https://github.com/cppalliance/capy
       8                 : //
       9                 : 
      10                 : #ifndef BOOST_CAPY_BUFFERS_HPP
      11                 : #define BOOST_CAPY_BUFFERS_HPP
      12                 : 
      13                 : #include <boost/capy/detail/config.hpp>
      14                 : #include <concepts>
      15                 : #include <cstddef>
      16                 : #include <iterator>
      17                 : #include <memory>
      18                 : #include <ranges>
      19                 : #include <type_traits>
      20                 : 
      21                 : // https://www.boost.org/doc/libs/1_65_0/doc/html/boost_asio/reference/ConstBufferSequence.html
      22                 : 
      23                 : namespace boost {
      24                 : 
      25                 : namespace asio {
      26                 : class const_buffer;
      27                 : class mutable_buffer;
      28                 : } // asio
      29                 : 
      30                 : namespace capy {
      31                 : 
      32                 : class const_buffer;
      33                 : class mutable_buffer;
      34                 : 
      35                 : /** A reference to a contiguous region of writable memory.
      36                 : 
      37                 :     Represents a pointer and size pair for a modifiable byte range.
      38                 :     Does not own the memory. Satisfies `MutableBufferSequence` (as a
      39                 :     single-element sequence) and is implicitly convertible to
      40                 :     `const_buffer`.
      41                 : 
      42                 :     @see const_buffer, MutableBufferSequence
      43                 : */
      44                 : class mutable_buffer
      45                 : {
      46                 :     unsigned char* p_ = nullptr;
      47                 :     std::size_t n_ = 0;
      48                 : 
      49                 : public:
      50                 :     /// Construct an empty buffer.
      51 HIT          29 :     mutable_buffer() = default;
      52                 : 
      53                 :     /// Construct a copy.
      54                 :     mutable_buffer(
      55                 :         mutable_buffer const&) = default;
      56                 : 
      57                 :     /// Assign by copying.
      58                 :     mutable_buffer& operator=(
      59                 :         mutable_buffer const&) = default;
      60                 : 
      61                 :     /// Construct from pointer and size.
      62           42543 :     constexpr mutable_buffer(
      63                 :         void* data, std::size_t size) noexcept
      64           42543 :         : p_(static_cast<unsigned char*>(data))
      65           42543 :         , n_(size)
      66                 :     {
      67           42543 :     }
      68                 : 
      69                 :     /// Return a pointer to the memory region.
      70           62282 :     constexpr void* data() const noexcept
      71                 :     {
      72           62282 :         return p_;
      73                 :     }
      74                 : 
      75                 :     /// Return the size in bytes.
      76           97921 :     constexpr std::size_t size() const noexcept
      77                 :     {
      78           97921 :         return n_;
      79                 :     }
      80                 : 
      81                 :     /** Advance the buffer start, shrinking the region.
      82                 : 
      83                 :         @param n Bytes to skip. Clamped to `size()`.
      84                 :     */
      85                 :     mutable_buffer&
      86           19795 :     operator+=(std::size_t n) noexcept
      87                 :     {
      88           19795 :         if( n > n_)
      89 MIS           0 :             n = n_;
      90 HIT       19795 :         p_ += n;
      91           19795 :         n_ -= n;
      92           19795 :         return *this;
      93                 :     }
      94                 : };
      95                 : 
      96                 : /** A reference to a contiguous region of read-only memory.
      97                 : 
      98                 :     Represents a pointer and size pair for a non-modifiable byte range.
      99                 :     Does not own the memory. Satisfies `ConstBufferSequence` (as a
     100                 :     single-element sequence). Implicitly constructible from
     101                 :     `mutable_buffer`.
     102                 : 
     103                 :     @see mutable_buffer, ConstBufferSequence
     104                 : */
     105                 : class const_buffer
     106                 : {
     107                 :     unsigned char const* p_ = nullptr;
     108                 :     std::size_t n_ = 0;
     109                 : 
     110                 : public:
     111                 :     /// Construct an empty buffer.
     112              57 :     const_buffer() = default;
     113                 : 
     114                 :     /// Construct a copy.
     115                 :     const_buffer(const_buffer const&) = default;
     116                 : 
     117                 :     /// Assign by copying.
     118                 :     const_buffer& operator=(
     119                 :         const_buffer const& other) = default;
     120                 : 
     121                 :     /// Construct from pointer and size.
     122           43395 :     constexpr const_buffer(
     123                 :         void const* data, std::size_t size) noexcept
     124           43395 :         : p_(static_cast<unsigned char const*>(data))
     125           43395 :         , n_(size)
     126                 :     {
     127           43395 :     }
     128                 : 
     129                 :     /// Construct from mutable_buffer.
     130           11963 :     constexpr const_buffer(
     131                 :         mutable_buffer const& b) noexcept
     132           11963 :         : p_(static_cast<unsigned char const*>(b.data()))
     133           11963 :         , n_(b.size())
     134                 :     {
     135           11963 :     }
     136                 : 
     137                 :     /// Return a pointer to the memory region.
     138           59932 :     constexpr void const* data() const noexcept
     139                 :     {
     140           59932 :         return p_;
     141                 :     }
     142                 : 
     143                 :     /// Return the size in bytes.
     144          112803 :     constexpr std::size_t size() const noexcept
     145                 :     {
     146          112803 :         return n_;
     147                 :     }
     148                 : 
     149                 :     /** Advance the buffer start, shrinking the region.
     150                 : 
     151                 :         @param n Bytes to skip. Clamped to `size()`.
     152                 :     */
     153                 :     const_buffer&
     154           19796 :     operator+=(std::size_t n) noexcept
     155                 :     {
     156           19796 :         if( n > n_)
     157 MIS           0 :             n = n_;
     158 HIT       19796 :         p_ += n;
     159           19796 :         n_ -= n;
     160           19796 :         return *this;
     161                 :     }
     162                 : };
     163                 : 
     164                 : /** Concept for sequences of read-only buffer regions.
     165                 : 
     166                 :     A type satisfies `ConstBufferSequence` if it represents one or more
     167                 :     contiguous memory regions that can be read. This includes single
     168                 :     buffers (convertible to `const_buffer`) and ranges of buffers.
     169                 : 
     170                 :     @par Syntactic Requirements
     171                 :     @li Convertible to `const_buffer`, OR
     172                 :     @li A bidirectional range with value type convertible to `const_buffer`
     173                 : 
     174                 :     @see const_buffer, MutableBufferSequence
     175                 : */
     176                 : template<typename T>
     177                 : concept ConstBufferSequence =
     178                 :     std::is_convertible_v<T, const_buffer> || (
     179                 :         std::ranges::bidirectional_range<T> &&
     180                 :         std::is_convertible_v<std::ranges::range_value_t<T>, const_buffer>);
     181                 : 
     182                 : /** Concept for sequences of writable buffer regions.
     183                 : 
     184                 :     A type satisfies `MutableBufferSequence` if it represents one or more
     185                 :     contiguous memory regions that can be written. This includes single
     186                 :     buffers (convertible to `mutable_buffer`) and ranges of buffers.
     187                 :     Every `MutableBufferSequence` also satisfies `ConstBufferSequence`.
     188                 : 
     189                 :     @par Syntactic Requirements
     190                 :     @li Convertible to `mutable_buffer`, OR
     191                 :     @li A bidirectional range with value type convertible to `mutable_buffer`
     192                 : 
     193                 :     @see mutable_buffer, ConstBufferSequence
     194                 : */
     195                 : template<typename T>
     196                 : concept MutableBufferSequence =
     197                 :     std::is_convertible_v<T, mutable_buffer> || (
     198                 :         std::ranges::bidirectional_range<T> &&
     199                 :         std::is_convertible_v<std::ranges::range_value_t<T>, mutable_buffer>);
     200                 : 
     201                 : /** Return an iterator to the first buffer in a sequence.
     202                 : 
     203                 :     Handles single buffers and ranges uniformly. For a single buffer,
     204                 :     returns a pointer to it (forming a one-element range).
     205                 : */
     206                 : constexpr struct begin_mrdocs_workaround_t
     207                 : {
     208                 :     template<std::convertible_to<const_buffer> ConvertibleToBuffer>
     209           13111 :     auto operator()(ConvertibleToBuffer const& b) const noexcept -> ConvertibleToBuffer const*
     210                 :     {
     211           13111 :         return std::addressof(b);
     212                 :     }
     213                 : 
     214                 :     template<ConstBufferSequence BS>
     215                 :         requires (!std::convertible_to<BS, const_buffer>)
     216           46366 :     auto operator()(BS const& bs) const noexcept
     217                 :     {
     218           46366 :         return std::ranges::begin(bs);
     219                 :     }
     220                 : 
     221                 :     template<ConstBufferSequence BS>
     222                 :         requires (!std::convertible_to<BS, const_buffer>)
     223            9192 :     auto operator()(BS& bs) const noexcept
     224                 :     {
     225            9192 :         return std::ranges::begin(bs);
     226                 :     }
     227                 : } begin {};
     228                 : 
     229                 : /** Return an iterator past the last buffer in a sequence.
     230                 : 
     231                 :     Handles single buffers and ranges uniformly. For a single buffer,
     232                 :     returns a pointer one past it.
     233                 : */
     234                 : constexpr struct end_mrdocs_workaround_t
     235                 : {
     236                 :     template<std::convertible_to<const_buffer> ConvertibleToBuffer>
     237           12871 :     auto operator()(ConvertibleToBuffer const& b) const noexcept -> ConvertibleToBuffer const*
     238                 :     {
     239           12871 :         return std::addressof(b) + 1;
     240                 :     }
     241                 : 
     242                 :     template<ConstBufferSequence BS>
     243                 :         requires (!std::convertible_to<BS, const_buffer>)
     244           46366 :     auto operator()(BS const& bs) const noexcept
     245                 :     {
     246           46366 :         return std::ranges::end(bs);
     247                 :     }
     248                 : 
     249                 :     template<ConstBufferSequence BS>
     250                 :         requires (!std::convertible_to<BS, const_buffer>)
     251            9192 :     auto operator()(BS& bs) const noexcept
     252                 :     {
     253            9192 :         return std::ranges::end(bs);
     254                 :     }
     255                 : } end {};
     256                 : 
     257                 : /** Return the total byte count across all buffers in a sequence.
     258                 : 
     259                 :     Sums the `size()` of each buffer in the sequence. This differs
     260                 :     from `buffer_length` which counts the number of buffer elements.
     261                 : 
     262                 :     @par Example
     263                 :     @code
     264                 :     std::array<mutable_buffer, 2> bufs = { ... };
     265                 :     std::size_t total = buffer_size( bufs );  // sum of both sizes
     266                 :     @endcode
     267                 : */
     268                 : constexpr struct buffer_size_mrdocs_workaround_t
     269                 : {
     270                 :     // GCC 13 falsely flags reads of arr_[i].n_ in detail::buffer_array
     271                 :     // when iterating here. The class uses union storage with placement
     272                 :     // new for slots 0..n_-1, so reads inside this bounded loop are
     273                 :     // well-defined, but the optimizer can't prove the loop bound and
     274                 :     // warns. The runtime cost of value-initializing all N slots is
     275                 :     // non-trivial for non-trivial value types, so we suppress instead.
     276                 : #if defined(__GNUC__) && !defined(__clang__)
     277                 : #pragma GCC diagnostic push
     278                 : #pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
     279                 : #endif
     280                 :     template<ConstBufferSequence CB>
     281           12812 :     constexpr std::size_t operator()(
     282                 :         CB const& bs) const noexcept
     283                 :     {
     284           12812 :         std::size_t n = 0;
     285           12812 :         auto const e = capy::end(bs);
     286           27149 :         for(auto it = capy::begin(bs); it != e; ++it)
     287           14337 :             n += const_buffer(*it).size();
     288           12812 :         return n;
     289                 :     }
     290                 : #if defined(__GNUC__) && !defined(__clang__)
     291                 : #pragma GCC diagnostic pop
     292                 : #endif
     293                 : } buffer_size {};
     294                 : 
     295                 : /** Check if a buffer sequence contains no data.
     296                 : 
     297                 :     @return `true` if all buffers have size zero or the sequence
     298                 :         is empty.
     299                 : */
     300                 : constexpr struct buffer_empty_mrdocs_workaround_t
     301                 : {
     302                 :     // See note on buffer_size above — same union-storage false positive.
     303                 : #if defined(__GNUC__) && !defined(__clang__)
     304                 : #pragma GCC diagnostic push
     305                 : #pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
     306                 : #endif
     307                 :     template<ConstBufferSequence CB>
     308            4243 :     constexpr bool operator()(
     309                 :         CB const& bs) const noexcept
     310                 :     {
     311            4243 :         auto it = begin(bs);
     312            4243 :         auto const end_ = end(bs);
     313            4284 :         while(it != end_)
     314                 :         {
     315            4244 :             const_buffer b(*it++);
     316            4244 :             if(b.size() != 0)
     317            4203 :                 return false;
     318                 :         }
     319              40 :         return true;
     320                 :     }
     321                 : #if defined(__GNUC__) && !defined(__clang__)
     322                 : #pragma GCC diagnostic pop
     323                 : #endif
     324                 : } buffer_empty {};
     325                 : 
     326                 : namespace detail {
     327                 : 
     328                 : template<class It>
     329                 : auto
     330             240 : length_impl(It first, It last, int)
     331                 :     -> decltype(static_cast<std::size_t>(last - first))
     332                 : {
     333             240 :     return static_cast<std::size_t>(last - first);
     334                 : }
     335                 : 
     336                 : template<class It>
     337                 : std::size_t
     338                 : length_impl(It first, It last, long)
     339                 : {
     340                 :     std::size_t n = 0;
     341                 :     while(first != last)
     342                 :     {
     343                 :         ++first;
     344                 :         ++n;
     345                 :     }
     346                 :     return n;
     347                 : }
     348                 : 
     349                 : } // detail
     350                 : 
     351                 : /** Return the number of buffer elements in a sequence.
     352                 : 
     353                 :     Counts the number of individual buffer objects, not bytes.
     354                 :     For a single buffer, returns 1. For a range, returns the
     355                 :     distance from `begin` to `end`.
     356                 : 
     357                 :     @see buffer_size
     358                 : */
     359                 : template<ConstBufferSequence CB>
     360                 : std::size_t
     361             240 : buffer_length(CB const& bs)
     362                 : {
     363             240 :     return detail::length_impl(
     364             240 :         begin(bs), end(bs), 0);
     365                 : }
     366                 : 
     367                 : /// Alias for `mutable_buffer` or `const_buffer` based on sequence type.
     368                 : template<typename BS>
     369                 : using buffer_type = std::conditional_t<
     370                 :     MutableBufferSequence<BS>,
     371                 :     mutable_buffer, const_buffer>;
     372                 : 
     373                 : } // capy
     374                 : } // boost
     375                 : 
     376                 : #endif
        

Generated by: LCOV version 2.3