mirror of
https://github.com/intel/llvm.git
synced 2026-02-04 03:26:06 +08:00
Fix Objective-C metadata for properties from class extensions after r251874
After, properties from class extensions no longer show up in ObjCInterfaceDecl::properties(). Make ObjCCommonMac::EmitPropertyList() explicitly look for properties in class extensions before looking at direct properties. Also add a test that passes both with clang before r251874 and after this patch (but fails with r251874 and without this patch). llvm-svn: 254622
This commit is contained in:
@@ -2908,15 +2908,26 @@ llvm::Constant *CGObjCCommonMac::EmitPropertyList(Twine Name,
|
||||
const ObjCCommonTypesHelper &ObjCTypes) {
|
||||
SmallVector<llvm::Constant *, 16> Properties;
|
||||
llvm::SmallPtrSet<const IdentifierInfo*, 16> PropertySet;
|
||||
|
||||
auto AddProperty = [&](const ObjCPropertyDecl *PD) {
|
||||
llvm::Constant *Prop[] = {GetPropertyName(PD->getIdentifier()),
|
||||
GetPropertyTypeString(PD, Container)};
|
||||
Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy, Prop));
|
||||
};
|
||||
if (const ObjCInterfaceDecl *OID = dyn_cast<ObjCInterfaceDecl>(OCD))
|
||||
for (const ObjCCategoryDecl *ClassExt : OID->known_extensions())
|
||||
for (auto *PD : ClassExt->properties()) {
|
||||
PropertySet.insert(PD->getIdentifier());
|
||||
AddProperty(PD);
|
||||
}
|
||||
for (const auto *PD : OCD->properties()) {
|
||||
PropertySet.insert(PD->getIdentifier());
|
||||
llvm::Constant *Prop[] = {
|
||||
GetPropertyName(PD->getIdentifier()),
|
||||
GetPropertyTypeString(PD, Container)
|
||||
};
|
||||
Properties.push_back(llvm::ConstantStruct::get(ObjCTypes.PropertyTy,
|
||||
Prop));
|
||||
// Don't emit duplicate metadata for properties that were already in a
|
||||
// class extension.
|
||||
if (!PropertySet.insert(PD->getIdentifier()).second)
|
||||
continue;
|
||||
AddProperty(PD);
|
||||
}
|
||||
|
||||
if (const ObjCInterfaceDecl *OID = dyn_cast<ObjCInterfaceDecl>(OCD)) {
|
||||
for (const auto *P : OID->all_referenced_protocols())
|
||||
PushProtocolProperties(PropertySet, Properties, Container, P, ObjCTypes);
|
||||
|
||||
39
clang/test/CodeGenObjC/property-list-in-extension.m
Normal file
39
clang/test/CodeGenObjC/property-list-in-extension.m
Normal file
@@ -0,0 +1,39 @@
|
||||
// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm %s -o - | FileCheck %s
|
||||
|
||||
// Checks metadata for properties in a few cases.
|
||||
|
||||
|
||||
// Property from a class extension:
|
||||
@interface Foo
|
||||
@end
|
||||
|
||||
@interface Foo()
|
||||
@property int myprop;
|
||||
@end
|
||||
|
||||
@implementation Foo
|
||||
@synthesize myprop = _myprop;
|
||||
@end
|
||||
// Metadata for _myprop should be present, and PROP_LIST for Foo should have
|
||||
// only one entry.
|
||||
// CHECK: = private global [12 x i8] c"Ti,V_myprop\00",
|
||||
// CHECK: @"\01l_OBJC_$_PROP_LIST_Foo" = private global { i32, i32, [1 x %struct._prop_t] }
|
||||
|
||||
// Readonly property in interface made readwrite in a category:
|
||||
@interface FooRO
|
||||
@property (readonly) int evolvingprop;
|
||||
@end
|
||||
|
||||
@interface FooRO ()
|
||||
@property int evolvingprop;
|
||||
@end
|
||||
|
||||
@implementation FooRO
|
||||
@synthesize evolvingprop = _evolvingprop;
|
||||
@end
|
||||
// Metadata for _evolvingprop should be present, and PROP_LIST for FooRO should
|
||||
// still have only one entry, and the one entry should point to the version of
|
||||
// the property with a getter and setter.
|
||||
// CHECK: [[getter:@OBJC_PROP_NAME_ATTR[^ ]+]] = private global [13 x i8] c"evolvingprop\00"
|
||||
// CHECK: [[setter:@OBJC_PROP_NAME_ATTR[^ ]+]] = private global [18 x i8] c"Ti,V_evolvingprop\00",
|
||||
// CHECK: @"\01l_OBJC_$_PROP_LIST_FooRO" = private global { i32, i32, [1 x %struct._prop_t] }{{.*}}[[getter]]{{.*}}[[setter]]
|
||||
Reference in New Issue
Block a user