99void SelectionDAG::DAGNodeDeletedListener::anchor() {}
100void SelectionDAG::DAGNodeInsertedListener::anchor() {}
102#define DEBUG_TYPE "selectiondag"
106 cl::desc(
"Gang up loads and stores generated by inlining of memcpy"));
109 cl::desc(
"Number limit for gluing ld/st of memcpy."),
114 cl::desc(
"DAG combiner limit number of steps when searching DAG "
115 "for predecessor nodes"));
153 if (
auto OptAPInt =
N->getOperand(0)->bitcastToAPInt()) {
155 N->getValueType(0).getVectorElementType().getSizeInBits();
156 SplatVal = OptAPInt->
trunc(EltSize);
166 unsigned SplatBitSize;
168 unsigned EltSize =
N->getValueType(0).getVectorElementType().getSizeInBits();
173 const bool IsBigEndian =
false;
174 return BV->isConstantSplat(SplatVal, SplatUndef, SplatBitSize, HasUndefs,
175 EltSize, IsBigEndian) &&
176 EltSize == SplatBitSize;
185 N =
N->getOperand(0).getNode();
194 unsigned i = 0, e =
N->getNumOperands();
197 while (i != e &&
N->getOperand(i).isUndef())
201 if (i == e)
return false;
213 unsigned EltSize =
N->getValueType(0).getScalarSizeInBits();
214 if (OptAPInt->countr_one() < EltSize)
222 for (++i; i != e; ++i)
223 if (
N->getOperand(i) != NotZero && !
N->getOperand(i).isUndef())
231 N =
N->getOperand(0).getNode();
240 bool IsAllUndef =
true;
253 if (
auto OptAPInt =
Op->bitcastToAPInt()) {
254 unsigned EltSize =
N->getValueType(0).getScalarSizeInBits();
255 if (OptAPInt->countr_zero() < EltSize)
303 assert(
N->getValueType(0).isVector() &&
"Expected a vector!");
305 unsigned EltSize =
N->getValueType(0).getScalarSizeInBits();
306 if (EltSize <= NewEltSize)
310 return (
N->getOperand(0).getValueType().getScalarSizeInBits() <=
315 return (
N->getOperand(0).getValueType().getScalarSizeInBits() <=
328 APInt C =
Op->getAsAPIntVal().trunc(EltSize);
329 if (
Signed &&
C.trunc(NewEltSize).sext(EltSize) !=
C)
331 if (!
Signed &&
C.trunc(NewEltSize).zext(EltSize) !=
C)
342 if (
N->getNumOperands() == 0)
348 return N->getOpcode() ==
ISD::FREEZE &&
N->getOperand(0).isUndef();
351template <
typename ConstNodeType>
353 std::function<
bool(ConstNodeType *)> Match,
354 bool AllowUndefs,
bool AllowTruncation) {
364 EVT SVT =
Op.getValueType().getScalarType();
365 for (
unsigned i = 0, e =
Op.getNumOperands(); i != e; ++i) {
366 if (AllowUndefs &&
Op.getOperand(i).isUndef()) {
373 if (!Cst || (!AllowTruncation && Cst->getValueType(0) != SVT) ||
388 bool AllowUndefs,
bool AllowTypeMismatch) {
389 if (!AllowTypeMismatch && LHS.getValueType() != RHS.getValueType())
395 return Match(LHSCst, RHSCst);
398 if (LHS.getOpcode() != RHS.getOpcode() ||
404 for (
unsigned i = 0, e = LHS.getNumOperands(); i != e; ++i) {
407 bool LHSUndef = AllowUndefs && LHSOp.
isUndef();
408 bool RHSUndef = AllowUndefs && RHSOp.
isUndef();
411 if ((!LHSCst && !LHSUndef) || (!RHSCst && !RHSUndef))
413 if (!AllowTypeMismatch && (LHSOp.
getValueType() != SVT ||
416 if (!Match(LHSCst, RHSCst))
453 switch (VecReduceOpcode) {
458 case ISD::VP_REDUCE_FADD:
459 case ISD::VP_REDUCE_SEQ_FADD:
463 case ISD::VP_REDUCE_FMUL:
464 case ISD::VP_REDUCE_SEQ_FMUL:
467 case ISD::VP_REDUCE_ADD:
470 case ISD::VP_REDUCE_MUL:
473 case ISD::VP_REDUCE_AND:
476 case ISD::VP_REDUCE_OR:
479 case ISD::VP_REDUCE_XOR:
482 case ISD::VP_REDUCE_SMAX:
485 case ISD::VP_REDUCE_SMIN:
488 case ISD::VP_REDUCE_UMAX:
491 case ISD::VP_REDUCE_UMIN:
494 case ISD::VP_REDUCE_FMAX:
497 case ISD::VP_REDUCE_FMIN:
500 case ISD::VP_REDUCE_FMAXIMUM:
503 case ISD::VP_REDUCE_FMINIMUM:
527#define BEGIN_REGISTER_VP_SDNODE(VPSD, ...) \
530#include "llvm/IR/VPIntrinsics.def"
538#define BEGIN_REGISTER_VP_SDNODE(VPSD, ...) case ISD::VPSD:
539#define VP_PROPERTY_BINARYOP return true;
540#define END_REGISTER_VP_SDNODE(VPSD) break;
541#include "llvm/IR/VPIntrinsics.def"
550 case ISD::VP_REDUCE_ADD:
551 case ISD::VP_REDUCE_MUL:
552 case ISD::VP_REDUCE_AND:
553 case ISD::VP_REDUCE_OR:
554 case ISD::VP_REDUCE_XOR:
555 case ISD::VP_REDUCE_SMAX:
556 case ISD::VP_REDUCE_SMIN:
557 case ISD::VP_REDUCE_UMAX:
558 case ISD::VP_REDUCE_UMIN:
559 case ISD::VP_REDUCE_FMAX:
560 case ISD::VP_REDUCE_FMIN:
561 case ISD::VP_REDUCE_FMAXIMUM:
562 case ISD::VP_REDUCE_FMINIMUM:
563 case ISD::VP_REDUCE_FADD:
564 case ISD::VP_REDUCE_FMUL:
565 case ISD::VP_REDUCE_SEQ_FADD:
566 case ISD::VP_REDUCE_SEQ_FMUL:
576#define BEGIN_REGISTER_VP_SDNODE(VPSD, LEGALPOS, TDNAME, MASKPOS, ...) \
579#include "llvm/IR/VPIntrinsics.def"
588#define BEGIN_REGISTER_VP_SDNODE(VPSD, LEGALPOS, TDNAME, MASKPOS, EVLPOS) \
591#include "llvm/IR/VPIntrinsics.def"
601#define BEGIN_REGISTER_VP_SDNODE(VPOPC, ...) case ISD::VPOPC:
602#define VP_PROPERTY_FUNCTIONAL_SDOPC(SDOPC) return ISD::SDOPC;
603#define END_REGISTER_VP_SDNODE(VPOPC) break;
604#include "llvm/IR/VPIntrinsics.def"
613#define BEGIN_REGISTER_VP_SDNODE(VPOPC, ...) break;
614#define VP_PROPERTY_FUNCTIONAL_SDOPC(SDOPC) case ISD::SDOPC:
615#define END_REGISTER_VP_SDNODE(VPOPC) return ISD::VPOPC;
616#include "llvm/IR/VPIntrinsics.def"
663 bool isIntegerLike) {
688 bool IsInteger =
Type.isInteger();
693 unsigned Op = Op1 | Op2;
709 bool IsInteger =
Type.isInteger();
744 ID.AddPointer(VTList.
VTs);
750 for (
const auto &
Op :
Ops) {
751 ID.AddPointer(
Op.getNode());
752 ID.AddInteger(
Op.getResNo());
759 for (
const auto &
Op :
Ops) {
760 ID.AddPointer(
Op.getNode());
761 ID.AddInteger(
Op.getResNo());
774 switch (
N->getOpcode()) {
783 ID.AddPointer(
C->getConstantIntValue());
784 ID.AddBoolean(
C->isOpaque());
848 ID.AddInteger(LD->getMemoryVT().getRawBits());
849 ID.AddInteger(LD->getRawSubclassData());
850 ID.AddInteger(LD->getPointerInfo().getAddrSpace());
851 ID.AddInteger(LD->getMemOperand()->getFlags());
856 ID.AddInteger(ST->getMemoryVT().getRawBits());
857 ID.AddInteger(ST->getRawSubclassData());
858 ID.AddInteger(ST->getPointerInfo().getAddrSpace());
859 ID.AddInteger(ST->getMemOperand()->getFlags());
870 case ISD::VP_LOAD_FF: {
872 ID.AddInteger(LD->getMemoryVT().getRawBits());
873 ID.AddInteger(LD->getRawSubclassData());
874 ID.AddInteger(LD->getPointerInfo().getAddrSpace());
875 ID.AddInteger(LD->getMemOperand()->getFlags());
878 case ISD::VP_STORE: {
886 case ISD::EXPERIMENTAL_VP_STRIDED_LOAD: {
893 case ISD::EXPERIMENTAL_VP_STRIDED_STORE: {
900 case ISD::VP_GATHER: {
908 case ISD::VP_SCATTER: {
1007 ID.AddInteger(MN->getRawSubclassData());
1008 ID.AddInteger(MN->getMemoryVT().getRawBits());
1010 ID.AddInteger(MMO->getPointerInfo().getAddrSpace());
1011 ID.AddInteger(MMO->getFlags());
1035 if (
N->getValueType(0) == MVT::Glue)
1038 switch (
N->getOpcode()) {
1046 for (
unsigned i = 1, e =
N->getNumValues(); i != e; ++i)
1047 if (
N->getValueType(i) == MVT::Glue)
1056 EVT VT = V.getValueType();
1075 if (
Node.use_empty())
1090 while (!DeadNodes.
empty()) {
1099 DUL->NodeDeleted(
N,
nullptr);
1102 RemoveNodeFromCSEMaps(
N);
1133 RemoveNodeFromCSEMaps(
N);
1137 DeleteNodeNotInCSEMaps(
N);
1140void SelectionDAG::DeleteNodeNotInCSEMaps(
SDNode *
N) {
1141 assert(
N->getIterator() != AllNodes.begin() &&
1142 "Cannot delete the entry node!");
1143 assert(
N->use_empty() &&
"Cannot delete a node that is not dead!");
1152 assert(!(V->isVariadic() && isParameter));
1154 ByvalParmDbgValues.push_back(V);
1156 DbgValues.push_back(V);
1159 DbgValMap[
Node].push_back(V);
1163 DbgValMapType::iterator
I = DbgValMap.find(
Node);
1164 if (
I == DbgValMap.end())
1166 for (
auto &Val:
I->second)
1167 Val->setIsInvalidated();
1171void SelectionDAG::DeallocateNode(
SDNode *
N) {
1194void SelectionDAG::verifyNode(
SDNode *
N)
const {
1195 switch (
N->getOpcode()) {
1197 if (
N->isTargetOpcode())
1201 EVT VT =
N->getValueType(0);
1202 assert(
N->getNumValues() == 1 &&
"Too many results!");
1204 "Wrong return type!");
1205 assert(
N->getNumOperands() == 2 &&
"Wrong number of operands!");
1206 assert(
N->getOperand(0).getValueType() ==
N->getOperand(1).getValueType() &&
1207 "Mismatched operand types!");
1209 "Wrong operand type!");
1211 "Wrong return type size");
1215 assert(
N->getNumValues() == 1 &&
"Too many results!");
1216 assert(
N->getValueType(0).isVector() &&
"Wrong return type!");
1217 assert(
N->getNumOperands() ==
N->getValueType(0).getVectorNumElements() &&
1218 "Wrong number of operands!");
1219 EVT EltVT =
N->getValueType(0).getVectorElementType();
1220 for (
const SDUse &
Op :
N->ops()) {
1221 assert((
Op.getValueType() == EltVT ||
1222 (EltVT.
isInteger() &&
Op.getValueType().isInteger() &&
1223 EltVT.
bitsLE(
Op.getValueType()))) &&
1224 "Wrong operand type!");
1225 assert(
Op.getValueType() ==
N->getOperand(0).getValueType() &&
1226 "Operands must all have the same type");
1234 assert(
N->getNumValues() == 2 &&
"Wrong number of results!");
1235 assert(
N->getVTList().NumVTs == 2 &&
N->getNumOperands() == 2 &&
1236 "Invalid add/sub overflow op!");
1237 assert(
N->getVTList().VTs[0].isInteger() &&
1238 N->getVTList().VTs[1].isInteger() &&
1239 N->getOperand(0).getValueType() ==
N->getOperand(1).getValueType() &&
1240 N->getOperand(0).getValueType() ==
N->getVTList().VTs[0] &&
1241 "Binary operator types must match!");
1251void SelectionDAG::InsertNode(SDNode *
N) {
1252 AllNodes.push_back(
N);
1254 N->PersistentId = NextPersistentId++;
1258 DUL->NodeInserted(
N);
1265bool SelectionDAG::RemoveNodeFromCSEMaps(SDNode *
N) {
1266 bool Erased =
false;
1267 switch (
N->getOpcode()) {
1271 "Cond code doesn't exist!");
1280 Erased = TargetExternalSymbols.erase(std::pair<std::string, unsigned>(
1286 Erased = MCSymbols.erase(MCSN->getMCSymbol());
1292 Erased = ExtendedValueTypeNodes.erase(VT);
1303 Erased = CSEMap.RemoveNode(
N);
1310 if (!Erased &&
N->getValueType(
N->getNumValues()-1) != MVT::Glue &&
1325SelectionDAG::AddModifiedNodeToCSEMaps(SDNode *
N) {
1329 SDNode *Existing = CSEMap.GetOrInsertNode(
N);
1330 if (Existing !=
N) {
1341 DUL->NodeDeleted(
N, Existing);
1342 DeleteNodeNotInCSEMaps(
N);
1349 DUL->NodeUpdated(
N);
1356SDNode *SelectionDAG::FindModifiedNodeSlot(SDNode *
N,
SDValue Op,
1362 FoldingSetNodeID
ID;
1365 SDNode *
Node = FindNodeOrInsertPos(
ID, SDLoc(
N), InsertPos);
1367 Node->intersectFlagsWith(
N->getFlags());
1375SDNode *SelectionDAG::FindModifiedNodeSlot(SDNode *
N,
1382 FoldingSetNodeID
ID;
1385 SDNode *
Node = FindNodeOrInsertPos(
ID, SDLoc(
N), InsertPos);
1387 Node->intersectFlagsWith(
N->getFlags());
1400 FoldingSetNodeID
ID;
1403 SDNode *
Node = FindNodeOrInsertPos(
ID, SDLoc(
N), InsertPos);
1405 Node->intersectFlagsWith(
N->getFlags());
1418 : TM(tm), OptLevel(OL), EntryNode(
ISD::EntryToken, 0,
DebugLoc(),
1421 InsertNode(&EntryNode);
1433 SDAGISelPass = PassPtr;
1437 LibInfo = LibraryInfo;
1438 Libcalls = LibcallsInfo;
1439 Context = &MF->getFunction().getContext();
1444 FnVarLocs = VarLocs;
1448 assert(!UpdateListeners &&
"Dangling registered DAGUpdateListeners");
1450 OperandRecycler.clear(OperandAllocator);
1458void SelectionDAG::allnodes_clear() {
1459 assert(&*AllNodes.begin() == &EntryNode);
1460 AllNodes.remove(AllNodes.begin());
1461 while (!AllNodes.empty())
1462 DeallocateNode(&AllNodes.front());
1464 NextPersistentId = 0;
1470 SDNode *
N = CSEMap.FindNodeOrInsertPos(
ID, InsertPos);
1472 switch (
N->getOpcode()) {
1477 "debug location. Use another overload.");
1484 const SDLoc &
DL,
void *&InsertPos) {
1485 SDNode *
N = CSEMap.FindNodeOrInsertPos(
ID, InsertPos);
1487 switch (
N->getOpcode()) {
1493 if (
N->getDebugLoc() !=
DL.getDebugLoc())
1500 if (
DL.getIROrder() &&
DL.getIROrder() <
N->getIROrder())
1501 N->setDebugLoc(
DL.getDebugLoc());
1510 OperandRecycler.clear(OperandAllocator);
1511 OperandAllocator.Reset();
1514 ExtendedValueTypeNodes.clear();
1515 ExternalSymbols.clear();
1516 TargetExternalSymbols.clear();
1522 EntryNode.UseList =
nullptr;
1523 InsertNode(&EntryNode);
1529 return VT.
bitsGT(
Op.getValueType())
1535std::pair<SDValue, SDValue>
1539 "Strict no-op FP extend/round not allowed.");
1546 return std::pair<SDValue, SDValue>(Res,
SDValue(Res.
getNode(), 1));
1550 return VT.
bitsGT(
Op.getValueType()) ?
1556 return VT.
bitsGT(
Op.getValueType()) ?
1562 return VT.
bitsGT(
Op.getValueType()) ?
1570 auto Type =
Op.getValueType();
1574 auto Size =
Op.getValueSizeInBits();
1585 auto Type =
Op.getValueType();
1589 auto Size =
Op.getValueSizeInBits();
1600 auto Type =
Op.getValueType();
1604 auto Size =
Op.getValueSizeInBits();
1618 return getNode(TLI->getExtendForContent(BType), SL, VT,
Op);
1622 EVT OpVT =
Op.getValueType();
1624 "Cannot getZeroExtendInReg FP types");
1626 "getZeroExtendInReg type should be vector iff the operand "
1630 "Vector element counts must match in getZeroExtendInReg");
1648 EVT OpVT =
Op.getValueType();
1650 "Cannot getVPZeroExtendInReg FP types");
1652 "getVPZeroExtendInReg type and operand type should be vector!");
1654 "Vector element counts must match in getZeroExtendInReg");
1693 return getNode(ISD::VP_XOR,
DL, VT, Val, TrueValue, Mask, EVL);
1704 return getNode(ISD::VP_ZERO_EXTEND,
DL, VT,
Op, Mask, EVL);
1706 return getNode(ISD::VP_TRUNCATE,
DL, VT,
Op, Mask, EVL);
1715 switch (TLI->getBooleanContents(OpVT)) {
1726 bool isT,
bool isO) {
1732 bool isT,
bool isO) {
1733 return getConstant(*ConstantInt::get(*Context, Val),
DL, VT, isT, isO);
1737 EVT VT,
bool isT,
bool isO) {
1754 EltVT = TLI->getTypeToTransformTo(*
getContext(), EltVT);
1760 Elt = ConstantInt::get(*
getContext(), NewVal);
1772 EVT ViaEltVT = TLI->getTypeToTransformTo(*
getContext(), EltVT);
1779 "Can only handle an even split!");
1783 for (
unsigned i = 0; i != Parts; ++i)
1785 NewVal.
extractBits(ViaEltSizeInBits, i * ViaEltSizeInBits),
DL,
1786 ViaEltVT, isT, isO));
1791 unsigned ViaVecNumElts = VT.
getSizeInBits() / ViaEltSizeInBits;
1802 NewVal.
extractBits(ViaEltSizeInBits, i * ViaEltSizeInBits),
DL,
1803 ViaEltVT, isT, isO));
1808 std::reverse(EltParts.
begin(), EltParts.
end());
1827 "APInt size does not match type size!");
1836 if ((
N = FindNodeOrInsertPos(
ID,
DL, IP)))
1841 N = newSDNode<ConstantSDNode>(isT, isO, Elt, VTs);
1843 N->setDebugLoc(
DL.getDebugLoc());
1844 CSEMap.InsertNode(
N, IP);
1856 bool isT,
bool isO) {
1864 IsTarget, IsOpaque);
1896 EVT VT,
bool isTarget) {
1917 if ((
N = FindNodeOrInsertPos(
ID,
DL, IP)))
1922 N = newSDNode<ConstantFPSDNode>(isTarget, Elt, VTs);
1923 CSEMap.InsertNode(
N, IP);
1937 if (EltVT == MVT::f32)
1939 if (EltVT == MVT::f64)
1941 if (EltVT == MVT::f80 || EltVT == MVT::f128 || EltVT == MVT::ppcf128 ||
1942 EltVT == MVT::f16 || EltVT == MVT::bf16) {
1953 EVT VT, int64_t
Offset,
bool isTargetGA,
1954 unsigned TargetFlags) {
1955 assert((TargetFlags == 0 || isTargetGA) &&
1956 "Cannot set target flags on target-independent globals");
1974 ID.AddInteger(TargetFlags);
1976 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP))
1979 auto *
N = newSDNode<GlobalAddressSDNode>(
1980 Opc,
DL.getIROrder(),
DL.getDebugLoc(), GV, VTs,
Offset, TargetFlags);
1981 CSEMap.InsertNode(
N, IP);
1995 auto *
N = newSDNode<DeactivationSymbolSDNode>(GV, VTs);
1996 CSEMap.InsertNode(
N, IP);
2008 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2011 auto *
N = newSDNode<FrameIndexSDNode>(FI, VTs, isTarget);
2012 CSEMap.InsertNode(
N, IP);
2018 unsigned TargetFlags) {
2019 assert((TargetFlags == 0 || isTarget) &&
2020 "Cannot set target flags on target-independent jump tables");
2026 ID.AddInteger(TargetFlags);
2028 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2031 auto *
N = newSDNode<JumpTableSDNode>(JTI, VTs, isTarget, TargetFlags);
2032 CSEMap.InsertNode(
N, IP);
2046 bool isTarget,
unsigned TargetFlags) {
2047 assert((TargetFlags == 0 || isTarget) &&
2048 "Cannot set target flags on target-independent globals");
2057 ID.AddInteger(Alignment->value());
2060 ID.AddInteger(TargetFlags);
2062 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2065 auto *
N = newSDNode<ConstantPoolSDNode>(isTarget,
C, VTs,
Offset, *Alignment,
2067 CSEMap.InsertNode(
N, IP);
2076 bool isTarget,
unsigned TargetFlags) {
2077 assert((TargetFlags == 0 || isTarget) &&
2078 "Cannot set target flags on target-independent globals");
2085 ID.AddInteger(Alignment->value());
2087 C->addSelectionDAGCSEId(
ID);
2088 ID.AddInteger(TargetFlags);
2090 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2093 auto *
N = newSDNode<ConstantPoolSDNode>(isTarget,
C, VTs,
Offset, *Alignment,
2095 CSEMap.InsertNode(
N, IP);
2105 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2108 auto *
N = newSDNode<BasicBlockSDNode>(
MBB);
2109 CSEMap.InsertNode(
N, IP);
2116 ValueTypeNodes.size())
2123 N = newSDNode<VTSDNode>(VT);
2129 SDNode *&
N = ExternalSymbols[Sym];
2131 N = newSDNode<ExternalSymbolSDNode>(
false, Sym, 0,
getVTList(VT));
2145 N = newSDNode<MCSymbolSDNode>(Sym,
getVTList(VT));
2151 unsigned TargetFlags) {
2153 TargetExternalSymbols[std::pair<std::string, unsigned>(Sym, TargetFlags)];
2155 N = newSDNode<ExternalSymbolSDNode>(
true, Sym, TargetFlags,
getVTList(VT));
2161 EVT VT,
unsigned TargetFlags) {
2167 if ((
unsigned)
Cond >= CondCodeNodes.size())
2168 CondCodeNodes.resize(
Cond+1);
2170 if (!CondCodeNodes[
Cond]) {
2171 auto *
N = newSDNode<CondCodeSDNode>(
Cond);
2172 CondCodeNodes[
Cond] =
N;
2181 "APInt size does not match type size!");
2199template <
typename Ty>
2201 EVT VT, Ty Quantity) {
2202 if (Quantity.isScalable())
2206 return DAG.
getConstant(Quantity.getKnownMinValue(),
DL, VT);
2232 const APInt &StepVal) {
2256 "Must have the same number of vector elements as mask elements!");
2258 "Invalid VECTOR_SHUFFLE");
2266 int NElts = Mask.size();
2268 [&](
int M) {
return M < (NElts * 2) && M >= -1; }) &&
2269 "Index out of range");
2277 for (
int i = 0; i != NElts; ++i)
2278 if (MaskVec[i] >= NElts) MaskVec[i] -= NElts;
2285 if (TLI->hasVectorBlend()) {
2294 for (
int i = 0; i < NElts; ++i) {
2295 if (MaskVec[i] <
Offset || MaskVec[i] >= (
Offset + NElts))
2299 if (UndefElements[MaskVec[i] -
Offset]) {
2305 if (!UndefElements[i])
2310 BlendSplat(N1BV, 0);
2312 BlendSplat(N2BV, NElts);
2317 bool AllLHS =
true, AllRHS =
true;
2319 for (
int i = 0; i != NElts; ++i) {
2320 if (MaskVec[i] >= NElts) {
2325 }
else if (MaskVec[i] >= 0) {
2329 if (AllLHS && AllRHS)
2331 if (AllLHS && !N2Undef)
2344 bool Identity =
true, AllSame =
true;
2345 for (
int i = 0; i != NElts; ++i) {
2346 if (MaskVec[i] >= 0 && MaskVec[i] != i) Identity =
false;
2347 if (MaskVec[i] != MaskVec[0]) AllSame =
false;
2349 if (Identity && NElts)
2382 if (AllSame && SameNumElts) {
2383 EVT BuildVT = BV->getValueType(0);
2400 for (
int i = 0; i != NElts; ++i)
2401 ID.AddInteger(MaskVec[i]);
2404 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP))
2410 int *MaskAlloc = OperandAllocator.Allocate<
int>(NElts);
2413 auto *
N = newSDNode<ShuffleVectorSDNode>(VTs, dl.
getIROrder(),
2415 createOperands(
N,
Ops);
2417 CSEMap.InsertNode(
N, IP);
2438 ID.AddInteger(Reg.id());
2440 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2443 auto *
N = newSDNode<RegisterSDNode>(Reg, VTs);
2444 N->SDNodeBits.IsDivergent = TLI->isSDNodeSourceOfDivergence(
N, FLI, UA);
2445 CSEMap.InsertNode(
N, IP);
2453 ID.AddPointer(RegMask);
2455 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2458 auto *
N = newSDNode<RegisterMaskSDNode>(RegMask);
2459 CSEMap.InsertNode(
N, IP);
2474 ID.AddPointer(Label);
2476 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2481 createOperands(
N,
Ops);
2483 CSEMap.InsertNode(
N, IP);
2489 int64_t
Offset,
bool isTarget,
2490 unsigned TargetFlags) {
2498 ID.AddInteger(TargetFlags);
2500 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2503 auto *
N = newSDNode<BlockAddressSDNode>(
Opc, VTs, BA,
Offset, TargetFlags);
2504 CSEMap.InsertNode(
N, IP);
2515 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2518 auto *
N = newSDNode<SrcValueSDNode>(V);
2519 CSEMap.InsertNode(
N, IP);
2530 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP))
2533 auto *
N = newSDNode<MDNodeSDNode>(MD);
2534 CSEMap.InsertNode(
N, IP);
2540 if (VT == V.getValueType())
2547 unsigned SrcAS,
unsigned DestAS) {
2552 ID.AddInteger(SrcAS);
2553 ID.AddInteger(DestAS);
2556 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP))
2560 VTs, SrcAS, DestAS);
2561 createOperands(
N,
Ops);
2563 CSEMap.InsertNode(
N, IP);
2582 EVT OpTy =
Op.getValueType();
2584 if (OpTy == ShTy || OpTy.
isVector())
return Op;
2593 EVT VT =
Node->getValueType(0);
2602 if (MA && *MA > TLI.getMinStackArgumentAlignment()) {
2640 Align RedAlign = UseABI ?
DL.getABITypeAlign(Ty) :
DL.getPrefTypeAlign(Ty);
2642 if (TLI->isTypeLegal(VT) || !VT.
isVector())
2650 if (RedAlign > StackAlign) {
2653 unsigned NumIntermediates;
2654 TLI->getVectorTypeBreakdown(*
getContext(), VT, IntermediateVT,
2655 NumIntermediates, RegisterVT);
2657 Align RedAlign2 = UseABI ?
DL.getABITypeAlign(Ty) :
DL.getPrefTypeAlign(Ty);
2658 if (RedAlign2 < RedAlign)
2659 RedAlign = RedAlign2;
2664 RedAlign = std::min(RedAlign, StackAlign);
2679 false,
nullptr, StackID);
2694 "Don't know how to choose the maximum size when creating a stack "
2703 Align Align = std::max(
DL.getPrefTypeAlign(Ty1),
DL.getPrefTypeAlign(Ty2));
2712 auto GetUndefBooleanConstant = [&]() {
2714 TLI->getBooleanContents(OpVT) ==
2751 return GetUndefBooleanConstant();
2756 return GetUndefBooleanConstant();
2765 const APInt &C2 = N2C->getAPIntValue();
2767 const APInt &C1 = N1C->getAPIntValue();
2777 if (N1CFP && N2CFP) {
2782 return GetUndefBooleanConstant();
2787 return GetUndefBooleanConstant();
2793 return GetUndefBooleanConstant();
2798 return GetUndefBooleanConstant();
2803 return GetUndefBooleanConstant();
2809 return GetUndefBooleanConstant();
2836 if (!TLI->isCondCodeLegal(SwappedCond, OpVT.
getSimpleVT()))
2838 return getSetCC(dl, VT, N2, N1, SwappedCond, {},
2840 }
else if ((N2CFP && N2CFP->getValueAPF().isNaN()) ||
2855 return GetUndefBooleanConstant();
2866 unsigned BitWidth =
Op.getScalarValueSizeInBits();
2875 unsigned Opc =
Op.getOpcode();
2884 return (NoFPClass & TestMask) == TestMask;
2891 return Op->getFlags().hasNoNaNs();
2917 unsigned Depth)
const {
2925 const APInt &DemandedElts,
2926 unsigned Depth)
const {
2933 unsigned Depth )
const {
2939 unsigned Depth)
const {
2944 const APInt &DemandedElts,
2945 unsigned Depth)
const {
2946 EVT VT =
Op.getValueType();
2953 for (
unsigned EltIdx = 0; EltIdx != NumElts; ++EltIdx) {
2954 if (!DemandedElts[EltIdx])
2958 KnownZeroElements.
setBit(EltIdx);
2960 return KnownZeroElements;
2970 unsigned Opcode = V.getOpcode();
2971 EVT VT = V.getValueType();
2974 "scalable demanded bits are ignored");
2986 UndefElts = V.getOperand(0).isUndef()
2995 APInt UndefLHS, UndefRHS;
3004 (DemandedElts & UndefLHS) == (DemandedElts & UndefRHS)) {
3005 UndefElts = UndefLHS | UndefRHS;
3019 return TLI->isSplatValueForTargetNode(V, DemandedElts, UndefElts, *
this,
3036 for (
unsigned i = 0; i != NumElts; ++i) {
3042 if (!DemandedElts[i])
3044 if (Scl && Scl !=
Op)
3055 for (
int i = 0; i != (int)NumElts; ++i) {
3061 if (!DemandedElts[i])
3063 if (M < (
int)NumElts)
3066 DemandedRHS.
setBit(M - NumElts);
3078 auto CheckSplatSrc = [&](
SDValue Src,
const APInt &SrcElts) {
3080 return (SrcElts.popcount() == 1) ||
3082 (SrcElts & SrcUndefs).
isZero());
3084 if (!DemandedLHS.
isZero())
3085 return CheckSplatSrc(V.getOperand(0), DemandedLHS);
3086 return CheckSplatSrc(V.getOperand(1), DemandedRHS);
3092 if (Src.getValueType().isScalableVector())
3094 uint64_t Idx = V.getConstantOperandVal(1);
3095 unsigned NumSrcElts = Src.getValueType().getVectorNumElements();
3097 APInt DemandedSrcElts = DemandedElts.
zext(NumSrcElts).
shl(Idx);
3099 UndefElts = UndefSrcElts.
extractBits(NumElts, Idx);
3110 if (Src.getValueType().isScalableVector())
3114 APInt DemandedSrcElts = DemandedElts.
zext(NumSrcElts);
3116 UndefElts = UndefSrcElts.
trunc(NumElts);
3123 EVT SrcVT = Src.getValueType();
3133 if ((
BitWidth % SrcBitWidth) == 0) {
3135 unsigned Scale =
BitWidth / SrcBitWidth;
3137 APInt ScaledDemandedElts =
3139 for (
unsigned I = 0;
I != Scale; ++
I) {
3143 SubDemandedElts &= ScaledDemandedElts;
3147 if (!SubUndefElts.
isZero())
3161 EVT VT = V.getValueType();
3171 (AllowUndefs || !UndefElts);
3177 EVT VT = V.getValueType();
3178 unsigned Opcode = V.getOpcode();
3199 SplatIdx = (UndefElts & DemandedElts).
countr_one();
3214 if (!SVN->isSplat())
3216 int Idx = SVN->getSplatIndex();
3217 int NumElts = V.getValueType().getVectorNumElements();
3218 SplatIdx = Idx % NumElts;
3219 return V.getOperand(Idx / NumElts);
3231 if (LegalTypes && !TLI->isTypeLegal(SVT)) {
3234 LegalSVT = TLI->getTypeToTransformTo(*
getContext(), LegalSVT);
3235 if (LegalSVT.
bitsLT(SVT))
3243std::optional<ConstantRange>
3245 unsigned Depth)
const {
3248 "Unknown shift node");
3250 unsigned BitWidth = V.getScalarValueSizeInBits();
3253 const APInt &ShAmt = Cst->getAPIntValue();
3255 return std::nullopt;
3260 const APInt *MinAmt =
nullptr, *MaxAmt =
nullptr;
3261 for (
unsigned i = 0, e = BV->getNumOperands(); i != e; ++i) {
3262 if (!DemandedElts[i])
3266 MinAmt = MaxAmt =
nullptr;
3269 const APInt &ShAmt = SA->getAPIntValue();
3271 return std::nullopt;
3272 if (!MinAmt || MinAmt->
ugt(ShAmt))
3274 if (!MaxAmt || MaxAmt->ult(ShAmt))
3277 assert(((!MinAmt && !MaxAmt) || (MinAmt && MaxAmt)) &&
3278 "Failed to find matching min/max shift amounts");
3279 if (MinAmt && MaxAmt)
3289 return std::nullopt;
3292std::optional<unsigned>
3294 unsigned Depth)
const {
3297 "Unknown shift node");
3298 if (std::optional<ConstantRange> AmtRange =
3300 if (
const APInt *ShAmt = AmtRange->getSingleElement())
3301 return ShAmt->getZExtValue();
3302 return std::nullopt;
3305std::optional<unsigned>
3311std::optional<unsigned>
3313 unsigned Depth)
const {
3316 "Unknown shift node");
3317 if (std::optional<ConstantRange> AmtRange =
3319 return AmtRange->getUnsignedMin().getZExtValue();
3320 return std::nullopt;
3323std::optional<unsigned>
3329std::optional<unsigned>
3331 unsigned Depth)
const {
3334 "Unknown shift node");
3335 if (std::optional<ConstantRange> AmtRange =
3337 return AmtRange->getUnsignedMax().getZExtValue();
3338 return std::nullopt;
3341std::optional<unsigned>
3359 unsigned Depth)
const {
3360 unsigned BitWidth =
Op.getScalarValueSizeInBits();
3364 if (
auto OptAPInt =
Op->bitcastToAPInt()) {
3374 assert((!
Op.getValueType().isScalableVector() || NumElts == 1) &&
3375 "DemandedElts for scalable vectors must be 1 to represent all lanes");
3376 assert((!
Op.getValueType().isFixedLengthVector() ||
3377 NumElts ==
Op.getValueType().getVectorNumElements()) &&
3378 "Unexpected vector size");
3383 unsigned Opcode =
Op.getOpcode();
3391 "Expected SPLAT_VECTOR implicit truncation");
3398 unsigned ScalarSize =
Op.getOperand(0).getScalarValueSizeInBits();
3400 "Expected SPLAT_VECTOR_PARTS scalars to cover element width");
3407 const APInt &Step =
Op.getConstantOperandAPInt(0);
3416 const APInt MinNumElts =
3422 .
umul_ov(MinNumElts, Overflow);
3426 const APInt MaxValue = (MaxNumElts - 1).
umul_ov(Step, Overflow);
3434 assert(!
Op.getValueType().isScalableVector());
3437 for (
unsigned i = 0, e =
Op.getNumOperands(); i != e; ++i) {
3438 if (!DemandedElts[i])
3447 "Expected BUILD_VECTOR implicit truncation");
3471 assert(!
Op.getValueType().isScalableVector());
3474 APInt DemandedLHS, DemandedRHS;
3478 DemandedLHS, DemandedRHS))
3483 if (!!DemandedLHS) {
3491 if (!!DemandedRHS) {
3500 const APInt &Multiplier =
Op.getConstantOperandAPInt(0);
3505 if (
Op.getValueType().isScalableVector())
3509 EVT SubVectorVT =
Op.getOperand(0).getValueType();
3511 unsigned NumSubVectors =
Op.getNumOperands();
3512 for (
unsigned i = 0; i != NumSubVectors; ++i) {
3514 DemandedElts.
extractBits(NumSubVectorElts, i * NumSubVectorElts);
3515 if (!!DemandedSub) {
3527 if (
Op.getValueType().isScalableVector())
3534 unsigned NumSubElts =
Sub.getValueType().getVectorNumElements();
3536 APInt DemandedSrcElts = DemandedElts;
3537 DemandedSrcElts.
clearBits(Idx, Idx + NumSubElts);
3540 if (!!DemandedSubElts) {
3545 if (!!DemandedSrcElts) {
3555 APInt DemandedSrcElts;
3556 if (Src.getValueType().isScalableVector())
3557 DemandedSrcElts =
APInt(1, 1);
3560 unsigned NumSrcElts = Src.getValueType().getVectorNumElements();
3561 DemandedSrcElts = DemandedElts.
zext(NumSrcElts).
shl(Idx);
3567 if (
Op.getValueType().isScalableVector())
3571 if (DemandedElts != 1)
3582 if (
Op.getValueType().isScalableVector())
3602 if ((
BitWidth % SubBitWidth) == 0) {
3609 unsigned SubScale =
BitWidth / SubBitWidth;
3610 APInt SubDemandedElts(NumElts * SubScale, 0);
3611 for (
unsigned i = 0; i != NumElts; ++i)
3612 if (DemandedElts[i])
3613 SubDemandedElts.
setBit(i * SubScale);
3615 for (
unsigned i = 0; i != SubScale; ++i) {
3618 unsigned Shifts = IsLE ? i : SubScale - 1 - i;
3619 Known.
insertBits(Known2, SubBitWidth * Shifts);
3624 if ((SubBitWidth %
BitWidth) == 0) {
3625 assert(
Op.getValueType().isVector() &&
"Expected bitcast to vector");
3630 unsigned SubScale = SubBitWidth /
BitWidth;
3631 APInt SubDemandedElts =
3636 for (
unsigned i = 0; i != NumElts; ++i)
3637 if (DemandedElts[i]) {
3638 unsigned Shifts = IsLE ? i : NumElts - 1 - i;
3669 bool SelfMultiply =
Op.getOperand(0) ==
Op.getOperand(1);
3680 if (
Op->getFlags().hasNoSignedWrap() &&
3681 Op.getOperand(0) ==
Op.getOperand(1) &&
3708 unsigned SignBits1 =
3712 unsigned SignBits0 =
3718 assert((
Op.getResNo() == 0 ||
Op.getResNo() == 1) &&
"Unknown result");
3721 bool SelfMultiply =
Op.getOperand(0) ==
Op.getOperand(1);
3722 if (
Op.getResNo() == 0)
3729 assert((
Op.getResNo() == 0 ||
Op.getResNo() == 1) &&
"Unknown result");
3732 bool SelfMultiply =
Op.getOperand(0) ==
Op.getOperand(1);
3733 if (
Op.getResNo() == 0)
3786 if (
Op.getResNo() != 1)
3792 if (TLI->getBooleanContents(
Op.getValueType().isVector(),
false) ==
3801 unsigned OpNo =
Op->isStrictFPOpcode() ? 1 : 0;
3803 if (TLI->getBooleanContents(
Op.getOperand(OpNo).getValueType()) ==
3813 bool NUW =
Op->getFlags().hasNoUnsignedWrap();
3814 bool NSW =
Op->getFlags().hasNoSignedWrap();
3821 if (std::optional<unsigned> ShMinAmt =
3830 Op->getFlags().hasExact());
3833 if (std::optional<unsigned> ShMinAmt =
3841 Op->getFlags().hasExact());
3847 unsigned Amt =
C->getAPIntValue().urem(
BitWidth);
3862 unsigned Amt =
C->getAPIntValue().urem(
BitWidth);
3868 DemandedElts,
Depth + 1);
3884 assert((
Op.getResNo() == 0 ||
Op.getResNo() == 1) &&
"Unknown result");
3887 unsigned LoBits =
Op.getOperand(0).getScalarValueSizeInBits();
3888 unsigned HiBits =
Op.getOperand(1).getScalarValueSizeInBits();
3891 Known = Known2.
concat(Known);
3905 if (
Op.getResNo() == 0)
3936 unsigned MinRedundantSignBits =
3940 Known =
Range.toKnownBits();
3988 const Constant *Cst = TLI->getTargetConstantFromLoad(LD);
3993 !
Op.getValueType().isScalableVector()) {
4006 for (
unsigned i = 0; i != NumElts; ++i) {
4007 if (!DemandedElts[i])
4017 APInt Value = CFP->getValueAPF().bitcastToAPInt();
4036 }
else if (
Op.getResNo() == 0) {
4037 unsigned ScalarMemorySize = LD->getMemoryVT().getScalarSizeInBits();
4038 KnownBits KnownScalarMemory(ScalarMemorySize);
4039 if (
const MDNode *MD = LD->getRanges())
4050 Known = KnownScalarMemory;
4057 if (
Op.getValueType().isScalableVector())
4059 EVT InVT =
Op.getOperand(0).getValueType();
4071 if (
Op.getValueType().isScalableVector())
4073 EVT InVT =
Op.getOperand(0).getValueType();
4089 if (
Op.getValueType().isScalableVector())
4091 EVT InVT =
Op.getOperand(0).getValueType();
4126 Known.
Zero |= (~InMask);
4127 Known.
One &= (~Known.Zero);
4147 if ((NoFPClass & NegativeTestMask) == NegativeTestMask) {
4153 if ((NoFPClass & PositiveTestMask) == PositiveTestMask) {
4173 bool SelfAdd =
Op.getOperand(0) ==
Op.getOperand(1) &&
4175 Op.getOperand(0), DemandedElts,
4178 Flags.hasNoUnsignedWrap(), SelfAdd);
4186 Flags.hasNoUnsignedWrap());
4193 if (
Op.getResNo() == 1) {
4195 if (TLI->getBooleanContents(
Op.getOperand(0).getValueType()) ==
4204 "We only compute knownbits for the difference here.");
4211 Borrow = Borrow.
trunc(1);
4225 if (
Op.getResNo() == 1) {
4227 if (TLI->getBooleanContents(
Op.getOperand(0).getValueType()) ==
4236 assert(
Op.getResNo() == 0 &&
"We only compute knownbits for the sum here.");
4246 Carry = Carry.
trunc(1);
4282 const unsigned Index =
Op.getConstantOperandVal(1);
4283 const unsigned EltBitWidth =
Op.getValueSizeInBits();
4290 Known = Known.
trunc(EltBitWidth);
4306 Known = Known.
trunc(EltBitWidth);
4312 if (ConstEltNo && ConstEltNo->getAPIntValue().ult(NumSrcElts))
4322 if (
Op.getValueType().isScalableVector())
4331 bool DemandedVal =
true;
4332 APInt DemandedVecElts = DemandedElts;
4334 if (CEltNo && CEltNo->getAPIntValue().ult(NumElts)) {
4335 unsigned EltIdx = CEltNo->getZExtValue();
4336 DemandedVal = !!DemandedElts[EltIdx];
4344 if (!!DemandedVecElts) {
4363 Known = Known2.
abs();
4396 if (CstLow && CstHigh) {
4401 const APInt &ValueHigh = CstHigh->getAPIntValue();
4402 if (ValueLow.
sle(ValueHigh)) {
4405 unsigned MinSignBits = std::min(LowSignBits, HighSignBits);
4428 if (IsMax && CstLow) {
4458 if (
Op.getResNo() == 0) {
4460 unsigned ScalarMemorySize = AT->getMemoryVT().getScalarSizeInBits();
4461 KnownBits KnownScalarMemory(ScalarMemorySize);
4462 if (
const MDNode *MD = AT->getRanges())
4465 switch (AT->getExtensionType()) {
4473 switch (TLI->getExtendForAtomicOps()) {
4486 Known = KnownScalarMemory;
4494 if (
Op.getResNo() == 1) {
4499 if (TLI->getBooleanContents(
Op.getValueType().isVector(),
false) ==
4520 if (
Op.getResNo() == 0) {
4522 unsigned MemBits = AT->getMemoryVT().getScalarSizeInBits();
4543 TLI->computeKnownBitsForTargetNode(
Op, Known, DemandedElts, *
this,
Depth);
4675 unsigned Depth)
const {
4681 const APInt &DemandedElts,
4683 unsigned Depth)
const {
4684 EVT VT =
Op.getValueType();
4688 return ConstantRange::getFull(
BitWidth);
4693 unsigned Opcode =
Op.getOpcode();
4697 const APInt &Multiplier =
Op.getConstantOperandAPInt(0);
4704 return ConstantRange::getFull(
BitWidth);
4709 unsigned Depth)
const {
4717 unsigned Depth)
const {
4727 unsigned Depth)
const {
4733 const APInt &DemandedElts,
4734 bool OrZero,
unsigned Depth)
const {
4740 [[maybe_unused]]
unsigned NumElts = DemandedElts.
getBitWidth();
4742 "DemandedElts for scalable vectors must be 1 to represent all lanes");
4745 "Unexpected vector size");
4749 return (OrZero && V.isZero()) || V.isPowerOf2();
4760 auto *C = dyn_cast<ConstantSDNode>(P.value());
4761 return !DemandedElts[P.index()] || (C && IsPowerOfTwoOrZero(C));
4769 if (IsPowerOfTwoOrZero(
C))
4787 APInt DemandedSrcElts =
4788 ConstEltNo && ConstEltNo->getAPIntValue().
ult(NumSrcElts)
4813 if (
C &&
C->getAPIntValue() == 1)
4824 if (
C &&
C->getAPIntValue().isSignMask())
4874 APInt DemandedLHS, DemandedRHS;
4878 DemandedLHS, DemandedRHS))
4902 return C1->getValueAPF().getExactLog2Abs() >= 0;
4916 unsigned Depth)
const {
4917 EVT VT =
Op.getValueType();
4922 unsigned FirstAnswer = 1;
4925 "DemandedElts for scalable vectors must be 1 to represent all lanes");
4928 const APInt &Val =
C->getAPIntValue();
4938 unsigned Opcode =
Op.getOpcode();
4943 return VTBits-Tmp+1;
4957 unsigned NumSrcBits =
Op.getOperand(0).getValueSizeInBits();
4959 if (NumSrcSignBits > (NumSrcBits - VTBits))
4960 return NumSrcSignBits - (NumSrcBits - VTBits);
4966 for (
unsigned i = 0, e =
Op.getNumOperands(); (i < e) && (Tmp > 1); ++i) {
4967 if (!DemandedElts[i])
4974 APInt T =
C->getAPIntValue().trunc(VTBits);
4975 Tmp2 =
T.getNumSignBits();
4979 if (
SrcOp.getValueSizeInBits() != VTBits) {
4981 "Expected BUILD_VECTOR implicit truncation");
4982 unsigned ExtraBits =
SrcOp.getValueSizeInBits() - VTBits;
4983 Tmp2 = (Tmp2 > ExtraBits ? Tmp2 - ExtraBits : 1);
4986 Tmp = std::min(Tmp, Tmp2);
4997 Tmp = std::min(Tmp, Tmp2);
5004 APInt DemandedLHS, DemandedRHS;
5008 DemandedLHS, DemandedRHS))
5011 Tmp = std::numeric_limits<unsigned>::max();
5014 if (!!DemandedRHS) {
5016 Tmp = std::min(Tmp, Tmp2);
5021 assert(Tmp <= VTBits &&
"Failed to determine minimum sign bits");
5037 if (VTBits == SrcBits)
5043 if ((SrcBits % VTBits) == 0) {
5046 unsigned Scale = SrcBits / VTBits;
5047 APInt SrcDemandedElts =
5057 for (
unsigned i = 0; i != NumElts; ++i)
5058 if (DemandedElts[i]) {
5059 unsigned SubOffset = i % Scale;
5060 SubOffset = (IsLE ? ((Scale - 1) - SubOffset) : SubOffset);
5061 SubOffset = SubOffset * VTBits;
5062 if (Tmp <= SubOffset)
5064 Tmp2 = std::min(Tmp2, Tmp - SubOffset);
5074 return VTBits - Tmp + 1;
5076 Tmp = VTBits -
Op.getOperand(0).getScalarValueSizeInBits();
5083 return std::max(Tmp, Tmp2);
5088 EVT SrcVT = Src.getValueType();
5096 if (std::optional<unsigned> ShAmt =
5098 Tmp = std::min(Tmp + *ShAmt, VTBits);
5101 if (std::optional<ConstantRange> ShAmtRange =
5103 unsigned MaxShAmt = ShAmtRange->getUnsignedMax().getZExtValue();
5104 unsigned MinShAmt = ShAmtRange->getUnsignedMin().getZExtValue();
5115 unsigned SizeDifference =
5117 if (SizeDifference <= MinShAmt) {
5118 Tmp = SizeDifference +
5121 return Tmp - MaxShAmt;
5127 return Tmp - MaxShAmt;
5137 FirstAnswer = std::min(Tmp, Tmp2);
5147 if (Tmp == 1)
return 1;
5149 return std::min(Tmp, Tmp2);
5152 if (Tmp == 1)
return 1;
5154 return std::min(Tmp, Tmp2);
5166 if (CstLow && CstHigh) {
5171 Tmp2 = CstHigh->getAPIntValue().getNumSignBits();
5172 return std::min(Tmp, Tmp2);
5181 return std::min(Tmp, Tmp2);
5189 return std::min(Tmp, Tmp2);
5193 if (
Op.getResNo() == 0 &&
Op.getOperand(0) ==
Op.getOperand(1))
5204 if (
Op.getResNo() != 1)
5210 if (TLI->getBooleanContents(VT.
isVector(),
false) ==
5218 unsigned OpNo =
Op->isStrictFPOpcode() ? 1 : 0;
5220 if (TLI->getBooleanContents(
Op.getOperand(OpNo).getValueType()) ==
5235 unsigned RotAmt =
C->getAPIntValue().urem(VTBits);
5239 RotAmt = (VTBits - RotAmt) % VTBits;
5243 if (Tmp > (RotAmt + 1))
return (Tmp - RotAmt);
5250 if (Tmp == 1)
return 1;
5255 if (CRHS->isAllOnes()) {
5261 if ((Known.
Zero | 1).isAllOnes())
5271 if (Tmp2 == 1)
return 1;
5275 return std::min(Tmp, Tmp2) - 1;
5278 if (Tmp2 == 1)
return 1;
5283 if (CLHS->isZero()) {
5288 if ((Known.
Zero | 1).isAllOnes())
5302 if (Tmp == 1)
return 1;
5303 return std::min(Tmp, Tmp2) - 1;
5307 if (SignBitsOp0 == 1)
5310 if (SignBitsOp1 == 1)
5312 unsigned OutValidBits =
5313 (VTBits - SignBitsOp0 + 1) + (VTBits - SignBitsOp1 + 1);
5314 return OutValidBits > VTBits ? 1 : VTBits - OutValidBits + 1;
5322 return std::min(Tmp, Tmp2);
5331 unsigned NumSrcBits =
Op.getOperand(0).getScalarValueSizeInBits();
5333 if (NumSrcSignBits > (NumSrcBits - VTBits))
5334 return NumSrcSignBits - (NumSrcBits - VTBits);
5341 const int BitWidth =
Op.getValueSizeInBits();
5342 const int Items =
Op.getOperand(0).getValueSizeInBits() /
BitWidth;
5346 const int rIndex = Items - 1 -
Op.getConstantOperandVal(1);
5361 bool DemandedVal =
true;
5362 APInt DemandedVecElts = DemandedElts;
5364 if (CEltNo && CEltNo->getAPIntValue().ult(NumElts)) {
5365 unsigned EltIdx = CEltNo->getZExtValue();
5366 DemandedVal = !!DemandedElts[EltIdx];
5369 Tmp = std::numeric_limits<unsigned>::max();
5375 Tmp = std::min(Tmp, Tmp2);
5377 if (!!DemandedVecElts) {
5379 Tmp = std::min(Tmp, Tmp2);
5381 assert(Tmp <= VTBits &&
"Failed to determine minimum sign bits");
5391 const unsigned BitWidth =
Op.getValueSizeInBits();
5392 const unsigned EltBitWidth =
Op.getOperand(0).getScalarValueSizeInBits();
5405 if (ConstEltNo && ConstEltNo->getAPIntValue().ult(NumSrcElts))
5415 APInt DemandedSrcElts;
5416 if (Src.getValueType().isScalableVector())
5417 DemandedSrcElts =
APInt(1, 1);
5420 unsigned NumSrcElts = Src.getValueType().getVectorNumElements();
5421 DemandedSrcElts = DemandedElts.
zext(NumSrcElts).
shl(Idx);
5430 Tmp = std::numeric_limits<unsigned>::max();
5431 EVT SubVectorVT =
Op.getOperand(0).getValueType();
5433 unsigned NumSubVectors =
Op.getNumOperands();
5434 for (
unsigned i = 0; (i < NumSubVectors) && (Tmp > 1); ++i) {
5436 DemandedElts.
extractBits(NumSubVectorElts, i * NumSubVectorElts);
5440 Tmp = std::min(Tmp, Tmp2);
5442 assert(Tmp <= VTBits &&
"Failed to determine minimum sign bits");
5453 unsigned NumSubElts =
Sub.getValueType().getVectorNumElements();
5455 APInt DemandedSrcElts = DemandedElts;
5456 DemandedSrcElts.
clearBits(Idx, Idx + NumSubElts);
5458 Tmp = std::numeric_limits<unsigned>::max();
5459 if (!!DemandedSubElts) {
5464 if (!!DemandedSrcElts) {
5466 Tmp = std::min(Tmp, Tmp2);
5468 assert(Tmp <= VTBits &&
"Failed to determine minimum sign bits");
5473 if (
Op.getResNo() != 0)
5477 if (
const MDNode *Ranges = LD->getRanges()) {
5478 if (DemandedElts != 1)
5483 switch (LD->getExtensionType()) {
5501 unsigned ExtType = LD->getExtensionType();
5506 Tmp = LD->getMemoryVT().getScalarSizeInBits();
5507 return VTBits - Tmp + 1;
5509 Tmp = LD->getMemoryVT().getScalarSizeInBits();
5510 return VTBits - Tmp;
5512 if (
const Constant *Cst = TLI->getTargetConstantFromLoad(LD)) {
5515 Type *CstTy = Cst->getType();
5520 for (
unsigned i = 0; i != NumElts; ++i) {
5521 if (!DemandedElts[i])
5526 Tmp = std::min(Tmp,
Value.getNumSignBits());
5530 APInt Value = CFP->getValueAPF().bitcastToAPInt();
5531 Tmp = std::min(Tmp,
Value.getNumSignBits());
5563 if (
Op.getResNo() == 0) {
5564 Tmp = AT->getMemoryVT().getScalarSizeInBits();
5570 switch (AT->getExtensionType()) {
5574 return VTBits - Tmp + 1;
5576 return VTBits - Tmp;
5581 return VTBits - Tmp + 1;
5583 return VTBits - Tmp;
5598 TLI->ComputeNumSignBitsForTargetNode(
Op, DemandedElts, *
this,
Depth);
5600 FirstAnswer = std::max(FirstAnswer, NumBits);
5611 unsigned Depth)
const {
5613 return Op.getScalarValueSizeInBits() - SignBits + 1;
5617 const APInt &DemandedElts,
5618 unsigned Depth)
const {
5620 return Op.getScalarValueSizeInBits() - SignBits + 1;
5625 unsigned Depth)
const {
5635 const APInt &DemandedElts,
5637 unsigned Depth)
const {
5638 unsigned Opcode =
Op.getOpcode();
5666 EVT SrcVT = Src.getValueType();
5667 EVT DstVT =
Op.getValueType();
5677 if (SrcEltBits == DstEltBits)
5681 if (SrcEltBits < DstEltBits) {
5682 if (DstEltBits % SrcEltBits != 0)
5685 assert(NumSrcElts == NumDstElts * (DstEltBits / SrcEltBits) &&
5686 "Unexpected vector bitcast");
5687 APInt DemandedSrcElts =
5693 if (SrcEltBits % DstEltBits != 0)
5696 assert(NumDstElts == NumSrcElts * (SrcEltBits / DstEltBits) &&
5697 "Unexpected vector bitcast");
5698 APInt DemandedSrcElts =
5707 for (
unsigned i = 0, e =
Op.getNumOperands(); i < e; ++i) {
5708 if (!DemandedElts[i])
5716 EVT VT =
Op.getValueType();
5720 EVT SubVT =
Op.getOperand(0).getValueType();
5722 for (
unsigned I = 0, E =
Op.getNumOperands();
I != E; ++
I) {
5723 APInt DemandedSubElts =
5725 if (!!DemandedSubElts &&
5735 if (Src.getValueType().isScalableVector())
5738 unsigned NumSrcElts = Src.getValueType().getVectorNumElements();
5739 APInt DemandedSrcElts = DemandedElts.
zext(NumSrcElts).
shl(Idx);
5745 if (
Op.getValueType().isScalableVector())
5750 unsigned NumSubElts =
Sub.getValueType().getVectorNumElements();
5752 APInt DemandedSrcElts = DemandedElts;
5753 DemandedSrcElts.
clearBits(Idx, Idx + NumSubElts);
5756 Sub, DemandedSubElts, Kind,
Depth + 1))
5759 Src, DemandedSrcElts, Kind,
Depth + 1))
5767 EVT SrcVT = Src.getValueType();
5771 IndexC->getZExtValue());
5786 if (DemandedElts[IndexC->getZExtValue()] &&
5789 APInt InVecDemandedElts = DemandedElts;
5790 InVecDemandedElts.
clearBit(IndexC->getZExtValue());
5791 if (!!InVecDemandedElts &&
5794 InVecDemandedElts, Kind,
Depth + 1))
5806 if (DemandedElts[0] &&
5826 APInt DemandedLHS, DemandedRHS;
5829 DemandedElts, DemandedLHS, DemandedRHS,
5832 if (!DemandedLHS.
isZero() &&
5836 if (!DemandedRHS.
isZero() &&
5884 return isGuaranteedNotToBeUndefOrPoison(V, DemandedElts, Kind,
5897 return TLI->isGuaranteedNotToBeUndefOrPoisonForTargetNode(
5898 Op, DemandedElts, *
this, Kind,
Depth);
5909 return isGuaranteedNotToBeUndefOrPoison(V, Kind, Depth + 1);
5915 unsigned Depth)
const {
5923 unsigned Depth)
const {
5924 if (ConsiderFlags &&
includesPoison(Kind) &&
Op->hasPoisonGeneratingFlags())
5927 unsigned Opcode =
Op.getOpcode();
6018 if (
Op.getOperand(0).getValueType().isInteger())
6025 unsigned CCOp = Opcode ==
ISD::SETCC ? 2 : 4;
6027 return (
unsigned)CCCode & 0x10U;
6087 EVT VecVT =
Op.getOperand(0).getValueType();
6098 for (
auto [Idx, Elt] :
enumerate(SVN->getMask()))
6099 if (Elt < 0 && DemandedElts[Idx])
6111 return TLI->canCreateUndefOrPoisonForTargetNode(
6112 Op, DemandedElts, *
this, Kind, ConsiderFlags,
Depth);
6121 unsigned Opcode =
Op.getOpcode();
6123 return Op->getFlags().hasDisjoint() ||
6137 unsigned Depth)
const {
6143 const APInt &DemandedElts,
6145 unsigned Depth)
const {
6157 EVT VT =
Op.getValueType();
6161 "Unexpected vector size");
6166 unsigned Opcode =
Op.getOpcode();
6175 InterestedClasses,
Depth + 1);
6182 for (
unsigned I = 0, E =
Op.getNumOperands();
I != E; ++
I) {
6183 if (!DemandedElts[
I])
6203 EVT SrcVT = Src.getValueType();
6229 EVT SrcVT =
Op.getOperand(0).getValueType();
6234 if (VTNumElts != SrcVTNumElts)
6243 InterestedClasses,
Depth + 1);
6249 InterestedClasses,
Depth + 1);
6251 InterestedClasses,
Depth + 1);
6257 InterestedClasses,
Depth + 1);
6265 EVT SrcVT = Src.getValueType();
6267 unsigned Idx =
Op.getConstantOperandVal(1);
6283 unsigned Idx =
Op.getConstantOperandVal(2);
6287 APInt DemandedMask =
6289 APInt DemandedSrcElts = DemandedElts & ~DemandedMask;
6292 if (!DemandedSrcElts.
isZero())
6294 InterestedClasses,
Depth + 1);
6295 if (!DemandedSubElts.
isZero()) {
6297 SubVector, DemandedSubElts, InterestedClasses,
Depth + 1);
6298 Known = DemandedSrcElts.
isZero() ? SubKnown : (Known | SubKnown);
6312 Op.getOperand(2), DemandedElts, InterestedClasses,
Depth + 1);
6316 Op.getOperand(1), DemandedElts, InterestedClasses,
Depth + 1);
6323 TLI->computeKnownFPClassForTargetNode(
Op, Known, DemandedElts, *
this,
6333 unsigned Depth)
const {
6339 bool SNaN,
unsigned Depth)
const {
6340 assert(!DemandedElts.
isZero() &&
"No demanded elements");
6343 if (
Op->getFlags().hasNoNaNs())
6349 unsigned Opcode =
Op.getOpcode();
6451 EVT SrcVT = Src.getValueType();
6455 Idx->getZExtValue());
6462 if (Src.getValueType().isFixedLengthVector()) {
6463 unsigned Idx =
Op.getConstantOperandVal(1);
6464 unsigned NumSrcElts = Src.getValueType().getVectorNumElements();
6465 APInt DemandedSrcElts = DemandedElts.
zext(NumSrcElts).
shl(Idx);
6475 unsigned Idx =
Op.getConstantOperandVal(2);
6481 APInt DemandedMask =
6483 APInt DemandedSrcElts = DemandedElts & ~DemandedMask;
6486 bool NeverNaN =
true;
6487 if (!DemandedSrcElts.
isZero())
6490 if (NeverNaN && !DemandedSubElts.
isZero())
6499 unsigned NumElts =
Op.getNumOperands();
6500 for (
unsigned I = 0;
I != NumElts; ++
I)
6501 if (DemandedElts[
I] &&
6520 return TLI->isKnownNeverNaNForTargetNode(
Op, DemandedElts, *
this, SNaN,
6537 const APInt &DemandedElts,
6538 unsigned Depth)
const {
6539 assert(!DemandedElts.
isZero() &&
"No demanded elements");
6540 EVT VT =
Op.getValueType();
6552 unsigned Depth)
const {
6556 EVT OpVT =
Op.getValueType();
6559 assert(!
Op.getValueType().isFloatingPoint() &&
6560 "Floating point types unsupported - use isKnownNeverLogicalZero");
6573 switch (
Op.getOpcode()) {
6580 auto *C = dyn_cast<ConstantSDNode>(P.value());
6581 return !DemandedElts[P.index()] || (C && IsNeverZero(C));
6608 if (ConstEltNo && ConstEltNo->getAPIntValue().ult(NumSrcElts))
6625 if (
Op->getFlags().hasNoSignedWrap() ||
Op->getFlags().hasNoUnsignedWrap())
6630 if (ValKnown.
One[0])
6642 if (
Op.getValueType().isScalableVector())
6650 APInt DemandedLHS, DemandedRHS;
6652 assert(NumElts == SVN->getMask().size() &&
"Unexpected vector size");
6654 DemandedLHS, DemandedRHS))
6657 return (!DemandedLHS ||
6716 if (
Op->getFlags().hasExact())
6734 if (
Op->getFlags().hasExact())
6739 if (
Op->getFlags().hasNoUnsignedWrap())
6757 if (
Op->getFlags().hasNoSignedWrap() ||
Op->getFlags().hasNoUnsignedWrap())
6768 const APInt &Multiplier =
Op.getConstantOperandAPInt(0);
6782 return !C1->isNegative();
6784 switch (
Op.getOpcode()) {
6798 assert(
Use.getValueType().isFloatingPoint());
6800 if (
User->getFlags().hasNoSignedZeros())
6805 switch (
User->getOpcode()) {
6813 return OperandNo == 0;
6831 if (
Op->getFlags().hasNoSignedZeros())
6836 if (
Op->use_size() > 2)
6839 [&](
const SDUse &
Use) { return canIgnoreSignBitOfZero(Use); });
6844 if (
A ==
B)
return true;
6849 if (CA->isZero() && CB->isZero())
return true;
6884 NotOperand = NotOperand->getOperand(0);
6886 if (
Other == NotOperand)
6889 return NotOperand ==
Other->getOperand(0) ||
6890 NotOperand ==
Other->getOperand(1);
6896 A =
A->getOperand(0);
6899 B =
B->getOperand(0);
6902 return MatchNoCommonBitsPattern(
A->getOperand(0),
A->getOperand(1),
B) ||
6903 MatchNoCommonBitsPattern(
A->getOperand(1),
A->getOperand(0),
B);
6909 assert(
A.getValueType() ==
B.getValueType() &&
6910 "Values must have the same type");
6932 "BUILD_VECTOR cannot be used with scalable types");
6934 "Incorrect element count in BUILD_VECTOR!");
6942 bool IsIdentity =
true;
6943 for (
int i = 0; i !=
NumOps; ++i) {
6946 (IdentitySrc &&
Ops[i].getOperand(0) != IdentitySrc) ||
6948 Ops[i].getConstantOperandAPInt(1) != i) {
6952 IdentitySrc =
Ops[i].getOperand(0);
6965 assert(!
Ops.empty() &&
"Can't concatenate an empty list of vectors!");
6968 return Ops[0].getValueType() ==
Op.getValueType();
6970 "Concatenation of vectors with inconsistent value types!");
6973 "Incorrect element count in vector concatenation!");
6975 if (
Ops.size() == 1)
6986 bool IsIdentity =
true;
6987 for (
unsigned i = 0, e =
Ops.size(); i != e; ++i) {
6989 unsigned IdentityIndex = i *
Op.getValueType().getVectorMinNumElements();
6991 Op.getOperand(0).getValueType() != VT ||
6992 (IdentitySrc &&
Op.getOperand(0) != IdentitySrc) ||
6993 Op.getConstantOperandVal(1) != IdentityIndex) {
6997 assert((!IdentitySrc || IdentitySrc ==
Op.getOperand(0)) &&
6998 "Unexpected identity source vector for concat of extracts");
6999 IdentitySrc =
Op.getOperand(0);
7002 assert(IdentitySrc &&
"Failed to set source vector of extracts");
7018 EVT OpVT =
Op.getValueType();
7034 SVT = (SVT.
bitsLT(
Op.getValueType()) ?
Op.getValueType() : SVT);
7058 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP))
7061 auto *
N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
7062 CSEMap.InsertNode(
N, IP);
7074 Flags = Inserter->getFlags();
7075 return getNode(Opcode,
DL, VT, N1, Flags);
7128 "STEP_VECTOR can only be used with scalable types");
7131 "Unexpected step operand");
7152 "Invalid FP cast!");
7156 "Vector element count mismatch!");
7174 "Invalid SIGN_EXTEND!");
7176 "SIGN_EXTEND result type type should be vector iff the operand "
7181 "Vector element count mismatch!");
7204 unsigned NumSignExtBits =
7215 "Invalid ZERO_EXTEND!");
7217 "ZERO_EXTEND result type type should be vector iff the operand "
7222 "Vector element count mismatch!");
7260 "Invalid ANY_EXTEND!");
7262 "ANY_EXTEND result type type should be vector iff the operand "
7267 "Vector element count mismatch!");
7292 "Invalid TRUNCATE!");
7294 "TRUNCATE result type type should be vector iff the operand "
7299 "Vector element count mismatch!");
7326 assert(VT.
isVector() &&
"This DAG node is restricted to vector types.");
7328 "The input must be the same size or smaller than the result.");
7331 "The destination vector type must have fewer lanes than the input.");
7340 "Invalid ABS_MIN_POISON!");
7347 "BSWAP types must be a multiple of 16 bits!");
7361 "Cannot BITCAST between types of different sizes!");
7374 "Illegal SCALAR_TO_VECTOR node!");
7435 "Wrong operand type!");
7442 if (VT != MVT::Glue) {
7446 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
7447 E->intersectFlagsWith(Flags);
7451 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
7453 createOperands(
N,
Ops);
7454 CSEMap.InsertNode(
N, IP);
7456 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
7457 createOperands(
N,
Ops);
7491 if (!C2.getBoolValue())
7495 if (!C2.getBoolValue())
7499 if (!C2.getBoolValue())
7503 if (!C2.getBoolValue())
7533 return std::nullopt;
7538 bool IsUndef1,
const APInt &C2,
7540 if (!(IsUndef1 || IsUndef2))
7548 return std::nullopt;
7556 if (!TLI->isOffsetFoldingLegal(GA))
7561 int64_t
Offset = C2->getSExtValue();
7581 assert(
Ops.size() == 2 &&
"Div/rem should have 2 operands");
7588 [](
SDValue V) { return V.isUndef() ||
7589 isNullConstant(V); });
7627 const APInt &Val =
C->getAPIntValue();
7631 C->isTargetOpcode(),
C->isOpaque());
7638 C->isTargetOpcode(),
C->isOpaque());
7643 C->isTargetOpcode(),
C->isOpaque());
7645 C->isTargetOpcode(),
C->isOpaque());
7674 C->isTargetOpcode(),
C->isOpaque());
7700 if (VT == MVT::f16 &&
C->getValueType(0) == MVT::i16)
7702 if (VT == MVT::f32 &&
C->getValueType(0) == MVT::i32)
7704 if (VT == MVT::f64 &&
C->getValueType(0) == MVT::i64)
7706 if (VT == MVT::f128 &&
C->getValueType(0) == MVT::i128)
7767 return getConstant(V.bitcastToAPInt().getZExtValue(),
DL, VT);
7770 if (VT == MVT::i16 &&
C->getValueType(0) == MVT::f16)
7773 if (VT == MVT::i16 &&
C->getValueType(0) == MVT::bf16)
7776 if (VT == MVT::i32 &&
C->getValueType(0) == MVT::f32)
7779 if (VT == MVT::i64 &&
C->getValueType(0) == MVT::f64)
7780 return getConstant(V.bitcastToAPInt().getZExtValue(),
DL, VT);
7797 if (C1->isOpaque() || C2->isOpaque())
7800 std::optional<APInt> FoldAttempt =
7801 FoldValue(Opcode, C1->getAPIntValue(), C2->getAPIntValue());
7807 "Can't fold vectors ops with scalar operands");
7815 if (TLI->isCommutativeBinOp(Opcode))
7831 const APInt &Val = C1->getAPIntValue();
7832 return SignExtendInReg(Val, VT);
7845 ScalarOps.
push_back(SignExtendInReg(Val, OpVT));
7853 SignExtendInReg(
Ops[0].getConstantOperandAPInt(0),
7864 if (C1 && C2 && C3) {
7865 if (C1->isOpaque() || C2->isOpaque() || C3->isOpaque())
7867 const APInt &
V1 = C1->getAPIntValue(), &V2 = C2->getAPIntValue(),
7868 &
V3 = C3->getAPIntValue();
7884 if (C1 && C2 && C3) {
7905 Ops[0].getValueType() == VT &&
Ops[1].getValueType() == VT &&
7918 if (BV1->getConstantRawBits(IsLE, EltBits, RawBits1, UndefElts1) &&
7919 BV2->getConstantRawBits(IsLE, EltBits, RawBits2, UndefElts2)) {
7923 Opcode, RawBits1[
I], UndefElts1[
I], RawBits2[
I], UndefElts2[
I]);
7934 BVEltVT = BV1->getOperand(0).getValueType();
7937 BVEltVT = BV2->getOperand(0).getValueType();
7943 DstBits, RawBits, DstUndefs,
7946 for (
unsigned I = 0, E = DstBits.
size();
I != E; ++
I) {
7971 ?
Ops[0].getConstantOperandAPInt(0) * RHSVal
7972 :
Ops[0].getConstantOperandAPInt(0) << RHSVal;
7977 auto IsScalarOrSameVectorSize = [NumElts](
const SDValue &
Op) {
7978 return !
Op.getValueType().isVector() ||
7979 Op.getValueType().getVectorElementCount() == NumElts;
7982 auto IsBuildVectorSplatVectorOrUndef = [](
const SDValue &
Op) {
8008 LegalSVT = TLI->getTypeToTransformTo(*
getContext(), LegalSVT);
8020 for (
unsigned I = 0;
I != NumVectorElts;
I++) {
8023 EVT InSVT =
Op.getValueType().getScalarType();
8066 if (LegalSVT != SVT)
8067 ScalarResult =
getNode(ExtendCode,
DL, LegalSVT, ScalarResult);
8081 if (
Ops.size() != 2)
8092 if (N1CFP && N2CFP) {
8143 if (N1C && N1C->getValueAPF().isNegZero() && N2.
isUndef())
8166 if (SrcEltVT == DstEltVT)
8174 if (SrcBitSize == DstBitSize) {
8179 if (
Op.getValueType() != SrcEltVT)
8222 for (
unsigned I = 0, E = RawBits.
size();
I != E; ++
I) {
8223 if (UndefElements[
I])
8244 ID.AddInteger(
A.value());
8247 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP))
8251 newSDNode<AssertAlignSDNode>(
DL.getIROrder(),
DL.getDebugLoc(), VTs,
A);
8252 createOperands(
N, {Val});
8254 CSEMap.InsertNode(
N, IP);
8266 Flags = Inserter->getFlags();
8267 return getNode(Opcode,
DL, VT, N1, N2, Flags);
8272 if (!TLI->isCommutativeBinOp(Opcode))
8281 if ((N1C && !N2C) || (N1CFP && !N2CFP))
8295 "Operand is DELETED_NODE!");
8311 N2.
getValueType() == MVT::Other &&
"Invalid token factor!");
8315 if (N1 == N2)
return N1;
8331 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8333 N1.
getValueType() == VT &&
"Binary operator types must match!");
8336 if (N2CV && N2CV->
isZero())
8346 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8348 N1.
getValueType() == VT &&
"Binary operator types must match!");
8358 if (N2CV && N2CV->
isZero())
8372 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8374 N1.
getValueType() == VT &&
"Binary operator types must match!");
8377 if (N2CV && N2CV->
isZero())
8381 const APInt &N2CImm = N2C->getAPIntValue();
8395 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8397 N1.
getValueType() == VT &&
"Binary operator types must match!");
8410 "Types of operands of UCMP/SCMP must match");
8412 "Operands and return type of must both be scalars or vectors");
8416 "Result and operands must have the same number of elements");
8422 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8424 N1.
getValueType() == VT &&
"Binary operator types must match!");
8428 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8430 N1.
getValueType() == VT &&
"Binary operator types must match!");
8436 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8438 N1.
getValueType() == VT &&
"Binary operator types must match!");
8444 assert(VT.
isInteger() &&
"This operator does not apply to FP types!");
8446 N1.
getValueType() == VT &&
"Binary operator types must match!");
8457 N1.
getValueType() == VT &&
"Binary operator types must match!");
8465 "Invalid FCOPYSIGN!");
8470 const APInt &ShiftImm = N2C->getAPIntValue();
8484 "Shift operators return type must be the same as their first arg");
8486 "Shifts only work on integers");
8488 "Vector shift amounts must be in the same as their first arg");
8495 "Invalid use of small shift amount with oversized value!");
8502 if (N2CV && N2CV->
isZero())
8508 (N2C->getZExtValue() == 0 || N2C->getZExtValue() == 1) &&
8514 "IS_FPCLASS is used for a non-floating type");
8529 "AssertNoFPClass is used for a non-floating type");
8534 "FPClassTest value too large");
8543 "Cannot *_EXTEND_INREG FP types");
8545 "AssertSExt/AssertZExt type should be the vector element type "
8546 "rather than the vector type!");
8555 "Cannot *_EXTEND_INREG FP types");
8557 "SIGN_EXTEND_INREG type should be vector iff the operand "
8561 "Vector element counts must match in SIGN_EXTEND_INREG");
8563 if (
EVT == VT)
return N1;
8571 "FP_TO_*INT_SAT type should be vector iff the operand type is "
8575 "Vector element counts must match in FP_TO_*INT_SAT");
8577 "Type to saturate to must be a scalar.");
8584 "The result of EXTRACT_VECTOR_ELT must be at least as wide as the \
8585 element type of the vector.");
8607 N2C->getZExtValue() % Factor);
8616 "BUILD_VECTOR used for scalable vectors");
8639 if (N1Op2C && N2C) {
8669 assert(N2C && (
unsigned)N2C->getZExtValue() < 2 &&
"Bad EXTRACT_ELEMENT!");
8673 "Wrong types for EXTRACT_ELEMENT!");
8684 unsigned Shift = ElementSize * N2C->getZExtValue();
8685 const APInt &Val = N1C->getAPIntValue();
8692 "Extract subvector VTs must be vectors!");
8694 "Extract subvector VTs must have the same element type!");
8696 "Cannot extract a scalable vector from a fixed length vector!");
8699 "Extract subvector must be from larger vector to smaller vector!");
8700 assert(N2C &&
"Extract subvector index must be a constant");
8704 "Extract subvector overflow!");
8705 assert(N2C->getAPIntValue().getBitWidth() ==
8707 "Constant index for EXTRACT_SUBVECTOR has an invalid size");
8709 "Extract index is not a multiple of the output vector length");
8724 return N1.
getOperand(N2C->getZExtValue() / Factor);
8765 if (TLI->isCommutativeBinOp(Opcode)) {
8844 if (VT != MVT::Glue) {
8848 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
8849 E->intersectFlagsWith(Flags);
8853 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
8855 createOperands(
N,
Ops);
8856 CSEMap.InsertNode(
N, IP);
8858 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
8859 createOperands(
N,
Ops);
8872 Flags = Inserter->getFlags();
8873 return getNode(Opcode,
DL, VT, N1, N2, N3, Flags);
8882 "Operand is DELETED_NODE!");
8901 "SETCC operands must have the same type!");
8903 "SETCC type should be vector iff the operand type is vector!");
8906 "SETCC vector element counts must match!");
8930 "INSERT_VECTOR_ELT vector type mismatch");
8932 "INSERT_VECTOR_ELT scalar fp/int mismatch");
8935 "INSERT_VECTOR_ELT fp scalar type mismatch");
8938 "INSERT_VECTOR_ELT int scalar size mismatch");
8984 "Dest and insert subvector source types must match!");
8986 "Insert subvector VTs must be vectors!");
8988 "Insert subvector VTs must have the same element type!");
8990 "Cannot insert a scalable vector into a fixed length vector!");
8993 "Insert subvector must be from smaller vector to larger vector!");
8995 "Insert subvector index must be constant");
8999 "Insert subvector overflow!");
9002 "Constant index for INSERT_SUBVECTOR has an invalid size");
9046 case ISD::VP_TRUNCATE:
9047 case ISD::VP_SIGN_EXTEND:
9048 case ISD::VP_ZERO_EXTEND:
9057 assert(VT == VecVT &&
"Vector and result type don't match.");
9059 "All inputs must be vectors.");
9060 assert(VecVT == PassthruVT &&
"Vector and passthru types don't match.");
9062 "Vector and mask must have same number of elements.");
9077 "Expected the second and third operands of the PARTIAL_REDUCE_MLA "
9078 "node to have the same type!");
9080 "Expected the first operand of the PARTIAL_REDUCE_MLA node to have "
9081 "the same type as its result!");
9084 "Expected the element count of the second and third operands of the "
9085 "PARTIAL_REDUCE_MLA node to be a positive integer multiple of the "
9086 "element count of the first operand and the result!");
9088 "Expected the second and third operands of the PARTIAL_REDUCE_MLA "
9089 "node to have an element type which is the same as or smaller than "
9090 "the element type of the first operand and result!");
9112 if (VT != MVT::Glue) {
9116 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
9117 E->intersectFlagsWith(Flags);
9121 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
9123 createOperands(
N,
Ops);
9124 CSEMap.InsertNode(
N, IP);
9126 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
9127 createOperands(
N,
Ops);
9147 Flags = Inserter->getFlags();
9148 return getNode(Opcode,
DL, VT, N1, N2, N3, N4, Flags);
9163 Flags = Inserter->getFlags();
9164 return getNode(Opcode,
DL, VT, N1, N2, N3, N4, N5, Flags);
9181 if (FI->getIndex() < 0)
9196 assert(
C->getAPIntValue().getBitWidth() == 8);
9201 return DAG.
getConstant(Val, dl, VT,
false, IsOpaque);
9206 assert(
Value.getValueType() == MVT::i8 &&
"memset with non-byte fill value?");
9222 if (VT !=
Value.getValueType())
9235 if (Slice.Array ==
nullptr) {
9244 unsigned NumVTBytes = NumVTBits / 8;
9245 unsigned NumBytes = std::min(NumVTBytes,
unsigned(Slice.Length));
9247 APInt Val(NumVTBits, 0);
9249 for (
unsigned i = 0; i != NumBytes; ++i)
9252 for (
unsigned i = 0; i != NumBytes; ++i)
9253 Val |= (
uint64_t)(
unsigned char)Slice[i] << (NumVTBytes-i-1)*8;
9276 if (TLI->shouldPreservePtrArith(this->getMachineFunction().getFunction(),
9291 else if (Src->isAnyAdd() &&
9295 SrcDelta = Src.getConstantOperandVal(1);
9301 SrcDelta +
G->getOffset());
9317 assert(OutLoadChains.
size() &&
"Missing loads in memcpy inlining");
9318 assert(OutStoreChains.
size() &&
"Missing stores in memcpy inlining");
9320 for (
unsigned i = From; i < To; ++i) {
9322 GluedLoadChains.
push_back(OutLoadChains[i]);
9329 for (
unsigned i = From; i < To; ++i) {
9332 ST->getBasePtr(), ST->getMemoryVT(),
9333 ST->getMemOperand());
9341 Align SrcAlign,
bool isVol,
bool AlwaysInline,
9357 std::vector<EVT> MemOps;
9358 bool DstAlignCanChange =
false;
9364 DstAlignCanChange =
true;
9369 bool isZeroConstant = CopyFromConstant && Slice.Array ==
nullptr;
9371 const MemOp Op = isZeroConstant
9375 SrcAlign, isVol, CopyFromConstant);
9381 if (DstAlignCanChange) {
9382 Type *Ty = MemOps[0].getTypeForEVT(
C);
9383 Align NewDstAlign =
DL.getABITypeAlign(Ty);
9389 if (!
TRI->hasStackRealignment(MF))
9391 NewDstAlign = std::min(NewDstAlign, *StackAlign);
9393 if (NewDstAlign > DstAlign) {
9397 DstAlign = NewDstAlign;
9407 BatchAA && SrcVal &&
9415 unsigned NumMemOps = MemOps.size();
9417 for (
unsigned i = 0; i != NumMemOps; ++i) {
9422 if (VTSize >
Size) {
9425 assert(i == NumMemOps-1 && i != 0);
9426 SrcOff -= VTSize -
Size;
9427 DstOff -= VTSize -
Size;
9430 if (CopyFromConstant &&
9438 if (SrcOff < Slice.Length) {
9440 SubSlice.
move(SrcOff);
9443 SubSlice.
Array =
nullptr;
9445 SubSlice.
Length = VTSize;
9448 if (
Value.getNode()) {
9452 DstPtrInfo.
getWithOffset(DstOff), DstAlign, MMOFlags, NewAAInfo);
9457 if (!Store.getNode()) {
9466 bool isDereferenceable =
9469 if (isDereferenceable)
9484 DstPtrInfo.
getWithOffset(DstOff), VT, DstAlign, MMOFlags, NewAAInfo);
9494 unsigned NumLdStInMemcpy = OutStoreChains.
size();
9496 if (NumLdStInMemcpy) {
9502 for (
unsigned i = 0; i < NumLdStInMemcpy; ++i) {
9508 if (NumLdStInMemcpy <= GluedLdStLimit) {
9510 NumLdStInMemcpy, OutLoadChains,
9513 unsigned NumberLdChain = NumLdStInMemcpy / GluedLdStLimit;
9514 unsigned RemainingLdStInMemcpy = NumLdStInMemcpy % GluedLdStLimit;
9515 unsigned GlueIter = 0;
9518 if (RemainingLdStInMemcpy) {
9520 DAG, dl, OutChains, NumLdStInMemcpy - RemainingLdStInMemcpy,
9521 NumLdStInMemcpy, OutLoadChains, OutStoreChains);
9524 for (
unsigned cnt = 0; cnt < NumberLdChain; ++cnt) {
9525 unsigned IndexFrom = NumLdStInMemcpy - RemainingLdStInMemcpy -
9526 GlueIter - GluedLdStLimit;
9527 unsigned IndexTo = NumLdStInMemcpy - RemainingLdStInMemcpy - GlueIter;
9529 OutLoadChains, OutStoreChains);
9530 GlueIter += GluedLdStLimit;
9553 std::vector<EVT> MemOps;
9554 bool DstAlignCanChange =
false;
9560 DstAlignCanChange =
true;
9570 if (DstAlignCanChange) {
9571 Type *Ty = MemOps[0].getTypeForEVT(
C);
9572 Align NewDstAlign =
DL.getABITypeAlign(Ty);
9578 if (!
TRI->hasStackRealignment(MF))
9580 NewDstAlign = std::min(NewDstAlign, *StackAlign);
9582 if (NewDstAlign > DstAlign) {
9586 DstAlign = NewDstAlign;
9600 unsigned NumMemOps = MemOps.size();
9601 for (
unsigned i = 0; i < NumMemOps; i++) {
9605 bool IsOverlapping =
false;
9607 if (i == NumMemOps - 1 && i != 0 && VTSize >
Size - SrcOff) {
9610 SrcOff =
Size - VTSize;
9611 IsOverlapping =
true;
9618 if (IsOverlapping) {
9623 SrcAlignAtOffset, MMOFlags,
9632 bool isDereferenceable =
9635 if (isDereferenceable)
9641 SrcMMOFlags, NewAAInfo);
9649 for (
unsigned i = 0; i < NumMemOps; i++) {
9653 bool IsOverlapping =
false;
9655 if (i == NumMemOps - 1 && i != 0 && VTSize >
Size - DstOff) {
9658 DstOff =
Size - VTSize;
9659 IsOverlapping =
true;
9666 if (IsOverlapping) {
9671 DstAlignAtOffset, MMOFlags,
9680 Chain, dl, LoadValues[i],
9682 DstPtrInfo.
getWithOffset(DstOff), DstAlignAtOffset, MMOFlags,
9723 std::vector<EVT> MemOps;
9724 bool DstAlignCanChange =
false;
9731 DstAlignCanChange =
true;
9738 MemOp::Set(
Size, DstAlignCanChange, Alignment, IsZeroVal, isVol),
9743 if (DstAlignCanChange) {
9746 Align NewAlign =
DL.getABITypeAlign(Ty);
9752 if (!
TRI->hasStackRealignment(MF))
9754 NewAlign = std::min(NewAlign, *StackAlign);
9756 if (NewAlign > Alignment) {
9760 Alignment = NewAlign;
9766 unsigned NumMemOps = MemOps.size();
9771 LargestVT = MemOps[0];
9772 for (
unsigned i = 1; i < NumMemOps; i++)
9773 if (MemOps[i].bitsGT(LargestVT))
9774 LargestVT = MemOps[i];
9782 for (
unsigned i = 0; i < NumMemOps; i++) {
9787 assert(
Size > 0 &&
"Target specified more stores than needed in "
9788 "findOptimalMemOpLowering");
9789 if (VTSize >
Size) {
9792 assert(i == NumMemOps-1 && i != 0);
9793 DstOff -= VTSize -
Size;
9800 if (VT.
bitsLT(LargestVT)) {
9820 assert(
Value.getValueType() == VT &&
"Value with wrong type.");
9831 if (VTSize >
Size) {
9840 assert(
Size == 0 &&
"Target's findOptimalMemOpLowering did not specify "
9841 "stores that exactly cover the memset size");
9858 bool AllowReturnsFirstArg) {
9864 AllowReturnsFirstArg &&
9868static std::pair<SDValue, SDValue>
9875 if (LCImpl == RTLIB::Unsupported)
9887 CI->
getType(), Callee, std::move(Args))
9900 RTLIB::STRCMP,
this, TLI);
9910 RTLIB::STRSTR,
this, TLI);
9926 RTLIB::MEMCCPY,
this, TLI);
9929std::pair<SDValue, SDValue>
9938 RTLIB::MEMCMP,
this, TLI);
9948 RTLIB::STRCPY,
this, TLI);
9959 RTLIB::STRLEN,
this, TLI);
9964 Align DstAlign,
Align SrcAlign,
bool isVol,
bool AlwaysInline,
9965 const CallInst *CI, std::optional<bool> OverrideTailCall,
9973 if (ConstantSize->
isZero())
9977 *
this, dl, Chain, Dst, Src, ConstantSize->
getZExtValue(), DstAlign,
9978 SrcAlign, isVol,
false, DstPtrInfo, SrcPtrInfo, AAInfo, BatchAA);
9979 if (Result.getNode())
9986 SDValue Result = TSI->EmitTargetCodeForMemcpy(
9987 *
this, dl, Chain, Dst, Src,
Size, DstAlign, SrcAlign, isVol,
9988 AlwaysInline, DstPtrInfo, SrcPtrInfo);
9989 if (Result.getNode())
9996 assert(ConstantSize &&
"AlwaysInline requires a constant size!");
9998 *
this, dl, Chain, Dst, Src, ConstantSize->
getZExtValue(), DstAlign,
9999 SrcAlign, isVol,
true, DstPtrInfo, SrcPtrInfo, AAInfo, BatchAA);
10014 Args.emplace_back(Dst, PtrTy);
10015 Args.emplace_back(Src, PtrTy);
10019 bool IsTailCall =
false;
10020 RTLIB::LibcallImpl MemCpyImpl = TLI->getMemcpyImpl();
10022 if (OverrideTailCall.has_value()) {
10023 IsTailCall = *OverrideTailCall;
10025 bool LowersToMemcpy = MemCpyImpl == RTLIB::impl_memcpy;
10032 Libcalls->getLibcallImplCallingConv(MemCpyImpl),
10033 Dst.getValueType().getTypeForEVT(*
getContext()),
10039 std::pair<SDValue,SDValue> CallResult = TLI->LowerCallTo(CLI);
10040 return CallResult.second;
10045 Type *SizeTy,
unsigned ElemSz,
10052 Args.emplace_back(Dst, ArgTy);
10053 Args.emplace_back(Src, ArgTy);
10054 Args.emplace_back(
Size, SizeTy);
10056 RTLIB::Libcall LibraryCall =
10058 RTLIB::LibcallImpl LibcallImpl = Libcalls->getLibcallImpl(LibraryCall);
10059 if (LibcallImpl == RTLIB::Unsupported)
10066 Libcalls->getLibcallImplCallingConv(LibcallImpl),
10073 std::pair<SDValue, SDValue> CallResult = TLI->LowerCallTo(CLI);
10074 return CallResult.second;
10080 std::optional<bool> OverrideTailCall,
10088 if (ConstantSize) {
10090 if (ConstantSize->
isZero())
10094 *
this, dl, Chain, Dst, Src, ConstantSize->
getZExtValue(), DstAlign,
10095 SrcAlign, isVol,
false, DstPtrInfo, SrcPtrInfo, AAInfo);
10096 if (Result.getNode())
10103 SDValue Result = TSI->EmitTargetCodeForMemmove(
10104 *
this, dl, Chain, Dst, Src,
Size, DstAlign, SrcAlign, isVol, DstPtrInfo,
10106 if (Result.getNode())
10119 Args.emplace_back(Dst, PtrTy);
10120 Args.emplace_back(Src, PtrTy);
10125 RTLIB::LibcallImpl MemmoveImpl = Libcalls->getLibcallImpl(RTLIB::MEMMOVE);
10127 bool IsTailCall =
false;
10128 if (OverrideTailCall.has_value()) {
10129 IsTailCall = *OverrideTailCall;
10131 bool LowersToMemmove = MemmoveImpl == RTLIB::impl_memmove;
10138 Libcalls->getLibcallImplCallingConv(MemmoveImpl),
10139 Dst.getValueType().getTypeForEVT(*
getContext()),
10145 std::pair<SDValue,SDValue> CallResult = TLI->LowerCallTo(CLI);
10146 return CallResult.second;
10151 Type *SizeTy,
unsigned ElemSz,
10160 Args.emplace_back(
Size, SizeTy);
10162 RTLIB::Libcall LibraryCall =
10164 RTLIB::LibcallImpl LibcallImpl = Libcalls->getLibcallImpl(LibraryCall);
10165 if (LibcallImpl == RTLIB::Unsupported)
10172 Libcalls->getLibcallImplCallingConv(LibcallImpl),
10179 std::pair<SDValue, SDValue> CallResult = TLI->LowerCallTo(CLI);
10180 return CallResult.second;
10185 bool isVol,
bool AlwaysInline,
10192 if (ConstantSize) {
10194 if (ConstantSize->
isZero())
10199 isVol,
false, DstPtrInfo, AAInfo);
10201 if (Result.getNode())
10208 SDValue Result = TSI->EmitTargetCodeForMemset(
10209 *
this, dl, Chain, Dst, Src,
Size, Alignment, isVol, AlwaysInline, DstPtrInfo);
10210 if (Result.getNode())
10216 if (AlwaysInline) {
10217 assert(ConstantSize &&
"AlwaysInline requires a constant size!");
10220 isVol,
true, DstPtrInfo, AAInfo);
10222 "getMemsetStores must return a valid sequence when AlwaysInline");
10236 RTLIB::LibcallImpl BzeroImpl = Libcalls->getLibcallImpl(RTLIB::BZERO);
10237 bool UseBZero = BzeroImpl != RTLIB::Unsupported &&
isNullConstant(Src);
10243 Args.emplace_back(
Size,
DL.getIntPtrType(Ctx));
10245 Libcalls->getLibcallImplCallingConv(BzeroImpl),
Type::getVoidTy(Ctx),
10248 RTLIB::LibcallImpl MemsetImpl = Libcalls->getLibcallImpl(RTLIB::MEMSET);
10252 Args.emplace_back(Src, Src.getValueType().getTypeForEVT(Ctx));
10253 Args.emplace_back(
Size,
DL.getIntPtrType(Ctx));
10254 CLI.
setLibCallee(Libcalls->getLibcallImplCallingConv(MemsetImpl),
10255 Dst.getValueType().getTypeForEVT(Ctx),
10260 RTLIB::LibcallImpl MemsetImpl = Libcalls->getLibcallImpl(RTLIB::MEMSET);
10261 bool LowersToMemset = MemsetImpl == RTLIB::impl_memset;
10272 std::pair<SDValue, SDValue> CallResult = TLI->LowerCallTo(CLI);
10273 return CallResult.second;
10278 Type *SizeTy,
unsigned ElemSz,
10285 Args.emplace_back(
Size, SizeTy);
10287 RTLIB::Libcall LibraryCall =
10289 RTLIB::LibcallImpl LibcallImpl = Libcalls->getLibcallImpl(LibraryCall);
10290 if (LibcallImpl == RTLIB::Unsupported)
10297 Libcalls->getLibcallImplCallingConv(LibcallImpl),
10304 std::pair<SDValue, SDValue> CallResult = TLI->LowerCallTo(CLI);
10305 return CallResult.second;
10315 ID.AddInteger(getSyntheticNodeSubclassData<AtomicSDNode>(
10316 dl.
getIROrder(), Opcode, VTList, MemVT, MMO, ExtType));
10319 void* IP =
nullptr;
10321 E->refineAlignment(MMO);
10322 E->refineRanges(MMO);
10327 VTList, MemVT, MMO, ExtType);
10328 createOperands(
N,
Ops);
10330 CSEMap.InsertNode(
N, IP);
10367 "Invalid Atomic Op");
10387 if (
Ops.size() == 1)
10402 if (
Size.hasValue() && !
Size.getValue())
10407 MF.getMachineMemOperand(PtrInfo, Flags,
Size, Alignment, AAInfo);
10423 assert(!MMOs.
empty() &&
"Must have at least one MMO");
10427 (Opcode <= (
unsigned)std::numeric_limits<int>::max() &&
10429 "Opcode is not a memory-accessing opcode!");
10432 if (MMOs.
size() == 1) {
10438 void *Buffer = Allocator.Allocate(AllocSize,
alignof(
size_t));
10439 size_t *CountPtr =
static_cast<size_t *
>(Buffer);
10440 *CountPtr = MMOs.
size();
10449 if (VTList.
VTs[VTList.
NumVTs-1] != MVT::Glue) {
10452 ID.AddInteger(getSyntheticNodeSubclassData<MemIntrinsicSDNode>(
10453 Opcode, dl.
getIROrder(), VTList, MemVT, MemRefs));
10456 ID.AddInteger(MMO->getPointerInfo().getAddrSpace());
10457 ID.AddInteger(MMO->getFlags());
10459 void *IP =
nullptr;
10460 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
10466 VTList, MemVT, MemRefs);
10467 createOperands(
N,
Ops);
10468 CSEMap.InsertNode(
N, IP);
10471 VTList, MemVT, MemRefs);
10472 createOperands(
N,
Ops);
10481 SDValue Chain,
int FrameIndex) {
10483 const auto VTs =
getVTList(MVT::Other);
10492 ID.AddInteger(FrameIndex);
10493 void *IP =
nullptr;
10494 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP))
10499 createOperands(
N,
Ops);
10500 CSEMap.InsertNode(
N, IP);
10511 const auto VTs =
getVTList(MVT::Other);
10516 ID.AddInteger(Index);
10517 void *IP =
nullptr;
10518 if (
SDNode *E = FindNodeOrInsertPos(
ID, Dl, IP))
10521 auto *
N = newSDNode<PseudoProbeSDNode>(
10523 createOperands(
N,
Ops);
10524 CSEMap.InsertNode(
N, IP);
10541 FI->getIndex(),
Offset);
10578 "Invalid chain type");
10590 Alignment, AAInfo, Ranges);
10591 return getLoad(AM, ExtType, VT, dl, Chain, Ptr,
Offset, MemVT, MMO);
10601 assert(VT == MemVT &&
"Non-extending load from different memory type!");
10605 "Should only be an extending load, not truncating!");
10607 "Cannot convert from FP to Int or Int -> FP!");
10609 "Cannot use an ext load to convert to or from a vector!");
10612 "Cannot use an ext load to change the number of vector elements!");
10619 "Range metadata and load type must match!");
10630 ID.AddInteger(getSyntheticNodeSubclassData<LoadSDNode>(
10631 dl.
getIROrder(), VTs, AM, ExtType, MemVT, MMO));
10634 void *IP =
nullptr;
10636 E->refineAlignment(MMO);
10637 E->refineRanges(MMO);
10641 ExtType, MemVT, MMO);
10642 createOperands(
N,
Ops);
10644 CSEMap.InsertNode(
N, IP);
10658 PtrInfo, VT, Alignment, MMOFlags, AAInfo, Ranges);
10676 MemVT, Alignment, MMOFlags, AAInfo);
10691 assert(LD->getOffset().isUndef() &&
"Load is already a indexed load!");
10694 LD->getMemOperand()->getFlags() &
10697 LD->getChain(),
Base,
Offset, LD->getPointerInfo(),
10698 LD->getMemoryVT(), LD->getAlign(), MMOFlags, LD->getAAInfo());
10717 MF.getMachineMemOperand(PtrInfo, MMOFlags,
Size, Alignment, AAInfo);
10718 return getStore(Chain, dl, Val, Ptr, MMO);
10731 bool IsTruncating) {
10735 IsTruncating =
false;
10736 }
else if (!IsTruncating) {
10737 assert(VT == SVT &&
"No-truncating store from different memory type!");
10740 "Should only be a truncating store, not extending!");
10743 "Cannot use trunc store to convert to or from a vector!");
10746 "Cannot use trunc store to change the number of vector elements!");
10757 ID.AddInteger(getSyntheticNodeSubclassData<StoreSDNode>(
10758 dl.
getIROrder(), VTs, AM, IsTruncating, SVT, MMO));
10761 void *IP =
nullptr;
10762 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
10767 IsTruncating, SVT, MMO);
10768 createOperands(
N,
Ops);
10770 CSEMap.InsertNode(
N, IP);
10783 "Invalid chain type");
10793 PtrInfo, MMOFlags, SVT.
getStoreSize(), Alignment, AAInfo);
10808 assert(ST->getOffset().isUndef() &&
"Store is already a indexed store!");
10810 ST->getMemoryVT(), ST->getMemOperand(), AM,
10811 ST->isTruncatingStore());
10819 const MDNode *Ranges,
bool IsExpanding) {
10830 Alignment, AAInfo, Ranges);
10831 return getLoadVP(AM, ExtType, VT, dl, Chain, Ptr,
Offset, Mask, EVL, MemVT,
10840 bool IsExpanding) {
10842 assert(Mask.getValueType().getVectorElementCount() ==
10844 "Vector width mismatch between mask and data");
10855 ID.AddInteger(getSyntheticNodeSubclassData<VPLoadSDNode>(
10856 dl.
getIROrder(), VTs, AM, ExtType, IsExpanding, MemVT, MMO));
10859 void *IP =
nullptr;
10861 E->refineAlignment(MMO);
10862 E->refineRanges(MMO);
10866 ExtType, IsExpanding, MemVT, MMO);
10867 createOperands(
N,
Ops);
10869 CSEMap.InsertNode(
N, IP);
10882 bool IsExpanding) {
10885 Mask, EVL, PtrInfo, VT, Alignment, MMOFlags, AAInfo, Ranges,
10894 Mask, EVL, VT, MMO, IsExpanding);
10903 const AAMDNodes &AAInfo,
bool IsExpanding) {
10906 EVL, PtrInfo, MemVT, Alignment, MMOFlags, AAInfo,
nullptr,
10916 EVL, MemVT, MMO, IsExpanding);
10923 assert(LD->getOffset().isUndef() &&
"Load is already a indexed load!");
10926 LD->getMemOperand()->getFlags() &
10929 LD->getChain(),
Base,
Offset, LD->getMask(),
10930 LD->getVectorLength(), LD->getPointerInfo(),
10931 LD->getMemoryVT(), LD->getAlign(), MMOFlags, LD->getAAInfo(),
10932 nullptr, LD->isExpandingLoad());
10939 bool IsCompressing) {
10941 assert(Mask.getValueType().getVectorElementCount() ==
10943 "Vector width mismatch between mask and data");
10953 ID.AddInteger(getSyntheticNodeSubclassData<VPStoreSDNode>(
10954 dl.
getIROrder(), VTs, AM, IsTruncating, IsCompressing, MemVT, MMO));
10957 void *IP =
nullptr;
10958 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
10963 IsTruncating, IsCompressing, MemVT, MMO);
10964 createOperands(
N,
Ops);
10966 CSEMap.InsertNode(
N, IP);
10979 bool IsCompressing) {
10990 PtrInfo, MMOFlags, SVT.
getStoreSize(), Alignment, AAInfo);
10999 bool IsCompressing) {
11006 false, IsCompressing);
11009 "Should only be a truncating store, not extending!");
11012 "Cannot use trunc store to convert to or from a vector!");
11015 "Cannot use trunc store to change the number of vector elements!");
11023 ID.AddInteger(getSyntheticNodeSubclassData<VPStoreSDNode>(
11027 void *IP =
nullptr;
11028 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11035 createOperands(
N,
Ops);
11037 CSEMap.InsertNode(
N, IP);
11048 assert(ST->getOffset().isUndef() &&
"Store is already an indexed store!");
11051 Offset, ST->getMask(), ST->getVectorLength()};
11054 ID.AddInteger(ST->getMemoryVT().getRawBits());
11055 ID.AddInteger(ST->getRawSubclassData());
11056 ID.AddInteger(ST->getPointerInfo().getAddrSpace());
11057 ID.AddInteger(ST->getMemOperand()->getFlags());
11058 void *IP =
nullptr;
11059 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP))
11062 auto *
N = newSDNode<VPStoreSDNode>(
11064 ST->isCompressingStore(), ST->getMemoryVT(), ST->getMemOperand());
11065 createOperands(
N,
Ops);
11067 CSEMap.InsertNode(
N, IP);
11087 ID.AddInteger(getSyntheticNodeSubclassData<VPStridedLoadSDNode>(
11088 DL.getIROrder(), VTs, AM, ExtType, IsExpanding, MemVT, MMO));
11091 void *IP =
nullptr;
11092 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
11098 newSDNode<VPStridedLoadSDNode>(
DL.getIROrder(),
DL.getDebugLoc(), VTs, AM,
11099 ExtType, IsExpanding, MemVT, MMO);
11100 createOperands(
N,
Ops);
11101 CSEMap.InsertNode(
N, IP);
11112 bool IsExpanding) {
11115 Undef, Stride, Mask, EVL, VT, MMO, IsExpanding);
11124 Stride, Mask, EVL, MemVT, MMO, IsExpanding);
11133 bool IsTruncating,
bool IsCompressing) {
11143 ID.AddInteger(getSyntheticNodeSubclassData<VPStridedStoreSDNode>(
11144 DL.getIROrder(), VTs, AM, IsTruncating, IsCompressing, MemVT, MMO));
11146 void *IP =
nullptr;
11147 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
11151 auto *
N = newSDNode<VPStridedStoreSDNode>(
DL.getIROrder(),
DL.getDebugLoc(),
11152 VTs, AM, IsTruncating,
11153 IsCompressing, MemVT, MMO);
11154 createOperands(
N,
Ops);
11156 CSEMap.InsertNode(
N, IP);
11168 bool IsCompressing) {
11175 false, IsCompressing);
11178 "Should only be a truncating store, not extending!");
11181 "Cannot use trunc store to convert to or from a vector!");
11184 "Cannot use trunc store to change the number of vector elements!");
11192 ID.AddInteger(getSyntheticNodeSubclassData<VPStridedStoreSDNode>(
11195 void *IP =
nullptr;
11196 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
11200 auto *
N = newSDNode<VPStridedStoreSDNode>(
DL.getIROrder(),
DL.getDebugLoc(),
11202 IsCompressing, SVT, MMO);
11203 createOperands(
N,
Ops);
11205 CSEMap.InsertNode(
N, IP);
11215 assert(
Ops.size() == 6 &&
"Incompatible number of operands");
11220 ID.AddInteger(getSyntheticNodeSubclassData<VPGatherSDNode>(
11224 void *IP =
nullptr;
11225 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11231 VT, MMO, IndexType);
11232 createOperands(
N,
Ops);
11234 assert(
N->getMask().getValueType().getVectorElementCount() ==
11235 N->getValueType(0).getVectorElementCount() &&
11236 "Vector width mismatch between mask and data");
11237 assert(
N->getIndex().getValueType().getVectorElementCount().isScalable() ==
11238 N->getValueType(0).getVectorElementCount().isScalable() &&
11239 "Scalable flags of index and data do not match");
11241 N->getIndex().getValueType().getVectorElementCount(),
11242 N->getValueType(0).getVectorElementCount()) &&
11243 "Vector width mismatch between index and data");
11245 N->getScale()->getAsAPIntVal().isPowerOf2() &&
11246 "Scale should be a constant power of 2");
11248 CSEMap.InsertNode(
N, IP);
11259 assert(
Ops.size() == 7 &&
"Incompatible number of operands");
11264 ID.AddInteger(getSyntheticNodeSubclassData<VPScatterSDNode>(
11268 void *IP =
nullptr;
11269 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11274 VT, MMO, IndexType);
11275 createOperands(
N,
Ops);
11277 assert(
N->getMask().getValueType().getVectorElementCount() ==
11278 N->getValue().getValueType().getVectorElementCount() &&
11279 "Vector width mismatch between mask and data");
11281 N->getIndex().getValueType().getVectorElementCount().isScalable() ==
11282 N->getValue().getValueType().getVectorElementCount().isScalable() &&
11283 "Scalable flags of index and data do not match");
11285 N->getIndex().getValueType().getVectorElementCount(),
11286 N->getValue().getValueType().getVectorElementCount()) &&
11287 "Vector width mismatch between index and data");
11289 N->getScale()->getAsAPIntVal().isPowerOf2() &&
11290 "Scale should be a constant power of 2");
11292 CSEMap.InsertNode(
N, IP);
11307 "Unindexed masked load with an offset!");
11314 ID.AddInteger(getSyntheticNodeSubclassData<MaskedLoadSDNode>(
11315 dl.
getIROrder(), VTs, AM, ExtTy, isExpanding, MemVT, MMO));
11318 void *IP =
nullptr;
11319 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11324 AM, ExtTy, isExpanding, MemVT, MMO);
11325 createOperands(
N,
Ops);
11327 CSEMap.InsertNode(
N, IP);
11338 assert(LD->getOffset().isUndef() &&
"Masked load is already a indexed load!");
11340 Offset, LD->getMask(), LD->getPassThru(),
11341 LD->getMemoryVT(), LD->getMemOperand(), AM,
11342 LD->getExtensionType(), LD->isExpandingLoad());
11350 bool IsCompressing) {
11352 "Invalid chain type");
11355 "Unindexed masked store with an offset!");
11362 ID.AddInteger(getSyntheticNodeSubclassData<MaskedStoreSDNode>(
11363 dl.
getIROrder(), VTs, AM, IsTruncating, IsCompressing, MemVT, MMO));
11366 void *IP =
nullptr;
11367 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11373 IsTruncating, IsCompressing, MemVT, MMO);
11374 createOperands(
N,
Ops);
11376 CSEMap.InsertNode(
N, IP);
11387 assert(ST->getOffset().isUndef() &&
11388 "Masked store is already a indexed store!");
11390 ST->getMask(), ST->getMemoryVT(), ST->getMemOperand(),
11391 AM, ST->isTruncatingStore(), ST->isCompressingStore());
11399 assert(
Ops.size() == 6 &&
"Incompatible number of operands");
11404 ID.AddInteger(getSyntheticNodeSubclassData<MaskedGatherSDNode>(
11405 dl.
getIROrder(), VTs, MemVT, MMO, IndexType, ExtTy));
11408 void *IP =
nullptr;
11409 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11415 VTs, MemVT, MMO, IndexType, ExtTy);
11416 createOperands(
N,
Ops);
11418 assert(
N->getPassThru().getValueType() ==
N->getValueType(0) &&
11419 "Incompatible type of the PassThru value in MaskedGatherSDNode");
11420 assert(
N->getMask().getValueType().getVectorElementCount() ==
11421 N->getValueType(0).getVectorElementCount() &&
11422 "Vector width mismatch between mask and data");
11423 assert(
N->getIndex().getValueType().getVectorElementCount().isScalable() ==
11424 N->getValueType(0).getVectorElementCount().isScalable() &&
11425 "Scalable flags of index and data do not match");
11427 N->getIndex().getValueType().getVectorElementCount(),
11428 N->getValueType(0).getVectorElementCount()) &&
11429 "Vector width mismatch between index and data");
11431 N->getScale()->getAsAPIntVal().isPowerOf2() &&
11432 "Scale should be a constant power of 2");
11434 CSEMap.InsertNode(
N, IP);
11446 assert(
Ops.size() == 6 &&
"Incompatible number of operands");
11451 ID.AddInteger(getSyntheticNodeSubclassData<MaskedScatterSDNode>(
11452 dl.
getIROrder(), VTs, MemVT, MMO, IndexType, IsTrunc));
11455 void *IP =
nullptr;
11456 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11462 VTs, MemVT, MMO, IndexType, IsTrunc);
11463 createOperands(
N,
Ops);
11465 assert(
N->getMask().getValueType().getVectorElementCount() ==
11466 N->getValue().getValueType().getVectorElementCount() &&
11467 "Vector width mismatch between mask and data");
11469 N->getIndex().getValueType().getVectorElementCount().isScalable() ==
11470 N->getValue().getValueType().getVectorElementCount().isScalable() &&
11471 "Scalable flags of index and data do not match");
11473 N->getIndex().getValueType().getVectorElementCount(),
11474 N->getValue().getValueType().getVectorElementCount()) &&
11475 "Vector width mismatch between index and data");
11477 N->getScale()->getAsAPIntVal().isPowerOf2() &&
11478 "Scale should be a constant power of 2");
11480 CSEMap.InsertNode(
N, IP);
11491 assert(
Ops.size() == 7 &&
"Incompatible number of operands");
11496 ID.AddInteger(getSyntheticNodeSubclassData<MaskedHistogramSDNode>(
11497 dl.
getIROrder(), VTs, MemVT, MMO, IndexType));
11500 void *IP =
nullptr;
11501 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP)) {
11507 VTs, MemVT, MMO, IndexType);
11508 createOperands(
N,
Ops);
11510 assert(
N->getMask().getValueType().getVectorElementCount() ==
11511 N->getIndex().getValueType().getVectorElementCount() &&
11512 "Vector width mismatch between mask and data");
11514 N->getScale()->getAsAPIntVal().isPowerOf2() &&
11515 "Scale should be a constant power of 2");
11516 assert(
N->getInc().getValueType().isInteger() &&
"Non integer update value");
11518 CSEMap.InsertNode(
N, IP);
11533 ID.AddInteger(getSyntheticNodeSubclassData<VPLoadFFSDNode>(
DL.getIROrder(),
11537 void *IP =
nullptr;
11538 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
11542 auto *
N = newSDNode<VPLoadFFSDNode>(
DL.getIROrder(),
DL.getDebugLoc(), VTs,
11544 createOperands(
N,
Ops);
11546 CSEMap.InsertNode(
N, IP);
11561 ID.AddInteger(getSyntheticNodeSubclassData<FPStateAccessSDNode>(
11565 void *IP =
nullptr;
11566 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP))
11571 createOperands(
N,
Ops);
11573 CSEMap.InsertNode(
N, IP);
11588 ID.AddInteger(getSyntheticNodeSubclassData<FPStateAccessSDNode>(
11592 void *IP =
nullptr;
11593 if (
SDNode *E = FindNodeOrInsertPos(
ID, dl, IP))
11598 createOperands(
N,
Ops);
11600 CSEMap.InsertNode(
N, IP);
11611 if (
Cond.isUndef())
11646 return !Val || Val->getAPIntValue().uge(
X.getScalarValueSizeInBits());
11652 if (
X.getValueType().getScalarType() == MVT::i1)
11665 bool HasNan = (XC && XC->
getValueAPF().isNaN()) ||
11667 bool HasInf = (XC && XC->
getValueAPF().isInfinity()) ||
11670 if (Flags.hasNoNaNs() && (HasNan ||
X.isUndef() ||
Y.isUndef()))
11673 if (Flags.hasNoInfs() && (HasInf ||
X.isUndef() ||
Y.isUndef()))
11696 if (Opcode ==
ISD::FMUL && Flags.hasNoNaNs() && Flags.hasNoSignedZeros())
11711 switch (
Ops.size()) {
11712 case 0:
return getNode(Opcode,
DL, VT);
11722 return getNode(Opcode,
DL, VT, NewOps);
11729 Flags = Inserter->getFlags();
11737 case 0:
return getNode(Opcode,
DL, VT);
11738 case 1:
return getNode(Opcode,
DL, VT,
Ops[0], Flags);
11745 for (
const auto &
Op :
Ops)
11747 "Operand is DELETED_NODE!");
11764 "LHS and RHS of condition must have same type!");
11766 "True and False arms of SelectCC must have same type!");
11768 "select_cc node must be of same type as true and false value!");
11772 "Expected select_cc with vector result to have the same sized "
11773 "comparison type!");
11778 "LHS/RHS of comparison should match types!");
11784 Opcode = ISD::VP_XOR;
11789 Opcode = ISD::VP_AND;
11791 case ISD::VP_REDUCE_MUL:
11794 Opcode = ISD::VP_REDUCE_AND;
11796 case ISD::VP_REDUCE_ADD:
11799 Opcode = ISD::VP_REDUCE_XOR;
11801 case ISD::VP_REDUCE_SMAX:
11802 case ISD::VP_REDUCE_UMIN:
11806 Opcode = ISD::VP_REDUCE_AND;
11808 case ISD::VP_REDUCE_SMIN:
11809 case ISD::VP_REDUCE_UMAX:
11813 Opcode = ISD::VP_REDUCE_OR;
11821 if (VT != MVT::Glue) {
11824 void *IP =
nullptr;
11826 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
11827 E->intersectFlagsWith(Flags);
11831 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
11832 createOperands(
N,
Ops);
11834 CSEMap.InsertNode(
N, IP);
11836 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
11837 createOperands(
N,
Ops);
11840 N->setFlags(Flags);
11851 Flags = Inserter->getFlags();
11865 Flags = Inserter->getFlags();
11875 for (
const auto &
Op :
Ops)
11877 "Operand is DELETED_NODE!");
11886 "Invalid add/sub overflow op!");
11888 Ops[0].getValueType() ==
Ops[1].getValueType() &&
11889 Ops[0].getValueType() == VTList.
VTs[0] &&
11890 "Binary operator types must match!");
11897 if (N2CV && N2CV->
isZero()) {
11928 "Invalid add/sub overflow op!");
11930 Ops[0].getValueType() ==
Ops[1].getValueType() &&
11931 Ops[0].getValueType() == VTList.
VTs[0] &&
11932 Ops[2].getValueType() == VTList.
VTs[1] &&
11933 "Binary operator types must match!");
11937 assert(VTList.
NumVTs == 2 &&
Ops.size() == 2 &&
"Invalid mul lo/hi op!");
11939 VTList.
VTs[0] ==
Ops[0].getValueType() &&
11940 VTList.
VTs[0] ==
Ops[1].getValueType() &&
11941 "Binary operator types must match!");
11947 unsigned OutWidth = Width * 2;
11948 APInt Val = LHS->getAPIntValue();
11951 Val = Val.
sext(OutWidth);
11952 Mul =
Mul.sext(OutWidth);
11954 Val = Val.
zext(OutWidth);
11955 Mul =
Mul.zext(OutWidth);
11967 assert(VTList.
NumVTs == 2 &&
Ops.size() == 1 &&
"Invalid ffrexp op!");
11969 VTList.
VTs[0] ==
Ops[0].getValueType() &&
"frexp type mismatch");
11977 DL, VTList.
VTs[1]);
11985 "Invalid STRICT_FP_EXTEND!");
11987 Ops[1].getValueType().isFloatingPoint() &&
"Invalid FP cast!");
11989 "STRICT_FP_EXTEND result type should be vector iff the operand "
11990 "type is vector!");
11993 Ops[1].getValueType().getVectorElementCount()) &&
11994 "Vector element count mismatch!");
11996 "Invalid fpext node, dst <= src!");
11999 assert(VTList.
NumVTs == 2 &&
Ops.size() == 3 &&
"Invalid STRICT_FP_ROUND!");
12001 "STRICT_FP_ROUND result type should be vector iff the operand "
12002 "type is vector!");
12005 Ops[1].getValueType().getVectorElementCount()) &&
12006 "Vector element count mismatch!");
12008 Ops[1].getValueType().isFloatingPoint() &&
12011 (
Ops[2]->getAsZExtVal() == 0 ||
Ops[2]->getAsZExtVal() == 1) &&
12012 "Invalid STRICT_FP_ROUND!");
12018 if (VTList.
VTs[VTList.
NumVTs-1] != MVT::Glue) {
12021 void *IP =
nullptr;
12022 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
12023 E->intersectFlagsWith(Flags);
12027 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTList);
12028 createOperands(
N,
Ops);
12029 CSEMap.InsertNode(
N, IP);
12031 N = newSDNode<SDNode>(Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTList);
12032 createOperands(
N,
Ops);
12035 N->setFlags(Flags);
12082 return makeVTList(&(*EVTs.insert(VT).first), 1);
12091 void *IP =
nullptr;
12094 EVT *Array = Allocator.Allocate<
EVT>(2);
12097 Result =
new (Allocator)
SDVTListNode(
ID.Intern(Allocator), Array, 2);
12098 VTListMap.InsertNode(Result, IP);
12100 return Result->getSDVTList();
12110 void *IP =
nullptr;
12113 EVT *Array = Allocator.Allocate<
EVT>(3);
12117 Result =
new (Allocator)
SDVTListNode(
ID.Intern(Allocator), Array, 3);
12118 VTListMap.InsertNode(Result, IP);
12120 return Result->getSDVTList();
12131 void *IP =
nullptr;
12134 EVT *Array = Allocator.Allocate<
EVT>(4);
12139 Result =
new (Allocator)
SDVTListNode(
ID.Intern(Allocator), Array, 4);
12140 VTListMap.InsertNode(Result, IP);
12142 return Result->getSDVTList();
12146 unsigned NumVTs = VTs.
size();
12148 ID.AddInteger(NumVTs);
12149 for (
unsigned index = 0; index < NumVTs; index++) {
12150 ID.AddInteger(VTs[index].getRawBits());
12153 void *IP =
nullptr;
12156 EVT *Array = Allocator.Allocate<
EVT>(NumVTs);
12158 Result =
new (Allocator)
SDVTListNode(
ID.Intern(Allocator), Array, NumVTs);
12159 VTListMap.InsertNode(Result, IP);
12161 return Result->getSDVTList();
12172 assert(
N->getNumOperands() == 1 &&
"Update with wrong number of operands");
12175 if (
Op ==
N->getOperand(0))
return N;
12178 void *InsertPos =
nullptr;
12179 if (
SDNode *Existing = FindModifiedNodeSlot(
N,
Op, InsertPos))
12184 if (!RemoveNodeFromCSEMaps(
N))
12185 InsertPos =
nullptr;
12188 N->OperandList[0].set(
Op);
12192 if (InsertPos) CSEMap.InsertNode(
N, InsertPos);
12197 assert(
N->getNumOperands() == 2 &&
"Update with wrong number of operands");
12200 if (Op1 ==
N->getOperand(0) && Op2 ==
N->getOperand(1))
12204 void *InsertPos =
nullptr;
12205 if (
SDNode *Existing = FindModifiedNodeSlot(
N, Op1, Op2, InsertPos))
12210 if (!RemoveNodeFromCSEMaps(
N))
12211 InsertPos =
nullptr;
12214 if (
N->OperandList[0] != Op1)
12215 N->OperandList[0].set(Op1);
12216 if (
N->OperandList[1] != Op2)
12217 N->OperandList[1].set(Op2);
12221 if (InsertPos) CSEMap.InsertNode(
N, InsertPos);
12241 SDValue Ops[] = { Op1, Op2, Op3, Op4, Op5 };
12249 "Update with wrong number of operands");
12252 if (std::equal(
Ops.begin(),
Ops.end(),
N->op_begin()))
12256 void *InsertPos =
nullptr;
12257 if (
SDNode *Existing = FindModifiedNodeSlot(
N,
Ops, InsertPos))
12262 if (!RemoveNodeFromCSEMaps(
N))
12263 InsertPos =
nullptr;
12266 for (
unsigned i = 0; i !=
NumOps; ++i)
12267 if (
N->OperandList[i] !=
Ops[i])
12268 N->OperandList[i].set(
Ops[i]);
12272 if (InsertPos) CSEMap.InsertNode(
N, InsertPos);
12289 if (NewMemRefs.
empty()) {
12295 if (NewMemRefs.
size() == 1) {
12296 N->MemRefs = NewMemRefs[0];
12302 Allocator.template Allocate<MachineMemOperand *>(NewMemRefs.
size());
12304 N->MemRefs = MemRefsBuffer;
12305 N->NumMemRefs =
static_cast<int>(NewMemRefs.
size());
12377 New->setNodeId(-1);
12397 unsigned Order = std::min(
N->getIROrder(), OLoc.
getIROrder());
12398 N->setIROrder(Order);
12421 void *IP =
nullptr;
12422 if (VTs.
VTs[VTs.
NumVTs-1] != MVT::Glue) {
12426 return UpdateSDLocOnMergeSDNode(ON,
SDLoc(
N));
12429 if (!RemoveNodeFromCSEMaps(
N))
12434 N->ValueList = VTs.
VTs;
12444 if (Used->use_empty())
12445 DeadNodeSet.
insert(Used);
12450 MN->clearMemRefs();
12454 createOperands(
N,
Ops);
12458 if (!DeadNodeSet.
empty()) {
12460 for (
SDNode *
N : DeadNodeSet)
12461 if (
N->use_empty())
12467 CSEMap.InsertNode(
N, IP);
12472 unsigned OrigOpc =
Node->getOpcode();
12477#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
12478 case ISD::STRICT_##DAGN: NewOpc = ISD::DAGN; break;
12479#define CMP_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
12480 case ISD::STRICT_##DAGN: NewOpc = ISD::SETCC; break;
12481#include "llvm/IR/ConstrainedOps.def"
12484 assert(
Node->getNumValues() == 2 &&
"Unexpected number of results!");
12492 for (
unsigned i = 1, e =
Node->getNumOperands(); i != e; ++i)
12493 Ops.push_back(
Node->getOperand(i));
12610 bool DoCSE = VTs.
VTs[VTs.
NumVTs-1] != MVT::Glue;
12612 void *IP =
nullptr;
12618 if (
SDNode *E = FindNodeOrInsertPos(
ID,
DL, IP)) {
12624 N = newSDNode<MachineSDNode>(~Opcode,
DL.getIROrder(),
DL.getDebugLoc(), VTs);
12625 createOperands(
N,
Ops);
12628 CSEMap.InsertNode(
N, IP);
12641 VT, Operand, SRIdxVal);
12651 VT, Operand, Subreg, SRIdxVal);
12659 bool AllowCommute) {
12662 Flags = Inserter->getFlags();
12669 bool AllowCommute) {
12670 if (VTList.
VTs[VTList.
NumVTs - 1] == MVT::Glue)
12676 void *IP =
nullptr;
12677 if (
SDNode *E = FindNodeOrInsertPos(
ID, IP)) {
12678 E->intersectFlagsWith(Flags);
12687 if (AllowCommute && TLI->isCommutativeBinOp(Opcode))
12696 if (VTList.
VTs[VTList.
NumVTs - 1] != MVT::Glue) {
12699 void *IP =
nullptr;
12700 if (FindNodeOrInsertPos(
ID,
SDLoc(), IP))
12710 SDNode *
N,
unsigned R,
bool IsIndirect,
12713 "Expected inlined-at fields to agree");
12714 return new (DbgInfo->getAlloc())
12716 {}, IsIndirect,
DL, O,
12726 "Expected inlined-at fields to agree");
12727 return new (DbgInfo->getAlloc())
12740 "Expected inlined-at fields to agree");
12752 "Expected inlined-at fields to agree");
12753 return new (DbgInfo->getAlloc())
12755 Dependencies, IsIndirect,
DL, O,
12764 "Expected inlined-at fields to agree");
12765 return new (DbgInfo->getAlloc())
12767 {}, IsIndirect,
DL, O,
12775 unsigned O,
bool IsVariadic) {
12777 "Expected inlined-at fields to agree");
12778 return new (DbgInfo->getAlloc())
12779 SDDbgValue(DbgInfo->getAlloc(), Var, Expr, Locs, Dependencies, IsIndirect,
12780 DL, O, IsVariadic);
12784 unsigned OffsetInBits,
unsigned SizeInBits,
12785 bool InvalidateDbg) {
12788 assert(FromNode && ToNode &&
"Can't modify dbg values");
12793 if (From == To || FromNode == ToNode)
12805 if (Dbg->isInvalidated())
12813 auto NewLocOps = Dbg->copyLocationOps();
12815 NewLocOps.begin(), NewLocOps.end(),
12817 bool Match = Op == FromLocOp;
12827 auto *Expr = Dbg->getExpression();
12833 if (
auto FI = Expr->getFragmentInfo())
12834 if (OffsetInBits + SizeInBits > FI->SizeInBits)
12843 auto AdditionalDependencies = Dbg->getAdditionalDependencies();
12846 Var, Expr, NewLocOps, AdditionalDependencies, Dbg->isIndirect(),
12847 Dbg->getDebugLoc(), std::max(ToNode->
getIROrder(), Dbg->getOrder()),
12848 Dbg->isVariadic());
12851 if (InvalidateDbg) {
12853 Dbg->setIsInvalidated();
12854 Dbg->setIsEmitted();
12860 "Transferred DbgValues should depend on the new SDNode");
12866 if (!
N.getHasDebugValue())
12869 auto GetLocationOperand = [](
SDNode *
Node,
unsigned ResNo) {
12877 if (DV->isInvalidated())
12879 switch (
N.getOpcode()) {
12889 Offset =
N.getConstantOperandVal(1);
12892 if (!RHSConstant && DV->isIndirect())
12899 auto *DIExpr = DV->getExpression();
12900 auto NewLocOps = DV->copyLocationOps();
12902 size_t OrigLocOpsSize = NewLocOps.size();
12903 for (
size_t i = 0; i < OrigLocOpsSize; ++i) {
12908 NewLocOps[i].getSDNode() != &
N)
12919 const auto *TmpDIExpr =
12927 NewLocOps.push_back(RHS);
12936 DV->isVariadic() || OrigLocOpsSize != NewLocOps.size();
12938 auto AdditionalDependencies = DV->getAdditionalDependencies();
12940 DV->getVariable(), DIExpr, NewLocOps, AdditionalDependencies,
12941 DV->isIndirect(), DV->getDebugLoc(), DV->getOrder(), IsVariadic);
12943 DV->setIsInvalidated();
12944 DV->setIsEmitted();
12946 N0.
getNode()->dumprFull(
this);
12947 dbgs() <<
" into " << *DIExpr <<
'\n');
12954 TypeSize ToSize =
N.getValueSizeInBits(0);
12958 auto NewLocOps = DV->copyLocationOps();
12960 for (
size_t i = 0; i < NewLocOps.size(); ++i) {
12962 NewLocOps[i].getSDNode() != &
N)
12974 DV->getAdditionalDependencies(), DV->isIndirect(),
12975 DV->getDebugLoc(), DV->getOrder(), DV->isVariadic());
12978 DV->setIsInvalidated();
12979 DV->setIsEmitted();
12981 dbgs() <<
" into " << *DbgExpression <<
'\n');
12988 assert((!Dbg->getSDNodes().empty() ||
12991 return Op.getKind() == SDDbgOperand::FRAMEIX;
12993 "Salvaged DbgValue should depend on a new SDNode");
13002 "Expected inlined-at fields to agree");
13003 return new (DbgInfo->getAlloc())
SDDbgLabel(Label,
DL, O);
13018 while (UI != UE &&
N == UI->
getUser())
13026 :
SelectionDAG::DAGUpdateListener(d), UI(ui), UE(ue) {}
13039 "Cannot replace with this method!");
13040 assert(From != To.
getNode() &&
"Cannot replace uses of with self");
13055 RAUWUpdateListener Listener(*
this, UI, UE);
13060 RemoveNodeFromCSEMaps(
User);
13075 AddModifiedNodeToCSEMaps(
User);
13091 for (
unsigned i = 0, e = From->
getNumValues(); i != e; ++i)
13094 "Cannot use this version of ReplaceAllUsesWith!");
13102 for (
unsigned i = 0, e = From->
getNumValues(); i != e; ++i)
13104 assert((i < To->getNumValues()) &&
"Invalid To location");
13113 RAUWUpdateListener Listener(*
this, UI, UE);
13118 RemoveNodeFromCSEMaps(
User);
13134 AddModifiedNodeToCSEMaps(
User);
13151 for (
unsigned i = 0, e = From->
getNumValues(); i != e; ++i) {
13161 RAUWUpdateListener Listener(*
this, UI, UE);
13166 RemoveNodeFromCSEMaps(
User);
13172 bool To_IsDivergent =
false;
13187 AddModifiedNodeToCSEMaps(
User);
13200 if (From == To)
return;
13216 RAUWUpdateListener Listener(*
this, UI, UE);
13219 bool UserRemovedFromCSEMaps =
false;
13236 if (!UserRemovedFromCSEMaps) {
13237 RemoveNodeFromCSEMaps(
User);
13238 UserRemovedFromCSEMaps =
true;
13248 if (!UserRemovedFromCSEMaps)
13253 AddModifiedNodeToCSEMaps(
User);
13272bool operator<(
const UseMemo &L,
const UseMemo &R) {
13273 return (intptr_t)L.User < (intptr_t)R.User;
13280 SmallVectorImpl<UseMemo> &
Uses;
13282 void NodeDeleted(SDNode *
N, SDNode *
E)
override {
13283 for (UseMemo &Memo :
Uses)
13284 if (Memo.User ==
N)
13285 Memo.User =
nullptr;
13289 RAUOVWUpdateListener(SelectionDAG &d, SmallVectorImpl<UseMemo> &uses)
13290 : SelectionDAG::DAGUpdateListener(d),
Uses(uses) {}
13297 switch (
Node->getOpcode()) {
13309 if (TLI->isSDNodeAlwaysUniform(
N)) {
13310 assert(!TLI->isSDNodeSourceOfDivergence(
N, FLI, UA) &&
13311 "Conflicting divergence information!");
13314 if (TLI->isSDNodeSourceOfDivergence(
N, FLI, UA))
13316 for (
const auto &
Op :
N->ops()) {
13317 EVT VT =
Op.getValueType();
13320 if (VT != MVT::Other &&
Op.getNode()->isDivergent() &&
13332 if (
N->SDNodeBits.IsDivergent != IsDivergent) {
13333 N->SDNodeBits.IsDivergent = IsDivergent;
13336 }
while (!Worklist.
empty());
13339void SelectionDAG::CreateTopologicalOrder(std::vector<SDNode *> &Order) {
13341 Order.reserve(AllNodes.size());
13343 unsigned NOps =
N.getNumOperands();
13346 Order.push_back(&
N);
13348 for (
size_t I = 0;
I != Order.size(); ++
I) {
13350 for (
auto *U :
N->users()) {
13351 unsigned &UnsortedOps = Degree[U];
13352 if (0 == --UnsortedOps)
13353 Order.push_back(U);
13358#if !defined(NDEBUG) && LLVM_ENABLE_ABI_BREAKING_CHECKS
13359void SelectionDAG::VerifyDAGDivergence() {
13360 std::vector<SDNode *> TopoOrder;
13361 CreateTopologicalOrder(TopoOrder);
13362 for (
auto *
N : TopoOrder) {
13364 "Divergence bit inconsistency detected");
13387 for (
unsigned i = 0; i != Num; ++i) {
13388 unsigned FromResNo = From[i].
getResNo();
13391 if (
Use.getResNo() == FromResNo) {
13393 Uses.push_back(Memo);
13400 RAUOVWUpdateListener Listener(*
this,
Uses);
13402 for (
unsigned UseIndex = 0, UseIndexEnd =
Uses.size();
13403 UseIndex != UseIndexEnd; ) {
13409 if (
User ==
nullptr) {
13415 RemoveNodeFromCSEMaps(
User);
13422 unsigned i =
Uses[UseIndex].Index;
13427 }
while (UseIndex != UseIndexEnd &&
Uses[UseIndex].
User ==
User);
13431 AddModifiedNodeToCSEMaps(
User);
13439 unsigned DAGSize = 0;
13455 unsigned Degree =
N.getNumOperands();
13458 N.setNodeId(DAGSize++);
13460 if (Q != SortedPos)
13461 SortedPos = AllNodes.insert(SortedPos, AllNodes.remove(Q));
13462 assert(SortedPos != AllNodes.end() &&
"Overran node list");
13466 N.setNodeId(Degree);
13478 unsigned Degree =
P->getNodeId();
13479 assert(Degree != 0 &&
"Invalid node degree");
13483 P->setNodeId(DAGSize++);
13484 if (
P->getIterator() != SortedPos)
13485 SortedPos = AllNodes.insert(SortedPos, AllNodes.remove(
P));
13486 assert(SortedPos != AllNodes.end() &&
"Overran node list");
13490 P->setNodeId(Degree);
13493 if (
Node.getIterator() == SortedPos) {
13497 dbgs() <<
"Overran sorted position:\n";
13499 dbgs() <<
"Checking if this is due to cycles\n";
13506 assert(SortedPos == AllNodes.end() &&
13507 "Topological sort incomplete!");
13509 "First node in topological sort is not the entry token!");
13510 assert(AllNodes.front().getNodeId() == 0 &&
13511 "First node in topological sort has non-zero id!");
13512 assert(AllNodes.front().getNumOperands() == 0 &&
13513 "First node in topological sort has operands!");
13514 assert(AllNodes.back().getNodeId() == (
int)DAGSize-1 &&
13515 "Last node in topologic sort has unexpected id!");
13516 assert(AllNodes.back().use_empty() &&
13517 "Last node in topologic sort has users!");
13524 SortedNodes.
clear();
13531 unsigned NumOperands =
N.getNumOperands();
13532 if (NumOperands == 0)
13536 RemainingOperands[&
N] = NumOperands;
13541 for (
unsigned i = 0U; i < SortedNodes.
size(); ++i) {
13542 const SDNode *
N = SortedNodes[i];
13543 for (
const SDNode *U :
N->users()) {
13548 unsigned &NumRemOperands = RemainingOperands[U];
13549 assert(NumRemOperands &&
"Invalid number of remaining operands");
13551 if (!NumRemOperands)
13556 assert(SortedNodes.
size() == AllNodes.size() &&
"Node count mismatch");
13558 "First node in topological sort is not the entry token");
13559 assert(SortedNodes.
front()->getNumOperands() == 0 &&
13560 "First node in topological sort has operands");
13566 for (
SDNode *SD : DB->getSDNodes()) {
13569 assert(DbgInfo->getSDDbgValues(SD).empty() || SD->getHasDebugValue());
13570 SD->setHasDebugValue(
true);
13572 DbgInfo->add(DB, isParameter);
13585 if (OldChain == NewMemOpChain || OldChain.
use_empty())
13586 return NewMemOpChain;
13589 OldChain, NewMemOpChain);
13592 return TokenFactor;
13611 if (OutFunction !=
nullptr)
13619 std::string ErrorStr;
13621 ErrorFormatter <<
"Undefined external symbol ";
13622 ErrorFormatter <<
'"' << Symbol <<
'"';
13632 return Const !=
nullptr && Const->isZero();
13641 return Const !=
nullptr && Const->isZero() && !Const->isNegative();
13646 return Const !=
nullptr && Const->isAllOnes();
13651 return Const !=
nullptr && Const->isOne();
13656 return Const !=
nullptr && Const->isMinSignedValue();
13660 SDValue V,
unsigned OperandNo,
13661 unsigned Depth)
const {
13668 unsigned OperandNo,
unsigned Depth)
const {
13671 if (V.getValueType().isInteger()) {
13680 return Const.isZero();
13682 return Const.isOne();
13685 return Const.isAllOnes();
13687 return Const.isMinSignedValue();
13689 return Const.isMaxSignedValue();
13694 return OperandNo == 1 && Const.isZero();
13697 return OperandNo == 1 && Const.isOne();
13703 return ConstFP->isZero() &&
13704 (Flags.hasNoSignedZeros() || ConstFP->isNegative());
13706 return OperandNo == 1 && ConstFP->isZero() &&
13707 (Flags.hasNoSignedZeros() || !ConstFP->isNegative());
13709 return ConstFP->isOne();
13711 return OperandNo == 1 && ConstFP->isOne();
13715 EVT VT = V.getValueType();
13723 return ConstFP->isExactlyValue(NeutralAF);
13737 while (V.getOpcode() ==
ISD::BITCAST && V.getOperand(0).hasOneUse())
13756 !DemandedElts[IndexC->getZExtValue()]) {
13775 unsigned NumBits = V.getScalarValueSizeInBits();
13778 return C && (
C->getAPIntValue().
countr_one() >= NumBits);
13782 bool AllowTruncation) {
13789 bool AllowTruncation) {
13796 EVT VecEltVT =
N->getValueType(0).getVectorElementType();
13798 EVT CVT = CN->getValueType(0);
13799 assert(CVT.
bitsGE(VecEltVT) &&
"Illegal splat_vector element extension");
13800 if (AllowTruncation || CVT == VecEltVT)
13807 ConstantSDNode *CN = BV->getConstantSplatNode(DemandedElts, &UndefElements);
13812 if (CN && (UndefElements.
none() || AllowUndefs)) {
13814 EVT NSVT =
N.getValueType().getScalarType();
13815 assert(CVT.
bitsGE(NSVT) &&
"Illegal build vector element extension");
13816 if (AllowTruncation || (CVT == NSVT))
13830 const APInt &DemandedElts,
13831 bool AllowUndefs) {
13838 BV->getConstantFPSplatNode(DemandedElts, &UndefElements);
13840 if (CN && (UndefElements.
none() || AllowUndefs))
13855 return C &&
C->isZero();
13861 return C &&
C->isOne();
13866 return C &&
C->isOne();
13871 unsigned BitWidth =
N.getScalarValueSizeInBits();
13874 return C &&
C->getAPIntValue().countTrailingOnes() >=
BitWidth;
13880 APInt(
C->getAPIntValue().getBitWidth(), 1));
13886 return C &&
C->isZero();
13891 return C &&
C->isZero();
13902 bool IsVolatile =
false;
13903 bool IsNonTemporal =
false;
13904 bool IsDereferenceable =
true;
13905 bool IsInvariant =
true;
13907 IsVolatile |= MMO->isVolatile();
13908 IsNonTemporal |= MMO->isNonTemporal();
13909 IsDereferenceable &= MMO->isDereferenceable();
13910 IsInvariant &= MMO->isInvariant();
13936 std::vector<EVT> VTs;
13949const EVT *SDNode::getValueTypeList(
MVT VT) {
13950 static EVTArray SimpleVTArray;
13953 return &SimpleVTArray.VTs[VT.
SimpleTy];
13962 if (U.getResNo() ==
Value)
14000 return any_of(
N->op_values(),
14001 [
this](
SDValue Op) { return this == Op.getNode(); });
14015 unsigned Depth)
const {
14016 if (*
this == Dest)
return true;
14020 if (
Depth == 0)
return false;
14040 return Op.reachesChainWithoutSideEffects(Dest, Depth - 1);
14046 if (Ld->isUnordered())
14047 return Ld->getChain().reachesChainWithoutSideEffects(Dest,
Depth-1);
14060 this->Flags &= Flags;
14066 bool AllowPartials) {
14081 unsigned CandidateBinOp =
Op.getOpcode();
14082 if (
Op.getValueType().isFloatingPoint()) {
14084 switch (CandidateBinOp) {
14086 if (!Flags.hasNoSignedZeros() || !Flags.hasAllowReassociation())
14096 auto PartialReduction = [&](
SDValue Op,
unsigned NumSubElts) {
14097 if (!AllowPartials || !
Op)
14099 EVT OpVT =
Op.getValueType();
14102 if (!TLI->isExtractSubvectorCheap(SubVT, OpVT, 0))
14121 unsigned Stages =
Log2_32(
Op.getValueType().getVectorNumElements());
14123 for (
unsigned i = 0; i < Stages; ++i) {
14124 unsigned MaskEnd = (1 << i);
14126 if (
Op.getOpcode() != CandidateBinOp)
14127 return PartialReduction(PrevOp, MaskEnd);
14143 return PartialReduction(PrevOp, MaskEnd);
14146 for (
int Index = 0; Index < (int)MaskEnd; ++Index)
14147 if (Shuffle->
getMaskElt(Index) != (
int)(MaskEnd + Index))
14148 return PartialReduction(PrevOp, MaskEnd);
14155 while (
Op.getOpcode() == CandidateBinOp) {
14156 unsigned NumElts =
Op.getValueType().getVectorNumElements();
14165 if (NumSrcElts != (2 * NumElts))
14180 EVT VT =
N->getValueType(0);
14189 else if (NE > ResNE)
14192 if (
N->getNumValues() == 2) {
14195 EVT VT1 =
N->getValueType(1);
14199 for (i = 0; i != NE; ++i) {
14200 for (
unsigned j = 0, e =
N->getNumOperands(); j != e; ++j) {
14201 SDValue Operand =
N->getOperand(j);
14209 SDValue EltOp =
getNode(
N->getOpcode(), dl, {EltVT, EltVT1}, Operands);
14214 for (; i < ResNE; ++i) {
14226 assert(
N->getNumValues() == 1 &&
14227 "Can't unroll a vector with multiple results!");
14233 for (i= 0; i != NE; ++i) {
14234 for (
unsigned j = 0, e =
N->getNumOperands(); j != e; ++j) {
14235 SDValue Operand =
N->getOperand(j);
14243 Operands[j] = Operand;
14247 switch (
N->getOpcode()) {
14275 ASC->getSrcAddressSpace(),
14276 ASC->getDestAddressSpace()));
14282 for (; i < ResNE; ++i)
14291 unsigned Opcode =
N->getOpcode();
14295 "Expected an overflow opcode");
14297 EVT ResVT =
N->getValueType(0);
14298 EVT OvVT =
N->getValueType(1);
14307 else if (NE > ResNE)
14319 for (
unsigned i = 0; i < NE; ++i) {
14320 SDValue Res =
getNode(Opcode, dl, VTs, LHSScalars[i], RHSScalars[i]);
14343 if (LD->isVolatile() ||
Base->isVolatile())
14346 if (!LD->isSimple())
14348 if (LD->isIndexed() ||
Base->isIndexed())
14350 if (LD->getChain() !=
Base->getChain())
14352 EVT VT = LD->getMemoryVT();
14360 if (BaseLocDecomp.equalBaseIndex(LocDecomp, *
this,
Offset))
14361 return (Dist * (int64_t)Bytes ==
Offset);
14370 int64_t GVOffset = 0;
14371 if (TLI->isGAPlusOffset(Ptr.
getNode(), GV, GVOffset)) {
14382 int FrameIdx = INT_MIN;
14383 int64_t FrameOffset = 0;
14385 FrameIdx = FI->getIndex();
14393 if (FrameIdx != INT_MIN) {
14398 return std::nullopt;
14408 "Split node must be a scalar type");
14413 return std::make_pair(
Lo,
Hi);
14422 LoVT = HiVT = TLI->getTypeToTransformTo(*
getContext(), VT);
14426 return std::make_pair(LoVT, HiVT);
14434 bool *HiIsEmpty)
const {
14444 "Mixing fixed width and scalable vectors when enveloping a type");
14449 *HiIsEmpty =
false;
14457 return std::make_pair(LoVT, HiVT);
14462std::pair<SDValue, SDValue>
14467 "Splitting vector with an invalid mixture of fixed and scalable "
14470 N.getValueType().getVectorMinNumElements() &&
14471 "More vector elements requested than available!");
14479 return std::make_pair(
Lo,
Hi);
14486 EVT VT =
N.getValueType();
14488 "Expecting the mask to be an evenly-sized vector");
14493 return std::make_pair(
Lo,
Hi);
14498 EVT VT =
N.getValueType();
14506 unsigned Start,
unsigned Count,
14508 EVT VT =
Op.getValueType();
14511 if (EltVT ==
EVT())
14514 for (
unsigned i = Start, e = Start +
Count; i != e; ++i) {
14526 return Val.MachineCPVal->getType();
14527 return Val.ConstVal->getType();
14531 unsigned &SplatBitSize,
14532 bool &HasAnyUndefs,
14533 unsigned MinSplatBits,
14534 bool IsBigEndian)
const {
14538 if (MinSplatBits > VecWidth)
14543 SplatValue =
APInt(VecWidth, 0);
14544 SplatUndef =
APInt(VecWidth, 0);
14551 assert(
NumOps > 0 &&
"isConstantSplat has 0-size build vector");
14554 for (
unsigned j = 0; j <
NumOps; ++j) {
14555 unsigned i = IsBigEndian ?
NumOps - 1 - j : j;
14557 unsigned BitPos = j * EltWidth;
14560 SplatUndef.
setBits(BitPos, BitPos + EltWidth);
14562 SplatValue.
insertBits(CN->getAPIntValue().zextOrTrunc(EltWidth), BitPos);
14564 SplatValue.
insertBits(CN->getValueAPF().bitcastToAPInt(), BitPos);
14571 HasAnyUndefs = (SplatUndef != 0);
14574 while (VecWidth > 8) {
14579 unsigned HalfSize = VecWidth / 2;
14586 if ((HighValue & ~LowUndef) != (LowValue & ~HighUndef) ||
14587 MinSplatBits > HalfSize)
14590 SplatValue = HighValue | LowValue;
14591 SplatUndef = HighUndef & LowUndef;
14593 VecWidth = HalfSize;
14602 SplatBitSize = VecWidth;
14609 if (UndefElements) {
14610 UndefElements->
clear();
14617 for (
unsigned i = 0; i !=
NumOps; ++i) {
14618 if (!DemandedElts[i])
14621 if (
Op.isUndef()) {
14623 (*UndefElements)[i] =
true;
14624 }
else if (!Splatted) {
14626 }
else if (Splatted !=
Op) {
14632 unsigned FirstDemandedIdx = DemandedElts.
countr_zero();
14634 "Can only have a splat without a constant for all undefs.");
14651 if (UndefElements) {
14652 UndefElements->
clear();
14663 (*UndefElements)[
I] =
true;
14666 for (
unsigned SeqLen = 1; SeqLen <
NumOps; SeqLen *= 2) {
14667 Sequence.append(SeqLen,
SDValue());
14668 for (
unsigned I = 0;
I !=
NumOps; ++
I) {
14669 if (!DemandedElts[
I])
14671 SDValue &SeqOp = Sequence[
I % SeqLen];
14673 if (
Op.isUndef()) {
14678 if (SeqOp && !SeqOp.
isUndef() && SeqOp !=
Op) {
14684 if (!Sequence.empty())
14688 assert(Sequence.empty() &&
"Failed to empty non-repeating sequence pattern");
14729 const APFloat &APF = CN->getValueAPF();
14735 return IntVal.exactLogBase2();
14741 bool IsLittleEndian,
unsigned DstEltSizeInBits,
14749 assert(((NumSrcOps * SrcEltSizeInBits) % DstEltSizeInBits) == 0 &&
14750 "Invalid bitcast scale");
14755 BitVector SrcUndeElements(NumSrcOps,
false);
14757 for (
unsigned I = 0;
I != NumSrcOps; ++
I) {
14759 if (
Op.isUndef()) {
14760 SrcUndeElements.
set(
I);
14765 assert((CInt || CFP) &&
"Unknown constant");
14766 SrcBitElements[
I] = CInt ? CInt->getAPIntValue().trunc(SrcEltSizeInBits)
14767 : CFP->getValueAPF().bitcastToAPInt();
14771 recastRawBits(IsLittleEndian, DstEltSizeInBits, RawBitElements,
14772 SrcBitElements, UndefElements, SrcUndeElements);
14777 unsigned DstEltSizeInBits,
14782 unsigned NumSrcOps = SrcBitElements.
size();
14783 unsigned SrcEltSizeInBits = SrcBitElements[0].getBitWidth();
14784 assert(((NumSrcOps * SrcEltSizeInBits) % DstEltSizeInBits) == 0 &&
14785 "Invalid bitcast scale");
14786 assert(NumSrcOps == SrcUndefElements.
size() &&
14787 "Vector size mismatch");
14789 unsigned NumDstOps = (NumSrcOps * SrcEltSizeInBits) / DstEltSizeInBits;
14790 DstUndefElements.
clear();
14791 DstUndefElements.
resize(NumDstOps,
false);
14795 if (SrcEltSizeInBits <= DstEltSizeInBits) {
14796 unsigned Scale = DstEltSizeInBits / SrcEltSizeInBits;
14797 for (
unsigned I = 0;
I != NumDstOps; ++
I) {
14798 DstUndefElements.
set(
I);
14799 APInt &DstBits = DstBitElements[
I];
14800 for (
unsigned J = 0; J != Scale; ++J) {
14801 unsigned Idx = (
I * Scale) + (IsLittleEndian ? J : (Scale - J - 1));
14802 if (SrcUndefElements[Idx])
14804 DstUndefElements.
reset(
I);
14805 const APInt &SrcBits = SrcBitElements[Idx];
14807 "Illegal constant bitwidths");
14808 DstBits.
insertBits(SrcBits, J * SrcEltSizeInBits);
14815 unsigned Scale = SrcEltSizeInBits / DstEltSizeInBits;
14816 for (
unsigned I = 0;
I != NumSrcOps; ++
I) {
14817 if (SrcUndefElements[
I]) {
14818 DstUndefElements.
set(
I * Scale, (
I + 1) * Scale);
14821 const APInt &SrcBits = SrcBitElements[
I];
14822 for (
unsigned J = 0; J != Scale; ++J) {
14823 unsigned Idx = (
I * Scale) + (IsLittleEndian ? J : (Scale - J - 1));
14824 APInt &DstBits = DstBitElements[Idx];
14825 DstBits = SrcBits.
extractBits(DstEltSizeInBits, J * DstEltSizeInBits);
14832 unsigned Opc =
Op.getOpcode();
14839std::optional<std::pair<APInt, APInt>>
14843 return std::nullopt;
14846 APInt Start, Stride;
14847 int FirstIdx = -1, SecondIdx = -1;
14851 for (
unsigned I = 0;
I <
NumOps; ++
I) {
14856 return std::nullopt;
14859 if (FirstIdx < 0) {
14862 }
else if (SecondIdx < 0) {
14868 unsigned IdxDiff =
I - FirstIdx;
14869 APInt ValDiff = Val - Start;
14874 return std::nullopt;
14875 IdxDiff >>= CommonPow2Bits;
14883 return std::nullopt;
14886 Start -= Stride * FirstIdx;
14889 if (Val != Start + Stride *
I)
14890 return std::nullopt;
14896 return std::nullopt;
14898 return std::make_pair(Start, Stride);
14904 for (i = 0, e = Mask.size(); i != e && Mask[i] < 0; ++i)
14914 for (
int Idx = Mask[i]; i != e; ++i)
14915 if (Mask[i] >= 0 && Mask[i] != Idx)
14923 SDValue N,
bool AllowOpaques)
const {
14927 return AllowOpaques || !
C->isOpaque();
14936 TLI->isOffsetFoldingLegal(GA))
14964 return std::nullopt;
14966 EVT VT =
N->getValueType(0);
14968 switch (TLI->getBooleanContents(
N.getValueType())) {
14974 return std::nullopt;
14980 return std::nullopt;
14988 assert(!
Node->OperandList &&
"Node already has operands");
14990 "too many operands to fit into SDNode");
14991 SDUse *
Ops = OperandRecycler.allocate(
14994 bool IsDivergent =
false;
14995 for (
unsigned I = 0;
I != Vals.
size(); ++
I) {
14997 Ops[
I].setInitial(Vals[
I]);
14998 EVT VT =
Ops[
I].getValueType();
15001 if (VT != MVT::Other &&
15004 IsDivergent =
true;
15009 if (!TLI->isSDNodeAlwaysUniform(Node)) {
15010 IsDivergent |= TLI->isSDNodeSourceOfDivergence(Node, FLI, UA);
15011 Node->SDNodeBits.IsDivergent = IsDivergent;
15019 while (Vals.
size() > Limit) {
15020 unsigned SliceIdx = Vals.
size() - Limit;
15091 "Unexpected opcode");
15112 const SDLoc &DLoc) {
15116 RTLIB::LibcallImpl LibcallImpl =
15117 Libcalls->getLibcallImpl(
static_cast<RTLIB::Libcall
>(LibFunc));
15118 if (LibcallImpl == RTLIB::Unsupported)
15125 Libcalls->getLibcallImplCallingConv(LibcallImpl),
15127 return TLI->LowerCallTo(CLI).second;
15131 assert(From && To &&
"Invalid SDNode; empty source SDValue?");
15132 auto I = SDEI.find(From);
15133 if (
I == SDEI.end())
15138 NodeExtraInfo NEI =
I->second;
15147 SDEI[To] = std::move(NEI);
15164 auto VisitFrom = [&](
auto &&Self,
const SDNode *
N,
int MaxDepth) {
15165 if (MaxDepth == 0) {
15171 if (!FromReach.
insert(
N).second)
15174 Self(Self,
Op.getNode(), MaxDepth - 1);
15179 auto DeepCopyTo = [&](
auto &&Self,
const SDNode *
N) {
15182 if (!Visited.
insert(
N).second)
15187 if (
N == To &&
Op.getNode() == EntrySDN) {
15192 if (!Self(Self,
Op.getNode()))
15196 SDEI[
N] = std::move(NEI);
15206 for (
int PrevDepth = 0, MaxDepth = 16; MaxDepth <= 1024;
15207 PrevDepth = MaxDepth, MaxDepth *= 2, Visited.
clear()) {
15212 for (
const SDNode *
N : StartFrom)
15213 VisitFrom(VisitFrom,
N, MaxDepth - PrevDepth);
15217 LLVM_DEBUG(
dbgs() << __func__ <<
": MaxDepth=" << MaxDepth <<
" too low\n");
15225 errs() <<
"warning: incomplete propagation of SelectionDAG::NodeExtraInfo\n";
15226 assert(
false &&
"From subgraph too complex - increase max. MaxDepth?");
15228 SDEI[To] = std::move(NEI);
15242 if (!Visited.
insert(
N).second) {
15243 errs() <<
"Detected cycle in SelectionDAG\n";
15244 dbgs() <<
"Offending node:\n";
15245 N->dumprFull(DAG);
dbgs() <<
"\n";
15261 bool check = force;
15262#ifdef EXPENSIVE_CHECKS
15266 assert(
N &&
"Checking nonexistent SDNode");
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static bool isConstant(const MachineInstr &MI)
This file declares a class to represent arbitrary precision floating point values and provide a varie...
This file implements a class to represent arbitrary precision integral constant values and operations...
This file implements the APSInt class, which is a simple class that represents an arbitrary sized int...
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
This file implements the BitVector class.
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
static std::optional< bool > isBigEndian(const SmallDenseMap< int64_t, int64_t, 8 > &MemOffset2Idx, int64_t LowestIdx)
Given a map from byte offsets in memory to indices in a load/store, determine if that map corresponds...
#define __asan_unpoison_memory_region(p, size)
#define LLVM_LIKELY(EXPR)
This file contains the declarations for the subclasses of Constant, which represent the different fla...
This file defines the DenseSet and SmallDenseSet classes.
This file contains constants used for implementing Dwarf debug support.
This file defines a hash set that can be used to remove duplication of nodes in a graph.
static MaybeAlign getAlign(Value *Ptr)
std::pair< Instruction::BinaryOps, Value * > OffsetOp
Find all possible pairs (BinOp, RHS) that BinOp V, RHS can be simplified.
static constexpr Value * getValue(Ty &ValueOrUse)
const size_t AbstractManglingParser< Derived, Alloc >::NumOps
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
static Register getMemsetValue(Register Val, LLT Ty, MachineIRBuilder &MIB)
static bool isZero(Value *V, const DataLayout &DL, DominatorTree *DT, AssumptionCache *AC)
static Align getPrefTypeAlign(EVT VT, SelectionDAG &DAG)
static bool isConstantSplatVector(SDValue N, APInt &SplatValue, unsigned MinSizeInBits)
This file declares the MachineConstantPool class which is an abstract constant pool to keep track of ...
Register const TargetRegisterInfo * TRI
This file provides utility analysis objects describing memory locations.
static MCRegister getReg(const MCDisassembler *D, unsigned RC, unsigned RegNo)
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
PowerPC Reduce CR logical Operation
const SmallVectorImpl< MachineOperand > & Cond
Remove Loads Into Fake Uses
static bool isValid(const char C)
Returns true if C is a valid mangled character: <0-9a-zA-Z_>.
Contains matchers for matching SelectionDAG nodes and values.
static Type * getValueType(Value *V, bool LookThroughCmp=false)
Returns the "element type" of the given value/instruction V.
static uint64_t umul_ov(uint64_t i, uint64_t j, bool &Overflow)
static bool shouldLowerMemFuncForSize(const MachineFunction &MF, SelectionDAG &DAG)
static SDValue getFixedOrScalableQuantity(SelectionDAG &DAG, const SDLoc &DL, EVT VT, Ty Quantity)
static std::pair< SDValue, SDValue > getRuntimeCallSDValueHelper(SDValue Chain, const SDLoc &dl, TargetLowering::ArgListTy &&Args, const CallInst *CI, RTLIB::Libcall Call, SelectionDAG *DAG, const TargetLowering *TLI)
static SDValue getMemsetStores(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Dst, SDValue Src, uint64_t Size, Align Alignment, bool isVol, bool AlwaysInline, MachinePointerInfo DstPtrInfo, const AAMDNodes &AAInfo)
Lower the call to 'memset' intrinsic function into a series of store operations.
static std::optional< APInt > FoldValueWithUndef(unsigned Opcode, const APInt &C1, bool IsUndef1, const APInt &C2, bool IsUndef2)
static SDValue FoldSTEP_VECTOR(const SDLoc &DL, EVT VT, SDValue Step, SelectionDAG &DAG)
static void AddNodeIDNode(FoldingSetNodeID &ID, unsigned OpC, SDVTList VTList, ArrayRef< SDValue > OpList)
static SDValue getMemsetStringVal(EVT VT, const SDLoc &dl, SelectionDAG &DAG, const TargetLowering &TLI, const ConstantDataArraySlice &Slice)
getMemsetStringVal - Similar to getMemsetValue.
static cl::opt< bool > EnableMemCpyDAGOpt("enable-memcpy-dag-opt", cl::Hidden, cl::init(true), cl::desc("Gang up loads and stores generated by inlining of memcpy"))
static bool haveNoCommonBitsSetCommutative(SDValue A, SDValue B)
static void AddNodeIDValueTypes(FoldingSetNodeID &ID, SDVTList VTList)
AddNodeIDValueTypes - Value type lists are intern'd so we can represent them solely with their pointe...
static void commuteShuffle(SDValue &N1, SDValue &N2, MutableArrayRef< int > M)
Swaps the values of N1 and N2.
static bool isMemSrcFromConstant(SDValue Src, ConstantDataArraySlice &Slice)
Returns true if memcpy source is constant data.
static SDValue getMemcpyLoadsAndStores(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Dst, SDValue Src, uint64_t Size, Align DstAlign, Align SrcAlign, bool isVol, bool AlwaysInline, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo, const AAMDNodes &AAInfo, BatchAAResults *BatchAA)
static void AddNodeIDOpcode(FoldingSetNodeID &ID, unsigned OpC)
AddNodeIDOpcode - Add the node opcode to the NodeID data.
static ISD::CondCode getSetCCInverseImpl(ISD::CondCode Op, bool isIntegerLike)
static bool doNotCSE(SDNode *N)
doNotCSE - Return true if CSE should not be performed for this node.
static cl::opt< int > MaxLdStGlue("ldstmemcpy-glue-max", cl::desc("Number limit for gluing ld/st of memcpy."), cl::Hidden, cl::init(0))
static void AddNodeIDOperands(FoldingSetNodeID &ID, ArrayRef< SDValue > Ops)
AddNodeIDOperands - Various routines for adding operands to the NodeID data.
static SDValue foldCONCAT_VECTORS(const SDLoc &DL, EVT VT, ArrayRef< SDValue > Ops, SelectionDAG &DAG)
Try to simplify vector concatenation to an input value, undef, or build vector.
static MachinePointerInfo InferPointerInfo(const MachinePointerInfo &Info, SelectionDAG &DAG, SDValue Ptr, int64_t Offset=0)
InferPointerInfo - If the specified ptr/offset is a frame index, infer a MachinePointerInfo record fr...
static bool isInTailCallPositionWrapper(const CallInst *CI, const SelectionDAG *SelDAG, bool AllowReturnsFirstArg)
static void AddNodeIDCustom(FoldingSetNodeID &ID, const SDNode *N)
If this is an SDNode with special info, add this info to the NodeID data.
static bool gluePropagatesDivergence(const SDNode *Node)
Return true if a glue output should propagate divergence information.
static void NewSDValueDbgMsg(SDValue V, StringRef Msg, SelectionDAG *G)
static SDVTList makeVTList(const EVT *VTs, unsigned NumVTs)
makeVTList - Return an instance of the SDVTList struct initialized with the specified members.
static void checkForCyclesHelper(const SDNode *N, SmallPtrSetImpl< const SDNode * > &Visited, SmallPtrSetImpl< const SDNode * > &Checked, const llvm::SelectionDAG *DAG)
static void chainLoadsAndStoresForMemcpy(SelectionDAG &DAG, const SDLoc &dl, SmallVector< SDValue, 32 > &OutChains, unsigned From, unsigned To, SmallVector< SDValue, 16 > &OutLoadChains, SmallVector< SDValue, 16 > &OutStoreChains)
static int isSignedOp(ISD::CondCode Opcode)
For an integer comparison, return 1 if the comparison is a signed operation and 2 if the result is an...
static std::optional< APInt > FoldValue(unsigned Opcode, const APInt &C1, const APInt &C2)
static SDValue FoldBUILD_VECTOR(const SDLoc &DL, EVT VT, ArrayRef< SDValue > Ops, SelectionDAG &DAG)
static void checkAddrSpaceIsValidForLibcall(const TargetLowering *TLI, unsigned AS)
static cl::opt< unsigned > MaxSteps("has-predecessor-max-steps", cl::Hidden, cl::init(8192), cl::desc("DAG combiner limit number of steps when searching DAG " "for predecessor nodes"))
static APInt getDemandAllEltsMask(SDValue V)
Construct a DemandedElts mask which demands all elements of V.
static SDValue getMemmoveLoadsAndStores(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Dst, SDValue Src, uint64_t Size, Align DstAlign, Align SrcAlign, bool isVol, bool AlwaysInline, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo, const AAMDNodes &AAInfo)
This file defines the SmallPtrSet class.
This file defines the SmallVector class.
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static SymbolRef::Type getType(const Symbol *Sym)
This file describes how to lower LLVM code to machine code.
static void removeOperands(MachineInstr &MI, unsigned i)
static OverflowResult mapOverflowResult(ConstantRange::OverflowResult OR)
Convert ConstantRange OverflowResult into ValueTracking OverflowResult.
static int Lookup(ArrayRef< TableEntry > Table, unsigned Opcode)
static unsigned getSize(unsigned Kind)
static const fltSemantics & IEEEsingle()
cmpResult
IEEE-754R 5.11: Floating Point Comparison Relations.
static constexpr roundingMode rmTowardZero
static const fltSemantics & BFloat()
static const fltSemantics & IEEEquad()
static const fltSemantics & IEEEdouble()
static constexpr roundingMode rmTowardNegative
static constexpr roundingMode rmNearestTiesToEven
static constexpr roundingMode rmTowardPositive
static const fltSemantics & IEEEhalf()
opStatus
IEEE-754R 7: Default exception handling.
static APFloat getQNaN(const fltSemantics &Sem, bool Negative=false, const APInt *payload=nullptr)
Factory for QNaN values.
opStatus divide(const APFloat &RHS, roundingMode RM)
void copySign(const APFloat &RHS)
LLVM_ABI opStatus convert(const fltSemantics &ToSemantics, roundingMode RM, bool *losesInfo)
opStatus subtract(const APFloat &RHS, roundingMode RM)
opStatus add(const APFloat &RHS, roundingMode RM)
opStatus convertFromAPInt(const APInt &Input, bool IsSigned, roundingMode RM)
opStatus multiply(const APFloat &RHS, roundingMode RM)
LLVM_READONLY bool isOne() const
static APFloat getLargest(const fltSemantics &Sem, bool Negative=false)
Returns the largest finite number in the given semantics.
opStatus convertToInteger(MutableArrayRef< integerPart > Input, unsigned int Width, bool IsSigned, roundingMode RM, bool *IsExact) const
static APFloat getInf(const fltSemantics &Sem, bool Negative=false)
Factory for Positive and Negative Infinity.
opStatus mod(const APFloat &RHS)
static APFloat getNaN(const fltSemantics &Sem, bool Negative=false, uint64_t payload=0)
Factory for NaN values.
Class for arbitrary precision integers.
LLVM_ABI APInt umul_ov(const APInt &RHS, bool &Overflow) const
LLVM_ABI APInt usub_sat(const APInt &RHS) const
LLVM_ABI APInt udiv(const APInt &RHS) const
Unsigned division operation.
static APInt getAllOnes(unsigned numBits)
Return an APInt of a specified width with all bits set.
void clearBit(unsigned BitPosition)
Set a given bit to 0.
LLVM_ABI APInt zext(unsigned width) const
Zero extend to a new width.
static APInt getSignMask(unsigned BitWidth)
Get the SignMask for a specific bit width.
bool isMinSignedValue() const
Determine if this is the smallest signed value.
uint64_t getZExtValue() const
Get zero extended value.
void setHighBits(unsigned hiBits)
Set the top hiBits bits.
unsigned popcount() const
Count the number of bits set.
void setBitsFrom(unsigned loBit)
Set the top bits starting from loBit.
LLVM_ABI APInt getHiBits(unsigned numBits) const
Compute an APInt containing numBits highbits from this APInt.
LLVM_ABI APInt zextOrTrunc(unsigned width) const
Zero extend or truncate to width.
unsigned getActiveBits() const
Compute the number of active bits in the value.
LLVM_ABI APInt trunc(unsigned width) const
Truncate to new width.
void setBit(unsigned BitPosition)
Set the given bit to 1 whose position is given as "bitPosition".
APInt abs() const
Get the absolute value.
LLVM_ABI APInt sadd_sat(const APInt &RHS) const
bool isAllOnes() const
Determine if all bits are set. This is true for zero-width values.
bool ugt(const APInt &RHS) const
Unsigned greater than comparison.
static APInt getBitsSet(unsigned numBits, unsigned loBit, unsigned hiBit)
Get a value with a block of bits set.
bool isZero() const
Determine if this value is zero, i.e. all bits are clear.
LLVM_ABI APInt urem(const APInt &RHS) const
Unsigned remainder operation.
unsigned getBitWidth() const
Return the number of bits in the APInt.
bool ult(const APInt &RHS) const
Unsigned less than comparison.
static APInt getSignedMaxValue(unsigned numBits)
Gets maximum signed value of APInt for a specific bit width.
bool isNegative() const
Determine sign of this APInt.
LLVM_ABI APInt sdiv(const APInt &RHS) const
Signed division function for APInt.
void clearAllBits()
Set every bit to 0.
LLVM_ABI APInt rotr(unsigned rotateAmt) const
Rotate right by rotateAmt.
LLVM_ABI APInt reverseBits() const
void ashrInPlace(unsigned ShiftAmt)
Arithmetic right-shift this APInt by ShiftAmt in place.
bool sle(const APInt &RHS) const
Signed less or equal comparison.
unsigned countr_zero() const
Count the number of trailing zero bits.
unsigned getNumSignBits() const
Computes the number of leading bits of this APInt that are equal to its sign bit.
unsigned countl_zero() const
The APInt version of std::countl_zero.
static LLVM_ABI APInt getSplat(unsigned NewLen, const APInt &V)
Return a value containing V broadcasted over NewLen bits.
static APInt getSignedMinValue(unsigned numBits)
Gets minimum signed value of APInt for a specific bit width.
LLVM_ABI APInt sshl_sat(const APInt &RHS) const
LLVM_ABI APInt ushl_sat(const APInt &RHS) const
LLVM_ABI APInt sextOrTrunc(unsigned width) const
Sign extend or truncate to width.
static bool isSameValue(const APInt &I1, const APInt &I2, bool SignedCompare=false)
Determine if two APInts have the same value, after zero-extending or sign-extending (if SignedCompare...
LLVM_ABI APInt rotl(unsigned rotateAmt) const
Rotate left by rotateAmt.
LLVM_ABI void insertBits(const APInt &SubBits, unsigned bitPosition)
Insert the bits from a smaller APInt starting at bitPosition.
void clearLowBits(unsigned loBits)
Set bottom loBits bits to 0.
unsigned logBase2() const
LLVM_ABI APInt uadd_sat(const APInt &RHS) const
APInt ashr(unsigned ShiftAmt) const
Arithmetic right-shift function.
LLVM_ABI APInt multiplicativeInverse() const
LLVM_ABI APInt srem(const APInt &RHS) const
Function for signed remainder operation.
bool isNonNegative() const
Determine if this APInt Value is non-negative (>= 0)
bool ule(const APInt &RHS) const
Unsigned less or equal comparison.
LLVM_ABI APInt sext(unsigned width) const
Sign extend to a new width.
void setBits(unsigned loBit, unsigned hiBit)
Set the bits from loBit (inclusive) to hiBit (exclusive) to 1.
APInt shl(unsigned shiftAmt) const
Left-shift function.
LLVM_ABI APInt byteSwap() const
bool isSubsetOf(const APInt &RHS) const
This operation checks that all bits set in this APInt are also set in RHS.
bool isPowerOf2() const
Check if this APInt's value is a power of two greater than zero.
static APInt getLowBitsSet(unsigned numBits, unsigned loBitsSet)
Constructs an APInt value that has the bottom loBitsSet bits set.
void clearBits(unsigned LoBit, unsigned HiBit)
Clear the bits from LoBit (inclusive) to HiBit (exclusive) to 0.
static APInt getZero(unsigned numBits)
Get the '0' value for the specified bit-width.
void setLowBits(unsigned loBits)
Set the bottom loBits bits.
LLVM_ABI APInt extractBits(unsigned numBits, unsigned bitPosition) const
Return an APInt with the extracted bits [bitPosition,bitPosition+numBits).
bool sge(const APInt &RHS) const
Signed greater or equal comparison.
bool isOne() const
Determine if this is a value of 1.
static APInt getBitsSetFrom(unsigned numBits, unsigned loBit)
Constructs an APInt value that has a contiguous range of bits set.
static APInt getOneBitSet(unsigned numBits, unsigned BitNo)
Return an APInt with exactly one bit set in the result.
void lshrInPlace(unsigned ShiftAmt)
Logical right-shift this APInt by ShiftAmt in place.
APInt lshr(unsigned shiftAmt) const
Logical right-shift function.
bool uge(const APInt &RHS) const
Unsigned greater or equal comparison.
LLVM_ABI APInt ssub_sat(const APInt &RHS) const
An arbitrary precision integer that knows its signedness.
unsigned getSrcAddressSpace() const
unsigned getDestAddressSpace() const
static Capacity get(size_t N)
Get the capacity of an array that can hold at least N elements.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
Get the array size.
bool empty() const
Check if the array is empty.
This is an SDNode representing atomic operations.
static LLVM_ABI BaseIndexOffset match(const SDNode *N, const SelectionDAG &DAG)
Parses tree in N for base, index, offset addresses.
This class is a wrapper over an AAResults, and it is intended to be used only when there are no IR ch...
bool pointsToConstantMemory(const MemoryLocation &Loc, bool OrLocal=false)
BitVector & reset()
Reset all bits in the bitvector.
void resize(unsigned N, bool t=false)
Grow or shrink the bitvector.
void clear()
Removes all bits from the bitvector.
BitVector & set()
Set all bits in the bitvector.
bool none() const
Returns true if none of the bits are set.
size_type size() const
Returns the number of bits in this bitvector.
int64_t getOffset() const
unsigned getTargetFlags() const
const BlockAddress * getBlockAddress() const
The address of a basic block.
BlockFrequencyInfo pass uses BlockFrequencyInfoImpl implementation to estimate IR basic block frequen...
A "pseudo-class" with methods for operating on BUILD_VECTORs.
LLVM_ABI bool getConstantRawBits(bool IsLittleEndian, unsigned DstEltSizeInBits, SmallVectorImpl< APInt > &RawBitElements, BitVector &UndefElements) const
Extract the raw bit data from a build vector of Undef, Constant or ConstantFP node elements.
static LLVM_ABI void recastRawBits(bool IsLittleEndian, unsigned DstEltSizeInBits, SmallVectorImpl< APInt > &DstBitElements, ArrayRef< APInt > SrcBitElements, BitVector &DstUndefElements, const BitVector &SrcUndefElements)
Recast bit data SrcBitElements to DstEltSizeInBits wide elements.
LLVM_ABI bool getRepeatedSequence(const APInt &DemandedElts, SmallVectorImpl< SDValue > &Sequence, BitVector *UndefElements=nullptr) const
Find the shortest repeating sequence of values in the build vector.
LLVM_ABI ConstantFPSDNode * getConstantFPSplatNode(const APInt &DemandedElts, BitVector *UndefElements=nullptr) const
Returns the demanded splatted constant FP or null if this is not a constant FP splat.
LLVM_ABI SDValue getSplatValue(const APInt &DemandedElts, BitVector *UndefElements=nullptr) const
Returns the demanded splatted value or a null value if this is not a splat.
LLVM_ABI bool isConstantSplat(APInt &SplatValue, APInt &SplatUndef, unsigned &SplatBitSize, bool &HasAnyUndefs, unsigned MinSplatBits=0, bool isBigEndian=false) const
Check if this is a constant splat, and if so, find the smallest element size that splats the vector.
LLVM_ABI ConstantSDNode * getConstantSplatNode(const APInt &DemandedElts, BitVector *UndefElements=nullptr) const
Returns the demanded splatted constant or null if this is not a constant splat.
LLVM_ABI int32_t getConstantFPSplatPow2ToLog2Int(BitVector *UndefElements, uint32_t BitWidth) const
If this is a constant FP splat and the splatted constant FP is an exact power or 2,...
LLVM_ABI std::optional< std::pair< APInt, APInt > > isArithmeticSequence() const
If this BuildVector is constant and represents an arithmetic sequence "<a, a+n, a+2n,...
LLVM_ABI bool isConstant() const
This class represents a function call, abstracting a target machine's calling convention.
static LLVM_ABI bool isValueValidForType(EVT VT, const APFloat &Val)
const APFloat & getValueAPF() const
bool isExactlyValue(double V) const
We don't rely on operator== working on double values, as it returns true for things that are clearly ...
ConstantFP - Floating Point Values [float, double].
const APFloat & getValue() const
This is the shared class of boolean and integer constants.
unsigned getBitWidth() const
getBitWidth - Return the scalar bitwidth of this constant.
const APInt & getValue() const
Return the constant as an APInt value reference.
MachineConstantPoolValue * getMachineCPVal() const
bool isMachineConstantPoolEntry() const
const Constant * getConstVal() const
LLVM_ABI Type * getType() const
unsigned getTargetFlags() const
This class represents a range of values.
PreferredRangeType
If represented precisely, the result of some range operations may consist of multiple disjoint ranges...
const APInt * getSingleElement() const
If this set contains a single element, return it, otherwise return null.
static LLVM_ABI ConstantRange fromKnownBits(const KnownBits &Known, bool IsSigned)
Initialize a range based on a known bits constraint.
LLVM_ABI OverflowResult unsignedSubMayOverflow(const ConstantRange &Other) const
Return whether unsigned sub of the two ranges always/never overflows.
LLVM_ABI OverflowResult unsignedAddMayOverflow(const ConstantRange &Other) const
Return whether unsigned add of the two ranges always/never overflows.
LLVM_ABI KnownBits toKnownBits() const
Return known bits for values in this range.
LLVM_ABI ConstantRange zeroExtend(uint32_t BitWidth) const
Return a new range in the specified integer type, which must be strictly larger than the current type...
LLVM_ABI APInt getSignedMin() const
Return the smallest signed value contained in the ConstantRange.
LLVM_ABI OverflowResult unsignedMulMayOverflow(const ConstantRange &Other) const
Return whether unsigned mul of the two ranges always/never overflows.
LLVM_ABI ConstantRange signExtend(uint32_t BitWidth) const
Return a new range in the specified integer type, which must be strictly larger than the current type...
LLVM_ABI ConstantRange multiply(const ConstantRange &Other, unsigned NoWrapKind=0) const
Return a new range representing the possible values resulting from a multiplication of a value in thi...
LLVM_ABI bool contains(const APInt &Val) const
Return true if the specified value is in the set.
LLVM_ABI APInt getUnsignedMax() const
Return the largest unsigned value contained in the ConstantRange.
LLVM_ABI ConstantRange intersectWith(const ConstantRange &CR, PreferredRangeType Type=Smallest) const
Return the range that results from the intersection of this range with another range.
LLVM_ABI APInt getSignedMax() const
Return the largest signed value contained in the ConstantRange.
OverflowResult
Represents whether an operation on the given constant range is known to always or never overflow.
@ NeverOverflows
Never overflows.
@ AlwaysOverflowsHigh
Always overflows in the direction of signed/unsigned max value.
@ AlwaysOverflowsLow
Always overflows in the direction of signed/unsigned min value.
@ MayOverflow
May or may not overflow.
uint32_t getBitWidth() const
Get the bit width of this ConstantRange.
LLVM_ABI OverflowResult signedSubMayOverflow(const ConstantRange &Other) const
Return whether signed sub of the two ranges always/never overflows.
uint64_t getZExtValue() const
const APInt & getAPIntValue() const
This is an important base class in LLVM.
LLVM_ABI Constant * getSplatValue(bool AllowPoison=false) const
If all elements of the vector constant have the same value, return that value.
LLVM_ABI Constant * getAggregateElement(unsigned Elt) const
For aggregates (struct/array/vector) return the constant that corresponds to the specified element if...
static LLVM_ABI ExtOps getExtOps(unsigned FromSize, unsigned ToSize, bool Signed)
Returns the ops for a zero- or sign-extension in a DIExpression.
static LLVM_ABI void appendOffset(SmallVectorImpl< uint64_t > &Ops, int64_t Offset)
Append Ops with operations to apply the Offset.
static LLVM_ABI DIExpression * appendOpsToArg(const DIExpression *Expr, ArrayRef< uint64_t > Ops, unsigned ArgNo, bool StackValue=false)
Create a copy of Expr by appending the given list of Ops to each instance of the operand DW_OP_LLVM_a...
static LLVM_ABI const DIExpression * convertToVariadicExpression(const DIExpression *Expr)
If Expr is a non-variadic expression (i.e.
static LLVM_ABI std::optional< DIExpression * > createFragmentExpression(const DIExpression *Expr, unsigned OffsetInBits, unsigned SizeInBits)
Create a DIExpression to describe one part of an aggregate variable that is fragmented across multipl...
Base class for variables.
A parsed version of the target data layout string in and methods for querying it.
bool isLittleEndian() const
Layout endianness...
LLVM_ABI IntegerType * getIntPtrType(LLVMContext &C, unsigned AddressSpace=0) const
Returns an integer type with size at least as big as that of a pointer in the given address space.
LLVM_ABI Align getABITypeAlign(Type *Ty) const
Returns the minimum ABI-required alignment for the specified type.
LLVM_ABI unsigned getPointerTypeSizeInBits(Type *) const
The pointer representation size in bits for this type.
LLVM_ABI Align getPrefTypeAlign(Type *Ty) const
Returns the preferred stack/global alignment for the specified type.
Implements a dense probed hash-table based set.
const char * getSymbol() const
unsigned getTargetFlags() const
This class is used to gather all the unique data bits of a node.
Data structure describing the variable locations in a function.
bool hasMinSize() const
Optimize this function for minimum size (-Oz).
AttributeList getAttributes() const
Return the attribute list for this Function.
int64_t getOffset() const
LLVM_ABI unsigned getAddressSpace() const
unsigned getTargetFlags() const
const GlobalValue * getGlobal() const
bool isThreadLocal() const
If the value is "Thread Local", its value isn't shared by the threads.
unsigned getAddressSpace() const
Module * getParent()
Get the module that this global value is contained inside of...
PointerType * getType() const
Global values are always pointers.
This class is used to form a handle around another node that is persistent and is updated across invo...
const SDValue & getValue() const
static LLVM_ABI bool compare(const APInt &LHS, const APInt &RHS, ICmpInst::Predicate Pred)
Return result of LHS Pred RHS comparison.
This is an important class for using LLVM in a threaded context.
Tracks which library functions to use for a particular subtarget.
CallingConv::ID getLibcallImplCallingConv(RTLIB::LibcallImpl Call) const
Get the CallingConv that should be used for the specified libcall.
RTLIB::LibcallImpl getLibcallImpl(RTLIB::Libcall Call) const
Return the lowering's selection of implementation call for Call.
This SDNode is used for LIFETIME_START/LIFETIME_END values.
This class is used to represent ISD::LOAD nodes.
static LocationSize precise(uint64_t Value)
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
const MDOperand & getOperand(unsigned I) const
static MVT getIntegerVT(unsigned BitWidth)
Abstract base class for all machine specific constantpool value subclasses.
virtual void addSelectionDAGCSEId(FoldingSetNodeID &ID)=0
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
LLVM_ABI int CreateStackObject(uint64_t Size, Align Alignment, bool isSpillSlot, const AllocaInst *Alloca=nullptr, uint8_t ID=0)
Create a new statically sized stack object, returning a nonnegative identifier to represent it.
Align getObjectAlign(int ObjectIdx) const
Return the alignment of the specified stack object.
bool isFixedObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a fixed stack object.
void setObjectAlignment(int ObjectIdx, Align Alignment)
setObjectAlignment - Change the alignment of the specified stack object.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
Function & getFunction()
Return the LLVM function that this machine code represents.
const TargetMachine & getTarget() const
getTarget - Return the target machine this machine code is compiled with
A description of a memory reference used in the backend.
const MDNode * getRanges() const
Return the range tag for the memory reference.
Flags
Flags values. These may be or'd together.
@ MOVolatile
The memory access is volatile.
@ MODereferenceable
The memory access is dereferenceable (i.e., doesn't trap).
@ MOLoad
The memory access reads data.
@ MOInvariant
The memory access always returns the same value (or traps).
@ MOStore
The memory access writes data.
const MachinePointerInfo & getPointerInfo() const
Flags getFlags() const
Return the raw flags of the source value,.
This class contains meta information specific to a module.
An SDNode that represents everything that will be needed to construct a MachineInstr.
This class is used to represent an MGATHER node.
This class is used to represent an MLOAD node.
This class is used to represent an MSCATTER node.
This class is used to represent an MSTORE node.
This SDNode is used for target intrinsics that touch memory and need an associated MachineMemOperand.
size_t getNumMemOperands() const
Return the number of memory operands.
LLVM_ABI MemSDNode(unsigned Opc, unsigned Order, const DebugLoc &dl, SDVTList VTs, EVT memvt, PointerUnion< MachineMemOperand *, MachineMemOperand ** > memrefs)
Constructor that supports single or multiple MMOs.
PointerUnion< MachineMemOperand *, MachineMemOperand ** > MemRefs
Memory reference information.
MachineMemOperand * getMemOperand() const
Return the unique MachineMemOperand object describing the memory reference performed by operation.
const MachinePointerInfo & getPointerInfo() const
ArrayRef< MachineMemOperand * > memoperands() const
Return the memory operands for this node.
unsigned getRawSubclassData() const
Return the SubclassData value, without HasDebugValue.
EVT getMemoryVT() const
Return the type of the in-memory value.
Representation for a specific memory location.
A Module instance is used to store all the information related to an LLVM module.
Function * getFunction(StringRef Name) const
Look up the specified function in the module symbol table.
Represent a mutable reference to an array (0 or more elements consecutively in memory),...
Pass interface - Implemented by all 'passes'.
Class to represent pointers.
static PointerType * getUnqual(Type *ElementType)
This constructs a pointer to an object of the specified type in the default address space (address sp...
unsigned getAddressSpace() const
Return the address space of the Pointer type.
static LLVM_ABI PointerType * get(Type *ElementType, unsigned AddressSpace)
This constructs a pointer to an object of the specified type in a numbered address space.
A discriminated union of two or more pointer types, with the discriminator in the low bits of the poi...
bool isNull() const
Test if the pointer held in the union is null, regardless of which type it is.
Analysis providing profile information.
void Deallocate(SubClass *E)
Deallocate - Release storage for the pointed-to object.
Wrapper class representing virtual and physical registers.
Keeps track of dbg_value information through SDISel.
LLVM_ABI void add(SDDbgValue *V, bool isParameter)
LLVM_ABI void erase(const SDNode *Node)
Invalidate all DbgValues attached to the node and remove it from the Node-to-DbgValues map.
Holds the information from a dbg_label node through SDISel.
Holds the information for a single machine location through SDISel; either an SDNode,...
static SDDbgOperand fromNode(SDNode *Node, unsigned ResNo)
static SDDbgOperand fromFrameIdx(unsigned FrameIdx)
static SDDbgOperand fromVReg(Register VReg)
static SDDbgOperand fromConst(const Value *Const)
@ SDNODE
Value is the result of an expression.
Holds the information from a dbg_value node through SDISel.
Wrapper class for IR location info (IR ordering and DebugLoc) to be passed into SDNode creation funct...
const DebugLoc & getDebugLoc() const
unsigned getIROrder() const
This class provides iterator support for SDUse operands that use a specific SDNode.
Represents one node in the SelectionDAG.
ArrayRef< SDUse > ops() const
const APInt & getAsAPIntVal() const
Helper method returns the APInt value of a ConstantSDNode.
LLVM_ABI void dumprFull(const SelectionDAG *G=nullptr) const
printrFull to dbgs().
unsigned getOpcode() const
Return the SelectionDAG opcode value for this node.
LLVM_ABI bool isOnlyUserOf(const SDNode *N) const
Return true if this node is the only use of N.
iterator_range< value_op_iterator > op_values() const
unsigned getIROrder() const
Return the node ordering.
static constexpr size_t getMaxNumOperands()
Return the maximum number of operands that a SDNode can hold.
iterator_range< use_iterator > uses()
MemSDNodeBitfields MemSDNodeBits
LLVM_ABI void Profile(FoldingSetNodeID &ID) const
Gather unique data for the node.
bool getHasDebugValue() const
SDNodeFlags getFlags() const
void setNodeId(int Id)
Set unique node id.
LLVM_ABI void intersectFlagsWith(const SDNodeFlags Flags)
Clear any flags in this node that aren't also set in Flags.
static bool hasPredecessorHelper(const SDNode *N, SmallPtrSetImpl< const SDNode * > &Visited, SmallVectorImpl< const SDNode * > &Worklist, unsigned int MaxSteps=0, bool TopologicalPrune=false)
Returns true if N is a predecessor of any node in Worklist.
uint64_t getAsZExtVal() const
Helper method returns the zero-extended integer value of a ConstantSDNode.
bool use_empty() const
Return true if there are no uses of this node.
unsigned getNumValues() const
Return the number of values defined/returned by this operator.
unsigned getNumOperands() const
Return the number of values used by this operation.
const SDValue & getOperand(unsigned Num) const
static LLVM_ABI bool areOnlyUsersOf(ArrayRef< const SDNode * > Nodes, const SDNode *N)
Return true if all the users of N are contained in Nodes.
use_iterator use_begin() const
Provide iteration support to walk over all uses of an SDNode.
LLVM_ABI bool isOperandOf(const SDNode *N) const
Return true if this node is an operand of N.
const APInt & getConstantOperandAPInt(unsigned Num) const
Helper method returns the APInt of a ConstantSDNode operand.
std::optional< APInt > bitcastToAPInt() const
LLVM_ABI bool hasPredecessor(const SDNode *N) const
Return true if N is a predecessor of this node.
LLVM_ABI bool hasAnyUseOfValue(unsigned Value) const
Return true if there are any use of the indicated value.
EVT getValueType(unsigned ResNo) const
Return the type of a specified result.
bool isUndef() const
Returns true if the node type is UNDEF or POISON.
op_iterator op_end() const
op_iterator op_begin() const
static use_iterator use_end()
LLVM_ABI void DropOperands()
Release the operands and set this node to have zero operands.
SDNode(unsigned Opc, unsigned Order, DebugLoc dl, SDVTList VTs)
Create an SDNode.
Represents a use of a SDNode.
SDNode * getUser()
This returns the SDNode that contains this Use.
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
bool hasOneUse() const
Return true if there is exactly one node using value ResNo of Node.
LLVM_ABI bool isOperandOf(const SDNode *N) const
Return true if the referenced return value is an operand of N.
LLVM_ABI bool reachesChainWithoutSideEffects(SDValue Dest, unsigned Depth=2) const
Return true if this operand (which must be a chain) reaches the specified operand without crossing an...
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.
const SDValue & getOperand(unsigned i) const
bool use_empty() const
Return true if there are no nodes using value ResNo of Node.
const APInt & getConstantOperandAPInt(unsigned i) const
uint64_t getScalarValueSizeInBits() const
unsigned getResNo() const
get the index which selects a specific result in the SDNode
uint64_t getConstantOperandVal(unsigned i) const
unsigned getOpcode() const
virtual void verifyTargetNode(const SelectionDAG &DAG, const SDNode *N) const
Checks that the given target-specific node is valid. Aborts if it is not.
This is used to represent a portion of an LLVM function in a low-level Data Dependence DAG representa...
LLVM_ABI SDValue getElementCount(const SDLoc &DL, EVT VT, ElementCount EC)
LLVM_ABI Align getReducedAlign(EVT VT, bool UseABI)
In most cases this function returns the ABI alignment for a given type, except for illegal vector typ...
LLVM_ABI SDValue getVPZeroExtendInReg(SDValue Op, SDValue Mask, SDValue EVL, const SDLoc &DL, EVT VT)
Return the expression required to zero extend the Op value assuming it was the smaller SrcTy value.
LLVM_ABI SDValue getShiftAmountOperand(EVT LHSTy, SDValue Op)
Return the specified value casted to the target's desired shift amount type.
LLVM_ABI SDValue getExtLoad(ISD::LoadExtType ExtType, const SDLoc &dl, EVT VT, SDValue Chain, SDValue Ptr, MachinePointerInfo PtrInfo, EVT MemVT, MaybeAlign Alignment=MaybeAlign(), MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes())
LLVM_ABI std::pair< SDValue, SDValue > getMemccpy(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue C, SDValue Size, const CallInst *CI)
Lower a memccpy operation into a target library call and return the resulting chain and call result a...
LLVM_ABI bool isKnownNeverLogicalZero(SDValue Op, const APInt &DemandedElts, unsigned Depth=0) const
Test whether the given floating point SDValue (or all elements of it, if it is a vector) is known to ...
LLVM_ABI SDValue getExtLoadVP(ISD::LoadExtType ExtType, const SDLoc &dl, EVT VT, SDValue Chain, SDValue Ptr, SDValue Mask, SDValue EVL, MachinePointerInfo PtrInfo, EVT MemVT, MaybeAlign Alignment, MachineMemOperand::Flags MMOFlags, const AAMDNodes &AAInfo, bool IsExpanding=false)
SDValue getExtractVectorElt(const SDLoc &DL, EVT VT, SDValue Vec, unsigned Idx)
Extract element at Idx from Vec.
LLVM_ABI SDValue getSplatSourceVector(SDValue V, int &SplatIndex)
If V is a splatted value, return the source vector and its splat index.
LLVM_ABI SDValue getLabelNode(unsigned Opcode, const SDLoc &dl, SDValue Root, MCSymbol *Label)
LLVM_ABI OverflowKind computeOverflowForUnsignedSub(SDValue N0, SDValue N1) const
Determine if the result of the unsigned sub of 2 nodes can overflow.
LLVM_ABI unsigned ComputeMaxSignificantBits(SDValue Op, unsigned Depth=0) const
Get the upper bound on bit size for this Value Op as a signed integer.
const SDValue & getRoot() const
Return the root tag of the SelectionDAG.
LLVM_ABI std::pair< SDValue, SDValue > getStrlen(SDValue Chain, const SDLoc &dl, SDValue Src, const CallInst *CI)
Lower a strlen operation into a target library call and return the resulting chain and call result as...
LLVM_ABI SDValue getMaskedGather(SDVTList VTs, EVT MemVT, const SDLoc &dl, ArrayRef< SDValue > Ops, MachineMemOperand *MMO, ISD::MemIndexType IndexType, ISD::LoadExtType ExtTy)
LLVM_ABI SDValue getAddrSpaceCast(const SDLoc &dl, EVT VT, SDValue Ptr, unsigned SrcAS, unsigned DestAS)
Return an AddrSpaceCastSDNode.
LLVM_ABI SDValue FoldSetCC(EVT VT, SDValue N1, SDValue N2, ISD::CondCode Cond, const SDLoc &dl, SDNodeFlags Flags={})
Constant fold a setcc to true or false.
bool isKnownNeverSNaN(SDValue Op, const APInt &DemandedElts, unsigned Depth=0) const
LLVM_ABI std::optional< bool > isBoolConstant(SDValue N) const
Check if a value \op N is a constant using the target's BooleanContent for its type.
LLVM_ABI SDValue getStackArgumentTokenFactor(SDValue Chain)
Compute a TokenFactor to force all the incoming stack arguments to be loaded from the stack.
const TargetSubtargetInfo & getSubtarget() const
LLVM_ABI ConstantRange computeConstantRange(SDValue Op, bool ForSigned, unsigned Depth=0) const
Determine the possible constant range of an integer or vector of integers.
LLVM_ABI SDValue getMergeValues(ArrayRef< SDValue > Ops, const SDLoc &dl)
Create a MERGE_VALUES node from the given operands.
LLVM_ABI SDVTList getVTList(EVT VT)
Return an SDVTList that represents the list of values specified.
LLVM_ABI SDValue getShiftAmountConstant(uint64_t Val, EVT VT, const SDLoc &DL)
LLVM_ABI void updateDivergence(SDNode *N)
LLVM_ABI SDValue getSplatValue(SDValue V, bool LegalTypes=false)
If V is a splat vector, return its scalar source operand by extracting that element from the source v...
LLVM_ABI SDValue getAllOnesConstant(const SDLoc &DL, EVT VT, bool IsTarget=false, bool IsOpaque=false)
LLVM_ABI MachineSDNode * getMachineNode(unsigned Opcode, const SDLoc &dl, EVT VT)
These are used for target selectors to create a new node with specified return type(s),...
LLVM_ABI void ExtractVectorElements(SDValue Op, SmallVectorImpl< SDValue > &Args, unsigned Start=0, unsigned Count=0, EVT EltVT=EVT())
Append the extracted elements from Start to Count out of the vector Op in Args.
LLVM_ABI SDValue getAtomicMemset(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Value, SDValue Size, Type *SizeTy, unsigned ElemSz, bool isTailCall, MachinePointerInfo DstPtrInfo)
LLVM_ABI SDValue getAtomicLoad(ISD::LoadExtType ExtType, const SDLoc &dl, EVT MemVT, EVT VT, SDValue Chain, SDValue Ptr, MachineMemOperand *MMO)
LLVM_ABI SDNode * getNodeIfExists(unsigned Opcode, SDVTList VTList, ArrayRef< SDValue > Ops, const SDNodeFlags Flags, bool AllowCommute=false)
Get the specified node if it's already available, or else return NULL.
LLVM_ABI SDValue getPseudoProbeNode(const SDLoc &Dl, SDValue Chain, uint64_t Guid, uint64_t Index, uint32_t Attr)
Creates a PseudoProbeSDNode with function GUID Guid and the index of the block Index it is probing,...
LLVM_ABI SDValue getFreeze(SDValue V)
Return a freeze using the SDLoc of the value operand.
LLVM_ABI SDNode * SelectNodeTo(SDNode *N, unsigned MachineOpc, EVT VT)
These are used for target selectors to mutate the specified node to have the specified return type,...
LLVM_ABI void init(MachineFunction &NewMF, OptimizationRemarkEmitter &NewORE, Pass *PassPtr, const TargetLibraryInfo *LibraryInfo, const LibcallLoweringInfo *LibcallsInfo, UniformityInfo *UA, ProfileSummaryInfo *PSIin, BlockFrequencyInfo *BFIin, MachineModuleInfo &MMI, FunctionVarLocs const *FnVarLocs)
Prepare this SelectionDAG to process code in the given MachineFunction.
LLVM_ABI SelectionDAG(const TargetMachine &TM, CodeGenOptLevel)
LLVM_ABI SDValue getMemset(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, Align Alignment, bool isVol, bool AlwaysInline, const CallInst *CI, MachinePointerInfo DstPtrInfo, const AAMDNodes &AAInfo=AAMDNodes())
LLVM_ABI SDValue getBitcastedSExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by first bitcasting (from potentia...
LLVM_ABI SDValue getConstantPool(const Constant *C, EVT VT, MaybeAlign Align=std::nullopt, int Offs=0, bool isT=false, unsigned TargetFlags=0)
LLVM_ABI SDValue getStridedLoadVP(ISD::MemIndexedMode AM, ISD::LoadExtType ExtType, EVT VT, const SDLoc &DL, SDValue Chain, SDValue Ptr, SDValue Offset, SDValue Stride, SDValue Mask, SDValue EVL, EVT MemVT, MachineMemOperand *MMO, bool IsExpanding=false)
LLVM_ABI SDValue getAtomicCmpSwap(unsigned Opcode, const SDLoc &dl, EVT MemVT, SDVTList VTs, SDValue Chain, SDValue Ptr, SDValue Cmp, SDValue Swp, MachineMemOperand *MMO)
Gets a node for an atomic cmpxchg op.
LLVM_ABI SDValue makeEquivalentMemoryOrdering(SDValue OldChain, SDValue NewMemOpChain)
If an existing load has uses of its chain, create a token factor node with that chain and the new mem...
LLVM_ABI bool isConstantIntBuildVectorOrConstantInt(SDValue N, bool AllowOpaques=true) const
Test whether the given value is a constant int or similar node.
LLVM_ABI void ReplaceAllUsesOfValuesWith(const SDValue *From, const SDValue *To, unsigned Num)
Like ReplaceAllUsesOfValueWith, but for multiple values at once.
LLVM_ABI SDValue getJumpTableDebugInfo(int JTI, SDValue Chain, const SDLoc &DL)
LLVM_ABI SDValue getSymbolFunctionGlobalAddress(SDValue Op, Function **TargetFunction=nullptr)
Return a GlobalAddress of the function from the current module with name matching the given ExternalS...
LLVM_ABI std::optional< unsigned > getValidMaximumShiftAmount(SDValue V, const APInt &DemandedElts, unsigned Depth=0) const
If a SHL/SRA/SRL node V has shift amounts that are all less than the element bit-width of the shift n...
LLVM_ABI SDValue UnrollVectorOp(SDNode *N, unsigned ResNE=0)
Utility function used by legalize and lowering to "unroll" a vector operation by splitting out the sc...
LLVM_ABI SDValue getVScale(const SDLoc &DL, EVT VT, APInt MulImm)
Return a node that represents the runtime scaling 'MulImm * RuntimeVL'.
LLVM_ABI SDValue getConstantFP(double Val, const SDLoc &DL, EVT VT, bool isTarget=false)
Create a ConstantFPSDNode wrapping a constant value.
OverflowKind
Used to represent the possible overflow behavior of an operation.
static LLVM_ABI unsigned getHasPredecessorMaxSteps()
LLVM_ABI bool haveNoCommonBitsSet(SDValue A, SDValue B) const
Return true if A and B have no common bits set.
SDValue getExtractSubvector(const SDLoc &DL, EVT VT, SDValue Vec, unsigned Idx)
Return the VT typed sub-vector of Vec at Idx.
LLVM_ABI bool cannotBeOrderedNegativeFP(SDValue Op) const
Test whether the given float value is known to be positive.
LLVM_ABI SDValue getRegister(Register Reg, EVT VT)
LLVM_ABI bool calculateDivergence(SDNode *N)
LLVM_ABI std::pair< SDValue, SDValue > getStrcmp(SDValue Chain, const SDLoc &dl, SDValue S0, SDValue S1, const CallInst *CI)
Lower a strcmp operation into a target library call and return the resulting chain and call result as...
LLVM_ABI SDValue getGetFPEnv(SDValue Chain, const SDLoc &dl, SDValue Ptr, EVT MemVT, MachineMemOperand *MMO)
LLVM_ABI SDValue getAssertAlign(const SDLoc &DL, SDValue V, Align A)
Return an AssertAlignSDNode.
LLVM_ABI SDNode * mutateStrictFPToFP(SDNode *Node)
Mutate the specified strict FP node to its non-strict equivalent, unlinking the node from its chain a...
LLVM_ABI SDValue getLoad(EVT VT, const SDLoc &dl, SDValue Chain, SDValue Ptr, MachinePointerInfo PtrInfo, MaybeAlign Alignment=MaybeAlign(), MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes(), const MDNode *Ranges=nullptr)
Loads are not normal binary operators: their result type is not determined by their operands,...
LLVM_ABI bool canIgnoreSignBitOfZero(const SDUse &Use) const
Check if a use of a float value is insensitive to signed zeros.
LLVM_ABI bool SignBitIsZeroFP(SDValue Op, unsigned Depth=0) const
Return true if the sign bit of Op is known to be zero, for a floating-point value.
LLVM_ABI SDValue getMemIntrinsicNode(unsigned Opcode, const SDLoc &dl, SDVTList VTList, ArrayRef< SDValue > Ops, EVT MemVT, MachinePointerInfo PtrInfo, Align Alignment, MachineMemOperand::Flags Flags=MachineMemOperand::MOLoad|MachineMemOperand::MOStore, LocationSize Size=LocationSize::precise(0), const AAMDNodes &AAInfo=AAMDNodes())
Creates a MemIntrinsicNode that may produce a result and takes a list of operands.
SDValue getInsertSubvector(const SDLoc &DL, SDValue Vec, SDValue SubVec, unsigned Idx)
Insert SubVec at the Idx element of Vec.
LLVM_ABI SDValue getBitcastedZExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by first bitcasting (from potentia...
LLVM_ABI SDValue getStepVector(const SDLoc &DL, EVT ResVT, const APInt &StepVal)
Returns a vector of type ResVT whose elements contain the linear sequence <0, Step,...
SDValue getSetCC(const SDLoc &DL, EVT VT, SDValue LHS, SDValue RHS, ISD::CondCode Cond, SDValue Chain=SDValue(), bool IsSignaling=false, SDNodeFlags Flags={})
Helper function to make it easier to build SetCC's if you just have an ISD::CondCode instead of an SD...
LLVM_ABI SDValue getAtomic(unsigned Opcode, const SDLoc &dl, EVT MemVT, SDValue Chain, SDValue Ptr, SDValue Val, MachineMemOperand *MMO)
Gets a node for an atomic op, produces result (if relevant) and chain and takes 2 operands.
LLVM_ABI Align getEVTAlign(EVT MemoryVT) const
Compute the default alignment value for the given type.
LLVM_ABI bool shouldOptForSize() const
LLVM_ABI SDValue getNOT(const SDLoc &DL, SDValue Val, EVT VT)
Create a bitwise NOT operation as (XOR Val, -1).
LLVM_ABI SDValue getVPZExtOrTrunc(const SDLoc &DL, EVT VT, SDValue Op, SDValue Mask, SDValue EVL)
Convert a vector-predicated Op, which must be an integer vector, to the vector-type VT,...
LLVM_ABI SDValue getMemcpy(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, Align DstAlign, Align SrcAlign, bool isVol, bool AlwaysInline, const CallInst *CI, std::optional< bool > OverrideTailCall, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo, const AAMDNodes &AAInfo=AAMDNodes(), BatchAAResults *BatchAA=nullptr)
const TargetLowering & getTargetLoweringInfo() const
LLVM_ABI bool isEqualTo(SDValue A, SDValue B) const
Test whether two SDValues are known to compare equal.
static constexpr unsigned MaxRecursionDepth
LLVM_ABI SDValue getStridedStoreVP(SDValue Chain, const SDLoc &DL, SDValue Val, SDValue Ptr, SDValue Offset, SDValue Stride, SDValue Mask, SDValue EVL, EVT MemVT, MachineMemOperand *MMO, ISD::MemIndexedMode AM, bool IsTruncating=false, bool IsCompressing=false)
bool isGuaranteedNotToBePoison(SDValue Op, unsigned Depth=0) const
Return true if this function can prove that Op is never poison.
LLVM_ABI SDValue getIdentityElement(unsigned Opcode, const SDLoc &DL, EVT VT, SDNodeFlags Flags)
Get the (commutative) identity element for the given opcode, if it exists.
LLVM_ABI SDValue expandVACopy(SDNode *Node)
Expand the specified ISD::VACOPY node as the Legalize pass would.
LLVM_ABI SDValue getIndexedMaskedLoad(SDValue OrigLoad, const SDLoc &dl, SDValue Base, SDValue Offset, ISD::MemIndexedMode AM)
LLVM_ABI APInt computeVectorKnownZeroElements(SDValue Op, const APInt &DemandedElts, unsigned Depth=0) const
For each demanded element of a vector, see if it is known to be zero.
LLVM_ABI void AddDbgValue(SDDbgValue *DB, bool isParameter)
Add a dbg_value SDNode.
bool NewNodesMustHaveLegalTypes
When true, additional steps are taken to ensure that getConstant() and similar functions return DAG n...
LLVM_ABI std::pair< EVT, EVT > GetSplitDestVTs(const EVT &VT) const
Compute the VTs needed for the low/hi parts of a type which is split (or expanded) into two not neces...
LLVM_ABI void salvageDebugInfo(SDNode &N)
To be invoked on an SDNode that is slated to be erased.
LLVM_ABI SDNode * MorphNodeTo(SDNode *N, unsigned Opc, SDVTList VTs, ArrayRef< SDValue > Ops)
This mutates the specified node to have the specified return type, opcode, and operands.
LLVM_ABI std::pair< SDValue, SDValue > UnrollVectorOverflowOp(SDNode *N, unsigned ResNE=0)
Like UnrollVectorOp(), but for the [US](ADD|SUB|MUL)O family of opcodes.
allnodes_const_iterator allnodes_begin() const
SDValue getUNDEF(EVT VT)
Return an UNDEF node. UNDEF does not have a useful SDLoc.
LLVM_ABI SDValue getGatherVP(SDVTList VTs, EVT VT, const SDLoc &dl, ArrayRef< SDValue > Ops, MachineMemOperand *MMO, ISD::MemIndexType IndexType)
SDValue getBuildVector(EVT VT, const SDLoc &DL, ArrayRef< SDValue > Ops)
Return an ISD::BUILD_VECTOR node.
LLVM_ABI SDValue getBitcastedAnyExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by first bitcasting (from potentia...
LLVM_ABI bool isSplatValue(SDValue V, const APInt &DemandedElts, APInt &UndefElts, unsigned Depth=0) const
Test whether V has a splatted value for all the demanded elements.
LLVM_ABI void DeleteNode(SDNode *N)
Remove the specified node from the system.
LLVM_ABI SDValue getBitcast(EVT VT, SDValue V)
Return a bitcast using the SDLoc of the value operand, and casting to the provided type.
LLVM_ABI SDDbgValue * getDbgValueList(DIVariable *Var, DIExpression *Expr, ArrayRef< SDDbgOperand > Locs, ArrayRef< SDNode * > Dependencies, bool IsIndirect, const DebugLoc &DL, unsigned O, bool IsVariadic)
Creates a SDDbgValue node from a list of locations.
LLVM_ABI std::pair< SDValue, SDValue > getStrcpy(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, const CallInst *CI)
Lower a strcpy operation into a target library call and return the resulting chain and call result as...
SDValue getSelect(const SDLoc &DL, EVT VT, SDValue Cond, SDValue LHS, SDValue RHS, SDNodeFlags Flags=SDNodeFlags())
Helper function to make it easier to build Select's if you just have operands and don't want to check...
LLVM_ABI SDValue getNegative(SDValue Val, const SDLoc &DL, EVT VT)
Create negative operation as (SUB 0, Val).
LLVM_ABI std::optional< unsigned > getValidShiftAmount(SDValue V, const APInt &DemandedElts, unsigned Depth=0) const
If a SHL/SRA/SRL node V has a uniform shift amount that is less than the element bit-width of the shi...
LLVM_ABI void setNodeMemRefs(MachineSDNode *N, ArrayRef< MachineMemOperand * > NewMemRefs)
Mutate the specified machine node's memory references to the provided list.
LLVM_ABI SDValue simplifySelect(SDValue Cond, SDValue TVal, SDValue FVal)
Try to simplify a select/vselect into 1 of its operands or a constant.
LLVM_ABI SDValue getZeroExtendInReg(SDValue Op, const SDLoc &DL, EVT VT)
Return the expression required to zero extend the Op value assuming it was the smaller SrcTy value.
LLVM_ABI bool isConstantFPBuildVectorOrConstantFP(SDValue N) const
Test whether the given value is a constant FP or similar node.
const DataLayout & getDataLayout() const
LLVM_ABI SDValue getPartialReduceMLS(unsigned Opc, const SDLoc &DL, SDValue Acc, SDValue LHS, SDValue RHS)
Get an expression that implements a partial multiply-subtract reduction.
LLVM_ABI SDValue expandVAArg(SDNode *Node)
Expand the specified ISD::VAARG node as the Legalize pass would.
LLVM_ABI SDValue getTokenFactor(const SDLoc &DL, SmallVectorImpl< SDValue > &Vals)
Creates a new TokenFactor containing Vals.
LLVM_ABI bool doesNodeExist(unsigned Opcode, SDVTList VTList, ArrayRef< SDValue > Ops)
Check if a node exists without modifying its flags.
LLVM_ABI ConstantRange computeConstantRangeIncludingKnownBits(SDValue Op, bool ForSigned, unsigned Depth=0) const
Combine constant ranges from computeConstantRange() and computeKnownBits().
const SelectionDAGTargetInfo & getSelectionDAGInfo() const
LLVM_ABI bool areNonVolatileConsecutiveLoads(LoadSDNode *LD, LoadSDNode *Base, unsigned Bytes, int Dist) const
Return true if loads are next to each other and can be merged.
LLVM_ABI SDValue getMaskedHistogram(SDVTList VTs, EVT MemVT, const SDLoc &dl, ArrayRef< SDValue > Ops, MachineMemOperand *MMO, ISD::MemIndexType IndexType)
LLVM_ABI SDDbgLabel * getDbgLabel(DILabel *Label, const DebugLoc &DL, unsigned O)
Creates a SDDbgLabel node.
LLVM_ABI SDValue getStoreVP(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Ptr, SDValue Offset, SDValue Mask, SDValue EVL, EVT MemVT, MachineMemOperand *MMO, ISD::MemIndexedMode AM, bool IsTruncating=false, bool IsCompressing=false)
LLVM_ABI OverflowKind computeOverflowForUnsignedMul(SDValue N0, SDValue N1) const
Determine if the result of the unsigned mul of 2 nodes can overflow.
LLVM_ABI void copyExtraInfo(SDNode *From, SDNode *To)
Copy extra info associated with one node to another.
LLVM_ABI SDValue getConstant(uint64_t Val, const SDLoc &DL, EVT VT, bool isTarget=false, bool isOpaque=false)
Create a ConstantSDNode wrapping a constant value.
LLVM_ABI SDValue getMemBasePlusOffset(SDValue Base, TypeSize Offset, const SDLoc &DL, const SDNodeFlags Flags=SDNodeFlags())
Returns sum of the base pointer and offset.
LLVM_ABI SDValue getGlobalAddress(const GlobalValue *GV, const SDLoc &DL, EVT VT, int64_t offset=0, bool isTargetGA=false, unsigned TargetFlags=0)
LLVM_ABI SDValue getVAArg(EVT VT, const SDLoc &dl, SDValue Chain, SDValue Ptr, SDValue SV, unsigned Align)
VAArg produces a result and token chain, and takes a pointer and a source value as input.
LLVM_ABI SDValue getTruncStore(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Ptr, MachinePointerInfo PtrInfo, EVT SVT, Align Alignment, MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes())
LLVM_ABI SDValue getLoadFFVP(EVT VT, const SDLoc &DL, SDValue Chain, SDValue Ptr, SDValue Mask, SDValue EVL, MachineMemOperand *MMO)
LLVM_ABI SDValue getTypeSize(const SDLoc &DL, EVT VT, TypeSize TS)
LLVM_ABI SDValue getMDNode(const MDNode *MD)
Return an MDNodeSDNode which holds an MDNode.
LLVM_ABI void clear()
Clear state and free memory necessary to make this SelectionDAG ready to process a new block.
LLVM_ABI std::pair< SDValue, SDValue > getMemcmp(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, const CallInst *CI)
Lower a memcmp operation into a target library call and return the resulting chain and call result as...
LLVM_ABI void ReplaceAllUsesWith(SDValue From, SDValue To)
Modify anything using 'From' to use 'To' instead.
LLVM_ABI SDValue getCommutedVectorShuffle(const ShuffleVectorSDNode &SV)
Returns an ISD::VECTOR_SHUFFLE node semantically equivalent to the shuffle node in input but with swa...
LLVM_ABI std::pair< SDValue, SDValue > SplitVector(const SDValue &N, const SDLoc &DL, const EVT &LoVT, const EVT &HiVT)
Split the vector with EXTRACT_SUBVECTOR using the provided VTs and return the low/high part.
LLVM_ABI SDValue makeStateFunctionCall(unsigned LibFunc, SDValue Ptr, SDValue InChain, const SDLoc &DLoc)
Helper used to make a call to a library function that has one argument of pointer type.
LLVM_ABI SDValue getStore(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Ptr, MachinePointerInfo PtrInfo, Align Alignment, MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes())
Helper function to build ISD::STORE nodes.
LLVM_ABI SDValue getSignedConstant(int64_t Val, const SDLoc &DL, EVT VT, bool isTarget=false, bool isOpaque=false)
LLVM_ABI SDValue getIndexedLoadVP(SDValue OrigLoad, const SDLoc &dl, SDValue Base, SDValue Offset, ISD::MemIndexedMode AM)
LLVM_ABI SDValue getSrcValue(const Value *v)
Construct a node to track a Value* through the backend.
SDValue getSplatVector(EVT VT, const SDLoc &DL, SDValue Op)
LLVM_ABI SDValue getAtomicMemcpy(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, Type *SizeTy, unsigned ElemSz, bool isTailCall, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo)
LLVM_ABI OverflowKind computeOverflowForSignedMul(SDValue N0, SDValue N1) const
Determine if the result of the signed mul of 2 nodes can overflow.
LLVM_ABI MaybeAlign InferPtrAlign(SDValue Ptr) const
Infer alignment of a load / store address.
LLVM_ABI void dump() const
Dump the textual format of this DAG.
LLVM_ABI bool MaskedValueIsAllOnes(SDValue Op, const APInt &Mask, unsigned Depth=0) const
Return true if '(Op & Mask) == Mask'.
LLVM_ABI bool SignBitIsZero(SDValue Op, unsigned Depth=0) const
Return true if the sign bit of Op is known to be zero.
LLVM_ABI void RemoveDeadNodes()
This method deletes all unreachable nodes in the SelectionDAG.
LLVM_ABI void RemoveDeadNode(SDNode *N)
Remove the specified node from the system.
LLVM_ABI void AddDbgLabel(SDDbgLabel *DB)
Add a dbg_label SDNode.
bool isConstantValueOfAnyType(SDValue N) const
LLVM_ABI bool canCreateUndefOrPoison(SDValue Op, const APInt &DemandedElts, UndefPoisonKind Kind=UndefPoisonKind::UndefOrPoison, bool ConsiderFlags=true, unsigned Depth=0) const
Return true if Op can create undef or poison from non-undef & non-poison operands.
LLVM_ABI SDValue getTargetExtractSubreg(int SRIdx, const SDLoc &DL, EVT VT, SDValue Operand)
A convenience function for creating TargetInstrInfo::EXTRACT_SUBREG nodes.
LLVM_ABI SDValue getBasicBlock(MachineBasicBlock *MBB)
LLVM_ABI SDValue getSExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either sign-extending or trunca...
LLVM_ABI SDDbgValue * getVRegDbgValue(DIVariable *Var, DIExpression *Expr, Register VReg, bool IsIndirect, const DebugLoc &DL, unsigned O)
Creates a VReg SDDbgValue node.
LLVM_ABI KnownFPClass computeKnownFPClass(SDValue Op, FPClassTest InterestedClasses, unsigned Depth=0) const
Determine floating-point class information about Op.
LLVM_ABI bool isIdentityElement(unsigned Opc, SDNodeFlags Flags, SDValue V, unsigned OperandNo, unsigned Depth=0) const
Returns true if V is an identity element of Opc with Flags.
LLVM_ABI SDValue getEHLabel(const SDLoc &dl, SDValue Root, MCSymbol *Label)
LLVM_ABI SDValue getIndexedStoreVP(SDValue OrigStore, const SDLoc &dl, SDValue Base, SDValue Offset, ISD::MemIndexedMode AM)
LLVM_ABI bool isGuaranteedNotToBeUndefOrPoison(SDValue Op, UndefPoisonKind Kind=UndefPoisonKind::UndefOrPoison, unsigned Depth=0) const
Return true if this function can prove that Op is never poison and, Kind can be used to track poison ...
LLVM_ABI bool isKnownNeverZero(SDValue Op, unsigned Depth=0) const
Test whether the given SDValue is known to contain non-zero value(s).
LLVM_ABI SDValue getIndexedStore(SDValue OrigStore, const SDLoc &dl, SDValue Base, SDValue Offset, ISD::MemIndexedMode AM)
LLVM_ABI SDValue FoldConstantArithmetic(unsigned Opcode, const SDLoc &DL, EVT VT, ArrayRef< SDValue > Ops, SDNodeFlags Flags=SDNodeFlags())
LLVM_ABI std::optional< unsigned > getValidMinimumShiftAmount(SDValue V, const APInt &DemandedElts, unsigned Depth=0) const
If a SHL/SRA/SRL node V has shift amounts that are all less than the element bit-width of the shift n...
LLVM_ABI SDValue getSetFPEnv(SDValue Chain, const SDLoc &dl, SDValue Ptr, EVT MemVT, MachineMemOperand *MMO)
LLVM_ABI SDValue getBoolExtOrTrunc(SDValue Op, const SDLoc &SL, EVT VT, EVT OpVT)
Convert Op, which must be of integer type, to the integer type VT, by using an extension appropriate ...
LLVM_ABI SDValue getMaskedStore(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Base, SDValue Offset, SDValue Mask, EVT MemVT, MachineMemOperand *MMO, ISD::MemIndexedMode AM, bool IsTruncating=false, bool IsCompressing=false)
LLVM_ABI SDValue getExternalSymbol(const char *Sym, EVT VT)
const TargetMachine & getTarget() const
LLVM_ABI std::pair< SDValue, SDValue > getStrictFPExtendOrRound(SDValue Op, SDValue Chain, const SDLoc &DL, EVT VT)
Convert Op, which must be a STRICT operation of float type, to the float type VT, by either extending...
LLVM_ABI std::pair< SDValue, SDValue > SplitEVL(SDValue N, EVT VecVT, const SDLoc &DL)
Split the explicit vector length parameter of a VP operation.
LLVM_ABI SDValue getPtrExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either truncating it or perform...
LLVM_ABI SDValue getVPLogicalNOT(const SDLoc &DL, SDValue Val, SDValue Mask, SDValue EVL, EVT VT)
Create a vector-predicated logical NOT operation as (VP_XOR Val, BooleanOne, Mask,...
LLVM_ABI SDValue getMaskFromElementCount(const SDLoc &DL, EVT VT, ElementCount Len)
Return a vector with the first 'Len' lanes set to true and remaining lanes set to false.
LLVM_ABI SDValue getAnyExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either any-extending or truncat...
iterator_range< allnodes_iterator > allnodes()
LLVM_ABI SDValue getBlockAddress(const BlockAddress *BA, EVT VT, int64_t Offset=0, bool isTarget=false, unsigned TargetFlags=0)
LLVM_ABI SDValue WidenVector(const SDValue &N, const SDLoc &DL)
Widen the vector up to the next power of two using INSERT_SUBVECTOR.
const LibcallLoweringInfo & getLibcalls() const
LLVM_ABI SDValue getLoadVP(ISD::MemIndexedMode AM, ISD::LoadExtType ExtType, EVT VT, const SDLoc &dl, SDValue Chain, SDValue Ptr, SDValue Offset, SDValue Mask, SDValue EVL, MachinePointerInfo PtrInfo, EVT MemVT, Align Alignment, MachineMemOperand::Flags MMOFlags, const AAMDNodes &AAInfo, const MDNode *Ranges=nullptr, bool IsExpanding=false)
LLVM_ABI SDValue getIntPtrConstant(uint64_t Val, const SDLoc &DL, bool isTarget=false)
LLVM_ABI SDDbgValue * getConstantDbgValue(DIVariable *Var, DIExpression *Expr, const Value *C, const DebugLoc &DL, unsigned O)
Creates a constant SDDbgValue node.
LLVM_ABI SDValue getScatterVP(SDVTList VTs, EVT VT, const SDLoc &dl, ArrayRef< SDValue > Ops, MachineMemOperand *MMO, ISD::MemIndexType IndexType)
LLVM_ABI SDValue getValueType(EVT)
LLVM_ABI SDValue getLifetimeNode(bool IsStart, const SDLoc &dl, SDValue Chain, int FrameIndex)
Creates a LifetimeSDNode that starts (IsStart==true) or ends (IsStart==false) the lifetime of the Fra...
ArrayRef< SDDbgValue * > GetDbgValues(const SDNode *SD) const
Get the debug values which reference the given SDNode.
LLVM_ABI SDValue getNode(unsigned Opcode, const SDLoc &DL, EVT VT, ArrayRef< SDUse > Ops)
Gets or creates the specified node.
LLVM_ABI OverflowKind computeOverflowForSignedAdd(SDValue N0, SDValue N1) const
Determine if the result of the signed addition of 2 nodes can overflow.
LLVM_ABI SDValue getFPExtendOrRound(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of float type, to the float type VT, by either extending or rounding (by tr...
LLVM_ABI unsigned AssignTopologicalOrder()
Topological-sort the AllNodes list and a assign a unique node id for each node in the DAG based on th...
ilist< SDNode >::size_type allnodes_size() const
LLVM_ABI bool isKnownNeverNaN(SDValue Op, const APInt &DemandedElts, bool SNaN=false, unsigned Depth=0) const
Test whether the given SDValue (or all elements of it, if it is a vector) is known to never be NaN in...
LLVM_ABI SDValue FoldConstantBuildVector(BuildVectorSDNode *BV, const SDLoc &DL, EVT DstEltVT)
Fold BUILD_VECTOR of constants/undefs to the destination type BUILD_VECTOR of constants/undefs elemen...
LLVM_ABI SDValue getAtomicMemmove(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, Type *SizeTy, unsigned ElemSz, bool isTailCall, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo)
LLVM_ABI SDValue getIndexedMaskedStore(SDValue OrigStore, const SDLoc &dl, SDValue Base, SDValue Offset, ISD::MemIndexedMode AM)
LLVM_ABI SDValue getTruncStoreVP(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Ptr, SDValue Mask, SDValue EVL, MachinePointerInfo PtrInfo, EVT SVT, Align Alignment, MachineMemOperand::Flags MMOFlags, const AAMDNodes &AAInfo, bool IsCompressing=false)
SDValue getTargetConstant(uint64_t Val, const SDLoc &DL, EVT VT, bool isOpaque=false)
LLVM_ABI unsigned ComputeNumSignBits(SDValue Op, unsigned Depth=0) const
Return the number of times the sign bit of the register is replicated into the other bits.
LLVM_ABI bool MaskedVectorIsZero(SDValue Op, const APInt &DemandedElts, unsigned Depth=0) const
Return true if 'Op' is known to be zero in DemandedElts.
LLVM_ABI SDValue getBoolConstant(bool V, const SDLoc &DL, EVT VT, EVT OpVT)
Create a true or false constant of type VT using the target's BooleanContent for type OpVT.
LLVM_ABI SDDbgValue * getFrameIndexDbgValue(DIVariable *Var, DIExpression *Expr, unsigned FI, bool IsIndirect, const DebugLoc &DL, unsigned O)
Creates a FrameIndex SDDbgValue node.
LLVM_ABI SDValue getExtStridedLoadVP(ISD::LoadExtType ExtType, const SDLoc &DL, EVT VT, SDValue Chain, SDValue Ptr, SDValue Stride, SDValue Mask, SDValue EVL, EVT MemVT, MachineMemOperand *MMO, bool IsExpanding=false)
LLVM_ABI SDValue getMemmove(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, Align DstAlign, Align SrcAlign, bool isVol, const CallInst *CI, std::optional< bool > OverrideTailCall, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo, const AAMDNodes &AAInfo=AAMDNodes(), BatchAAResults *BatchAA=nullptr)
LLVM_ABI SDValue getJumpTable(int JTI, EVT VT, bool isTarget=false, unsigned TargetFlags=0)
LLVM_ABI bool isBaseWithConstantOffset(SDValue Op) const
Return true if the specified operand is an ISD::ADD with a ConstantSDNode on the right-hand side,...
LLVM_ABI SDValue getVPPtrExtOrTrunc(const SDLoc &DL, EVT VT, SDValue Op, SDValue Mask, SDValue EVL)
Convert a vector-predicated Op, which must be of integer type, to the vector-type integer type VT,...
LLVM_ABI SDValue getVectorIdxConstant(uint64_t Val, const SDLoc &DL, bool isTarget=false)
LLVM_ABI void getTopologicallyOrderedNodes(SmallVectorImpl< const SDNode * > &SortedNodes) const
Get all the nodes in their topological order without modifying any states.
LLVM_ABI void ReplaceAllUsesOfValueWith(SDValue From, SDValue To)
Replace any uses of From with To, leaving uses of other values produced by From.getNode() alone.
MachineFunction & getMachineFunction() const
LLVM_ABI std::pair< SDValue, SDValue > getStrstr(SDValue Chain, const SDLoc &dl, SDValue S0, SDValue S1, const CallInst *CI)
Lower a strstr operation into a target library call and return the resulting chain and call result as...
LLVM_ABI SDValue getPtrExtendInReg(SDValue Op, const SDLoc &DL, EVT VT)
Return the expression required to extend the Op as a pointer value assuming it was the smaller SrcTy ...
LLVM_ABI OverflowKind computeOverflowForUnsignedAdd(SDValue N0, SDValue N1) const
Determine if the result of the unsigned addition of 2 nodes can overflow.
SDValue getPOISON(EVT VT)
Return a POISON node. POISON does not have a useful SDLoc.
SDValue getSplatBuildVector(EVT VT, const SDLoc &DL, SDValue Op)
Return a splat ISD::BUILD_VECTOR node, consisting of Op splatted to all elements.
LLVM_ABI SDValue getFrameIndex(int FI, EVT VT, bool isTarget=false)
LLVM_ABI SDValue getTruncStridedStoreVP(SDValue Chain, const SDLoc &DL, SDValue Val, SDValue Ptr, SDValue Stride, SDValue Mask, SDValue EVL, EVT SVT, MachineMemOperand *MMO, bool IsCompressing=false)
LLVM_ABI void canonicalizeCommutativeBinop(unsigned Opcode, SDValue &N1, SDValue &N2) const
Swap N1 and N2 if Opcode is a commutative binary opcode and the canonical form expects the opposite o...
LLVM_ABI KnownBits computeKnownBits(SDValue Op, unsigned Depth=0) const
Determine which bits of Op are known to be either zero or one and return them in Known.
LLVM_ABI SDValue getRegisterMask(const uint32_t *RegMask)
LLVM_ABI SDValue getZExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either zero-extending or trunca...
LLVM_ABI SDValue getCondCode(ISD::CondCode Cond)
LLVM_ABI bool MaskedValueIsZero(SDValue Op, const APInt &Mask, unsigned Depth=0) const
Return true if 'Op & Mask' is known to be zero.
LLVM_ABI bool isKnownToBeAPowerOfTwoFP(SDValue Val, unsigned Depth=0) const
Test if the given fp value is known to be an integer power-of-2, either positive or negative.
LLVM_ABI OverflowKind computeOverflowForSignedSub(SDValue N0, SDValue N1) const
Determine if the result of the signed sub of 2 nodes can overflow.
SDValue getObjectPtrOffset(const SDLoc &SL, SDValue Ptr, TypeSize Offset)
Create an add instruction with appropriate flags when used for addressing some offset of an object.
LLVMContext * getContext() const
LLVM_ABI SDValue simplifyFPBinop(unsigned Opcode, SDValue X, SDValue Y, SDNodeFlags Flags)
Try to simplify a floating-point binary operation into 1 of its operands or a constant.
const SDValue & setRoot(SDValue N)
Set the current root tag of the SelectionDAG.
LLVM_ABI bool isKnownToBeAPowerOfTwo(SDValue Val, bool OrZero=false, unsigned Depth=0) const
Test if the given value is known to have exactly one bit set.
LLVM_ABI SDValue getDeactivationSymbol(const GlobalValue *GV)
LLVM_ABI SDValue getTargetExternalSymbol(const char *Sym, EVT VT, unsigned TargetFlags=0)
LLVM_ABI SDValue getMCSymbol(MCSymbol *Sym, EVT VT)
LLVM_ABI bool isUndef(unsigned Opcode, ArrayRef< SDValue > Ops)
Return true if the result of this operation is always undefined.
LLVM_ABI SDValue CreateStackTemporary(TypeSize Bytes, Align Alignment)
Create a stack temporary based on the size in bytes and the alignment.
LLVM_ABI SDNode * UpdateNodeOperands(SDNode *N, SDValue Op)
Mutate the specified node in-place to have the specified operands.
LLVM_ABI std::pair< EVT, EVT > GetDependentSplitDestVTs(const EVT &VT, const EVT &EnvVT, bool *HiIsEmpty) const
Compute the VTs needed for the low/hi parts of a type, dependent on an enveloping VT that has been sp...
LLVM_ABI SDValue foldConstantFPMath(unsigned Opcode, const SDLoc &DL, EVT VT, ArrayRef< SDValue > Ops)
Fold floating-point operations when all operands are constants and/or undefined.
LLVM_ABI std::optional< ConstantRange > getValidShiftAmountRange(SDValue V, const APInt &DemandedElts, unsigned Depth) const
If a SHL/SRA/SRL node V has shift amounts that are all less than the element bit-width of the shift n...
LLVM_ABI SDValue FoldSymbolOffset(unsigned Opcode, EVT VT, const GlobalAddressSDNode *GA, const SDNode *N2)
LLVM_ABI SDValue getIndexedLoad(SDValue OrigLoad, const SDLoc &dl, SDValue Base, SDValue Offset, ISD::MemIndexedMode AM)
LLVM_ABI SDValue getTargetInsertSubreg(int SRIdx, const SDLoc &DL, EVT VT, SDValue Operand, SDValue Subreg)
A convenience function for creating TargetInstrInfo::INSERT_SUBREG nodes.
SDValue getEntryNode() const
Return the token chain corresponding to the entry of the function.
LLVM_ABI SDDbgValue * getDbgValue(DIVariable *Var, DIExpression *Expr, SDNode *N, unsigned R, bool IsIndirect, const DebugLoc &DL, unsigned O)
Creates a SDDbgValue node.
LLVM_ABI SDValue getMaskedLoad(EVT VT, const SDLoc &dl, SDValue Chain, SDValue Base, SDValue Offset, SDValue Mask, SDValue Src0, EVT MemVT, MachineMemOperand *MMO, ISD::MemIndexedMode AM, ISD::LoadExtType, bool IsExpanding=false)
DenormalMode getDenormalMode(EVT VT) const
Return the current function's default denormal handling kind for the given floating point type.
SDValue getSplat(EVT VT, const SDLoc &DL, SDValue Op)
Returns a node representing a splat of one value into all lanes of the provided vector type.
LLVM_ABI std::pair< SDValue, SDValue > SplitScalar(const SDValue &N, const SDLoc &DL, const EVT &LoVT, const EVT &HiVT)
Split the scalar node with EXTRACT_ELEMENT using the provided VTs and return the low/high part.
LLVM_ABI SDValue matchBinOpReduction(SDNode *Extract, ISD::NodeType &BinOp, ArrayRef< ISD::NodeType > CandidateBinOps, bool AllowPartials=false)
Match a binop + shuffle pyramid that represents a horizontal reduction over the elements of a vector ...
LLVM_ABI bool isADDLike(SDValue Op, bool NoWrap=false) const
Return true if the specified operand is an ISD::OR or ISD::XOR node that can be treated as an ISD::AD...
LLVM_ABI SDValue getVectorShuffle(EVT VT, const SDLoc &dl, SDValue N1, SDValue N2, ArrayRef< int > Mask)
Return an ISD::VECTOR_SHUFFLE node.
LLVM_ABI SDValue simplifyShift(SDValue X, SDValue Y)
Try to simplify a shift into 1 of its operands or a constant.
LLVM_ABI void transferDbgValues(SDValue From, SDValue To, unsigned OffsetInBits=0, unsigned SizeInBits=0, bool InvalidateDbg=true)
Transfer debug values from one node to another, while optionally generating fragment expressions for ...
LLVM_ABI SDValue getLogicalNOT(const SDLoc &DL, SDValue Val, EVT VT)
Create a logical NOT operation as (XOR Val, BooleanOne).
LLVM_ABI SDValue getMaskedScatter(SDVTList VTs, EVT MemVT, const SDLoc &dl, ArrayRef< SDValue > Ops, MachineMemOperand *MMO, ISD::MemIndexType IndexType, bool IsTruncating=false)
ilist< SDNode >::iterator allnodes_iterator
This SDNode is used to implement the code generator support for the llvm IR shufflevector instruction...
int getMaskElt(unsigned Idx) const
ArrayRef< int > getMask() const
static void commuteMask(MutableArrayRef< int > Mask)
Change values in a shuffle permute mask assuming the two vector operands have swapped position.
static LLVM_ABI bool isSplatMask(ArrayRef< int > Mask)
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
bool erase(PtrType Ptr)
Remove pointer from the set.
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void assign(size_type NumElts, ValueParamT Elt)
reference emplace_back(ArgTypes &&... Args)
void reserve(size_type N)
iterator erase(const_iterator CI)
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
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.
Represent a constant reference to a string, i.e.
constexpr const char * data() const
Get a pointer to the start of the string (which may not be null terminated).
Information about stack frame layout on the target.
virtual TargetStackID::Value getStackIDForScalableVectors() const
Returns the StackID that scalable vectors should be associated with.
Align getStackAlign() const
getStackAlignment - This method returns the number of bytes to which the stack pointer must be aligne...
Completely target-dependent object reference.
int64_t getOffset() const
unsigned getTargetFlags() const
Provides information about what library functions are available for the current target.
virtual bool shouldConvertConstantLoadToIntImm(const APInt &Imm, Type *Ty) const
Return true if it is beneficial to convert a load of a constant to just the constant itself.
const TargetMachine & getTargetMachine() const
virtual bool isZExtFree(Type *FromTy, Type *ToTy) const
Return true if any actual instruction that defines a value of type FromTy implicitly zero-extends the...
unsigned getMaxStoresPerMemcpy(bool OptSize) const
Get maximum # of store operations permitted for llvm.memcpy.
unsigned getMaxStoresPerMemset(bool OptSize) const
Get maximum # of store operations permitted for llvm.memset.
virtual bool allowsMisalignedMemoryAccesses(EVT, unsigned AddrSpace=0, Align Alignment=Align(1), MachineMemOperand::Flags Flags=MachineMemOperand::MONone, unsigned *=nullptr) const
Determine if the target supports unaligned memory accesses.
virtual bool shallExtractConstSplatVectorElementToStore(Type *VectorTy, unsigned ElemSizeInBits, unsigned &Index) const
Return true if the target shall perform extract vector element and store given that the vector is kno...
virtual bool isTruncateFree(Type *FromTy, Type *ToTy) const
Return true if it's free to truncate a value of type FromTy to type ToTy.
virtual EVT getTypeToTransformTo(LLVMContext &Context, EVT VT) const
For types supported by the target, this is an identity function.
bool isTypeLegal(EVT VT) const
Return true if the target has native support for the specified value type.
virtual MVT getPointerTy(const DataLayout &DL, uint32_t AS=0) const
Return the pointer type for the given address space, defaults to the pointer type from the data layou...
BooleanContent
Enum that describes how the target represents true/false values.
@ ZeroOrOneBooleanContent
@ UndefinedBooleanContent
@ ZeroOrNegativeOneBooleanContent
virtual unsigned getMaxGluedStoresPerMemcpy() const
Get maximum # of store operations to be glued together.
std::vector< ArgListEntry > ArgListTy
unsigned getMaxStoresPerMemmove(bool OptSize) const
Get maximum # of store operations permitted for llvm.memmove.
virtual bool isLegalStoreImmediate(int64_t Value) const
Return true if the specified immediate is legal for the value input of a store instruction.
static ISD::NodeType getExtendForContent(BooleanContent Content)
This class defines information used to lower LLVM code to legal SelectionDAG operators that the targe...
virtual bool findOptimalMemOpLowering(LLVMContext &Context, std::vector< EVT > &MemOps, unsigned Limit, const MemOp &Op, unsigned DstAS, unsigned SrcAS, const AttributeList &FuncAttributes, EVT *LargestVT=nullptr) const
Determines the optimal series of memory ops to replace the memset / memcpy.
std::pair< SDValue, SDValue > LowerCallTo(CallLoweringInfo &CLI) const
This function lowers an abstract call to a function into an actual call.
Primary interface to the complete machine description for the target machine.
virtual bool isNoopAddrSpaceCast(unsigned SrcAS, unsigned DestAS) const
Returns true if a cast between SrcAS and DestAS is a noop.
const Triple & getTargetTriple() const
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
virtual const SelectionDAGTargetInfo * getSelectionDAGInfo() const
virtual const TargetRegisterInfo * getRegisterInfo() const =0
Return the target's register information.
virtual const TargetLowering * getTargetLowering() const
bool isOSDarwin() const
Is this a "Darwin" OS (macOS, iOS, tvOS, watchOS, DriverKit, XROS, or bridgeOS).
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
static constexpr TypeSize getFixed(ScalarTy ExactSize)
The instances of the Type class are immutable: once they are created, they are never changed.
bool isVectorTy() const
True if this is an instance of VectorType.
static LLVM_ABI IntegerType * getInt32Ty(LLVMContext &C)
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
static LLVM_ABI IntegerType * getInt8Ty(LLVMContext &C)
LLVM_ABI TypeSize getPrimitiveSizeInBits() const LLVM_READONLY
Return the basic size of this type if it is a primitive type.
LLVM_ABI unsigned getScalarSizeInBits() const LLVM_READONLY
If this is a vector type, return the getPrimitiveSizeInBits value for the element type.
A Use represents the edge between a Value definition and its users.
LLVM_ABI unsigned getOperandNo() const
Return the operand # of this use in its User.
LLVM_ABI void set(Value *Val)
User * getUser() const
Returns the User that contains this Use.
Value * getOperand(unsigned i) const
This class is used to represent an VP_GATHER node.
This class is used to represent a VP_LOAD node.
This class is used to represent an VP_SCATTER node.
This class is used to represent a VP_STORE node.
This class is used to represent an EXPERIMENTAL_VP_STRIDED_LOAD node.
This class is used to represent an EXPERIMENTAL_VP_STRIDED_STORE node.
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
std::pair< iterator, bool > insert(const ValueT &V)
bool contains(const_arg_type_t< ValueT > V) const
Check if the set contains the given element.
constexpr bool hasKnownScalarFactor(const FixedOrScalableQuantity &RHS) const
Returns true if there exists a value X where RHS.multiplyCoefficientBy(X) will result in a value whos...
constexpr ScalarTy getFixedValue() const
static constexpr bool isKnownLE(const FixedOrScalableQuantity &LHS, const FixedOrScalableQuantity &RHS)
constexpr bool isScalable() const
Returns whether the quantity is scaled by a runtime quantity (vscale).
constexpr bool isKnownEven() const
A return value of true indicates we know at compile time that the number of elements (vscale * Min) i...
constexpr ScalarTy getKnownMinValue() const
Returns the minimum value this quantity can represent.
constexpr LeafTy divideCoefficientBy(ScalarTy RHS) const
We do not provide the '/' operator here because division for polynomial types does not work in the sa...
static constexpr bool isKnownGE(const FixedOrScalableQuantity &LHS, const FixedOrScalableQuantity &RHS)
A raw_ostream that writes to an std::string.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
LLVM_ABI APInt clmulr(const APInt &LHS, const APInt &RHS)
Perform a reversed carry-less multiply.
LLVM_ABI APInt mulhu(const APInt &C1, const APInt &C2)
Performs (2*N)-bit multiplication on zero-extended operands.
LLVM_ABI APInt avgCeilU(const APInt &C1, const APInt &C2)
Compute the ceil of the unsigned average of C1 and C2.
LLVM_ABI APInt avgFloorU(const APInt &C1, const APInt &C2)
Compute the floor of the unsigned average of C1 and C2.
LLVM_ABI APInt pext(const APInt &Val, const APInt &Mask)
Perform a "compress" operation, also known as pext or bext.
LLVM_ABI APInt fshr(const APInt &Hi, const APInt &Lo, const APInt &Shift)
Perform a funnel shift right.
LLVM_ABI APInt mulhs(const APInt &C1, const APInt &C2)
Performs (2*N)-bit multiplication on sign-extended operands.
LLVM_ABI APInt clmul(const APInt &LHS, const APInt &RHS)
Perform a carry-less multiply, also known as XOR multiplication, and return low-bits.
LLVM_ABI APInt pdep(const APInt &Val, const APInt &Mask)
Perform an "expand" operation, also known as pdep or bdep.
APInt abds(const APInt &A, const APInt &B)
Determine the absolute difference of two APInts considered to be signed.
LLVM_ABI APInt fshl(const APInt &Hi, const APInt &Lo, const APInt &Shift)
Perform a funnel shift left.
LLVM_ABI APInt ScaleBitMask(const APInt &A, unsigned NewBitWidth, bool MatchAllBits=false)
Splat/Merge neighboring bits to widen/narrow the bitmask represented by.
LLVM_ABI APInt clmulh(const APInt &LHS, const APInt &RHS)
Perform a carry-less multiply, and return high-bits.
APInt abdu(const APInt &A, const APInt &B)
Determine the absolute difference of two APInts considered to be unsigned.
LLVM_ABI APInt avgFloorS(const APInt &C1, const APInt &C2)
Compute the floor of the signed average of C1 and C2.
LLVM_ABI APInt avgCeilS(const APInt &C1, const APInt &C2)
Compute the ceil of the signed average of C1 and C2.
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.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ Fast
Attempts to make calls as fast as possible (e.g.
@ C
The default llvm calling convention, compatible with C.
LLVM_ABI CondCode getSetCCInverse(CondCode Operation, bool isIntegerLike)
Return the operation corresponding to !(X op Y), where 'op' is a valid SetCC operation.
ISD namespace - This namespace contains an enum which represents all of the SelectionDAG node types a...
LLVM_ABI CondCode getSetCCAndOperation(CondCode Op1, CondCode Op2, EVT Type)
Return the result of a logical AND between different comparisons of identical values: ((X op1 Y) & (X...
LLVM_ABI bool isConstantSplatVectorAllOnes(const SDNode *N, bool BuildVectorOnly=false)
Return true if the specified node is a BUILD_VECTOR or SPLAT_VECTOR where all of the elements are ~0 ...
bool isNON_EXTLoad(const SDNode *N)
Returns true if the specified node is a non-extending load.
NodeType
ISD::NodeType enum - This enum defines the target-independent operators for a SelectionDAG.
@ 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...
@ MDNODE_SDNODE
MDNODE_SDNODE - This is a node that holdes an MDNode*, which is used to reference metadata in the IR.
@ STRICT_FSETCC
STRICT_FSETCC/STRICT_FSETCCS - Constrained versions of SETCC, used for floating-point operands only.
@ PTRADD
PTRADD represents pointer arithmetic semantics, for targets that opt in using shouldPreservePtrArith(...
@ DELETED_NODE
DELETED_NODE - This is an illegal value that is used to catch errors.
@ POISON
POISON - A poison node.
@ PARTIAL_REDUCE_SMLA
PARTIAL_REDUCE_[U|S]MLA(Accumulator, Input1, Input2) The partial reduction nodes sign or zero extend ...
@ VECREDUCE_SEQ_FADD
Generic reduction nodes.
@ MLOAD
Masked load and store - consecutive vector load and store operations with additional mask operand tha...
@ FGETSIGN
INT = FGETSIGN(FP) - Return the sign bit of the specified floating point value as an integer 0/1 valu...
@ 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.
@ JUMP_TABLE_DEBUG_INFO
JUMP_TABLE_DEBUG_INFO - Jumptable debug info.
@ BSWAP
Byte Swap and Counting operators.
@ DEACTIVATION_SYMBOL
Untyped node storing deactivation symbol reference (DeactivationSymbolSDNode).
@ ATOMIC_STORE
OUTCHAIN = ATOMIC_STORE(INCHAIN, val, ptr) This corresponds to "store atomic" instruction.
@ ADDC
Carry-setting nodes for multiple precision addition and subtraction.
@ FMAD
FMAD - Perform a * b + c, while getting the same result as the separately rounded operations.
@ 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...
@ ANY_EXTEND
ANY_EXTEND - Used for integer types. The high bits are undefined.
@ FMA
FMA - Perform a * b + c with no intermediate rounding step.
@ FATAN2
FATAN2 - atan2, inspired by libm.
@ INTRINSIC_VOID
OUTCHAIN = INTRINSIC_VOID(INCHAIN, INTRINSICID, arg1, arg2, ...) This node represents a target intrin...
@ ATOMIC_CMP_SWAP_WITH_SUCCESS
Val, Success, OUTCHAIN = ATOMIC_CMP_SWAP_WITH_SUCCESS(INCHAIN, ptr, cmp, swap) N.b.
@ SINT_TO_FP
[SU]INT_TO_FP - These operators convert integers (whose interpreted sign depends on the first letter)...
@ CONCAT_VECTORS
CONCAT_VECTORS(VECTOR0, VECTOR1, ...) - Given a number of values of vector type with the same length ...
@ 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 ...
@ FP16_TO_FP
FP16_TO_FP, FP_TO_FP16 - These operators are used to perform promotions and truncation for half-preci...
@ FMULADD
FMULADD - Performs a * b + c, with, or without, intermediate rounding.
@ BITCAST
BITCAST - This operator converts between integer, vector and FP values, as if the value was stored to...
@ BUILD_PAIR
BUILD_PAIR - This is the opposite of EXTRACT_ELEMENT in some ways.
@ CLMUL
Carry-less multiplication operations.
@ FLDEXP
FLDEXP - ldexp, inspired by libm (op0 * 2**op1).
@ BUILTIN_OP_END
BUILTIN_OP_END - This must be the last enum value in this list.
@ SRCVALUE
SRCVALUE - This is a node type that holds a Value* that is used to make reference to a value in the L...
@ EH_LABEL
EH_LABEL - Represents a label in mid basic block used to track locations needed for debug and excepti...
@ SIGN_EXTEND
Conversion operators.
@ AVGCEILS
AVGCEILS/AVGCEILU - Rounding averaging add - Add two integers using an integer of type i[N+2],...
@ SCALAR_TO_VECTOR
SCALAR_TO_VECTOR(VAL) - This represents the operation of loading a scalar value into element 0 of the...
@ VECREDUCE_FADD
These reductions have relaxed evaluation order semantics, and have a single vector operand.
@ TargetIndex
TargetIndex - Like a constant pool entry, but with completely target-dependent semantics.
@ PREFETCH
PREFETCH - This corresponds to a prefetch intrinsic.
@ SETCCCARRY
Like SetCC, ops #0 and #1 are the LHS and RHS operands to compare, but op #2 is a boolean indicating ...
@ FNEG
Perform various unary floating-point operations inspired by libm.
@ BR_CC
BR_CC - Conditional branch.
@ 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.
@ IS_FPCLASS
Performs a check of floating point class property, defined by IEEE-754.
@ SSUBSAT
RESULT = [US]SUBSAT(LHS, RHS) - Perform saturation subtraction on 2 integers with the same bit width ...
@ SELECT
Select(COND, TRUEVAL, FALSEVAL).
@ ATOMIC_LOAD
Val, OUTCHAIN = ATOMIC_LOAD(INCHAIN, ptr) This corresponds to "load atomic" instruction.
@ UNDEF
UNDEF - An undefined node.
@ EXTRACT_ELEMENT
EXTRACT_ELEMENT - This is used to get the lower or upper (determined by a Constant,...
@ SPLAT_VECTOR
SPLAT_VECTOR(VAL) - Returns a vector with the scalar value VAL duplicated in all lanes.
@ AssertAlign
AssertAlign - These nodes record if a register contains a value that has a known alignment and the tr...
@ GET_ACTIVE_LANE_MASK
GET_ACTIVE_LANE_MASK - this corrosponds to the llvm.get.active.lane.mask intrinsic.
@ BasicBlock
Various leaf nodes.
@ CopyFromReg
CopyFromReg - This node indicates that the input value is a virtual or physical register that is defi...
@ SADDO
RESULT, BOOL = [SU]ADDO(LHS, RHS) - Overflow-aware nodes for addition.
@ TargetGlobalAddress
TargetGlobalAddress - Like GlobalAddress, but the DAG does no folding or anything else with this node...
@ ARITH_FENCE
ARITH_FENCE - This corresponds to a arithmetic fence intrinsic.
@ CTLS
Count leading redundant sign bits.
@ 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.
@ AssertNoFPClass
AssertNoFPClass - These nodes record if a register contains a float value that is known to be not som...
@ VECTOR_SHUFFLE
VECTOR_SHUFFLE(VEC1, VEC2) - Returns a vector, of the same type as VEC1/VEC2.
@ EXTRACT_SUBVECTOR
EXTRACT_SUBVECTOR(VECTOR, IDX) - Returns a subvector from VECTOR.
@ FMINNUM_IEEE
FMINNUM_IEEE/FMAXNUM_IEEE - Perform floating-point minimumNumber or maximumNumber on two values,...
@ EntryToken
EntryToken - This is the marker used to indicate the start of a region.
@ EXTRACT_VECTOR_ELT
EXTRACT_VECTOR_ELT(VECTOR, IDX) - Returns a single element from VECTOR identified by the (potentially...
@ CopyToReg
CopyToReg - This node has three operands: a chain, a register number to set to this value,...
@ 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) ...
@ VSCALE
VSCALE(IMM) - Returns the runtime scaling factor used to calculate the number of elements within a sc...
@ ATOMIC_CMP_SWAP
Val, OUTCHAIN = ATOMIC_CMP_SWAP(INCHAIN, ptr, cmp, swap) For double-word atomic operations: ValLo,...
@ FMINNUM
FMINNUM/FMAXNUM - Perform floating-point minimum maximum on two values, following IEEE-754 definition...
@ SSHLSAT
RESULT = [US]SHLSAT(LHS, RHS) - Perform saturation left shift.
@ SMULO
Same for multiplication.
@ VECTOR_SPLICE_LEFT
VECTOR_SPLICE_LEFT(VEC1, VEC2, OFFSET) - Shifts CONCAT_VECTORS(VEC1, VEC2) left by OFFSET elements an...
@ 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.
@ MASKED_UDIV
Masked vector arithmetic that returns poison on disabled lanes.
@ LIFETIME_START
This corresponds to the llvm.lifetime.
@ 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...
@ UADDO_CARRY
Carry-using nodes for multiple precision addition and subtraction.
@ MGATHER
Masked gather and scatter - load and store operations for a vector of random addresses with additiona...
@ HANDLENODE
HANDLENODE node - Used as a handle for various purposes.
@ BF16_TO_FP
BF16_TO_FP, FP_TO_BF16 - These operators are used to perform promotions and truncation for bfloat16.
@ PEXT
Parallel bit extract (compress) and parallel bit deposit (expand).
@ STRICT_FP_ROUND
X = STRICT_FP_ROUND(Y, TRUNC) - Rounding 'Y' from a larger floating point type down to the precision ...
@ 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.
@ TargetConstant
TargetConstant* - Like Constant*, but the DAG does not do any folding, simplification,...
@ 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.
@ INTRINSIC_WO_CHAIN
RESULT = INTRINSIC_WO_CHAIN(INTRINSICID, arg1, arg2, ...) This node represents a target intrinsic fun...
@ GET_FPENV_MEM
Gets the current floating-point environment.
@ PSEUDO_PROBE
Pseudo probe for AutoFDO, as a place holder in a basic block to improve the sample counts quality.
@ 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],...
@ VECTOR_SPLICE_RIGHT
VECTOR_SPLICE_RIGHT(VEC1, VEC2, OFFSET) - Shifts CONCAT_VECTORS(VEC1,VEC2) right by OFFSET elements a...
@ ADDE
Carry-using nodes for multiple precision addition and subtraction.
@ SPLAT_VECTOR_PARTS
SPLAT_VECTOR_PARTS(SCALAR1, SCALAR2, ...) - Returns a vector with the scalar values joined together a...
@ FREEZE
FREEZE - FREEZE(VAL) returns an arbitrary value if VAL is UNDEF (or is evaluated to UNDEF),...
@ INSERT_VECTOR_ELT
INSERT_VECTOR_ELT(VECTOR, VAL, IDX) - Returns VECTOR with the element at IDX replaced with VAL.
@ TokenFactor
TokenFactor - This node takes multiple tokens as input and produces a single token result.
@ ATOMIC_SWAP
Val, OUTCHAIN = ATOMIC_SWAP(INCHAIN, ptr, amt) Val, OUTCHAIN = ATOMIC_LOAD_[OpName](INCHAIN,...
@ CTTZ_ZERO_POISON
Bit counting operators with a poisoned result for zero inputs.
@ 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 ...
@ ADDRSPACECAST
ADDRSPACECAST - This operator converts between pointers of different address spaces.
@ EXPERIMENTAL_VECTOR_HISTOGRAM
Experimental vector histogram intrinsic Operands: Input Chain, Inc, Mask, Base, Index,...
@ 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.
@ VAARG
VAARG - VAARG has four operands: an input chain, a pointer, a SRCVALUE, and the alignment.
@ SHL_PARTS
SHL_PARTS/SRA_PARTS/SRL_PARTS - These operators are used for expanded integer shift operations.
@ 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)...
@ SET_FPENV_MEM
Sets the current floating point environment.
@ FMINIMUMNUM
FMINIMUMNUM/FMAXIMUMNUM - minimumnum/maximumnum that is same with FMINNUM_IEEE and FMAXNUM_IEEE besid...
@ TRUNCATE_SSAT_S
TRUNCATE_[SU]SAT_[SU] - Truncate for saturated operand [SU] located in middle, prefix for SAT means i...
@ ABDS
ABDS/ABDU - Absolute difference - Return the absolute difference between two numbers interpreted as s...
@ SADDO_CARRY
Carry-using overflow-aware nodes for multiple precision addition and subtraction.
@ INTRINSIC_W_CHAIN
RESULT,OUTCHAIN = INTRINSIC_W_CHAIN(INCHAIN, INTRINSICID, arg1, ...) This node represents a target in...
@ ABS_MIN_POISON
ABS with a poison result for INT_MIN.
@ BUILD_VECTOR
BUILD_VECTOR(ELT0, ELT1, ELT2, ELT3,...) - Return a fixed-width vector with the specified,...
LLVM_ABI NodeType getOppositeSignednessMinMaxOpcode(unsigned MinMaxOpc)
Given a MinMaxOpc of ISD::(U|S)MIN or ISD::(U|S)MAX, returns the corresponding opcode with the opposi...
LLVM_ABI bool isBuildVectorOfConstantSDNodes(const SDNode *N)
Return true if the specified node is a BUILD_VECTOR node of all ConstantSDNode or undef.
LLVM_ABI NodeType getExtForLoadExtType(bool IsFP, LoadExtType)
bool isZEXTLoad(const SDNode *N)
Returns true if the specified node is a ZEXTLOAD.
bool isExtOpcode(unsigned Opcode)
LLVM_ABI bool isConstantSplatVectorAllZeros(const SDNode *N, bool BuildVectorOnly=false)
Return true if the specified node is a BUILD_VECTOR or SPLAT_VECTOR where all of the elements are 0 o...
LLVM_ABI NodeType getUnmaskedBinOpOpcode(unsigned MaskedOpc)
Given a MaskedOpc of ISD::MASKED_(U|S)(DIV|REM), returns the unmasked ISD::(U|S)(DIV|REM).
LLVM_ABI bool isVectorShrinkable(const SDNode *N, unsigned NewEltSize, bool Signed)
Returns true if the specified node is a vector where all elements can be truncated to the specified e...
LLVM_ABI bool isVPBinaryOp(unsigned Opcode)
Whether this is a vector-predicated binary operation opcode.
LLVM_ABI CondCode getSetCCInverse(CondCode Operation, EVT Type)
Return the operation corresponding to !(X op Y), where 'op' is a valid SetCC operation.
LLVM_ABI std::optional< unsigned > getBaseOpcodeForVP(unsigned Opcode, bool hasFPExcept)
Translate this VP Opcode to its corresponding non-VP Opcode.
bool isBitwiseLogicOp(unsigned Opcode)
Whether this is bitwise logic opcode.
bool isTrueWhenEqual(CondCode Cond)
Return true if the specified condition returns true if the two operands to the condition are equal.
LLVM_ABI std::optional< unsigned > getVPMaskIdx(unsigned Opcode)
The operand position of the vector mask.
unsigned getUnorderedFlavor(CondCode Cond)
This function returns 0 if the condition is always false if an operand is a NaN, 1 if the condition i...
LLVM_ABI std::optional< unsigned > getVPExplicitVectorLengthIdx(unsigned Opcode)
The operand position of the explicit vector length parameter.
bool isEXTLoad(const SDNode *N)
Returns true if the specified node is a EXTLOAD.
LLVM_ABI bool allOperandsUndef(const SDNode *N)
Return true if the node has at least one operand and all operands of the specified node are ISD::UNDE...
LLVM_ABI bool isFreezeUndef(const SDNode *N)
Return true if the specified node is FREEZE(UNDEF).
LLVM_ABI CondCode getSetCCSwappedOperands(CondCode Operation)
Return the operation corresponding to (Y op X) when given the operation for (X op Y).
LLVM_ABI std::optional< unsigned > getVPForBaseOpcode(unsigned Opcode)
Translate this non-VP Opcode to its corresponding VP Opcode.
MemIndexType
MemIndexType enum - This enum defines how to interpret MGATHER/SCATTER's index parameter when calcula...
LLVM_ABI bool isBuildVectorAllZeros(const SDNode *N)
Return true if the specified node is a BUILD_VECTOR where all of the elements are 0 or undef.
bool matchUnaryPredicateImpl(SDValue Op, std::function< bool(ConstNodeType *)> Match, bool AllowUndefs=false, bool AllowTruncation=false)
Attempt to match a unary predicate against a scalar/splat constant or every element of a constant BUI...
LLVM_ABI bool isConstantSplatVector(const SDNode *N, APInt &SplatValue)
Node predicates.
LLVM_ABI NodeType getInverseMinMaxOpcode(unsigned MinMaxOpc)
Given a MinMaxOpc of ISD::(U|S)MIN or ISD::(U|S)MAX, returns ISD::(U|S)MAX and ISD::(U|S)MIN,...
LLVM_ABI bool matchBinaryPredicate(SDValue LHS, SDValue RHS, std::function< bool(ConstantSDNode *, ConstantSDNode *)> Match, bool AllowUndefs=false, bool AllowTypeMismatch=false)
Attempt to match a binary predicate against a pair of scalar/splat constants or every element of a pa...
LLVM_ABI bool isVPReduction(unsigned Opcode)
Whether this is a vector-predicated reduction opcode.
bool matchUnaryPredicate(SDValue Op, std::function< bool(ConstantSDNode *)> Match, bool AllowUndefs=false, bool AllowTruncation=false)
Hook for matching ConstantSDNode predicate.
MemIndexedMode
MemIndexedMode enum - This enum defines the load / store indexed addressing modes.
LLVM_ABI bool isBuildVectorOfConstantFPSDNodes(const SDNode *N)
Return true if the specified node is a BUILD_VECTOR node of all ConstantFPSDNode or undef.
bool isSEXTLoad(const SDNode *N)
Returns true if the specified node is a SEXTLOAD.
CondCode
ISD::CondCode enum - These are ordered carefully to make the bitfields below work out,...
LLVM_ABI bool isBuildVectorAllOnes(const SDNode *N)
Return true if the specified node is a BUILD_VECTOR where all of the elements are ~0 or undef.
LLVM_ABI NodeType getVecReduceBaseOpcode(unsigned VecReduceOpcode)
Get underlying scalar opcode for VECREDUCE opcode.
LoadExtType
LoadExtType enum - This enum defines the three variants of LOADEXT (load with extension).
LLVM_ABI bool isVPOpcode(unsigned Opcode)
Whether this is a vector-predicated Opcode.
LLVM_ABI CondCode getSetCCOrOperation(CondCode Op1, CondCode Op2, EVT Type)
Return the result of a logical OR between different comparisons of identical values: ((X op1 Y) | (X ...
BinaryOp_match< SpecificConstantMatch, SrcTy, TargetOpcode::G_SUB > m_Neg(const SrcTy &&Src)
Matches a register negated by a G_SUB.
BinaryOp_match< LHS, RHS, Instruction::And > m_And(const LHS &L, const RHS &R)
match_deferred< Value > m_Deferred(Value *const &V)
Like m_Specific(), but works if the specific value to match is determined as part of the same match()...
auto m_Value()
Match an arbitrary value and ignore it.
BinaryOp_match< LHS, RHS, Instruction::Sub > m_Sub(const LHS &L, const RHS &R)
LLVM_ABI Libcall getMEMCPY_ELEMENT_UNORDERED_ATOMIC(uint64_t ElementSize)
getMEMCPY_ELEMENT_UNORDERED_ATOMIC - Return MEMCPY_ELEMENT_UNORDERED_ATOMIC_* value for the given ele...
LLVM_ABI Libcall getMEMSET_ELEMENT_UNORDERED_ATOMIC(uint64_t ElementSize)
getMEMSET_ELEMENT_UNORDERED_ATOMIC - Return MEMSET_ELEMENT_UNORDERED_ATOMIC_* value for the given ele...
LLVM_ABI Libcall getMEMMOVE_ELEMENT_UNORDERED_ATOMIC(uint64_t ElementSize)
getMEMMOVE_ELEMENT_UNORDERED_ATOMIC - Return MEMMOVE_ELEMENT_UNORDERED_ATOMIC_* value for the given e...
bool sd_match(SDNode *N, const SelectionDAG *DAG, Pattern &&P)
initializer< Ty > init(const Ty &Val)
@ DW_OP_LLVM_arg
Only used in LLVM metadata.
std::enable_if_t< detail::IsValidPointer< X, Y >::value, X * > extract(Y &&MD)
Extract a Value from Metadata.
NodeAddr< NodeBase * > Node
This is an optimization pass for GlobalISel generic memory operations.
GenericUniformityInfo< SSAContext > UniformityInfo
unsigned Log2_32_Ceil(uint32_t Value)
Return the ceil log base 2 of the specified value, 32 if the value is zero.
bool operator<(int64_t V1, const APSInt &V2)
LLVM_ABI ISD::CondCode getICmpCondCode(ICmpInst::Predicate Pred)
getICmpCondCode - Return the ISD condition code corresponding to the given LLVM IR integer condition ...
void fill(R &&Range, T &&Value)
Provide wrappers to std::fill which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI SDValue peekThroughExtractSubvectors(SDValue V)
Return the non-extracted vector source operand of V if it exists.
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI bool isNullConstant(SDValue V)
Returns true if V is a constant integer zero.
LLVM_ABI bool isAllOnesOrAllOnesSplat(const MachineInstr &MI, const MachineRegisterInfo &MRI, bool AllowUndefs=false)
Return true if the value is a constant -1 integer or a splatted vector of a constant -1 integer (with...
LLVM_ABI SDValue getBitwiseNotOperand(SDValue V, SDValue Mask, bool AllowUndefs)
If V is a bitwise not, returns the inverted operand.
@ Undef
Value of the register doesn't matter.
LLVM_ABI SDValue peekThroughBitcasts(SDValue V)
Return the non-bitcasted source operand of V if it exists.
auto enumerate(FirstRange &&First, RestRanges &&...Rest)
Given two or more input ranges, returns a new range whose values are tuples (A, B,...
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
int countr_one(T Value)
Count the number of ones from the least significant bit to the first zero bit.
bool isIntOrFPConstant(SDValue V)
Return true if V is either a integer or FP constant.
auto dyn_cast_if_present(const Y &Val)
dyn_cast_if_present<X> - Functionally identical to dyn_cast, except that a null (or none in the case ...
LLVM_ABI bool getConstantDataArrayInfo(const Value *V, ConstantDataArraySlice &Slice, unsigned ElementSize, uint64_t Offset=0)
Returns true if the value V is a pointer into a ConstantDataArray.
LLVM_ABI bool isOneOrOneSplatFP(SDValue V, bool AllowUndefs=false)
Return true if the value is a constant floating-point value, or a splatted vector of a constant float...
int bit_width(T Value)
Returns the number of bits needed to represent Value if Value is nonzero.
LLVM_READONLY APFloat maximum(const APFloat &A, const APFloat &B)
Implements IEEE 754-2019 maximum semantics.
void append_range(Container &C, Range &&R)
Wrapper function to append range R to container C.
constexpr bool isUIntN(unsigned N, uint64_t x)
Checks if an unsigned integer fits into the given (dynamic) bit width.
LLVM_ABI bool shouldOptimizeForSize(const MachineFunction *MF, ProfileSummaryInfo *PSI, const MachineBlockFrequencyInfo *BFI, PGSOQueryType QueryType=PGSOQueryType::Other)
Returns true if machine function MF is suggested to be size-optimized based on the profile.
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
auto cast_or_null(const Y &Val)
LLVM_ABI bool isNullOrNullSplat(const MachineInstr &MI, const MachineRegisterInfo &MRI, bool AllowUndefs=false)
Return true if the value is a constant 0 integer or a splatted vector of a constant 0 integer (with n...
LLVM_ABI bool isMinSignedConstant(SDValue V)
Returns true if V is a constant min signed integer value.
LLVM_ABI ConstantFPSDNode * isConstOrConstSplatFP(SDValue N, bool AllowUndefs=false)
Returns the SDNode if it is a constant splat BuildVector or constant float.
LLVM_ABI ConstantRange getConstantRangeFromMetadata(const MDNode &RangeMD)
Parse out a conservative ConstantRange from !range metadata.
APFloat frexp(const APFloat &X, int &Exp, APFloat::roundingMode RM)
Equivalent of C standard library function.
int countr_zero(T Val)
Count number of 0's from the least significant bit to the most stopping at the first 1.
auto dyn_cast_or_null(const Y &Val)
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI bool getShuffleDemandedElts(int SrcWidth, ArrayRef< int > Mask, const APInt &DemandedElts, APInt &DemandedLHS, APInt &DemandedRHS, bool AllowUndefElts=false)
Transform a shuffle mask's output demanded element mask into demanded element masks for the 2 operand...
LLVM_READONLY APFloat maxnum(const APFloat &A, const APFloat &B)
Implements IEEE-754 2008 maxNum semantics.
unsigned Log2_32(uint32_t Value)
Return the floor log base 2 of the specified value, -1 if the value is zero.
LLVM_ABI bool isBitwiseNot(SDValue V, bool AllowUndefs=false)
Returns true if V is a bitwise not operation.
LLVM_ABI SDValue peekThroughInsertVectorElt(SDValue V, const APInt &DemandedElts)
Recursively peek through INSERT_VECTOR_ELT nodes, returning the source vector operand of V,...
constexpr bool isPowerOf2_32(uint32_t Value)
Return true if the argument is a power of two > 0.
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
LLVM_ABI void checkForCycles(const SelectionDAG *DAG, bool force=false)
void sort(IteratorTy Start, IteratorTy End)
LLVM_READONLY APFloat minimumnum(const APFloat &A, const APFloat &B)
Implements IEEE 754-2019 minimumNumber semantics.
FPClassTest
Floating-point class tests, supported by 'is_fpclass' intrinsic.
LLVM_ABI void computeKnownBits(const Value *V, KnownBits &Known, const DataLayout &DL, AssumptionCache *AC=nullptr, const Instruction *CxtI=nullptr, const DominatorTree *DT=nullptr, bool UseInstrInfo=true, unsigned Depth=0)
Determine which bits of V are known to be either zero or one and return them in the KnownZero/KnownOn...
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI SDValue peekThroughTruncates(SDValue V)
Return the non-truncated source operand of V if it exists.
bool none_of(R &&Range, UnaryPredicate P)
Provide wrappers to std::none_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
constexpr std::underlying_type_t< Enum > to_underlying(Enum E)
Returns underlying integer value of an enum.
LLVM_ABI ConstantRange getVScaleRange(const Function *F, unsigned BitWidth)
Determine the possible constant range of vscale with the given bit width, based on the vscale_range f...
LLVM_ABI SDValue peekThroughOneUseBitcasts(SDValue V)
Return the non-bitcasted and one-use source operand of V if it exists.
CodeGenOptLevel
Code generation optimization level.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
bool includesPoison(UndefPoisonKind Kind)
Returns true if Kind includes the Poison bit.
LLVM_ABI bool isOneOrOneSplat(SDValue V, bool AllowUndefs=false)
Return true if the value is a constant 1 integer or a splatted vector of a constant 1 integer (with n...
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ First
Helpers to iterate all locations in the MemoryEffectsBase class.
bool includesUndef(UndefPoisonKind Kind)
Returns true if Kind includes the Undef bit.
LLVM_READONLY APFloat minnum(const APFloat &A, const APFloat &B)
Implements IEEE-754 2008 minNum semantics.
@ Mul
Product of integers.
@ Sub
Subtraction of integers.
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr Count
LLVM_ABI bool isNullConstantOrUndef(SDValue V)
Returns true if V is a constant integer zero or an UNDEF node.
LLVM_ABI bool isInTailCallPosition(const CallBase &Call, const TargetMachine &TM, bool ReturnsFirstArg=false)
Test if the given instruction is in a position to be optimized with a tail-call.
DWARFExpression::Operation Op
ArrayRef(const T &OneElt) -> ArrayRef< T >
LLVM_ABI ConstantSDNode * isConstOrConstSplat(SDValue N, bool AllowUndefs=false, bool AllowTruncation=false)
Returns the SDNode if it is a constant splat BuildVector or constant int.
OutputIt copy(R &&Range, OutputIt Out)
constexpr unsigned BitWidth
LLVM_ABI bool funcReturnsFirstArgOfCall(const CallInst &CI)
Returns true if the parent of CI returns CI's first argument after calling CI.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI bool isZeroOrZeroSplat(SDValue N, bool AllowUndefs=false)
Return true if the value is a constant 0 integer or a splatted vector of a constant 0 integer (with n...
LLVM_ABI bool isOneConstant(SDValue V)
Returns true if V is a constant integer one.
UndefPoisonKind
Enumeration to track whether we are interested in Undef, Poison, or both.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
Align commonAlignment(Align A, uint64_t Offset)
Returns the alignment that satisfies both alignments.
LLVM_ABI bool isNullFPConstant(SDValue V)
Returns true if V is an FP constant with a value of positive zero.
constexpr int64_t SignExtend64(uint64_t x)
Sign-extend the number in the bottom B bits of X to a 64-bit integer.
unsigned Log2(Align A)
Returns the log2 of the alignment.
LLVM_ABI bool isZeroOrZeroSplatFP(SDValue N, bool AllowUndefs=false)
Return true if the value is a constant (+/-)0.0 floating-point value or a splatted vector thereof (wi...
LLVM_ABI void computeKnownBitsFromRangeMetadata(const MDNode &Ranges, KnownBits &Known)
Compute known bits from the range metadata.
LLVM_READONLY APFloat minimum(const APFloat &A, const APFloat &B)
Implements IEEE 754-2019 minimum semantics.
LLVM_READONLY APFloat maximumnum(const APFloat &A, const APFloat &B)
Implements IEEE 754-2019 maximumNumber semantics.
LLVM_ABI bool isOnesOrOnesSplat(SDValue N, bool AllowUndefs=false)
Return true if the value is a constant 1 integer or a splatted vector of a constant 1 integer (with n...
LLVM_ABI bool isAllOnesConstant(SDValue V)
Returns true if V is an integer constant with all bits set.
constexpr uint64_t NextPowerOf2(uint64_t A)
Returns the next power of two (in 64-bits) that is strictly greater than A.
LLVM_ABI void reportFatalUsageError(Error Err)
Report a fatal error that does not indicate a bug in LLVM.
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.
A collection of metadata nodes that might be associated with a memory access used by the alias-analys...
MDNode * TBAAStruct
The tag for type-based alias analysis (tbaa struct).
MDNode * TBAA
The tag for type-based alias analysis.
This struct is a compact representation of a valid (non-zero power of two) alignment.
constexpr uint64_t value() const
This is a hole in the type system and should not be abused.
Represents offset+length into a ConstantDataArray.
uint64_t Length
Length of the slice.
uint64_t Offset
Slice starts at this Offset.
void move(uint64_t Delta)
Moves the Offset and adjusts Length accordingly.
const ConstantDataArray * Array
ConstantDataArray pointer.
TypeSize getStoreSize() const
Return the number of bytes overwritten by a store of the specified value type.
bool isSimple() const
Test if the given EVT is simple (as opposed to being extended).
intptr_t getRawBits() const
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.
EVT changeTypeToInteger() const
Return the type converted to an equivalently sized integer or vector with integer element type.
bool bitsGT(EVT VT) const
Return true if this has more bits than VT.
bool bitsLT(EVT VT) const
Return true if this has less bits than VT.
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
ElementCount getVectorElementCount() const
TypeSize getSizeInBits() const
Return the size of the specified value type in bits.
unsigned getVectorMinNumElements() const
Given a vector type, return the minimum number of elements it contains.
uint64_t getScalarSizeInBits() const
MVT getSimpleVT() const
Return the SimpleValueType held in the specified simple EVT.
static EVT getIntegerVT(LLVMContext &Context, unsigned BitWidth)
Returns the EVT that represents an integer with the given number of bits.
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.
bool bitsGE(EVT VT) const
Return true if this has no less bits than VT.
bool bitsEq(EVT VT) const
Return true if this has the same number of bits as VT.
LLVM_ABI 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.
bool isExtended() const
Test if the given EVT is extended (as opposed to being simple).
LLVM_ABI const fltSemantics & getFltSemantics() const
Returns an APFloat semantics tag appropriate for the value type.
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.
EVT getHalfNumVectorElementsVT(LLVMContext &Context) const
bool isInteger() const
Return true if this is an integer or a vector integer type.
static KnownBits makeConstant(const APInt &C)
Create known bits from a known constant.
LLVM_ABI KnownBits sextInReg(unsigned SrcBitWidth) const
Return known bits for a in-register sign extension of the value we're tracking.
static LLVM_ABI KnownBits mulhu(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits from zero-extended multiply-hi.
unsigned countMinSignBits() const
Returns the number of times the sign bit is replicated into the other bits.
static LLVM_ABI KnownBits smax(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for smax(LHS, RHS).
bool isNonNegative() const
Returns true if this value is known to be non-negative.
bool isZero() const
Returns true if value is all zero.
void makeNonNegative()
Make this value non-negative.
static LLVM_ABI KnownBits usub_sat(const KnownBits &LHS, const KnownBits &RHS)
Compute knownbits resulting from llvm.usub.sat(LHS, RHS)
unsigned countMinTrailingZeros() const
Returns the minimum number of trailing zero bits.
static LLVM_ABI KnownBits ashr(const KnownBits &LHS, const KnownBits &RHS, bool ShAmtNonZero=false, bool Exact=false)
Compute known bits for ashr(LHS, RHS).
static LLVM_ABI KnownBits urem(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for urem(LHS, RHS).
bool isUnknown() const
Returns true if we don't know any bits.
unsigned countMaxTrailingZeros() const
Returns the maximum number of trailing zero bits possible.
static LLVM_ABI std::optional< bool > ne(const KnownBits &LHS, const KnownBits &RHS)
Determine if these known bits always give the same ICMP_NE result.
void makeNegative()
Make this value negative.
void setAllConflict()
Make all bits known to be both zero and one.
KnownBits trunc(unsigned BitWidth) const
Return known bits for a truncation of the value we're tracking.
KnownBits byteSwap() const
static LLVM_ABI KnownBits fshl(const KnownBits &LHS, const KnownBits &RHS, const APInt &Amt)
Compute known bits for fshl(LHS, RHS, Amt).
unsigned countMaxPopulation() const
Returns the maximum number of bits that could be one.
void setAllZero()
Make all bits known to be zero and discard any previous information.
KnownBits reverseBits() const
KnownBits concat(const KnownBits &Lo) const
Concatenate the bits from Lo onto the bottom of *this.
unsigned getBitWidth() const
Get the bit width of this value.
static LLVM_ABI KnownBits umax(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for umax(LHS, RHS).
KnownBits zext(unsigned BitWidth) const
Return known bits for a zero extension of the value we're tracking.
bool isConstant() const
Returns true if we know the value of all bits.
void resetAll()
Resets the known state of all bits.
static KnownBits add(const KnownBits &LHS, const KnownBits &RHS, bool NSW=false, bool NUW=false, bool SelfAdd=false)
Compute knownbits resulting from addition of LHS and RHS.
static LLVM_ABI KnownBits lshr(const KnownBits &LHS, const KnownBits &RHS, bool ShAmtNonZero=false, bool Exact=false)
Compute known bits for lshr(LHS, RHS).
bool isNonZero() const
Returns true if this value is known to be non-zero.
static LLVM_ABI KnownBits abdu(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for abdu(LHS, RHS).
KnownBits extractBits(unsigned NumBits, unsigned BitPosition) const
Return a subset of the known bits from [bitPosition,bitPosition+numBits).
static LLVM_ABI KnownBits pdep(const KnownBits &Val, const KnownBits &Mask)
Compute known bits for pdep(Val, Mask).
static LLVM_ABI KnownBits avgFloorU(const KnownBits &LHS, const KnownBits &RHS)
Compute knownbits resulting from APIntOps::avgFloorU.
KnownBits intersectWith(const KnownBits &RHS) const
Returns KnownBits information that is known to be true for both this and RHS.
KnownBits sext(unsigned BitWidth) const
Return known bits for a sign extension of the value we're tracking.
static LLVM_ABI KnownBits computeForSubBorrow(const KnownBits &LHS, KnownBits RHS, const KnownBits &Borrow)
Compute known bits results from subtracting RHS from LHS with 1-bit Borrow.
KnownBits zextOrTrunc(unsigned BitWidth) const
Return known bits for a zero extension or truncation of the value we're tracking.
APInt getMaxValue() const
Return the maximal unsigned value possible given these KnownBits.
static LLVM_ABI KnownBits fshr(const KnownBits &LHS, const KnownBits &RHS, const APInt &Amt)
Compute known bits for fshr(LHS, RHS, Amt).
static LLVM_ABI KnownBits abds(KnownBits LHS, KnownBits RHS)
Compute known bits for abds(LHS, RHS).
static LLVM_ABI KnownBits smin(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for smin(LHS, RHS).
static LLVM_ABI KnownBits mulhs(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits from sign-extended multiply-hi.
static LLVM_ABI KnownBits srem(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for srem(LHS, RHS).
static LLVM_ABI KnownBits udiv(const KnownBits &LHS, const KnownBits &RHS, bool Exact=false)
Compute known bits for udiv(LHS, RHS).
bool isStrictlyPositive() const
Returns true if this value is known to be positive.
static LLVM_ABI KnownBits sdiv(const KnownBits &LHS, const KnownBits &RHS, bool Exact=false)
Compute known bits for sdiv(LHS, RHS).
static LLVM_ABI KnownBits avgFloorS(const KnownBits &LHS, const KnownBits &RHS)
Compute knownbits resulting from APIntOps::avgFloorS.
static bool haveNoCommonBitsSet(const KnownBits &LHS, const KnownBits &RHS)
Return true if LHS and RHS have no common bits set.
bool isNegative() const
Returns true if this value is known to be negative.
LLVM_ABI KnownBits truncSSat(unsigned BitWidth) const
Truncate with signed saturation (signed input -> signed output)
static LLVM_ABI KnownBits computeForAddCarry(const KnownBits &LHS, const KnownBits &RHS, const KnownBits &Carry)
Compute known bits resulting from adding LHS, RHS and a 1-bit Carry.
static KnownBits sub(const KnownBits &LHS, const KnownBits &RHS, bool NSW=false, bool NUW=false)
Compute knownbits resulting from subtraction of LHS and RHS.
unsigned countMaxLeadingZeros() const
Returns the maximum number of leading zero bits possible.
void insertBits(const KnownBits &SubBits, unsigned BitPosition)
Insert the bits from a smaller known bits starting at bitPosition.
static LLVM_ABI KnownBits avgCeilU(const KnownBits &LHS, const KnownBits &RHS)
Compute knownbits resulting from APIntOps::avgCeilU.
static LLVM_ABI KnownBits mul(const KnownBits &LHS, const KnownBits &RHS, bool NoUndefSelfMultiply=false)
Compute known bits resulting from multiplying LHS and RHS.
KnownBits anyext(unsigned BitWidth) const
Return known bits for an "any" extension of the value we're tracking, where we don't know anything ab...
static LLVM_ABI KnownBits clmul(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for clmul(LHS, RHS).
LLVM_ABI KnownBits abs(bool IntMinIsPoison=false) const
Compute known bits for the absolute value.
LLVM_ABI KnownBits truncUSat(unsigned BitWidth) const
Truncate with unsigned saturation (unsigned input -> unsigned output)
static LLVM_ABI KnownBits shl(const KnownBits &LHS, const KnownBits &RHS, bool NUW=false, bool NSW=false, bool ShAmtNonZero=false)
Compute known bits for shl(LHS, RHS).
static LLVM_ABI KnownBits umin(const KnownBits &LHS, const KnownBits &RHS)
Compute known bits for umin(LHS, RHS).
LLVM_ABI KnownBits truncSSatU(unsigned BitWidth) const
Truncate with signed saturation to unsigned (signed input -> unsigned output)
static LLVM_ABI KnownBits pext(const KnownBits &Val, const KnownBits &Mask)
Compute known bits for pext(Val, Mask).
static LLVM_ABI KnownBits avgCeilS(const KnownBits &LHS, const KnownBits &RHS)
Compute knownbits resulting from APIntOps::avgCeilS.
const APInt & getConstant() const
Returns the value when all bits have a known value.
FPClassTest KnownFPClasses
Floating-point classes the value could be one of.
void copysign(const KnownFPClass &Sign)
LLVM_ABI bool isKnownNeverLogicalZero(DenormalMode Mode) const
Return true if it's known this can never be interpreted as a zero.
KnownFPClass intersectWith(const KnownFPClass &RHS) const
std::optional< bool > SignBit
std::nullopt if the sign bit is unknown, true if the sign bit is definitely set or false if the sign ...
bool isKnownNever(FPClassTest Mask) const
Return true if it's known this can never be one of the mask entries.
static LLVM_ABI KnownFPClass bitcast(const fltSemantics &FltSemantics, const KnownBits &Bits)
Report known values for a bitcast into a float with provided semantics.
This class contains a discriminated union of information about pointers in memory operands,...
LLVM_ABI bool isDereferenceable(unsigned Size, LLVMContext &C, const DataLayout &DL) const
Return true if memory region [V, V+Offset+Size) is known to be dereferenceable.
LLVM_ABI unsigned getAddrSpace() const
Return the LLVM IR address space number that this pointer points into.
PointerUnion< const Value *, const PseudoSourceValue * > V
This is the IR pointer value for the access, or it is null if unknown.
MachinePointerInfo getWithOffset(int64_t O) const
static LLVM_ABI MachinePointerInfo getFixedStack(MachineFunction &MF, int FI, int64_t Offset=0)
Return a MachinePointerInfo record that refers to the specified FrameIndex.
This struct is a compact representation of a valid (power of two) or undefined (0) alignment.
Align valueOrOne() const
For convenience, returns a valid alignment or 1 if undefined.
static MemOp Set(uint64_t Size, bool DstAlignCanChange, Align DstAlign, bool IsZeroMemset, bool IsVolatile)
static MemOp Copy(uint64_t Size, bool DstAlignCanChange, Align DstAlign, Align SrcAlign, bool IsVolatile, bool MemcpyStrSrc=false)
static StringRef getLibcallImplName(RTLIB::LibcallImpl CallImpl)
Get the libcall routine name for the specified libcall implementation.
These are IR-level optimization flags that may be propagated to SDNodes.
This represents a list of ValueType's that has been intern'd by a SelectionDAG.
Clients of various APIs that cause global effects on the DAG can optionally implement this interface.
DAGUpdateListener *const Next
virtual void NodeDeleted(SDNode *N, SDNode *E)
The node N that was deleted and, if E is not null, an equivalent node E that replaced it.
virtual void NodeInserted(SDNode *N)
The node N that was inserted.
virtual void NodeUpdated(SDNode *N)
The node N that was updated.
This structure contains all information that is necessary for lowering calls.
CallLoweringInfo & setLibCallee(CallingConv::ID CC, Type *ResultType, SDValue Target, ArgListTy &&ArgsList)
CallLoweringInfo & setDiscardResult(bool Value=true)
CallLoweringInfo & setDebugLoc(const SDLoc &dl)
CallLoweringInfo & setTailCall(bool Value=true)
CallLoweringInfo & setChain(SDValue InChain)