| // Protocol Buffers - Google's data interchange format |
| // Copyright 2023 Google LLC. 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 "protos/protos_extension_lock.h" |
| |
| #include <atomic> |
| #include <mutex> |
| #include <string> |
| #include <thread> |
| |
| #include <gmock/gmock.h> |
| #include <gtest/gtest.h> |
| #include "absl/hash/hash.h" |
| #include "absl/log/absl_check.h" |
| #include "protos/protos.h" |
| #include "protos_generator/tests/test_model.upb.proto.h" |
| #include "upb/mem/arena.hpp" |
| |
| #ifndef ASSERT_OK |
| #define ASSERT_OK(x) ASSERT_TRUE(x.ok()) |
| #endif // ASSERT_OK |
| #ifndef EXPECT_OK |
| #define EXPECT_OK(x) EXPECT_TRUE(x.ok()) |
| #endif // EXPECT_OK |
| |
| namespace protos_generator::test::protos { |
| |
| namespace { |
| |
| std::string GenerateTestData() { |
| TestModel model; |
| model.set_str1("str"); |
| ThemeExtension extension1; |
| extension1.set_ext_name("theme"); |
| ABSL_CHECK_OK(::protos::SetExtension(&model, theme, extension1)); |
| ThemeExtension extension2; |
| extension2.set_ext_name("theme_extension"); |
| ABSL_CHECK_OK(::protos::SetExtension(&model, ThemeExtension::theme_extension, |
| extension2)); |
| ::upb::Arena arena; |
| auto bytes = ::protos::Serialize(&model, arena); |
| ABSL_CHECK_OK(bytes); |
| return std::string(bytes->data(), bytes->size()); |
| } |
| |
| std::mutex m[8]; |
| void unlock_func(const void* msg) { m[absl::HashOf(msg) & 0x7].unlock(); } |
| ::protos::internal::UpbExtensionUnlocker lock_func(const void* msg) { |
| m[absl::HashOf(msg) & 0x7].lock(); |
| return &unlock_func; |
| } |
| |
| void TestConcurrentExtensionAccess(::protos::ExtensionRegistry registry) { |
| ::protos::internal::upb_extension_locker_global.store( |
| &lock_func, std::memory_order_release); |
| const std::string payload = GenerateTestData(); |
| TestModel parsed_model = |
| ::protos::Parse<TestModel>(payload, registry).value(); |
| const auto test_main = [&] { EXPECT_EQ("str", parsed_model.str1()); }; |
| const auto test_theme = [&] { |
| ASSERT_TRUE(::protos::HasExtension(&parsed_model, theme)); |
| auto ext = ::protos::GetExtension(&parsed_model, theme); |
| ASSERT_OK(ext); |
| EXPECT_EQ((*ext)->ext_name(), "theme"); |
| }; |
| const auto test_theme_extension = [&] { |
| auto ext = |
| ::protos::GetExtension(&parsed_model, ThemeExtension::theme_extension); |
| ASSERT_OK(ext); |
| EXPECT_EQ((*ext)->ext_name(), "theme_extension"); |
| }; |
| const auto test_serialize = [&] { |
| ::upb::Arena arena; |
| EXPECT_OK(::protos::Serialize(&parsed_model, arena)); |
| }; |
| const auto test_copy_constructor = [&] { |
| TestModel copy_a = parsed_model; |
| TestModel copy_b = parsed_model; |
| EXPECT_EQ(copy_a.has_str1(), copy_b.has_str1()); |
| }; |
| std::thread t1(test_main); |
| std::thread t2(test_main); |
| std::thread t3(test_theme); |
| std::thread t4(test_theme); |
| std::thread t5(test_theme_extension); |
| std::thread t6(test_theme_extension); |
| std::thread t7(test_serialize); |
| std::thread t8(test_copy_constructor); |
| t1.join(); |
| t2.join(); |
| t3.join(); |
| t4.join(); |
| t5.join(); |
| t6.join(); |
| t7.join(); |
| t8.join(); |
| test_main(); |
| test_theme(); |
| test_theme_extension(); |
| } |
| |
| TEST(CppGeneratedCode, ConcurrentAccessDoesNotRaceBothLazy) { |
| ::upb::Arena arena; |
| TestConcurrentExtensionAccess({{}, arena}); |
| } |
| |
| TEST(CppGeneratedCode, ConcurrentAccessDoesNotRaceOneLazyOneEager) { |
| ::upb::Arena arena; |
| TestConcurrentExtensionAccess({{&theme}, arena}); |
| TestConcurrentExtensionAccess({{&ThemeExtension::theme_extension}, arena}); |
| } |
| |
| TEST(CppGeneratedCode, ConcurrentAccessDoesNotRaceBothEager) { |
| ::upb::Arena arena; |
| TestConcurrentExtensionAccess( |
| {{&theme, &ThemeExtension::theme_extension}, arena}); |
| } |
| |
| } // namespace |
| } // namespace protos_generator::test::protos |