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();
1000 NumParts, RegisterVT, V,
CallConv, ExtendKind);
1006 for (
unsigned i = 0; i != NumRegs; ++i) {
1018 if (NumRegs == 1 || Glue)
1029 Chain = Chains[NumRegs-1];
1035 unsigned MatchingIdx,
const SDLoc &dl,
1037 std::vector<SDValue> &
Ops)
const {
1042 Flag.setMatchingOp(MatchingIdx);
1043 else if (!
Regs.empty() &&
Regs.front().isVirtual()) {
1051 Flag.setRegClass(RC->
getID());
1062 "No 1:1 mapping from clobbers to regs?");
1065 for (
unsigned I = 0, E =
ValueVTs.size();
I != E; ++
I) {
1070 "If we clobbered the stack pointer, MFI should know about it.");
1079 for (
unsigned i = 0; i != NumRegs; ++i) {
1080 assert(Reg <
Regs.size() &&
"Mismatch in # registers expected");
1092 unsigned RegCount = std::get<0>(CountAndVT);
1093 MVT RegisterVT = std::get<1>(CountAndVT);
1111 SL->init(
DAG.getTargetLoweringInfo(), TM,
DAG.getDataLayout());
1113 *
DAG.getMachineFunction().getFunction().getParent());
1118 UnusedArgNodeMap.clear();
1120 PendingExports.clear();
1121 PendingConstrainedFP.clear();
1122 PendingConstrainedFPStrict.clear();
1130 DanglingDebugInfoMap.clear();
1137 if (Pending.
empty())
1143 unsigned i = 0, e = Pending.
size();
1144 for (; i != e; ++i) {
1146 if (Pending[i].
getNode()->getOperand(0) == Root)
1154 if (Pending.
size() == 1)
1181 if (!PendingConstrainedFPStrict.empty()) {
1182 assert(PendingConstrainedFP.empty());
1183 updateRoot(PendingConstrainedFPStrict);
1196 if (!PendingConstrainedFP.empty()) {
1197 assert(PendingConstrainedFPStrict.empty());
1198 updateRoot(PendingConstrainedFP);
1202 return DAG.getRoot();
1210 PendingConstrainedFP.size() +
1211 PendingConstrainedFPStrict.size());
1213 PendingConstrainedFP.end());
1214 PendingLoads.append(PendingConstrainedFPStrict.begin(),
1215 PendingConstrainedFPStrict.end());
1216 PendingConstrainedFP.clear();
1217 PendingConstrainedFPStrict.clear();
1224 PendingExports.append(PendingConstrainedFPStrict.begin(),
1225 PendingConstrainedFPStrict.end());
1226 PendingConstrainedFPStrict.clear();
1227 return updateRoot(PendingExports);
1234 assert(Variable &&
"Missing variable");
1241 <<
"dbg_declare: Dropping debug info (bad/undef/unused-arg address)\n");
1257 if (IsParameter && FINode) {
1259 SDV =
DAG.getFrameIndexDbgValue(Variable,
Expression, FINode->getIndex(),
1260 true,
DL, SDNodeOrder);
1265 FuncArgumentDbgValueKind::Declare,
N);
1268 SDV =
DAG.getDbgValue(Variable,
Expression,
N.getNode(),
N.getResNo(),
1269 true,
DL, SDNodeOrder);
1271 DAG.AddDbgValue(SDV, IsParameter);
1276 FuncArgumentDbgValueKind::Declare,
N)) {
1278 <<
" (could not emit func-arg dbg_value)\n");
1289 for (
auto It = FnVarLocs->locs_begin(&
I), End = FnVarLocs->locs_end(&
I);
1291 auto *Var = FnVarLocs->getDILocalVariable(It->VariableID);
1293 if (It->Values.isKillLocation(It->Expr)) {
1299 It->Values.hasArgList())) {
1302 FnVarLocs->getDILocalVariable(It->VariableID),
1303 It->Expr, Vals.
size() > 1, It->DL, SDNodeOrder);
1316 bool SkipDbgVariableRecords =
DAG.getFunctionVarLocs();
1319 for (
DbgRecord &DR :
I.getDbgRecordRange()) {
1321 assert(DLR->getLabel() &&
"Missing label");
1323 DAG.getDbgLabel(DLR->getLabel(), DLR->getDebugLoc(), SDNodeOrder);
1324 DAG.AddDbgLabel(SDV);
1328 if (SkipDbgVariableRecords)
1336 if (
FuncInfo.PreprocessedDVRDeclares.contains(&DVR))
1338 LLVM_DEBUG(
dbgs() <<
"SelectionDAG visiting dbg_declare: " << DVR
1347 if (Values.
empty()) {
1364 SDNodeOrder, IsVariadic)) {
1375 if (
I.isTerminator()) {
1376 HandlePHINodesInSuccessorBlocks(
I.getParent());
1383 bool NodeInserted =
false;
1384 std::unique_ptr<SelectionDAG::DAGNodeInsertedListener> InsertedListener;
1385 MDNode *PCSectionsMD =
I.getMetadata(LLVMContext::MD_pcsections);
1386 MDNode *MMRA =
I.getMetadata(LLVMContext::MD_mmra);
1387 if (PCSectionsMD || MMRA) {
1388 InsertedListener = std::make_unique<SelectionDAG::DAGNodeInsertedListener>(
1389 DAG, [&](
SDNode *) { NodeInserted =
true; });
1399 if (PCSectionsMD || MMRA) {
1400 auto It = NodeMap.find(&
I);
1401 if (It != NodeMap.end()) {
1403 DAG.addPCSections(It->second.getNode(), PCSectionsMD);
1405 DAG.addMMRAMetadata(It->second.getNode(), MMRA);
1406 }
else if (NodeInserted) {
1409 errs() <<
"warning: loosing !pcsections and/or !mmra metadata ["
1410 <<
I.getModule()->getName() <<
"]\n";
1419void SelectionDAGBuilder::visitPHI(
const PHINode &) {
1429#define HANDLE_INST(NUM, OPCODE, CLASS) \
1430 case Instruction::OPCODE: visit##OPCODE((const CLASS&)I); break;
1431#include "llvm/IR/Instruction.def"
1443 for (
const Value *V : Values) {
1468 DanglingDebugInfoMap[Values[0]].emplace_back(Var, Expr,
DL, Order);
1473 auto isMatchingDbgValue = [&](DanglingDebugInfo &DDI) {
1474 DIVariable *DanglingVariable = DDI.getVariable();
1476 if (DanglingVariable == Variable && Expr->
fragmentsOverlap(DanglingExpr)) {
1478 << printDDI(
nullptr, DDI) <<
"\n");
1484 for (
auto &DDIMI : DanglingDebugInfoMap) {
1485 DanglingDebugInfoVector &DDIV = DDIMI.second;
1489 for (
auto &DDI : DDIV)
1490 if (isMatchingDbgValue(DDI))
1493 erase_if(DDIV, isMatchingDbgValue);
1501 auto DanglingDbgInfoIt = DanglingDebugInfoMap.find(V);
1502 if (DanglingDbgInfoIt == DanglingDebugInfoMap.end())
1505 DanglingDebugInfoVector &DDIV = DanglingDbgInfoIt->second;
1506 for (
auto &DDI : DDIV) {
1508 unsigned DbgSDNodeOrder = DDI.getSDNodeOrder();
1511 assert(Variable->isValidLocationForIntrinsic(
DL) &&
1512 "Expected inlined-at fields to agree");
1522 if (!EmitFuncArgumentDbgValue(V, Variable, Expr,
DL,
1523 FuncArgumentDbgValueKind::Value, Val)) {
1525 << printDDI(V, DDI) <<
"\n");
1532 <<
"changing SDNodeOrder from " << DbgSDNodeOrder <<
" to "
1533 << ValSDNodeOrder <<
"\n");
1534 SDV = getDbgValue(Val, Variable, Expr,
DL,
1535 std::max(DbgSDNodeOrder, ValSDNodeOrder));
1536 DAG.AddDbgValue(SDV,
false);
1540 <<
" in EmitFuncArgumentDbgValue\n");
1542 LLVM_DEBUG(
dbgs() <<
"Dropping debug info for " << printDDI(V, DDI)
1546 DAG.getConstantDbgValue(Variable, Expr,
Poison,
DL, DbgSDNodeOrder);
1547 DAG.AddDbgValue(SDV,
false);
1554 DanglingDebugInfo &DDI) {
1559 const Value *OrigV = V;
1563 unsigned SDOrder = DDI.getSDNodeOrder();
1567 bool StackValue =
true;
1592 if (!AdditionalValues.
empty())
1602 dbgs() <<
"Salvaged debug location info for:\n " << *Var <<
"\n"
1603 << *OrigV <<
"\nBy stripping back to:\n " << *V <<
"\n");
1611 assert(OrigV &&
"V shouldn't be null");
1613 auto *SDV =
DAG.getConstantDbgValue(Var, Expr,
Poison,
DL, SDNodeOrder);
1614 DAG.AddDbgValue(SDV,
false);
1616 << printDDI(OrigV, DDI) <<
"\n");
1633 unsigned Order,
bool IsVariadic) {
1638 if (visitEntryValueDbgValue(Values, Var, Expr, DbgLoc))
1643 for (
const Value *V : Values) {
1653 if (CE->getOpcode() == Instruction::IntToPtr) {
1672 N = UnusedArgNodeMap[V];
1677 EmitFuncArgumentDbgValue(V, Var, Expr, DbgLoc,
1678 FuncArgumentDbgValueKind::Value,
N))
1705 bool IsParamOfFunc =
1713 auto VMI =
FuncInfo.ValueMap.find(V);
1714 if (VMI !=
FuncInfo.ValueMap.end()) {
1719 V->getType(), std::nullopt);
1725 unsigned BitsToDescribe = 0;
1727 BitsToDescribe = *VarSize;
1729 BitsToDescribe = Fragment->SizeInBits;
1732 if (
Offset >= BitsToDescribe)
1735 unsigned RegisterSize = RegAndSize.second;
1736 unsigned FragmentSize = (
Offset + RegisterSize > BitsToDescribe)
1737 ? BitsToDescribe -
Offset
1740 Expr,
Offset, FragmentSize);
1744 Var, *FragmentExpr, RegAndSize.first,
false, DbgLoc, Order);
1745 DAG.AddDbgValue(SDV,
false);
1761 DAG.getDbgValueList(Var, Expr, LocationOps, Dependencies,
1762 false, DbgLoc, Order, IsVariadic);
1763 DAG.AddDbgValue(SDV,
false);
1769 for (
auto &Pair : DanglingDebugInfoMap)
1770 for (
auto &DDI : Pair.second)
1781 if (It !=
FuncInfo.ValueMap.end()) {
1785 DAG.getDataLayout(), InReg, Ty,
1802 if (
N.getNode())
return N;
1862 return DAG.getSplatBuildVector(
1865 return DAG.getConstant(*CI,
DL, VT);
1874 getValue(CPA->getAddrDiscriminator()),
1875 getValue(CPA->getDiscriminator()));
1891 visit(CE->getOpcode(), *CE);
1893 assert(N1.
getNode() &&
"visit didn't populate the NodeMap!");
1899 for (
const Use &U :
C->operands()) {
1905 for (
unsigned i = 0, e = Val->
getNumValues(); i != e; ++i)
1906 Constants.push_back(
SDValue(Val, i));
1915 for (
uint64_t i = 0, e = CDS->getNumElements(); i != e; ++i) {
1919 for (
unsigned i = 0, e = Val->
getNumValues(); i != e; ++i)
1928 if (
C->getType()->isStructTy() ||
C->getType()->isArrayTy()) {
1930 "Unknown struct or array constant!");
1934 unsigned NumElts = ValueVTs.
size();
1938 for (
unsigned i = 0; i != NumElts; ++i) {
1939 EVT EltVT = ValueVTs[i];
1941 Constants[i] =
DAG.getUNDEF(EltVT);
1952 return DAG.getBlockAddress(BA, VT);
1955 return getValue(Equiv->getGlobalValue());
1960 if (VT == MVT::aarch64svcount) {
1961 assert(
C->isNullValue() &&
"Can only zero this target type!");
1967 assert(
C->isNullValue() &&
"Can only zero this target type!");
1984 for (
unsigned i = 0; i != NumElements; ++i)
2012 return DAG.getFrameIndex(
2020 Inst->getType(), std::nullopt);
2034void SelectionDAGBuilder::visitCatchPad(
const CatchPadInst &
I) {
2047 if (IsMSVCCXX || IsCoreCLR)
2053 MachineBasicBlock *TargetMBB =
FuncInfo.getMBB(
I.getSuccessor());
2054 FuncInfo.MBB->addSuccessor(TargetMBB);
2061 if (TargetMBB != NextBlock(
FuncInfo.MBB) ||
2070 DAG.getMachineFunction().setHasEHContTarget(
true);
2076 Value *ParentPad =
I.getCatchSwitchParentPad();
2079 SuccessorColor = &
FuncInfo.Fn->getEntryBlock();
2082 assert(SuccessorColor &&
"No parent funclet for catchret!");
2083 MachineBasicBlock *SuccessorColorMBB =
FuncInfo.getMBB(SuccessorColor);
2084 assert(SuccessorColorMBB &&
"No MBB for SuccessorColor!");
2089 DAG.getBasicBlock(SuccessorColorMBB));
2093void SelectionDAGBuilder::visitCleanupPad(
const CleanupPadInst &CPI) {
2099 FuncInfo.MBB->setIsEHFuncletEntry();
2100 FuncInfo.MBB->setIsCleanupFuncletEntry();
2129 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2135 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2136 UnwindDests.back().first->setIsEHScopeEntry();
2139 UnwindDests.back().first->setIsEHFuncletEntry();
2143 for (
const BasicBlock *CatchPadBB : CatchSwitch->handlers()) {
2144 UnwindDests.emplace_back(FuncInfo.
getMBB(CatchPadBB), Prob);
2146 if (IsMSVCCXX || IsCoreCLR)
2147 UnwindDests.back().first->setIsEHFuncletEntry();
2149 UnwindDests.back().first->setIsEHScopeEntry();
2151 NewEHPadBB = CatchSwitch->getUnwindDest();
2157 if (BPI && NewEHPadBB)
2159 EHPadBB = NewEHPadBB;
2166 auto UnwindDest =
I.getUnwindDest();
2167 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
2168 BranchProbability UnwindDestProb =
2173 for (
auto &UnwindDest : UnwindDests) {
2174 UnwindDest.first->setIsEHPad();
2175 addSuccessorWithProb(
FuncInfo.MBB, UnwindDest.first, UnwindDest.second);
2177 FuncInfo.MBB->normalizeSuccProbs();
2180 MachineBasicBlock *CleanupPadMBB =
2181 FuncInfo.getMBB(
I.getCleanupPad()->getParent());
2187void SelectionDAGBuilder::visitCatchSwitch(
const CatchSwitchInst &CSI) {
2191void SelectionDAGBuilder::visitRet(
const ReturnInst &
I) {
2192 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
2193 auto &
DL =
DAG.getDataLayout();
2205 if (
I.getParent()->getTerminatingDeoptimizeCall()) {
2222 SmallVector<uint64_t, 4>
Offsets;
2225 unsigned NumValues = ValueVTs.
size();
2228 Align BaseAlign =
DL.getPrefTypeAlign(
I.getOperand(0)->getType());
2229 for (
unsigned i = 0; i != NumValues; ++i) {
2236 if (MemVTs[i] != ValueVTs[i])
2238 Chains[i] =
DAG.getStore(
2246 MVT::Other, Chains);
2247 }
else if (
I.getNumOperands() != 0) {
2250 unsigned NumValues =
Types.size();
2254 const Function *
F =
I.getParent()->getParent();
2257 I.getOperand(0)->getType(),
F->getCallingConv(),
2261 if (
F->getAttributes().hasRetAttr(Attribute::SExt))
2263 else if (
F->getAttributes().hasRetAttr(Attribute::ZExt))
2266 LLVMContext &
Context =
F->getContext();
2267 bool RetInReg =
F->getAttributes().hasRetAttr(Attribute::InReg);
2269 for (
unsigned j = 0;
j != NumValues; ++
j) {
2282 &Parts[0], NumParts, PartVT, &
I, CC, ExtendKind);
2285 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2289 if (
I.getOperand(0)->getType()->isPointerTy()) {
2291 Flags.setPointerAddrSpace(
2295 if (NeedsRegBlock) {
2296 Flags.setInConsecutiveRegs();
2297 if (j == NumValues - 1)
2298 Flags.setInConsecutiveRegsLast();
2306 else if (
F->getAttributes().hasRetAttr(Attribute::NoExt))
2309 for (
unsigned i = 0; i < NumParts; ++i) {
2312 VT, Types[j], 0, 0));
2322 const Function *
F =
I.getParent()->getParent();
2324 F->getAttributes().hasAttrSomewhere(Attribute::SwiftError)) {
2326 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2327 Flags.setSwiftError();
2339 bool isVarArg =
DAG.getMachineFunction().getFunction().isVarArg();
2341 DAG.getMachineFunction().getFunction().getCallingConv();
2342 Chain =
DAG.getTargetLoweringInfo().LowerReturn(
2347 "LowerReturn didn't return a valid chain!");
2358 if (V->getType()->isEmptyTy())
2362 if (VMI !=
FuncInfo.ValueMap.end()) {
2364 "Unused value assigned virtual registers!");
2377 if (
FuncInfo.isExportedInst(V))
return;
2389 if (VI->getParent() == FromBB)
2415 const BasicBlock *SrcBB = Src->getBasicBlock();
2416 const BasicBlock *DstBB = Dst->getBasicBlock();
2420 auto SuccSize = std::max<uint32_t>(
succ_size(SrcBB), 1);
2430 Src->addSuccessorWithoutProb(Dst);
2433 Prob = getEdgeProbability(Src, Dst);
2434 Src->addSuccessor(Dst, Prob);
2440 return I->getParent() == BB;
2464 if (CurBB == SwitchBB ||
2470 InvertCond ? IC->getInversePredicate() : IC->getPredicate();
2475 InvertCond ? FC->getInversePredicate() : FC->getPredicate();
2477 if (FC->hasNoNaNs() ||
2485 CaseBlock CB(Condition, BOp->getOperand(0), BOp->getOperand(1),
nullptr,
2487 SL->SwitchCases.push_back(CB);
2496 SL->SwitchCases.push_back(CB);
2504 unsigned Depth = 0) {
2513 if (Necessary !=
nullptr) {
2516 if (Necessary->contains(
I))
2535 if (
I.getNumSuccessors() != 2)
2538 if (!
I.isConditional())
2550 if (BPI !=
nullptr) {
2556 std::optional<bool> Likely;
2559 else if (BPI->
isEdgeHot(
I.getParent(), IfFalse))
2563 if (
Opc == (*Likely ? Instruction::And : Instruction::Or))
2575 if (CostThresh <= 0)
2596 Value *BrCond =
I.getCondition();
2597 auto ShouldCountInsn = [&RhsDeps, &BrCond](
const Instruction *Ins) {
2598 for (
const auto *U : Ins->users()) {
2601 if (UIns != BrCond && !RhsDeps.
contains(UIns))
2614 for (
unsigned PruneIters = 0; PruneIters < MaxPruneIters; ++PruneIters) {
2616 for (
const auto &InsPair : RhsDeps) {
2617 if (!ShouldCountInsn(InsPair.first)) {
2618 ToDrop = InsPair.first;
2622 if (ToDrop ==
nullptr)
2624 RhsDeps.erase(ToDrop);
2627 for (
const auto &InsPair : RhsDeps) {
2632 CostOfIncluding +=
TTI->getInstructionCost(
2635 if (CostOfIncluding > CostThresh)
2661 const Value *BOpOp0, *BOpOp1;
2675 if (BOpc == Instruction::And)
2676 BOpc = Instruction::Or;
2677 else if (BOpc == Instruction::Or)
2678 BOpc = Instruction::And;
2684 bool BOpIsInOrAndTree = BOpc && BOpc ==
Opc && BOp->
hasOneUse();
2689 TProb, FProb, InvertCond);
2699 if (
Opc == Instruction::Or) {
2720 auto NewTrueProb = TProb / 2;
2721 auto NewFalseProb = TProb / 2 + FProb;
2724 NewFalseProb, InvertCond);
2731 Probs[1], InvertCond);
2733 assert(
Opc == Instruction::And &&
"Unknown merge op!");
2753 auto NewTrueProb = TProb + FProb / 2;
2754 auto NewFalseProb = FProb / 2;
2757 NewFalseProb, InvertCond);
2764 Probs[1], InvertCond);
2773 if (Cases.size() != 2)
return true;
2777 if ((Cases[0].CmpLHS == Cases[1].CmpLHS &&
2778 Cases[0].CmpRHS == Cases[1].CmpRHS) ||
2779 (Cases[0].CmpRHS == Cases[1].CmpLHS &&
2780 Cases[0].CmpLHS == Cases[1].CmpRHS)) {
2786 if (Cases[0].CmpRHS == Cases[1].CmpRHS &&
2787 Cases[0].CC == Cases[1].CC &&
2790 if (Cases[0].CC ==
ISD::SETEQ && Cases[0].TrueBB == Cases[1].ThisBB)
2792 if (Cases[0].CC ==
ISD::SETNE && Cases[0].FalseBB == Cases[1].ThisBB)
2799void SelectionDAGBuilder::visitBr(
const BranchInst &
I) {
2805 if (
I.isUnconditional()) {
2811 if (Succ0MBB != NextBlock(BrMBB) ||
2824 const Value *CondVal =
I.getCondition();
2825 MachineBasicBlock *Succ1MBB =
FuncInfo.getMBB(
I.getSuccessor(1));
2844 bool IsUnpredictable =
I.hasMetadata(LLVMContext::MD_unpredictable);
2846 if (!
DAG.getTargetLoweringInfo().isJumpExpensive() && BOp &&
2849 const Value *BOp0, *BOp1;
2852 Opcode = Instruction::And;
2854 Opcode = Instruction::Or;
2861 DAG.getTargetLoweringInfo().getJumpConditionMergingParams(
2862 Opcode, BOp0, BOp1))) {
2864 getEdgeProbability(BrMBB, Succ0MBB),
2865 getEdgeProbability(BrMBB, Succ1MBB),
2870 assert(
SL->SwitchCases[0].ThisBB == BrMBB &&
"Unexpected lowering!");
2874 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i) {
2881 SL->SwitchCases.erase(
SL->SwitchCases.begin());
2887 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i)
2888 FuncInfo.MF->erase(
SL->SwitchCases[i].ThisBB);
2890 SL->SwitchCases.clear();
2896 nullptr, Succ0MBB, Succ1MBB, BrMBB,
getCurSDLoc(),
2917 if (CB.
TrueBB != NextBlock(SwitchBB)) {
2924 auto &TLI =
DAG.getTargetLoweringInfo();
2948 Cond =
DAG.getSetCC(dl, MVT::i1, CondLHS, CondRHS, CB.
CC);
2960 Cond =
DAG.getSetCC(dl, MVT::i1, CmpOp,
DAG.getConstant(
High, dl, VT),
2964 VT, CmpOp,
DAG.getConstant(
Low, dl, VT));
2965 Cond =
DAG.getSetCC(dl, MVT::i1, SUB,
2980 if (CB.
TrueBB == NextBlock(SwitchBB)) {
2996 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
2999 DAG.setRoot(BrCond);
3005 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3006 assert(JT.
Reg &&
"Should lower JT Header first!");
3007 EVT PTy =
DAG.getTargetLoweringInfo().getJumpTableRegTy(
DAG.getDataLayout());
3011 Index.getValue(1), Table, Index);
3012 DAG.setRoot(BrJumpTable);
3020 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3027 DAG.getConstant(JTH.
First, dl, VT));
3042 JT.
Reg = JumpTableReg;
3050 Sub.getValueType()),
3054 MVT::Other, CopyTo, CMP,
3058 if (JT.
MBB != NextBlock(SwitchBB))
3059 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
3060 DAG.getBasicBlock(JT.
MBB));
3062 DAG.setRoot(BrCond);
3065 if (JT.
MBB != NextBlock(SwitchBB))
3067 DAG.getBasicBlock(JT.
MBB)));
3069 DAG.setRoot(CopyTo);
3093 if (PtrTy != PtrMemTy)
3109 auto &
DL =
DAG.getDataLayout();
3118 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3125 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3139 assert(GuardCheckFn &&
"Guard check function is null");
3150 Entry.IsInReg =
true;
3151 Args.push_back(Entry);
3157 getValue(GuardCheckFn), std::move(Args));
3159 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
3160 DAG.setRoot(Result.second);
3172 Guard =
DAG.getLoad(PtrMemTy, dl, Chain, GuardPtr,
3178 Guard =
DAG.getPOISON(PtrMemTy);
3220 auto &
DL =
DAG.getDataLayout();
3228 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3234 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3249 if (GuardCheckFn->hasParamAttribute(0, Attribute::AttrKind::InReg))
3250 Entry.IsInReg =
true;
3251 Args.push_back(Entry);
3257 getValue(GuardCheckFn), std::move(Args));
3263 Chain = TLI.
makeLibCall(
DAG, RTLIB::STACKPROTECTOR_CHECK_FAIL, MVT::isVoid,
3286 DAG.getNode(
ISD::SUB, dl, VT, SwitchOp,
DAG.getConstant(
B.First, dl, VT));
3290 bool UsePtrType =
false;
3314 if (!
B.FallthroughUnreachable)
3315 addSuccessorWithProb(SwitchBB,
B.Default,
B.DefaultProb);
3316 addSuccessorWithProb(SwitchBB,
MBB,
B.Prob);
3320 if (!
B.FallthroughUnreachable) {
3329 DAG.getBasicBlock(
B.Default));
3333 if (
MBB != NextBlock(SwitchBB))
3351 if (PopCount == 1) {
3358 }
else if (PopCount == BB.
Range) {
3366 DAG.getConstant(1, dl, VT), ShiftOp);
3370 VT, SwitchVal,
DAG.getConstant(
B.Mask, dl, VT));
3377 addSuccessorWithProb(SwitchBB,
B.TargetBB,
B.ExtraProb);
3379 addSuccessorWithProb(SwitchBB, NextMBB, BranchProbToNext);
3387 Cmp,
DAG.getBasicBlock(
B.TargetBB));
3390 if (NextMBB != NextBlock(SwitchBB))
3391 BrAnd =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrAnd,
3392 DAG.getBasicBlock(NextMBB));
3397void SelectionDAGBuilder::visitInvoke(
const InvokeInst &
I) {
3415 const Value *Callee(
I.getCalledOperand());
3418 visitInlineAsm(
I, EHPadBB);
3423 case Intrinsic::donothing:
3425 case Intrinsic::seh_try_begin:
3426 case Intrinsic::seh_scope_begin:
3427 case Intrinsic::seh_try_end:
3428 case Intrinsic::seh_scope_end:
3434 case Intrinsic::experimental_patchpoint_void:
3435 case Intrinsic::experimental_patchpoint:
3436 visitPatchpoint(
I, EHPadBB);
3438 case Intrinsic::experimental_gc_statepoint:
3444 case Intrinsic::wasm_throw: {
3446 std::array<SDValue, 4>
Ops = {
3457 case Intrinsic::wasm_rethrow: {
3458 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3459 std::array<SDValue, 2>
Ops = {
3468 }
else if (
I.hasDeoptState()) {
3489 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
3490 BranchProbability EHPadBBProb =
3496 addSuccessorWithProb(InvokeMBB, Return);
3497 for (
auto &UnwindDest : UnwindDests) {
3498 UnwindDest.first->setIsEHPad();
3499 addSuccessorWithProb(InvokeMBB, UnwindDest.first, UnwindDest.second);
3505 DAG.getBasicBlock(Return)));
3514void SelectionDAGBuilder::visitCallBrIntrinsic(
const CallBrInst &
I) {
3517 DAG.getTargetLoweringInfo().getTgtMemIntrinsic(
3518 Infos,
I,
DAG.getMachineFunction(),
I.getIntrinsicID());
3519 assert(Infos.
empty() &&
"Intrinsic touches memory");
3522 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
3525 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad);
3526 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
3530 getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
3531 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
3536void SelectionDAGBuilder::visitCallBr(
const CallBrInst &
I) {
3537 MachineBasicBlock *CallBrMBB =
FuncInfo.MBB;
3539 if (
I.isInlineAsm()) {
3546 assert(!
I.hasOperandBundles() &&
3547 "Can't have operand bundles for intrinsics");
3548 visitCallBrIntrinsic(
I);
3553 SmallPtrSet<BasicBlock *, 8> Dests;
3554 Dests.
insert(
I.getDefaultDest());
3564 if (
I.isInlineAsm()) {
3565 for (BasicBlock *Dest :
I.getIndirectDests()) {
3567 Target->setIsInlineAsmBrIndirectTarget();
3573 Target->setLabelMustBeEmitted();
3575 if (Dests.
insert(Dest).second)
3584 DAG.getBasicBlock(Return)));
3587void SelectionDAGBuilder::visitResume(
const ResumeInst &RI) {
3588 llvm_unreachable(
"SelectionDAGBuilder shouldn't visit resume instructions!");
3591void SelectionDAGBuilder::visitLandingPad(
const LandingPadInst &LP) {
3593 "Call to landingpad not in landing pad!");
3597 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3613 assert(ValueVTs.
size() == 2 &&
"Only two-valued landingpads are supported");
3618 if (
FuncInfo.ExceptionPointerVirtReg) {
3619 Ops[0] =
DAG.getZExtOrTrunc(
3620 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3627 Ops[1] =
DAG.getZExtOrTrunc(
3628 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3635 DAG.getVTList(ValueVTs),
Ops);
3643 if (JTB.first.HeaderBB ==
First)
3644 JTB.first.HeaderBB =
Last;
3657 for (
unsigned i = 0, e =
I.getNumSuccessors(); i != e; ++i) {
3659 bool Inserted =
Done.insert(BB).second;
3664 addSuccessorWithProb(IndirectBrMBB, Succ);
3674 if (!
I.shouldLowerToTrap(
DAG.getTarget().Options.TrapUnreachable,
3675 DAG.getTarget().Options.NoTrapAfterNoreturn))
3681void SelectionDAGBuilder::visitUnary(
const User &
I,
unsigned Opcode) {
3684 Flags.copyFMF(*FPOp);
3692void SelectionDAGBuilder::visitBinary(
const User &
I,
unsigned Opcode) {
3695 Flags.setNoSignedWrap(OFBinOp->hasNoSignedWrap());
3696 Flags.setNoUnsignedWrap(OFBinOp->hasNoUnsignedWrap());
3699 Flags.setExact(ExactOp->isExact());
3701 Flags.setDisjoint(DisjointOp->isDisjoint());
3703 Flags.copyFMF(*FPOp);
3712void SelectionDAGBuilder::visitShift(
const User &
I,
unsigned Opcode) {
3716 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
3721 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
3723 "Unexpected shift type");
3733 if (
const OverflowingBinaryOperator *OFBinOp =
3735 nuw = OFBinOp->hasNoUnsignedWrap();
3736 nsw = OFBinOp->hasNoSignedWrap();
3738 if (
const PossiblyExactOperator *ExactOp =
3740 exact = ExactOp->isExact();
3743 Flags.setExact(exact);
3744 Flags.setNoSignedWrap(nsw);
3745 Flags.setNoUnsignedWrap(nuw);
3751void SelectionDAGBuilder::visitSDiv(
const User &
I) {
3762void SelectionDAGBuilder::visitICmp(
const ICmpInst &
I) {
3768 auto &TLI =
DAG.getTargetLoweringInfo();
3781 Flags.setSameSign(
I.hasSameSign());
3782 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
3784 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3789void SelectionDAGBuilder::visitFCmp(
const FCmpInst &
I) {
3796 if (FPMO->hasNoNaNs() ||
3797 (
DAG.isKnownNeverNaN(Op1) &&
DAG.isKnownNeverNaN(Op2)))
3801 Flags.copyFMF(*FPMO);
3802 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
3804 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3814 return isa<SelectInst>(V);
3818void SelectionDAGBuilder::visitSelect(
const User &
I) {
3822 unsigned NumValues = ValueVTs.
size();
3823 if (NumValues == 0)
return;
3833 bool IsUnaryAbs =
false;
3834 bool Negate =
false;
3838 Flags.copyFMF(*FPOp);
3840 Flags.setUnpredictable(
3845 EVT VT = ValueVTs[0];
3846 LLVMContext &Ctx = *
DAG.getContext();
3847 auto &TLI =
DAG.getTargetLoweringInfo();
3857 bool UseScalarMinMax = VT.
isVector() &&
3866 switch (SPR.Flavor) {
3875 switch (SPR.NaNBehavior) {
3880 Flags.setNoSignedZeros(
true);
3894 switch (SPR.NaNBehavior) {
3899 Flags.setNoSignedZeros(
true);
3941 for (
unsigned i = 0; i != NumValues; ++i) {
3947 Values[i] =
DAG.getNegative(Values[i], dl, VT);
3950 for (
unsigned i = 0; i != NumValues; ++i) {
3954 Values[i] =
DAG.getNode(
3961 DAG.getVTList(ValueVTs), Values));
3964void SelectionDAGBuilder::visitTrunc(
const User &
I) {
3967 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3971 Flags.setNoSignedWrap(Trunc->hasNoSignedWrap());
3972 Flags.setNoUnsignedWrap(Trunc->hasNoUnsignedWrap());
3978void SelectionDAGBuilder::visitZExt(
const User &
I) {
3982 auto &TLI =
DAG.getTargetLoweringInfo();
3987 Flags.setNonNeg(PNI->hasNonNeg());
3992 if (
Flags.hasNonNeg() &&
4001void SelectionDAGBuilder::visitSExt(
const User &
I) {
4005 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4010void SelectionDAGBuilder::visitFPTrunc(
const User &
I) {
4016 Flags.copyFMF(*FPOp);
4017 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4020 DAG.getTargetConstant(
4025void SelectionDAGBuilder::visitFPExt(
const User &
I) {
4028 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4032 Flags.copyFMF(*FPOp);
4036void SelectionDAGBuilder::visitFPToUI(
const User &
I) {
4039 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4044void SelectionDAGBuilder::visitFPToSI(
const User &
I) {
4047 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4052void SelectionDAGBuilder::visitUIToFP(
const User &
I) {
4055 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4059 Flags.setNonNeg(PNI->hasNonNeg());
4064void SelectionDAGBuilder::visitSIToFP(
const User &
I) {
4067 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]);
4791 if (!ConstantMemory) {
4801 DAG.getVTList(ValueVTs), Values));
4804void SelectionDAGBuilder::visitStoreToSwiftError(
const StoreInst &
I) {
4805 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4806 "call visitStoreToSwiftError when backend supports swifterror");
4809 SmallVector<uint64_t, 4>
Offsets;
4810 const Value *SrcV =
I.getOperand(0);
4812 SrcV->
getType(), ValueVTs,
nullptr, &Offsets, 0);
4813 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4814 "expect a single EVT for swifterror");
4823 SDValue(Src.getNode(), Src.getResNo()));
4824 DAG.setRoot(CopyNode);
4827void SelectionDAGBuilder::visitLoadFromSwiftError(
const LoadInst &
I) {
4828 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4829 "call visitLoadFromSwiftError when backend supports swifterror");
4832 !
I.hasMetadata(LLVMContext::MD_nontemporal) &&
4833 !
I.hasMetadata(LLVMContext::MD_invariant_load) &&
4834 "Support volatile, non temporal, invariant for load_from_swift_error");
4836 const Value *SV =
I.getOperand(0);
4837 Type *Ty =
I.getType();
4840 !
BatchAA->pointsToConstantMemory(MemoryLocation(
4842 I.getAAMetadata()))) &&
4843 "load_from_swift_error should not be constant memory");
4846 SmallVector<uint64_t, 4>
Offsets;
4848 ValueVTs,
nullptr, &Offsets, 0);
4849 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4850 "expect a single EVT for swifterror");
4860void SelectionDAGBuilder::visitStore(
const StoreInst &
I) {
4862 return visitAtomicStore(
I);
4864 const Value *SrcV =
I.getOperand(0);
4865 const Value *PtrV =
I.getOperand(1);
4867 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4872 if (Arg->hasSwiftErrorAttr())
4873 return visitStoreToSwiftError(
I);
4877 if (Alloca->isSwiftError())
4878 return visitStoreToSwiftError(
I);
4885 SrcV->
getType(), ValueVTs, &MemVTs, &Offsets);
4886 unsigned NumValues = ValueVTs.
size();
4899 Align Alignment =
I.getAlign();
4900 AAMDNodes AAInfo =
I.getAAMetadata();
4904 unsigned ChainI = 0;
4905 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4915 MachinePointerInfo PtrInfo =
4917 ? MachinePointerInfo(PtrV, Offsets[i].getKnownMinValue())
4918 : MachinePointerInfo();
4922 if (MemVTs[i] != ValueVTs[i])
4923 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVTs[i]);
4925 DAG.getStore(Root, dl, Val,
Add, PtrInfo, Alignment, MMOFlags, AAInfo);
4926 Chains[ChainI] = St;
4932 DAG.setRoot(StoreNode);
4935void SelectionDAGBuilder::visitMaskedStore(
const CallInst &
I,
4936 bool IsCompressing) {
4939 Value *Src0Operand =
I.getArgOperand(0);
4940 Value *PtrOperand =
I.getArgOperand(1);
4941 Value *MaskOperand =
I.getArgOperand(2);
4942 Align Alignment =
I.getParamAlign(1).valueOrOne();
4952 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
4955 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
4956 MachinePointerInfo(PtrOperand), MMOFlags,
4959 const auto &TLI =
DAG.getTargetLoweringInfo();
4962 !IsCompressing &&
TTI->hasConditionalLoadStoreForType(
4963 I.getArgOperand(0)->getType(),
true)
4969 DAG.setRoot(StoreNode);
4999 C =
C->getSplatValue();
5013 if (!
GEP ||
GEP->getParent() != CurBB)
5016 if (
GEP->getNumOperands() != 2)
5019 const Value *BasePtr =
GEP->getPointerOperand();
5020 const Value *IndexVal =
GEP->getOperand(
GEP->getNumOperands() - 1);
5026 TypeSize ScaleVal =
DL.getTypeAllocSize(
GEP->getResultElementType());
5031 if (ScaleVal != 1 &&
5043void SelectionDAGBuilder::visitMaskedScatter(
const CallInst &
I) {
5047 const Value *Ptr =
I.getArgOperand(1);
5051 Align Alignment =
I.getParamAlign(1).valueOrOne();
5052 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5061 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5071 EVT IdxVT =
Index.getValueType();
5079 SDValue Scatter =
DAG.getMaskedScatter(
DAG.getVTList(MVT::Other), VT, sdl,
5081 DAG.setRoot(Scatter);
5085void SelectionDAGBuilder::visitMaskedLoad(
const CallInst &
I,
bool IsExpanding) {
5088 Value *PtrOperand =
I.getArgOperand(0);
5089 Value *MaskOperand =
I.getArgOperand(1);
5090 Value *Src0Operand =
I.getArgOperand(2);
5091 Align Alignment =
I.getParamAlign(0).valueOrOne();
5099 AAMDNodes AAInfo =
I.getAAMetadata();
5106 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
5109 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
5111 if (
I.hasMetadata(LLVMContext::MD_invariant_load))
5114 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5115 MachinePointerInfo(PtrOperand), MMOFlags,
5118 const auto &TLI =
DAG.getTargetLoweringInfo();
5125 TTI->hasConditionalLoadStoreForType(Src0Operand->
getType(),
5130 DAG.getMaskedLoad(VT, sdl, InChain, Ptr,
Offset, Mask, Src0, VT, MMO,
5137void SelectionDAGBuilder::visitMaskedGather(
const CallInst &
I) {
5141 const Value *Ptr =
I.getArgOperand(0);
5145 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5147 Align Alignment =
I.getParamAlign(0).valueOrOne();
5158 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5170 EVT IdxVT =
Index.getValueType();
5179 DAG.getMaskedGather(
DAG.getVTList(VT, MVT::Other), VT, sdl,
Ops, MMO,
5195 SDVTList VTs =
DAG.getVTList(MemVT, MVT::i1, MVT::Other);
5197 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5200 MachineFunction &MF =
DAG.getMachineFunction();
5202 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5203 DAG.getEVTAlign(MemVT), AAMDNodes(),
nullptr, SSID, SuccessOrdering,
5207 dl, MemVT, VTs, InChain,
5215 DAG.setRoot(OutChain);
5218void SelectionDAGBuilder::visitAtomicRMW(
const AtomicRMWInst &
I) {
5221 switch (
I.getOperation()) {
5263 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5266 MachineFunction &MF =
DAG.getMachineFunction();
5268 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5269 DAG.getEVTAlign(MemVT), AAMDNodes(),
nullptr, SSID, Ordering);
5272 DAG.getAtomic(NT, dl, MemVT, InChain,
5279 DAG.setRoot(OutChain);
5282void SelectionDAGBuilder::visitFence(
const FenceInst &
I) {
5284 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5287 Ops[1] =
DAG.getTargetConstant((
unsigned)
I.getOrdering(), dl,
5289 Ops[2] =
DAG.getTargetConstant(
I.getSyncScopeID(), dl,
5296void SelectionDAGBuilder::visitAtomicLoad(
const LoadInst &
I) {
5303 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5314 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5315 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5316 I.getAlign(), AAMDNodes(), Ranges, SSID, Order);
5326 L =
DAG.getPtrExtOrTrunc(L, dl, VT);
5329 DAG.setRoot(OutChain);
5332void SelectionDAGBuilder::visitAtomicStore(
const StoreInst &
I) {
5340 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5350 MachineFunction &MF =
DAG.getMachineFunction();
5352 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5353 I.getAlign(), AAMDNodes(),
nullptr, SSID, Ordering);
5357 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVT);
5364 DAG.setRoot(OutChain);
5372std::pair<bool, bool>
5373SelectionDAGBuilder::getTargetIntrinsicCallProperties(
const CallBase &
I) {
5375 bool HasChain = !
F->doesNotAccessMemory();
5377 HasChain &&
F->onlyReadsMemory() &&
F->willReturn() &&
F->doesNotThrow();
5379 return {HasChain, OnlyLoad};
5383 const CallBase &
I,
bool HasChain,
bool OnlyLoad,
5385 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5392 Ops.push_back(
DAG.getRoot());
5405 for (
unsigned i = 0, e =
I.arg_size(); i != e; ++i) {
5406 const Value *Arg =
I.getArgOperand(i);
5407 if (!
I.paramHasAttr(i, Attribute::ImmArg)) {
5415 assert(CI->getBitWidth() <= 64 &&
5416 "large intrinsic immediates not handled");
5417 Ops.push_back(
DAG.getTargetConstant(*CI, SDLoc(), VT));
5424 if (std::optional<OperandBundleUse> Bundle =
5426 auto *Sym = Bundle->Inputs[0].get();
5429 Ops.push_back(SDSym);
5432 if (std::optional<OperandBundleUse> Bundle =
5434 Value *Token = Bundle->Inputs[0].get();
5436 assert(
Ops.back().getValueType() != MVT::Glue &&
5437 "Did not expect another glue node here.");
5440 Ops.push_back(ConvControlToken);
5448 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5456 return DAG.getVTList(ValueVTs);
5460SDValue SelectionDAGBuilder::getTargetNonMemIntrinsicNode(
5483 if (
I.getType()->isVoidTy())
5498void SelectionDAGBuilder::visitTargetIntrinsic(
const CallInst &
I,
5500 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
5504 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5507 TargetLowering::IntrinsicInfo *
Info = !Infos.
empty() ? &Infos[0] :
nullptr;
5510 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad, Info);
5511 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
5516 Flags.copyFMF(*FPMO);
5517 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
5524 if (!Infos.
empty()) {
5527 MachineFunction &MF =
DAG.getMachineFunction();
5529 for (
const auto &Info : Infos) {
5532 MachinePointerInfo MPI;
5534 MPI = MachinePointerInfo(
Info.ptrVal,
Info.offset);
5535 else if (
Info.fallbackAddressSpace)
5536 MPI = MachinePointerInfo(*
Info.fallbackAddressSpace);
5537 EVT MemVT =
Info.memVT;
5539 if (
Size.hasValue() && !
Size.getValue())
5541 Align Alignment =
Info.align.value_or(
DAG.getEVTAlign(MemVT));
5543 MPI,
Info.flags,
Size, Alignment,
I.getAAMetadata(),
5551 Result = getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
5554 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
5611 SDValue TwoToFractionalPartOfX;
5688 if (
Op.getValueType() == MVT::f32 &&
5712 if (
Op.getValueType() == MVT::f32 &&
5811 if (
Op.getValueType() == MVT::f32 &&
5895 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log2ofMantissa);
5908 if (
Op.getValueType() == MVT::f32 &&
5985 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log10ofMantissa);
5996 if (
Op.getValueType() == MVT::f32 &&
6009 bool IsExp10 =
false;
6010 if (
LHS.getValueType() == MVT::f32 &&
RHS.getValueType() == MVT::f32 &&
6014 IsExp10 = LHSC->isExactlyValue(Ten);
6041 unsigned Val = RHSC->getSExtValue();
6070 CurSquare, CurSquare);
6075 if (RHSC->getSExtValue() < 0)
6089 EVT VT =
LHS.getValueType();
6112 if ((ScaleInt > 0 || (Saturating &&
Signed)) &&
6116 Opcode, VT, ScaleInt);
6151 switch (
N.getOpcode()) {
6155 Op.getValueType().getSizeInBits());
6180bool SelectionDAGBuilder::EmitFuncArgumentDbgValue(
6187 MachineFunction &MF =
DAG.getMachineFunction();
6188 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
6192 auto MakeVRegDbgValue = [&](
Register Reg, DIExpression *FragExpr,
6197 auto &Inst =
TII->get(TargetOpcode::DBG_INSTR_REF);
6204 auto *NewDIExpr = FragExpr;
6211 return BuildMI(MF,
DL, Inst,
false, MOs, Variable, NewDIExpr);
6214 auto &Inst =
TII->get(TargetOpcode::DBG_VALUE);
6215 return BuildMI(MF,
DL, Inst, Indirect,
Reg, Variable, FragExpr);
6219 if (Kind == FuncArgumentDbgValueKind::Value) {
6224 if (!IsInEntryBlock)
6240 bool VariableIsFunctionInputArg =
Variable->isParameter() &&
6241 !
DL->getInlinedAt();
6243 if (!IsInPrologue && !VariableIsFunctionInputArg)
6277 if (VariableIsFunctionInputArg) {
6279 if (ArgNo >=
FuncInfo.DescribedArgs.size())
6280 FuncInfo.DescribedArgs.resize(ArgNo + 1,
false);
6281 else if (!IsInPrologue &&
FuncInfo.DescribedArgs.test(ArgNo))
6282 return !NodeMap[
V].getNode();
6287 bool IsIndirect =
false;
6288 std::optional<MachineOperand>
Op;
6290 int FI =
FuncInfo.getArgumentFrameIndex(Arg);
6291 if (FI != std::numeric_limits<int>::max())
6295 if (!
Op &&
N.getNode()) {
6298 if (ArgRegsAndSizes.
size() == 1)
6299 Reg = ArgRegsAndSizes.
front().first;
6302 MachineRegisterInfo &RegInfo = MF.
getRegInfo();
6309 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6313 if (!
Op &&
N.getNode()) {
6317 if (FrameIndexSDNode *FINode =
6324 auto splitMultiRegDbgValue =
6337 uint64_t ExprFragmentSizeInBits = ExprFragmentInfo->SizeInBits;
6340 if (
Offset >= ExprFragmentSizeInBits)
6344 if (
Offset + RegFragmentSizeInBits > ExprFragmentSizeInBits) {
6345 RegFragmentSizeInBits = ExprFragmentSizeInBits -
Offset;
6350 Expr,
Offset, RegFragmentSizeInBits);
6354 if (!FragmentExpr) {
6355 SDDbgValue *SDV =
DAG.getConstantDbgValue(
6357 DAG.AddDbgValue(SDV,
false);
6360 MachineInstr *NewMI = MakeVRegDbgValue(
6361 Reg, *FragmentExpr, Kind != FuncArgumentDbgValueKind::Value);
6362 FuncInfo.ArgDbgValues.push_back(NewMI);
6371 if (VMI !=
FuncInfo.ValueMap.end()) {
6372 const auto &TLI =
DAG.getTargetLoweringInfo();
6373 RegsForValue RFV(
V->getContext(), TLI,
DAG.getDataLayout(), VMI->second,
6374 V->getType(), std::nullopt);
6375 if (RFV.occupiesMultipleRegs())
6376 return splitMultiRegDbgValue(RFV.getRegsAndSizes());
6379 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6380 }
else if (ArgRegsAndSizes.
size() > 1) {
6383 return splitMultiRegDbgValue(ArgRegsAndSizes);
6391 "Expected inlined-at fields to agree");
6392 MachineInstr *NewMI =
nullptr;
6395 NewMI = MakeVRegDbgValue(
Op->getReg(), Expr, IsIndirect);
6397 NewMI =
BuildMI(MF,
DL,
TII->get(TargetOpcode::DBG_VALUE),
true, *
Op,
6401 FuncInfo.ArgDbgValues.push_back(NewMI);
6410 unsigned DbgSDNodeOrder) {
6422 return DAG.getFrameIndexDbgValue(Variable, Expr, FISDN->getIndex(),
6423 false, dl, DbgSDNodeOrder);
6425 return DAG.getDbgValue(Variable, Expr,
N.getNode(),
N.getResNo(),
6426 false, dl, DbgSDNodeOrder);
6431 case Intrinsic::smul_fix:
6433 case Intrinsic::umul_fix:
6435 case Intrinsic::smul_fix_sat:
6437 case Intrinsic::umul_fix_sat:
6439 case Intrinsic::sdiv_fix:
6441 case Intrinsic::udiv_fix:
6443 case Intrinsic::sdiv_fix_sat:
6445 case Intrinsic::udiv_fix_sat:
6458 "expected call_preallocated_setup Value");
6459 for (
const auto *U : PreallocatedSetup->
users()) {
6461 const Function *Fn = UseCall->getCalledFunction();
6462 if (!Fn || Fn->
getIntrinsicID() != Intrinsic::call_preallocated_arg) {
6472bool SelectionDAGBuilder::visitEntryValueDbgValue(
6482 auto ArgIt =
FuncInfo.ValueMap.find(Arg);
6483 if (ArgIt ==
FuncInfo.ValueMap.end()) {
6485 dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6486 "couldn't find an associated register for the Argument\n");
6489 Register ArgVReg = ArgIt->getSecond();
6491 for (
auto [PhysReg, VirtReg] :
FuncInfo.RegInfo->liveins())
6492 if (ArgVReg == VirtReg || ArgVReg == PhysReg) {
6493 SDDbgValue *SDV =
DAG.getVRegDbgValue(
6494 Variable, Expr, PhysReg,
false , DbgLoc, SDNodeOrder);
6495 DAG.AddDbgValue(SDV,
false );
6498 LLVM_DEBUG(
dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6499 "couldn't find a physical register\n");
6504void SelectionDAGBuilder::visitConvergenceControl(
const CallInst &
I,
6507 switch (Intrinsic) {
6508 case Intrinsic::experimental_convergence_anchor:
6511 case Intrinsic::experimental_convergence_entry:
6514 case Intrinsic::experimental_convergence_loop: {
6516 auto *Token = Bundle->Inputs[0].get();
6524void SelectionDAGBuilder::visitVectorHistogram(
const CallInst &
I,
6525 unsigned IntrinsicID) {
6528 assert(IntrinsicID == Intrinsic::experimental_vector_histogram_add &&
6529 "Tried to lower unsupported histogram type");
6535 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6536 DataLayout TargetDL =
DAG.getDataLayout();
6538 Align Alignment =
DAG.getEVTAlign(VT);
6551 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
6552 MachinePointerInfo(AS),
6563 EVT IdxVT =
Index.getValueType();
6574 SDValue ID =
DAG.getTargetConstant(IntrinsicID, sdl, MVT::i32);
6577 SDValue Histogram =
DAG.getMaskedHistogram(
DAG.getVTList(MVT::Other), VT, sdl,
6581 DAG.setRoot(Histogram);
6584void SelectionDAGBuilder::visitVectorExtractLastActive(
const CallInst &
I,
6586 assert(Intrinsic == Intrinsic::experimental_vector_extract_last_active &&
6587 "Tried lowering invalid vector extract last");
6589 const DataLayout &Layout =
DAG.getDataLayout();
6593 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6603 EVT BoolVT =
Mask.getValueType().getScalarType();
6605 Result =
DAG.getSelect(sdl, ResVT, AnyActive, Result, PassThru);
6612void SelectionDAGBuilder::visitIntrinsicCall(
const CallInst &
I,
6614 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6621 Flags.copyFMF(*FPOp);
6623 switch (Intrinsic) {
6626 visitTargetIntrinsic(
I, Intrinsic);
6628 case Intrinsic::vscale: {
6633 case Intrinsic::vastart: visitVAStart(
I);
return;
6634 case Intrinsic::vaend: visitVAEnd(
I);
return;
6635 case Intrinsic::vacopy: visitVACopy(
I);
return;
6636 case Intrinsic::returnaddress:
6641 case Intrinsic::addressofreturnaddress:
6646 case Intrinsic::sponentry:
6651 case Intrinsic::frameaddress:
6656 case Intrinsic::read_volatile_register:
6657 case Intrinsic::read_register: {
6658 Value *
Reg =
I.getArgOperand(0);
6664 DAG.getVTList(VT, MVT::Other), Chain,
RegName);
6669 case Intrinsic::write_register: {
6670 Value *
Reg =
I.getArgOperand(0);
6671 Value *RegValue =
I.getArgOperand(1);
6679 case Intrinsic::memcpy:
6680 case Intrinsic::memcpy_inline: {
6686 "memcpy_inline needs constant size");
6688 Align DstAlign = MCI.getDestAlign().valueOrOne();
6689 Align SrcAlign = MCI.getSourceAlign().valueOrOne();
6690 Align Alignment = std::min(DstAlign, SrcAlign);
6691 bool isVol = MCI.isVolatile();
6695 SDValue MC =
DAG.getMemcpy(Root, sdl, Dst, Src,
Size, Alignment, isVol,
6696 MCI.isForceInlined(), &
I, std::nullopt,
6697 MachinePointerInfo(
I.getArgOperand(0)),
6698 MachinePointerInfo(
I.getArgOperand(1)),
6700 updateDAGForMaybeTailCall(MC);
6703 case Intrinsic::memset:
6704 case Intrinsic::memset_inline: {
6710 "memset_inline needs constant size");
6712 Align DstAlign = MSII.getDestAlign().valueOrOne();
6713 bool isVol = MSII.isVolatile();
6716 Root, sdl, Dst, Value,
Size, DstAlign, isVol, MSII.isForceInlined(),
6717 &
I, MachinePointerInfo(
I.getArgOperand(0)),
I.getAAMetadata());
6718 updateDAGForMaybeTailCall(MC);
6721 case Intrinsic::memmove: {
6727 Align DstAlign = MMI.getDestAlign().valueOrOne();
6728 Align SrcAlign = MMI.getSourceAlign().valueOrOne();
6729 Align Alignment = std::min(DstAlign, SrcAlign);
6730 bool isVol = MMI.isVolatile();
6734 SDValue MM =
DAG.getMemmove(Root, sdl, Op1, Op2, Op3, Alignment, isVol, &
I,
6736 MachinePointerInfo(
I.getArgOperand(0)),
6737 MachinePointerInfo(
I.getArgOperand(1)),
6739 updateDAGForMaybeTailCall(MM);
6742 case Intrinsic::memcpy_element_unordered_atomic: {
6748 Type *LengthTy =
MI.getLength()->getType();
6749 unsigned ElemSz =
MI.getElementSizeInBytes();
6753 isTC, MachinePointerInfo(
MI.getRawDest()),
6754 MachinePointerInfo(
MI.getRawSource()));
6755 updateDAGForMaybeTailCall(MC);
6758 case Intrinsic::memmove_element_unordered_atomic: {
6764 Type *LengthTy =
MI.getLength()->getType();
6765 unsigned ElemSz =
MI.getElementSizeInBytes();
6769 isTC, MachinePointerInfo(
MI.getRawDest()),
6770 MachinePointerInfo(
MI.getRawSource()));
6771 updateDAGForMaybeTailCall(MC);
6774 case Intrinsic::memset_element_unordered_atomic: {
6780 Type *LengthTy =
MI.getLength()->getType();
6781 unsigned ElemSz =
MI.getElementSizeInBytes();
6785 isTC, MachinePointerInfo(
MI.getRawDest()));
6786 updateDAGForMaybeTailCall(MC);
6789 case Intrinsic::call_preallocated_setup: {
6791 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6798 case Intrinsic::call_preallocated_arg: {
6800 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6814 case Intrinsic::eh_typeid_for: {
6817 unsigned TypeID =
DAG.getMachineFunction().getTypeIDFor(GV);
6818 Res =
DAG.getConstant(
TypeID, sdl, MVT::i32);
6823 case Intrinsic::eh_return_i32:
6824 case Intrinsic::eh_return_i64:
6825 DAG.getMachineFunction().setCallsEHReturn(
true);
6832 case Intrinsic::eh_unwind_init:
6833 DAG.getMachineFunction().setCallsUnwindInit(
true);
6835 case Intrinsic::eh_dwarf_cfa:
6840 case Intrinsic::eh_sjlj_callsite: {
6842 assert(
FuncInfo.getCurrentCallSite() == 0 &&
"Overlapping call sites!");
6847 case Intrinsic::eh_sjlj_functioncontext: {
6849 MachineFrameInfo &MFI =
DAG.getMachineFunction().getFrameInfo();
6852 int FI =
FuncInfo.StaticAllocaMap[FnCtx];
6856 case Intrinsic::eh_sjlj_setjmp: {
6861 DAG.getVTList(MVT::i32, MVT::Other),
Ops);
6863 DAG.setRoot(
Op.getValue(1));
6866 case Intrinsic::eh_sjlj_longjmp:
6870 case Intrinsic::eh_sjlj_setup_dispatch:
6874 case Intrinsic::masked_gather:
6875 visitMaskedGather(
I);
6877 case Intrinsic::masked_load:
6880 case Intrinsic::masked_scatter:
6881 visitMaskedScatter(
I);
6883 case Intrinsic::masked_store:
6884 visitMaskedStore(
I);
6886 case Intrinsic::masked_expandload:
6887 visitMaskedLoad(
I,
true );
6889 case Intrinsic::masked_compressstore:
6890 visitMaskedStore(
I,
true );
6892 case Intrinsic::powi:
6896 case Intrinsic::log:
6899 case Intrinsic::log2:
6903 case Intrinsic::log10:
6907 case Intrinsic::exp:
6910 case Intrinsic::exp2:
6914 case Intrinsic::pow:
6918 case Intrinsic::sqrt:
6919 case Intrinsic::fabs:
6920 case Intrinsic::sin:
6921 case Intrinsic::cos:
6922 case Intrinsic::tan:
6923 case Intrinsic::asin:
6924 case Intrinsic::acos:
6925 case Intrinsic::atan:
6926 case Intrinsic::sinh:
6927 case Intrinsic::cosh:
6928 case Intrinsic::tanh:
6929 case Intrinsic::exp10:
6930 case Intrinsic::floor:
6931 case Intrinsic::ceil:
6932 case Intrinsic::trunc:
6933 case Intrinsic::rint:
6934 case Intrinsic::nearbyint:
6935 case Intrinsic::round:
6936 case Intrinsic::roundeven:
6937 case Intrinsic::canonicalize: {
6940 switch (Intrinsic) {
6942 case Intrinsic::sqrt: Opcode =
ISD::FSQRT;
break;
6943 case Intrinsic::fabs: Opcode =
ISD::FABS;
break;
6944 case Intrinsic::sin: Opcode =
ISD::FSIN;
break;
6945 case Intrinsic::cos: Opcode =
ISD::FCOS;
break;
6946 case Intrinsic::tan: Opcode =
ISD::FTAN;
break;
6947 case Intrinsic::asin: Opcode =
ISD::FASIN;
break;
6948 case Intrinsic::acos: Opcode =
ISD::FACOS;
break;
6949 case Intrinsic::atan: Opcode =
ISD::FATAN;
break;
6950 case Intrinsic::sinh: Opcode =
ISD::FSINH;
break;
6951 case Intrinsic::cosh: Opcode =
ISD::FCOSH;
break;
6952 case Intrinsic::tanh: Opcode =
ISD::FTANH;
break;
6953 case Intrinsic::exp10: Opcode =
ISD::FEXP10;
break;
6954 case Intrinsic::floor: Opcode =
ISD::FFLOOR;
break;
6955 case Intrinsic::ceil: Opcode =
ISD::FCEIL;
break;
6956 case Intrinsic::trunc: Opcode =
ISD::FTRUNC;
break;
6957 case Intrinsic::rint: Opcode =
ISD::FRINT;
break;
6959 case Intrinsic::round: Opcode =
ISD::FROUND;
break;
6966 getValue(
I.getArgOperand(0)).getValueType(),
6970 case Intrinsic::atan2:
6972 getValue(
I.getArgOperand(0)).getValueType(),
6976 case Intrinsic::lround:
6977 case Intrinsic::llround:
6978 case Intrinsic::lrint:
6979 case Intrinsic::llrint: {
6982 switch (Intrinsic) {
6984 case Intrinsic::lround: Opcode =
ISD::LROUND;
break;
6986 case Intrinsic::lrint: Opcode =
ISD::LRINT;
break;
6987 case Intrinsic::llrint: Opcode =
ISD::LLRINT;
break;
6996 case Intrinsic::minnum:
6998 getValue(
I.getArgOperand(0)).getValueType(),
7002 case Intrinsic::maxnum:
7004 getValue(
I.getArgOperand(0)).getValueType(),
7008 case Intrinsic::minimum:
7010 getValue(
I.getArgOperand(0)).getValueType(),
7014 case Intrinsic::maximum:
7016 getValue(
I.getArgOperand(0)).getValueType(),
7020 case Intrinsic::minimumnum:
7022 getValue(
I.getArgOperand(0)).getValueType(),
7026 case Intrinsic::maximumnum:
7028 getValue(
I.getArgOperand(0)).getValueType(),
7032 case Intrinsic::copysign:
7034 getValue(
I.getArgOperand(0)).getValueType(),
7038 case Intrinsic::ldexp:
7040 getValue(
I.getArgOperand(0)).getValueType(),
7044 case Intrinsic::modf:
7045 case Intrinsic::sincos:
7046 case Intrinsic::sincospi:
7047 case Intrinsic::frexp: {
7049 switch (Intrinsic) {
7052 case Intrinsic::sincos:
7055 case Intrinsic::sincospi:
7058 case Intrinsic::modf:
7061 case Intrinsic::frexp:
7067 SDVTList VTs =
DAG.getVTList(ValueVTs);
7069 &
I,
DAG.getNode(Opcode, sdl, VTs,
getValue(
I.getArgOperand(0)), Flags));
7072 case Intrinsic::arithmetic_fence: {
7074 getValue(
I.getArgOperand(0)).getValueType(),
7078 case Intrinsic::fma:
7084#define INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC) \
7085 case Intrinsic::INTRINSIC:
7086#include "llvm/IR/ConstrainedOps.def"
7089#define BEGIN_REGISTER_VP_INTRINSIC(VPID, ...) case Intrinsic::VPID:
7090#include "llvm/IR/VPIntrinsics.def"
7093 case Intrinsic::fptrunc_round: {
7097 std::optional<RoundingMode> RoundMode =
7105 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
7110 DAG.getTargetConstant((
int)*RoundMode, sdl, MVT::i32));
7115 case Intrinsic::fmuladd: {
7120 getValue(
I.getArgOperand(0)).getValueType(),
7127 getValue(
I.getArgOperand(0)).getValueType(),
7143 case Intrinsic::fptosi_sat: {
7150 case Intrinsic::fptoui_sat: {
7157 case Intrinsic::convert_from_arbitrary_fp: {
7162 const fltSemantics *SrcSem =
7165 DAG.getContext()->emitError(
7166 "convert_from_arbitrary_fp: not implemented format '" + FormatStr +
7177 DAG.getTargetConstant(
static_cast<int>(SemEnum), sdl, MVT::i32);
7182 case Intrinsic::set_rounding:
7188 case Intrinsic::is_fpclass: {
7189 const DataLayout DLayout =
DAG.getDataLayout();
7191 EVT ArgVT = TLI.
getValueType(DLayout,
I.getArgOperand(0)->getType());
7194 MachineFunction &MF =
DAG.getMachineFunction();
7198 Flags.setNoFPExcept(
7199 !
F.getAttributes().hasFnAttr(llvm::Attribute::StrictFP));
7215 case Intrinsic::get_fpenv: {
7216 const DataLayout DLayout =
DAG.getDataLayout();
7218 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7233 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7236 Chain =
DAG.getGetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7237 Res =
DAG.getLoad(EnvVT, sdl, Chain, Temp, MPI);
7243 case Intrinsic::set_fpenv: {
7244 const DataLayout DLayout =
DAG.getDataLayout();
7247 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7260 Chain =
DAG.getStore(Chain, sdl, Env, Temp, MPI, TempAlign,
7262 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7265 Chain =
DAG.getSetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7270 case Intrinsic::reset_fpenv:
7273 case Intrinsic::get_fpmode:
7282 case Intrinsic::set_fpmode:
7287 case Intrinsic::reset_fpmode: {
7292 case Intrinsic::pcmarker: {
7297 case Intrinsic::readcyclecounter: {
7300 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7305 case Intrinsic::readsteadycounter: {
7308 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7313 case Intrinsic::bitreverse:
7315 getValue(
I.getArgOperand(0)).getValueType(),
7318 case Intrinsic::bswap:
7320 getValue(
I.getArgOperand(0)).getValueType(),
7323 case Intrinsic::cttz: {
7331 case Intrinsic::ctlz: {
7339 case Intrinsic::ctpop: {
7345 case Intrinsic::fshl:
7346 case Intrinsic::fshr: {
7347 bool IsFSHL =
Intrinsic == Intrinsic::fshl;
7351 EVT VT =
X.getValueType();
7362 case Intrinsic::clmul: {
7368 case Intrinsic::sadd_sat: {
7374 case Intrinsic::uadd_sat: {
7380 case Intrinsic::ssub_sat: {
7386 case Intrinsic::usub_sat: {
7392 case Intrinsic::sshl_sat:
7393 case Intrinsic::ushl_sat: {
7397 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
7402 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
7405 "Unexpected shift type");
7414 case Intrinsic::smul_fix:
7415 case Intrinsic::umul_fix:
7416 case Intrinsic::smul_fix_sat:
7417 case Intrinsic::umul_fix_sat: {
7425 case Intrinsic::sdiv_fix:
7426 case Intrinsic::udiv_fix:
7427 case Intrinsic::sdiv_fix_sat:
7428 case Intrinsic::udiv_fix_sat: {
7433 Op1, Op2, Op3,
DAG, TLI));
7436 case Intrinsic::smax: {
7442 case Intrinsic::smin: {
7448 case Intrinsic::umax: {
7454 case Intrinsic::umin: {
7460 case Intrinsic::abs: {
7466 case Intrinsic::scmp: {
7473 case Intrinsic::ucmp: {
7480 case Intrinsic::stackaddress:
7481 case Intrinsic::stacksave: {
7486 Res =
DAG.getNode(SDOpcode, sdl,
DAG.getVTList(VT, MVT::Other),
Op);
7491 case Intrinsic::stackrestore:
7495 case Intrinsic::get_dynamic_area_offset: {
7504 case Intrinsic::stackguard: {
7505 MachineFunction &MF =
DAG.getMachineFunction();
7511 Res =
DAG.getPtrExtOrTrunc(Res, sdl, PtrTy);
7515 LLVMContext &Ctx = *
DAG.getContext();
7516 Ctx.
diagnose(DiagnosticInfoGeneric(
"unable to lower stackguard"));
7523 MachinePointerInfo(
Global, 0), Align,
7532 case Intrinsic::stackprotector: {
7534 MachineFunction &MF =
DAG.getMachineFunction();
7554 Chain, sdl, Src, FIN,
7561 case Intrinsic::objectsize:
7564 case Intrinsic::is_constant:
7567 case Intrinsic::annotation:
7568 case Intrinsic::ptr_annotation:
7569 case Intrinsic::launder_invariant_group:
7570 case Intrinsic::strip_invariant_group:
7575 case Intrinsic::type_test:
7576 case Intrinsic::public_type_test:
7578 "LowerTypeTests pass before code generation");
7581 case Intrinsic::assume:
7582 case Intrinsic::experimental_noalias_scope_decl:
7583 case Intrinsic::var_annotation:
7584 case Intrinsic::sideeffect:
7589 case Intrinsic::codeview_annotation: {
7591 MachineFunction &MF =
DAG.getMachineFunction();
7600 case Intrinsic::init_trampoline: {
7608 Ops[4] =
DAG.getSrcValue(
I.getArgOperand(0));
7616 case Intrinsic::adjust_trampoline:
7621 case Intrinsic::gcroot: {
7622 assert(
DAG.getMachineFunction().getFunction().hasGC() &&
7623 "only valid in functions with gc specified, enforced by Verifier");
7625 const Value *Alloca =
I.getArgOperand(0)->stripPointerCasts();
7632 case Intrinsic::gcread:
7633 case Intrinsic::gcwrite:
7635 case Intrinsic::get_rounding:
7641 case Intrinsic::expect:
7642 case Intrinsic::expect_with_probability:
7648 case Intrinsic::ubsantrap:
7649 case Intrinsic::debugtrap:
7650 case Intrinsic::trap: {
7651 StringRef TrapFuncName =
7652 I.getAttributes().getFnAttr(
"trap-func-name").getValueAsString();
7653 if (TrapFuncName.
empty()) {
7654 switch (Intrinsic) {
7655 case Intrinsic::trap:
7658 case Intrinsic::debugtrap:
7661 case Intrinsic::ubsantrap:
7664 DAG.getTargetConstant(
7670 DAG.addNoMergeSiteInfo(
DAG.getRoot().getNode(),
7671 I.hasFnAttr(Attribute::NoMerge));
7675 if (Intrinsic == Intrinsic::ubsantrap) {
7676 Value *Arg =
I.getArgOperand(0);
7680 TargetLowering::CallLoweringInfo CLI(
DAG);
7681 CLI.setDebugLoc(sdl).setChain(
getRoot()).setLibCallee(
7683 DAG.getExternalSymbol(TrapFuncName.
data(),
7686 CLI.NoMerge =
I.hasFnAttr(Attribute::NoMerge);
7692 case Intrinsic::allow_runtime_check:
7693 case Intrinsic::allow_ubsan_check:
7697 case Intrinsic::uadd_with_overflow:
7698 case Intrinsic::sadd_with_overflow:
7699 case Intrinsic::usub_with_overflow:
7700 case Intrinsic::ssub_with_overflow:
7701 case Intrinsic::umul_with_overflow:
7702 case Intrinsic::smul_with_overflow: {
7704 switch (Intrinsic) {
7706 case Intrinsic::uadd_with_overflow:
Op =
ISD::UADDO;
break;
7707 case Intrinsic::sadd_with_overflow:
Op =
ISD::SADDO;
break;
7708 case Intrinsic::usub_with_overflow:
Op =
ISD::USUBO;
break;
7709 case Intrinsic::ssub_with_overflow:
Op =
ISD::SSUBO;
break;
7710 case Intrinsic::umul_with_overflow:
Op =
ISD::UMULO;
break;
7711 case Intrinsic::smul_with_overflow:
Op =
ISD::SMULO;
break;
7719 SDVTList VTs =
DAG.getVTList(ResultVT, OverflowVT);
7723 case Intrinsic::prefetch: {
7738 std::nullopt, Flags);
7744 DAG.setRoot(Result);
7747 case Intrinsic::lifetime_start:
7748 case Intrinsic::lifetime_end: {
7749 bool IsStart = (
Intrinsic == Intrinsic::lifetime_start);
7755 if (!LifetimeObject)
7760 auto SI =
FuncInfo.StaticAllocaMap.find(LifetimeObject);
7761 if (SI ==
FuncInfo.StaticAllocaMap.end())
7765 Res =
DAG.getLifetimeNode(IsStart, sdl,
getRoot(), FrameIndex);
7769 case Intrinsic::pseudoprobe: {
7777 case Intrinsic::invariant_start:
7782 case Intrinsic::invariant_end:
7785 case Intrinsic::clear_cache: {
7790 {InputChain, StartVal, EndVal});
7795 case Intrinsic::donothing:
7796 case Intrinsic::seh_try_begin:
7797 case Intrinsic::seh_scope_begin:
7798 case Intrinsic::seh_try_end:
7799 case Intrinsic::seh_scope_end:
7802 case Intrinsic::experimental_stackmap:
7805 case Intrinsic::experimental_patchpoint_void:
7806 case Intrinsic::experimental_patchpoint:
7809 case Intrinsic::experimental_gc_statepoint:
7812 case Intrinsic::experimental_gc_result:
7815 case Intrinsic::experimental_gc_relocate:
7818 case Intrinsic::instrprof_cover:
7820 case Intrinsic::instrprof_increment:
7822 case Intrinsic::instrprof_timestamp:
7824 case Intrinsic::instrprof_value_profile:
7826 case Intrinsic::instrprof_mcdc_parameters:
7828 case Intrinsic::instrprof_mcdc_tvbitmap_update:
7830 case Intrinsic::localescape: {
7831 MachineFunction &MF =
DAG.getMachineFunction();
7832 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
7836 for (
unsigned Idx = 0,
E =
I.arg_size(); Idx <
E; ++Idx) {
7842 "can only escape static allocas");
7847 TII->get(TargetOpcode::LOCAL_ESCAPE))
7855 case Intrinsic::localrecover: {
7857 MachineFunction &MF =
DAG.getMachineFunction();
7863 unsigned(Idx->getLimitedValue(std::numeric_limits<int>::max()));
7867 Value *
FP =
I.getArgOperand(1);
7873 SDValue OffsetSym =
DAG.getMCSymbol(FrameAllocSym, PtrVT);
7878 SDValue Add =
DAG.getMemBasePlusOffset(FPVal, OffsetVal, sdl);
7884 case Intrinsic::fake_use: {
7885 Value *
V =
I.getArgOperand(0);
7890 auto FakeUseValue = [&]() ->
SDValue {
7904 if (!FakeUseValue || FakeUseValue.isUndef())
7907 Ops[1] = FakeUseValue;
7916 case Intrinsic::reloc_none: {
7921 DAG.getTargetExternalSymbol(
7927 case Intrinsic::cond_loop: {
7937 case Intrinsic::eh_exceptionpointer:
7938 case Intrinsic::eh_exceptioncode: {
7944 SDValue N =
DAG.getCopyFromReg(
DAG.getEntryNode(), sdl, VReg, PtrVT);
7945 if (Intrinsic == Intrinsic::eh_exceptioncode)
7946 N =
DAG.getZExtOrTrunc(
N, sdl, MVT::i32);
7950 case Intrinsic::xray_customevent: {
7953 const auto &Triple =
DAG.getTarget().getTargetTriple();
7962 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
7964 Ops.push_back(LogEntryVal);
7965 Ops.push_back(StrSizeVal);
7966 Ops.push_back(Chain);
7972 MachineSDNode *MN =
DAG.getMachineNode(TargetOpcode::PATCHABLE_EVENT_CALL,
7975 DAG.setRoot(patchableNode);
7979 case Intrinsic::xray_typedevent: {
7982 const auto &Triple =
DAG.getTarget().getTargetTriple();
7994 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
7996 Ops.push_back(LogTypeId);
7997 Ops.push_back(LogEntryVal);
7998 Ops.push_back(StrSizeVal);
7999 Ops.push_back(Chain);
8005 MachineSDNode *MN =
DAG.getMachineNode(
8006 TargetOpcode::PATCHABLE_TYPED_EVENT_CALL, sdl, NodeTys,
Ops);
8008 DAG.setRoot(patchableNode);
8012 case Intrinsic::experimental_deoptimize:
8015 case Intrinsic::stepvector:
8018 case Intrinsic::vector_reduce_fadd:
8019 case Intrinsic::vector_reduce_fmul:
8020 case Intrinsic::vector_reduce_add:
8021 case Intrinsic::vector_reduce_mul:
8022 case Intrinsic::vector_reduce_and:
8023 case Intrinsic::vector_reduce_or:
8024 case Intrinsic::vector_reduce_xor:
8025 case Intrinsic::vector_reduce_smax:
8026 case Intrinsic::vector_reduce_smin:
8027 case Intrinsic::vector_reduce_umax:
8028 case Intrinsic::vector_reduce_umin:
8029 case Intrinsic::vector_reduce_fmax:
8030 case Intrinsic::vector_reduce_fmin:
8031 case Intrinsic::vector_reduce_fmaximum:
8032 case Intrinsic::vector_reduce_fminimum:
8033 visitVectorReduce(
I, Intrinsic);
8036 case Intrinsic::icall_branch_funnel: {
8042 I.getArgOperand(1),
Offset,
DAG.getDataLayout()));
8045 "llvm.icall.branch.funnel operand must be a GlobalValue");
8046 Ops.push_back(
DAG.getTargetGlobalAddress(
Base, sdl, MVT::i64, 0));
8048 struct BranchFunnelTarget {
8054 for (
unsigned Op = 1,
N =
I.arg_size();
Op !=
N;
Op += 2) {
8057 if (ElemBase !=
Base)
8059 "to the same GlobalValue");
8065 "llvm.icall.branch.funnel operand must be a GlobalValue");
8071 [](
const BranchFunnelTarget &
T1,
const BranchFunnelTarget &T2) {
8072 return T1.Offset < T2.Offset;
8075 for (
auto &
T : Targets) {
8076 Ops.push_back(
DAG.getTargetConstant(
T.Offset, sdl, MVT::i32));
8077 Ops.push_back(
T.Target);
8080 Ops.push_back(
DAG.getRoot());
8081 SDValue N(
DAG.getMachineNode(TargetOpcode::ICALL_BRANCH_FUNNEL, sdl,
8090 case Intrinsic::wasm_landingpad_index:
8096 case Intrinsic::aarch64_settag:
8097 case Intrinsic::aarch64_settag_zero: {
8098 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
8099 bool ZeroMemory =
Intrinsic == Intrinsic::aarch64_settag_zero;
8102 getValue(
I.getArgOperand(1)), MachinePointerInfo(
I.getArgOperand(0)),
8108 case Intrinsic::amdgcn_cs_chain: {
8113 Type *RetTy =
I.getType();
8123 for (
unsigned Idx : {2, 3, 1}) {
8124 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8126 Arg.setAttributes(&
I, Idx);
8127 Args.push_back(Arg);
8130 assert(Args[0].IsInReg &&
"SGPR args should be marked inreg");
8131 assert(!Args[1].IsInReg &&
"VGPR args should not be marked inreg");
8132 Args[2].IsInReg =
true;
8135 for (
unsigned Idx = 4; Idx <
I.arg_size(); ++Idx) {
8136 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8138 Arg.setAttributes(&
I, Idx);
8139 Args.push_back(Arg);
8142 TargetLowering::CallLoweringInfo CLI(
DAG);
8145 .setCallee(CC, RetTy, Callee, std::move(Args))
8148 .setConvergent(
I.isConvergent());
8150 std::pair<SDValue, SDValue>
Result =
8154 "Should've lowered as tail call");
8159 case Intrinsic::amdgcn_call_whole_wave: {
8161 bool isTailCall =
I.isTailCall();
8164 for (
unsigned Idx = 1; Idx <
I.arg_size(); ++Idx) {
8165 TargetLowering::ArgListEntry Arg(
getValue(
I.getArgOperand(Idx)),
8166 I.getArgOperand(Idx)->getType());
8167 Arg.setAttributes(&
I, Idx);
8174 Args.push_back(Arg);
8179 auto *Token = Bundle->Inputs[0].get();
8180 ConvControlToken =
getValue(Token);
8183 TargetLowering::CallLoweringInfo CLI(
DAG);
8187 getValue(
I.getArgOperand(0)), std::move(Args))
8191 .setConvergent(
I.isConvergent())
8192 .setConvergenceControlToken(ConvControlToken);
8195 std::pair<SDValue, SDValue>
Result =
8198 if (
Result.first.getNode())
8202 case Intrinsic::ptrmask: {
8218 auto HighOnes =
DAG.getNode(
8219 ISD::SHL, sdl, PtrVT,
DAG.getAllOnesConstant(sdl, PtrVT),
8220 DAG.getShiftAmountConstant(
Mask.getValueType().getFixedSizeInBits(),
8223 DAG.getZExtOrTrunc(Mask, sdl, PtrVT), HighOnes);
8224 }
else if (
Mask.getValueType() != PtrVT)
8225 Mask =
DAG.getPtrExtOrTrunc(Mask, sdl, PtrVT);
8231 case Intrinsic::threadlocal_address: {
8235 case Intrinsic::get_active_lane_mask: {
8239 EVT ElementVT =
Index.getValueType();
8250 SDValue VectorIndex =
DAG.getSplat(VecTy, sdl, Index);
8251 SDValue VectorTripCount =
DAG.getSplat(VecTy, sdl, TripCount);
8252 SDValue VectorStep =
DAG.getStepVector(sdl, VecTy);
8255 SDValue SetCC =
DAG.getSetCC(sdl, CCVT, VectorInduction,
8260 case Intrinsic::experimental_get_vector_length: {
8262 "Expected positive VF");
8267 EVT CountVT =
Count.getValueType();
8270 visitTargetIntrinsic(
I, Intrinsic);
8279 if (CountVT.
bitsLT(VT)) {
8284 SDValue MaxEVL =
DAG.getElementCount(sdl, CountVT,
8294 case Intrinsic::vector_partial_reduce_add: {
8302 case Intrinsic::vector_partial_reduce_fadd: {
8310 case Intrinsic::experimental_cttz_elts: {
8313 EVT OpVT =
Op.getValueType();
8316 visitTargetIntrinsic(
I, Intrinsic);
8332 ConstantRange VScaleRange(1,
true);
8361 case Intrinsic::vector_insert: {
8369 if (
Index.getValueType() != VectorIdxTy)
8370 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8377 case Intrinsic::vector_extract: {
8385 if (
Index.getValueType() != VectorIdxTy)
8386 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8392 case Intrinsic::experimental_vector_match: {
8398 EVT ResVT =
Mask.getValueType();
8404 visitTargetIntrinsic(
I, Intrinsic);
8408 SDValue Ret =
DAG.getConstant(0, sdl, ResVT);
8410 for (
unsigned i = 0; i < SearchSize; ++i) {
8413 DAG.getVectorIdxConstant(i, sdl));
8416 Ret =
DAG.getNode(
ISD::OR, sdl, ResVT, Ret, Cmp);
8422 case Intrinsic::vector_reverse:
8423 visitVectorReverse(
I);
8425 case Intrinsic::vector_splice_left:
8426 case Intrinsic::vector_splice_right:
8427 visitVectorSplice(
I);
8429 case Intrinsic::callbr_landingpad:
8430 visitCallBrLandingPad(
I);
8432 case Intrinsic::vector_interleave2:
8433 visitVectorInterleave(
I, 2);
8435 case Intrinsic::vector_interleave3:
8436 visitVectorInterleave(
I, 3);
8438 case Intrinsic::vector_interleave4:
8439 visitVectorInterleave(
I, 4);
8441 case Intrinsic::vector_interleave5:
8442 visitVectorInterleave(
I, 5);
8444 case Intrinsic::vector_interleave6:
8445 visitVectorInterleave(
I, 6);
8447 case Intrinsic::vector_interleave7:
8448 visitVectorInterleave(
I, 7);
8450 case Intrinsic::vector_interleave8:
8451 visitVectorInterleave(
I, 8);
8453 case Intrinsic::vector_deinterleave2:
8454 visitVectorDeinterleave(
I, 2);
8456 case Intrinsic::vector_deinterleave3:
8457 visitVectorDeinterleave(
I, 3);
8459 case Intrinsic::vector_deinterleave4:
8460 visitVectorDeinterleave(
I, 4);
8462 case Intrinsic::vector_deinterleave5:
8463 visitVectorDeinterleave(
I, 5);
8465 case Intrinsic::vector_deinterleave6:
8466 visitVectorDeinterleave(
I, 6);
8468 case Intrinsic::vector_deinterleave7:
8469 visitVectorDeinterleave(
I, 7);
8471 case Intrinsic::vector_deinterleave8:
8472 visitVectorDeinterleave(
I, 8);
8474 case Intrinsic::experimental_vector_compress:
8476 getValue(
I.getArgOperand(0)).getValueType(),
8481 case Intrinsic::experimental_convergence_anchor:
8482 case Intrinsic::experimental_convergence_entry:
8483 case Intrinsic::experimental_convergence_loop:
8484 visitConvergenceControl(
I, Intrinsic);
8486 case Intrinsic::experimental_vector_histogram_add: {
8487 visitVectorHistogram(
I, Intrinsic);
8490 case Intrinsic::experimental_vector_extract_last_active: {
8491 visitVectorExtractLastActive(
I, Intrinsic);
8494 case Intrinsic::loop_dependence_war_mask:
8499 DAG.getConstant(0, sdl, MVT::i64)));
8501 case Intrinsic::loop_dependence_raw_mask:
8506 DAG.getConstant(0, sdl, MVT::i64)));
8511void SelectionDAGBuilder::pushFPOpOutChain(
SDValue Result,
8527 PendingConstrainedFP.push_back(OutChain);
8530 PendingConstrainedFPStrict.push_back(OutChain);
8535void SelectionDAGBuilder::visitConstrainedFPIntrinsic(
8549 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8551 SDVTList VTs =
DAG.getVTList(VT, MVT::Other);
8555 Flags.setNoFPExcept(
true);
8558 Flags.copyFMF(*FPOp);
8563#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
8564 case Intrinsic::INTRINSIC: \
8565 Opcode = ISD::STRICT_##DAGN; \
8567#include "llvm/IR/ConstrainedOps.def"
8568 case Intrinsic::experimental_constrained_fmuladd: {
8575 pushFPOpOutChain(
Mul, EB);
8598 if (
DAG.isKnownNeverNaN(Opers[1]) &&
DAG.isKnownNeverNaN(Opers[2]))
8606 pushFPOpOutChain(Result, EB);
8613 std::optional<unsigned> ResOPC;
8615 case Intrinsic::vp_ctlz: {
8617 ResOPC = IsZeroUndef ? ISD::VP_CTLZ_ZERO_UNDEF : ISD::VP_CTLZ;
8620 case Intrinsic::vp_cttz: {
8622 ResOPC = IsZeroUndef ? ISD::VP_CTTZ_ZERO_UNDEF : ISD::VP_CTTZ;
8625 case Intrinsic::vp_cttz_elts: {
8627 ResOPC = IsZeroPoison ? ISD::VP_CTTZ_ELTS_ZERO_UNDEF : ISD::VP_CTTZ_ELTS;
8630#define HELPER_MAP_VPID_TO_VPSD(VPID, VPSD) \
8631 case Intrinsic::VPID: \
8632 ResOPC = ISD::VPSD; \
8634#include "llvm/IR/VPIntrinsics.def"
8639 "Inconsistency: no SDNode available for this VPIntrinsic!");
8641 if (*ResOPC == ISD::VP_REDUCE_SEQ_FADD ||
8642 *ResOPC == ISD::VP_REDUCE_SEQ_FMUL) {
8644 return *ResOPC == ISD::VP_REDUCE_SEQ_FADD ? ISD::VP_REDUCE_FADD
8645 : ISD::VP_REDUCE_FMUL;
8651void SelectionDAGBuilder::visitVPLoad(
8663 Alignment =
DAG.getEVTAlign(VT);
8666 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8667 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8670 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8671 MachinePointerInfo(PtrOperand), MMOFlags,
8673 LD =
DAG.getLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8680void SelectionDAGBuilder::visitVPLoadFF(
8683 assert(OpValues.
size() == 3 &&
"Unexpected number of operands");
8693 Alignment =
DAG.getEVTAlign(VT);
8696 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8697 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8700 LD =
DAG.getLoadFFVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8705 setValue(&VPIntrin,
DAG.getMergeValues({LD.getValue(0), Trunc},
DL));
8708void SelectionDAGBuilder::visitVPGather(
8712 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8724 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8726 *Alignment, AAInfo, Ranges);
8736 EVT IdxVT =
Index.getValueType();
8742 LD =
DAG.getGatherVP(
8743 DAG.getVTList(VT, MVT::Other), VT,
DL,
8744 {DAG.getRoot(), Base, Index, Scale, OpValues[1], OpValues[2]}, MMO,
8750void SelectionDAGBuilder::visitVPStore(
8754 EVT VT = OpValues[0].getValueType();
8759 Alignment =
DAG.getEVTAlign(VT);
8762 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8765 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8766 MachinePointerInfo(PtrOperand), MMOFlags,
8775void SelectionDAGBuilder::visitVPScatter(
8778 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8780 EVT VT = OpValues[0].getValueType();
8790 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8792 *Alignment, AAInfo);
8802 EVT IdxVT =
Index.getValueType();
8808 ST =
DAG.getScatterVP(
DAG.getVTList(MVT::Other), VT,
DL,
8809 {getMemoryRoot(), OpValues[0], Base, Index, Scale,
8810 OpValues[2], OpValues[3]},
8816void SelectionDAGBuilder::visitVPStridedLoad(
8828 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8830 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8833 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8835 *Alignment, AAInfo, Ranges);
8837 SDValue LD =
DAG.getStridedLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1],
8838 OpValues[2], OpValues[3], MMO,
8846void SelectionDAGBuilder::visitVPStridedStore(
8850 EVT VT = OpValues[0].getValueType();
8856 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8859 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8861 *Alignment, AAInfo);
8865 DAG.getUNDEF(OpValues[1].getValueType()), OpValues[2], OpValues[3],
8873void SelectionDAGBuilder::visitVPCmp(
const VPCmpIntrinsic &VPIntrin) {
8874 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8889 "Unexpected target EVL type");
8892 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
8894 if (
DAG.isKnownNeverNaN(Op1) &&
DAG.isKnownNeverNaN(Op2))
8897 DAG.getSetCCVP(
DL, DestVT, Op1, Op2, Condition, MaskOp, EVL));
8900void SelectionDAGBuilder::visitVectorPredicationIntrinsic(
8908 return visitVPCmp(*CmpI);
8911 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8913 SDVTList VTs =
DAG.getVTList(ValueVTs);
8919 "Unexpected target EVL type");
8923 for (
unsigned I = 0;
I < VPIntrin.
arg_size(); ++
I) {
8925 if (
I == EVLParamPos)
8932 SDNodeFlags SDFlags;
8940 visitVPLoad(VPIntrin, ValueVTs[0], OpValues);
8942 case ISD::VP_LOAD_FF:
8943 visitVPLoadFF(VPIntrin, ValueVTs[0], ValueVTs[1], OpValues);
8945 case ISD::VP_GATHER:
8946 visitVPGather(VPIntrin, ValueVTs[0], OpValues);
8948 case ISD::EXPERIMENTAL_VP_STRIDED_LOAD:
8949 visitVPStridedLoad(VPIntrin, ValueVTs[0], OpValues);
8952 visitVPStore(VPIntrin, OpValues);
8954 case ISD::VP_SCATTER:
8955 visitVPScatter(VPIntrin, OpValues);
8957 case ISD::EXPERIMENTAL_VP_STRIDED_STORE:
8958 visitVPStridedStore(VPIntrin, OpValues);
8960 case ISD::VP_FMULADD: {
8961 assert(OpValues.
size() == 5 &&
"Unexpected number of operands");
8962 SDNodeFlags SDFlags;
8967 setValue(&VPIntrin,
DAG.getNode(ISD::VP_FMA,
DL, VTs, OpValues, SDFlags));
8970 ISD::VP_FMUL,
DL, VTs,
8971 {OpValues[0], OpValues[1], OpValues[3], OpValues[4]}, SDFlags);
8973 DAG.getNode(ISD::VP_FADD,
DL, VTs,
8974 {
Mul, OpValues[2], OpValues[3], OpValues[4]}, SDFlags);
8979 case ISD::VP_IS_FPCLASS: {
8980 const DataLayout DLayout =
DAG.getDataLayout();
8982 auto Constant = OpValues[1]->getAsZExtVal();
8985 {OpValues[0],
Check, OpValues[2], OpValues[3]});
8989 case ISD::VP_INTTOPTR: {
9000 case ISD::VP_PTRTOINT: {
9002 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9015 case ISD::VP_CTLZ_ZERO_UNDEF:
9017 case ISD::VP_CTTZ_ZERO_UNDEF:
9018 case ISD::VP_CTTZ_ELTS_ZERO_UNDEF:
9019 case ISD::VP_CTTZ_ELTS: {
9021 DAG.getNode(Opcode,
DL, VTs, {OpValues[0], OpValues[2], OpValues[3]});
9031 MachineFunction &MF =
DAG.getMachineFunction();
9039 unsigned CallSiteIndex =
FuncInfo.getCurrentCallSite();
9040 if (CallSiteIndex) {
9054 assert(BeginLabel &&
"BeginLabel should've been set");
9056 MachineFunction &MF =
DAG.getMachineFunction();
9068 assert(
II &&
"II should've been set");
9079std::pair<SDValue, SDValue>
9093 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
9096 "Non-null chain expected with non-tail call!");
9097 assert((Result.second.getNode() || !Result.first.getNode()) &&
9098 "Null value expected with tail call!");
9100 if (!Result.second.getNode()) {
9107 PendingExports.clear();
9109 DAG.setRoot(Result.second);
9127 if (!isMustTailCall &&
9128 Caller->getFnAttribute(
"disable-tail-calls").getValueAsBool())
9134 if (
DAG.getTargetLoweringInfo().supportSwiftError() &&
9135 Caller->getAttributes().hasAttrSomewhere(Attribute::SwiftError))
9144 bool isTailCall,
bool isMustTailCall,
9147 auto &
DL =
DAG.getDataLayout();
9154 const Value *SwiftErrorVal =
nullptr;
9161 const Value *V = *
I;
9164 if (V->getType()->isEmptyTy())
9169 Entry.setAttributes(&CB,
I - CB.
arg_begin());
9181 Args.push_back(Entry);
9192 Value *V = Bundle->Inputs[0];
9194 Entry.IsCFGuardTarget =
true;
9195 Args.push_back(Entry);
9208 "Target doesn't support calls with kcfi operand bundles.");
9216 auto *Token = Bundle->Inputs[0].get();
9217 ConvControlToken =
getValue(Token);
9228 .
setCallee(RetTy, FTy, Callee, std::move(Args), CB)
9241 "This target doesn't support calls with ptrauth operand bundles.");
9245 std::pair<SDValue, SDValue> Result =
lowerInvokable(CLI, EHPadBB);
9247 if (Result.first.getNode()) {
9262 DAG.setRoot(CopyNode);
9278 LoadTy, Builder.DAG.getDataLayout()))
9279 return Builder.getValue(LoadCst);
9285 bool ConstantMemory =
false;
9288 if (Builder.BatchAA && Builder.BatchAA->pointsToConstantMemory(PtrVal)) {
9289 Root = Builder.DAG.getEntryNode();
9290 ConstantMemory =
true;
9293 Root = Builder.DAG.getRoot();
9298 Builder.DAG.getLoad(LoadVT, Builder.getCurSDLoc(), Root, Ptr,
9301 if (!ConstantMemory)
9302 Builder.PendingLoads.push_back(LoadVal.
getValue(1));
9308void SelectionDAGBuilder::processIntegerCallValue(
const Instruction &
I,
9311 EVT VT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9322bool SelectionDAGBuilder::visitMemCmpBCmpCall(
const CallInst &
I) {
9323 const Value *
LHS =
I.getArgOperand(0), *
RHS =
I.getArgOperand(1);
9324 const Value *
Size =
I.getArgOperand(2);
9327 EVT CallVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9333 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9337 if (Res.first.getNode()) {
9338 processIntegerCallValue(
I, Res.first,
true);
9352 auto hasFastLoadsAndCompare = [&](
unsigned NumBits) {
9353 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9375 switch (NumBitsToCompare) {
9387 LoadVT = hasFastLoadsAndCompare(NumBitsToCompare);
9400 LoadL =
DAG.getBitcast(CmpVT, LoadL);
9401 LoadR =
DAG.getBitcast(CmpVT, LoadR);
9405 processIntegerCallValue(
I, Cmp,
false);
9414bool SelectionDAGBuilder::visitMemChrCall(
const CallInst &
I) {
9415 const Value *Src =
I.getArgOperand(0);
9416 const Value *
Char =
I.getArgOperand(1);
9417 const Value *
Length =
I.getArgOperand(2);
9419 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9420 std::pair<SDValue, SDValue> Res =
9423 MachinePointerInfo(Src));
9424 if (Res.first.getNode()) {
9438bool SelectionDAGBuilder::visitMemCCpyCall(
const CallInst &
I) {
9439 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9446 processIntegerCallValue(
I, Res.first,
true);
9458bool SelectionDAGBuilder::visitMemPCpyCall(
const CallInst &
I) {
9463 Align DstAlign =
DAG.InferPtrAlign(Dst).valueOrOne();
9464 Align SrcAlign =
DAG.InferPtrAlign(Src).valueOrOne();
9466 Align Alignment = std::min(DstAlign, SrcAlign);
9475 Root, sdl, Dst, Src,
Size, Alignment,
false,
false,
nullptr,
9476 std::nullopt, MachinePointerInfo(
I.getArgOperand(0)),
9477 MachinePointerInfo(
I.getArgOperand(1)),
I.getAAMetadata());
9479 "** memcpy should not be lowered as TailCall in mempcpy context **");
9483 Size =
DAG.getSExtOrTrunc(
Size, sdl, Dst.getValueType());
9496bool SelectionDAGBuilder::visitStrCpyCall(
const CallInst &
I,
bool isStpcpy) {
9497 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9499 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9502 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), isStpcpy, &
I);
9503 if (Res.first.getNode()) {
9505 DAG.setRoot(Res.second);
9517bool SelectionDAGBuilder::visitStrCmpCall(
const CallInst &
I) {
9518 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9520 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9523 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), &
I);
9524 if (Res.first.getNode()) {
9525 processIntegerCallValue(
I, Res.first,
true);
9538bool SelectionDAGBuilder::visitStrLenCall(
const CallInst &
I) {
9539 const Value *Arg0 =
I.getArgOperand(0);
9541 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9544 if (Res.first.getNode()) {
9545 processIntegerCallValue(
I, Res.first,
false);
9558bool SelectionDAGBuilder::visitStrNLenCall(
const CallInst &
I) {
9559 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9561 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9562 std::pair<SDValue, SDValue> Res =
9565 MachinePointerInfo(Arg0));
9566 if (Res.first.getNode()) {
9567 processIntegerCallValue(
I, Res.first,
false);
9580bool SelectionDAGBuilder::visitStrstrCall(
const CallInst &
I) {
9581 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9582 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9586 processIntegerCallValue(
I, Res.first,
false);
9598bool SelectionDAGBuilder::visitUnaryFloatCall(
const CallInst &
I,
9603 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9620bool SelectionDAGBuilder::visitBinaryFloatCall(
const CallInst &
I,
9625 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9638void SelectionDAGBuilder::visitCall(
const CallInst &
I) {
9640 if (
I.isInlineAsm()) {
9647 if (Function *
F =
I.getCalledFunction()) {
9648 if (
F->isDeclaration()) {
9650 if (
unsigned IID =
F->getIntrinsicID()) {
9651 visitIntrinsicCall(
I, IID);
9662 if (!
I.isNoBuiltin() && !
F->hasLocalLinkage() &&
F->hasName() &&
9663 LibInfo->getLibFunc(*
F, Func) &&
LibInfo->hasOptimizedCodeGen(Func)) {
9667 if (visitMemCmpBCmpCall(
I))
9670 case LibFunc_copysign:
9671 case LibFunc_copysignf:
9672 case LibFunc_copysignl:
9675 if (
I.onlyReadsMemory()) {
9720 case LibFunc_atan2f:
9721 case LibFunc_atan2l:
9746 case LibFunc_sqrt_finite:
9747 case LibFunc_sqrtf_finite:
9748 case LibFunc_sqrtl_finite:
9765 case LibFunc_exp10f:
9766 case LibFunc_exp10l:
9771 case LibFunc_ldexpf:
9772 case LibFunc_ldexpl:
9776 case LibFunc_strstr:
9777 if (visitStrstrCall(
I))
9780 case LibFunc_memcmp:
9781 if (visitMemCmpBCmpCall(
I))
9784 case LibFunc_memccpy:
9785 if (visitMemCCpyCall(
I))
9788 case LibFunc_mempcpy:
9789 if (visitMemPCpyCall(
I))
9792 case LibFunc_memchr:
9793 if (visitMemChrCall(
I))
9796 case LibFunc_strcpy:
9797 if (visitStrCpyCall(
I,
false))
9800 case LibFunc_stpcpy:
9801 if (visitStrCpyCall(
I,
true))
9804 case LibFunc_strcmp:
9805 if (visitStrCmpCall(
I))
9808 case LibFunc_strlen:
9809 if (visitStrLenCall(
I))
9812 case LibFunc_strnlen:
9813 if (visitStrNLenCall(
I))
9837 if (
I.hasDeoptState())
9854 const Value *Discriminator = PAB->Inputs[1];
9856 assert(
Key->getType()->isIntegerTy(32) &&
"Invalid ptrauth key");
9857 assert(Discriminator->getType()->isIntegerTy(64) &&
9858 "Invalid ptrauth discriminator");
9863 if (CalleeCPA->isKnownCompatibleWith(
Key, Discriminator,
9864 DAG.getDataLayout()))
9904 for (
const auto &Code : Codes)
9919 SDISelAsmOperandInfo &MatchingOpInfo,
9921 if (OpInfo.ConstraintVT == MatchingOpInfo.ConstraintVT)
9927 std::pair<unsigned, const TargetRegisterClass *> MatchRC =
9929 OpInfo.ConstraintVT);
9930 std::pair<unsigned, const TargetRegisterClass *> InputRC =
9932 MatchingOpInfo.ConstraintVT);
9933 const bool OutOpIsIntOrFP =
9934 OpInfo.ConstraintVT.isInteger() || OpInfo.ConstraintVT.isFloatingPoint();
9935 const bool InOpIsIntOrFP = MatchingOpInfo.ConstraintVT.isInteger() ||
9936 MatchingOpInfo.ConstraintVT.isFloatingPoint();
9937 if ((OutOpIsIntOrFP != InOpIsIntOrFP) || (MatchRC.second != InputRC.second)) {
9940 " with a matching output constraint of"
9941 " incompatible type!");
9943 MatchingOpInfo.ConstraintVT = OpInfo.ConstraintVT;
9950 SDISelAsmOperandInfo &OpInfo,
9963 const Value *OpVal = OpInfo.CallOperandVal;
9981 DL.getPrefTypeAlign(Ty),
false,
9984 Chain = DAG.
getTruncStore(Chain, Location, OpInfo.CallOperand, StackSlot,
9987 OpInfo.CallOperand = StackSlot;
10000static std::optional<unsigned>
10002 SDISelAsmOperandInfo &OpInfo,
10003 SDISelAsmOperandInfo &RefOpInfo) {
10014 return std::nullopt;
10018 unsigned AssignedReg;
10021 &
TRI, RefOpInfo.ConstraintCode, RefOpInfo.ConstraintVT);
10024 return std::nullopt;
10029 const MVT RegVT = *
TRI.legalclasstypes_begin(*RC);
10031 if (OpInfo.ConstraintVT != MVT::Other && RegVT != MVT::Untyped) {
10040 !
TRI.isTypeLegalForClass(*RC, OpInfo.ConstraintVT)) {
10045 if (RegVT.
getSizeInBits() == OpInfo.ConstraintVT.getSizeInBits()) {
10050 OpInfo.CallOperand =
10052 OpInfo.ConstraintVT = RegVT;
10056 }
else if (RegVT.
isInteger() && OpInfo.ConstraintVT.isFloatingPoint()) {
10059 OpInfo.CallOperand =
10061 OpInfo.ConstraintVT = VT;
10068 if (OpInfo.isMatchingInputConstraint())
10069 return std::nullopt;
10071 EVT ValueVT = OpInfo.ConstraintVT;
10072 if (OpInfo.ConstraintVT == MVT::Other)
10076 unsigned NumRegs = 1;
10077 if (OpInfo.ConstraintVT != MVT::Other)
10092 I = std::find(
I, RC->
end(), AssignedReg);
10093 if (
I == RC->
end()) {
10096 return {AssignedReg};
10100 for (; NumRegs; --NumRegs, ++
I) {
10101 assert(
I != RC->
end() &&
"Ran out of registers to allocate!");
10106 OpInfo.AssignedRegs =
RegsForValue(Regs, RegVT, ValueVT);
10107 return std::nullopt;
10112 const std::vector<SDValue> &AsmNodeOperands) {
10115 for (; OperandNo; --OperandNo) {
10117 unsigned OpFlag = AsmNodeOperands[CurOp]->getAsZExtVal();
10120 (
F.isRegDefKind() ||
F.isRegDefEarlyClobberKind() ||
F.isMemKind()) &&
10121 "Skipped past definitions?");
10122 CurOp +=
F.getNumOperandRegisters() + 1;
10130 unsigned Flags = 0;
10133 explicit ExtraFlags(
const CallBase &
Call) {
10135 if (
IA->hasSideEffects())
10137 if (
IA->isAlignStack())
10139 if (
IA->canThrow())
10146 void update(
const TargetLowering::AsmOperandInfo &OpInfo) {
10162 unsigned get()
const {
return Flags; }
10185void SelectionDAGBuilder::visitInlineAsm(
const CallBase &
Call,
10192 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10194 DAG.getDataLayout(),
DAG.getSubtarget().getRegisterInfo(),
Call);
10198 bool HasSideEffect =
IA->hasSideEffects();
10199 ExtraFlags ExtraInfo(
Call);
10201 for (
auto &
T : TargetConstraints) {
10202 ConstraintOperands.
push_back(SDISelAsmOperandInfo(
T));
10203 SDISelAsmOperandInfo &OpInfo = ConstraintOperands.
back();
10205 if (OpInfo.CallOperandVal)
10206 OpInfo.CallOperand =
getValue(OpInfo.CallOperandVal);
10208 if (!HasSideEffect)
10209 HasSideEffect = OpInfo.hasMemory(TLI);
10221 return emitInlineAsmError(
Call,
"constraint '" + Twine(
T.ConstraintCode) +
10222 "' expects an integer constant "
10225 ExtraInfo.update(
T);
10233 if (EmitEHLabels) {
10234 assert(EHPadBB &&
"InvokeInst must have an EHPadBB");
10238 if (IsCallBr || EmitEHLabels) {
10246 if (EmitEHLabels) {
10247 Chain = lowerStartEH(Chain, EHPadBB, BeginLabel);
10252 IA->collectAsmStrs(AsmStrs);
10255 for (SDISelAsmOperandInfo &OpInfo : ConstraintOperands) {
10263 if (OpInfo.hasMatchingInput()) {
10264 SDISelAsmOperandInfo &Input = ConstraintOperands[OpInfo.MatchingInput];
10295 if (OpInfo.isIndirect &&
isFunction(OpInfo.CallOperand) &&
10298 OpInfo.isIndirect =
false;
10305 !OpInfo.isIndirect) {
10306 assert((OpInfo.isMultipleAlternative ||
10308 "Can only indirectify direct input operands!");
10314 OpInfo.CallOperandVal =
nullptr;
10317 OpInfo.isIndirect =
true;
10323 std::vector<SDValue> AsmNodeOperands;
10324 AsmNodeOperands.push_back(
SDValue());
10325 AsmNodeOperands.push_back(
DAG.getTargetExternalSymbol(
10332 AsmNodeOperands.push_back(
DAG.getMDNode(SrcLoc));
10336 AsmNodeOperands.push_back(
DAG.getTargetConstant(
10341 for (SDISelAsmOperandInfo &OpInfo : ConstraintOperands) {
10343 SDISelAsmOperandInfo &RefOpInfo =
10344 OpInfo.isMatchingInputConstraint()
10345 ? ConstraintOperands[OpInfo.getMatchedOperand()]
10347 const auto RegError =
10350 const MachineFunction &MF =
DAG.getMachineFunction();
10352 const char *
RegName =
TRI.getName(*RegError);
10353 emitInlineAsmError(
Call,
"register '" + Twine(
RegName) +
10354 "' allocated for constraint '" +
10355 Twine(OpInfo.ConstraintCode) +
10356 "' does not match required type");
10360 auto DetectWriteToReservedRegister = [&]() {
10361 const MachineFunction &MF =
DAG.getMachineFunction();
10366 emitInlineAsmError(
Call,
"write to reserved register '" +
10375 !OpInfo.isMatchingInputConstraint())) &&
10376 "Only address as input operand is allowed.");
10378 switch (OpInfo.Type) {
10384 "Failed to convert memory constraint code to constraint id.");
10388 OpFlags.setMemConstraint(ConstraintID);
10389 AsmNodeOperands.push_back(
DAG.getTargetConstant(OpFlags,
getCurSDLoc(),
10391 AsmNodeOperands.push_back(OpInfo.CallOperand);
10396 if (OpInfo.AssignedRegs.
Regs.empty()) {
10397 emitInlineAsmError(
10398 Call,
"couldn't allocate output register for constraint '" +
10399 Twine(OpInfo.ConstraintCode) +
"'");
10403 if (DetectWriteToReservedRegister())
10417 SDValue InOperandVal = OpInfo.CallOperand;
10419 if (OpInfo.isMatchingInputConstraint()) {
10424 InlineAsm::Flag
Flag(AsmNodeOperands[CurOp]->getAsZExtVal());
10425 if (
Flag.isRegDefKind() ||
Flag.isRegDefEarlyClobberKind()) {
10426 if (OpInfo.isIndirect) {
10428 emitInlineAsmError(
Call,
"inline asm not supported yet: "
10429 "don't know how to handle tied "
10430 "indirect register inputs");
10435 MachineFunction &MF =
DAG.getMachineFunction();
10440 MVT RegVT =
R->getSimpleValueType(0);
10441 const TargetRegisterClass *RC =
10444 :
TRI.getMinimalPhysRegClass(TiedReg);
10445 for (
unsigned i = 0, e =
Flag.getNumOperandRegisters(); i != e; ++i)
10448 RegsForValue MatchedRegs(Regs, RegVT, InOperandVal.
getValueType());
10452 MatchedRegs.getCopyToRegs(InOperandVal,
DAG, dl, Chain, &Glue, &
Call);
10454 OpInfo.getMatchedOperand(), dl,
DAG,
10459 assert(
Flag.isMemKind() &&
"Unknown matching constraint!");
10460 assert(
Flag.getNumOperandRegisters() == 1 &&
10461 "Unexpected number of operands");
10464 Flag.clearMemConstraint();
10465 Flag.setMatchingOp(OpInfo.getMatchedOperand());
10466 AsmNodeOperands.push_back(
DAG.getTargetConstant(
10468 AsmNodeOperands.push_back(AsmNodeOperands[CurOp+1]);
10479 std::vector<SDValue>
Ops;
10485 emitInlineAsmError(
Call,
"value out of range for constraint '" +
10486 Twine(OpInfo.ConstraintCode) +
"'");
10490 emitInlineAsmError(
Call,
10491 "invalid operand for inline asm constraint '" +
10492 Twine(OpInfo.ConstraintCode) +
"'");
10498 AsmNodeOperands.push_back(
DAG.getTargetConstant(
10505 assert((OpInfo.isIndirect ||
10507 "Operand must be indirect to be a mem!");
10510 "Memory operands expect pointer values");
10515 "Failed to convert memory constraint code to constraint id.");
10519 ResOpType.setMemConstraint(ConstraintID);
10520 AsmNodeOperands.push_back(
DAG.getTargetConstant(ResOpType,
10523 AsmNodeOperands.push_back(InOperandVal);
10531 "Failed to convert memory constraint code to constraint id.");
10535 SDValue AsmOp = InOperandVal;
10539 AsmOp =
DAG.getTargetGlobalAddress(GA->getGlobal(),
getCurSDLoc(),
10545 ResOpType.setMemConstraint(ConstraintID);
10547 AsmNodeOperands.push_back(
10550 AsmNodeOperands.push_back(AsmOp);
10556 emitInlineAsmError(
Call,
"unknown asm constraint '" +
10557 Twine(OpInfo.ConstraintCode) +
"'");
10562 if (OpInfo.isIndirect) {
10563 emitInlineAsmError(
10564 Call,
"Don't know how to handle indirect register inputs yet "
10565 "for constraint '" +
10566 Twine(OpInfo.ConstraintCode) +
"'");
10571 if (OpInfo.AssignedRegs.
Regs.empty()) {
10572 emitInlineAsmError(
Call,
10573 "couldn't allocate input reg for constraint '" +
10574 Twine(OpInfo.ConstraintCode) +
"'");
10578 if (DetectWriteToReservedRegister())
10587 0, dl,
DAG, AsmNodeOperands);
10593 if (!OpInfo.AssignedRegs.
Regs.empty())
10603 if (Glue.
getNode()) AsmNodeOperands.push_back(Glue);
10607 DAG.getVTList(MVT::Other, MVT::Glue), AsmNodeOperands);
10619 ResultTypes = StructResult->elements();
10620 else if (!CallResultType->
isVoidTy())
10621 ResultTypes =
ArrayRef(CallResultType);
10623 auto CurResultType = ResultTypes.
begin();
10624 auto handleRegAssign = [&](
SDValue V) {
10625 assert(CurResultType != ResultTypes.
end() &&
"Unexpected value");
10626 assert((*CurResultType)->isSized() &&
"Unexpected unsized type");
10627 EVT ResultVT = TLI.
getValueType(
DAG.getDataLayout(), *CurResultType);
10639 if (ResultVT !=
V.getValueType() &&
10642 else if (ResultVT !=
V.getValueType() && ResultVT.
isInteger() &&
10643 V.getValueType().isInteger()) {
10649 assert(ResultVT ==
V.getValueType() &&
"Asm result value mismatch!");
10655 for (SDISelAsmOperandInfo &OpInfo : ConstraintOperands) {
10659 if (OpInfo.AssignedRegs.
Regs.empty())
10662 switch (OpInfo.ConstraintType) {
10666 Chain, &Glue, &
Call);
10678 assert(
false &&
"Unexpected unknown constraint");
10682 if (OpInfo.isIndirect) {
10683 const Value *Ptr = OpInfo.CallOperandVal;
10684 assert(Ptr &&
"Expected value CallOperandVal for indirect asm operand");
10686 MachinePointerInfo(Ptr));
10693 handleRegAssign(V);
10695 handleRegAssign(Val);
10701 if (!ResultValues.
empty()) {
10702 assert(CurResultType == ResultTypes.
end() &&
10703 "Mismatch in number of ResultTypes");
10705 "Mismatch in number of output operands in asm result");
10708 DAG.getVTList(ResultVTs), ResultValues);
10713 if (!OutChains.
empty())
10716 if (EmitEHLabels) {
10721 if (ResultValues.
empty() || HasSideEffect || !OutChains.
empty() || IsCallBr ||
10723 DAG.setRoot(Chain);
10726void SelectionDAGBuilder::emitInlineAsmError(
const CallBase &
Call,
10727 const Twine &Message) {
10728 LLVMContext &Ctx = *
DAG.getContext();
10732 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10736 if (ValueVTs.
empty())
10740 for (
const EVT &VT : ValueVTs)
10741 Ops.push_back(
DAG.getUNDEF(VT));
10746void SelectionDAGBuilder::visitVAStart(
const CallInst &
I) {
10750 DAG.getSrcValue(
I.getArgOperand(0))));
10753void SelectionDAGBuilder::visitVAArg(
const VAArgInst &
I) {
10754 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10755 const DataLayout &
DL =
DAG.getDataLayout();
10759 DL.getABITypeAlign(
I.getType()).value());
10760 DAG.setRoot(
V.getValue(1));
10762 if (
I.getType()->isPointerTy())
10763 V =
DAG.getPtrExtOrTrunc(
10768void SelectionDAGBuilder::visitVAEnd(
const CallInst &
I) {
10772 DAG.getSrcValue(
I.getArgOperand(0))));
10775void SelectionDAGBuilder::visitVACopy(
const CallInst &
I) {
10780 DAG.getSrcValue(
I.getArgOperand(0)),
10781 DAG.getSrcValue(
I.getArgOperand(1))));
10787 std::optional<ConstantRange> CR =
getRange(
I);
10789 if (!CR || CR->isFullSet() || CR->isEmptySet() || CR->isUpperWrapped())
10792 APInt Lo = CR->getUnsignedMin();
10793 if (!
Lo.isMinValue())
10796 APInt Hi = CR->getUnsignedMax();
10797 unsigned Bits = std::max(
Hi.getActiveBits(),
10805 DAG.getValueType(SmallVT));
10806 unsigned NumVals =
Op.getNode()->getNumValues();
10812 Ops.push_back(ZExt);
10813 for (
unsigned I = 1;
I != NumVals; ++
I)
10814 Ops.push_back(
Op.getValue(
I));
10816 return DAG.getMergeValues(
Ops,
SL);
10826 SDValue TestConst =
DAG.getTargetConstant(Classes,
SDLoc(), MVT::i32);
10834 for (
unsigned I = 0, E =
Ops.size();
I != E; ++
I) {
10837 MergeOp, TestConst);
10840 return DAG.getMergeValues(
Ops,
SL);
10851 unsigned ArgIdx,
unsigned NumArgs,
SDValue Callee,
Type *ReturnTy,
10854 Args.reserve(NumArgs);
10858 for (
unsigned ArgI = ArgIdx, ArgE = ArgIdx + NumArgs;
10859 ArgI != ArgE; ++ArgI) {
10860 const Value *V =
Call->getOperand(ArgI);
10862 assert(!V->getType()->isEmptyTy() &&
"Empty type passed to intrinsic.");
10865 Entry.setAttributes(
Call, ArgI);
10866 Args.push_back(Entry);
10871 .
setCallee(
Call->getCallingConv(), ReturnTy, Callee, std::move(Args),
10900 for (
unsigned I = StartIdx;
I <
Call.arg_size();
I++) {
10909 Ops.push_back(Builder.getValue(
Call.getArgOperand(
I)));
10915void SelectionDAGBuilder::visitStackmap(
const CallInst &CI) {
10941 Ops.push_back(Chain);
10942 Ops.push_back(InGlue);
10949 assert(
ID.getValueType() == MVT::i64);
10951 DAG.getTargetConstant(
ID->getAsZExtVal(),
DL,
ID.getValueType());
10952 Ops.push_back(IDConst);
10958 Ops.push_back(ShadConst);
10964 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
10968 Chain =
DAG.getCALLSEQ_END(Chain, 0, 0, InGlue,
DL);
10973 DAG.setRoot(Chain);
10976 FuncInfo.MF->getFrameInfo().setHasStackMap();
10980void SelectionDAGBuilder::visitPatchpoint(
const CallBase &CB,
10997 Callee =
DAG.getIntPtrConstant(ConstCallee->getZExtValue(), dl,
11000 Callee =
DAG.getTargetGlobalAddress(SymbolicCallee->getGlobal(),
11001 SDLoc(SymbolicCallee),
11002 SymbolicCallee->getValueType(0));
11012 "Not enough arguments provided to the patchpoint intrinsic");
11015 unsigned NumCallArgs = IsAnyRegCC ? 0 : NumArgs;
11019 TargetLowering::CallLoweringInfo CLI(
DAG);
11024 SDNode *CallEnd =
Result.second.getNode();
11033 "Expected a callseq node.");
11035 bool HasGlue =
Call->getGluedNode();
11060 Ops.push_back(Callee);
11066 NumCallRegArgs = IsAnyRegCC ? NumArgs : NumCallRegArgs;
11067 Ops.push_back(
DAG.getTargetConstant(NumCallRegArgs, dl, MVT::i32));
11070 Ops.push_back(
DAG.getTargetConstant((
unsigned)CC, dl, MVT::i32));
11075 for (
unsigned i = NumMetaOpers, e = NumMetaOpers + NumArgs; i !=
e; ++i)
11086 if (IsAnyRegCC && HasDef) {
11088 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11091 assert(ValueVTs.
size() == 1 &&
"Expected only one return value type.");
11096 NodeTys =
DAG.getVTList(ValueVTs);
11098 NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
11115 if (IsAnyRegCC && HasDef) {
11118 DAG.ReplaceAllUsesOfValuesWith(From, To, 2);
11124 FuncInfo.MF->getFrameInfo().setHasPatchPoint();
11127void SelectionDAGBuilder::visitVectorReduce(
const CallInst &
I,
11129 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11132 if (
I.arg_size() > 1)
11137 SDNodeFlags SDFlags;
11141 switch (Intrinsic) {
11142 case Intrinsic::vector_reduce_fadd:
11150 case Intrinsic::vector_reduce_fmul:
11158 case Intrinsic::vector_reduce_add:
11161 case Intrinsic::vector_reduce_mul:
11164 case Intrinsic::vector_reduce_and:
11167 case Intrinsic::vector_reduce_or:
11170 case Intrinsic::vector_reduce_xor:
11173 case Intrinsic::vector_reduce_smax:
11176 case Intrinsic::vector_reduce_smin:
11179 case Intrinsic::vector_reduce_umax:
11182 case Intrinsic::vector_reduce_umin:
11185 case Intrinsic::vector_reduce_fmax:
11188 case Intrinsic::vector_reduce_fmin:
11191 case Intrinsic::vector_reduce_fmaximum:
11194 case Intrinsic::vector_reduce_fminimum:
11208 Attrs.push_back(Attribute::SExt);
11210 Attrs.push_back(Attribute::ZExt);
11212 Attrs.push_back(Attribute::InReg);
11214 return AttributeList::get(CLI.
RetTy->
getContext(), AttributeList::ReturnIndex,
11222std::pair<SDValue, SDValue>
11236 "Only supported for non-aggregate returns");
11239 for (
Type *Ty : RetOrigTys)
11248 RetOrigTys.
swap(OldRetOrigTys);
11249 RetVTs.
swap(OldRetVTs);
11250 Offsets.swap(OldOffsets);
11252 for (
size_t i = 0, e = OldRetVTs.
size(); i != e; ++i) {
11253 EVT RetVT = OldRetVTs[i];
11257 unsigned RegisterVTByteSZ = RegisterVT.
getSizeInBits() / 8;
11258 RetOrigTys.
append(NumRegs, OldRetOrigTys[i]);
11259 RetVTs.
append(NumRegs, RegisterVT);
11260 for (
unsigned j = 0; j != NumRegs; ++j)
11273 int DemoteStackIdx = -100;
11286 ArgListEntry Entry(DemoteStackSlot, StackSlotPtrType);
11287 Entry.IsSRet =
true;
11288 Entry.Alignment = Alignment;
11300 for (
unsigned I = 0, E = RetVTs.
size();
I != E; ++
I) {
11302 if (NeedsRegBlock) {
11303 Flags.setInConsecutiveRegs();
11304 if (
I == RetVTs.
size() - 1)
11305 Flags.setInConsecutiveRegsLast();
11307 EVT VT = RetVTs[
I];
11311 for (
unsigned i = 0; i != NumRegs; ++i) {
11325 CLI.
Ins.push_back(Ret);
11334 if (Arg.IsSwiftError) {
11340 CLI.
Ins.push_back(Ret);
11348 for (
unsigned i = 0, e = Args.size(); i != e; ++i) {
11352 Type *FinalType = Args[i].Ty;
11353 if (Args[i].IsByVal)
11354 FinalType = Args[i].IndirectType;
11357 for (
unsigned Value = 0, NumValues = OrigArgTys.
size();
Value != NumValues;
11360 Type *ArgTy = OrigArgTy;
11361 if (Args[i].Ty != Args[i].OrigTy) {
11362 assert(
Value == 0 &&
"Only supported for non-aggregate arguments");
11363 ArgTy = Args[i].Ty;
11368 Args[i].Node.getResNo() +
Value);
11375 Flags.setOrigAlign(OriginalAlignment);
11380 Flags.setPointer();
11383 if (Args[i].IsZExt)
11385 if (Args[i].IsSExt)
11387 if (Args[i].IsNoExt)
11389 if (Args[i].IsInReg) {
11396 Flags.setHvaStart();
11402 if (Args[i].IsSRet)
11404 if (Args[i].IsSwiftSelf)
11405 Flags.setSwiftSelf();
11406 if (Args[i].IsSwiftAsync)
11407 Flags.setSwiftAsync();
11408 if (Args[i].IsSwiftError)
11409 Flags.setSwiftError();
11410 if (Args[i].IsCFGuardTarget)
11411 Flags.setCFGuardTarget();
11412 if (Args[i].IsByVal)
11414 if (Args[i].IsByRef)
11416 if (Args[i].IsPreallocated) {
11417 Flags.setPreallocated();
11425 if (Args[i].IsInAlloca) {
11426 Flags.setInAlloca();
11435 if (Args[i].IsByVal || Args[i].IsInAlloca || Args[i].IsPreallocated) {
11436 unsigned FrameSize =
DL.getTypeAllocSize(Args[i].IndirectType);
11437 Flags.setByValSize(FrameSize);
11440 if (
auto MA = Args[i].Alignment)
11444 }
else if (
auto MA = Args[i].Alignment) {
11447 MemAlign = OriginalAlignment;
11449 Flags.setMemAlign(MemAlign);
11450 if (Args[i].IsNest)
11453 Flags.setInConsecutiveRegs();
11456 unsigned NumParts =
11461 if (Args[i].IsSExt)
11463 else if (Args[i].IsZExt)
11468 if (Args[i].IsReturned && !
Op.getValueType().isVector() &&
11473 Args[i].Ty->getPointerAddressSpace())) &&
11474 RetVTs.
size() == NumValues &&
"unexpected use of 'returned'");
11487 CLI.
RetZExt == Args[i].IsZExt))
11488 Flags.setReturned();
11494 for (
unsigned j = 0; j != NumParts; ++j) {
11500 j * Parts[j].
getValueType().getStoreSize().getKnownMinValue());
11501 if (NumParts > 1 && j == 0)
11505 if (j == NumParts - 1)
11509 CLI.
Outs.push_back(MyFlags);
11510 CLI.
OutVals.push_back(Parts[j]);
11513 if (NeedsRegBlock &&
Value == NumValues - 1)
11514 CLI.
Outs[CLI.
Outs.size() - 1].Flags.setInConsecutiveRegsLast();
11526 "LowerCall didn't return a valid chain!");
11528 "LowerCall emitted a return value for a tail call!");
11530 "LowerCall didn't emit the correct number of values!");
11542 for (
unsigned i = 0, e = CLI.
Ins.size(); i != e; ++i) {
11543 assert(InVals[i].
getNode() &&
"LowerCall emitted a null value!");
11544 assert(
EVT(CLI.
Ins[i].VT) == InVals[i].getValueType() &&
11545 "LowerCall emitted a value with the wrong type!");
11555 unsigned NumValues = RetVTs.
size();
11556 ReturnValues.
resize(NumValues);
11563 for (
unsigned i = 0; i < NumValues; ++i) {
11570 DemoteStackIdx, Offsets[i]),
11572 ReturnValues[i] = L;
11573 Chains[i] = L.getValue(1);
11580 std::optional<ISD::NodeType> AssertOp;
11585 unsigned CurReg = 0;
11586 for (
EVT VT : RetVTs) {
11592 CLI.
DAG, CLI.
DL, &InVals[CurReg], NumRegs, RegisterVT, VT,
nullptr,
11600 if (ReturnValues.
empty())
11606 return std::make_pair(Res, CLI.
Chain);
11623 if (
N->getNumValues() == 1) {
11631 "Lowering returned the wrong number of results!");
11634 for (
unsigned I = 0, E =
N->getNumValues();
I != E; ++
I)
11648 "Copy from a reg to the same reg!");
11649 assert(!Reg.isPhysical() &&
"Is a physreg");
11655 RegsForValue RFV(V->getContext(), TLI,
DAG.getDataLayout(), Reg, V->getType(),
11660 auto PreferredExtendIt =
FuncInfo.PreferredExtendType.find(V);
11661 if (PreferredExtendIt !=
FuncInfo.PreferredExtendType.end())
11662 ExtendType = PreferredExtendIt->second;
11665 PendingExports.push_back(Chain);
11677 return A->use_empty();
11679 const BasicBlock &Entry =
A->getParent()->front();
11680 for (
const User *U :
A->users())
11689 std::pair<const AllocaInst *, const StoreInst *>>;
11701 enum StaticAllocaInfo {
Unknown, Clobbered, Elidable };
11703 unsigned NumArgs = FuncInfo->
Fn->
arg_size();
11704 StaticAllocas.
reserve(NumArgs * 2);
11706 auto GetInfoIfStaticAlloca = [&](
const Value *V) -> StaticAllocaInfo * {
11709 V = V->stripPointerCasts();
11711 if (!AI || !AI->isStaticAlloca() || !FuncInfo->
StaticAllocaMap.count(AI))
11714 return &Iter.first->second;
11731 if (
I.isDebugOrPseudoInst())
11735 for (
const Use &U :
I.operands()) {
11736 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(U))
11737 *Info = StaticAllocaInfo::Clobbered;
11743 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(
SI->getValueOperand()))
11744 *Info = StaticAllocaInfo::Clobbered;
11747 const Value *Dst =
SI->getPointerOperand()->stripPointerCasts();
11748 StaticAllocaInfo *Info = GetInfoIfStaticAlloca(Dst);
11754 if (*Info != StaticAllocaInfo::Unknown)
11762 const Value *Val =
SI->getValueOperand()->stripPointerCasts();
11765 if (!Arg || Arg->hasPassPointeeByValueCopyAttr() ||
11767 DL.getTypeStoreSize(Arg->
getType()) != *AllocaSize ||
11768 !
DL.typeSizeEqualsStoreSize(Arg->
getType()) ||
11769 ArgCopyElisionCandidates.count(Arg)) {
11770 *Info = StaticAllocaInfo::Clobbered;
11774 LLVM_DEBUG(
dbgs() <<
"Found argument copy elision candidate: " << *AI
11778 *Info = StaticAllocaInfo::Elidable;
11779 ArgCopyElisionCandidates.insert({Arg, {AI,
SI}});
11784 if (ArgCopyElisionCandidates.size() == NumArgs)
11808 auto ArgCopyIter = ArgCopyElisionCandidates.find(&Arg);
11809 assert(ArgCopyIter != ArgCopyElisionCandidates.end());
11810 const AllocaInst *AI = ArgCopyIter->second.first;
11811 int FixedIndex = FINode->getIndex();
11813 int OldIndex = AllocaIndex;
11817 dbgs() <<
" argument copy elision failed due to bad fixed stack "
11823 LLVM_DEBUG(
dbgs() <<
" argument copy elision failed: alignment of alloca "
11824 "greater than stack argument alignment ("
11825 <<
DebugStr(RequiredAlignment) <<
" vs "
11833 dbgs() <<
"Eliding argument copy from " << Arg <<
" to " << *AI <<
'\n'
11834 <<
" Replacing frame index " << OldIndex <<
" with " << FixedIndex
11840 AllocaIndex = FixedIndex;
11841 ArgCopyElisionFrameIndexMap.
insert({OldIndex, FixedIndex});
11842 for (
SDValue ArgVal : ArgVals)
11846 const StoreInst *
SI = ArgCopyIter->second.second;
11859void SelectionDAGISel::LowerArguments(
const Function &
F) {
11860 SelectionDAG &DAG =
SDB->DAG;
11861 SDLoc dl =
SDB->getCurSDLoc();
11866 if (
F.hasFnAttribute(Attribute::Naked))
11871 MVT ValueVT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
11873 ISD::ArgFlagsTy
Flags;
11875 MVT RegisterVT =
TLI->getRegisterType(*DAG.
getContext(), ValueVT);
11876 ISD::InputArg RetArg(Flags, RegisterVT, ValueVT,
F.getReturnType(),
true,
11886 ArgCopyElisionCandidates);
11889 for (
const Argument &Arg :
F.args()) {
11890 unsigned ArgNo = Arg.getArgNo();
11893 bool isArgValueUsed = !Arg.
use_empty();
11895 if (Arg.hasAttribute(Attribute::ByVal))
11896 FinalType = Arg.getParamByValType();
11897 bool NeedsRegBlock =
TLI->functionArgumentNeedsConsecutiveRegisters(
11898 FinalType,
F.getCallingConv(),
F.isVarArg(),
DL);
11899 for (
unsigned Value = 0, NumValues =
Types.size();
Value != NumValues;
11902 EVT VT =
TLI->getValueType(
DL, ArgTy);
11903 ISD::ArgFlagsTy
Flags;
11906 Flags.setPointer();
11909 if (Arg.hasAttribute(Attribute::ZExt))
11911 if (Arg.hasAttribute(Attribute::SExt))
11913 if (Arg.hasAttribute(Attribute::InReg)) {
11920 Flags.setHvaStart();
11926 if (Arg.hasAttribute(Attribute::StructRet))
11928 if (Arg.hasAttribute(Attribute::SwiftSelf))
11929 Flags.setSwiftSelf();
11930 if (Arg.hasAttribute(Attribute::SwiftAsync))
11931 Flags.setSwiftAsync();
11932 if (Arg.hasAttribute(Attribute::SwiftError))
11933 Flags.setSwiftError();
11934 if (Arg.hasAttribute(Attribute::ByVal))
11936 if (Arg.hasAttribute(Attribute::ByRef))
11938 if (Arg.hasAttribute(Attribute::InAlloca)) {
11939 Flags.setInAlloca();
11947 if (Arg.hasAttribute(Attribute::Preallocated)) {
11948 Flags.setPreallocated();
11960 const Align OriginalAlignment(
11961 TLI->getABIAlignmentForCallingConv(ArgTy,
DL));
11962 Flags.setOrigAlign(OriginalAlignment);
11965 Type *ArgMemTy =
nullptr;
11966 if (
Flags.isByVal() ||
Flags.isInAlloca() ||
Flags.isPreallocated() ||
11969 ArgMemTy = Arg.getPointeeInMemoryValueType();
11971 uint64_t MemSize =
DL.getTypeAllocSize(ArgMemTy);
11976 if (
auto ParamAlign = Arg.getParamStackAlign())
11977 MemAlign = *ParamAlign;
11978 else if ((ParamAlign = Arg.getParamAlign()))
11979 MemAlign = *ParamAlign;
11981 MemAlign =
TLI->getByValTypeAlignment(ArgMemTy,
DL);
11982 if (
Flags.isByRef())
11983 Flags.setByRefSize(MemSize);
11985 Flags.setByValSize(MemSize);
11986 }
else if (
auto ParamAlign = Arg.getParamStackAlign()) {
11987 MemAlign = *ParamAlign;
11989 MemAlign = OriginalAlignment;
11991 Flags.setMemAlign(MemAlign);
11993 if (Arg.hasAttribute(Attribute::Nest))
11996 Flags.setInConsecutiveRegs();
11997 if (ArgCopyElisionCandidates.count(&Arg))
11998 Flags.setCopyElisionCandidate();
11999 if (Arg.hasAttribute(Attribute::Returned))
12000 Flags.setReturned();
12002 MVT RegisterVT =
TLI->getRegisterTypeForCallingConv(
12003 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12004 unsigned NumRegs =
TLI->getNumRegistersForCallingConv(
12005 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12006 for (
unsigned i = 0; i != NumRegs; ++i) {
12010 ISD::InputArg MyFlags(
12011 Flags, RegisterVT, VT, ArgTy, isArgValueUsed, ArgNo,
12013 if (NumRegs > 1 && i == 0)
12014 MyFlags.Flags.setSplit();
12017 MyFlags.Flags.setOrigAlign(
Align(1));
12018 if (i == NumRegs - 1)
12019 MyFlags.Flags.setSplitEnd();
12023 if (NeedsRegBlock &&
Value == NumValues - 1)
12024 Ins[Ins.
size() - 1].Flags.setInConsecutiveRegsLast();
12030 SDValue NewRoot =
TLI->LowerFormalArguments(
12031 DAG.
getRoot(),
F.getCallingConv(),
F.isVarArg(), Ins, dl, DAG, InVals);
12035 "LowerFormalArguments didn't return a valid chain!");
12037 "LowerFormalArguments didn't emit the correct number of values!");
12039 for (
unsigned i = 0, e = Ins.
size(); i != e; ++i) {
12041 "LowerFormalArguments emitted a null value!");
12043 "LowerFormalArguments emitted a value with the wrong type!");
12055 MVT VT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
12056 MVT RegVT =
TLI->getRegisterType(*
CurDAG->getContext(), VT);
12057 std::optional<ISD::NodeType> AssertOp;
12060 F.getCallingConv(), AssertOp);
12062 MachineFunction&
MF =
SDB->DAG.getMachineFunction();
12063 MachineRegisterInfo&
RegInfo =
MF.getRegInfo();
12065 RegInfo.createVirtualRegister(
TLI->getRegClassFor(RegVT));
12066 FuncInfo->DemoteRegister = SRetReg;
12068 SDB->DAG.getCopyToReg(NewRoot,
SDB->getCurSDLoc(), SRetReg, ArgValue);
12076 DenseMap<int, int> ArgCopyElisionFrameIndexMap;
12077 for (
const Argument &Arg :
F.args()) {
12081 unsigned NumValues = ValueVTs.
size();
12082 if (NumValues == 0)
12089 if (Ins[i].
Flags.isCopyElisionCandidate()) {
12090 unsigned NumParts = 0;
12091 for (EVT VT : ValueVTs)
12092 NumParts +=
TLI->getNumRegistersForCallingConv(*
CurDAG->getContext(),
12093 F.getCallingConv(), VT);
12097 ArrayRef(&InVals[i], NumParts), ArgHasUses);
12102 bool isSwiftErrorArg =
12103 TLI->supportSwiftError() &&
12104 Arg.hasAttribute(Attribute::SwiftError);
12105 if (!ArgHasUses && !isSwiftErrorArg) {
12106 SDB->setUnusedArgValue(&Arg, InVals[i]);
12109 if (FrameIndexSDNode *FI =
12111 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12114 for (
unsigned Val = 0; Val != NumValues; ++Val) {
12115 EVT VT = ValueVTs[Val];
12116 MVT PartVT =
TLI->getRegisterTypeForCallingConv(*
CurDAG->getContext(),
12117 F.getCallingConv(), VT);
12118 unsigned NumParts =
TLI->getNumRegistersForCallingConv(
12119 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12124 if (ArgHasUses || isSwiftErrorArg) {
12125 std::optional<ISD::NodeType> AssertOp;
12126 if (Arg.hasAttribute(Attribute::SExt))
12128 else if (Arg.hasAttribute(Attribute::ZExt))
12133 NewRoot,
F.getCallingConv(), AssertOp);
12136 if (NoFPClass !=
fcNone) {
12138 static_cast<uint64_t
>(NoFPClass), dl, MVT::i32);
12140 OutVal, SDNoFPClass);
12149 if (ArgValues.
empty())
12153 if (FrameIndexSDNode *FI =
12155 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12158 SDB->getCurSDLoc());
12160 SDB->setValue(&Arg, Res);
12170 if (LoadSDNode *LNode =
12172 if (FrameIndexSDNode *FI =
12174 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12202 FuncInfo->InitializeRegForValue(&Arg);
12203 SDB->CopyToExportRegsIfNeeded(&Arg);
12207 if (!Chains.
empty()) {
12214 assert(i == InVals.
size() &&
"Argument register count mismatch!");
12218 if (!ArgCopyElisionFrameIndexMap.
empty()) {
12219 for (MachineFunction::VariableDbgInfo &VI :
12220 MF->getInStackSlotVariableDbgInfo()) {
12221 auto I = ArgCopyElisionFrameIndexMap.
find(
VI.getStackSlot());
12222 if (
I != ArgCopyElisionFrameIndexMap.
end())
12223 VI.updateStackSlot(
I->second);
12238SelectionDAGBuilder::HandlePHINodesInSuccessorBlocks(
const BasicBlock *LLVMBB) {
12239 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12241 SmallPtrSet<MachineBasicBlock *, 4> SuccsHandled;
12247 MachineBasicBlock *SuccMBB =
FuncInfo.getMBB(SuccBB);
12251 if (!SuccsHandled.
insert(SuccMBB).second)
12259 for (
const PHINode &PN : SuccBB->phis()) {
12261 if (PN.use_empty())
12265 if (PN.getType()->isEmptyTy())
12269 const Value *PHIOp = PN.getIncomingValueForBlock(LLVMBB);
12274 RegOut =
FuncInfo.CreateRegs(&PN);
12292 "Didn't codegen value into a register!??");
12302 for (EVT VT : ValueVTs) {
12304 for (
unsigned i = 0; i != NumRegisters; ++i)
12306 Reg += NumRegisters;
12326void SelectionDAGBuilder::updateDAGForMaybeTailCall(
SDValue MaybeTC) {
12328 if (MaybeTC.
getNode() !=
nullptr)
12329 DAG.setRoot(MaybeTC);
12334void SelectionDAGBuilder::lowerWorkItem(SwitchWorkListItem W,
Value *
Cond,
12337 MachineFunction *CurMF =
FuncInfo.MF;
12338 MachineBasicBlock *NextMBB =
nullptr;
12343 unsigned Size =
W.LastCluster -
W.FirstCluster + 1;
12345 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12347 if (
Size == 2 &&
W.MBB == SwitchMBB) {
12355 CaseCluster &
Small = *
W.FirstCluster;
12356 CaseCluster &
Big = *
W.LastCluster;
12360 const APInt &SmallValue =
Small.Low->getValue();
12361 const APInt &BigValue =
Big.Low->getValue();
12364 APInt CommonBit = BigValue ^ SmallValue;
12371 DAG.getConstant(CommonBit,
DL, VT));
12373 DL, MVT::i1,
Or,
DAG.getConstant(BigValue | SmallValue,
DL, VT),
12379 addSuccessorWithProb(SwitchMBB,
Small.MBB,
Small.Prob +
Big.Prob);
12381 addSuccessorWithProb(
12382 SwitchMBB, DefaultMBB,
12386 addSuccessorWithProb(SwitchMBB, DefaultMBB);
12394 DAG.getBasicBlock(DefaultMBB));
12396 DAG.setRoot(BrCond);
12408 [](
const CaseCluster &a,
const CaseCluster &b) {
12409 return a.Prob != b.Prob ?
12411 a.Low->getValue().slt(b.Low->getValue());
12418 if (
I->Prob >
W.LastCluster->Prob)
12420 if (
I->Kind ==
CC_Range &&
I->MBB == NextMBB) {
12428 BranchProbability DefaultProb =
W.DefaultProb;
12429 BranchProbability UnhandledProbs = DefaultProb;
12431 UnhandledProbs +=
I->Prob;
12433 MachineBasicBlock *CurMBB =
W.MBB;
12435 bool FallthroughUnreachable =
false;
12436 MachineBasicBlock *Fallthrough;
12437 if (
I ==
W.LastCluster) {
12439 Fallthrough = DefaultMBB;
12444 CurMF->
insert(BBI, Fallthrough);
12448 UnhandledProbs -=
I->Prob;
12453 JumpTableHeader *JTH = &
SL->JTCases[
I->JTCasesIndex].first;
12454 SwitchCG::JumpTable *JT = &
SL->JTCases[
I->JTCasesIndex].second;
12457 MachineBasicBlock *JumpMBB = JT->
MBB;
12458 CurMF->
insert(BBI, JumpMBB);
12460 auto JumpProb =
I->Prob;
12461 auto FallthroughProb = UnhandledProbs;
12469 if (*SI == DefaultMBB) {
12470 JumpProb += DefaultProb / 2;
12471 FallthroughProb -= DefaultProb / 2;
12489 if (FallthroughUnreachable) {
12496 addSuccessorWithProb(CurMBB, Fallthrough, FallthroughProb);
12497 addSuccessorWithProb(CurMBB, JumpMBB, JumpProb);
12506 if (CurMBB == SwitchMBB) {
12514 BitTestBlock *BTB = &
SL->BitTestCases[
I->BTCasesIndex];
12517 for (BitTestCase &BTC : BTB->
Cases)
12529 BTB->
Prob += DefaultProb / 2;
12533 if (FallthroughUnreachable)
12537 if (CurMBB == SwitchMBB) {
12544 const Value *
RHS, *
LHS, *MHS;
12546 if (
I->Low ==
I->High) {
12561 if (FallthroughUnreachable)
12565 CaseBlock CB(CC,
LHS,
RHS, MHS,
I->MBB, Fallthrough, CurMBB,
12568 if (CurMBB == SwitchMBB)
12571 SL->SwitchCases.push_back(CB);
12576 CurMBB = Fallthrough;
12580void SelectionDAGBuilder::splitWorkItem(
SwitchWorkList &WorkList,
12581 const SwitchWorkListItem &W,
12584 assert(
W.FirstCluster->Low->getValue().slt(
W.LastCluster->Low->getValue()) &&
12585 "Clusters not sorted?");
12586 assert(
W.LastCluster -
W.FirstCluster + 1 >= 2 &&
"Too small to split!");
12588 auto [LastLeft, FirstRight, LeftProb, RightProb] =
12589 SL->computeSplitWorkItemInfo(W);
12594 assert(PivotCluster >
W.FirstCluster);
12595 assert(PivotCluster <=
W.LastCluster);
12600 const ConstantInt *Pivot = PivotCluster->Low;
12609 MachineBasicBlock *LeftMBB;
12610 if (FirstLeft == LastLeft && FirstLeft->Kind ==
CC_Range &&
12611 FirstLeft->Low ==
W.GE &&
12612 (FirstLeft->High->getValue() + 1LL) == Pivot->
getValue()) {
12613 LeftMBB = FirstLeft->MBB;
12615 LeftMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12616 FuncInfo.MF->insert(BBI, LeftMBB);
12618 {LeftMBB, FirstLeft, LastLeft,
W.GE, Pivot,
W.DefaultProb / 2});
12626 MachineBasicBlock *RightMBB;
12627 if (FirstRight == LastRight && FirstRight->Kind ==
CC_Range &&
12628 W.LT && (FirstRight->High->getValue() + 1ULL) ==
W.LT->getValue()) {
12629 RightMBB = FirstRight->MBB;
12631 RightMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12632 FuncInfo.MF->insert(BBI, RightMBB);
12634 {RightMBB, FirstRight, LastRight, Pivot,
W.LT,
W.DefaultProb / 2});
12640 CaseBlock CB(
ISD::SETLT,
Cond, Pivot,
nullptr, LeftMBB, RightMBB,
W.MBB,
12643 if (
W.MBB == SwitchMBB)
12646 SL->SwitchCases.push_back(CB);
12671 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12679 unsigned PeeledCaseIndex = 0;
12680 bool SwitchPeeled =
false;
12681 for (
unsigned Index = 0;
Index < Clusters.size(); ++
Index) {
12682 CaseCluster &CC = Clusters[
Index];
12683 if (CC.
Prob < TopCaseProb)
12685 TopCaseProb = CC.
Prob;
12686 PeeledCaseIndex =
Index;
12687 SwitchPeeled =
true;
12692 LLVM_DEBUG(
dbgs() <<
"Peeled one top case in switch stmt, prob: "
12693 << TopCaseProb <<
"\n");
12698 MachineBasicBlock *PeeledSwitchMBB =
12700 FuncInfo.MF->insert(BBI, PeeledSwitchMBB);
12703 auto PeeledCaseIt = Clusters.begin() + PeeledCaseIndex;
12704 SwitchWorkListItem
W = {SwitchMBB, PeeledCaseIt, PeeledCaseIt,
12705 nullptr,
nullptr, TopCaseProb.
getCompl()};
12706 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, PeeledSwitchMBB);
12708 Clusters.erase(PeeledCaseIt);
12709 for (CaseCluster &CC : Clusters) {
12711 dbgs() <<
"Scale the probablity for one cluster, before scaling: "
12712 << CC.
Prob <<
"\n");
12716 PeeledCaseProb = TopCaseProb;
12717 return PeeledSwitchMBB;
12720void SelectionDAGBuilder::visitSwitch(
const SwitchInst &
SI) {
12722 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12724 Clusters.reserve(
SI.getNumCases());
12725 for (
auto I :
SI.cases()) {
12726 MachineBasicBlock *Succ =
FuncInfo.getMBB(
I.getCaseSuccessor());
12727 const ConstantInt *CaseVal =
I.getCaseValue();
12728 BranchProbability Prob =
12730 : BranchProbability(1,
SI.getNumCases() + 1);
12734 MachineBasicBlock *DefaultMBB =
FuncInfo.getMBB(
SI.getDefaultDest());
12743 MachineBasicBlock *PeeledSwitchMBB =
12744 peelDominantCaseCluster(SI, Clusters, PeeledCaseProb);
12747 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12748 if (Clusters.empty()) {
12749 assert(PeeledSwitchMBB == SwitchMBB);
12751 if (DefaultMBB != NextBlock(SwitchMBB)) {
12758 SL->findJumpTables(Clusters, &SI,
getCurSDLoc(), DefaultMBB,
DAG.getPSI(),
12760 SL->findBitTestClusters(Clusters, &SI);
12763 dbgs() <<
"Case clusters: ";
12764 for (
const CaseCluster &
C : Clusters) {
12770 C.Low->getValue().print(
dbgs(),
true);
12771 if (
C.Low !=
C.High) {
12773 C.High->getValue().print(
dbgs(),
true);
12780 assert(!Clusters.empty());
12784 auto DefaultProb = getEdgeProbability(PeeledSwitchMBB, DefaultMBB);
12788 DefaultMBB ==
FuncInfo.getMBB(
SI.getDefaultDest()))
12791 {PeeledSwitchMBB,
First,
Last,
nullptr,
nullptr, DefaultProb});
12793 while (!WorkList.
empty()) {
12795 unsigned NumClusters =
W.LastCluster -
W.FirstCluster + 1;
12800 splitWorkItem(WorkList, W,
SI.getCondition(), SwitchMBB);
12804 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, DefaultMBB);
12808void SelectionDAGBuilder::visitStepVector(
const CallInst &
I) {
12809 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12815void SelectionDAGBuilder::visitVectorReverse(
const CallInst &
I) {
12816 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12821 assert(VT ==
V.getValueType() &&
"Malformed vector.reverse!");
12830 SmallVector<int, 8>
Mask;
12832 for (
unsigned i = 0; i != NumElts; ++i)
12833 Mask.push_back(NumElts - 1 - i);
12838void SelectionDAGBuilder::visitVectorDeinterleave(
const CallInst &
I,
12847 EVT OutVT = ValueVTs[0];
12851 for (
unsigned i = 0; i != Factor; ++i) {
12852 assert(ValueVTs[i] == OutVT &&
"Expected VTs to be the same");
12854 DAG.getVectorIdxConstant(OutNumElts * i,
DL));
12860 SDValue Even =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
12862 SDValue Odd =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
12870 DAG.getVTList(ValueVTs), SubVecs);
12874void SelectionDAGBuilder::visitVectorInterleave(
const CallInst &
I,
12877 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12882 for (
unsigned i = 0; i < Factor; ++i) {
12885 "Expected VTs to be the same");
12903 for (
unsigned i = 0; i < Factor; ++i)
12910void SelectionDAGBuilder::visitFreeze(
const FreezeInst &
I) {
12914 unsigned NumValues = ValueVTs.
size();
12915 if (NumValues == 0)
return;
12920 for (
unsigned i = 0; i != NumValues; ++i)
12925 DAG.getVTList(ValueVTs), Values));
12928void SelectionDAGBuilder::visitVectorSplice(
const CallInst &
I) {
12929 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12935 const bool IsLeft =
I.getIntrinsicID() == Intrinsic::vector_splice_left;
12950 uint64_t Idx = IsLeft ?
Imm : NumElts -
Imm;
12953 SmallVector<int, 8>
Mask;
12954 for (
unsigned i = 0; i < NumElts; ++i)
12955 Mask.push_back(Idx + i);
12983 assert(
MI->getOpcode() == TargetOpcode::COPY &&
12984 "start of copy chain MUST be COPY");
12985 Reg =
MI->getOperand(1).getReg();
12988 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
12992 if (
MI->getOpcode() == TargetOpcode::COPY) {
12993 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
12994 Reg =
MI->getOperand(1).getReg();
12995 assert(
Reg.isPhysical() &&
"expected COPY of physical register");
12998 assert(
MI->getOpcode() == TargetOpcode::INLINEASM_BR &&
12999 "end of copy chain MUST be INLINEASM_BR");
13009void SelectionDAGBuilder::visitCallBrLandingPad(
const CallInst &
I) {
13015 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13016 const TargetRegisterInfo *
TRI =
DAG.getSubtarget().getRegisterInfo();
13017 MachineRegisterInfo &MRI =
DAG.getMachineFunction().getRegInfo();
13025 for (
auto &
T : TargetConstraints) {
13026 SDISelAsmOperandInfo OpInfo(
T);
13034 switch (OpInfo.ConstraintType) {
13045 FuncInfo.MBB->addLiveIn(OriginalDef);
13053 ResultVTs.
push_back(OpInfo.ConstraintVT);
13062 ResultVTs.
push_back(OpInfo.ConstraintVT);
13070 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.
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)
Returns the 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 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 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 TableGen::Emitter::OptClass< SkeletonEmitter > X("gen-skeleton-class", "Generate example skeleton class")
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.
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
size - Get the array size.
bool empty() const
empty - 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
@ FMax
*p = maxnum(old, v) maxnum matches the behavior of llvm.maxnum.
@ UDecWrap
Decrement one until a minimum value or zero.
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 if the block is well formed or null if the block is not well forme...
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.
Conditional or Unconditional Branch instruction.
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.
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 > iterator
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.
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).
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 ...
@ 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.
bool contains(const KeyT &Key) const
std::pair< iterator, bool > try_emplace(const KeyT &Key, Ts &&...Args)
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 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.
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.
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)
bool shouldKeepJumpConditionsTogether(const FunctionLoweringInfo &FuncInfo, const BranchInst &I, Instruction::BinaryOps Opc, const Value *Lhs, const Value *Rhs, TargetLoweringBase::CondMergingParams Params) const
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 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 getBasicBlock(MachineBasicBlock *MBB)
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.
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.
StringRef - Represent a constant reference to a string, i.e.
constexpr bool empty() const
empty - Check if the string is empty.
constexpr const char * data() const
data - 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.
unsigned getBitWidthForCttzElements(Type *RetTy, ElementCount EC, bool ZeroIsPoison, const ConstantRange *VScaleRange) const
Return the minimum number of bits required to hold the maximum possible number of trailing zero vecto...
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.
MachineMemOperand::Flags getLoadMemOperandFlags(const LoadInst &LI, const DataLayout &DL, AssumptionCache *AC=nullptr, const TargetLibraryInfo *LibInfo=nullptr) const
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 shouldExpandCttzElements(EVT VT) const
Return true if the @llvm.experimental.cttz.elts intrinsic should be expanded using generic code in Se...
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 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...
CodeGenOptLevel getOptLevel() const
Returns the optimization level: None, Less, Default, or Aggressive.
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'.
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.
@ 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.
@ 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.
@ CTTZ_ZERO_UNDEF
Bit counting operators with an undefined result for zero inputs.
@ 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.
@ 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.
@ 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.
@ 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,...
@ 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.
@ 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...
@ 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_LogicalOr()
Matches L || R where L and R are arbitrary values.
class_match< Value > m_Value()
Match an arbitrary value and ignore it.
auto m_LogicalAnd()
Matches L && R where L and R are arbitrary values.
std::pair< JumpTableHeader, JumpTable > JumpTableBlock
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
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.
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)
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)
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.
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)
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 ConstantRange getVScaleRange(const Function *F, unsigned BitWidth)
Determine the possible constant range of vscale with the given bit width, based on the vscale_range f...
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.
@ And
Bitwise or logical AND of integers.
@ Sub
Subtraction of integers.
uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
@ 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.
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
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...
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...
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 values are 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.
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.
void addIPToStateRange(const InvokeInst *II, MCSymbol *InvokeBegin, MCSymbol *InvokeEnd)