| // Protocol Buffers - Google's data interchange format |
| // Copyright 2008 Google Inc. All rights reserved. |
| // |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file or at |
| // https://developers.google.com/open-source/licenses/bsd |
| |
| #include <array> |
| #include <string> |
| #include <type_traits> |
| #include <utility> |
| #include <vector> |
| |
| #include "absl/container/flat_hash_set.h" |
| #include "google/protobuf/arena_test_util.h" |
| #include "google/protobuf/internal_visibility_for_testing.h" |
| #include "google/protobuf/map_proto2_unittest.pb.h" |
| #include "google/protobuf/map_unittest.pb.h" |
| #include "google/protobuf/reflection_tester.h" |
| #include "google/protobuf/unittest.pb.h" |
| #include "google/protobuf/unittest_import.pb.h" |
| |
| |
| #define BRIDGE_UNITTEST ::google::protobuf::bridge_unittest |
| #define UNITTEST ::protobuf_unittest |
| #define UNITTEST_IMPORT ::protobuf_unittest_import |
| #define UNITTEST_PACKAGE_NAME "protobuf_unittest" |
| |
| // Must include after defining UNITTEST, etc. |
| // clang-format off |
| #include "google/protobuf/test_util.inc" |
| #include "google/protobuf/map_test_util.inc" |
| #include "google/protobuf/map_test.inc" |
| // clang-format on |
| |
| // Must be included last. |
| #include "google/protobuf/port_def.inc" |
| |
| namespace google { |
| namespace protobuf { |
| namespace internal { |
| |
| struct AlignedAsDefault { |
| int x; |
| }; |
| struct alignas(8) AlignedAs8 { |
| int x; |
| }; |
| |
| template <> |
| struct is_internal_map_value_type<AlignedAsDefault> : std::true_type {}; |
| template <> |
| struct is_internal_map_value_type<AlignedAs8> : std::true_type {}; |
| |
| namespace { |
| |
| using ::testing::FieldsAre; |
| using ::testing::UnorderedElementsAre; |
| |
| |
| TEST(MapTest, CopyConstructIntegers) { |
| auto token = internal::InternalVisibilityForTesting{}; |
| using MapType = Map<int32_t, int32_t>; |
| MapType original; |
| original[1] = 2; |
| original[2] = 3; |
| |
| MapType map1(original); |
| ASSERT_EQ(map1.size(), 2); |
| EXPECT_EQ(map1[1], 2); |
| EXPECT_EQ(map1[2], 3); |
| |
| MapType map2(token, nullptr, original); |
| ASSERT_EQ(map2.size(), 2); |
| EXPECT_EQ(map2[1], 2); |
| EXPECT_EQ(map2[2], 3); |
| } |
| |
| TEST(MapTest, CopyConstructStrings) { |
| auto token = internal::InternalVisibilityForTesting{}; |
| using MapType = Map<std::string, std::string>; |
| MapType original; |
| original["1"] = "2"; |
| original["2"] = "3"; |
| |
| MapType map1(original); |
| ASSERT_EQ(map1.size(), 2); |
| EXPECT_EQ(map1["1"], "2"); |
| EXPECT_EQ(map1["2"], "3"); |
| |
| MapType map2(token, nullptr, original); |
| ASSERT_EQ(map2.size(), 2); |
| EXPECT_EQ(map2["1"], "2"); |
| EXPECT_EQ(map2["2"], "3"); |
| } |
| |
| TEST(MapTest, CopyConstructMessages) { |
| auto token = internal::InternalVisibilityForTesting{}; |
| using MapType = Map<std::string, TestAllTypes>; |
| MapType original; |
| original["1"].set_optional_int32(1); |
| original["2"].set_optional_int32(2); |
| |
| MapType map1(original); |
| ASSERT_EQ(map1.size(), 2); |
| EXPECT_EQ(map1["1"].optional_int32(), 1); |
| EXPECT_EQ(map1["2"].optional_int32(), 2); |
| |
| MapType map2(token, nullptr, original); |
| ASSERT_EQ(map2.size(), 2); |
| EXPECT_EQ(map2["1"].optional_int32(), 1); |
| EXPECT_EQ(map2["2"].optional_int32(), 2); |
| } |
| |
| TEST(MapTest, CopyConstructIntegersWithArena) { |
| auto token = internal::InternalVisibilityForTesting{}; |
| using MapType = Map<int32_t, int32_t>; |
| MapType original; |
| original[1] = 2; |
| original[2] = 3; |
| |
| Arena arena; |
| alignas(MapType) char mem1[sizeof(MapType)]; |
| MapType& map1 = *new (mem1) MapType(token, &arena, original); |
| ASSERT_EQ(map1.size(), 2); |
| EXPECT_EQ(map1[1], 2); |
| EXPECT_EQ(map1[2], 3); |
| EXPECT_EQ(map1[2], 3); |
| } |
| |
| TEST(MapTest, CopyConstructStringsWithArena) { |
| auto token = internal::InternalVisibilityForTesting{}; |
| using MapType = Map<std::string, std::string>; |
| MapType original; |
| original["1"] = "2"; |
| original["2"] = "3"; |
| |
| Arena arena; |
| alignas(MapType) char mem1[sizeof(MapType)]; |
| MapType& map1 = *new (mem1) MapType(token, &arena, original); |
| ASSERT_EQ(map1.size(), 2); |
| EXPECT_EQ(map1["1"], "2"); |
| EXPECT_EQ(map1["2"], "3"); |
| } |
| |
| TEST(MapTest, CopyConstructMessagesWithArena) { |
| auto token = internal::InternalVisibilityForTesting{}; |
| using MapType = Map<std::string, TestAllTypes>; |
| MapType original; |
| original["1"].set_optional_int32(1); |
| original["2"].set_optional_int32(2); |
| |
| Arena arena; |
| alignas(MapType) char mem1[sizeof(MapType)]; |
| MapType& map1 = *new (mem1) MapType(token, &arena, original); |
| ASSERT_EQ(map1.size(), 2); |
| EXPECT_EQ(map1["1"].optional_int32(), 1); |
| EXPECT_EQ(map1["1"].GetArena(), &arena); |
| EXPECT_EQ(map1["2"].optional_int32(), 2); |
| EXPECT_EQ(map1["2"].GetArena(), &arena); |
| } |
| |
| // We changed the internal implementation to use a smaller size type, but the |
| // public API will still use size_t to avoid changing the API. Test that. |
| TEST(MapTest, SizeTypeIsSizeT) { |
| using M = Map<int, int>; |
| EXPECT_TRUE((std::is_same<M::size_type, size_t>::value)); |
| EXPECT_TRUE((std::is_same<decltype(M().size()), size_t>::value)); |
| size_t x = 0; |
| x = std::max(M().size(), x); |
| (void)x; |
| } |
| |
| template <typename Aligned, bool on_arena = false> |
| void MapTest_Aligned() { |
| Arena arena; |
| constexpr size_t align_mask = alignof(Aligned) - 1; |
| Map<int, Aligned> map(on_arena ? &arena : nullptr); |
| map.insert({1, Aligned{}}); |
| auto it = map.find(1); |
| ASSERT_NE(it, map.end()); |
| ASSERT_EQ(reinterpret_cast<intptr_t>(&it->second) & align_mask, 0); |
| map.clear(); |
| } |
| |
| TEST(MapTest, Aligned) { MapTest_Aligned<AlignedAsDefault>(); } |
| TEST(MapTest, AlignedOnArena) { MapTest_Aligned<AlignedAsDefault, true>(); } |
| TEST(MapTest, Aligned8) { MapTest_Aligned<AlignedAs8>(); } |
| TEST(MapTest, Aligned8OnArena) { MapTest_Aligned<AlignedAs8, true>(); } |
| |
| |
| |
| } // namespace |
| } // namespace internal |
| } // namespace protobuf |
| } // namespace google |
| |
| #include "google/protobuf/port_undef.inc" |