mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-04-03 18:32:50 +00:00
Enhance type legalization on bitcast from vector to integer
- Find a legal vector type before casting and extracting element from it. - As the new vector type may have more than 2 elements, build the final hi/lo pair by BFS pairing them from bottom to top. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163830 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
da0e8219b7
commit
092122f124
@ -94,14 +94,43 @@ void DAGTypeLegalizer::ExpandRes_BITCAST(SDNode *N, SDValue &Lo, SDValue &Hi) {
|
||||
if (InVT.isVector() && OutVT.isInteger()) {
|
||||
// Handle cases like i64 = BITCAST v1i64 on x86, where the operand
|
||||
// is legal but the result is not.
|
||||
EVT NVT = EVT::getVectorVT(*DAG.getContext(), NOutVT, 2);
|
||||
unsigned NumElems = 2;
|
||||
EVT NVT = EVT::getVectorVT(*DAG.getContext(), NOutVT, NumElems);
|
||||
|
||||
// If <NOutVT * N> is not a legal type, try <NOutVT/2 * (N*2)>.
|
||||
while (!isTypeLegal(NVT)) {
|
||||
unsigned NewSizeInBits = NOutVT.getSizeInBits() / 2;
|
||||
// If the element size is smaller than byte, bail.
|
||||
if (NewSizeInBits < 8)
|
||||
break;
|
||||
NumElems *= 2;
|
||||
NOutVT = EVT::getIntegerVT(*DAG.getContext(), NewSizeInBits);
|
||||
NVT = EVT::getVectorVT(*DAG.getContext(), NOutVT, NumElems);
|
||||
}
|
||||
|
||||
if (isTypeLegal(NVT)) {
|
||||
SDValue CastInOp = DAG.getNode(ISD::BITCAST, dl, NVT, InOp);
|
||||
Lo = DAG.getNode(ISD::EXTRACT_VECTOR_ELT, dl, NOutVT, CastInOp,
|
||||
DAG.getIntPtrConstant(0));
|
||||
Hi = DAG.getNode(ISD::EXTRACT_VECTOR_ELT, dl, NOutVT, CastInOp,
|
||||
DAG.getIntPtrConstant(1));
|
||||
|
||||
SmallVector<SDValue, 8> Vals;
|
||||
for (unsigned i = 0; i < NumElems; ++i)
|
||||
Vals.push_back(DAG.getNode(ISD::EXTRACT_VECTOR_ELT, dl, NOutVT,
|
||||
CastInOp, DAG.getIntPtrConstant(i)));
|
||||
|
||||
// Build Lo, Hi pair by pairing extracted elements if needed.
|
||||
unsigned Slot = 0;
|
||||
for (unsigned e = Vals.size(); e - Slot > 2; Slot += 2, e += 1) {
|
||||
// Each iteration will BUILD_PAIR two nodes and append the result until
|
||||
// there are only two nodes left, i.e. Lo and Hi.
|
||||
SDValue LHS = Vals[Slot];
|
||||
SDValue RHS = Vals[Slot + 1];
|
||||
Vals.push_back(DAG.getNode(ISD::BUILD_PAIR, dl,
|
||||
EVT::getIntegerVT(
|
||||
*DAG.getContext(),
|
||||
LHS.getValueType().getSizeInBits() << 1),
|
||||
LHS, RHS));
|
||||
}
|
||||
Lo = Vals[Slot++];
|
||||
Hi = Vals[Slot++];
|
||||
|
||||
if (TLI.isBigEndian())
|
||||
std::swap(Lo, Hi);
|
||||
|
11
test/CodeGen/X86/bitcast-i256.ll
Normal file
11
test/CodeGen/X86/bitcast-i256.ll
Normal file
@ -0,0 +1,11 @@
|
||||
; RUN: llc -mtriple=x86_64-unknown-unknown -mcpu=core-avx-i < %s | FileCheck %s --check-prefix CHECK
|
||||
|
||||
define i256 @foo(<8 x i32> %a) {
|
||||
%r = bitcast <8 x i32> %a to i256
|
||||
ret i256 %r
|
||||
; CHECK: foo
|
||||
; CHECK: vextractf128
|
||||
; CHECK: vpextrq
|
||||
; CHECK: vpextrq
|
||||
; CHECK: ret
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user