forked from OSchip/llvm-project
[RISCV] Add support for bitcasts between scalars and fixed-length vectors
This patch supports bitcasts from scalar types to fixed-length vectors and vice versa. It custom-lowers and custom-legalizes them to EXTRACT_VECTOR_ELT/INSERT_VECTOR_ELT operations, using a single-element vectors to hold the scalar where appropriate. Previously, some of these would fail to select, others would be expanded through stack loads and stores. Effort was made to ensure the codegen avoids the stack for both legal and illegal scalar types. Some of the codegen could be improved, but on first glance it looks like a general optimization of EXTRACT_VECTOR_ELT when extracting an i64 element on RV32. Reviewed By: craig.topper Differential Revision: https://reviews.llvm.org/D99667
This commit is contained in:
parent
c590a9880d
commit
af3a839c70
|
@ -732,6 +732,15 @@ RISCVTargetLowering::RISCVTargetLowering(const TargetMachine &TM,
|
||||||
setOperationAction(ISD::VECREDUCE_FADD, VT, Custom);
|
setOperationAction(ISD::VECREDUCE_FADD, VT, Custom);
|
||||||
setOperationAction(ISD::VECREDUCE_SEQ_FADD, VT, Custom);
|
setOperationAction(ISD::VECREDUCE_SEQ_FADD, VT, Custom);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Custom-legalize bitcasts from fixed-length vectors to scalar types.
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::i8, Custom);
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::i16, Custom);
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::i32, Custom);
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::i64, Custom);
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::f16, Custom);
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::f32, Custom);
|
||||||
|
setOperationAction(ISD::BITCAST, MVT::f64, Custom);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1540,29 +1549,43 @@ SDValue RISCVTargetLowering::LowerOperation(SDValue Op,
|
||||||
case ISD::SRL_PARTS:
|
case ISD::SRL_PARTS:
|
||||||
return lowerShiftRightParts(Op, DAG, false);
|
return lowerShiftRightParts(Op, DAG, false);
|
||||||
case ISD::BITCAST: {
|
case ISD::BITCAST: {
|
||||||
|
SDLoc DL(Op);
|
||||||
|
EVT VT = Op.getValueType();
|
||||||
SDValue Op0 = Op.getOperand(0);
|
SDValue Op0 = Op.getOperand(0);
|
||||||
// We can handle fixed length vector bitcasts with a simple replacement
|
EVT Op0VT = Op0.getValueType();
|
||||||
// in isel.
|
MVT XLenVT = Subtarget.getXLenVT();
|
||||||
if (Op.getValueType().isFixedLengthVector()) {
|
if (VT.isFixedLengthVector()) {
|
||||||
if (Op0.getValueType().isFixedLengthVector())
|
// We can handle fixed length vector bitcasts with a simple replacement
|
||||||
|
// in isel.
|
||||||
|
if (Op0VT.isFixedLengthVector())
|
||||||
return Op;
|
return Op;
|
||||||
|
// When bitcasting from scalar to fixed-length vector, insert the scalar
|
||||||
|
// into a one-element vector of the result type, and perform a vector
|
||||||
|
// bitcast.
|
||||||
|
if (!Op0VT.isVector()) {
|
||||||
|
auto BVT = EVT::getVectorVT(*DAG.getContext(), Op0VT, 1);
|
||||||
|
return DAG.getBitcast(VT, DAG.getNode(ISD::INSERT_VECTOR_ELT, DL, BVT,
|
||||||
|
DAG.getUNDEF(BVT), Op0,
|
||||||
|
DAG.getConstant(0, DL, XLenVT)));
|
||||||
|
}
|
||||||
return SDValue();
|
return SDValue();
|
||||||
}
|
}
|
||||||
assert(((Subtarget.is64Bit() && Subtarget.hasStdExtF()) ||
|
// Custom-legalize bitcasts from fixed-length vector types to scalar types
|
||||||
Subtarget.hasStdExtZfh()) &&
|
// thus: bitcast the vector to a one-element vector type whose element type
|
||||||
"Unexpected custom legalisation");
|
// is the same as the result type, and extract the first element.
|
||||||
SDLoc DL(Op);
|
if (!VT.isVector() && Op0VT.isFixedLengthVector()) {
|
||||||
if (Op.getValueType() == MVT::f16 && Subtarget.hasStdExtZfh()) {
|
LLVMContext &Context = *DAG.getContext();
|
||||||
if (Op0.getValueType() != MVT::i16)
|
SDValue BVec = DAG.getBitcast(EVT::getVectorVT(Context, VT, 1), Op0);
|
||||||
return SDValue();
|
return DAG.getNode(ISD::EXTRACT_VECTOR_ELT, DL, VT, BVec,
|
||||||
SDValue NewOp0 =
|
DAG.getConstant(0, DL, XLenVT));
|
||||||
DAG.getNode(ISD::ANY_EXTEND, DL, Subtarget.getXLenVT(), Op0);
|
}
|
||||||
|
if (VT == MVT::f16 && Op0VT == MVT::i16 && Subtarget.hasStdExtZfh()) {
|
||||||
|
SDValue NewOp0 = DAG.getNode(ISD::ANY_EXTEND, DL, XLenVT, Op0);
|
||||||
SDValue FPConv = DAG.getNode(RISCVISD::FMV_H_X, DL, MVT::f16, NewOp0);
|
SDValue FPConv = DAG.getNode(RISCVISD::FMV_H_X, DL, MVT::f16, NewOp0);
|
||||||
return FPConv;
|
return FPConv;
|
||||||
} else if (Op.getValueType() == MVT::f32 && Subtarget.is64Bit() &&
|
}
|
||||||
Subtarget.hasStdExtF()) {
|
if (VT == MVT::f32 && Op0VT == MVT::i32 && Subtarget.is64Bit() &&
|
||||||
if (Op0.getValueType() != MVT::i32)
|
Subtarget.hasStdExtF()) {
|
||||||
return SDValue();
|
|
||||||
SDValue NewOp0 = DAG.getNode(ISD::ANY_EXTEND, DL, MVT::i64, Op0);
|
SDValue NewOp0 = DAG.getNode(ISD::ANY_EXTEND, DL, MVT::i64, Op0);
|
||||||
SDValue FPConv =
|
SDValue FPConv =
|
||||||
DAG.getNode(RISCVISD::FMV_W_X_RV64, DL, MVT::f32, NewOp0);
|
DAG.getNode(RISCVISD::FMV_W_X_RV64, DL, MVT::f32, NewOp0);
|
||||||
|
@ -4092,24 +4115,27 @@ void RISCVTargetLowering::ReplaceNodeResults(SDNode *N,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case ISD::BITCAST: {
|
case ISD::BITCAST: {
|
||||||
assert(((N->getValueType(0) == MVT::i32 && Subtarget.is64Bit() &&
|
EVT VT = N->getValueType(0);
|
||||||
Subtarget.hasStdExtF()) ||
|
|
||||||
(N->getValueType(0) == MVT::i16 && Subtarget.hasStdExtZfh())) &&
|
|
||||||
"Unexpected custom legalisation");
|
|
||||||
SDValue Op0 = N->getOperand(0);
|
SDValue Op0 = N->getOperand(0);
|
||||||
if (N->getValueType(0) == MVT::i16 && Subtarget.hasStdExtZfh()) {
|
EVT Op0VT = Op0.getValueType();
|
||||||
if (Op0.getValueType() != MVT::f16)
|
MVT XLenVT = Subtarget.getXLenVT();
|
||||||
return;
|
if (VT == MVT::i16 && Op0VT == MVT::f16 && Subtarget.hasStdExtZfh()) {
|
||||||
SDValue FPConv =
|
SDValue FPConv = DAG.getNode(RISCVISD::FMV_X_ANYEXTH, DL, XLenVT, Op0);
|
||||||
DAG.getNode(RISCVISD::FMV_X_ANYEXTH, DL, Subtarget.getXLenVT(), Op0);
|
|
||||||
Results.push_back(DAG.getNode(ISD::TRUNCATE, DL, MVT::i16, FPConv));
|
Results.push_back(DAG.getNode(ISD::TRUNCATE, DL, MVT::i16, FPConv));
|
||||||
} else if (N->getValueType(0) == MVT::i32 && Subtarget.is64Bit() &&
|
} else if (VT == MVT::i32 && Op0VT == MVT::f32 && Subtarget.is64Bit() &&
|
||||||
Subtarget.hasStdExtF()) {
|
Subtarget.hasStdExtF()) {
|
||||||
if (Op0.getValueType() != MVT::f32)
|
|
||||||
return;
|
|
||||||
SDValue FPConv =
|
SDValue FPConv =
|
||||||
DAG.getNode(RISCVISD::FMV_X_ANYEXTW_RV64, DL, MVT::i64, Op0);
|
DAG.getNode(RISCVISD::FMV_X_ANYEXTW_RV64, DL, MVT::i64, Op0);
|
||||||
Results.push_back(DAG.getNode(ISD::TRUNCATE, DL, MVT::i32, FPConv));
|
Results.push_back(DAG.getNode(ISD::TRUNCATE, DL, MVT::i32, FPConv));
|
||||||
|
} else if (!VT.isVector() && Op0VT.isFixedLengthVector()) {
|
||||||
|
// Custom-legalize bitcasts from fixed-length vector types to illegal
|
||||||
|
// scalar types in order to improve codegen. Bitcast the vector to a
|
||||||
|
// one-element vector type whose element type is the same as the result
|
||||||
|
// type, and extract the first element.
|
||||||
|
LLVMContext &Context = *DAG.getContext();
|
||||||
|
SDValue BVec = DAG.getBitcast(EVT::getVectorVT(Context, VT, 1), Op0);
|
||||||
|
Results.push_back(DAG.getNode(ISD::EXTRACT_VECTOR_ELT, DL, VT, BVec,
|
||||||
|
DAG.getConstant(0, DL, XLenVT)));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
|
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
|
||||||
; RUN: llc -mtriple=riscv32 -mattr=+experimental-v -verify-machineinstrs -riscv-v-vector-bits-min=128 < %s | FileCheck %s
|
; RUN: llc -mtriple=riscv32 -mattr=+experimental-v -verify-machineinstrs -riscv-v-vector-bits-min=128 < %s | FileCheck %s --check-prefixes=CHECK,RV32
|
||||||
; RUN: llc -mtriple=riscv64 -mattr=+experimental-v -verify-machineinstrs -riscv-v-vector-bits-min=128 < %s | FileCheck %s
|
; RUN: llc -mtriple=riscv64 -mattr=+experimental-v -verify-machineinstrs -riscv-v-vector-bits-min=128 < %s | FileCheck %s --check-prefixes=CHECK,RV64
|
||||||
|
|
||||||
define <32 x i1> @bitcast_v4i8_v32i1(<4 x i8> %a, <32 x i1> %b) {
|
define <32 x i1> @bitcast_v4i8_v32i1(<4 x i8> %a, <32 x i1> %b) {
|
||||||
; CHECK-LABEL: bitcast_v4i8_v32i1:
|
; CHECK-LABEL: bitcast_v4i8_v32i1:
|
||||||
|
@ -13,3 +13,367 @@ define <32 x i1> @bitcast_v4i8_v32i1(<4 x i8> %a, <32 x i1> %b) {
|
||||||
%d = xor <32 x i1> %b, %c
|
%d = xor <32 x i1> %b, %c
|
||||||
ret <32 x i1> %d
|
ret <32 x i1> %d
|
||||||
}
|
}
|
||||||
|
|
||||||
|
define i8 @bitcast_v1i8_i8(<1 x i8> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1i8_i8:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e8,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x i8> %a to i8
|
||||||
|
ret i8 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i16 @bitcast_v2i8_i16(<2 x i8> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v2i8_i16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <2 x i8> %a to i16
|
||||||
|
ret i16 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i16 @bitcast_v1i16_i16(<1 x i16> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1i16_i16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x i16> %a to i16
|
||||||
|
ret i16 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @bitcast_v4i8_i32(<4 x i8> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v4i8_i32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <4 x i8> %a to i32
|
||||||
|
ret i32 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @bitcast_v2i16_i32(<2 x i16> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v2i16_i32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <2 x i16> %a to i32
|
||||||
|
ret i32 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @bitcast_v1i32_i32(<1 x i32> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1i32_i32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x i32> %a to i32
|
||||||
|
ret i32 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v8i8_i64(<8 x i8> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v8i8_i64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v8i8_i64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <8 x i8> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v4i16_i64(<4 x i16> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v4i16_i64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v4i16_i64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <4 x i16> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v2i32_i64(<2 x i32> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v2i32_i64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v2i32_i64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <2 x i32> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v1i64_i64(<1 x i64> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v1i64_i64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v1i64_i64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <1 x i64> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define half @bitcast_v2i8_f16(<2 x i8> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v2i8_f16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <2 x i8> %a to half
|
||||||
|
ret half %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define half @bitcast_v1i16_f16(<1 x i16> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1i16_f16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x i16> %a to half
|
||||||
|
ret half %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define float @bitcast_v4i8_f32(<4 x i8> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v4i8_f32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <4 x i8> %a to float
|
||||||
|
ret float %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define float @bitcast_v2i16_f32(<2 x i16> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v2i16_f32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <2 x i16> %a to float
|
||||||
|
ret float %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define float @bitcast_v1i32_f32(<1 x i32> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1i32_f32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x i32> %a to float
|
||||||
|
ret float %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v8i8_f64(<8 x i8> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v8i8_f64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v8i8_f64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <8 x i8> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v4i16_f64(<4 x i16> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v4i16_f64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v4i16_f64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <4 x i16> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v2i32_f64(<2 x i32> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v2i32_f64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v2i32_f64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <2 x i32> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v1i64_f64(<1 x i64> %a) {
|
||||||
|
; RV32-LABEL: bitcast_v1i64_f64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_v1i64_f64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast <1 x i64> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x i16> @bitcast_i16_v1i16(i16 %a) {
|
||||||
|
; CHECK-LABEL: bitcast_i16_v1i16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetivli a1, 1, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.v.x v8, a0
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast i16 %a to <1 x i16>
|
||||||
|
ret <1 x i16> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x i16> @bitcast_i32_v2i16(i32 %a) {
|
||||||
|
; RV32-LABEL: bitcast_i32_v2i16:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_i32_v2i16:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.v.x v8, a0
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast i32 %a to <2 x i16>
|
||||||
|
ret <2 x i16> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x i32> @bitcast_i32_v1i32(i32 %a) {
|
||||||
|
; RV32-LABEL: bitcast_i32_v1i32:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_i32_v1i32:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.v.x v8, a0
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast i32 %a to <1 x i32>
|
||||||
|
ret <1 x i32> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <4 x i16> @bitcast_i64_v4i16(i64 %a) {
|
||||||
|
; RV32-LABEL: bitcast_i64_v4i16:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: vsetivli a2, 2, e32,m1,ta,mu
|
||||||
|
; RV32-NEXT: vmv.v.i v25, 0
|
||||||
|
; RV32-NEXT: vslide1up.vx v26, v25, a1
|
||||||
|
; RV32-NEXT: vslide1up.vx v25, v26, a0
|
||||||
|
; RV32-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vslideup.vi v8, v25, 0
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_i64_v4i16:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast i64 %a to <4 x i16>
|
||||||
|
ret <4 x i16> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x i32> @bitcast_i64_v2i32(i64 %a) {
|
||||||
|
; RV32-LABEL: bitcast_i64_v2i32:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: vsetivli a2, 2, e32,m1,ta,mu
|
||||||
|
; RV32-NEXT: vmv.v.i v25, 0
|
||||||
|
; RV32-NEXT: vslide1up.vx v26, v25, a1
|
||||||
|
; RV32-NEXT: vslide1up.vx v25, v26, a0
|
||||||
|
; RV32-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vslideup.vi v8, v25, 0
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_i64_v2i32:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast i64 %a to <2 x i32>
|
||||||
|
ret <2 x i32> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x i64> @bitcast_i64_v1i64(i64 %a) {
|
||||||
|
; RV32-LABEL: bitcast_i64_v1i64:
|
||||||
|
; RV32: # %bb.0:
|
||||||
|
; RV32-NEXT: vsetivli a2, 2, e32,m1,ta,mu
|
||||||
|
; RV32-NEXT: vmv.v.i v25, 0
|
||||||
|
; RV32-NEXT: vslide1up.vx v26, v25, a1
|
||||||
|
; RV32-NEXT: vslide1up.vx v25, v26, a0
|
||||||
|
; RV32-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-NEXT: vslideup.vi v8, v25, 0
|
||||||
|
; RV32-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-LABEL: bitcast_i64_v1i64:
|
||||||
|
; RV64: # %bb.0:
|
||||||
|
; RV64-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-NEXT: ret
|
||||||
|
%b = bitcast i64 %a to <1 x i64>
|
||||||
|
ret <1 x i64> %b
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,525 @@
|
||||||
|
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
|
||||||
|
; RUN: llc -mtriple=riscv32 -mattr=+d,+experimental-zfh,+experimental-v -verify-machineinstrs -riscv-v-vector-bits-min=128 < %s | FileCheck %s --check-prefixes=CHECK,RV32-FP
|
||||||
|
; RUN: llc -mtriple=riscv64 -mattr=+d,+experimental-zfh,+experimental-v -verify-machineinstrs -riscv-v-vector-bits-min=128 < %s | FileCheck %s --check-prefixes=CHECK,RV64-FP
|
||||||
|
|
||||||
|
define i16 @bitcast_v1f16_i16(<1 x half> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1f16_i16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x half> %a to i16
|
||||||
|
ret i16 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define half @bitcast_v1f16_f16(<1 x half> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1f16_f16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vfmv.f.s ft0, v8
|
||||||
|
; CHECK-NEXT: fmv.x.h a0, ft0
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x half> %a to half
|
||||||
|
ret half %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @bitcast_v2f16_i32(<2 x half> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v2f16_i32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <2 x half> %a to i32
|
||||||
|
ret i32 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @bitcast_v1f32_i32(<1 x float> %a) {
|
||||||
|
; CHECK-LABEL: bitcast_v1f32_i32:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.x.s a0, v8
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast <1 x float> %a to i32
|
||||||
|
ret i32 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define float @bitcast_v2f16_f32(<2 x half> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v2f16_f32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v2f16_f32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vfmv.f.s ft0, v8
|
||||||
|
; RV64-FP-NEXT: fmv.x.w a0, ft0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <2 x half> %a to float
|
||||||
|
ret float %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define float @bitcast_v1f32_f32(<1 x float> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v1f32_f32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v1f32_f32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vfmv.f.s ft0, v8
|
||||||
|
; RV64-FP-NEXT: fmv.x.w a0, ft0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <1 x float> %a to float
|
||||||
|
ret float %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v4f16_i64(<4 x half> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v4f16_i64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v4f16_i64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <4 x half> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v2f32_i64(<2 x float> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v2f32_i64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v2f32_i64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <2 x float> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define i64 @bitcast_v1f64_i64(<1 x double> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v1f64_i64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi a0, zero, 32
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vsrl.vx v25, v8, a0
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a1, v25
|
||||||
|
; RV32-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v1f64_i64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <1 x double> %a to i64
|
||||||
|
ret i64 %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v4f16_f64(<4 x half> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v4f16_f64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.f.s ft0, v8
|
||||||
|
; RV32-FP-NEXT: fsd ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: lw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: lw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v4f16_f64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <4 x half> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v2f32_f64(<2 x float> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v2f32_f64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.f.s ft0, v8
|
||||||
|
; RV32-FP-NEXT: fsd ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: lw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: lw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v2f32_f64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <2 x float> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @bitcast_v1f64_f64(<1 x double> %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_v1f64_f64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.f.s ft0, v8
|
||||||
|
; RV32-FP-NEXT: fsd ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: lw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: lw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_v1f64_f64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetvli zero, zero, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.x.s a0, v8
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast <1 x double> %a to double
|
||||||
|
ret double %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x half> @bitcast_i16_v1f16(i16 %a) {
|
||||||
|
; CHECK-LABEL: bitcast_i16_v1f16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: vsetivli a1, 1, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vmv.v.x v8, a0
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast i16 %a to <1 x half>
|
||||||
|
ret <1 x half> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x half> @bitcast_i32_v2f16(i32 %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_i32_v2f16:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_i32_v2f16:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.v.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast i32 %a to <2 x half>
|
||||||
|
ret <2 x half> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x float> @bitcast_i32_v1f32(i32 %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_i32_v1f32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_i32_v1f32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.v.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast i32 %a to <1 x float>
|
||||||
|
ret <1 x float> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <4 x half> @bitcast_i64_v4f16(i64 %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_i64_v4f16:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a2, 2, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.v.i v25, 0
|
||||||
|
; RV32-FP-NEXT: vslide1up.vx v26, v25, a1
|
||||||
|
; RV32-FP-NEXT: vslide1up.vx v25, v26, a0
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vslideup.vi v8, v25, 0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_i64_v4f16:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast i64 %a to <4 x half>
|
||||||
|
ret <4 x half> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x float> @bitcast_i64_v2f32(i64 %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_i64_v2f32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a2, 2, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.v.i v25, 0
|
||||||
|
; RV32-FP-NEXT: vslide1up.vx v26, v25, a1
|
||||||
|
; RV32-FP-NEXT: vslide1up.vx v25, v26, a0
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vslideup.vi v8, v25, 0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_i64_v2f32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast i64 %a to <2 x float>
|
||||||
|
ret <2 x float> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x double> @bitcast_i64_v1f64(i64 %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_i64_v1f64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a2, 2, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.v.i v25, 0
|
||||||
|
; RV32-FP-NEXT: vslide1up.vx v26, v25, a1
|
||||||
|
; RV32-FP-NEXT: vslide1up.vx v25, v26, a0
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vslideup.vi v8, v25, 0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_i64_v1f64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast i64 %a to <1 x double>
|
||||||
|
ret <1 x double> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x i16> @bitcast_f16_v1i16(half %a) {
|
||||||
|
; CHECK-LABEL: bitcast_f16_v1i16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: fmv.h.x ft0, a0
|
||||||
|
; CHECK-NEXT: vsetivli a0, 1, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast half %a to <1 x i16>
|
||||||
|
ret <1 x i16> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x half> @bitcast_f16_v1f16(half %a) {
|
||||||
|
; CHECK-LABEL: bitcast_f16_v1f16:
|
||||||
|
; CHECK: # %bb.0:
|
||||||
|
; CHECK-NEXT: fmv.h.x ft0, a0
|
||||||
|
; CHECK-NEXT: vsetivli a0, 1, e16,m1,ta,mu
|
||||||
|
; CHECK-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; CHECK-NEXT: ret
|
||||||
|
%b = bitcast half %a to <1 x half>
|
||||||
|
ret <1 x half> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x i16> @bitcast_f32_v2i16(float %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f32_v2i16:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f32_v2i16:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: fmv.w.x ft0, a0
|
||||||
|
; RV64-FP-NEXT: vsetivli a0, 1, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast float %a to <2 x i16>
|
||||||
|
ret <2 x i16> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x half> @bitcast_f32_v2f16(float %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f32_v2f16:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f32_v2f16:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: fmv.w.x ft0, a0
|
||||||
|
; RV64-FP-NEXT: vsetivli a0, 1, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast float %a to <2 x half>
|
||||||
|
ret <2 x half> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x i32> @bitcast_f32_v1i32(float %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f32_v1i32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f32_v1i32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: fmv.w.x ft0, a0
|
||||||
|
; RV64-FP-NEXT: vsetivli a0, 1, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast float %a to <1 x i32>
|
||||||
|
ret <1 x i32> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x float> @bitcast_f32_v1f32(float %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f32_v1f32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: vsetivli a1, 1, e32,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f32_v1f32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: fmv.w.x ft0, a0
|
||||||
|
; RV64-FP-NEXT: vsetivli a0, 1, e32,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast float %a to <1 x float>
|
||||||
|
ret <1 x float> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <4 x i16> @bitcast_f64_v4i16(double %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f64_v4i16:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: sw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: sw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: fld ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f64_v4i16:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast double %a to <4 x i16>
|
||||||
|
ret <4 x i16> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <4 x half> @bitcast_f64_v4f16(double %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f64_v4f16:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: sw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: sw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: fld ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f64_v4f16:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast double %a to <4 x half>
|
||||||
|
ret <4 x half> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x i32> @bitcast_f64_v2i32(double %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f64_v2i32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: sw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: sw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: fld ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f64_v2i32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast double %a to <2 x i32>
|
||||||
|
ret <2 x i32> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <2 x float> @bitcast_f64_v2f32(double %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f64_v2f32:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: sw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: sw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: fld ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f64_v2f32:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast double %a to <2 x float>
|
||||||
|
ret <2 x float> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x i64> @bitcast_f64_v1i64(double %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f64_v1i64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: sw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: sw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: fld ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f64_v1i64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast double %a to <1 x i64>
|
||||||
|
ret <1 x i64> %b
|
||||||
|
}
|
||||||
|
|
||||||
|
define <1 x double> @bitcast_f64_v1f64(double %a) {
|
||||||
|
; RV32-FP-LABEL: bitcast_f64_v1f64:
|
||||||
|
; RV32-FP: # %bb.0:
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, -16
|
||||||
|
; RV32-FP-NEXT: .cfi_def_cfa_offset 16
|
||||||
|
; RV32-FP-NEXT: sw a0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: sw a1, 12(sp)
|
||||||
|
; RV32-FP-NEXT: fld ft0, 8(sp)
|
||||||
|
; RV32-FP-NEXT: vsetivli a0, 1, e64,m1,ta,mu
|
||||||
|
; RV32-FP-NEXT: vfmv.s.f v8, ft0
|
||||||
|
; RV32-FP-NEXT: addi sp, sp, 16
|
||||||
|
; RV32-FP-NEXT: ret
|
||||||
|
;
|
||||||
|
; RV64-FP-LABEL: bitcast_f64_v1f64:
|
||||||
|
; RV64-FP: # %bb.0:
|
||||||
|
; RV64-FP-NEXT: vsetivli a1, 1, e64,m1,ta,mu
|
||||||
|
; RV64-FP-NEXT: vmv.s.x v8, a0
|
||||||
|
; RV64-FP-NEXT: ret
|
||||||
|
%b = bitcast double %a to <1 x double>
|
||||||
|
ret <1 x double> %b
|
||||||
|
}
|
Loading…
Reference in New Issue