|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2008 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 "GPBUnknownFieldSet_PackagePrivate.h"
|
|
|
|
|
|
|
|
#import "GPBCodedInputStream_PackagePrivate.h"
|
|
|
|
#import "GPBCodedOutputStream.h"
|
|
|
|
#import "GPBUnknownField_PackagePrivate.h"
|
|
|
|
#import "GPBUtilities.h"
|
|
|
|
#import "GPBWireFormat.h"
|
|
|
|
|
|
|
|
#pragma mark Helpers
|
|
|
|
|
|
|
|
static void checkNumber(int32_t number) {
|
|
|
|
if (number == 0) {
|
|
|
|
[NSException raise:NSInvalidArgumentException format:@"Zero is not a valid field number."];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@implementation GPBUnknownFieldSet {
|
|
|
|
@package
|
|
|
|
CFMutableDictionaryRef fields_;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void CopyWorker(__unused const void *key, const void *value, void *context) {
|
|
|
|
GPBUnknownField *field = value;
|
|
|
|
GPBUnknownFieldSet *result = context;
|
|
|
|
|
|
|
|
GPBUnknownField *copied = [field copy];
|
|
|
|
[result addField:copied];
|
|
|
|
[copied release];
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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"
|
|
|
|
|
|
|
|
- (id)copyWithZone:(NSZone *)zone {
|
|
|
|
GPBUnknownFieldSet *result = [[GPBUnknownFieldSet allocWithZone:zone] init];
|
|
|
|
if (fields_) {
|
|
|
|
CFDictionaryApplyFunction(fields_, CopyWorker, result);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc {
|
|
|
|
if (fields_) {
|
|
|
|
CFRelease(fields_);
|
|
|
|
}
|
|
|
|
[super dealloc];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isEqual:(id)object {
|
|
|
|
BOOL equal = NO;
|
|
|
|
if ([object isKindOfClass:[GPBUnknownFieldSet class]]) {
|
|
|
|
GPBUnknownFieldSet *set = (GPBUnknownFieldSet *)object;
|
|
|
|
if ((fields_ == NULL) && (set->fields_ == NULL)) {
|
|
|
|
equal = YES;
|
|
|
|
} else if ((fields_ != NULL) && (set->fields_ != NULL)) {
|
|
|
|
equal = CFEqual(fields_, set->fields_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return equal;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSUInteger)hash {
|
|
|
|
// Return the hash of the fields dictionary (or just some value).
|
|
|
|
if (fields_) {
|
|
|
|
return CFHash(fields_);
|
|
|
|
}
|
|
|
|
return (NSUInteger)[GPBUnknownFieldSet class];
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - Public Methods
|
|
|
|
|
|
|
|
- (BOOL)hasField:(int32_t)number {
|
|
|
|
ssize_t key = number;
|
|
|
|
return fields_ ? (CFDictionaryGetValue(fields_, (void *)key) != nil) : NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (GPBUnknownField *)getField:(int32_t)number {
|
|
|
|
ssize_t key = number;
|
|
|
|
GPBUnknownField *result = fields_ ? CFDictionaryGetValue(fields_, (void *)key) : nil;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSUInteger)countOfFields {
|
|
|
|
return fields_ ? CFDictionaryGetCount(fields_) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSArray *)sortedFields {
|
|
|
|
if (!fields_) return [NSArray array];
|
|
|
|
size_t count = CFDictionaryGetCount(fields_);
|
|
|
|
ssize_t keys[count];
|
|
|
|
GPBUnknownField *values[count];
|
|
|
|
CFDictionaryGetKeysAndValues(fields_, (const void **)keys, (const void **)values);
|
|
|
|
struct GPBFieldPair {
|
|
|
|
ssize_t key;
|
|
|
|
GPBUnknownField *value;
|
|
|
|
} pairs[count];
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
pairs[i].key = keys[i];
|
|
|
|
pairs[i].value = values[i];
|
|
|
|
};
|
|
|
|
qsort_b(pairs, count, sizeof(struct GPBFieldPair), ^(const void *first, const void *second) {
|
|
|
|
const struct GPBFieldPair *a = first;
|
|
|
|
const struct GPBFieldPair *b = second;
|
|
|
|
return (a->key > b->key) ? 1 : ((a->key == b->key) ? 0 : -1);
|
|
|
|
});
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
values[i] = pairs[i].value;
|
|
|
|
};
|
|
|
|
return [NSArray arrayWithObjects:values count:count];
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - Internal Methods
|
|
|
|
|
|
|
|
- (void)writeToCodedOutputStream:(GPBCodedOutputStream *)output {
|
|
|
|
if (!fields_) return;
|
|
|
|
size_t count = CFDictionaryGetCount(fields_);
|
|
|
|
ssize_t keys[count];
|
|
|
|
GPBUnknownField *values[count];
|
|
|
|
CFDictionaryGetKeysAndValues(fields_, (const void **)keys, (const void **)values);
|
|
|
|
if (count > 1) {
|
|
|
|
struct GPBFieldPair {
|
|
|
|
ssize_t key;
|
|
|
|
GPBUnknownField *value;
|
|
|
|
} pairs[count];
|
|
|
|
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
pairs[i].key = keys[i];
|
|
|
|
pairs[i].value = values[i];
|
|
|
|
};
|
|
|
|
qsort_b(pairs, count, sizeof(struct GPBFieldPair), ^(const void *first, const void *second) {
|
|
|
|
const struct GPBFieldPair *a = first;
|
|
|
|
const struct GPBFieldPair *b = second;
|
|
|
|
return (a->key > b->key) ? 1 : ((a->key == b->key) ? 0 : -1);
|
|
|
|
});
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
GPBUnknownField *value = pairs[i].value;
|
|
|
|
[value writeToOutput:output];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
[values[0] writeToOutput:output];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *)description {
|
|
|
|
NSMutableString *description =
|
|
|
|
[NSMutableString stringWithFormat:@"<%@ %p>: TextFormat: {\n", [self class], self];
|
|
|
|
NSString *textFormat = GPBTextFormatForUnknownFieldSet(self, @" ");
|
|
|
|
[description appendString:textFormat];
|
|
|
|
[description appendString:@"}"];
|
|
|
|
return description;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GPBUnknownFieldSetSerializedSize(__unused const void *key, const void *value,
|
|
|
|
void *context) {
|
|
|
|
GPBUnknownField *field = value;
|
|
|
|
size_t *result = context;
|
|
|
|
*result += [field serializedSize];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (size_t)serializedSize {
|
|
|
|
size_t result = 0;
|
|
|
|
if (fields_) {
|
|
|
|
CFDictionaryApplyFunction(fields_, GPBUnknownFieldSetSerializedSize, &result);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GPBUnknownFieldSetWriteAsMessageSetTo(__unused const void *key, const void *value,
|
|
|
|
void *context) {
|
|
|
|
GPBUnknownField *field = value;
|
|
|
|
GPBCodedOutputStream *output = context;
|
|
|
|
[field writeAsMessageSetExtensionToOutput:output];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)writeAsMessageSetTo:(GPBCodedOutputStream *)output {
|
|
|
|
if (fields_) {
|
|
|
|
CFDictionaryApplyFunction(fields_, GPBUnknownFieldSetWriteAsMessageSetTo, output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GPBUnknownFieldSetSerializedSizeAsMessageSet(__unused const void *key,
|
|
|
|
const void *value, void *context) {
|
|
|
|
GPBUnknownField *field = value;
|
|
|
|
size_t *result = context;
|
|
|
|
*result += [field serializedSizeAsMessageSetExtension];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (size_t)serializedSizeAsMessageSet {
|
|
|
|
size_t result = 0;
|
|
|
|
if (fields_) {
|
|
|
|
CFDictionaryApplyFunction(fields_, GPBUnknownFieldSetSerializedSizeAsMessageSet, &result);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSData *)data {
|
|
|
|
NSMutableData *data = [NSMutableData dataWithLength:self.serializedSize];
|
|
|
|
GPBCodedOutputStream *output = [[GPBCodedOutputStream alloc] initWithData:data];
|
|
|
|
[self writeToCodedOutputStream:output];
|
|
|
|
[output release];
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (BOOL)isFieldTag:(int32_t)tag {
|
|
|
|
return GPBWireFormatGetTagWireType(tag) != GPBWireFormatEndGroup;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)addField:(GPBUnknownField *)field {
|
|
|
|
int32_t number = [field number];
|
|
|
|
checkNumber(number);
|
|
|
|
if (!fields_) {
|
|
|
|
// Use a custom dictionary here because the keys are numbers and conversion
|
|
|
|
// back and forth from NSNumber isn't worth the cost.
|
|
|
|
fields_ =
|
|
|
|
CFDictionaryCreateMutable(kCFAllocatorDefault, 0, NULL, &kCFTypeDictionaryValueCallBacks);
|
|
|
|
}
|
|
|
|
ssize_t key = number;
|
|
|
|
CFDictionarySetValue(fields_, (const void *)key, field);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (GPBUnknownField *)mutableFieldForNumber:(int32_t)number create:(BOOL)create {
|
|
|
|
ssize_t key = number;
|
|
|
|
GPBUnknownField *existing = fields_ ? CFDictionaryGetValue(fields_, (const void *)key) : nil;
|
|
|
|
if (!existing && create) {
|
|
|
|
existing = [[GPBUnknownField alloc] initWithNumber:number];
|
|
|
|
// This retains existing.
|
|
|
|
[self addField:existing];
|
|
|
|
[existing release];
|
|
|
|
}
|
|
|
|
return existing;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GPBUnknownFieldSetMergeUnknownFields(__unused const void *key, const void *value,
|
|
|
|
void *context) {
|
|
|
|
GPBUnknownField *field = value;
|
|
|
|
GPBUnknownFieldSet *self = context;
|
|
|
|
|
|
|
|
int32_t number = [field number];
|
|
|
|
checkNumber(number);
|
|
|
|
GPBUnknownField *oldField = [self mutableFieldForNumber:number create:NO];
|
|
|
|
if (oldField) {
|
|
|
|
[oldField mergeFromField:field];
|
|
|
|
} else {
|
|
|
|
// Merge only comes from GPBMessage's mergeFrom:, so it means we are on
|
|
|
|
// mutable message and are an mutable instance, so make sure we need
|
|
|
|
// mutable fields.
|
|
|
|
GPBUnknownField *fieldCopy = [field copy];
|
|
|
|
[self addField:fieldCopy];
|
|
|
|
[fieldCopy release];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mergeUnknownFields:(GPBUnknownFieldSet *)other {
|
|
|
|
if (other && other->fields_) {
|
|
|
|
CFDictionaryApplyFunction(other->fields_, GPBUnknownFieldSetMergeUnknownFields, self);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mergeVarintField:(int32_t)number value:(int32_t)value {
|
|
|
|
checkNumber(number);
|
|
|
|
[[self mutableFieldForNumber:number create:YES] addVarint:value];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)mergeFieldFrom:(int32_t)tag input:(GPBCodedInputStream *)input {
|
|
|
|
NSAssert(GPBWireFormatIsValidTag(tag), @"Got passed an invalid tag");
|
|
|
|
int32_t number = GPBWireFormatGetTagFieldNumber(tag);
|
|
|
|
GPBCodedInputStreamState *state = &input->state_;
|
|
|
|
switch (GPBWireFormatGetTagWireType(tag)) {
|
|
|
|
case GPBWireFormatVarint: {
|
|
|
|
GPBUnknownField *field = [self mutableFieldForNumber:number create:YES];
|
|
|
|
[field addVarint:GPBCodedInputStreamReadInt64(state)];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
case GPBWireFormatFixed64: {
|
|
|
|
GPBUnknownField *field = [self mutableFieldForNumber:number create:YES];
|
|
|
|
[field addFixed64:GPBCodedInputStreamReadFixed64(state)];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
case GPBWireFormatLengthDelimited: {
|
|
|
|
NSData *data = GPBCodedInputStreamReadRetainedBytes(state);
|
|
|
|
GPBUnknownField *field = [self mutableFieldForNumber:number create:YES];
|
|
|
|
[field addLengthDelimited:data];
|
|
|
|
[data release];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
case GPBWireFormatStartGroup: {
|
|
|
|
GPBUnknownFieldSet *unknownFieldSet = [[GPBUnknownFieldSet alloc] init];
|
|
|
|
GPBUnknownField *field = [self mutableFieldForNumber:number create:YES];
|
|
|
|
[field addGroup:unknownFieldSet];
|
|
|
|
// The field will now retain unknownFieldSet, so go ahead and release it in case
|
|
|
|
// -readUnknownGroup:message: throws so it won't be leaked.
|
|
|
|
[unknownFieldSet release];
|
|
|
|
[input readUnknownGroup:number message:unknownFieldSet];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
case GPBWireFormatEndGroup:
|
|
|
|
return NO;
|
|
|
|
case GPBWireFormatFixed32: {
|
|
|
|
GPBUnknownField *field = [self mutableFieldForNumber:number create:YES];
|
|
|
|
[field addFixed32:GPBCodedInputStreamReadFixed32(state)];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mergeMessageSetMessage:(int32_t)number data:(NSData *)messageData {
|
|
|
|
[[self mutableFieldForNumber:number create:YES] addLengthDelimited:messageData];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)addUnknownMapEntry:(int32_t)fieldNum value:(NSData *)data {
|
|
|
|
GPBUnknownField *field = [self mutableFieldForNumber:fieldNum create:YES];
|
|
|
|
[field addLengthDelimited:data];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)mergeFromCodedInputStream:(GPBCodedInputStream *)input {
|
|
|
|
while (YES) {
|
|
|
|
int32_t tag = GPBCodedInputStreamReadTag(&input->state_);
|
|
|
|
if (tag == 0 || ![self mergeFieldFrom:tag input:input]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)getTags:(int32_t *)tags {
|
|
|
|
if (!fields_) return;
|
|
|
|
size_t count = CFDictionaryGetCount(fields_);
|
|
|
|
ssize_t keys[count];
|
|
|
|
CFDictionaryGetKeysAndValues(fields_, (const void **)keys, NULL);
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
tags[i] = (int32_t)keys[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma clang diagnostic pop
|
|
|
|
|
|
|
|
@end
|