|
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317 |
- // 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 "GPBMessage_PackagePrivate.h"
-
- #import <objc/runtime.h>
- #import <objc/message.h>
- #import <stdatomic.h>
-
- #import "GPBArray_PackagePrivate.h"
- #import "GPBCodedInputStream_PackagePrivate.h"
- #import "GPBCodedOutputStream_PackagePrivate.h"
- #import "GPBDescriptor_PackagePrivate.h"
- #import "GPBDictionary_PackagePrivate.h"
- #import "GPBExtensionInternals.h"
- #import "GPBExtensionRegistry.h"
- #import "GPBRootObject_PackagePrivate.h"
- #import "GPBUnknownFieldSet_PackagePrivate.h"
- #import "GPBUtilities_PackagePrivate.h"
-
- // 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"
-
- NSString *const GPBMessageErrorDomain =
- GPBNSStringifySymbol(GPBMessageErrorDomain);
-
- NSString *const GPBErrorReasonKey = @"Reason";
-
- static NSString *const kGPBDataCoderKey = @"GPBData";
-
- //
- // PLEASE REMEMBER:
- //
- // This is the base class for *all* messages generated, so any selector defined,
- // *public* or *private* could end up colliding with a proto message field. So
- // avoid using selectors that could match a property, use C functions to hide
- // them, etc.
- //
-
- @interface GPBMessage () {
- @package
- GPBUnknownFieldSet *unknownFields_;
- NSMutableDictionary *extensionMap_;
- NSMutableDictionary *autocreatedExtensionMap_;
-
- // If the object was autocreated, we remember the creator so that if we get
- // mutated, we can inform the creator to make our field visible.
- GPBMessage *autocreator_;
- GPBFieldDescriptor *autocreatorField_;
- GPBExtensionDescriptor *autocreatorExtension_;
-
- // A lock to provide mutual exclusion from internal data that can be modified
- // by *read* operations such as getters (autocreation of message fields and
- // message extensions, not setting of values). Used to guarantee thread safety
- // for concurrent reads on the message.
- // NOTE: OSSpinLock may seem like a good fit here but Apple engineers have
- // pointed out that they are vulnerable to live locking on iOS in cases of
- // priority inversion:
- // http://mjtsai.com/blog/2015/12/16/osspinlock-is-unsafe/
- // https://lists.swift.org/pipermail/swift-dev/Week-of-Mon-20151214/000372.html
- // Use of readOnlySemaphore_ must be prefaced by a call to
- // GPBPrepareReadOnlySemaphore to ensure it has been created. This allows
- // readOnlySemaphore_ to be only created when actually needed.
- _Atomic(dispatch_semaphore_t) readOnlySemaphore_;
- }
- @end
-
- static id CreateArrayForField(GPBFieldDescriptor *field,
- GPBMessage *autocreator)
- __attribute__((ns_returns_retained));
- static id GetOrCreateArrayIvarWithField(GPBMessage *self,
- GPBFieldDescriptor *field,
- GPBFileSyntax syntax);
- static id GetArrayIvarWithField(GPBMessage *self, GPBFieldDescriptor *field);
- static id CreateMapForField(GPBFieldDescriptor *field,
- GPBMessage *autocreator)
- __attribute__((ns_returns_retained));
- static id GetOrCreateMapIvarWithField(GPBMessage *self,
- GPBFieldDescriptor *field,
- GPBFileSyntax syntax);
- static id GetMapIvarWithField(GPBMessage *self, GPBFieldDescriptor *field);
- static NSMutableDictionary *CloneExtensionMap(NSDictionary *extensionMap,
- NSZone *zone)
- __attribute__((ns_returns_retained));
-
- #ifdef DEBUG
- static NSError *MessageError(NSInteger code, NSDictionary *userInfo) {
- return [NSError errorWithDomain:GPBMessageErrorDomain
- code:code
- userInfo:userInfo];
- }
- #endif
-
- static NSError *ErrorFromException(NSException *exception) {
- NSError *error = nil;
-
- if ([exception.name isEqual:GPBCodedInputStreamException]) {
- NSDictionary *exceptionInfo = exception.userInfo;
- error = exceptionInfo[GPBCodedInputStreamUnderlyingErrorKey];
- }
-
- if (!error) {
- NSString *reason = exception.reason;
- NSDictionary *userInfo = nil;
- if ([reason length]) {
- userInfo = @{ GPBErrorReasonKey : reason };
- }
-
- error = [NSError errorWithDomain:GPBMessageErrorDomain
- code:GPBMessageErrorCodeOther
- userInfo:userInfo];
- }
- return error;
- }
-
- static void CheckExtension(GPBMessage *self,
- GPBExtensionDescriptor *extension) {
- if (![self isKindOfClass:extension.containingMessageClass]) {
- [NSException
- raise:NSInvalidArgumentException
- format:@"Extension %@ used on wrong class (%@ instead of %@)",
- extension.singletonName,
- [self class], extension.containingMessageClass];
- }
- }
-
- static NSMutableDictionary *CloneExtensionMap(NSDictionary *extensionMap,
- NSZone *zone) {
- if (extensionMap.count == 0) {
- return nil;
- }
- NSMutableDictionary *result = [[NSMutableDictionary allocWithZone:zone]
- initWithCapacity:extensionMap.count];
-
- for (GPBExtensionDescriptor *extension in extensionMap) {
- id value = [extensionMap objectForKey:extension];
- BOOL isMessageExtension = GPBExtensionIsMessage(extension);
-
- if (extension.repeated) {
- if (isMessageExtension) {
- NSMutableArray *list =
- [[NSMutableArray alloc] initWithCapacity:[value count]];
- for (GPBMessage *listValue in value) {
- GPBMessage *copiedValue = [listValue copyWithZone:zone];
- [list addObject:copiedValue];
- [copiedValue release];
- }
- [result setObject:list forKey:extension];
- [list release];
- } else {
- NSMutableArray *copiedValue = [value mutableCopyWithZone:zone];
- [result setObject:copiedValue forKey:extension];
- [copiedValue release];
- }
- } else {
- if (isMessageExtension) {
- GPBMessage *copiedValue = [value copyWithZone:zone];
- [result setObject:copiedValue forKey:extension];
- [copiedValue release];
- } else {
- [result setObject:value forKey:extension];
- }
- }
- }
-
- return result;
- }
-
- static id CreateArrayForField(GPBFieldDescriptor *field,
- GPBMessage *autocreator) {
- id result;
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- switch (fieldDataType) {
- case GPBDataTypeBool:
- result = [[GPBBoolArray alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBUInt32Array alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBInt32Array alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBUInt64Array alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBInt64Array alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBFloatArray alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBDoubleArray alloc] init];
- break;
-
- case GPBDataTypeEnum:
- result = [[GPBEnumArray alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
-
- case GPBDataTypeBytes:
- case GPBDataTypeGroup:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- if (autocreator) {
- result = [[GPBAutocreatedArray alloc] init];
- } else {
- result = [[NSMutableArray alloc] init];
- }
- break;
- }
-
- if (autocreator) {
- if (GPBDataTypeIsObject(fieldDataType)) {
- GPBAutocreatedArray *autoArray = result;
- autoArray->_autocreator = autocreator;
- } else {
- GPBInt32Array *gpbArray = result;
- gpbArray->_autocreator = autocreator;
- }
- }
-
- return result;
- }
-
- static id CreateMapForField(GPBFieldDescriptor *field,
- GPBMessage *autocreator) {
- id result;
- GPBDataType keyDataType = field.mapKeyDataType;
- GPBDataType valueDataType = GPBGetFieldDataType(field);
- switch (keyDataType) {
- case GPBDataTypeBool:
- switch (valueDataType) {
- case GPBDataTypeBool:
- result = [[GPBBoolBoolDictionary alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBBoolUInt32Dictionary alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBBoolInt32Dictionary alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBBoolUInt64Dictionary alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBBoolInt64Dictionary alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBBoolFloatDictionary alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBBoolDoubleDictionary alloc] init];
- break;
- case GPBDataTypeEnum:
- result = [[GPBBoolEnumDictionary alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- result = [[GPBBoolObjectDictionary alloc] init];
- break;
- case GPBDataTypeGroup:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- switch (valueDataType) {
- case GPBDataTypeBool:
- result = [[GPBUInt32BoolDictionary alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBUInt32UInt32Dictionary alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBUInt32Int32Dictionary alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBUInt32UInt64Dictionary alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBUInt32Int64Dictionary alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBUInt32FloatDictionary alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBUInt32DoubleDictionary alloc] init];
- break;
- case GPBDataTypeEnum:
- result = [[GPBUInt32EnumDictionary alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- result = [[GPBUInt32ObjectDictionary alloc] init];
- break;
- case GPBDataTypeGroup:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- switch (valueDataType) {
- case GPBDataTypeBool:
- result = [[GPBInt32BoolDictionary alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBInt32UInt32Dictionary alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBInt32Int32Dictionary alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBInt32UInt64Dictionary alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBInt32Int64Dictionary alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBInt32FloatDictionary alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBInt32DoubleDictionary alloc] init];
- break;
- case GPBDataTypeEnum:
- result = [[GPBInt32EnumDictionary alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- result = [[GPBInt32ObjectDictionary alloc] init];
- break;
- case GPBDataTypeGroup:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- switch (valueDataType) {
- case GPBDataTypeBool:
- result = [[GPBUInt64BoolDictionary alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBUInt64UInt32Dictionary alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBUInt64Int32Dictionary alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBUInt64UInt64Dictionary alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBUInt64Int64Dictionary alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBUInt64FloatDictionary alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBUInt64DoubleDictionary alloc] init];
- break;
- case GPBDataTypeEnum:
- result = [[GPBUInt64EnumDictionary alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- result = [[GPBUInt64ObjectDictionary alloc] init];
- break;
- case GPBDataTypeGroup:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- switch (valueDataType) {
- case GPBDataTypeBool:
- result = [[GPBInt64BoolDictionary alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBInt64UInt32Dictionary alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBInt64Int32Dictionary alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBInt64UInt64Dictionary alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBInt64Int64Dictionary alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBInt64FloatDictionary alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBInt64DoubleDictionary alloc] init];
- break;
- case GPBDataTypeEnum:
- result = [[GPBInt64EnumDictionary alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- result = [[GPBInt64ObjectDictionary alloc] init];
- break;
- case GPBDataTypeGroup:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
- break;
- case GPBDataTypeString:
- switch (valueDataType) {
- case GPBDataTypeBool:
- result = [[GPBStringBoolDictionary alloc] init];
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- result = [[GPBStringUInt32Dictionary alloc] init];
- break;
- case GPBDataTypeInt32:
- case GPBDataTypeSFixed32:
- case GPBDataTypeSInt32:
- result = [[GPBStringInt32Dictionary alloc] init];
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- result = [[GPBStringUInt64Dictionary alloc] init];
- break;
- case GPBDataTypeInt64:
- case GPBDataTypeSFixed64:
- case GPBDataTypeSInt64:
- result = [[GPBStringInt64Dictionary alloc] init];
- break;
- case GPBDataTypeFloat:
- result = [[GPBStringFloatDictionary alloc] init];
- break;
- case GPBDataTypeDouble:
- result = [[GPBStringDoubleDictionary alloc] init];
- break;
- case GPBDataTypeEnum:
- result = [[GPBStringEnumDictionary alloc]
- initWithValidationFunction:field.enumDescriptor.enumVerifier];
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeMessage:
- case GPBDataTypeString:
- if (autocreator) {
- result = [[GPBAutocreatedDictionary alloc] init];
- } else {
- result = [[NSMutableDictionary alloc] init];
- }
- break;
- case GPBDataTypeGroup:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
- break;
-
- case GPBDataTypeFloat:
- case GPBDataTypeDouble:
- case GPBDataTypeEnum:
- case GPBDataTypeBytes:
- case GPBDataTypeGroup:
- case GPBDataTypeMessage:
- NSCAssert(NO, @"shouldn't happen");
- return nil;
- }
-
- if (autocreator) {
- if ((keyDataType == GPBDataTypeString) &&
- GPBDataTypeIsObject(valueDataType)) {
- GPBAutocreatedDictionary *autoDict = result;
- autoDict->_autocreator = autocreator;
- } else {
- GPBInt32Int32Dictionary *gpbDict = result;
- gpbDict->_autocreator = autocreator;
- }
- }
-
- return result;
- }
-
- #if !defined(__clang_analyzer__)
- // These functions are blocked from the analyzer because the analyzer sees the
- // GPBSetRetainedObjectIvarWithFieldInternal() call as consuming the array/map,
- // so use of the array/map after the call returns is flagged as a use after
- // free.
- // But GPBSetRetainedObjectIvarWithFieldInternal() is "consuming" the retain
- // count be holding onto the object (it is transfering it), the object is
- // still valid after returning from the call. The other way to avoid this
- // would be to add a -retain/-autorelease, but that would force every
- // repeated/map field parsed into the autorelease pool which is both a memory
- // and performance hit.
-
- static id GetOrCreateArrayIvarWithField(GPBMessage *self,
- GPBFieldDescriptor *field,
- GPBFileSyntax syntax) {
- id array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!array) {
- // No lock needed, this is called from places expecting to mutate
- // so no threading protection is needed.
- array = CreateArrayForField(field, nil);
- GPBSetRetainedObjectIvarWithFieldInternal(self, field, array, syntax);
- }
- return array;
- }
-
- // This is like GPBGetObjectIvarWithField(), but for arrays, it should
- // only be used to wire the method into the class.
- static id GetArrayIvarWithField(GPBMessage *self, GPBFieldDescriptor *field) {
- id array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!array) {
- // Check again after getting the lock.
- GPBPrepareReadOnlySemaphore(self);
- dispatch_semaphore_wait(self->readOnlySemaphore_, DISPATCH_TIME_FOREVER);
- array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!array) {
- array = CreateArrayForField(field, self);
- GPBSetAutocreatedRetainedObjectIvarWithField(self, field, array);
- }
- dispatch_semaphore_signal(self->readOnlySemaphore_);
- }
- return array;
- }
-
- static id GetOrCreateMapIvarWithField(GPBMessage *self,
- GPBFieldDescriptor *field,
- GPBFileSyntax syntax) {
- id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!dict) {
- // No lock needed, this is called from places expecting to mutate
- // so no threading protection is needed.
- dict = CreateMapForField(field, nil);
- GPBSetRetainedObjectIvarWithFieldInternal(self, field, dict, syntax);
- }
- return dict;
- }
-
- // This is like GPBGetObjectIvarWithField(), but for maps, it should
- // only be used to wire the method into the class.
- static id GetMapIvarWithField(GPBMessage *self, GPBFieldDescriptor *field) {
- id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!dict) {
- // Check again after getting the lock.
- GPBPrepareReadOnlySemaphore(self);
- dispatch_semaphore_wait(self->readOnlySemaphore_, DISPATCH_TIME_FOREVER);
- dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!dict) {
- dict = CreateMapForField(field, self);
- GPBSetAutocreatedRetainedObjectIvarWithField(self, field, dict);
- }
- dispatch_semaphore_signal(self->readOnlySemaphore_);
- }
- return dict;
- }
-
- #endif // !defined(__clang_analyzer__)
-
- GPBMessage *GPBCreateMessageWithAutocreator(Class msgClass,
- GPBMessage *autocreator,
- GPBFieldDescriptor *field) {
- GPBMessage *message = [[msgClass alloc] init];
- message->autocreator_ = autocreator;
- message->autocreatorField_ = [field retain];
- return message;
- }
-
- static GPBMessage *CreateMessageWithAutocreatorForExtension(
- Class msgClass, GPBMessage *autocreator, GPBExtensionDescriptor *extension)
- __attribute__((ns_returns_retained));
-
- static GPBMessage *CreateMessageWithAutocreatorForExtension(
- Class msgClass, GPBMessage *autocreator,
- GPBExtensionDescriptor *extension) {
- GPBMessage *message = [[msgClass alloc] init];
- message->autocreator_ = autocreator;
- message->autocreatorExtension_ = [extension retain];
- return message;
- }
-
- BOOL GPBWasMessageAutocreatedBy(GPBMessage *message, GPBMessage *parent) {
- return (message->autocreator_ == parent);
- }
-
- void GPBBecomeVisibleToAutocreator(GPBMessage *self) {
- // Message objects that are implicitly created by accessing a message field
- // are initially not visible via the hasX selector. This method makes them
- // visible.
- if (self->autocreator_) {
- // This will recursively make all parent messages visible until it reaches a
- // super-creator that's visible.
- if (self->autocreatorField_) {
- GPBFileSyntax syntax = [self->autocreator_ descriptor].file.syntax;
- GPBSetObjectIvarWithFieldInternal(self->autocreator_,
- self->autocreatorField_, self, syntax);
- } else {
- [self->autocreator_ setExtension:self->autocreatorExtension_ value:self];
- }
- }
- }
-
- void GPBAutocreatedArrayModified(GPBMessage *self, id array) {
- // When one of our autocreated arrays adds elements, make it visible.
- GPBDescriptor *descriptor = [[self class] descriptor];
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (field.fieldType == GPBFieldTypeRepeated) {
- id curArray = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (curArray == array) {
- if (GPBFieldDataTypeIsObject(field)) {
- GPBAutocreatedArray *autoArray = array;
- autoArray->_autocreator = nil;
- } else {
- GPBInt32Array *gpbArray = array;
- gpbArray->_autocreator = nil;
- }
- GPBBecomeVisibleToAutocreator(self);
- return;
- }
- }
- }
- NSCAssert(NO, @"Unknown autocreated %@ for %@.", [array class], self);
- }
-
- void GPBAutocreatedDictionaryModified(GPBMessage *self, id dictionary) {
- // When one of our autocreated dicts adds elements, make it visible.
- GPBDescriptor *descriptor = [[self class] descriptor];
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (field.fieldType == GPBFieldTypeMap) {
- id curDict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (curDict == dictionary) {
- if ((field.mapKeyDataType == GPBDataTypeString) &&
- GPBFieldDataTypeIsObject(field)) {
- GPBAutocreatedDictionary *autoDict = dictionary;
- autoDict->_autocreator = nil;
- } else {
- GPBInt32Int32Dictionary *gpbDict = dictionary;
- gpbDict->_autocreator = nil;
- }
- GPBBecomeVisibleToAutocreator(self);
- return;
- }
- }
- }
- NSCAssert(NO, @"Unknown autocreated %@ for %@.", [dictionary class], self);
- }
-
- void GPBClearMessageAutocreator(GPBMessage *self) {
- if ((self == nil) || !self->autocreator_) {
- return;
- }
-
- #if defined(DEBUG) && DEBUG && !defined(NS_BLOCK_ASSERTIONS)
- // Either the autocreator must have its "has" flag set to YES, or it must be
- // NO and not equal to ourselves.
- BOOL autocreatorHas =
- (self->autocreatorField_
- ? GPBGetHasIvarField(self->autocreator_, self->autocreatorField_)
- : [self->autocreator_ hasExtension:self->autocreatorExtension_]);
- GPBMessage *autocreatorFieldValue =
- (self->autocreatorField_
- ? GPBGetObjectIvarWithFieldNoAutocreate(self->autocreator_,
- self->autocreatorField_)
- : [self->autocreator_->autocreatedExtensionMap_
- objectForKey:self->autocreatorExtension_]);
- NSCAssert(autocreatorHas || autocreatorFieldValue != self,
- @"Cannot clear autocreator because it still refers to self, self: %@.",
- self);
-
- #endif // DEBUG && !defined(NS_BLOCK_ASSERTIONS)
-
- self->autocreator_ = nil;
- [self->autocreatorField_ release];
- self->autocreatorField_ = nil;
- [self->autocreatorExtension_ release];
- self->autocreatorExtension_ = nil;
- }
-
- // Call this before using the readOnlySemaphore_. This ensures it is created only once.
- void GPBPrepareReadOnlySemaphore(GPBMessage *self) {
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdirect-ivar-access"
-
- // Create the semaphore on demand (rather than init) as developers might not cause them
- // to be needed, and the heap usage can add up. The atomic swap is used to avoid needing
- // another lock around creating it.
- if (self->readOnlySemaphore_ == nil) {
- dispatch_semaphore_t worker = dispatch_semaphore_create(1);
- dispatch_semaphore_t expected = nil;
- if (!atomic_compare_exchange_strong(&self->readOnlySemaphore_, &expected, worker)) {
- dispatch_release(worker);
- }
- #if defined(__clang_analyzer__)
- // The Xcode 9.2 (and 9.3 beta) static analyzer thinks worker is leaked
- // (doesn't seem to know about atomic_compare_exchange_strong); so just
- // for the analyzer, let it think worker is also released in this case.
- else { dispatch_release(worker); }
- #endif
- }
-
- #pragma clang diagnostic pop
- }
-
- static GPBUnknownFieldSet *GetOrMakeUnknownFields(GPBMessage *self) {
- if (!self->unknownFields_) {
- self->unknownFields_ = [[GPBUnknownFieldSet alloc] init];
- GPBBecomeVisibleToAutocreator(self);
- }
- return self->unknownFields_;
- }
-
- @implementation GPBMessage
-
- + (void)initialize {
- Class pbMessageClass = [GPBMessage class];
- if ([self class] == pbMessageClass) {
- // This is here to start up the "base" class descriptor.
- [self descriptor];
- // Message shares extension method resolving with GPBRootObject so insure
- // it is started up at the same time.
- (void)[GPBRootObject class];
- } else if ([self superclass] == pbMessageClass) {
- // This is here to start up all the "message" subclasses. Just needs to be
- // done for the messages, not any of the subclasses.
- // This must be done in initialize to enforce thread safety of start up of
- // the protocol buffer library.
- // Note: The generated code for -descriptor calls
- // +[GPBDescriptor allocDescriptorForClass:...], passing the GPBRootObject
- // subclass for the file. That call chain is what ensures that *Root class
- // is started up to support extension resolution off the message class
- // (+resolveClassMethod: below) in a thread safe manner.
- [self descriptor];
- }
- }
-
- + (instancetype)allocWithZone:(NSZone *)zone {
- // Override alloc to allocate our classes with the additional storage
- // required for the instance variables.
- GPBDescriptor *descriptor = [self descriptor];
- return NSAllocateObject(self, descriptor->storageSize_, zone);
- }
-
- + (instancetype)alloc {
- return [self allocWithZone:nil];
- }
-
- + (GPBDescriptor *)descriptor {
- // This is thread safe because it is called from +initialize.
- static GPBDescriptor *descriptor = NULL;
- static GPBFileDescriptor *fileDescriptor = NULL;
- if (!descriptor) {
- // Use a dummy file that marks it as proto2 syntax so when used generically
- // it supports unknowns/etc.
- fileDescriptor =
- [[GPBFileDescriptor alloc] initWithPackage:@"internal"
- syntax:GPBFileSyntaxProto2];
-
- descriptor = [GPBDescriptor allocDescriptorForClass:[GPBMessage class]
- rootClass:Nil
- file:fileDescriptor
- fields:NULL
- fieldCount:0
- storageSize:0
- flags:0];
- }
- return descriptor;
- }
-
- + (instancetype)message {
- return [[[self alloc] init] autorelease];
- }
-
- - (instancetype)init {
- if ((self = [super init])) {
- messageStorage_ = (GPBMessage_StoragePtr)(
- ((uint8_t *)self) + class_getInstanceSize([self class]));
- }
-
- return self;
- }
-
- - (instancetype)initWithData:(NSData *)data error:(NSError **)errorPtr {
- return [self initWithData:data extensionRegistry:nil error:errorPtr];
- }
-
- - (instancetype)initWithData:(NSData *)data
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry
- error:(NSError **)errorPtr {
- if ((self = [self init])) {
- @try {
- [self mergeFromData:data extensionRegistry:extensionRegistry];
- if (errorPtr) {
- *errorPtr = nil;
- }
- }
- @catch (NSException *exception) {
- [self release];
- self = nil;
- if (errorPtr) {
- *errorPtr = ErrorFromException(exception);
- }
- }
- #ifdef DEBUG
- if (self && !self.initialized) {
- [self release];
- self = nil;
- if (errorPtr) {
- *errorPtr = MessageError(GPBMessageErrorCodeMissingRequiredField, nil);
- }
- }
- #endif
- }
- return self;
- }
-
- - (instancetype)initWithCodedInputStream:(GPBCodedInputStream *)input
- extensionRegistry:
- (GPBExtensionRegistry *)extensionRegistry
- error:(NSError **)errorPtr {
- if ((self = [self init])) {
- @try {
- [self mergeFromCodedInputStream:input extensionRegistry:extensionRegistry];
- if (errorPtr) {
- *errorPtr = nil;
- }
- }
- @catch (NSException *exception) {
- [self release];
- self = nil;
- if (errorPtr) {
- *errorPtr = ErrorFromException(exception);
- }
- }
- #ifdef DEBUG
- if (self && !self.initialized) {
- [self release];
- self = nil;
- if (errorPtr) {
- *errorPtr = MessageError(GPBMessageErrorCodeMissingRequiredField, nil);
- }
- }
- #endif
- }
- return self;
- }
-
- - (void)dealloc {
- [self internalClear:NO];
- NSCAssert(!autocreator_, @"Autocreator was not cleared before dealloc.");
- if (readOnlySemaphore_) {
- dispatch_release(readOnlySemaphore_);
- }
- [super dealloc];
- }
-
- - (void)copyFieldsInto:(GPBMessage *)message
- zone:(NSZone *)zone
- descriptor:(GPBDescriptor *)descriptor {
- // Copy all the storage...
- memcpy(message->messageStorage_, messageStorage_, descriptor->storageSize_);
-
- GPBFileSyntax syntax = descriptor.file.syntax;
-
- // Loop over the fields doing fixup...
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (GPBFieldIsMapOrArray(field)) {
- id value = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (value) {
- // We need to copy the array/map, but the catch is for message fields,
- // we also need to ensure all the messages as those need copying also.
- id newValue;
- if (GPBFieldDataTypeIsMessage(field)) {
- if (field.fieldType == GPBFieldTypeRepeated) {
- NSArray *existingArray = (NSArray *)value;
- NSMutableArray *newArray =
- [[NSMutableArray alloc] initWithCapacity:existingArray.count];
- newValue = newArray;
- for (GPBMessage *msg in existingArray) {
- GPBMessage *copiedMsg = [msg copyWithZone:zone];
- [newArray addObject:copiedMsg];
- [copiedMsg release];
- }
- } else {
- if (field.mapKeyDataType == GPBDataTypeString) {
- // Map is an NSDictionary.
- NSDictionary *existingDict = value;
- NSMutableDictionary *newDict = [[NSMutableDictionary alloc]
- initWithCapacity:existingDict.count];
- newValue = newDict;
- [existingDict enumerateKeysAndObjectsUsingBlock:^(NSString *key,
- GPBMessage *msg,
- BOOL *stop) {
- #pragma unused(stop)
- GPBMessage *copiedMsg = [msg copyWithZone:zone];
- [newDict setObject:copiedMsg forKey:key];
- [copiedMsg release];
- }];
- } else {
- // Is one of the GPB*ObjectDictionary classes. Type doesn't
- // matter, just need one to invoke the selector.
- GPBInt32ObjectDictionary *existingDict = value;
- newValue = [existingDict deepCopyWithZone:zone];
- }
- }
- } else {
- // Not messages (but is a map/array)...
- if (field.fieldType == GPBFieldTypeRepeated) {
- if (GPBFieldDataTypeIsObject(field)) {
- // NSArray
- newValue = [value mutableCopyWithZone:zone];
- } else {
- // GPB*Array
- newValue = [value copyWithZone:zone];
- }
- } else {
- if ((field.mapKeyDataType == GPBDataTypeString) &&
- GPBFieldDataTypeIsObject(field)) {
- // NSDictionary
- newValue = [value mutableCopyWithZone:zone];
- } else {
- // Is one of the GPB*Dictionary classes. Type doesn't matter,
- // just need one to invoke the selector.
- GPBInt32Int32Dictionary *existingDict = value;
- newValue = [existingDict copyWithZone:zone];
- }
- }
- }
- // We retain here because the memcpy picked up the pointer value and
- // the next call to SetRetainedObject... will release the current value.
- [value retain];
- GPBSetRetainedObjectIvarWithFieldInternal(message, field, newValue,
- syntax);
- }
- } else if (GPBFieldDataTypeIsMessage(field)) {
- // For object types, if we have a value, copy it. If we don't,
- // zero it to remove the pointer to something that was autocreated
- // (and the ptr just got memcpyed).
- if (GPBGetHasIvarField(self, field)) {
- GPBMessage *value = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- GPBMessage *newValue = [value copyWithZone:zone];
- // We retain here because the memcpy picked up the pointer value and
- // the next call to SetRetainedObject... will release the current value.
- [value retain];
- GPBSetRetainedObjectIvarWithFieldInternal(message, field, newValue,
- syntax);
- } else {
- uint8_t *storage = (uint8_t *)message->messageStorage_;
- id *typePtr = (id *)&storage[field->description_->offset];
- *typePtr = NULL;
- }
- } else if (GPBFieldDataTypeIsObject(field) &&
- GPBGetHasIvarField(self, field)) {
- // A set string/data value (message picked off above), copy it.
- id value = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- id newValue = [value copyWithZone:zone];
- // We retain here because the memcpy picked up the pointer value and
- // the next call to SetRetainedObject... will release the current value.
- [value retain];
- GPBSetRetainedObjectIvarWithFieldInternal(message, field, newValue,
- syntax);
- } else {
- // memcpy took care of the rest of the primitive fields if they were set.
- }
- } // for (field in descriptor->fields_)
- }
-
- - (id)copyWithZone:(NSZone *)zone {
- GPBDescriptor *descriptor = [self descriptor];
- GPBMessage *result = [[descriptor.messageClass allocWithZone:zone] init];
-
- [self copyFieldsInto:result zone:zone descriptor:descriptor];
- // Make immutable copies of the extra bits.
- result->unknownFields_ = [unknownFields_ copyWithZone:zone];
- result->extensionMap_ = CloneExtensionMap(extensionMap_, zone);
- return result;
- }
-
- - (void)clear {
- [self internalClear:YES];
- }
-
- - (void)internalClear:(BOOL)zeroStorage {
- GPBDescriptor *descriptor = [self descriptor];
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (GPBFieldIsMapOrArray(field)) {
- id arrayOrMap = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (arrayOrMap) {
- if (field.fieldType == GPBFieldTypeRepeated) {
- if (GPBFieldDataTypeIsObject(field)) {
- if ([arrayOrMap isKindOfClass:[GPBAutocreatedArray class]]) {
- GPBAutocreatedArray *autoArray = arrayOrMap;
- if (autoArray->_autocreator == self) {
- autoArray->_autocreator = nil;
- }
- }
- } else {
- // Type doesn't matter, it is a GPB*Array.
- GPBInt32Array *gpbArray = arrayOrMap;
- if (gpbArray->_autocreator == self) {
- gpbArray->_autocreator = nil;
- }
- }
- } else {
- if ((field.mapKeyDataType == GPBDataTypeString) &&
- GPBFieldDataTypeIsObject(field)) {
- if ([arrayOrMap isKindOfClass:[GPBAutocreatedDictionary class]]) {
- GPBAutocreatedDictionary *autoDict = arrayOrMap;
- if (autoDict->_autocreator == self) {
- autoDict->_autocreator = nil;
- }
- }
- } else {
- // Type doesn't matter, it is a GPB*Dictionary.
- GPBInt32Int32Dictionary *gpbDict = arrayOrMap;
- if (gpbDict->_autocreator == self) {
- gpbDict->_autocreator = nil;
- }
- }
- }
- [arrayOrMap release];
- }
- } else if (GPBFieldDataTypeIsMessage(field)) {
- GPBClearAutocreatedMessageIvarWithField(self, field);
- GPBMessage *value = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [value release];
- } else if (GPBFieldDataTypeIsObject(field) &&
- GPBGetHasIvarField(self, field)) {
- id value = GPBGetObjectIvarWithField(self, field);
- [value release];
- }
- }
-
- // GPBClearMessageAutocreator() expects that its caller has already been
- // removed from autocreatedExtensionMap_ so we set to nil first.
- NSArray *autocreatedValues = [autocreatedExtensionMap_ allValues];
- [autocreatedExtensionMap_ release];
- autocreatedExtensionMap_ = nil;
-
- // Since we're clearing all of our extensions, make sure that we clear the
- // autocreator on any that we've created so they no longer refer to us.
- for (GPBMessage *value in autocreatedValues) {
- NSCAssert(GPBWasMessageAutocreatedBy(value, self),
- @"Autocreated extension does not refer back to self.");
- GPBClearMessageAutocreator(value);
- }
-
- [extensionMap_ release];
- extensionMap_ = nil;
- [unknownFields_ release];
- unknownFields_ = nil;
-
- // Note that clearing does not affect autocreator_. If we are being cleared
- // because of a dealloc, then autocreator_ should be nil anyway. If we are
- // being cleared because someone explicitly clears us, we don't want to
- // sever our relationship with our autocreator.
-
- if (zeroStorage) {
- memset(messageStorage_, 0, descriptor->storageSize_);
- }
- }
-
- - (BOOL)isInitialized {
- GPBDescriptor *descriptor = [self descriptor];
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (field.isRequired) {
- if (!GPBGetHasIvarField(self, field)) {
- return NO;
- }
- }
- if (GPBFieldDataTypeIsMessage(field)) {
- GPBFieldType fieldType = field.fieldType;
- if (fieldType == GPBFieldTypeSingle) {
- if (field.isRequired) {
- GPBMessage *message = GPBGetMessageMessageField(self, field);
- if (!message.initialized) {
- return NO;
- }
- } else {
- NSAssert(field.isOptional,
- @"%@: Single message field %@ not required or optional?",
- [self class], field.name);
- if (GPBGetHasIvarField(self, field)) {
- GPBMessage *message = GPBGetMessageMessageField(self, field);
- if (!message.initialized) {
- return NO;
- }
- }
- }
- } else if (fieldType == GPBFieldTypeRepeated) {
- NSArray *array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- for (GPBMessage *message in array) {
- if (!message.initialized) {
- return NO;
- }
- }
- } else { // fieldType == GPBFieldTypeMap
- if (field.mapKeyDataType == GPBDataTypeString) {
- NSDictionary *map =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (map && !GPBDictionaryIsInitializedInternalHelper(map, field)) {
- return NO;
- }
- } else {
- // Real type is GPB*ObjectDictionary, exact type doesn't matter.
- GPBInt32ObjectDictionary *map =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (map && ![map isInitialized]) {
- return NO;
- }
- }
- }
- }
- }
-
- __block BOOL result = YES;
- [extensionMap_
- enumerateKeysAndObjectsUsingBlock:^(GPBExtensionDescriptor *extension,
- id obj,
- BOOL *stop) {
- if (GPBExtensionIsMessage(extension)) {
- if (extension.isRepeated) {
- for (GPBMessage *msg in obj) {
- if (!msg.initialized) {
- result = NO;
- *stop = YES;
- break;
- }
- }
- } else {
- GPBMessage *asMsg = obj;
- if (!asMsg.initialized) {
- result = NO;
- *stop = YES;
- }
- }
- }
- }];
- return result;
- }
-
- - (GPBDescriptor *)descriptor {
- return [[self class] descriptor];
- }
-
- - (NSData *)data {
- #ifdef DEBUG
- if (!self.initialized) {
- return nil;
- }
- #endif
- NSMutableData *data = [NSMutableData dataWithLength:[self serializedSize]];
- GPBCodedOutputStream *stream =
- [[GPBCodedOutputStream alloc] initWithData:data];
- @try {
- [self writeToCodedOutputStream:stream];
- }
- @catch (NSException *exception) {
- // This really shouldn't happen. The only way writeToCodedOutputStream:
- // could throw is if something in the library has a bug and the
- // serializedSize was wrong.
- #ifdef DEBUG
- NSLog(@"%@: Internal exception while building message data: %@",
- [self class], exception);
- #endif
- data = nil;
- }
- [stream release];
- return data;
- }
-
- - (NSData *)delimitedData {
- size_t serializedSize = [self serializedSize];
- size_t varintSize = GPBComputeRawVarint32SizeForInteger(serializedSize);
- NSMutableData *data =
- [NSMutableData dataWithLength:(serializedSize + varintSize)];
- GPBCodedOutputStream *stream =
- [[GPBCodedOutputStream alloc] initWithData:data];
- @try {
- [self writeDelimitedToCodedOutputStream:stream];
- }
- @catch (NSException *exception) {
- // This really shouldn't happen. The only way writeToCodedOutputStream:
- // could throw is if something in the library has a bug and the
- // serializedSize was wrong.
- #ifdef DEBUG
- NSLog(@"%@: Internal exception while building message delimitedData: %@",
- [self class], exception);
- #endif
- // If it happens, truncate.
- data.length = 0;
- }
- [stream release];
- return data;
- }
-
- - (void)writeToOutputStream:(NSOutputStream *)output {
- GPBCodedOutputStream *stream =
- [[GPBCodedOutputStream alloc] initWithOutputStream:output];
- [self writeToCodedOutputStream:stream];
- [stream release];
- }
-
- - (void)writeToCodedOutputStream:(GPBCodedOutputStream *)output {
- GPBDescriptor *descriptor = [self descriptor];
- NSArray *fieldsArray = descriptor->fields_;
- NSUInteger fieldCount = fieldsArray.count;
- const GPBExtensionRange *extensionRanges = descriptor.extensionRanges;
- NSUInteger extensionRangesCount = descriptor.extensionRangesCount;
- for (NSUInteger i = 0, j = 0; i < fieldCount || j < extensionRangesCount;) {
- if (i == fieldCount) {
- [self writeExtensionsToCodedOutputStream:output
- range:extensionRanges[j++]];
- } else if (j == extensionRangesCount ||
- GPBFieldNumber(fieldsArray[i]) < extensionRanges[j].start) {
- [self writeField:fieldsArray[i++] toCodedOutputStream:output];
- } else {
- [self writeExtensionsToCodedOutputStream:output
- range:extensionRanges[j++]];
- }
- }
- if (descriptor.isWireFormat) {
- [unknownFields_ writeAsMessageSetTo:output];
- } else {
- [unknownFields_ writeToCodedOutputStream:output];
- }
- }
-
- - (void)writeDelimitedToOutputStream:(NSOutputStream *)output {
- GPBCodedOutputStream *codedOutput =
- [[GPBCodedOutputStream alloc] initWithOutputStream:output];
- [self writeDelimitedToCodedOutputStream:codedOutput];
- [codedOutput release];
- }
-
- - (void)writeDelimitedToCodedOutputStream:(GPBCodedOutputStream *)output {
- [output writeRawVarintSizeTAs32:[self serializedSize]];
- [self writeToCodedOutputStream:output];
- }
-
- - (void)writeField:(GPBFieldDescriptor *)field
- toCodedOutputStream:(GPBCodedOutputStream *)output {
- GPBFieldType fieldType = field.fieldType;
- if (fieldType == GPBFieldTypeSingle) {
- BOOL has = GPBGetHasIvarField(self, field);
- if (!has) {
- return;
- }
- }
- uint32_t fieldNumber = GPBFieldNumber(field);
-
- //%PDDM-DEFINE FIELD_CASE(TYPE, REAL_TYPE)
- //%FIELD_CASE_FULL(TYPE, REAL_TYPE, REAL_TYPE)
- //%PDDM-DEFINE FIELD_CASE_FULL(TYPE, REAL_TYPE, ARRAY_TYPE)
- //% case GPBDataType##TYPE:
- //% if (fieldType == GPBFieldTypeRepeated) {
- //% uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- //% GPB##ARRAY_TYPE##Array *array =
- //% GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- //% [output write##TYPE##Array:fieldNumber values:array tag:tag];
- //% } else if (fieldType == GPBFieldTypeSingle) {
- //% [output write##TYPE:fieldNumber
- //% TYPE$S value:GPBGetMessage##REAL_TYPE##Field(self, field)];
- //% } else { // fieldType == GPBFieldTypeMap
- //% // Exact type here doesn't matter.
- //% GPBInt32##ARRAY_TYPE##Dictionary *dict =
- //% GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- //% [dict writeToCodedOutputStream:output asField:field];
- //% }
- //% break;
- //%
- //%PDDM-DEFINE FIELD_CASE2(TYPE)
- //% case GPBDataType##TYPE:
- //% if (fieldType == GPBFieldTypeRepeated) {
- //% NSArray *array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- //% [output write##TYPE##Array:fieldNumber values:array];
- //% } else if (fieldType == GPBFieldTypeSingle) {
- //% // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has check
- //% // again.
- //% [output write##TYPE:fieldNumber
- //% TYPE$S value:GPBGetObjectIvarWithFieldNoAutocreate(self, field)];
- //% } else { // fieldType == GPBFieldTypeMap
- //% // Exact type here doesn't matter.
- //% id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- //% GPBDataType mapKeyDataType = field.mapKeyDataType;
- //% if (mapKeyDataType == GPBDataTypeString) {
- //% GPBDictionaryWriteToStreamInternalHelper(output, dict, field);
- //% } else {
- //% [dict writeToCodedOutputStream:output asField:field];
- //% }
- //% }
- //% break;
- //%
-
- switch (GPBGetFieldDataType(field)) {
-
- //%PDDM-EXPAND FIELD_CASE(Bool, Bool)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeBool:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBBoolArray *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeBoolArray:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeBool:fieldNumber
- value:GPBGetMessageBoolField(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32BoolDictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(Fixed32, UInt32)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeFixed32:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBUInt32Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeFixed32Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeFixed32:fieldNumber
- value:GPBGetMessageUInt32Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32UInt32Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(SFixed32, Int32)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeSFixed32:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBInt32Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeSFixed32Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeSFixed32:fieldNumber
- value:GPBGetMessageInt32Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32Int32Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(Float, Float)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeFloat:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBFloatArray *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeFloatArray:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeFloat:fieldNumber
- value:GPBGetMessageFloatField(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32FloatDictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(Fixed64, UInt64)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeFixed64:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBUInt64Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeFixed64Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeFixed64:fieldNumber
- value:GPBGetMessageUInt64Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32UInt64Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(SFixed64, Int64)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeSFixed64:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBInt64Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeSFixed64Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeSFixed64:fieldNumber
- value:GPBGetMessageInt64Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32Int64Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(Double, Double)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeDouble:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBDoubleArray *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeDoubleArray:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeDouble:fieldNumber
- value:GPBGetMessageDoubleField(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32DoubleDictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(Int32, Int32)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeInt32:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBInt32Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeInt32Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeInt32:fieldNumber
- value:GPBGetMessageInt32Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32Int32Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(Int64, Int64)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeInt64:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBInt64Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeInt64Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeInt64:fieldNumber
- value:GPBGetMessageInt64Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32Int64Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(SInt32, Int32)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeSInt32:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBInt32Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeSInt32Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeSInt32:fieldNumber
- value:GPBGetMessageInt32Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32Int32Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(SInt64, Int64)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeSInt64:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBInt64Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeSInt64Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeSInt64:fieldNumber
- value:GPBGetMessageInt64Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32Int64Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(UInt32, UInt32)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeUInt32:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBUInt32Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeUInt32Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeUInt32:fieldNumber
- value:GPBGetMessageUInt32Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32UInt32Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE(UInt64, UInt64)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeUInt64:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBUInt64Array *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeUInt64Array:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeUInt64:fieldNumber
- value:GPBGetMessageUInt64Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32UInt64Dictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE_FULL(Enum, Int32, Enum)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeEnum:
- if (fieldType == GPBFieldTypeRepeated) {
- uint32_t tag = field.isPackable ? GPBFieldTag(field) : 0;
- GPBEnumArray *array =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeEnumArray:fieldNumber values:array tag:tag];
- } else if (fieldType == GPBFieldTypeSingle) {
- [output writeEnum:fieldNumber
- value:GPBGetMessageInt32Field(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- GPBInt32EnumDictionary *dict =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [dict writeToCodedOutputStream:output asField:field];
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE2(Bytes)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeBytes:
- if (fieldType == GPBFieldTypeRepeated) {
- NSArray *array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeBytesArray:fieldNumber values:array];
- } else if (fieldType == GPBFieldTypeSingle) {
- // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has check
- // again.
- [output writeBytes:fieldNumber
- value:GPBGetObjectIvarWithFieldNoAutocreate(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- GPBDataType mapKeyDataType = field.mapKeyDataType;
- if (mapKeyDataType == GPBDataTypeString) {
- GPBDictionaryWriteToStreamInternalHelper(output, dict, field);
- } else {
- [dict writeToCodedOutputStream:output asField:field];
- }
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE2(String)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeString:
- if (fieldType == GPBFieldTypeRepeated) {
- NSArray *array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeStringArray:fieldNumber values:array];
- } else if (fieldType == GPBFieldTypeSingle) {
- // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has check
- // again.
- [output writeString:fieldNumber
- value:GPBGetObjectIvarWithFieldNoAutocreate(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- GPBDataType mapKeyDataType = field.mapKeyDataType;
- if (mapKeyDataType == GPBDataTypeString) {
- GPBDictionaryWriteToStreamInternalHelper(output, dict, field);
- } else {
- [dict writeToCodedOutputStream:output asField:field];
- }
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE2(Message)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeMessage:
- if (fieldType == GPBFieldTypeRepeated) {
- NSArray *array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeMessageArray:fieldNumber values:array];
- } else if (fieldType == GPBFieldTypeSingle) {
- // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has check
- // again.
- [output writeMessage:fieldNumber
- value:GPBGetObjectIvarWithFieldNoAutocreate(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- GPBDataType mapKeyDataType = field.mapKeyDataType;
- if (mapKeyDataType == GPBDataTypeString) {
- GPBDictionaryWriteToStreamInternalHelper(output, dict, field);
- } else {
- [dict writeToCodedOutputStream:output asField:field];
- }
- }
- break;
-
- //%PDDM-EXPAND FIELD_CASE2(Group)
- // This block of code is generated, do not edit it directly.
-
- case GPBDataTypeGroup:
- if (fieldType == GPBFieldTypeRepeated) {
- NSArray *array = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [output writeGroupArray:fieldNumber values:array];
- } else if (fieldType == GPBFieldTypeSingle) {
- // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has check
- // again.
- [output writeGroup:fieldNumber
- value:GPBGetObjectIvarWithFieldNoAutocreate(self, field)];
- } else { // fieldType == GPBFieldTypeMap
- // Exact type here doesn't matter.
- id dict = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- GPBDataType mapKeyDataType = field.mapKeyDataType;
- if (mapKeyDataType == GPBDataTypeString) {
- GPBDictionaryWriteToStreamInternalHelper(output, dict, field);
- } else {
- [dict writeToCodedOutputStream:output asField:field];
- }
- }
- break;
-
- //%PDDM-EXPAND-END (18 expansions)
- }
- }
-
- #pragma mark - Extensions
-
- - (id)getExtension:(GPBExtensionDescriptor *)extension {
- CheckExtension(self, extension);
- id value = [extensionMap_ objectForKey:extension];
- if (value != nil) {
- return value;
- }
-
- // No default for repeated.
- if (extension.isRepeated) {
- return nil;
- }
- // Non messages get their default.
- if (!GPBExtensionIsMessage(extension)) {
- return extension.defaultValue;
- }
-
- // Check for an autocreated value.
- GPBPrepareReadOnlySemaphore(self);
- dispatch_semaphore_wait(readOnlySemaphore_, DISPATCH_TIME_FOREVER);
- value = [autocreatedExtensionMap_ objectForKey:extension];
- if (!value) {
- // Auto create the message extensions to match normal fields.
- value = CreateMessageWithAutocreatorForExtension(extension.msgClass, self,
- extension);
-
- if (autocreatedExtensionMap_ == nil) {
- autocreatedExtensionMap_ = [[NSMutableDictionary alloc] init];
- }
-
- // We can't simply call setExtension here because that would clear the new
- // value's autocreator.
- [autocreatedExtensionMap_ setObject:value forKey:extension];
- [value release];
- }
-
- dispatch_semaphore_signal(readOnlySemaphore_);
- return value;
- }
-
- - (id)getExistingExtension:(GPBExtensionDescriptor *)extension {
- // This is an internal method so we don't need to call CheckExtension().
- return [extensionMap_ objectForKey:extension];
- }
-
- - (BOOL)hasExtension:(GPBExtensionDescriptor *)extension {
- #if defined(DEBUG) && DEBUG
- CheckExtension(self, extension);
- #endif // DEBUG
- return nil != [extensionMap_ objectForKey:extension];
- }
-
- - (NSArray *)extensionsCurrentlySet {
- return [extensionMap_ allKeys];
- }
-
- - (void)writeExtensionsToCodedOutputStream:(GPBCodedOutputStream *)output
- range:(GPBExtensionRange)range {
- NSArray *sortedExtensions = [[extensionMap_ allKeys]
- sortedArrayUsingSelector:@selector(compareByFieldNumber:)];
- uint32_t start = range.start;
- uint32_t end = range.end;
- for (GPBExtensionDescriptor *extension in sortedExtensions) {
- uint32_t fieldNumber = extension.fieldNumber;
- if (fieldNumber >= start && fieldNumber < end) {
- id value = [extensionMap_ objectForKey:extension];
- GPBWriteExtensionValueToOutputStream(extension, value, output);
- }
- }
- }
-
- - (void)setExtension:(GPBExtensionDescriptor *)extension value:(id)value {
- if (!value) {
- [self clearExtension:extension];
- return;
- }
-
- CheckExtension(self, extension);
-
- if (extension.repeated) {
- [NSException raise:NSInvalidArgumentException
- format:@"Must call addExtension() for repeated types."];
- }
-
- if (extensionMap_ == nil) {
- extensionMap_ = [[NSMutableDictionary alloc] init];
- }
-
- // This pointless cast is for CLANG_WARN_NULLABLE_TO_NONNULL_CONVERSION.
- // Without it, the compiler complains we're passing an id nullable when
- // setObject:forKey: requires a id nonnull for the value. The check for
- // !value at the start of the method ensures it isn't nil, but the check
- // isn't smart enough to realize that.
- [extensionMap_ setObject:(id)value forKey:extension];
-
- GPBExtensionDescriptor *descriptor = extension;
-
- if (GPBExtensionIsMessage(descriptor) && !descriptor.isRepeated) {
- GPBMessage *autocreatedValue =
- [[autocreatedExtensionMap_ objectForKey:extension] retain];
- // Must remove from the map before calling GPBClearMessageAutocreator() so
- // that GPBClearMessageAutocreator() knows its safe to clear.
- [autocreatedExtensionMap_ removeObjectForKey:extension];
- GPBClearMessageAutocreator(autocreatedValue);
- [autocreatedValue release];
- }
-
- GPBBecomeVisibleToAutocreator(self);
- }
-
- - (void)addExtension:(GPBExtensionDescriptor *)extension value:(id)value {
- CheckExtension(self, extension);
-
- if (!extension.repeated) {
- [NSException raise:NSInvalidArgumentException
- format:@"Must call setExtension() for singular types."];
- }
-
- if (extensionMap_ == nil) {
- extensionMap_ = [[NSMutableDictionary alloc] init];
- }
- NSMutableArray *list = [extensionMap_ objectForKey:extension];
- if (list == nil) {
- list = [NSMutableArray array];
- [extensionMap_ setObject:list forKey:extension];
- }
-
- [list addObject:value];
- GPBBecomeVisibleToAutocreator(self);
- }
-
- - (void)setExtension:(GPBExtensionDescriptor *)extension
- index:(NSUInteger)idx
- value:(id)value {
- CheckExtension(self, extension);
-
- if (!extension.repeated) {
- [NSException raise:NSInvalidArgumentException
- format:@"Must call setExtension() for singular types."];
- }
-
- if (extensionMap_ == nil) {
- extensionMap_ = [[NSMutableDictionary alloc] init];
- }
-
- NSMutableArray *list = [extensionMap_ objectForKey:extension];
-
- [list replaceObjectAtIndex:idx withObject:value];
- GPBBecomeVisibleToAutocreator(self);
- }
-
- - (void)clearExtension:(GPBExtensionDescriptor *)extension {
- CheckExtension(self, extension);
-
- // Only become visible if there was actually a value to clear.
- if ([extensionMap_ objectForKey:extension]) {
- [extensionMap_ removeObjectForKey:extension];
- GPBBecomeVisibleToAutocreator(self);
- }
- }
-
- #pragma mark - mergeFrom
-
- - (void)mergeFromData:(NSData *)data
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry {
- GPBCodedInputStream *input = [[GPBCodedInputStream alloc] initWithData:data];
- [self mergeFromCodedInputStream:input extensionRegistry:extensionRegistry];
- [input checkLastTagWas:0];
- [input release];
- }
-
- #pragma mark - mergeDelimitedFrom
-
- - (void)mergeDelimitedFromCodedInputStream:(GPBCodedInputStream *)input
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry {
- GPBCodedInputStreamState *state = &input->state_;
- if (GPBCodedInputStreamIsAtEnd(state)) {
- return;
- }
- NSData *data = GPBCodedInputStreamReadRetainedBytesNoCopy(state);
- if (data == nil) {
- return;
- }
- [self mergeFromData:data extensionRegistry:extensionRegistry];
- [data release];
- }
-
- #pragma mark - Parse From Data Support
-
- + (instancetype)parseFromData:(NSData *)data error:(NSError **)errorPtr {
- return [self parseFromData:data extensionRegistry:nil error:errorPtr];
- }
-
- + (instancetype)parseFromData:(NSData *)data
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry
- error:(NSError **)errorPtr {
- return [[[self alloc] initWithData:data
- extensionRegistry:extensionRegistry
- error:errorPtr] autorelease];
- }
-
- + (instancetype)parseFromCodedInputStream:(GPBCodedInputStream *)input
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry
- error:(NSError **)errorPtr {
- return
- [[[self alloc] initWithCodedInputStream:input
- extensionRegistry:extensionRegistry
- error:errorPtr] autorelease];
- }
-
- #pragma mark - Parse Delimited From Data Support
-
- + (instancetype)parseDelimitedFromCodedInputStream:(GPBCodedInputStream *)input
- extensionRegistry:
- (GPBExtensionRegistry *)extensionRegistry
- error:(NSError **)errorPtr {
- GPBMessage *message = [[[self alloc] init] autorelease];
- @try {
- [message mergeDelimitedFromCodedInputStream:input
- extensionRegistry:extensionRegistry];
- if (errorPtr) {
- *errorPtr = nil;
- }
- }
- @catch (NSException *exception) {
- message = nil;
- if (errorPtr) {
- *errorPtr = ErrorFromException(exception);
- }
- }
- #ifdef DEBUG
- if (message && !message.initialized) {
- message = nil;
- if (errorPtr) {
- *errorPtr = MessageError(GPBMessageErrorCodeMissingRequiredField, nil);
- }
- }
- #endif
- return message;
- }
-
- #pragma mark - Unknown Field Support
-
- - (GPBUnknownFieldSet *)unknownFields {
- return unknownFields_;
- }
-
- - (void)setUnknownFields:(GPBUnknownFieldSet *)unknownFields {
- if (unknownFields != unknownFields_) {
- [unknownFields_ release];
- unknownFields_ = [unknownFields copy];
- GPBBecomeVisibleToAutocreator(self);
- }
- }
-
- - (void)parseMessageSet:(GPBCodedInputStream *)input
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry {
- uint32_t typeId = 0;
- NSData *rawBytes = nil;
- GPBExtensionDescriptor *extension = nil;
- GPBCodedInputStreamState *state = &input->state_;
- while (true) {
- uint32_t tag = GPBCodedInputStreamReadTag(state);
- if (tag == 0) {
- break;
- }
-
- if (tag == GPBWireFormatMessageSetTypeIdTag) {
- typeId = GPBCodedInputStreamReadUInt32(state);
- if (typeId != 0) {
- extension = [extensionRegistry extensionForDescriptor:[self descriptor]
- fieldNumber:typeId];
- }
- } else if (tag == GPBWireFormatMessageSetMessageTag) {
- rawBytes =
- [GPBCodedInputStreamReadRetainedBytesNoCopy(state) autorelease];
- } else {
- if (![input skipField:tag]) {
- break;
- }
- }
- }
-
- [input checkLastTagWas:GPBWireFormatMessageSetItemEndTag];
-
- if (rawBytes != nil && typeId != 0) {
- if (extension != nil) {
- GPBCodedInputStream *newInput =
- [[GPBCodedInputStream alloc] initWithData:rawBytes];
- GPBExtensionMergeFromInputStream(extension,
- extension.packable,
- newInput,
- extensionRegistry,
- self);
- [newInput release];
- } else {
- GPBUnknownFieldSet *unknownFields = GetOrMakeUnknownFields(self);
- // rawBytes was created via a NoCopy, so it can be reusing a
- // subrange of another NSData that might go out of scope as things
- // unwind, so a copy is needed to ensure what is saved in the
- // unknown fields stays valid.
- NSData *cloned = [NSData dataWithData:rawBytes];
- [unknownFields mergeMessageSetMessage:typeId data:cloned];
- }
- }
- }
-
- - (BOOL)parseUnknownField:(GPBCodedInputStream *)input
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry
- tag:(uint32_t)tag {
- GPBWireFormat wireType = GPBWireFormatGetTagWireType(tag);
- int32_t fieldNumber = GPBWireFormatGetTagFieldNumber(tag);
-
- GPBDescriptor *descriptor = [self descriptor];
- GPBExtensionDescriptor *extension =
- [extensionRegistry extensionForDescriptor:descriptor
- fieldNumber:fieldNumber];
- if (extension == nil) {
- if (descriptor.wireFormat && GPBWireFormatMessageSetItemTag == tag) {
- [self parseMessageSet:input extensionRegistry:extensionRegistry];
- return YES;
- }
- } else {
- if (extension.wireType == wireType) {
- GPBExtensionMergeFromInputStream(extension,
- extension.packable,
- input,
- extensionRegistry,
- self);
- return YES;
- }
- // Primitive, repeated types can be packed on unpacked on the wire, and are
- // parsed either way.
- if ([extension isRepeated] &&
- !GPBDataTypeIsObject(extension->description_->dataType) &&
- (extension.alternateWireType == wireType)) {
- GPBExtensionMergeFromInputStream(extension,
- !extension.packable,
- input,
- extensionRegistry,
- self);
- return YES;
- }
- }
- if ([GPBUnknownFieldSet isFieldTag:tag]) {
- GPBUnknownFieldSet *unknownFields = GetOrMakeUnknownFields(self);
- return [unknownFields mergeFieldFrom:tag input:input];
- } else {
- return NO;
- }
- }
-
- - (void)addUnknownMapEntry:(int32_t)fieldNum value:(NSData *)data {
- GPBUnknownFieldSet *unknownFields = GetOrMakeUnknownFields(self);
- [unknownFields addUnknownMapEntry:fieldNum value:data];
- }
-
- #pragma mark - MergeFromCodedInputStream Support
-
- static void MergeSingleFieldFromCodedInputStream(
- GPBMessage *self, GPBFieldDescriptor *field, GPBFileSyntax syntax,
- GPBCodedInputStream *input, GPBExtensionRegistry *extensionRegistry) {
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- switch (fieldDataType) {
- #define CASE_SINGLE_POD(NAME, TYPE, FUNC_TYPE) \
- case GPBDataType##NAME: { \
- TYPE val = GPBCodedInputStreamRead##NAME(&input->state_); \
- GPBSet##FUNC_TYPE##IvarWithFieldInternal(self, field, val, syntax); \
- break; \
- }
- #define CASE_SINGLE_OBJECT(NAME) \
- case GPBDataType##NAME: { \
- id val = GPBCodedInputStreamReadRetained##NAME(&input->state_); \
- GPBSetRetainedObjectIvarWithFieldInternal(self, field, val, syntax); \
- break; \
- }
- CASE_SINGLE_POD(Bool, BOOL, Bool)
- CASE_SINGLE_POD(Fixed32, uint32_t, UInt32)
- CASE_SINGLE_POD(SFixed32, int32_t, Int32)
- CASE_SINGLE_POD(Float, float, Float)
- CASE_SINGLE_POD(Fixed64, uint64_t, UInt64)
- CASE_SINGLE_POD(SFixed64, int64_t, Int64)
- CASE_SINGLE_POD(Double, double, Double)
- CASE_SINGLE_POD(Int32, int32_t, Int32)
- CASE_SINGLE_POD(Int64, int64_t, Int64)
- CASE_SINGLE_POD(SInt32, int32_t, Int32)
- CASE_SINGLE_POD(SInt64, int64_t, Int64)
- CASE_SINGLE_POD(UInt32, uint32_t, UInt32)
- CASE_SINGLE_POD(UInt64, uint64_t, UInt64)
- CASE_SINGLE_OBJECT(Bytes)
- CASE_SINGLE_OBJECT(String)
- #undef CASE_SINGLE_POD
- #undef CASE_SINGLE_OBJECT
-
- case GPBDataTypeMessage: {
- if (GPBGetHasIvarField(self, field)) {
- // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has
- // check again.
- GPBMessage *message =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [input readMessage:message extensionRegistry:extensionRegistry];
- } else {
- GPBMessage *message = [[field.msgClass alloc] init];
- [input readMessage:message extensionRegistry:extensionRegistry];
- GPBSetRetainedObjectIvarWithFieldInternal(self, field, message, syntax);
- }
- break;
- }
-
- case GPBDataTypeGroup: {
- if (GPBGetHasIvarField(self, field)) {
- // GPBGetObjectIvarWithFieldNoAutocreate() avoids doing the has
- // check again.
- GPBMessage *message =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [input readGroup:GPBFieldNumber(field)
- message:message
- extensionRegistry:extensionRegistry];
- } else {
- GPBMessage *message = [[field.msgClass alloc] init];
- [input readGroup:GPBFieldNumber(field)
- message:message
- extensionRegistry:extensionRegistry];
- GPBSetRetainedObjectIvarWithFieldInternal(self, field, message, syntax);
- }
- break;
- }
-
- case GPBDataTypeEnum: {
- int32_t val = GPBCodedInputStreamReadEnum(&input->state_);
- if (GPBHasPreservingUnknownEnumSemantics(syntax) ||
- [field isValidEnumValue:val]) {
- GPBSetInt32IvarWithFieldInternal(self, field, val, syntax);
- } else {
- GPBUnknownFieldSet *unknownFields = GetOrMakeUnknownFields(self);
- [unknownFields mergeVarintField:GPBFieldNumber(field) value:val];
- }
- }
- } // switch
- }
-
- static void MergeRepeatedPackedFieldFromCodedInputStream(
- GPBMessage *self, GPBFieldDescriptor *field, GPBFileSyntax syntax,
- GPBCodedInputStream *input) {
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- GPBCodedInputStreamState *state = &input->state_;
- id genericArray = GetOrCreateArrayIvarWithField(self, field, syntax);
- int32_t length = GPBCodedInputStreamReadInt32(state);
- size_t limit = GPBCodedInputStreamPushLimit(state, length);
- while (GPBCodedInputStreamBytesUntilLimit(state) > 0) {
- switch (fieldDataType) {
- #define CASE_REPEATED_PACKED_POD(NAME, TYPE, ARRAY_TYPE) \
- case GPBDataType##NAME: { \
- TYPE val = GPBCodedInputStreamRead##NAME(state); \
- [(GPB##ARRAY_TYPE##Array *)genericArray addValue:val]; \
- break; \
- }
- CASE_REPEATED_PACKED_POD(Bool, BOOL, Bool)
- CASE_REPEATED_PACKED_POD(Fixed32, uint32_t, UInt32)
- CASE_REPEATED_PACKED_POD(SFixed32, int32_t, Int32)
- CASE_REPEATED_PACKED_POD(Float, float, Float)
- CASE_REPEATED_PACKED_POD(Fixed64, uint64_t, UInt64)
- CASE_REPEATED_PACKED_POD(SFixed64, int64_t, Int64)
- CASE_REPEATED_PACKED_POD(Double, double, Double)
- CASE_REPEATED_PACKED_POD(Int32, int32_t, Int32)
- CASE_REPEATED_PACKED_POD(Int64, int64_t, Int64)
- CASE_REPEATED_PACKED_POD(SInt32, int32_t, Int32)
- CASE_REPEATED_PACKED_POD(SInt64, int64_t, Int64)
- CASE_REPEATED_PACKED_POD(UInt32, uint32_t, UInt32)
- CASE_REPEATED_PACKED_POD(UInt64, uint64_t, UInt64)
- #undef CASE_REPEATED_PACKED_POD
-
- case GPBDataTypeBytes:
- case GPBDataTypeString:
- case GPBDataTypeMessage:
- case GPBDataTypeGroup:
- NSCAssert(NO, @"Non primitive types can't be packed");
- break;
-
- case GPBDataTypeEnum: {
- int32_t val = GPBCodedInputStreamReadEnum(state);
- if (GPBHasPreservingUnknownEnumSemantics(syntax) ||
- [field isValidEnumValue:val]) {
- [(GPBEnumArray*)genericArray addRawValue:val];
- } else {
- GPBUnknownFieldSet *unknownFields = GetOrMakeUnknownFields(self);
- [unknownFields mergeVarintField:GPBFieldNumber(field) value:val];
- }
- break;
- }
- } // switch
- } // while(BytesUntilLimit() > 0)
- GPBCodedInputStreamPopLimit(state, limit);
- }
-
- static void MergeRepeatedNotPackedFieldFromCodedInputStream(
- GPBMessage *self, GPBFieldDescriptor *field, GPBFileSyntax syntax,
- GPBCodedInputStream *input, GPBExtensionRegistry *extensionRegistry) {
- GPBCodedInputStreamState *state = &input->state_;
- id genericArray = GetOrCreateArrayIvarWithField(self, field, syntax);
- switch (GPBGetFieldDataType(field)) {
- #define CASE_REPEATED_NOT_PACKED_POD(NAME, TYPE, ARRAY_TYPE) \
- case GPBDataType##NAME: { \
- TYPE val = GPBCodedInputStreamRead##NAME(state); \
- [(GPB##ARRAY_TYPE##Array *)genericArray addValue:val]; \
- break; \
- }
- #define CASE_REPEATED_NOT_PACKED_OBJECT(NAME) \
- case GPBDataType##NAME: { \
- id val = GPBCodedInputStreamReadRetained##NAME(state); \
- [(NSMutableArray*)genericArray addObject:val]; \
- [val release]; \
- break; \
- }
- CASE_REPEATED_NOT_PACKED_POD(Bool, BOOL, Bool)
- CASE_REPEATED_NOT_PACKED_POD(Fixed32, uint32_t, UInt32)
- CASE_REPEATED_NOT_PACKED_POD(SFixed32, int32_t, Int32)
- CASE_REPEATED_NOT_PACKED_POD(Float, float, Float)
- CASE_REPEATED_NOT_PACKED_POD(Fixed64, uint64_t, UInt64)
- CASE_REPEATED_NOT_PACKED_POD(SFixed64, int64_t, Int64)
- CASE_REPEATED_NOT_PACKED_POD(Double, double, Double)
- CASE_REPEATED_NOT_PACKED_POD(Int32, int32_t, Int32)
- CASE_REPEATED_NOT_PACKED_POD(Int64, int64_t, Int64)
- CASE_REPEATED_NOT_PACKED_POD(SInt32, int32_t, Int32)
- CASE_REPEATED_NOT_PACKED_POD(SInt64, int64_t, Int64)
- CASE_REPEATED_NOT_PACKED_POD(UInt32, uint32_t, UInt32)
- CASE_REPEATED_NOT_PACKED_POD(UInt64, uint64_t, UInt64)
- CASE_REPEATED_NOT_PACKED_OBJECT(Bytes)
- CASE_REPEATED_NOT_PACKED_OBJECT(String)
- #undef CASE_REPEATED_NOT_PACKED_POD
- #undef CASE_NOT_PACKED_OBJECT
- case GPBDataTypeMessage: {
- GPBMessage *message = [[field.msgClass alloc] init];
- [input readMessage:message extensionRegistry:extensionRegistry];
- [(NSMutableArray*)genericArray addObject:message];
- [message release];
- break;
- }
- case GPBDataTypeGroup: {
- GPBMessage *message = [[field.msgClass alloc] init];
- [input readGroup:GPBFieldNumber(field)
- message:message
- extensionRegistry:extensionRegistry];
- [(NSMutableArray*)genericArray addObject:message];
- [message release];
- break;
- }
- case GPBDataTypeEnum: {
- int32_t val = GPBCodedInputStreamReadEnum(state);
- if (GPBHasPreservingUnknownEnumSemantics(syntax) ||
- [field isValidEnumValue:val]) {
- [(GPBEnumArray*)genericArray addRawValue:val];
- } else {
- GPBUnknownFieldSet *unknownFields = GetOrMakeUnknownFields(self);
- [unknownFields mergeVarintField:GPBFieldNumber(field) value:val];
- }
- break;
- }
- } // switch
- }
-
- - (void)mergeFromCodedInputStream:(GPBCodedInputStream *)input
- extensionRegistry:(GPBExtensionRegistry *)extensionRegistry {
- GPBDescriptor *descriptor = [self descriptor];
- GPBFileSyntax syntax = descriptor.file.syntax;
- GPBCodedInputStreamState *state = &input->state_;
- uint32_t tag = 0;
- NSUInteger startingIndex = 0;
- NSArray *fields = descriptor->fields_;
- NSUInteger numFields = fields.count;
- while (YES) {
- BOOL merged = NO;
- tag = GPBCodedInputStreamReadTag(state);
- if (tag == 0) {
- break; // Reached end.
- }
- for (NSUInteger i = 0; i < numFields; ++i) {
- if (startingIndex >= numFields) startingIndex = 0;
- GPBFieldDescriptor *fieldDescriptor = fields[startingIndex];
- if (GPBFieldTag(fieldDescriptor) == tag) {
- GPBFieldType fieldType = fieldDescriptor.fieldType;
- if (fieldType == GPBFieldTypeSingle) {
- MergeSingleFieldFromCodedInputStream(self, fieldDescriptor, syntax,
- input, extensionRegistry);
- // Well formed protos will only have a single field once, advance
- // the starting index to the next field.
- startingIndex += 1;
- } else if (fieldType == GPBFieldTypeRepeated) {
- if (fieldDescriptor.isPackable) {
- MergeRepeatedPackedFieldFromCodedInputStream(
- self, fieldDescriptor, syntax, input);
- // Well formed protos will only have a repeated field that is
- // packed once, advance the starting index to the next field.
- startingIndex += 1;
- } else {
- MergeRepeatedNotPackedFieldFromCodedInputStream(
- self, fieldDescriptor, syntax, input, extensionRegistry);
- }
- } else { // fieldType == GPBFieldTypeMap
- // GPB*Dictionary or NSDictionary, exact type doesn't matter at this
- // point.
- id map = GetOrCreateMapIvarWithField(self, fieldDescriptor, syntax);
- [input readMapEntry:map
- extensionRegistry:extensionRegistry
- field:fieldDescriptor
- parentMessage:self];
- }
- merged = YES;
- break;
- } else {
- startingIndex += 1;
- }
- } // for(i < numFields)
-
- if (!merged && (tag != 0)) {
- // Primitive, repeated types can be packed on unpacked on the wire, and
- // are parsed either way. The above loop covered tag in the preferred
- // for, so this need to check the alternate form.
- for (NSUInteger i = 0; i < numFields; ++i) {
- if (startingIndex >= numFields) startingIndex = 0;
- GPBFieldDescriptor *fieldDescriptor = fields[startingIndex];
- if ((fieldDescriptor.fieldType == GPBFieldTypeRepeated) &&
- !GPBFieldDataTypeIsObject(fieldDescriptor) &&
- (GPBFieldAlternateTag(fieldDescriptor) == tag)) {
- BOOL alternateIsPacked = !fieldDescriptor.isPackable;
- if (alternateIsPacked) {
- MergeRepeatedPackedFieldFromCodedInputStream(
- self, fieldDescriptor, syntax, input);
- // Well formed protos will only have a repeated field that is
- // packed once, advance the starting index to the next field.
- startingIndex += 1;
- } else {
- MergeRepeatedNotPackedFieldFromCodedInputStream(
- self, fieldDescriptor, syntax, input, extensionRegistry);
- }
- merged = YES;
- break;
- } else {
- startingIndex += 1;
- }
- }
- }
-
- if (!merged) {
- if (tag == 0) {
- // zero signals EOF / limit reached
- return;
- } else {
- if (![self parseUnknownField:input
- extensionRegistry:extensionRegistry
- tag:tag]) {
- // it's an endgroup tag
- return;
- }
- }
- } // if(!merged)
-
- } // while(YES)
- }
-
- #pragma mark - MergeFrom Support
-
- - (void)mergeFrom:(GPBMessage *)other {
- Class selfClass = [self class];
- Class otherClass = [other class];
- if (!([selfClass isSubclassOfClass:otherClass] ||
- [otherClass isSubclassOfClass:selfClass])) {
- [NSException raise:NSInvalidArgumentException
- format:@"Classes must match %@ != %@", selfClass, otherClass];
- }
-
- // We assume something will be done and become visible.
- GPBBecomeVisibleToAutocreator(self);
-
- GPBDescriptor *descriptor = [[self class] descriptor];
- GPBFileSyntax syntax = descriptor.file.syntax;
-
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- GPBFieldType fieldType = field.fieldType;
- if (fieldType == GPBFieldTypeSingle) {
- int32_t hasIndex = GPBFieldHasIndex(field);
- uint32_t fieldNumber = GPBFieldNumber(field);
- if (!GPBGetHasIvar(other, hasIndex, fieldNumber)) {
- // Other doesn't have the field set, on to the next.
- continue;
- }
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- switch (fieldDataType) {
- case GPBDataTypeBool:
- GPBSetBoolIvarWithFieldInternal(
- self, field, GPBGetMessageBoolField(other, field), syntax);
- break;
- case GPBDataTypeSFixed32:
- case GPBDataTypeEnum:
- case GPBDataTypeInt32:
- case GPBDataTypeSInt32:
- GPBSetInt32IvarWithFieldInternal(
- self, field, GPBGetMessageInt32Field(other, field), syntax);
- break;
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- GPBSetUInt32IvarWithFieldInternal(
- self, field, GPBGetMessageUInt32Field(other, field), syntax);
- break;
- case GPBDataTypeSFixed64:
- case GPBDataTypeInt64:
- case GPBDataTypeSInt64:
- GPBSetInt64IvarWithFieldInternal(
- self, field, GPBGetMessageInt64Field(other, field), syntax);
- break;
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- GPBSetUInt64IvarWithFieldInternal(
- self, field, GPBGetMessageUInt64Field(other, field), syntax);
- break;
- case GPBDataTypeFloat:
- GPBSetFloatIvarWithFieldInternal(
- self, field, GPBGetMessageFloatField(other, field), syntax);
- break;
- case GPBDataTypeDouble:
- GPBSetDoubleIvarWithFieldInternal(
- self, field, GPBGetMessageDoubleField(other, field), syntax);
- break;
- case GPBDataTypeBytes:
- case GPBDataTypeString: {
- id otherVal = GPBGetObjectIvarWithFieldNoAutocreate(other, field);
- GPBSetObjectIvarWithFieldInternal(self, field, otherVal, syntax);
- break;
- }
- case GPBDataTypeMessage:
- case GPBDataTypeGroup: {
- id otherVal = GPBGetObjectIvarWithFieldNoAutocreate(other, field);
- if (GPBGetHasIvar(self, hasIndex, fieldNumber)) {
- GPBMessage *message =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- [message mergeFrom:otherVal];
- } else {
- GPBMessage *message = [otherVal copy];
- GPBSetRetainedObjectIvarWithFieldInternal(self, field, message,
- syntax);
- }
- break;
- }
- } // switch()
- } else if (fieldType == GPBFieldTypeRepeated) {
- // In the case of a list, they need to be appended, and there is no
- // _hasIvar to worry about setting.
- id otherArray =
- GPBGetObjectIvarWithFieldNoAutocreate(other, field);
- if (otherArray) {
- GPBDataType fieldDataType = field->description_->dataType;
- if (GPBDataTypeIsObject(fieldDataType)) {
- NSMutableArray *resultArray =
- GetOrCreateArrayIvarWithField(self, field, syntax);
- [resultArray addObjectsFromArray:otherArray];
- } else if (fieldDataType == GPBDataTypeEnum) {
- GPBEnumArray *resultArray =
- GetOrCreateArrayIvarWithField(self, field, syntax);
- [resultArray addRawValuesFromArray:otherArray];
- } else {
- // The array type doesn't matter, that all implment
- // -addValuesFromArray:.
- GPBInt32Array *resultArray =
- GetOrCreateArrayIvarWithField(self, field, syntax);
- [resultArray addValuesFromArray:otherArray];
- }
- }
- } else { // fieldType = GPBFieldTypeMap
- // In the case of a map, they need to be merged, and there is no
- // _hasIvar to worry about setting.
- id otherDict = GPBGetObjectIvarWithFieldNoAutocreate(other, field);
- if (otherDict) {
- GPBDataType keyDataType = field.mapKeyDataType;
- GPBDataType valueDataType = field->description_->dataType;
- if (GPBDataTypeIsObject(keyDataType) &&
- GPBDataTypeIsObject(valueDataType)) {
- NSMutableDictionary *resultDict =
- GetOrCreateMapIvarWithField(self, field, syntax);
- [resultDict addEntriesFromDictionary:otherDict];
- } else if (valueDataType == GPBDataTypeEnum) {
- // The exact type doesn't matter, just need to know it is a
- // GPB*EnumDictionary.
- GPBInt32EnumDictionary *resultDict =
- GetOrCreateMapIvarWithField(self, field, syntax);
- [resultDict addRawEntriesFromDictionary:otherDict];
- } else {
- // The exact type doesn't matter, they all implement
- // -addEntriesFromDictionary:.
- GPBInt32Int32Dictionary *resultDict =
- GetOrCreateMapIvarWithField(self, field, syntax);
- [resultDict addEntriesFromDictionary:otherDict];
- }
- }
- } // if (fieldType)..else if...else
- } // for(fields)
-
- // Unknown fields.
- if (!unknownFields_) {
- [self setUnknownFields:other.unknownFields];
- } else {
- [unknownFields_ mergeUnknownFields:other.unknownFields];
- }
-
- // Extensions
-
- if (other->extensionMap_.count == 0) {
- return;
- }
-
- if (extensionMap_ == nil) {
- extensionMap_ =
- CloneExtensionMap(other->extensionMap_, NSZoneFromPointer(self));
- } else {
- for (GPBExtensionDescriptor *extension in other->extensionMap_) {
- id otherValue = [other->extensionMap_ objectForKey:extension];
- id value = [extensionMap_ objectForKey:extension];
- BOOL isMessageExtension = GPBExtensionIsMessage(extension);
-
- if (extension.repeated) {
- NSMutableArray *list = value;
- if (list == nil) {
- list = [[NSMutableArray alloc] init];
- [extensionMap_ setObject:list forKey:extension];
- [list release];
- }
- if (isMessageExtension) {
- for (GPBMessage *otherListValue in otherValue) {
- GPBMessage *copiedValue = [otherListValue copy];
- [list addObject:copiedValue];
- [copiedValue release];
- }
- } else {
- [list addObjectsFromArray:otherValue];
- }
- } else {
- if (isMessageExtension) {
- if (value) {
- [(GPBMessage *)value mergeFrom:(GPBMessage *)otherValue];
- } else {
- GPBMessage *copiedValue = [otherValue copy];
- [extensionMap_ setObject:copiedValue forKey:extension];
- [copiedValue release];
- }
- } else {
- [extensionMap_ setObject:otherValue forKey:extension];
- }
- }
-
- if (isMessageExtension && !extension.isRepeated) {
- GPBMessage *autocreatedValue =
- [[autocreatedExtensionMap_ objectForKey:extension] retain];
- // Must remove from the map before calling GPBClearMessageAutocreator()
- // so that GPBClearMessageAutocreator() knows its safe to clear.
- [autocreatedExtensionMap_ removeObjectForKey:extension];
- GPBClearMessageAutocreator(autocreatedValue);
- [autocreatedValue release];
- }
- }
- }
- }
-
- #pragma mark - isEqual: & hash Support
-
- - (BOOL)isEqual:(id)other {
- if (other == self) {
- return YES;
- }
- if (![other isKindOfClass:[GPBMessage class]]) {
- return NO;
- }
- GPBMessage *otherMsg = other;
- GPBDescriptor *descriptor = [[self class] descriptor];
- if ([[otherMsg class] descriptor] != descriptor) {
- return NO;
- }
- uint8_t *selfStorage = (uint8_t *)messageStorage_;
- uint8_t *otherStorage = (uint8_t *)otherMsg->messageStorage_;
-
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (GPBFieldIsMapOrArray(field)) {
- // In the case of a list or map, there is no _hasIvar to worry about.
- // NOTE: These are NSArray/GPB*Array or NSDictionary/GPB*Dictionary, but
- // the type doesn't really matter as the objects all support -count and
- // -isEqual:.
- NSArray *resultMapOrArray =
- GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- NSArray *otherMapOrArray =
- GPBGetObjectIvarWithFieldNoAutocreate(other, field);
- // nil and empty are equal
- if (resultMapOrArray.count != 0 || otherMapOrArray.count != 0) {
- if (![resultMapOrArray isEqual:otherMapOrArray]) {
- return NO;
- }
- }
- } else { // Single field
- int32_t hasIndex = GPBFieldHasIndex(field);
- uint32_t fieldNum = GPBFieldNumber(field);
- BOOL selfHas = GPBGetHasIvar(self, hasIndex, fieldNum);
- BOOL otherHas = GPBGetHasIvar(other, hasIndex, fieldNum);
- if (selfHas != otherHas) {
- return NO; // Differing has values, not equal.
- }
- if (!selfHas) {
- // Same has values, was no, nothing else to check for this field.
- continue;
- }
- // Now compare the values.
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- size_t fieldOffset = field->description_->offset;
- switch (fieldDataType) {
- case GPBDataTypeBool: {
- // Bools are stored in has_bits to avoid needing explicit space in
- // the storage structure.
- // (the field number passed to the HasIvar helper doesn't really
- // matter since the offset is never negative)
- BOOL selfValue = GPBGetHasIvar(self, (int32_t)(fieldOffset), 0);
- BOOL otherValue = GPBGetHasIvar(other, (int32_t)(fieldOffset), 0);
- if (selfValue != otherValue) {
- return NO;
- }
- break;
- }
- case GPBDataTypeSFixed32:
- case GPBDataTypeInt32:
- case GPBDataTypeSInt32:
- case GPBDataTypeEnum:
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- case GPBDataTypeFloat: {
- GPBInternalCompileAssert(sizeof(float) == sizeof(uint32_t), float_not_32_bits);
- // These are all 32bit, signed/unsigned doesn't matter for equality.
- uint32_t *selfValPtr = (uint32_t *)&selfStorage[fieldOffset];
- uint32_t *otherValPtr = (uint32_t *)&otherStorage[fieldOffset];
- if (*selfValPtr != *otherValPtr) {
- return NO;
- }
- break;
- }
- case GPBDataTypeSFixed64:
- case GPBDataTypeInt64:
- case GPBDataTypeSInt64:
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- case GPBDataTypeDouble: {
- GPBInternalCompileAssert(sizeof(double) == sizeof(uint64_t), double_not_64_bits);
- // These are all 64bit, signed/unsigned doesn't matter for equality.
- uint64_t *selfValPtr = (uint64_t *)&selfStorage[fieldOffset];
- uint64_t *otherValPtr = (uint64_t *)&otherStorage[fieldOffset];
- if (*selfValPtr != *otherValPtr) {
- return NO;
- }
- break;
- }
- case GPBDataTypeBytes:
- case GPBDataTypeString:
- case GPBDataTypeMessage:
- case GPBDataTypeGroup: {
- // Type doesn't matter here, they all implement -isEqual:.
- id *selfValPtr = (id *)&selfStorage[fieldOffset];
- id *otherValPtr = (id *)&otherStorage[fieldOffset];
- if (![*selfValPtr isEqual:*otherValPtr]) {
- return NO;
- }
- break;
- }
- } // switch()
- } // if(mapOrArray)...else
- } // for(fields)
-
- // nil and empty are equal
- if (extensionMap_.count != 0 || otherMsg->extensionMap_.count != 0) {
- if (![extensionMap_ isEqual:otherMsg->extensionMap_]) {
- return NO;
- }
- }
-
- // nil and empty are equal
- GPBUnknownFieldSet *otherUnknowns = otherMsg->unknownFields_;
- if ([unknownFields_ countOfFields] != 0 ||
- [otherUnknowns countOfFields] != 0) {
- if (![unknownFields_ isEqual:otherUnknowns]) {
- return NO;
- }
- }
-
- return YES;
- }
-
- // It is very difficult to implement a generic hash for ProtoBuf messages that
- // will perform well. If you need hashing on your ProtoBufs (eg you are using
- // them as dictionary keys) you will probably want to implement a ProtoBuf
- // message specific hash as a category on your protobuf class. Do not make it a
- // category on GPBMessage as you will conflict with this hash, and will possibly
- // override hash for all generated protobufs. A good implementation of hash will
- // be really fast, so we would recommend only hashing protobufs that have an
- // identifier field of some kind that you can easily hash. If you implement
- // hash, we would strongly recommend overriding isEqual: in your category as
- // well, as the default implementation of isEqual: is extremely slow, and may
- // drastically affect performance in large sets.
- - (NSUInteger)hash {
- GPBDescriptor *descriptor = [[self class] descriptor];
- const NSUInteger prime = 19;
- uint8_t *storage = (uint8_t *)messageStorage_;
-
- // Start with the descriptor and then mix it with some instance info.
- // Hopefully that will give a spread based on classes and what fields are set.
- NSUInteger result = (NSUInteger)descriptor;
-
- for (GPBFieldDescriptor *field in descriptor->fields_) {
- if (GPBFieldIsMapOrArray(field)) {
- // Exact type doesn't matter, just check if there are any elements.
- NSArray *mapOrArray = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- NSUInteger count = mapOrArray.count;
- if (count) {
- // NSArray/NSDictionary use count, use the field number and the count.
- result = prime * result + GPBFieldNumber(field);
- result = prime * result + count;
- }
- } else if (GPBGetHasIvarField(self, field)) {
- // Just using the field number seemed simple/fast, but then a small
- // message class where all the same fields are always set (to different
- // things would end up all with the same hash, so pull in some data).
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- size_t fieldOffset = field->description_->offset;
- switch (fieldDataType) {
- case GPBDataTypeBool: {
- // Bools are stored in has_bits to avoid needing explicit space in
- // the storage structure.
- // (the field number passed to the HasIvar helper doesn't really
- // matter since the offset is never negative)
- BOOL value = GPBGetHasIvar(self, (int32_t)(fieldOffset), 0);
- result = prime * result + value;
- break;
- }
- case GPBDataTypeSFixed32:
- case GPBDataTypeInt32:
- case GPBDataTypeSInt32:
- case GPBDataTypeEnum:
- case GPBDataTypeFixed32:
- case GPBDataTypeUInt32:
- case GPBDataTypeFloat: {
- GPBInternalCompileAssert(sizeof(float) == sizeof(uint32_t), float_not_32_bits);
- // These are all 32bit, just mix it in.
- uint32_t *valPtr = (uint32_t *)&storage[fieldOffset];
- result = prime * result + *valPtr;
- break;
- }
- case GPBDataTypeSFixed64:
- case GPBDataTypeInt64:
- case GPBDataTypeSInt64:
- case GPBDataTypeFixed64:
- case GPBDataTypeUInt64:
- case GPBDataTypeDouble: {
- GPBInternalCompileAssert(sizeof(double) == sizeof(uint64_t), double_not_64_bits);
- // These are all 64bit, just mix what fits into an NSUInteger in.
- uint64_t *valPtr = (uint64_t *)&storage[fieldOffset];
- result = prime * result + (NSUInteger)(*valPtr);
- break;
- }
- case GPBDataTypeBytes:
- case GPBDataTypeString: {
- // Type doesn't matter here, they both implement -hash:.
- id *valPtr = (id *)&storage[fieldOffset];
- result = prime * result + [*valPtr hash];
- break;
- }
-
- case GPBDataTypeMessage:
- case GPBDataTypeGroup: {
- GPBMessage **valPtr = (GPBMessage **)&storage[fieldOffset];
- // Could call -hash on the sub message, but that could recurse pretty
- // deep; follow the lead of NSArray/NSDictionary and don't really
- // recurse for hash, instead use the field number and the descriptor
- // of the sub message. Yes, this could suck for a bunch of messages
- // where they all only differ in the sub messages, but if you are
- // using a message with sub messages for something that needs -hash,
- // odds are you are also copying them as keys, and that deep copy
- // will also suck.
- result = prime * result + GPBFieldNumber(field);
- result = prime * result + (NSUInteger)[[*valPtr class] descriptor];
- break;
- }
- } // switch()
- }
- }
-
- // Unknowns and extensions are not included.
-
- return result;
- }
-
- #pragma mark - Description Support
-
- - (NSString *)description {
- NSString *textFormat = GPBTextFormatForMessage(self, @" ");
- NSString *description = [NSString
- stringWithFormat:@"<%@ %p>: {\n%@}", [self class], self, textFormat];
- return description;
- }
-
- #if defined(DEBUG) && DEBUG
-
- // Xcode 5.1 added support for custom quick look info.
- // https://developer.apple.com/library/ios/documentation/IDEs/Conceptual/CustomClassDisplay_in_QuickLook/CH01-quick_look_for_custom_objects/CH01-quick_look_for_custom_objects.html#//apple_ref/doc/uid/TP40014001-CH2-SW1
- - (id)debugQuickLookObject {
- return GPBTextFormatForMessage(self, nil);
- }
-
- #endif // DEBUG
-
- #pragma mark - SerializedSize
-
- - (size_t)serializedSize {
- GPBDescriptor *descriptor = [[self class] descriptor];
- size_t result = 0;
-
- // Has check is done explicitly, so GPBGetObjectIvarWithFieldNoAutocreate()
- // avoids doing the has check again.
-
- // Fields.
- for (GPBFieldDescriptor *fieldDescriptor in descriptor->fields_) {
- GPBFieldType fieldType = fieldDescriptor.fieldType;
- GPBDataType fieldDataType = GPBGetFieldDataType(fieldDescriptor);
-
- // Single Fields
- if (fieldType == GPBFieldTypeSingle) {
- BOOL selfHas = GPBGetHasIvarField(self, fieldDescriptor);
- if (!selfHas) {
- continue; // Nothing to do.
- }
-
- uint32_t fieldNumber = GPBFieldNumber(fieldDescriptor);
-
- switch (fieldDataType) {
- #define CASE_SINGLE_POD(NAME, TYPE, FUNC_TYPE) \
- case GPBDataType##NAME: { \
- TYPE fieldVal = GPBGetMessage##FUNC_TYPE##Field(self, fieldDescriptor); \
- result += GPBCompute##NAME##Size(fieldNumber, fieldVal); \
- break; \
- }
- #define CASE_SINGLE_OBJECT(NAME) \
- case GPBDataType##NAME: { \
- id fieldVal = GPBGetObjectIvarWithFieldNoAutocreate(self, fieldDescriptor); \
- result += GPBCompute##NAME##Size(fieldNumber, fieldVal); \
- break; \
- }
- CASE_SINGLE_POD(Bool, BOOL, Bool)
- CASE_SINGLE_POD(Fixed32, uint32_t, UInt32)
- CASE_SINGLE_POD(SFixed32, int32_t, Int32)
- CASE_SINGLE_POD(Float, float, Float)
- CASE_SINGLE_POD(Fixed64, uint64_t, UInt64)
- CASE_SINGLE_POD(SFixed64, int64_t, Int64)
- CASE_SINGLE_POD(Double, double, Double)
- CASE_SINGLE_POD(Int32, int32_t, Int32)
- CASE_SINGLE_POD(Int64, int64_t, Int64)
- CASE_SINGLE_POD(SInt32, int32_t, Int32)
- CASE_SINGLE_POD(SInt64, int64_t, Int64)
- CASE_SINGLE_POD(UInt32, uint32_t, UInt32)
- CASE_SINGLE_POD(UInt64, uint64_t, UInt64)
- CASE_SINGLE_OBJECT(Bytes)
- CASE_SINGLE_OBJECT(String)
- CASE_SINGLE_OBJECT(Message)
- CASE_SINGLE_OBJECT(Group)
- CASE_SINGLE_POD(Enum, int32_t, Int32)
- #undef CASE_SINGLE_POD
- #undef CASE_SINGLE_OBJECT
- }
-
- // Repeated Fields
- } else if (fieldType == GPBFieldTypeRepeated) {
- id genericArray =
- GPBGetObjectIvarWithFieldNoAutocreate(self, fieldDescriptor);
- NSUInteger count = [genericArray count];
- if (count == 0) {
- continue; // Nothing to add.
- }
- __block size_t dataSize = 0;
-
- switch (fieldDataType) {
- #define CASE_REPEATED_POD(NAME, TYPE, ARRAY_TYPE) \
- CASE_REPEATED_POD_EXTRA(NAME, TYPE, ARRAY_TYPE, )
- #define CASE_REPEATED_POD_EXTRA(NAME, TYPE, ARRAY_TYPE, ARRAY_ACCESSOR_NAME) \
- case GPBDataType##NAME: { \
- GPB##ARRAY_TYPE##Array *array = genericArray; \
- [array enumerate##ARRAY_ACCESSOR_NAME##ValuesWithBlock:^(TYPE value, NSUInteger idx, BOOL *stop) { \
- _Pragma("unused(idx, stop)"); \
- dataSize += GPBCompute##NAME##SizeNoTag(value); \
- }]; \
- break; \
- }
- #define CASE_REPEATED_OBJECT(NAME) \
- case GPBDataType##NAME: { \
- for (id value in genericArray) { \
- dataSize += GPBCompute##NAME##SizeNoTag(value); \
- } \
- break; \
- }
- CASE_REPEATED_POD(Bool, BOOL, Bool)
- CASE_REPEATED_POD(Fixed32, uint32_t, UInt32)
- CASE_REPEATED_POD(SFixed32, int32_t, Int32)
- CASE_REPEATED_POD(Float, float, Float)
- CASE_REPEATED_POD(Fixed64, uint64_t, UInt64)
- CASE_REPEATED_POD(SFixed64, int64_t, Int64)
- CASE_REPEATED_POD(Double, double, Double)
- CASE_REPEATED_POD(Int32, int32_t, Int32)
- CASE_REPEATED_POD(Int64, int64_t, Int64)
- CASE_REPEATED_POD(SInt32, int32_t, Int32)
- CASE_REPEATED_POD(SInt64, int64_t, Int64)
- CASE_REPEATED_POD(UInt32, uint32_t, UInt32)
- CASE_REPEATED_POD(UInt64, uint64_t, UInt64)
- CASE_REPEATED_OBJECT(Bytes)
- CASE_REPEATED_OBJECT(String)
- CASE_REPEATED_OBJECT(Message)
- CASE_REPEATED_OBJECT(Group)
- CASE_REPEATED_POD_EXTRA(Enum, int32_t, Enum, Raw)
- #undef CASE_REPEATED_POD
- #undef CASE_REPEATED_POD_EXTRA
- #undef CASE_REPEATED_OBJECT
- } // switch
- result += dataSize;
- size_t tagSize = GPBComputeTagSize(GPBFieldNumber(fieldDescriptor));
- if (fieldDataType == GPBDataTypeGroup) {
- // Groups have both a start and an end tag.
- tagSize *= 2;
- }
- if (fieldDescriptor.isPackable) {
- result += tagSize;
- result += GPBComputeSizeTSizeAsInt32NoTag(dataSize);
- } else {
- result += count * tagSize;
- }
-
- // Map<> Fields
- } else { // fieldType == GPBFieldTypeMap
- if (GPBDataTypeIsObject(fieldDataType) &&
- (fieldDescriptor.mapKeyDataType == GPBDataTypeString)) {
- // If key type was string, then the map is an NSDictionary.
- NSDictionary *map =
- GPBGetObjectIvarWithFieldNoAutocreate(self, fieldDescriptor);
- if (map) {
- result += GPBDictionaryComputeSizeInternalHelper(map, fieldDescriptor);
- }
- } else {
- // Type will be GPB*GroupDictionary, exact type doesn't matter.
- GPBInt32Int32Dictionary *map =
- GPBGetObjectIvarWithFieldNoAutocreate(self, fieldDescriptor);
- result += [map computeSerializedSizeAsField:fieldDescriptor];
- }
- }
- } // for(fields)
-
- // Add any unknown fields.
- if (descriptor.wireFormat) {
- result += [unknownFields_ serializedSizeAsMessageSet];
- } else {
- result += [unknownFields_ serializedSize];
- }
-
- // Add any extensions.
- for (GPBExtensionDescriptor *extension in extensionMap_) {
- id value = [extensionMap_ objectForKey:extension];
- result += GPBComputeExtensionSerializedSizeIncludingTag(extension, value);
- }
-
- return result;
- }
-
- #pragma mark - Resolve Methods Support
-
- typedef struct ResolveIvarAccessorMethodResult {
- IMP impToAdd;
- SEL encodingSelector;
- } ResolveIvarAccessorMethodResult;
-
- // |field| can be __unsafe_unretained because they are created at startup
- // and are essentially global. No need to pay for retain/release when
- // they are captured in blocks.
- static void ResolveIvarGet(__unsafe_unretained GPBFieldDescriptor *field,
- ResolveIvarAccessorMethodResult *result) {
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- switch (fieldDataType) {
- #define CASE_GET(NAME, TYPE, TRUE_NAME) \
- case GPBDataType##NAME: { \
- result->impToAdd = imp_implementationWithBlock(^(id obj) { \
- return GPBGetMessage##TRUE_NAME##Field(obj, field); \
- }); \
- result->encodingSelector = @selector(get##NAME); \
- break; \
- }
- #define CASE_GET_OBJECT(NAME, TYPE, TRUE_NAME) \
- case GPBDataType##NAME: { \
- result->impToAdd = imp_implementationWithBlock(^(id obj) { \
- return GPBGetObjectIvarWithField(obj, field); \
- }); \
- result->encodingSelector = @selector(get##NAME); \
- break; \
- }
- CASE_GET(Bool, BOOL, Bool)
- CASE_GET(Fixed32, uint32_t, UInt32)
- CASE_GET(SFixed32, int32_t, Int32)
- CASE_GET(Float, float, Float)
- CASE_GET(Fixed64, uint64_t, UInt64)
- CASE_GET(SFixed64, int64_t, Int64)
- CASE_GET(Double, double, Double)
- CASE_GET(Int32, int32_t, Int32)
- CASE_GET(Int64, int64_t, Int64)
- CASE_GET(SInt32, int32_t, Int32)
- CASE_GET(SInt64, int64_t, Int64)
- CASE_GET(UInt32, uint32_t, UInt32)
- CASE_GET(UInt64, uint64_t, UInt64)
- CASE_GET_OBJECT(Bytes, id, Object)
- CASE_GET_OBJECT(String, id, Object)
- CASE_GET_OBJECT(Message, id, Object)
- CASE_GET_OBJECT(Group, id, Object)
- CASE_GET(Enum, int32_t, Enum)
- #undef CASE_GET
- }
- }
-
- // See comment about __unsafe_unretained on ResolveIvarGet.
- static void ResolveIvarSet(__unsafe_unretained GPBFieldDescriptor *field,
- GPBFileSyntax syntax,
- ResolveIvarAccessorMethodResult *result) {
- GPBDataType fieldDataType = GPBGetFieldDataType(field);
- switch (fieldDataType) {
- #define CASE_SET(NAME, TYPE, TRUE_NAME) \
- case GPBDataType##NAME: { \
- result->impToAdd = imp_implementationWithBlock(^(id obj, TYPE value) { \
- return GPBSet##TRUE_NAME##IvarWithFieldInternal(obj, field, value, syntax); \
- }); \
- result->encodingSelector = @selector(set##NAME:); \
- break; \
- }
- CASE_SET(Bool, BOOL, Bool)
- CASE_SET(Fixed32, uint32_t, UInt32)
- CASE_SET(SFixed32, int32_t, Int32)
- CASE_SET(Float, float, Float)
- CASE_SET(Fixed64, uint64_t, UInt64)
- CASE_SET(SFixed64, int64_t, Int64)
- CASE_SET(Double, double, Double)
- CASE_SET(Int32, int32_t, Int32)
- CASE_SET(Int64, int64_t, Int64)
- CASE_SET(SInt32, int32_t, Int32)
- CASE_SET(SInt64, int64_t, Int64)
- CASE_SET(UInt32, uint32_t, UInt32)
- CASE_SET(UInt64, uint64_t, UInt64)
- CASE_SET(Bytes, id, Object)
- CASE_SET(String, id, Object)
- CASE_SET(Message, id, Object)
- CASE_SET(Group, id, Object)
- CASE_SET(Enum, int32_t, Enum)
- #undef CASE_SET
- }
- }
-
- + (BOOL)resolveInstanceMethod:(SEL)sel {
- const GPBDescriptor *descriptor = [self descriptor];
- if (!descriptor) {
- return [super resolveInstanceMethod:sel];
- }
-
- // NOTE: hasOrCountSel_/setHasSel_ will be NULL if the field for the given
- // message should not have has support (done in GPBDescriptor.m), so there is
- // no need for checks here to see if has*/setHas* are allowed.
- ResolveIvarAccessorMethodResult result = {NULL, NULL};
-
- // See comment about __unsafe_unretained on ResolveIvarGet.
- for (__unsafe_unretained GPBFieldDescriptor *field in descriptor->fields_) {
- BOOL isMapOrArray = GPBFieldIsMapOrArray(field);
- if (!isMapOrArray) {
- // Single fields.
- if (sel == field->getSel_) {
- ResolveIvarGet(field, &result);
- break;
- } else if (sel == field->setSel_) {
- ResolveIvarSet(field, descriptor.file.syntax, &result);
- break;
- } else if (sel == field->hasOrCountSel_) {
- int32_t index = GPBFieldHasIndex(field);
- uint32_t fieldNum = GPBFieldNumber(field);
- result.impToAdd = imp_implementationWithBlock(^(id obj) {
- return GPBGetHasIvar(obj, index, fieldNum);
- });
- result.encodingSelector = @selector(getBool);
- break;
- } else if (sel == field->setHasSel_) {
- result.impToAdd = imp_implementationWithBlock(^(id obj, BOOL value) {
- if (value) {
- [NSException raise:NSInvalidArgumentException
- format:@"%@: %@ can only be set to NO (to clear field).",
- [obj class],
- NSStringFromSelector(field->setHasSel_)];
- }
- GPBClearMessageField(obj, field);
- });
- result.encodingSelector = @selector(setBool:);
- break;
- } else {
- GPBOneofDescriptor *oneof = field->containingOneof_;
- if (oneof && (sel == oneof->caseSel_)) {
- int32_t index = GPBFieldHasIndex(field);
- result.impToAdd = imp_implementationWithBlock(^(id obj) {
- return GPBGetHasOneof(obj, index);
- });
- result.encodingSelector = @selector(getEnum);
- break;
- }
- }
- } else {
- // map<>/repeated fields.
- if (sel == field->getSel_) {
- if (field.fieldType == GPBFieldTypeRepeated) {
- result.impToAdd = imp_implementationWithBlock(^(id obj) {
- return GetArrayIvarWithField(obj, field);
- });
- } else {
- result.impToAdd = imp_implementationWithBlock(^(id obj) {
- return GetMapIvarWithField(obj, field);
- });
- }
- result.encodingSelector = @selector(getArray);
- break;
- } else if (sel == field->setSel_) {
- // Local for syntax so the block can directly capture it and not the
- // full lookup.
- const GPBFileSyntax syntax = descriptor.file.syntax;
- result.impToAdd = imp_implementationWithBlock(^(id obj, id value) {
- return GPBSetObjectIvarWithFieldInternal(obj, field, value, syntax);
- });
- result.encodingSelector = @selector(setArray:);
- break;
- } else if (sel == field->hasOrCountSel_) {
- result.impToAdd = imp_implementationWithBlock(^(id obj) {
- // Type doesn't matter, all *Array and *Dictionary types support
- // -count.
- NSArray *arrayOrMap =
- GPBGetObjectIvarWithFieldNoAutocreate(obj, field);
- return [arrayOrMap count];
- });
- result.encodingSelector = @selector(getArrayCount);
- break;
- }
- }
- }
- if (result.impToAdd) {
- const char *encoding =
- GPBMessageEncodingForSelector(result.encodingSelector, YES);
- Class msgClass = descriptor.messageClass;
- BOOL methodAdded = class_addMethod(msgClass, sel, result.impToAdd, encoding);
- // class_addMethod() is documented as also failing if the method was already
- // added; so we check if the method is already there and return success so
- // the method dispatch will still happen. Why would it already be added?
- // Two threads could cause the same method to be bound at the same time,
- // but only one will actually bind it; the other still needs to return true
- // so things will dispatch.
- if (!methodAdded) {
- methodAdded = GPBClassHasSel(msgClass, sel);
- }
- return methodAdded;
- }
- return [super resolveInstanceMethod:sel];
- }
-
- + (BOOL)resolveClassMethod:(SEL)sel {
- // Extensions scoped to a Message and looked up via class methods.
- if (GPBResolveExtensionClassMethod([self descriptor].messageClass, sel)) {
- return YES;
- }
- return [super resolveClassMethod:sel];
- }
-
- #pragma mark - NSCoding Support
-
- + (BOOL)supportsSecureCoding {
- return YES;
- }
-
- - (instancetype)initWithCoder:(NSCoder *)aDecoder {
- self = [self init];
- if (self) {
- NSData *data =
- [aDecoder decodeObjectOfClass:[NSData class] forKey:kGPBDataCoderKey];
- if (data.length) {
- [self mergeFromData:data extensionRegistry:nil];
- }
- }
- return self;
- }
-
- - (void)encodeWithCoder:(NSCoder *)aCoder {
- NSData *data = [self data];
- if (data.length) {
- [aCoder encodeObject:data forKey:kGPBDataCoderKey];
- }
- }
-
- #pragma mark - KVC Support
-
- + (BOOL)accessInstanceVariablesDirectly {
- // Make sure KVC doesn't use instance variables.
- return NO;
- }
-
- @end
-
- #pragma mark - Messages from GPBUtilities.h but defined here for access to helpers.
-
- // Only exists for public api, no core code should use this.
- id GPBGetMessageRepeatedField(GPBMessage *self, GPBFieldDescriptor *field) {
- #if defined(DEBUG) && DEBUG
- if (field.fieldType != GPBFieldTypeRepeated) {
- [NSException raise:NSInvalidArgumentException
- format:@"%@.%@ is not a repeated field.",
- [self class], field.name];
- }
- #endif
- GPBDescriptor *descriptor = [[self class] descriptor];
- GPBFileSyntax syntax = descriptor.file.syntax;
- return GetOrCreateArrayIvarWithField(self, field, syntax);
- }
-
- // Only exists for public api, no core code should use this.
- id GPBGetMessageMapField(GPBMessage *self, GPBFieldDescriptor *field) {
- #if defined(DEBUG) && DEBUG
- if (field.fieldType != GPBFieldTypeMap) {
- [NSException raise:NSInvalidArgumentException
- format:@"%@.%@ is not a map<> field.",
- [self class], field.name];
- }
- #endif
- GPBDescriptor *descriptor = [[self class] descriptor];
- GPBFileSyntax syntax = descriptor.file.syntax;
- return GetOrCreateMapIvarWithField(self, field, syntax);
- }
-
- id GPBGetObjectIvarWithField(GPBMessage *self, GPBFieldDescriptor *field) {
- NSCAssert(!GPBFieldIsMapOrArray(field), @"Shouldn't get here");
- if (GPBGetHasIvarField(self, field)) {
- uint8_t *storage = (uint8_t *)self->messageStorage_;
- id *typePtr = (id *)&storage[field->description_->offset];
- return *typePtr;
- }
- // Not set...
-
- // Non messages (string/data), get their default.
- if (!GPBFieldDataTypeIsMessage(field)) {
- return field.defaultValue.valueMessage;
- }
-
- GPBPrepareReadOnlySemaphore(self);
- dispatch_semaphore_wait(self->readOnlySemaphore_, DISPATCH_TIME_FOREVER);
- GPBMessage *result = GPBGetObjectIvarWithFieldNoAutocreate(self, field);
- if (!result) {
- // For non repeated messages, create the object, set it and return it.
- // This object will not initially be visible via GPBGetHasIvar, so
- // we save its creator so it can become visible if it's mutated later.
- result = GPBCreateMessageWithAutocreator(field.msgClass, self, field);
- GPBSetAutocreatedRetainedObjectIvarWithField(self, field, result);
- }
- dispatch_semaphore_signal(self->readOnlySemaphore_);
- return result;
- }
-
- #pragma clang diagnostic pop
|