220 lines
9.3 KiB
LLVM
220 lines
9.3 KiB
LLVM
; 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)
|