| // 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__ |