diff --git a/cmake/abseil-cpp.cmake b/cmake/abseil-cpp.cmake index 523517cdc4..ff32501bbe 100644 --- a/cmake/abseil-cpp.cmake +++ b/cmake/abseil-cpp.cmake @@ -64,6 +64,7 @@ else() absl::strings absl::synchronization absl::time + absl::type_traits absl::utility absl::variant ) diff --git a/src/google/protobuf/BUILD.bazel b/src/google/protobuf/BUILD.bazel index c60caa0f5a..e2b97f3147 100644 --- a/src/google/protobuf/BUILD.bazel +++ b/src/google/protobuf/BUILD.bazel @@ -178,6 +178,9 @@ cc_library( "//:__subpackages__", "//src/google/protobuf:__subpackages__", ], + deps = [ + "@com_google_absl//absl/meta:type_traits", + ], ) cc_library( @@ -334,6 +337,7 @@ cc_library( "//src/google/protobuf/stubs:lite", "@com_google_absl//absl/container:btree", "@com_google_absl//absl/container:flat_hash_set", + "@com_google_absl//absl/meta:type_traits", "@com_google_absl//absl/numeric:bits", "@com_google_absl//absl/strings:internal", "@com_google_absl//absl/synchronization", diff --git a/src/google/protobuf/arena.h b/src/google/protobuf/arena.h index 748b23e36b..757d02c0cf 100644 --- a/src/google/protobuf/arena.h +++ b/src/google/protobuf/arena.h @@ -47,7 +47,7 @@ using type_info = ::type_info; #include #endif -#include +#include "absl/meta/type_traits.h" #include "google/protobuf/arena_align.h" #include "google/protobuf/arena_config.h" #include "google/protobuf/port.h" diff --git a/src/google/protobuf/map.h b/src/google/protobuf/map.h index a1577b53eb..cb3ec472fd 100644 --- a/src/google/protobuf/map.h +++ b/src/google/protobuf/map.h @@ -55,8 +55,9 @@ #endif #include "google/protobuf/stubs/common.h" -#include "google/protobuf/arena.h" #include "absl/container/btree_map.h" +#include "absl/meta/type_traits.h" +#include "google/protobuf/arena.h" #include "google/protobuf/generated_enum_util.h" #include "google/protobuf/map_type_handler.h" #include "google/protobuf/port.h" @@ -98,6 +99,14 @@ class DynamicMapField; class GeneratedMessageReflection; +// Internal type traits that can be used to define custom key/value types. These +// are only be specialized by protobuf internals, and never by users. +template +struct is_internal_map_key_type : std::false_type {}; + +template +struct is_internal_map_value_type : std::false_type {}; + // re-implement std::allocator to use arena allocator for memory allocation. // Used for Map implementation. Users should not use this class // directly. @@ -987,8 +996,8 @@ class Map { using size_type = size_t; using hasher = typename internal::TransparentSupport::hash; - constexpr Map() : elements_(nullptr) {} - explicit Map(Arena* arena) : elements_(arena) {} + constexpr Map() : elements_(nullptr) { StaticValidityCheck(); } + explicit Map(Arena* arena) : elements_(arena) { StaticValidityCheck(); } Map(const Map& other) : Map() { insert(other.begin(), other.end()); } @@ -1016,9 +1025,47 @@ class Map { insert(first, last); } - ~Map() {} + ~Map() { + // Fail-safe in case we miss calling this in a constructor. Note: this one + // won't trigger for leaked maps that never get destructed. + StaticValidityCheck(); + } private: +#ifdef PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS + static_assert(!std::is_const::value && + !std::is_const::value, + "We do not support const types."); + static_assert(!std::is_volatile::value && + !std::is_volatile::value, + "We do not support volatile types."); + static_assert(!std::is_pointer::value && + !std::is_pointer::value, + "We do not support pointer types."); + static_assert(!std::is_reference::value && + !std::is_reference::value, + "We do not support reference types."); +#endif // PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS + static constexpr PROTOBUF_ALWAYS_INLINE void StaticValidityCheck() { +#ifdef PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS + static_assert(alignof(internal::NodeBase) >= alignof(mapped_type), + "Alignment of mapped type is too high."); + static_assert( + absl::disjunction, + internal::is_supported_string_type, + internal::is_internal_map_key_type>::value, + "We only support integer, string, or designated internal key " + "types."); + static_assert(absl::disjunction< + internal::is_supported_scalar_type, + is_proto_enum, + internal::is_supported_message_type, + internal::is_internal_map_value_type>::value, + "We only support scalar, Message, and designated internal " + "mapped types."); +#endif // PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS + } + template struct SameAsElementReference : std::is_same #include +#include #include "google/protobuf/arena.h" #include "google/protobuf/port.h" @@ -274,6 +275,11 @@ class PROTOBUF_EXPORT MapKey { FieldDescriptor::CppType type_; }; +namespace internal { +template <> +struct is_internal_map_key_type : std::true_type {}; +} // namespace internal + } // namespace protobuf } // namespace google namespace std { @@ -937,6 +943,13 @@ class PROTOBUF_EXPORT MapIterator { MapValueRef value_; }; +namespace internal { +template <> +struct is_internal_map_value_type : std::true_type {}; +template <> +struct is_internal_map_value_type : std::true_type {}; +} // namespace internal + } // namespace protobuf } // namespace google diff --git a/src/google/protobuf/map_test.cc b/src/google/protobuf/map_test.cc index 5f78a3e42b..205e57ee92 100644 --- a/src/google/protobuf/map_test.cc +++ b/src/google/protobuf/map_test.cc @@ -28,6 +28,8 @@ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +#include + #include "absl/container/flat_hash_set.h" #include "google/protobuf/map_proto2_unittest.pb.h" #include "google/protobuf/map_unittest.pb.h" @@ -54,8 +56,6 @@ namespace google { namespace protobuf { namespace internal { -namespace { - struct AlignedAsDefault { int x; @@ -64,6 +64,14 @@ struct alignas(8) AlignedAs8 { int x; }; +template <> +struct is_internal_map_value_type : std::true_type {}; +template <> +struct is_internal_map_value_type : std::true_type {}; + +namespace { + + template void MapTest_Aligned() { Arena arena; diff --git a/src/google/protobuf/map_test.inc b/src/google/protobuf/map_test.inc index c1afd81045..4ff1845227 100644 --- a/src/google/protobuf/map_test.inc +++ b/src/google/protobuf/map_test.inc @@ -90,6 +90,91 @@ void MapTestForceDeterministic() { io::CodedOutputStream::SetDefaultSerializationDeterministic(); } +struct MoveTestKey { + MoveTestKey(int data, int* copies) : data(data), copies(copies) {} + + MoveTestKey(const MoveTestKey& other) + : data(other.data), copies(other.copies) { + ++*copies; + } + + MoveTestKey(MoveTestKey&& other) noexcept + : data(other.data), copies(other.copies) {} + + friend bool operator==(const MoveTestKey& lhs, const MoveTestKey& rhs) { + return lhs.data == rhs.data; + } + friend bool operator<(const MoveTestKey& lhs, const MoveTestKey& rhs) { + return lhs.data < rhs.data; + } + + int data; + int* copies; +}; + +enum class ConstructorType { + kDefault, + kCopy, + kMove, +}; + +struct ConstructorTag { + ConstructorTag() : invoked_constructor(ConstructorType::kDefault) {} + ConstructorTag(const ConstructorTag&) + : invoked_constructor(ConstructorType::kCopy) {} + ConstructorTag(ConstructorTag&&) + : invoked_constructor(ConstructorType::kMove) {} + + ConstructorType invoked_constructor; +}; + +struct CountedInstance { + CountedInstance() { ++num_created; } + CountedInstance(const CountedInstance&) : CountedInstance() {} + CountedInstance(CountedInstance&&) : CountedInstance() {} + + CountedInstance& operator=(const CountedInstance&) { + ++num_assigned; + return *this; + } + + explicit CountedInstance(int x) : CountedInstance() {} + + static int num_created; + static int num_assigned; +}; + +int CountedInstance::num_created = 0; +int CountedInstance::num_assigned = 0; + +struct ArenaConstructible { + using InternalArenaConstructable_ = void; + using DestructorSkippable_ = void; + + ArenaConstructible() = default; + ArenaConstructible(const ArenaConstructible&) = default; + ArenaConstructible(Arena*) : ArenaConstructible() {} + + ArenaConstructible& operator=(const ArenaConstructible&) = default; + + explicit ArenaConstructible(int) : ArenaConstructible() {} + + Arena* arena() const { return nullptr; } + + CountedInstance unused; +}; + +template <> +struct is_internal_map_key_type : std::true_type {}; +template <> +struct is_internal_map_key_type : std::true_type {}; +template <> +struct is_internal_map_value_type : std::true_type {}; +template <> +struct is_internal_map_value_type : std::true_type {}; +template <> +struct is_internal_map_value_type : std::true_type {}; + namespace { using internal::DownCast; @@ -188,28 +273,6 @@ TEST_F(MapImplTest, OperatorBracket) { ExpectSingleElement(key, value2); } -struct MoveTestKey { - MoveTestKey(int data, int* copies) : data(data), copies(copies) {} - - MoveTestKey(const MoveTestKey& other) - : data(other.data), copies(other.copies) { - ++*copies; - } - - MoveTestKey(MoveTestKey&& other) noexcept - : data(other.data), copies(other.copies) {} - - friend bool operator==(const MoveTestKey& lhs, const MoveTestKey& rhs) { - return lhs.data == rhs.data; - } - friend bool operator<(const MoveTestKey& lhs, const MoveTestKey& rhs) { - return lhs.data < rhs.data; - } - - int data; - int* copies; -}; - } // namespace } // namespace internal } // namespace protobuf @@ -768,29 +831,6 @@ TEST_F(MapImplTest, EmplaceKeyOnly) { #else -TEST_F(MapImplTest, ValueTypeNoImplicitConversion) { - using vt = typename Map::value_type; - - EXPECT_FALSE((std::is_convertible< - vt, std::pair>>::value)); -} - -enum class ConstructorType { - kDefault, - kCopy, - kMove, -}; - -struct ConstructorTag { - ConstructorTag() : invoked_constructor(ConstructorType::kDefault) {} - ConstructorTag(const ConstructorTag&) - : invoked_constructor(ConstructorType::kCopy) {} - ConstructorTag(ConstructorTag&&) - : invoked_constructor(ConstructorType::kMove) {} - - ConstructorType invoked_constructor; -}; - TEST_F(MapImplTest, ValueTypeHasMoveConstructor) { using vt = typename Map::value_type; ConstructorTag l, r; @@ -803,25 +843,6 @@ TEST_F(MapImplTest, ValueTypeHasMoveConstructor) { #endif // !PROTOBUF_FUTURE_MAP_PAIR_UPGRADE -struct CountedInstance { - CountedInstance() { ++num_created; } - CountedInstance(const CountedInstance&) : CountedInstance() {} - CountedInstance(CountedInstance&&) : CountedInstance() {} - - CountedInstance& operator=(const CountedInstance&) { - ++num_assigned; - return *this; - } - - explicit CountedInstance(int x) : CountedInstance() {} - - static int num_created; - static int num_assigned; -}; - -int CountedInstance::num_created = 0; -int CountedInstance::num_assigned = 0; - TEST_F(MapImplTest, TryEmplaceExisting) { Map m; @@ -836,23 +857,6 @@ TEST_F(MapImplTest, TryEmplaceExisting) { EXPECT_EQ(CountedInstance::num_assigned, 0); } -struct ArenaConstructible { - using InternalArenaConstructable_ = void; - using DestructorSkippable_ = void; - - ArenaConstructible() = default; - ArenaConstructible(const ArenaConstructible&) = default; - ArenaConstructible(Arena*) : ArenaConstructible() {} - - ArenaConstructible& operator=(const ArenaConstructible&) = default; - - explicit ArenaConstructible(int) : ArenaConstructible() {} - - Arena* arena() const { return nullptr; } - - CountedInstance unused; -}; - TEST_F(MapImplTest, TryEmplaceArenaConstructible) { ASSERT_TRUE(Arena::is_arena_constructable::value); diff --git a/src/google/protobuf/port.h b/src/google/protobuf/port.h index 195f7b554e..2fbca5dc68 100644 --- a/src/google/protobuf/port.h +++ b/src/google/protobuf/port.h @@ -39,15 +39,21 @@ #include #include #include +#include #include +#include "absl/meta/type_traits.h" + // must be last #include "google/protobuf/port_def.inc" namespace google { namespace protobuf { + +class MessageLite; + namespace internal { @@ -110,6 +116,33 @@ inline ToRef DownCast(From& f) { return *static_cast(&f); } +// Helpers for identifying our supported types. +template +struct is_supported_integral_type + : absl::disjunction, std::is_same, + std::is_same, std::is_same, + std::is_same> {}; + +template +struct is_supported_floating_point_type + : absl::disjunction, std::is_same> {}; + +template +struct is_supported_string_type + : absl::disjunction> {}; + +template +struct is_supported_scalar_type + : absl::disjunction, + is_supported_floating_point_type, + is_supported_string_type> {}; + +template +struct is_supported_message_type + : absl::disjunction> { + static constexpr auto force_complete_type = sizeof(T); +}; + // To prevent sharing cache lines between threads #ifdef __cpp_aligned_new enum { kCacheAlignment = 64 }; diff --git a/src/google/protobuf/port_def.inc b/src/google/protobuf/port_def.inc index 63a6a9848b..2a43812a5a 100644 --- a/src/google/protobuf/port_def.inc +++ b/src/google/protobuf/port_def.inc @@ -204,6 +204,10 @@ static_assert(PROTOBUF_CPLUSPLUS_MIN(201402L), "Protobuf only supports C++14 and #ifdef PROTOBUF_FUTURE_BREAKING_CHANGES +// Include static asserts to lock down assumptions about our containers. +// Owner: mkruskal@ +#define PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS 1 + // Used to upgrade google::protobuf::MapPair to std::pair. // Owner: mordberg@ #define PROTOBUF_FUTURE_MAP_PAIR_UPGRADE 1 diff --git a/src/google/protobuf/repeated_field.h b/src/google/protobuf/repeated_field.h index d0f6ca448b..8e6295e885 100644 --- a/src/google/protobuf/repeated_field.h +++ b/src/google/protobuf/repeated_field.h @@ -148,6 +148,16 @@ class RepeatedField final { static_assert( alignof(Arena) >= alignof(Element), "We only support types that have an alignment smaller than Arena"); +#ifndef PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS + static_assert(!std::is_const::value, + "We do not support const value types."); + static_assert(!std::is_volatile::value, + "We do not support volatile value types."); + static_assert(!std::is_pointer::value, + "We do not support pointer value types."); + static_assert(!std::is_reference::value, + "We do not support reference value types."); +#endif // !PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS public: constexpr RepeatedField(); diff --git a/src/google/protobuf/repeated_ptr_field.h b/src/google/protobuf/repeated_ptr_field.h index 66efc68ddc..6d40e9f545 100644 --- a/src/google/protobuf/repeated_ptr_field.h +++ b/src/google/protobuf/repeated_ptr_field.h @@ -902,6 +902,16 @@ class StringTypeHandler { // Messages. template class RepeatedPtrField final : private internal::RepeatedPtrFieldBase { +#ifndef PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS + static_assert(!std::is_const::value, + "We do not support const value types."); + static_assert(!std::is_volatile::value, + "We do not support volatile value types."); + static_assert(!std::is_pointer::value, + "We do not support pointer value types."); + static_assert(!std::is_reference::value, + "We do not support reference value types."); +#endif // !PROTOBUF_FUTURE_CONTAINER_STATIC_ASSERTS public: constexpr RepeatedPtrField();