2021-03-08 14:23:20 +01:00
|
|
|
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
2020-08-04 10:07:28 +02:00
|
|
|
; RUN: opt < %s -jump-threading -S -verify | FileCheck %s
|
|
|
|
|
|
|
|
target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128"
|
|
|
|
target triple = "x86_64-unknown-linux-gnu"
|
|
|
|
|
|
|
|
@a = global i32 0, align 4
|
|
|
|
|
2021-03-08 14:23:20 +01:00
|
|
|
; Verify that we branch (twice) on cond2 without checking ptr.
|
|
|
|
; Verify that we eliminate "bb.file".
|
|
|
|
|
2020-08-04 10:07:28 +02:00
|
|
|
define void @foo(i32 %cond1, i32 %cond2) {
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK-LABEL: @foo(
|
|
|
|
; CHECK-NEXT: entry:
|
|
|
|
; CHECK-NEXT: [[TOBOOL:%.*]] = icmp eq i32 [[COND1:%.*]], 0
|
|
|
|
; CHECK-NEXT: br i1 [[TOBOOL]], label [[BB_COND2_THREAD:%.*]], label [[BB_COND2:%.*]]
|
|
|
|
; CHECK: bb.cond2:
|
|
|
|
; CHECK-NEXT: call void @f1()
|
|
|
|
; CHECK-NEXT: [[TOBOOL1:%.*]] = icmp eq i32 [[COND2:%.*]], 0
|
|
|
|
; CHECK-NEXT: br i1 [[TOBOOL1]], label [[BB_F4:%.*]], label [[BB_F2:%.*]]
|
|
|
|
; CHECK: bb.cond2.thread:
|
|
|
|
; CHECK-NEXT: [[TOBOOL12:%.*]] = icmp eq i32 [[COND2]], 0
|
|
|
|
; CHECK-NEXT: br i1 [[TOBOOL12]], label [[BB_F3:%.*]], label [[BB_F2]]
|
|
|
|
; CHECK: bb.f2:
|
|
|
|
; CHECK-NEXT: call void @f2()
|
|
|
|
; CHECK-NEXT: br label [[EXIT:%.*]]
|
|
|
|
; CHECK: bb.f3:
|
|
|
|
; CHECK-NEXT: call void @f3()
|
|
|
|
; CHECK-NEXT: br label [[EXIT]]
|
|
|
|
; CHECK: bb.f4:
|
|
|
|
; CHECK-NEXT: [[PTR3:%.*]] = phi i32* [ null, [[BB_COND2]] ]
|
|
|
|
; CHECK-NEXT: call void @f4()
|
|
|
|
; CHECK-NEXT: br label [[EXIT]]
|
|
|
|
; CHECK: exit:
|
|
|
|
; CHECK-NEXT: ret void
|
|
|
|
;
|
2020-08-04 10:07:28 +02:00
|
|
|
entry:
|
|
|
|
%tobool = icmp eq i32 %cond1, 0
|
|
|
|
br i1 %tobool, label %bb.cond2, label %bb.f1
|
|
|
|
|
|
|
|
bb.f1:
|
|
|
|
call void @f1()
|
|
|
|
br label %bb.cond2
|
|
|
|
|
|
|
|
bb.cond2:
|
|
|
|
%ptr = phi i32* [ null, %bb.f1 ], [ @a, %entry ]
|
|
|
|
%tobool1 = icmp eq i32 %cond2, 0
|
|
|
|
br i1 %tobool1, label %bb.file, label %bb.f2
|
|
|
|
|
|
|
|
bb.f2:
|
|
|
|
call void @f2()
|
|
|
|
br label %exit
|
|
|
|
|
|
|
|
bb.file:
|
|
|
|
%cmp = icmp eq i32* %ptr, null
|
|
|
|
br i1 %cmp, label %bb.f4, label %bb.f3
|
|
|
|
|
|
|
|
bb.f3:
|
|
|
|
call void @f3()
|
|
|
|
br label %exit
|
|
|
|
|
|
|
|
bb.f4:
|
|
|
|
call void @f4()
|
|
|
|
br label %exit
|
|
|
|
|
|
|
|
exit:
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
declare void @f1()
|
|
|
|
declare void @f2()
|
|
|
|
declare void @f3()
|
|
|
|
declare void @f4()
|
|
|
|
|
|
|
|
|
2021-03-08 14:23:20 +01:00
|
|
|
; Verify that we branch (twice) on cond2 without checking tobool again.
|
|
|
|
; Verify that we eliminate "bb.cond1again".
|
|
|
|
|
2020-08-04 10:07:28 +02:00
|
|
|
define void @foo2(i32 %cond1, i32 %cond2) {
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK-LABEL: @foo2(
|
|
|
|
; CHECK-NEXT: entry:
|
|
|
|
; CHECK-NEXT: [[TOBOOL:%.*]] = icmp ne i32 [[COND1:%.*]], 0
|
|
|
|
; CHECK-NEXT: br i1 [[TOBOOL]], label [[BB_COND2:%.*]], label [[BB_COND2_THREAD:%.*]]
|
|
|
|
; CHECK: bb.cond2:
|
|
|
|
; CHECK-NEXT: call void @f1()
|
|
|
|
; CHECK-NEXT: [[TOBOOL1:%.*]] = icmp eq i32 [[COND2:%.*]], 0
|
|
|
|
; CHECK-NEXT: br i1 [[TOBOOL1]], label [[EXIT:%.*]], label [[BB_F3:%.*]]
|
|
|
|
; CHECK: bb.cond2.thread:
|
|
|
|
; CHECK-NEXT: call void @f2()
|
|
|
|
; CHECK-NEXT: [[TOBOOL11:%.*]] = icmp eq i32 [[COND2]], 0
|
|
|
|
; CHECK-NEXT: br i1 [[TOBOOL11]], label [[EXIT]], label [[BB_F4:%.*]]
|
|
|
|
; CHECK: bb.f3:
|
|
|
|
; CHECK-NEXT: call void @f3()
|
|
|
|
; CHECK-NEXT: br label [[EXIT]]
|
|
|
|
; CHECK: bb.f4:
|
|
|
|
; CHECK-NEXT: call void @f4()
|
|
|
|
; CHECK-NEXT: br label [[EXIT]]
|
|
|
|
; CHECK: exit:
|
|
|
|
; CHECK-NEXT: ret void
|
|
|
|
;
|
2020-08-04 10:07:28 +02:00
|
|
|
entry:
|
|
|
|
%tobool = icmp ne i32 %cond1, 0
|
|
|
|
br i1 %tobool, label %bb.f1, label %bb.f2
|
|
|
|
|
|
|
|
bb.f1:
|
|
|
|
call void @f1()
|
|
|
|
br label %bb.cond2
|
|
|
|
|
|
|
|
bb.f2:
|
|
|
|
call void @f2()
|
|
|
|
br label %bb.cond2
|
|
|
|
|
|
|
|
bb.cond2:
|
|
|
|
%tobool1 = icmp eq i32 %cond2, 0
|
|
|
|
br i1 %tobool1, label %exit, label %bb.cond1again
|
|
|
|
|
|
|
|
bb.cond1again:
|
|
|
|
br i1 %tobool, label %bb.f3, label %bb.f4
|
|
|
|
|
|
|
|
bb.f3:
|
|
|
|
call void @f3()
|
|
|
|
br label %exit
|
|
|
|
|
|
|
|
bb.f4:
|
|
|
|
call void @f4()
|
|
|
|
br label %exit
|
|
|
|
|
|
|
|
exit:
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
; Verify that we do *not* thread any edge. We used to evaluate
|
|
|
|
; constant expressions like:
|
|
|
|
;
|
|
|
|
; icmp ugt i8* null, inttoptr (i64 4 to i8*)
|
|
|
|
;
|
|
|
|
; as "true", causing jump threading to a wrong destination.
|
2021-03-08 14:23:20 +01:00
|
|
|
|
2021-03-08 14:39:56 +01:00
|
|
|
define void @icmp_ult_null_constexpr(i8* %arg1, i8* %arg2) {
|
|
|
|
; CHECK-LABEL: @icmp_ult_null_constexpr(
|
|
|
|
; CHECK-NEXT: entry:
|
|
|
|
; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8* [[ARG1:%.*]], null
|
|
|
|
; CHECK-NEXT: br i1 [[CMP1]], label [[BB_BAR1:%.*]], label [[BB_END:%.*]]
|
|
|
|
; CHECK: bb_bar1:
|
|
|
|
; CHECK-NEXT: call void @bar(i32 1)
|
|
|
|
; CHECK-NEXT: br label [[BB_END]]
|
|
|
|
; CHECK: bb_end:
|
|
|
|
; CHECK-NEXT: [[CMP2:%.*]] = icmp ne i8* [[ARG2:%.*]], null
|
|
|
|
; CHECK-NEXT: br i1 [[CMP2]], label [[BB_CONT:%.*]], label [[BB_BAR2:%.*]]
|
|
|
|
; CHECK: bb_bar2:
|
|
|
|
; CHECK-NEXT: call void @bar(i32 2)
|
|
|
|
; CHECK-NEXT: br label [[BB_EXIT:%.*]]
|
|
|
|
; CHECK: bb_cont:
|
|
|
|
; CHECK-NEXT: [[CMP3:%.*]] = icmp ult i8* [[ARG1]], inttoptr (i64 4 to i8*)
|
|
|
|
; CHECK-NEXT: br i1 [[CMP3]], label [[BB_EXIT]], label [[BB_BAR3:%.*]]
|
|
|
|
; CHECK: bb_bar3:
|
|
|
|
; CHECK-NEXT: call void @bar(i32 3)
|
|
|
|
; CHECK-NEXT: br label [[BB_EXIT]]
|
|
|
|
; CHECK: bb_exit:
|
|
|
|
; CHECK-NEXT: ret void
|
|
|
|
;
|
|
|
|
entry:
|
|
|
|
%cmp1 = icmp eq i8* %arg1, null
|
|
|
|
br i1 %cmp1, label %bb_bar1, label %bb_end
|
|
|
|
|
|
|
|
bb_bar1:
|
|
|
|
call void @bar(i32 1)
|
|
|
|
br label %bb_end
|
|
|
|
|
|
|
|
bb_end:
|
|
|
|
%cmp2 = icmp ne i8* %arg2, null
|
|
|
|
br i1 %cmp2, label %bb_cont, label %bb_bar2
|
|
|
|
|
|
|
|
bb_bar2:
|
|
|
|
call void @bar(i32 2)
|
|
|
|
br label %bb_exit
|
|
|
|
|
|
|
|
bb_cont:
|
|
|
|
%cmp3 = icmp ult i8* %arg1, inttoptr (i64 4 to i8*)
|
|
|
|
br i1 %cmp3, label %bb_exit, label %bb_bar3
|
|
|
|
|
|
|
|
bb_bar3:
|
|
|
|
call void @bar(i32 3)
|
|
|
|
br label %bb_exit
|
|
|
|
|
|
|
|
bb_exit:
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2021-03-08 14:52:12 +01:00
|
|
|
; This is a special-case of the above pattern:
|
2021-03-08 14:39:56 +01:00
|
|
|
; Null is guaranteed to be unsigned <= all values.
|
|
|
|
|
|
|
|
define void @icmp_ule_null_constexpr(i8* %arg1, i8* %arg2) {
|
|
|
|
; CHECK-LABEL: @icmp_ule_null_constexpr(
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK-NEXT: entry:
|
|
|
|
; CHECK-NEXT: [[CMP1:%.*]] = icmp eq i8* [[ARG1:%.*]], null
|
2021-03-08 14:52:12 +01:00
|
|
|
; CHECK-NEXT: br i1 [[CMP1]], label [[BB_END_THREAD:%.*]], label [[BB_END:%.*]]
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK: bb_end:
|
|
|
|
; CHECK-NEXT: [[CMP2:%.*]] = icmp ne i8* [[ARG2:%.*]], null
|
|
|
|
; CHECK-NEXT: br i1 [[CMP2]], label [[BB_CONT:%.*]], label [[BB_BAR2:%.*]]
|
2021-03-08 14:52:12 +01:00
|
|
|
; CHECK: bb_end.thread:
|
|
|
|
; CHECK-NEXT: call void @bar(i32 1)
|
|
|
|
; CHECK-NEXT: [[CMP21:%.*]] = icmp ne i8* [[ARG2]], null
|
|
|
|
; CHECK-NEXT: br i1 [[CMP21]], label [[BB_EXIT:%.*]], label [[BB_BAR2]]
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK: bb_bar2:
|
|
|
|
; CHECK-NEXT: call void @bar(i32 2)
|
2021-03-08 14:52:12 +01:00
|
|
|
; CHECK-NEXT: br label [[BB_EXIT]]
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK: bb_cont:
|
|
|
|
; CHECK-NEXT: [[CMP3:%.*]] = icmp ule i8* [[ARG1]], inttoptr (i64 4 to i8*)
|
|
|
|
; CHECK-NEXT: br i1 [[CMP3]], label [[BB_EXIT]], label [[BB_BAR3:%.*]]
|
|
|
|
; CHECK: bb_bar3:
|
|
|
|
; CHECK-NEXT: call void @bar(i32 3)
|
|
|
|
; CHECK-NEXT: br label [[BB_EXIT]]
|
|
|
|
; CHECK: bb_exit:
|
|
|
|
; CHECK-NEXT: ret void
|
|
|
|
;
|
2020-08-04 10:07:28 +02:00
|
|
|
entry:
|
|
|
|
%cmp1 = icmp eq i8* %arg1, null
|
|
|
|
br i1 %cmp1, label %bb_bar1, label %bb_end
|
|
|
|
|
|
|
|
bb_bar1:
|
|
|
|
call void @bar(i32 1)
|
|
|
|
br label %bb_end
|
|
|
|
|
|
|
|
bb_end:
|
|
|
|
%cmp2 = icmp ne i8* %arg2, null
|
|
|
|
br i1 %cmp2, label %bb_cont, label %bb_bar2
|
|
|
|
|
|
|
|
bb_bar2:
|
|
|
|
call void @bar(i32 2)
|
|
|
|
br label %bb_exit
|
|
|
|
|
|
|
|
bb_cont:
|
|
|
|
%cmp3 = icmp ule i8* %arg1, inttoptr (i64 4 to i8*)
|
|
|
|
br i1 %cmp3, label %bb_exit, label %bb_bar3
|
|
|
|
|
|
|
|
bb_bar3:
|
|
|
|
call void @bar(i32 3)
|
|
|
|
br label %bb_exit
|
|
|
|
|
|
|
|
bb_exit:
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
declare void @bar(i32)
|
|
|
|
|
|
|
|
|
|
|
|
;; Test that we skip unconditional PredBB when threading jumps through two
|
|
|
|
;; successive basic blocks.
|
|
|
|
|
|
|
|
define i32 @foo4(i32* %0) {
|
2021-03-08 14:23:20 +01:00
|
|
|
; CHECK-LABEL: @foo4(
|
|
|
|
; CHECK-NEXT: entry:
|
|
|
|
; CHECK-NEXT: [[SIZE:%.*]] = call i64 @get_size(i32* [[TMP0:%.*]])
|
|
|
|
; CHECK-NEXT: [[GOOD:%.*]] = icmp ugt i64 [[SIZE]], 3
|
|
|
|
; CHECK-NEXT: br i1 [[GOOD]], label [[PRED_BB:%.*]], label [[PRED_PRED_BB:%.*]]
|
|
|
|
; CHECK: pred.pred.bb:
|
|
|
|
; CHECK-NEXT: call void @effect()
|
|
|
|
; CHECK-NEXT: br label [[PRED_BB]]
|
|
|
|
; CHECK: pred.bb:
|
|
|
|
; CHECK-NEXT: [[V:%.*]] = load i32, i32* [[TMP0]], align 4
|
|
|
|
; CHECK-NEXT: br label [[BB:%.*]]
|
|
|
|
; CHECK: bb:
|
|
|
|
; CHECK-NEXT: call void @effect1(i8* blockaddress(@foo4, [[BB]]))
|
|
|
|
; CHECK-NEXT: br i1 [[GOOD]], label [[EXIT:%.*]], label [[EXIT]]
|
|
|
|
; CHECK: exit:
|
|
|
|
; CHECK-NEXT: ret i32 [[V]]
|
|
|
|
;
|
2020-08-04 10:07:28 +02:00
|
|
|
entry:
|
|
|
|
%size = call i64 @get_size(i32* %0)
|
|
|
|
%good = icmp ugt i64 %size, 3
|
|
|
|
br i1 %good, label %pred.bb, label %pred.pred.bb
|
|
|
|
|
|
|
|
pred.pred.bb: ; preds = %entry
|
|
|
|
call void @effect()
|
|
|
|
br label %pred.bb
|
|
|
|
pred.bb: ; preds = %pred.pred.bb, %entry
|
|
|
|
%v = load i32, i32* %0
|
|
|
|
br label %bb
|
|
|
|
|
|
|
|
bb: ; preds = %pred.bb
|
|
|
|
call void @effect1(i8* blockaddress(@foo4, %bb))
|
|
|
|
br i1 %good, label %cont2, label %cont1
|
|
|
|
|
|
|
|
cont1: ; preds = %bb
|
|
|
|
br i1 %good, label %exit, label %cont2
|
|
|
|
cont2: ; preds = %bb
|
|
|
|
br label %exit
|
|
|
|
exit: ; preds = %cont1, %cont2
|
|
|
|
ret i32 %v
|
|
|
|
}
|
|
|
|
|
|
|
|
declare i64 @get_size(i32*)
|
|
|
|
declare void @effect()
|
|
|
|
declare void @effect1(i8*)
|