| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391 | 
							- // 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.
 
- #import "GPBExtensionInternals.h"
 
- #import <objc/runtime.h>
 
- #import "GPBCodedInputStream_PackagePrivate.h"
 
- #import "GPBCodedOutputStream_PackagePrivate.h"
 
- #import "GPBDescriptor_PackagePrivate.h"
 
- #import "GPBMessage_PackagePrivate.h"
 
- #import "GPBUtilities_PackagePrivate.h"
 
- static id NewSingleValueFromInputStream(GPBExtensionDescriptor *extension,
 
-                                         GPBCodedInputStream *input,
 
-                                         GPBExtensionRegistry *extensionRegistry,
 
-                                         GPBMessage *existingValue)
 
-     __attribute__((ns_returns_retained));
 
- GPB_INLINE size_t DataTypeSize(GPBDataType dataType) {
 
- #pragma clang diagnostic push
 
- #pragma clang diagnostic ignored "-Wswitch-enum"
 
-   switch (dataType) {
 
-     case GPBDataTypeBool:
 
-       return 1;
 
-     case GPBDataTypeFixed32:
 
-     case GPBDataTypeSFixed32:
 
-     case GPBDataTypeFloat:
 
-       return 4;
 
-     case GPBDataTypeFixed64:
 
-     case GPBDataTypeSFixed64:
 
-     case GPBDataTypeDouble:
 
-       return 8;
 
-     default:
 
-       return 0;
 
-   }
 
- #pragma clang diagnostic pop
 
- }
 
- static size_t ComputePBSerializedSizeNoTagOfObject(GPBDataType dataType, id object) {
 
- #define FIELD_CASE(TYPE, ACCESSOR)                                     \
 
-   case GPBDataType##TYPE:                                              \
 
-     return GPBCompute##TYPE##SizeNoTag([(NSNumber *)object ACCESSOR]);
 
- #define FIELD_CASE2(TYPE)                                              \
 
-   case GPBDataType##TYPE:                                              \
 
-     return GPBCompute##TYPE##SizeNoTag(object);
 
-   switch (dataType) {
 
-     FIELD_CASE(Bool, boolValue)
 
-     FIELD_CASE(Float, floatValue)
 
-     FIELD_CASE(Double, doubleValue)
 
-     FIELD_CASE(Int32, intValue)
 
-     FIELD_CASE(SFixed32, intValue)
 
-     FIELD_CASE(SInt32, intValue)
 
-     FIELD_CASE(Enum, intValue)
 
-     FIELD_CASE(Int64, longLongValue)
 
-     FIELD_CASE(SInt64, longLongValue)
 
-     FIELD_CASE(SFixed64, longLongValue)
 
-     FIELD_CASE(UInt32, unsignedIntValue)
 
-     FIELD_CASE(Fixed32, unsignedIntValue)
 
-     FIELD_CASE(UInt64, unsignedLongLongValue)
 
-     FIELD_CASE(Fixed64, unsignedLongLongValue)
 
-     FIELD_CASE2(Bytes)
 
-     FIELD_CASE2(String)
 
-     FIELD_CASE2(Message)
 
-     FIELD_CASE2(Group)
 
-   }
 
- #undef FIELD_CASE
 
- #undef FIELD_CASE2
 
- }
 
- static size_t ComputeSerializedSizeIncludingTagOfObject(
 
-     GPBExtensionDescription *description, id object) {
 
- #define FIELD_CASE(TYPE, ACCESSOR)                                   \
 
-   case GPBDataType##TYPE:                                            \
 
-     return GPBCompute##TYPE##Size(description->fieldNumber,          \
 
-                                   [(NSNumber *)object ACCESSOR]);
 
- #define FIELD_CASE2(TYPE)                                            \
 
-   case GPBDataType##TYPE:                                            \
 
-     return GPBCompute##TYPE##Size(description->fieldNumber, object);
 
