GPBRootObject.m 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  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. #import "GPBRootObject_PackagePrivate.h"
  31. #import <objc/runtime.h>
  32. #import <libkern/OSAtomic.h>
  33. #import <CoreFoundation/CoreFoundation.h>
  34. #import "GPBDescriptor.h"
  35. #import "GPBExtensionRegistry.h"
  36. #import "GPBUtilities_PackagePrivate.h"
  37. @interface GPBExtensionDescriptor (GPBRootObject)
  38. // Get singletonName as a c string.
  39. - (const char *)singletonNameC;
  40. @end
  41. @implementation GPBRootObject
  42. // Taken from http://www.burtleburtle.net/bob/hash/doobs.html
  43. // Public Domain
  44. static uint32_t jenkins_one_at_a_time_hash(const char *key) {
  45. uint32_t hash = 0;
  46. for (uint32_t i = 0; key[i] != '\0'; ++i) {
  47. hash += key[i];
  48. hash += (hash << 10);
  49. hash ^= (hash >> 6);
  50. }
  51. hash += (hash << 3);
  52. hash ^= (hash >> 11);
  53. hash += (hash << 15);
  54. return hash;
  55. }
  56. // Key methods for our custom CFDictionary.
  57. // Note that the dictionary lasts for the lifetime of our app, so no need
  58. // to worry about deallocation. All of the items are added to it at
  59. // startup, and so the keys don't need to be retained/released.
  60. // Keys are NULL terminated char *.
  61. static const void *GPBRootExtensionKeyRetain(CFAllocatorRef allocator,
  62. const void *value) {
  63. #pragma unused(allocator)
  64. return value;
  65. }
  66. static void GPBRootExtensionKeyRelease(CFAllocatorRef allocator,
  67. const void *value) {
  68. #pragma unused(allocator)
  69. #pragma unused(value)
  70. }
  71. static CFStringRef GPBRootExtensionCopyKeyDescription(const void *value) {
  72. const char *key = (const char *)value;
  73. return CFStringCreateWithCString(kCFAllocatorDefault, key,
  74. kCFStringEncodingUTF8);
  75. }
  76. static Boolean GPBRootExtensionKeyEqual(const void *value1,
  77. const void *value2) {
  78. const char *key1 = (const char *)value1;
  79. const char *key2 = (const char *)value2;
  80. return strcmp(key1, key2) == 0;
  81. }
  82. static CFHashCode GPBRootExtensionKeyHash(const void *value) {
  83. const char *key = (const char *)value;
  84. return jenkins_one_at_a_time_hash(key);
  85. }
  86. static OSSpinLock gExtensionSingletonDictionaryLock_ = OS_SPINLOCK_INIT;
  87. static CFMutableDictionaryRef gExtensionSingletonDictionary = NULL;
  88. static GPBExtensionRegistry *gDefaultExtensionRegistry = NULL;
  89. + (void)initialize {
  90. // Ensure the global is started up.
  91. if (!gExtensionSingletonDictionary) {
  92. CFDictionaryKeyCallBacks keyCallBacks = {
  93. // See description above for reason for using custom dictionary.
  94. 0,
  95. GPBRootExtensionKeyRetain,
  96. GPBRootExtensionKeyRelease,
  97. GPBRootExtensionCopyKeyDescription,
  98. GPBRootExtensionKeyEqual,
  99. GPBRootExtensionKeyHash,
  100. };
  101. gExtensionSingletonDictionary =
  102. CFDictionaryCreateMutable(kCFAllocatorDefault, 0, &keyCallBacks,
  103. &kCFTypeDictionaryValueCallBacks);
  104. gDefaultExtensionRegistry = [[GPBExtensionRegistry alloc] init];
  105. }
  106. if ([self superclass] == [GPBRootObject class]) {
  107. // This is here to start up all the per file "Root" subclasses.
  108. // This must be done in initialize to enforce thread safety of start up of
  109. // the protocol buffer library.
  110. [self extensionRegistry];
  111. }
  112. }
  113. + (GPBExtensionRegistry *)extensionRegistry {
  114. // Is overridden in all the subclasses that provide extensions to provide the
  115. // per class one.
  116. return gDefaultExtensionRegistry;
  117. }
  118. + (void)globallyRegisterExtension:(GPBExtensionDescriptor *)field {
  119. const char *key = [field singletonNameC];
  120. OSSpinLockLock(&gExtensionSingletonDictionaryLock_);
  121. CFDictionarySetValue(gExtensionSingletonDictionary, key, field);
  122. OSSpinLockUnlock(&gExtensionSingletonDictionaryLock_);
  123. }
  124. static id ExtensionForName(id self, SEL _cmd) {
  125. // Really fast way of doing "classname_selName".
  126. // This came up as a hotspot (creation of NSString *) when accessing a
  127. // lot of extensions.
  128. const char *selName = sel_getName(_cmd);
  129. if (selName[0] == '_') {
  130. return nil; // Apple internal selector.
  131. }
  132. size_t selNameLen = 0;
  133. while (1) {
  134. char c = selName[selNameLen];
  135. if (c == '\0') { // String end.
  136. break;
  137. }
  138. if (c == ':') {
  139. return nil; // Selector took an arg, not one of the runtime methods.
  140. }
  141. ++selNameLen;
  142. }
  143. const char *className = class_getName(self);
  144. size_t classNameLen = strlen(className);
  145. char key[classNameLen + selNameLen + 2];
  146. memcpy(key, className, classNameLen);
  147. key[classNameLen] = '_';
  148. memcpy(&key[classNameLen + 1], selName, selNameLen);
  149. key[classNameLen + 1 + selNameLen] = '\0';
  150. OSSpinLockLock(&gExtensionSingletonDictionaryLock_);
  151. id extension = (id)CFDictionaryGetValue(gExtensionSingletonDictionary, key);
  152. if (extension) {
  153. // The method is getting wired in to the class, so no need to keep it in
  154. // the dictionary.
  155. CFDictionaryRemoveValue(gExtensionSingletonDictionary, key);
  156. }
  157. OSSpinLockUnlock(&gExtensionSingletonDictionaryLock_);
  158. return extension;
  159. }
  160. BOOL GPBResolveExtensionClassMethod(Class self, SEL sel) {
  161. // Another option would be to register the extensions with the class at
  162. // globallyRegisterExtension:
  163. // Timing the two solutions, this solution turned out to be much faster
  164. // and reduced startup time, and runtime memory.
  165. // The advantage to globallyRegisterExtension is that it would reduce the
  166. // size of the protos somewhat because the singletonNameC wouldn't need
  167. // to include the class name. For a class with a lot of extensions it
  168. // can add up. You could also significantly reduce the code complexity of this
  169. // file.
  170. id extension = ExtensionForName(self, sel);
  171. if (extension != nil) {
  172. const char *encoding =
  173. GPBMessageEncodingForSelector(@selector(getClassValue), NO);
  174. Class metaClass = objc_getMetaClass(class_getName(self));
  175. IMP imp = imp_implementationWithBlock(^(id obj) {
  176. #pragma unused(obj)
  177. return extension;
  178. });
  179. if (class_addMethod(metaClass, sel, imp, encoding)) {
  180. return YES;
  181. }
  182. }
  183. return NO;
  184. }
  185. + (BOOL)resolveClassMethod:(SEL)sel {
  186. if (GPBResolveExtensionClassMethod(self, sel)) {
  187. return YES;
  188. }
  189. return [super resolveClassMethod:sel];
  190. }
  191. @end