// 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 #ifndef UPB_MESSAGE_INTERNAL_EXTENSION_H_ #define UPB_MESSAGE_INTERNAL_EXTENSION_H_ #include #include "upb/mem/arena.h" #include "upb/message/internal/array.h" #include "upb/message/internal/map.h" #include "upb/message/internal/types.h" #include "upb/message/value.h" #include "upb/mini_table/extension.h" #include "upb/mini_table/internal/field.h" // Must be last. #include "upb/port/def.inc" // The internal representation of an extension is self-describing: it contains // enough information that we can serialize it to binary format without needing // to look it up in a upb_ExtensionRegistry. // // This representation allocates 16 bytes to data on 64-bit platforms. // This is rather wasteful for scalars (in the extreme case of bool, // it wastes 15 bytes). We accept this because we expect messages to be // the most common extension type. typedef struct { const upb_MiniTableExtension* ext; upb_MessageValue data; } upb_Extension; #ifdef __cplusplus extern "C" { #endif // Adds the given extension data to the given message. // |ext| is copied into the message instance. // This logically replaces any previously-added extension with this number. upb_Extension* UPB_PRIVATE(_upb_Message_GetOrCreateExtension)( struct upb_Message* msg, const upb_MiniTableExtension* ext, upb_Arena* arena); // Returns an extension for a message with a given mini table, // or NULL if no extension exists with this mini table. const upb_Extension* UPB_PRIVATE(_upb_Message_Getext)( const struct upb_Message* msg, const upb_MiniTableExtension* ext); UPB_INLINE bool UPB_PRIVATE(_upb_Extension_IsEmpty)(const upb_Extension* ext) { switch ( UPB_PRIVATE(_upb_MiniTableField_Mode)(&ext->ext->UPB_PRIVATE(field))) { case kUpb_FieldMode_Scalar: return false; case kUpb_FieldMode_Array: return upb_Array_Size(ext->data.array_val) == 0; case kUpb_FieldMode_Map: return _upb_Map_Size(ext->data.map_val) == 0; } UPB_UNREACHABLE(); } #ifdef __cplusplus } /* extern "C" */ #endif #include "upb/port/undef.inc" #endif /* UPB_MESSAGE_INTERNAL_EXTENSION_H_ */