|  | // Protocol Buffers - Google's data interchange format | 
|  | // Copyright 2024 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 | 
|  |  | 
|  | #ifndef GOOGLE_PROTOBUF_HPB_EXTENSION_H__ | 
|  | #define GOOGLE_PROTOBUF_HPB_EXTENSION_H__ | 
|  |  | 
|  | #include <cstdint> | 
|  | #include <type_traits> | 
|  |  | 
|  | #include "absl/base/attributes.h" | 
|  | #include "hpb/arena.h" | 
|  | #include "hpb/backend/upb/extension.h" | 
|  | #include "hpb/backend/upb/interop.h" | 
|  | #include "hpb/internal/message_lock.h" | 
|  | #include "hpb/internal/template_help.h" | 
|  | #include "hpb/multibackend.h" | 
|  | #include "hpb/ptr.h" | 
|  | #include "upb/message/accessors.h" | 
|  | #include "upb/mini_table/extension_registry.h" | 
|  |  | 
|  | namespace hpb { | 
|  | // upb has a notion of an ExtensionRegistry. We expect most callers to use | 
|  | // the generated registry, which utilizes upb linker arrays. It is also possible | 
|  | // to call hpb funcs with hpb::ExtensionRegistry::empty_registry(). | 
|  | // | 
|  | // Since google::protobuf::cpp only has the generated registry, hpb funcs | 
|  | // that use an extension registry must be invoked with | 
|  | // hpb::ExtensionRegistry::generated_registry(). Note that | 
|  | // hpb::ExtensionRegistry::empty_registry() does not even exist | 
|  | // for the cpp backend. | 
|  | class ExtensionRegistry { | 
|  | public: | 
|  | #if HPB_INTERNAL_BACKEND == HPB_INTERNAL_BACKEND_UPB | 
|  | // The lifetimes of the ExtensionRegistry and the Arena are disparate, but | 
|  | // the Arena must outlive the ExtensionRegistry. | 
|  | explicit ExtensionRegistry(const hpb::Arena& arena) | 
|  | : registry_( | 
|  | upb_ExtensionRegistry_New(hpb::interop::upb::UnwrapArena(arena))) {} | 
|  |  | 
|  | template <typename ExtensionIdentifier> | 
|  | void AddExtension(const ExtensionIdentifier& id) { | 
|  | if (registry_) { | 
|  | auto* extension = id.mini_table_ext(); | 
|  | upb_ExtensionRegistryStatus status = | 
|  | upb_ExtensionRegistry_AddArray(registry_, &extension, 1); | 
|  | if (status != kUpb_ExtensionRegistryStatus_Ok) { | 
|  | registry_ = nullptr; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static const ExtensionRegistry& empty_registry() { | 
|  | static const ExtensionRegistry* r = new ExtensionRegistry(); | 
|  | return *r; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | static const ExtensionRegistry& generated_registry() { | 
|  | static const ExtensionRegistry* r = NewGeneratedRegistry(); | 
|  | return *r; | 
|  | } | 
|  |  | 
|  | private: | 
|  | #if HPB_INTERNAL_BACKEND == HPB_INTERNAL_BACKEND_UPB | 
|  | friend upb_ExtensionRegistry* ::hpb::internal::GetUpbExtensions( | 
|  | const ExtensionRegistry& extension_registry); | 
|  | upb_ExtensionRegistry* registry_; | 
|  | #endif | 
|  | // TODO: b/379100963 - Introduce ShutdownHpbLibrary | 
|  | static const ExtensionRegistry* NewGeneratedRegistry() { | 
|  | #if HPB_INTERNAL_BACKEND == HPB_INTERNAL_BACKEND_UPB | 
|  | static hpb::Arena* global_arena = new hpb::Arena(); | 
|  | ExtensionRegistry* registry = new ExtensionRegistry(*global_arena); | 
|  | upb_ExtensionRegistry_AddAllLinkedExtensions(registry->registry_); | 
|  | return registry; | 
|  | #elif HPB_INTERNAL_BACKEND == HPB_INTERNAL_BACKEND_CPP | 
|  | ExtensionRegistry* registry = new ExtensionRegistry(); | 
|  | return registry; | 
|  | #else | 
|  | #error "Unsupported hpb backend" | 
|  | #endif | 
|  | } | 
|  | explicit ExtensionRegistry() = default; | 
|  | }; | 
|  |  | 
|  | template <typename T, typename Extendee, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>> | 
|  | ABSL_MUST_USE_RESULT bool HasExtension( | 
|  | Ptr<T> message, | 
|  | const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) { | 
|  | return ::hpb::internal::HasExtensionOrUnknown( | 
|  | hpb::interop::upb::GetMessage(message), id.mini_table_ext()); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extendee, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>> | 
|  | ABSL_MUST_USE_RESULT bool HasExtension( | 
|  | const T* message, | 
|  | const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) { | 
|  | return HasExtension(Ptr(message), id); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>, | 
|  | typename = hpb::internal::EnableIfMutableProto<T>> | 
|  | void ClearExtension( | 
|  | Ptr<T> message, | 
|  | const ::hpb::internal::ExtensionIdentifier<T, Extension>& id) { | 
|  | static_assert(!std::is_const_v<T>, ""); | 
|  | upb_Message_ClearExtension(hpb::interop::upb::GetMessage(message), | 
|  | id.mini_table_ext()); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>> | 
|  | void ClearExtension( | 
|  | T* message, const ::hpb::internal::ExtensionIdentifier<T, Extension>& id) { | 
|  | ClearExtension(Ptr(message), id); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Sets the extension to provided value. | 
|  | * | 
|  | * `message` is the model and may be passed in as a `T*` or a `Ptr<T>`. | 
|  | * | 
|  | * `id` is the ExtensionIdentifier provided by hpb gencode. | 
|  | * | 
|  | * `value` is the value to set the extension to. | 
|  | *  For message extension it can bind to `const Input&`, `Input&&`, | 
|  | *  or `Ptr<const Input>`. | 
|  | *  For rvalue references, if the arenas match, the extension is moved. | 
|  | *  If the arenas differ, a deep copy is performed. | 
|  | */ | 
|  | template <int&... DeductionBarrier, typename T, typename Extension, | 
|  | typename Input> | 
|  | auto SetExtension( | 
|  | hpb::internal::PtrOrRawMutable<T> message, | 
|  | const internal::ExtensionIdentifier<internal::RemovePtrT<T>, Extension>& id, | 
|  | Input&& value) | 
|  | -> decltype(internal::UpbExtensionTrait<Extension>::Set( | 
|  | message, id, std::forward<Input>(value))) { | 
|  | return internal::UpbExtensionTrait<Extension>::Set( | 
|  | message, id, std::forward<Input>(value)); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>, | 
|  | typename = hpb::internal::EnableIfMutableProto<T>> | 
|  | void SetAliasExtension( | 
|  | Ptr<T> message, | 
|  | const ::hpb::internal::ExtensionIdentifier<T, Extension>& id, | 
|  | Ptr<Extension> value) { | 
|  | static_assert(!std::is_const_v<T>); | 
|  | auto* message_arena = hpb::interop::upb::GetArena(message); | 
|  | auto* extension_arena = hpb::interop::upb::GetArena(value); | 
|  | return ::hpb::internal::SetAliasExtension( | 
|  | hpb::interop::upb::GetMessage(message), message_arena, | 
|  | id.mini_table_ext(), hpb::interop::upb::GetMessage(value), | 
|  | extension_arena); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extendee, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>> | 
|  | absl::StatusOr<typename internal::UpbExtensionTrait<Extension>::ReturnType> | 
|  | GetExtension( | 
|  | Ptr<T> message, | 
|  | const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) { | 
|  | return hpb::internal::UpbExtensionTrait<Extension>::Get(message, id); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extendee, typename Extension, | 
|  | typename = hpb::internal::EnableIfHpbClassThatHasExtensions<T>> | 
|  | absl::StatusOr<typename internal::UpbExtensionTrait<Extension>::ReturnType> | 
|  | GetExtension( | 
|  | const T* message, | 
|  | const hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) { | 
|  | return GetExtension(Ptr(message), id); | 
|  | } | 
|  |  | 
|  | template <typename T, typename Extension> | 
|  | constexpr uint32_t ExtensionNumber( | 
|  | const internal::ExtensionIdentifier<T, Extension>& id) { | 
|  | return internal::PrivateAccess::GetExtensionNumber(id); | 
|  | } | 
|  |  | 
|  | }  // namespace hpb | 
|  |  | 
|  | #endif  // GOOGLE_PROTOBUF_HPB_EXTENSION_H__ |