xref: /llvm-project/llvm/test/DebugInfo/WebAssembly/stackified-debug.ll (revision 341d4cdeb6a7b6526858ac6c86750101e2747e8a)
1*341d4cdeSHeejin Ahn; RUN: llc -verify-machineinstrs < %s | FileCheck %s
2*341d4cdeSHeejin Ahn; RUN: llc -filetype=obj %s -o - | llvm-dwarfdump - | FileCheck %s --check-prefix DWARF
3*341d4cdeSHeejin Ahn
4*341d4cdeSHeejin Ahn; Input C code:
5*341d4cdeSHeejin Ahn
6*341d4cdeSHeejin Ahn;  int i = input();  // Nested case
7*341d4cdeSHeejin Ahn;  int j = input();  // Trivial def-use.
8*341d4cdeSHeejin Ahn;  output(i, j);
9*341d4cdeSHeejin Ahn
10*341d4cdeSHeejin Ahn; The ll below generates 330 lines of .S, so relevant parts that the
11*341d4cdeSHeejin Ahn; WebAssemblyDebugFixup pass affects:
12*341d4cdeSHeejin Ahn
13*341d4cdeSHeejin Ahn; CHECK: 	call	input
14*341d4cdeSHeejin Ahn; CHECK: .Ltmp0:
15*341d4cdeSHeejin Ahn; CHECK: 	call	input
16*341d4cdeSHeejin Ahn; CHECK: .Ltmp1:
17*341d4cdeSHeejin Ahn; CHECK: 	call	output
18*341d4cdeSHeejin Ahn; CHECK: .Ltmp2:
19*341d4cdeSHeejin Ahn
20*341d4cdeSHeejin Ahn; This defines variable "i" which is live on the stack between Ltmp0 and Ltmp2,
21*341d4cdeSHeejin Ahn; 2 = TI_OPERAND_STACK and 0 = stack offset.
22*341d4cdeSHeejin Ahn
23*341d4cdeSHeejin Ahn; CHECK: 	.section	.debug_loc,"",@
24*341d4cdeSHeejin Ahn; CHECK: .Ldebug_loc0:
25*341d4cdeSHeejin Ahn; CHECK: 	.int32	.Ltmp0-.Lfunc_begin0
26*341d4cdeSHeejin Ahn; CHECK: 	.int32	.Ltmp2-.Lfunc_begin0
27*341d4cdeSHeejin Ahn; CHECK: 	.int16	4                       # Loc expr size
28*341d4cdeSHeejin Ahn; CHECK: 	.int8	237                       # DW_OP_WASM_location
29*341d4cdeSHeejin Ahn; CHECK: 	.int8	2                         # 2
30*341d4cdeSHeejin Ahn; CHECK: 	.int8	0                         # 0
31*341d4cdeSHeejin Ahn; CHECK: 	.int8	159                       # DW_OP_stack_value
32*341d4cdeSHeejin Ahn
33*341d4cdeSHeejin Ahn; This defines variable "j" which is live on the stack between Ltmp1 and Ltmp2,
34*341d4cdeSHeejin Ahn; 2 = TI_OPERAND_STACK and 1 = stack offset.
35*341d4cdeSHeejin Ahn
36*341d4cdeSHeejin Ahn; CHECK: .Ldebug_loc1:
37*341d4cdeSHeejin Ahn; CHECK: 	.int32	.Ltmp1-.Lfunc_begin0
38*341d4cdeSHeejin Ahn; CHECK: 	.int32	.Ltmp2-.Lfunc_begin0
39*341d4cdeSHeejin Ahn; CHECK: 	.int16	4                       # Loc expr size
40*341d4cdeSHeejin Ahn; CHECK: 	.int8	237                       # DW_OP_WASM_location
41*341d4cdeSHeejin Ahn; CHECK: 	.int8	2                         # 2
42*341d4cdeSHeejin Ahn; CHECK: 	.int8	1                         # 1
43*341d4cdeSHeejin Ahn; CHECK: 	.int8	159                       # DW_OP_stack_value
44*341d4cdeSHeejin Ahn
45*341d4cdeSHeejin Ahn
46*341d4cdeSHeejin Ahnsource_filename = "stackified.c"
47*341d4cdeSHeejin Ahntarget triple = "wasm32-unknown-unknown"
48*341d4cdeSHeejin Ahn
49*341d4cdeSHeejin Ahndefine void @foo() !dbg !12 {
50*341d4cdeSHeejin Ahnentry:
51*341d4cdeSHeejin Ahn  %call = call i32 @input(), !dbg !18
52*341d4cdeSHeejin Ahn  call void @llvm.dbg.value(metadata i32 %call, metadata !16, metadata !DIExpression()), !dbg !19
53*341d4cdeSHeejin Ahn  %call1 = call i32 @input(), !dbg !20
54*341d4cdeSHeejin Ahn  call void @llvm.dbg.value(metadata i32 %call1, metadata !17, metadata !DIExpression()), !dbg !19
55*341d4cdeSHeejin Ahn  call void @output(i32 %call, i32 %call1), !dbg !21
56*341d4cdeSHeejin Ahn  ret void, !dbg !22
57*341d4cdeSHeejin Ahn}
58*341d4cdeSHeejin Ahn
59*341d4cdeSHeejin Ahn; DebugFixup pass should not add a DBG_VALUE after the BR_IF instruction at the
60*341d4cdeSHeejin Ahn; end of the 'entry' BB, because it is not allowed to have more instructions
61*341d4cdeSHeejin Ahn; after a terminator and debug ranges are terminated at the end of a BB anyway.
62*341d4cdeSHeejin Ahn; If this passes 'llc -verify-machineinstrs', that means the DBG_VALUE
63*341d4cdeSHeejin Ahn; instruction is correctly omitted.
64*341d4cdeSHeejin Ahn
65*341d4cdeSHeejin Ahn; DWARF-LABEL: DW_AT_name ("no_dbg_value_after_terminator")
66*341d4cdeSHeejin Ahn; DWARF:       DW_TAG_variable
67*341d4cdeSHeejin Ahn; DWARF-NEXT:    DW_AT_location
68*341d4cdeSHeejin Ahn; DWARF-NEXT:      [
69*341d4cdeSHeejin Ahn; DWARF-NEXT:    DW_AT_name ("myvar")
70*341d4cdeSHeejin Ahndefine void @no_dbg_value_after_terminator(i32 %a, i32 %b) !dbg !23 {
71*341d4cdeSHeejin Ahnentry:
72*341d4cdeSHeejin Ahn  %cmp = icmp ne i32 %a, %b, !dbg !25
73*341d4cdeSHeejin Ahn  call void @llvm.dbg.value(metadata i1 %cmp, metadata !27, metadata !DIExpression(DW_OP_LLVM_convert, 1, DW_ATE_unsigned, DW_OP_LLVM_convert, 8, DW_ATE_unsigned, DW_OP_stack_value)), !dbg !25
74*341d4cdeSHeejin Ahn  br i1 %cmp, label %bb.1, label %bb.0, !dbg !25
75*341d4cdeSHeejin Ahn
76*341d4cdeSHeejin Ahnbb.0:                                             ; preds = %entry
77*341d4cdeSHeejin Ahn  unreachable
78*341d4cdeSHeejin Ahn
79*341d4cdeSHeejin Ahnbb.1:                                             ; preds = %entry
80*341d4cdeSHeejin Ahn  ret void
81*341d4cdeSHeejin Ahn}
82*341d4cdeSHeejin Ahn
83*341d4cdeSHeejin Ahndeclare i32 @input()
84*341d4cdeSHeejin Ahndeclare !dbg !4 void @output(i32, i32)
85*341d4cdeSHeejin Ahndeclare void @llvm.dbg.value(metadata, metadata, metadata)
86*341d4cdeSHeejin Ahn
87*341d4cdeSHeejin Ahn!llvm.dbg.cu = !{!0}
88*341d4cdeSHeejin Ahn!llvm.module.flags = !{!8, !9, !10}
89*341d4cdeSHeejin Ahn!llvm.ident = !{!11}
90*341d4cdeSHeejin Ahn
91*341d4cdeSHeejin Ahn!0 = distinct !DICompileUnit(language: DW_LANG_C99, file: !1, producer: "clang version 11.0.0 (https://github.com/llvm/llvm-project.git ed7aaf832444411ce93aa0443425ce401f5c7a8e)", isOptimized: true, runtimeVersion: 0, emissionKind: FullDebug, enums: !2, retainedTypes: !3, nameTableKind: None)
92*341d4cdeSHeejin Ahn!1 = !DIFile(filename: "stackified.c", directory: "C:\\stuff\\llvm-project")
93*341d4cdeSHeejin Ahn!2 = !{}
94*341d4cdeSHeejin Ahn!3 = !{!4}
95*341d4cdeSHeejin Ahn!4 = !DISubprogram(name: "output", scope: !1, file: !1, line: 2, type: !5, flags: DIFlagPrototyped, spFlags: DISPFlagOptimized, retainedNodes: !2)
96*341d4cdeSHeejin Ahn!5 = !DISubroutineType(types: !6)
97*341d4cdeSHeejin Ahn!6 = !{null, !7, !7}
98*341d4cdeSHeejin Ahn!7 = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed)
99*341d4cdeSHeejin Ahn!8 = !{i32 7, !"Dwarf Version", i32 4}
100*341d4cdeSHeejin Ahn!9 = !{i32 2, !"Debug Info Version", i32 3}
101*341d4cdeSHeejin Ahn!10 = !{i32 1, !"wchar_size", i32 4}
102*341d4cdeSHeejin Ahn!11 = !{!"clang version 11.0.0 (https://github.com/llvm/llvm-project.git ed7aaf832444411ce93aa0443425ce401f5c7a8e)"}
103*341d4cdeSHeejin Ahn!12 = distinct !DISubprogram(name: "foo", scope: !1, file: !1, line: 3, type: !13, scopeLine: 3, flags: DIFlagAllCallsDescribed, spFlags: DISPFlagDefinition | DISPFlagOptimized, unit: !0, retainedNodes: !15)
104*341d4cdeSHeejin Ahn!13 = !DISubroutineType(types: !14)
105*341d4cdeSHeejin Ahn!14 = !{null}
106*341d4cdeSHeejin Ahn!15 = !{!16, !17}
107*341d4cdeSHeejin Ahn!16 = !DILocalVariable(name: "i", scope: !12, file: !1, line: 4, type: !7)
108*341d4cdeSHeejin Ahn!17 = !DILocalVariable(name: "j", scope: !12, file: !1, line: 5, type: !7)
109*341d4cdeSHeejin Ahn!18 = !DILocation(line: 4, column: 11, scope: !12)
110*341d4cdeSHeejin Ahn!19 = !DILocation(line: 0, scope: !12)
111*341d4cdeSHeejin Ahn!20 = !DILocation(line: 5, column: 11, scope: !12)
112*341d4cdeSHeejin Ahn!21 = !DILocation(line: 6, column: 3, scope: !12)
113*341d4cdeSHeejin Ahn!22 = !DILocation(line: 7, column: 1, scope: !12)
114*341d4cdeSHeejin Ahn!23 = distinct !DISubprogram(name: "no_dbg_value_after_terminator", scope: null, type: !24, spFlags: DISPFlagDefinition, unit: !0)
115*341d4cdeSHeejin Ahn!24 = !DISubroutineType(types: !2)
116*341d4cdeSHeejin Ahn!25 = !DILocation(line: 0, scope: !26)
117*341d4cdeSHeejin Ahn!26 = distinct !DILexicalBlock(scope: !23)
118*341d4cdeSHeejin Ahn!27 = !DILocalVariable(name: "myvar", scope: !26, type: !28)
119*341d4cdeSHeejin Ahn!28 = !DIBasicType(name: "bool", size: 8, encoding: DW_ATE_boolean)
120