| // Protocol Buffers - Google's data interchange format |
| // Copyright 2024 Google Inc. 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 |
| |
| #import "GPBUnknownFields.h" |
| #import "GPBUnknownFields_PackagePrivate.h" |
| |
| #import <Foundation/Foundation.h> |
| |
| #import "GPBCodedInputStream.h" |
| #import "GPBCodedInputStream_PackagePrivate.h" |
| #import "GPBCodedOutputStream.h" |
| #import "GPBCodedOutputStream_PackagePrivate.h" |
| #import "GPBDescriptor.h" |
| #import "GPBMessage.h" |
| #import "GPBMessage_PackagePrivate.h" |
| #import "GPBUnknownField.h" |
| #import "GPBUnknownField_PackagePrivate.h" |
| #import "GPBWireFormat.h" |
| |
| #define CHECK_FIELD_NUMBER(number) \ |
| if (number <= 0) { \ |
| [NSException raise:NSInvalidArgumentException format:@"Not a valid field number."]; \ |
| } |
| |
| // TODO: Consider using on other functions to reduce bloat when |
| // some compiler optimizations are enabled. |
| #define GPB_NOINLINE __attribute__((noinline)) |
| |
| @interface GPBUnknownFields () { |
| @package |
| NSMutableArray<GPBUnknownField *> *fields_; |
| } |
| @end |
| |
| // 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_NOINLINE |
| static size_t ComputeSerializeSize(GPBUnknownFields *_Nonnull self) { |
| size_t result = 0; |
| for (GPBUnknownField *field in self->fields_) { |
| uint32_t fieldNumber = field->number_; |
| switch (field->type_) { |
| case GPBUnknownFieldTypeVarint: |
| result += GPBComputeUInt64Size(fieldNumber, field->storage_.intValue); |
| break; |
| case GPBUnknownFieldTypeFixed32: |
| result += GPBComputeFixed32Size(fieldNumber, (uint32_t)field->storage_.intValue); |
| break; |
| case GPBUnknownFieldTypeFixed64: |
| result += GPBComputeFixed64Size(fieldNumber, field->storage_.intValue); |
| break; |
| case GPBUnknownFieldTypeLengthDelimited: |
| result += GPBComputeBytesSize(fieldNumber, field->storage_.lengthDelimited); |
| break; |
| case GPBUnknownFieldTypeGroup: |
| result += |
| (GPBComputeTagSize(fieldNumber) * 2) + ComputeSerializeSize(field->storage_.group); |
| break; |
| } |
| } |
| return result; |
| } |
| |
| GPB_NOINLINE |
| static void WriteToCoddedOutputStream(GPBUnknownFields *_Nonnull self, |
| GPBCodedOutputStream *_Nonnull output) { |
| for (GPBUnknownField *field in self->fields_) { |
| uint32_t fieldNumber = field->number_; |
| switch (field->type_) { |
| case GPBUnknownFieldTypeVarint: |
| [output writeUInt64:fieldNumber value:field->storage_.intValue]; |
| break; |
| case GPBUnknownFieldTypeFixed32: |
| [output writeFixed32:fieldNumber value:(uint32_t)field->storage_.intValue]; |
| break; |
| case GPBUnknownFieldTypeFixed64: |
| [output writeFixed64:fieldNumber value:field->storage_.intValue]; |
| break; |
| case GPBUnknownFieldTypeLengthDelimited: |
| [output writeBytes:fieldNumber value:field->storage_.lengthDelimited]; |
| break; |
| case GPBUnknownFieldTypeGroup: |
| [output writeRawVarint32:GPBWireFormatMakeTag(fieldNumber, GPBWireFormatStartGroup)]; |
| WriteToCoddedOutputStream(field->storage_.group, output); |
| [output writeRawVarint32:GPBWireFormatMakeTag(fieldNumber, GPBWireFormatEndGroup)]; |
| break; |
| } |
| } |
| } |
| |
| GPB_NOINLINE |
| static BOOL MergeFromInputStream(GPBUnknownFields *self, GPBCodedInputStream *input, |
| uint32_t endTag) { |
| #if defined(DEBUG) && DEBUG |
| NSCAssert(endTag == 0 || GPBWireFormatGetTagWireType(endTag) == GPBWireFormatEndGroup, |
| @"Internal error:Invalid end tag: %u", endTag); |
| #endif |
| GPBCodedInputStreamState *state = &input->state_; |
| NSMutableArray<GPBUnknownField *> *fields = self->fields_; |
| @try { |
| while (YES) { |
| uint32_t tag = GPBCodedInputStreamReadTag(state); |
| if (tag == endTag) { |
| return YES; |
| } |
| if (tag == 0) { |
| // Reached end of input without finding the end tag. |
| return NO; |
| } |
| GPBWireFormat wireType = GPBWireFormatGetTagWireType(tag); |
| int32_t fieldNumber = GPBWireFormatGetTagFieldNumber(tag); |
| switch (wireType) { |
| case GPBWireFormatVarint: { |
| uint64_t value = GPBCodedInputStreamReadInt64(state); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber |
| varint:value]; |
| [fields addObject:field]; |
| [field release]; |
| break; |
| } |
| case GPBWireFormatFixed32: { |
| uint32_t value = GPBCodedInputStreamReadFixed32(state); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber |
| fixed32:value]; |
| [fields addObject:field]; |
| [field release]; |
| break; |
| } |
| case GPBWireFormatFixed64: { |
| uint64_t value = GPBCodedInputStreamReadFixed64(state); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber |
| fixed64:value]; |
| [fields addObject:field]; |
| [field release]; |
| break; |
| } |
| case GPBWireFormatLengthDelimited: { |
| NSData *data = GPBCodedInputStreamReadRetainedBytes(state); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber |
| lengthDelimited:data]; |
| [fields addObject:field]; |
| [field release]; |
| [data release]; |
| break; |
| } |
| case GPBWireFormatStartGroup: { |
| GPBUnknownFields *group = [[GPBUnknownFields alloc] init]; |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber group:group]; |
| [fields addObject:field]; |
| [field release]; |
| [group release]; // Still will be held in the field/fields. |
| uint32_t endGroupTag = GPBWireFormatMakeTag(fieldNumber, GPBWireFormatEndGroup); |
| if (MergeFromInputStream(group, input, endGroupTag)) { |
| GPBCodedInputStreamCheckLastTagWas(state, endGroupTag); |
| } else { |
| [NSException |
| raise:NSInternalInconsistencyException |
| format:@"Internal error: Unknown field data for nested group was malformed."]; |
| } |
| break; |
| } |
| case GPBWireFormatEndGroup: |
| [NSException raise:NSInternalInconsistencyException |
| format:@"Unexpected end group tag: %u", tag]; |
| break; |
| } |
| } |
| } @catch (NSException *exception) { |
| #if defined(DEBUG) && DEBUG |
| NSLog(@"%@: Internal exception while parsing unknown data, this shouldn't happen!: %@", |
| [self class], exception); |
| #endif |
| } |
| } |
| |
| @implementation GPBUnknownFields |
| |
| - (instancetype)initFromMessage:(nonnull GPBMessage *)message { |
| self = [super init]; |
| if (self) { |
| fields_ = [[NSMutableArray alloc] init]; |
| // This shouldn't happen with the annotations, but just incase something claiming nonnull |
| // does return nil, block it. |
| if (!message) { |
| [self release]; |
| [NSException raise:NSInvalidArgumentException format:@"Message cannot be nil"]; |
| } |
| NSData *data = GPBMessageUnknownFieldsData(message); |
| if (data) { |
| GPBCodedInputStream *input = [[GPBCodedInputStream alloc] initWithData:data]; |
| // Parse until the end of the data (tag will be zero). |
| if (!MergeFromInputStream(self, input, 0)) { |
| [input release]; |
| [self release]; |
| [NSException raise:NSInternalInconsistencyException |
| format:@"Internal error: Unknown field data from message was malformed."]; |
| } |
| [input release]; |
| } |
| } |
| return self; |
| } |
| |
| - (instancetype)init { |
| self = [super init]; |
| if (self) { |
| fields_ = [[NSMutableArray alloc] init]; |
| } |
| return self; |
| } |
| |
| - (id)copyWithZone:(NSZone *)zone { |
| GPBUnknownFields *copy = [[GPBUnknownFields allocWithZone:zone] init]; |
| copy->fields_ = [[NSMutableArray allocWithZone:zone] initWithArray:fields_ copyItems:YES]; |
| return copy; |
| } |
| |
| - (void)dealloc { |
| [fields_ release]; |
| [super dealloc]; |
| } |
| |
| - (BOOL)isEqual:(id)object { |
| if (![object isKindOfClass:[GPBUnknownFields class]]) { |
| return NO; |
| } |
| GPBUnknownFields *ufs = (GPBUnknownFields *)object; |
| // The type is defined with order of fields mattering, so just compare the arrays. |
| return [fields_ isEqual:ufs->fields_]; |
| } |
| |
| - (NSUInteger)hash { |
| return [fields_ hash]; |
| } |
| |
| - (NSString *)description { |
| return [NSString |
| stringWithFormat:@"<%@ %p>: %lu fields", [self class], self, (unsigned long)fields_.count]; |
| } |
| |
| #pragma mark - Public Methods |
| |
| - (NSUInteger)count { |
| return fields_.count; |
| } |
| |
| - (BOOL)empty { |
| return fields_.count == 0; |
| } |
| |
| - (void)clear { |
| [fields_ removeAllObjects]; |
| } |
| |
| - (NSArray<GPBUnknownField *> *)fields:(int32_t)fieldNumber { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| NSMutableArray<GPBUnknownField *> *result = [[NSMutableArray alloc] init]; |
| for (GPBUnknownField *field in fields_) { |
| if (field.number == fieldNumber) { |
| [result addObject:field]; |
| } |
| } |
| if (result.count == 0) { |
| [result release]; |
| return nil; |
| } |
| return [result autorelease]; |
| } |
| |
| - (void)addFieldNumber:(int32_t)fieldNumber varint:(uint64_t)value { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber varint:value]; |
| [fields_ addObject:field]; |
| [field release]; |
| } |
| |
| - (void)addFieldNumber:(int32_t)fieldNumber fixed32:(uint32_t)value { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber fixed32:value]; |
| [fields_ addObject:field]; |
| [field release]; |
| } |
| |
| - (void)addFieldNumber:(int32_t)fieldNumber fixed64:(uint64_t)value { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber fixed64:value]; |
| [fields_ addObject:field]; |
| [field release]; |
| } |
| |
| - (void)addFieldNumber:(int32_t)fieldNumber lengthDelimited:(NSData *)value { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber |
| lengthDelimited:value]; |
| [fields_ addObject:field]; |
| [field release]; |
| } |
| |
| - (GPBUnknownFields *)addGroupWithFieldNumber:(int32_t)fieldNumber { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| GPBUnknownFields *group = [[GPBUnknownFields alloc] init]; |
| GPBUnknownField *field = [[GPBUnknownField alloc] initWithNumber:fieldNumber group:group]; |
| [fields_ addObject:field]; |
| [field release]; |
| return [group autorelease]; |
| } |
| |
| - (GPBUnknownField *)addCopyOfField:(nonnull GPBUnknownField *)field { |
| GPBUnknownField *result = [field copy]; |
| [fields_ addObject:result]; |
| return [result autorelease]; |
| } |
| |
| - (void)removeField:(nonnull GPBUnknownField *)field { |
| NSUInteger count = fields_.count; |
| [fields_ removeObjectIdenticalTo:field]; |
| if (count == fields_.count) { |
| [NSException raise:NSInvalidArgumentException format:@"The field was not present."]; |
| } |
| } |
| |
| - (void)clearFieldNumber:(int32_t)fieldNumber { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| NSMutableIndexSet *toRemove = nil; |
| NSUInteger idx = 0; |
| for (GPBUnknownField *field in fields_) { |
| if (field->number_ == fieldNumber) { |
| if (toRemove == nil) { |
| toRemove = [[NSMutableIndexSet alloc] initWithIndex:idx]; |
| } else { |
| [toRemove addIndex:idx]; |
| } |
| } |
| ++idx; |
| } |
| if (toRemove) { |
| [fields_ removeObjectsAtIndexes:toRemove]; |
| [toRemove release]; |
| } |
| } |
| |
| #pragma mark - NSFastEnumeration protocol |
| |
| - (NSUInteger)countByEnumeratingWithState:(NSFastEnumerationState *)state |
| objects:(__unsafe_unretained id _Nonnull *)stackbuf |
| count:(NSUInteger)len { |
| return [fields_ countByEnumeratingWithState:state objects:stackbuf count:len]; |
| } |
| |
| #pragma mark - Internal Methods |
| |
| - (NSData *)serializeAsData { |
| if (fields_.count == 0) { |
| return [NSData data]; |
| } |
| size_t expectedSize = ComputeSerializeSize(self); |
| NSMutableData *data = [NSMutableData dataWithLength:expectedSize]; |
| GPBCodedOutputStream *stream = [[GPBCodedOutputStream alloc] initWithData:data]; |
| @try { |
| WriteToCoddedOutputStream(self, stream); |
| [stream flush]; |
| } @catch (NSException *exception) { |
| #if defined(DEBUG) && DEBUG |
| NSLog(@"Internal exception while building GPBUnknownFields serialized data: %@", exception); |
| #endif |
| } |
| #if defined(DEBUG) && DEBUG |
| NSAssert([stream bytesWritten] == expectedSize, @"Internal error within the library"); |
| #endif |
| [stream release]; |
| return data; |
| } |
| |
| @end |
| |
| @implementation GPBUnknownFields (AccessHelpers) |
| |
| - (BOOL)getFirst:(int32_t)fieldNumber varint:(nonnull uint64_t *)outValue { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| for (GPBUnknownField *field in fields_) { |
| if (field.number == fieldNumber && field.type == GPBUnknownFieldTypeVarint) { |
| *outValue = field.varint; |
| return YES; |
| } |
| } |
| return NO; |
| } |
| |
| - (BOOL)getFirst:(int32_t)fieldNumber fixed32:(nonnull uint32_t *)outValue { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| for (GPBUnknownField *field in fields_) { |
| if (field.number == fieldNumber && field.type == GPBUnknownFieldTypeFixed32) { |
| *outValue = field.fixed32; |
| return YES; |
| } |
| } |
| return NO; |
| } |
| |
| - (BOOL)getFirst:(int32_t)fieldNumber fixed64:(nonnull uint64_t *)outValue { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| for (GPBUnknownField *field in fields_) { |
| if (field.number == fieldNumber && field.type == GPBUnknownFieldTypeFixed64) { |
| *outValue = field.fixed64; |
| return YES; |
| } |
| } |
| return NO; |
| } |
| |
| - (nullable NSData *)firstLengthDelimited:(int32_t)fieldNumber { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| for (GPBUnknownField *field in fields_) { |
| if (field.number == fieldNumber && field.type == GPBUnknownFieldTypeLengthDelimited) { |
| return field.lengthDelimited; |
| } |
| } |
| return nil; |
| } |
| |
| - (nullable GPBUnknownFields *)firstGroup:(int32_t)fieldNumber { |
| CHECK_FIELD_NUMBER(fieldNumber); |
| for (GPBUnknownField *field in fields_) { |
| if (field.number == fieldNumber && field.type == GPBUnknownFieldTypeGroup) { |
| return field.group; |
| } |
| } |
| return nil; |
| } |
| |
| @end |
| |
| #pragma clang diagnostic pop |