| 1 | // RUN: %clang_cc1 -emit-llvm -fblocks -debug-info-kind=limited -triple x86_64-apple-darwin10 -fobjc-dispatch-method=mixed -x objective-c < %s -o - | FileCheck %s |
| 2 | |
| 3 | // rdar://problem/9279956 |
| 4 | // Test that we generate the proper debug location for a captured self. |
| 5 | // The second half of this test is in llvm/tests/DebugInfo/debug-info-blocks.ll |
| 6 | |
| 7 | // CHECK: define {{.*}}_block_invoke |
| 8 | // CHECK: %[[BLOCK:.*]] = bitcast i8* %.block_descriptor to <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>*, !dbg |
| 9 | // CHECK-NEXT: store <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %[[BLOCK]], <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>** %[[ALLOCA:.*]], align |
| 10 | // CHECK-NEXT: call void @llvm.dbg.declare(metadata <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>** %[[ALLOCA]], metadata ![[SELF:[0-9]+]], metadata !{{.*}}) |
| 11 | // CHECK-NEXT: call void @llvm.dbg.declare(metadata %1** %d, metadata ![[D:[0-9]+]], metadata !{{.*}}) |
| 12 | |
| 13 | // Test that we do emit scope info for the helper functions, and that the |
| 14 | // parameters to these functions are marked as artificial (so the debugger |
| 15 | // doesn't accidentally step into the function). |
| 16 | // CHECK: define {{.*}} @__copy_helper_block_{{.*}}(i8*, i8*) |
| 17 | // CHECK-NOT: ret |
| 18 | // CHECK: call {{.*}}, !dbg ![[DBG_LINE:[0-9]+]] |
| 19 | // CHECK-NOT: ret |
| 20 | // CHECK: load {{.*}}, !dbg ![[COPY_LINE:[0-9]+]] |
| 21 | // CHECK: ret void, !dbg ![[COPY_LINE]] |
| 22 | // CHECK: define {{.*}} @__destroy_helper_block_{{.*}}(i8*) |
| 23 | // CHECK-NOT: ret |
| 24 | // CHECK: load {{.*}}, !dbg ![[DESTROY_LINE:[0-9]+]] |
| 25 | // CHECK: ret void, !dbg ![[DESTROY_LINE]] |
| 26 | |
| 27 | typedef unsigned int NSUInteger; |
| 28 | |
| 29 | @protocol NSObject |
| 30 | @end |
| 31 | |
| 32 | @interface NSObject <NSObject> |
| 33 | - (id)init; |
| 34 | + (id)alloc; |
| 35 | @end |
| 36 | |
| 37 | @interface NSDictionary : NSObject |
| 38 | - (NSUInteger)count; |
| 39 | @end |
| 40 | |
| 41 | @interface NSMutableDictionary : NSDictionary |
| 42 | @end |
| 43 | |
| 44 | @interface A : NSObject { |
| 45 | @public |
| 46 | int ivar; |
| 47 | } |
| 48 | @end |
| 49 | |
| 50 | static void run(void (^block)(void)) |
| 51 | { |
| 52 | block(); |
| 53 | } |
| 54 | |
| 55 | @implementation A |
| 56 | |
| 57 | - (id)init |
| 58 | { |
| 59 | if ((self = [super init])) { |
| 60 | // CHECK-DAG: [[DBG_LINE]] = !DILocation(line: 0, scope: ![[COPY_SP:[0-9]+]]) |
| 61 | // CHECK-DAG: [[COPY_LINE]] = !DILocation(line: [[@LINE+7]], scope: ![[COPY_SP:[0-9]+]]) |
| 62 | // CHECK-DAG: [[COPY_SP]] = distinct !DISubprogram(name: "__copy_helper_block_8_32o" |
| 63 | // CHECK-DAG: [[DESTROY_LINE]] = !DILocation(line: [[@LINE+5]], scope: ![[DESTROY_SP:[0-9]+]]) |
| 64 | // CHECK-DAG: [[DESTROY_SP]] = distinct !DISubprogram(name: "__destroy_helper_block_8_32o" |
| 65 | // CHECK-DAG: !DILocalVariable(arg: 1, scope: ![[COPY_SP]], {{.*}}, flags: DIFlagArtificial) |
| 66 | // CHECK-DAG: !DILocalVariable(arg: 2, scope: ![[COPY_SP]], {{.*}}, flags: DIFlagArtificial) |
| 67 | // CHECK-DAG: !DILocalVariable(arg: 1, scope: ![[DESTROY_SP]], {{.*}}, flags: DIFlagArtificial) |
| 68 | run(^{ |
| 69 | // CHECK-DAG: ![[SELF]] = !DILocalVariable(name: "self", scope:{{.*}}, line: [[@LINE+4]], |
| 70 | // CHECK-DAG: ![[D]] = !DILocalVariable(name: "d", scope:{{.*}}, line: [[@LINE+1]], |
| 71 | NSMutableDictionary *d = [[NSMutableDictionary alloc] init]; |
| 72 | ivar = 42 + (int)[d count]; |
| 73 | }); |
| 74 | } |
| 75 | return self; |
| 76 | } |
| 77 | |
| 78 | @end |
| 79 | |
| 80 | int main() |
| 81 | { |
| 82 | A *a = [[A alloc] init]; |
| 83 | return 0; |
| 84 | } |
| 85 | |