blob: 7b73abacdf5566ac2e2ade889a3de17b62c8acfd [file] [log] [blame]
David Blaikie488393f2017-05-12 01:13:45 +00001; RUN: llc -mtriple=x86_64-linux -split-dwarf-cross-cu-references -split-dwarf-file=foo.dwo -filetype=obj -o %t < %s
2; RUN: llvm-objdump -r %t | FileCheck %s
Adrian Prantl3ae35eb2017-09-13 22:09:01 +00003; RUN: llvm-dwarfdump -v -debug-info %t | FileCheck --check-prefix=ALL --check-prefix=INFO --check-prefix=DWO --check-prefix=CROSS %s
Adrian Prantl16aa4cf2017-09-11 23:05:20 +00004; RUN: llvm-dwarfdump -v -debug-info %t | FileCheck --check-prefix=ALL --check-prefix=INFO %s
David Blaikie488393f2017-05-12 01:13:45 +00005
6; RUN: llc -mtriple=x86_64-linux -split-dwarf-file=foo.dwo -filetype=obj -o %t < %s
7; RUN: llvm-objdump -r %t | FileCheck %s
Adrian Prantl3ae35eb2017-09-13 22:09:01 +00008; RUN: llvm-dwarfdump -v -debug-info %t | FileCheck --check-prefix=ALL --check-prefix=DWO --check-prefix=NOCROSS %s
Adrian Prantl16aa4cf2017-09-11 23:05:20 +00009; RUN: llvm-dwarfdump -v -debug-info %t | FileCheck --check-prefix=ALL --check-prefix=INFO %s
David Blaikie488393f2017-05-12 01:13:45 +000010
11; Testing cross-CU references for types, subprograms, and variables
12; Built from code something like this:
13; foo.cpp:
14; struct t1 { int i; };
15; void f();
16; __attribute__((always_inline)) void f1(t1 t) {
17; f();
18; }
19; void foo(t1 t) {
20; f1(t);
21; }
22; bar.cpp:
23; struct t1 { int i; };
24; void f1(t1);
25; void bar(t1 t) {
26; f1(t);
27; }
28; $ clang++-tot -emit-llvm -S {foo,bar}.cpp -g
29; $ llvm-link-tot {foo,bar}.ll -S -o foobar.ll
30; $ clang++-tot -emit-llvm foobar.ll -o foobar.opt.ll -S -c
31;
32; Then manually removing the original f1 definition, to simplify the DWARF a bit
33; (so it only has the inlined definitions, no concrete definition)
34
35; Check that:
36; * no relocations are emitted for the debug_info.dwo section no matter what
37; * one debug_info->debug_info relocation in debug_info no matter what (for
38; split dwarf inlining)
39; * debug_info uses relocations and ref_addr no matter what
40; * debug_info.dwo uses relocations for types as well as abstract subprograms
41; and variables when -split-dwarf-cross-cu-references is used
42; * debug_info.dwo contains duplicate types, abstract subprograms and abstract
43; variables otherwise to avoid the need for cross-cu references
David Blaikie85366ac2017-04-22 07:53:44 +000044
Adrian Prantl3ae35eb2017-09-13 22:09:01 +000045; DWO: .debug_info.dwo contents:
David Blaikie5477b972017-04-22 08:17:39 +000046; CHECK-NOT: .rel{{a?}}.debug_info.dwo
George Rimar5c922f62019-05-07 13:14:18 +000047; CHECK: RELOCATION RECORDS FOR [.debug_info]:
David Blaikie85366ac2017-04-22 07:53:44 +000048; CHECK-NOT: RELOCATION RECORDS
David Blaikie488393f2017-05-12 01:13:45 +000049; Expect one relocation in debug_info, from the inlined f1 in foo to its
50; abstract origin in bar
David Blaikie85366ac2017-04-22 07:53:44 +000051; CHECK: R_X86_64_32 .debug_info
David Blaikie488393f2017-05-12 01:13:45 +000052; CHECK-NOT: RELOCATION RECORDS
David Blaikie85366ac2017-04-22 07:53:44 +000053; CHECK-NOT: .debug_info
54; CHECK: RELOCATION RECORDS
David Blaikie5477b972017-04-22 08:17:39 +000055; CHECK-NOT: .rel{{a?}}.debug_info.dwo
David Blaikie85366ac2017-04-22 07:53:44 +000056
David Blaikie488393f2017-05-12 01:13:45 +000057; ALL: Compile Unit
58; ALL: DW_TAG_compile_unit
59; DWO: DW_AT_name {{.*}} "foo.cpp"
60; ALL: 0x[[F1:.*]]: DW_TAG_subprogram
61; ALL: DW_AT_name {{.*}} "f1"
62; DWO: 0x[[F1T:.*]]: DW_TAG_formal_parameter
63; DWO: DW_AT_name {{.*}} "t"
64; DWO: DW_AT_type [DW_FORM_ref4] {{.*}}{0x[[T1:.*]]}
65; DWO: NULL
66; DWO: 0x[[T1]]: DW_TAG_structure_type
67; DWO: DW_AT_name {{.*}} "t1"
68; ALL: DW_TAG_subprogram
69; ALL: DW_AT_name {{.*}} "foo"
70; DWO: DW_TAG_formal_parameter
71; DWO: DW_AT_name {{.*}} "t"
72; DWO: DW_AT_type [DW_FORM_ref4] {{.*}}{0x[[T1]]}
73; ALL: DW_TAG_inlined_subroutine
74; ALL: DW_AT_abstract_origin [DW_FORM_ref4] {{.*}}{0x[[F1]]}
75; DWO: DW_TAG_formal_parameter
76; DWO: DW_AT_abstract_origin [DW_FORM_ref4] {{.*}}{0x[[F1T]]}
David Blaikie85366ac2017-04-22 07:53:44 +000077
David Blaikie488393f2017-05-12 01:13:45 +000078; ALL: Compile Unit
79; ALL: DW_TAG_compile_unit
80; DWO: DW_AT_name {{.*}} "bar.cpp"
81; NOCROSS: 0x[[BAR_F1:.*]]: DW_TAG_subprogram
82; NOCROSS: DW_AT_name {{.*}} "f1"
83; NOCROSS: 0x[[BAR_F1T:.*]]: DW_TAG_formal_parameter
84; NOCROSS: DW_AT_name {{.*}} "t"
85; NOCROSS: DW_AT_type [DW_FORM_ref4] {{.*}}{0x[[BAR_T1:.*]]}
86; NOCROSS: NULL
87; NOCROSS: 0x[[BAR_T1]]: DW_TAG_structure_type
88; NOCROSS: DW_AT_name {{.*}} "t1"
89; ALL: DW_TAG_subprogram
90; ALL: DW_AT_name {{.*}} "bar"
91; DWO: DW_TAG_formal_parameter
92; DWO: DW_AT_name {{.*}} "t"
93; CROSS: DW_AT_type [DW_FORM_ref_addr] (0x00000000[[T1]]
94; NOCROSS: DW_AT_type [DW_FORM_ref4] {{.*}}{0x[[BAR_T1]]}
95; ALL: DW_TAG_inlined_subroutine
96; INFO: DW_AT_abstract_origin [DW_FORM_ref_addr] (0x00000000[[F1]]
97; NOCROSS: DW_AT_abstract_origin [DW_FORM_ref4] {{.*}}{0x[[BAR_F1]]}
98; DWO: DW_TAG_formal_parameter
99; CROSS: DW_AT_abstract_origin [DW_FORM_ref_addr] (0x00000000[[F1T]]
100; NOCROSS: DW_AT_abstract_origin [DW_FORM_ref4] {{.*}}{0x[[BAR_F1T]]
101
102%struct.t1 = type { i32 }
103
104; Function Attrs: nounwind readnone speculatable
105declare void @llvm.dbg.declare(metadata, metadata, metadata) #1
106
107declare void @_Z1fv() #2
David Blaikie85366ac2017-04-22 07:53:44 +0000108
109; Function Attrs: noinline uwtable
David Blaikie488393f2017-05-12 01:13:45 +0000110define void @_Z3foo2t1(i32 %t.coerce) #3 !dbg !20 {
David Blaikie85366ac2017-04-22 07:53:44 +0000111entry:
David Blaikie488393f2017-05-12 01:13:45 +0000112 %t.i = alloca %struct.t1, align 4
113 call void @llvm.dbg.declare(metadata %struct.t1* %t.i, metadata !15, metadata !16), !dbg !21
114 %t = alloca %struct.t1, align 4
115 %agg.tmp = alloca %struct.t1, align 4
116 %coerce.dive = getelementptr inbounds %struct.t1, %struct.t1* %t, i32 0, i32 0
117 store i32 %t.coerce, i32* %coerce.dive, align 4
118 call void @llvm.dbg.declare(metadata %struct.t1* %t, metadata !23, metadata !16), !dbg !24
119 %0 = bitcast %struct.t1* %agg.tmp to i8*, !dbg !25
120 %1 = bitcast %struct.t1* %t to i8*, !dbg !25
Daniel Neilson1e687242018-01-19 17:13:12 +0000121 call void @llvm.memcpy.p0i8.p0i8.i64(i8* align 4 %0, i8* align 4 %1, i64 4, i1 false), !dbg !25
David Blaikie488393f2017-05-12 01:13:45 +0000122 %coerce.dive1 = getelementptr inbounds %struct.t1, %struct.t1* %agg.tmp, i32 0, i32 0, !dbg !26
123 %2 = load i32, i32* %coerce.dive1, align 4, !dbg !26
124 %coerce.dive.i = getelementptr inbounds %struct.t1, %struct.t1* %t.i, i32 0, i32 0
125 store i32 %2, i32* %coerce.dive.i, align 4
126 call void @_Z1fv(), !dbg !27
127 ret void, !dbg !28
128}
129
130; Function Attrs: argmemonly nounwind
Daniel Neilson1e687242018-01-19 17:13:12 +0000131declare void @llvm.memcpy.p0i8.p0i8.i64(i8* nocapture writeonly, i8* nocapture readonly, i64, i1) #4
David Blaikie488393f2017-05-12 01:13:45 +0000132
133; Function Attrs: noinline uwtable
134define void @_Z3bar2t1(i32 %t.coerce) #3 !dbg !29 {
135entry:
136 %t.i = alloca %struct.t1, align 4
137 call void @llvm.dbg.declare(metadata %struct.t1* %t.i, metadata !15, metadata !16), !dbg !30
138 %t = alloca %struct.t1, align 4
139 %agg.tmp = alloca %struct.t1, align 4
140 %coerce.dive = getelementptr inbounds %struct.t1, %struct.t1* %t, i32 0, i32 0
141 store i32 %t.coerce, i32* %coerce.dive, align 4
142 call void @llvm.dbg.declare(metadata %struct.t1* %t, metadata !32, metadata !16), !dbg !33
143 %0 = bitcast %struct.t1* %agg.tmp to i8*, !dbg !34
144 %1 = bitcast %struct.t1* %t to i8*, !dbg !34
Daniel Neilson1e687242018-01-19 17:13:12 +0000145 call void @llvm.memcpy.p0i8.p0i8.i64(i8* align 4 %0, i8* align 4 %1, i64 4, i1 false), !dbg !34
David Blaikie488393f2017-05-12 01:13:45 +0000146 %coerce.dive1 = getelementptr inbounds %struct.t1, %struct.t1* %agg.tmp, i32 0, i32 0, !dbg !35
147 %2 = load i32, i32* %coerce.dive1, align 4, !dbg !35
148 %coerce.dive.i = getelementptr inbounds %struct.t1, %struct.t1* %t.i, i32 0, i32 0
149 store i32 %2, i32* %coerce.dive.i, align 4
150 call void @_Z1fv(), !dbg !36
151 ret void, !dbg !37
David Blaikie85366ac2017-04-22 07:53:44 +0000152}
153
154!llvm.dbg.cu = !{!0, !3}
155!llvm.ident = !{!5, !5}
David Blaikie488393f2017-05-12 01:13:45 +0000156!llvm.module.flags = !{!6, !7}
David Blaikie85366ac2017-04-22 07:53:44 +0000157
David Blaikie488393f2017-05-12 01:13:45 +0000158!0 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !1, producer: "clang version 5.0.0 (trunk 302809) (llvm/trunk 302815)", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, enums: !2, splitDebugInlining: true)
159!1 = !DIFile(filename: "foo.cpp", directory: "/usr/local/google/home/blaikie/dev/scratch")
David Blaikie85366ac2017-04-22 07:53:44 +0000160!2 = !{}
David Blaikie488393f2017-05-12 01:13:45 +0000161!3 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !4, producer: "clang version 5.0.0 (trunk 302809) (llvm/trunk 302815)", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, enums: !2, splitDebugInlining: true)
162!4 = !DIFile(filename: "bar.cpp", directory: "/usr/local/google/home/blaikie/dev/scratch")
163!5 = !{!"clang version 5.0.0 (trunk 302809) (llvm/trunk 302815)"}
164!6 = !{i32 2, !"Dwarf Version", i32 4}
165!7 = !{i32 2, !"Debug Info Version", i32 3}
Shiva Chen2c864552018-05-09 02:40:45 +0000166!8 = distinct !DISubprogram(name: "f1", linkageName: "_Z2f12t1", scope: !1, file: !1, line: 3, type: !9, isLocal: false, isDefinition: true, scopeLine: 3, flags: DIFlagPrototyped, isOptimized: false, unit: !0, retainedNodes: !2)
David Blaikie488393f2017-05-12 01:13:45 +0000167!9 = !DISubroutineType(types: !10)
168!10 = !{null, !11}
169!11 = distinct !DICompositeType(tag: DW_TAG_structure_type, name: "t1", file: !1, line: 1, size: 32, elements: !12, identifier: "_ZTS2t1")
170!12 = !{!13}
171!13 = !DIDerivedType(tag: DW_TAG_member, name: "i", scope: !11, file: !1, line: 1, baseType: !14, size: 32)
172!14 = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed)
173!15 = !DILocalVariable(name: "t", arg: 1, scope: !8, file: !1, line: 3, type: !11)
174!16 = !DIExpression()
175!17 = !DILocation(line: 3, column: 43, scope: !8)
176!18 = !DILocation(line: 4, column: 3, scope: !8)
177!19 = !DILocation(line: 5, column: 1, scope: !8)
Shiva Chen2c864552018-05-09 02:40:45 +0000178!20 = distinct !DISubprogram(name: "foo", linkageName: "_Z3foo2t1", scope: !1, file: !1, line: 6, type: !9, isLocal: false, isDefinition: true, scopeLine: 6, flags: DIFlagPrototyped, isOptimized: false, unit: !0, retainedNodes: !2)
David Blaikie488393f2017-05-12 01:13:45 +0000179!21 = !DILocation(line: 3, column: 43, scope: !8, inlinedAt: !22)
180!22 = distinct !DILocation(line: 7, column: 3, scope: !20)
181!23 = !DILocalVariable(name: "t", arg: 1, scope: !20, file: !1, line: 6, type: !11)
182!24 = !DILocation(line: 6, column: 13, scope: !20)
183!25 = !DILocation(line: 7, column: 6, scope: !20)
184!26 = !DILocation(line: 7, column: 3, scope: !20)
185!27 = !DILocation(line: 4, column: 3, scope: !8, inlinedAt: !22)
186!28 = !DILocation(line: 8, column: 1, scope: !20)
Shiva Chen2c864552018-05-09 02:40:45 +0000187!29 = distinct !DISubprogram(name: "bar", linkageName: "_Z3bar2t1", scope: !4, file: !4, line: 3, type: !9, isLocal: false, isDefinition: true, scopeLine: 3, flags: DIFlagPrototyped, isOptimized: false, unit: !3, retainedNodes: !2)
David Blaikie488393f2017-05-12 01:13:45 +0000188!30 = !DILocation(line: 3, column: 43, scope: !8, inlinedAt: !31)
189!31 = distinct !DILocation(line: 4, column: 3, scope: !29)
190!32 = !DILocalVariable(name: "t", arg: 1, scope: !29, file: !4, line: 3, type: !11)
191!33 = !DILocation(line: 3, column: 13, scope: !29)
192!34 = !DILocation(line: 4, column: 6, scope: !29)
193!35 = !DILocation(line: 4, column: 3, scope: !29)
194!36 = !DILocation(line: 4, column: 3, scope: !8, inlinedAt: !31)
195!37 = !DILocation(line: 5, column: 1, scope: !29)