This patch updates SCCP to use the value ranges of AddInst operands to try to prove the AddInst does not overflow in the signed sense and adds the NSW flag. The reasoning is done with makeGuaranteedNoWrapRegion (thanks @nikic for point it out!). Follow-ups will include extending this to more OverflowingBinaryOperators. Depends on D142387. Reviewed By: nikic Differential Revision: https://reviews.llvm.org/D142390
141 lines
4.9 KiB
LLVM
141 lines
4.9 KiB
LLVM
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --function-signature
|
|
; RUN: opt -passes=ipsccp -S %s -o -| FileCheck %s
|
|
|
|
define void @caller.1(ptr %arg) {
|
|
; CHECK-LABEL: define {{[^@]+}}@caller.1
|
|
; CHECK-SAME: (ptr [[ARG:%.*]]) {
|
|
; CHECK-NEXT: [[R_1:%.*]] = tail call i32 @callee.1(i32 4)
|
|
; CHECK-NEXT: [[R_2:%.*]] = tail call i32 @callee.1(i32 2)
|
|
; CHECK-NEXT: call void @use(i32 20)
|
|
; CHECK-NEXT: ret void
|
|
;
|
|
%r.1 = tail call i32 @callee.1(i32 4)
|
|
%r.2 = tail call i32 @callee.1(i32 2)
|
|
%r.3 = add i32 %r.1, %r.2
|
|
call void @use(i32 %r.3)
|
|
ret void
|
|
}
|
|
|
|
define internal i32 @callee.1(i32 %arg) {
|
|
; CHECK-LABEL: define {{[^@]+}}@callee.1
|
|
; CHECK-SAME: (i32 [[ARG:%.*]]) {
|
|
; CHECK-NEXT: [[SEL:%.*]] = select i1 false, i32 16, i32 [[ARG]]
|
|
; CHECK-NEXT: br label [[BB10:%.*]]
|
|
; CHECK: bb10:
|
|
; CHECK-NEXT: ret i32 undef
|
|
;
|
|
%c.1 = icmp slt i32 %arg, 0
|
|
%sel = select i1 %c.1, i32 16, i32 %arg
|
|
%c.2 = icmp eq i32 %sel, 0
|
|
br i1 %c.2, label %bb12, label %bb10
|
|
|
|
bb10: ; preds = %bb8
|
|
ret i32 10
|
|
|
|
bb12: ; preds = %bb8, %bb3, %bb
|
|
ret i32 12
|
|
}
|
|
|
|
declare void @use(i32)
|
|
|
|
define internal i1 @f1(i32 %x, i32 %y, i1 %cmp) {
|
|
; CHECK-LABEL: define {{[^@]+}}@f1
|
|
; CHECK-SAME: (i32 [[X:%.*]], i32 [[Y:%.*]], i1 [[CMP:%.*]]) {
|
|
; CHECK-NEXT: [[SEL_1:%.*]] = select i1 [[CMP]], i32 [[X]], i32 [[Y]]
|
|
; CHECK-NEXT: [[C_2:%.*]] = icmp sgt i32 [[SEL_1]], 100
|
|
; CHECK-NEXT: [[C_3:%.*]] = icmp eq i32 [[SEL_1]], 50
|
|
; CHECK-NEXT: [[RES_1:%.*]] = add nuw nsw i1 false, [[C_2]]
|
|
; CHECK-NEXT: [[RES_2:%.*]] = add i1 [[RES_1]], [[C_3]]
|
|
; CHECK-NEXT: [[RES_3:%.*]] = add nuw nsw i1 [[RES_2]], false
|
|
; CHECK-NEXT: ret i1 [[RES_3]]
|
|
;
|
|
%sel.1 = select i1 %cmp, i32 %x, i32 %y
|
|
%c.1 = icmp sgt i32 %sel.1, 300
|
|
%c.2 = icmp sgt i32 %sel.1, 100
|
|
%c.3 = icmp eq i32 %sel.1, 50
|
|
%c.4 = icmp slt i32 %sel.1, 9
|
|
%res.1 = add i1 %c.1, %c.2
|
|
%res.2 = add i1 %res.1, %c.3
|
|
%res.3 = add i1 %res.2, %c.4
|
|
ret i1 %res.3
|
|
}
|
|
|
|
define i1 @caller1(i1 %cmp) {
|
|
; CHECK-LABEL: define {{[^@]+}}@caller1
|
|
; CHECK-SAME: (i1 [[CMP:%.*]]) {
|
|
; CHECK-NEXT: [[CALL_1:%.*]] = tail call i1 @f1(i32 10, i32 100, i1 [[CMP]])
|
|
; CHECK-NEXT: [[CALL_2:%.*]] = tail call i1 @f1(i32 20, i32 200, i1 [[CMP]])
|
|
; CHECK-NEXT: [[RES:%.*]] = and i1 [[CALL_1]], [[CALL_2]]
|
|
; CHECK-NEXT: ret i1 [[RES]]
|
|
;
|
|
%call.1 = tail call i1 @f1(i32 10, i32 100, i1 %cmp)
|
|
%call.2 = tail call i1 @f1(i32 20, i32 200, i1 %cmp)
|
|
%res = and i1 %call.1, %call.2
|
|
ret i1 %res
|
|
}
|
|
|
|
|
|
define i1 @f2(i32 %x, i32 %y, i1 %cmp) {
|
|
; CHECK-LABEL: define {{[^@]+}}@f2
|
|
; CHECK-SAME: (i32 [[X:%.*]], i32 [[Y:%.*]], i1 [[CMP:%.*]]) {
|
|
; CHECK-NEXT: [[SEL_1:%.*]] = select i1 [[CMP]], i32 [[X]], i32 [[Y]]
|
|
; CHECK-NEXT: [[C_1:%.*]] = icmp sgt i32 [[SEL_1]], 300
|
|
; CHECK-NEXT: [[C_2:%.*]] = icmp sgt i32 [[SEL_1]], 100
|
|
; CHECK-NEXT: [[C_3:%.*]] = icmp eq i32 [[SEL_1]], 50
|
|
; CHECK-NEXT: [[C_4:%.*]] = icmp slt i32 [[SEL_1]], 9
|
|
; CHECK-NEXT: [[RES_1:%.*]] = add i1 [[C_1]], [[C_2]]
|
|
; CHECK-NEXT: [[RES_2:%.*]] = add i1 [[RES_1]], [[C_3]]
|
|
; CHECK-NEXT: [[RES_3:%.*]] = add i1 [[RES_2]], [[C_4]]
|
|
; CHECK-NEXT: ret i1 [[RES_3]]
|
|
;
|
|
%sel.1 = select i1 %cmp, i32 %x, i32 %y
|
|
%c.1 = icmp sgt i32 %sel.1, 300
|
|
%c.2 = icmp sgt i32 %sel.1, 100
|
|
%c.3 = icmp eq i32 %sel.1, 50
|
|
%c.4 = icmp slt i32 %sel.1, 9
|
|
%res.1 = add i1 %c.1, %c.2
|
|
%res.2 = add i1 %res.1, %c.3
|
|
%res.3 = add i1 %res.2, %c.4
|
|
ret i1 %res.3
|
|
}
|
|
|
|
define i1 @caller2(i32 %y, i1 %cmp) {
|
|
; CHECK-LABEL: define {{[^@]+}}@caller2
|
|
; CHECK-SAME: (i32 [[Y:%.*]], i1 [[CMP:%.*]]) {
|
|
; CHECK-NEXT: [[CALL_1:%.*]] = tail call i1 @f2(i32 10, i32 [[Y]], i1 [[CMP]])
|
|
; CHECK-NEXT: [[CALL_2:%.*]] = tail call i1 @f2(i32 20, i32 [[Y]], i1 [[CMP]])
|
|
; CHECK-NEXT: [[RES:%.*]] = and i1 [[CALL_1]], [[CALL_2]]
|
|
; CHECK-NEXT: ret i1 [[RES]]
|
|
;
|
|
%call.1 = tail call i1 @f2(i32 10, i32 %y, i1 %cmp)
|
|
%call.2 = tail call i1 @f2(i32 20, i32 %y, i1 %cmp)
|
|
%res = and i1 %call.1, %call.2
|
|
ret i1 %res
|
|
}
|
|
|
|
@GV = common global i32 0, align 4
|
|
|
|
define i32 @f3_constantexpr_cond(i32 %x, i32 %y) {
|
|
; CHECK-LABEL: define {{[^@]+}}@f3_constantexpr_cond
|
|
; CHECK-SAME: (i32 [[X:%.*]], i32 [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[SEL_1:%.*]] = select i1 icmp eq (ptr @f3_constantexpr_cond, ptr @GV), i32 [[X]], i32 [[Y]]
|
|
; CHECK-NEXT: ret i32 [[SEL_1]]
|
|
;
|
|
%sel.1 = select i1 icmp eq (ptr @f3_constantexpr_cond, ptr @GV), i32 %x, i32 %y
|
|
ret i32 %sel.1
|
|
}
|
|
|
|
define i32 @caller3(i32 %y) {
|
|
; CHECK-LABEL: define {{[^@]+}}@caller3
|
|
; CHECK-SAME: (i32 [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[CALL_1:%.*]] = tail call i32 @f3_constantexpr_cond(i32 10, i32 [[Y]])
|
|
; CHECK-NEXT: [[CALL_2:%.*]] = tail call i32 @f3_constantexpr_cond(i32 20, i32 [[Y]])
|
|
; CHECK-NEXT: [[RES:%.*]] = and i32 [[CALL_1]], [[CALL_2]]
|
|
; CHECK-NEXT: ret i32 [[RES]]
|
|
;
|
|
%call.1 = tail call i32 @f3_constantexpr_cond(i32 10, i32 %y)
|
|
%call.2 = tail call i32 @f3_constantexpr_cond(i32 20, i32 %y)
|
|
%res = and i32 %call.1, %call.2
|
|
ret i32 %res
|
|
}
|