mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-15 23:31:37 +00:00
More movsldup/movshdup cleanup. Rewrite the mask matching function and add
support for 256-bit versions (but no instruction selection yet, coming next). git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@136050 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
5f6c440e53
commit
9123c6fea0
@ -3457,51 +3457,58 @@ static bool isCommutedMOVL(ShuffleVectorSDNode *N, bool V2IsSplat = false,
|
|||||||
|
|
||||||
/// isMOVSHDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
/// isMOVSHDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
||||||
/// specifies a shuffle of elements that is suitable for input to MOVSHDUP.
|
/// specifies a shuffle of elements that is suitable for input to MOVSHDUP.
|
||||||
bool X86::isMOVSHDUPMask(ShuffleVectorSDNode *N) {
|
/// Masks to match: <1, 1, 3, 3> or <1, 1, 3, 3, 5, 5, 7, 7>
|
||||||
if (N->getValueType(0).getVectorNumElements() != 4)
|
bool X86::isMOVSHDUPMask(ShuffleVectorSDNode *N,
|
||||||
|
const X86Subtarget *Subtarget) {
|
||||||
|
if (!Subtarget->hasSSE3() && !Subtarget->hasAVX())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Expect 1, 1, 3, 3
|
// The second vector must be undef
|
||||||
for (unsigned i = 0; i < 2; ++i) {
|
if (N->getOperand(1).getOpcode() != ISD::UNDEF)
|
||||||
int Elt = N->getMaskElt(i);
|
return false;
|
||||||
if (Elt >= 0 && Elt != 1)
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool HasHi = false;
|
EVT VT = N->getValueType(0);
|
||||||
for (unsigned i = 2; i < 4; ++i) {
|
unsigned NumElems = VT.getVectorNumElements();
|
||||||
int Elt = N->getMaskElt(i);
|
|
||||||
if (Elt >= 0 && Elt != 3)
|
if ((VT.getSizeInBits() == 128 && NumElems != 4) ||
|
||||||
|
(VT.getSizeInBits() == 256 && NumElems != 8))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// "i+1" is the value the indexed mask element must have
|
||||||
|
for (unsigned i = 0; i < NumElems; i += 2)
|
||||||
|
if (!isUndefOrEqual(N->getMaskElt(i), i+1) ||
|
||||||
|
!isUndefOrEqual(N->getMaskElt(i+1), i+1))
|
||||||
return false;
|
return false;
|
||||||
if (Elt == 3)
|
|
||||||
HasHi = true;
|
return true;
|
||||||
}
|
|
||||||
// Don't use movshdup if it can be done with a shufps.
|
|
||||||
// FIXME: verify that matching u, u, 3, 3 is what we want.
|
|
||||||
return HasHi;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// isMOVSLDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
/// isMOVSLDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
||||||
/// specifies a shuffle of elements that is suitable for input to MOVSLDUP.
|
/// specifies a shuffle of elements that is suitable for input to MOVSLDUP.
|
||||||
bool X86::isMOVSLDUPMask(ShuffleVectorSDNode *N) {
|
/// Masks to match: <0, 0, 2, 2> or <0, 0, 2, 2, 4, 4, 6, 6>
|
||||||
if (N->getValueType(0).getVectorNumElements() != 4)
|
bool X86::isMOVSLDUPMask(ShuffleVectorSDNode *N,
|
||||||
|
const X86Subtarget *Subtarget) {
|
||||||
|
if (!Subtarget->hasSSE3() && !Subtarget->hasAVX())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Expect 0, 0, 2, 2
|
// The second vector must be undef
|
||||||
for (unsigned i = 0; i < 2; ++i)
|
if (N->getOperand(1).getOpcode() != ISD::UNDEF)
|
||||||
if (N->getMaskElt(i) > 0)
|
return false;
|
||||||
|
|
||||||
|
EVT VT = N->getValueType(0);
|
||||||
|
unsigned NumElems = VT.getVectorNumElements();
|
||||||
|
|
||||||
|
if ((VT.getSizeInBits() == 128 && NumElems != 4) ||
|
||||||
|
(VT.getSizeInBits() == 256 && NumElems != 8))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// "i" is the value the indexed mask element must have
|
||||||
|
for (unsigned i = 0; i < NumElems; i += 2)
|
||||||
|
if (!isUndefOrEqual(N->getMaskElt(i), i) ||
|
||||||
|
!isUndefOrEqual(N->getMaskElt(i+1), i))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
bool HasHi = false;
|
return true;
|
||||||
for (unsigned i = 2; i < 4; ++i) {
|
|
||||||
int Elt = N->getMaskElt(i);
|
|
||||||
if (Elt >= 0 && Elt != 2)
|
|
||||||
return false;
|
|
||||||
if (Elt == 2)
|
|
||||||
HasHi = true;
|
|
||||||
}
|
|
||||||
// Don't use movsldup if it can be done with a shufps.
|
|
||||||
return HasHi;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// isMOVDDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
/// isMOVDDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
||||||
@ -5942,10 +5949,10 @@ X86TargetLowering::LowerVECTOR_SHUFFLE(SDValue Op, SelectionDAG &DAG) const {
|
|||||||
if (X86::isMOVHLPSMask(SVOp))
|
if (X86::isMOVHLPSMask(SVOp))
|
||||||
return getMOVHighToLow(Op, dl, DAG);
|
return getMOVHighToLow(Op, dl, DAG);
|
||||||
|
|
||||||
if (X86::isMOVSHDUPMask(SVOp) && HasSSE3 && V2IsUndef && NumElems == 4)
|
if (X86::isMOVSHDUPMask(SVOp, Subtarget))
|
||||||
return getTargetShuffleNode(X86ISD::MOVSHDUP, dl, VT, V1, DAG);
|
return getTargetShuffleNode(X86ISD::MOVSHDUP, dl, VT, V1, DAG);
|
||||||
|
|
||||||
if (X86::isMOVSLDUPMask(SVOp) && HasSSE3 && V2IsUndef && NumElems == 4)
|
if (X86::isMOVSLDUPMask(SVOp, Subtarget))
|
||||||
return getTargetShuffleNode(X86ISD::MOVSLDUP, dl, VT, V1, DAG);
|
return getTargetShuffleNode(X86ISD::MOVSLDUP, dl, VT, V1, DAG);
|
||||||
|
|
||||||
if (X86::isMOVLPMask(SVOp))
|
if (X86::isMOVLPMask(SVOp))
|
||||||
|
@ -406,11 +406,11 @@ namespace llvm {
|
|||||||
|
|
||||||
/// isMOVSHDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
/// isMOVSHDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
||||||
/// specifies a shuffle of elements that is suitable for input to MOVSHDUP.
|
/// specifies a shuffle of elements that is suitable for input to MOVSHDUP.
|
||||||
bool isMOVSHDUPMask(ShuffleVectorSDNode *N);
|
bool isMOVSHDUPMask(ShuffleVectorSDNode *N, const X86Subtarget *Subtarget);
|
||||||
|
|
||||||
/// isMOVSLDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
/// isMOVSLDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
||||||
/// specifies a shuffle of elements that is suitable for input to MOVSLDUP.
|
/// specifies a shuffle of elements that is suitable for input to MOVSLDUP.
|
||||||
bool isMOVSLDUPMask(ShuffleVectorSDNode *N);
|
bool isMOVSLDUPMask(ShuffleVectorSDNode *N, const X86Subtarget *Subtarget);
|
||||||
|
|
||||||
/// isMOVDDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
/// isMOVDDUPMask - Return true if the specified VECTOR_SHUFFLE operand
|
||||||
/// specifies a shuffle of elements that is suitable for input to MOVDDUP.
|
/// specifies a shuffle of elements that is suitable for input to MOVDDUP.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user