Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1413)

Side by Side Diff: third_party/protobuf/objectivec/GPBRootObject.m

Issue 1842653006: Update //third_party/protobuf to version 3. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: merge Created 4 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2008 Google Inc. All rights reserved.
3 // https://developers.google.com/protocol-buffers/
4 //
5 // Redistribution and use in source and binary forms, with or without
6 // modification, are permitted provided that the following conditions are
7 // met:
8 //
9 // * Redistributions of source code must retain the above copyright
10 // notice, this list of conditions and the following disclaimer.
11 // * Redistributions in binary form must reproduce the above
12 // copyright notice, this list of conditions and the following disclaimer
13 // in the documentation and/or other materials provided with the
14 // distribution.
15 // * Neither the name of Google Inc. nor the names of its
16 // contributors may be used to endorse or promote products derived from
17 // this software without specific prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 #import "GPBRootObject_PackagePrivate.h"
32
33 #import <objc/runtime.h>
34
35 #import <CoreFoundation/CoreFoundation.h>
36
37 #import "GPBDescriptor.h"
38 #import "GPBExtensionRegistry.h"
39 #import "GPBUtilities_PackagePrivate.h"
40
41 @interface GPBExtensionDescriptor (GPBRootObject)
42 // Get singletonName as a c string.
43 - (const char *)singletonNameC;
44 @end
45
46 @implementation GPBRootObject
47
48 // Taken from http://www.burtleburtle.net/bob/hash/doobs.html
49 // Public Domain
50 static uint32_t jenkins_one_at_a_time_hash(const char *key) {
51 uint32_t hash = 0;
52 for (uint32_t i = 0; key[i] != '\0'; ++i) {
53 hash += key[i];
54 hash += (hash << 10);
55 hash ^= (hash >> 6);
56 }
57 hash += (hash << 3);
58 hash ^= (hash >> 11);
59 hash += (hash << 15);
60 return hash;
61 }
62
63 // Key methods for our custom CFDictionary.
64 // Note that the dictionary lasts for the lifetime of our app, so no need
65 // to worry about deallocation. All of the items are added to it at
66 // startup, and so the keys don't need to be retained/released.
67 // Keys are NULL terminated char *.
68 static const void *GPBRootExtensionKeyRetain(CFAllocatorRef allocator,
69 const void *value) {
70 #pragma unused(allocator)
71 return value;
72 }
73
74 static void GPBRootExtensionKeyRelease(CFAllocatorRef allocator,
75 const void *value) {
76 #pragma unused(allocator)
77 #pragma unused(value)
78 }
79
80 static CFStringRef GPBRootExtensionCopyKeyDescription(const void *value) {
81 const char *key = (const char *)value;
82 return CFStringCreateWithCString(kCFAllocatorDefault, key,
83 kCFStringEncodingUTF8);
84 }
85
86 static Boolean GPBRootExtensionKeyEqual(const void *value1,
87 const void *value2) {
88 const char *key1 = (const char *)value1;
89 const char *key2 = (const char *)value2;
90 return strcmp(key1, key2) == 0;
91 }
92
93 static CFHashCode GPBRootExtensionKeyHash(const void *value) {
94 const char *key = (const char *)value;
95 return jenkins_one_at_a_time_hash(key);
96 }
97
98 // NOTE: OSSpinLock may seem like a good fit here but Apple engineers have
99 // pointed out that they are vulnerable to live locking on iOS in cases of
100 // priority inversion:
101 // http://mjtsai.com/blog/2015/12/16/osspinlock-is-unsafe/
102 // https://lists.swift.org/pipermail/swift-dev/Week-of-Mon-20151214/000372.htm l
103 static dispatch_semaphore_t gExtensionSingletonDictionarySemaphore;
104 static CFMutableDictionaryRef gExtensionSingletonDictionary = NULL;
105 static GPBExtensionRegistry *gDefaultExtensionRegistry = NULL;
106
107 + (void)initialize {
108 // Ensure the global is started up.
109 if (!gExtensionSingletonDictionary) {
110 gExtensionSingletonDictionarySemaphore = dispatch_semaphore_create(1);
111 CFDictionaryKeyCallBacks keyCallBacks = {
112 // See description above for reason for using custom dictionary.
113 0,
114 GPBRootExtensionKeyRetain,
115 GPBRootExtensionKeyRelease,
116 GPBRootExtensionCopyKeyDescription,
117 GPBRootExtensionKeyEqual,
118 GPBRootExtensionKeyHash,
119 };
120 gExtensionSingletonDictionary =
121 CFDictionaryCreateMutable(kCFAllocatorDefault, 0, &keyCallBacks,
122 &kCFTypeDictionaryValueCallBacks);
123 gDefaultExtensionRegistry = [[GPBExtensionRegistry alloc] init];
124 }
125
126 if ([self superclass] == [GPBRootObject class]) {
127 // This is here to start up all the per file "Root" subclasses.
128 // This must be done in initialize to enforce thread safety of start up of
129 // the protocol buffer library.
130 [self extensionRegistry];
131 }
132 }
133
134 + (GPBExtensionRegistry *)extensionRegistry {
135 // Is overridden in all the subclasses that provide extensions to provide the
136 // per class one.
137 return gDefaultExtensionRegistry;
138 }
139
140 + (void)globallyRegisterExtension:(GPBExtensionDescriptor *)field {
141 const char *key = [field singletonNameC];
142 dispatch_semaphore_wait(gExtensionSingletonDictionarySemaphore,
143 DISPATCH_TIME_FOREVER);
144 CFDictionarySetValue(gExtensionSingletonDictionary, key, field);
145 dispatch_semaphore_signal(gExtensionSingletonDictionarySemaphore);
146 }
147
148 static id ExtensionForName(id self, SEL _cmd) {
149 // Really fast way of doing "classname_selName".
150 // This came up as a hotspot (creation of NSString *) when accessing a
151 // lot of extensions.
152 const char *selName = sel_getName(_cmd);
153 if (selName[0] == '_') {
154 return nil; // Apple internal selector.
155 }
156 size_t selNameLen = 0;
157 while (1) {
158 char c = selName[selNameLen];
159 if (c == '\0') { // String end.
160 break;
161 }
162 if (c == ':') {
163 return nil; // Selector took an arg, not one of the runtime methods.
164 }
165 ++selNameLen;
166 }
167
168 const char *className = class_getName(self);
169 size_t classNameLen = strlen(className);
170 char key[classNameLen + selNameLen + 2];
171 memcpy(key, className, classNameLen);
172 key[classNameLen] = '_';
173 memcpy(&key[classNameLen + 1], selName, selNameLen);
174 key[classNameLen + 1 + selNameLen] = '\0';
175
176 // NOTE: Even though this method is called from another C function,
177 // gExtensionSingletonDictionarySemaphore and gExtensionSingletonDictionary
178 // will always be initialized. This is because this call flow is just to
179 // lookup the Extension, meaning the code is calling an Extension class
180 // message on a Message or Root class. This guarantees that the class was
181 // initialized and Message classes ensure their Root was also initialized.
182 NSAssert(gExtensionSingletonDictionary, @"Startup order broken!");
183
184 dispatch_semaphore_wait(gExtensionSingletonDictionarySemaphore,
185 DISPATCH_TIME_FOREVER);
186 id extension = (id)CFDictionaryGetValue(gExtensionSingletonDictionary, key);
187 if (extension) {
188 // The method is getting wired in to the class, so no need to keep it in
189 // the dictionary.
190 CFDictionaryRemoveValue(gExtensionSingletonDictionary, key);
191 }
192 dispatch_semaphore_signal(gExtensionSingletonDictionarySemaphore);
193 return extension;
194 }
195
196 BOOL GPBResolveExtensionClassMethod(Class self, SEL sel) {
197 // Another option would be to register the extensions with the class at
198 // globallyRegisterExtension:
199 // Timing the two solutions, this solution turned out to be much faster
200 // and reduced startup time, and runtime memory.
201 // The advantage to globallyRegisterExtension is that it would reduce the
202 // size of the protos somewhat because the singletonNameC wouldn't need
203 // to include the class name. For a class with a lot of extensions it
204 // can add up. You could also significantly reduce the code complexity of this
205 // file.
206 id extension = ExtensionForName(self, sel);
207 if (extension != nil) {
208 const char *encoding =
209 GPBMessageEncodingForSelector(@selector(getClassValue), NO);
210 Class metaClass = objc_getMetaClass(class_getName(self));
211 IMP imp = imp_implementationWithBlock(^(id obj) {
212 #pragma unused(obj)
213 return extension;
214 });
215 if (class_addMethod(metaClass, sel, imp, encoding)) {
216 return YES;
217 }
218 }
219 return NO;
220 }
221
222
223 + (BOOL)resolveClassMethod:(SEL)sel {
224 if (GPBResolveExtensionClassMethod(self, sel)) {
225 return YES;
226 }
227 return [super resolveClassMethod:sel];
228 }
229
230 @end
OLDNEW
« no previous file with comments | « third_party/protobuf/objectivec/GPBRootObject.h ('k') | third_party/protobuf/objectivec/GPBRootObject_PackagePrivate.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698