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 std::optional<CallingConv::ID> CallConv;
2022 if (CB && !CB->isInlineAsm())
2023 CallConv = CB->getCallingConv();
2026 Inst->getType(), CallConv);
2040void SelectionDAGBuilder::visitCatchPad(
const CatchPadInst &
I) {
2053 if (IsMSVCCXX || IsCoreCLR)
2059 MachineBasicBlock *TargetMBB =
FuncInfo.getMBB(
I.getSuccessor());
2060 FuncInfo.MBB->addSuccessor(TargetMBB);
2067 if (TargetMBB != NextBlock(
FuncInfo.MBB) ||
2076 DAG.getMachineFunction().setHasEHContTarget(
true);
2082 Value *ParentPad =
I.getCatchSwitchParentPad();
2085 SuccessorColor = &
FuncInfo.Fn->getEntryBlock();
2088 assert(SuccessorColor &&
"No parent funclet for catchret!");
2089 MachineBasicBlock *SuccessorColorMBB =
FuncInfo.getMBB(SuccessorColor);
2090 assert(SuccessorColorMBB &&
"No MBB for SuccessorColor!");
2095 DAG.getBasicBlock(SuccessorColorMBB));
2099void SelectionDAGBuilder::visitCleanupPad(
const CleanupPadInst &CPI) {
2105 FuncInfo.MBB->setIsEHFuncletEntry();
2106 FuncInfo.MBB->setIsCleanupFuncletEntry();
2135 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2141 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2142 UnwindDests.back().first->setIsEHScopeEntry();
2145 UnwindDests.back().first->setIsEHFuncletEntry();
2149 for (
const BasicBlock *CatchPadBB : CatchSwitch->handlers()) {
2150 UnwindDests.emplace_back(FuncInfo.
getMBB(CatchPadBB), Prob);
2152 if (IsMSVCCXX || IsCoreCLR)
2153 UnwindDests.back().first->setIsEHFuncletEntry();
2155 UnwindDests.back().first->setIsEHScopeEntry();
2157 NewEHPadBB = CatchSwitch->getUnwindDest();
2163 if (BPI && NewEHPadBB)
2165 EHPadBB = NewEHPadBB;
2172 auto UnwindDest =
I.getUnwindDest();
2173 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
2174 BranchProbability UnwindDestProb =
2179 for (
auto &UnwindDest : UnwindDests) {
2180 UnwindDest.first->setIsEHPad();
2181 addSuccessorWithProb(
FuncInfo.MBB, UnwindDest.first, UnwindDest.second);
2183 FuncInfo.MBB->normalizeSuccProbs();
2186 MachineBasicBlock *CleanupPadMBB =
2187 FuncInfo.getMBB(
I.getCleanupPad()->getParent());
2193void SelectionDAGBuilder::visitCatchSwitch(
const CatchSwitchInst &CSI) {
2197void SelectionDAGBuilder::visitRet(
const ReturnInst &
I) {
2198 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
2199 auto &
DL =
DAG.getDataLayout();
2211 if (
I.getParent()->getTerminatingDeoptimizeCall()) {
2228 SmallVector<uint64_t, 4>
Offsets;
2231 unsigned NumValues = ValueVTs.
size();
2234 Align BaseAlign =
DL.getPrefTypeAlign(
I.getOperand(0)->getType());
2235 for (
unsigned i = 0; i != NumValues; ++i) {
2242 if (MemVTs[i] != ValueVTs[i])
2244 Chains[i] =
DAG.getStore(
2252 MVT::Other, Chains);
2253 }
else if (
I.getNumOperands() != 0) {
2256 unsigned NumValues =
Types.size();
2260 const Function *
F =
I.getParent()->getParent();
2263 I.getOperand(0)->getType(),
F->getCallingConv(),
2267 if (
F->getAttributes().hasRetAttr(Attribute::SExt))
2269 else if (
F->getAttributes().hasRetAttr(Attribute::ZExt))
2272 LLVMContext &
Context =
F->getContext();
2273 bool RetInReg =
F->getAttributes().hasRetAttr(Attribute::InReg);
2275 for (
unsigned j = 0;
j != NumValues; ++
j) {
2288 &Parts[0], NumParts, PartVT, &
I, CC, ExtendKind);
2291 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2295 if (
I.getOperand(0)->getType()->isPointerTy()) {
2297 Flags.setPointerAddrSpace(
2301 if (NeedsRegBlock) {
2302 Flags.setInConsecutiveRegs();
2303 if (j == NumValues - 1)
2304 Flags.setInConsecutiveRegsLast();
2312 else if (
F->getAttributes().hasRetAttr(Attribute::NoExt))
2315 for (
unsigned i = 0; i < NumParts; ++i) {
2318 VT, Types[j], 0, 0));
2328 const Function *
F =
I.getParent()->getParent();
2330 F->getAttributes().hasAttrSomewhere(Attribute::SwiftError)) {
2332 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2333 Flags.setSwiftError();
2345 bool isVarArg =
DAG.getMachineFunction().getFunction().isVarArg();
2347 DAG.getMachineFunction().getFunction().getCallingConv();
2348 Chain =
DAG.getTargetLoweringInfo().LowerReturn(
2353 "LowerReturn didn't return a valid chain!");
2364 if (V->getType()->isEmptyTy())
2368 if (VMI !=
FuncInfo.ValueMap.end()) {
2370 "Unused value assigned virtual registers!");
2383 if (
FuncInfo.isExportedInst(V))
return;
2395 if (VI->getParent() == FromBB)
2421 const BasicBlock *SrcBB = Src->getBasicBlock();
2422 const BasicBlock *DstBB = Dst->getBasicBlock();
2426 auto SuccSize = std::max<uint32_t>(
succ_size(SrcBB), 1);
2436 Src->addSuccessorWithoutProb(Dst);
2439 Prob = getEdgeProbability(Src, Dst);
2440 Src->addSuccessor(Dst, Prob);
2446 return I->getParent() == BB;
2470 if (CurBB == SwitchBB ||
2476 InvertCond ? IC->getInversePredicate() : IC->getPredicate();
2481 InvertCond ? FC->getInversePredicate() : FC->getPredicate();
2483 if (FC->hasNoNaNs() ||
2491 CaseBlock CB(Condition, BOp->getOperand(0), BOp->getOperand(1),
nullptr,
2493 SL->SwitchCases.push_back(CB);
2502 SL->SwitchCases.push_back(CB);
2510 unsigned Depth = 0) {
2519 if (Necessary !=
nullptr) {
2522 if (Necessary->contains(
I))
2541 if (
I.getNumSuccessors() != 2)
2544 if (!
I.isConditional())
2556 if (BPI !=
nullptr) {
2562 std::optional<bool> Likely;
2565 else if (BPI->
isEdgeHot(
I.getParent(), IfFalse))
2569 if (
Opc == (*Likely ? Instruction::And : Instruction::Or))
2581 if (CostThresh <= 0)
2602 Value *BrCond =
I.getCondition();
2603 auto ShouldCountInsn = [&RhsDeps, &BrCond](
const Instruction *Ins) {
2604 for (
const auto *U : Ins->users()) {
2607 if (UIns != BrCond && !RhsDeps.
contains(UIns))
2620 for (
unsigned PruneIters = 0; PruneIters < MaxPruneIters; ++PruneIters) {
2622 for (
const auto &InsPair : RhsDeps) {
2623 if (!ShouldCountInsn(InsPair.first)) {
2624 ToDrop = InsPair.first;
2628 if (ToDrop ==
nullptr)
2630 RhsDeps.erase(ToDrop);
2633 for (
const auto &InsPair : RhsDeps) {
2638 CostOfIncluding +=
TTI->getInstructionCost(
2641 if (CostOfIncluding > CostThresh)
2667 const Value *BOpOp0, *BOpOp1;
2681 if (BOpc == Instruction::And)
2682 BOpc = Instruction::Or;
2683 else if (BOpc == Instruction::Or)
2684 BOpc = Instruction::And;
2690 bool BOpIsInOrAndTree = BOpc && BOpc ==
Opc && BOp->
hasOneUse();
2695 TProb, FProb, InvertCond);
2705 if (
Opc == Instruction::Or) {
2726 auto NewTrueProb = TProb / 2;
2727 auto NewFalseProb = TProb / 2 + FProb;
2730 NewFalseProb, InvertCond);
2737 Probs[1], InvertCond);
2739 assert(
Opc == Instruction::And &&
"Unknown merge op!");
2759 auto NewTrueProb = TProb + FProb / 2;
2760 auto NewFalseProb = FProb / 2;
2763 NewFalseProb, InvertCond);
2770 Probs[1], InvertCond);
2779 if (Cases.size() != 2)
return true;
2783 if ((Cases[0].CmpLHS == Cases[1].CmpLHS &&
2784 Cases[0].CmpRHS == Cases[1].CmpRHS) ||
2785 (Cases[0].CmpRHS == Cases[1].CmpLHS &&
2786 Cases[0].CmpLHS == Cases[1].CmpRHS)) {
2792 if (Cases[0].CmpRHS == Cases[1].CmpRHS &&
2793 Cases[0].CC == Cases[1].CC &&
2796 if (Cases[0].CC ==
ISD::SETEQ && Cases[0].TrueBB == Cases[1].ThisBB)
2798 if (Cases[0].CC ==
ISD::SETNE && Cases[0].FalseBB == Cases[1].ThisBB)
2805void SelectionDAGBuilder::visitBr(
const BranchInst &
I) {
2811 if (
I.isUnconditional()) {
2817 if (Succ0MBB != NextBlock(BrMBB) ||
2830 const Value *CondVal =
I.getCondition();
2831 MachineBasicBlock *Succ1MBB =
FuncInfo.getMBB(
I.getSuccessor(1));
2850 bool IsUnpredictable =
I.hasMetadata(LLVMContext::MD_unpredictable);
2852 if (!
DAG.getTargetLoweringInfo().isJumpExpensive() && BOp &&
2855 const Value *BOp0, *BOp1;
2858 Opcode = Instruction::And;
2860 Opcode = Instruction::Or;
2867 DAG.getTargetLoweringInfo().getJumpConditionMergingParams(
2868 Opcode, BOp0, BOp1))) {
2870 getEdgeProbability(BrMBB, Succ0MBB),
2871 getEdgeProbability(BrMBB, Succ1MBB),
2876 assert(
SL->SwitchCases[0].ThisBB == BrMBB &&
"Unexpected lowering!");
2880 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i) {
2887 SL->SwitchCases.erase(
SL->SwitchCases.begin());
2893 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i)
2894 FuncInfo.MF->erase(
SL->SwitchCases[i].ThisBB);
2896 SL->SwitchCases.clear();
2902 nullptr, Succ0MBB, Succ1MBB, BrMBB,
getCurSDLoc(),
2923 if (CB.
TrueBB != NextBlock(SwitchBB)) {
2930 auto &TLI =
DAG.getTargetLoweringInfo();
2954 Cond =
DAG.getSetCC(dl, MVT::i1, CondLHS, CondRHS, CB.
CC);
2966 Cond =
DAG.getSetCC(dl, MVT::i1, CmpOp,
DAG.getConstant(
High, dl, VT),
2970 VT, CmpOp,
DAG.getConstant(
Low, dl, VT));
2971 Cond =
DAG.getSetCC(dl, MVT::i1, SUB,
2986 if (CB.
TrueBB == NextBlock(SwitchBB)) {
3002 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
3005 DAG.setRoot(BrCond);
3011 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3012 assert(JT.
Reg &&
"Should lower JT Header first!");
3013 EVT PTy =
DAG.getTargetLoweringInfo().getJumpTableRegTy(
DAG.getDataLayout());
3017 Index.getValue(1), Table, Index);
3018 DAG.setRoot(BrJumpTable);
3026 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3033 DAG.getConstant(JTH.
First, dl, VT));
3048 JT.
Reg = JumpTableReg;
3056 Sub.getValueType()),
3060 MVT::Other, CopyTo, CMP,
3064 if (JT.
MBB != NextBlock(SwitchBB))
3065 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
3066 DAG.getBasicBlock(JT.
MBB));
3068 DAG.setRoot(BrCond);
3071 if (JT.
MBB != NextBlock(SwitchBB))
3073 DAG.getBasicBlock(JT.
MBB)));
3075 DAG.setRoot(CopyTo);
3099 if (PtrTy != PtrMemTy)
3115 auto &
DL =
DAG.getDataLayout();
3124 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3131 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3145 assert(GuardCheckFn &&
"Guard check function is null");
3156 Entry.IsInReg =
true;
3157 Args.push_back(Entry);
3163 getValue(GuardCheckFn), std::move(Args));
3165 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
3166 DAG.setRoot(Result.second);
3178 Guard =
DAG.getLoad(PtrMemTy, dl, Chain, GuardPtr,
3184 Guard =
DAG.getPOISON(PtrMemTy);
3226 auto &
DL =
DAG.getDataLayout();
3234 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3240 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3255 if (GuardCheckFn->hasParamAttribute(0, Attribute::AttrKind::InReg))
3256 Entry.IsInReg =
true;
3257 Args.push_back(Entry);
3263 getValue(GuardCheckFn), std::move(Args));
3269 Chain = TLI.
makeLibCall(
DAG, RTLIB::STACKPROTECTOR_CHECK_FAIL, MVT::isVoid,
3292 DAG.getNode(
ISD::SUB, dl, VT, SwitchOp,
DAG.getConstant(
B.First, dl, VT));
3296 bool UsePtrType =
false;
3320 if (!
B.FallthroughUnreachable)
3321 addSuccessorWithProb(SwitchBB,
B.Default,
B.DefaultProb);
3322 addSuccessorWithProb(SwitchBB,
MBB,
B.Prob);
3326 if (!
B.FallthroughUnreachable) {
3335 DAG.getBasicBlock(
B.Default));
3339 if (
MBB != NextBlock(SwitchBB))
3357 if (PopCount == 1) {
3364 }
else if (PopCount == BB.
Range) {
3372 DAG.getConstant(1, dl, VT), ShiftOp);
3376 VT, SwitchVal,
DAG.getConstant(
B.Mask, dl, VT));
3383 addSuccessorWithProb(SwitchBB,
B.TargetBB,
B.ExtraProb);
3385 addSuccessorWithProb(SwitchBB, NextMBB, BranchProbToNext);
3393 Cmp,
DAG.getBasicBlock(
B.TargetBB));
3396 if (NextMBB != NextBlock(SwitchBB))
3397 BrAnd =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrAnd,
3398 DAG.getBasicBlock(NextMBB));
3403void SelectionDAGBuilder::visitInvoke(
const InvokeInst &
I) {
3421 const Value *Callee(
I.getCalledOperand());
3424 visitInlineAsm(
I, EHPadBB);
3429 case Intrinsic::donothing:
3431 case Intrinsic::seh_try_begin:
3432 case Intrinsic::seh_scope_begin:
3433 case Intrinsic::seh_try_end:
3434 case Intrinsic::seh_scope_end:
3440 case Intrinsic::experimental_patchpoint_void:
3441 case Intrinsic::experimental_patchpoint:
3442 visitPatchpoint(
I, EHPadBB);
3444 case Intrinsic::experimental_gc_statepoint:
3450 case Intrinsic::wasm_throw: {
3452 std::array<SDValue, 4>
Ops = {
3463 case Intrinsic::wasm_rethrow: {
3464 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3465 std::array<SDValue, 2>
Ops = {
3474 }
else if (
I.hasDeoptState()) {
3495 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
3496 BranchProbability EHPadBBProb =
3502 addSuccessorWithProb(InvokeMBB, Return);
3503 for (
auto &UnwindDest : UnwindDests) {
3504 UnwindDest.first->setIsEHPad();
3505 addSuccessorWithProb(InvokeMBB, UnwindDest.first, UnwindDest.second);
3511 DAG.getBasicBlock(Return)));
3520void SelectionDAGBuilder::visitCallBrIntrinsic(
const CallBrInst &
I) {
3523 DAG.getTargetLoweringInfo().getTgtMemIntrinsic(
3524 Infos,
I,
DAG.getMachineFunction(),
I.getIntrinsicID());
3525 assert(Infos.
empty() &&
"Intrinsic touches memory");
3528 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
3531 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad);
3532 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
3536 getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
3537 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
3542void SelectionDAGBuilder::visitCallBr(
const CallBrInst &
I) {
3543 MachineBasicBlock *CallBrMBB =
FuncInfo.MBB;
3545 if (
I.isInlineAsm()) {
3552 assert(!
I.hasOperandBundles() &&
3553 "Can't have operand bundles for intrinsics");
3554 visitCallBrIntrinsic(
I);
3559 SmallPtrSet<BasicBlock *, 8> Dests;
3560 Dests.
insert(
I.getDefaultDest());
3570 if (
I.isInlineAsm()) {
3571 for (BasicBlock *Dest :
I.getIndirectDests()) {
3573 Target->setIsInlineAsmBrIndirectTarget();
3579 Target->setLabelMustBeEmitted();
3581 if (Dests.
insert(Dest).second)
3590 DAG.getBasicBlock(Return)));
3593void SelectionDAGBuilder::visitResume(
const ResumeInst &RI) {
3594 llvm_unreachable(
"SelectionDAGBuilder shouldn't visit resume instructions!");
3597void SelectionDAGBuilder::visitLandingPad(
const LandingPadInst &LP) {
3599 "Call to landingpad not in landing pad!");
3603 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3619 assert(ValueVTs.
size() == 2 &&
"Only two-valued landingpads are supported");
3624 if (
FuncInfo.ExceptionPointerVirtReg) {
3625 Ops[0] =
DAG.getZExtOrTrunc(
3626 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3633 Ops[1] =
DAG.getZExtOrTrunc(
3634 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3641 DAG.getVTList(ValueVTs),
Ops);
3649 if (JTB.first.HeaderBB ==
First)
3650 JTB.first.HeaderBB =
Last;
3663 for (
unsigned i = 0, e =
I.getNumSuccessors(); i != e; ++i) {
3665 bool Inserted =
Done.insert(BB).second;
3670 addSuccessorWithProb(IndirectBrMBB, Succ);
3680 if (!
I.shouldLowerToTrap(
DAG.getTarget().Options.TrapUnreachable,
3681 DAG.getTarget().Options.NoTrapAfterNoreturn))
3687void SelectionDAGBuilder::visitUnary(
const User &
I,
unsigned Opcode) {
3690 Flags.copyFMF(*FPOp);
3698void SelectionDAGBuilder::visitBinary(
const User &
I,
unsigned Opcode) {
3701 Flags.setNoSignedWrap(OFBinOp->hasNoSignedWrap());
3702 Flags.setNoUnsignedWrap(OFBinOp->hasNoUnsignedWrap());
3705 Flags.setExact(ExactOp->isExact());
3707 Flags.setDisjoint(DisjointOp->isDisjoint());
3709 Flags.copyFMF(*FPOp);
3718void SelectionDAGBuilder::visitShift(
const User &
I,
unsigned Opcode) {
3722 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
3727 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
3729 "Unexpected shift type");
3739 if (
const OverflowingBinaryOperator *OFBinOp =
3741 nuw = OFBinOp->hasNoUnsignedWrap();
3742 nsw = OFBinOp->hasNoSignedWrap();
3744 if (
const PossiblyExactOperator *ExactOp =
3746 exact = ExactOp->isExact();
3749 Flags.setExact(exact);
3750 Flags.setNoSignedWrap(nsw);
3751 Flags.setNoUnsignedWrap(nuw);
3757void SelectionDAGBuilder::visitSDiv(
const User &
I) {
3768void SelectionDAGBuilder::visitICmp(
const ICmpInst &
I) {
3774 auto &TLI =
DAG.getTargetLoweringInfo();
3787 Flags.setSameSign(
I.hasSameSign());
3788 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
3790 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3795void SelectionDAGBuilder::visitFCmp(
const FCmpInst &
I) {
3802 if (FPMO->hasNoNaNs() ||
3803 (
DAG.isKnownNeverNaN(Op1) &&
DAG.isKnownNeverNaN(Op2)))
3807 Flags.copyFMF(*FPMO);
3808 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
3810 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3820 return isa<SelectInst>(V);
3824void SelectionDAGBuilder::visitSelect(
const User &
I) {
3828 unsigned NumValues = ValueVTs.
size();
3829 if (NumValues == 0)
return;
3839 bool IsUnaryAbs =
false;
3840 bool Negate =
false;
3844 Flags.copyFMF(*FPOp);
3846 Flags.setUnpredictable(
3851 EVT VT = ValueVTs[0];
3852 LLVMContext &Ctx = *
DAG.getContext();
3853 auto &TLI =
DAG.getTargetLoweringInfo();
3863 bool UseScalarMinMax = VT.
isVector() &&
3872 switch (SPR.Flavor) {
3878 switch (SPR.NaNBehavior) {
3891 switch (SPR.NaNBehavior) {
3935 for (
unsigned i = 0; i != NumValues; ++i) {
3941 Values[i] =
DAG.getNegative(Values[i], dl, VT);
3944 for (
unsigned i = 0; i != NumValues; ++i) {
3948 Values[i] =
DAG.getNode(
3955 DAG.getVTList(ValueVTs), Values));
3958void SelectionDAGBuilder::visitTrunc(
const User &
I) {
3961 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3965 Flags.setNoSignedWrap(Trunc->hasNoSignedWrap());
3966 Flags.setNoUnsignedWrap(Trunc->hasNoUnsignedWrap());
3972void SelectionDAGBuilder::visitZExt(
const User &
I) {
3976 auto &TLI =
DAG.getTargetLoweringInfo();
3981 Flags.setNonNeg(PNI->hasNonNeg());
3986 if (
Flags.hasNonNeg() &&
3995void SelectionDAGBuilder::visitSExt(
const User &
I) {
3999 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4004void SelectionDAGBuilder::visitFPTrunc(
const User &
I) {
4010 Flags.copyFMF(*FPOp);
4011 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4014 DAG.getTargetConstant(
4019void SelectionDAGBuilder::visitFPExt(
const User &
I) {
4022 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4026 Flags.copyFMF(*FPOp);
4030void SelectionDAGBuilder::visitFPToUI(
const User &
I) {
4033 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4038void SelectionDAGBuilder::visitFPToSI(
const User &
I) {
4041 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4046void SelectionDAGBuilder::visitUIToFP(
const User &
I) {
4049 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4053 Flags.setNonNeg(PNI->hasNonNeg());
4058void SelectionDAGBuilder::visitSIToFP(
const User &
I) {
4061 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4066void SelectionDAGBuilder::visitPtrToAddr(
const User &
I) {
4069 const auto &TLI =
DAG.getTargetLoweringInfo();
4077void SelectionDAGBuilder::visitPtrToInt(
const User &
I) {
4081 auto &TLI =
DAG.getTargetLoweringInfo();
4082 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4091void SelectionDAGBuilder::visitIntToPtr(
const User &
I) {
4095 auto &TLI =
DAG.getTargetLoweringInfo();
4103void SelectionDAGBuilder::visitBitCast(
const User &
I) {
4106 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4111 if (DestVT !=
N.getValueType())
4119 setValue(&
I,
DAG.getConstant(
C->getValue(), dl, DestVT,
false,
4125void SelectionDAGBuilder::visitAddrSpaceCast(
const User &
I) {
4126 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4127 const Value *SV =
I.getOperand(0);
4132 unsigned DestAS =
I.getType()->getPointerAddressSpace();
4134 if (!TM.isNoopAddrSpaceCast(SrcAS, DestAS))
4140void SelectionDAGBuilder::visitInsertElement(
const User &
I) {
4141 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4148 InVec, InVal, InIdx));
4151void SelectionDAGBuilder::visitExtractElement(
const User &
I) {
4152 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4161void SelectionDAGBuilder::visitShuffleVector(
const User &
I) {
4166 Mask = SVI->getShuffleMask();
4170 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4178 DAG.getVectorIdxConstant(0,
DL));
4189 unsigned MaskNumElts =
Mask.size();
4191 if (SrcNumElts == MaskNumElts) {
4197 if (SrcNumElts < MaskNumElts) {
4201 if (MaskNumElts % SrcNumElts == 0) {
4205 unsigned NumConcat = MaskNumElts / SrcNumElts;
4206 bool IsConcat =
true;
4207 SmallVector<int, 8> ConcatSrcs(NumConcat, -1);
4208 for (
unsigned i = 0; i != MaskNumElts; ++i) {
4214 if ((Idx % SrcNumElts != (i % SrcNumElts)) ||
4215 (ConcatSrcs[i / SrcNumElts] >= 0 &&
4216 ConcatSrcs[i / SrcNumElts] != (
int)(Idx / SrcNumElts))) {
4221 ConcatSrcs[i / SrcNumElts] = Idx / SrcNumElts;
4228 for (
auto Src : ConcatSrcs) {
4241 unsigned PaddedMaskNumElts =
alignTo(MaskNumElts, SrcNumElts);
4242 unsigned NumConcat = PaddedMaskNumElts / SrcNumElts;
4258 SmallVector<int, 8> MappedOps(PaddedMaskNumElts, -1);
4259 for (
unsigned i = 0; i != MaskNumElts; ++i) {
4261 if (Idx >= (
int)SrcNumElts)
4262 Idx -= SrcNumElts - PaddedMaskNumElts;
4270 if (MaskNumElts != PaddedMaskNumElts)
4272 DAG.getVectorIdxConstant(0,
DL));
4278 assert(SrcNumElts > MaskNumElts);
4282 int StartIdx[2] = {-1, -1};
4283 bool CanExtract =
true;
4284 for (
int Idx : Mask) {
4289 if (Idx >= (
int)SrcNumElts) {
4297 int NewStartIdx =
alignDown(Idx, MaskNumElts);
4298 if (NewStartIdx + MaskNumElts > SrcNumElts ||
4299 (StartIdx[Input] >= 0 && StartIdx[Input] != NewStartIdx))
4303 StartIdx[Input] = NewStartIdx;
4306 if (StartIdx[0] < 0 && StartIdx[1] < 0) {
4312 for (
unsigned Input = 0; Input < 2; ++Input) {
4313 SDValue &Src = Input == 0 ? Src1 : Src2;
4314 if (StartIdx[Input] < 0)
4315 Src =
DAG.getUNDEF(VT);
4318 DAG.getVectorIdxConstant(StartIdx[Input],
DL));
4323 SmallVector<int, 8> MappedOps(Mask);
4324 for (
int &Idx : MappedOps) {
4325 if (Idx >= (
int)SrcNumElts)
4326 Idx -= SrcNumElts + StartIdx[1] - MaskNumElts;
4331 setValue(&
I,
DAG.getVectorShuffle(VT,
DL, Src1, Src2, MappedOps));
4340 for (
int Idx : Mask) {
4344 Res =
DAG.getUNDEF(EltVT);
4346 SDValue &Src = Idx < (int)SrcNumElts ? Src1 : Src2;
4347 if (Idx >= (
int)SrcNumElts) Idx -= SrcNumElts;
4350 DAG.getVectorIdxConstant(Idx,
DL));
4360 ArrayRef<unsigned> Indices =
I.getIndices();
4361 const Value *Op0 =
I.getOperand(0);
4363 Type *AggTy =
I.getType();
4370 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4376 unsigned NumAggValues = AggValueVTs.
size();
4377 unsigned NumValValues = ValValueVTs.
size();
4381 if (!NumAggValues) {
4389 for (; i != LinearIndex; ++i)
4390 Values[i] = IntoUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4395 for (; i != LinearIndex + NumValValues; ++i)
4396 Values[i] = FromUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4400 for (; i != NumAggValues; ++i)
4401 Values[i] = IntoUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4405 DAG.getVTList(AggValueVTs), Values));
4409 ArrayRef<unsigned> Indices =
I.getIndices();
4410 const Value *Op0 =
I.getOperand(0);
4412 Type *ValTy =
I.getType();
4417 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4421 unsigned NumValValues = ValValueVTs.
size();
4424 if (!NumValValues) {
4433 for (
unsigned i = LinearIndex; i != LinearIndex + NumValValues; ++i)
4434 Values[i - LinearIndex] =
4440 DAG.getVTList(ValValueVTs), Values));
4443void SelectionDAGBuilder::visitGetElementPtr(
const User &
I) {
4444 Value *Op0 =
I.getOperand(0);
4450 auto &TLI =
DAG.getTargetLoweringInfo();
4455 bool IsVectorGEP =
I.getType()->isVectorTy();
4456 ElementCount VectorElementCount =
4462 const Value *Idx = GTI.getOperand();
4463 if (StructType *StTy = GTI.getStructTypeOrNull()) {
4468 DAG.getDataLayout().getStructLayout(StTy)->getElementOffset(
Field);
4478 N =
DAG.getMemBasePlusOffset(
4479 N,
DAG.getConstant(
Offset, dl,
N.getValueType()), dl, Flags);
4485 unsigned IdxSize =
DAG.getDataLayout().getIndexSizeInBits(AS);
4487 TypeSize ElementSize =
4488 GTI.getSequentialElementStride(
DAG.getDataLayout());
4493 bool ElementScalable = ElementSize.
isScalable();
4499 C =
C->getSplatValue();
4502 if (CI && CI->isZero())
4504 if (CI && !ElementScalable) {
4505 APInt Offs = ElementMul * CI->getValue().sextOrTrunc(IdxSize);
4508 if (
N.getValueType().isVector())
4509 OffsVal =
DAG.getConstant(
4512 OffsVal =
DAG.getConstant(Offs, dl, IdxTy);
4519 Flags.setNoUnsignedWrap(
true);
4522 OffsVal =
DAG.getSExtOrTrunc(OffsVal, dl,
N.getValueType());
4524 N =
DAG.getMemBasePlusOffset(
N, OffsVal, dl, Flags);
4532 if (
N.getValueType().isVector()) {
4534 VectorElementCount);
4535 IdxN =
DAG.getSplat(VT, dl, IdxN);
4539 N =
DAG.getSplat(VT, dl,
N);
4545 IdxN =
DAG.getSExtOrTrunc(IdxN, dl,
N.getValueType());
4547 SDNodeFlags ScaleFlags;
4556 if (ElementScalable) {
4557 EVT VScaleTy =
N.getValueType().getScalarType();
4560 DAG.getConstant(ElementMul.getZExtValue(), dl, VScaleTy));
4561 if (
N.getValueType().isVector())
4562 VScale =
DAG.getSplatVector(
N.getValueType(), dl, VScale);
4563 IdxN =
DAG.getNode(
ISD::MUL, dl,
N.getValueType(), IdxN, VScale,
4568 if (ElementMul != 1) {
4569 if (ElementMul.isPowerOf2()) {
4570 unsigned Amt = ElementMul.logBase2();
4573 DAG.getShiftAmountConstant(Amt,
N.getValueType(), dl),
4576 SDValue Scale =
DAG.getConstant(ElementMul.getZExtValue(), dl,
4578 IdxN =
DAG.getNode(
ISD::MUL, dl,
N.getValueType(), IdxN, Scale,
4588 SDNodeFlags AddFlags;
4592 N =
DAG.getMemBasePlusOffset(
N, IdxN, dl, AddFlags);
4596 if (IsVectorGEP && !
N.getValueType().isVector()) {
4598 N =
DAG.getSplat(VT, dl,
N);
4609 N =
DAG.getPtrExtendInReg(
N, dl, PtrMemTy);
4614void SelectionDAGBuilder::visitAlloca(
const AllocaInst &
I) {
4621 Type *Ty =
I.getAllocatedType();
4622 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4623 auto &
DL =
DAG.getDataLayout();
4624 TypeSize TySize =
DL.getTypeAllocSize(Ty);
4625 MaybeAlign Alignment =
I.getAlign();
4631 AllocSize =
DAG.getZExtOrTrunc(AllocSize, dl, IntPtr);
4633 AllocSize =
DAG.getNode(
4635 DAG.getZExtOrTrunc(
DAG.getTypeSize(dl, MVT::i64, TySize), dl, IntPtr));
4640 Align StackAlign =
DAG.getSubtarget().getFrameLowering()->getStackAlign();
4641 if (*Alignment <= StackAlign)
4642 Alignment = std::nullopt;
4644 const uint64_t StackAlignMask = StackAlign.
value() - 1U;
4649 DAG.getConstant(StackAlignMask, dl, IntPtr),
4654 DAG.getSignedConstant(~StackAlignMask, dl, IntPtr));
4658 DAG.getConstant(Alignment ? Alignment->value() : 0, dl, IntPtr)};
4668 return I.getMetadata(LLVMContext::MD_range);
4673 if (std::optional<ConstantRange> CR = CB->getRange())
4677 return std::nullopt;
4682 return CB->getRetNoFPClass();
4686void SelectionDAGBuilder::visitLoad(
const LoadInst &
I) {
4688 return visitAtomicLoad(
I);
4690 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4691 const Value *SV =
I.getOperand(0);
4696 if (Arg->hasSwiftErrorAttr())
4697 return visitLoadFromSwiftError(
I);
4701 if (Alloca->isSwiftError())
4702 return visitLoadFromSwiftError(
I);
4708 Type *Ty =
I.getType();
4712 unsigned NumValues = ValueVTs.
size();
4716 Align Alignment =
I.getAlign();
4717 AAMDNodes AAInfo =
I.getAAMetadata();
4719 bool isVolatile =
I.isVolatile();
4724 bool ConstantMemory =
false;
4731 BatchAA->pointsToConstantMemory(MemoryLocation(
4736 Root =
DAG.getEntryNode();
4737 ConstantMemory =
true;
4741 Root =
DAG.getRoot();
4752 unsigned ChainI = 0;
4753 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4769 MachinePointerInfo PtrInfo =
4771 ? MachinePointerInfo(SV, Offsets[i].getKnownMinValue())
4772 : MachinePointerInfo();
4774 SDValue A =
DAG.getObjectPtrOffset(dl, Ptr, Offsets[i]);
4775 SDValue L =
DAG.getLoad(MemVTs[i], dl, Root,
A, PtrInfo, Alignment,
4776 MMOFlags, AAInfo, Ranges);
4777 Chains[ChainI] =
L.getValue(1);
4779 if (MemVTs[i] != ValueVTs[i])
4780 L =
DAG.getPtrExtOrTrunc(L, dl, ValueVTs[i]);
4785 if (!ConstantMemory) {
4795 DAG.getVTList(ValueVTs), Values));
4798void SelectionDAGBuilder::visitStoreToSwiftError(
const StoreInst &
I) {
4799 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4800 "call visitStoreToSwiftError when backend supports swifterror");
4803 SmallVector<uint64_t, 4>
Offsets;
4804 const Value *SrcV =
I.getOperand(0);
4806 SrcV->
getType(), ValueVTs,
nullptr, &Offsets, 0);
4807 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4808 "expect a single EVT for swifterror");
4817 SDValue(Src.getNode(), Src.getResNo()));
4818 DAG.setRoot(CopyNode);
4821void SelectionDAGBuilder::visitLoadFromSwiftError(
const LoadInst &
I) {
4822 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4823 "call visitLoadFromSwiftError when backend supports swifterror");
4826 !
I.hasMetadata(LLVMContext::MD_nontemporal) &&
4827 !
I.hasMetadata(LLVMContext::MD_invariant_load) &&
4828 "Support volatile, non temporal, invariant for load_from_swift_error");
4830 const Value *SV =
I.getOperand(0);
4831 Type *Ty =
I.getType();
4834 !
BatchAA->pointsToConstantMemory(MemoryLocation(
4836 I.getAAMetadata()))) &&
4837 "load_from_swift_error should not be constant memory");
4840 SmallVector<uint64_t, 4>
Offsets;
4842 ValueVTs,
nullptr, &Offsets, 0);
4843 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4844 "expect a single EVT for swifterror");
4854void SelectionDAGBuilder::visitStore(
const StoreInst &
I) {
4856 return visitAtomicStore(
I);
4858 const Value *SrcV =
I.getOperand(0);
4859 const Value *PtrV =
I.getOperand(1);
4861 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4866 if (Arg->hasSwiftErrorAttr())
4867 return visitStoreToSwiftError(
I);
4871 if (Alloca->isSwiftError())
4872 return visitStoreToSwiftError(
I);
4879 SrcV->
getType(), ValueVTs, &MemVTs, &Offsets);
4880 unsigned NumValues = ValueVTs.
size();
4893 Align Alignment =
I.getAlign();
4894 AAMDNodes AAInfo =
I.getAAMetadata();
4898 unsigned ChainI = 0;
4899 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4909 MachinePointerInfo PtrInfo =
4911 ? MachinePointerInfo(PtrV, Offsets[i].getKnownMinValue())
4912 : MachinePointerInfo();
4916 if (MemVTs[i] != ValueVTs[i])
4917 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVTs[i]);
4919 DAG.getStore(Root, dl, Val,
Add, PtrInfo, Alignment, MMOFlags, AAInfo);
4920 Chains[ChainI] = St;
4926 DAG.setRoot(StoreNode);
4929void SelectionDAGBuilder::visitMaskedStore(
const CallInst &
I,
4930 bool IsCompressing) {
4933 Value *Src0Operand =
I.getArgOperand(0);
4934 Value *PtrOperand =
I.getArgOperand(1);
4935 Value *MaskOperand =
I.getArgOperand(2);
4936 Align Alignment =
I.getParamAlign(1).valueOrOne();
4946 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
4949 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
4950 MachinePointerInfo(PtrOperand), MMOFlags,
4953 const auto &TLI =
DAG.getTargetLoweringInfo();
4956 !IsCompressing &&
TTI->hasConditionalLoadStoreForType(
4957 I.getArgOperand(0)->getType(),
true)
4963 DAG.setRoot(StoreNode);
4993 C =
C->getSplatValue();
5007 if (!
GEP ||
GEP->getParent() != CurBB)
5010 if (
GEP->getNumOperands() != 2)
5013 const Value *BasePtr =
GEP->getPointerOperand();
5014 const Value *IndexVal =
GEP->getOperand(
GEP->getNumOperands() - 1);
5020 TypeSize ScaleVal =
DL.getTypeAllocSize(
GEP->getResultElementType());
5025 if (ScaleVal != 1 &&
5037void SelectionDAGBuilder::visitMaskedScatter(
const CallInst &
I) {
5041 const Value *Ptr =
I.getArgOperand(1);
5045 Align Alignment =
I.getParamAlign(1).valueOrOne();
5046 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5055 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5065 EVT IdxVT =
Index.getValueType();
5073 SDValue Scatter =
DAG.getMaskedScatter(
DAG.getVTList(MVT::Other), VT, sdl,
5075 DAG.setRoot(Scatter);
5079void SelectionDAGBuilder::visitMaskedLoad(
const CallInst &
I,
bool IsExpanding) {
5082 Value *PtrOperand =
I.getArgOperand(0);
5083 Value *MaskOperand =
I.getArgOperand(1);
5084 Value *Src0Operand =
I.getArgOperand(2);
5085 Align Alignment =
I.getParamAlign(0).valueOrOne();
5093 AAMDNodes AAInfo =
I.getAAMetadata();
5100 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
5103 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
5105 if (
I.hasMetadata(LLVMContext::MD_invariant_load))
5108 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5109 MachinePointerInfo(PtrOperand), MMOFlags,
5112 const auto &TLI =
DAG.getTargetLoweringInfo();
5119 TTI->hasConditionalLoadStoreForType(Src0Operand->
getType(),
5124 DAG.getMaskedLoad(VT, sdl, InChain, Ptr,
Offset, Mask, Src0, VT, MMO,
5131void SelectionDAGBuilder::visitMaskedGather(
const CallInst &
I) {
5135 const Value *Ptr =
I.getArgOperand(0);
5139 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5141 Align Alignment =
I.getParamAlign(0).valueOrOne();
5152 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5164 EVT IdxVT =
Index.getValueType();
5173 DAG.getMaskedGather(
DAG.getVTList(VT, MVT::Other), VT, sdl,
Ops, MMO,
5189 SDVTList VTs =
DAG.getVTList(MemVT, MVT::i1, MVT::Other);
5191 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5194 MachineFunction &MF =
DAG.getMachineFunction();
5196 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5197 DAG.getEVTAlign(MemVT), AAMDNodes(),
nullptr, SSID, SuccessOrdering,
5201 dl, MemVT, VTs, InChain,
5209 DAG.setRoot(OutChain);
5212void SelectionDAGBuilder::visitAtomicRMW(
const AtomicRMWInst &
I) {
5215 switch (
I.getOperation()) {
5257 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5260 MachineFunction &MF =
DAG.getMachineFunction();
5262 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5263 DAG.getEVTAlign(MemVT), AAMDNodes(),
nullptr, SSID, Ordering);
5266 DAG.getAtomic(NT, dl, MemVT, InChain,
5273 DAG.setRoot(OutChain);
5276void SelectionDAGBuilder::visitFence(
const FenceInst &
I) {
5278 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5281 Ops[1] =
DAG.getTargetConstant((
unsigned)
I.getOrdering(), dl,
5283 Ops[2] =
DAG.getTargetConstant(
I.getSyncScopeID(), dl,
5290void SelectionDAGBuilder::visitAtomicLoad(
const LoadInst &
I) {
5297 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5308 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5309 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5310 I.getAlign(), AAMDNodes(), Ranges, SSID, Order);
5320 L =
DAG.getPtrExtOrTrunc(L, dl, VT);
5323 DAG.setRoot(OutChain);
5326void SelectionDAGBuilder::visitAtomicStore(
const StoreInst &
I) {
5334 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5344 MachineFunction &MF =
DAG.getMachineFunction();
5346 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5347 I.getAlign(), AAMDNodes(),
nullptr, SSID, Ordering);
5351 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVT);
5358 DAG.setRoot(OutChain);
5366std::pair<bool, bool>
5367SelectionDAGBuilder::getTargetIntrinsicCallProperties(
const CallBase &
I) {
5369 bool HasChain = !
F->doesNotAccessMemory();
5371 HasChain &&
F->onlyReadsMemory() &&
F->willReturn() &&
F->doesNotThrow();
5373 return {HasChain, OnlyLoad};
5377 const CallBase &
I,
bool HasChain,
bool OnlyLoad,
5379 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5386 Ops.push_back(
DAG.getRoot());
5399 for (
unsigned i = 0, e =
I.arg_size(); i != e; ++i) {
5400 const Value *Arg =
I.getArgOperand(i);
5401 if (!
I.paramHasAttr(i, Attribute::ImmArg)) {
5409 assert(CI->getBitWidth() <= 64 &&
5410 "large intrinsic immediates not handled");
5411 Ops.push_back(
DAG.getTargetConstant(*CI, SDLoc(), VT));
5418 if (std::optional<OperandBundleUse> Bundle =
5420 auto *Sym = Bundle->Inputs[0].get();
5423 Ops.push_back(SDSym);
5426 if (std::optional<OperandBundleUse> Bundle =
5428 Value *Token = Bundle->Inputs[0].get();
5430 assert(
Ops.back().getValueType() != MVT::Glue &&
5431 "Did not expect another glue node here.");
5434 Ops.push_back(ConvControlToken);
5442 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5450 return DAG.getVTList(ValueVTs);
5454SDValue SelectionDAGBuilder::getTargetNonMemIntrinsicNode(
5477 if (
I.getType()->isVoidTy())
5492void SelectionDAGBuilder::visitTargetIntrinsic(
const CallInst &
I,
5494 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
5498 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5501 TargetLowering::IntrinsicInfo *
Info = !Infos.
empty() ? &Infos[0] :
nullptr;
5504 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad, Info);
5505 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
5510 Flags.copyFMF(*FPMO);
5511 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
5518 if (!Infos.
empty()) {
5521 MachineFunction &MF =
DAG.getMachineFunction();
5523 for (
const auto &Info : Infos) {
5526 MachinePointerInfo MPI;
5528 MPI = MachinePointerInfo(
Info.ptrVal,
Info.offset);
5529 else if (
Info.fallbackAddressSpace)
5530 MPI = MachinePointerInfo(*
Info.fallbackAddressSpace);
5531 EVT MemVT =
Info.memVT;
5533 if (
Size.hasValue() && !
Size.getValue())
5535 Align Alignment =
Info.align.value_or(
DAG.getEVTAlign(MemVT));
5537 MPI,
Info.flags,
Size, Alignment,
I.getAAMetadata(),
5545 Result = getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
5548 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
5605 SDValue TwoToFractionalPartOfX;
5682 if (
Op.getValueType() == MVT::f32 &&
5706 if (
Op.getValueType() == MVT::f32 &&
5805 if (
Op.getValueType() == MVT::f32 &&
5889 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log2ofMantissa);
5902 if (
Op.getValueType() == MVT::f32 &&
5979 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log10ofMantissa);
5990 if (
Op.getValueType() == MVT::f32 &&
6003 bool IsExp10 =
false;
6004 if (
LHS.getValueType() == MVT::f32 &&
RHS.getValueType() == MVT::f32 &&
6008 IsExp10 = LHSC->isExactlyValue(Ten);
6035 unsigned Val = RHSC->getSExtValue();
6064 CurSquare, CurSquare);
6069 if (RHSC->getSExtValue() < 0)
6083 EVT VT =
LHS.getValueType();
6106 if ((ScaleInt > 0 || (Saturating &&
Signed)) &&
6110 Opcode, VT, ScaleInt);
6145 switch (
N.getOpcode()) {
6149 Op.getValueType().getSizeInBits());
6174bool SelectionDAGBuilder::EmitFuncArgumentDbgValue(
6181 MachineFunction &MF =
DAG.getMachineFunction();
6182 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
6186 auto MakeVRegDbgValue = [&](
Register Reg, DIExpression *FragExpr,
6191 auto &Inst =
TII->get(TargetOpcode::DBG_INSTR_REF);
6198 auto *NewDIExpr = FragExpr;
6205 return BuildMI(MF,
DL, Inst,
false, MOs, Variable, NewDIExpr);
6208 auto &Inst =
TII->get(TargetOpcode::DBG_VALUE);
6209 return BuildMI(MF,
DL, Inst, Indirect,
Reg, Variable, FragExpr);
6213 if (Kind == FuncArgumentDbgValueKind::Value) {
6218 if (!IsInEntryBlock)
6234 bool VariableIsFunctionInputArg =
Variable->isParameter() &&
6235 !
DL->getInlinedAt();
6237 if (!IsInPrologue && !VariableIsFunctionInputArg)
6271 if (VariableIsFunctionInputArg) {
6273 if (ArgNo >=
FuncInfo.DescribedArgs.size())
6274 FuncInfo.DescribedArgs.resize(ArgNo + 1,
false);
6275 else if (!IsInPrologue &&
FuncInfo.DescribedArgs.test(ArgNo))
6276 return !NodeMap[
V].getNode();
6281 bool IsIndirect =
false;
6282 std::optional<MachineOperand>
Op;
6284 int FI =
FuncInfo.getArgumentFrameIndex(Arg);
6285 if (FI != std::numeric_limits<int>::max())
6289 if (!
Op &&
N.getNode()) {
6292 if (ArgRegsAndSizes.
size() == 1)
6293 Reg = ArgRegsAndSizes.
front().first;
6296 MachineRegisterInfo &RegInfo = MF.
getRegInfo();
6303 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6307 if (!
Op &&
N.getNode()) {
6311 if (FrameIndexSDNode *FINode =
6318 auto splitMultiRegDbgValue =
6331 uint64_t ExprFragmentSizeInBits = ExprFragmentInfo->SizeInBits;
6334 if (
Offset >= ExprFragmentSizeInBits)
6338 if (
Offset + RegFragmentSizeInBits > ExprFragmentSizeInBits) {
6339 RegFragmentSizeInBits = ExprFragmentSizeInBits -
Offset;
6344 Expr,
Offset, RegFragmentSizeInBits);
6348 if (!FragmentExpr) {
6349 SDDbgValue *SDV =
DAG.getConstantDbgValue(
6351 DAG.AddDbgValue(SDV,
false);
6354 MachineInstr *NewMI = MakeVRegDbgValue(
6355 Reg, *FragmentExpr, Kind != FuncArgumentDbgValueKind::Value);
6356 FuncInfo.ArgDbgValues.push_back(NewMI);
6365 if (VMI !=
FuncInfo.ValueMap.end()) {
6366 const auto &TLI =
DAG.getTargetLoweringInfo();
6367 RegsForValue RFV(
V->getContext(), TLI,
DAG.getDataLayout(), VMI->second,
6368 V->getType(), std::nullopt);
6369 if (RFV.occupiesMultipleRegs())
6370 return splitMultiRegDbgValue(RFV.getRegsAndSizes());
6373 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6374 }
else if (ArgRegsAndSizes.
size() > 1) {
6377 return splitMultiRegDbgValue(ArgRegsAndSizes);
6385 "Expected inlined-at fields to agree");
6386 MachineInstr *NewMI =
nullptr;
6389 NewMI = MakeVRegDbgValue(
Op->getReg(), Expr, IsIndirect);
6391 NewMI =
BuildMI(MF,
DL,
TII->get(TargetOpcode::DBG_VALUE),
true, *
Op,
6395 FuncInfo.ArgDbgValues.push_back(NewMI);
6404 unsigned DbgSDNodeOrder) {
6416 return DAG.getFrameIndexDbgValue(Variable, Expr, FISDN->getIndex(),
6417 false, dl, DbgSDNodeOrder);
6419 return DAG.getDbgValue(Variable, Expr,
N.getNode(),
N.getResNo(),
6420 false, dl, DbgSDNodeOrder);
6425 case Intrinsic::smul_fix:
6427 case Intrinsic::umul_fix:
6429 case Intrinsic::smul_fix_sat:
6431 case Intrinsic::umul_fix_sat:
6433 case Intrinsic::sdiv_fix:
6435 case Intrinsic::udiv_fix:
6437 case Intrinsic::sdiv_fix_sat:
6439 case Intrinsic::udiv_fix_sat:
6452 "expected call_preallocated_setup Value");
6453 for (
const auto *U : PreallocatedSetup->
users()) {
6455 const Function *Fn = UseCall->getCalledFunction();
6456 if (!Fn || Fn->
getIntrinsicID() != Intrinsic::call_preallocated_arg) {
6466bool SelectionDAGBuilder::visitEntryValueDbgValue(
6476 auto ArgIt =
FuncInfo.ValueMap.find(Arg);
6477 if (ArgIt ==
FuncInfo.ValueMap.end()) {
6479 dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6480 "couldn't find an associated register for the Argument\n");
6483 Register ArgVReg = ArgIt->getSecond();
6485 for (
auto [PhysReg, VirtReg] :
FuncInfo.RegInfo->liveins())
6486 if (ArgVReg == VirtReg || ArgVReg == PhysReg) {
6487 SDDbgValue *SDV =
DAG.getVRegDbgValue(
6488 Variable, Expr, PhysReg,
false , DbgLoc, SDNodeOrder);
6489 DAG.AddDbgValue(SDV,
false );
6492 LLVM_DEBUG(
dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6493 "couldn't find a physical register\n");
6498void SelectionDAGBuilder::visitConvergenceControl(
const CallInst &
I,
6501 switch (Intrinsic) {
6502 case Intrinsic::experimental_convergence_anchor:
6505 case Intrinsic::experimental_convergence_entry:
6508 case Intrinsic::experimental_convergence_loop: {
6510 auto *Token = Bundle->Inputs[0].get();
6518void SelectionDAGBuilder::visitVectorHistogram(
const CallInst &
I,
6519 unsigned IntrinsicID) {
6522 assert(IntrinsicID == Intrinsic::experimental_vector_histogram_add &&
6523 "Tried to lower unsupported histogram type");
6529 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6530 DataLayout TargetDL =
DAG.getDataLayout();
6532 Align Alignment =
DAG.getEVTAlign(VT);
6545 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
6546 MachinePointerInfo(AS),
6557 EVT IdxVT =
Index.getValueType();
6568 SDValue ID =
DAG.getTargetConstant(IntrinsicID, sdl, MVT::i32);
6571 SDValue Histogram =
DAG.getMaskedHistogram(
DAG.getVTList(MVT::Other), VT, sdl,
6575 DAG.setRoot(Histogram);
6578void SelectionDAGBuilder::visitVectorExtractLastActive(
const CallInst &
I,
6580 assert(Intrinsic == Intrinsic::experimental_vector_extract_last_active &&
6581 "Tried lowering invalid vector extract last");
6583 const DataLayout &Layout =
DAG.getDataLayout();
6587 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6597 EVT BoolVT =
Mask.getValueType().getScalarType();
6599 Result =
DAG.getSelect(sdl, ResVT, AnyActive, Result, PassThru);
6606void SelectionDAGBuilder::visitIntrinsicCall(
const CallInst &
I,
6608 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6615 Flags.copyFMF(*FPOp);
6617 switch (Intrinsic) {
6620 visitTargetIntrinsic(
I, Intrinsic);
6622 case Intrinsic::vscale: {
6627 case Intrinsic::vastart: visitVAStart(
I);
return;
6628 case Intrinsic::vaend: visitVAEnd(
I);
return;
6629 case Intrinsic::vacopy: visitVACopy(
I);
return;
6630 case Intrinsic::returnaddress:
6635 case Intrinsic::addressofreturnaddress:
6640 case Intrinsic::sponentry:
6645 case Intrinsic::frameaddress:
6650 case Intrinsic::read_volatile_register:
6651 case Intrinsic::read_register: {
6652 Value *
Reg =
I.getArgOperand(0);
6658 DAG.getVTList(VT, MVT::Other), Chain,
RegName);
6663 case Intrinsic::write_register: {
6664 Value *
Reg =
I.getArgOperand(0);
6665 Value *RegValue =
I.getArgOperand(1);
6673 case Intrinsic::memcpy:
6674 case Intrinsic::memcpy_inline: {
6680 "memcpy_inline needs constant size");
6682 Align DstAlign = MCI.getDestAlign().valueOrOne();
6683 Align SrcAlign = MCI.getSourceAlign().valueOrOne();
6684 Align Alignment = std::min(DstAlign, SrcAlign);
6685 bool isVol = MCI.isVolatile();
6689 SDValue MC =
DAG.getMemcpy(Root, sdl, Dst, Src,
Size, Alignment, isVol,
6690 MCI.isForceInlined(), &
I, std::nullopt,
6691 MachinePointerInfo(
I.getArgOperand(0)),
6692 MachinePointerInfo(
I.getArgOperand(1)),
6694 updateDAGForMaybeTailCall(MC);
6697 case Intrinsic::memset:
6698 case Intrinsic::memset_inline: {
6704 "memset_inline needs constant size");
6706 Align DstAlign = MSII.getDestAlign().valueOrOne();
6707 bool isVol = MSII.isVolatile();
6710 Root, sdl, Dst, Value,
Size, DstAlign, isVol, MSII.isForceInlined(),
6711 &
I, MachinePointerInfo(
I.getArgOperand(0)),
I.getAAMetadata());
6712 updateDAGForMaybeTailCall(MC);
6715 case Intrinsic::memmove: {
6721 Align DstAlign = MMI.getDestAlign().valueOrOne();
6722 Align SrcAlign = MMI.getSourceAlign().valueOrOne();
6723 Align Alignment = std::min(DstAlign, SrcAlign);
6724 bool isVol = MMI.isVolatile();
6728 SDValue MM =
DAG.getMemmove(Root, sdl, Op1, Op2, Op3, Alignment, isVol, &
I,
6730 MachinePointerInfo(
I.getArgOperand(0)),
6731 MachinePointerInfo(
I.getArgOperand(1)),
6733 updateDAGForMaybeTailCall(MM);
6736 case Intrinsic::memcpy_element_unordered_atomic: {
6742 Type *LengthTy =
MI.getLength()->getType();
6743 unsigned ElemSz =
MI.getElementSizeInBytes();
6747 isTC, MachinePointerInfo(
MI.getRawDest()),
6748 MachinePointerInfo(
MI.getRawSource()));
6749 updateDAGForMaybeTailCall(MC);
6752 case Intrinsic::memmove_element_unordered_atomic: {
6758 Type *LengthTy =
MI.getLength()->getType();
6759 unsigned ElemSz =
MI.getElementSizeInBytes();
6763 isTC, MachinePointerInfo(
MI.getRawDest()),
6764 MachinePointerInfo(
MI.getRawSource()));
6765 updateDAGForMaybeTailCall(MC);
6768 case Intrinsic::memset_element_unordered_atomic: {
6774 Type *LengthTy =
MI.getLength()->getType();
6775 unsigned ElemSz =
MI.getElementSizeInBytes();
6779 isTC, MachinePointerInfo(
MI.getRawDest()));
6780 updateDAGForMaybeTailCall(MC);
6783 case Intrinsic::call_preallocated_setup: {
6785 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6792 case Intrinsic::call_preallocated_arg: {
6794 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6808 case Intrinsic::eh_typeid_for: {
6811 unsigned TypeID =
DAG.getMachineFunction().getTypeIDFor(GV);
6812 Res =
DAG.getConstant(
TypeID, sdl, MVT::i32);
6817 case Intrinsic::eh_return_i32:
6818 case Intrinsic::eh_return_i64:
6819 DAG.getMachineFunction().setCallsEHReturn(
true);
6826 case Intrinsic::eh_unwind_init:
6827 DAG.getMachineFunction().setCallsUnwindInit(
true);
6829 case Intrinsic::eh_dwarf_cfa:
6834 case Intrinsic::eh_sjlj_callsite: {
6836 assert(
FuncInfo.getCurrentCallSite() == 0 &&
"Overlapping call sites!");
6841 case Intrinsic::eh_sjlj_functioncontext: {
6843 MachineFrameInfo &MFI =
DAG.getMachineFunction().getFrameInfo();
6846 int FI =
FuncInfo.StaticAllocaMap[FnCtx];
6850 case Intrinsic::eh_sjlj_setjmp: {
6855 DAG.getVTList(MVT::i32, MVT::Other),
Ops);
6857 DAG.setRoot(
Op.getValue(1));
6860 case Intrinsic::eh_sjlj_longjmp:
6864 case Intrinsic::eh_sjlj_setup_dispatch:
6868 case Intrinsic::masked_gather:
6869 visitMaskedGather(
I);
6871 case Intrinsic::masked_load:
6874 case Intrinsic::masked_scatter:
6875 visitMaskedScatter(
I);
6877 case Intrinsic::masked_store:
6878 visitMaskedStore(
I);
6880 case Intrinsic::masked_expandload:
6881 visitMaskedLoad(
I,
true );
6883 case Intrinsic::masked_compressstore:
6884 visitMaskedStore(
I,
true );
6886 case Intrinsic::powi:
6890 case Intrinsic::log:
6893 case Intrinsic::log2:
6897 case Intrinsic::log10:
6901 case Intrinsic::exp:
6904 case Intrinsic::exp2:
6908 case Intrinsic::pow:
6912 case Intrinsic::sqrt:
6913 case Intrinsic::fabs:
6914 case Intrinsic::sin:
6915 case Intrinsic::cos:
6916 case Intrinsic::tan:
6917 case Intrinsic::asin:
6918 case Intrinsic::acos:
6919 case Intrinsic::atan:
6920 case Intrinsic::sinh:
6921 case Intrinsic::cosh:
6922 case Intrinsic::tanh:
6923 case Intrinsic::exp10:
6924 case Intrinsic::floor:
6925 case Intrinsic::ceil:
6926 case Intrinsic::trunc:
6927 case Intrinsic::rint:
6928 case Intrinsic::nearbyint:
6929 case Intrinsic::round:
6930 case Intrinsic::roundeven:
6931 case Intrinsic::canonicalize: {
6934 switch (Intrinsic) {
6936 case Intrinsic::sqrt: Opcode =
ISD::FSQRT;
break;
6937 case Intrinsic::fabs: Opcode =
ISD::FABS;
break;
6938 case Intrinsic::sin: Opcode =
ISD::FSIN;
break;
6939 case Intrinsic::cos: Opcode =
ISD::FCOS;
break;
6940 case Intrinsic::tan: Opcode =
ISD::FTAN;
break;
6941 case Intrinsic::asin: Opcode =
ISD::FASIN;
break;
6942 case Intrinsic::acos: Opcode =
ISD::FACOS;
break;
6943 case Intrinsic::atan: Opcode =
ISD::FATAN;
break;
6944 case Intrinsic::sinh: Opcode =
ISD::FSINH;
break;
6945 case Intrinsic::cosh: Opcode =
ISD::FCOSH;
break;
6946 case Intrinsic::tanh: Opcode =
ISD::FTANH;
break;
6947 case Intrinsic::exp10: Opcode =
ISD::FEXP10;
break;
6948 case Intrinsic::floor: Opcode =
ISD::FFLOOR;
break;
6949 case Intrinsic::ceil: Opcode =
ISD::FCEIL;
break;
6950 case Intrinsic::trunc: Opcode =
ISD::FTRUNC;
break;
6951 case Intrinsic::rint: Opcode =
ISD::FRINT;
break;
6953 case Intrinsic::round: Opcode =
ISD::FROUND;
break;
6960 getValue(
I.getArgOperand(0)).getValueType(),
6964 case Intrinsic::atan2:
6966 getValue(
I.getArgOperand(0)).getValueType(),
6970 case Intrinsic::lround:
6971 case Intrinsic::llround:
6972 case Intrinsic::lrint:
6973 case Intrinsic::llrint: {
6976 switch (Intrinsic) {
6978 case Intrinsic::lround: Opcode =
ISD::LROUND;
break;
6980 case Intrinsic::lrint: Opcode =
ISD::LRINT;
break;
6981 case Intrinsic::llrint: Opcode =
ISD::LLRINT;
break;
6990 case Intrinsic::minnum:
6992 getValue(
I.getArgOperand(0)).getValueType(),
6996 case Intrinsic::maxnum:
6998 getValue(
I.getArgOperand(0)).getValueType(),
7002 case Intrinsic::minimum:
7004 getValue(
I.getArgOperand(0)).getValueType(),
7008 case Intrinsic::maximum:
7010 getValue(
I.getArgOperand(0)).getValueType(),
7014 case Intrinsic::minimumnum:
7016 getValue(
I.getArgOperand(0)).getValueType(),
7020 case Intrinsic::maximumnum:
7022 getValue(
I.getArgOperand(0)).getValueType(),
7026 case Intrinsic::copysign:
7028 getValue(
I.getArgOperand(0)).getValueType(),
7032 case Intrinsic::ldexp:
7034 getValue(
I.getArgOperand(0)).getValueType(),
7038 case Intrinsic::modf:
7039 case Intrinsic::sincos:
7040 case Intrinsic::sincospi:
7041 case Intrinsic::frexp: {
7043 switch (Intrinsic) {
7046 case Intrinsic::sincos:
7049 case Intrinsic::sincospi:
7052 case Intrinsic::modf:
7055 case Intrinsic::frexp:
7061 SDVTList VTs =
DAG.getVTList(ValueVTs);
7063 &
I,
DAG.getNode(Opcode, sdl, VTs,
getValue(
I.getArgOperand(0)), Flags));
7066 case Intrinsic::arithmetic_fence: {
7068 getValue(
I.getArgOperand(0)).getValueType(),
7072 case Intrinsic::fma:
7078#define INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC) \
7079 case Intrinsic::INTRINSIC:
7080#include "llvm/IR/ConstrainedOps.def"
7083#define BEGIN_REGISTER_VP_INTRINSIC(VPID, ...) case Intrinsic::VPID:
7084#include "llvm/IR/VPIntrinsics.def"
7087 case Intrinsic::fptrunc_round: {
7091 std::optional<RoundingMode> RoundMode =
7099 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
7104 DAG.getTargetConstant((
int)*RoundMode, sdl, MVT::i32));
7109 case Intrinsic::fmuladd: {
7114 getValue(
I.getArgOperand(0)).getValueType(),
7121 getValue(
I.getArgOperand(0)).getValueType(),
7137 case Intrinsic::fptosi_sat: {
7144 case Intrinsic::fptoui_sat: {
7151 case Intrinsic::set_rounding:
7157 case Intrinsic::is_fpclass: {
7158 const DataLayout DLayout =
DAG.getDataLayout();
7160 EVT ArgVT = TLI.
getValueType(DLayout,
I.getArgOperand(0)->getType());
7163 MachineFunction &MF =
DAG.getMachineFunction();
7167 Flags.setNoFPExcept(
7168 !
F.getAttributes().hasFnAttr(llvm::Attribute::StrictFP));
7184 case Intrinsic::get_fpenv: {
7185 const DataLayout DLayout =
DAG.getDataLayout();
7187 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7202 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7205 Chain =
DAG.getGetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7206 Res =
DAG.getLoad(EnvVT, sdl, Chain, Temp, MPI);
7212 case Intrinsic::set_fpenv: {
7213 const DataLayout DLayout =
DAG.getDataLayout();
7216 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7229 Chain =
DAG.getStore(Chain, sdl, Env, Temp, MPI, TempAlign,
7231 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7234 Chain =
DAG.getSetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7239 case Intrinsic::reset_fpenv:
7242 case Intrinsic::get_fpmode:
7251 case Intrinsic::set_fpmode:
7256 case Intrinsic::reset_fpmode: {
7261 case Intrinsic::pcmarker: {
7266 case Intrinsic::readcyclecounter: {
7269 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7274 case Intrinsic::readsteadycounter: {
7277 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7282 case Intrinsic::bitreverse:
7284 getValue(
I.getArgOperand(0)).getValueType(),
7287 case Intrinsic::bswap:
7289 getValue(
I.getArgOperand(0)).getValueType(),
7292 case Intrinsic::cttz: {
7300 case Intrinsic::ctlz: {
7308 case Intrinsic::ctpop: {
7314 case Intrinsic::fshl:
7315 case Intrinsic::fshr: {
7316 bool IsFSHL =
Intrinsic == Intrinsic::fshl;
7320 EVT VT =
X.getValueType();
7331 case Intrinsic::clmul: {
7337 case Intrinsic::sadd_sat: {
7343 case Intrinsic::uadd_sat: {
7349 case Intrinsic::ssub_sat: {
7355 case Intrinsic::usub_sat: {
7361 case Intrinsic::sshl_sat:
7362 case Intrinsic::ushl_sat: {
7366 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
7371 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
7374 "Unexpected shift type");
7383 case Intrinsic::smul_fix:
7384 case Intrinsic::umul_fix:
7385 case Intrinsic::smul_fix_sat:
7386 case Intrinsic::umul_fix_sat: {
7394 case Intrinsic::sdiv_fix:
7395 case Intrinsic::udiv_fix:
7396 case Intrinsic::sdiv_fix_sat:
7397 case Intrinsic::udiv_fix_sat: {
7402 Op1, Op2, Op3,
DAG, TLI));
7405 case Intrinsic::smax: {
7411 case Intrinsic::smin: {
7417 case Intrinsic::umax: {
7423 case Intrinsic::umin: {
7429 case Intrinsic::abs: {
7435 case Intrinsic::scmp: {
7442 case Intrinsic::ucmp: {
7449 case Intrinsic::stackaddress:
7450 case Intrinsic::stacksave: {
7455 Res =
DAG.getNode(SDOpcode, sdl,
DAG.getVTList(VT, MVT::Other),
Op);
7460 case Intrinsic::stackrestore:
7464 case Intrinsic::get_dynamic_area_offset: {
7473 case Intrinsic::stackguard: {
7474 MachineFunction &MF =
DAG.getMachineFunction();
7480 Res =
DAG.getPtrExtOrTrunc(Res, sdl, PtrTy);
7484 LLVMContext &Ctx = *
DAG.getContext();
7485 Ctx.
diagnose(DiagnosticInfoGeneric(
"unable to lower stackguard"));
7492 MachinePointerInfo(
Global, 0), Align,
7501 case Intrinsic::stackprotector: {
7503 MachineFunction &MF =
DAG.getMachineFunction();
7523 Chain, sdl, Src, FIN,
7530 case Intrinsic::objectsize:
7533 case Intrinsic::is_constant:
7536 case Intrinsic::annotation:
7537 case Intrinsic::ptr_annotation:
7538 case Intrinsic::launder_invariant_group:
7539 case Intrinsic::strip_invariant_group:
7544 case Intrinsic::type_test:
7545 case Intrinsic::public_type_test:
7547 "LowerTypeTests pass before code generation");
7550 case Intrinsic::assume:
7551 case Intrinsic::experimental_noalias_scope_decl:
7552 case Intrinsic::var_annotation:
7553 case Intrinsic::sideeffect:
7558 case Intrinsic::codeview_annotation: {
7560 MachineFunction &MF =
DAG.getMachineFunction();
7569 case Intrinsic::init_trampoline: {
7577 Ops[4] =
DAG.getSrcValue(
I.getArgOperand(0));
7585 case Intrinsic::adjust_trampoline:
7590 case Intrinsic::gcroot: {
7591 assert(
DAG.getMachineFunction().getFunction().hasGC() &&
7592 "only valid in functions with gc specified, enforced by Verifier");
7594 const Value *Alloca =
I.getArgOperand(0)->stripPointerCasts();
7601 case Intrinsic::gcread:
7602 case Intrinsic::gcwrite:
7604 case Intrinsic::get_rounding:
7610 case Intrinsic::expect:
7611 case Intrinsic::expect_with_probability:
7617 case Intrinsic::ubsantrap:
7618 case Intrinsic::debugtrap:
7619 case Intrinsic::trap: {
7620 StringRef TrapFuncName =
7621 I.getAttributes().getFnAttr(
"trap-func-name").getValueAsString();
7622 if (TrapFuncName.
empty()) {
7623 switch (Intrinsic) {
7624 case Intrinsic::trap:
7627 case Intrinsic::debugtrap:
7630 case Intrinsic::ubsantrap:
7633 DAG.getTargetConstant(
7639 DAG.addNoMergeSiteInfo(
DAG.getRoot().getNode(),
7640 I.hasFnAttr(Attribute::NoMerge));
7644 if (Intrinsic == Intrinsic::ubsantrap) {
7645 Value *Arg =
I.getArgOperand(0);
7649 TargetLowering::CallLoweringInfo CLI(
DAG);
7650 CLI.setDebugLoc(sdl).setChain(
getRoot()).setLibCallee(
7652 DAG.getExternalSymbol(TrapFuncName.
data(),
7655 CLI.NoMerge =
I.hasFnAttr(Attribute::NoMerge);
7661 case Intrinsic::allow_runtime_check:
7662 case Intrinsic::allow_ubsan_check:
7666 case Intrinsic::uadd_with_overflow:
7667 case Intrinsic::sadd_with_overflow:
7668 case Intrinsic::usub_with_overflow:
7669 case Intrinsic::ssub_with_overflow:
7670 case Intrinsic::umul_with_overflow:
7671 case Intrinsic::smul_with_overflow: {
7673 switch (Intrinsic) {
7675 case Intrinsic::uadd_with_overflow:
Op =
ISD::UADDO;
break;
7676 case Intrinsic::sadd_with_overflow:
Op =
ISD::SADDO;
break;
7677 case Intrinsic::usub_with_overflow:
Op =
ISD::USUBO;
break;
7678 case Intrinsic::ssub_with_overflow:
Op =
ISD::SSUBO;
break;
7679 case Intrinsic::umul_with_overflow:
Op =
ISD::UMULO;
break;
7680 case Intrinsic::smul_with_overflow:
Op =
ISD::SMULO;
break;
7688 SDVTList VTs =
DAG.getVTList(ResultVT, OverflowVT);
7692 case Intrinsic::prefetch: {
7707 std::nullopt, Flags);
7713 DAG.setRoot(Result);
7716 case Intrinsic::lifetime_start:
7717 case Intrinsic::lifetime_end: {
7718 bool IsStart = (
Intrinsic == Intrinsic::lifetime_start);
7724 if (!LifetimeObject)
7729 auto SI =
FuncInfo.StaticAllocaMap.find(LifetimeObject);
7730 if (SI ==
FuncInfo.StaticAllocaMap.end())
7734 Res =
DAG.getLifetimeNode(IsStart, sdl,
getRoot(), FrameIndex);
7738 case Intrinsic::pseudoprobe: {
7746 case Intrinsic::invariant_start:
7751 case Intrinsic::invariant_end:
7754 case Intrinsic::clear_cache: {
7759 {InputChain, StartVal, EndVal});
7764 case Intrinsic::donothing:
7765 case Intrinsic::seh_try_begin:
7766 case Intrinsic::seh_scope_begin:
7767 case Intrinsic::seh_try_end:
7768 case Intrinsic::seh_scope_end:
7771 case Intrinsic::experimental_stackmap:
7774 case Intrinsic::experimental_patchpoint_void:
7775 case Intrinsic::experimental_patchpoint:
7778 case Intrinsic::experimental_gc_statepoint:
7781 case Intrinsic::experimental_gc_result:
7784 case Intrinsic::experimental_gc_relocate:
7787 case Intrinsic::instrprof_cover:
7789 case Intrinsic::instrprof_increment:
7791 case Intrinsic::instrprof_timestamp:
7793 case Intrinsic::instrprof_value_profile:
7795 case Intrinsic::instrprof_mcdc_parameters:
7797 case Intrinsic::instrprof_mcdc_tvbitmap_update:
7799 case Intrinsic::localescape: {
7800 MachineFunction &MF =
DAG.getMachineFunction();
7801 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
7805 for (
unsigned Idx = 0,
E =
I.arg_size(); Idx <
E; ++Idx) {
7811 "can only escape static allocas");
7816 TII->get(TargetOpcode::LOCAL_ESCAPE))
7824 case Intrinsic::localrecover: {
7826 MachineFunction &MF =
DAG.getMachineFunction();
7832 unsigned(Idx->getLimitedValue(std::numeric_limits<int>::max()));
7836 Value *
FP =
I.getArgOperand(1);
7842 SDValue OffsetSym =
DAG.getMCSymbol(FrameAllocSym, PtrVT);
7847 SDValue Add =
DAG.getMemBasePlusOffset(FPVal, OffsetVal, sdl);
7853 case Intrinsic::fake_use: {
7854 Value *
V =
I.getArgOperand(0);
7859 auto FakeUseValue = [&]() ->
SDValue {
7873 if (!FakeUseValue || FakeUseValue.isUndef())
7876 Ops[1] = FakeUseValue;
7885 case Intrinsic::reloc_none: {
7890 DAG.getTargetExternalSymbol(
7896 case Intrinsic::cond_loop: {
7906 case Intrinsic::eh_exceptionpointer:
7907 case Intrinsic::eh_exceptioncode: {
7913 SDValue N =
DAG.getCopyFromReg(
DAG.getEntryNode(), sdl, VReg, PtrVT);
7914 if (Intrinsic == Intrinsic::eh_exceptioncode)
7915 N =
DAG.getZExtOrTrunc(
N, sdl, MVT::i32);
7919 case Intrinsic::xray_customevent: {
7922 const auto &Triple =
DAG.getTarget().getTargetTriple();
7931 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
7933 Ops.push_back(LogEntryVal);
7934 Ops.push_back(StrSizeVal);
7935 Ops.push_back(Chain);
7941 MachineSDNode *MN =
DAG.getMachineNode(TargetOpcode::PATCHABLE_EVENT_CALL,
7944 DAG.setRoot(patchableNode);
7948 case Intrinsic::xray_typedevent: {
7951 const auto &Triple =
DAG.getTarget().getTargetTriple();
7963 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
7965 Ops.push_back(LogTypeId);
7966 Ops.push_back(LogEntryVal);
7967 Ops.push_back(StrSizeVal);
7968 Ops.push_back(Chain);
7974 MachineSDNode *MN =
DAG.getMachineNode(
7975 TargetOpcode::PATCHABLE_TYPED_EVENT_CALL, sdl, NodeTys,
Ops);
7977 DAG.setRoot(patchableNode);
7981 case Intrinsic::experimental_deoptimize:
7984 case Intrinsic::stepvector:
7987 case Intrinsic::vector_reduce_fadd:
7988 case Intrinsic::vector_reduce_fmul:
7989 case Intrinsic::vector_reduce_add:
7990 case Intrinsic::vector_reduce_mul:
7991 case Intrinsic::vector_reduce_and:
7992 case Intrinsic::vector_reduce_or:
7993 case Intrinsic::vector_reduce_xor:
7994 case Intrinsic::vector_reduce_smax:
7995 case Intrinsic::vector_reduce_smin:
7996 case Intrinsic::vector_reduce_umax:
7997 case Intrinsic::vector_reduce_umin:
7998 case Intrinsic::vector_reduce_fmax:
7999 case Intrinsic::vector_reduce_fmin:
8000 case Intrinsic::vector_reduce_fmaximum:
8001 case Intrinsic::vector_reduce_fminimum:
8002 visitVectorReduce(
I, Intrinsic);
8005 case Intrinsic::icall_branch_funnel: {
8011 I.getArgOperand(1),
Offset,
DAG.getDataLayout()));
8014 "llvm.icall.branch.funnel operand must be a GlobalValue");
8015 Ops.push_back(
DAG.getTargetGlobalAddress(
Base, sdl, MVT::i64, 0));
8017 struct BranchFunnelTarget {
8023 for (
unsigned Op = 1,
N =
I.arg_size();
Op !=
N;
Op += 2) {
8026 if (ElemBase !=
Base)
8028 "to the same GlobalValue");
8034 "llvm.icall.branch.funnel operand must be a GlobalValue");
8040 [](
const BranchFunnelTarget &
T1,
const BranchFunnelTarget &T2) {
8041 return T1.Offset < T2.Offset;
8044 for (
auto &
T : Targets) {
8045 Ops.push_back(
DAG.getTargetConstant(
T.Offset, sdl, MVT::i32));
8046 Ops.push_back(
T.Target);
8049 Ops.push_back(
DAG.getRoot());
8050 SDValue N(
DAG.getMachineNode(TargetOpcode::ICALL_BRANCH_FUNNEL, sdl,
8059 case Intrinsic::wasm_landingpad_index:
8065 case Intrinsic::aarch64_settag:
8066 case Intrinsic::aarch64_settag_zero: {
8067 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
8068 bool ZeroMemory =
Intrinsic == Intrinsic::aarch64_settag_zero;
8071 getValue(
I.getArgOperand(1)), MachinePointerInfo(
I.getArgOperand(0)),
8077 case Intrinsic::amdgcn_cs_chain: {
8082 Type *RetTy =
I.getType();
8092 for (
unsigned Idx : {2, 3, 1}) {
8093 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8095 Arg.setAttributes(&
I, Idx);
8096 Args.push_back(Arg);
8099 assert(Args[0].IsInReg &&
"SGPR args should be marked inreg");
8100 assert(!Args[1].IsInReg &&
"VGPR args should not be marked inreg");
8101 Args[2].IsInReg =
true;
8104 for (
unsigned Idx = 4; Idx <
I.arg_size(); ++Idx) {
8105 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8107 Arg.setAttributes(&
I, Idx);
8108 Args.push_back(Arg);
8111 TargetLowering::CallLoweringInfo CLI(
DAG);
8114 .setCallee(CC, RetTy, Callee, std::move(Args))
8117 .setConvergent(
I.isConvergent());
8119 std::pair<SDValue, SDValue>
Result =
8123 "Should've lowered as tail call");
8128 case Intrinsic::amdgcn_call_whole_wave: {
8130 bool isTailCall =
I.isTailCall();
8133 for (
unsigned Idx = 1; Idx <
I.arg_size(); ++Idx) {
8134 TargetLowering::ArgListEntry Arg(
getValue(
I.getArgOperand(Idx)),
8135 I.getArgOperand(Idx)->getType());
8136 Arg.setAttributes(&
I, Idx);
8143 Args.push_back(Arg);
8148 auto *Token = Bundle->Inputs[0].get();
8149 ConvControlToken =
getValue(Token);
8152 TargetLowering::CallLoweringInfo CLI(
DAG);
8156 getValue(
I.getArgOperand(0)), std::move(Args))
8160 .setConvergent(
I.isConvergent())
8161 .setConvergenceControlToken(ConvControlToken);
8164 std::pair<SDValue, SDValue>
Result =
8167 if (
Result.first.getNode())
8171 case Intrinsic::ptrmask: {
8187 auto HighOnes =
DAG.getNode(
8188 ISD::SHL, sdl, PtrVT,
DAG.getAllOnesConstant(sdl, PtrVT),
8189 DAG.getShiftAmountConstant(
Mask.getValueType().getFixedSizeInBits(),
8192 DAG.getZExtOrTrunc(Mask, sdl, PtrVT), HighOnes);
8193 }
else if (
Mask.getValueType() != PtrVT)
8194 Mask =
DAG.getPtrExtOrTrunc(Mask, sdl, PtrVT);
8200 case Intrinsic::threadlocal_address: {
8204 case Intrinsic::get_active_lane_mask: {
8208 EVT ElementVT =
Index.getValueType();
8219 SDValue VectorIndex =
DAG.getSplat(VecTy, sdl, Index);
8220 SDValue VectorTripCount =
DAG.getSplat(VecTy, sdl, TripCount);
8221 SDValue VectorStep =
DAG.getStepVector(sdl, VecTy);
8224 SDValue SetCC =
DAG.getSetCC(sdl, CCVT, VectorInduction,
8229 case Intrinsic::experimental_get_vector_length: {
8231 "Expected positive VF");
8236 EVT CountVT =
Count.getValueType();
8239 visitTargetIntrinsic(
I, Intrinsic);
8248 if (CountVT.
bitsLT(VT)) {
8253 SDValue MaxEVL =
DAG.getElementCount(sdl, CountVT,
8263 case Intrinsic::vector_partial_reduce_add: {
8271 case Intrinsic::vector_partial_reduce_fadd: {
8279 case Intrinsic::experimental_cttz_elts: {
8282 EVT OpVT =
Op.getValueType();
8285 visitTargetIntrinsic(
I, Intrinsic);
8301 ConstantRange VScaleRange(1,
true);
8330 case Intrinsic::vector_insert: {
8338 if (
Index.getValueType() != VectorIdxTy)
8339 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8346 case Intrinsic::vector_extract: {
8354 if (
Index.getValueType() != VectorIdxTy)
8355 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8361 case Intrinsic::experimental_vector_match: {
8367 EVT ResVT =
Mask.getValueType();
8373 visitTargetIntrinsic(
I, Intrinsic);
8377 SDValue Ret =
DAG.getConstant(0, sdl, ResVT);
8379 for (
unsigned i = 0; i < SearchSize; ++i) {
8382 DAG.getVectorIdxConstant(i, sdl));
8385 Ret =
DAG.getNode(
ISD::OR, sdl, ResVT, Ret, Cmp);
8391 case Intrinsic::vector_reverse:
8392 visitVectorReverse(
I);
8394 case Intrinsic::vector_splice_left:
8395 case Intrinsic::vector_splice_right:
8396 visitVectorSplice(
I);
8398 case Intrinsic::callbr_landingpad:
8399 visitCallBrLandingPad(
I);
8401 case Intrinsic::vector_interleave2:
8402 visitVectorInterleave(
I, 2);
8404 case Intrinsic::vector_interleave3:
8405 visitVectorInterleave(
I, 3);
8407 case Intrinsic::vector_interleave4:
8408 visitVectorInterleave(
I, 4);
8410 case Intrinsic::vector_interleave5:
8411 visitVectorInterleave(
I, 5);
8413 case Intrinsic::vector_interleave6:
8414 visitVectorInterleave(
I, 6);
8416 case Intrinsic::vector_interleave7:
8417 visitVectorInterleave(
I, 7);
8419 case Intrinsic::vector_interleave8:
8420 visitVectorInterleave(
I, 8);
8422 case Intrinsic::vector_deinterleave2:
8423 visitVectorDeinterleave(
I, 2);
8425 case Intrinsic::vector_deinterleave3:
8426 visitVectorDeinterleave(
I, 3);
8428 case Intrinsic::vector_deinterleave4:
8429 visitVectorDeinterleave(
I, 4);
8431 case Intrinsic::vector_deinterleave5:
8432 visitVectorDeinterleave(
I, 5);
8434 case Intrinsic::vector_deinterleave6:
8435 visitVectorDeinterleave(
I, 6);
8437 case Intrinsic::vector_deinterleave7:
8438 visitVectorDeinterleave(
I, 7);
8440 case Intrinsic::vector_deinterleave8:
8441 visitVectorDeinterleave(
I, 8);
8443 case Intrinsic::experimental_vector_compress:
8445 getValue(
I.getArgOperand(0)).getValueType(),
8450 case Intrinsic::experimental_convergence_anchor:
8451 case Intrinsic::experimental_convergence_entry:
8452 case Intrinsic::experimental_convergence_loop:
8453 visitConvergenceControl(
I, Intrinsic);
8455 case Intrinsic::experimental_vector_histogram_add: {
8456 visitVectorHistogram(
I, Intrinsic);
8459 case Intrinsic::experimental_vector_extract_last_active: {
8460 visitVectorExtractLastActive(
I, Intrinsic);
8463 case Intrinsic::loop_dependence_war_mask:
8468 DAG.getConstant(0, sdl, MVT::i64)));
8470 case Intrinsic::loop_dependence_raw_mask:
8475 DAG.getConstant(0, sdl, MVT::i64)));
8480void SelectionDAGBuilder::pushFPOpOutChain(
SDValue Result,
8496 PendingConstrainedFP.push_back(OutChain);
8499 PendingConstrainedFPStrict.push_back(OutChain);
8504void SelectionDAGBuilder::visitConstrainedFPIntrinsic(
8518 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8520 SDVTList VTs =
DAG.getVTList(VT, MVT::Other);
8524 Flags.setNoFPExcept(
true);
8527 Flags.copyFMF(*FPOp);
8532#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
8533 case Intrinsic::INTRINSIC: \
8534 Opcode = ISD::STRICT_##DAGN; \
8536#include "llvm/IR/ConstrainedOps.def"
8537 case Intrinsic::experimental_constrained_fmuladd: {
8544 pushFPOpOutChain(
Mul, EB);
8567 if (
DAG.isKnownNeverNaN(Opers[1]) &&
DAG.isKnownNeverNaN(Opers[2]))
8575 pushFPOpOutChain(Result, EB);
8582 std::optional<unsigned> ResOPC;
8584 case Intrinsic::vp_ctlz: {
8586 ResOPC = IsZeroUndef ? ISD::VP_CTLZ_ZERO_UNDEF : ISD::VP_CTLZ;
8589 case Intrinsic::vp_cttz: {
8591 ResOPC = IsZeroUndef ? ISD::VP_CTTZ_ZERO_UNDEF : ISD::VP_CTTZ;
8594 case Intrinsic::vp_cttz_elts: {
8596 ResOPC = IsZeroPoison ? ISD::VP_CTTZ_ELTS_ZERO_UNDEF : ISD::VP_CTTZ_ELTS;
8599#define HELPER_MAP_VPID_TO_VPSD(VPID, VPSD) \
8600 case Intrinsic::VPID: \
8601 ResOPC = ISD::VPSD; \
8603#include "llvm/IR/VPIntrinsics.def"
8608 "Inconsistency: no SDNode available for this VPIntrinsic!");
8610 if (*ResOPC == ISD::VP_REDUCE_SEQ_FADD ||
8611 *ResOPC == ISD::VP_REDUCE_SEQ_FMUL) {
8613 return *ResOPC == ISD::VP_REDUCE_SEQ_FADD ? ISD::VP_REDUCE_FADD
8614 : ISD::VP_REDUCE_FMUL;
8620void SelectionDAGBuilder::visitVPLoad(
8632 Alignment =
DAG.getEVTAlign(VT);
8635 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8636 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8639 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8640 MachinePointerInfo(PtrOperand), MMOFlags,
8642 LD =
DAG.getLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8649void SelectionDAGBuilder::visitVPLoadFF(
8652 assert(OpValues.
size() == 3 &&
"Unexpected number of operands");
8662 Alignment =
DAG.getEVTAlign(VT);
8665 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8666 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8669 LD =
DAG.getLoadFFVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8674 setValue(&VPIntrin,
DAG.getMergeValues({LD.getValue(0), Trunc},
DL));
8677void SelectionDAGBuilder::visitVPGather(
8681 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8693 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8695 *Alignment, AAInfo, Ranges);
8705 EVT IdxVT =
Index.getValueType();
8711 LD =
DAG.getGatherVP(
8712 DAG.getVTList(VT, MVT::Other), VT,
DL,
8713 {DAG.getRoot(), Base, Index, Scale, OpValues[1], OpValues[2]}, MMO,
8719void SelectionDAGBuilder::visitVPStore(
8723 EVT VT = OpValues[0].getValueType();
8728 Alignment =
DAG.getEVTAlign(VT);
8731 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8734 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8735 MachinePointerInfo(PtrOperand), MMOFlags,
8744void SelectionDAGBuilder::visitVPScatter(
8747 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8749 EVT VT = OpValues[0].getValueType();
8759 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8761 *Alignment, AAInfo);
8771 EVT IdxVT =
Index.getValueType();
8777 ST =
DAG.getScatterVP(
DAG.getVTList(MVT::Other), VT,
DL,
8778 {getMemoryRoot(), OpValues[0], Base, Index, Scale,
8779 OpValues[2], OpValues[3]},
8785void SelectionDAGBuilder::visitVPStridedLoad(
8797 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8799 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8802 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8804 *Alignment, AAInfo, Ranges);
8806 SDValue LD =
DAG.getStridedLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1],
8807 OpValues[2], OpValues[3], MMO,
8815void SelectionDAGBuilder::visitVPStridedStore(
8819 EVT VT = OpValues[0].getValueType();
8825 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8828 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8830 *Alignment, AAInfo);
8834 DAG.getUNDEF(OpValues[1].getValueType()), OpValues[2], OpValues[3],
8842void SelectionDAGBuilder::visitVPCmp(
const VPCmpIntrinsic &VPIntrin) {
8843 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8858 "Unexpected target EVL type");
8861 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
8863 if (
DAG.isKnownNeverNaN(Op1) &&
DAG.isKnownNeverNaN(Op2))
8866 DAG.getSetCCVP(
DL, DestVT, Op1, Op2, Condition, MaskOp, EVL));
8869void SelectionDAGBuilder::visitVectorPredicationIntrinsic(
8877 return visitVPCmp(*CmpI);
8880 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8882 SDVTList VTs =
DAG.getVTList(ValueVTs);
8888 "Unexpected target EVL type");
8892 for (
unsigned I = 0;
I < VPIntrin.
arg_size(); ++
I) {
8894 if (
I == EVLParamPos)
8901 SDNodeFlags SDFlags;
8909 visitVPLoad(VPIntrin, ValueVTs[0], OpValues);
8911 case ISD::VP_LOAD_FF:
8912 visitVPLoadFF(VPIntrin, ValueVTs[0], ValueVTs[1], OpValues);
8914 case ISD::VP_GATHER:
8915 visitVPGather(VPIntrin, ValueVTs[0], OpValues);
8917 case ISD::EXPERIMENTAL_VP_STRIDED_LOAD:
8918 visitVPStridedLoad(VPIntrin, ValueVTs[0], OpValues);
8921 visitVPStore(VPIntrin, OpValues);
8923 case ISD::VP_SCATTER:
8924 visitVPScatter(VPIntrin, OpValues);
8926 case ISD::EXPERIMENTAL_VP_STRIDED_STORE:
8927 visitVPStridedStore(VPIntrin, OpValues);
8929 case ISD::VP_FMULADD: {
8930 assert(OpValues.
size() == 5 &&
"Unexpected number of operands");
8931 SDNodeFlags SDFlags;
8936 setValue(&VPIntrin,
DAG.getNode(ISD::VP_FMA,
DL, VTs, OpValues, SDFlags));
8939 ISD::VP_FMUL,
DL, VTs,
8940 {OpValues[0], OpValues[1], OpValues[3], OpValues[4]}, SDFlags);
8942 DAG.getNode(ISD::VP_FADD,
DL, VTs,
8943 {
Mul, OpValues[2], OpValues[3], OpValues[4]}, SDFlags);
8948 case ISD::VP_IS_FPCLASS: {
8949 const DataLayout DLayout =
DAG.getDataLayout();
8951 auto Constant = OpValues[1]->getAsZExtVal();
8954 {OpValues[0],
Check, OpValues[2], OpValues[3]});
8958 case ISD::VP_INTTOPTR: {
8969 case ISD::VP_PTRTOINT: {
8971 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
8984 case ISD::VP_CTLZ_ZERO_UNDEF:
8986 case ISD::VP_CTTZ_ZERO_UNDEF:
8987 case ISD::VP_CTTZ_ELTS_ZERO_UNDEF:
8988 case ISD::VP_CTTZ_ELTS: {
8990 DAG.getNode(Opcode,
DL, VTs, {OpValues[0], OpValues[2], OpValues[3]});
9000 MachineFunction &MF =
DAG.getMachineFunction();
9008 unsigned CallSiteIndex =
FuncInfo.getCurrentCallSite();
9009 if (CallSiteIndex) {
9023 assert(BeginLabel &&
"BeginLabel should've been set");
9025 MachineFunction &MF =
DAG.getMachineFunction();
9037 assert(
II &&
"II should've been set");
9048std::pair<SDValue, SDValue>
9062 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
9065 "Non-null chain expected with non-tail call!");
9066 assert((Result.second.getNode() || !Result.first.getNode()) &&
9067 "Null value expected with tail call!");
9069 if (!Result.second.getNode()) {
9076 PendingExports.clear();
9078 DAG.setRoot(Result.second);
9096 if (!isMustTailCall &&
9097 Caller->getFnAttribute(
"disable-tail-calls").getValueAsBool())
9103 if (
DAG.getTargetLoweringInfo().supportSwiftError() &&
9104 Caller->getAttributes().hasAttrSomewhere(Attribute::SwiftError))
9113 bool isTailCall,
bool isMustTailCall,
9116 auto &
DL =
DAG.getDataLayout();
9123 const Value *SwiftErrorVal =
nullptr;
9130 const Value *V = *
I;
9133 if (V->getType()->isEmptyTy())
9138 Entry.setAttributes(&CB,
I - CB.
arg_begin());
9150 Args.push_back(Entry);
9161 Value *V = Bundle->Inputs[0];
9163 Entry.IsCFGuardTarget =
true;
9164 Args.push_back(Entry);
9177 "Target doesn't support calls with kcfi operand bundles.");
9185 auto *Token = Bundle->Inputs[0].get();
9186 ConvControlToken =
getValue(Token);
9197 .
setCallee(RetTy, FTy, Callee, std::move(Args), CB)
9210 "This target doesn't support calls with ptrauth operand bundles.");
9214 std::pair<SDValue, SDValue> Result =
lowerInvokable(CLI, EHPadBB);
9216 if (Result.first.getNode()) {
9231 DAG.setRoot(CopyNode);
9247 LoadTy, Builder.DAG.getDataLayout()))
9248 return Builder.getValue(LoadCst);
9254 bool ConstantMemory =
false;
9257 if (Builder.BatchAA && Builder.BatchAA->pointsToConstantMemory(PtrVal)) {
9258 Root = Builder.DAG.getEntryNode();
9259 ConstantMemory =
true;
9262 Root = Builder.DAG.getRoot();
9267 Builder.DAG.getLoad(LoadVT, Builder.getCurSDLoc(), Root, Ptr,
9270 if (!ConstantMemory)
9271 Builder.PendingLoads.push_back(LoadVal.
getValue(1));
9277void SelectionDAGBuilder::processIntegerCallValue(
const Instruction &
I,
9280 EVT VT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9291bool SelectionDAGBuilder::visitMemCmpBCmpCall(
const CallInst &
I) {
9292 const Value *
LHS =
I.getArgOperand(0), *
RHS =
I.getArgOperand(1);
9293 const Value *
Size =
I.getArgOperand(2);
9296 EVT CallVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9302 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9306 if (Res.first.getNode()) {
9307 processIntegerCallValue(
I, Res.first,
true);
9321 auto hasFastLoadsAndCompare = [&](
unsigned NumBits) {
9322 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9344 switch (NumBitsToCompare) {
9356 LoadVT = hasFastLoadsAndCompare(NumBitsToCompare);
9369 LoadL =
DAG.getBitcast(CmpVT, LoadL);
9370 LoadR =
DAG.getBitcast(CmpVT, LoadR);
9374 processIntegerCallValue(
I, Cmp,
false);
9383bool SelectionDAGBuilder::visitMemChrCall(
const CallInst &
I) {
9384 const Value *Src =
I.getArgOperand(0);
9385 const Value *
Char =
I.getArgOperand(1);
9386 const Value *
Length =
I.getArgOperand(2);
9388 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9389 std::pair<SDValue, SDValue> Res =
9392 MachinePointerInfo(Src));
9393 if (Res.first.getNode()) {
9407bool SelectionDAGBuilder::visitMemCCpyCall(
const CallInst &
I) {
9408 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9415 processIntegerCallValue(
I, Res.first,
true);
9427bool SelectionDAGBuilder::visitMemPCpyCall(
const CallInst &
I) {
9432 Align DstAlign =
DAG.InferPtrAlign(Dst).valueOrOne();
9433 Align SrcAlign =
DAG.InferPtrAlign(Src).valueOrOne();
9435 Align Alignment = std::min(DstAlign, SrcAlign);
9444 Root, sdl, Dst, Src,
Size, Alignment,
false,
false,
nullptr,
9445 std::nullopt, MachinePointerInfo(
I.getArgOperand(0)),
9446 MachinePointerInfo(
I.getArgOperand(1)),
I.getAAMetadata());
9448 "** memcpy should not be lowered as TailCall in mempcpy context **");
9452 Size =
DAG.getSExtOrTrunc(
Size, sdl, Dst.getValueType());
9465bool SelectionDAGBuilder::visitStrCpyCall(
const CallInst &
I,
bool isStpcpy) {
9466 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9468 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9471 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), isStpcpy, &
I);
9472 if (Res.first.getNode()) {
9474 DAG.setRoot(Res.second);
9486bool SelectionDAGBuilder::visitStrCmpCall(
const CallInst &
I) {
9487 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9489 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9492 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), &
I);
9493 if (Res.first.getNode()) {
9494 processIntegerCallValue(
I, Res.first,
true);
9507bool SelectionDAGBuilder::visitStrLenCall(
const CallInst &
I) {
9508 const Value *Arg0 =
I.getArgOperand(0);
9510 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9513 if (Res.first.getNode()) {
9514 processIntegerCallValue(
I, Res.first,
false);
9527bool SelectionDAGBuilder::visitStrNLenCall(
const CallInst &
I) {
9528 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9530 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9531 std::pair<SDValue, SDValue> Res =
9534 MachinePointerInfo(Arg0));
9535 if (Res.first.getNode()) {
9536 processIntegerCallValue(
I, Res.first,
false);
9549bool SelectionDAGBuilder::visitStrstrCall(
const CallInst &
I) {
9550 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9551 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9555 processIntegerCallValue(
I, Res.first,
false);
9567bool SelectionDAGBuilder::visitUnaryFloatCall(
const CallInst &
I,
9572 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9589bool SelectionDAGBuilder::visitBinaryFloatCall(
const CallInst &
I,
9594 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9607void SelectionDAGBuilder::visitCall(
const CallInst &
I) {
9609 if (
I.isInlineAsm()) {
9616 if (Function *
F =
I.getCalledFunction()) {
9617 if (
F->isDeclaration()) {
9619 if (
unsigned IID =
F->getIntrinsicID()) {
9620 visitIntrinsicCall(
I, IID);
9631 if (!
I.isNoBuiltin() && !
F->hasLocalLinkage() &&
F->hasName() &&
9632 LibInfo->getLibFunc(*
F, Func) &&
LibInfo->hasOptimizedCodeGen(Func)) {
9636 if (visitMemCmpBCmpCall(
I))
9639 case LibFunc_copysign:
9640 case LibFunc_copysignf:
9641 case LibFunc_copysignl:
9644 if (
I.onlyReadsMemory()) {
9689 case LibFunc_atan2f:
9690 case LibFunc_atan2l:
9715 case LibFunc_sqrt_finite:
9716 case LibFunc_sqrtf_finite:
9717 case LibFunc_sqrtl_finite:
9734 case LibFunc_exp10f:
9735 case LibFunc_exp10l:
9740 case LibFunc_ldexpf:
9741 case LibFunc_ldexpl:
9745 case LibFunc_strstr:
9746 if (visitStrstrCall(
I))
9749 case LibFunc_memcmp:
9750 if (visitMemCmpBCmpCall(
I))
9753 case LibFunc_memccpy:
9754 if (visitMemCCpyCall(
I))
9757 case LibFunc_mempcpy:
9758 if (visitMemPCpyCall(
I))
9761 case LibFunc_memchr:
9762 if (visitMemChrCall(
I))
9765 case LibFunc_strcpy:
9766 if (visitStrCpyCall(
I,
false))
9769 case LibFunc_stpcpy:
9770 if (visitStrCpyCall(
I,
true))
9773 case LibFunc_strcmp:
9774 if (visitStrCmpCall(
I))
9777 case LibFunc_strlen:
9778 if (visitStrLenCall(
I))
9781 case LibFunc_strnlen:
9782 if (visitStrNLenCall(
I))
9806 if (
I.hasDeoptState())
9823 const Value *Discriminator = PAB->Inputs[1];
9825 assert(
Key->getType()->isIntegerTy(32) &&
"Invalid ptrauth key");
9826 assert(Discriminator->getType()->isIntegerTy(64) &&
9827 "Invalid ptrauth discriminator");
9832 if (CalleeCPA->isKnownCompatibleWith(
Key, Discriminator,
9833 DAG.getDataLayout()))
9873 for (
const auto &Code : Codes)
9888 SDISelAsmOperandInfo &MatchingOpInfo,
9890 if (OpInfo.ConstraintVT == MatchingOpInfo.ConstraintVT)
9896 std::pair<unsigned, const TargetRegisterClass *> MatchRC =
9898 OpInfo.ConstraintVT);
9899 std::pair<unsigned, const TargetRegisterClass *> InputRC =
9901 MatchingOpInfo.ConstraintVT);
9902 const bool OutOpIsIntOrFP =
9903 OpInfo.ConstraintVT.isInteger() || OpInfo.ConstraintVT.isFloatingPoint();
9904 const bool InOpIsIntOrFP = MatchingOpInfo.ConstraintVT.isInteger() ||
9905 MatchingOpInfo.ConstraintVT.isFloatingPoint();
9906 if ((OutOpIsIntOrFP != InOpIsIntOrFP) || (MatchRC.second != InputRC.second)) {
9909 " with a matching output constraint of"
9910 " incompatible type!");
9912 MatchingOpInfo.ConstraintVT = OpInfo.ConstraintVT;
9919 SDISelAsmOperandInfo &OpInfo,
9932 const Value *OpVal = OpInfo.CallOperandVal;
9950 DL.getPrefTypeAlign(Ty),
false,
9953 Chain = DAG.
getTruncStore(Chain, Location, OpInfo.CallOperand, StackSlot,
9956 OpInfo.CallOperand = StackSlot;
9969static std::optional<unsigned>
9971 SDISelAsmOperandInfo &OpInfo,
9972 SDISelAsmOperandInfo &RefOpInfo) {
9983 return std::nullopt;
9987 unsigned AssignedReg;
9990 &
TRI, RefOpInfo.ConstraintCode, RefOpInfo.ConstraintVT);
9993 return std::nullopt;
9998 const MVT RegVT = *
TRI.legalclasstypes_begin(*RC);
10000 if (OpInfo.ConstraintVT != MVT::Other && RegVT != MVT::Untyped) {
10009 !
TRI.isTypeLegalForClass(*RC, OpInfo.ConstraintVT)) {
10014 if (RegVT.
getSizeInBits() == OpInfo.ConstraintVT.getSizeInBits()) {
10019 OpInfo.CallOperand =
10021 OpInfo.ConstraintVT = RegVT;
10025 }
else if (RegVT.
isInteger() && OpInfo.ConstraintVT.isFloatingPoint()) {
10028 OpInfo.CallOperand =
10030 OpInfo.ConstraintVT = VT;
10037 if (OpInfo.isMatchingInputConstraint())
10038 return std::nullopt;
10040 EVT ValueVT = OpInfo.ConstraintVT;
10041 if (OpInfo.ConstraintVT == MVT::Other)
10045 unsigned NumRegs = 1;
10046 if (OpInfo.ConstraintVT != MVT::Other)
10061 I = std::find(
I, RC->
end(), AssignedReg);
10062 if (
I == RC->
end()) {
10065 return {AssignedReg};
10069 for (; NumRegs; --NumRegs, ++
I) {
10070 assert(
I != RC->
end() &&
"Ran out of registers to allocate!");
10075 OpInfo.AssignedRegs =
RegsForValue(Regs, RegVT, ValueVT);
10076 return std::nullopt;
10081 const std::vector<SDValue> &AsmNodeOperands) {
10084 for (; OperandNo; --OperandNo) {
10086 unsigned OpFlag = AsmNodeOperands[CurOp]->getAsZExtVal();
10089 (
F.isRegDefKind() ||
F.isRegDefEarlyClobberKind() ||
F.isMemKind()) &&
10090 "Skipped past definitions?");
10091 CurOp +=
F.getNumOperandRegisters() + 1;
10099 unsigned Flags = 0;
10102 explicit ExtraFlags(
const CallBase &
Call) {
10104 if (
IA->hasSideEffects())
10106 if (
IA->isAlignStack())
10108 if (
IA->canThrow())
10115 void update(
const TargetLowering::AsmOperandInfo &OpInfo) {
10131 unsigned get()
const {
return Flags; }
10154void SelectionDAGBuilder::visitInlineAsm(
const CallBase &
Call,
10161 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10163 DAG.getDataLayout(),
DAG.getSubtarget().getRegisterInfo(),
Call);
10167 bool HasSideEffect =
IA->hasSideEffects();
10168 ExtraFlags ExtraInfo(
Call);
10170 for (
auto &
T : TargetConstraints) {
10171 ConstraintOperands.
push_back(SDISelAsmOperandInfo(
T));
10172 SDISelAsmOperandInfo &OpInfo = ConstraintOperands.
back();
10174 if (OpInfo.CallOperandVal)
10175 OpInfo.CallOperand =
getValue(OpInfo.CallOperandVal);
10177 if (!HasSideEffect)
10178 HasSideEffect = OpInfo.hasMemory(TLI);
10190 return emitInlineAsmError(
Call,
"constraint '" + Twine(
T.ConstraintCode) +
10191 "' expects an integer constant "
10194 ExtraInfo.update(
T);
10202 if (EmitEHLabels) {
10203 assert(EHPadBB &&
"InvokeInst must have an EHPadBB");
10207 if (IsCallBr || EmitEHLabels) {
10215 if (EmitEHLabels) {
10216 Chain = lowerStartEH(Chain, EHPadBB, BeginLabel);
10221 IA->collectAsmStrs(AsmStrs);
10224 for (SDISelAsmOperandInfo &OpInfo : ConstraintOperands) {
10232 if (OpInfo.hasMatchingInput()) {
10233 SDISelAsmOperandInfo &Input = ConstraintOperands[OpInfo.MatchingInput];
10264 if (OpInfo.isIndirect &&
isFunction(OpInfo.CallOperand) &&
10267 OpInfo.isIndirect =
false;
10274 !OpInfo.isIndirect) {
10275 assert((OpInfo.isMultipleAlternative ||
10277 "Can only indirectify direct input operands!");
10283 OpInfo.CallOperandVal =
nullptr;
10286 OpInfo.isIndirect =
true;
10292 std::vector<SDValue> AsmNodeOperands;
10293 AsmNodeOperands.push_back(
SDValue());
10294 AsmNodeOperands.push_back(
DAG.getTargetExternalSymbol(
10301 AsmNodeOperands.push_back(
DAG.getMDNode(SrcLoc));
10305 AsmNodeOperands.push_back(
DAG.getTargetConstant(
10310 for (SDISelAsmOperandInfo &OpInfo : ConstraintOperands) {
10312 SDISelAsmOperandInfo &RefOpInfo =
10313 OpInfo.isMatchingInputConstraint()
10314 ? ConstraintOperands[OpInfo.getMatchedOperand()]
10316 const auto RegError =
10319 const MachineFunction &MF =
DAG.getMachineFunction();
10321 const char *
RegName =
TRI.getName(*RegError);
10322 emitInlineAsmError(
Call,
"register '" + Twine(
RegName) +
10323 "' allocated for constraint '" +
10324 Twine(OpInfo.ConstraintCode) +
10325 "' does not match required type");
10329 auto DetectWriteToReservedRegister = [&]() {
10330 const MachineFunction &MF =
DAG.getMachineFunction();
10335 emitInlineAsmError(
Call,
"write to reserved register '" +
10344 !OpInfo.isMatchingInputConstraint())) &&
10345 "Only address as input operand is allowed.");
10347 switch (OpInfo.Type) {
10353 "Failed to convert memory constraint code to constraint id.");
10357 OpFlags.setMemConstraint(ConstraintID);
10358 AsmNodeOperands.push_back(
DAG.getTargetConstant(OpFlags,
getCurSDLoc(),
10360 AsmNodeOperands.push_back(OpInfo.CallOperand);
10365 if (OpInfo.AssignedRegs.
Regs.empty()) {
10366 emitInlineAsmError(
10367 Call,
"couldn't allocate output register for constraint '" +
10368 Twine(OpInfo.ConstraintCode) +
"'");
10372 if (DetectWriteToReservedRegister())
10386 SDValue InOperandVal = OpInfo.CallOperand;
10388 if (OpInfo.isMatchingInputConstraint()) {
10393 InlineAsm::Flag
Flag(AsmNodeOperands[CurOp]->getAsZExtVal());
10394 if (
Flag.isRegDefKind() ||
Flag.isRegDefEarlyClobberKind()) {
10395 if (OpInfo.isIndirect) {
10397 emitInlineAsmError(
Call,
"inline asm not supported yet: "
10398 "don't know how to handle tied "
10399 "indirect register inputs");
10404 MachineFunction &MF =
DAG.getMachineFunction();
10409 MVT RegVT =
R->getSimpleValueType(0);
10410 const TargetRegisterClass *RC =
10413 :
TRI.getMinimalPhysRegClass(TiedReg);
10414 for (
unsigned i = 0, e =
Flag.getNumOperandRegisters(); i != e; ++i)
10417 RegsForValue MatchedRegs(Regs, RegVT, InOperandVal.
getValueType());
10421 MatchedRegs.getCopyToRegs(InOperandVal,
DAG, dl, Chain, &Glue, &
Call);
10423 OpInfo.getMatchedOperand(), dl,
DAG,
10428 assert(
Flag.isMemKind() &&
"Unknown matching constraint!");
10429 assert(
Flag.getNumOperandRegisters() == 1 &&
10430 "Unexpected number of operands");
10433 Flag.clearMemConstraint();
10434 Flag.setMatchingOp(OpInfo.getMatchedOperand());
10435 AsmNodeOperands.push_back(
DAG.getTargetConstant(
10437 AsmNodeOperands.push_back(AsmNodeOperands[CurOp+1]);
10448 std::vector<SDValue>
Ops;
10454 emitInlineAsmError(
Call,
"value out of range for constraint '" +
10455 Twine(OpInfo.ConstraintCode) +
"'");
10459 emitInlineAsmError(
Call,
10460 "invalid operand for inline asm constraint '" +
10461 Twine(OpInfo.ConstraintCode) +
"'");
10467 AsmNodeOperands.push_back(
DAG.getTargetConstant(
10474 assert((OpInfo.isIndirect ||
10476 "Operand must be indirect to be a mem!");
10479 "Memory operands expect pointer values");
10484 "Failed to convert memory constraint code to constraint id.");
10488 ResOpType.setMemConstraint(ConstraintID);
10489 AsmNodeOperands.push_back(
DAG.getTargetConstant(ResOpType,
10492 AsmNodeOperands.push_back(InOperandVal);
10500 "Failed to convert memory constraint code to constraint id.");
10504 SDValue AsmOp = InOperandVal;
10508 AsmOp =
DAG.getTargetGlobalAddress(GA->getGlobal(),
getCurSDLoc(),
10514 ResOpType.setMemConstraint(ConstraintID);
10516 AsmNodeOperands.push_back(
10519 AsmNodeOperands.push_back(AsmOp);
10525 emitInlineAsmError(
Call,
"unknown asm constraint '" +
10526 Twine(OpInfo.ConstraintCode) +
"'");
10531 if (OpInfo.isIndirect) {
10532 emitInlineAsmError(
10533 Call,
"Don't know how to handle indirect register inputs yet "
10534 "for constraint '" +
10535 Twine(OpInfo.ConstraintCode) +
"'");
10540 if (OpInfo.AssignedRegs.
Regs.empty()) {
10541 emitInlineAsmError(
Call,
10542 "couldn't allocate input reg for constraint '" +
10543 Twine(OpInfo.ConstraintCode) +
"'");
10547 if (DetectWriteToReservedRegister())
10556 0, dl,
DAG, AsmNodeOperands);
10562 if (!OpInfo.AssignedRegs.
Regs.empty())
10572 if (Glue.
getNode()) AsmNodeOperands.push_back(Glue);
10576 DAG.getVTList(MVT::Other, MVT::Glue), AsmNodeOperands);
10588 ResultTypes = StructResult->elements();
10589 else if (!CallResultType->
isVoidTy())
10590 ResultTypes =
ArrayRef(CallResultType);
10592 auto CurResultType = ResultTypes.
begin();
10593 auto handleRegAssign = [&](
SDValue V) {
10594 assert(CurResultType != ResultTypes.
end() &&
"Unexpected value");
10595 assert((*CurResultType)->isSized() &&
"Unexpected unsized type");
10596 EVT ResultVT = TLI.
getValueType(
DAG.getDataLayout(), *CurResultType);
10608 if (ResultVT !=
V.getValueType() &&
10611 else if (ResultVT !=
V.getValueType() && ResultVT.
isInteger() &&
10612 V.getValueType().isInteger()) {
10618 assert(ResultVT ==
V.getValueType() &&
"Asm result value mismatch!");
10624 for (SDISelAsmOperandInfo &OpInfo : ConstraintOperands) {
10628 if (OpInfo.AssignedRegs.
Regs.empty())
10631 switch (OpInfo.ConstraintType) {
10635 Chain, &Glue, &
Call);
10647 assert(
false &&
"Unexpected unknown constraint");
10651 if (OpInfo.isIndirect) {
10652 const Value *Ptr = OpInfo.CallOperandVal;
10653 assert(Ptr &&
"Expected value CallOperandVal for indirect asm operand");
10655 MachinePointerInfo(Ptr));
10662 handleRegAssign(V);
10664 handleRegAssign(Val);
10670 if (!ResultValues.
empty()) {
10671 assert(CurResultType == ResultTypes.
end() &&
10672 "Mismatch in number of ResultTypes");
10674 "Mismatch in number of output operands in asm result");
10677 DAG.getVTList(ResultVTs), ResultValues);
10682 if (!OutChains.
empty())
10685 if (EmitEHLabels) {
10690 if (ResultValues.
empty() || HasSideEffect || !OutChains.
empty() || IsCallBr ||
10692 DAG.setRoot(Chain);
10695void SelectionDAGBuilder::emitInlineAsmError(
const CallBase &
Call,
10696 const Twine &Message) {
10697 LLVMContext &Ctx = *
DAG.getContext();
10701 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10705 if (ValueVTs.
empty())
10709 for (
const EVT &VT : ValueVTs)
10710 Ops.push_back(
DAG.getUNDEF(VT));
10715void SelectionDAGBuilder::visitVAStart(
const CallInst &
I) {
10719 DAG.getSrcValue(
I.getArgOperand(0))));
10722void SelectionDAGBuilder::visitVAArg(
const VAArgInst &
I) {
10723 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10724 const DataLayout &
DL =
DAG.getDataLayout();
10728 DL.getABITypeAlign(
I.getType()).value());
10729 DAG.setRoot(
V.getValue(1));
10731 if (
I.getType()->isPointerTy())
10732 V =
DAG.getPtrExtOrTrunc(
10737void SelectionDAGBuilder::visitVAEnd(
const CallInst &
I) {
10741 DAG.getSrcValue(
I.getArgOperand(0))));
10744void SelectionDAGBuilder::visitVACopy(
const CallInst &
I) {
10749 DAG.getSrcValue(
I.getArgOperand(0)),
10750 DAG.getSrcValue(
I.getArgOperand(1))));
10756 std::optional<ConstantRange> CR =
getRange(
I);
10758 if (!CR || CR->isFullSet() || CR->isEmptySet() || CR->isUpperWrapped())
10761 APInt Lo = CR->getUnsignedMin();
10762 if (!
Lo.isMinValue())
10765 APInt Hi = CR->getUnsignedMax();
10766 unsigned Bits = std::max(
Hi.getActiveBits(),
10774 DAG.getValueType(SmallVT));
10775 unsigned NumVals =
Op.getNode()->getNumValues();
10781 Ops.push_back(ZExt);
10782 for (
unsigned I = 1;
I != NumVals; ++
I)
10783 Ops.push_back(
Op.getValue(
I));
10785 return DAG.getMergeValues(
Ops,
SL);
10795 SDValue TestConst =
DAG.getTargetConstant(Classes,
SDLoc(), MVT::i32);
10803 for (
unsigned I = 0, E =
Ops.size();
I != E; ++
I) {
10806 MergeOp, TestConst);
10809 return DAG.getMergeValues(
Ops,
SL);
10820 unsigned ArgIdx,
unsigned NumArgs,
SDValue Callee,
Type *ReturnTy,
10823 Args.reserve(NumArgs);
10827 for (
unsigned ArgI = ArgIdx, ArgE = ArgIdx + NumArgs;
10828 ArgI != ArgE; ++ArgI) {
10829 const Value *V =
Call->getOperand(ArgI);
10831 assert(!V->getType()->isEmptyTy() &&
"Empty type passed to intrinsic.");
10834 Entry.setAttributes(
Call, ArgI);
10835 Args.push_back(Entry);
10840 .
setCallee(
Call->getCallingConv(), ReturnTy, Callee, std::move(Args),
10869 for (
unsigned I = StartIdx;
I <
Call.arg_size();
I++) {
10878 Ops.push_back(Builder.getValue(
Call.getArgOperand(
I)));
10884void SelectionDAGBuilder::visitStackmap(
const CallInst &CI) {
10910 Ops.push_back(Chain);
10911 Ops.push_back(InGlue);
10918 assert(
ID.getValueType() == MVT::i64);
10920 DAG.getTargetConstant(
ID->getAsZExtVal(),
DL,
ID.getValueType());
10921 Ops.push_back(IDConst);
10927 Ops.push_back(ShadConst);
10933 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
10937 Chain =
DAG.getCALLSEQ_END(Chain, 0, 0, InGlue,
DL);
10942 DAG.setRoot(Chain);
10945 FuncInfo.MF->getFrameInfo().setHasStackMap();
10949void SelectionDAGBuilder::visitPatchpoint(
const CallBase &CB,
10966 Callee =
DAG.getIntPtrConstant(ConstCallee->getZExtValue(), dl,
10969 Callee =
DAG.getTargetGlobalAddress(SymbolicCallee->getGlobal(),
10970 SDLoc(SymbolicCallee),
10971 SymbolicCallee->getValueType(0));
10981 "Not enough arguments provided to the patchpoint intrinsic");
10984 unsigned NumCallArgs = IsAnyRegCC ? 0 : NumArgs;
10988 TargetLowering::CallLoweringInfo CLI(
DAG);
10993 SDNode *CallEnd =
Result.second.getNode();
11002 "Expected a callseq node.");
11004 bool HasGlue =
Call->getGluedNode();
11029 Ops.push_back(Callee);
11035 NumCallRegArgs = IsAnyRegCC ? NumArgs : NumCallRegArgs;
11036 Ops.push_back(
DAG.getTargetConstant(NumCallRegArgs, dl, MVT::i32));
11039 Ops.push_back(
DAG.getTargetConstant((
unsigned)CC, dl, MVT::i32));
11044 for (
unsigned i = NumMetaOpers, e = NumMetaOpers + NumArgs; i !=
e; ++i)
11055 if (IsAnyRegCC && HasDef) {
11057 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11060 assert(ValueVTs.
size() == 1 &&
"Expected only one return value type.");
11065 NodeTys =
DAG.getVTList(ValueVTs);
11067 NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
11084 if (IsAnyRegCC && HasDef) {
11087 DAG.ReplaceAllUsesOfValuesWith(From, To, 2);
11093 FuncInfo.MF->getFrameInfo().setHasPatchPoint();
11096void SelectionDAGBuilder::visitVectorReduce(
const CallInst &
I,
11098 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11101 if (
I.arg_size() > 1)
11106 SDNodeFlags SDFlags;
11110 switch (Intrinsic) {
11111 case Intrinsic::vector_reduce_fadd:
11119 case Intrinsic::vector_reduce_fmul:
11127 case Intrinsic::vector_reduce_add:
11130 case Intrinsic::vector_reduce_mul:
11133 case Intrinsic::vector_reduce_and:
11136 case Intrinsic::vector_reduce_or:
11139 case Intrinsic::vector_reduce_xor:
11142 case Intrinsic::vector_reduce_smax:
11145 case Intrinsic::vector_reduce_smin:
11148 case Intrinsic::vector_reduce_umax:
11151 case Intrinsic::vector_reduce_umin:
11154 case Intrinsic::vector_reduce_fmax:
11157 case Intrinsic::vector_reduce_fmin:
11160 case Intrinsic::vector_reduce_fmaximum:
11163 case Intrinsic::vector_reduce_fminimum:
11177 Attrs.push_back(Attribute::SExt);
11179 Attrs.push_back(Attribute::ZExt);
11181 Attrs.push_back(Attribute::InReg);
11183 return AttributeList::get(CLI.
RetTy->
getContext(), AttributeList::ReturnIndex,
11191std::pair<SDValue, SDValue>
11205 "Only supported for non-aggregate returns");
11208 for (
Type *Ty : RetOrigTys)
11217 RetOrigTys.
swap(OldRetOrigTys);
11218 RetVTs.
swap(OldRetVTs);
11219 Offsets.swap(OldOffsets);
11221 for (
size_t i = 0, e = OldRetVTs.
size(); i != e; ++i) {
11222 EVT RetVT = OldRetVTs[i];
11226 unsigned RegisterVTByteSZ = RegisterVT.
getSizeInBits() / 8;
11227 RetOrigTys.
append(NumRegs, OldRetOrigTys[i]);
11228 RetVTs.
append(NumRegs, RegisterVT);
11229 for (
unsigned j = 0; j != NumRegs; ++j)
11242 int DemoteStackIdx = -100;
11255 ArgListEntry Entry(DemoteStackSlot, StackSlotPtrType);
11256 Entry.IsSRet =
true;
11257 Entry.Alignment = Alignment;
11269 for (
unsigned I = 0, E = RetVTs.
size();
I != E; ++
I) {
11271 if (NeedsRegBlock) {
11272 Flags.setInConsecutiveRegs();
11273 if (
I == RetVTs.
size() - 1)
11274 Flags.setInConsecutiveRegsLast();
11276 EVT VT = RetVTs[
I];
11280 for (
unsigned i = 0; i != NumRegs; ++i) {
11294 CLI.
Ins.push_back(Ret);
11303 if (Arg.IsSwiftError) {
11309 CLI.
Ins.push_back(Ret);
11317 for (
unsigned i = 0, e = Args.size(); i != e; ++i) {
11321 Type *FinalType = Args[i].Ty;
11322 if (Args[i].IsByVal)
11323 FinalType = Args[i].IndirectType;
11326 for (
unsigned Value = 0, NumValues = OrigArgTys.
size();
Value != NumValues;
11329 Type *ArgTy = OrigArgTy;
11330 if (Args[i].Ty != Args[i].OrigTy) {
11331 assert(
Value == 0 &&
"Only supported for non-aggregate arguments");
11332 ArgTy = Args[i].Ty;
11337 Args[i].Node.getResNo() +
Value);
11344 Flags.setOrigAlign(OriginalAlignment);
11349 Flags.setPointer();
11352 if (Args[i].IsZExt)
11354 if (Args[i].IsSExt)
11356 if (Args[i].IsNoExt)
11358 if (Args[i].IsInReg) {
11365 Flags.setHvaStart();
11371 if (Args[i].IsSRet)
11373 if (Args[i].IsSwiftSelf)
11374 Flags.setSwiftSelf();
11375 if (Args[i].IsSwiftAsync)
11376 Flags.setSwiftAsync();
11377 if (Args[i].IsSwiftError)
11378 Flags.setSwiftError();
11379 if (Args[i].IsCFGuardTarget)
11380 Flags.setCFGuardTarget();
11381 if (Args[i].IsByVal)
11383 if (Args[i].IsByRef)
11385 if (Args[i].IsPreallocated) {
11386 Flags.setPreallocated();
11394 if (Args[i].IsInAlloca) {
11395 Flags.setInAlloca();
11404 if (Args[i].IsByVal || Args[i].IsInAlloca || Args[i].IsPreallocated) {
11405 unsigned FrameSize =
DL.getTypeAllocSize(Args[i].IndirectType);
11406 Flags.setByValSize(FrameSize);
11409 if (
auto MA = Args[i].Alignment)
11413 }
else if (
auto MA = Args[i].Alignment) {
11416 MemAlign = OriginalAlignment;
11418 Flags.setMemAlign(MemAlign);
11419 if (Args[i].IsNest)
11422 Flags.setInConsecutiveRegs();
11425 unsigned NumParts =
11430 if (Args[i].IsSExt)
11432 else if (Args[i].IsZExt)
11437 if (Args[i].IsReturned && !
Op.getValueType().isVector() &&
11442 Args[i].Ty->getPointerAddressSpace())) &&
11443 RetVTs.
size() == NumValues &&
"unexpected use of 'returned'");
11456 CLI.
RetZExt == Args[i].IsZExt))
11457 Flags.setReturned();
11463 for (
unsigned j = 0; j != NumParts; ++j) {
11469 j * Parts[j].
getValueType().getStoreSize().getKnownMinValue());
11470 if (NumParts > 1 && j == 0)
11474 if (j == NumParts - 1)
11478 CLI.
Outs.push_back(MyFlags);
11479 CLI.
OutVals.push_back(Parts[j]);
11482 if (NeedsRegBlock &&
Value == NumValues - 1)
11483 CLI.
Outs[CLI.
Outs.size() - 1].Flags.setInConsecutiveRegsLast();
11495 "LowerCall didn't return a valid chain!");
11497 "LowerCall emitted a return value for a tail call!");
11499 "LowerCall didn't emit the correct number of values!");
11511 for (
unsigned i = 0, e = CLI.
Ins.size(); i != e; ++i) {
11512 assert(InVals[i].
getNode() &&
"LowerCall emitted a null value!");
11513 assert(
EVT(CLI.
Ins[i].VT) == InVals[i].getValueType() &&
11514 "LowerCall emitted a value with the wrong type!");
11524 unsigned NumValues = RetVTs.
size();
11525 ReturnValues.
resize(NumValues);
11532 for (
unsigned i = 0; i < NumValues; ++i) {
11539 DemoteStackIdx, Offsets[i]),
11541 ReturnValues[i] = L;
11542 Chains[i] = L.getValue(1);
11549 std::optional<ISD::NodeType> AssertOp;
11554 unsigned CurReg = 0;
11555 for (
EVT VT : RetVTs) {
11561 CLI.
DAG, CLI.
DL, &InVals[CurReg], NumRegs, RegisterVT, VT,
nullptr,
11569 if (ReturnValues.
empty())
11575 return std::make_pair(Res, CLI.
Chain);
11592 if (
N->getNumValues() == 1) {
11600 "Lowering returned the wrong number of results!");
11603 for (
unsigned I = 0, E =
N->getNumValues();
I != E; ++
I)
11617 "Copy from a reg to the same reg!");
11618 assert(!Reg.isPhysical() &&
"Is a physreg");
11624 RegsForValue RFV(V->getContext(), TLI,
DAG.getDataLayout(), Reg, V->getType(),
11629 auto PreferredExtendIt =
FuncInfo.PreferredExtendType.find(V);
11630 if (PreferredExtendIt !=
FuncInfo.PreferredExtendType.end())
11631 ExtendType = PreferredExtendIt->second;
11634 PendingExports.push_back(Chain);
11646 return A->use_empty();
11648 const BasicBlock &Entry =
A->getParent()->front();
11649 for (
const User *U :
A->users())
11658 std::pair<const AllocaInst *, const StoreInst *>>;
11670 enum StaticAllocaInfo {
Unknown, Clobbered, Elidable };
11672 unsigned NumArgs = FuncInfo->
Fn->
arg_size();
11673 StaticAllocas.
reserve(NumArgs * 2);
11675 auto GetInfoIfStaticAlloca = [&](
const Value *V) -> StaticAllocaInfo * {
11678 V = V->stripPointerCasts();
11680 if (!AI || !AI->isStaticAlloca() || !FuncInfo->
StaticAllocaMap.count(AI))
11683 return &Iter.first->second;
11700 if (
I.isDebugOrPseudoInst())
11704 for (
const Use &U :
I.operands()) {
11705 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(U))
11706 *Info = StaticAllocaInfo::Clobbered;
11712 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(
SI->getValueOperand()))
11713 *Info = StaticAllocaInfo::Clobbered;
11716 const Value *Dst =
SI->getPointerOperand()->stripPointerCasts();
11717 StaticAllocaInfo *Info = GetInfoIfStaticAlloca(Dst);
11723 if (*Info != StaticAllocaInfo::Unknown)
11731 const Value *Val =
SI->getValueOperand()->stripPointerCasts();
11734 if (!Arg || Arg->hasPassPointeeByValueCopyAttr() ||
11736 DL.getTypeStoreSize(Arg->
getType()) != *AllocaSize ||
11737 !
DL.typeSizeEqualsStoreSize(Arg->
getType()) ||
11738 ArgCopyElisionCandidates.count(Arg)) {
11739 *Info = StaticAllocaInfo::Clobbered;
11743 LLVM_DEBUG(
dbgs() <<
"Found argument copy elision candidate: " << *AI
11747 *Info = StaticAllocaInfo::Elidable;
11748 ArgCopyElisionCandidates.insert({Arg, {AI,
SI}});
11753 if (ArgCopyElisionCandidates.size() == NumArgs)
11777 auto ArgCopyIter = ArgCopyElisionCandidates.find(&Arg);
11778 assert(ArgCopyIter != ArgCopyElisionCandidates.end());
11779 const AllocaInst *AI = ArgCopyIter->second.first;
11780 int FixedIndex = FINode->getIndex();
11782 int OldIndex = AllocaIndex;
11786 dbgs() <<
" argument copy elision failed due to bad fixed stack "
11792 LLVM_DEBUG(
dbgs() <<
" argument copy elision failed: alignment of alloca "
11793 "greater than stack argument alignment ("
11794 <<
DebugStr(RequiredAlignment) <<
" vs "
11802 dbgs() <<
"Eliding argument copy from " << Arg <<
" to " << *AI <<
'\n'
11803 <<
" Replacing frame index " << OldIndex <<
" with " << FixedIndex
11809 AllocaIndex = FixedIndex;
11810 ArgCopyElisionFrameIndexMap.
insert({OldIndex, FixedIndex});
11811 for (
SDValue ArgVal : ArgVals)
11815 const StoreInst *
SI = ArgCopyIter->second.second;
11828void SelectionDAGISel::LowerArguments(
const Function &
F) {
11829 SelectionDAG &DAG =
SDB->DAG;
11830 SDLoc dl =
SDB->getCurSDLoc();
11835 if (
F.hasFnAttribute(Attribute::Naked))
11840 MVT ValueVT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
11842 ISD::ArgFlagsTy
Flags;
11844 MVT RegisterVT =
TLI->getRegisterType(*DAG.
getContext(), ValueVT);
11845 ISD::InputArg RetArg(Flags, RegisterVT, ValueVT,
F.getReturnType(),
true,
11855 ArgCopyElisionCandidates);
11858 for (
const Argument &Arg :
F.args()) {
11859 unsigned ArgNo = Arg.getArgNo();
11862 bool isArgValueUsed = !Arg.
use_empty();
11864 if (Arg.hasAttribute(Attribute::ByVal))
11865 FinalType = Arg.getParamByValType();
11866 bool NeedsRegBlock =
TLI->functionArgumentNeedsConsecutiveRegisters(
11867 FinalType,
F.getCallingConv(),
F.isVarArg(),
DL);
11868 for (
unsigned Value = 0, NumValues =
Types.size();
Value != NumValues;
11871 EVT VT =
TLI->getValueType(
DL, ArgTy);
11872 ISD::ArgFlagsTy
Flags;
11875 Flags.setPointer();
11878 if (Arg.hasAttribute(Attribute::ZExt))
11880 if (Arg.hasAttribute(Attribute::SExt))
11882 if (Arg.hasAttribute(Attribute::InReg)) {
11889 Flags.setHvaStart();
11895 if (Arg.hasAttribute(Attribute::StructRet))
11897 if (Arg.hasAttribute(Attribute::SwiftSelf))
11898 Flags.setSwiftSelf();
11899 if (Arg.hasAttribute(Attribute::SwiftAsync))
11900 Flags.setSwiftAsync();
11901 if (Arg.hasAttribute(Attribute::SwiftError))
11902 Flags.setSwiftError();
11903 if (Arg.hasAttribute(Attribute::ByVal))
11905 if (Arg.hasAttribute(Attribute::ByRef))
11907 if (Arg.hasAttribute(Attribute::InAlloca)) {
11908 Flags.setInAlloca();
11916 if (Arg.hasAttribute(Attribute::Preallocated)) {
11917 Flags.setPreallocated();
11929 const Align OriginalAlignment(
11930 TLI->getABIAlignmentForCallingConv(ArgTy,
DL));
11931 Flags.setOrigAlign(OriginalAlignment);
11934 Type *ArgMemTy =
nullptr;
11935 if (
Flags.isByVal() ||
Flags.isInAlloca() ||
Flags.isPreallocated() ||
11938 ArgMemTy = Arg.getPointeeInMemoryValueType();
11940 uint64_t MemSize =
DL.getTypeAllocSize(ArgMemTy);
11945 if (
auto ParamAlign = Arg.getParamStackAlign())
11946 MemAlign = *ParamAlign;
11947 else if ((ParamAlign = Arg.getParamAlign()))
11948 MemAlign = *ParamAlign;
11950 MemAlign =
TLI->getByValTypeAlignment(ArgMemTy,
DL);
11951 if (
Flags.isByRef())
11952 Flags.setByRefSize(MemSize);
11954 Flags.setByValSize(MemSize);
11955 }
else if (
auto ParamAlign = Arg.getParamStackAlign()) {
11956 MemAlign = *ParamAlign;
11958 MemAlign = OriginalAlignment;
11960 Flags.setMemAlign(MemAlign);
11962 if (Arg.hasAttribute(Attribute::Nest))
11965 Flags.setInConsecutiveRegs();
11966 if (ArgCopyElisionCandidates.count(&Arg))
11967 Flags.setCopyElisionCandidate();
11968 if (Arg.hasAttribute(Attribute::Returned))
11969 Flags.setReturned();
11971 MVT RegisterVT =
TLI->getRegisterTypeForCallingConv(
11972 *
CurDAG->getContext(),
F.getCallingConv(), VT);
11973 unsigned NumRegs =
TLI->getNumRegistersForCallingConv(
11974 *
CurDAG->getContext(),
F.getCallingConv(), VT);
11975 for (
unsigned i = 0; i != NumRegs; ++i) {
11979 ISD::InputArg MyFlags(
11980 Flags, RegisterVT, VT, ArgTy, isArgValueUsed, ArgNo,
11982 if (NumRegs > 1 && i == 0)
11983 MyFlags.Flags.setSplit();
11986 MyFlags.Flags.setOrigAlign(
Align(1));
11987 if (i == NumRegs - 1)
11988 MyFlags.Flags.setSplitEnd();
11992 if (NeedsRegBlock &&
Value == NumValues - 1)
11993 Ins[Ins.
size() - 1].Flags.setInConsecutiveRegsLast();
11999 SDValue NewRoot =
TLI->LowerFormalArguments(
12000 DAG.
getRoot(),
F.getCallingConv(),
F.isVarArg(), Ins, dl, DAG, InVals);
12004 "LowerFormalArguments didn't return a valid chain!");
12006 "LowerFormalArguments didn't emit the correct number of values!");
12008 for (
unsigned i = 0, e = Ins.
size(); i != e; ++i) {
12010 "LowerFormalArguments emitted a null value!");
12012 "LowerFormalArguments emitted a value with the wrong type!");
12024 MVT VT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
12025 MVT RegVT =
TLI->getRegisterType(*
CurDAG->getContext(), VT);
12026 std::optional<ISD::NodeType> AssertOp;
12029 F.getCallingConv(), AssertOp);
12031 MachineFunction&
MF =
SDB->DAG.getMachineFunction();
12032 MachineRegisterInfo&
RegInfo =
MF.getRegInfo();
12034 RegInfo.createVirtualRegister(
TLI->getRegClassFor(RegVT));
12035 FuncInfo->DemoteRegister = SRetReg;
12037 SDB->DAG.getCopyToReg(NewRoot,
SDB->getCurSDLoc(), SRetReg, ArgValue);
12045 DenseMap<int, int> ArgCopyElisionFrameIndexMap;
12046 for (
const Argument &Arg :
F.args()) {
12050 unsigned NumValues = ValueVTs.
size();
12051 if (NumValues == 0)
12058 if (Ins[i].
Flags.isCopyElisionCandidate()) {
12059 unsigned NumParts = 0;
12060 for (EVT VT : ValueVTs)
12061 NumParts +=
TLI->getNumRegistersForCallingConv(*
CurDAG->getContext(),
12062 F.getCallingConv(), VT);
12066 ArrayRef(&InVals[i], NumParts), ArgHasUses);
12071 bool isSwiftErrorArg =
12072 TLI->supportSwiftError() &&
12073 Arg.hasAttribute(Attribute::SwiftError);
12074 if (!ArgHasUses && !isSwiftErrorArg) {
12075 SDB->setUnusedArgValue(&Arg, InVals[i]);
12078 if (FrameIndexSDNode *FI =
12080 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12083 for (
unsigned Val = 0; Val != NumValues; ++Val) {
12084 EVT VT = ValueVTs[Val];
12085 MVT PartVT =
TLI->getRegisterTypeForCallingConv(*
CurDAG->getContext(),
12086 F.getCallingConv(), VT);
12087 unsigned NumParts =
TLI->getNumRegistersForCallingConv(
12088 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12093 if (ArgHasUses || isSwiftErrorArg) {
12094 std::optional<ISD::NodeType> AssertOp;
12095 if (Arg.hasAttribute(Attribute::SExt))
12097 else if (Arg.hasAttribute(Attribute::ZExt))
12102 NewRoot,
F.getCallingConv(), AssertOp);
12105 if (NoFPClass !=
fcNone) {
12107 static_cast<uint64_t
>(NoFPClass), dl, MVT::i32);
12109 OutVal, SDNoFPClass);
12118 if (ArgValues.
empty())
12122 if (FrameIndexSDNode *FI =
12124 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12127 SDB->getCurSDLoc());
12129 SDB->setValue(&Arg, Res);
12139 if (LoadSDNode *LNode =
12141 if (FrameIndexSDNode *FI =
12143 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12171 FuncInfo->InitializeRegForValue(&Arg);
12172 SDB->CopyToExportRegsIfNeeded(&Arg);
12176 if (!Chains.
empty()) {
12183 assert(i == InVals.
size() &&
"Argument register count mismatch!");
12187 if (!ArgCopyElisionFrameIndexMap.
empty()) {
12188 for (MachineFunction::VariableDbgInfo &VI :
12189 MF->getInStackSlotVariableDbgInfo()) {
12190 auto I = ArgCopyElisionFrameIndexMap.
find(
VI.getStackSlot());
12191 if (
I != ArgCopyElisionFrameIndexMap.
end())
12192 VI.updateStackSlot(
I->second);
12207SelectionDAGBuilder::HandlePHINodesInSuccessorBlocks(
const BasicBlock *LLVMBB) {
12208 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12210 SmallPtrSet<MachineBasicBlock *, 4> SuccsHandled;
12216 MachineBasicBlock *SuccMBB =
FuncInfo.getMBB(SuccBB);
12220 if (!SuccsHandled.
insert(SuccMBB).second)
12228 for (
const PHINode &PN : SuccBB->phis()) {
12230 if (PN.use_empty())
12234 if (PN.getType()->isEmptyTy())
12238 const Value *PHIOp = PN.getIncomingValueForBlock(LLVMBB);
12243 RegOut =
FuncInfo.CreateRegs(&PN);
12261 "Didn't codegen value into a register!??");
12271 for (EVT VT : ValueVTs) {
12273 for (
unsigned i = 0; i != NumRegisters; ++i)
12275 Reg += NumRegisters;
12295void SelectionDAGBuilder::updateDAGForMaybeTailCall(
SDValue MaybeTC) {
12297 if (MaybeTC.
getNode() !=
nullptr)
12298 DAG.setRoot(MaybeTC);
12303void SelectionDAGBuilder::lowerWorkItem(SwitchWorkListItem W,
Value *
Cond,
12306 MachineFunction *CurMF =
FuncInfo.MF;
12307 MachineBasicBlock *NextMBB =
nullptr;
12312 unsigned Size =
W.LastCluster -
W.FirstCluster + 1;
12314 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12316 if (
Size == 2 &&
W.MBB == SwitchMBB) {
12324 CaseCluster &
Small = *
W.FirstCluster;
12325 CaseCluster &
Big = *
W.LastCluster;
12329 const APInt &SmallValue =
Small.Low->getValue();
12330 const APInt &BigValue =
Big.Low->getValue();
12333 APInt CommonBit = BigValue ^ SmallValue;
12340 DAG.getConstant(CommonBit,
DL, VT));
12342 DL, MVT::i1,
Or,
DAG.getConstant(BigValue | SmallValue,
DL, VT),
12348 addSuccessorWithProb(SwitchMBB,
Small.MBB,
Small.Prob +
Big.Prob);
12350 addSuccessorWithProb(
12351 SwitchMBB, DefaultMBB,
12355 addSuccessorWithProb(SwitchMBB, DefaultMBB);
12363 DAG.getBasicBlock(DefaultMBB));
12365 DAG.setRoot(BrCond);
12377 [](
const CaseCluster &a,
const CaseCluster &b) {
12378 return a.Prob != b.Prob ?
12380 a.Low->getValue().slt(b.Low->getValue());
12387 if (
I->Prob >
W.LastCluster->Prob)
12389 if (
I->Kind ==
CC_Range &&
I->MBB == NextMBB) {
12397 BranchProbability DefaultProb =
W.DefaultProb;
12398 BranchProbability UnhandledProbs = DefaultProb;
12400 UnhandledProbs +=
I->Prob;
12402 MachineBasicBlock *CurMBB =
W.MBB;
12404 bool FallthroughUnreachable =
false;
12405 MachineBasicBlock *Fallthrough;
12406 if (
I ==
W.LastCluster) {
12408 Fallthrough = DefaultMBB;
12413 CurMF->
insert(BBI, Fallthrough);
12417 UnhandledProbs -=
I->Prob;
12422 JumpTableHeader *JTH = &
SL->JTCases[
I->JTCasesIndex].first;
12423 SwitchCG::JumpTable *JT = &
SL->JTCases[
I->JTCasesIndex].second;
12426 MachineBasicBlock *JumpMBB = JT->
MBB;
12427 CurMF->
insert(BBI, JumpMBB);
12429 auto JumpProb =
I->Prob;
12430 auto FallthroughProb = UnhandledProbs;
12438 if (*SI == DefaultMBB) {
12439 JumpProb += DefaultProb / 2;
12440 FallthroughProb -= DefaultProb / 2;
12458 if (FallthroughUnreachable) {
12465 addSuccessorWithProb(CurMBB, Fallthrough, FallthroughProb);
12466 addSuccessorWithProb(CurMBB, JumpMBB, JumpProb);
12475 if (CurMBB == SwitchMBB) {
12483 BitTestBlock *BTB = &
SL->BitTestCases[
I->BTCasesIndex];
12486 for (BitTestCase &BTC : BTB->
Cases)
12498 BTB->
Prob += DefaultProb / 2;
12502 if (FallthroughUnreachable)
12506 if (CurMBB == SwitchMBB) {
12513 const Value *
RHS, *
LHS, *MHS;
12515 if (
I->Low ==
I->High) {
12530 if (FallthroughUnreachable)
12534 CaseBlock CB(CC,
LHS,
RHS, MHS,
I->MBB, Fallthrough, CurMBB,
12537 if (CurMBB == SwitchMBB)
12540 SL->SwitchCases.push_back(CB);
12545 CurMBB = Fallthrough;
12549void SelectionDAGBuilder::splitWorkItem(
SwitchWorkList &WorkList,
12550 const SwitchWorkListItem &W,
12553 assert(
W.FirstCluster->Low->getValue().slt(
W.LastCluster->Low->getValue()) &&
12554 "Clusters not sorted?");
12555 assert(
W.LastCluster -
W.FirstCluster + 1 >= 2 &&
"Too small to split!");
12557 auto [LastLeft, FirstRight, LeftProb, RightProb] =
12558 SL->computeSplitWorkItemInfo(W);
12563 assert(PivotCluster >
W.FirstCluster);
12564 assert(PivotCluster <=
W.LastCluster);
12569 const ConstantInt *Pivot = PivotCluster->Low;
12578 MachineBasicBlock *LeftMBB;
12579 if (FirstLeft == LastLeft && FirstLeft->Kind ==
CC_Range &&
12580 FirstLeft->Low ==
W.GE &&
12581 (FirstLeft->High->getValue() + 1LL) == Pivot->
getValue()) {
12582 LeftMBB = FirstLeft->MBB;
12584 LeftMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12585 FuncInfo.MF->insert(BBI, LeftMBB);
12587 {LeftMBB, FirstLeft, LastLeft,
W.GE, Pivot,
W.DefaultProb / 2});
12595 MachineBasicBlock *RightMBB;
12596 if (FirstRight == LastRight && FirstRight->Kind ==
CC_Range &&
12597 W.LT && (FirstRight->High->getValue() + 1ULL) ==
W.LT->getValue()) {
12598 RightMBB = FirstRight->MBB;
12600 RightMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12601 FuncInfo.MF->insert(BBI, RightMBB);
12603 {RightMBB, FirstRight, LastRight, Pivot,
W.LT,
W.DefaultProb / 2});
12609 CaseBlock CB(
ISD::SETLT,
Cond, Pivot,
nullptr, LeftMBB, RightMBB,
W.MBB,
12612 if (
W.MBB == SwitchMBB)
12615 SL->SwitchCases.push_back(CB);
12640 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12648 unsigned PeeledCaseIndex = 0;
12649 bool SwitchPeeled =
false;
12650 for (
unsigned Index = 0;
Index < Clusters.size(); ++
Index) {
12651 CaseCluster &CC = Clusters[
Index];
12652 if (CC.
Prob < TopCaseProb)
12654 TopCaseProb = CC.
Prob;
12655 PeeledCaseIndex =
Index;
12656 SwitchPeeled =
true;
12661 LLVM_DEBUG(
dbgs() <<
"Peeled one top case in switch stmt, prob: "
12662 << TopCaseProb <<
"\n");
12667 MachineBasicBlock *PeeledSwitchMBB =
12669 FuncInfo.MF->insert(BBI, PeeledSwitchMBB);
12672 auto PeeledCaseIt = Clusters.begin() + PeeledCaseIndex;
12673 SwitchWorkListItem
W = {SwitchMBB, PeeledCaseIt, PeeledCaseIt,
12674 nullptr,
nullptr, TopCaseProb.
getCompl()};
12675 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, PeeledSwitchMBB);
12677 Clusters.erase(PeeledCaseIt);
12678 for (CaseCluster &CC : Clusters) {
12680 dbgs() <<
"Scale the probablity for one cluster, before scaling: "
12681 << CC.
Prob <<
"\n");
12685 PeeledCaseProb = TopCaseProb;
12686 return PeeledSwitchMBB;
12689void SelectionDAGBuilder::visitSwitch(
const SwitchInst &
SI) {
12691 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12693 Clusters.reserve(
SI.getNumCases());
12694 for (
auto I :
SI.cases()) {
12695 MachineBasicBlock *Succ =
FuncInfo.getMBB(
I.getCaseSuccessor());
12696 const ConstantInt *CaseVal =
I.getCaseValue();
12697 BranchProbability Prob =
12699 : BranchProbability(1,
SI.getNumCases() + 1);
12703 MachineBasicBlock *DefaultMBB =
FuncInfo.getMBB(
SI.getDefaultDest());
12712 MachineBasicBlock *PeeledSwitchMBB =
12713 peelDominantCaseCluster(SI, Clusters, PeeledCaseProb);
12716 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12717 if (Clusters.empty()) {
12718 assert(PeeledSwitchMBB == SwitchMBB);
12720 if (DefaultMBB != NextBlock(SwitchMBB)) {
12727 SL->findJumpTables(Clusters, &SI,
getCurSDLoc(), DefaultMBB,
DAG.getPSI(),
12729 SL->findBitTestClusters(Clusters, &SI);
12732 dbgs() <<
"Case clusters: ";
12733 for (
const CaseCluster &
C : Clusters) {
12739 C.Low->getValue().print(
dbgs(),
true);
12740 if (
C.Low !=
C.High) {
12742 C.High->getValue().print(
dbgs(),
true);
12749 assert(!Clusters.empty());
12753 auto DefaultProb = getEdgeProbability(PeeledSwitchMBB, DefaultMBB);
12757 DefaultMBB ==
FuncInfo.getMBB(
SI.getDefaultDest()))
12760 {PeeledSwitchMBB,
First,
Last,
nullptr,
nullptr, DefaultProb});
12762 while (!WorkList.
empty()) {
12764 unsigned NumClusters =
W.LastCluster -
W.FirstCluster + 1;
12769 splitWorkItem(WorkList, W,
SI.getCondition(), SwitchMBB);
12773 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, DefaultMBB);
12777void SelectionDAGBuilder::visitStepVector(
const CallInst &
I) {
12778 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12784void SelectionDAGBuilder::visitVectorReverse(
const CallInst &
I) {
12785 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12790 assert(VT ==
V.getValueType() &&
"Malformed vector.reverse!");
12799 SmallVector<int, 8>
Mask;
12801 for (
unsigned i = 0; i != NumElts; ++i)
12802 Mask.push_back(NumElts - 1 - i);
12807void SelectionDAGBuilder::visitVectorDeinterleave(
const CallInst &
I,
12816 EVT OutVT = ValueVTs[0];
12820 for (
unsigned i = 0; i != Factor; ++i) {
12821 assert(ValueVTs[i] == OutVT &&
"Expected VTs to be the same");
12823 DAG.getVectorIdxConstant(OutNumElts * i,
DL));
12829 SDValue Even =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
12831 SDValue Odd =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
12839 DAG.getVTList(ValueVTs), SubVecs);
12843void SelectionDAGBuilder::visitVectorInterleave(
const CallInst &
I,
12846 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12851 for (
unsigned i = 0; i < Factor; ++i) {
12854 "Expected VTs to be the same");
12872 for (
unsigned i = 0; i < Factor; ++i)
12879void SelectionDAGBuilder::visitFreeze(
const FreezeInst &
I) {
12883 unsigned NumValues = ValueVTs.
size();
12884 if (NumValues == 0)
return;
12889 for (
unsigned i = 0; i != NumValues; ++i)
12894 DAG.getVTList(ValueVTs), Values));
12897void SelectionDAGBuilder::visitVectorSplice(
const CallInst &
I) {
12898 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12904 const bool IsLeft =
I.getIntrinsicID() == Intrinsic::vector_splice_left;
12919 uint64_t Idx = IsLeft ?
Imm : NumElts -
Imm;
12922 SmallVector<int, 8>
Mask;
12923 for (
unsigned i = 0; i < NumElts; ++i)
12924 Mask.push_back(Idx + i);
12952 assert(
MI->getOpcode() == TargetOpcode::COPY &&
12953 "start of copy chain MUST be COPY");
12954 Reg =
MI->getOperand(1).getReg();
12957 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
12958 MI =
MRI.def_begin(
Reg)->getParent();
12961 if (
MI->getOpcode() == TargetOpcode::COPY) {
12962 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
12963 Reg =
MI->getOperand(1).getReg();
12964 assert(
Reg.isPhysical() &&
"expected COPY of physical register");
12967 assert(
MI->getOpcode() == TargetOpcode::INLINEASM_BR &&
12968 "end of copy chain MUST be INLINEASM_BR");
12978void SelectionDAGBuilder::visitCallBrLandingPad(
const CallInst &
I) {
12984 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12985 const TargetRegisterInfo *
TRI =
DAG.getSubtarget().getRegisterInfo();
12986 MachineRegisterInfo &
MRI =
DAG.getMachineFunction().getRegInfo();
12994 for (
auto &
T : TargetConstraints) {
12995 SDISelAsmOperandInfo OpInfo(
T);
13003 switch (OpInfo.ConstraintType) {
13014 FuncInfo.MBB->addLiveIn(OriginalDef);
13022 ResultVTs.
push_back(OpInfo.ConstraintVT);
13031 ResultVTs.
push_back(OpInfo.ConstraintVT);
13039 DAG.getVTList(ResultVTs), ResultValues);
unsigned const MachineRegisterInfo * MRI
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()
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.
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,...
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 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...
@ 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)