-   switch (description->dataType) {
 
-     FIELD_CASE(Bool, boolValue)
 
-     FIELD_CASE(Float, floatValue)
 
-     FIELD_CASE(Double, doubleValue)
 
-     FIELD_CASE(Int32, intValue)
 
-     FIELD_CASE(SFixed32, intValue)
 
-     FIELD_CASE(SInt32, intValue)
 
-     FIELD_CASE(Enum, intValue)
 
-     FIELD_CASE(Int64, longLongValue)
 
-     FIELD_CASE(SInt64, longLongValue)
 
-     FIELD_CASE(SFixed64, longLongValue)
 
-     FIELD_CASE(UInt32, unsignedIntValue)
 
-     FIELD_CASE(Fixed32, unsignedIntValue)
 
-     FIELD_CASE(UInt64, unsignedLongLongValue)
 
-     FIELD_CASE(Fixed64, unsignedLongLongValue)
 
-     FIELD_CASE2(Bytes)
 
-     FIELD_CASE2(String)
 
-     FIELD_CASE2(Group)
 
-     case GPBDataTypeMessage:
 
-       if (GPBExtensionIsWireFormat(description)) {
 
-         return GPBComputeMessageSetExtensionSize(description->fieldNumber,
 
-                                                  object);
 
-       } else {
 
-         return GPBComputeMessageSize(description->fieldNumber, object);
 
-       }
 
-   }
 
- #undef FIELD_CASE
 
- #undef FIELD_CASE2
 
- }
 
- static size_t ComputeSerializedSizeIncludingTagOfArray(
 
-     GPBExtensionDescription *description, NSArray *values) {
 
-   if (GPBExtensionIsPacked(description)) {
 
-     size_t size = 0;
 
-     size_t typeSize = DataTypeSize(description->dataType);
 
-     if (typeSize != 0) {
 
-       size = values.count * typeSize;
 
-     } else {
 
-       for (id value in values) {
 
-         size +=
 
-             ComputePBSerializedSizeNoTagOfObject(description->dataType, value);
 
-       }
 
-     }
 
-     return size + GPBComputeTagSize(description->fieldNumber) +
 
-            GPBComputeRawVarint32SizeForInteger(size);
 
-   } else {
 
-     size_t size = 0;
 
-     for (id value in values) {
 
-       size += ComputeSerializedSizeIncludingTagOfObject(description, value);
 
-     }
 
-     return size;
 
-   }
 
- }
 
