
Currently this optimization only occurs for `mul`, but this generalizes that for any operation that has a fixed point of `0`. There is similar logic within `EarlyCSE` pass, but that is stricter in terms of `poison` propagation so will not optimize for many operations. Alive2 Proofs: `and`: https://alive2.llvm.org/ce/z/RraasX ; base-case https://alive2.llvm.org/ce/z/gzfFTX ; commuted-case https://alive2.llvm.org/ce/z/63XaoX ; compare against undef https://alive2.llvm.org/ce/z/MVRVNd ; select undef https://alive2.llvm.org/ce/z/2bsoYG ; vector https://alive2.llvm.org/ce/z/xByeX- ; vector compare against undef https://alive2.llvm.org/ce/z/zNdzmZ ; vector select undef `fshl`: https://alive2.llvm.org/ce/z/U3_PG3 ; base-case https://alive2.llvm.org/ce/z/BWCnxT ; compare against undef https://alive2.llvm.org/ce/z/8HGAE_ ; select undef ; vector times out `fshr`: https://alive2.llvm.org/ce/z/o6F47G ; base-case https://alive2.llvm.org/ce/z/fVnBXy ; compare against undef https://alive2.llvm.org/ce/z/suymYJ ; select undef ; vector times out `umin`: https://alive2.llvm.org/ce/z/GGMqf6 ; base-case https://alive2.llvm.org/ce/z/6cx5-k ; commuted-case https://alive2.llvm.org/ce/z/W5d9tz ; compare against undef https://alive2.llvm.org/ce/z/nKbaUn ; select undef https://alive2.llvm.org/ce/z/gxEGqc ; vector https://alive2.llvm.org/ce/z/_SDpi_ ; vector compare against undef `sdiv`: https://alive2.llvm.org/ce/z/5XGs3q `srem`: https://alive2.llvm.org/ce/z/vXAnQM `udiv`: https://alive2.llvm.org/ce/z/e6_8Ug `urem`: https://alive2.llvm.org/ce/z/VmM2SL `shl`: https://alive2.llvm.org/ce/z/aCZr3u ; Argument with range https://alive2.llvm.org/ce/z/YgDy8C ; Instruction with known bits https://alive2.llvm.org/ce/z/6pIxR6 ; Constant `lshr`: https://alive2.llvm.org/ce/z/WCCBej `ashr: https://alive2.llvm.org/ce/z/egV4TR --------- Co-authored-by: Ryan Buchner <rbuchner@ventanamicro.com> Co-authored-by: Yingwei Zheng <dtcxzyw@qq.com>
222 lines
7.0 KiB
LLVM
222 lines
7.0 KiB
LLVM
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
|
; RUN: opt -S -passes=instcombine < %s | FileCheck %s
|
|
|
|
; (select (icmp x, 0, eq), 0, (umin x, y)) -> (umin x, y)
|
|
define i64 @umin_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @umin_select(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[UMIN:%.*]] = call i64 @llvm.umin.i64(i64 [[A:%.*]], i64 [[B_FR]])
|
|
; CHECK-NEXT: ret i64 [[UMIN]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%umin = call i64 @llvm.umin.i64(i64 %a, i64 %b)
|
|
%select = select i1 %cond, i64 0, i64 %umin
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (mul x, y)) -> (mul x, y)
|
|
define i64 @mul_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @mul_select(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[MUL:%.*]] = mul i64 [[A:%.*]], [[B_FR]]
|
|
; CHECK-NEXT: ret i64 [[MUL]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%mul = mul i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %mul
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (mul x, y)) -> (mul x, y)
|
|
define i64 @mul_select_comm(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @mul_select_comm(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[MUL:%.*]] = mul i64 [[B_FR]], [[A:%.*]]
|
|
; CHECK-NEXT: ret i64 [[MUL]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%mul = mul i64 %b, %a
|
|
%select = select i1 %cond, i64 0, i64 %mul
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (shl x, y)) -> (shl x, y)
|
|
define i64 @shl_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @shl_select(
|
|
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[A:%.*]], 0
|
|
; CHECK-NEXT: [[SHL:%.*]] = shl i64 [[A]], [[B_FR:%.*]]
|
|
; CHECK-NEXT: [[SELECT:%.*]] = select i1 [[COND]], i64 0, i64 [[SHL]]
|
|
; CHECK-NEXT: ret i64 [[SELECT]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%shl = shl i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %shl
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (and x, y)) -> (and x, y)
|
|
define i64 @and_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @and_select(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[AND:%.*]] = and i64 [[A:%.*]], [[B_FR]]
|
|
; CHECK-NEXT: ret i64 [[AND]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%and = and i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %and
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (and x, y)) -> (and x, y)
|
|
define i64 @and_select_comm(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @and_select_comm(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[AND:%.*]] = and i64 [[B_FR]], [[A:%.*]]
|
|
; CHECK-NEXT: ret i64 [[AND]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%and = and i64 %b, %a
|
|
%select = select i1 %cond, i64 0, i64 %and
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, ne), (ashr x, y), 0) -> (ashr x, y)
|
|
define i64 @ashr_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @ashr_select(
|
|
; CHECK-NEXT: [[COND_NOT:%.*]] = icmp eq i64 [[A:%.*]], 0
|
|
; CHECK-NEXT: [[ASHR:%.*]] = ashr i64 [[A]], [[B_FR:%.*]]
|
|
; CHECK-NEXT: [[SELECT:%.*]] = select i1 [[COND_NOT]], i64 0, i64 [[ASHR]]
|
|
; CHECK-NEXT: ret i64 [[SELECT]]
|
|
;
|
|
%cond = icmp ne i64 0, %a
|
|
%ashr = ashr i64 %a, %b
|
|
%select = select i1 %cond, i64 %ashr, i64 0
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, ne), (lshr x, y), 0) -> (lshr x, y)
|
|
define i64 @lshr_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @lshr_select(
|
|
; CHECK-NEXT: [[COND_NOT:%.*]] = icmp eq i64 [[A:%.*]], 0
|
|
; CHECK-NEXT: [[LSHR:%.*]] = lshr i64 [[A]], [[B_FR:%.*]]
|
|
; CHECK-NEXT: [[SELECT:%.*]] = select i1 [[COND_NOT]], i64 0, i64 [[LSHR]]
|
|
; CHECK-NEXT: ret i64 [[SELECT]]
|
|
;
|
|
%cond = icmp ne i64 0, %a
|
|
%lshr = lshr i64 %a, %b
|
|
%select = select i1 %cond, i64 %lshr, i64 0
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, fshr(x, x, y)) -> fshr(x, x, y)
|
|
define i64 @fshr_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @fshr_select(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[FSHR:%.*]] = call i64 @llvm.fshr.i64(i64 [[A:%.*]], i64 [[A]], i64 [[B_FR]])
|
|
; CHECK-NEXT: ret i64 [[FSHR]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%fshr = call i64 @llvm.fshr.i64(i64 %a, i64 %a, i64 %b)
|
|
%select = select i1 %cond, i64 0, i64 %fshr
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (fshl x, x, y)) -> (fshl x, x, y)
|
|
define i64 @fshl_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @fshl_select(
|
|
; CHECK-NEXT: [[B_FR:%.*]] = freeze i64 [[B:%.*]]
|
|
; CHECK-NEXT: [[FSHL:%.*]] = call i64 @llvm.fshl.i64(i64 [[A:%.*]], i64 [[A]], i64 [[B_FR]])
|
|
; CHECK-NEXT: ret i64 [[FSHL]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%fshl = call i64 @llvm.fshl.i64(i64 %a, i64 %a, i64 %b)
|
|
%select = select i1 %cond, i64 0, i64 %fshl
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (fshr x, z, y)) -> leave as is
|
|
define i64 @fshr_select_no_combine(i64 %a, i64 %b, i64 %c) {
|
|
; CHECK-LABEL: @fshr_select_no_combine(
|
|
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[A:%.*]], 0
|
|
; CHECK-NEXT: [[FSHR:%.*]] = call i64 @llvm.fshr.i64(i64 [[A]], i64 [[B:%.*]], i64 [[C:%.*]])
|
|
; CHECK-NEXT: [[SELECT:%.*]] = select i1 [[COND]], i64 0, i64 [[FSHR]]
|
|
; CHECK-NEXT: ret i64 [[SELECT]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%fshr = call i64 @llvm.fshr.i64(i64 %a, i64 %b, i64 %c)
|
|
%select = select i1 %cond, i64 0, i64 %fshr
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (sdiv x, y)) -> (sdiv x, y)
|
|
define i64 @sdiv_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @sdiv_select(
|
|
; CHECK-NEXT: [[DIV:%.*]] = sdiv i64 [[A:%.*]], [[B_FR:%.*]]
|
|
; CHECK-NEXT: ret i64 [[DIV]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%div = sdiv i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %div
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (udiv x, y)) -> (udiv x, y)
|
|
define i64 @udiv_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @udiv_select(
|
|
; CHECK-NEXT: [[DIV:%.*]] = udiv i64 [[A:%.*]], [[B_FR:%.*]]
|
|
; CHECK-NEXT: ret i64 [[DIV]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%div = udiv i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %div
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (srem x, y)) -> (srem x, y)
|
|
define i64 @srem_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @srem_select(
|
|
; CHECK-NEXT: [[REM:%.*]] = srem i64 [[A:%.*]], [[B:%.*]]
|
|
; CHECK-NEXT: ret i64 [[REM]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%rem = srem i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %rem
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (urem x, y)) -> (urem x, y)
|
|
define i64 @urem_select(i64 %a, i64 %b) {
|
|
; CHECK-LABEL: @urem_select(
|
|
; CHECK-NEXT: [[REM:%.*]] = urem i64 [[A:%.*]], [[B:%.*]]
|
|
; CHECK-NEXT: ret i64 [[REM]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%rem = urem i64 %a, %b
|
|
%select = select i1 %cond, i64 0, i64 %rem
|
|
ret i64 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (icmp x, 0, slt)) -> (icmp x, 0, slt)
|
|
define i1 @icmp_slt_select(i64 %a) {
|
|
; CHECK-LABEL: @icmp_slt_select(
|
|
; CHECK-NEXT: [[ICMP:%.*]] = icmp slt i64 [[A:%.*]], 0
|
|
; CHECK-NEXT: ret i1 [[ICMP]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%icmp = icmp slt i64 %a, 0
|
|
%select = select i1 %cond, i1 0, i1 %icmp
|
|
ret i1 %select
|
|
}
|
|
|
|
; (select (icmp x, 0, eq), 0, (sub 0, x)) -> (sub 0, x)
|
|
define i64 @sub_select(i64 %a) {
|
|
; CHECK-LABEL: @sub_select(
|
|
; CHECK-NEXT: [[SUB:%.*]] = sub i64 0, [[A:%.*]]
|
|
; CHECK-NEXT: ret i64 [[SUB]]
|
|
;
|
|
%cond = icmp eq i64 %a, 0
|
|
%sub = sub i64 0, %a
|
|
%select = select i1 %cond, i64 0, i64 %sub
|
|
ret i64 %select
|
|
}
|