|  | // 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_BACKEND_UPB_UPB_H__ | 
|  | #define GOOGLE_PROTOBUF_HPB_BACKEND_UPB_UPB_H__ | 
|  |  | 
|  | #include <cstddef> | 
|  |  | 
|  | #include "absl/status/statusor.h" | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "hpb/arena.h" | 
|  | #include "hpb/backend/types.h" | 
|  | #include "hpb/backend/upb/interop.h" | 
|  | #include "hpb/extension.h" | 
|  | #include "hpb/internal/internal.h" | 
|  | #include "hpb/internal/message_lock.h" | 
|  | #include "hpb/internal/template_help.h" | 
|  | #include "hpb/options.h" | 
|  | #include "hpb/ptr.h" | 
|  | #include "hpb/status.h" | 
|  | #include "upb/mem/arena.h" | 
|  | #include "upb/message/message.h" | 
|  | #include "upb/wire/decode.h" | 
|  |  | 
|  | // Must be last. | 
|  | #include "upb/port/def.inc" | 
|  |  | 
|  | namespace hpb::internal::backend::upb { | 
|  |  | 
|  | template <size_t N> | 
|  | class DefaultInstance { | 
|  | public: | 
|  | static const upb_Message* msg() { return (upb_Message*)buffer_; } | 
|  | static upb_Arena* arena() { return arena_; } | 
|  |  | 
|  | private: | 
|  | alignas(UPB_MALLOC_ALIGN) static constexpr char buffer_[N] = {0}; | 
|  | static inline upb_Arena* arena_ = upb_Arena_New(); | 
|  | }; | 
|  |  | 
|  | template <typename T> | 
|  | typename T::Proxy CreateMessage(hpb::Arena& arena) { | 
|  | return PrivateAccess::CreateMessage<T>(hpb::interop::upb::UnwrapArena(arena)); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | typename T::Proxy CloneMessage(Ptr<T> message, hpb::Arena& arena) { | 
|  | return internal::PrivateAccess::Proxy<T>( | 
|  | internal::DeepClone(interop::upb::GetMessage(message), T::minitable(), | 
|  | hpb::interop::upb::UnwrapArena(arena)), | 
|  | hpb::interop::upb::UnwrapArena(arena)); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | void ClearMessage(PtrOrRawMutable<T> message) { | 
|  | auto ptr = Ptr(message); | 
|  | auto minitable = interop::upb::GetMiniTable(ptr); | 
|  | upb_Message_Clear(interop::upb::GetMessage(ptr), minitable); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | void DeepCopy(Ptr<const T> source_message, Ptr<T> target_message) { | 
|  | static_assert(!std::is_const_v<T>); | 
|  | internal::DeepCopy(interop::upb::GetMessage(target_message), | 
|  | interop::upb::GetMessage(source_message), T::minitable(), | 
|  | interop::upb::GetArena(target_message)); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | absl::StatusOr<absl::string_view> Serialize(PtrOrRaw<T> message, | 
|  | hpb::Arena& arena) { | 
|  | return hpb::internal::Serialize(interop::upb::GetMessage(message), | 
|  | interop::upb::GetMiniTable(message), | 
|  | hpb::interop::upb::UnwrapArena(arena), 0); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | bool Parse(PtrOrRaw<T> message, absl::string_view bytes, | 
|  | const ExtensionRegistry& extension_registry) { | 
|  | static_assert(!std::is_const_v<T>); | 
|  | upb_Message_Clear(interop::upb::GetMessage(message), | 
|  | interop::upb::GetMiniTable(message)); | 
|  | auto* arena = interop::upb::GetArena(message); | 
|  | return upb_Decode(bytes.data(), bytes.size(), | 
|  | interop::upb::GetMessage(message), | 
|  | interop::upb::GetMiniTable(message), | 
|  | internal::GetUpbExtensions(extension_registry), | 
|  | /* options= */ 0, arena) == kUpb_DecodeStatus_Ok; | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | absl::StatusOr<T> Parse(absl::string_view bytes, | 
|  | const ExtensionRegistry& extension_registry) { | 
|  | T message; | 
|  | auto* arena = interop::upb::GetArena(&message); | 
|  | upb_DecodeStatus status = | 
|  | upb_Decode(bytes.data(), bytes.size(), interop::upb::GetMessage(&message), | 
|  | interop::upb::GetMiniTable(&message), | 
|  | internal::GetUpbExtensions(extension_registry), | 
|  | /* options= */ 0, arena); | 
|  | if (status == kUpb_DecodeStatus_Ok) { | 
|  | return message; | 
|  | } | 
|  | return MessageDecodeError(status); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | hpb::StatusOr<T> Parse(absl::string_view bytes, ParseOptions options) { | 
|  | int flags = 0; | 
|  | if (options.alias_string) { | 
|  | flags |= kUpb_DecodeOption_AliasString; | 
|  | } | 
|  | T message; | 
|  | auto* arena = interop::upb::GetArena(&message); | 
|  | upb_DecodeStatus status = upb_Decode( | 
|  | bytes.data(), bytes.size(), interop::upb::GetMessage(&message), | 
|  | interop::upb::GetMiniTable(&message), | 
|  | internal::GetUpbExtensions(options.extension_registry), flags, arena); | 
|  | if (status == kUpb_DecodeStatus_Ok) { | 
|  | return hpb::StatusOr<T>(std::move(message)); | 
|  | } | 
|  | return hpb::StatusOr<T>(internal::backend::Error(status)); | 
|  | } | 
|  |  | 
|  | }  // namespace hpb::internal::backend::upb | 
|  |  | 
|  | #include "upb/port/undef.inc" | 
|  |  | 
|  | #endif  // GOOGLE_PROTOBUF_HPB_BACKEND_UPB_UPB_H__ |