diff options
author | Fariborz Jahanian <fjahanian@apple.com> | 2012-03-27 18:41:05 +0000 |
---|---|---|
committer | Fariborz Jahanian <fjahanian@apple.com> | 2012-03-27 18:41:05 +0000 |
commit | e0335786c17a6ca45966f66094146e9d8ba2d54a (patch) | |
tree | 44c4c5c42e1f888337dcb635219e54754452aea0 | |
parent | f78c0f9aaabf32f5b5f8633b0066e611b24640ee (diff) |
objective-c modern translator: move all inithooks into a single array
// rdar://11124354
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@153526 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | lib/Rewrite/RewriteModernObjC.cpp | 71 | ||||
-rw-r--r-- | test/Rewriter/objc-modern-class-init-hooks.mm | 36 |
2 files changed, 83 insertions, 24 deletions
diff --git a/lib/Rewrite/RewriteModernObjC.cpp b/lib/Rewrite/RewriteModernObjC.cpp index 40b806d69d..83337df29d 100644 --- a/lib/Rewrite/RewriteModernObjC.cpp +++ b/lib/Rewrite/RewriteModernObjC.cpp @@ -387,10 +387,13 @@ namespace { StringRef prefix, StringRef ClassName, std::string &Result); virtual void RewriteObjCClassMetaData(ObjCImplementationDecl *IDecl, std::string &Result); + virtual void RewriteClassSetupInitHook(std::string &Result); + virtual void RewriteMetaDataIntoBuffer(std::string &Result); virtual void WriteImageInfo(std::string &Result); virtual void RewriteObjCCategoryImplDecl(ObjCCategoryImplDecl *CDecl, std::string &Result); + virtual void RewriteCategorySetupInitHook(std::string &Result); // Rewriting ivar virtual void RewriteIvarOffsetComputation(ObjCIvarDecl *ivar, @@ -5767,25 +5770,17 @@ static void Write_class_t(ASTContext *Context, std::string &Result, Result += "\tOBJC_CLASS_$_"; Result += CDecl->getNameAsString(); Result += ".cache = "; Result += "&_objc_empty_cache"; Result += ";\n"; Result += "}\n"; - - Result += "#pragma section(\".objc_inithooks$B\", long, read, write)\n"; - Result += "__declspec(allocate(\".objc_inithooks$B\")) "; - Result += "static void *OBJC_CLASS_SETUP2_$_"; - Result += CDecl->getNameAsString(); - Result += " = (void *)&OBJC_CLASS_SETUP_$_"; - Result += CDecl->getNameAsString(); - Result += ";\n\n"; } static void Write_category_t(RewriteModernObjC &RewriteObj, ASTContext *Context, std::string &Result, - StringRef CatName, + ObjCCategoryDecl *CatDecl, ObjCInterfaceDecl *ClassDecl, ArrayRef<ObjCMethodDecl *> InstanceMethods, ArrayRef<ObjCMethodDecl *> ClassMethods, ArrayRef<ObjCProtocolDecl *> RefedProtocols, ArrayRef<ObjCPropertyDecl *> ClassProperties) { - + StringRef CatName = CatDecl->getName(); StringRef ClassName = ClassDecl->getName(); // must declare an extern class object in case this class is not implemented // in this TU. @@ -5855,18 +5850,6 @@ static void Write_category_t(RewriteModernObjC &RewriteObj, ASTContext *Context, Result += CatName; Result += ".cls = "; Result += "&OBJC_CLASS_$_"; Result += ClassName; Result += ";\n}\n"; - - Result += "#pragma section(\".objc_inithooks$B\", long, read, write)\n"; - Result += "__declspec(allocate(\".objc_inithooks$B\")) "; - Result += "static void *OBJC_CATEGORY_SETUP2_$_"; - Result += ClassDecl->getNameAsString(); - Result += "_$_"; - Result += CatName; - Result += " = (void *)&OBJC_CATEGORY_SETUP_$_"; - Result += ClassDecl->getNameAsString(); - Result += "_$_"; - Result += CatName; - Result += ";\n\n"; } static void Write__extendedMethodTypes_initializer(RewriteModernObjC &RewriteObj, @@ -6385,6 +6368,22 @@ void RewriteModernObjC::RewriteObjCClassMetaData(ObjCImplementationDecl *IDecl, } +void RewriteModernObjC::RewriteClassSetupInitHook(std::string &Result) { + int ClsDefCount = ClassImplementation.size(); + if (!ClsDefCount) + return; + Result += "#pragma section(\".objc_inithooks$B\", long, read, write)\n"; + Result += "__declspec(allocate(\".objc_inithooks$B\")) "; + Result += "static void *OBJC_CLASS_SETUP[] = {\n"; + for (int i = 0; i < ClsDefCount; i++) { + ObjCImplementationDecl *IDecl = ClassImplementation[i]; + ObjCInterfaceDecl *CDecl = IDecl->getClassInterface(); + Result += "\t(void *)&OBJC_CLASS_SETUP_$_"; + Result += CDecl->getName(); Result += ",\n"; + } + Result += "};\n"; +} + void RewriteModernObjC::RewriteMetaDataIntoBuffer(std::string &Result) { int ClsDefCount = ClassImplementation.size(); int CatDefCount = CategoryImplementation.size(); @@ -6393,10 +6392,14 @@ void RewriteModernObjC::RewriteMetaDataIntoBuffer(std::string &Result) { for (int i = 0; i < ClsDefCount; i++) RewriteObjCClassMetaData(ClassImplementation[i], Result); + RewriteClassSetupInitHook(Result); + // For each implemented category, write out all its meta data. for (int i = 0; i < CatDefCount; i++) RewriteObjCCategoryImplDecl(CategoryImplementation[i], Result); + RewriteCategorySetupInitHook(Result); + if (ClsDefCount > 0) { if (LangOpts.MicrosoftExt) Result += "__declspec(allocate(\".objc_classlist$B\")) "; @@ -6550,7 +6553,7 @@ void RewriteModernObjC::RewriteObjCCategoryImplDecl(ObjCCategoryImplDecl *IDecl, FullCategoryName); Write_category_t(*this, Context, Result, - CDecl->getNameAsString(), + CDecl, ClassDecl, InstanceMethods, ClassMethods, @@ -6560,7 +6563,27 @@ void RewriteModernObjC::RewriteObjCCategoryImplDecl(ObjCCategoryImplDecl *IDecl, // Determine if this category is also "non-lazy". if (ImplementationIsNonLazy(IDecl)) DefinedNonLazyCategories.push_back(CDecl); - + +} + +void RewriteModernObjC::RewriteCategorySetupInitHook(std::string &Result) { + int CatDefCount = CategoryImplementation.size(); + if (!CatDefCount) + return; + Result += "#pragma section(\".objc_inithooks$B\", long, read, write)\n"; + Result += "__declspec(allocate(\".objc_inithooks$B\")) "; + Result += "static void *OBJC_CATEGORY_SETUP[] = {\n"; + for (int i = 0; i < CatDefCount; i++) { + ObjCCategoryImplDecl *IDecl = CategoryImplementation[i]; + ObjCCategoryDecl *CatDecl= IDecl->getCategoryDecl(); + ObjCInterfaceDecl *ClassDecl = IDecl->getClassInterface(); + Result += "\t(void *)&OBJC_CATEGORY_SETUP_$_"; + Result += ClassDecl->getName(); + Result += "_$_"; + Result += CatDecl->getName(); + Result += ",\n"; + } + Result += "};\n"; } // RewriteObjCMethodsMetaData - Rewrite methods metadata for instance or diff --git a/test/Rewriter/objc-modern-class-init-hooks.mm b/test/Rewriter/objc-modern-class-init-hooks.mm new file mode 100644 index 0000000000..c294c79e2e --- /dev/null +++ b/test/Rewriter/objc-modern-class-init-hooks.mm @@ -0,0 +1,36 @@ +// RUN: %clang_cc1 -E %s -o %t.mm +// RUN: %clang_cc1 -x objective-c++ -fblocks -fms-extensions -rewrite-objc %t.mm -o - | FileCheck %s +// rdar:// 11124354 + +@interface Root @end + +@interface Super : Root +@end + +@interface Sub : Super +@end + +@implementation Sub @end + +@implementation Root @end + +@interface Root(Cat) @end + +@interface Sub(Cat) @end + +@implementation Root(Cat) @end + +@implementation Sub(Cat) @end + + +// CHECK: #pragma section(".objc_inithooks$B", long, read, write) +// CHECK: __declspec(allocate(".objc_inithooks$B")) static void *OBJC_CLASS_SETUP[] = { +// CHECK: (void *)&OBJC_CLASS_SETUP_$_Sub, +// CHECK: (void *)&OBJC_CLASS_SETUP_$_Root, +// CHECK: }; + +// CHECK: #pragma section(".objc_inithooks$B", long, read, write) +// CHECK: __declspec(allocate(".objc_inithooks$B")) static void *OBJC_CATEGORY_SETUP[] = { +// CHECK: (void *)&OBJC_CATEGORY_SETUP_$_Root_$_Cat, +// CHECK: (void *)&OBJC_CATEGORY_SETUP_$_Sub_$_Cat, +// CHECK: }; |