100.00% Lines (36/36) 100.00% Functions (2/2)
TLA Baseline Branch
Line Hits Code Line Hits Code
1   // 1   //
2   // Copyright (c) 2023 Vinnie Falco (vinnie.falco@gmail.com) 2   // Copyright (c) 2023 Vinnie Falco (vinnie.falco@gmail.com)
3   // 3   //
4   // Distributed under the Boost Software License, Version 1.0. (See accompanying 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) 5   // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6   // 6   //
7   // Official repository: https://github.com/cppalliance/capy 7   // Official repository: https://github.com/cppalliance/capy
8   // 8   //
9   9  
10   #ifndef BOOST_CAPY_BUFFERS_BUFFER_COPY_HPP 10   #ifndef BOOST_CAPY_BUFFERS_BUFFER_COPY_HPP
11   #define BOOST_CAPY_BUFFERS_BUFFER_COPY_HPP 11   #define BOOST_CAPY_BUFFERS_BUFFER_COPY_HPP
12   12  
13   #include <boost/capy/detail/config.hpp> 13   #include <boost/capy/detail/config.hpp>
14   #include <boost/capy/buffers.hpp> 14   #include <boost/capy/buffers.hpp>
15   #include <cstring> 15   #include <cstring>
16   #include <utility> 16   #include <utility>
17   17  
18   namespace boost { 18   namespace boost {
19   namespace capy { 19   namespace capy {
20   20  
21   /** Copy the contents of a buffer sequence into another buffer sequence. 21   /** Copy the contents of a buffer sequence into another buffer sequence.
22   22  
23 - This function copies bytes from the constant buffer sequence `src` into 23 + @functionobject
24 - the mutable buffer sequence `dest`, stopping when any limit is reached. 24 + */
  25 + constexpr struct
  26 + {
  27 + /** Copy the contents of a buffer sequence into another buffer sequence.
25   28  
26 - @par Constraints 29 + Copies bytes from the constant buffer sequence `src` into the
27 - @code 30 + mutable buffer sequence `dest`, stopping when any limit is
28 - MutableBufferSequence<decltype(dest)> && 31 + reached.
29 - ConstBufferSequence<decltype(src)>  
30 - @endcode  
31   32  
32 - @return The number of bytes copied, equal to 33 + @param dest The destination buffer sequence.
33 - `std::min(size(dest), size(src), at_most)`.  
34   34  
35 - @param dest The destination buffer sequence. 35 + @param src The source buffer sequence.
36 - @param src The source buffer sequence. 36 +
37 - @param at_most The maximum bytes to copy. Default copies all available. 37 + @param at_most The maximum bytes to copy. Default copies all
38 - */ 38 + available.
39 - constexpr struct buffer_copy_mrdocs_workaround_t 39 +
40 - { 40 + @return The number of bytes copied, equal to
  41 + `std::min(size(dest), size(src), at_most)`.
  42 + */
41   template< 43   template<
42   MutableBufferSequence MB, 44   MutableBufferSequence MB,
43   ConstBufferSequence CB> 45   ConstBufferSequence CB>
44   std::size_t 46   std::size_t
HITCBC 45   12256 operator()( 47   12256 operator()(
46   MB const& dest, 48   MB const& dest,
47   CB const& src, 49   CB const& src,
48   std::size_t at_most = std::size_t(-1)) const noexcept 50   std::size_t at_most = std::size_t(-1)) const noexcept
49   { 51   {
HITCBC 50   12256 std::size_t total = 0; 52   12256 std::size_t total = 0;
HITCBC 51   12256 std::size_t pos0 = 0; 53   12256 std::size_t pos0 = 0;
HITCBC 52   12256 std::size_t pos1 = 0; 54   12256 std::size_t pos1 = 0;
HITCBC 53   12256 auto const end0 = end(src); 55   12256 auto const end0 = end(src);
HITCBC 54   12256 auto const end1 = end(dest); 56   12256 auto const end1 = end(dest);
HITCBC 55   12256 auto it0 = begin(src); 57   12256 auto it0 = begin(src);
HITCBC 56   12256 auto it1 = begin(dest); 58   12256 auto it1 = begin(dest);
HITCBC 57   12256 while( 59   12256 while(
HITCBC 58   38002 total < at_most && 60   38002 total < at_most &&
HITCBC 59   54380 it0 != end0 && 61   54380 it0 != end0 &&
HITCBC 60   9402 it1 != end1) 62   9402 it1 != end1)
61   { 63   {
HITCBC 62   19805 const_buffer b0 = *it0; 64   19805 const_buffer b0 = *it0;
HITCBC 63   19805 mutable_buffer b1 = *it1; 65   19805 mutable_buffer b1 = *it1;
HITCBC 64   19805 b0 += pos0; 66   19805 b0 += pos0;
HITCBC 65   19805 b1 += pos1; 67   19805 b1 += pos1;
66   std::size_t const amount = 68   std::size_t const amount =
HITCBC 67   59415 [&] 69   59415 [&]
68   { 70   {
HITCBC 69   19805 std::size_t n = b0.size(); 71   19805 std::size_t n = b0.size();
HITCBC 70   19805 if( n > b1.size()) 72   19805 if( n > b1.size())
HITCBC 71   5750 n = b1.size(); 73   5750 n = b1.size();
HITCBC 72   19805 if( n > at_most - total) 74   19805 if( n > at_most - total)
HITCBC 73   4902 n = at_most - total; 75   4902 n = at_most - total;
HITCBC 74   19805 if(n != 0) 76   19805 if(n != 0)
HITCBC 75   18953 std::memcpy( 77   18953 std::memcpy(
76   b1.data(), 78   b1.data(),
77   b0.data(), 79   b0.data(),
78   n); 80   n);
HITCBC 79   19805 return n; 81   19805 return n;
HITCBC 80   19805 }(); 82   19805 }();
HITCBC 81   19805 total += amount; 83   19805 total += amount;
HITCBC 82   19805 if(amount == b1.size()) 84   19805 if(amount == b1.size())
83   { 85   {
HITCBC 84   7301 ++it1; 86   7301 ++it1;
HITCBC 85   7301 pos1 = 0; 87   7301 pos1 = 0;
86   } 88   }
87   else 89   else
88   { 90   {
HITCBC 89   12504 pos1 += amount; 91   12504 pos1 += amount;
90   } 92   }
HITCBC 91   19805 if(amount == b0.size()) 93   19805 if(amount == b0.size())
92   { 94   {
HITCBC 93   10569 ++it0; 95   10569 ++it0;
HITCBC 94   10569 pos0 = 0; 96   10569 pos0 = 0;
95   } 97   }
96   else 98   else
97   { 99   {
HITCBC 98   9236 pos0 += amount; 100   9236 pos0 += amount;
99   } 101   }
100   } 102   }
HITCBC 101   12256 return total; 103   12256 return total;
102   } 104   }
103   } buffer_copy {}; 105   } buffer_copy {};
104   106  
105   } // capy 107   } // capy
106   } // boost 108   } // boost
107   109  
108   #endif 110   #endif