From d90849f11fa642a71f053a10660512d195a3ffca Mon Sep 17 00:00:00 2001 From: Matthias Schimek Date: Mon, 16 Sep 2024 12:10:14 +0200 Subject: [PATCH 1/6] In const owning data buffers: do not store the data as consts as this enforces copying the data once it is moved out --- include/kamping/data_buffer.hpp | 25 ++++-- tests/data_buffer_test.cpp | 2 + tests/named_parameters_test.cpp | 131 +++++++++++++++++++++++-------- tests/parameter_objects_test.cpp | 37 +++++++-- 4 files changed, 150 insertions(+), 45 deletions(-) diff --git a/include/kamping/data_buffer.hpp b/include/kamping/data_buffer.hpp index a81840434..af726adba 100644 --- a/include/kamping/data_buffer.hpp +++ b/include/kamping/data_buffer.hpp @@ -171,8 +171,8 @@ static constexpr bool is_vector_bool_v< template static constexpr bool is_vector_bool_v>>>::type> = - is_specialization>, std::vector>::value&& - std::is_same_v>::value_type, bool>; + is_specialization>, std::vector>::value + && std::is_same_v>::value_type, bool>; KAMPING_MAKE_HAS_MEMBER(resize) @@ -370,6 +370,8 @@ template < typename ValueType = default_value_type_tag> class DataBuffer : private CopyMoveEnabler>, private Extractable { public: + static_assert(!std::is_const_v, "Member Type should not be const qualified."); + static constexpr TParameterType parameter_type = parameter_type_param; ///< The type of parameter this buffer represents. @@ -411,6 +413,16 @@ class DataBuffer : private CopyMoveEnabler MemberTypeWithConst, MemberTypeWithConst&>; ///< The ContainerType as const or non-const (see ContainerTypeWithConst) and ///< reference or non-reference depending on ownership. + /// + using StorageType = std::conditional_t< + is_owning, + MemberType, + MemberTypeWithConstAndRef>; ///< The type as which the underlying container will be stored. If the buffer is + ///< owning, i.e. the underlying data is not referenced but stored directly, the + ///< potential constness of the data is not reflected in StorageType as this would + ///< enforce copying of the \c const data once it will be extracted. Modifying const + ///< data is instead prevented by giving only const qualified access via + ///< underlying() or data() in such case. using value_type = typename ValueTypeWrapper::value_type; ///< Value type of the buffer. @@ -507,7 +519,8 @@ class DataBuffer : private CopyMoveEnabler /// is not called if the buffer's resize policy is BufferResizePolicy::no_resize. template void resize_if_requested(SizeFunc&& compute_required_size) { - if constexpr (resize_policy == BufferResizePolicy::resize_to_fit || resize_policy == BufferResizePolicy::grow_only) { + if constexpr (resize_policy == BufferResizePolicy::resize_to_fit + || resize_policy == BufferResizePolicy::grow_only) { resize(compute_required_size()); } } @@ -586,21 +599,21 @@ class DataBuffer : private CopyMoveEnabler /// /// @return Moves the underlying container out of the DataBuffer. template = true> - MemberTypeWithConst extract() { + StorageType extract() { static_assert( ownership == BufferOwnership::owning, "Moving out of a reference should not be done because it would leave " "a users container in an unspecified state." ); kassert_not_extracted("Cannot extract a buffer that has already been extracted."); - auto extracted = std::move(underlying()); + auto extracted = std::move(_data); // we set is_extracted here because otherwise the call to underlying() would fail set_extracted(); return extracted; } private: - MemberTypeWithConstAndRef _data; ///< Container which holds the actual data. + StorageType _data; ///< Container which holds the actual data. }; /// @brief A more generic version of a DataBuffer which stores an object of type \tparam MemberType with its associcated diff --git a/tests/data_buffer_test.cpp b/tests/data_buffer_test.cpp index 4954aad3a..a5a5ff78a 100644 --- a/tests/data_buffer_test.cpp +++ b/tests/data_buffer_test.cpp @@ -1185,3 +1185,5 @@ TEST(DataBufferTest, referencing_buffers_are_copyable) { [[maybe_unused]] auto buffer2 = std::move(buffer_based_on_const_int_vector); } } + + using StorageType = std::conditional_t; diff --git a/tests/named_parameters_test.cpp b/tests/named_parameters_test.cpp index 8aa9e41fb..7de974687 100644 --- a/tests/named_parameters_test.cpp +++ b/tests/named_parameters_test.cpp @@ -14,6 +14,7 @@ #include #include +#include #include #include @@ -30,8 +31,8 @@ using namespace ::kamping::internal; namespace testing { template -void test_const_buffer( - GeneratedBuffer const& generated_buffer, +void test_const_referencing_buffer( + GeneratedBuffer& generated_buffer, kamping::internal::ParameterType expected_parameter_type, kamping::internal::BufferType expected_buffer_type, Span& expected_span @@ -40,9 +41,19 @@ void test_const_buffer( static_assert(std::is_same_v); EXPECT_FALSE(GeneratedBuffer::is_modifiable); + EXPECT_FALSE(GeneratedBuffer::is_owning); EXPECT_EQ(GeneratedBuffer::parameter_type, expected_parameter_type); EXPECT_EQ(GeneratedBuffer::buffer_type, expected_buffer_type); + static_assert( + std::is_const_v>, + "Member data() of the generated buffer does not point to const memory." + ); + static_assert( + std::is_const_v>, + "Member underlying() of the generated buffer provides access to non-const memory." + ); + auto span = generated_buffer.get(); static_assert(std::is_pointer_v, "Member ptr of internal::Span is not a pointer."); static_assert( @@ -59,8 +70,8 @@ void test_const_buffer( } template -void test_owning_buffer( - GeneratedBuffer const& generated_buffer, +void test_const_owning_buffer( + GeneratedBuffer& generated_buffer, kamping::internal::ParameterType expected_parameter_type, kamping::internal::BufferType expected_buffer_type, ExpectedValueContainer&& expected_value_container @@ -69,9 +80,19 @@ void test_owning_buffer( static_assert(std::is_same_v); EXPECT_FALSE(GeneratedBuffer::is_modifiable); + EXPECT_TRUE(GeneratedBuffer::is_owning); EXPECT_EQ(GeneratedBuffer::parameter_type, expected_parameter_type); EXPECT_EQ(GeneratedBuffer::buffer_type, expected_buffer_type); + static_assert( + std::is_const_v>, + "Member data() of the generated buffer does not point to const memory." + ); + static_assert( + std::is_const_v>, + "Member underlying() of the generated buffer provides access to non-const memory." + ); + auto span = generated_buffer.get(); static_assert(std::is_pointer_v, "Member ptr of internal::Span is not a pointer."); static_assert( @@ -205,7 +226,7 @@ TEST(ParameterFactoriesTest, send_buf_basics_int_vector) { auto gen_via_int_vec = send_buf(int_vec).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_int_vec, ParameterType::send_buf, internal::BufferType::in_buffer, @@ -218,7 +239,7 @@ TEST(ParameterFactoriesTest, send_buf_basics_const_int_vector) { auto gen_via_const_int_vec = send_buf(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::send_buf, internal::BufferType::in_buffer, @@ -231,7 +252,7 @@ TEST(ParameterFactoriesTest, send_buf_basics_moved_vector) { std::vector const expected = const_int_vec; auto gen_via_moved_vec = send_buf(std::move(const_int_vec)).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_moved_vec, ParameterType::send_buf, internal::BufferType::in_buffer, @@ -247,7 +268,7 @@ TEST(ParameterFactoriesTest, send_buf_basics_vector_from_function) { std::vector const expected = make_vector(); auto gen_via_vec_from_function = send_buf(make_vector()).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_vec_from_function, ParameterType::send_buf, internal::BufferType::in_buffer, @@ -259,7 +280,7 @@ TEST(ParameterFactoriesTest, send_buf_basics_vector_from_initializer_list) { std::vector expected = {1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_vec_from_function = send_buf({1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_vec_from_function, ParameterType::send_buf, internal::BufferType::in_buffer, @@ -384,12 +405,28 @@ TEST(ParameterFactoriesTest, send_buf_ignored) { EXPECT_EQ(ignored_send_buf.get().size(), 0); } +TEST(ParameterFactoriesTest, send_buf_owning_move_only_data) { + // test that data within the buffer is still treated as constant but can be returned without being copied + testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer const + expected_vec{1, 2, 3, 4}; // required as original data will be moved to buffer + auto send_buffer = send_buf(std::move(vec)).construct_buffer_or_rebind(); + testing::test_const_owning_buffer( + send_buffer, + ParameterType::send_buf, + internal::BufferType::in_buffer, + expected_vec + ); + auto extracted_vec = send_buffer.extract(); + EXPECT_THAT(extracted_vec, testing::ElementsAre(1, 2, 3, 4)); +} + TEST(ParameterFactoriesTest, send_counts_basics_int_vector) { std::vector int_vec{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_int_vec = send_counts(int_vec).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_int_vec, ParameterType::send_counts, internal::BufferType::in_buffer, @@ -402,7 +439,7 @@ TEST(ParameterFactoriesTest, send_counts_basics_const_int_vector) { auto gen_via_const_int_vec = send_counts(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::send_counts, internal::BufferType::in_buffer, @@ -416,7 +453,7 @@ TEST(ParameterFactoriesTest, send_counts_basics_moved_int_vector) { auto gen_via_int_vec = send_counts(std::move(int_vec)).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_int_vec, ParameterType::send_counts, internal::BufferType::in_buffer, @@ -428,7 +465,7 @@ TEST(ParameterFactoriesTest, send_counts_basics_initializer_list) { std::vector expected{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_int_initializer_list = send_counts({1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_int_initializer_list, ParameterType::send_counts, internal::BufferType::in_buffer, @@ -436,12 +473,28 @@ TEST(ParameterFactoriesTest, send_counts_basics_initializer_list) { ); } +TEST(ParameterFactoriesTest, send_counts_owning_move_only_data) { + // test that data within the buffer is still treated as constant but can be returned without being copied + testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer const + expected_vec{1, 2, 3, 4}; // required as original data will be moved to buffer + auto send_buffer = send_counts(std::move(vec)).construct_buffer_or_rebind(); + testing::test_const_owning_buffer( + send_buffer, + ParameterType::send_counts, + internal::BufferType::in_buffer, + expected_vec + ); + auto extracted_vec = send_buffer.extract(); + EXPECT_THAT(extracted_vec, testing::ElementsAre(1, 2, 3, 4)); +} + TEST(ParameterFactoriesTest, recv_counts_in_basics_int_vector) { std::vector int_vec{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_int_vec = recv_counts(int_vec).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_int_vec, ParameterType::recv_counts, internal::BufferType::in_buffer, @@ -454,7 +507,7 @@ TEST(ParameterFactoriesTest, recv_counts_in_basics_const_int_vector) { auto gen_via_const_int_vec = recv_counts(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::recv_counts, internal::BufferType::in_buffer, @@ -467,7 +520,7 @@ TEST(ParameterFactoriesTest, recv_counts_in_basics_moved_vector) { auto expected = int_vec; auto gen_via_moved_vec = recv_counts(std::move(int_vec)).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_moved_vec, ParameterType::recv_counts, internal::BufferType::in_buffer, @@ -479,7 +532,7 @@ TEST(ParameterFactoriesTest, recv_counts_in_basics_initializer_list) { std::vector expected{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_initializer_list = recv_counts({1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_initializer_list, ParameterType::recv_counts, internal::BufferType::in_buffer, @@ -492,7 +545,7 @@ TEST(ParameterFactoriesTest, send_displs_in_basics_int_vector) { auto gen_via_int_vec = send_displs(int_vec).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_int_vec, ParameterType::send_displs, internal::BufferType::in_buffer, @@ -505,7 +558,7 @@ TEST(ParameterFactoriesTest, send_displs_in_basics_const_int_vector) { auto gen_via_const_int_vec = send_displs(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::send_displs, internal::BufferType::in_buffer, @@ -518,7 +571,7 @@ TEST(ParameterFactoriesTest, send_displs_in_basics_moved_vector) { auto expected = int_vec; auto gen_via_moved_vec = send_displs(std::move(int_vec)).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_moved_vec, ParameterType::send_displs, internal::BufferType::in_buffer, @@ -530,7 +583,7 @@ TEST(ParameterFactoriesTest, send_displs_in_basics_initializer_list) { std::vector expected{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_initializer_list = send_displs({1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_initializer_list, ParameterType::send_displs, internal::BufferType::in_buffer, @@ -538,12 +591,28 @@ TEST(ParameterFactoriesTest, send_displs_in_basics_initializer_list) { ); } +TEST(ParameterFactoriesTest, send_displs_owning_move_only_data) { + // test that data within the buffer is still treated as constant but can be returned without being copied + testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer const + expected_vec{1, 2, 3, 4}; // required as original data will be moved to buffer + auto send_buffer = send_displs(std::move(vec)).construct_buffer_or_rebind(); + testing::test_const_owning_buffer( + send_buffer, + ParameterType::send_displs, + internal::BufferType::in_buffer, + expected_vec + ); + auto extracted_vec = send_buffer.extract(); + EXPECT_THAT(extracted_vec, testing::ElementsAre(1, 2, 3, 4)); +} + TEST(ParameterFactoriesTest, recv_displs_in_basics_int_vector) { std::vector int_vec{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_int_vec = recv_displs(int_vec).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_int_vec, ParameterType::recv_displs, internal::BufferType::in_buffer, @@ -556,7 +625,7 @@ TEST(ParameterFactoriesTest, recv_displs_in_basics_const_int_vector) { auto gen_via_const_int_vec = recv_displs(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::recv_displs, internal::BufferType::in_buffer, @@ -569,7 +638,7 @@ TEST(ParameterFactoriesTest, recv_displs_in_basics_moved_vector) { auto expected = int_vec; auto gen_via_moved_vec = recv_displs(std::move(int_vec)).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_moved_vec, ParameterType::recv_displs, internal::BufferType::in_buffer, @@ -581,7 +650,7 @@ TEST(ParameterFactoriesTest, recv_displs_in_basics_initializer_list) { std::vector expected{1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_initializer_list = recv_displs({1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_initializer_list, ParameterType::recv_displs, internal::BufferType::in_buffer, @@ -1218,7 +1287,7 @@ TEST(ParameterFactoriesTest, send_recv_buf_basics_const_int_vector) { auto gen_via_const_int_vec = send_recv_buf(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::send_recv_buf, internal::BufferType::in_out_buffer, @@ -1514,7 +1583,7 @@ TEST(ParameterFactoriesTest, values_on_rank_0_basics_int_vector) { auto gen_via_int_vec = values_on_rank_0(int_vec).construct_buffer_or_rebind(); Span expected_span{int_vec.data(), int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_int_vec, ParameterType::values_on_rank_0, BufferType::in_buffer, @@ -1527,7 +1596,7 @@ TEST(ParameterFactoriesTest, values_on_rank_0_basics_const_int_vector) { auto gen_via_const_int_vec = values_on_rank_0(const_int_vec).construct_buffer_or_rebind(); Span expected_span{const_int_vec.data(), const_int_vec.size()}; using ExpectedValueType = int; - testing::test_const_buffer( + testing::test_const_referencing_buffer( gen_via_const_int_vec, ParameterType::values_on_rank_0, BufferType::in_buffer, @@ -1540,7 +1609,7 @@ TEST(ParameterFactoriesTest, values_on_rank_0_basics_moved_vector) { std::vector const expected = const_int_vec; auto gen_via_moved_vec = values_on_rank_0(std::move(const_int_vec)).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_moved_vec, ParameterType::values_on_rank_0, BufferType::in_buffer, @@ -1556,7 +1625,7 @@ TEST(ParameterFactoriesTest, values_on_rank_0_basics_vector_from_function) { std::vector const expected = make_vector(); auto gen_via_vec_from_function = values_on_rank_0(make_vector()).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_vec_from_function, ParameterType::values_on_rank_0, BufferType::in_buffer, @@ -1568,7 +1637,7 @@ TEST(ParameterFactoriesTest, values_on_rank_0_basics_vector_from_initializer_lis std::vector expected = {1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}; auto gen_via_vec_from_function = values_on_rank_0({1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1}).construct_buffer_or_rebind(); using ExpectedValueType = int; - testing::test_owning_buffer( + testing::test_const_owning_buffer( gen_via_vec_from_function, ParameterType::values_on_rank_0, BufferType::in_buffer, diff --git a/tests/parameter_objects_test.cpp b/tests/parameter_objects_test.cpp index f87976a7e..75a15f33a 100644 --- a/tests/parameter_objects_test.cpp +++ b/tests/parameter_objects_test.cpp @@ -59,10 +59,10 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_noncopyable_type) { { // by reference testing::NonCopyableOwnContainer container{1, 2, 3, 4}; auto b = make_data_buffer_builder< - kamping::internal::ParameterType::recv_buf, - kamping::internal::BufferModifiability::modifiable, - kamping::internal::BufferType::out_buffer, - resize_to_fit>(container); + kamping::internal::ParameterType::recv_buf, + kamping::internal::BufferModifiability::modifiable, + kamping::internal::BufferType::out_buffer, + resize_to_fit>(container); container[0] = 42; EXPECT_EQ(b.size(), 4); auto buffer = b.construct_buffer_or_rebind(); @@ -72,10 +72,10 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_noncopyable_type) { { // by value testing::NonCopyableOwnContainer container{1, 2, 3, 4}; auto b = make_data_buffer_builder< - kamping::internal::ParameterType::recv_buf, - kamping::internal::BufferModifiability::modifiable, - kamping::internal::BufferType::out_buffer, - resize_to_fit>(std::move(container)); + kamping::internal::ParameterType::recv_buf, + kamping::internal::BufferModifiability::modifiable, + kamping::internal::BufferType::out_buffer, + resize_to_fit>(std::move(container)); container.resize(1); container[0] = 42; EXPECT_EQ(b.size(), 4); @@ -92,3 +92,24 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_noncopyable_type) { EXPECT_TRUE(is_non_copyable_own_container>); } } + +TEST(ParameterObjectsTest, DataBufferBuilder_with_const_owning_noncopyable_type) { + { + testing::NonCopyableOwnContainer container{1, 2, 3, 4}; + auto b = make_data_buffer_builder< + kamping::internal::ParameterType::recv_buf, + kamping::internal::BufferModifiability::constant, + kamping::internal::BufferType::out_buffer, + no_resize>(std::move(container)); + container.resize(1); + container[0] = 42; + EXPECT_EQ(b.size(), 4); + auto buffer = b.construct_buffer_or_rebind(); + constexpr bool is_data_ptr_constant = std::is_const_v>; + constexpr bool is_underlying_constant = std::is_const_v<>; + EXPECT_TRUE(is_data_ptr_constant); + EXPECT_TRUE(is_underlying_constant); + EXPECT_THAT(buffer.underlying(), ElementsAre(1, 2, 3, 4)); + EXPECT_THAT(buffer.extract(), ElementsAre(1, 2, 3, 4)); + } +} From 20acc40912cd1f02dc40d5bd4fc663939d3b11bc Mon Sep 17 00:00:00 2001 From: Matthias Schimek Date: Mon, 16 Sep 2024 12:10:48 +0200 Subject: [PATCH 2/6] fix formatting --- include/kamping/data_buffer.hpp | 7 +++---- tests/data_buffer_test.cpp | 2 +- tests/named_parameters_test.cpp | 27 ++++++++++++++++++--------- tests/parameter_objects_test.cpp | 26 +++++++++++++------------- 4 files changed, 35 insertions(+), 27 deletions(-) diff --git a/include/kamping/data_buffer.hpp b/include/kamping/data_buffer.hpp index af726adba..0fcc1740f 100644 --- a/include/kamping/data_buffer.hpp +++ b/include/kamping/data_buffer.hpp @@ -171,8 +171,8 @@ static constexpr bool is_vector_bool_v< template static constexpr bool is_vector_bool_v>>>::type> = - is_specialization>, std::vector>::value - && std::is_same_v>::value_type, bool>; + is_specialization>, std::vector>::value&& + std::is_same_v>::value_type, bool>; KAMPING_MAKE_HAS_MEMBER(resize) @@ -519,8 +519,7 @@ class DataBuffer : private CopyMoveEnabler /// is not called if the buffer's resize policy is BufferResizePolicy::no_resize. template void resize_if_requested(SizeFunc&& compute_required_size) { - if constexpr (resize_policy == BufferResizePolicy::resize_to_fit - || resize_policy == BufferResizePolicy::grow_only) { + if constexpr (resize_policy == BufferResizePolicy::resize_to_fit || resize_policy == BufferResizePolicy::grow_only) { resize(compute_required_size()); } } diff --git a/tests/data_buffer_test.cpp b/tests/data_buffer_test.cpp index a5a5ff78a..35f292bdb 100644 --- a/tests/data_buffer_test.cpp +++ b/tests/data_buffer_test.cpp @@ -1186,4 +1186,4 @@ TEST(DataBufferTest, referencing_buffers_are_copyable) { } } - using StorageType = std::conditional_t; +using StorageType = std::conditional_t; diff --git a/tests/named_parameters_test.cpp b/tests/named_parameters_test.cpp index 7de974687..2ab884e40 100644 --- a/tests/named_parameters_test.cpp +++ b/tests/named_parameters_test.cpp @@ -407,9 +407,12 @@ TEST(ParameterFactoriesTest, send_buf_ignored) { TEST(ParameterFactoriesTest, send_buf_owning_move_only_data) { // test that data within the buffer is still treated as constant but can be returned without being copied - testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer - testing::NonCopyableOwnContainer const - expected_vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer const expected_vec{ + 1, + 2, + 3, + 4}; // required as original data will be moved to buffer auto send_buffer = send_buf(std::move(vec)).construct_buffer_or_rebind(); testing::test_const_owning_buffer( send_buffer, @@ -475,9 +478,12 @@ TEST(ParameterFactoriesTest, send_counts_basics_initializer_list) { TEST(ParameterFactoriesTest, send_counts_owning_move_only_data) { // test that data within the buffer is still treated as constant but can be returned without being copied - testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer - testing::NonCopyableOwnContainer const - expected_vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer const expected_vec{ + 1, + 2, + 3, + 4}; // required as original data will be moved to buffer auto send_buffer = send_counts(std::move(vec)).construct_buffer_or_rebind(); testing::test_const_owning_buffer( send_buffer, @@ -593,9 +599,12 @@ TEST(ParameterFactoriesTest, send_displs_in_basics_initializer_list) { TEST(ParameterFactoriesTest, send_displs_owning_move_only_data) { // test that data within the buffer is still treated as constant but can be returned without being copied - testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer - testing::NonCopyableOwnContainer const - expected_vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer vec{1, 2, 3, 4}; // required as original data will be moved to buffer + testing::NonCopyableOwnContainer const expected_vec{ + 1, + 2, + 3, + 4}; // required as original data will be moved to buffer auto send_buffer = send_displs(std::move(vec)).construct_buffer_or_rebind(); testing::test_const_owning_buffer( send_buffer, diff --git a/tests/parameter_objects_test.cpp b/tests/parameter_objects_test.cpp index 75a15f33a..b7d4d9763 100644 --- a/tests/parameter_objects_test.cpp +++ b/tests/parameter_objects_test.cpp @@ -59,10 +59,10 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_noncopyable_type) { { // by reference testing::NonCopyableOwnContainer container{1, 2, 3, 4}; auto b = make_data_buffer_builder< - kamping::internal::ParameterType::recv_buf, - kamping::internal::BufferModifiability::modifiable, - kamping::internal::BufferType::out_buffer, - resize_to_fit>(container); + kamping::internal::ParameterType::recv_buf, + kamping::internal::BufferModifiability::modifiable, + kamping::internal::BufferType::out_buffer, + resize_to_fit>(container); container[0] = 42; EXPECT_EQ(b.size(), 4); auto buffer = b.construct_buffer_or_rebind(); @@ -72,10 +72,10 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_noncopyable_type) { { // by value testing::NonCopyableOwnContainer container{1, 2, 3, 4}; auto b = make_data_buffer_builder< - kamping::internal::ParameterType::recv_buf, - kamping::internal::BufferModifiability::modifiable, - kamping::internal::BufferType::out_buffer, - resize_to_fit>(std::move(container)); + kamping::internal::ParameterType::recv_buf, + kamping::internal::BufferModifiability::modifiable, + kamping::internal::BufferType::out_buffer, + resize_to_fit>(std::move(container)); container.resize(1); container[0] = 42; EXPECT_EQ(b.size(), 4); @@ -97,16 +97,16 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_const_owning_noncopyable_type) { testing::NonCopyableOwnContainer container{1, 2, 3, 4}; auto b = make_data_buffer_builder< - kamping::internal::ParameterType::recv_buf, - kamping::internal::BufferModifiability::constant, - kamping::internal::BufferType::out_buffer, - no_resize>(std::move(container)); + kamping::internal::ParameterType::recv_buf, + kamping::internal::BufferModifiability::constant, + kamping::internal::BufferType::out_buffer, + no_resize>(std::move(container)); container.resize(1); container[0] = 42; EXPECT_EQ(b.size(), 4); auto buffer = b.construct_buffer_or_rebind(); constexpr bool is_data_ptr_constant = std::is_const_v>; - constexpr bool is_underlying_constant = std::is_const_v<>; + constexpr bool is_underlying_constant = std::is_const_v << decltype(buffer.underlying()) >> ; EXPECT_TRUE(is_data_ptr_constant); EXPECT_TRUE(is_underlying_constant); EXPECT_THAT(buffer.underlying(), ElementsAre(1, 2, 3, 4)); From 7a5386463a2564cf1d671cdbb063e76133350721 Mon Sep 17 00:00:00 2001 From: Matthias Schimek Date: Mon, 16 Sep 2024 12:18:31 +0200 Subject: [PATCH 3/6] fix compile error --- tests/parameter_objects_test.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/parameter_objects_test.cpp b/tests/parameter_objects_test.cpp index b7d4d9763..f14ead52c 100644 --- a/tests/parameter_objects_test.cpp +++ b/tests/parameter_objects_test.cpp @@ -106,7 +106,7 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_const_owning_noncopyable_type) EXPECT_EQ(b.size(), 4); auto buffer = b.construct_buffer_or_rebind(); constexpr bool is_data_ptr_constant = std::is_const_v>; - constexpr bool is_underlying_constant = std::is_const_v << decltype(buffer.underlying()) >> ; + constexpr bool is_underlying_constant = std::is_const_v>; EXPECT_TRUE(is_data_ptr_constant); EXPECT_TRUE(is_underlying_constant); EXPECT_THAT(buffer.underlying(), ElementsAre(1, 2, 3, 4)); From 729a4d58513c3dbe1a1fd333f0cd6454f1102162 Mon Sep 17 00:00:00 2001 From: Matthias Schimek Date: Mon, 16 Sep 2024 12:27:01 +0200 Subject: [PATCH 4/6] remove accidentaly made edit --- tests/data_buffer_test.cpp | 2 -- 1 file changed, 2 deletions(-) diff --git a/tests/data_buffer_test.cpp b/tests/data_buffer_test.cpp index 35f292bdb..4954aad3a 100644 --- a/tests/data_buffer_test.cpp +++ b/tests/data_buffer_test.cpp @@ -1185,5 +1185,3 @@ TEST(DataBufferTest, referencing_buffers_are_copyable) { [[maybe_unused]] auto buffer2 = std::move(buffer_based_on_const_int_vector); } } - -using StorageType = std::conditional_t; From a5d5b3c51c4ff13a3218f47e193507b9517a9cbe Mon Sep 17 00:00:00 2001 From: Matthias Schimek Date: Mon, 16 Sep 2024 12:37:23 +0200 Subject: [PATCH 5/6] fix test --- tests/parameter_objects_test.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/parameter_objects_test.cpp b/tests/parameter_objects_test.cpp index f14ead52c..bf73f1d99 100644 --- a/tests/parameter_objects_test.cpp +++ b/tests/parameter_objects_test.cpp @@ -105,7 +105,7 @@ TEST(ParameterObjectsTest, DataBufferBuilder_with_const_owning_noncopyable_type) container[0] = 42; EXPECT_EQ(b.size(), 4); auto buffer = b.construct_buffer_or_rebind(); - constexpr bool is_data_ptr_constant = std::is_const_v>; + constexpr bool is_data_ptr_constant = std::is_const_v>; constexpr bool is_underlying_constant = std::is_const_v>; EXPECT_TRUE(is_data_ptr_constant); EXPECT_TRUE(is_underlying_constant); From abd62de137847ad8a5b602899950940f093b2757 Mon Sep 17 00:00:00 2001 From: Matthias Schimek Date: Mon, 30 Sep 2024 08:09:05 +0200 Subject: [PATCH 6/6] add check for vector --- include/kamping/data_buffer.hpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/include/kamping/data_buffer.hpp b/include/kamping/data_buffer.hpp index 0fcc1740f..1f83fa4a1 100644 --- a/include/kamping/data_buffer.hpp +++ b/include/kamping/data_buffer.hpp @@ -604,6 +604,10 @@ class DataBuffer : private CopyMoveEnabler "Moving out of a reference should not be done because it would leave " "a users container in an unspecified state." ); + static_assert( + !is_vector_bool_v, + "Buffers based on std::vector are not supported, use std::vector instead." + ); kassert_not_extracted("Cannot extract a buffer that has already been extracted."); auto extracted = std::move(_data); // we set is_extracted here because otherwise the call to underlying() would fail