|
| 1 | +; -stats requires asserts |
| 2 | +; REQUIRES: asserts |
| 3 | + |
| 4 | +; Check that we can still devirtualize outside LTO mode |
| 5 | +; Check that we skip devirtualization for empty functions outside LTO mode |
| 6 | + |
| 7 | +; RUN: opt -S -passes=wholeprogramdevirt -wholeprogramdevirt-nolto -pass-remarks=wholeprogramdevirt -stats %s 2>&1 | FileCheck %s |
| 8 | + |
| 9 | +target datalayout = "e-p:64:64" |
| 10 | +target triple = "x86_64-unknown-linux-gnu" |
| 11 | + |
| 12 | +; CHECK: remark: devirt-single.cc:30:32: single-impl: devirtualized a call to vf |
| 13 | +; CHECK: remark: devirt-single.cc:41:32: single-impl: devirtualized a call to vf |
| 14 | +; CHECK: remark: devirt-single.cc:51:32: single-impl: devirtualized a call to vf |
| 15 | +; CHECK: remark: devirt-single.cc:13:0: devirtualized vf |
| 16 | +; CHECK-NOT: devirtualized |
| 17 | + |
| 18 | +@vt1 = constant [1 x ptr] [ptr @vf], !type !8 |
| 19 | +@vt2 = constant [1 x ptr] [ptr @vf_empty], !type !12 |
| 20 | + |
| 21 | +define i1 @vf(ptr %this) #0 !dbg !7 { |
| 22 | + ret i1 true |
| 23 | +} |
| 24 | + |
| 25 | +; This should NOT be devietualized because during non-lto empty functions |
| 26 | +; are skipped. |
| 27 | +define void @vf_empty(ptr %this) !dbg !11 { |
| 28 | + ret void |
| 29 | +} |
| 30 | + |
| 31 | +; CHECK: define void @call |
| 32 | +define void @call(ptr %obj) #1 !dbg !5 { |
| 33 | + %vtable = load ptr, ptr %obj |
| 34 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid") |
| 35 | + call void @llvm.assume(i1 %p) |
| 36 | + %fptr = load ptr, ptr %vtable |
| 37 | + ; CHECK: if.true.direct_targ: |
| 38 | + ; CHECK: call i1 @vf( |
| 39 | + ; CHECK: if.false.orig_indirect: |
| 40 | + ; CHECK: call i1 %fptr( |
| 41 | + call i1 %fptr(ptr %obj), !dbg !6 |
| 42 | + ret void |
| 43 | +} |
| 44 | + |
| 45 | + |
| 46 | +; CHECK: define void @call1 |
| 47 | +define void @call1(ptr %obj) #1 !dbg !9 { |
| 48 | + %vtable = load ptr, ptr %obj |
| 49 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid1") |
| 50 | + call void @llvm.assume(i1 %p) |
| 51 | + %fptr = load ptr, ptr %vtable, align 8 |
| 52 | + ; CHECK: call i1 %fptr |
| 53 | + %1 = call i1 %fptr(ptr %obj), !dbg !10 |
| 54 | + ret void |
| 55 | +} |
| 56 | +declare ptr @llvm.load.relative.i32(ptr, i32) |
| 57 | + |
| 58 | +@vt3 = private unnamed_addr constant [1 x i32] [ |
| 59 | + i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf to i64), i64 ptrtoint (ptr @vt3 to i64)) to i32) |
| 60 | +], align 4, !type !15 |
| 61 | + |
| 62 | +; CHECK: define void @call2 |
| 63 | +define void @call2(ptr %obj) #1 !dbg !13 { |
| 64 | + %vtable = load ptr, ptr %obj |
| 65 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid2") |
| 66 | + call void @llvm.assume(i1 %p) |
| 67 | + %fptr = call ptr @llvm.load.relative.i32(ptr %vtable, i32 0) |
| 68 | + ; CHECK: if.true.direct_targ: |
| 69 | + ; CHECK: call i1 @vf( |
| 70 | + ; CHECK: if.false.orig_indirect: |
| 71 | + ; CHECK: call i1 %fptr( |
| 72 | + call i1 %fptr(ptr %obj), !dbg !14 |
| 73 | + ret void |
| 74 | +} |
| 75 | + |
| 76 | +@_ZTV1A.local = private unnamed_addr constant { [3 x i32] } { [3 x i32] [ |
| 77 | + i32 0, ; offset to top |
| 78 | + i32 0, ; rtti |
| 79 | + i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf to i64), i64 ptrtoint (ptr getelementptr inbounds ({ [3 x i32] }, ptr @_ZTV1A.local, i32 0, i32 0, i32 2) to i64)) to i32) ; vf_emptyunc offset |
| 80 | +] }, align 4, !type !18 |
| 81 | + |
| 82 | +; CHECK: define void @call3 |
| 83 | +define void @call3(ptr %obj) #1 !dbg !16 { |
| 84 | + %vtable = load ptr, ptr %obj |
| 85 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid3") |
| 86 | + call void @llvm.assume(i1 %p) |
| 87 | + %fptr = call ptr @llvm.load.relative.i32(ptr %vtable, i32 8) |
| 88 | + ; CHECK: if.true.direct_targ: |
| 89 | + ; CHECK: call i1 @vf( |
| 90 | + ; CHECK: if.false.orig_indirect: |
| 91 | + ; CHECK: call i1 %fptr( |
| 92 | + call i1 %fptr(ptr %obj), !dbg !17 |
| 93 | + ret void |
| 94 | +} |
| 95 | + |
| 96 | + |
| 97 | +declare i1 @llvm.type.test(ptr, metadata) |
| 98 | +declare void @llvm.assume(i1) |
| 99 | + |
| 100 | +!llvm.dbg.cu = !{!0} |
| 101 | +!llvm.module.flags = !{!2, !3} |
| 102 | +!llvm.ident = !{!4} |
| 103 | + |
| 104 | +!0 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !1, producer: "clang version 4.0.0 (trunk 278098)", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug) |
| 105 | +!1 = !DIFile(filename: "devirt-single.cc", directory: ".") |
| 106 | +!2 = !{i32 2, !"Dwarf Version", i32 4} |
| 107 | +!3 = !{i32 2, !"Debug Info Version", i32 3} |
| 108 | +!4 = !{!"clang version 4.0.0 (trunk 278098)"} |
| 109 | +!5 = distinct !DISubprogram(name: "call", linkageName: "_Z4callPv", scope: !1, file: !1, line: 29, isLocal: false, isDefinition: true, scopeLine: 9, flags: DIFlagPrototyped, isOptimized: false, unit: !0) |
| 110 | +!6 = !DILocation(line: 30, column: 32, scope: !5) |
| 111 | +!7 = distinct !DISubprogram(name: "vf", linkageName: "_ZN3vt12vfEv", scope: !1, file: !1, line: 13, isLocal: false, isDefinition: true, scopeLine: 13, flags: DIFlagPrototyped, isOptimized: false, unit: !0) |
| 112 | +!8 = !{i32 0, !"typeid"} |
| 113 | + |
| 114 | +!9 = distinct !DISubprogram(name: "call1", linkageName: "_Z5call1Pv", scope: !1, file: !1, line: 31, isLocal: false, isDefinition: true, scopeLine: 9, flags: DIFlagPrototyped, isOptimized: false, unit: !0) |
| 115 | +!10 = !DILocation(line: 35, column: 32, scope: !9) |
| 116 | +!11 = distinct !DISubprogram(name: "vf_empty", linkageName: "_ZN3vt18vf_emptyEv", scope: !1, file: !1, line: 23, isLocal: false, isDefinition: true, scopeLine: 23, flags: DIFlagPrototyped, isOptimized: false, unit: !0) |
| 117 | +!12 = !{i32 0, !"typeid1"} |
| 118 | + |
| 119 | +!13 = distinct !DISubprogram(name: "call2", linkageName: "_Z5call2Pv", scope: !1, file: !1, line: 40, isLocal: false, isDefinition: true, scopeLine: 9, flags: DIFlagPrototyped, isOptimized: false, unit: !0) |
| 120 | +!14 = !DILocation(line: 41, column: 32, scope: !13) |
| 121 | +!15 = !{i32 0, !"typeid2"} |
| 122 | + |
| 123 | +!16 = distinct !DISubprogram(name: "call3", linkageName: "_Z5call3Pv", scope: !1, file: !1, line: 50, isLocal: false, isDefinition: true, scopeLine: 9, flags: DIFlagPrototyped, isOptimized: false, unit: !0) |
| 124 | +!17 = !DILocation(line: 51, column: 32, scope: !16) |
| 125 | +!18 = !{i32 0, !"typeid3"} |
| 126 | + |
| 127 | + |
| 128 | + |
| 129 | +; CHECK: 1 wholeprogramdevirt - Number of whole program devirtualization targets |
| 130 | +; CHECK: 3 wholeprogramdevirt - Number of single implementation devirtualizations |
0 commit comments