1 ; RUN: opt -S -instcombine %s | FileCheck %s -check-prefix=RUN-ONCE
3 ; This example was reduced from a test case in which InstCombine ran at least
5 ; - The first InstCombine run converted dbg.declares to dbg.values using the
6 ; LowerDbgDeclare utility. This produced a dbg.value(i32* %2, DW_OP_deref)
7 ; (this happens when the contents of an alloca are passed by-value), and a
8 ; dbg.value(i32 %0) (due to the store of %0 into the alloca).
9 ; - The second InstCombine run deleted the alloca (%2).
10 ; Check that the DW_OP_deref dbg.value is deleted, just like a dbg.declare would
13 ; RUN-ONCE-LABEL: @t1(
14 ; RUN-ONCE-NEXT: llvm.dbg.value(metadata i32 %0, metadata [[t1_arg0:![0-9]+]], metadata !DIExpression())
15 ; RUN-ONCE-NEXT: llvm.dbg.value(metadata i32* undef, metadata [[t1_fake_ptr:![0-9]+]], metadata !DIExpression())
16 ; RUN-ONCE-NEXT: ret void
17 define void @t1(i32) !dbg !9 {
18 %2 = alloca i32, align 4
19 store i32 %0, i32* %2, align 4
20 call void @llvm.dbg.value(metadata i32 %0, metadata !14, metadata !DIExpression()), !dbg !15
21 call void @llvm.dbg.value(metadata i32* %2, metadata !14, metadata !DIExpression(DW_OP_deref)), !dbg !15
22 call void @llvm.dbg.value(metadata i32* %2, metadata !20, metadata !DIExpression()), !dbg !15
26 ; This example is closer to an end-to-end test: the IR looks like it could have
27 ; been produced by a frontend compiling at -O0.
29 ; Here's what happens:
30 ; 1) We run InstCombine. This puts a dbg.value(i32* %x.addr, DW_OP_deref)
31 ; before the call to @use, and a dbg.value(i32 %x) after the store.
33 ; 3) We run InstCombine again. The alloca %x.addr is erased. We should just get
34 ; dbg.value(i32 %x). There should be no leftover dbg.value(metadata i32*
37 ;;; define void @use(i32* %addr) alwaysinline { ret void }
38 ;;; define void @t2(i32 %x) !dbg !17 {
39 ;;; %x.addr = alloca i32, align 4
40 ;;; store i32 %x, i32* %x.addr, align 4
41 ;;; call void @llvm.dbg.declare(metadata i32* %x.addr, metadata !18, metadata !DIExpression()), !dbg !19
42 ;;; call void @use(i32* %x.addr)
46 declare void @llvm.dbg.value(metadata, metadata, metadata)
47 declare void @llvm.dbg.declare(metadata, metadata, metadata)
49 !llvm.module.flags = !{!0, !1, !2, !3, !4}
53 ; RUN-ONCE: [[t1_arg0]] = !DILocalVariable(name: "a"
54 ; RUN-ONCE: [[t1_fake_ptr]] = !DILocalVariable(name: "fake_ptr"
56 !0 = !{i32 2, !"SDK Version", [2 x i32] [i32 10, i32 14]}
57 !1 = !{i32 2, !"Dwarf Version", i32 4}
58 !2 = !{i32 2, !"Debug Info Version", i32 3}
59 !3 = !{i32 1, !"wchar_size", i32 4}
60 !4 = !{i32 7, !"PIC Level", i32 2}
61 !5 = distinct !DICompileUnit(language: DW_LANG_C99, file: !6, producer: "", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, enums: !7, nameTableKind: GNU)
62 !6 = !DIFile(filename: "-", directory: "/")
65 !9 = distinct !DISubprogram(name: "t1", scope: !10, file: !10, line: 1, type: !11, scopeLine: 1, flags: DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !5, retainedNodes: !7)
66 !10 = !DIFile(filename: "<stdin>", directory: "/")
67 !11 = !DISubroutineType(types: !12)
69 !13 = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed)
70 !14 = !DILocalVariable(name: "a", arg: 1, scope: !9, file: !10, line: 1, type: !13)
71 !15 = !DILocation(line: 1, column: 13, scope: !9)
72 !16 = !DILocation(line: 1, column: 17, scope: !9)
73 !17 = distinct !DISubprogram(name: "t2", scope: !10, file: !10, line: 1, type: !11, scopeLine: 1, flags: DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !5, retainedNodes: !7)
74 !18 = !DILocalVariable(name: "x", arg: 1, scope: !17, file: !10, line: 1, type: !13)
75 !19 = !DILocation(line: 1, column: 1, scope: !17)
76 !20 = !DILocalVariable(name: "fake_ptr", scope: !9, file: !10, line: 1, type: !13)