1; RUN: opt %s -sroa -verify -S -o - | FileCheck %s
2;
3; Test that we can partial emit debug info for aggregates repeatedly
4; split up by SROA.
5;
6;    // Compile with -O1
7;    typedef struct {
8;      int a;
9;      long int b;
10;    } Inner;
11;
12;    typedef struct {
13;      Inner inner[2];
14;    } Outer;
15;
16;    int foo(Outer outer) {
17;      Inner i1 = outer.inner[1];
18;      return i1.a;
19;    }
20;
21
22; Verify that SROA creates a variable piece when splitting i1.
23; CHECK: %[[I1:.*]] = alloca [12 x i8], align 4
24; CHECK: call void @llvm.dbg.declare(metadata [12 x i8]* %[[I1]], metadata ![[VAR:[0-9]+]], metadata !DIExpression(DW_OP_LLVM_fragment, 32, 96))
25; CHECK: call void @llvm.dbg.value(metadata i32 %[[A:.*]], metadata ![[VAR]], metadata !DIExpression(DW_OP_LLVM_fragment, 0, 32))
26; CHECK: ret i32 %[[A]]
27; Read Var and Piece:
28; CHECK: ![[VAR]] = !DILocalVariable(name: "i1",{{.*}} line: 11,
29
30target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128"
31target triple = "x86_64-apple-macosx10.9.0"
32
33%struct.Outer = type { [2 x %struct.Inner] }
34%struct.Inner = type { i32, i64 }
35
36; Function Attrs: nounwind ssp uwtable
37define i32 @foo(%struct.Outer* byval(%struct.Outer) align 8 %outer) #0 !dbg !4 {
38entry:
39  %i1 = alloca %struct.Inner, align 8
40  call void @llvm.dbg.declare(metadata %struct.Outer* %outer, metadata !25, metadata !2), !dbg !26
41  call void @llvm.dbg.declare(metadata %struct.Inner* %i1, metadata !27, metadata !2), !dbg !28
42  %inner = getelementptr inbounds %struct.Outer, %struct.Outer* %outer, i32 0, i32 0, !dbg !28
43  %arrayidx = getelementptr inbounds [2 x %struct.Inner], [2 x %struct.Inner]* %inner, i32 0, i64 1, !dbg !28
44  %0 = bitcast %struct.Inner* %i1 to i8*, !dbg !28
45  %1 = bitcast %struct.Inner* %arrayidx to i8*, !dbg !28
46  call void @llvm.memcpy.p0i8.p0i8.i64(i8* align 8 %0, i8* align 8 %1, i64 16, i1 false), !dbg !28
47  %a = getelementptr inbounds %struct.Inner, %struct.Inner* %i1, i32 0, i32 0, !dbg !29
48  %2 = load i32, i32* %a, align 4, !dbg !29
49  ret i32 %2, !dbg !29
50}
51
52; Function Attrs: nounwind readnone
53declare void @llvm.dbg.declare(metadata, metadata, metadata) #1
54
55; Function Attrs: nounwind
56declare void @llvm.memcpy.p0i8.p0i8.i64(i8* nocapture, i8* nocapture readonly, i64, i1) #2
57
58attributes #0 = { nounwind ssp uwtable }
59attributes #1 = { nounwind readnone }
60attributes #2 = { nounwind }
61
62!llvm.dbg.cu = !{!0}
63!llvm.module.flags = !{!22, !23}
64!llvm.ident = !{!24}
65
66!0 = distinct !DICompileUnit(language: DW_LANG_C99, producer: "clang version 3.5.0 ", isOptimized: false, emissionKind: FullDebug, file: !1, enums: !{}, retainedTypes: !{}, globals: !{}, imports: !{})
67!1 = !DIFile(filename: "sroasplit-1.c", directory: "")
68!2 = !DIExpression()
69!4 = distinct !DISubprogram(name: "foo", line: 10, isLocal: false, isDefinition: true, virtualIndex: 6, flags: DIFlagPrototyped, isOptimized: false, unit: !0, scopeLine: 10, file: !1, scope: !5, type: !6, retainedNodes: !{})
70!5 = !DIFile(filename: "sroasplit-1.c", directory: "")
71!6 = !DISubroutineType(types: !7)
72!7 = !{!8, !9}
73!8 = !DIBasicType(tag: DW_TAG_base_type, name: "int", size: 32, align: 32, encoding: DW_ATE_signed)
74!9 = !DIDerivedType(tag: DW_TAG_typedef, name: "Outer", line: 8, file: !1, baseType: !10)
75!10 = !DICompositeType(tag: DW_TAG_structure_type, line: 6, size: 256, align: 64, file: !1, elements: !11)
76!11 = !{!12}
77!12 = !DIDerivedType(tag: DW_TAG_member, name: "inner", line: 7, size: 256, align: 64, file: !1, scope: !10, baseType: !13)
78!13 = !DICompositeType(tag: DW_TAG_array_type, size: 256, align: 64, baseType: !14, elements: !20)
79!14 = !DIDerivedType(tag: DW_TAG_typedef, name: "Inner", line: 4, file: !1, baseType: !15)
80!15 = !DICompositeType(tag: DW_TAG_structure_type, line: 1, size: 128, align: 64, file: !1, elements: !16)
81!16 = !{!17, !18}
82!17 = !DIDerivedType(tag: DW_TAG_member, name: "a", line: 2, size: 32, align: 32, file: !1, scope: !15, baseType: !8)
83!18 = !DIDerivedType(tag: DW_TAG_member, name: "b", line: 3, size: 64, align: 64, offset: 64, file: !1, scope: !15, baseType: !19)
84!19 = !DIBasicType(tag: DW_TAG_base_type, name: "long int", size: 64, align: 64, encoding: DW_ATE_signed)
85!20 = !{!21}
86!21 = !DISubrange(count: 2)
87!22 = !{i32 2, !"Dwarf Version", i32 2}
88!23 = !{i32 1, !"Debug Info Version", i32 3}
89!24 = !{!"clang version 3.5.0 "}
90!25 = !DILocalVariable(name: "outer", line: 10, arg: 1, scope: !4, file: !5, type: !9)
91!26 = !DILocation(line: 10, scope: !4)
92!27 = !DILocalVariable(name: "i1", line: 11, scope: !4, file: !5, type: !14)
93!28 = !DILocation(line: 11, scope: !4)
94!29 = !DILocation(line: 12, scope: !4)
95