51#define DEBUG_TYPE "legalizevectorops"
55class VectorLegalizer {
67 LegalizedNodes.
insert(std::make_pair(
From, To));
70 LegalizedNodes.
insert(std::make_pair(To, To));
139 std::pair<SDValue, SDValue> ExpandLoad(
SDNode *
N);
197bool VectorLegalizer::Run() {
199 bool HasVectors =
false;
201 E = std::prev(DAG.allnodes_end());
I != std::next(E); ++
I) {
221 DAG.AssignTopologicalOrder();
223 E = std::prev(DAG.allnodes_end());
I != std::next(E); ++
I)
227 SDValue OldRoot = DAG.getRoot();
228 assert(LegalizedNodes.count(OldRoot) &&
"Root didn't get legalized?");
229 DAG.setRoot(LegalizedNodes[OldRoot]);
231 LegalizedNodes.clear();
234 DAG.RemoveDeadNodes();
241 "Unexpected number of results");
243 for (
unsigned i = 0, e =
Op->getNumValues(); i != e; ++i)
244 AddLegalizedOperand(
Op.getValue(i),
SDValue(Result, i));
249VectorLegalizer::RecursivelyLegalizeResults(
SDValue Op,
252 "Unexpected number of results");
254 for (
unsigned i = 0, e =
Results.
size(); i != e; ++i) {
256 AddLegalizedOperand(
Op.getValue(i),
Results[i]);
266 if (
I != LegalizedNodes.end())
return I->second;
270 for (
const SDValue &Oper :
Op->op_values())
273 SDNode *
Node = DAG.UpdateNodeOperands(
Op.getNode(), Ops);
275 bool HasVectorValueOrOp =
278 [](
SDValue O) { return O.getValueType().isVector(); });
279 if (!HasVectorValueOrOp)
280 return TranslateLegalizeResults(
Op,
Node);
284 switch (
Op.getOpcode()) {
286 return TranslateLegalizeResults(
Op,
Node);
290 EVT LoadedVT =
LD->getMemoryVT();
292 Action = TLI.getLoadExtAction(ExtType,
LD->getValueType(0), LoadedVT);
297 EVT StVT =
ST->getMemoryVT();
298 MVT ValVT =
ST->getValue().getSimpleValueType();
299 if (StVT.
isVector() &&
ST->isTruncatingStore())
300 Action = TLI.getTruncStoreAction(ValVT, StVT);
304 Action = TLI.getOperationAction(
Node->getOpcode(),
Node->getValueType(0));
307 if (Action == TargetLowering::Legal)
308 Action = TargetLowering::Expand;
310#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
311 case ISD::STRICT_##DAGN:
312#include "llvm/IR/ConstrainedOps.def"
313 ValVT =
Node->getValueType(0);
316 ValVT =
Node->getOperand(1).getValueType();
319 MVT OpVT =
Node->getOperand(1).getSimpleValueType();
321 Action = TLI.getCondCodeAction(CCCode, OpVT);
322 if (Action == TargetLowering::Legal)
323 Action = TLI.getOperationAction(
Node->getOpcode(), OpVT);
325 Action = TLI.getOperationAction(
Node->getOpcode(), ValVT);
332 if (Action == TargetLowering::Expand && !TLI.isStrictFPEnabled() &&
333 TLI.getStrictFPOperationAction(
Node->getOpcode(), ValVT) ==
334 TargetLowering::Legal) {
336 if (TLI.getOperationAction(
Node->getOpcode(), EltVT)
337 == TargetLowering::Expand &&
338 TLI.getStrictFPOperationAction(
Node->getOpcode(), EltVT)
339 == TargetLowering::Legal)
340 Action = TargetLowering::Legal;
461 Action = TLI.getOperationAction(
Node->getOpcode(),
Node->getValueType(0));
471 unsigned Scale =
Node->getConstantOperandVal(2);
472 Action = TLI.getFixedPointOperationAction(
Node->getOpcode(),
473 Node->getValueType(0), Scale);
497 Action = TLI.getOperationAction(
Node->getOpcode(),
498 Node->getOperand(0).getValueType());
502 Action = TLI.getOperationAction(
Node->getOpcode(),
503 Node->getOperand(1).getValueType());
506 MVT OpVT =
Node->getOperand(0).getSimpleValueType();
508 Action = TLI.getCondCodeAction(CCCode, OpVT);
509 if (Action == TargetLowering::Legal)
510 Action = TLI.getOperationAction(
Node->getOpcode(), OpVT);
514#define BEGIN_REGISTER_VP_SDNODE(VPID, LEGALPOS, ...) \
516 EVT LegalizeVT = LEGALPOS < 0 ? Node->getValueType(-(1 + LEGALPOS)) \
517 : Node->getOperand(LEGALPOS).getValueType(); \
518 if (ISD::VPID == ISD::VP_SETCC) { \
519 ISD::CondCode CCCode = cast<CondCodeSDNode>(Node->getOperand(2))->get(); \
520 Action = TLI.getCondCodeAction(CCCode, LegalizeVT.getSimpleVT()); \
521 if (Action != TargetLowering::Legal) \
525 if (!Node->getValueType(0).isVector() && \
526 Node->getValueType(0) != MVT::Other) { \
527 Action = TargetLowering::Legal; \
530 Action = TLI.getOperationAction(Node->getOpcode(), LegalizeVT); \
532#include "llvm/IR/VPIntrinsics.def"
540 case TargetLowering::Promote:
542 "This action is not supported yet!");
544 Promote(
Node, ResultVals);
545 assert(!ResultVals.
empty() &&
"No results for promotion?");
547 case TargetLowering::Legal:
550 case TargetLowering::Custom:
552 if (LowerOperationWrapper(
Node, ResultVals))
556 case TargetLowering::Expand:
558 Expand(
Node, ResultVals);
562 if (ResultVals.
empty())
563 return TranslateLegalizeResults(
Op,
Node);
566 return RecursivelyLegalizeResults(
Op, ResultVals);
571bool VectorLegalizer::LowerOperationWrapper(
SDNode *
Node,
583 if (
Node->getNumValues() == 1) {
591 "Lowering returned the wrong number of results!");
594 for (
unsigned I = 0, E =
Node->getNumValues();
I != E; ++
I)
600void VectorLegalizer::PromoteSETCC(
SDNode *
Node,
602 MVT VecVT =
Node->getOperand(0).getSimpleValueType();
603 MVT NewVecVT = TLI.getTypeToPromoteTo(
Node->getOpcode(), VecVT);
610 Operands[0] = DAG.getNode(ExtOp,
DL, NewVecVT,
Node->getOperand(0));
611 Operands[1] = DAG.getNode(ExtOp,
DL, NewVecVT,
Node->getOperand(1));
614 if (
Node->getOpcode() == ISD::VP_SETCC) {
625void VectorLegalizer::PromoteSTRICT(
SDNode *
Node,
627 MVT VecVT =
Node->getOperand(1).getSimpleValueType();
628 MVT NewVecVT = TLI.getTypeToPromoteTo(
Node->getOpcode(), VecVT);
636 for (
unsigned j = 1;
j !=
Node->getNumOperands(); ++
j)
637 if (
Node->getOperand(j).getValueType().isVector() &&
644 {
Node->getOperand(0),
Node->getOperand(j)});
650 SDVTList VTs = DAG.getVTList(NewVecVT,
Node->getValueType(1));
660 DAG.getIntPtrConstant(0,
DL,
true)});
669 switch (
Node->getOpcode()) {
710 "Can't promote a vector with multiple results!");
711 MVT VT =
Node->getSimpleValueType(0);
712 MVT NVT = TLI.getTypeToPromoteTo(
Node->getOpcode(), VT);
716 for (
unsigned j = 0;
j !=
Node->getNumOperands(); ++
j) {
720 if (
Node->getOperand(j).getValueType().isVector() && !SkipPromote)
721 if (
Node->getOperand(j)
723 .getVectorElementType()
724 .isFloatingPoint() &&
740 DAG.getIntPtrConstant(0, dl,
true));
747void VectorLegalizer::PromoteINT_TO_FP(
SDNode *
Node,
751 bool IsStrict =
Node->isStrictFPOpcode();
752 MVT VT =
Node->getOperand(IsStrict ? 1 : 0).getSimpleValueType();
753 MVT NVT = TLI.getTypeToPromoteTo(
Node->getOpcode(), VT);
755 "Vectors have different number of elements!");
764 for (
unsigned j = 0;
j !=
Node->getNumOperands(); ++
j) {
765 if (
Node->getOperand(j).getValueType().isVector())
766 Operands[j] = DAG.getNode(Opc, dl, NVT,
Node->getOperand(j));
788void VectorLegalizer::PromoteFP_TO_INT(
SDNode *
Node,
790 MVT VT =
Node->getSimpleValueType(0);
791 MVT NVT = TLI.getTypeToPromoteTo(
Node->getOpcode(), VT);
792 bool IsStrict =
Node->isStrictFPOpcode();
794 "Vectors have different number of elements!");
796 unsigned NewOpc =
Node->getOpcode();
810 Promoted = DAG.
getNode(NewOpc, dl, {NVT, MVT::Other},
811 {
Node->getOperand(0),
Node->getOperand(1)});
814 Promoted = DAG.
getNode(NewOpc, dl, NVT,
Node->getOperand(0));
825 Promoted = DAG.
getNode(NewOpc, dl, NVT, Promoted,
833std::pair<SDValue, SDValue> VectorLegalizer::ExpandLoad(
SDNode *
N) {
835 return TLI.scalarizeVectorLoad(LD, DAG);
840 SDValue TF = TLI.scalarizeVectorStore(ST, DAG);
845 switch (
Node->getOpcode()) {
847 std::pair<SDValue, SDValue> Tmp = ExpandLoad(
Node);
856 for (
unsigned i = 0, e =
Node->getNumValues(); i != e; ++i)
863 Results.push_back(ExpandANY_EXTEND_VECTOR_INREG(
Node));
866 Results.push_back(ExpandSIGN_EXTEND_VECTOR_INREG(
Node));
869 Results.push_back(ExpandZERO_EXTEND_VECTOR_INREG(
Node));
894 if (
Node->getValueType(0).isScalableVector()) {
895 EVT CondVT = TLI.getSetCCResultType(
896 DAG.getDataLayout(), *DAG.getContext(),
Node->getValueType(0));
899 Node->getOperand(1),
Node->getOperand(4));
902 Node->getOperand(3)));
948 case ISD::VP_BITREVERSE:
949 if (
SDValue Expanded = TLI.expandVPBITREVERSE(
Node, DAG)) {
955 if (
SDValue Expanded = TLI.expandCTPOP(
Node, DAG)) {
961 if (
SDValue Expanded = TLI.expandVPCTPOP(
Node, DAG)) {
968 if (
SDValue Expanded = TLI.expandCTLZ(
Node, DAG)) {
974 case ISD::VP_CTLZ_ZERO_UNDEF:
975 if (
SDValue Expanded = TLI.expandVPCTLZ(
Node, DAG)) {
982 if (
SDValue Expanded = TLI.expandCTTZ(
Node, DAG)) {
988 case ISD::VP_CTTZ_ZERO_UNDEF:
989 if (
SDValue Expanded = TLI.expandVPCTTZ(
Node, DAG)) {
998 if (
SDValue Expanded = TLI.expandFunnelShift(
Node, DAG)) {
1005 if (
SDValue Expanded = TLI.expandROT(
Node,
false , DAG)) {
1012 if (
SDValue Expanded = TLI.expandFMINNUM_FMAXNUM(
Node, DAG)) {
1019 Results.push_back(TLI.expandFMINIMUM_FMAXIMUM(
Node, DAG));
1025 if (
SDValue Expanded = TLI.expandIntMINMAX(
Node, DAG)) {
1046 if (
SDValue Expanded = TLI.expandAddSubSat(
Node, DAG)) {
1053 if (
SDValue Expanded = TLI.expandShlSat(
Node, DAG)) {
1061 if (
Node->getValueType(0).isScalableVector()) {
1062 if (
SDValue Expanded = TLI.expandFP_TO_INT_SAT(
Node, DAG)) {
1070 if (
SDValue Expanded = TLI.expandFixedPointMul(
Node, DAG)) {
1089#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
1090 case ISD::STRICT_##DAGN:
1091#include "llvm/IR/ConstrainedOps.def"
1109 Results.push_back(TLI.expandVecReduce(
Node, DAG));
1113 Results.push_back(TLI.expandVecReduceSeq(
Node, DAG));
1123 if (tryExpandVecMathCall(
Node, RTLIB::REM_F32, RTLIB::REM_F64,
1124 RTLIB::REM_F80, RTLIB::REM_F128,
1130 Results.push_back(TLI.expandVECTOR_COMPRESS(
Node, DAG));
1135 if (
Node->getNumValues() == 1) {
1139 "VectorLegalizer Expand returned wrong number of results!");
1149 EVT VT =
Node->getValueType(0);
1167 if (TLI.getOperationAction(
ISD::AND, VT) == TargetLowering::Expand ||
1168 TLI.getOperationAction(
ISD::XOR, VT) == TargetLowering::Expand ||
1169 TLI.getOperationAction(
ISD::OR, VT) == TargetLowering::Expand ||
1172 VT) == TargetLowering::Expand)
1173 return DAG.UnrollVectorOp(
Node);
1181 Mask = DAG.getSelect(
DL, BitTy, Mask, DAG.getAllOnesConstant(
DL, BitTy),
1182 DAG.getConstant(0,
DL, BitTy));
1185 Mask = DAG.getSplat(MaskTy,
DL, Mask);
1193 SDValue NotMask = DAG.getNOT(
DL, Mask, MaskTy);
1202 EVT VT =
Node->getValueType(0);
1205 if (TLI.getOperationAction(
ISD::SRA, VT) == TargetLowering::Expand ||
1206 TLI.getOperationAction(
ISD::SHL, VT) == TargetLowering::Expand)
1207 return DAG.UnrollVectorOp(
Node);
1210 EVT OrigTy = cast<VTSDNode>(
Node->getOperand(1))->getVT();
1214 SDValue ShiftSz = DAG.getConstant(BW - OrigBW,
DL, VT);
1224 EVT VT =
Node->getValueType(0);
1227 EVT SrcVT = Src.getValueType();
1234 "ANY_EXTEND_VECTOR_INREG vector size mismatch");
1239 Src, DAG.getVectorIdxConstant(0,
DL));
1244 ShuffleMask.
resize(NumSrcElements, -1);
1247 int ExtLaneScale = NumSrcElements / NumElements;
1248 int EndianOffset = DAG.getDataLayout().isBigEndian() ? ExtLaneScale - 1 : 0;
1249 for (
int i = 0; i < NumElements; ++i)
1250 ShuffleMask[i * ExtLaneScale + EndianOffset] = i;
1254 DAG.getVectorShuffle(SrcVT,
DL, Src, DAG.getUNDEF(SrcVT), ShuffleMask));
1259 EVT VT =
Node->getValueType(0);
1261 EVT SrcVT = Src.getValueType();
1272 SDValue ShiftAmount = DAG.getConstant(EltWidth - SrcEltWidth,
DL, VT);
1283 EVT VT =
Node->getValueType(0);
1286 EVT SrcVT = Src.getValueType();
1293 "ZERO_EXTEND_VECTOR_INREG vector size mismatch");
1298 Src, DAG.getVectorIdxConstant(0,
DL));
1306 auto ShuffleMask = llvm::to_vector<16>(llvm::seq<int>(0, NumSrcElements));
1308 int ExtLaneScale = NumSrcElements / NumElements;
1309 int EndianOffset = DAG.getDataLayout().isBigEndian() ? ExtLaneScale - 1 : 0;
1310 for (
int i = 0; i < NumElements; ++i)
1311 ShuffleMask[i * ExtLaneScale + EndianOffset] = NumSrcElements + i;
1314 DAG.getVectorShuffle(SrcVT,
DL, Zero, Src, ShuffleMask));
1320 for (
int J = ScalarSizeInBytes - 1; J >= 0; --J)
1321 ShuffleMask.
push_back((
I * ScalarSizeInBytes) + J);
1325 EVT VT =
Node->getValueType(0);
1329 return TLI.expandBSWAP(
Node, DAG);
1337 if (TLI.isShuffleMaskLegal(ShuffleMask, ByteVT)) {
1340 Op = DAG.getVectorShuffle(ByteVT,
DL,
Op, DAG.getUNDEF(ByteVT), ShuffleMask);
1346 if (TLI.isOperationLegalOrCustom(
ISD::SHL, VT) &&
1347 TLI.isOperationLegalOrCustom(
ISD::SRL, VT) &&
1348 TLI.isOperationLegalOrCustomOrPromote(
ISD::AND, VT) &&
1349 TLI.isOperationLegalOrCustomOrPromote(
ISD::OR, VT))
1350 return TLI.expandBSWAP(
Node, DAG);
1353 return DAG.UnrollVectorOp(
Node);
1356void VectorLegalizer::ExpandBITREVERSE(
SDNode *
Node,
1358 EVT VT =
Node->getValueType(0);
1362 Results.push_back(TLI.expandBITREVERSE(
Node, DAG));
1377 if (ScalarSizeInBits > 8 && (ScalarSizeInBits % 8) == 0) {
1382 if (TLI.isShuffleMaskLegal(BSWAPMask, ByteVT) &&
1384 (TLI.isOperationLegalOrCustom(
ISD::SHL, ByteVT) &&
1385 TLI.isOperationLegalOrCustom(
ISD::SRL, ByteVT) &&
1386 TLI.isOperationLegalOrCustomOrPromote(
ISD::AND, ByteVT) &&
1387 TLI.isOperationLegalOrCustomOrPromote(
ISD::OR, ByteVT)))) {
1390 Op = DAG.getVectorShuffle(ByteVT,
DL,
Op, DAG.getUNDEF(ByteVT),
1401 if (TLI.isOperationLegalOrCustom(
ISD::SHL, VT) &&
1402 TLI.isOperationLegalOrCustom(
ISD::SRL, VT) &&
1403 TLI.isOperationLegalOrCustomOrPromote(
ISD::AND, VT) &&
1404 TLI.isOperationLegalOrCustomOrPromote(
ISD::OR, VT)) {
1405 Results.push_back(TLI.expandBITREVERSE(
Node, DAG));
1429 if (TLI.getOperationAction(
ISD::AND, VT) == TargetLowering::Expand ||
1430 TLI.getOperationAction(
ISD::XOR, VT) == TargetLowering::Expand ||
1431 TLI.getOperationAction(
ISD::OR, VT) == TargetLowering::Expand)
1432 return DAG.UnrollVectorOp(
Node);
1438 auto BoolContents = TLI.getBooleanContents(Op1.
getValueType());
1439 if (BoolContents != TargetLowering::ZeroOrNegativeOneBooleanContent &&
1440 !(BoolContents == TargetLowering::ZeroOrOneBooleanContent &&
1442 return DAG.UnrollVectorOp(
Node);
1448 return DAG.UnrollVectorOp(
Node);
1456 SDValue NotMask = DAG.getNOT(
DL, Mask, VT);
1478 if (TLI.getOperationAction(ISD::VP_AND, VT) == TargetLowering::Expand ||
1479 TLI.getOperationAction(ISD::VP_XOR, VT) == TargetLowering::Expand ||
1480 TLI.getOperationAction(ISD::VP_OR, VT) == TargetLowering::Expand)
1481 return DAG.UnrollVectorOp(
Node);
1485 return DAG.UnrollVectorOp(
Node);
1487 SDValue Ones = DAG.getAllOnesConstant(
DL, VT);
1490 Op1 = DAG.
getNode(ISD::VP_AND,
DL, VT, Op1, Mask, Ones, EVL);
1491 Op2 = DAG.
getNode(ISD::VP_AND,
DL, VT, Op2, NotMask, Ones, EVL);
1492 return DAG.getNode(ISD::VP_OR,
DL, VT, Op1, Op2, Ones, EVL);
1507 EVT MaskVT =
Mask.getValueType();
1519 return DAG.UnrollVectorOp(
Node);
1523 if (TLI.getSetCCResultType(DAG.getDataLayout(), *DAG.getContext(),
1524 EVLVecVT) != MaskVT)
1525 return DAG.UnrollVectorOp(
Node);
1527 SDValue StepVec = DAG.getStepVector(
DL, EVLVecVT);
1528 SDValue SplatEVL = DAG.getSplat(EVLVecVT,
DL, EVL);
1530 DAG.getSetCC(
DL, MaskVT, StepVec, SplatEVL, ISD::CondCode::SETULT);
1533 return DAG.getSelect(
DL,
Node->getValueType(0), FullMask, Op1, Op2);
1538 EVT VT =
Node->getValueType(0);
1540 unsigned DivOpc =
Node->getOpcode() == ISD::VP_SREM ? ISD::VP_SDIV : ISD::VP_UDIV;
1542 if (!TLI.isOperationLegalOrCustom(DivOpc, VT) ||
1543 !TLI.isOperationLegalOrCustom(ISD::VP_MUL, VT) ||
1544 !TLI.isOperationLegalOrCustom(ISD::VP_SUB, VT))
1556 SDValue Mul = DAG.getNode(ISD::VP_MUL,
DL, VT, Divisor, Div, Mask, EVL);
1557 return DAG.getNode(ISD::VP_SUB,
DL, VT, Dividend,
Mul, Mask, EVL);
1560void VectorLegalizer::ExpandFP_TO_UINT(
SDNode *
Node,
1564 if (TLI.expandFP_TO_UINT(
Node, Result, Chain, DAG)) {
1566 if (
Node->isStrictFPOpcode())
1572 if (
Node->isStrictFPOpcode()) {
1580void VectorLegalizer::ExpandUINT_TO_FLOAT(
SDNode *
Node,
1582 bool IsStrict =
Node->isStrictFPOpcode();
1583 unsigned OpNo = IsStrict ? 1 : 0;
1585 EVT VT = Src.getValueType();
1591 if (TLI.expandUINT_TO_FP(
Node, Result, Chain, DAG)) {
1600 TargetLowering::Expand) ||
1602 TargetLowering::Expand)) ||
1603 TLI.getOperationAction(
ISD::SRL, VT) == TargetLowering::Expand) {
1614 assert((BW == 64 || BW == 32) &&
1615 "Elements in vector-UINT_TO_FP must be 32 or 64 bits wide");
1617 SDValue HalfWord = DAG.getConstant(BW / 2,
DL, VT);
1622 uint64_t HWMask = (BW == 64) ? 0x00000000FFFFFFFF : 0x0000FFFF;
1623 SDValue HalfWordMask = DAG.getConstant(HWMask,
DL, VT);
1627 DAG.getConstantFP(1ULL << (BW / 2),
DL,
Node->getValueType(0));
1638 {
Node->getValueType(0), MVT::Other},
1639 {
Node->getOperand(0),
HI});
1643 {
Node->getValueType(0), MVT::Other},
1644 {
Node->getOperand(0),
LO});
1672 if (TLI.isOperationLegalOrCustom(
ISD::FSUB,
Node->getValueType(0))) {
1677 Node->getOperand(0));
1679 return DAG.UnrollVectorOp(
Node);
1682void VectorLegalizer::ExpandFSUB(
SDNode *
Node,
1687 EVT VT =
Node->getValueType(0);
1688 if (TLI.isOperationLegalOrCustom(
ISD::FNEG, VT) &&
1689 TLI.isOperationLegalOrCustom(
ISD::FADD, VT))
1696void VectorLegalizer::ExpandSETCC(
SDNode *
Node,
1698 bool NeedInvert =
false;
1699 bool IsVP =
Node->getOpcode() == ISD::VP_SETCC;
1703 unsigned Offset = IsStrict ? 1 : 0;
1710 MVT OpVT =
LHS.getSimpleValueType();
1713 if (TLI.getCondCodeAction(CCCode, OpVT) != TargetLowering::Expand) {
1730 TLI.LegalizeSetCCCondCode(DAG,
Node->getValueType(0), LHS, RHS,
CC, Mask,
1731 EVL, NeedInvert, dl, Chain, IsSignaling);
1738 LHS = DAG.getNode(
Node->getOpcode(), dl,
Node->getVTList(),
1739 {Chain, LHS, RHS, CC},
Node->getFlags());
1740 Chain =
LHS.getValue(1);
1742 LHS = DAG.getNode(ISD::VP_SETCC, dl,
Node->getValueType(0),
1743 {LHS, RHS, CC, Mask, EVL},
Node->getFlags());
1754 LHS = DAG.getLogicalNOT(dl, LHS,
LHS->getValueType(0));
1756 LHS = DAG.getVPLogicalNOT(dl, LHS, Mask, EVL,
LHS->getValueType(0));
1759 assert(!IsStrict &&
"Don't know how to expand for strict nodes.");
1763 EVT VT =
Node->getValueType(0);
1766 DAG.getBoolConstant(
true, dl, VT,
LHS.getValueType()),
1767 DAG.getBoolConstant(
false, dl, VT,
LHS.getValueType()),
CC);
1768 LHS->setFlags(
Node->getFlags());
1776void VectorLegalizer::ExpandUADDSUBO(
SDNode *
Node,
1779 TLI.expandUADDSUBO(
Node, Result, Overflow, DAG);
1784void VectorLegalizer::ExpandSADDSUBO(
SDNode *
Node,
1787 TLI.expandSADDSUBO(
Node, Result, Overflow, DAG);
1792void VectorLegalizer::ExpandMULO(
SDNode *
Node,
1795 if (!TLI.expandMULO(
Node, Result, Overflow, DAG))
1796 std::tie(Result, Overflow) = DAG.UnrollVectorOverflowOp(
Node);
1802void VectorLegalizer::ExpandFixedPointDiv(
SDNode *
Node,
1805 if (
SDValue Expanded = TLI.expandFixedPointDiv(
N->getOpcode(),
SDLoc(
N),
1806 N->getOperand(0),
N->getOperand(1),
N->getConstantOperandVal(2), DAG))
1810void VectorLegalizer::ExpandStrictFPOp(
SDNode *
Node,
1833 "Expected REM node");
1836 if (!TLI.expandREM(
Node, Result, DAG))
1851 assert(!
Node->isStrictFPOpcode() &&
"Unexpected strict fp operation!");
1853 const char *LCName = TLI.getLibcallName(LC);
1856 LLVM_DEBUG(
dbgs() <<
"Looking for vector variant of " << LCName <<
"\n");
1858 EVT VT =
Node->getValueType(0);
1876 for (
unsigned i = 0; i <
Node->getNumOperands(); ++i) {
1877 assert(
Node->getOperand(i).getValueType() == VT &&
1878 "Expected matching vector types!");
1881 FunctionType *ScalarFTy = FunctionType::get(ScalarTy, ArgTys,
false);
1893 if (OptVFInfo->Shape.Parameters.size() !=
1902 Entry.IsSExt =
false;
1903 Entry.IsZExt =
false;
1906 for (
auto &VFParam : OptVFInfo->Shape.Parameters) {
1907 if (VFParam.ParamKind == VFParamKind::GlobalPredicate) {
1908 EVT MaskVT = TLI.getSetCCResultType(DAG.getDataLayout(), *Ctx, VT);
1909 Entry.Node = DAG.getBoolConstant(
true,
DL, MaskVT, VT);
1911 Args.push_back(Entry);
1916 if (VFParam.ParamKind != VFParamKind::Vector)
1921 Args.push_back(Entry);
1926 TLI.getPointerTy(DAG.getDataLayout()));
1929 .setChain(DAG.getEntryNode())
1932 std::pair<SDValue, SDValue> CallResult = TLI.LowerCallTo(CLI);
1933 Results.push_back(CallResult.first);
1938bool VectorLegalizer::tryExpandVecMathCall(
1943 Node->getValueType(0).getVectorElementType(), Call_F32, Call_F64,
1944 Call_F80, Call_F128, Call_PPCF128);
1946 if (LC == RTLIB::UNKNOWN_LIBCALL)
1952void VectorLegalizer::UnrollStrictFPOp(
SDNode *
Node,
1954 EVT VT =
Node->getValueType(0);
1957 unsigned NumOpers =
Node->getNumOperands();
1960 EVT TmpEltVT = EltVT;
1964 *DAG.getContext(), TmpEltVT);
1966 EVT ValueVTs[] = {TmpEltVT, MVT::Other};
1972 for (
unsigned i = 0; i < NumElems; ++i) {
1974 SDValue Idx = DAG.getVectorIdxConstant(i, dl);
1980 for (
unsigned j = 1;
j < NumOpers; ++
j) {
1997 ScalarResult = DAG.getSelect(dl, EltVT, ScalarResult,
1998 DAG.getAllOnesConstant(dl, EltVT),
1999 DAG.getConstant(0, dl, EltVT));
2013 EVT VT =
Node->getValueType(0);
2019 EVT TmpEltVT =
LHS.getValueType().getVectorElementType();
2022 for (
unsigned i = 0; i < NumElems; ++i) {
2024 DAG.getVectorIdxConstant(i, dl));
2026 DAG.getVectorIdxConstant(i, dl));
2029 *DAG.getContext(), TmpEltVT),
2030 LHSElem, RHSElem,
CC);
2031 Ops[i] = DAG.getSelect(dl, EltVT, Ops[i], DAG.getAllOnesConstant(dl, EltVT),
2032 DAG.getConstant(0, dl, EltVT));
2034 return DAG.getBuildVector(VT, dl, Ops);
2038 return VectorLegalizer(*this).Run();
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
Function Alias Analysis Results
BlockVerifier::State From
Returns the sub type a function will return at a given Idx Should correspond to the result type of an ExtractValue instruction executed with just that one unsigned Idx
This file defines the DenseMap class.
static void createBSWAPShuffleMask(EVT VT, SmallVectorImpl< int > &ShuffleMask)
mir Rename Register Operands
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file defines the SmallVector class.
This file describes how to lower LLVM code to machine code.
DEMANGLE_DUMP_METHOD void dump() const
This class represents an Operation in the Expression.
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
This is an important class for using LLVM in a threaded context.
This class is used to represent ISD::LOAD nodes.
unsigned getVectorNumElements() const
bool isVector() const
Return true if this is a vector value type.
MVT getVectorElementType() const
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
MVT getScalarType() const
If this is a vector, return the element type, otherwise return this.
MutableArrayRef - Represent a mutable reference to an array (0 or more elements consecutively in memo...
Wrapper class for IR location info (IR ordering and DebugLoc) to be passed into SDNode creation funct...
Represents one node in the SelectionDAG.
unsigned getNumValues() const
Return the number of values defined/returned by this operator.
Unlike LLVM values, Selection DAG nodes may return multiple values as the result of a computation.
SDNode * getNode() const
get the SDNode which holds the desired result
SDValue getValue(unsigned R) const
EVT getValueType() const
Return the ValueType of the referenced return value.
TypeSize getValueSizeInBits() const
Returns the size of the value in bits.
This is used to represent a portion of an LLVM function in a low-level Data Dependence DAG representa...
bool LegalizeVectors()
This transforms the SelectionDAG into a SelectionDAG that only uses vector math operations supported ...
const TargetLowering & getTargetLoweringInfo() const
ilist< SDNode >::iterator allnodes_iterator
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
This class is used to represent ISD::STORE nodes.
constexpr const char * data() const
data - Get a pointer to the start of the string (which may not be null terminated).
Provides information about what library functions are available for the current target.
const VecDesc * getVectorMappingInfo(StringRef F, const ElementCount &VF, bool Masked) const
LegalizeAction
This enum indicates whether operations are valid for a target, and if not, what action should be used...
virtual EVT getSetCCResultType(const DataLayout &DL, LLVMContext &Context, EVT VT) const
Return the ValueType of the result of SETCC operations.
std::vector< ArgListEntry > ArgListTy
This class defines information used to lower LLVM code to legal SelectionDAG operators that the targe...
The instances of the Type class are immutable: once they are created, they are never changed.
Type * getScalarType() const
If this is a vector type, return the element type, otherwise return 'this'.
Provides info so a possible vectorization of a function can be computed.
std::string getVectorFunctionABIVariantString() const
Returns a vector function ABI variant string on the form: ZGV<isa><mask><vlen><vparams><scalarname>(<...
StringRef getVectorFnName() const
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
constexpr char Args[]
Key for Kernel::Metadata::mArgs.
constexpr std::underlying_type_t< E > Mask()
Get a bitmask with 1s in all places up to the high-order bit of E's largest value.
@ C
The default llvm calling convention, compatible with C.
@ SETCC
SetCC operator - This evaluates to a true value iff the condition is true.
@ MERGE_VALUES
MERGE_VALUES - This node takes multiple discrete operands and returns them all as its individual resu...
@ STRICT_FSETCC
STRICT_FSETCC/STRICT_FSETCCS - Constrained versions of SETCC, used for floating-point operands only.
@ VECREDUCE_SEQ_FADD
Generic reduction nodes.
@ SMUL_LOHI
SMUL_LOHI/UMUL_LOHI - Multiply two integers of type iN, producing a signed/unsigned value of type i[2...
@ INSERT_SUBVECTOR
INSERT_SUBVECTOR(VECTOR1, VECTOR2, IDX) - Returns a vector with VECTOR2 inserted into VECTOR1.
@ BSWAP
Byte Swap and Counting operators.
@ SMULFIX
RESULT = [US]MULFIX(LHS, RHS, SCALE) - Perform fixed point multiplication on 2 integers with the same...
@ ADD
Simple integer binary arithmetic operators.
@ LOAD
LOAD and STORE have token chains as their first operand, then the same operands as an LLVM load/store...
@ SMULFIXSAT
Same as the corresponding unsaturated fixed point instructions, but the result is clamped between the...
@ ANY_EXTEND
ANY_EXTEND - Used for integer types. The high bits are undefined.
@ FMA
FMA - Perform a * b + c with no intermediate rounding step.
@ SINT_TO_FP
[SU]INT_TO_FP - These operators convert integers (whose interpreted sign depends on the first letter)...
@ VECREDUCE_FMAX
FMIN/FMAX nodes can have flags, for NaN/NoNaN variants.
@ FADD
Simple binary floating point operators.
@ VECREDUCE_FMAXIMUM
FMINIMUM/FMAXIMUM nodes propatate NaNs and signed zeroes using the llvm.minimum and llvm....
@ ABS
ABS - Determine the unsigned absolute value of a signed integer value of the same bitwidth.
@ SIGN_EXTEND_VECTOR_INREG
SIGN_EXTEND_VECTOR_INREG(Vector) - This operator represents an in-register sign-extension of the low ...
@ SDIVREM
SDIVREM/UDIVREM - Divide two integers and produce both a quotient and remainder result.
@ FPTRUNC_ROUND
FPTRUNC_ROUND - This corresponds to the fptrunc_round intrinsic.
@ BITCAST
BITCAST - This operator converts between integer, vector and FP values, as if the value was stored to...
@ FLDEXP
FLDEXP - ldexp, inspired by libm (op0 * 2**op1).
@ SDIVFIX
RESULT = [US]DIVFIX(LHS, RHS, SCALE) - Perform fixed point division on 2 integers with the same width...
@ STRICT_FSQRT
Constrained versions of libm-equivalent floating point intrinsics.
@ SIGN_EXTEND
Conversion operators.
@ AVGCEILS
AVGCEILS/AVGCEILU - Rounding averaging add - Add two integers using an integer of type i[N+2],...
@ VECREDUCE_FADD
These reductions have relaxed evaluation order semantics, and have a single vector operand.
@ CTTZ_ZERO_UNDEF
Bit counting operators with an undefined result for zero inputs.
@ FNEG
Perform various unary floating-point operations inspired by libm.
@ SSUBO
Same for subtraction.
@ STEP_VECTOR
STEP_VECTOR(IMM) - Returns a scalable vector whose lanes are comprised of a linear sequence of unsign...
@ FCANONICALIZE
Returns platform specific canonical encoding of a floating point number.
@ SSUBSAT
RESULT = [US]SUBSAT(LHS, RHS) - Perform saturation subtraction on 2 integers with the same bit width ...
@ SELECT
Select(COND, TRUEVAL, FALSEVAL).
@ SPLAT_VECTOR
SPLAT_VECTOR(VAL) - Returns a vector with the scalar value VAL duplicated in all lanes.
@ SADDO
RESULT, BOOL = [SU]ADDO(LHS, RHS) - Overflow-aware nodes for addition.
@ VECREDUCE_ADD
Integer reductions may have a result type larger than the vector element type.
@ MULHU
MULHU/MULHS - Multiply high - Multiply two integers of type iN, producing an unsigned/signed value of...
@ SHL
Shift and rotation operations.
@ FMINNUM_IEEE
FMINNUM_IEEE/FMAXNUM_IEEE - Perform floating-point minimumNumber or maximumNumber on two values,...
@ EXTRACT_VECTOR_ELT
EXTRACT_VECTOR_ELT(VECTOR, IDX) - Returns a single element from VECTOR identified by the (potentially...
@ ZERO_EXTEND
ZERO_EXTEND - Used for integer types, zeroing the new bits.
@ SELECT_CC
Select with condition operator - This selects between a true value and a false value (ops #2 and #3) ...
@ FMINNUM
FMINNUM/FMAXNUM - Perform floating-point minimum or maximum on two values.
@ SSHLSAT
RESULT = [US]SHLSAT(LHS, RHS) - Perform saturation left shift.
@ SMULO
Same for multiplication.
@ ANY_EXTEND_VECTOR_INREG
ANY_EXTEND_VECTOR_INREG(Vector) - This operator represents an in-register any-extension of the low la...
@ SIGN_EXTEND_INREG
SIGN_EXTEND_INREG - This operator atomically performs a SHL/SRA pair to sign extend a small value in ...
@ SMIN
[US]{MIN/MAX} - Binary minimum or maximum of signed or unsigned integers.
@ SDIVFIXSAT
Same as the corresponding unsaturated fixed point instructions, but the result is clamped between the...
@ FP_EXTEND
X = FP_EXTEND(Y) - Extend a smaller FP type into a larger FP type.
@ VSELECT
Select with a vector condition (op #0) and two vector operands (ops #1 and #2), returning a vector re...
@ STRICT_SINT_TO_FP
STRICT_[US]INT_TO_FP - Convert a signed or unsigned integer to a floating point value.
@ MGATHER
Masked gather and scatter - load and store operations for a vector of random addresses with additiona...
@ STRICT_FP_ROUND
X = STRICT_FP_ROUND(Y, TRUNC) - Rounding 'Y' from a larger floating point type down to the precision ...
@ STRICT_FP_TO_SINT
STRICT_FP_TO_[US]INT - Convert a floating point value to a signed or unsigned integer.
@ FMINIMUM
FMINIMUM/FMAXIMUM - NaN-propagating minimum/maximum that also treat -0.0 as less than 0....
@ FP_TO_SINT
FP_TO_[US]INT - Convert a floating point value to a signed or unsigned integer.
@ STRICT_FP_EXTEND
X = STRICT_FP_EXTEND(Y) - Extend a smaller FP type into a larger FP type.
@ AND
Bitwise operators - logical and, logical or, logical xor.
@ SCMP
[US]CMP - 3-way comparison of signed or unsigned integers.
@ AVGFLOORS
AVGFLOORS/AVGFLOORU - Averaging add - Add two integers using an integer of type i[N+1],...
@ STRICT_FADD
Constrained versions of the binary floating point operators.
@ TokenFactor
TokenFactor - This node takes multiple tokens as input and produces a single token result.
@ FFREXP
FFREXP - frexp, extract fractional and exponent component of a floating-point value.
@ FP_ROUND
X = FP_ROUND(Y, TRUNC) - Rounding 'Y' from a larger floating point type down to the precision of the ...
@ VECTOR_COMPRESS
VECTOR_COMPRESS(Vec, Mask, Passthru) consecutively place vector elements based on mask e....
@ ZERO_EXTEND_VECTOR_INREG
ZERO_EXTEND_VECTOR_INREG(Vector) - This operator represents an in-register zero-extension of the low ...
@ FP_TO_SINT_SAT
FP_TO_[US]INT_SAT - Convert floating point value in operand 0 to a signed or unsigned scalar integer ...
@ TRUNCATE
TRUNCATE - Completely drop the high bits.
@ AssertSext
AssertSext, AssertZext - These nodes record if a register contains a value that has already been zero...
@ FCOPYSIGN
FCOPYSIGN(X, Y) - Return the value of X with the sign of Y.
@ SADDSAT
RESULT = [US]ADDSAT(LHS, RHS) - Perform saturation addition on 2 integers with the same bit width (W)...
@ ABDS
ABDS/ABDU - Absolute difference - Return the absolute difference between two numbers interpreted as s...
@ BUILD_VECTOR
BUILD_VECTOR(ELT0, ELT1, ELT2, ELT3,...) - Return a fixed-width vector with the specified,...
std::optional< unsigned > getVPMaskIdx(unsigned Opcode)
The operand position of the vector mask.
CondCode
ISD::CondCode enum - These are ordered carefully to make the bitfields below work out,...
LoadExtType
LoadExtType enum - This enum defines the three variants of LOADEXT (load with extension).
bool isVPOpcode(unsigned Opcode)
Whether this is a vector-predicated Opcode.
Libcall
RTLIB::Libcall enum - This enum defines all of the runtime library calls the backend can emit.
Libcall getFPLibCall(EVT VT, Libcall Call_F32, Libcall Call_F64, Libcall Call_F80, Libcall Call_F128, Libcall Call_PPCF128)
GetFPLibCall - Helper to return the right libcall for the given floating point type,...
ManagedStatic< cl::opt< FnT >, OptCreatorT > Action
std::optional< VFInfo > tryDemangleForVFABI(StringRef MangledName, const FunctionType *FTy)
Function to construct a VFInfo out of a mangled names in the following format:
This is an optimization pass for GlobalISel generic memory operations.
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
EVT changeVectorElementTypeToInteger() const
Return a vector with the same number of elements as this vector, but with the element type converted ...
static EVT getVectorVT(LLVMContext &Context, EVT VT, unsigned NumElements, bool IsScalable=false)
Returns the EVT that represents a vector NumElements in length, where each element is of type VT.
ElementCount getVectorElementCount() const
TypeSize getSizeInBits() const
Return the size of the specified value type in bits.
uint64_t getScalarSizeInBits() const
bool isFixedLengthVector() const
bool isVector() const
Return true if this is a vector value type.
EVT getScalarType() const
If this is a vector type, return the element type, otherwise return this.
Type * getTypeForEVT(LLVMContext &Context) const
This method returns an LLVM type corresponding to the specified EVT.
bool isScalableVector() const
Return true if this is a vector type where the runtime length is machine dependent.
EVT getVectorElementType() const
Given a vector type, return the type of each element.
unsigned getVectorNumElements() const
Given a vector type, return the number of elements it contains.
bool bitsLE(EVT VT) const
Return true if this has no more bits than VT.
This represents a list of ValueType's that has been intern'd by a SelectionDAG.
This structure contains all information that is necessary for lowering calls.