• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Protocol Buffers - Google's data interchange format
2// Copyright 2008 Google Inc.  All rights reserved.
3//
4// Use of this source code is governed by a BSD-style
5// license that can be found in the LICENSE file or at
6// https://developers.google.com/open-source/licenses/bsd
7
8#import "GPBRootObject.h"
9#import "GPBRootObject_PackagePrivate.h"
10
11#import <CoreFoundation/CoreFoundation.h>
12#import <objc/runtime.h>
13#import <os/lock.h>
14
15#import "GPBDescriptor.h"
16#import "GPBExtensionRegistry.h"
17#import "GPBUtilities.h"
18#import "GPBUtilities_PackagePrivate.h"
19
20@interface GPBExtensionDescriptor (GPBRootObject)
21// Get singletonName as a c string.
22- (const char *)singletonNameC;
23@end
24
25// We need some object to conform to the MessageSignatureProtocol to make sure
26// the selectors in it are recorded in our Objective C runtime information.
27// GPBMessage is arguably the more "obvious" choice, but given that all messages
28// inherit from GPBMessage, conflicts seem likely, so we are using GPBRootObject
29// instead.
30@interface GPBRootObject () <GPBMessageSignatureProtocol>
31@end
32
33@implementation GPBRootObject
34
35// Taken from http://www.burtleburtle.net/bob/hash/doobs.html
36// Public Domain
37static uint32_t jenkins_one_at_a_time_hash(const char *key) {
38  uint32_t hash = 0;
39  for (uint32_t i = 0; key[i] != '\0'; ++i) {
40    hash += key[i];
41    hash += (hash << 10);
42    hash ^= (hash >> 6);
43  }
44  hash += (hash << 3);
45  hash ^= (hash >> 11);
46  hash += (hash << 15);
47  return hash;
48}
49
50// Key methods for our custom CFDictionary.
51// Note that the dictionary lasts for the lifetime of our app, so no need
52// to worry about deallocation. All of the items are added to it at
53// startup, and so the keys don't need to be retained/released.
54// Keys are NULL terminated char *.
55static const void *GPBRootExtensionKeyRetain(__unused CFAllocatorRef allocator, const void *value) {
56  return value;
57}
58
59static void GPBRootExtensionKeyRelease(__unused CFAllocatorRef allocator,
60                                       __unused const void *value) {}
61
62static CFStringRef GPBRootExtensionCopyKeyDescription(const void *value) {
63  const char *key = (const char *)value;
64  return CFStringCreateWithCString(kCFAllocatorDefault, key, kCFStringEncodingUTF8);
65}
66
67static Boolean GPBRootExtensionKeyEqual(const void *value1, const void *value2) {
68  const char *key1 = (const char *)value1;
69  const char *key2 = (const char *)value2;
70  return strcmp(key1, key2) == 0;
71}
72
73static CFHashCode GPBRootExtensionKeyHash(const void *value) {
74  const char *key = (const char *)value;
75  return jenkins_one_at_a_time_hash(key);
76}
77
78// Long ago, this was an OSSpinLock, but then it came to light that there were issues for that on
79// iOS:
80//   http://mjtsai.com/blog/2015/12/16/osspinlock-is-unsafe/
81//   https://lists.swift.org/pipermail/swift-dev/Week-of-Mon-20151214/000372.html
82// It was changed to a dispatch_semaphore_t, but that has potential for priority inversion issues.
83// The minOS versions are now high enough that os_unfair_lock can be used, and should provide
84// all the support we need. For more information in the concurrency/locking space see:
85//   https://gist.github.com/tclementdev/6af616354912b0347cdf6db159c37057
86//   https://developer.apple.com/library/archive/documentation/Performance/Conceptual/EnergyGuide-iOS/PrioritizeWorkWithQoS.html
87//   https://developer.apple.com/videos/play/wwdc2017/706/
88static os_unfair_lock gExtensionSingletonDictionaryLock = OS_UNFAIR_LOCK_INIT;
89static CFMutableDictionaryRef gExtensionSingletonDictionary = NULL;
90static GPBExtensionRegistry *gDefaultExtensionRegistry = NULL;
91
92+ (void)initialize {
93  // Ensure the global is started up.
94  if (!gExtensionSingletonDictionary) {
95    CFDictionaryKeyCallBacks keyCallBacks = {
96        // See description above for reason for using custom dictionary.
97        0,
98        GPBRootExtensionKeyRetain,
99        GPBRootExtensionKeyRelease,
100        GPBRootExtensionCopyKeyDescription,
101        GPBRootExtensionKeyEqual,
102        GPBRootExtensionKeyHash,
103    };
104    gExtensionSingletonDictionary = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, &keyCallBacks,
105                                                              &kCFTypeDictionaryValueCallBacks);
106    gDefaultExtensionRegistry = [[GPBExtensionRegistry alloc] init];
107  }
108
109  if ([self superclass] == [GPBRootObject class]) {
110    // This is here to start up all the per file "Root" subclasses.
111    // This must be done in initialize to enforce thread safety of start up of
112    // the protocol buffer library.
113    [self extensionRegistry];
114  }
115}
116
117+ (GPBExtensionRegistry *)extensionRegistry {
118  // Is overridden in all the subclasses that provide extensions to provide the
119  // per class one.
120  return gDefaultExtensionRegistry;
121}
122
123+ (void)globallyRegisterExtension:(GPBExtensionDescriptor *)field {
124  const char *key = [field singletonNameC];
125  os_unfair_lock_lock(&gExtensionSingletonDictionaryLock);
126  CFDictionarySetValue(gExtensionSingletonDictionary, key, field);
127  os_unfair_lock_unlock(&gExtensionSingletonDictionaryLock);
128}
129
130static id ExtensionForName(id self, SEL _cmd) {
131  // Really fast way of doing "classname_selName".
132  // This came up as a hotspot (creation of NSString *) when accessing a
133  // lot of extensions.
134  const char *selName = sel_getName(_cmd);
135  if (selName[0] == '_') {
136    return nil;  // Apple internal selector.
137  }
138  size_t selNameLen = 0;
139  while (1) {
140    char c = selName[selNameLen];
141    if (c == '\0') {  // String end.
142      break;
143    }
144    if (c == ':') {
145      return nil;  // Selector took an arg, not one of the runtime methods.
146    }
147    ++selNameLen;
148  }
149
150  const char *className = class_getName(self);
151  size_t classNameLen = strlen(className);
152  char key[classNameLen + selNameLen + 2];
153  memcpy(key, className, classNameLen);
154  key[classNameLen] = '_';
155  memcpy(&key[classNameLen + 1], selName, selNameLen);
156  key[classNameLen + 1 + selNameLen] = '\0';
157
158  // NOTE: Even though this method is called from another C function,
159  // gExtensionSingletonDictionaryLock and gExtensionSingletonDictionary
160  // will always be initialized. This is because this call flow is just to
161  // lookup the Extension, meaning the code is calling an Extension class
162  // message on a Message or Root class. This guarantees that the class was
163  // initialized and Message classes ensure their Root was also initialized.
164  NSAssert(gExtensionSingletonDictionary, @"Startup order broken!");
165
166  os_unfair_lock_lock(&gExtensionSingletonDictionaryLock);
167  id extension = (id)CFDictionaryGetValue(gExtensionSingletonDictionary, key);
168  // We can't remove the key from the dictionary here (as an optimization),
169  // two threads could have gone into +resolveClassMethod: for the same method,
170  // and ended up here; there's no way to ensure both return YES without letting
171  // both try to wire in the method.
172  os_unfair_lock_unlock(&gExtensionSingletonDictionaryLock);
173  return extension;
174}
175
176BOOL GPBResolveExtensionClassMethod(Class self, SEL sel) {
177  // Another option would be to register the extensions with the class at
178  // globallyRegisterExtension:
179  // Timing the two solutions, this solution turned out to be much faster
180  // and reduced startup time, and runtime memory.
181  // The advantage to globallyRegisterExtension is that it would reduce the
182  // size of the protos somewhat because the singletonNameC wouldn't need
183  // to include the class name. For a class with a lot of extensions it
184  // can add up. You could also significantly reduce the code complexity of this
185  // file.
186  id extension = ExtensionForName(self, sel);
187  if (extension != nil) {
188    const char *encoding = GPBMessageEncodingForSelector(@selector(getClassValue), NO);
189    Class metaClass = objc_getMetaClass(class_getName(self));
190    IMP imp = imp_implementationWithBlock(^(__unused id obj) {
191      return extension;
192    });
193    BOOL methodAdded = class_addMethod(metaClass, sel, imp, encoding);
194    // class_addMethod() is documented as also failing if the method was already
195    // added; so we check if the method is already there and return success so
196    // the method dispatch will still happen.  Why would it already be added?
197    // Two threads could cause the same method to be bound at the same time,
198    // but only one will actually bind it; the other still needs to return true
199    // so things will dispatch.
200    if (!methodAdded) {
201      methodAdded = GPBClassHasSel(metaClass, sel);
202    }
203    return methodAdded;
204  }
205  return NO;
206}
207
208+ (BOOL)resolveClassMethod:(SEL)sel {
209  if (GPBResolveExtensionClassMethod(self, sel)) {
210    return YES;
211  }
212  return [super resolveClassMethod:sel];
213}
214
215@end
216