|
| 1 | +; A variant of branch-funnel.ll where we just check that the funnels' entry counts |
| 2 | +; are correctly set. |
| 3 | +; |
| 4 | +; RUN: opt -S -passes=wholeprogramdevirt -whole-program-visibility %s | FileCheck --check-prefixes=RETP %s |
| 5 | +; RUN: sed -e 's,+retpoline,-retpoline,g' %s | opt -S -passes=wholeprogramdevirt -whole-program-visibility | FileCheck --check-prefixes=NORETP %s |
| 6 | +; RUN: opt -passes=wholeprogramdevirt -whole-program-visibility -wholeprogramdevirt-summary-action=export -wholeprogramdevirt-read-summary=%S/Inputs/export.yaml -wholeprogramdevirt-write-summary=%t -S -o - %s | FileCheck --check-prefixes=RETP %s |
| 7 | +; RUN: opt -passes='wholeprogramdevirt,default<O3>' -whole-program-visibility -wholeprogramdevirt-summary-action=export -wholeprogramdevirt-read-summary=%S/Inputs/export.yaml -wholeprogramdevirt-write-summary=%t -S -o - %s | FileCheck --check-prefixes=O3 %s |
| 8 | + |
| 9 | +; RETP: define hidden void @__typeid_typeid1_0_branch_funnel(ptr nest %0, ...) !prof !11 |
| 10 | +; RETP: define hidden void @__typeid_typeid1_rv_0_branch_funnel(ptr nest %0, ...) !prof !11 |
| 11 | +; RETP: define internal void @branch_funnel(ptr nest %0, ...) !prof !10 |
| 12 | +; RETP: define internal void @branch_funnel.1(ptr nest %0, ...) !prof !10 |
| 13 | +; RETP: !10 = !{!"function_entry_count", i64 1000} |
| 14 | +; RETP: !11 = !{!"function_entry_count", i64 3000} |
| 15 | + |
| 16 | +; NORETP: define hidden void @__typeid_typeid1_0_branch_funnel(ptr nest %0, ...) !prof !11 |
| 17 | +; NORETP: define hidden void @__typeid_typeid1_rv_0_branch_funnel(ptr nest %0, ...) !prof !11 |
| 18 | +; NORETP: define internal void @branch_funnel(ptr nest %0, ...) !prof !11 |
| 19 | +; NORETP: define internal void @branch_funnel.1(ptr nest %0, ...) !prof !11 |
| 20 | +; NORETP: !11 = !{!"unknown"} |
| 21 | + |
| 22 | +; O3: define hidden void @__typeid_typeid1_0_branch_funnel(ptr nest %0, ...) local_unnamed_addr #5 !prof !11 |
| 23 | +; O3: define hidden void @__typeid_typeid1_rv_0_branch_funnel(ptr nest %0, ...) local_unnamed_addr #5 !prof !11 |
| 24 | +; O3: define internal void @branch_funnel(ptr nest %0, ...) unnamed_addr #5 !prof !10 |
| 25 | +; O3: define internal void @branch_funnel.1(ptr nest %0, ...) unnamed_addr #5 !prof !10 |
| 26 | +; O3: define hidden void @__typeid_typeid3_0_branch_funnel(ptr nest %0, ...) local_unnamed_addr #5 !prof !12 |
| 27 | +; O3: define hidden void @__typeid_typeid3_rv_0_branch_funnel(ptr nest %0, ...) local_unnamed_addr #5 !prof !12 |
| 28 | +; O3: !10 = !{!"function_entry_count", i64 1000} |
| 29 | +; O3: !11 = !{!"function_entry_count", i64 3000} |
| 30 | +; O3: !12 = !{!"unknown"} |
| 31 | + |
| 32 | +target datalayout = "e-p:64:64" |
| 33 | +target triple = "x86_64-unknown-linux-gnu" |
| 34 | + |
| 35 | +@vt1_1 = constant [1 x ptr] [ptr @vf1_1], !type !0 |
| 36 | +@vt1_2 = constant [1 x ptr] [ptr @vf1_2], !type !0 |
| 37 | + |
| 38 | +declare i32 @vf1_1(ptr %this, i32 %arg) |
| 39 | +declare i32 @vf1_2(ptr %this, i32 %arg) |
| 40 | + |
| 41 | +@vt2_1 = constant [1 x ptr] [ptr @vf2_1], !type !1 |
| 42 | +@vt2_2 = constant [1 x ptr] [ptr @vf2_2], !type !1 |
| 43 | +@vt2_3 = constant [1 x ptr] [ptr @vf2_3], !type !1 |
| 44 | +@vt2_4 = constant [1 x ptr] [ptr @vf2_4], !type !1 |
| 45 | +@vt2_5 = constant [1 x ptr] [ptr @vf2_5], !type !1 |
| 46 | +@vt2_6 = constant [1 x ptr] [ptr @vf2_6], !type !1 |
| 47 | +@vt2_7 = constant [1 x ptr] [ptr @vf2_7], !type !1 |
| 48 | +@vt2_8 = constant [1 x ptr] [ptr @vf2_8], !type !1 |
| 49 | +@vt2_9 = constant [1 x ptr] [ptr @vf2_9], !type !1 |
| 50 | +@vt2_10 = constant [1 x ptr] [ptr @vf2_10], !type !1 |
| 51 | +@vt2_11 = constant [1 x ptr] [ptr @vf2_11], !type !1 |
| 52 | + |
| 53 | +declare i32 @vf2_1(ptr %this, i32 %arg) |
| 54 | +declare i32 @vf2_2(ptr %this, i32 %arg) |
| 55 | +declare i32 @vf2_3(ptr %this, i32 %arg) |
| 56 | +declare i32 @vf2_4(ptr %this, i32 %arg) |
| 57 | +declare i32 @vf2_5(ptr %this, i32 %arg) |
| 58 | +declare i32 @vf2_6(ptr %this, i32 %arg) |
| 59 | +declare i32 @vf2_7(ptr %this, i32 %arg) |
| 60 | +declare i32 @vf2_8(ptr %this, i32 %arg) |
| 61 | +declare i32 @vf2_9(ptr %this, i32 %arg) |
| 62 | +declare i32 @vf2_10(ptr %this, i32 %arg) |
| 63 | +declare i32 @vf2_11(ptr %this, i32 %arg) |
| 64 | + |
| 65 | +@vt3_1 = constant [1 x ptr] [ptr @vf3_1], !type !2 |
| 66 | +@vt3_2 = constant [1 x ptr] [ptr @vf3_2], !type !2 |
| 67 | + |
| 68 | +declare i32 @vf3_1(ptr %this, i32 %arg) |
| 69 | +declare i32 @vf3_2(ptr %this, i32 %arg) |
| 70 | + |
| 71 | +@vt4_1 = constant [1 x ptr] [ptr @vf4_1], !type !3 |
| 72 | +@vt4_2 = constant [1 x ptr] [ptr @vf4_2], !type !3 |
| 73 | + |
| 74 | +declare i32 @vf4_1(ptr %this, i32 %arg) |
| 75 | +declare i32 @vf4_2(ptr %this, i32 %arg) |
| 76 | + |
| 77 | +declare ptr @llvm.load.relative.i32(ptr, i32) |
| 78 | + |
| 79 | +;; These are relative vtables equivalent to the ones above. |
| 80 | +@vt1_1_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf1_1 to i64), i64 ptrtoint (ptr @vt1_1_rv to i64)) to i32)], !type !5 |
| 81 | +@vt1_2_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf1_2 to i64), i64 ptrtoint (ptr @vt1_2_rv to i64)) to i32)], !type !5 |
| 82 | + |
| 83 | +@vt2_1_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_1 to i64), i64 ptrtoint (ptr @vt2_1_rv to i64)) to i32)], !type !6 |
| 84 | +@vt2_2_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_2 to i64), i64 ptrtoint (ptr @vt2_2_rv to i64)) to i32)], !type !6 |
| 85 | +@vt2_3_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_3 to i64), i64 ptrtoint (ptr @vt2_3_rv to i64)) to i32)], !type !6 |
| 86 | +@vt2_4_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_4 to i64), i64 ptrtoint (ptr @vt2_4_rv to i64)) to i32)], !type !6 |
| 87 | +@vt2_5_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_5 to i64), i64 ptrtoint (ptr @vt2_5_rv to i64)) to i32)], !type !6 |
| 88 | +@vt2_6_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_6 to i64), i64 ptrtoint (ptr @vt2_6_rv to i64)) to i32)], !type !6 |
| 89 | +@vt2_7_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_7 to i64), i64 ptrtoint (ptr @vt2_7_rv to i64)) to i32)], !type !6 |
| 90 | +@vt2_8_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_8 to i64), i64 ptrtoint (ptr @vt2_8_rv to i64)) to i32)], !type !6 |
| 91 | +@vt2_9_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_9 to i64), i64 ptrtoint (ptr @vt2_9_rv to i64)) to i32)], !type !6 |
| 92 | +@vt2_10_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_10 to i64), i64 ptrtoint (ptr @vt2_10_rv to i64)) to i32)], !type !6 |
| 93 | +@vt2_11_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf2_11 to i64), i64 ptrtoint (ptr @vt2_11_rv to i64)) to i32)], !type !6 |
| 94 | + |
| 95 | +@vt3_1_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf3_1 to i64), i64 ptrtoint (ptr @vt3_1_rv to i64)) to i32)], !type !7 |
| 96 | +@vt3_2_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf3_2 to i64), i64 ptrtoint (ptr @vt3_2_rv to i64)) to i32)], !type !7 |
| 97 | + |
| 98 | +@vt4_1_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf4_1 to i64), i64 ptrtoint (ptr @vt4_1_rv to i64)) to i32)], !type !8 |
| 99 | +@vt4_2_rv = constant [1 x i32] [i32 trunc (i64 sub (i64 ptrtoint (ptr dso_local_equivalent @vf4_2 to i64), i64 ptrtoint (ptr @vt4_2_rv to i64)) to i32)], !type !8 |
| 100 | + |
| 101 | + |
| 102 | +define i32 @fn1(ptr %obj) #0 !prof !10 { |
| 103 | + %vtable = load ptr, ptr %obj |
| 104 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid1") |
| 105 | + call void @llvm.assume(i1 %p) |
| 106 | + %fptr = load ptr, ptr %vtable |
| 107 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 108 | + ret i32 %result |
| 109 | +} |
| 110 | + |
| 111 | +define i32 @fn1_rv(ptr %obj) #0 !prof !10 { |
| 112 | + %vtable = load ptr, ptr %obj |
| 113 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid1_rv") |
| 114 | + call void @llvm.assume(i1 %p) |
| 115 | + %fptr = call ptr @llvm.load.relative.i32(ptr %vtable, i32 0) |
| 116 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 117 | + ret i32 %result |
| 118 | +} |
| 119 | + |
| 120 | +define i32 @fn2(ptr %obj) #0 !prof !10 { |
| 121 | + %vtable = load ptr, ptr %obj |
| 122 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid2") |
| 123 | + call void @llvm.assume(i1 %p) |
| 124 | + %fptr = load ptr, ptr %vtable |
| 125 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 126 | + ret i32 %result |
| 127 | +} |
| 128 | + |
| 129 | +define i32 @fn2_rv(ptr %obj) #0 !prof !10 { |
| 130 | + %vtable = load ptr, ptr %obj |
| 131 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !"typeid2_rv") |
| 132 | + call void @llvm.assume(i1 %p) |
| 133 | + %fptr = call ptr @llvm.load.relative.i32(ptr %vtable, i32 0) |
| 134 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 135 | + ret i32 %result |
| 136 | +} |
| 137 | + |
| 138 | +define i32 @fn3(ptr %obj) #0 !prof !10 { |
| 139 | + %vtable = load ptr, ptr %obj |
| 140 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !4) |
| 141 | + call void @llvm.assume(i1 %p) |
| 142 | + %fptr = load ptr, ptr %vtable |
| 143 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 144 | + ret i32 %result |
| 145 | +} |
| 146 | + |
| 147 | +define i32 @fn3_rv(ptr %obj) #0 !prof !10 { |
| 148 | + %vtable = load ptr, ptr %obj |
| 149 | + %p = call i1 @llvm.type.test(ptr %vtable, metadata !9) |
| 150 | + call void @llvm.assume(i1 %p) |
| 151 | + %fptr = call ptr @llvm.load.relative.i32(ptr %vtable, i32 0) |
| 152 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 153 | + ret i32 %result |
| 154 | +} |
| 155 | + |
| 156 | +define i32 @fn4(ptr %obj) #0 !prof !10 { |
| 157 | + %p = call i1 @llvm.type.test(ptr @vt1_1, metadata !"typeid1") |
| 158 | + call void @llvm.assume(i1 %p) |
| 159 | + %fptr = load ptr, ptr @vt1_1 |
| 160 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 161 | + ret i32 %result |
| 162 | +} |
| 163 | + |
| 164 | +define i32 @fn4_cpy(ptr %obj) #0 !prof !10 { |
| 165 | + %p = call i1 @llvm.type.test(ptr @vt1_1, metadata !"typeid1") |
| 166 | + call void @llvm.assume(i1 %p) |
| 167 | + %fptr = load ptr, ptr @vt1_1 |
| 168 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 169 | + ret i32 %result |
| 170 | +} |
| 171 | + |
| 172 | +define i32 @fn4_rv(ptr %obj) #0 !prof !10 { |
| 173 | + %p = call i1 @llvm.type.test(ptr @vt1_1_rv, metadata !"typeid1_rv") |
| 174 | + call void @llvm.assume(i1 %p) |
| 175 | + %fptr = call ptr @llvm.load.relative.i32(ptr @vt1_1_rv, i32 0) |
| 176 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 177 | + ret i32 %result |
| 178 | +} |
| 179 | + |
| 180 | +define i32 @fn4_rv_cpy(ptr %obj) #0 !prof !10 { |
| 181 | + %p = call i1 @llvm.type.test(ptr @vt1_1_rv, metadata !"typeid1_rv") |
| 182 | + call void @llvm.assume(i1 %p) |
| 183 | + %fptr = call ptr @llvm.load.relative.i32(ptr @vt1_1_rv, i32 0) |
| 184 | + %result = call i32 %fptr(ptr %obj, i32 1) |
| 185 | + ret i32 %result |
| 186 | +} |
| 187 | + |
| 188 | +declare i1 @llvm.type.test(ptr, metadata) |
| 189 | +declare void @llvm.assume(i1) |
| 190 | + |
| 191 | +!0 = !{i32 0, !"typeid1"} |
| 192 | +!1 = !{i32 0, !"typeid2"} |
| 193 | +!2 = !{i32 0, !"typeid3"} |
| 194 | +!3 = !{i32 0, !4} |
| 195 | +!4 = distinct !{} |
| 196 | +!5 = !{i32 0, !"typeid1_rv"} |
| 197 | +!6 = !{i32 0, !"typeid2_rv"} |
| 198 | +!7 = !{i32 0, !"typeid3_rv"} |
| 199 | +!8 = !{i32 0, !9} |
| 200 | +!9 = distinct !{} |
| 201 | +!10 = !{!"function_entry_count", i64 1000} |
| 202 | + |
| 203 | +attributes #0 = { "target-features"="+retpoline" } |
0 commit comments