1; RUN: opt -passes=instcombine %s -S -o - | FileCheck %s 2; RUN: opt -passes=instcombine %s -S -o - --try-experimental-debuginfo-iterators | FileCheck %s 3; Verify that the eliminated instructions (bitcast, gep, load) are salvaged into 4; a DIExpression. 5; 6; Originally created from the following C source and then heavily isolated/reduced. 7; 8; struct entry { 9; struct entry *next; 10; }; 11; void scan(struct entry *queue, struct entry *end) 12; { 13; struct entry *entry; 14; for (entry = (struct entry *)((char *)(queue->next) - 8); 15; &entry->next == end; 16; entry = (struct entry *)((char *)(entry->next) - 8)) { 17; } 18; } 19 20; ModuleID = '<stdin>' 21source_filename = "test.c" 22target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128" 23target triple = "x86_64-apple-macosx10.12.0" 24 25%struct.entry = type { ptr } 26 27; This salvage can't currently occur safely (PR40628), however if/when that's 28; ever fixed, then this is definitely a piece of test coverage that should 29; be maintained. 30define void @salvage_load(ptr %queue) local_unnamed_addr #0 !dbg !14 { 31entry: 32 %im_not_dead = alloca ptr 33 %0 = load ptr, ptr %queue, align 8, !dbg !19 34 %1 = load ptr, ptr %queue, align 8, !dbg !19 35 call void @llvm.dbg.value(metadata ptr %1, metadata !18, metadata !20), !dbg !19 36; CHECK: define void @salvage_load 37; CHECK-NEXT: entry: 38; CHECK-NEXT: #dbg_value(ptr poison 39 store ptr %1, ptr %im_not_dead, align 8 40 ret void, !dbg !21 41} 42 43define void @salvage_bitcast(ptr %queue) local_unnamed_addr #0 !dbg !22 { 44entry: 45 %im_not_dead = alloca ptr 46 call void @llvm.dbg.value(metadata ptr %queue, metadata !24, metadata !20), !dbg !23 47; CHECK: define void @salvage_bitcast 48; CHECK-NEXT: entry: 49; CHECK-NEXT: #dbg_value(ptr %queue, 50; CHECK-SAME: !DIExpression(DW_OP_plus_uconst, 0), 51 store ptr %queue, ptr %im_not_dead, align 8 52 ret void, !dbg !23 53} 54 55define void @salvage_gep0(ptr %queue, ptr %end) local_unnamed_addr #0 !dbg !25 { 56entry: 57 %im_not_dead = alloca ptr 58 %0 = getelementptr inbounds %struct.entry, ptr %queue, i32 -1, i32 0, !dbg !26 59 %1 = getelementptr inbounds %struct.entry, ptr %queue, i32 -1, i32 0, !dbg !26 60 call void @llvm.dbg.value(metadata ptr %1, metadata !27, metadata !20), !dbg !26 61; CHECK: define void @salvage_gep0 62; CHECK-NEXT: entry: 63; CHECK-NEXT: #dbg_value(ptr %queue, 64; CHECK-SAME: !DIExpression(DW_OP_constu, 8, DW_OP_minus, DW_OP_stack_value), 65 store ptr %1, ptr %im_not_dead, align 8 66 ret void, !dbg !26 67} 68 69define void @salvage_gep1(ptr %queue, ptr %end) local_unnamed_addr #0 !dbg !28 { 70entry: 71 %im_not_dead = alloca ptr 72 %0 = getelementptr inbounds %struct.entry, ptr %queue, i32 -1, i32 0, !dbg !29 73 %1 = getelementptr inbounds %struct.entry, ptr %queue, i32 -1, i32 0, !dbg !29 74 call void @llvm.dbg.value(metadata ptr %1, metadata !30, metadata !DIExpression(DW_OP_LLVM_fragment, 0, 32)), !dbg !29 75; CHECK: define void @salvage_gep1 76; CHECK-NEXT: entry: 77; CHECK-NEXT: #dbg_value(ptr %queue, 78; CHECK-SAME: !DIExpression(DW_OP_constu, 8, DW_OP_minus, DW_OP_stack_value, DW_OP_LLVM_fragment, 0, 32), 79 store ptr %1, ptr %im_not_dead, align 8 80 ret void, !dbg !29 81} 82 83define void @salvage_gep2(ptr %queue, ptr %end) local_unnamed_addr #0 !dbg !31 { 84entry: 85 %im_not_dead = alloca ptr 86 %0 = getelementptr inbounds %struct.entry, ptr %queue, i32 -1, i32 0, !dbg !32 87 %1 = getelementptr inbounds %struct.entry, ptr %queue, i32 -1, i32 0, !dbg !32 88 call void @llvm.dbg.value(metadata ptr %1, metadata !33, metadata !DIExpression(DW_OP_stack_value)), !dbg !32 89; CHECK: define void @salvage_gep2 90; CHECK-NEXT: entry: 91; CHECK-NEXT: #dbg_value(ptr %queue, 92; CHECK-SAME: !DIExpression(DW_OP_constu, 8, DW_OP_minus, DW_OP_stack_value), 93 store ptr %1, ptr %im_not_dead, align 8 94 ret void, !dbg !32 95} 96 97; Function Attrs: nounwind readnone 98declare void @llvm.dbg.value(metadata, metadata, metadata) #1 99 100attributes #0 = { nounwind ssp uwtable } 101attributes #1 = { nounwind readnone } 102 103!llvm.dbg.cu = !{!0} 104!llvm.module.flags = !{!10, !11, !12} 105!llvm.ident = !{!13} 106 107!0 = distinct !DICompileUnit(language: DW_LANG_C99, file: !1, producer: "clang version 5.0.0 (trunk 297628) (llvm/trunk 297643)", isOptimized: true, runtimeVersion: 0, emissionKind: FullDebug, enums: !2, retainedTypes: !3) 108!1 = !DIFile(filename: "test.c", directory: "/") 109!2 = !{} 110!3 = !{!4, !8} 111!4 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !5, size: 64) 112!5 = distinct !DICompositeType(tag: DW_TAG_structure_type, name: "entry", file: !1, line: 1, size: 64, elements: !6) 113!6 = !{!7} 114!7 = !DIDerivedType(tag: DW_TAG_member, name: "next", scope: !5, file: !1, line: 2, baseType: !4, size: 64) 115!8 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !9, size: 64) 116!9 = !DIBasicType(name: "char", size: 8, encoding: DW_ATE_signed_char) 117!10 = !{i32 2, !"Dwarf Version", i32 4} 118!11 = !{i32 2, !"Debug Info Version", i32 3} 119!12 = !{i32 1, !"PIC Level", i32 2} 120!13 = !{!"clang version 5.0.0 (trunk 297628) (llvm/trunk 297643)"} 121!14 = distinct !DISubprogram(name: "scan", scope: !1, file: !1, line: 4, type: !15, isLocal: false, isDefinition: true, scopeLine: 5, flags: DIFlagPrototyped, isOptimized: true, unit: !0, retainedNodes: !17) 122!15 = !DISubroutineType(types: !16) 123!16 = !{null, !4, !4} 124!17 = !{!18} 125!18 = !DILocalVariable(name: "entry", scope: !14, file: !1, line: 6, type: !4) 126!19 = !DILocation(line: 6, column: 17, scope: !14) 127!20 = !DIExpression(DW_OP_plus_uconst, 0) 128!21 = !DILocation(line: 11, column: 1, scope: !14) 129!22 = distinct !DISubprogram(name: "scan", scope: !1, file: !1, line: 4, type: !15, isLocal: false, isDefinition: true, scopeLine: 5, flags: DIFlagPrototyped, isOptimized: true, unit: !0, retainedNodes: !17) 130!23 = !DILocation(line: 6, column: 17, scope: !22) 131!24 = !DILocalVariable(name: "entry", scope: !22, file: !1, line: 6, type: !4) 132!25 = distinct !DISubprogram(name: "scan", scope: !1, file: !1, line: 4, type: !15, isLocal: false, isDefinition: true, scopeLine: 5, flags: DIFlagPrototyped, isOptimized: true, unit: !0, retainedNodes: !17) 133!26 = !DILocation(line: 6, column: 17, scope: !25) 134!27 = !DILocalVariable(name: "entry", scope: !25, file: !1, line: 6, type: !4) 135!28 = distinct !DISubprogram(name: "scan", scope: !1, file: !1, line: 4, type: !15, isLocal: false, isDefinition: true, scopeLine: 5, flags: DIFlagPrototyped, isOptimized: true, unit: !0, retainedNodes: !17) 136!29 = !DILocation(line: 6, column: 17, scope: !28) 137!30 = !DILocalVariable(name: "entry", scope: !28, file: !1, line: 6, type: !4) 138!31 = distinct !DISubprogram(name: "scan", scope: !1, file: !1, line: 4, type: !15, isLocal: false, isDefinition: true, scopeLine: 5, flags: DIFlagPrototyped, isOptimized: true, unit: !0, retainedNodes: !17) 139!32 = !DILocation(line: 6, column: 17, scope: !31) 140!33 = !DILocalVariable(name: "entry", scope: !31, file: !1, line: 6, type: !4) 141