129 lines
3.6 KiB
LLVM
129 lines
3.6 KiB
LLVM
|
; RUN: opt -S -jump-threading < %s | FileCheck %s
|
||
|
|
||
|
; Check that based solely on static profile estimation we don't update the
|
||
|
; branch-weight metadata. Even if the function has an entry frequency, a
|
||
|
; completely cold part of the CFG may be statically estimated.
|
||
|
|
||
|
; For example in the loop below, jump threading would update the weight of the
|
||
|
; loop-exiting branch to 0, drastically inflating the frequency of the loop
|
||
|
; (in the range of billions).
|
||
|
;
|
||
|
; This is the CFG of the loop. There is no run-time profile info for edges
|
||
|
; inside the loop, so branch and block frequencies are estimated as shown:
|
||
|
;
|
||
|
; check_1 (16)
|
||
|
; (8) / |
|
||
|
; eq_1 | (8)
|
||
|
; \ |
|
||
|
; check_2 (16)
|
||
|
; (8) / |
|
||
|
; eq_2 | (8)
|
||
|
; \ |
|
||
|
; check_3 (16)
|
||
|
; (1) / |
|
||
|
; (loop exit) | (15)
|
||
|
; |
|
||
|
; latch
|
||
|
; |
|
||
|
; (back edge)
|
||
|
;
|
||
|
; First we thread eq_1->check_2 to check_3. Frequencies are updated to remove
|
||
|
; the frequency of eq_1 from check_2 and then the false edge leaving check_2
|
||
|
; (changed frequencies are highlighted with * *):
|
||
|
;
|
||
|
; check_1 (16)
|
||
|
; (8) / |
|
||
|
; eq_1~ | (8)
|
||
|
; / |
|
||
|
; / check_2 (*8*)
|
||
|
; / (8) / |
|
||
|
; \ eq_2 | (*0*)
|
||
|
; \ \ |
|
||
|
; ` --- check_3 (16)
|
||
|
; (1) / |
|
||
|
; (loop exit) | (15)
|
||
|
; |
|
||
|
; latch
|
||
|
; |
|
||
|
; (back edge)
|
||
|
;
|
||
|
; Next we thread eq_1->check_3 and eq_2->check_3 to check_1 as new edges to
|
||
|
; the loop latch. Frequencies are updated to remove the frequency of eq_1
|
||
|
; and eq_3 from check_3 and then the false edge leaving check_3 (changed
|
||
|
; frequencies are highlighted with * *):
|
||
|
;
|
||
|
; check_1 (16)
|
||
|
; (8) / |
|
||
|
; eq_1~ | (8)
|
||
|
; / |
|
||
|
; / check_2 (*8*)
|
||
|
; / (8) / |
|
||
|
; /-- eq_2~ | (*0*)
|
||
|
; / |
|
||
|
; / check_3 (*0*)
|
||
|
; / (*0*) / |
|
||
|
; | (loop exit) | (*0*)
|
||
|
; \ |
|
||
|
; `--------- latch
|
||
|
; |
|
||
|
; (back edge)
|
||
|
;
|
||
|
; As a result, the loop exit edge ends up with 0 frequency which in turn makes
|
||
|
; the loop header to have maximum frequency.
|
||
|
|
||
|
declare void @bar()
|
||
|
|
||
|
define void @foo(i32 *%p, i32 %n) !prof !0 {
|
||
|
entry:
|
||
|
%enter_loop = icmp eq i32 %n, 0
|
||
|
br i1 %enter_loop, label %exit, label %check_1, !prof !1
|
||
|
; CHECK: br i1 %enter_loop, label %exit, label %check_1, !prof !1
|
||
|
|
||
|
check_1:
|
||
|
%v = load i32, i32* %p
|
||
|
%cond1 = icmp eq i32 %v, 1
|
||
|
br i1 %cond1, label %eq_1, label %check_2
|
||
|
; No metadata:
|
||
|
; CHECK: br i1 %cond1, label %check_2.thread, label %check_2{{$}}
|
||
|
|
||
|
eq_1:
|
||
|
call void @bar()
|
||
|
br label %check_2
|
||
|
; Verify the new edge:
|
||
|
; CHECK: check_2.thread:
|
||
|
; CHECK-NEXT: call void @bar()
|
||
|
; CHECK-NEXT: br label %latch
|
||
|
|
||
|
check_2:
|
||
|
%cond2 = icmp eq i32 %v, 2
|
||
|
br i1 %cond2, label %eq_2, label %check_3
|
||
|
; No metadata:
|
||
|
; CHECK: br i1 %cond2, label %eq_2, label %check_3{{$}}
|
||
|
|
||
|
eq_2:
|
||
|
call void @bar()
|
||
|
br label %check_3
|
||
|
; Verify the new edge:
|
||
|
; CHECK: eq_2:
|
||
|
; CHECK-NEXT: call void @bar()
|
||
|
; CHECK-NEXT: br label %latch
|
||
|
|
||
|
check_3:
|
||
|
%condE = icmp eq i32 %v, 3
|
||
|
br i1 %condE, label %exit, label %latch
|
||
|
; No metadata:
|
||
|
; CHECK: br i1 %condE, label %exit, label %latch{{$}}
|
||
|
|
||
|
latch:
|
||
|
br label %check_1
|
||
|
|
||
|
exit:
|
||
|
ret void
|
||
|
}
|
||
|
|
||
|
!0 = !{!"function_entry_count", i64 120}
|
||
|
; CHECK-NOT: branch_weights
|
||
|
!1 = !{!"branch_weights", i32 119, i32 1}
|
||
|
; CHECK: !1 = !{!"branch_weights", i32 119, i32 1}
|
||
|
; CHECK-NOT: branch_weights
|