2012-11-07 05:10:22 +08:00
|
|
|
// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -fobjc-gc -emit-llvm -o - %s | FileCheck %s
|
|
|
|
// RUN: %clang_cc1 -x objective-c++ -triple x86_64-apple-darwin10 -fobjc-gc -emit-llvm -o - %s | FileCheck %s
|
2009-05-04 05:05:10 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
Here is a handy command for looking at llvm-gcc's output:
|
|
|
|
llvm-gcc -m64 -fobjc-gc -emit-llvm -S -o - ivar-layout-64.m | \
|
|
|
|
grep 'OBJC_CLASS_NAME.* =.*global' | \
|
|
|
|
sed -e 's#, section.*# ...#' | \
|
2009-05-04 12:10:48 +08:00
|
|
|
sed -e 's#_[0-9]*"#_NNN#' | \
|
2009-05-04 05:05:10 +08:00
|
|
|
sort
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
@interface B @end
|
|
|
|
|
|
|
|
@interface A {
|
|
|
|
struct s0 {
|
|
|
|
int f0;
|
|
|
|
int f1;
|
|
|
|
} f0;
|
|
|
|
id f1;
|
|
|
|
__weak B *f2;
|
|
|
|
int f3 : 5;
|
|
|
|
struct s1 {
|
|
|
|
int *f0;
|
|
|
|
int *f1;
|
|
|
|
} f4[2][1];
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
2009-05-04 12:10:48 +08:00
|
|
|
@interface C : A
|
|
|
|
@property int p3;
|
|
|
|
@end
|
|
|
|
|
2014-11-06 21:30:38 +08:00
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"C\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\11p\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"!`\00"
|
2012-11-07 05:10:22 +08:00
|
|
|
|
|
|
|
|
2009-05-04 12:10:48 +08:00
|
|
|
@implementation C
|
|
|
|
@synthesize p3 = _p3;
|
|
|
|
@end
|
|
|
|
|
2009-05-04 05:05:10 +08:00
|
|
|
@interface A()
|
|
|
|
@property int p0;
|
|
|
|
@property (assign) __strong id p1;
|
|
|
|
@property (assign) __weak id p2;
|
|
|
|
@end
|
|
|
|
|
2014-11-06 21:30:38 +08:00
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"A\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\11q\10\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"!q\00"
|
2012-11-07 05:10:22 +08:00
|
|
|
|
2009-05-04 05:05:10 +08:00
|
|
|
@implementation A
|
2009-05-04 12:10:48 +08:00
|
|
|
@synthesize p0 = _p0;
|
|
|
|
@synthesize p1 = _p1;
|
|
|
|
@synthesize p2 = _p2;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface D : A
|
|
|
|
@property int p3;
|
|
|
|
@end
|
|
|
|
|
2014-11-06 21:30:38 +08:00
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"D\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\11p\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"!`\00"
|
2012-11-07 05:10:22 +08:00
|
|
|
|
2009-05-04 12:10:48 +08:00
|
|
|
@implementation D
|
|
|
|
@synthesize p3 = _p3;
|
2009-05-04 05:05:10 +08:00
|
|
|
@end
|
2009-05-05 05:46:27 +08:00
|
|
|
|
|
|
|
typedef unsigned short UInt16;
|
|
|
|
|
|
|
|
|
|
|
|
typedef signed char BOOL;
|
|
|
|
typedef unsigned int FSCatalogInfoBitmap;
|
|
|
|
|
|
|
|
@interface NSFileLocationComponent {
|
|
|
|
@private
|
|
|
|
|
|
|
|
id _specifierOrStandardizedPath;
|
|
|
|
BOOL _carbonCatalogInfoAndNameAreValid;
|
|
|
|
FSCatalogInfoBitmap _carbonCatalogInfoMask;
|
|
|
|
id _name;
|
|
|
|
id _containerComponent;
|
|
|
|
id _presentableName;
|
|
|
|
id _iconAsAttributedString;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
2014-11-06 21:30:38 +08:00
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"NSFileLocationComponent\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\01\14\00"
|
2012-11-07 05:10:22 +08:00
|
|
|
|
2009-05-05 05:46:27 +08:00
|
|
|
@implementation NSFileLocationComponent @end
|
|
|
|
|
2012-11-07 05:10:22 +08:00
|
|
|
@interface NSObject {
|
|
|
|
id isa;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface Foo : NSObject {
|
|
|
|
id ivar;
|
|
|
|
|
|
|
|
unsigned long bitfield :31;
|
|
|
|
unsigned long bitfield2 :1;
|
|
|
|
unsigned long bitfield3 :32;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
2014-11-06 21:30:38 +08:00
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"Foo\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\02\10\00"
|
2012-11-07 05:10:22 +08:00
|
|
|
|
|
|
|
@implementation Foo @end
|
Fix and stylize the emission of GC/ARC ivar and GC block layout strings.
Specifically, handle under-aligned object references (by explicitly
ignoring them, because this just isn't representable in the format;
yes, this means that GC silently ignores such references), descend
into anonymous structs and unions, stop classifying fields of
pointer-to-strong/weak type as strong/weak in ARC mode, and emit
skips to cover the entirety of block layouts in GC mode. As a
cleanup, extract this code into a helper class, avoid a number of
unnecessary copies and layout queries, generate skips implicitly
instead of explicitly tracking them, and clarify the bitmap-creation
logic.
llvm-svn: 250919
2015-10-22 02:06:47 +08:00
|
|
|
|
|
|
|
// GC layout strings aren't capable of expressing __strong ivars at
|
|
|
|
// non-word alignments.
|
|
|
|
struct __attribute__((packed)) PackedStruct {
|
|
|
|
char c;
|
|
|
|
__strong id x;
|
|
|
|
};
|
|
|
|
@interface Packed : NSObject {
|
|
|
|
struct PackedStruct _packed;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation Packed @end
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"Packed\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\01 \00"
|
|
|
|
// ' ' == 0x20
|
|
|
|
|
|
|
|
// Ensure that layout descends into anonymous unions and structs.
|
|
|
|
// Hilariously, anonymous unions and structs that appear directly as ivars
|
|
|
|
// are completely ignored by layout.
|
|
|
|
|
|
|
|
@interface AnonymousUnion : NSObject {
|
|
|
|
struct {
|
|
|
|
union {
|
|
|
|
id _object;
|
|
|
|
void *_ptr;
|
|
|
|
};
|
|
|
|
} a;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation AnonymousUnion @end
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"AnonymousUnion\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\02\00"
|
|
|
|
|
|
|
|
@interface AnonymousStruct : NSObject {
|
|
|
|
struct {
|
|
|
|
struct {
|
|
|
|
id _object;
|
|
|
|
__weak id _weakref;
|
|
|
|
};
|
|
|
|
} a;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation AnonymousStruct @end
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"AnonymousStruct\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"\02\10\00"
|
|
|
|
// CHECK: @OBJC_CLASS_NAME_{{.*}} = private global {{.*}} c"!\00"
|
|
|
|
// '!' == 0x21
|