; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 4
; RUN: opt < %s -passes=instcombine -S | FileCheck %s
define i32 @test_or_fshl(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_or_fshl(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], [[C]]
; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[B]], [[D]]
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.fshl.i32(i32 [[TMP1]], i32 [[TMP2]], i32 [[SH]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
%val2 = call i32 @llvm.fshl.i32(i32 %c, i32 %d, i32 %sh)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_and_fshl(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_and_fshl(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[A]], [[C]]
; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[B]], [[D]]
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.fshl.i32(i32 [[TMP1]], i32 [[TMP2]], i32 [[SH]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
%val2 = call i32 @llvm.fshl.i32(i32 %c, i32 %d, i32 %sh)
%ret = and i32 %val1, %val2
ret i32 %ret
}
define i32 @test_xor_fshl(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_xor_fshl(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[A]], [[C]]
; CHECK-NEXT: [[TMP2:%.*]] = xor i32 [[B]], [[D]]
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.fshl.i32(i32 [[TMP1]], i32 [[TMP2]], i32 [[SH]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
%val2 = call i32 @llvm.fshl.i32(i32 %c, i32 %d, i32 %sh)
%ret = xor i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_fshr(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_or_fshr(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], [[C]]
; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[B]], [[D]]
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.fshr.i32(i32 [[TMP1]], i32 [[TMP2]], i32 [[SH]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshr.i32(i32 %a, i32 %b, i32 %sh)
%val2 = call i32 @llvm.fshr.i32(i32 %c, i32 %d, i32 %sh)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_fshl_cascade(i32 %a, i32 %b, i32 %c) {
; CHECK-LABEL: define i32 @test_or_fshl_cascade(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], [[B]]
; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[A]], [[B]]
; CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP1]], [[C]]
; CHECK-NEXT: [[TMP4:%.*]] = or i32 [[TMP2]], [[C]]
; CHECK-NEXT: [[OR2:%.*]] = call i32 @llvm.fshl.i32(i32 [[TMP3]], i32 [[TMP4]], i32 24)
; CHECK-NEXT: ret i32 [[OR2]]
;
%fshl1 = call i32 @llvm.fshl.i32(i32 %a, i32 %a, i32 24)
%fshl2 = call i32 @llvm.fshl.i32(i32 %b, i32 %b, i32 24)
%fshl3 = call i32 @llvm.fshl.i32(i32 %c, i32 %c, i32 24)
%or1 = or i32 %fshl1, %fshl2
%or2 = or i32 %or1, %fshl3
ret i32 %or2
}
define i32 @test_or_bitreverse(i32 %a, i32 %b) {
; CHECK-LABEL: define i32 @test_or_bitreverse(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], [[B]]
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.bitreverse.i32(i32 [[TMP1]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.bitreverse.i32(i32 %a)
%val2 = call i32 @llvm.bitreverse.i32(i32 %b)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_bitreverse_constant(i32 %a, i32 %b) {
; CHECK-LABEL: define i32 @test_or_bitreverse_constant(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], 255
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.bitreverse.i32(i32 [[TMP1]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.bitreverse.i32(i32 %a)
%ret = or i32 %val1, 4278190080
ret i32 %ret
}
define i32 @test_or_bswap(i32 %a, i32 %b) {
; CHECK-LABEL: define i32 @test_or_bswap(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], [[B]]
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.bswap.i32(i32 [[TMP1]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.bswap.i32(i32 %a)
%val2 = call i32 @llvm.bswap.i32(i32 %b)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_bswap_constant(i32 %a, i32 %b) {
; CHECK-LABEL: define i32 @test_or_bswap_constant(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[A]], 255
; CHECK-NEXT: [[RET:%.*]] = call i32 @llvm.bswap.i32(i32 [[TMP1]])
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.bswap.i32(i32 %a)
%ret = or i32 %val1, 4278190080
ret i32 %ret
}
; Negative tests
define i32 @test_or_fshl_fshr(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_or_fshl_fshr(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.fshl.i32(i32 [[A]], i32 [[B]], i32 [[SH]])
; CHECK-NEXT: [[VAL2:%.*]] = call i32 @llvm.fshr.i32(i32 [[C]], i32 [[D]], i32 [[SH]])
; CHECK-NEXT: [[RET:%.*]] = or i32 [[VAL1]], [[VAL2]]
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
%val2 = call i32 @llvm.fshr.i32(i32 %c, i32 %d, i32 %sh)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_bitreverse_bswap(i32 %a, i32 %b) {
; CHECK-LABEL: define i32 @test_or_bitreverse_bswap(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.bitreverse.i32(i32 [[A]])
; CHECK-NEXT: [[VAL2:%.*]] = call i32 @llvm.bswap.i32(i32 [[B]])
; CHECK-NEXT: [[RET:%.*]] = or i32 [[VAL1]], [[VAL2]]
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.bitreverse.i32(i32 %a)
%val2 = call i32 @llvm.bswap.i32(i32 %b)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_fshl_mismatched_shamt(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh1, i32 %sh2) {
; CHECK-LABEL: define i32 @test_or_fshl_mismatched_shamt(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH1:%.*]], i32 [[SH2:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.fshl.i32(i32 [[A]], i32 [[B]], i32 [[SH1]])
; CHECK-NEXT: [[VAL2:%.*]] = call i32 @llvm.fshl.i32(i32 [[C]], i32 [[D]], i32 [[SH2]])
; CHECK-NEXT: [[RET:%.*]] = or i32 [[VAL1]], [[VAL2]]
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh1)
%val2 = call i32 @llvm.fshl.i32(i32 %c, i32 %d, i32 %sh2)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_add_fshl(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_add_fshl(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.fshl.i32(i32 [[A]], i32 [[B]], i32 [[SH]])
; CHECK-NEXT: [[VAL2:%.*]] = call i32 @llvm.fshl.i32(i32 [[C]], i32 [[D]], i32 [[SH]])
; CHECK-NEXT: [[RET:%.*]] = add i32 [[VAL1]], [[VAL2]]
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
%val2 = call i32 @llvm.fshl.i32(i32 %c, i32 %d, i32 %sh)
%ret = add i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_fshl_multiuse(i32 %a, i32 %b, i32 %c, i32 %d, i32 %sh) {
; CHECK-LABEL: define i32 @test_or_fshl_multiuse(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[C:%.*]], i32 [[D:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.fshl.i32(i32 [[A]], i32 [[B]], i32 [[SH]])
; CHECK-NEXT: call void @use(i32 [[VAL1]])
; CHECK-NEXT: [[VAL2:%.*]] = call i32 @llvm.fshl.i32(i32 [[C]], i32 [[D]], i32 [[SH]])
; CHECK-NEXT: [[RET:%.*]] = or i32 [[VAL1]], [[VAL2]]
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
call void @use(i32 %val1)
%val2 = call i32 @llvm.fshl.i32(i32 %c, i32 %d, i32 %sh)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_bitreverse_multiuse(i32 %a, i32 %b) {
; CHECK-LABEL: define i32 @test_or_bitreverse_multiuse(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.bitreverse.i32(i32 [[A]])
; CHECK-NEXT: call void @use(i32 [[VAL1]])
; CHECK-NEXT: [[VAL2:%.*]] = call i32 @llvm.bitreverse.i32(i32 [[B]])
; CHECK-NEXT: [[RET:%.*]] = or i32 [[VAL1]], [[VAL2]]
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.bitreverse.i32(i32 %a)
call void @use(i32 %val1)
%val2 = call i32 @llvm.bitreverse.i32(i32 %b)
%ret = or i32 %val1, %val2
ret i32 %ret
}
define i32 @test_or_fshl_constant(i32 %a, i32 %b, i32 %sh) {
; CHECK-LABEL: define i32 @test_or_fshl_constant(
; CHECK-SAME: i32 [[A:%.*]], i32 [[B:%.*]], i32 [[SH:%.*]]) {
; CHECK-NEXT: [[VAL1:%.*]] = call i32 @llvm.fshl.i32(i32 [[A]], i32 [[B]], i32 [[SH]])
; CHECK-NEXT: [[RET:%.*]] = or i32 [[VAL1]], -16777216
; CHECK-NEXT: ret i32 [[RET]]
;
%val1 = call i32 @llvm.fshl.i32(i32 %a, i32 %b, i32 %sh)
%ret = or i32 %val1, 4278190080
ret i32 %ret
}
declare void @use(i32)
declare i32 @llvm.fshl.i32(i32, i32, i32)
declare i32 @llvm.fshr.i32(i32, i32, i32)
declare i32 @llvm.bitreverse.i32(i32)
declare i32 @llvm.bswap.i32(i32)