- static void WriteObjectIncludingTagToCodedOutputStream(
 
-     id object, GPBExtensionDescription *description,
 
-     GPBCodedOutputStream *output) {
 
- #define FIELD_CASE(TYPE, ACCESSOR)                      \
 
-   case GPBDataType##TYPE:                               \
 
-     [output write##TYPE:description->fieldNumber        \
 
-                   value:[(NSNumber *)object ACCESSOR]]; \
 
-     return;
 
- #define FIELD_CASE2(TYPE)                                       \
 
-   case GPBDataType##TYPE:                                       \
 
-     [output write##TYPE:description->fieldNumber value:object]; \
 
-     return;
 
-   switch (description->dataType) {
 
-     FIELD_CASE(Bool, boolValue)
 
-     FIELD_CASE(Float, floatValue)
 
-     FIELD_CASE(Double, doubleValue)
 
-     FIELD_CASE(Int32, intValue)
 
-     FIELD_CASE(SFixed32, intValue)
 
-     FIELD_CASE(SInt32, intValue)
 
-     FIELD_CASE(Enum, intValue)
 
-     FIELD_CASE(Int64, longLongValue)
 
-     FIELD_CASE(SInt64, longLongValue)
 
-     FIELD_CASE(SFixed64, longLongValue)
 
-     FIELD_CASE(UInt32, unsignedIntValue)
 
-     FIELD_CASE(Fixed32, unsignedIntValue)
 
-     FIELD_CASE(UInt64, unsignedLongLongValue)
 
-     FIELD_CASE(Fixed64, unsignedLongLongValue)
 
-     FIELD_CASE2(Bytes)
 
-     FIELD_CASE2(String)
 
-     FIELD_CASE2(Group)
 
-     case GPBDataTypeMessage:
 
-       if (GPBExtensionIsWireFormat(description)) {
 
-         [output writeMessageSetExtension:description->fieldNumber value:object];
 
-       } else {
 
-         [output writeMessage:description->fieldNumber value:object];
 
-       }
 
-       return;
 
-   }
 
- #undef FIELD_CASE
 
- #undef FIELD_CASE2
 
- }
 
- static void WriteObjectNoTagToCodedOutputStream(
 
-     id object, GPBExtensionDescription *description,
 
-     GPBCodedOutputStream *output) {
 
- #define FIELD_CASE(TYPE, ACCESSOR)                             \
 
-   case GPBDataType##TYPE:                                      \
 
-     [output write##TYPE##NoTag:[(NSNumber *)object ACCESSOR]]; \
 
-     return;
 
- #define FIELD_CASE2(TYPE)               \
 
-   case GPBDataType##TYPE:               \
 
-     [output write##TYPE##NoTag:object]; \
 
-     return;
 
-   switch (description->dataType) {
 
-     FIELD_CASE(Bool, boolValue)
 
-     FIELD_CASE(Float, floatValue)
 
-     FIELD_CASE(Double, doubleValue)
 
-     FIELD_CASE(Int32, intValue)
 
-     FIELD_CASE(SFixed32, intValue)
 
-     FIELD_CASE(SInt32, intValue)
 
-     FIELD_CASE(Enum, intValue)
 
-     FIELD_CASE(Int64, longLongValue)
 
-     FIELD_CASE(SInt64, longLongValue)
 
-     FIELD_CASE(SFixed64, longLongValue)
 
-     FIELD_CASE(UInt32, unsignedIntValue)
 
-     FIELD_CASE(Fixed32, unsignedIntValue)
 
-     FIELD_CASE(UInt64, unsignedLongLongValue)
 
-     FIELD_CASE(Fixed64, unsignedLongLongValue)
 
-     FIELD_CASE2(Bytes)
 
-     FIELD_CASE2(String)
 
-     FIELD_CASE2(Message)
 
-     case GPBDataTypeGroup:
 
-       [output writeGroupNoTag:description->fieldNumber value:object];
 
-       return;
 
-   }
 
- #undef FIELD_CASE
 
- #undef FIELD_CASE2
 
- }
 
- static void WriteArrayIncludingTagsToCodedOutputStream(
 
-     NSArray *values, GPBExtensionDescription *description,
 
-     GPBCodedOutputStream *output) {
 
-   if (GPBExtensionIsPacked(description)) {
 
-     [output writeTag:description->fieldNumber
 
-               format:GPBWireFormatLengthDelimited];
 
-     size_t dataSize = 0;
 
-     size_t typeSize = DataTypeSize(description->dataType);
 
-     if (typeSize != 0) {
 
-       dataSize = values.count * typeSize;
 
-     } else {
 
-       for (id value in values) {
 
-         dataSize +=
 
-             ComputePBSerializedSizeNoTagOfObject(description->dataType, value);
 
-       }
 
-     }
 
-     [output writeRawVarintSizeTAs32:dataSize];
 
-     for (id value in values) {
 
-       WriteObjectNoTagToCodedOutputStream(value, description, output);
 
-     }
 
-   } else {
 
-     for (id value in values) {
 
-       WriteObjectIncludingTagToCodedOutputStream(value, description, output);
 
-     }
 
-   }
 
- }
 
- // 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"
 
- void GPBExtensionMergeFromInputStream(GPBExtensionDescriptor *extension,
 
-                                       BOOL isPackedOnStream,
 
-                                       GPBCodedInputStream *input,
 
-                                       GPBExtensionRegistry *extensionRegistry,
 
-                                       GPBMessage *message) {
 
-   GPBExtensionDescription *description = extension->description_;
 
-   GPBCodedInputStreamState *state = &input->state_;
 
-   if (isPackedOnStream) {
 
-     NSCAssert(GPBExtensionIsRepeated(description),
 
-               @"How was it packed if it isn't repeated?");
 
-     int32_t length = GPBCodedInputStreamReadInt32(state);
 
-     size_t limit = GPBCodedInputStreamPushLimit(state, length);
 
-     while (GPBCodedInputStreamBytesUntilLimit(state) > 0) {
 
-       id value = NewSingleValueFromInputStream(extension,
 
-                                                input,
 
-                                                extensionRegistry,
 
-                                                nil);
 
-       [message addExtension:extension value:value];
 
-       [value release];
 
-     }
 
-     GPBCodedInputStreamPopLimit(state, limit);
 
-   } else {
 
-     id existingValue = nil;
 
-     BOOL isRepeated = GPBExtensionIsRepeated(description);
 
-     if (!isRepeated && GPBDataTypeIsMessage(description->dataType)) {
 
-       existingValue = [message getExistingExtension:extension];
 
-     }
 
-     id value = NewSingleValueFromInputStream(extension,
 
-                                              input,
 
-                                              extensionRegistry,
 
-                                              existingValue);
 
-     if (isRepeated) {
 
-       [message addExtension:extension value:value];
 
-     } else {
 
-       [message setExtension:extension value:value];
 
-     }
 
-     [value release];
 
-   }
 
- }
 
- void GPBWriteExtensionValueToOutputStream(GPBExtensionDescriptor *extension,
 
-                                           id value,
 
-                                           GPBCodedOutputStream *output) {
 
-   GPBExtensionDescription *description = extension->description_;
 
-   if (GPBExtensionIsRepeated(description)) {
 
-     WriteArrayIncludingTagsToCodedOutputStream(value, description, output);
 
-   } else {
 
-     WriteObjectIncludingTagToCodedOutputStream(value, description, output);
 
-   }
 
- }
 
- size_t GPBComputeExtensionSerializedSizeIncludingTag(
 
-     GPBExtensionDescriptor *extension, id value) {
 
-   GPBExtensionDescription *description = extension->description_;
 
-   if (GPBExtensionIsRepeated(description)) {
 
-     return ComputeSerializedSizeIncludingTagOfArray(description, value);
 
-   } else {
 
-     return ComputeSerializedSizeIncludingTagOfObject(description, value);
 
-   }
 
- }
 
- // Note that this returns a retained value intentionally.
 
- static id NewSingleValueFromInputStream(GPBExtensionDescriptor *extension,
 
-                                         GPBCodedInputStream *input,
 
-                                         GPBExtensionRegistry *extensionRegistry,
 
-                                         GPBMessage *existingValue) {
 
-   GPBExtensionDescription *description = extension->description_;
 
-   GPBCodedInputStreamState *state = &input->state_;
 
-   switch (description->dataType) {
 
-     case GPBDataTypeBool:     return [[NSNumber alloc] initWithBool:GPBCodedInputStreamReadBool(state)];
 
-     case GPBDataTypeFixed32:  return [[NSNumber alloc] initWithUnsignedInt:GPBCodedInputStreamReadFixed32(state)];
 
-     case GPBDataTypeSFixed32: return [[NSNumber alloc] initWithInt:GPBCodedInputStreamReadSFixed32(state)];
 
-     case GPBDataTypeFloat:    return [[NSNumber alloc] initWithFloat:GPBCodedInputStreamReadFloat(state)];
 
-     case GPBDataTypeFixed64:  return [[NSNumber alloc] initWithUnsignedLongLong:GPBCodedInputStreamReadFixed64(state)];
 
-     case GPBDataTypeSFixed64: return [[NSNumber alloc] initWithLongLong:GPBCodedInputStreamReadSFixed64(state)];
 
-     case GPBDataTypeDouble:   return [[NSNumber alloc] initWithDouble:GPBCodedInputStreamReadDouble(state)];
 
-     case GPBDataTypeInt32:    return [[NSNumber alloc] initWithInt:GPBCodedInputStreamReadInt32(state)];
 
-     case GPBDataTypeInt64:    return [[NSNumber alloc] initWithLongLong:GPBCodedInputStreamReadInt64(state)];
 
-     case GPBDataTypeSInt32:   return [[NSNumber alloc] initWithInt:GPBCodedInputStreamReadSInt32(state)];
 
-     case GPBDataTypeSInt64:   return [[NSNumber alloc] initWithLongLong:GPBCodedInputStreamReadSInt64(state)];
 
-     case GPBDataTypeUInt32:   return [[NSNumber alloc] initWithUnsignedInt:GPBCodedInputStreamReadUInt32(state)];
 
-     case GPBDataTypeUInt64:   return [[NSNumber alloc] initWithUnsignedLongLong:GPBCodedInputStreamReadUInt64(state)];
 
-     case GPBDataTypeBytes:    return GPBCodedInputStreamReadRetainedBytes(state);
 
-     case GPBDataTypeString:   return GPBCodedInputStreamReadRetainedString(state);
 
-     case GPBDataTypeEnum:     return [[NSNumber alloc] initWithInt:GPBCodedInputStreamReadEnum(state)];
 
-     case GPBDataTypeGroup:
 
-     case GPBDataTypeMessage: {
 
-       GPBMessage *message;
 
-       if (existingValue) {
 
-         message = [existingValue retain];
 
-       } else {
 
-         GPBDescriptor *decriptor = [extension.msgClass descriptor];
 
-         message = [[decriptor.messageClass alloc] init];
 
-       }
 
-       if (description->dataType == GPBDataTypeGroup) {
 
-         [input readGroup:description->fieldNumber
 
-                  message:message
 
-             extensionRegistry:extensionRegistry];
 
-       } else {
 
-         // description->dataType == GPBDataTypeMessage
 
-         if (GPBExtensionIsWireFormat(description)) {
 
-           // For MessageSet fields the message length will have already been
 
-           // read.
 
-           [message mergeFromCodedInputStream:input
 
-                            extensionRegistry:extensionRegistry];
 
-         } else {
 
-           [input readMessage:message extensionRegistry:extensionRegistry];
 
-         }
 
-       }
 
-       return message;
 
-     }
 
-   }
 
-   return nil;
 
- }
 
- #pragma clang diagnostic pop
 
 
  |