123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- // Protocol Buffers - Google's data interchange format
- // Copyright 2008 Google Inc. All rights reserved.
- // https://developers.google.com/protocol-buffers/
- //
- // Redistribution and use in source and binary forms, with or without
- // modification, are permitted provided that the following conditions are
- // met:
- //
- // * Redistributions of source code must retain the above copyright
- // notice, this list of conditions and the following disclaimer.
- // * Redistributions in binary form must reproduce the above
- // copyright notice, this list of conditions and the following disclaimer
- // in the documentation and/or other materials provided with the
- // distribution.
- // * Neither the name of Google Inc. nor the names of its
- // contributors may be used to endorse or promote products derived from
- // this software without specific prior written permission.
- //
- // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- // This header is private to the ProtobolBuffers library and must NOT be
- // included by any sources outside this library. The contents of this file are
- // subject to change at any time without notice.
- #import "GPBDescriptor.h"
- #import "GPBWireFormat.h"
- // Describes attributes of the field.
- typedef NS_OPTIONS(uint16_t, GPBFieldFlags) {
- GPBFieldNone = 0,
- // These map to standard protobuf concepts.
- GPBFieldRequired = 1 << 0,
- GPBFieldRepeated = 1 << 1,
- GPBFieldPacked = 1 << 2,
- GPBFieldOptional = 1 << 3,
- GPBFieldHasDefaultValue = 1 << 4,
- // Indicate that the field should "clear" when set to zero value. This is the
- // proto3 non optional behavior for singular data (ints, data, string, enum)
- // fields.
- GPBFieldClearHasIvarOnZero = 1 << 5,
- // Indicates the field needs custom handling for the TextFormat name, if not
- // set, the name can be derived from the ObjC name.
- GPBFieldTextFormatNameCustom = 1 << 6,
- // Indicates the field has an enum descriptor.
- GPBFieldHasEnumDescriptor = 1 << 7,
- // These are not standard protobuf concepts, they are specific to the
- // Objective C runtime.
- // These bits are used to mark the field as a map and what the key
- // type is.
- GPBFieldMapKeyMask = 0xF << 8,
- GPBFieldMapKeyInt32 = 1 << 8,
- GPBFieldMapKeyInt64 = 2 << 8,
- GPBFieldMapKeyUInt32 = 3 << 8,
- GPBFieldMapKeyUInt64 = 4 << 8,
- GPBFieldMapKeySInt32 = 5 << 8,
- GPBFieldMapKeySInt64 = 6 << 8,
- GPBFieldMapKeyFixed32 = 7 << 8,
- GPBFieldMapKeyFixed64 = 8 << 8,
- GPBFieldMapKeySFixed32 = 9 << 8,
- GPBFieldMapKeySFixed64 = 10 << 8,
- GPBFieldMapKeyBool = 11 << 8,
- GPBFieldMapKeyString = 12 << 8,
- };
- // NOTE: The structures defined here have their members ordered to minimize
- // their size. This directly impacts the size of apps since these exist per
- // field/extension.
- // Describes a single field in a protobuf as it is represented as an ivar.
- typedef struct GPBMessageFieldDescription {
- // Name of ivar.
- const char *name;
- union {
- // className is deprecated and will be removed in favor of clazz.
- // kept around right now for backwards compatibility.
- // clazz is used iff GPBDescriptorInitializationFlag_UsesClassRefs is set.
- char *className; // Name of the class of the message.
- Class clazz; // Class of the message.
- // For enums only: If EnumDescriptors are compiled in, it will be that,
- // otherwise it will be the verifier.
- GPBEnumDescriptorFunc enumDescFunc;
- GPBEnumValidationFunc enumVerifier;
- } dataTypeSpecific;
- // The field number for the ivar.
- uint32_t number;
- // The index (in bits) into _has_storage_.
- // >= 0: the bit to use for a value being set.
- // = GPBNoHasBit(INT32_MAX): no storage used.
- // < 0: in a oneOf, use a full int32 to record the field active.
- int32_t hasIndex;
- // Offset of the variable into it's structure struct.
- uint32_t offset;
- // Field flags. Use accessor functions below.
- GPBFieldFlags flags;
- // Data type of the ivar.
- GPBDataType dataType;
- } GPBMessageFieldDescription;
- // Fields in messages defined in a 'proto2' syntax file can provide a default
- // value. This struct provides the default along with the field info.
- typedef struct GPBMessageFieldDescriptionWithDefault {
- // Default value for the ivar.
- GPBGenericValue defaultValue;
- GPBMessageFieldDescription core;
- } GPBMessageFieldDescriptionWithDefault;
- // Describes attributes of the extension.
- typedef NS_OPTIONS(uint8_t, GPBExtensionOptions) {
- GPBExtensionNone = 0,
- // These map to standard protobuf concepts.
- GPBExtensionRepeated = 1 << 0,
- GPBExtensionPacked = 1 << 1,
- GPBExtensionSetWireFormat = 1 << 2,
- };
- // An extension
- typedef struct GPBExtensionDescription {
- GPBGenericValue defaultValue;
- const char *singletonName;
- // Before 3.12, `extendedClass` was just a `const char *`. Thanks to nested
- // initialization (https://en.cppreference.com/w/c/language/struct_initialization#Nested_initialization)
- // old generated code with `.extendedClass = GPBStringifySymbol(Something)`
- // still works; and the current generator can use `extendedClass.clazz`, to
- // pass a Class reference.
- union {
- const char *name;
- Class clazz;
- } extendedClass;
- // Before 3.12, this was `const char *messageOrGroupClassName`. In the
- // initial 3.12 release, we moved the `union messageOrGroupClass`, and failed
- // to realize that would break existing source code for extensions. So to
- // keep existing source code working, we added an unnamed union (C11) to
- // provide both the old field name and the new union. This keeps both older
- // and newer code working.
- // Background: https://github.com/protocolbuffers/protobuf/issues/7555
- union {
- const char *messageOrGroupClassName;
- union {
- const char *name;
- Class clazz;
- } messageOrGroupClass;
- };
- GPBEnumDescriptorFunc enumDescriptorFunc;
- int32_t fieldNumber;
- GPBDataType dataType;
- GPBExtensionOptions options;
- } GPBExtensionDescription;
- typedef NS_OPTIONS(uint32_t, GPBDescriptorInitializationFlags) {
- GPBDescriptorInitializationFlag_None = 0,
- GPBDescriptorInitializationFlag_FieldsWithDefault = 1 << 0,
- GPBDescriptorInitializationFlag_WireFormat = 1 << 1,
- // This is used as a stopgap as we move from using class names to class
- // references. The runtime needs to support both until we allow a
- // breaking change in the runtime.
- GPBDescriptorInitializationFlag_UsesClassRefs = 1 << 2,
- // This flag is used to indicate that the generated sources already contain
- // the `GPBFieldClearHasIvarOnZero` flag and it doesn't have to be computed
- // at startup. This allows older generated code to still work with the
- // current runtime library.
- GPBDescriptorInitializationFlag_Proto3OptionalKnown = 1 << 3,
- };
- @interface GPBDescriptor () {
- @package
- NSArray *fields_;
- NSArray *oneofs_;
- uint32_t storageSize_;
- }
- // fieldDescriptions have to be long lived, they are held as raw pointers.
- + (instancetype)
- allocDescriptorForClass:(Class)messageClass
- rootClass:(Class)rootClass
- file:(GPBFileDescriptor *)file
- fields:(void *)fieldDescriptions
- fieldCount:(uint32_t)fieldCount
- storageSize:(uint32_t)storageSize
- flags:(GPBDescriptorInitializationFlags)flags;
- - (instancetype)initWithClass:(Class)messageClass
- file:(GPBFileDescriptor *)file
- fields:(NSArray *)fields
- storageSize:(uint32_t)storage
- wireFormat:(BOOL)wireFormat;
- // Called right after init to provide extra information to avoid init having
- // an explosion of args. These pointers are recorded, so they are expected
- // to live for the lifetime of the app.
- - (void)setupOneofs:(const char **)oneofNames
- count:(uint32_t)count
- firstHasIndex:(int32_t)firstHasIndex;
- - (void)setupExtraTextInfo:(const char *)extraTextFormatInfo;
- - (void)setupExtensionRanges:(const GPBExtensionRange *)ranges count:(int32_t)count;
- - (void)setupContainingMessageClass:(Class)msgClass;
- - (void)setupMessageClassNameSuffix:(NSString *)suffix;
- // Deprecated. Use setupContainingMessageClass instead.
- - (void)setupContainingMessageClassName:(const char *)msgClassName;
- @end
- @interface GPBFileDescriptor ()
- - (instancetype)initWithPackage:(NSString *)package
- objcPrefix:(NSString *)objcPrefix
- syntax:(GPBFileSyntax)syntax;
- - (instancetype)initWithPackage:(NSString *)package
- syntax:(GPBFileSyntax)syntax;
- @end
- @interface GPBOneofDescriptor () {
- @package
- const char *name_;
- NSArray *fields_;
- SEL caseSel_;
- }
- // name must be long lived.
- - (instancetype)initWithName:(const char *)name fields:(NSArray *)fields;
- @end
- @interface GPBFieldDescriptor () {
- @package
- GPBMessageFieldDescription *description_;
- GPB_UNSAFE_UNRETAINED GPBOneofDescriptor *containingOneof_;
- SEL getSel_;
- SEL setSel_;
- SEL hasOrCountSel_; // *Count for map<>/repeated fields, has* otherwise.
- SEL setHasSel_;
- }
- // Single initializer
- // description has to be long lived, it is held as a raw pointer.
- - (instancetype)initWithFieldDescription:(void *)description
- includesDefault:(BOOL)includesDefault
- usesClassRefs:(BOOL)usesClassRefs
- proto3OptionalKnown:(BOOL)proto3OptionalKnown
- syntax:(GPBFileSyntax)syntax;
- @end
- @interface GPBEnumDescriptor ()
- // valueNames, values and extraTextFormatInfo have to be long lived, they are
- // held as raw pointers.
- + (instancetype)
- allocDescriptorForName:(NSString *)name
- valueNames:(const char *)valueNames
- values:(const int32_t *)values
- count:(uint32_t)valueCount
- enumVerifier:(GPBEnumValidationFunc)enumVerifier;
- + (instancetype)
- allocDescriptorForName:(NSString *)name
- valueNames:(const char *)valueNames
- values:(const int32_t *)values
- count:(uint32_t)valueCount
- enumVerifier:(GPBEnumValidationFunc)enumVerifier
- extraTextFormatInfo:(const char *)extraTextFormatInfo;
- - (instancetype)initWithName:(NSString *)name
- valueNames:(const char *)valueNames
- values:(const int32_t *)values
- count:(uint32_t)valueCount
- enumVerifier:(GPBEnumValidationFunc)enumVerifier;
- @end
- @interface GPBExtensionDescriptor () {
- @package
- GPBExtensionDescription *description_;
- }
- @property(nonatomic, readonly) GPBWireFormat wireType;
- // For repeated extensions, alternateWireType is the wireType with the opposite
- // value for the packable property. i.e. - if the extension was marked packed
- // it would be the wire type for unpacked; if the extension was marked unpacked,
- // it would be the wire type for packed.
- @property(nonatomic, readonly) GPBWireFormat alternateWireType;
- // description has to be long lived, it is held as a raw pointer.
- - (instancetype)initWithExtensionDescription:(GPBExtensionDescription *)desc
- usesClassRefs:(BOOL)usesClassRefs;
- // Deprecated. Calls above with `usesClassRefs = NO`
- - (instancetype)initWithExtensionDescription:(GPBExtensionDescription *)desc;
- - (NSComparisonResult)compareByFieldNumber:(GPBExtensionDescriptor *)other;
- @end
- CF_EXTERN_C_BEGIN
- // Direct access is use for speed, to avoid even internally declaring things
- // read/write, etc. The warning is enabled in the project to ensure code calling
- // protos can turn on -Wdirect-ivar-access without issues.
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdirect-ivar-access"
- GPB_INLINE BOOL GPBFieldIsMapOrArray(GPBFieldDescriptor *field) {
- return (field->description_->flags &
- (GPBFieldRepeated | GPBFieldMapKeyMask)) != 0;
- }
- GPB_INLINE GPBDataType GPBGetFieldDataType(GPBFieldDescriptor *field) {
- return field->description_->dataType;
- }
- GPB_INLINE int32_t GPBFieldHasIndex(GPBFieldDescriptor *field) {
- return field->description_->hasIndex;
- }
- GPB_INLINE uint32_t GPBFieldNumber(GPBFieldDescriptor *field) {
- return field->description_->number;
- }
- #pragma clang diagnostic pop
- uint32_t GPBFieldTag(GPBFieldDescriptor *self);
- // For repeated fields, alternateWireType is the wireType with the opposite
- // value for the packable property. i.e. - if the field was marked packed it
- // would be the wire type for unpacked; if the field was marked unpacked, it
- // would be the wire type for packed.
- uint32_t GPBFieldAlternateTag(GPBFieldDescriptor *self);
- GPB_INLINE BOOL GPBHasPreservingUnknownEnumSemantics(GPBFileSyntax syntax) {
- return syntax == GPBFileSyntaxProto3;
- }
- GPB_INLINE BOOL GPBExtensionIsRepeated(GPBExtensionDescription *description) {
- return (description->options & GPBExtensionRepeated) != 0;
- }
- GPB_INLINE BOOL GPBExtensionIsPacked(GPBExtensionDescription *description) {
- return (description->options & GPBExtensionPacked) != 0;
- }
- GPB_INLINE BOOL GPBExtensionIsWireFormat(GPBExtensionDescription *description) {
- return (description->options & GPBExtensionSetWireFormat) != 0;
- }
- // Helper for compile time assets.
- #ifndef GPBInternalCompileAssert
- #if __has_feature(c_static_assert) || __has_extension(c_static_assert)
- #define GPBInternalCompileAssert(test, msg) _Static_assert((test), #msg)
- #else
- // Pre-Xcode 7 support.
- #define GPBInternalCompileAssertSymbolInner(line, msg) GPBInternalCompileAssert ## line ## __ ## msg
- #define GPBInternalCompileAssertSymbol(line, msg) GPBInternalCompileAssertSymbolInner(line, msg)
- #define GPBInternalCompileAssert(test, msg) \
- typedef char GPBInternalCompileAssertSymbol(__LINE__, msg) [ ((test) ? 1 : -1) ]
- #endif // __has_feature(c_static_assert) || __has_extension(c_static_assert)
- #endif // GPBInternalCompileAssert
- // Sanity check that there isn't padding between the field description
- // structures with and without a default.
- GPBInternalCompileAssert(sizeof(GPBMessageFieldDescriptionWithDefault) ==
- (sizeof(GPBGenericValue) +
- sizeof(GPBMessageFieldDescription)),
- DescriptionsWithDefault_different_size_than_expected);
- CF_EXTERN_C_END
|