forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			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
 |