79#include "llvm/IR/IntrinsicsAArch64.h"
80#include "llvm/IR/IntrinsicsAMDGPU.h"
81#include "llvm/IR/IntrinsicsWebAssembly.h"
114#define DEBUG_TYPE "isel"
122 cl::desc(
"Insert the experimental `assertalign` node."),
127 cl::desc(
"Generate low-precision inline sequences "
128 "for some float libcalls"),
134 cl::desc(
"Set the case probability threshold for peeling the case from a "
135 "switch statement. A value greater than 100 will void this "
155 const SDValue *Parts,
unsigned NumParts,
158 std::optional<CallingConv::ID> CC);
167 unsigned NumParts,
MVT PartVT,
EVT ValueVT,
const Value *V,
169 std::optional<CallingConv::ID> CC = std::nullopt,
170 std::optional<ISD::NodeType> AssertOp = std::nullopt) {
174 PartVT, ValueVT, CC))
181 assert(NumParts > 0 &&
"No parts to assemble!");
192 unsigned RoundBits = PartBits * RoundParts;
193 EVT RoundVT = RoundBits == ValueBits ?
199 if (RoundParts > 2) {
203 PartVT, HalfVT, V, InChain);
214 if (RoundParts < NumParts) {
216 unsigned OddParts = NumParts - RoundParts;
219 OddVT, V, InChain, CC);
235 assert(ValueVT ==
EVT(MVT::ppcf128) && PartVT == MVT::f64 &&
246 !PartVT.
isVector() &&
"Unexpected split");
258 if (PartEVT == ValueVT)
262 ValueVT.
bitsLT(PartEVT)) {
275 if (ValueVT.
bitsLT(PartEVT)) {
280 Val = DAG.
getNode(*AssertOp,
DL, PartEVT, Val,
295 llvm::Attribute::StrictFP)) {
297 DAG.
getVTList(ValueVT, MVT::Other), InChain, Val,
309 if (PartEVT == MVT::x86mmx && ValueVT.
isInteger() &&
310 ValueVT.
bitsLT(PartEVT)) {
319 const Twine &ErrMsg) {
322 return Ctx.emitError(ErrMsg);
325 if (CI->isInlineAsm()) {
327 *CI, ErrMsg +
", possible invalid constraint for vector type"));
330 return Ctx.emitError(
I, ErrMsg);
339 const SDValue *Parts,
unsigned NumParts,
342 std::optional<CallingConv::ID> CallConv) {
344 assert(NumParts > 0 &&
"No parts to assemble!");
345 const bool IsABIRegCopy = CallConv.has_value();
354 unsigned NumIntermediates;
359 *DAG.
getContext(), *CallConv, ValueVT, IntermediateVT,
360 NumIntermediates, RegisterVT);
364 NumIntermediates, RegisterVT);
367 assert(NumRegs == NumParts &&
"Part count doesn't match vector breakdown!");
369 assert(RegisterVT == PartVT &&
"Part type doesn't match vector breakdown!");
372 "Part type sizes don't match!");
376 if (NumIntermediates == NumParts) {
379 for (
unsigned i = 0; i != NumParts; ++i)
381 V, InChain, CallConv);
382 }
else if (NumParts > 0) {
385 assert(NumParts % NumIntermediates == 0 &&
386 "Must expand into a divisible number of parts!");
387 unsigned Factor = NumParts / NumIntermediates;
388 for (
unsigned i = 0; i != NumIntermediates; ++i)
390 IntermediateVT, V, InChain, CallConv);
405 DL, BuiltVectorTy,
Ops);
411 if (PartEVT == ValueVT)
427 "Cannot narrow, it would be a lossy transformation");
433 if (PartEVT == ValueVT)
458 }
else if (ValueVT.
bitsLT(PartEVT)) {
467 *DAG.
getContext(), V,
"non-trivial scalar-to-vector conversion");
498 std::optional<CallingConv::ID> CallConv);
505 unsigned NumParts,
MVT PartVT,
const Value *V,
506 std::optional<CallingConv::ID> CallConv = std::nullopt,
520 unsigned OrigNumParts = NumParts;
522 "Copying to an illegal type!");
528 EVT PartEVT = PartVT;
529 if (PartEVT == ValueVT) {
530 assert(NumParts == 1 &&
"No-op copy with multiple parts!");
539 assert(NumParts == 1 &&
"Do not know what to promote to!");
550 "Unknown mismatch!");
552 Val = DAG.
getNode(ExtendKind,
DL, ValueVT, Val);
553 if (PartVT == MVT::x86mmx)
558 assert(NumParts == 1 && PartEVT != ValueVT);
564 "Unknown mismatch!");
567 if (PartVT == MVT::x86mmx)
574 "Failed to tile the value with PartVT!");
577 if (PartEVT != ValueVT) {
579 "scalar-to-vector conversion failed");
588 if (NumParts & (NumParts - 1)) {
591 "Do not know what to expand to!");
593 unsigned RoundBits = RoundParts * PartBits;
594 unsigned OddParts = NumParts - RoundParts;
603 std::reverse(Parts + RoundParts, Parts + NumParts);
605 NumParts = RoundParts;
617 for (
unsigned StepSize = NumParts; StepSize > 1; StepSize /= 2) {
618 for (
unsigned i = 0; i < NumParts; i += StepSize) {
619 unsigned ThisBits = StepSize * PartBits / 2;
622 SDValue &Part1 = Parts[i+StepSize/2];
629 if (ThisBits == PartBits && ThisVT != PartVT) {
637 std::reverse(Parts, Parts + OrigNumParts);
659 if (ValueEVT == MVT::bf16 && PartEVT == MVT::f16) {
661 "Cannot widen to illegal type");
665 }
else if (PartEVT != ValueEVT) {
680 Ops.append((PartNumElts - ValueNumElts).getFixedValue(), EltUndef);
691 std::optional<CallingConv::ID> CallConv) {
695 const bool IsABIRegCopy = CallConv.has_value();
698 EVT PartEVT = PartVT;
699 if (PartEVT == ValueVT) {
745 "lossy conversion of vector to scalar type");
760 unsigned NumIntermediates;
764 *DAG.
getContext(), *CallConv, ValueVT, IntermediateVT, NumIntermediates,
769 NumIntermediates, RegisterVT);
772 assert(NumRegs == NumParts &&
"Part count doesn't match vector breakdown!");
774 assert(RegisterVT == PartVT &&
"Part type doesn't match vector breakdown!");
777 "Mixing scalable and fixed vectors when copying in parts");
779 std::optional<ElementCount> DestEltCnt;
789 if (ValueVT == BuiltVectorTy) {
813 for (
unsigned i = 0; i != NumIntermediates; ++i) {
828 if (NumParts == NumIntermediates) {
831 for (
unsigned i = 0; i != NumParts; ++i)
833 }
else if (NumParts > 0) {
836 assert(NumIntermediates != 0 &&
"division by zero");
837 assert(NumParts % NumIntermediates == 0 &&
838 "Must expand into a divisible number of parts!");
839 unsigned Factor = NumParts / NumIntermediates;
840 for (
unsigned i = 0; i != NumIntermediates; ++i)
848 if (
I.hasOperandBundlesOtherThan(AllowedBundles)) {
852 for (
unsigned i = 0, e =
I.getNumOperandBundles(); i != e; ++i) {
855 OS << LS << U.getTagName();
858 Twine(
"cannot lower ", Name)
864 EVT valuevt, std::optional<CallingConv::ID> CC)
870 std::optional<CallingConv::ID> CC) {
884 for (
unsigned i = 0; i != NumRegs; ++i)
885 Regs.push_back(Reg + i);
886 RegVTs.push_back(RegisterVT);
888 Reg = Reg.id() + NumRegs;
915 for (
unsigned i = 0; i != NumRegs; ++i) {
921 *Glue =
P.getValue(2);
924 Chain =
P.getValue(1);
952 EVT FromVT(MVT::Other);
956 }
else if (NumSignBits > 1) {
964 assert(FromVT != MVT::Other);
970 RegisterVT, ValueVT, V, Chain,
CallConv);
986 unsigned NumRegs =
Regs.size();
1001 NumParts, RegisterVT, V,
CallConv, ExtendKind);
1007 for (
unsigned i = 0; i != NumRegs; ++i) {
1019 if (NumRegs == 1 || Glue)
1030 Chain = Chains[NumRegs-1];
1036 unsigned MatchingIdx,
const SDLoc &dl,
1038 std::vector<SDValue> &
Ops)
const {
1043 Flag.setMatchingOp(MatchingIdx);
1044 else if (!
Regs.empty() &&
Regs.front().isVirtual()) {
1052 Flag.setRegClass(RC->
getID());
1063 "No 1:1 mapping from clobbers to regs?");
1066 for (
unsigned I = 0, E =
ValueVTs.size();
I != E; ++
I) {
1071 "If we clobbered the stack pointer, MFI should know about it.");
1080 for (
unsigned i = 0; i != NumRegs; ++i) {
1081 assert(Reg <
Regs.size() &&
"Mismatch in # registers expected");
1093 unsigned RegCount = std::get<0>(CountAndVT);
1094 MVT RegisterVT = std::get<1>(CountAndVT);
1112 SL->init(
DAG.getTargetLoweringInfo(), TM,
DAG.getDataLayout());
1114 *
DAG.getMachineFunction().getFunction().getParent());
1119 UnusedArgNodeMap.clear();
1121 PendingExports.clear();
1122 PendingConstrainedFP.clear();
1123 PendingConstrainedFPStrict.clear();
1131 DanglingDebugInfoMap.clear();
1138 if (Pending.
empty())
1144 unsigned i = 0, e = Pending.
size();
1145 for (; i != e; ++i) {
1147 if (Pending[i].
getNode()->getOperand(0) == Root)
1155 if (Pending.
size() == 1)
1182 if (!PendingConstrainedFPStrict.empty()) {
1183 assert(PendingConstrainedFP.empty());
1184 updateRoot(PendingConstrainedFPStrict);
1197 if (!PendingConstrainedFP.empty()) {
1198 assert(PendingConstrainedFPStrict.empty());
1199 updateRoot(PendingConstrainedFP);
1203 return DAG.getRoot();
1211 PendingConstrainedFP.size() +
1212 PendingConstrainedFPStrict.size());
1214 PendingConstrainedFP.end());
1215 PendingLoads.append(PendingConstrainedFPStrict.begin(),
1216 PendingConstrainedFPStrict.end());
1217 PendingConstrainedFP.clear();
1218 PendingConstrainedFPStrict.clear();
1225 PendingExports.append(PendingConstrainedFPStrict.begin(),
1226 PendingConstrainedFPStrict.end());
1227 PendingConstrainedFPStrict.clear();
1228 return updateRoot(PendingExports);
1235 assert(Variable &&
"Missing variable");
1242 <<
"dbg_declare: Dropping debug info (bad/undef/unused-arg address)\n");
1258 if (IsParameter && FINode) {
1260 SDV =
DAG.getFrameIndexDbgValue(Variable,
Expression, FINode->getIndex(),
1261 true,
DL, SDNodeOrder);
1266 FuncArgumentDbgValueKind::Declare,
N);
1269 SDV =
DAG.getDbgValue(Variable,
Expression,
N.getNode(),
N.getResNo(),
1270 true,
DL, SDNodeOrder);
1272 DAG.AddDbgValue(SDV, IsParameter);
1277 FuncArgumentDbgValueKind::Declare,
N)) {
1279 <<
" (could not emit func-arg dbg_value)\n");
1290 for (
auto It = FnVarLocs->locs_begin(&
I), End = FnVarLocs->locs_end(&
I);
1292 auto *Var = FnVarLocs->getDILocalVariable(It->VariableID);
1294 if (It->Values.isKillLocation(It->Expr)) {
1300 It->Values.hasArgList())) {
1303 FnVarLocs->getDILocalVariable(It->VariableID),
1304 It->Expr, Vals.
size() > 1, It->DL, SDNodeOrder);
1317 bool SkipDbgVariableRecords =
DAG.getFunctionVarLocs();
1320 for (
DbgRecord &DR :
I.getDbgRecordRange()) {
1322 assert(DLR->getLabel() &&
"Missing label");
1324 DAG.getDbgLabel(DLR->getLabel(), DLR->getDebugLoc(), SDNodeOrder);
1325 DAG.AddDbgLabel(SDV);
1329 if (SkipDbgVariableRecords)
1337 if (
FuncInfo.PreprocessedDVRDeclares.contains(&DVR))
1339 LLVM_DEBUG(
dbgs() <<
"SelectionDAG visiting dbg_declare: " << DVR
1348 if (Values.
empty()) {
1365 SDNodeOrder, IsVariadic)) {
1376 if (
I.isTerminator()) {
1377 HandlePHINodesInSuccessorBlocks(
I.getParent());
1384 bool NodeInserted =
false;
1385 std::unique_ptr<SelectionDAG::DAGNodeInsertedListener> InsertedListener;
1386 MDNode *PCSectionsMD =
I.getMetadata(LLVMContext::MD_pcsections);
1387 MDNode *MMRA =
I.getMetadata(LLVMContext::MD_mmra);
1388 if (PCSectionsMD || MMRA) {
1389 InsertedListener = std::make_unique<SelectionDAG::DAGNodeInsertedListener>(
1390 DAG, [&](
SDNode *) { NodeInserted =
true; });
1400 if (PCSectionsMD || MMRA) {
1401 auto It = NodeMap.find(&
I);
1402 if (It != NodeMap.end()) {
1404 DAG.addPCSections(It->second.getNode(), PCSectionsMD);
1406 DAG.addMMRAMetadata(It->second.getNode(), MMRA);
1407 }
else if (NodeInserted) {
1410 errs() <<
"warning: loosing !pcsections and/or !mmra metadata ["
1411 <<
I.getModule()->getName() <<
"]\n";
1420void SelectionDAGBuilder::visitPHI(
const PHINode &) {
1430#define HANDLE_INST(NUM, OPCODE, CLASS) \
1431 case Instruction::OPCODE: visit##OPCODE((const CLASS&)I); break;
1432#include "llvm/IR/Instruction.def"
1444 for (
const Value *V : Values) {
1469 DanglingDebugInfoMap[Values[0]].emplace_back(Var, Expr,
DL, Order);
1474 auto isMatchingDbgValue = [&](DanglingDebugInfo &DDI) {
1475 DIVariable *DanglingVariable = DDI.getVariable();
1477 if (DanglingVariable == Variable && Expr->
fragmentsOverlap(DanglingExpr)) {
1479 << printDDI(
nullptr, DDI) <<
"\n");
1485 for (
auto &DDIMI : DanglingDebugInfoMap) {
1486 DanglingDebugInfoVector &DDIV = DDIMI.second;
1490 for (
auto &DDI : DDIV)
1491 if (isMatchingDbgValue(DDI))
1494 erase_if(DDIV, isMatchingDbgValue);
1502 auto DanglingDbgInfoIt = DanglingDebugInfoMap.find(V);
1503 if (DanglingDbgInfoIt == DanglingDebugInfoMap.end())
1506 DanglingDebugInfoVector &DDIV = DanglingDbgInfoIt->second;
1507 for (
auto &DDI : DDIV) {
1509 unsigned DbgSDNodeOrder = DDI.getSDNodeOrder();
1512 assert(Variable->isValidLocationForIntrinsic(
DL) &&
1513 "Expected inlined-at fields to agree");
1523 if (!EmitFuncArgumentDbgValue(V, Variable, Expr,
DL,
1524 FuncArgumentDbgValueKind::Value, Val)) {
1526 << printDDI(V, DDI) <<
"\n");
1533 <<
"changing SDNodeOrder from " << DbgSDNodeOrder <<
" to "
1534 << ValSDNodeOrder <<
"\n");
1535 SDV = getDbgValue(Val, Variable, Expr,
DL,
1536 std::max(DbgSDNodeOrder, ValSDNodeOrder));
1537 DAG.AddDbgValue(SDV,
false);
1541 <<
" in EmitFuncArgumentDbgValue\n");
1543 LLVM_DEBUG(
dbgs() <<
"Dropping debug info for " << printDDI(V, DDI)
1547 DAG.getConstantDbgValue(Variable, Expr,
Poison,
DL, DbgSDNodeOrder);
1548 DAG.AddDbgValue(SDV,
false);
1555 DanglingDebugInfo &DDI) {
1560 const Value *OrigV = V;
1564 unsigned SDOrder = DDI.getSDNodeOrder();
1568 bool StackValue =
true;
1593 if (!AdditionalValues.
empty())
1603 dbgs() <<
"Salvaged debug location info for:\n " << *Var <<
"\n"
1604 << *OrigV <<
"\nBy stripping back to:\n " << *V <<
"\n");
1612 assert(OrigV &&
"V shouldn't be null");
1614 auto *SDV =
DAG.getConstantDbgValue(Var, Expr,
Poison,
DL, SDNodeOrder);
1615 DAG.AddDbgValue(SDV,
false);
1617 << printDDI(OrigV, DDI) <<
"\n");
1634 unsigned Order,
bool IsVariadic) {
1639 if (visitEntryValueDbgValue(Values, Var, Expr, DbgLoc))
1644 for (
const Value *V : Values) {
1654 if (CE->getOpcode() == Instruction::IntToPtr) {
1673 N = UnusedArgNodeMap[V];
1678 EmitFuncArgumentDbgValue(V, Var, Expr, DbgLoc,
1679 FuncArgumentDbgValueKind::Value,
N))
1706 bool IsParamOfFunc =
1714 auto VMI =
FuncInfo.ValueMap.find(V);
1715 if (VMI !=
FuncInfo.ValueMap.end()) {
1720 V->getType(), std::nullopt);
1726 unsigned BitsToDescribe = 0;
1728 BitsToDescribe = *VarSize;
1730 BitsToDescribe = Fragment->SizeInBits;
1733 if (
Offset >= BitsToDescribe)
1736 unsigned RegisterSize = RegAndSize.second;
1737 unsigned FragmentSize = (
Offset + RegisterSize > BitsToDescribe)
1738 ? BitsToDescribe -
Offset
1741 Expr,
Offset, FragmentSize);
1745 Var, *FragmentExpr, RegAndSize.first,
false, DbgLoc, Order);
1746 DAG.AddDbgValue(SDV,
false);
1762 DAG.getDbgValueList(Var, Expr, LocationOps, Dependencies,
1763 false, DbgLoc, Order, IsVariadic);
1764 DAG.AddDbgValue(SDV,
false);
1770 for (
auto &Pair : DanglingDebugInfoMap)
1771 for (
auto &DDI : Pair.second)
1779 auto It =
FuncInfo.ValueMap.find(V);
1782 if (It !=
FuncInfo.ValueMap.end()) {
1786 DAG.getDataLayout(), InReg, Ty,
1803 if (
N.getNode())
return N;
1863 return DAG.getSplatBuildVector(
1866 return DAG.getConstant(*CI,
DL, VT);
1878 getValue(CPA->getAddrDiscriminator()),
1879 getValue(CPA->getDiscriminator()));
1895 visit(CE->getOpcode(), *CE);
1897 assert(N1.
getNode() &&
"visit didn't populate the NodeMap!");
1903 for (
const Use &U :
C->operands()) {
1909 for (
unsigned i = 0, e = Val->
getNumValues(); i != e; ++i)
1910 Constants.push_back(
SDValue(Val, i));
1919 for (
uint64_t i = 0, e = CDS->getNumElements(); i != e; ++i) {
1923 for (
unsigned i = 0, e = Val->
getNumValues(); i != e; ++i)
1932 if (
C->getType()->isStructTy() ||
C->getType()->isArrayTy()) {
1934 "Unknown struct or array constant!");
1938 unsigned NumElts = ValueVTs.
size();
1942 for (
unsigned i = 0; i != NumElts; ++i) {
1943 EVT EltVT = ValueVTs[i];
1945 Constants[i] =
DAG.getUNDEF(EltVT);
1956 return DAG.getBlockAddress(BA, VT);
1959 return getValue(Equiv->getGlobalValue());
1964 if (VT == MVT::aarch64svcount) {
1965 assert(
C->isNullValue() &&
"Can only zero this target type!");
1971 assert(
C->isNullValue() &&
"Can only zero this target type!");
1988 for (
unsigned i = 0; i != NumElements; ++i)
2015 return DAG.getFrameIndex(
2023 Inst->getType(), std::nullopt);
2037void SelectionDAGBuilder::visitCatchPad(
const CatchPadInst &
I) {
2050 if (IsMSVCCXX || IsCoreCLR)
2056 MachineBasicBlock *TargetMBB =
FuncInfo.getMBB(
I.getSuccessor());
2057 FuncInfo.MBB->addSuccessor(TargetMBB);
2064 if (TargetMBB != NextBlock(
FuncInfo.MBB) ||
2073 DAG.getMachineFunction().setHasEHContTarget(
true);
2079 Value *ParentPad =
I.getCatchSwitchParentPad();
2082 SuccessorColor = &
FuncInfo.Fn->getEntryBlock();
2085 assert(SuccessorColor &&
"No parent funclet for catchret!");
2086 MachineBasicBlock *SuccessorColorMBB =
FuncInfo.getMBB(SuccessorColor);
2087 assert(SuccessorColorMBB &&
"No MBB for SuccessorColor!");
2092 DAG.getBasicBlock(SuccessorColorMBB));
2096void SelectionDAGBuilder::visitCleanupPad(
const CleanupPadInst &CPI) {
2102 FuncInfo.MBB->setIsEHFuncletEntry();
2103 FuncInfo.MBB->setIsCleanupFuncletEntry();
2132 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2138 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2139 UnwindDests.back().first->setIsEHScopeEntry();
2142 UnwindDests.back().first->setIsEHFuncletEntry();
2146 for (
const BasicBlock *CatchPadBB : CatchSwitch->handlers()) {
2147 UnwindDests.emplace_back(FuncInfo.
getMBB(CatchPadBB), Prob);
2149 if (IsMSVCCXX || IsCoreCLR)
2150 UnwindDests.back().first->setIsEHFuncletEntry();
2152 UnwindDests.back().first->setIsEHScopeEntry();
2154 NewEHPadBB = CatchSwitch->getUnwindDest();
2160 if (BPI && NewEHPadBB)
2162 EHPadBB = NewEHPadBB;
2169 auto UnwindDest =
I.getUnwindDest();
2170 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
2171 BranchProbability UnwindDestProb =
2176 for (
auto &UnwindDest : UnwindDests) {
2177 UnwindDest.first->setIsEHPad();
2178 addSuccessorWithProb(
FuncInfo.MBB, UnwindDest.first, UnwindDest.second);
2180 FuncInfo.MBB->normalizeSuccProbs();
2183 MachineBasicBlock *CleanupPadMBB =
2184 FuncInfo.getMBB(
I.getCleanupPad()->getParent());
2190void SelectionDAGBuilder::visitCatchSwitch(
const CatchSwitchInst &CSI) {
2194void SelectionDAGBuilder::visitRet(
const ReturnInst &
I) {
2195 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
2196 auto &
DL =
DAG.getDataLayout();
2208 if (
I.getParent()->getTerminatingDeoptimizeCall()) {
2225 SmallVector<uint64_t, 4>
Offsets;
2228 unsigned NumValues = ValueVTs.
size();
2231 Align BaseAlign =
DL.getPrefTypeAlign(
I.getOperand(0)->getType());
2232 for (
unsigned i = 0; i != NumValues; ++i) {
2239 if (MemVTs[i] != ValueVTs[i])
2241 Chains[i] =
DAG.getStore(
2249 MVT::Other, Chains);
2250 }
else if (
I.getNumOperands() != 0) {
2253 unsigned NumValues =
Types.size();
2257 const Function *
F =
I.getParent()->getParent();
2260 I.getOperand(0)->getType(),
F->getCallingConv(),
2264 if (
F->getAttributes().hasRetAttr(Attribute::SExt))
2266 else if (
F->getAttributes().hasRetAttr(Attribute::ZExt))
2269 LLVMContext &
Context =
F->getContext();
2270 bool RetInReg =
F->getAttributes().hasRetAttr(Attribute::InReg);
2272 for (
unsigned j = 0;
j != NumValues; ++
j) {
2285 &Parts[0], NumParts, PartVT, &
I, CC, ExtendKind);
2288 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2292 if (
I.getOperand(0)->getType()->isPointerTy()) {
2294 Flags.setPointerAddrSpace(
2298 if (NeedsRegBlock) {
2299 Flags.setInConsecutiveRegs();
2300 if (j == NumValues - 1)
2301 Flags.setInConsecutiveRegsLast();
2309 else if (
F->getAttributes().hasRetAttr(Attribute::NoExt))
2312 for (
unsigned i = 0; i < NumParts; ++i) {
2315 VT, Types[j], 0, 0));
2325 const Function *
F =
I.getParent()->getParent();
2327 F->getAttributes().hasAttrSomewhere(Attribute::SwiftError)) {
2329 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2330 Flags.setSwiftError();
2342 bool isVarArg =
DAG.getMachineFunction().getFunction().isVarArg();
2344 DAG.getMachineFunction().getFunction().getCallingConv();
2345 Chain =
DAG.getTargetLoweringInfo().LowerReturn(
2350 "LowerReturn didn't return a valid chain!");
2361 if (V->getType()->isEmptyTy())
2364 auto VMI =
FuncInfo.ValueMap.find(V);
2365 if (VMI !=
FuncInfo.ValueMap.end()) {
2367 "Unused value assigned virtual registers!");
2380 if (
FuncInfo.isExportedInst(V))
return;
2392 if (VI->getParent() == FromBB)
2418 const BasicBlock *SrcBB = Src->getBasicBlock();
2419 const BasicBlock *DstBB = Dst->getBasicBlock();
2423 auto SuccSize = std::max<uint32_t>(
succ_size(SrcBB), 1);
2433 Src->addSuccessorWithoutProb(Dst);
2436 Prob = getEdgeProbability(Src, Dst);
2437 Src->addSuccessor(Dst, Prob);
2443 return I->getParent() == BB;
2467 if (CurBB == SwitchBB ||
2473 InvertCond ? IC->getInversePredicate() : IC->getPredicate();
2478 InvertCond ? FC->getInversePredicate() : FC->getPredicate();
2480 if (FC->hasNoNaNs() ||
2488 CaseBlock CB(Condition, BOp->getOperand(0), BOp->getOperand(1),
nullptr,
2490 SL->SwitchCases.push_back(CB);
2499 SL->SwitchCases.push_back(CB);
2507 unsigned Depth = 0) {
2516 if (Necessary !=
nullptr) {
2519 if (Necessary->contains(
I))
2547 if (BPI !=
nullptr) {
2553 std::optional<bool> Likely;
2556 else if (BPI->
isEdgeHot(
I.getParent(), IfFalse))
2560 if (
Opc == (*Likely ? Instruction::And : Instruction::Or))
2572 if (CostThresh <= 0)
2593 Value *BrCond =
I.getCondition();
2594 auto ShouldCountInsn = [&RhsDeps, &BrCond](
const Instruction *Ins) {
2595 for (
const auto *U : Ins->users()) {
2598 if (UIns != BrCond && !RhsDeps.
contains(UIns))
2611 for (
unsigned PruneIters = 0; PruneIters < MaxPruneIters; ++PruneIters) {
2613 for (
const auto &InsPair : RhsDeps) {
2614 if (!ShouldCountInsn(InsPair.first)) {
2615 ToDrop = InsPair.first;
2619 if (ToDrop ==
nullptr)
2621 RhsDeps.erase(ToDrop);
2624 for (
const auto &InsPair : RhsDeps) {
2629 CostOfIncluding +=
TTI->getInstructionCost(
2632 if (CostOfIncluding > CostThresh)
2658 const Value *BOpOp0, *BOpOp1;
2672 if (BOpc == Instruction::And)
2673 BOpc = Instruction::Or;
2674 else if (BOpc == Instruction::Or)
2675 BOpc = Instruction::And;
2681 bool BOpIsInOrAndTree = BOpc && BOpc ==
Opc && BOp->
hasOneUse();
2686 TProb, FProb, InvertCond);
2696 if (
Opc == Instruction::Or) {
2717 auto NewTrueProb = TProb / 2;
2718 auto NewFalseProb = TProb / 2 + FProb;
2721 NewFalseProb, InvertCond);
2728 Probs[1], InvertCond);
2730 assert(
Opc == Instruction::And &&
"Unknown merge op!");
2750 auto NewTrueProb = TProb + FProb / 2;
2751 auto NewFalseProb = FProb / 2;
2754 NewFalseProb, InvertCond);
2761 Probs[1], InvertCond);
2770 if (Cases.size() != 2)
return true;
2774 if ((Cases[0].CmpLHS == Cases[1].CmpLHS &&
2775 Cases[0].CmpRHS == Cases[1].CmpRHS) ||
2776 (Cases[0].CmpRHS == Cases[1].CmpLHS &&
2777 Cases[0].CmpLHS == Cases[1].CmpRHS)) {
2783 if (Cases[0].CmpRHS == Cases[1].CmpRHS &&
2784 Cases[0].CC == Cases[1].CC &&
2787 if (Cases[0].CC ==
ISD::SETEQ && Cases[0].TrueBB == Cases[1].ThisBB)
2789 if (Cases[0].CC ==
ISD::SETNE && Cases[0].FalseBB == Cases[1].ThisBB)
2796void SelectionDAGBuilder::visitUncondBr(
const UncondBrInst &
I) {
2806 if (Succ0MBB != NextBlock(BrMBB) ||
2815void SelectionDAGBuilder::visitCondBr(
const CondBrInst &
I) {
2816 MachineBasicBlock *BrMBB =
FuncInfo.MBB;
2818 MachineBasicBlock *Succ0MBB =
FuncInfo.getMBB(
I.getSuccessor(0));
2822 const Value *CondVal =
I.getCondition();
2823 MachineBasicBlock *Succ1MBB =
FuncInfo.getMBB(
I.getSuccessor(1));
2842 bool IsUnpredictable =
I.hasMetadata(LLVMContext::MD_unpredictable);
2844 if (!
DAG.getTargetLoweringInfo().isJumpExpensive() && BOp &&
2847 const Value *BOp0, *BOp1;
2850 Opcode = Instruction::And;
2852 Opcode = Instruction::Or;
2859 DAG.getTargetLoweringInfo().getJumpConditionMergingParams(
2860 Opcode, BOp0, BOp1))) {
2862 getEdgeProbability(BrMBB, Succ0MBB),
2863 getEdgeProbability(BrMBB, Succ1MBB),
2868 assert(
SL->SwitchCases[0].ThisBB == BrMBB &&
"Unexpected lowering!");
2872 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i) {
2879 SL->SwitchCases.erase(
SL->SwitchCases.begin());
2885 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i)
2886 FuncInfo.MF->erase(
SL->SwitchCases[i].ThisBB);
2888 SL->SwitchCases.clear();
2894 nullptr, Succ0MBB, Succ1MBB, BrMBB,
getCurSDLoc(),
2915 if (CB.
TrueBB != NextBlock(SwitchBB)) {
2922 auto &TLI =
DAG.getTargetLoweringInfo();
2946 Cond =
DAG.getSetCC(dl, MVT::i1, CondLHS, CondRHS, CB.
CC);
2958 Cond =
DAG.getSetCC(dl, MVT::i1, CmpOp,
DAG.getConstant(
High, dl, VT),
2962 VT, CmpOp,
DAG.getConstant(
Low, dl, VT));
2963 Cond =
DAG.getSetCC(dl, MVT::i1, SUB,
2978 if (CB.
TrueBB == NextBlock(SwitchBB)) {
2994 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
2997 DAG.setRoot(BrCond);
3003 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3004 assert(JT.
Reg &&
"Should lower JT Header first!");
3005 EVT PTy =
DAG.getTargetLoweringInfo().getJumpTableRegTy(
DAG.getDataLayout());
3009 Index.getValue(1), Table, Index);
3010 DAG.setRoot(BrJumpTable);
3018 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3025 DAG.getConstant(JTH.
First, dl, VT));
3040 JT.
Reg = JumpTableReg;
3048 Sub.getValueType()),
3052 MVT::Other, CopyTo, CMP,
3056 if (JT.
MBB != NextBlock(SwitchBB))
3057 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
3058 DAG.getBasicBlock(JT.
MBB));
3060 DAG.setRoot(BrCond);
3063 if (JT.
MBB != NextBlock(SwitchBB))
3065 DAG.getBasicBlock(JT.
MBB)));
3067 DAG.setRoot(CopyTo);
3091 if (PtrTy != PtrMemTy)
3107 auto &
DL =
DAG.getDataLayout();
3116 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3123 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3137 assert(GuardCheckFn &&
"Guard check function is null");
3148 Entry.IsInReg =
true;
3149 Args.push_back(Entry);
3155 getValue(GuardCheckFn), std::move(Args));
3157 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
3158 DAG.setRoot(Result.second);
3170 Guard =
DAG.getLoad(PtrMemTy, dl, Chain, GuardPtr,
3176 Guard =
DAG.getPOISON(PtrMemTy);
3218 auto &
DL =
DAG.getDataLayout();
3226 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3232 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3247 if (GuardCheckFn->hasParamAttribute(0, Attribute::AttrKind::InReg))
3248 Entry.IsInReg =
true;
3249 Args.push_back(Entry);
3255 getValue(GuardCheckFn), std::move(Args));
3261 Chain = TLI.
makeLibCall(
DAG, RTLIB::STACKPROTECTOR_CHECK_FAIL, MVT::isVoid,
3284 DAG.getNode(
ISD::SUB, dl, VT, SwitchOp,
DAG.getConstant(
B.First, dl, VT));
3288 bool UsePtrType =
false;
3312 if (!
B.FallthroughUnreachable)
3313 addSuccessorWithProb(SwitchBB,
B.Default,
B.DefaultProb);
3314 addSuccessorWithProb(SwitchBB,
MBB,
B.Prob);
3318 if (!
B.FallthroughUnreachable) {
3327 DAG.getBasicBlock(
B.Default));
3331 if (
MBB != NextBlock(SwitchBB))
3349 if (PopCount == 1) {
3356 }
else if (PopCount == BB.
Range) {
3364 DAG.getConstant(1, dl, VT), ShiftOp);
3368 VT, SwitchVal,
DAG.getConstant(
B.Mask, dl, VT));
3375 addSuccessorWithProb(SwitchBB,
B.TargetBB,
B.ExtraProb);
3377 addSuccessorWithProb(SwitchBB, NextMBB, BranchProbToNext);
3385 Cmp,
DAG.getBasicBlock(
B.TargetBB));
3388 if (NextMBB != NextBlock(SwitchBB))
3389 BrAnd =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrAnd,
3390 DAG.getBasicBlock(NextMBB));
3395void SelectionDAGBuilder::visitInvoke(
const InvokeInst &
I) {
3413 const Value *Callee(
I.getCalledOperand());
3416 visitInlineAsm(
I, EHPadBB);
3421 case Intrinsic::donothing:
3423 case Intrinsic::seh_try_begin:
3424 case Intrinsic::seh_scope_begin:
3425 case Intrinsic::seh_try_end:
3426 case Intrinsic::seh_scope_end:
3432 case Intrinsic::experimental_patchpoint_void:
3433 case Intrinsic::experimental_patchpoint:
3434 visitPatchpoint(
I, EHPadBB);
3436 case Intrinsic::experimental_gc_statepoint:
3442 case Intrinsic::wasm_throw: {
3444 std::array<SDValue, 4>
Ops = {
3455 case Intrinsic::wasm_rethrow: {
3456 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3457 std::array<SDValue, 2>
Ops = {
3466 }
else if (
I.hasDeoptState()) {
3487 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
3488 BranchProbability EHPadBBProb =
3494 addSuccessorWithProb(InvokeMBB, Return);
3495 for (
auto &UnwindDest : UnwindDests) {
3496 UnwindDest.first->setIsEHPad();
3497 addSuccessorWithProb(InvokeMBB, UnwindDest.first, UnwindDest.second);
3503 DAG.getBasicBlock(Return)));
3512void SelectionDAGBuilder::visitCallBrIntrinsic(
const CallBrInst &
I) {
3515 DAG.getTargetLoweringInfo().getTgtMemIntrinsic(
3516 Infos,
I,
DAG.getMachineFunction(),
I.getIntrinsicID());
3517 assert(Infos.
empty() &&
"Intrinsic touches memory");
3520 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
3523 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad);
3524 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
3528 getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
3529 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
3534void SelectionDAGBuilder::visitCallBr(
const CallBrInst &
I) {
3535 MachineBasicBlock *CallBrMBB =
FuncInfo.MBB;
3537 if (
I.isInlineAsm()) {
3544 assert(!
I.hasOperandBundles() &&
3545 "Can't have operand bundles for intrinsics");
3546 visitCallBrIntrinsic(
I);
3551 SmallPtrSet<BasicBlock *, 8> Dests;
3552 Dests.
insert(
I.getDefaultDest());
3562 if (
I.isInlineAsm()) {
3563 for (BasicBlock *Dest :
I.getIndirectDests()) {
3565 Target->setIsInlineAsmBrIndirectTarget();
3571 Target->setLabelMustBeEmitted();
3573 if (Dests.
insert(Dest).second)
3582 DAG.getBasicBlock(Return)));
3585void SelectionDAGBuilder::visitResume(
const ResumeInst &RI) {
3586 llvm_unreachable(
"SelectionDAGBuilder shouldn't visit resume instructions!");
3589void SelectionDAGBuilder::visitLandingPad(
const LandingPadInst &LP) {
3591 "Call to landingpad not in landing pad!");
3595 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3611 assert(ValueVTs.
size() == 2 &&
"Only two-valued landingpads are supported");
3616 if (
FuncInfo.ExceptionPointerVirtReg) {
3617 Ops[0] =
DAG.getZExtOrTrunc(
3618 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3625 Ops[1] =
DAG.getZExtOrTrunc(
3626 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3633 DAG.getVTList(ValueVTs),
Ops);
3641 if (JTB.first.HeaderBB ==
First)
3642 JTB.first.HeaderBB =
Last;
3655 for (
unsigned i = 0, e =
I.getNumSuccessors(); i != e; ++i) {
3657 bool Inserted =
Done.insert(BB).second;
3662 addSuccessorWithProb(IndirectBrMBB, Succ);
3672 if (!
I.shouldLowerToTrap(
DAG.getTarget().Options.TrapUnreachable,
3673 DAG.getTarget().Options.NoTrapAfterNoreturn))
3679void SelectionDAGBuilder::visitUnary(
const User &
I,
unsigned Opcode) {
3682 Flags.copyFMF(*FPOp);
3690void SelectionDAGBuilder::visitBinary(
const User &
I,
unsigned Opcode) {
3693 Flags.setNoSignedWrap(OFBinOp->hasNoSignedWrap());
3694 Flags.setNoUnsignedWrap(OFBinOp->hasNoUnsignedWrap());
3697 Flags.setExact(ExactOp->isExact());
3699 Flags.setDisjoint(DisjointOp->isDisjoint());
3701 Flags.copyFMF(*FPOp);
3710void SelectionDAGBuilder::visitShift(
const User &
I,
unsigned Opcode) {
3714 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
3719 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
3721 "Unexpected shift type");
3731 if (
const OverflowingBinaryOperator *OFBinOp =
3733 nuw = OFBinOp->hasNoUnsignedWrap();
3734 nsw = OFBinOp->hasNoSignedWrap();
3736 if (
const PossiblyExactOperator *ExactOp =
3738 exact = ExactOp->isExact();
3741 Flags.setExact(exact);
3742 Flags.setNoSignedWrap(nsw);
3743 Flags.setNoUnsignedWrap(nuw);
3749void SelectionDAGBuilder::visitSDiv(
const User &
I) {
3760void SelectionDAGBuilder::visitICmp(
const ICmpInst &
I) {
3766 auto &TLI =
DAG.getTargetLoweringInfo();
3779 Flags.setSameSign(
I.hasSameSign());
3781 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3787void SelectionDAGBuilder::visitFCmp(
const FCmpInst &
I) {
3794 if (FPMO->hasNoNaNs() ||
3795 (
DAG.isKnownNeverNaN(Op1) &&
DAG.isKnownNeverNaN(Op2)))
3799 Flags.copyFMF(*FPMO);
3801 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3811 return isa<SelectInst>(V);
3815void SelectionDAGBuilder::visitSelect(
const User &
I) {
3819 unsigned NumValues = ValueVTs.
size();
3820 if (NumValues == 0)
return;
3830 bool IsUnaryAbs =
false;
3831 bool Negate =
false;
3835 Flags.copyFMF(*FPOp);
3837 Flags.setUnpredictable(
3842 EVT VT = ValueVTs[0];
3843 LLVMContext &Ctx = *
DAG.getContext();
3844 auto &TLI =
DAG.getTargetLoweringInfo();
3854 bool UseScalarMinMax = VT.
isVector() &&
3863 switch (SPR.Flavor) {
3872 switch (SPR.NaNBehavior) {
3877 Flags.setNoSignedZeros(
true);
3891 switch (SPR.NaNBehavior) {
3896 Flags.setNoSignedZeros(
true);
3938 for (
unsigned i = 0; i != NumValues; ++i) {
3944 Values[i] =
DAG.getNegative(Values[i], dl, VT);
3947 for (
unsigned i = 0; i != NumValues; ++i) {
3951 Values[i] =
DAG.getNode(
3958 DAG.getVTList(ValueVTs), Values));
3961void SelectionDAGBuilder::visitTrunc(
const User &
I) {
3964 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3968 Flags.setNoSignedWrap(Trunc->hasNoSignedWrap());
3969 Flags.setNoUnsignedWrap(Trunc->hasNoUnsignedWrap());
3975void SelectionDAGBuilder::visitZExt(
const User &
I) {
3979 auto &TLI =
DAG.getTargetLoweringInfo();
3984 Flags.setNonNeg(PNI->hasNonNeg());
3989 if (
Flags.hasNonNeg() &&
3998void SelectionDAGBuilder::visitSExt(
const User &
I) {
4002 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4007void SelectionDAGBuilder::visitFPTrunc(
const User &
I) {
4013 Flags.copyFMF(*FPOp);
4014 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4017 DAG.getTargetConstant(
4022void SelectionDAGBuilder::visitFPExt(
const User &
I) {
4025 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4029 Flags.copyFMF(*FPOp);
4033void SelectionDAGBuilder::visitFPToUI(
const User &
I) {
4036 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4041void SelectionDAGBuilder::visitFPToSI(
const User &
I) {
4044 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4049void SelectionDAGBuilder::visitUIToFP(
const User &
I) {
4052 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4061void SelectionDAGBuilder::visitSIToFP(
const User &
I) {
4064 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4072void SelectionDAGBuilder::visitPtrToAddr(
const User &
I) {
4075 const auto &TLI =
DAG.getTargetLoweringInfo();
4083void SelectionDAGBuilder::visitPtrToInt(
const User &
I) {
4087 auto &TLI =
DAG.getTargetLoweringInfo();
4088 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4097void SelectionDAGBuilder::visitIntToPtr(
const User &
I) {
4101 auto &TLI =
DAG.getTargetLoweringInfo();
4109void SelectionDAGBuilder::visitBitCast(
const User &
I) {
4112 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4117 if (DestVT !=
N.getValueType())
4125 setValue(&
I,
DAG.getConstant(
C->getValue(), dl, DestVT,
false,
4131void SelectionDAGBuilder::visitAddrSpaceCast(
const User &
I) {
4132 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4133 const Value *SV =
I.getOperand(0);
4138 unsigned DestAS =
I.getType()->getPointerAddressSpace();
4140 if (!TM.isNoopAddrSpaceCast(SrcAS, DestAS))
4146void SelectionDAGBuilder::visitInsertElement(
const User &
I) {
4147 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4154 InVec, InVal, InIdx));
4157void SelectionDAGBuilder::visitExtractElement(
const User &
I) {
4158 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4167void SelectionDAGBuilder::visitShuffleVector(
const User &
I) {
4172 Mask = SVI->getShuffleMask();
4176 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4184 DAG.getVectorIdxConstant(0,
DL));
4195 unsigned MaskNumElts =
Mask.size();
4197 if (SrcNumElts == MaskNumElts) {
4203 if (SrcNumElts < MaskNumElts) {
4207 if (MaskNumElts % SrcNumElts == 0) {
4211 unsigned NumConcat = MaskNumElts / SrcNumElts;
4212 bool IsConcat =
true;
4213 SmallVector<int, 8> ConcatSrcs(NumConcat, -1);
4214 for (
unsigned i = 0; i != MaskNumElts; ++i) {
4220 if ((Idx % SrcNumElts != (i % SrcNumElts)) ||
4221 (ConcatSrcs[i / SrcNumElts] >= 0 &&
4222 ConcatSrcs[i / SrcNumElts] != (
int)(Idx / SrcNumElts))) {
4227 ConcatSrcs[i / SrcNumElts] = Idx / SrcNumElts;
4234 for (
auto Src : ConcatSrcs) {
4247 unsigned PaddedMaskNumElts =
alignTo(MaskNumElts, SrcNumElts);
4248 unsigned NumConcat = PaddedMaskNumElts / SrcNumElts;
4264 SmallVector<int, 8> MappedOps(PaddedMaskNumElts, -1);
4265 for (
unsigned i = 0; i != MaskNumElts; ++i) {
4267 if (Idx >= (
int)SrcNumElts)
4268 Idx -= SrcNumElts - PaddedMaskNumElts;
4276 if (MaskNumElts != PaddedMaskNumElts)
4278 DAG.getVectorIdxConstant(0,
DL));
4284 assert(SrcNumElts > MaskNumElts);
4288 int StartIdx[2] = {-1, -1};
4289 bool CanExtract =
true;
4290 for (
int Idx : Mask) {
4295 if (Idx >= (
int)SrcNumElts) {
4303 int NewStartIdx =
alignDown(Idx, MaskNumElts);
4304 if (NewStartIdx + MaskNumElts > SrcNumElts ||
4305 (StartIdx[Input] >= 0 && StartIdx[Input] != NewStartIdx))
4309 StartIdx[Input] = NewStartIdx;
4312 if (StartIdx[0] < 0 && StartIdx[1] < 0) {
4318 for (
unsigned Input = 0; Input < 2; ++Input) {
4319 SDValue &Src = Input == 0 ? Src1 : Src2;
4320 if (StartIdx[Input] < 0)
4321 Src =
DAG.getUNDEF(VT);
4324 DAG.getVectorIdxConstant(StartIdx[Input],
DL));
4329 SmallVector<int, 8> MappedOps(Mask);
4330 for (
int &Idx : MappedOps) {
4331 if (Idx >= (
int)SrcNumElts)
4332 Idx -= SrcNumElts + StartIdx[1] - MaskNumElts;
4337 setValue(&
I,
DAG.getVectorShuffle(VT,
DL, Src1, Src2, MappedOps));
4346 for (
int Idx : Mask) {
4350 Res =
DAG.getUNDEF(EltVT);
4352 SDValue &Src = Idx < (int)SrcNumElts ? Src1 : Src2;
4353 if (Idx >= (
int)SrcNumElts) Idx -= SrcNumElts;
4356 DAG.getVectorIdxConstant(Idx,
DL));
4366 ArrayRef<unsigned> Indices =
I.getIndices();
4367 const Value *Op0 =
I.getOperand(0);
4369 Type *AggTy =
I.getType();
4376 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4382 unsigned NumAggValues = AggValueVTs.
size();
4383 unsigned NumValValues = ValValueVTs.
size();
4387 if (!NumAggValues) {
4395 for (; i != LinearIndex; ++i)
4396 Values[i] = IntoUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4401 for (; i != LinearIndex + NumValValues; ++i)
4402 Values[i] = FromUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4406 for (; i != NumAggValues; ++i)
4407 Values[i] = IntoUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4411 DAG.getVTList(AggValueVTs), Values));
4415 ArrayRef<unsigned> Indices =
I.getIndices();
4416 const Value *Op0 =
I.getOperand(0);
4418 Type *ValTy =
I.getType();
4423 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4427 unsigned NumValValues = ValValueVTs.
size();
4430 if (!NumValValues) {
4439 for (
unsigned i = LinearIndex; i != LinearIndex + NumValValues; ++i)
4440 Values[i - LinearIndex] =
4446 DAG.getVTList(ValValueVTs), Values));
4449void SelectionDAGBuilder::visitGetElementPtr(
const User &
I) {
4450 Value *Op0 =
I.getOperand(0);
4456 auto &TLI =
DAG.getTargetLoweringInfo();
4461 bool IsVectorGEP =
I.getType()->isVectorTy();
4462 ElementCount VectorElementCount =
4468 const Value *Idx = GTI.getOperand();
4469 if (StructType *StTy = GTI.getStructTypeOrNull()) {
4474 DAG.getDataLayout().getStructLayout(StTy)->getElementOffset(
Field);
4484 N =
DAG.getMemBasePlusOffset(
4485 N,
DAG.getConstant(
Offset, dl,
N.getValueType()), dl, Flags);
4491 unsigned IdxSize =
DAG.getDataLayout().getIndexSizeInBits(AS);
4493 TypeSize ElementSize =
4494 GTI.getSequentialElementStride(
DAG.getDataLayout());
4499 bool ElementScalable = ElementSize.
isScalable();
4505 C =
C->getSplatValue();
4508 if (CI && CI->isZero())
4510 if (CI && !ElementScalable) {
4511 APInt Offs = ElementMul * CI->getValue().sextOrTrunc(IdxSize);
4514 if (
N.getValueType().isVector())
4515 OffsVal =
DAG.getConstant(
4518 OffsVal =
DAG.getConstant(Offs, dl, IdxTy);
4525 Flags.setNoUnsignedWrap(
true);
4528 OffsVal =
DAG.getSExtOrTrunc(OffsVal, dl,
N.getValueType());
4530 N =
DAG.getMemBasePlusOffset(
N, OffsVal, dl, Flags);
4538 if (
N.getValueType().isVector()) {
4540 VectorElementCount);
4541 IdxN =
DAG.getSplat(VT, dl, IdxN);
4545 N =
DAG.getSplat(VT, dl,
N);
4551 IdxN =
DAG.getSExtOrTrunc(IdxN, dl,
N.getValueType());
4553 SDNodeFlags ScaleFlags;
4562 if (ElementScalable) {
4563 EVT VScaleTy =
N.getValueType().getScalarType();
4566 DAG.getConstant(ElementMul.getZExtValue(), dl, VScaleTy));
4567 if (
N.getValueType().isVector())
4568 VScale =
DAG.getSplatVector(
N.getValueType(), dl, VScale);
4569 IdxN =
DAG.getNode(
ISD::MUL, dl,
N.getValueType(), IdxN, VScale,
4574 if (ElementMul != 1) {
4575 if (ElementMul.isPowerOf2()) {
4576 unsigned Amt = ElementMul.logBase2();
4579 DAG.getShiftAmountConstant(Amt,
N.getValueType(), dl),
4582 SDValue Scale =
DAG.getConstant(ElementMul.getZExtValue(), dl,
4584 IdxN =
DAG.getNode(
ISD::MUL, dl,
N.getValueType(), IdxN, Scale,
4594 SDNodeFlags AddFlags;
4598 N =
DAG.getMemBasePlusOffset(
N, IdxN, dl, AddFlags);
4602 if (IsVectorGEP && !
N.getValueType().isVector()) {
4604 N =
DAG.getSplat(VT, dl,
N);
4615 N =
DAG.getPtrExtendInReg(
N, dl, PtrMemTy);
4620void SelectionDAGBuilder::visitAlloca(
const AllocaInst &
I) {
4627 Type *Ty =
I.getAllocatedType();
4628 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4629 auto &
DL =
DAG.getDataLayout();
4630 TypeSize TySize =
DL.getTypeAllocSize(Ty);
4631 MaybeAlign Alignment =
I.getAlign();
4637 AllocSize =
DAG.getZExtOrTrunc(AllocSize, dl, IntPtr);
4639 AllocSize =
DAG.getNode(
4641 DAG.getZExtOrTrunc(
DAG.getTypeSize(dl, MVT::i64, TySize), dl, IntPtr));
4646 Align StackAlign =
DAG.getSubtarget().getFrameLowering()->getStackAlign();
4647 if (*Alignment <= StackAlign)
4648 Alignment = std::nullopt;
4650 const uint64_t StackAlignMask = StackAlign.
value() - 1U;
4655 DAG.getConstant(StackAlignMask, dl, IntPtr),
4660 DAG.getSignedConstant(~StackAlignMask, dl, IntPtr));
4664 DAG.getConstant(Alignment ? Alignment->value() : 0, dl, IntPtr)};
4674 return I.getMetadata(LLVMContext::MD_range);
4679 if (std::optional<ConstantRange> CR = CB->getRange())
4683 return std::nullopt;
4688 return CB->getRetNoFPClass();
4692void SelectionDAGBuilder::visitLoad(
const LoadInst &
I) {
4694 return visitAtomicLoad(
I);
4696 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4697 const Value *SV =
I.getOperand(0);
4702 if (Arg->hasSwiftErrorAttr())
4703 return visitLoadFromSwiftError(
I);
4707 if (Alloca->isSwiftError())
4708 return visitLoadFromSwiftError(
I);
4714 Type *Ty =
I.getType();
4718 unsigned NumValues = ValueVTs.
size();
4722 Align Alignment =
I.getAlign();
4723 AAMDNodes AAInfo =
I.getAAMetadata();
4725 bool isVolatile =
I.isVolatile();
4730 bool ConstantMemory =
false;
4737 BatchAA->pointsToConstantMemory(MemoryLocation(
4742 Root =
DAG.getEntryNode();
4743 ConstantMemory =
true;
4747 Root =
DAG.getRoot();
4758 unsigned ChainI = 0;
4759 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4775 MachinePointerInfo PtrInfo =
4777 ? MachinePointerInfo(SV, Offsets[i].getKnownMinValue())
4778 : MachinePointerInfo();
4780 SDValue A =
DAG.getObjectPtrOffset(dl, Ptr, Offsets[i]);
4781 SDValue L =
DAG.getLoad(MemVTs[i], dl, Root,
A, PtrInfo, Alignment,
4782 MMOFlags, AAInfo, Ranges);
4783 Chains[ChainI] =
L.getValue(1);
4785 if (MemVTs[i] != ValueVTs[i])
4786 L =
DAG.getPtrExtOrTrunc(L, dl, ValueVTs[i]);
4788 if (MDNode *NoFPClassMD =
I.getMetadata(LLVMContext::MD_nofpclass)) {
4789 uint64_t FPTestInt =
4791 cast<ConstantAsMetadata>(NoFPClassMD->getOperand(0))->getValue())
4793 if (FPTestInt != fcNone) {
4794 SDValue FPTestConst =
4795 DAG.getTargetConstant(FPTestInt, SDLoc(), MVT::i32);
4796 L = DAG.getNode(ISD::AssertNoFPClass, dl, L.getValueType(), L,
4803 if (!ConstantMemory) {
4809 PendingLoads.push_back(Chain);
4813 DAG.getVTList(ValueVTs), Values));
4816void SelectionDAGBuilder::visitStoreToSwiftError(
const StoreInst &
I) {
4817 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4818 "call visitStoreToSwiftError when backend supports swifterror");
4821 SmallVector<uint64_t, 4>
Offsets;
4822 const Value *SrcV =
I.getOperand(0);
4824 SrcV->
getType(), ValueVTs,
nullptr, &Offsets, 0);
4825 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4826 "expect a single EVT for swifterror");
4835 SDValue(Src.getNode(), Src.getResNo()));
4836 DAG.setRoot(CopyNode);
4839void SelectionDAGBuilder::visitLoadFromSwiftError(
const LoadInst &
I) {
4840 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4841 "call visitLoadFromSwiftError when backend supports swifterror");
4844 !
I.hasMetadata(LLVMContext::MD_nontemporal) &&
4845 !
I.hasMetadata(LLVMContext::MD_invariant_load) &&
4846 "Support volatile, non temporal, invariant for load_from_swift_error");
4848 const Value *SV =
I.getOperand(0);
4849 Type *Ty =
I.getType();
4852 !
BatchAA->pointsToConstantMemory(MemoryLocation(
4854 I.getAAMetadata()))) &&
4855 "load_from_swift_error should not be constant memory");
4858 SmallVector<uint64_t, 4>
Offsets;
4860 ValueVTs,
nullptr, &Offsets, 0);
4861 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4862 "expect a single EVT for swifterror");
4872void SelectionDAGBuilder::visitStore(
const StoreInst &
I) {
4874 return visitAtomicStore(
I);
4876 const Value *SrcV =
I.getOperand(0);
4877 const Value *PtrV =
I.getOperand(1);
4879 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4884 if (Arg->hasSwiftErrorAttr())
4885 return visitStoreToSwiftError(
I);
4889 if (Alloca->isSwiftError())
4890 return visitStoreToSwiftError(
I);
4897 SrcV->
getType(), ValueVTs, &MemVTs, &Offsets);
4898 unsigned NumValues = ValueVTs.
size();
4911 Align Alignment =
I.getAlign();
4912 AAMDNodes AAInfo =
I.getAAMetadata();
4916 unsigned ChainI = 0;
4917 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4927 MachinePointerInfo PtrInfo =
4929 ? MachinePointerInfo(PtrV, Offsets[i].getKnownMinValue())
4930 : MachinePointerInfo();
4934 if (MemVTs[i] != ValueVTs[i])
4935 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVTs[i]);
4937 DAG.getStore(Root, dl, Val,
Add, PtrInfo, Alignment, MMOFlags, AAInfo);
4938 Chains[ChainI] = St;
4944 DAG.setRoot(StoreNode);
4947void SelectionDAGBuilder::visitMaskedStore(
const CallInst &
I,
4948 bool IsCompressing) {
4951 Value *Src0Operand =
I.getArgOperand(0);
4952 Value *PtrOperand =
I.getArgOperand(1);
4953 Value *MaskOperand =
I.getArgOperand(2);
4954 Align Alignment =
I.getParamAlign(1).valueOrOne();
4964 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
4967 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
4968 MachinePointerInfo(PtrOperand), MMOFlags,
4972 const auto &TLI =
DAG.getTargetLoweringInfo();
4975 !IsCompressing &&
TTI->hasConditionalLoadStoreForType(
4976 I.getArgOperand(0)->getType(),
true)
4982 DAG.setRoot(StoreNode);
5012 C =
C->getSplatValue();
5026 if (!
GEP ||
GEP->getParent() != CurBB)
5029 if (
GEP->getNumOperands() != 2)
5032 const Value *BasePtr =
GEP->getPointerOperand();
5033 const Value *IndexVal =
GEP->getOperand(
GEP->getNumOperands() - 1);
5039 TypeSize ScaleVal =
DL.getTypeAllocSize(
GEP->getResultElementType());
5044 if (ScaleVal != 1 &&
5056void SelectionDAGBuilder::visitMaskedScatter(
const CallInst &
I) {
5060 const Value *Ptr =
I.getArgOperand(1);
5064 Align Alignment =
I.getParamAlign(1).valueOrOne();
5065 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5074 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5084 EVT IdxVT =
Index.getValueType();
5092 SDValue Scatter =
DAG.getMaskedScatter(
DAG.getVTList(MVT::Other), VT, sdl,
5094 DAG.setRoot(Scatter);
5098void SelectionDAGBuilder::visitMaskedLoad(
const CallInst &
I,
bool IsExpanding) {
5101 Value *PtrOperand =
I.getArgOperand(0);
5102 Value *MaskOperand =
I.getArgOperand(1);
5103 Value *Src0Operand =
I.getArgOperand(2);
5104 Align Alignment =
I.getParamAlign(0).valueOrOne();
5112 AAMDNodes AAInfo =
I.getAAMetadata();
5119 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
5122 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
5124 if (
I.hasMetadata(LLVMContext::MD_invariant_load))
5127 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5128 MachinePointerInfo(PtrOperand), MMOFlags,
5131 const auto &TLI =
DAG.getTargetLoweringInfo();
5138 TTI->hasConditionalLoadStoreForType(Src0Operand->
getType(),
5143 DAG.getMaskedLoad(VT, sdl, InChain, Ptr,
Offset, Mask, Src0, VT, MMO,
5150void SelectionDAGBuilder::visitMaskedGather(
const CallInst &
I) {
5154 const Value *Ptr =
I.getArgOperand(0);
5158 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5160 Align Alignment =
I.getParamAlign(0).valueOrOne();
5171 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5183 EVT IdxVT =
Index.getValueType();
5192 DAG.getMaskedGather(
DAG.getVTList(VT, MVT::Other), VT, sdl,
Ops, MMO,
5208 SDVTList VTs =
DAG.getVTList(MemVT, MVT::i1, MVT::Other);
5210 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5213 MachineFunction &MF =
DAG.getMachineFunction();
5214 MachineMemOperand *MMO =
5217 nullptr, SSID, SuccessOrdering, FailureOrdering);
5220 dl, MemVT, VTs, InChain,
5228 DAG.setRoot(OutChain);
5231void SelectionDAGBuilder::visitAtomicRMW(
const AtomicRMWInst &
I) {
5234 switch (
I.getOperation()) {
5282 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5285 MachineFunction &MF =
DAG.getMachineFunction();
5287 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5288 I.getAlign(), AAMDNodes(),
nullptr, SSID, Ordering);
5291 DAG.getAtomic(NT, dl, MemVT, InChain,
5298 DAG.setRoot(OutChain);
5301void SelectionDAGBuilder::visitFence(
const FenceInst &
I) {
5303 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5306 Ops[1] =
DAG.getTargetConstant((
unsigned)
I.getOrdering(), dl,
5308 Ops[2] =
DAG.getTargetConstant(
I.getSyncScopeID(), dl,
5315void SelectionDAGBuilder::visitAtomicLoad(
const LoadInst &
I) {
5322 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5333 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5334 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5335 I.getAlign(), AAMDNodes(), Ranges, SSID, Order);
5345 L =
DAG.getPtrExtOrTrunc(L, dl, VT);
5348 DAG.setRoot(OutChain);
5351void SelectionDAGBuilder::visitAtomicStore(
const StoreInst &
I) {
5359 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5369 MachineFunction &MF =
DAG.getMachineFunction();
5371 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5372 I.getAlign(), AAMDNodes(),
nullptr, SSID, Ordering);
5376 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVT);
5383 DAG.setRoot(OutChain);
5391std::pair<bool, bool>
5392SelectionDAGBuilder::getTargetIntrinsicCallProperties(
const CallBase &
I) {
5394 bool HasChain = !
F->doesNotAccessMemory();
5396 HasChain &&
F->onlyReadsMemory() &&
F->willReturn() &&
F->doesNotThrow();
5398 return {HasChain, OnlyLoad};
5402 const CallBase &
I,
bool HasChain,
bool OnlyLoad,
5404 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5411 Ops.push_back(
DAG.getRoot());
5424 for (
unsigned i = 0, e =
I.arg_size(); i != e; ++i) {
5425 const Value *Arg =
I.getArgOperand(i);
5426 if (!
I.paramHasAttr(i, Attribute::ImmArg)) {
5434 assert(CI->getBitWidth() <= 64 &&
5435 "large intrinsic immediates not handled");
5436 Ops.push_back(
DAG.getTargetConstant(*CI, SDLoc(), VT));
5443 if (std::optional<OperandBundleUse> Bundle =
5445 auto *Sym = Bundle->Inputs[0].get();
5448 Ops.push_back(SDSym);
5451 if (std::optional<OperandBundleUse> Bundle =
5453 Value *Token = Bundle->Inputs[0].get();
5455 assert(
Ops.back().getValueType() != MVT::Glue &&
5456 "Did not expect another glue node here.");
5459 Ops.push_back(ConvControlToken);
5467 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5475 return DAG.getVTList(ValueVTs);
5479SDValue SelectionDAGBuilder::getTargetNonMemIntrinsicNode(
5502 if (
I.getType()->isVoidTy())
5517void SelectionDAGBuilder::visitTargetIntrinsic(
const CallInst &
I,
5519 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
5523 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5526 TargetLowering::IntrinsicInfo *
Info = !Infos.
empty() ? &Infos[0] :
nullptr;
5529 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad, Info);
5530 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
5535 Flags.copyFMF(*FPMO);
5536 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
5543 if (!Infos.
empty()) {
5546 MachineFunction &MF =
DAG.getMachineFunction();
5548 for (
const auto &Info : Infos) {
5551 MachinePointerInfo MPI;
5553 MPI = MachinePointerInfo(
Info.ptrVal,
Info.offset);
5554 else if (
Info.fallbackAddressSpace)
5555 MPI = MachinePointerInfo(*
Info.fallbackAddressSpace);
5556 EVT MemVT =
Info.memVT;
5558 if (
Size.hasValue() && !
Size.getValue())
5560 Align Alignment =
Info.align.value_or(
DAG.getEVTAlign(MemVT));
5562 MPI,
Info.flags,
Size, Alignment,
I.getAAMetadata(),
5570 Result = getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
5573 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
5630 SDValue TwoToFractionalPartOfX;
5707 if (
Op.getValueType() == MVT::f32 &&
5731 if (
Op.getValueType() == MVT::f32 &&
5830 if (
Op.getValueType() == MVT::f32 &&
5914 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log2ofMantissa);
5927 if (
Op.getValueType() == MVT::f32 &&
6004 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log10ofMantissa);
6015 if (
Op.getValueType() == MVT::f32 &&
6028 bool IsExp10 =
false;
6029 if (
LHS.getValueType() == MVT::f32 &&
RHS.getValueType() == MVT::f32 &&
6033 IsExp10 = LHSC->isExactlyValue(Ten);
6060 unsigned Val = RHSC->getSExtValue();
6089 CurSquare, CurSquare);
6094 if (RHSC->getSExtValue() < 0)
6108 EVT VT =
LHS.getValueType();
6131 if ((ScaleInt > 0 || (Saturating &&
Signed)) &&
6135 Opcode, VT, ScaleInt);
6170 switch (
N.getOpcode()) {
6174 Op.getValueType().getSizeInBits());
6199bool SelectionDAGBuilder::EmitFuncArgumentDbgValue(
6206 MachineFunction &MF =
DAG.getMachineFunction();
6207 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
6211 auto MakeVRegDbgValue = [&](
Register Reg, DIExpression *FragExpr,
6216 auto &Inst =
TII->get(TargetOpcode::DBG_INSTR_REF);
6223 auto *NewDIExpr = FragExpr;
6230 return BuildMI(MF,
DL, Inst,
false, MOs, Variable, NewDIExpr);
6233 auto &Inst =
TII->get(TargetOpcode::DBG_VALUE);
6234 return BuildMI(MF,
DL, Inst, Indirect,
Reg, Variable, FragExpr);
6238 if (Kind == FuncArgumentDbgValueKind::Value) {
6243 if (!IsInEntryBlock)
6259 bool VariableIsFunctionInputArg =
Variable->isParameter() &&
6260 !
DL->getInlinedAt();
6262 if (!IsInPrologue && !VariableIsFunctionInputArg)
6296 if (VariableIsFunctionInputArg) {
6298 if (ArgNo >=
FuncInfo.DescribedArgs.size())
6299 FuncInfo.DescribedArgs.resize(ArgNo + 1,
false);
6300 else if (!IsInPrologue &&
FuncInfo.DescribedArgs.test(ArgNo))
6301 return !NodeMap[
V].getNode();
6306 bool IsIndirect =
false;
6307 std::optional<MachineOperand>
Op;
6309 int FI =
FuncInfo.getArgumentFrameIndex(Arg);
6310 if (FI != std::numeric_limits<int>::max())
6314 if (!
Op &&
N.getNode()) {
6317 if (ArgRegsAndSizes.
size() == 1)
6318 Reg = ArgRegsAndSizes.
front().first;
6321 MachineRegisterInfo &RegInfo = MF.
getRegInfo();
6328 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6332 if (!
Op &&
N.getNode()) {
6336 if (FrameIndexSDNode *FINode =
6343 auto splitMultiRegDbgValue =
6356 uint64_t ExprFragmentSizeInBits = ExprFragmentInfo->SizeInBits;
6359 if (
Offset >= ExprFragmentSizeInBits)
6363 if (
Offset + RegFragmentSizeInBits > ExprFragmentSizeInBits) {
6364 RegFragmentSizeInBits = ExprFragmentSizeInBits -
Offset;
6369 Expr,
Offset, RegFragmentSizeInBits);
6373 if (!FragmentExpr) {
6374 SDDbgValue *SDV =
DAG.getConstantDbgValue(
6376 DAG.AddDbgValue(SDV,
false);
6379 MachineInstr *NewMI = MakeVRegDbgValue(
6380 Reg, *FragmentExpr, Kind != FuncArgumentDbgValueKind::Value);
6381 FuncInfo.ArgDbgValues.push_back(NewMI);
6390 if (VMI !=
FuncInfo.ValueMap.end()) {
6391 const auto &TLI =
DAG.getTargetLoweringInfo();
6392 RegsForValue RFV(
V->getContext(), TLI,
DAG.getDataLayout(), VMI->second,
6393 V->getType(), std::nullopt);
6394 if (RFV.occupiesMultipleRegs())
6395 return splitMultiRegDbgValue(RFV.getRegsAndSizes());
6398 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6399 }
else if (ArgRegsAndSizes.
size() > 1) {
6402 return splitMultiRegDbgValue(ArgRegsAndSizes);
6410 "Expected inlined-at fields to agree");
6411 MachineInstr *NewMI =
nullptr;
6414 NewMI = MakeVRegDbgValue(
Op->getReg(), Expr, IsIndirect);
6416 NewMI =
BuildMI(MF,
DL,
TII->get(TargetOpcode::DBG_VALUE),
true, *
Op,
6420 FuncInfo.ArgDbgValues.push_back(NewMI);
6429 unsigned DbgSDNodeOrder) {
6441 return DAG.getFrameIndexDbgValue(Variable, Expr, FISDN->getIndex(),
6442 false, dl, DbgSDNodeOrder);
6444 return DAG.getDbgValue(Variable, Expr,
N.getNode(),
N.getResNo(),
6445 false, dl, DbgSDNodeOrder);
6450 case Intrinsic::smul_fix:
6452 case Intrinsic::umul_fix:
6454 case Intrinsic::smul_fix_sat:
6456 case Intrinsic::umul_fix_sat:
6458 case Intrinsic::sdiv_fix:
6460 case Intrinsic::udiv_fix:
6462 case Intrinsic::sdiv_fix_sat:
6464 case Intrinsic::udiv_fix_sat:
6477 "expected call_preallocated_setup Value");
6478 for (
const auto *U : PreallocatedSetup->
users()) {
6480 const Function *Fn = UseCall->getCalledFunction();
6481 if (!Fn || Fn->
getIntrinsicID() != Intrinsic::call_preallocated_arg) {
6491bool SelectionDAGBuilder::visitEntryValueDbgValue(
6501 auto ArgIt =
FuncInfo.ValueMap.find(Arg);
6502 if (ArgIt ==
FuncInfo.ValueMap.end()) {
6504 dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6505 "couldn't find an associated register for the Argument\n");
6508 Register ArgVReg = ArgIt->getSecond();
6510 for (
auto [PhysReg, VirtReg] :
FuncInfo.RegInfo->liveins())
6511 if (ArgVReg == VirtReg || ArgVReg == PhysReg) {
6512 SDDbgValue *SDV =
DAG.getVRegDbgValue(
6513 Variable, Expr, PhysReg,
false , DbgLoc, SDNodeOrder);
6514 DAG.AddDbgValue(SDV,
false );
6517 LLVM_DEBUG(
dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6518 "couldn't find a physical register\n");
6523void SelectionDAGBuilder::visitConvergenceControl(
const CallInst &
I,
6526 switch (Intrinsic) {
6527 case Intrinsic::experimental_convergence_anchor:
6530 case Intrinsic::experimental_convergence_entry:
6533 case Intrinsic::experimental_convergence_loop: {
6535 auto *Token = Bundle->Inputs[0].get();
6543void SelectionDAGBuilder::visitVectorHistogram(
const CallInst &
I,
6544 unsigned IntrinsicID) {
6547 assert(IntrinsicID == Intrinsic::experimental_vector_histogram_add &&
6548 "Tried to lower unsupported histogram type");
6554 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6555 DataLayout TargetDL =
DAG.getDataLayout();
6557 Align Alignment =
DAG.getEVTAlign(VT);
6570 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
6571 MachinePointerInfo(AS),
6582 EVT IdxVT =
Index.getValueType();
6593 SDValue ID =
DAG.getTargetConstant(IntrinsicID, sdl, MVT::i32);
6596 SDValue Histogram =
DAG.getMaskedHistogram(
DAG.getVTList(MVT::Other), VT, sdl,
6600 DAG.setRoot(Histogram);
6603void SelectionDAGBuilder::visitVectorExtractLastActive(
const CallInst &
I,
6605 assert(Intrinsic == Intrinsic::experimental_vector_extract_last_active &&
6606 "Tried lowering invalid vector extract last");
6608 const DataLayout &Layout =
DAG.getDataLayout();
6612 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6622 EVT BoolVT =
Mask.getValueType().getScalarType();
6624 Result =
DAG.getSelect(sdl, ResVT, AnyActive, Result, PassThru);
6631void SelectionDAGBuilder::visitIntrinsicCall(
const CallInst &
I,
6633 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6640 Flags.copyFMF(*FPOp);
6642 switch (Intrinsic) {
6645 visitTargetIntrinsic(
I, Intrinsic);
6647 case Intrinsic::vscale: {
6652 case Intrinsic::vastart: visitVAStart(
I);
return;
6653 case Intrinsic::vaend: visitVAEnd(
I);
return;
6654 case Intrinsic::vacopy: visitVACopy(
I);
return;
6655 case Intrinsic::returnaddress:
6660 case Intrinsic::addressofreturnaddress:
6665 case Intrinsic::sponentry:
6670 case Intrinsic::frameaddress:
6675 case Intrinsic::read_volatile_register:
6676 case Intrinsic::read_register: {
6677 Value *
Reg =
I.getArgOperand(0);
6683 DAG.getVTList(VT, MVT::Other), Chain,
RegName);
6688 case Intrinsic::write_register: {
6689 Value *
Reg =
I.getArgOperand(0);
6690 Value *RegValue =
I.getArgOperand(1);
6698 case Intrinsic::write_volatile_register: {
6699 Value *
Reg =
I.getArgOperand(0);
6700 Value *RegValue =
I.getArgOperand(1);
6713 const MachineFunction &MF =
DAG.getMachineFunction();
6717 const TargetRegisterInfo *TRI = MF.getSubtarget().getRegisterInfo();
6718 const TargetRegisterClass *RC = TRI->getMinimalPhysRegClass(PhysReg);
6719 MVT RegVT = *TRI->legalclasstypes_begin(*RC);
6720 DAG.setRoot(DAG.getNode(ISD::FAKE_USE, sdl, MVT::Other,
6721 {WriteChain, DAG.getRegister(PhysReg, RegVT)}));
6723 DAG.setRoot(WriteChain);
6727 case Intrinsic::memcpy:
6728 case Intrinsic::memcpy_inline: {
6734 "memcpy_inline needs constant size");
6736 Align DstAlign = MCI.getDestAlign().valueOrOne();
6737 Align SrcAlign = MCI.getSourceAlign().valueOrOne();
6738 bool isVol = MCI.isVolatile();
6740 SDValue MC =
DAG.getMemcpy(Root, sdl, Dst, Src,
Size, DstAlign, SrcAlign,
6741 isVol, MCI.isForceInlined(), &
I, std::nullopt,
6742 MachinePointerInfo(
I.getArgOperand(0)),
6743 MachinePointerInfo(
I.getArgOperand(1)),
6745 updateDAGForMaybeTailCall(MC);
6748 case Intrinsic::memset:
6749 case Intrinsic::memset_inline: {
6755 "memset_inline needs constant size");
6757 Align DstAlign = MSII.getDestAlign().valueOrOne();
6758 bool isVol = MSII.isVolatile();
6761 Root, sdl, Dst, Value,
Size, DstAlign, isVol, MSII.isForceInlined(),
6762 &
I, MachinePointerInfo(
I.getArgOperand(0)),
I.getAAMetadata());
6763 updateDAGForMaybeTailCall(MC);
6766 case Intrinsic::memmove: {
6772 Align DstAlign = MMI.getDestAlign().valueOrOne();
6773 Align SrcAlign = MMI.getSourceAlign().valueOrOne();
6774 bool isVol = MMI.isVolatile();
6777 Root, sdl, Op1, Op2, Op3, DstAlign, SrcAlign, isVol, &
I,
6779 MachinePointerInfo(
I.getArgOperand(0)),
6780 MachinePointerInfo(
I.getArgOperand(1)),
I.getAAMetadata(),
BatchAA);
6781 updateDAGForMaybeTailCall(MM);
6784 case Intrinsic::memcpy_element_unordered_atomic: {
6790 Type *LengthTy =
MI.getLength()->getType();
6791 unsigned ElemSz =
MI.getElementSizeInBytes();
6795 isTC, MachinePointerInfo(
MI.getRawDest()),
6796 MachinePointerInfo(
MI.getRawSource()));
6797 updateDAGForMaybeTailCall(MC);
6800 case Intrinsic::memmove_element_unordered_atomic: {
6806 Type *LengthTy =
MI.getLength()->getType();
6807 unsigned ElemSz =
MI.getElementSizeInBytes();
6811 isTC, MachinePointerInfo(
MI.getRawDest()),
6812 MachinePointerInfo(
MI.getRawSource()));
6813 updateDAGForMaybeTailCall(MC);
6816 case Intrinsic::memset_element_unordered_atomic: {
6822 Type *LengthTy =
MI.getLength()->getType();
6823 unsigned ElemSz =
MI.getElementSizeInBytes();
6827 isTC, MachinePointerInfo(
MI.getRawDest()));
6828 updateDAGForMaybeTailCall(MC);
6831 case Intrinsic::call_preallocated_setup: {
6833 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6840 case Intrinsic::call_preallocated_arg: {
6842 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6856 case Intrinsic::eh_typeid_for: {
6859 unsigned TypeID =
DAG.getMachineFunction().getTypeIDFor(GV);
6860 Res =
DAG.getConstant(
TypeID, sdl, MVT::i32);
6865 case Intrinsic::eh_return_i32:
6866 case Intrinsic::eh_return_i64:
6867 DAG.getMachineFunction().setCallsEHReturn(
true);
6874 case Intrinsic::eh_unwind_init:
6875 DAG.getMachineFunction().setCallsUnwindInit(
true);
6877 case Intrinsic::eh_dwarf_cfa:
6882 case Intrinsic::eh_sjlj_callsite: {
6884 assert(
FuncInfo.getCurrentCallSite() == 0 &&
"Overlapping call sites!");
6889 case Intrinsic::eh_sjlj_functioncontext: {
6891 MachineFrameInfo &MFI =
DAG.getMachineFunction().getFrameInfo();
6894 int FI =
FuncInfo.StaticAllocaMap[FnCtx];
6898 case Intrinsic::eh_sjlj_setjmp: {
6903 DAG.getVTList(MVT::i32, MVT::Other),
Ops);
6905 DAG.setRoot(
Op.getValue(1));
6908 case Intrinsic::eh_sjlj_longjmp:
6912 case Intrinsic::eh_sjlj_setup_dispatch:
6916 case Intrinsic::masked_gather:
6917 visitMaskedGather(
I);
6919 case Intrinsic::masked_load:
6922 case Intrinsic::masked_scatter:
6923 visitMaskedScatter(
I);
6925 case Intrinsic::masked_store:
6926 visitMaskedStore(
I);
6928 case Intrinsic::masked_expandload:
6929 visitMaskedLoad(
I,
true );
6931 case Intrinsic::masked_compressstore:
6932 visitMaskedStore(
I,
true );
6934 case Intrinsic::powi:
6938 case Intrinsic::log:
6941 case Intrinsic::log2:
6945 case Intrinsic::log10:
6949 case Intrinsic::exp:
6952 case Intrinsic::exp2:
6956 case Intrinsic::pow:
6960 case Intrinsic::sqrt:
6961 case Intrinsic::fabs:
6962 case Intrinsic::sin:
6963 case Intrinsic::cos:
6964 case Intrinsic::tan:
6965 case Intrinsic::asin:
6966 case Intrinsic::acos:
6967 case Intrinsic::atan:
6968 case Intrinsic::sinh:
6969 case Intrinsic::cosh:
6970 case Intrinsic::tanh:
6971 case Intrinsic::exp10:
6972 case Intrinsic::floor:
6973 case Intrinsic::ceil:
6974 case Intrinsic::trunc:
6975 case Intrinsic::rint:
6976 case Intrinsic::nearbyint:
6977 case Intrinsic::round:
6978 case Intrinsic::roundeven:
6979 case Intrinsic::canonicalize: {
6982 switch (Intrinsic) {
6984 case Intrinsic::sqrt: Opcode =
ISD::FSQRT;
break;
6985 case Intrinsic::fabs: Opcode =
ISD::FABS;
break;
6986 case Intrinsic::sin: Opcode =
ISD::FSIN;
break;
6987 case Intrinsic::cos: Opcode =
ISD::FCOS;
break;
6988 case Intrinsic::tan: Opcode =
ISD::FTAN;
break;
6989 case Intrinsic::asin: Opcode =
ISD::FASIN;
break;
6990 case Intrinsic::acos: Opcode =
ISD::FACOS;
break;
6991 case Intrinsic::atan: Opcode =
ISD::FATAN;
break;
6992 case Intrinsic::sinh: Opcode =
ISD::FSINH;
break;
6993 case Intrinsic::cosh: Opcode =
ISD::FCOSH;
break;
6994 case Intrinsic::tanh: Opcode =
ISD::FTANH;
break;
6995 case Intrinsic::exp10: Opcode =
ISD::FEXP10;
break;
6996 case Intrinsic::floor: Opcode =
ISD::FFLOOR;
break;
6997 case Intrinsic::ceil: Opcode =
ISD::FCEIL;
break;
6998 case Intrinsic::trunc: Opcode =
ISD::FTRUNC;
break;
6999 case Intrinsic::rint: Opcode =
ISD::FRINT;
break;
7001 case Intrinsic::round: Opcode =
ISD::FROUND;
break;
7008 getValue(
I.getArgOperand(0)).getValueType(),
7012 case Intrinsic::atan2:
7014 getValue(
I.getArgOperand(0)).getValueType(),
7018 case Intrinsic::lround:
7019 case Intrinsic::llround:
7020 case Intrinsic::lrint:
7021 case Intrinsic::llrint: {
7024 switch (Intrinsic) {
7026 case Intrinsic::lround: Opcode =
ISD::LROUND;
break;
7028 case Intrinsic::lrint: Opcode =
ISD::LRINT;
break;
7029 case Intrinsic::llrint: Opcode =
ISD::LLRINT;
break;
7038 case Intrinsic::minnum:
7040 getValue(
I.getArgOperand(0)).getValueType(),
7044 case Intrinsic::maxnum:
7046 getValue(
I.getArgOperand(0)).getValueType(),
7050 case Intrinsic::minimum:
7052 getValue(
I.getArgOperand(0)).getValueType(),
7056 case Intrinsic::maximum:
7058 getValue(
I.getArgOperand(0)).getValueType(),
7062 case Intrinsic::minimumnum:
7064 getValue(
I.getArgOperand(0)).getValueType(),
7068 case Intrinsic::maximumnum:
7070 getValue(
I.getArgOperand(0)).getValueType(),
7074 case Intrinsic::copysign:
7076 getValue(
I.getArgOperand(0)).getValueType(),
7080 case Intrinsic::ldexp:
7082 getValue(
I.getArgOperand(0)).getValueType(),
7086 case Intrinsic::modf:
7087 case Intrinsic::sincos:
7088 case Intrinsic::sincospi:
7089 case Intrinsic::frexp: {
7091 switch (Intrinsic) {
7094 case Intrinsic::sincos:
7097 case Intrinsic::sincospi:
7100 case Intrinsic::modf:
7103 case Intrinsic::frexp:
7109 SDVTList VTs =
DAG.getVTList(ValueVTs);
7111 &
I,
DAG.getNode(Opcode, sdl, VTs,
getValue(
I.getArgOperand(0)), Flags));
7114 case Intrinsic::arithmetic_fence: {
7116 getValue(
I.getArgOperand(0)).getValueType(),
7120 case Intrinsic::fma:
7126#define INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC) \
7127 case Intrinsic::INTRINSIC:
7128#include "llvm/IR/ConstrainedOps.def"
7131#define BEGIN_REGISTER_VP_INTRINSIC(VPID, ...) case Intrinsic::VPID:
7132#include "llvm/IR/VPIntrinsics.def"
7135 case Intrinsic::fptrunc_round: {
7139 std::optional<RoundingMode> RoundMode =
7147 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
7152 DAG.getTargetConstant((
int)*RoundMode, sdl, MVT::i32));
7157 case Intrinsic::fmuladd: {
7162 getValue(
I.getArgOperand(0)).getValueType(),
7169 getValue(
I.getArgOperand(0)).getValueType(),
7185 case Intrinsic::fptosi_sat: {
7192 case Intrinsic::fptoui_sat: {
7199 case Intrinsic::convert_from_arbitrary_fp: {
7204 const fltSemantics *SrcSem =
7207 DAG.getContext()->emitError(
7208 "convert_from_arbitrary_fp: not implemented format '" + FormatStr +
7219 DAG.getTargetConstant(
static_cast<int>(SemEnum), sdl, MVT::i32);
7224 case Intrinsic::convert_to_arbitrary_fp: {
7229 const fltSemantics *DstSem =
7232 DAG.getContext()->emitError(
7233 "convert_to_arbitrary_fp: not implemented format '" + FormatStr +
7245 "Dynamic rounding mode should have been rejected by the verifier");
7253 DAG.getTargetConstant(
static_cast<int>(SemEnum), sdl, MVT::i32);
7255 DAG.getTargetConstant(
static_cast<int>(*RoundMode), sdl, MVT::i32);
7256 SDValue SatConst =
DAG.getTargetConstant(Saturate, sdl, MVT::i32);
7258 SemConst, RoundConst, SatConst));
7261 case Intrinsic::set_rounding:
7267 case Intrinsic::is_fpclass: {
7268 const DataLayout DLayout =
DAG.getDataLayout();
7270 EVT ArgVT = TLI.
getValueType(DLayout,
I.getArgOperand(0)->getType());
7273 MachineFunction &MF =
DAG.getMachineFunction();
7277 Flags.setNoFPExcept(
7278 !
F.getAttributes().hasFnAttr(llvm::Attribute::StrictFP));
7294 case Intrinsic::get_fpenv: {
7295 const DataLayout DLayout =
DAG.getDataLayout();
7297 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7312 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7315 Chain =
DAG.getGetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7316 Res =
DAG.getLoad(EnvVT, sdl, Chain, Temp, MPI);
7322 case Intrinsic::set_fpenv: {
7323 const DataLayout DLayout =
DAG.getDataLayout();
7326 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7339 Chain =
DAG.getStore(Chain, sdl, Env, Temp, MPI, TempAlign,
7341 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7344 Chain =
DAG.getSetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7349 case Intrinsic::reset_fpenv:
7352 case Intrinsic::get_fpmode:
7361 case Intrinsic::set_fpmode:
7366 case Intrinsic::reset_fpmode: {
7371 case Intrinsic::pcmarker: {
7376 case Intrinsic::readcyclecounter: {
7379 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7384 case Intrinsic::readsteadycounter: {
7387 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7392 case Intrinsic::bitreverse:
7394 getValue(
I.getArgOperand(0)).getValueType(),
7397 case Intrinsic::bswap:
7399 getValue(
I.getArgOperand(0)).getValueType(),
7402 case Intrinsic::cttz: {
7410 case Intrinsic::ctlz: {
7418 case Intrinsic::ctpop: {
7424 case Intrinsic::fshl:
7425 case Intrinsic::fshr: {
7426 bool IsFSHL =
Intrinsic == Intrinsic::fshl;
7430 EVT VT =
X.getValueType();
7441 case Intrinsic::clmul: {
7447 case Intrinsic::pext: {
7453 case Intrinsic::pdep: {
7459 case Intrinsic::sadd_sat: {
7465 case Intrinsic::uadd_sat: {
7471 case Intrinsic::ssub_sat: {
7477 case Intrinsic::usub_sat: {
7483 case Intrinsic::sshl_sat:
7484 case Intrinsic::ushl_sat: {
7488 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
7493 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
7496 "Unexpected shift type");
7505 case Intrinsic::smul_fix:
7506 case Intrinsic::umul_fix:
7507 case Intrinsic::smul_fix_sat:
7508 case Intrinsic::umul_fix_sat: {
7516 case Intrinsic::sdiv_fix:
7517 case Intrinsic::udiv_fix:
7518 case Intrinsic::sdiv_fix_sat:
7519 case Intrinsic::udiv_fix_sat: {
7524 Op1, Op2, Op3,
DAG, TLI));
7527 case Intrinsic::smax: {
7533 case Intrinsic::smin: {
7539 case Intrinsic::umax: {
7545 case Intrinsic::umin: {
7551 case Intrinsic::abs: {
7558 case Intrinsic::scmp: {
7565 case Intrinsic::ucmp: {
7572 case Intrinsic::stackaddress:
7573 case Intrinsic::stacksave: {
7578 Res =
DAG.getNode(SDOpcode, sdl,
DAG.getVTList(VT, MVT::Other),
Op);
7583 case Intrinsic::stackrestore:
7587 case Intrinsic::get_dynamic_area_offset: {
7596 case Intrinsic::stackguard: {
7597 MachineFunction &MF =
DAG.getMachineFunction();
7603 Res =
DAG.getPtrExtOrTrunc(Res, sdl, PtrTy);
7607 LLVMContext &Ctx = *
DAG.getContext();
7608 Ctx.
diagnose(DiagnosticInfoGeneric(
"unable to lower stackguard"));
7615 MachinePointerInfo(
Global, 0), Align,
7624 case Intrinsic::stackprotector: {
7626 MachineFunction &MF =
DAG.getMachineFunction();
7646 Chain, sdl, Src, FIN,
7653 case Intrinsic::objectsize:
7656 case Intrinsic::is_constant:
7659 case Intrinsic::annotation:
7660 case Intrinsic::ptr_annotation:
7661 case Intrinsic::launder_invariant_group:
7662 case Intrinsic::strip_invariant_group:
7667 case Intrinsic::type_test:
7668 case Intrinsic::public_type_test:
7670 "LowerTypeTests pass before code generation");
7673 case Intrinsic::assume:
7674 case Intrinsic::experimental_noalias_scope_decl:
7675 case Intrinsic::var_annotation:
7676 case Intrinsic::sideeffect:
7681 case Intrinsic::codeview_annotation: {
7683 MachineFunction &MF =
DAG.getMachineFunction();
7692 case Intrinsic::init_trampoline: {
7700 Ops[4] =
DAG.getSrcValue(
I.getArgOperand(0));
7708 case Intrinsic::adjust_trampoline:
7713 case Intrinsic::gcroot: {
7714 assert(
DAG.getMachineFunction().getFunction().hasGC() &&
7715 "only valid in functions with gc specified, enforced by Verifier");
7717 const Value *Alloca =
I.getArgOperand(0)->stripPointerCasts();
7724 case Intrinsic::gcread:
7725 case Intrinsic::gcwrite:
7727 case Intrinsic::get_rounding:
7733 case Intrinsic::expect:
7734 case Intrinsic::expect_with_probability:
7740 case Intrinsic::ubsantrap:
7741 case Intrinsic::debugtrap:
7742 case Intrinsic::trap: {
7743 StringRef TrapFuncName =
7744 I.getAttributes().getFnAttr(
"trap-func-name").getValueAsString();
7745 if (TrapFuncName.
empty()) {
7746 switch (Intrinsic) {
7747 case Intrinsic::trap:
7750 case Intrinsic::debugtrap:
7753 case Intrinsic::ubsantrap:
7756 DAG.getTargetConstant(
7762 DAG.addNoMergeSiteInfo(
DAG.getRoot().getNode(),
7763 I.hasFnAttr(Attribute::NoMerge));
7767 if (Intrinsic == Intrinsic::ubsantrap) {
7768 Value *Arg =
I.getArgOperand(0);
7772 TargetLowering::CallLoweringInfo CLI(
DAG);
7773 CLI.setDebugLoc(sdl).setChain(
getRoot()).setLibCallee(
7775 DAG.getExternalSymbol(TrapFuncName.
data(),
7778 CLI.NoMerge =
I.hasFnAttr(Attribute::NoMerge);
7784 case Intrinsic::allow_runtime_check:
7785 case Intrinsic::allow_ubsan_check:
7789 case Intrinsic::uadd_with_overflow:
7790 case Intrinsic::sadd_with_overflow:
7791 case Intrinsic::usub_with_overflow:
7792 case Intrinsic::ssub_with_overflow:
7793 case Intrinsic::umul_with_overflow:
7794 case Intrinsic::smul_with_overflow: {
7796 switch (Intrinsic) {
7798 case Intrinsic::uadd_with_overflow:
Op =
ISD::UADDO;
break;
7799 case Intrinsic::sadd_with_overflow:
Op =
ISD::SADDO;
break;
7800 case Intrinsic::usub_with_overflow:
Op =
ISD::USUBO;
break;
7801 case Intrinsic::ssub_with_overflow:
Op =
ISD::SSUBO;
break;
7802 case Intrinsic::umul_with_overflow:
Op =
ISD::UMULO;
break;
7803 case Intrinsic::smul_with_overflow:
Op =
ISD::SMULO;
break;
7811 SDVTList VTs =
DAG.getVTList(ResultVT, OverflowVT);
7815 case Intrinsic::prefetch: {
7830 std::nullopt, Flags);
7836 DAG.setRoot(Result);
7839 case Intrinsic::lifetime_start:
7840 case Intrinsic::lifetime_end: {
7841 bool IsStart = (
Intrinsic == Intrinsic::lifetime_start);
7847 if (!LifetimeObject)
7852 auto SI =
FuncInfo.StaticAllocaMap.find(LifetimeObject);
7853 if (SI ==
FuncInfo.StaticAllocaMap.end())
7857 Res =
DAG.getLifetimeNode(IsStart, sdl,
getRoot(), FrameIndex);
7861 case Intrinsic::pseudoprobe: {
7869 case Intrinsic::invariant_start:
7874 case Intrinsic::invariant_end:
7877 case Intrinsic::clear_cache: {
7882 {InputChain, StartVal, EndVal});
7887 case Intrinsic::donothing:
7888 case Intrinsic::seh_try_begin:
7889 case Intrinsic::seh_scope_begin:
7890 case Intrinsic::seh_try_end:
7891 case Intrinsic::seh_scope_end:
7894 case Intrinsic::experimental_stackmap:
7897 case Intrinsic::experimental_patchpoint_void:
7898 case Intrinsic::experimental_patchpoint:
7901 case Intrinsic::experimental_gc_statepoint:
7904 case Intrinsic::experimental_gc_result:
7907 case Intrinsic::experimental_gc_relocate:
7910 case Intrinsic::instrprof_cover:
7912 case Intrinsic::instrprof_increment:
7914 case Intrinsic::instrprof_timestamp:
7916 case Intrinsic::instrprof_value_profile:
7918 case Intrinsic::instrprof_mcdc_parameters:
7920 case Intrinsic::instrprof_mcdc_tvbitmap_update:
7922 case Intrinsic::localescape: {
7923 MachineFunction &MF =
DAG.getMachineFunction();
7924 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
7928 for (
unsigned Idx = 0,
E =
I.arg_size(); Idx <
E; ++Idx) {
7934 "can only escape static allocas");
7939 TII->get(TargetOpcode::LOCAL_ESCAPE))
7947 case Intrinsic::localrecover: {
7949 MachineFunction &MF =
DAG.getMachineFunction();
7955 unsigned(Idx->getLimitedValue(std::numeric_limits<int>::max()));
7959 Value *
FP =
I.getArgOperand(1);
7965 SDValue OffsetSym =
DAG.getMCSymbol(FrameAllocSym, PtrVT);
7970 SDValue Add =
DAG.getMemBasePlusOffset(FPVal, OffsetVal, sdl);
7976 case Intrinsic::fake_use: {
7977 Value *
V =
I.getArgOperand(0);
7982 auto FakeUseValue = [&]() ->
SDValue {
7996 if (!FakeUseValue || FakeUseValue.isUndef())
7999 Ops[1] = FakeUseValue;
8008 case Intrinsic::reloc_none: {
8013 DAG.getTargetExternalSymbol(
8019 case Intrinsic::cond_loop: {
8029 case Intrinsic::eh_exceptionpointer:
8030 case Intrinsic::eh_exceptioncode: {
8036 SDValue N =
DAG.getCopyFromReg(
DAG.getEntryNode(), sdl, VReg, PtrVT);
8037 if (Intrinsic == Intrinsic::eh_exceptioncode)
8038 N =
DAG.getZExtOrTrunc(
N, sdl, MVT::i32);
8042 case Intrinsic::xray_customevent: {
8045 const auto &Triple =
DAG.getTarget().getTargetTriple();
8046 if (!Triple.isAArch64(64) && Triple.getArch() !=
Triple::x86_64 &&
8055 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
8057 Ops.push_back(LogEntryVal);
8058 Ops.push_back(StrSizeVal);
8059 Ops.push_back(Chain);
8065 MachineSDNode *MN =
DAG.getMachineNode(TargetOpcode::PATCHABLE_EVENT_CALL,
8068 DAG.setRoot(patchableNode);
8072 case Intrinsic::xray_typedevent: {
8075 const auto &Triple =
DAG.getTarget().getTargetTriple();
8076 if (!Triple.isAArch64(64) && Triple.getArch() !=
Triple::x86_64 &&
8088 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
8090 Ops.push_back(LogTypeId);
8091 Ops.push_back(LogEntryVal);
8092 Ops.push_back(StrSizeVal);
8093 Ops.push_back(Chain);
8099 MachineSDNode *MN =
DAG.getMachineNode(
8100 TargetOpcode::PATCHABLE_TYPED_EVENT_CALL, sdl, NodeTys,
Ops);
8102 DAG.setRoot(patchableNode);
8106 case Intrinsic::experimental_deoptimize:
8109 case Intrinsic::stepvector:
8112 case Intrinsic::vector_reduce_fadd:
8113 case Intrinsic::vector_reduce_fmul:
8114 case Intrinsic::vector_reduce_add:
8115 case Intrinsic::vector_reduce_mul:
8116 case Intrinsic::vector_reduce_and:
8117 case Intrinsic::vector_reduce_or:
8118 case Intrinsic::vector_reduce_xor:
8119 case Intrinsic::vector_reduce_smax:
8120 case Intrinsic::vector_reduce_smin:
8121 case Intrinsic::vector_reduce_umax:
8122 case Intrinsic::vector_reduce_umin:
8123 case Intrinsic::vector_reduce_fmax:
8124 case Intrinsic::vector_reduce_fmin:
8125 case Intrinsic::vector_reduce_fmaximum:
8126 case Intrinsic::vector_reduce_fminimum:
8127 visitVectorReduce(
I, Intrinsic);
8130 case Intrinsic::icall_branch_funnel: {
8136 I.getArgOperand(1),
Offset,
DAG.getDataLayout()));
8139 "llvm.icall.branch.funnel operand must be a GlobalValue");
8140 Ops.push_back(
DAG.getTargetGlobalAddress(
Base, sdl, MVT::i64, 0));
8142 struct BranchFunnelTarget {
8148 for (
unsigned Op = 1,
N =
I.arg_size();
Op !=
N;
Op += 2) {
8151 if (ElemBase !=
Base)
8153 "to the same GlobalValue");
8159 "llvm.icall.branch.funnel operand must be a GlobalValue");
8165 [](
const BranchFunnelTarget &
T1,
const BranchFunnelTarget &T2) {
8166 return T1.Offset < T2.Offset;
8169 for (
auto &
T : Targets) {
8170 Ops.push_back(
DAG.getTargetConstant(
T.Offset, sdl, MVT::i32));
8171 Ops.push_back(
T.Target);
8174 Ops.push_back(
DAG.getRoot());
8175 SDValue N(
DAG.getMachineNode(TargetOpcode::ICALL_BRANCH_FUNNEL, sdl,
8184 case Intrinsic::wasm_landingpad_index:
8190 case Intrinsic::aarch64_settag:
8191 case Intrinsic::aarch64_settag_zero: {
8192 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
8193 bool ZeroMemory =
Intrinsic == Intrinsic::aarch64_settag_zero;
8196 getValue(
I.getArgOperand(1)), MachinePointerInfo(
I.getArgOperand(0)),
8202 case Intrinsic::amdgcn_cs_chain: {
8207 Type *RetTy =
I.getType();
8217 for (
unsigned Idx : {2, 3, 1}) {
8218 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8220 Arg.setAttributes(&
I, Idx);
8221 Args.push_back(Arg);
8224 assert(Args[0].IsInReg &&
"SGPR args should be marked inreg");
8225 assert(!Args[1].IsInReg &&
"VGPR args should not be marked inreg");
8226 Args[2].IsInReg =
true;
8229 for (
unsigned Idx = 4; Idx <
I.arg_size(); ++Idx) {
8230 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8232 Arg.setAttributes(&
I, Idx);
8233 Args.push_back(Arg);
8236 TargetLowering::CallLoweringInfo CLI(
DAG);
8239 .setCallee(CC, RetTy, Callee, std::move(Args))
8242 .setConvergent(
I.isConvergent());
8244 std::pair<SDValue, SDValue>
Result =
8248 "Should've lowered as tail call");
8253 case Intrinsic::amdgcn_call_whole_wave: {
8255 bool isTailCall =
I.isTailCall();
8258 for (
unsigned Idx = 1; Idx <
I.arg_size(); ++Idx) {
8259 TargetLowering::ArgListEntry Arg(
getValue(
I.getArgOperand(Idx)),
8260 I.getArgOperand(Idx)->getType());
8261 Arg.setAttributes(&
I, Idx);
8268 Args.push_back(Arg);
8273 auto *Token = Bundle->Inputs[0].get();
8274 ConvControlToken =
getValue(Token);
8277 TargetLowering::CallLoweringInfo CLI(
DAG);
8281 getValue(
I.getArgOperand(0)), std::move(Args))
8285 .setConvergent(
I.isConvergent())
8286 .setConvergenceControlToken(ConvControlToken);
8289 std::pair<SDValue, SDValue>
Result =
8292 if (
Result.first.getNode())
8296 case Intrinsic::ptrmask: {
8312 auto HighOnes =
DAG.getNode(
8313 ISD::SHL, sdl, PtrVT,
DAG.getAllOnesConstant(sdl, PtrVT),
8314 DAG.getShiftAmountConstant(
Mask.getValueType().getFixedSizeInBits(),
8317 DAG.getZExtOrTrunc(Mask, sdl, PtrVT), HighOnes);
8318 }
else if (
Mask.getValueType() != PtrVT)
8319 Mask =
DAG.getPtrExtOrTrunc(Mask, sdl, PtrVT);
8325 case Intrinsic::threadlocal_address: {
8329 case Intrinsic::get_active_lane_mask: {
8333 EVT ElementVT =
Index.getValueType();
8344 SDValue VectorIndex =
DAG.getSplat(VecTy, sdl, Index);
8345 SDValue VectorTripCount =
DAG.getSplat(VecTy, sdl, TripCount);
8346 SDValue VectorStep =
DAG.getStepVector(sdl, VecTy);
8349 SDValue SetCC =
DAG.getSetCC(sdl, CCVT, VectorInduction,
8354 case Intrinsic::experimental_get_vector_length: {
8356 "Expected positive VF");
8361 EVT CountVT =
Count.getValueType();
8364 visitTargetIntrinsic(
I, Intrinsic);
8373 if (CountVT.
bitsLT(VT)) {
8378 SDValue MaxEVL =
DAG.getElementCount(sdl, CountVT,
8388 case Intrinsic::vector_partial_reduce_add: {
8396 case Intrinsic::vector_partial_reduce_fadd: {
8404 case Intrinsic::experimental_cttz_elts: {
8406 EVT OpVT =
Op.getValueType();
8413 SDValue AllZero =
DAG.getConstant(0, sdl, OpVT);
8422 case Intrinsic::vector_insert: {
8430 if (
Index.getValueType() != VectorIdxTy)
8431 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8438 case Intrinsic::vector_extract: {
8446 if (
Index.getValueType() != VectorIdxTy)
8447 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8453 case Intrinsic::experimental_vector_match: {
8459 EVT ResVT =
Mask.getValueType();
8465 visitTargetIntrinsic(
I, Intrinsic);
8469 SDValue Ret =
DAG.getConstant(0, sdl, ResVT);
8471 for (
unsigned i = 0; i < SearchSize; ++i) {
8474 DAG.getVectorIdxConstant(i, sdl));
8477 Ret =
DAG.getNode(
ISD::OR, sdl, ResVT, Ret, Cmp);
8483 case Intrinsic::vector_reverse:
8484 visitVectorReverse(
I);
8486 case Intrinsic::vector_splice_left:
8487 case Intrinsic::vector_splice_right:
8488 visitVectorSplice(
I);
8490 case Intrinsic::callbr_landingpad:
8491 visitCallBrLandingPad(
I);
8493 case Intrinsic::vector_interleave2:
8494 visitVectorInterleave(
I, 2);
8496 case Intrinsic::vector_interleave3:
8497 visitVectorInterleave(
I, 3);
8499 case Intrinsic::vector_interleave4:
8500 visitVectorInterleave(
I, 4);
8502 case Intrinsic::vector_interleave5:
8503 visitVectorInterleave(
I, 5);
8505 case Intrinsic::vector_interleave6:
8506 visitVectorInterleave(
I, 6);
8508 case Intrinsic::vector_interleave7:
8509 visitVectorInterleave(
I, 7);
8511 case Intrinsic::vector_interleave8:
8512 visitVectorInterleave(
I, 8);
8514 case Intrinsic::vector_deinterleave2:
8515 visitVectorDeinterleave(
I, 2);
8517 case Intrinsic::vector_deinterleave3:
8518 visitVectorDeinterleave(
I, 3);
8520 case Intrinsic::vector_deinterleave4:
8521 visitVectorDeinterleave(
I, 4);
8523 case Intrinsic::vector_deinterleave5:
8524 visitVectorDeinterleave(
I, 5);
8526 case Intrinsic::vector_deinterleave6:
8527 visitVectorDeinterleave(
I, 6);
8529 case Intrinsic::vector_deinterleave7:
8530 visitVectorDeinterleave(
I, 7);
8532 case Intrinsic::vector_deinterleave8:
8533 visitVectorDeinterleave(
I, 8);
8535 case Intrinsic::experimental_vector_compress:
8537 getValue(
I.getArgOperand(0)).getValueType(),
8542 case Intrinsic::experimental_convergence_anchor:
8543 case Intrinsic::experimental_convergence_entry:
8544 case Intrinsic::experimental_convergence_loop:
8545 visitConvergenceControl(
I, Intrinsic);
8547 case Intrinsic::experimental_vector_histogram_add: {
8548 visitVectorHistogram(
I, Intrinsic);
8551 case Intrinsic::experimental_vector_extract_last_active: {
8552 visitVectorExtractLastActive(
I, Intrinsic);
8555 case Intrinsic::loop_dependence_war_mask:
8560 DAG.getConstant(0, sdl, MVT::i64)));
8562 case Intrinsic::loop_dependence_raw_mask:
8567 DAG.getConstant(0, sdl, MVT::i64)));
8569 case Intrinsic::masked_udiv:
8575 case Intrinsic::masked_sdiv:
8581 case Intrinsic::masked_urem:
8587 case Intrinsic::masked_srem:
8596void SelectionDAGBuilder::pushFPOpOutChain(
SDValue Result,
8612 PendingConstrainedFP.push_back(OutChain);
8615 PendingConstrainedFPStrict.push_back(OutChain);
8620void SelectionDAGBuilder::visitConstrainedFPIntrinsic(
8634 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8636 SDVTList VTs =
DAG.getVTList(VT, MVT::Other);
8640 Flags.setNoFPExcept(
true);
8643 Flags.copyFMF(*FPOp);
8648#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
8649 case Intrinsic::INTRINSIC: \
8650 Opcode = ISD::STRICT_##DAGN; \
8652#include "llvm/IR/ConstrainedOps.def"
8653 case Intrinsic::experimental_constrained_fmuladd: {
8660 pushFPOpOutChain(
Mul, EB);
8683 if (
DAG.isKnownNeverNaN(Opers[1]) &&
DAG.isKnownNeverNaN(Opers[2]))
8691 pushFPOpOutChain(Result, EB);
8698 std::optional<unsigned> ResOPC;
8700 case Intrinsic::vp_ctlz: {
8702 ResOPC = IsZeroUndef ? ISD::VP_CTLZ_ZERO_POISON : ISD::VP_CTLZ;
8705 case Intrinsic::vp_cttz: {
8707 ResOPC = IsZeroUndef ? ISD::VP_CTTZ_ZERO_POISON : ISD::VP_CTTZ;
8710 case Intrinsic::vp_cttz_elts: {
8712 ResOPC = IsZeroPoison ? ISD::VP_CTTZ_ELTS_ZERO_POISON : ISD::VP_CTTZ_ELTS;
8715#define HELPER_MAP_VPID_TO_VPSD(VPID, VPSD) \
8716 case Intrinsic::VPID: \
8717 ResOPC = ISD::VPSD; \
8719#include "llvm/IR/VPIntrinsics.def"
8724 "Inconsistency: no SDNode available for this VPIntrinsic!");
8726 if (*ResOPC == ISD::VP_REDUCE_SEQ_FADD ||
8727 *ResOPC == ISD::VP_REDUCE_SEQ_FMUL) {
8729 return *ResOPC == ISD::VP_REDUCE_SEQ_FADD ? ISD::VP_REDUCE_FADD
8730 : ISD::VP_REDUCE_FMUL;
8736void SelectionDAGBuilder::visitVPLoad(
8748 Alignment =
DAG.getEVTAlign(VT);
8751 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8752 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8755 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8756 MachinePointerInfo(PtrOperand), MMOFlags,
8758 LD =
DAG.getLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8765void SelectionDAGBuilder::visitVPLoadFF(
8768 assert(OpValues.
size() == 3 &&
"Unexpected number of operands");
8778 Alignment =
DAG.getEVTAlign(VT);
8781 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8782 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8785 LD =
DAG.getLoadFFVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8790 setValue(&VPIntrin,
DAG.getMergeValues({LD.getValue(0), Trunc},
DL));
8793void SelectionDAGBuilder::visitVPGather(
8797 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8809 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8811 *Alignment, AAInfo, Ranges);
8821 EVT IdxVT =
Index.getValueType();
8827 LD =
DAG.getGatherVP(
8828 DAG.getVTList(VT, MVT::Other), VT,
DL,
8829 {DAG.getRoot(), Base, Index, Scale, OpValues[1], OpValues[2]}, MMO,
8835void SelectionDAGBuilder::visitVPStore(
8839 EVT VT = OpValues[0].getValueType();
8844 Alignment =
DAG.getEVTAlign(VT);
8847 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8850 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8851 MachinePointerInfo(PtrOperand), MMOFlags,
8860void SelectionDAGBuilder::visitVPScatter(
8863 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8865 EVT VT = OpValues[0].getValueType();
8875 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8877 *Alignment, AAInfo);
8887 EVT IdxVT =
Index.getValueType();
8893 ST =
DAG.getScatterVP(
DAG.getVTList(MVT::Other), VT,
DL,
8894 {getMemoryRoot(), OpValues[0], Base, Index, Scale,
8895 OpValues[2], OpValues[3]},
8901void SelectionDAGBuilder::visitVPStridedLoad(
8913 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8915 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8918 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8920 *Alignment, AAInfo, Ranges);
8922 SDValue LD =
DAG.getStridedLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1],
8923 OpValues[2], OpValues[3], MMO,
8931void SelectionDAGBuilder::visitVPStridedStore(
8935 EVT VT = OpValues[0].getValueType();
8941 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8944 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8946 *Alignment, AAInfo);
8950 DAG.getUNDEF(OpValues[1].getValueType()), OpValues[2], OpValues[3],
8958void SelectionDAGBuilder::visitVPCmp(
const VPCmpIntrinsic &VPIntrin) {
8959 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8972 "Unexpected target EVL type");
8977 SimplifyQuery SQ(
DAG.getDataLayout(), &VPIntrin);
8984 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
8987 Condition, MaskOp, EVL));
8990void SelectionDAGBuilder::visitVectorPredicationIntrinsic(
8998 return visitVPCmp(*CmpI);
9001 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9003 SDVTList VTs =
DAG.getVTList(ValueVTs);
9009 "Unexpected target EVL type");
9013 for (
unsigned I = 0;
I < VPIntrin.
arg_size(); ++
I) {
9015 if (
I == EVLParamPos)
9022 SDNodeFlags SDFlags;
9030 visitVPLoad(VPIntrin, ValueVTs[0], OpValues);
9032 case ISD::VP_LOAD_FF:
9033 visitVPLoadFF(VPIntrin, ValueVTs[0], ValueVTs[1], OpValues);
9035 case ISD::VP_GATHER:
9036 visitVPGather(VPIntrin, ValueVTs[0], OpValues);
9038 case ISD::EXPERIMENTAL_VP_STRIDED_LOAD:
9039 visitVPStridedLoad(VPIntrin, ValueVTs[0], OpValues);
9042 visitVPStore(VPIntrin, OpValues);
9044 case ISD::VP_SCATTER:
9045 visitVPScatter(VPIntrin, OpValues);
9047 case ISD::EXPERIMENTAL_VP_STRIDED_STORE:
9048 visitVPStridedStore(VPIntrin, OpValues);
9050 case ISD::VP_FMULADD: {
9051 assert(OpValues.
size() == 5 &&
"Unexpected number of operands");
9052 SDNodeFlags SDFlags;
9057 setValue(&VPIntrin,
DAG.getNode(ISD::VP_FMA,
DL, VTs, OpValues, SDFlags));
9060 ISD::VP_FMUL,
DL, VTs,
9061 {OpValues[0], OpValues[1], OpValues[3], OpValues[4]}, SDFlags);
9063 DAG.getNode(ISD::VP_FADD,
DL, VTs,
9064 {
Mul, OpValues[2], OpValues[3], OpValues[4]}, SDFlags);
9069 case ISD::VP_IS_FPCLASS: {
9070 const DataLayout DLayout =
DAG.getDataLayout();
9072 auto Constant = OpValues[1]->getAsZExtVal();
9075 {OpValues[0],
Check, OpValues[2], OpValues[3]});
9079 case ISD::VP_INTTOPTR: {
9090 case ISD::VP_PTRTOINT: {
9092 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9105 case ISD::VP_CTLZ_ZERO_POISON:
9107 case ISD::VP_CTTZ_ZERO_POISON:
9108 case ISD::VP_CTTZ_ELTS_ZERO_POISON:
9109 case ISD::VP_CTTZ_ELTS: {
9111 DAG.getNode(Opcode,
DL, VTs, {OpValues[0], OpValues[2], OpValues[3]});
9129 unsigned CallSiteIndex =
FuncInfo.getCurrentCallSite();
9130 if (CallSiteIndex) {
9144 assert(BeginLabel &&
"BeginLabel should've been set");
9158 assert(
II &&
"II should've been set");
9169std::pair<SDValue, SDValue>
9183 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
9186 "Non-null chain expected with non-tail call!");
9187 assert((Result.second.getNode() || !Result.first.getNode()) &&
9188 "Null value expected with tail call!");
9190 if (!Result.second.getNode()) {
9197 PendingExports.clear();
9199 DAG.setRoot(Result.second);
9217 if (!isMustTailCall &&
9218 Caller->getFnAttribute(
"disable-tail-calls").getValueAsBool())
9224 if (
DAG.getTargetLoweringInfo().supportSwiftError() &&
9225 Caller->getAttributes().hasAttrSomewhere(Attribute::SwiftError))
9234 bool isTailCall,
bool isMustTailCall,
9237 auto &
DL =
DAG.getDataLayout();
9244 const Value *SwiftErrorVal =
nullptr;
9251 const Value *V = *
I;
9254 if (V->getType()->isEmptyTy())
9259 Entry.setAttributes(&CB,
I - CB.
arg_begin());
9271 Args.push_back(Entry);
9282 Value *V = Bundle->Inputs[0];
9284 Entry.IsCFGuardTarget =
true;
9285 Args.push_back(Entry);
9298 "Target doesn't support calls with kcfi operand bundles.");
9306 auto *Token = Bundle->Inputs[0].get();
9307 ConvControlToken =
getValue(Token);
9318 .
setCallee(RetTy, FTy, Callee, std::move(Args), CB)
9331 "This target doesn't support calls with ptrauth operand bundles.");
9335 std::pair<SDValue, SDValue> Result =
lowerInvokable(CLI, EHPadBB);
9337 if (Result.first.getNode()) {
9352 DAG.setRoot(CopyNode);
9368 LoadTy, Builder.DAG.getDataLayout()))
9369 return Builder.getValue(LoadCst);
9375 bool ConstantMemory =
false;
9378 if (Builder.BatchAA && Builder.BatchAA->pointsToConstantMemory(PtrVal)) {
9379 Root = Builder.DAG.getEntryNode();
9380 ConstantMemory =
true;
9383 Root = Builder.DAG.getRoot();
9388 Builder.DAG.getLoad(LoadVT, Builder.getCurSDLoc(), Root, Ptr,
9391 if (!ConstantMemory)
9392 Builder.PendingLoads.push_back(LoadVal.
getValue(1));
9398void SelectionDAGBuilder::processIntegerCallValue(
const Instruction &
I,
9401 EVT VT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9412bool SelectionDAGBuilder::visitMemCmpBCmpCall(
const CallInst &
I) {
9413 const Value *
LHS =
I.getArgOperand(0), *
RHS =
I.getArgOperand(1);
9414 const Value *
Size =
I.getArgOperand(2);
9417 EVT CallVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9423 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9427 if (Res.first.getNode()) {
9428 processIntegerCallValue(
I, Res.first,
true);
9442 auto hasFastLoadsAndCompare = [&](
unsigned NumBits) {
9443 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9465 switch (NumBitsToCompare) {
9477 LoadVT = hasFastLoadsAndCompare(NumBitsToCompare);
9490 LoadL =
DAG.getBitcast(CmpVT, LoadL);
9491 LoadR =
DAG.getBitcast(CmpVT, LoadR);
9495 processIntegerCallValue(
I, Cmp,
false);
9504bool SelectionDAGBuilder::visitMemChrCall(
const CallInst &
I) {
9505 const Value *Src =
I.getArgOperand(0);
9506 const Value *
Char =
I.getArgOperand(1);
9507 const Value *
Length =
I.getArgOperand(2);
9509 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9510 std::pair<SDValue, SDValue> Res =
9513 MachinePointerInfo(Src));
9514 if (Res.first.getNode()) {
9528bool SelectionDAGBuilder::visitMemCCpyCall(
const CallInst &
I) {
9529 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9536 processIntegerCallValue(
I, Res.first,
true);
9548bool SelectionDAGBuilder::visitMemPCpyCall(
const CallInst &
I) {
9553 Align DstAlign =
DAG.InferPtrAlign(Dst).valueOrOne();
9554 Align SrcAlign =
DAG.InferPtrAlign(Src).valueOrOne();
9563 Root, sdl, Dst, Src,
Size, DstAlign, SrcAlign,
false,
false,
9564 nullptr, std::nullopt, MachinePointerInfo(
I.getArgOperand(0)),
9565 MachinePointerInfo(
I.getArgOperand(1)),
I.getAAMetadata());
9567 "** memcpy should not be lowered as TailCall in mempcpy context **");
9571 Size =
DAG.getSExtOrTrunc(
Size, sdl, Dst.getValueType());
9584bool SelectionDAGBuilder::visitStrCpyCall(
const CallInst &
I,
bool isStpcpy) {
9585 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9587 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9590 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), isStpcpy, &
I);
9591 if (Res.first.getNode()) {
9593 DAG.setRoot(Res.second);
9605bool SelectionDAGBuilder::visitStrCmpCall(
const CallInst &
I) {
9606 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9608 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9611 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), &
I);
9612 if (Res.first.getNode()) {
9613 processIntegerCallValue(
I, Res.first,
true);
9626bool SelectionDAGBuilder::visitStrLenCall(
const CallInst &
I) {
9627 const Value *Arg0 =
I.getArgOperand(0);
9629 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9632 if (Res.first.getNode()) {
9633 processIntegerCallValue(
I, Res.first,
false);
9646bool SelectionDAGBuilder::visitStrNLenCall(
const CallInst &
I) {
9647 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9649 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9650 std::pair<SDValue, SDValue> Res =
9653 MachinePointerInfo(Arg0));
9654 if (Res.first.getNode()) {
9655 processIntegerCallValue(
I, Res.first,
false);
9668bool SelectionDAGBuilder::visitStrstrCall(
const CallInst &
I) {
9669 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9670 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9674 processIntegerCallValue(
I, Res.first,
false);
9686bool SelectionDAGBuilder::visitUnaryFloatCall(
const CallInst &
I,
9691 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9708bool SelectionDAGBuilder::visitBinaryFloatCall(
const CallInst &
I,
9713 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9726void SelectionDAGBuilder::visitCall(
const CallInst &
I) {
9728 if (
I.isInlineAsm()) {
9735 if (Function *
F =
I.getCalledFunction()) {
9736 if (
F->isDeclaration()) {
9738 if (
unsigned IID =
F->getIntrinsicID()) {
9739 visitIntrinsicCall(
I, IID);
9750 if (!
I.isNoBuiltin() && !
F->hasLocalLinkage() &&
F->hasName() &&
9751 LibInfo->getLibFunc(*
F, Func) &&
LibInfo->hasOptimizedCodeGen(Func)) {
9755 if (visitMemCmpBCmpCall(
I))
9758 case LibFunc_copysign:
9759 case LibFunc_copysignf:
9760 case LibFunc_copysignl:
9763 if (
I.onlyReadsMemory()) {
9808 case LibFunc_atan2f:
9809 case LibFunc_atan2l:
9834 case LibFunc_sqrt_finite:
9835 case LibFunc_sqrtf_finite:
9836 case LibFunc_sqrtl_finite:
9853 case LibFunc_exp10f:
9854 case LibFunc_exp10l:
9859 case LibFunc_ldexpf:
9860 case LibFunc_ldexpl:
9864 case LibFunc_strstr:
9865 if (visitStrstrCall(
I))
9868 case LibFunc_memcmp:
9869 if (visitMemCmpBCmpCall(
I))
9872 case LibFunc_memccpy:
9873 if (visitMemCCpyCall(
I))
9876 case LibFunc_mempcpy:
9877 if (visitMemPCpyCall(
I))
9880 case LibFunc_memchr:
9881 if (visitMemChrCall(
I))
9884 case LibFunc_strcpy:
9885 if (visitStrCpyCall(
I,
false))
9888 case LibFunc_stpcpy:
9889 if (visitStrCpyCall(
I,
true))
9892 case LibFunc_strcmp:
9893 if (visitStrCmpCall(
I))
9896 case LibFunc_strlen:
9897 if (visitStrLenCall(
I))
9900 case LibFunc_strnlen:
9901 if (visitStrNLenCall(
I))
9925 if (
I.hasDeoptState())
9942 const Value *Discriminator = PAB->Inputs[1];
9944 assert(
Key->getType()->isIntegerTy(32) &&
"Invalid ptrauth key");
9945 assert(Discriminator->getType()->isIntegerTy(64) &&
9946 "Invalid ptrauth discriminator");
9951 if (CalleeCPA->isKnownCompatibleWith(
Key, Discriminator,
9952 DAG.getDataLayout()))
9992 for (
const auto &Code : Codes)
10007 SDISelAsmOperandInfo &MatchingOpInfo,
10009 if (OpInfo.ConstraintVT == MatchingOpInfo.ConstraintVT)
10015 std::pair<unsigned, const TargetRegisterClass *> MatchRC =
10017 OpInfo.ConstraintVT);
10018 std::pair<unsigned, const TargetRegisterClass *> InputRC =
10020 MatchingOpInfo.ConstraintVT);
10021 const bool OutOpIsIntOrFP =
10022 OpInfo.ConstraintVT.isInteger() || OpInfo.ConstraintVT.isFloatingPoint();
10023 const bool InOpIsIntOrFP = MatchingOpInfo.ConstraintVT.isInteger() ||
10024 MatchingOpInfo.ConstraintVT.isFloatingPoint();
10025 if ((OutOpIsIntOrFP != InOpIsIntOrFP) || (MatchRC.second != InputRC.second)) {
10028 " with a matching output constraint of"
10029 " incompatible type!");
10031 MatchingOpInfo.ConstraintVT = OpInfo.ConstraintVT;
10038 SDISelAsmOperandInfo &OpInfo,
10051 const Value *OpVal = OpInfo.CallOperandVal;
10069 DL.getPrefTypeAlign(Ty),
false,
10072 Chain = DAG.
getTruncStore(Chain, Location, OpInfo.CallOperand, StackSlot,
10075 OpInfo.CallOperand = StackSlot;
10088static std::optional<unsigned>
10090 SDISelAsmOperandInfo &OpInfo,
10091 SDISelAsmOperandInfo &RefOpInfo) {
10102 return std::nullopt;
10106 unsigned AssignedReg;
10109 &
TRI, RefOpInfo.ConstraintCode, RefOpInfo.ConstraintVT);
10112 return std::nullopt;
10117 const MVT RegVT = *
TRI.legalclasstypes_begin(*RC);
10119 if (OpInfo.ConstraintVT != MVT::Other && RegVT != MVT::Untyped) {
10128 !
TRI.isTypeLegalForClass(*RC, OpInfo.ConstraintVT)) {
10133 if (RegVT.
getSizeInBits() == OpInfo.ConstraintVT.getSizeInBits()) {
10138 OpInfo.CallOperand =
10140 OpInfo.ConstraintVT = RegVT;
10144 }
else if (RegVT.
isInteger() && OpInfo.ConstraintVT.isFloatingPoint()) {
10147 OpInfo.CallOperand =
10149 OpInfo.ConstraintVT = VT;
10156 if (OpInfo.isMatchingInputConstraint())
10157 return std::nullopt;
10159 EVT ValueVT = OpInfo.ConstraintVT;
10160 if (OpInfo.ConstraintVT == MVT::Other)
10164 unsigned NumRegs = 1;
10165 if (OpInfo.ConstraintVT != MVT::Other)
10180 I = std::find(
I, RC->
end(), AssignedReg);
10181 if (
I == RC->
end()) {
10184 return {AssignedReg};
10188 for (; NumRegs; --NumRegs, ++
I) {
10189 assert(
I != RC->
end() &&
"Ran out of registers to allocate!");
10194 OpInfo.AssignedRegs =
RegsForValue(Regs, RegVT, ValueVT);
10195 return std::nullopt;
10200 const std::vector<SDValue> &AsmNodeOperands) {
10203 for (; OperandNo; --OperandNo) {
10205 unsigned OpFlag = AsmNodeOperands[CurOp]->getAsZExtVal();
10208 (
F.isRegDefKind() ||
F.isRegDefEarlyClobberKind() ||
F.isMemKind()) &&
10209 "Skipped past definitions?");
10210 CurOp +=
F.getNumOperandRegisters() + 1;
10218 unsigned Flags = 0;
10221 explicit ExtraFlags(
const CallBase &
Call) {
10223 if (
IA->hasSideEffects())
10225 if (
IA->isAlignStack())
10227 if (
IA->canThrow())
10234 void update(
const TargetLowering::AsmOperandInfo &OpInfo) {
10250 unsigned get()
const {
return Flags; }
10274struct ConstraintDecisionInfo {
10276 std::vector<SDValue> AsmNodeOperands;
10278 bool HasSideEffect =
false;
10281 SmallVector<char> Buffer;
10282 raw_svector_ostream ErrorMsg;
10284 ConstraintDecisionInfo() : ErrorMsg(Buffer) {}
10294 ExtraFlags &ExtraInfo) {
10295 for (
auto &
T : TargetConstraints) {
10296 Info.ConstraintOperands.push_back(SDISelAsmOperandInfo(
T));
10297 SDISelAsmOperandInfo &OpInfo = Info.ConstraintOperands.back();
10299 if (OpInfo.CallOperandVal)
10300 OpInfo.CallOperand = Builder.getValue(OpInfo.CallOperandVal);
10302 if (!Info.HasSideEffect)
10303 Info.HasSideEffect = OpInfo.hasMemory(TLI);
10315 Info.ErrorMsg <<
"constraint '" <<
T.ConstraintCode
10316 <<
"' expects an integer constant expression";
10320 ExtraInfo.update(
T);
10334 IA->collectAsmStrs(AsmStrs);
10337 for (SDISelAsmOperandInfo &OpInfo : Info.ConstraintOperands) {
10345 if (OpInfo.hasMatchingInput()) {
10346 SDISelAsmOperandInfo &
Input =
10347 Info.ConstraintOperands[OpInfo.MatchingInput];
10378 if (OpInfo.isIndirect &&
isFunction(OpInfo.CallOperand) &&
10381 OpInfo.isIndirect =
false;
10388 !OpInfo.isIndirect) {
10389 assert((OpInfo.isMultipleAlternative ||
10391 "Can only indirectify direct input operands!");
10398 OpInfo.CallOperandVal =
nullptr;
10401 OpInfo.isIndirect =
true;
10413 SDLoc DL = Builder.getCurSDLoc();
10414 for (SDISelAsmOperandInfo &OpInfo : Info.ConstraintOperands) {
10416 SDISelAsmOperandInfo &RefOpInfo =
10417 OpInfo.isMatchingInputConstraint()
10418 ? Info.ConstraintOperands[OpInfo.getMatchedOperand()]
10424 const char *
RegName =
TRI.getName(*RegError);
10425 Info.ErrorMsg <<
"register '" <<
RegName <<
"' allocated for constraint '"
10426 << OpInfo.ConstraintCode
10427 <<
"' does not match required type";
10431 auto DetectWriteToReservedRegister = [&]() {
10436 if (
Reg.isPhysical() &&
TRI.isInlineAsmReadOnlyReg(MF,
Reg)) {
10437 Info.ErrorMsg <<
"write to reserved register '"
10438 <<
TRI.getRegAsmName(
Reg) <<
"'";
10447 !OpInfo.isMatchingInputConstraint())) &&
10448 "Only address as input operand is allowed.");
10450 switch (OpInfo.Type) {
10456 "Failed to convert memory constraint code to constraint id.");
10461 Info.AsmNodeOperands.push_back(
10463 Info.AsmNodeOperands.push_back(OpInfo.CallOperand);
10468 if (OpInfo.AssignedRegs.Regs.empty()) {
10469 Info.ErrorMsg <<
"could not allocate output register for "
10470 <<
"constraint '" << OpInfo.ConstraintCode <<
"'";
10474 if (DetectWriteToReservedRegister())
10479 OpInfo.AssignedRegs.AddInlineAsmOperands(
10482 false, 0,
DL, DAG, Info.AsmNodeOperands);
10488 SDValue InOperandVal = OpInfo.CallOperand;
10490 if (OpInfo.isMatchingInputConstraint()) {
10494 Info.AsmNodeOperands);
10496 if (Flag.isRegDefKind() || Flag.isRegDefEarlyClobberKind()) {
10497 if (OpInfo.isIndirect) {
10499 Info.ErrorMsg <<
"inline asm not supported yet: cannot handle "
10500 <<
"tied indirect register inputs";
10510 MVT RegVT = R->getSimpleValueType(0);
10514 :
TRI.getMinimalPhysRegClass(TiedReg);
10515 for (
unsigned I = 0,
E = Flag.getNumOperandRegisters();
I !=
E; ++
I)
10522 &Info.Glue, &
Call);
10524 OpInfo.getMatchedOperand(),
DL, DAG,
10525 Info.AsmNodeOperands);
10529 assert(Flag.isMemKind() &&
"Unknown matching constraint!");
10530 assert(Flag.getNumOperandRegisters() == 1 &&
10531 "Unexpected number of operands");
10535 Flag.clearMemConstraint();
10536 Flag.setMatchingOp(OpInfo.getMatchedOperand());
10539 Info.AsmNodeOperands.push_back(Info.AsmNodeOperands[CurOp + 1]);
10550 std::vector<SDValue>
Ops;
10556 Info.ErrorMsg <<
"value out of range for constraint '"
10557 << OpInfo.ConstraintCode <<
"'";
10561 Info.ErrorMsg <<
"invalid operand for inline asm constraint '"
10562 << OpInfo.ConstraintCode <<
"'";
10575 assert((OpInfo.isIndirect ||
10577 "Operand must be indirect to be a mem!");
10580 "Memory operands expect pointer values");
10585 "Failed to convert memory constraint code to constraint id.");
10590 Info.AsmNodeOperands.push_back(
10592 Info.AsmNodeOperands.push_back(InOperandVal);
10600 "Failed to convert memory constraint code to constraint id.");
10604 SDValue AsmOp = InOperandVal;
10616 Info.AsmNodeOperands.push_back(
10618 Info.AsmNodeOperands.push_back(AsmOp);
10624 Info.ErrorMsg <<
"unknown asm constraint '" << OpInfo.ConstraintCode
10630 if (OpInfo.isIndirect) {
10631 Info.ErrorMsg <<
"cannot handle indirect register inputs yet for "
10632 <<
"constraint '" << OpInfo.ConstraintCode <<
"'";
10637 if (OpInfo.AssignedRegs.Regs.empty()) {
10638 Info.ErrorMsg <<
"could not allocate input reg for constraint '"
10639 << OpInfo.ConstraintCode <<
"'";
10643 if (DetectWriteToReservedRegister())
10646 OpInfo.AssignedRegs.getCopyToRegs(InOperandVal, DAG,
DL, Info.Chain,
10647 &Info.Glue, &
Call);
10648 OpInfo.AssignedRegs.AddInlineAsmOperands(
10656 if (!OpInfo.AssignedRegs.Regs.empty())
10657 OpInfo.AssignedRegs.AddInlineAsmOperands(
10674 ExtraFlags ExtraInfo(
Call);
10677 Info.HasSideEffect = IA->hasSideEffects();
10683 Info.Chain = Info.HasSideEffect ? Builder.getRoot() : DAG.
getRoot();
10687 if (IsCallBr || EmitEHLabels)
10691 Info.Chain = Builder.getControlRoot();
10694 Info.Chain = Builder.lowerStartEH(Info.Chain, EHPadBB, Info.BeginLabel);
10700 Info.AsmNodeOperands.push_back(
SDValue());
10707 const MDNode *SrcLoc =
Call.getMetadata(
"srcloc");
10708 Info.AsmNodeOperands.push_back(DAG.
getMDNode(SrcLoc));
10712 Info.AsmNodeOperands.push_back(
10721void SelectionDAGBuilder::visitInlineAsm(
const CallBase &
Call,
10723 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10725 DAG.getDataLayout(),
DAG.getSubtarget().getRegisterInfo(),
Call);
10728 "InvokeInst must have an EHPadBB");
10730 ConstraintDecisionInfo
Info;
10733 return emitInlineAsmError(
Call,
Info.ErrorMsg.str());
10741 Info.AsmNodeOperands.push_back(Glue);
10747 Info.AsmNodeOperands);
10759 ResultTypes = StructResult->elements();
10760 else if (!CallResultType->
isVoidTy())
10761 ResultTypes =
ArrayRef(CallResultType);
10763 auto CurResultType = ResultTypes.
begin();
10764 auto handleRegAssign = [&](
SDValue V) {
10765 assert(CurResultType != ResultTypes.
end() &&
"Unexpected value");
10766 assert((*CurResultType)->isSized() &&
"Unexpected unsized type");
10767 EVT ResultVT = TLI.
getValueType(
DAG.getDataLayout(), *CurResultType);
10779 if (ResultVT !=
V.getValueType() &&
10782 else if (ResultVT !=
V.getValueType() && ResultVT.
isInteger() &&
10783 V.getValueType().isInteger()) {
10789 assert(ResultVT ==
V.getValueType() &&
"Asm result value mismatch!");
10795 for (SDISelAsmOperandInfo &OpInfo :
Info.ConstraintOperands) {
10799 if (OpInfo.AssignedRegs.
Regs.empty())
10802 switch (OpInfo.ConstraintType) {
10806 Chain, &Glue, &
Call);
10818 assert(
false &&
"Unexpected unknown constraint");
10822 if (OpInfo.isIndirect) {
10823 const Value *Ptr = OpInfo.CallOperandVal;
10824 assert(Ptr &&
"Expected value CallOperandVal for indirect asm operand");
10826 MachinePointerInfo(Ptr));
10833 handleRegAssign(V);
10835 handleRegAssign(Val);
10841 if (!ResultValues.
empty()) {
10842 assert(CurResultType == ResultTypes.
end() &&
10843 "Mismatch in number of ResultTypes");
10845 "Mismatch in number of output operands in asm result");
10848 DAG.getVTList(ResultVTs), ResultValues);
10853 if (!OutChains.
empty())
10857 Chain = lowerEndEH(Chain,
II, EHPadBB,
Info.BeginLabel);
10860 if (ResultValues.
empty() ||
Info.HasSideEffect || !OutChains.
empty() ||
10862 DAG.setRoot(Chain);
10865void SelectionDAGBuilder::emitInlineAsmError(
const CallBase &
Call,
10866 const Twine &Message) {
10867 LLVMContext &Ctx = *
DAG.getContext();
10871 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10875 if (ValueVTs.
empty())
10879 for (
const EVT &VT : ValueVTs)
10880 Ops.push_back(
DAG.getUNDEF(VT));
10885void SelectionDAGBuilder::visitVAStart(
const CallInst &
I) {
10889 DAG.getSrcValue(
I.getArgOperand(0))));
10892void SelectionDAGBuilder::visitVAArg(
const VAArgInst &
I) {
10893 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10894 const DataLayout &
DL =
DAG.getDataLayout();
10898 DL.getABITypeAlign(
I.getType()).value());
10899 DAG.setRoot(
V.getValue(1));
10901 if (
I.getType()->isPointerTy())
10902 V =
DAG.getPtrExtOrTrunc(
10907void SelectionDAGBuilder::visitVAEnd(
const CallInst &
I) {
10911 DAG.getSrcValue(
I.getArgOperand(0))));
10914void SelectionDAGBuilder::visitVACopy(
const CallInst &
I) {
10919 DAG.getSrcValue(
I.getArgOperand(0)),
10920 DAG.getSrcValue(
I.getArgOperand(1))));
10926 std::optional<ConstantRange> CR =
getRange(
I);
10928 if (!CR || CR->isFullSet() || CR->isEmptySet() || CR->isUpperWrapped())
10931 APInt Hi = CR->getUnsignedMax();
10932 unsigned Bits = std::max(
Hi.getActiveBits(),
10940 DAG.getValueType(SmallVT));
10941 unsigned NumVals =
Op.getNode()->getNumValues();
10947 Ops.push_back(ZExt);
10948 for (
unsigned I = 1;
I != NumVals; ++
I)
10949 Ops.push_back(
Op.getValue(
I));
10951 return DAG.getMergeValues(
Ops,
SL);
10961 SDValue TestConst =
DAG.getTargetConstant(Classes,
SDLoc(), MVT::i32);
10969 for (
unsigned I = 0, E =
Ops.size();
I != E; ++
I) {
10972 MergeOp, TestConst);
10975 return DAG.getMergeValues(
Ops,
SL);
10986 unsigned ArgIdx,
unsigned NumArgs,
SDValue Callee,
Type *ReturnTy,
10989 Args.reserve(NumArgs);
10993 for (
unsigned ArgI = ArgIdx, ArgE = ArgIdx + NumArgs;
10994 ArgI != ArgE; ++ArgI) {
10995 const Value *V =
Call->getOperand(ArgI);
10997 assert(!V->getType()->isEmptyTy() &&
"Empty type passed to intrinsic.");
11000 Entry.setAttributes(
Call, ArgI);
11001 Args.push_back(Entry);
11006 .
setCallee(
Call->getCallingConv(), ReturnTy, Callee, std::move(Args),
11035 for (
unsigned I = StartIdx;
I <
Call.arg_size();
I++) {
11044 Ops.push_back(Builder.getValue(
Call.getArgOperand(
I)));
11050void SelectionDAGBuilder::visitStackmap(
const CallInst &CI) {
11076 Ops.push_back(Chain);
11077 Ops.push_back(InGlue);
11084 assert(
ID.getValueType() == MVT::i64);
11086 DAG.getTargetConstant(
ID->getAsZExtVal(),
DL,
ID.getValueType());
11087 Ops.push_back(IDConst);
11093 Ops.push_back(ShadConst);
11099 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
11103 Chain =
DAG.getCALLSEQ_END(Chain, 0, 0, InGlue,
DL);
11108 DAG.setRoot(Chain);
11111 FuncInfo.MF->getFrameInfo().setHasStackMap();
11115void SelectionDAGBuilder::visitPatchpoint(
const CallBase &CB,
11132 Callee =
DAG.getIntPtrConstant(ConstCallee->getZExtValue(), dl,
11135 Callee =
DAG.getTargetGlobalAddress(SymbolicCallee->getGlobal(),
11136 SDLoc(SymbolicCallee),
11137 SymbolicCallee->getValueType(0));
11147 "Not enough arguments provided to the patchpoint intrinsic");
11150 unsigned NumCallArgs = IsAnyRegCC ? 0 : NumArgs;
11154 TargetLowering::CallLoweringInfo CLI(
DAG);
11159 SDNode *CallEnd =
Result.second.getNode();
11168 "Expected a callseq node.");
11170 bool HasGlue =
Call->getGluedNode();
11195 Ops.push_back(Callee);
11201 NumCallRegArgs = IsAnyRegCC ? NumArgs : NumCallRegArgs;
11202 Ops.push_back(
DAG.getTargetConstant(NumCallRegArgs, dl, MVT::i32));
11205 Ops.push_back(
DAG.getTargetConstant((
unsigned)CC, dl, MVT::i32));
11210 for (
unsigned i = NumMetaOpers, e = NumMetaOpers + NumArgs; i !=
e; ++i)
11221 if (IsAnyRegCC && HasDef) {
11223 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11226 assert(ValueVTs.
size() == 1 &&
"Expected only one return value type.");
11231 NodeTys =
DAG.getVTList(ValueVTs);
11233 NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
11250 if (IsAnyRegCC && HasDef) {
11253 DAG.ReplaceAllUsesOfValuesWith(From, To, 2);
11259 FuncInfo.MF->getFrameInfo().setHasPatchPoint();
11262void SelectionDAGBuilder::visitVectorReduce(
const CallInst &
I,
11264 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11267 if (
I.arg_size() > 1)
11272 SDNodeFlags SDFlags;
11276 switch (Intrinsic) {
11277 case Intrinsic::vector_reduce_fadd:
11285 case Intrinsic::vector_reduce_fmul:
11293 case Intrinsic::vector_reduce_add:
11296 case Intrinsic::vector_reduce_mul:
11299 case Intrinsic::vector_reduce_and:
11302 case Intrinsic::vector_reduce_or:
11305 case Intrinsic::vector_reduce_xor:
11308 case Intrinsic::vector_reduce_smax:
11311 case Intrinsic::vector_reduce_smin:
11314 case Intrinsic::vector_reduce_umax:
11317 case Intrinsic::vector_reduce_umin:
11320 case Intrinsic::vector_reduce_fmax:
11323 case Intrinsic::vector_reduce_fmin:
11326 case Intrinsic::vector_reduce_fmaximum:
11329 case Intrinsic::vector_reduce_fminimum:
11343 Attrs.push_back(Attribute::SExt);
11345 Attrs.push_back(Attribute::ZExt);
11347 Attrs.push_back(Attribute::InReg);
11349 return AttributeList::get(CLI.
RetTy->
getContext(), AttributeList::ReturnIndex,
11357std::pair<SDValue, SDValue>
11371 "Only supported for non-aggregate returns");
11374 for (
Type *Ty : RetOrigTys)
11383 RetOrigTys.
swap(OldRetOrigTys);
11384 RetVTs.
swap(OldRetVTs);
11385 Offsets.swap(OldOffsets);
11387 for (
size_t i = 0, e = OldRetVTs.
size(); i != e; ++i) {
11388 EVT RetVT = OldRetVTs[i];
11392 unsigned RegisterVTByteSZ = RegisterVT.
getSizeInBits() / 8;
11393 RetOrigTys.
append(NumRegs, OldRetOrigTys[i]);
11394 RetVTs.
append(NumRegs, RegisterVT);
11395 for (
unsigned j = 0; j != NumRegs; ++j)
11408 int DemoteStackIdx = -100;
11421 ArgListEntry Entry(DemoteStackSlot, StackSlotPtrType);
11422 Entry.IsSRet =
true;
11423 Entry.Alignment = Alignment;
11435 for (
unsigned I = 0, E = RetVTs.
size();
I != E; ++
I) {
11437 if (NeedsRegBlock) {
11438 Flags.setInConsecutiveRegs();
11439 if (
I == RetVTs.
size() - 1)
11440 Flags.setInConsecutiveRegsLast();
11442 EVT VT = RetVTs[
I];
11446 for (
unsigned i = 0; i != NumRegs; ++i) {
11460 CLI.
Ins.push_back(Ret);
11469 if (Arg.IsSwiftError) {
11475 CLI.
Ins.push_back(Ret);
11483 for (
unsigned i = 0, e = Args.size(); i != e; ++i) {
11487 Type *FinalType = Args[i].Ty;
11488 if (Args[i].IsByVal)
11489 FinalType = Args[i].IndirectType;
11492 for (
unsigned Value = 0, NumValues = OrigArgTys.
size();
Value != NumValues;
11495 Type *ArgTy = OrigArgTy;
11496 if (Args[i].Ty != Args[i].OrigTy) {
11497 assert(
Value == 0 &&
"Only supported for non-aggregate arguments");
11498 ArgTy = Args[i].Ty;
11503 Args[i].Node.getResNo() +
Value);
11510 Flags.setOrigAlign(OriginalAlignment);
11515 Flags.setPointer();
11518 if (Args[i].IsZExt)
11520 if (Args[i].IsSExt)
11522 if (Args[i].IsNoExt)
11524 if (Args[i].IsInReg) {
11531 Flags.setHvaStart();
11537 if (Args[i].IsSRet)
11539 if (Args[i].IsSwiftSelf)
11540 Flags.setSwiftSelf();
11541 if (Args[i].IsSwiftAsync)
11542 Flags.setSwiftAsync();
11543 if (Args[i].IsSwiftError)
11544 Flags.setSwiftError();
11545 if (Args[i].IsCFGuardTarget)
11546 Flags.setCFGuardTarget();
11547 if (Args[i].IsByVal)
11549 if (Args[i].IsByRef)
11551 if (Args[i].IsPreallocated) {
11552 Flags.setPreallocated();
11560 if (Args[i].IsInAlloca) {
11561 Flags.setInAlloca();
11570 if (Args[i].IsByVal || Args[i].IsInAlloca || Args[i].IsPreallocated) {
11571 unsigned FrameSize =
DL.getTypeAllocSize(Args[i].IndirectType);
11572 Flags.setByValSize(FrameSize);
11575 if (
auto MA = Args[i].Alignment)
11579 }
else if (
auto MA = Args[i].Alignment) {
11582 MemAlign = OriginalAlignment;
11584 Flags.setMemAlign(MemAlign);
11585 if (Args[i].IsNest)
11588 Flags.setInConsecutiveRegs();
11591 unsigned NumParts =
11596 if (Args[i].IsSExt)
11598 else if (Args[i].IsZExt)
11603 if (Args[i].IsReturned && !
Op.getValueType().isVector() &&
11608 Args[i].Ty->getPointerAddressSpace())) &&
11609 RetVTs.
size() == NumValues &&
"unexpected use of 'returned'");
11622 CLI.
RetZExt == Args[i].IsZExt))
11623 Flags.setReturned();
11629 for (
unsigned j = 0; j != NumParts; ++j) {
11635 j * Parts[j].
getValueType().getStoreSize().getKnownMinValue());
11636 if (NumParts > 1 && j == 0)
11640 if (j == NumParts - 1)
11644 CLI.
Outs.push_back(MyFlags);
11645 CLI.
OutVals.push_back(Parts[j]);
11648 if (NeedsRegBlock &&
Value == NumValues - 1)
11649 CLI.
Outs[CLI.
Outs.size() - 1].Flags.setInConsecutiveRegsLast();
11661 "LowerCall didn't return a valid chain!");
11663 "LowerCall emitted a return value for a tail call!");
11665 "LowerCall didn't emit the correct number of values!");
11677 for (
unsigned i = 0, e = CLI.
Ins.size(); i != e; ++i) {
11678 assert(InVals[i].
getNode() &&
"LowerCall emitted a null value!");
11679 assert(
EVT(CLI.
Ins[i].VT) == InVals[i].getValueType() &&
11680 "LowerCall emitted a value with the wrong type!");
11690 unsigned NumValues = RetVTs.
size();
11691 ReturnValues.
resize(NumValues);
11698 for (
unsigned i = 0; i < NumValues; ++i) {
11705 DemoteStackIdx, Offsets[i]),
11707 ReturnValues[i] = L;
11708 Chains[i] = L.getValue(1);
11715 std::optional<ISD::NodeType> AssertOp;
11720 unsigned CurReg = 0;
11721 for (
EVT VT : RetVTs) {
11727 CLI.
DAG, CLI.
DL, &InVals[CurReg], NumRegs, RegisterVT, VT,
nullptr,
11735 if (ReturnValues.
empty())
11741 return std::make_pair(Res, CLI.
Chain);
11758 if (
N->getNumValues() == 1) {
11766 "Lowering returned the wrong number of results!");
11769 for (
unsigned I = 0, E =
N->getNumValues();
I != E; ++
I)
11783 "Copy from a reg to the same reg!");
11784 assert(!Reg.isPhysical() &&
"Is a physreg");
11790 RegsForValue RFV(V->getContext(), TLI,
DAG.getDataLayout(), Reg, V->getType(),
11795 auto PreferredExtendIt =
FuncInfo.PreferredExtendType.find(V);
11796 if (PreferredExtendIt !=
FuncInfo.PreferredExtendType.end())
11797 ExtendType = PreferredExtendIt->second;
11800 PendingExports.push_back(Chain);
11812 return A->use_empty();
11814 const BasicBlock &Entry =
A->getParent()->front();
11815 for (
const User *U :
A->users())
11824 std::pair<const AllocaInst *, const StoreInst *>>;
11836 enum StaticAllocaInfo {
Unknown, Clobbered, Elidable };
11838 unsigned NumArgs = FuncInfo->
Fn->
arg_size();
11839 StaticAllocas.
reserve(NumArgs * 2);
11841 auto GetInfoIfStaticAlloca = [&](
const Value *V) -> StaticAllocaInfo * {
11844 V = V->stripPointerCasts();
11846 if (!AI || !AI->isStaticAlloca() || !FuncInfo->
StaticAllocaMap.count(AI))
11849 return &Iter.first->second;
11866 if (
I.isDebugOrPseudoInst())
11870 for (
const Use &U :
I.operands()) {
11871 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(U))
11872 *Info = StaticAllocaInfo::Clobbered;
11878 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(
SI->getValueOperand()))
11879 *Info = StaticAllocaInfo::Clobbered;
11882 const Value *Dst =
SI->getPointerOperand()->stripPointerCasts();
11883 StaticAllocaInfo *Info = GetInfoIfStaticAlloca(Dst);
11889 if (*Info != StaticAllocaInfo::Unknown)
11897 const Value *Val =
SI->getValueOperand()->stripPointerCasts();
11900 if (!Arg || Arg->hasPassPointeeByValueCopyAttr() ||
11902 DL.getTypeStoreSize(Arg->
getType()) != *AllocaSize ||
11903 !
DL.typeSizeEqualsStoreSize(Arg->
getType()) ||
11904 ArgCopyElisionCandidates.count(Arg)) {
11905 *Info = StaticAllocaInfo::Clobbered;
11909 LLVM_DEBUG(
dbgs() <<
"Found argument copy elision candidate: " << *AI
11913 *Info = StaticAllocaInfo::Elidable;
11914 ArgCopyElisionCandidates.insert({Arg, {AI,
SI}});
11919 if (ArgCopyElisionCandidates.size() == NumArgs)
11943 auto ArgCopyIter = ArgCopyElisionCandidates.find(&Arg);
11944 assert(ArgCopyIter != ArgCopyElisionCandidates.end());
11945 const AllocaInst *AI = ArgCopyIter->second.first;
11946 int FixedIndex = FINode->getIndex();
11948 int OldIndex = AllocaIndex;
11952 dbgs() <<
" argument copy elision failed due to bad fixed stack "
11958 LLVM_DEBUG(
dbgs() <<
" argument copy elision failed: alignment of alloca "
11959 "greater than stack argument alignment ("
11960 <<
DebugStr(RequiredAlignment) <<
" vs "
11968 dbgs() <<
"Eliding argument copy from " << Arg <<
" to " << *AI <<
'\n'
11969 <<
" Replacing frame index " << OldIndex <<
" with " << FixedIndex
11975 AllocaIndex = FixedIndex;
11976 ArgCopyElisionFrameIndexMap.
insert({OldIndex, FixedIndex});
11977 for (
SDValue ArgVal : ArgVals)
11981 const StoreInst *
SI = ArgCopyIter->second.second;
11994void SelectionDAGISel::LowerArguments(
const Function &
F) {
11995 SelectionDAG &DAG =
SDB->DAG;
11996 SDLoc dl =
SDB->getCurSDLoc();
12001 if (
F.hasFnAttribute(Attribute::Naked))
12006 MVT ValueVT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
12008 ISD::ArgFlagsTy
Flags;
12010 MVT RegisterVT =
TLI->getRegisterType(*DAG.
getContext(), ValueVT);
12011 ISD::InputArg RetArg(Flags, RegisterVT, ValueVT,
F.getReturnType(),
true,
12021 ArgCopyElisionCandidates);
12024 for (
const Argument &Arg :
F.args()) {
12025 unsigned ArgNo = Arg.getArgNo();
12028 bool isArgValueUsed = !Arg.
use_empty();
12030 if (Arg.hasAttribute(Attribute::ByVal))
12031 FinalType = Arg.getParamByValType();
12032 bool NeedsRegBlock =
TLI->functionArgumentNeedsConsecutiveRegisters(
12033 FinalType,
F.getCallingConv(),
F.isVarArg(),
DL);
12034 for (
unsigned Value = 0, NumValues =
Types.size();
Value != NumValues;
12037 EVT VT =
TLI->getValueType(
DL, ArgTy);
12038 ISD::ArgFlagsTy
Flags;
12041 Flags.setPointer();
12044 if (Arg.hasAttribute(Attribute::ZExt))
12046 if (Arg.hasAttribute(Attribute::SExt))
12048 if (Arg.hasAttribute(Attribute::InReg)) {
12055 Flags.setHvaStart();
12061 if (Arg.hasAttribute(Attribute::StructRet))
12063 if (Arg.hasAttribute(Attribute::SwiftSelf))
12064 Flags.setSwiftSelf();
12065 if (Arg.hasAttribute(Attribute::SwiftAsync))
12066 Flags.setSwiftAsync();
12067 if (Arg.hasAttribute(Attribute::SwiftError))
12068 Flags.setSwiftError();
12069 if (Arg.hasAttribute(Attribute::ByVal))
12071 if (Arg.hasAttribute(Attribute::ByRef))
12073 if (Arg.hasAttribute(Attribute::InAlloca)) {
12074 Flags.setInAlloca();
12082 if (Arg.hasAttribute(Attribute::Preallocated)) {
12083 Flags.setPreallocated();
12095 const Align OriginalAlignment(
12096 TLI->getABIAlignmentForCallingConv(ArgTy,
DL));
12097 Flags.setOrigAlign(OriginalAlignment);
12100 Type *ArgMemTy =
nullptr;
12101 if (
Flags.isByVal() ||
Flags.isInAlloca() ||
Flags.isPreallocated() ||
12104 ArgMemTy = Arg.getPointeeInMemoryValueType();
12106 uint64_t MemSize =
DL.getTypeAllocSize(ArgMemTy);
12111 if (
auto ParamAlign = Arg.getParamStackAlign())
12112 MemAlign = *ParamAlign;
12113 else if ((ParamAlign = Arg.getParamAlign()))
12114 MemAlign = *ParamAlign;
12116 MemAlign =
TLI->getByValTypeAlignment(ArgMemTy,
DL);
12117 if (
Flags.isByRef())
12118 Flags.setByRefSize(MemSize);
12120 Flags.setByValSize(MemSize);
12121 }
else if (
auto ParamAlign = Arg.getParamStackAlign()) {
12122 MemAlign = *ParamAlign;
12124 MemAlign = OriginalAlignment;
12126 Flags.setMemAlign(MemAlign);
12128 if (Arg.hasAttribute(Attribute::Nest))
12131 Flags.setInConsecutiveRegs();
12132 if (ArgCopyElisionCandidates.count(&Arg))
12133 Flags.setCopyElisionCandidate();
12134 if (Arg.hasAttribute(Attribute::Returned))
12135 Flags.setReturned();
12137 MVT RegisterVT =
TLI->getRegisterTypeForCallingConv(
12138 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12139 unsigned NumRegs =
TLI->getNumRegistersForCallingConv(
12140 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12141 for (
unsigned i = 0; i != NumRegs; ++i) {
12145 ISD::InputArg MyFlags(
12146 Flags, RegisterVT, VT, ArgTy, isArgValueUsed, ArgNo,
12148 if (NumRegs > 1 && i == 0)
12149 MyFlags.Flags.setSplit();
12152 MyFlags.Flags.setOrigAlign(
Align(1));
12153 if (i == NumRegs - 1)
12154 MyFlags.Flags.setSplitEnd();
12158 if (NeedsRegBlock &&
Value == NumValues - 1)
12159 Ins[Ins.
size() - 1].Flags.setInConsecutiveRegsLast();
12165 SDValue NewRoot =
TLI->LowerFormalArguments(
12166 DAG.
getRoot(),
F.getCallingConv(),
F.isVarArg(), Ins, dl, DAG, InVals);
12170 "LowerFormalArguments didn't return a valid chain!");
12172 "LowerFormalArguments didn't emit the correct number of values!");
12174 "LowerFormalArguments emitted a null value!");
12184 MVT VT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
12185 MVT RegVT =
TLI->getRegisterType(*
CurDAG->getContext(), VT);
12186 std::optional<ISD::NodeType> AssertOp;
12189 F.getCallingConv(), AssertOp);
12191 MachineFunction&
MF =
SDB->DAG.getMachineFunction();
12192 MachineRegisterInfo&
RegInfo =
MF.getRegInfo();
12194 RegInfo.createVirtualRegister(
TLI->getRegClassFor(RegVT));
12195 FuncInfo->DemoteRegister = SRetReg;
12197 SDB->DAG.getCopyToReg(NewRoot,
SDB->getCurSDLoc(), SRetReg, ArgValue);
12205 DenseMap<int, int> ArgCopyElisionFrameIndexMap;
12206 for (
const Argument &Arg :
F.args()) {
12210 unsigned NumValues = ValueVTs.
size();
12211 if (NumValues == 0)
12218 if (Ins[i].
Flags.isCopyElisionCandidate()) {
12219 unsigned NumParts = 0;
12220 for (EVT VT : ValueVTs)
12221 NumParts +=
TLI->getNumRegistersForCallingConv(*
CurDAG->getContext(),
12222 F.getCallingConv(), VT);
12226 ArrayRef(&InVals[i], NumParts), ArgHasUses);
12231 bool isSwiftErrorArg =
12232 TLI->supportSwiftError() &&
12233 Arg.hasAttribute(Attribute::SwiftError);
12234 if (!ArgHasUses && !isSwiftErrorArg) {
12235 SDB->setUnusedArgValue(&Arg, InVals[i]);
12238 if (FrameIndexSDNode *FI =
12240 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12243 for (
unsigned Val = 0; Val != NumValues; ++Val) {
12244 EVT VT = ValueVTs[Val];
12245 MVT PartVT =
TLI->getRegisterTypeForCallingConv(*
CurDAG->getContext(),
12246 F.getCallingConv(), VT);
12247 unsigned NumParts =
TLI->getNumRegistersForCallingConv(
12248 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12253 if (ArgHasUses || isSwiftErrorArg) {
12254 std::optional<ISD::NodeType> AssertOp;
12255 if (Arg.hasAttribute(Attribute::SExt))
12257 else if (Arg.hasAttribute(Attribute::ZExt))
12262 NewRoot,
F.getCallingConv(), AssertOp);
12265 if (NoFPClass !=
fcNone) {
12267 static_cast<uint64_t
>(NoFPClass), dl, MVT::i32);
12269 OutVal, SDNoFPClass);
12278 if (ArgValues.
empty())
12282 if (FrameIndexSDNode *FI =
12284 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12287 SDB->getCurSDLoc());
12289 SDB->setValue(&Arg, Res);
12299 if (LoadSDNode *LNode =
12301 if (FrameIndexSDNode *FI =
12303 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12331 FuncInfo->InitializeRegForValue(&Arg);
12332 SDB->CopyToExportRegsIfNeeded(&Arg);
12336 if (!Chains.
empty()) {
12343 assert(i == InVals.
size() &&
"Argument register count mismatch!");
12347 if (!ArgCopyElisionFrameIndexMap.
empty()) {
12348 for (MachineFunction::VariableDbgInfo &VI :
12349 MF->getInStackSlotVariableDbgInfo()) {
12350 auto I = ArgCopyElisionFrameIndexMap.
find(
VI.getStackSlot());
12351 if (
I != ArgCopyElisionFrameIndexMap.
end())
12352 VI.updateStackSlot(
I->second);
12367SelectionDAGBuilder::HandlePHINodesInSuccessorBlocks(
const BasicBlock *LLVMBB) {
12368 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12370 SmallPtrSet<MachineBasicBlock *, 4> SuccsHandled;
12376 MachineBasicBlock *SuccMBB =
FuncInfo.getMBB(SuccBB);
12380 if (!SuccsHandled.
insert(SuccMBB).second)
12388 for (
const PHINode &PN : SuccBB->phis()) {
12390 if (PN.use_empty())
12394 if (PN.getType()->isEmptyTy())
12398 const Value *PHIOp = PN.getIncomingValueForBlock(LLVMBB);
12403 RegOut =
FuncInfo.CreateRegs(&PN);
12414 auto I =
FuncInfo.ValueMap.find(PHIOp);
12420 "Didn't codegen value into a register!??");
12430 for (EVT VT : ValueVTs) {
12432 for (
unsigned i = 0; i != NumRegisters; ++i)
12434 Reg += NumRegisters;
12454void SelectionDAGBuilder::updateDAGForMaybeTailCall(
SDValue MaybeTC) {
12456 if (MaybeTC.
getNode() !=
nullptr)
12457 DAG.setRoot(MaybeTC);
12462void SelectionDAGBuilder::lowerWorkItem(SwitchWorkListItem W,
Value *
Cond,
12465 MachineFunction *CurMF =
FuncInfo.MF;
12466 MachineBasicBlock *NextMBB =
nullptr;
12471 unsigned Size =
W.LastCluster -
W.FirstCluster + 1;
12473 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12475 if (
Size == 2 &&
W.MBB == SwitchMBB) {
12483 CaseCluster &
Small = *
W.FirstCluster;
12484 CaseCluster &
Big = *
W.LastCluster;
12488 const APInt &SmallValue =
Small.Low->getValue();
12489 const APInt &BigValue =
Big.Low->getValue();
12492 APInt CommonBit = BigValue ^ SmallValue;
12499 DAG.getConstant(CommonBit,
DL, VT));
12501 DL, MVT::i1,
Or,
DAG.getConstant(BigValue | SmallValue,
DL, VT),
12507 addSuccessorWithProb(SwitchMBB,
Small.MBB,
Small.Prob +
Big.Prob);
12509 addSuccessorWithProb(
12510 SwitchMBB, DefaultMBB,
12514 addSuccessorWithProb(SwitchMBB, DefaultMBB);
12522 DAG.getBasicBlock(DefaultMBB));
12524 DAG.setRoot(BrCond);
12536 [](
const CaseCluster &a,
const CaseCluster &b) {
12537 return a.Prob != b.Prob ?
12539 a.Low->getValue().slt(b.Low->getValue());
12546 if (
I->Prob >
W.LastCluster->Prob)
12548 if (
I->Kind ==
CC_Range &&
I->MBB == NextMBB) {
12556 BranchProbability DefaultProb =
W.DefaultProb;
12557 BranchProbability UnhandledProbs = DefaultProb;
12559 UnhandledProbs +=
I->Prob;
12561 MachineBasicBlock *CurMBB =
W.MBB;
12563 bool FallthroughUnreachable =
false;
12564 MachineBasicBlock *Fallthrough;
12565 if (
I ==
W.LastCluster) {
12567 Fallthrough = DefaultMBB;
12572 CurMF->
insert(BBI, Fallthrough);
12576 UnhandledProbs -=
I->Prob;
12581 JumpTableHeader *JTH = &
SL->JTCases[
I->JTCasesIndex].first;
12582 SwitchCG::JumpTable *JT = &
SL->JTCases[
I->JTCasesIndex].second;
12585 MachineBasicBlock *JumpMBB = JT->
MBB;
12586 CurMF->
insert(BBI, JumpMBB);
12588 auto JumpProb =
I->Prob;
12589 auto FallthroughProb = UnhandledProbs;
12597 if (*SI == DefaultMBB) {
12598 JumpProb += DefaultProb / 2;
12599 FallthroughProb -= DefaultProb / 2;
12617 if (FallthroughUnreachable) {
12624 addSuccessorWithProb(CurMBB, Fallthrough, FallthroughProb);
12625 addSuccessorWithProb(CurMBB, JumpMBB, JumpProb);
12634 if (CurMBB == SwitchMBB) {
12642 BitTestBlock *BTB = &
SL->BitTestCases[
I->BTCasesIndex];
12645 for (BitTestCase &BTC : BTB->
Cases)
12657 BTB->
Prob += DefaultProb / 2;
12661 if (FallthroughUnreachable)
12665 if (CurMBB == SwitchMBB) {
12672 const Value *
RHS, *
LHS, *MHS;
12674 if (
I->Low ==
I->High) {
12689 if (FallthroughUnreachable)
12693 CaseBlock CB(CC,
LHS,
RHS, MHS,
I->MBB, Fallthrough, CurMBB,
12696 if (CurMBB == SwitchMBB)
12699 SL->SwitchCases.push_back(CB);
12704 CurMBB = Fallthrough;
12708void SelectionDAGBuilder::splitWorkItem(
SwitchWorkList &WorkList,
12709 const SwitchWorkListItem &W,
12712 assert(
W.FirstCluster->Low->getValue().slt(
W.LastCluster->Low->getValue()) &&
12713 "Clusters not sorted?");
12714 assert(
W.LastCluster -
W.FirstCluster + 1 >= 2 &&
"Too small to split!");
12716 auto [LastLeft, FirstRight, LeftProb, RightProb] =
12717 SL->computeSplitWorkItemInfo(W);
12722 assert(PivotCluster >
W.FirstCluster);
12723 assert(PivotCluster <=
W.LastCluster);
12728 const ConstantInt *Pivot = PivotCluster->Low;
12737 MachineBasicBlock *LeftMBB;
12738 if (FirstLeft == LastLeft && FirstLeft->Kind ==
CC_Range &&
12739 FirstLeft->Low ==
W.GE &&
12740 (FirstLeft->High->getValue() + 1LL) == Pivot->
getValue()) {
12741 LeftMBB = FirstLeft->MBB;
12743 LeftMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12744 FuncInfo.MF->insert(BBI, LeftMBB);
12746 {LeftMBB, FirstLeft, LastLeft,
W.GE, Pivot,
W.DefaultProb / 2});
12754 MachineBasicBlock *RightMBB;
12755 if (FirstRight == LastRight && FirstRight->Kind ==
CC_Range &&
12756 W.LT && (FirstRight->High->getValue() + 1ULL) ==
W.LT->getValue()) {
12757 RightMBB = FirstRight->MBB;
12759 RightMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12760 FuncInfo.MF->insert(BBI, RightMBB);
12762 {RightMBB, FirstRight, LastRight, Pivot,
W.LT,
W.DefaultProb / 2});
12768 CaseBlock CB(
ISD::SETLT,
Cond, Pivot,
nullptr, LeftMBB, RightMBB,
W.MBB,
12771 if (
W.MBB == SwitchMBB)
12774 SL->SwitchCases.push_back(CB);
12799 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12807 unsigned PeeledCaseIndex = 0;
12808 bool SwitchPeeled =
false;
12809 for (
unsigned Index = 0;
Index < Clusters.size(); ++
Index) {
12810 CaseCluster &CC = Clusters[
Index];
12811 if (CC.
Prob < TopCaseProb)
12813 TopCaseProb = CC.
Prob;
12814 PeeledCaseIndex =
Index;
12815 SwitchPeeled =
true;
12820 LLVM_DEBUG(
dbgs() <<
"Peeled one top case in switch stmt, prob: "
12821 << TopCaseProb <<
"\n");
12826 MachineBasicBlock *PeeledSwitchMBB =
12828 FuncInfo.MF->insert(BBI, PeeledSwitchMBB);
12831 auto PeeledCaseIt = Clusters.begin() + PeeledCaseIndex;
12832 SwitchWorkListItem
W = {SwitchMBB, PeeledCaseIt, PeeledCaseIt,
12833 nullptr,
nullptr, TopCaseProb.
getCompl()};
12834 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, PeeledSwitchMBB);
12836 Clusters.erase(PeeledCaseIt);
12837 for (CaseCluster &CC : Clusters) {
12839 dbgs() <<
"Scale the probablity for one cluster, before scaling: "
12840 << CC.
Prob <<
"\n");
12844 PeeledCaseProb = TopCaseProb;
12845 return PeeledSwitchMBB;
12848void SelectionDAGBuilder::visitSwitch(
const SwitchInst &
SI) {
12850 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12852 Clusters.reserve(
SI.getNumCases());
12853 for (
auto I :
SI.cases()) {
12854 MachineBasicBlock *Succ =
FuncInfo.getMBB(
I.getCaseSuccessor());
12855 const ConstantInt *CaseVal =
I.getCaseValue();
12856 BranchProbability Prob =
12858 : BranchProbability(1,
SI.getNumCases() + 1);
12862 MachineBasicBlock *DefaultMBB =
FuncInfo.getMBB(
SI.getDefaultDest());
12871 MachineBasicBlock *PeeledSwitchMBB =
12872 peelDominantCaseCluster(SI, Clusters, PeeledCaseProb);
12875 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12876 if (Clusters.empty()) {
12877 assert(PeeledSwitchMBB == SwitchMBB);
12879 if (DefaultMBB != NextBlock(SwitchMBB)) {
12886 SL->findJumpTables(Clusters, &SI,
getCurSDLoc(), DefaultMBB,
DAG.getPSI(),
12888 SL->findBitTestClusters(Clusters, &SI);
12891 dbgs() <<
"Case clusters: ";
12892 for (
const CaseCluster &
C : Clusters) {
12898 C.Low->getValue().print(
dbgs(),
true);
12899 if (
C.Low !=
C.High) {
12901 C.High->getValue().print(
dbgs(),
true);
12908 assert(!Clusters.empty());
12912 auto DefaultProb = getEdgeProbability(PeeledSwitchMBB, DefaultMBB);
12916 DefaultMBB ==
FuncInfo.getMBB(
SI.getDefaultDest()))
12919 {PeeledSwitchMBB,
First,
Last,
nullptr,
nullptr, DefaultProb});
12921 while (!WorkList.
empty()) {
12923 unsigned NumClusters =
W.LastCluster -
W.FirstCluster + 1;
12928 splitWorkItem(WorkList, W,
SI.getCondition(), SwitchMBB);
12932 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, DefaultMBB);
12936void SelectionDAGBuilder::visitStepVector(
const CallInst &
I) {
12937 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12943void SelectionDAGBuilder::visitVectorReverse(
const CallInst &
I) {
12944 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12949 assert(VT ==
V.getValueType() &&
"Malformed vector.reverse!");
12958 SmallVector<int, 8>
Mask;
12960 for (
unsigned i = 0; i != NumElts; ++i)
12961 Mask.push_back(NumElts - 1 - i);
12966void SelectionDAGBuilder::visitVectorDeinterleave(
const CallInst &
I,
12975 EVT OutVT = ValueVTs[0];
12979 for (
unsigned i = 0; i != Factor; ++i) {
12980 assert(ValueVTs[i] == OutVT &&
"Expected VTs to be the same");
12982 DAG.getVectorIdxConstant(OutNumElts * i,
DL));
12988 SDValue Even =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
12990 SDValue Odd =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
12998 DAG.getVTList(ValueVTs), SubVecs);
13002void SelectionDAGBuilder::visitVectorInterleave(
const CallInst &
I,
13005 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13010 for (
unsigned i = 0; i < Factor; ++i) {
13013 "Expected VTs to be the same");
13031 for (
unsigned i = 0; i < Factor; ++i)
13038void SelectionDAGBuilder::visitFreeze(
const FreezeInst &
I) {
13042 unsigned NumValues = ValueVTs.
size();
13043 if (NumValues == 0)
return;
13048 for (
unsigned i = 0; i != NumValues; ++i)
13053 DAG.getVTList(ValueVTs), Values));
13056void SelectionDAGBuilder::visitVectorSplice(
const CallInst &
I) {
13057 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13063 const bool IsLeft =
I.getIntrinsicID() == Intrinsic::vector_splice_left;
13078 uint64_t Idx = IsLeft ?
Imm : NumElts -
Imm;
13081 SmallVector<int, 8>
Mask;
13082 for (
unsigned i = 0; i < NumElts; ++i)
13083 Mask.push_back(Idx + i);
13111 assert(
MI->getOpcode() == TargetOpcode::COPY &&
13112 "start of copy chain MUST be COPY");
13113 Reg =
MI->getOperand(1).getReg();
13116 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
13120 if (
MI->getOpcode() == TargetOpcode::COPY) {
13121 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
13122 Reg =
MI->getOperand(1).getReg();
13123 assert(
Reg.isPhysical() &&
"expected COPY of physical register");
13126 assert(
MI->getOpcode() == TargetOpcode::INLINEASM_BR &&
13127 "end of copy chain MUST be INLINEASM_BR");
13137void SelectionDAGBuilder::visitCallBrLandingPad(
const CallInst &
I) {
13143 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13144 const TargetRegisterInfo *
TRI =
DAG.getSubtarget().getRegisterInfo();
13145 MachineRegisterInfo &MRI =
DAG.getMachineFunction().getRegInfo();
13153 for (
auto &
T : TargetConstraints) {
13154 SDISelAsmOperandInfo OpInfo(
T);
13162 switch (OpInfo.ConstraintType) {
13173 FuncInfo.MBB->addLiveIn(OriginalDef);
13181 ResultVTs.
push_back(OpInfo.ConstraintVT);
13190 ResultVTs.
push_back(OpInfo.ConstraintVT);
13198 DAG.getVTList(ResultVTs), ResultValues);
static unsigned getIntrinsicID(const SDNode *N)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static msgpack::DocNode getNode(msgpack::DocNode DN, msgpack::Type Type, MCValue Val)
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...
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
MachineBasicBlock MachineBasicBlock::iterator MBBI
Function Alias Analysis Results
Atomic ordering constants.
This file contains the simple types necessary to represent the attributes associated with functions a...
static const Function * getParent(const Value *V)
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")
This file contains the declarations for the subclasses of Constant, which represent the different fla...
static AttributeList getReturnAttrs(FastISel::CallLoweringInfo &CLI)
Returns an AttributeList representing the attributes applied to the return value of the given call.
static Value * getCondition(Instruction *I)
const HexagonInstrInfo * TII
This file provides various utilities for inspecting and working with the control flow graph in LLVM I...
Module.h This file contains the declarations for the Module class.
static void getRegistersForValue(MachineFunction &MF, MachineIRBuilder &MIRBuilder, GISelAsmOperandInfo &OpInfo, GISelAsmOperandInfo &RefOpInfo)
Assign virtual/physical registers for the specified register operand.
static void computeConstraintToUse(const TargetLowering *TLI, TargetLowering::AsmOperandInfo &OpInfo)
This file defines an InstructionCost class that is used when calculating the cost of an instruction,...
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
Machine Check Debug Module
static bool isUndef(const MachineInstr &MI)
Register const TargetRegisterInfo * TRI
Promote Memory to Register
static const Function * getCalledFunction(const Value *V)
This file provides utility analysis objects describing memory locations.
This file provides utility for Memory Model Relaxation Annotations (MMRAs).
static MCRegister getReg(const MCDisassembler *D, unsigned RC, unsigned RegNo)
static unsigned getAddressSpace(const Value *V, unsigned MaxLookup)
MachineInstr unsigned OpIdx
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
uint64_t IntrinsicInst * II
OptimizedStructLayoutField Field
const SmallVectorImpl< MachineOperand > MachineBasicBlock * TBB
const SmallVectorImpl< MachineOperand > & Cond
static Type * getValueType(Value *V, bool LookThroughCmp=false)
Returns the "element type" of the given value/instruction V.
static bool hasOnlySelectUsers(const Value *Cond)
static SDValue getLoadStackGuard(SelectionDAG &DAG, const SDLoc &DL, SDValue &Chain)
Create a LOAD_STACK_GUARD node, and let it carry the target specific global variable if there exists ...
static bool getUniformBase(const Value *Ptr, SDValue &Base, SDValue &Index, SDValue &Scale, SelectionDAGBuilder *SDB, const BasicBlock *CurBB, uint64_t ElemSize)
static void failForInvalidBundles(const CallBase &I, StringRef Name, ArrayRef< uint32_t > AllowedBundles)
static void addStackMapLiveVars(const CallBase &Call, unsigned StartIdx, const SDLoc &DL, SmallVectorImpl< SDValue > &Ops, SelectionDAGBuilder &Builder)
Add a stack map intrinsic call's live variable operands to a stackmap or patchpoint target node's ope...
static const unsigned MaxParallelChains
static SDValue expandPow(const SDLoc &dl, SDValue LHS, SDValue RHS, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
visitPow - Lower a pow intrinsic.
static const CallBase * FindPreallocatedCall(const Value *PreallocatedSetup)
Given a @llvm.call.preallocated.setup, return the corresponding preallocated call.
static cl::opt< unsigned > SwitchPeelThreshold("switch-peel-threshold", cl::Hidden, cl::init(66), cl::desc("Set the case probability threshold for peeling the case from a " "switch statement. A value greater than 100 will void this " "optimization"))
static cl::opt< bool > InsertAssertAlign("insert-assert-align", cl::init(true), cl::desc("Insert the experimental `assertalign` node."), cl::ReallyHidden)
static unsigned getISDForVPIntrinsic(const VPIntrinsic &VPIntrin)
static bool handleDanglingVariadicDebugInfo(SelectionDAG &DAG, DILocalVariable *Variable, DebugLoc DL, unsigned Order, SmallVectorImpl< Value * > &Values, DIExpression *Expression)
static bool prepareDAGLevelOperands(ConstraintDecisionInfo &Info, const CallBase &Call, SelectionDAGBuilder &Builder, const TargetLowering &TLI, SelectionDAG &DAG)
Prepare DAG-level operands.
static unsigned findMatchingInlineAsmOperand(unsigned OperandNo, const std::vector< SDValue > &AsmNodeOperands)
static void patchMatchingInput(const SDISelAsmOperandInfo &OpInfo, SDISelAsmOperandInfo &MatchingOpInfo, SelectionDAG &DAG)
Make sure that the output operand OpInfo and its corresponding input operand MatchingOpInfo have comp...
static void findUnwindDestinations(FunctionLoweringInfo &FuncInfo, const BasicBlock *EHPadBB, BranchProbability Prob, SmallVectorImpl< std::pair< MachineBasicBlock *, BranchProbability > > &UnwindDests)
When an invoke or a cleanupret unwinds to the next EH pad, there are many places it could ultimately ...
static unsigned FixedPointIntrinsicToOpcode(unsigned Intrinsic)
static BranchProbability scaleCaseProbality(BranchProbability CaseProb, BranchProbability PeeledCaseProb)
static SDValue expandExp2(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandExp2 - Lower an exp2 intrinsic.
static SDValue expandDivFix(unsigned Opcode, const SDLoc &DL, SDValue LHS, SDValue RHS, SDValue Scale, SelectionDAG &DAG, const TargetLowering &TLI)
static SDValue getF32Constant(SelectionDAG &DAG, unsigned Flt, const SDLoc &dl)
getF32Constant - Get 32-bit floating point constant.
static SDValue widenVectorToPartType(SelectionDAG &DAG, SDValue Val, const SDLoc &DL, EVT PartVT)
static SDValue expandLog10(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandLog10 - Lower a log10 intrinsic.
DenseMap< const Argument *, std::pair< const AllocaInst *, const StoreInst * > > ArgCopyElisionMapTy
static void getCopyToPartsVector(SelectionDAG &DAG, const SDLoc &dl, SDValue Val, SDValue *Parts, unsigned NumParts, MVT PartVT, const Value *V, std::optional< CallingConv::ID > CallConv)
getCopyToPartsVector - Create a series of nodes that contain the specified value split into legal par...
static void getUnderlyingArgRegs(SmallVectorImpl< std::pair< Register, TypeSize > > &Regs, const SDValue &N)
static void getCopyToParts(SelectionDAG &DAG, const SDLoc &DL, SDValue Val, SDValue *Parts, unsigned NumParts, MVT PartVT, const Value *V, std::optional< CallingConv::ID > CallConv=std::nullopt, ISD::NodeType ExtendKind=ISD::ANY_EXTEND)
getCopyToParts - Create a series of nodes that contain the specified value split into legal parts.
static SDValue getMemCmpLoad(const Value *PtrVal, MVT LoadVT, SelectionDAGBuilder &Builder)
static SDValue expandLog2(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandLog2 - Lower a log2 intrinsic.
static SDValue getAddressForMemoryInput(SDValue Chain, const SDLoc &Location, SDISelAsmOperandInfo &OpInfo, SelectionDAG &DAG)
Get a direct memory input to behave well as an indirect operand.
static bool isOnlyUsedInEntryBlock(const Argument *A, bool FastISel)
isOnlyUsedInEntryBlock - If the specified argument is only used in the entry block,...
static void diagnosePossiblyInvalidConstraint(LLVMContext &Ctx, const Value *V, const Twine &ErrMsg)
static bool collectInstructionDeps(SmallMapVector< const Instruction *, bool, 8 > *Deps, const Value *V, SmallMapVector< const Instruction *, bool, 8 > *Necessary=nullptr, unsigned Depth=0)
static void findArgumentCopyElisionCandidates(const DataLayout &DL, FunctionLoweringInfo *FuncInfo, ArgCopyElisionMapTy &ArgCopyElisionCandidates)
Scan the entry block of the function in FuncInfo for arguments that look like copies into a local all...
static bool isFunction(SDValue Op)
static SDValue GetExponent(SelectionDAG &DAG, SDValue Op, const TargetLowering &TLI, const SDLoc &dl)
GetExponent - Get the exponent:
static Register FollowCopyChain(MachineRegisterInfo &MRI, Register Reg)
static SDValue ExpandPowI(const SDLoc &DL, SDValue LHS, SDValue RHS, SelectionDAG &DAG)
ExpandPowI - Expand a llvm.powi intrinsic.
static SDValue expandLog(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandLog - Lower a log intrinsic.
static SDValue getCopyFromParts(SelectionDAG &DAG, const SDLoc &DL, const SDValue *Parts, unsigned NumParts, MVT PartVT, EVT ValueVT, const Value *V, SDValue InChain, std::optional< CallingConv::ID > CC=std::nullopt, std::optional< ISD::NodeType > AssertOp=std::nullopt)
getCopyFromParts - Create a value that contains the specified legal parts combined into the value the...
static SDValue getLimitedPrecisionExp2(SDValue t0, const SDLoc &dl, SelectionDAG &DAG)
static bool determineConstraints(ConstraintDecisionInfo &Info, TargetLowering::AsmOperandInfoVector &TargetConstraints, const CallBase &Call, SelectionDAGBuilder &Builder, const TargetLowering &TLI, const TargetMachine &TM, SelectionDAG &DAG, const BasicBlock *EHPadBB)
DetermineConstraints - Find the constraints to use for inline asm operands.
static bool constructOperandInfo(ConstraintDecisionInfo &Info, TargetLowering::AsmOperandInfoVector &TargetConstraints, SelectionDAGBuilder &Builder, const TargetLowering &TLI, ExtraFlags &ExtraInfo)
Construct operand info objects.
static SDValue GetSignificand(SelectionDAG &DAG, SDValue Op, const SDLoc &dl)
GetSignificand - Get the significand and build it into a floating-point number with exponent of 1:
static SDValue expandExp(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandExp - Lower an exp intrinsic.
static const MDNode * getRangeMetadata(const Instruction &I)
static cl::opt< unsigned, true > LimitFPPrecision("limit-float-precision", cl::desc("Generate low-precision inline sequences " "for some float libcalls"), cl::location(LimitFloatPrecision), cl::Hidden, cl::init(0))
static void tryToElideArgumentCopy(FunctionLoweringInfo &FuncInfo, SmallVectorImpl< SDValue > &Chains, DenseMap< int, int > &ArgCopyElisionFrameIndexMap, SmallPtrSetImpl< const Instruction * > &ElidedArgCopyInstrs, ArgCopyElisionMapTy &ArgCopyElisionCandidates, const Argument &Arg, ArrayRef< SDValue > ArgVals, bool &ArgHasUses)
Try to elide argument copies from memory into a local alloca.
static unsigned LimitFloatPrecision
LimitFloatPrecision - Generate low-precision inline sequences for some float libcalls (6,...
static SDValue getCopyFromPartsVector(SelectionDAG &DAG, const SDLoc &DL, const SDValue *Parts, unsigned NumParts, MVT PartVT, EVT ValueVT, const Value *V, SDValue InChain, std::optional< CallingConv::ID > CC)
getCopyFromPartsVector - Create a value that contains the specified legal parts combined into the val...
static bool InBlock(const Value *V, const BasicBlock *BB)
static FPClassTest getNoFPClass(const Instruction &I)
static LLVM_ATTRIBUTE_ALWAYS_INLINE MVT::SimpleValueType getSimpleVT(const uint8_t *MatcherTable, size_t &MatcherIndex)
getSimpleVT - Decode a value in MatcherTable, if it's a VBR encoded value, use GetVBR to decode it.
This file defines the SmallPtrSet class.
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static SymbolRef::Type getType(const Symbol *Sym)
uint16_t RegSizeInBits(const MCRegisterInfo &MRI, MCRegister RegNo)
static const fltSemantics & IEEEsingle()
static LLVM_ABI Semantics SemanticsToEnum(const llvm::fltSemantics &Sem)
static LLVM_ABI const fltSemantics * getArbitraryFPSemantics(StringRef Format)
Returns the fltSemantics for a given arbitrary FP format string, or nullptr if invalid.
Class for arbitrary precision integers.
bool isNonNegative() const
Determine if this APInt Value is non-negative (>= 0)
bool isPowerOf2() const
Check if this APInt's value is a power of two greater than zero.
an instruction to allocate memory on the stack
Align getAlign() const
Return the alignment of the memory that is being allocated by the instruction.
LLVM_ABI std::optional< TypeSize > getAllocationSize(const DataLayout &DL) const
Get allocation size in bytes.
This class represents an incoming formal argument to a Function.
LLVM_ABI bool hasAttribute(Attribute::AttrKind Kind) const
Check if an argument has a given attribute.
unsigned getArgNo() const
Return the index of this formal argument in its containing function.
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.
A cache of @llvm.assume calls within a function.
An instruction that atomically checks whether a specified value is in a memory location,...
an instruction that atomically reads a memory location, combines it with another value,...
@ USubCond
Subtract only if no unsigned overflow.
@ FMinimum
*p = minimum(old, v) minimum matches the behavior of llvm.minimum.
@ Min
*p = old <signed v ? old : v
@ USubSat
*p = usub.sat(old, v) usub.sat matches the behavior of llvm.usub.sat.
@ FMaximum
*p = maximum(old, v) maximum matches the behavior of llvm.maximum.
@ UIncWrap
Increment one up to a maximum value.
@ Max
*p = old >signed v ? old : v
@ UMin
*p = old <unsigned v ? old : v
@ FMin
*p = minnum(old, v) minnum matches the behavior of llvm.minnum.
@ UMax
*p = old >unsigned v ? old : v
@ FMaximumNum
*p = maximumnum(old, v) maximumnum matches the behavior of llvm.maximumnum.
@ FMax
*p = maxnum(old, v) maxnum matches the behavior of llvm.maxnum.
@ UDecWrap
Decrement one until a minimum value or zero.
@ FMinimumNum
*p = minimumnum(old, v) minimumnum matches the behavior of llvm.minimumnum.
This class holds the attributes for a particular argument, parameter, function, or return value.
LLVM Basic Block Representation.
const Function * getParent() const
Return the enclosing method, or null if none.
LLVM_ABI InstListType::const_iterator getFirstNonPHIIt() const
Returns an iterator to the first instruction in this block that is not a PHINode instruction.
InstListType::const_iterator const_iterator
LLVM_ABI bool isEntryBlock() const
Return true if this is the entry block of the containing function.
LLVM_ABI InstListType::const_iterator getFirstNonPHIOrDbg(bool SkipPseudoOp=true) const
Returns a pointer to the first instruction in this block that is not a PHINode or a debug intrinsic,...
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction; assumes that the block is well-formed.
This class is a wrapper over an AAResults, and it is intended to be used only when there are no IR ch...
This class represents a no-op cast from one type to another.
The address of a basic block.
Analysis providing branch probability information.
LLVM_ABI BranchProbability getEdgeProbability(const BasicBlock *Src, unsigned IndexInSuccessors) const
Get an edge's probability, relative to other out-edges of the Src.
LLVM_ABI bool isEdgeHot(const BasicBlock *Src, const BasicBlock *Dst) const
Test if an edge is hot relative to other out-edges of the Src.
static uint32_t getDenominator()
static BranchProbability getOne()
static BranchProbability getUnknown()
uint32_t getNumerator() const
LLVM_ABI uint64_t scale(uint64_t Num) const
Scale a large integer.
BranchProbability getCompl() const
static BranchProbability getZero()
static void normalizeProbabilities(ProbabilityIter Begin, ProbabilityIter End)
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
std::optional< OperandBundleUse > getOperandBundle(StringRef Name) const
Return an operand bundle by name, if present.
CallingConv::ID getCallingConv() const
User::op_iterator arg_begin()
Return the iterator pointing to the beginning of the argument list.
LLVM_ABI bool isMustTailCall() const
Tests if this call site must be tail call optimized.
LLVM_ABI bool isIndirectCall() const
Return true if the callsite is an indirect call.
unsigned countOperandBundlesOfType(StringRef Name) const
Return the number of operand bundles with the tag Name attached to this instruction.
Value * getCalledOperand() const
Value * getArgOperand(unsigned i) const
User::op_iterator arg_end()
Return the iterator pointing to the end of the argument list.
bool isConvergent() const
Determine if the invoke is convergent.
FunctionType * getFunctionType() const
unsigned arg_size() const
AttributeList getAttributes() const
Return the attributes for this call.
LLVM_ABI bool isTailCall() const
Tests if this call site is marked as a tail call.
CallBr instruction, tracking function calls that may not return control but instead transfer it to a ...
This class represents a function call, abstracting a target machine's calling convention.
This class is the base class for the comparison instructions.
Predicate
This enumeration lists the possible predicates for CmpInst subclasses.
Conditional Branch instruction.
Class for constant bytes.
ConstantDataSequential - A vector or array constant whose element type is a simple 1/2/4/8-byte integ...
A constant value that is initialized with an expression using other constant values.
ConstantFP - Floating Point Values [float, double].
This is the shared class of boolean and integer constants.
static LLVM_ABI ConstantInt * getTrue(LLVMContext &Context)
bool isZero() const
This is just a convenience method to make client code smaller for a common code.
static LLVM_ABI ConstantInt * getFalse(LLVMContext &Context)
uint64_t getZExtValue() const
Return the constant as a 64-bit unsigned integer value after it has been zero extended as appropriate...
const APInt & getValue() const
Return the constant as an APInt value reference.
A signed pointer, in the ptrauth sense.
uint64_t getZExtValue() const
Constant Vector Declarations.
This is an important base class in LLVM.
This is the common base class for constrained floating point intrinsics.
LLVM_ABI std::optional< fp::ExceptionBehavior > getExceptionBehavior() const
LLVM_ABI unsigned getNonMetadataArgCount() const
LLVM_ABI bool isEntryValue() const
Check if the expression consists of exactly one entry value operand.
static bool fragmentsOverlap(const FragmentInfo &A, const FragmentInfo &B)
Check if fragments overlap between a pair of FragmentInfos.
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 std::optional< FragmentInfo > getFragmentInfo(expr_op_iterator Start, expr_op_iterator End)
Retrieve the details of this fragment expression.
LLVM_ABI uint64_t getNumLocationOperands() const
Return the number of unique location operands referred to (via DW_OP_LLVM_arg) in this expression; th...
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...
static LLVM_ABI const DIExpression * convertToUndefExpression(const DIExpression *Expr)
Removes all elements from Expr that do not apply to an undef debug value, which includes every operat...
static LLVM_ABI DIExpression * prepend(const DIExpression *Expr, uint8_t Flags, int64_t Offset=0)
Prepend DIExpr with a deref and offset operation and optionally turn it into a stack value or/and an ...
static LLVM_ABI DIExpression * prependOpcodes(const DIExpression *Expr, SmallVectorImpl< uint64_t > &Ops, bool StackValue=false, bool EntryValue=false)
Prepend DIExpr with the given opcodes and optionally turn it into a stack value.
Base class for variables.
LLVM_ABI std::optional< uint64_t > getSizeInBits() const
Determines the size of the variable's type.
A parsed version of the target data layout string in and methods for querying it.
Records a position in IR for a source label (DILabel).
Base class for non-instruction debug metadata records that have positions within IR.
DebugLoc getDebugLoc() const
Record of a variable value-assignment, aka a non instruction representation of the dbg....
LocationType getType() const
LLVM_ABI Value * getVariableLocationOp(unsigned OpIdx) const
DIExpression * getExpression() const
DILocalVariable * getVariable() const
LLVM_ABI iterator_range< location_op_iterator > location_ops() const
Get the locations corresponding to the variable referenced by the debug info intrinsic.
LLVM_ABI DILocation * getInlinedAt() const
iterator find(const_arg_type_t< KeyT > Val)
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT, true > const_iterator
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
void reserve(size_type NumEntries)
Grow the densemap so that it can contain at least NumEntries items before resizing again.
Diagnostic information for inline asm reporting.
static constexpr ElementCount getFixed(ScalarTy MinVal)
static constexpr ElementCount get(ScalarTy MinVal, bool Scalable)
constexpr bool isScalar() const
Exactly one element.
Lightweight error class with error context and mandatory checking.
Class representing an expression and its matching format.
This instruction compares its operands according to the predicate given to the constructor.
This is a fast-path instruction selection class that generates poor code and doesn't support illegal ...
bool allowReassoc() const
Flag queries.
An instruction for ordering other memory operations.
static LLVM_ABI FixedVectorType * get(Type *ElementType, unsigned NumElts)
This class represents a freeze function that returns random concrete value if an operand is either a ...
FunctionLoweringInfo - This contains information that is global to a function that is used when lower...
BranchProbabilityInfo * BPI
MachineBasicBlock * getMBB(const BasicBlock *BB) const
DenseMap< const AllocaInst *, int > StaticAllocaMap
StaticAllocaMap - Keep track of frame indices for fixed sized allocas in the entry block.
const LiveOutInfo * GetLiveOutRegInfo(Register Reg)
GetLiveOutRegInfo - Gets LiveOutInfo for a register, returning NULL if the register is a PHI destinat...
MachineBasicBlock * MBB
MBB - The current block.
Class to represent function types.
unsigned getNumParams() const
Return the number of fixed parameters this function type requires.
Type * getParamType(unsigned i) const
Parameter type accessors.
Type * getReturnType() const
Data structure describing the variable locations in a function.
const BasicBlock & getEntryBlock() const
FunctionType * getFunctionType() const
Returns the FunctionType for me.
Intrinsic::ID getIntrinsicID() const LLVM_READONLY
getIntrinsicID - This method returns the ID number of the specified function, or Intrinsic::not_intri...
bool hasMinSize() const
Optimize this function for minimum size (-Oz).
bool hasParamAttribute(unsigned ArgNo, Attribute::AttrKind Kind) const
check if an attributes is in the list of attributes.
CallingConv::ID getCallingConv() const
getCallingConv()/setCallingConv(CC) - These method get and set the calling convention of this functio...
Constant * getPersonalityFn() const
Get the personality function associated with this function.
AttributeList getAttributes() const
Return the attribute list for this Function.
bool isIntrinsic() const
isIntrinsic - Returns true if the function's name starts with "llvm.".
bool hasFnAttribute(Attribute::AttrKind Kind) const
Return true if the function has the attribute.
Garbage collection metadata for a single function.
bool hasNoUnsignedSignedWrap() const
bool hasNoUnsignedWrap() const
an instruction for type-safe pointer arithmetic to access elements of arrays and structs
static StringRef dropLLVMManglingEscape(StringRef Name)
If the given string begins with the GlobalValue name mangling escape character '\1',...
bool hasDLLImportStorageClass() const
Module * getParent()
Get the module that this global value is contained inside of...
This instruction compares its operands according to the predicate given to the constructor.
Indirect Branch Instruction.
void setMemConstraint(ConstraintCode C)
setMemConstraint - Augment an existing flag with the constraint code for a memory constraint.
This instruction inserts a struct field of array element value into an aggregate value.
MDNode * getMetadata(unsigned KindID) const
Get the metadata of given kind attached to this Instruction.
LLVM_ABI FastMathFlags getFastMathFlags() const LLVM_READONLY
Convenience function for getting all the fast-math flags, which must be an operator which supports th...
LLVM_ABI AAMDNodes getAAMetadata() const
Returns the AA metadata for this instruction.
@ MIN_INT_BITS
Minimum number of bits that can be specified.
Intrinsic::ID getIntrinsicID() const
Return the intrinsic ID of this intrinsic.
This is an important class for using LLVM in a threaded context.
@ OB_clang_arc_attachedcall
LLVM_ABI void diagnose(const DiagnosticInfo &DI)
Report a message to the currently installed diagnostic handler.
The landingpad instruction holds all of the information necessary to generate correct exception handl...
A helper class to return the specified delimiter string after the first invocation of operator String...
An instruction for reading from memory.
static LocationSize precise(uint64_t Value)
static constexpr LocationSize beforeOrAfterPointer()
Any location before or after the base pointer (but still within the underlying object).
static LocationSize upperBound(uint64_t Value)
LLVM_ABI MCSymbol * createTempSymbol()
Create a temporary symbol with a unique name.
LLVM_ABI MCSymbol * getOrCreateFrameAllocSymbol(const Twine &FuncName, unsigned Idx)
Gets a symbol that will be defined to the final stack offset of a local variable after codegen.
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
const MDOperand & getOperand(unsigned I) const
LLVM_ABI StringRef getString() const
@ INVALID_SIMPLE_VALUE_TYPE
uint64_t getScalarSizeInBits() const
unsigned getVectorNumElements() const
bool isVector() const
Return true if this is a vector value type.
bool isInteger() const
Return true if this is an integer or a vector integer type.
TypeSize getSizeInBits() const
Returns the size of the specified MVT in bits.
uint64_t getFixedSizeInBits() const
Return the size of the specified fixed width value type in bits.
ElementCount getVectorElementCount() const
TypeSize getStoreSize() const
Return the number of bytes overwritten by a store of the specified value type.
bool bitsGE(MVT VT) const
Return true if this has no less bits than VT.
bool isScalarInteger() const
Return true if this is an integer, not including vectors.
static MVT getVectorVT(MVT VT, unsigned NumElements)
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
static MVT getIntegerVT(unsigned BitWidth)
void normalizeSuccProbs()
Normalize probabilities of all successors so that the sum of them becomes one.
const BasicBlock * getBasicBlock() const
Return the LLVM basic block that this instance corresponded to originally.
LLVM_ABI void setSuccProbability(succ_iterator I, BranchProbability Prob)
Set successor probability of a given iterator.
succ_iterator succ_begin()
LLVM_ABI void addSuccessor(MachineBasicBlock *Succ, BranchProbability Prob=BranchProbability::getUnknown())
Add Succ as a successor of this MachineBasicBlock.
SmallVectorImpl< MachineBasicBlock * >::iterator succ_iterator
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
void setIsEHContTarget(bool V=true)
Indicates if this is a target of Windows EH Continuation Guard.
void setIsEHFuncletEntry(bool V=true)
Indicates if this is the entry block of an EH funclet.
MachineInstrBundleIterator< MachineInstr > iterator
void setIsEHScopeEntry(bool V=true)
Indicates if this is the entry block of an EH scope, i.e., the block that that used to have a catchpa...
void setMachineBlockAddressTaken()
Set this block to indicate that its address is used as something other than the target of a terminato...
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
void setIsImmutableObjectIndex(int ObjectIdx, bool IsImmutable)
Marks the immutability of an object.
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.
bool hasOpaqueSPAdjustment() const
Returns true if the function contains opaque dynamic stack adjustments.
int getStackProtectorIndex() const
Return the index for the stack protector object.
void setStackProtectorIndex(int I)
void setIsAliasedObjectIndex(int ObjectIdx, bool IsAliased)
Set "maybe pointed to by an LLVM IR value" for an object.
Align getObjectAlign(int ObjectIdx) const
Return the alignment of the specified stack object.
int64_t getObjectSize(int ObjectIdx) const
Return the size of the specified object.
void RemoveStackObject(int ObjectIdx)
Remove or mark dead a statically sized stack object.
void setFunctionContextIndex(int I)
const WinEHFuncInfo * getWinEHFuncInfo() const
getWinEHFuncInfo - Return information about how the current function uses Windows exception handling.
bool useDebugInstrRef() const
Returns true if the function's variable locations are tracked with instruction referencing.
void setCallSiteBeginLabel(MCSymbol *BeginLabel, unsigned Site)
Map the begin label for a call site.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineMemOperand * getMachineMemOperand(MachinePointerInfo PtrInfo, MachineMemOperand::Flags f, LLT MemTy, Align base_alignment, const AAMDNodes &AAInfo=AAMDNodes(), const MDNode *Ranges=nullptr, SyncScope::ID SSID=SyncScope::System, AtomicOrdering Ordering=AtomicOrdering::NotAtomic, AtomicOrdering FailureOrdering=AtomicOrdering::NotAtomic)
getMachineMemOperand - Allocate a new MachineMemOperand.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MCContext & getContext() const
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
void addCodeViewAnnotation(MCSymbol *Label, MDNode *MD)
Record annotations associated with a particular label.
Function & getFunction()
Return the LLVM function that this machine code represents.
BasicBlockListType::iterator iterator
bool hasEHFunclets() const
void setHasEHContTarget(bool V)
void addInvoke(MachineBasicBlock *LandingPad, MCSymbol *BeginLabel, MCSymbol *EndLabel)
Provide the begin and end labels of an invoke style call and associate it with a try landing pad bloc...
MachineBasicBlock * CreateMachineBasicBlock(const BasicBlock *BB=nullptr, std::optional< UniqueBBID > BBID=std::nullopt)
CreateMachineInstr - Allocate a new MachineInstr.
void insert(iterator MBBI, MachineBasicBlock *MBB)
const MachineInstrBuilder & addSym(MCSymbol *Sym, unsigned char TargetFlags=0) const
const MachineInstrBuilder & addFrameIndex(int Idx) const
Representation of each machine instruction.
A description of a memory reference used in the backend.
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.
@ MONonTemporal
The memory access is non-temporal.
@ MOInvariant
The memory access always returns the same value (or traps).
@ MOStore
The memory access writes data.
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
static MachineOperand CreateReg(Register Reg, bool isDef, bool isImp=false, bool isKill=false, bool isDead=false, bool isUndef=false, bool isEarlyClobber=false, unsigned SubReg=0, bool isDebug=false, bool isInternalRead=false, bool isRenamable=false)
static MachineOperand CreateFI(int Idx)
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
const TargetRegisterClass * getRegClass(Register Reg) const
Return the register class of the specified virtual register.
def_iterator def_begin(Register RegNo) const
LLVM_ABI Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
LLVM_ABI MCRegister getLiveInPhysReg(Register VReg) const
getLiveInPhysReg - If VReg is a live-in virtual register, return the corresponding live-in physical r...
An SDNode that represents everything that will be needed to construct a MachineInstr.
std::pair< iterator, bool > try_emplace(const KeyT &Key, Ts &&...Args)
bool contains(const KeyT &Key) const
static MemoryLocation getAfter(const Value *Ptr, const AAMDNodes &AATags=AAMDNodes())
Return a location that may access any location after Ptr, while remaining within the underlying objec...
A Module instance is used to store all the information related to an LLVM module.
static PointerType * getUnqual(Type *ElementType)
This constructs a pointer to an object of the specified type in the default address space (address sp...
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.
static LLVM_ABI PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified type.
Wrapper class representing virtual and physical registers.
constexpr bool isValid() const
constexpr bool isVirtual() const
Return true if the specified register number is in the virtual register namespace.
constexpr bool isPhysical() const
Return true if the specified register number is in the physical register namespace.
Resume the propagation of an exception.
Return a value (possibly void), from a function.
Holds the information from a dbg_label node through SDISel.
static SDDbgOperand fromNode(SDNode *Node, unsigned ResNo)
static SDDbgOperand fromFrameIdx(unsigned FrameIdx)
static SDDbgOperand fromVReg(Register VReg)
static SDDbgOperand fromConst(const Value *Const)
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...
Represents one node in the SelectionDAG.
unsigned getOpcode() const
Return the SelectionDAG opcode value for this node.
iterator_range< value_op_iterator > op_values() const
unsigned getIROrder() const
Return the node ordering.
uint64_t getAsZExtVal() const
Helper method returns the zero-extended integer value of a ConstantSDNode.
unsigned getNumValues() const
Return the number of values defined/returned by this operator.
const SDValue & getOperand(unsigned Num) const
EVT getValueType(unsigned ResNo) const
Return the type of a specified result.
Unlike LLVM values, Selection DAG nodes may return multiple values as the result of a computation.
SDNode * getNode() const
get the SDNode which holds the desired result
SDValue getValue(unsigned R) const
EVT getValueType() const
Return the ValueType of the referenced return value.
TypeSize getValueSizeInBits() const
Returns the size of the value in bits.
const SDValue & getOperand(unsigned i) const
unsigned getResNo() const
get the index which selects a specific result in the SDNode
MVT getSimpleValueType() const
Return the simple ValueType of the referenced return value.
unsigned getOpcode() const
SelectionDAGBuilder - This is the common target-independent lowering implementation that is parameter...
SDValue getValue(const Value *V)
getValue - Return an SDValue for the given Value.
bool shouldKeepJumpConditionsTogether(const FunctionLoweringInfo &FuncInfo, const CondBrInst &I, Instruction::BinaryOps Opc, const Value *Lhs, const Value *Rhs, TargetLoweringBase::CondMergingParams Params) const
DenseMap< const Constant *, Register > ConstantsOut
void addDanglingDebugInfo(SmallVectorImpl< Value * > &Values, DILocalVariable *Var, DIExpression *Expr, bool IsVariadic, DebugLoc DL, unsigned Order)
Register a dbg_value which relies on a Value which we have not yet seen.
void visitDbgInfo(const Instruction &I)
void clearDanglingDebugInfo()
Clear the dangling debug information map.
SDValue lowerStartEH(SDValue Chain, const BasicBlock *EHPadBB, MCSymbol *&BeginLabel)
void LowerCallTo(const CallBase &CB, SDValue Callee, bool IsTailCall, bool IsMustTailCall, const BasicBlock *EHPadBB=nullptr, const TargetLowering::PtrAuthInfo *PAI=nullptr)
void clear()
Clear out the current SelectionDAG and the associated state and prepare this SelectionDAGBuilder obje...
void visitBitTestHeader(SwitchCG::BitTestBlock &B, MachineBasicBlock *SwitchBB)
visitBitTestHeader - This function emits necessary code to produce value suitable for "bit tests"
void LowerStatepoint(const GCStatepointInst &I, const BasicBlock *EHPadBB=nullptr)
std::unique_ptr< SDAGSwitchLowering > SL
SDValue lowerRangeToAssertZExt(SelectionDAG &DAG, const Instruction &I, SDValue Op)
bool HasTailCall
This is set to true if a call in the current block has been translated as a tail call.
bool ShouldEmitAsBranches(const std::vector< SwitchCG::CaseBlock > &Cases)
If the set of cases should be emitted as a series of branches, return true.
void EmitBranchForMergedCondition(const Value *Cond, MachineBasicBlock *TBB, MachineBasicBlock *FBB, MachineBasicBlock *CurBB, MachineBasicBlock *SwitchBB, BranchProbability TProb, BranchProbability FProb, bool InvertCond)
EmitBranchForMergedCondition - Helper method for FindMergedConditions.
void LowerDeoptimizeCall(const CallInst *CI)
void LowerCallSiteWithDeoptBundle(const CallBase *Call, SDValue Callee, const BasicBlock *EHPadBB)
SwiftErrorValueTracking & SwiftError
Information about the swifterror values used throughout the function.
SDValue getNonRegisterValue(const Value *V)
getNonRegisterValue - Return an SDValue for the given Value, but don't look in FuncInfo....
const TargetTransformInfo * TTI
DenseMap< MachineBasicBlock *, SmallVector< unsigned, 4 > > LPadToCallSiteMap
Map a landing pad to the call site indexes.
SDValue lowerNoFPClassToAssertNoFPClass(SelectionDAG &DAG, const Instruction &I, SDValue Op)
void handleDebugDeclare(Value *Address, DILocalVariable *Variable, DIExpression *Expression, DebugLoc DL)
StatepointLoweringState StatepointLowering
State used while lowering a statepoint sequence (gc_statepoint, gc_relocate, and gc_result).
void visitBitTestCase(SwitchCG::BitTestBlock &BB, MachineBasicBlock *NextMBB, BranchProbability BranchProbToNext, Register Reg, SwitchCG::BitTestCase &B, MachineBasicBlock *SwitchBB)
visitBitTestCase - this function produces one "bit test"
bool canTailCall(const CallBase &CB) const
void populateCallLoweringInfo(TargetLowering::CallLoweringInfo &CLI, const CallBase *Call, unsigned ArgIdx, unsigned NumArgs, SDValue Callee, Type *ReturnTy, AttributeSet RetAttrs, bool IsPatchPoint)
Populate a CallLowerinInfo (into CLI) based on the properties of the call being lowered.
void CopyValueToVirtualRegister(const Value *V, Register Reg, ISD::NodeType ExtendType=ISD::ANY_EXTEND)
void salvageUnresolvedDbgValue(const Value *V, DanglingDebugInfo &DDI)
For the given dangling debuginfo record, perform last-ditch efforts to resolve the debuginfo to somet...
SmallVector< SDValue, 8 > PendingLoads
Loads are not emitted to the program immediately.
GCFunctionInfo * GFI
Garbage collection metadata for the function.
void init(GCFunctionInfo *gfi, BatchAAResults *BatchAA, AssumptionCache *AC, const TargetLibraryInfo *li, const TargetTransformInfo &TTI)
SDValue getRoot()
Similar to getMemoryRoot, but also flushes PendingConstrainedFP(Strict) items.
void ExportFromCurrentBlock(const Value *V)
ExportFromCurrentBlock - If this condition isn't known to be exported from the current basic block,...
DebugLoc getCurDebugLoc() const
void resolveOrClearDbgInfo()
Evict any dangling debug information, attempting to salvage it first.
std::pair< SDValue, SDValue > lowerInvokable(TargetLowering::CallLoweringInfo &CLI, const BasicBlock *EHPadBB=nullptr)
SDValue getMemoryRoot()
Return the current virtual root of the Selection DAG, flushing any PendingLoad items.
void resolveDanglingDebugInfo(const Value *V, SDValue Val)
If we saw an earlier dbg_value referring to V, generate the debug data structures now that we've seen...
SDLoc getCurSDLoc() const
void visit(const Instruction &I)
void dropDanglingDebugInfo(const DILocalVariable *Variable, const DIExpression *Expr)
If we have dangling debug info that describes Variable, or an overlapping part of variable considerin...
SDValue getCopyFromRegs(const Value *V, Type *Ty)
If there was virtual register allocated for the value V emit CopyFromReg of the specified type Ty.
void CopyToExportRegsIfNeeded(const Value *V)
CopyToExportRegsIfNeeded - If the given value has virtual registers created for it,...
void handleKillDebugValue(DILocalVariable *Var, DIExpression *Expr, DebugLoc DbgLoc, unsigned Order)
Create a record for a kill location debug intrinsic.
void visitJumpTable(SwitchCG::JumpTable &JT)
visitJumpTable - Emit JumpTable node in the current MBB
SDValue getFPOperationRoot(fp::ExceptionBehavior EB)
Return the current virtual root of the Selection DAG, flushing PendingConstrainedFP or PendingConstra...
void visitJumpTableHeader(SwitchCG::JumpTable &JT, SwitchCG::JumpTableHeader &JTH, MachineBasicBlock *SwitchBB)
visitJumpTableHeader - This function emits necessary code to produce index in the JumpTable from swit...
void LowerCallSiteWithPtrAuthBundle(const CallBase &CB, const BasicBlock *EHPadBB)
static const unsigned LowestSDNodeOrder
Lowest valid SDNodeOrder.
void LowerDeoptimizingReturn()
FunctionLoweringInfo & FuncInfo
Information about the function as a whole.
void setValue(const Value *V, SDValue NewN)
void FindMergedConditions(const Value *Cond, MachineBasicBlock *TBB, MachineBasicBlock *FBB, MachineBasicBlock *CurBB, MachineBasicBlock *SwitchBB, Instruction::BinaryOps Opc, BranchProbability TProb, BranchProbability FProb, bool InvertCond)
const TargetLibraryInfo * LibInfo
bool isExportableFromCurrentBlock(const Value *V, const BasicBlock *FromBB)
void visitSPDescriptorParent(StackProtectorDescriptor &SPD, MachineBasicBlock *ParentBB)
Codegen a new tail for a stack protector check ParentMBB which has had its tail spliced into a stack ...
bool handleDebugValue(ArrayRef< const Value * > Values, DILocalVariable *Var, DIExpression *Expr, DebugLoc DbgLoc, unsigned Order, bool IsVariadic)
For a given list of Values, attempt to create and record a SDDbgValue in the SelectionDAG.
SDValue getControlRoot()
Similar to getRoot, but instead of flushing all the PendingLoad items, flush all the PendingExports (...
void UpdateSplitBlock(MachineBasicBlock *First, MachineBasicBlock *Last)
When an MBB was split during scheduling, update the references that need to refer to the last resulti...
SDValue getValueImpl(const Value *V)
getValueImpl - Helper function for getValue and getNonRegisterValue.
void visitSwitchCase(SwitchCG::CaseBlock &CB, MachineBasicBlock *SwitchBB)
visitSwitchCase - Emits the necessary code to represent a single node in the binary search tree resul...
void visitSPDescriptorFailure(StackProtectorDescriptor &SPD)
Codegen the failure basic block for a stack protector check.
std::unique_ptr< FunctionLoweringInfo > FuncInfo
SmallPtrSet< const Instruction *, 4 > ElidedArgCopyInstrs
const TargetLowering * TLI
MachineRegisterInfo * RegInfo
std::unique_ptr< SwiftErrorValueTracking > SwiftError
virtual void emitFunctionEntryCode()
std::unique_ptr< SelectionDAGBuilder > SDB
virtual std::pair< SDValue, SDValue > EmitTargetCodeForMemccpy(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Dst, SDValue Src, SDValue C, SDValue Size, const CallInst *CI) const
Emit target-specific code that performs a memccpy, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrnlen(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, SDValue Src, SDValue MaxLength, MachinePointerInfo SrcPtrInfo) const
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrlen(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, SDValue Src, const CallInst *CI) const
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrstr(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Op1, SDValue Op2, const CallInst *CI) const
Emit target-specific code that performs a strstr, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForMemchr(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Src, SDValue Char, SDValue Length, MachinePointerInfo SrcPtrInfo) const
Emit target-specific code that performs a memchr, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrcmp(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Op1, SDValue Op2, MachinePointerInfo Op1PtrInfo, MachinePointerInfo Op2PtrInfo, const CallInst *CI) const
Emit target-specific code that performs a strcmp, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForMemcmp(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Op1, SDValue Op2, SDValue Op3, const CallInst *CI) const
Emit target-specific code that performs a memcmp/bcmp, in cases where that is faster than a libcall.
virtual SDValue EmitTargetCodeForSetTag(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Addr, SDValue Size, MachinePointerInfo DstPtrInfo, bool ZeroData) const
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrcpy(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, SDValue Dest, SDValue Src, MachinePointerInfo DestPtrInfo, MachinePointerInfo SrcPtrInfo, bool isStpcpy, const CallInst *CI) const
Emit target-specific code that performs a strcpy or stpcpy, in cases where that is faster than a libc...
This is used to represent a portion of an LLVM function in a low-level Data Dependence DAG representa...
SDValue getTargetGlobalAddress(const GlobalValue *GV, const SDLoc &DL, EVT VT, int64_t offset=0, unsigned TargetFlags=0)
SDValue getExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT, unsigned Opcode)
Convert Op, which must be of integer type, to the integer type VT, by either any/sign/zero-extending ...
const SDValue & getRoot() const
Return the root tag of the SelectionDAG.
const TargetSubtargetInfo & getSubtarget() const
SDValue getCopyToReg(SDValue Chain, const SDLoc &dl, Register Reg, SDValue N)
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 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 getConstantPool(const Constant *C, EVT VT, MaybeAlign Align=std::nullopt, int Offs=0, bool isT=false, unsigned TargetFlags=0)
LLVM_ABI SDValue getConstantFP(double Val, const SDLoc &DL, EVT VT, bool isTarget=false)
Create a ConstantFPSDNode wrapping a constant value.
LLVM_ABI SDValue getRegister(Register Reg, EVT VT)
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 Align getEVTAlign(EVT MemoryVT) const
Compute the default alignment value for the given type.
LLVM_ABI bool shouldOptForSize() const
const TargetLowering & getTargetLoweringInfo() const
static constexpr unsigned MaxRecursionDepth
LLVM_ABI void AddDbgValue(SDDbgValue *DB, bool isParameter)
Add a dbg_value SDNode.
SDValue getUNDEF(EVT VT)
Return an UNDEF node. UNDEF does not have a useful SDLoc.
SDValue getBuildVector(EVT VT, const SDLoc &DL, ArrayRef< SDValue > Ops)
Return an ISD::BUILD_VECTOR node.
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.
SDValue getCopyFromReg(SDValue Chain, const SDLoc &dl, Register Reg, EVT VT)
LLVM_ABI void setNodeMemRefs(MachineSDNode *N, ArrayRef< MachineMemOperand * > NewMemRefs)
Mutate the specified machine node's memory references to the provided list.
const DataLayout & getDataLayout() const
SDValue getTargetFrameIndex(int FI, EVT VT)
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 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 getMDNode(const MDNode *MD)
Return an MDNodeSDNode which holds an MDNode.
LLVM_ABI SDValue getBasicBlock(MachineBasicBlock *MBB)
LLVM_ABI SDValue getEHLabel(const SDLoc &dl, SDValue Root, MCSymbol *Label)
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 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...
const LibcallLoweringInfo & getLibcalls() const
LLVM_ABI SDValue getIntPtrConstant(uint64_t Val, const SDLoc &DL, bool isTarget=false)
LLVM_ABI SDValue getValueType(EVT)
LLVM_ABI SDValue getNode(unsigned Opcode, const SDLoc &DL, EVT VT, ArrayRef< SDUse > Ops)
Gets or creates the specified node.
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...
SDValue getTargetConstant(uint64_t Val, const SDLoc &DL, EVT VT, bool isOpaque=false)
LLVM_ABI SDValue getVectorIdxConstant(uint64_t Val, const SDLoc &DL, bool isTarget=false)
MachineFunction & getMachineFunction() const
LLVM_ABI SDValue getFrameIndex(int FI, EVT VT, bool isTarget=false)
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...
LLVMContext * getContext() const
const SDValue & setRoot(SDValue N)
Set the current root tag of the SelectionDAG.
LLVM_ABI SDValue getTargetExternalSymbol(const char *Sym, EVT VT, unsigned TargetFlags=0)
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
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...
reference emplace_back(ArgTypes &&... Args)
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
void swap(SmallVectorImpl &RHS)
void push_back(const T &Elt)
pointer data()
Return a pointer to the vector's buffer, even if empty().
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Encapsulates all of the information needed to generate a stack protector check, and signals to isel w...
MachineBasicBlock * getSuccessMBB()
MachineBasicBlock * getFailureMBB()
MachineBasicBlock * getParentMBB()
bool shouldEmitFunctionBasedCheckStackProtector() const
An instruction for storing to memory.
Represent a constant reference to a string, i.e.
constexpr bool empty() const
Check if the string is empty.
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.
Provides information about what library functions are available for the current target.
virtual Align getByValTypeAlignment(Type *Ty, const DataLayout &DL) const
Returns the desired alignment for ByVal or InAlloca aggregate function arguments in the caller parame...
virtual bool isFMAFasterThanFMulAndFAdd(const MachineFunction &MF, EVT) const
Return true if an FMA operation is faster than a pair of fmul and fadd instructions.
EVT getMemValueType(const DataLayout &DL, Type *Ty, bool AllowUnknown=false) const
Function * getSSPStackGuardCheck(const Module &M, const LibcallLoweringInfo &Libcalls) const
If the target has a standard stack protection check function that performs validation and error handl...
EVT getValueType(const DataLayout &DL, Type *Ty, bool AllowUnknown=false) const
Return the EVT corresponding to this LLVM type.
LegalizeAction
This enum indicates whether operations are valid for a target, and if not, what action should be used...
virtual bool useStackGuardXorFP() const
If this function returns true, stack protection checks should XOR the frame pointer (or whichever poi...
virtual const TargetRegisterClass * getRegClassFor(MVT VT, bool isDivergent=false) const
Return the register class that should be used for the specified value type.
virtual bool isLegalScaleForGatherScatter(uint64_t Scale, uint64_t ElemSize) const
virtual bool isSExtCheaperThanZExt(EVT FromTy, EVT ToTy) const
Return true if sign-extension from FromTy to ToTy is cheaper than zero-extension.
MVT getVectorIdxTy(const DataLayout &DL) const
Returns the type to be used for the index operand of: ISD::INSERT_VECTOR_ELT, ISD::EXTRACT_VECTOR_ELT...
virtual unsigned getNumRegistersForCallingConv(LLVMContext &Context, CallingConv::ID CC, EVT VT) const
Certain targets require unusual breakdowns of certain types.
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...
virtual MVT getRegisterTypeForCallingConv(LLVMContext &Context, CallingConv::ID CC, EVT VT) const
Certain combinations of ABIs, Targets and features require that types are legal for some operations a...
virtual unsigned getNumRegisters(LLVMContext &Context, EVT VT, std::optional< MVT > RegisterVT=std::nullopt) const
Return the number of registers that this ValueType will eventually require.
MachineMemOperand::Flags getLoadMemOperandFlags(const LoadInst &LI, const DataLayout &DL, AssumptionCache *AC=nullptr, const TargetLibraryInfo *LibInfo=nullptr, CodeGenOptLevel OptLevel=CodeGenOptLevel::Default) const
virtual bool shouldExtendGSIndex(EVT VT, EVT &EltTy) const
Returns true if the index type for a masked gather/scatter requires extending.
virtual unsigned getVectorTypeBreakdownForCallingConv(LLVMContext &Context, CallingConv::ID CC, EVT VT, EVT &IntermediateVT, unsigned &NumIntermediates, MVT &RegisterVT) const
Certain targets such as MIPS require that some types such as vectors are always broken down into scal...
Register getStackPointerRegisterToSaveRestore() const
If a physical register, this specifies the register that llvm.savestack/llvm.restorestack should save...
LegalizeAction getFixedPointOperationAction(unsigned Op, EVT VT, unsigned Scale) const
Some fixed point operations may be natively supported by the target but only for specific scales.
MachineMemOperand::Flags getAtomicMemOperandFlags(const Instruction &AI, const DataLayout &DL) const
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.
bool isOperationCustom(unsigned Op, EVT VT) const
Return true if the operation uses custom lowering, regardless of whether the type is legal or not.
bool hasBigEndianPartOrdering(EVT VT, const DataLayout &DL) const
When splitting a value of the specified type into parts, does the Lo or Hi part come first?
EVT getShiftAmountTy(EVT LHSTy, const DataLayout &DL) const
Returns the type for the shift amount of a shift opcode.
virtual Align getABIAlignmentForCallingConv(Type *ArgTy, const DataLayout &DL) const
Certain targets have context sensitive alignment requirements, where one type has the alignment requi...
MachineMemOperand::Flags getVPIntrinsicMemOperandFlags(const VPIntrinsic &VPIntrin) const
virtual bool shouldExpandGetActiveLaneMask(EVT VT, EVT OpVT) const
Return true if the @llvm.get.active.lane.mask intrinsic should be expanded using generic code in Sele...
virtual EVT getSetCCResultType(const DataLayout &DL, LLVMContext &Context, EVT VT) const
Return the ValueType of the result of SETCC operations.
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.
MVT getProgramPointerTy(const DataLayout &DL) const
Return the type for code pointers, which is determined by the program address space specified through...
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...
bool isOperationLegal(unsigned Op, EVT VT) const
Return true if the specified operation is legal on this target.
virtual bool shouldExpandVectorMatch(EVT VT, unsigned SearchSize) const
Return true if the @llvm.experimental.vector.match intrinsic should be expanded for vector type ‘VT’ ...
virtual bool isProfitableToCombineMinNumMaxNum(EVT VT) const
virtual MVT getFenceOperandTy(const DataLayout &DL) const
Return the type for operands of fence.
virtual bool shouldExpandGetVectorLength(EVT CountVT, unsigned VF, bool IsScalable) const
bool isOperationLegalOrCustom(unsigned Op, EVT VT, bool LegalOnly=false) const
Return true if the specified operation is legal on this target or can be made legal with custom lower...
virtual MVT hasFastEqualityCompare(unsigned NumBits) const
Return the preferred operand type if the target has a quick way to compare integer values of the give...
MachineMemOperand::Flags getStoreMemOperandFlags(const StoreInst &SI, const DataLayout &DL) const
virtual void getTgtMemIntrinsic(SmallVectorImpl< IntrinsicInfo > &Infos, const CallBase &I, MachineFunction &MF, unsigned Intrinsic) const
Given an intrinsic, checks if on the target the intrinsic will need to map to a MemIntrinsicNode (tou...
virtual bool signExtendConstant(const ConstantInt *C) const
Return true if this constant should be sign extended when promoting to a larger type.
virtual Value * getSDagStackGuard(const Module &M, const LibcallLoweringInfo &Libcalls) const
Return the variable that's previously inserted by insertSSPDeclarations, if any, otherwise return nul...
LegalizeTypeAction getTypeAction(LLVMContext &Context, EVT VT) const
Return how we should legalize values of this type, either it is already legal (return 'Legal') or we ...
virtual Register getExceptionPointerRegister(const Constant *PersonalityFn) const
If a physical register, this returns the register that receives the exception address on entry to an ...
bool supportsUnalignedAtomics() const
Whether the target supports unaligned atomic operations.
std::vector< ArgListEntry > ArgListTy
bool isBeneficialToExpandPowI(int64_t Exponent, bool OptForSize) const
Return true if it is beneficial to expand an @llvm.powi.
MVT getFrameIndexTy(const DataLayout &DL) const
Return the type for frame index, which is determined by the alloca address space specified through th...
virtual Register getExceptionSelectorRegister(const Constant *PersonalityFn) const
If a physical register, this returns the register that receives the exception typeid on entry to a la...
virtual MVT getPointerMemTy(const DataLayout &DL, uint32_t AS=0) const
Return the in-memory pointer type for the given address space, defaults to the pointer type from the ...
MVT getRegisterType(MVT VT) const
Return the type of registers that this ValueType will eventually require.
unsigned getVectorTypeBreakdown(LLVMContext &Context, EVT VT, EVT &IntermediateVT, unsigned &NumIntermediates, MVT &RegisterVT) const
Vector types are broken down into some number of legal first class types.
virtual MVT getVPExplicitVectorLengthTy() const
Returns the type to be used for the EVL/AVL operand of VP nodes: ISD::VP_ADD, ISD::VP_SUB,...
This class defines information used to lower LLVM code to legal SelectionDAG operators that the targe...
virtual bool supportKCFIBundles() const
Return true if the target supports kcfi operand bundles.
virtual bool supportPtrAuthBundles() const
Return true if the target supports ptrauth operand bundles.
virtual bool supportSwiftError() const
Return true if the target supports swifterror attribute.
virtual SDValue visitMaskedLoad(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, MachineMemOperand *MMO, SDValue &NewLoad, SDValue Ptr, SDValue PassThru, SDValue Mask) const
virtual SDValue emitStackGuardXorFP(SelectionDAG &DAG, SDValue Val, const SDLoc &DL) const
virtual EVT getTypeForExtReturn(LLVMContext &Context, EVT VT, ISD::NodeType) const
Return the type that should be used to zero or sign extend a zeroext/signext integer return value.
virtual Register getRegisterByName(const char *RegName, LLT Ty, const MachineFunction &MF) const
Return the register ID of the name passed in.
virtual InlineAsm::ConstraintCode getInlineAsmMemConstraint(StringRef ConstraintCode) const
std::vector< AsmOperandInfo > AsmOperandInfoVector
SDValue expandIS_FPCLASS(EVT ResultVT, SDValue Op, FPClassTest Test, SDNodeFlags Flags, const SDLoc &DL, SelectionDAG &DAG) const
Expand check for floating point class.
virtual SDValue prepareVolatileOrAtomicLoad(SDValue Chain, const SDLoc &DL, SelectionDAG &DAG) const
This callback is used to prepare for a volatile or atomic load.
virtual ConstraintType getConstraintType(StringRef Constraint) const
Given a constraint, return the type of constraint it is for this target.
virtual bool splitValueIntoRegisterParts(SelectionDAG &DAG, const SDLoc &DL, SDValue Val, SDValue *Parts, unsigned NumParts, MVT PartVT, std::optional< CallingConv::ID > CC) const
Target-specific splitting of values into parts that fit a register storing a legal type.
virtual SDValue joinRegisterPartsIntoValue(SelectionDAG &DAG, const SDLoc &DL, const SDValue *Parts, unsigned NumParts, MVT PartVT, EVT ValueVT, std::optional< CallingConv::ID > CC) const
Target-specific combining of register parts into its original value.
virtual SDValue LowerCall(CallLoweringInfo &, SmallVectorImpl< SDValue > &) const
This hook must be implemented to lower calls into the specified DAG.
std::pair< SDValue, SDValue > LowerCallTo(CallLoweringInfo &CLI) const
This function lowers an abstract call to a function into an actual call.
virtual SDValue LowerAsmOutputForConstraint(SDValue &Chain, SDValue &Glue, const SDLoc &DL, const AsmOperandInfo &OpInfo, SelectionDAG &DAG) const
virtual std::pair< unsigned, const TargetRegisterClass * > getRegForInlineAsmConstraint(const TargetRegisterInfo *TRI, StringRef Constraint, MVT VT) const
Given a physical register constraint (e.g.
virtual AsmOperandInfoVector ParseConstraints(const DataLayout &DL, const TargetRegisterInfo *TRI, const CallBase &Call) const
Split up the constraint string from the inline assembly value into the specific constraints and their...
virtual SDValue LowerOperation(SDValue Op, SelectionDAG &DAG) const
This callback is invoked for operations that are unsupported by the target, which are registered to u...
virtual bool functionArgumentNeedsConsecutiveRegisters(Type *Ty, CallingConv::ID CallConv, bool isVarArg, const DataLayout &DL) const
For some targets, an LLVM struct type must be broken down into multiple simple types,...
virtual void ComputeConstraintToUse(AsmOperandInfo &OpInfo, SDValue Op, SelectionDAG *DAG=nullptr) const
Determines the constraint code and constraint type to use for the specific AsmOperandInfo,...
virtual void CollectTargetIntrinsicOperands(const CallInst &I, SmallVectorImpl< SDValue > &Ops, SelectionDAG &DAG) const
virtual SDValue visitMaskedStore(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, MachineMemOperand *MMO, SDValue Ptr, SDValue Val, SDValue Mask) const
virtual bool useLoadStackGuardNode(const Module &M) const
If this function returns true, SelectionDAGBuilder emits a LOAD_STACK_GUARD node when it is lowering ...
virtual void LowerAsmOperandForConstraint(SDValue Op, StringRef Constraint, std::vector< SDValue > &Ops, SelectionDAG &DAG) const
Lower the specified operand into the Ops vector.
std::pair< SDValue, SDValue > makeLibCall(SelectionDAG &DAG, RTLIB::LibcallImpl LibcallImpl, EVT RetVT, ArrayRef< SDValue > Ops, MakeLibCallOptions CallOptions, const SDLoc &dl, SDValue Chain=SDValue()) const
Returns a pair of (return value, chain).
virtual void LowerOperationWrapper(SDNode *N, SmallVectorImpl< SDValue > &Results, SelectionDAG &DAG) const
This callback is invoked by the type legalizer to legalize nodes with an illegal operand type but leg...
virtual bool isInlineAsmTargetBranch(const SmallVectorImpl< StringRef > &AsmStrs, unsigned OpNo) const
On x86, return true if the operand with index OpNo is a CALL or JUMP instruction, which can use eithe...
virtual MVT getJumpTableRegTy(const DataLayout &DL) const
virtual bool CanLowerReturn(CallingConv::ID, MachineFunction &, bool, const SmallVectorImpl< ISD::OutputArg > &, LLVMContext &, const Type *RetTy) const
This hook should be implemented to check whether the return values described by the Outs array can fi...
Primary interface to the complete machine description for the target machine.
CodeGenOptLevel getOptLevel() const
Returns the optimization level: None, Less, Default, or Aggressive.
CodeModel::Model getCodeModel() const
Returns the code model.
unsigned NoTrapAfterNoreturn
Do not emit a trap instruction for 'unreachable' IR instructions behind noreturn calls,...
unsigned TrapUnreachable
Emit target-specific trap instruction for 'unreachable' IR instructions.
unsigned getID() const
Return the register class ID number.
const MCPhysReg * iterator
iterator begin() const
begin/end - Return all of the registers in this class.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
virtual const TargetFrameLowering * getFrameLowering() const
virtual const TargetRegisterInfo * getRegisterInfo() const =0
Return the target's register information.
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.
LLVM_ABI bool isEmptyTy() const
Return true if this type is empty, that is, it has no elements or all of its elements are empty.
bool isVectorTy() const
True if this is an instance of VectorType.
bool isPointerTy() const
True if this is an instance of PointerType.
LLVM_ABI unsigned getPointerAddressSpace() const
Get the address space of this pointer or pointer vector type.
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
Type * getScalarType() const
If this is a vector type, return the element type, otherwise return 'this'.
LLVMContext & getContext() const
Return the LLVMContext in which this type was uniqued.
static LLVM_ABI IntegerType * getInt1Ty(LLVMContext &C)
bool isIntegerTy() const
True if this is an instance of IntegerType.
bool isTokenTy() const
Return true if this is 'token'.
static LLVM_ABI IntegerType * getIntNTy(LLVMContext &C, unsigned N)
bool isFPOrFPVectorTy() const
Return true if this is a FP type or a vector of FP.
bool isVoidTy() const
Return true if this is 'void'.
Unconditional Branch instruction.
This function has undefined behavior.
A Use represents the edge between a Value definition and its users.
Value * getOperand(unsigned i) const
unsigned getNumOperands() const
This class represents the va_arg llvm instruction, which returns an argument of the specified type gi...
LLVM_ABI CmpInst::Predicate getPredicate() const
This is the common base class for vector predication intrinsics.
static LLVM_ABI std::optional< unsigned > getVectorLengthParamPos(Intrinsic::ID IntrinsicID)
LLVM_ABI MaybeAlign getPointerAlignment() const
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
bool hasOneUse() const
Return true if there is exactly one use of this value.
LLVMContext & getContext() const
All values hold a context through their type.
iterator_range< user_iterator > users()
LLVM_ABI const Value * stripPointerCasts() const
Strip off pointer casts, all-zero GEPs and address space casts.
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
Base class of all SIMD vector types.
Type * getElementType() const
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 ScalarTy getKnownMinValue() const
Returns the minimum value this quantity can represent.
const ParentTy * getParent() const
A raw_ostream that writes to an std::string.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
constexpr char Align[]
Key for Kernel::Arg::Metadata::mAlign.
constexpr char Args[]
Key for Kernel::Metadata::mArgs.
constexpr char SymbolName[]
Key for Kernel::Metadata::mSymbolName.
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.
@ AnyReg
OBSOLETED - Used for stack based JavaScript calls.
@ AMDGPU_CS_Chain
Used on AMDGPUs to give the middle-end more control over argument placement.
@ X86_VectorCall
MSVC calling convention that passes vectors and vector aggregates in SSE registers.
@ C
The default llvm calling convention, compatible with C.
NodeType
ISD::NodeType enum - This enum defines the target-independent operators for a SelectionDAG.
@ MERGE_VALUES
MERGE_VALUES - This node takes multiple discrete operands and returns them all as its individual resu...
@ STACKRESTORE
STACKRESTORE has two operands, an input chain and a pointer to restore to it returns an output chain.
@ STACKSAVE
STACKSAVE - STACKSAVE has one operand, an input chain.
@ CONVERGENCECTRL_ANCHOR
The llvm.experimental.convergence.* intrinsics.
@ STRICT_FSETCC
STRICT_FSETCC/STRICT_FSETCCS - Constrained versions of SETCC, used for floating-point operands only.
@ DELETED_NODE
DELETED_NODE - This is an illegal value that is used to catch errors.
@ SET_FPENV
Sets the current floating-point environment.
@ ATOMIC_LOAD_FMINIMUMNUM
@ LOOP_DEPENDENCE_RAW_MASK
@ VECREDUCE_SEQ_FADD
Generic reduction nodes.
@ COND_LOOP
COND_LOOP is a conditional branch to self, used for implementing efficient conditional traps.
@ EH_SJLJ_LONGJMP
OUTCHAIN = EH_SJLJ_LONGJMP(INCHAIN, buffer) This corresponds to the eh.sjlj.longjmp intrinsic.
@ INSERT_SUBVECTOR
INSERT_SUBVECTOR(VECTOR1, VECTOR2, IDX) - Returns a vector with VECTOR2 inserted into VECTOR1.
@ STACKADDRESS
STACKADDRESS - Represents the llvm.stackaddress intrinsic.
@ BSWAP
Byte Swap and Counting operators.
@ SMULFIX
RESULT = [US]MULFIX(LHS, RHS, SCALE) - Perform fixed point multiplication on 2 integers with the same...
@ VAEND
VAEND, VASTART - VAEND and VASTART have three operands: an input chain, pointer, and a SRCVALUE.
@ ATOMIC_STORE
OUTCHAIN = ATOMIC_STORE(INCHAIN, val, ptr) This corresponds to "store atomic" instruction.
@ RESET_FPENV
Set floating-point environment to default state.
@ ADD
Simple integer binary arithmetic operators.
@ SMULFIXSAT
Same as the corresponding unsaturated fixed point instructions, but the result is clamped between the...
@ SET_FPMODE
Sets the current dynamic floating-point control modes.
@ ANY_EXTEND
ANY_EXTEND - Used for integer types. The high bits are undefined.
@ CTTZ_ELTS
Returns the number of number of trailing (least significant) zero elements in a vector.
@ FMA
FMA - Perform a * b + c with no intermediate rounding step.
@ VECTOR_FIND_LAST_ACTIVE
Finds the index of the last active mask element Operands: Mask.
@ FMODF
FMODF - Decomposes the operand into integral and fractional parts, each having the same type and sign...
@ FATAN2
FATAN2 - atan2, inspired by libm.
@ FSINCOSPI
FSINCOSPI - Compute both the sine and cosine times pi more accurately than FSINCOS(pi*x),...
@ INTRINSIC_VOID
OUTCHAIN = INTRINSIC_VOID(INCHAIN, INTRINSICID, arg1, arg2, ...) This node represents a target intrin...
@ EH_SJLJ_SETUP_DISPATCH
OUTCHAIN = EH_SJLJ_SETUP_DISPATCH(INCHAIN) The target initializes the dispatch table here.
@ 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.
@ ATOMIC_FENCE
OUTCHAIN = ATOMIC_FENCE(INCHAIN, ordering, scope) This corresponds to the fence instruction.
@ RESET_FPMODE
Sets default dynamic floating-point control modes.
@ FMULADD
FMULADD - Performs a * b + c, with, or without, intermediate rounding.
@ FPTRUNC_ROUND
FPTRUNC_ROUND - This corresponds to the fptrunc_round intrinsic.
@ FAKE_USE
FAKE_USE represents a use of the operand but does not do anything.
@ 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.
@ INIT_TRAMPOLINE
INIT_TRAMPOLINE - This corresponds to the init_trampoline intrinsic.
@ FLDEXP
FLDEXP - ldexp, inspired by libm (op0 * 2**op1).
@ SDIVFIX
RESULT = [US]DIVFIX(LHS, RHS, SCALE) - Perform fixed point division on 2 integers with the same width...
@ CONVERT_FROM_ARBITRARY_FP
CONVERT_FROM_ARBITRARY_FP - This operator converts from an arbitrary floating-point represented as an...
@ EH_LABEL
EH_LABEL - Represents a label in mid basic block used to track locations needed for debug and excepti...
@ EH_RETURN
OUTCHAIN = EH_RETURN(INCHAIN, OFFSET, HANDLER) - This node represents 'eh_return' gcc dwarf builtin,...
@ ANNOTATION_LABEL
ANNOTATION_LABEL - Represents a mid basic block label used by annotations.
@ SET_ROUNDING
Set rounding mode.
@ CONVERGENCECTRL_GLUE
This does not correspond to any convergence control intrinsic.
@ SIGN_EXTEND
Conversion operators.
@ PREALLOCATED_SETUP
PREALLOCATED_SETUP - This has 2 operands: an input chain and a SRCVALUE with the preallocated call Va...
@ READSTEADYCOUNTER
READSTEADYCOUNTER - This corresponds to the readfixedcounter intrinsic.
@ ADDROFRETURNADDR
ADDROFRETURNADDR - Represents the llvm.addressofreturnaddress intrinsic.
@ BR
Control flow instructions. These all have token chains.
@ VECREDUCE_FADD
These reductions have relaxed evaluation order semantics, and have a single vector operand.
@ PREFETCH
PREFETCH - This corresponds to a prefetch intrinsic.
@ FSINCOS
FSINCOS - Compute both fsin and fcos as a single operation.
@ SSUBO
Same for subtraction.
@ PREALLOCATED_ARG
PREALLOCATED_ARG - This has 3 operands: an input chain, a SRCVALUE with the preallocated call Value,...
@ BRIND
BRIND - Indirect branch.
@ BR_JT
BR_JT - Jumptable branch.
@ VECTOR_INTERLEAVE
VECTOR_INTERLEAVE(VEC1, VEC2, ...) - Returns N vectors from N input vectors, where N is the factor to...
@ 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).
@ 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.
@ VACOPY
VACOPY - VACOPY has 5 operands: an input chain, a destination pointer, a source pointer,...
@ 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.
@ ARITH_FENCE
ARITH_FENCE - This corresponds to a arithmetic fence intrinsic.
@ VECREDUCE_ADD
Integer reductions may have a result type larger than the vector element type.
@ GET_ROUNDING
Returns current rounding mode: -1 Undefined 0 Round to 0 1 Round to nearest, ties to even 2 Round to ...
@ CLEANUPRET
CLEANUPRET - Represents a return from a cleanup block funclet.
@ GET_FPMODE
Reads the current dynamic floating-point control modes.
@ GET_FPENV
Gets the current floating-point environment.
@ SHL
Shift and rotation operations.
@ AssertNoFPClass
AssertNoFPClass - These nodes record if a register contains a float value that is known to be not som...
@ PtrAuthGlobalAddress
A ptrauth constant.
@ EXTRACT_SUBVECTOR
EXTRACT_SUBVECTOR(VECTOR, IDX) - Returns a subvector from VECTOR.
@ EntryToken
EntryToken - This is the marker used to indicate the start of a region.
@ READ_REGISTER
READ_REGISTER, WRITE_REGISTER - This node represents llvm.register on the DAG, which implements the n...
@ EXTRACT_VECTOR_ELT
EXTRACT_VECTOR_ELT(VECTOR, IDX) - Returns a single element from VECTOR identified by the (potentially...
@ ZERO_EXTEND
ZERO_EXTEND - Used for integer types, zeroing the new bits.
@ DEBUGTRAP
DEBUGTRAP - Trap intended to get the attention of a debugger.
@ VSCALE
VSCALE(IMM) - Returns the runtime scaling factor used to calculate the number of elements within a sc...
@ LOCAL_RECOVER
LOCAL_RECOVER - Represents the llvm.localrecover intrinsic.
@ FMINNUM
FMINNUM/FMAXNUM - Perform floating-point minimum maximum on two values, following IEEE-754 definition...
@ UBSANTRAP
UBSANTRAP - Trap with an immediate describing the kind of sanitizer failure.
@ SSHLSAT
RESULT = [US]SHLSAT(LHS, RHS) - Perform saturation left shift.
@ PATCHPOINT
The llvm.experimental.patchpoint.
@ SMULO
Same for multiplication.
@ DYNAMIC_STACKALLOC
DYNAMIC_STACKALLOC - Allocate some number of bytes on the stack aligned to a specified boundary.
@ VECTOR_SPLICE_LEFT
VECTOR_SPLICE_LEFT(VEC1, VEC2, OFFSET) - Shifts CONCAT_VECTORS(VEC1, VEC2) left by OFFSET elements an...
@ SMIN
[US]{MIN/MAX} - Binary minimum or maximum of signed or unsigned integers.
@ MASKED_UDIV
Masked vector arithmetic that returns poison on disabled lanes.
@ VECTOR_REVERSE
VECTOR_REVERSE(VECTOR) - Returns a vector, of the same type as VECTOR, whose elements are shuffled us...
@ SDIVFIXSAT
Same as the corresponding unsaturated fixed point instructions, but the result is clamped between the...
@ FP_EXTEND
X = FP_EXTEND(Y) - Extend a smaller FP type into a larger FP type.
@ VSELECT
Select with a vector condition (op #0) and two vector operands (ops #1 and #2), returning a vector re...
@ PCMARKER
PCMARKER - This corresponds to the pcmarker intrinsic.
@ INLINEASM_BR
INLINEASM_BR - Branching version of inline asm. Used by asm-goto.
@ ATOMIC_LOAD_FMAXIMUMNUM
@ EH_DWARF_CFA
EH_DWARF_CFA - This node represents the pointer to the DWARF Canonical Frame Address (CFA),...
@ FRAMEADDR
FRAMEADDR, RETURNADDR - These nodes represent llvm.frameaddress and llvm.returnaddress on the DAG.
@ 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.
@ READCYCLECOUNTER
READCYCLECOUNTER - This corresponds to the readcyclecounter intrinsic.
@ RELOC_NONE
Issue a no-op relocation against a given symbol at the current location.
@ AND
Bitwise operators - logical and, logical or, logical xor.
@ TRAP
TRAP - Trapping instruction.
@ INTRINSIC_WO_CHAIN
RESULT = INTRINSIC_WO_CHAIN(INTRINSICID, arg1, arg2, ...) This node represents a target intrinsic fun...
@ SCMP
[US]CMP - 3-way comparison of signed or unsigned integers.
@ VECTOR_SPLICE_RIGHT
VECTOR_SPLICE_RIGHT(VEC1, VEC2, OFFSET) - Shifts CONCAT_VECTORS(VEC1,VEC2) right by OFFSET elements a...
@ STRICT_FADD
Constrained versions of the binary floating point operators.
@ STACKMAP
The llvm.experimental.stackmap intrinsic.
@ 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....
@ SPONENTRY
SPONENTRY - Represents the llvm.sponentry intrinsic.
@ CLEAR_CACHE
llvm.clear_cache intrinsic Operands: Input Chain, Start Addres, End Address Outputs: Output Chain
@ INLINEASM
INLINEASM - Represents an inline asm block.
@ FP_TO_SINT_SAT
FP_TO_[US]INT_SAT - Convert floating point value in operand 0 to a signed or unsigned scalar integer ...
@ EH_SJLJ_SETJMP
RESULT, OUTCHAIN = EH_SJLJ_SETJMP(INCHAIN, buffer) This corresponds to the eh.sjlj....
@ TRUNCATE
TRUNCATE - Completely drop the high bits.
@ BRCOND
BRCOND - Conditional branch.
@ CONVERT_TO_ARBITRARY_FP
CONVERT_TO_ARBITRARY_FP - Converts a native FP value to an arbitrary floating-point format,...
@ CATCHRET
CATCHRET - Represents a return from a catch block funclet.
@ 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)...
@ VECTOR_DEINTERLEAVE
VECTOR_DEINTERLEAVE(VEC1, VEC2, ...) - Returns N vectors from N input vectors, where N is the factor ...
@ GET_DYNAMIC_AREA_OFFSET
GET_DYNAMIC_AREA_OFFSET - get offset from native SP to the address of the most recent dynamic alloca.
@ FMINIMUMNUM
FMINIMUMNUM/FMAXIMUMNUM - minimumnum/maximumnum that is same with FMINNUM_IEEE and FMAXNUM_IEEE besid...
@ ADJUST_TRAMPOLINE
ADJUST_TRAMPOLINE - This corresponds to the adjust_trampoline intrinsic.
@ 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,...
@ LOOP_DEPENDENCE_WAR_MASK
The llvm.loop.dependence.
CondCode
ISD::CondCode enum - These are ordered carefully to make the bitfields below work out,...
This namespace contains an enum with a value for every intrinsic/builtin function known by LLVM.
Flag
These should be considered private to the implementation of the MCInstrDesc class.
BinaryOp_match< SrcTy, SpecificConstantMatch, TargetOpcode::G_XOR, true > m_Not(const SrcTy &&Src)
Matches a register not-ed by a G_XOR.
OneUse_match< SubPat > m_OneUse(const SubPat &SP)
bool match(Val *V, const Pattern &P)
specificval_ty m_Specific(const Value *V)
Match if we have a specific specified value.
TwoOps_match< Val_t, Idx_t, Instruction::ExtractElement > m_ExtractElt(const Val_t &Val, const Idx_t &Idx)
Matches ExtractElementInst.
IntrinsicID_match m_VScale()
Matches a call to llvm.vscale().
auto m_Value()
Match an arbitrary value and ignore it.
auto m_LogicalOr()
Matches L || R where L and R are arbitrary values.
auto m_LogicalAnd()
Matches L && R where L and R are arbitrary values.
std::pair< JumpTableHeader, JumpTable > JumpTableBlock
LLVM_ABI void sortAndRangeify(CaseClusterVector &Clusters)
Sort Clusters and merge adjacent cases.
std::vector< CaseCluster > CaseClusterVector
@ CC_Range
A cluster of adjacent case labels with the same destination, or just one case.
@ CC_JumpTable
A cluster of cases suitable for jump table lowering.
@ CC_BitTests
A cluster of cases suitable for bit test lowering.
SmallVector< SwitchWorkListItem, 4 > SwitchWorkList
CaseClusterVector::iterator CaseClusterIt
initializer< Ty > init(const Ty &Val)
LocationClass< Ty > location(Ty &L)
@ DW_OP_LLVM_arg
Only used in LLVM metadata.
ExceptionBehavior
Exception behavior used for floating point operations.
@ ebStrict
This corresponds to "fpexcept.strict".
@ ebMayTrap
This corresponds to "fpexcept.maytrap".
@ ebIgnore
This corresponds to "fpexcept.ignore".
NodeAddr< FuncNode * > Func
friend class Instruction
Iterator for Instructions in a `BasicBlock.
This is an optimization pass for GlobalISel generic memory operations.
@ Low
Lower the current thread's priority such that it does not affect foreground tasks significantly.
unsigned Log2_32_Ceil(uint32_t Value)
Return the ceil log base 2 of the specified value, 32 if the value is zero.
FunctionAddr VTableAddr Value
LLVM_ABI ISD::CondCode getICmpCondCode(ICmpInst::Predicate Pred)
getICmpCondCode - Return the ISD condition code corresponding to the given LLVM IR integer condition ...
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
SDValue peekThroughFreeze(SDValue V)
Return the non-frozen source operand of V if it exists.
LLVM_ABI void GetReturnInfo(CallingConv::ID CC, Type *ReturnType, AttributeList attr, SmallVectorImpl< ISD::OutputArg > &Outs, const TargetLowering &TLI, const DataLayout &DL)
Given an LLVM IR type and return type attributes, compute the return value EVTs and flags,...
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
LLVM_ABI bool isOnlyUsedInZeroEqualityComparison(const Instruction *CxtI)
LLVM_ABI void ComputeValueVTs(const TargetLowering &TLI, const DataLayout &DL, Type *Ty, SmallVectorImpl< EVT > &ValueVTs, SmallVectorImpl< EVT > *MemVTs=nullptr, SmallVectorImpl< TypeSize > *Offsets=nullptr, TypeSize StartingOffset=TypeSize::getZero())
ComputeValueVTs - Given an LLVM IR type, compute a sequence of EVTs that represent all the individual...
LLVM_ABI SDValue peekThroughBitcasts(SDValue V)
Return the non-bitcasted source operand of V if it exists.
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.
LLVM_ABI void diagnoseDontCall(const CallInst &CI)
auto successors(const MachineBasicBlock *BB)
bool isIntOrFPConstant(SDValue V)
Return true if V is either a integer or FP constant.
static ConstantRange getRange(Value *Op, SCCPSolver &Solver, const SmallPtrSetImpl< Value * > &InsertedValues)
Helper for getting ranges from Solver.
void append_range(Container &C, Range &&R)
Wrapper function to append range R to container C.
Value * GetPointerBaseWithConstantOffset(Value *Ptr, int64_t &Offset, const DataLayout &DL, bool AllowNonInbounds=true)
Analyze the specified pointer to see if it can be expressed as a base pointer plus a constant offset.
constexpr bool isUIntN(unsigned N, uint64_t x)
Checks if an unsigned integer fits into the given (dynamic) bit width.
auto cast_or_null(const Y &Val)
constexpr T alignDown(U Value, V Align, W Skew=0)
Returns the largest unsigned integer less than or equal to Value and is Skew mod Align.
gep_type_iterator gep_type_end(const User *GEP)
LLVM_ABI LLT getLLTForMVT(MVT Ty)
Get a rough equivalent of an LLT for a given MVT.
constexpr auto equal_to(T &&Arg)
Functor variant of std::equal_to that can be used as a UnaryPredicate in functional algorithms like a...
constexpr int popcount(T Value) noexcept
Count the number of set bits in a value.
LLVM_ABI ConstantRange getConstantRangeFromMetadata(const MDNode &RangeMD)
Parse out a conservative ConstantRange from !range metadata.
detail::concat_range< ValueT, RangeTs... > concat(RangeTs &&...Ranges)
Returns a concatenated range across two or more ranges.
bool isScopedEHPersonality(EHPersonality Pers)
Returns true if this personality uses scope-style EH IR instructions: catchswitch,...
int countr_zero(T Val)
Count number of 0's from the least significant bit to the most stopping at the first 1.
LLVM_ABI void ComputeValueTypes(const DataLayout &DL, Type *Ty, SmallVectorImpl< Type * > &Types, SmallVectorImpl< TypeSize > *Offsets=nullptr, TypeSize StartingOffset=TypeSize::getZero())
Given an LLVM IR type, compute non-aggregate subtypes.
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 llvm::SmallVector< int, 16 > createStrideMask(unsigned Start, unsigned Stride, unsigned VF)
Create a stride shuffle mask.
@ SPF_ABS
Floating point maxnum.
@ SPF_NABS
Absolute value.
@ SPF_FMAXNUM
Floating point minnum.
@ SPF_UMIN
Signed minimum.
@ SPF_UMAX
Signed maximum.
@ SPF_SMAX
Unsigned minimum.
@ SPF_FMINNUM
Unsigned maximum.
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
detail::zippy< detail::zip_first, T, U, Args... > zip_first(T &&t, U &&u, Args &&...args)
zip iterator that, for the sake of efficiency, assumes the first iteratee to be the shortest.
void sort(IteratorTy Start, IteratorTy End)
FPClassTest
Floating-point class tests, supported by 'is_fpclass' intrinsic.
LLVM_ABI SelectPatternResult matchSelectPattern(Value *V, Value *&LHS, Value *&RHS, Instruction::CastOps *CastOp=nullptr, unsigned Depth=0)
Pattern match integer [SU]MIN, [SU]MAX and ABS idioms, returning the kind and providing the out param...
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
constexpr uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
generic_gep_type_iterator<> gep_type_iterator
FunctionAddr VTableAddr Count
auto succ_size(const MachineBasicBlock *BB)
bool hasSingleElement(ContainerTy &&C)
Returns true if the given container only contains a single element.
LLVM_ABI ISD::CondCode getFCmpCondCode(FCmpInst::Predicate Pred)
getFCmpCondCode - Return the ISD condition code corresponding to the given LLVM IR floating-point con...
LLVM_ABI EHPersonality classifyEHPersonality(const Value *Pers)
See if the given exception handling personality function is one that we understand.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
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...
LLVM_ATTRIBUTE_VISIBILITY_DEFAULT AnalysisKey InnerAnalysisManagerProxy< AnalysisManagerT, IRUnitT, ExtraArgTs... >::Key
LLVM_ABI Value * salvageDebugInfoImpl(Instruction &I, uint64_t CurrentLocOps, SmallVectorImpl< uint64_t > &Ops, SmallVectorImpl< Value * > &AdditionalValues)
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ Global
Append to llvm.global_dtors.
AtomicOrdering
Atomic ordering for LLVM's memory model.
@ First
Helpers to iterate all locations in the MemoryEffectsBase class.
bool isFuncletEHPersonality(EHPersonality Pers)
Returns true if this is a personality function that invokes handler funclets (which must return to it...
FunctionAddr VTableAddr uintptr_t uintptr_t Data
LLVM_ABI bool isAssignmentTrackingEnabled(const Module &M)
Return true if assignment tracking is enabled for module M.
LLVM_ABI llvm::SmallVector< int, 16 > createInterleaveMask(unsigned VF, unsigned NumVecs)
Create an interleave shuffle mask.
@ UMin
Unsigned integer min implemented in terms of select(cmp()).
@ Or
Bitwise or logical OR of integers.
@ Mul
Product of integers.
@ Sub
Subtraction of integers.
@ SPNB_RETURNS_NAN
NaN behavior not applicable.
@ SPNB_RETURNS_OTHER
Given one NaN input, returns the NaN.
@ SPNB_RETURNS_ANY
Given one NaN input, returns the non-NaN.
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
@ Dynamic
Denotes mode unknown at compile time.
LLVM_ABI ISD::CondCode getFCmpCodeWithoutNaN(ISD::CondCode CC)
getFCmpCodeWithoutNaN - Given an ISD condition code comparing floats, return the equivalent code if w...
ArrayRef(const T &OneElt) -> ArrayRef< T >
bool isAsynchronousEHPersonality(EHPersonality Pers)
Returns true if this personality function catches asynchronous exceptions.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI bool isKnownNeverNaN(const Value *V, const SimplifyQuery &SQ, unsigned Depth=0)
Return true if the floating-point scalar value is not a NaN or if the floating-point vector value has...
LLVM_ABI std::optional< RoundingMode > convertStrToRoundingMode(StringRef)
Returns a valid RoundingMode enumerator when given a string that is valid as input in constrained int...
gep_type_iterator gep_type_begin(const User *GEP)
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
LLVM_ABI GlobalValue * ExtractTypeInfo(Value *V)
ExtractTypeInfo - Returns the type info, possibly bitcast, encoded in V.
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.
bool all_equal(std::initializer_list< T > Values)
Returns true if all Values in the initializer lists are equal or the list.
LLVM_ABI Constant * ConstantFoldLoadFromConstPtr(Constant *C, Type *Ty, APInt Offset, const DataLayout &DL)
Return the value that a load from C with offset Offset would produce if it is constant and determinab...
LLVM_ABI unsigned ComputeLinearIndex(Type *Ty, const unsigned *Indices, const unsigned *IndicesEnd, unsigned CurIndex=0)
Compute the linearized index of a member in a nested aggregate/struct/array.
T bit_floor(T Value)
Returns the largest integral power of two no greater than Value if Value is nonzero.
@ Default
The result value is uniform if and only if all operands are uniform.
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.
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.
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).
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.
uint64_t getScalarStoreSize() const
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
static LLVM_ABI EVT getEVT(Type *Ty, bool HandleUnknown=false)
Return the value type corresponding to the specified type.
EVT changeVectorElementType(LLVMContext &Context, EVT EltVT) const
Return a VT for a vector type whose attributes match ourselves with the exception of the element type...
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 isRISCVVectorTuple() const
Return true if this is a vector value type.
uint64_t getFixedSizeInBits() const
Return the size of the specified fixed width value type in 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 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.
EVT changeElementType(LLVMContext &Context, EVT EltVT) const
Return a VT for a type whose attributes match ourselves with the exception of the element type that i...
bool isScalarInteger() const
Return true if this is an integer, but not a vector.
unsigned getVectorNumElements() const
Given a vector type, return the number of elements it contains.
bool isInteger() const
Return true if this is an integer or a vector integer type.
void setPointerAddrSpace(unsigned AS)
void setOrigAlign(Align A)
OutputArg - This struct carries flags and a value for a single outgoing (actual) argument or outgoing...
ConstraintPrefix Type
Type - The basic type of the constraint: input/output/clobber/label.
unsigned countMinLeadingZeros() const
Returns the minimum number of leading zero bits.
This class contains a discriminated union of information about pointers in memory operands,...
static LLVM_ABI MachinePointerInfo getUnknownStack(MachineFunction &MF)
Stack memory without other information.
static LLVM_ABI MachinePointerInfo getFixedStack(MachineFunction &MF, int FI, int64_t Offset=0)
Return a MachinePointerInfo record that refers to the specified FrameIndex.
A lightweight accessor for an operand bundle meant to be passed around by value.
This struct represents the registers (physical or virtual) that a particular set of values is assigne...
SmallVector< std::pair< Register, TypeSize >, 4 > getRegsAndSizes() const
Return a list of registers and their sizes.
SmallVector< unsigned, 4 > RegCount
This list holds the number of registers for each value.
bool isABIMangled() const
SmallVector< EVT, 4 > ValueVTs
The value types of the values, which may not be legal, and may need be promoted or synthesized from o...
SmallVector< Register, 4 > Regs
This list holds the registers assigned to the values.
void AddInlineAsmOperands(InlineAsm::Kind Code, bool HasMatching, unsigned MatchingIdx, const SDLoc &dl, SelectionDAG &DAG, std::vector< SDValue > &Ops) const
Add this value to the specified inlineasm node operand list.
SDValue getCopyFromRegs(SelectionDAG &DAG, FunctionLoweringInfo &FuncInfo, const SDLoc &dl, SDValue &Chain, SDValue *Glue, const Value *V=nullptr) const
Emit a series of CopyFromReg nodes that copies from this value and returns the result as a ValueVTs v...
SmallVector< MVT, 4 > RegVTs
The value types of the registers.
void getCopyToRegs(SDValue Val, SelectionDAG &DAG, const SDLoc &dl, SDValue &Chain, SDValue *Glue, const Value *V=nullptr, ISD::NodeType PreferredExtendType=ISD::ANY_EXTEND) const
Emit a series of CopyToReg nodes that copies the specified value into the registers specified by this...
std::optional< CallingConv::ID > CallConv
Records if this value needs to be treated in an ABI dependant manner, different to normal type legali...
bool occupiesMultipleRegs() const
Check if the total RegCount is greater than one.
These are IR-level optimization flags that may be propagated to SDNodes.
void copyFMF(const FPMathOperator &FPMO)
Propagate the fast-math-flags from an IR FPMathOperator.
void setUnpredictable(bool b)
bool hasAllowReassociation() const
void setNoUnsignedWrap(bool b)
void setNoSignedWrap(bool b)
This represents a list of ValueType's that has been intern'd by a SelectionDAG.
A MapVector that performs no allocations if smaller than a certain size.
MachineBasicBlock * Default
BranchProbability DefaultProb
MachineBasicBlock * Parent
bool FallthroughUnreachable
MachineBasicBlock * ThisBB
This structure is used to communicate between SelectionDAGBuilder and SDISel for the code generation ...
BranchProbability TrueProb
BranchProbability FalseProb
MachineBasicBlock * TrueBB
MachineBasicBlock * FalseBB
SDLoc DL
The debug location of the instruction this CaseBlock was produced from.
static CaseCluster range(const ConstantInt *Low, const ConstantInt *High, MachineBasicBlock *MBB, BranchProbability Prob)
Register Reg
The virtual register containing the index of the jump table entry to jump to.
MachineBasicBlock * Default
The MBB of the default bb, which is a successor of the range check MBB.
unsigned JTI
The JumpTableIndex for this jump table in the function.
MachineBasicBlock * MBB
The MBB into which to emit the code for the indirect jump.
std::optional< SDLoc > SL
The debug location of the instruction this JumpTable was produced from.
This contains information for each constraint that we are lowering.
TargetLowering::ConstraintType ConstraintType
Information about the constraint code, e.g.
This structure contains all information that is necessary for lowering calls.
CallLoweringInfo & setConvergent(bool Value=true)
CallLoweringInfo & setDeactivationSymbol(GlobalValue *Sym)
CallLoweringInfo & setCFIType(const ConstantInt *Type)
SmallVector< ISD::InputArg, 32 > Ins
bool IsPostTypeLegalization
SmallVector< SDValue, 4 > InVals
Type * OrigRetTy
Original unlegalized return type.
CallLoweringInfo & setDiscardResult(bool Value=true)
CallLoweringInfo & setIsPatchPoint(bool Value=true)
CallLoweringInfo & setDebugLoc(const SDLoc &dl)
CallLoweringInfo & setTailCall(bool Value=true)
CallLoweringInfo & setIsPreallocated(bool Value=true)
CallLoweringInfo & setConvergenceControlToken(SDValue Token)
SmallVector< ISD::OutputArg, 32 > Outs
SmallVector< SDValue, 32 > OutVals
Type * RetTy
Same as OrigRetTy, or partially legalized for soft float libcalls.
CallLoweringInfo & setChain(SDValue InChain)
CallLoweringInfo & setPtrAuth(PtrAuthInfo Value)
CallLoweringInfo & setCallee(CallingConv::ID CC, Type *ResultType, SDValue Target, ArgListTy &&ArgsList, AttributeSet ResultAttrs={})
This structure is used to pass arguments to makeLibCall function.
MakeLibCallOptions & setDiscardResult(bool Value=true)
This structure contains the information necessary for lowering pointer-authenticating indirect calls.
LLVM_ABI void addIPToStateRange(const InvokeInst *II, MCSymbol *InvokeBegin, MCSymbol *InvokeEnd)