Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
440 lines
14 KiB
440 lines
14 KiB
// 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 <Foundation/Foundation.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 "GPBUnknownFieldSet_PackagePrivate.h" |
|
#import "GPBUnknownField_PackagePrivate.h" |
|
#import "GPBUnknownFields_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; |
|
case GPBUnknownFieldTypeLegacy: |
|
#if defined(DEBUG) && DEBUG |
|
NSCAssert(NO, @"Internal error within the library"); |
|
#endif |
|
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; |
|
case GPBUnknownFieldTypeLegacy: |
|
#if defined(DEBUG) && DEBUG |
|
NSCAssert(NO, @"Internal error within the library"); |
|
#endif |
|
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]; |
|
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 alloc] init]; |
|
// Fields are r/o in this api, so just copy the array. |
|
copy->fields_ = [fields_ mutableCopyWithZone:zone]; |
|
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]; |
|
} |
|
|
|
- (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
|
|
|