; 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)