79#include "llvm/IR/IntrinsicsAArch64.h"
80#include "llvm/IR/IntrinsicsAMDGPU.h"
81#include "llvm/IR/IntrinsicsWebAssembly.h"
114#define DEBUG_TYPE "isel"
122 cl::desc(
"Insert the experimental `assertalign` node."),
127 cl::desc(
"Generate low-precision inline sequences "
128 "for some float libcalls"),
134 cl::desc(
"Set the case probability threshold for peeling the case from a "
135 "switch statement. A value greater than 100 will void this "
155 const SDValue *Parts,
unsigned NumParts,
158 std::optional<CallingConv::ID> CC);
167 unsigned NumParts,
MVT PartVT,
EVT ValueVT,
const Value *V,
169 std::optional<CallingConv::ID> CC = std::nullopt,
170 std::optional<ISD::NodeType> AssertOp = std::nullopt) {
174 PartVT, ValueVT, CC))
181 assert(NumParts > 0 &&
"No parts to assemble!");
192 unsigned RoundBits = PartBits * RoundParts;
193 EVT RoundVT = RoundBits == ValueBits ?
199 if (RoundParts > 2) {
203 PartVT, HalfVT, V, InChain);
214 if (RoundParts < NumParts) {
216 unsigned OddParts = NumParts - RoundParts;
219 OddVT, V, InChain, CC);
235 assert(ValueVT ==
EVT(MVT::ppcf128) && PartVT == MVT::f64 &&
246 !PartVT.
isVector() &&
"Unexpected split");
258 if (PartEVT == ValueVT)
262 ValueVT.
bitsLT(PartEVT)) {
275 if (ValueVT.
bitsLT(PartEVT)) {
280 Val = DAG.
getNode(*AssertOp,
DL, PartEVT, Val,
295 llvm::Attribute::StrictFP)) {
297 DAG.
getVTList(ValueVT, MVT::Other), InChain, Val,
309 if (PartEVT == MVT::x86mmx && ValueVT.
isInteger() &&
310 ValueVT.
bitsLT(PartEVT)) {
319 const Twine &ErrMsg) {
322 return Ctx.emitError(ErrMsg);
325 if (CI->isInlineAsm()) {
327 *CI, ErrMsg +
", possible invalid constraint for vector type"));
330 return Ctx.emitError(
I, ErrMsg);
339 const SDValue *Parts,
unsigned NumParts,
342 std::optional<CallingConv::ID> CallConv) {
344 assert(NumParts > 0 &&
"No parts to assemble!");
345 const bool IsABIRegCopy = CallConv.has_value();
354 unsigned NumIntermediates;
359 *DAG.
getContext(), *CallConv, ValueVT, IntermediateVT,
360 NumIntermediates, RegisterVT);
364 NumIntermediates, RegisterVT);
367 assert(NumRegs == NumParts &&
"Part count doesn't match vector breakdown!");
369 assert(RegisterVT == PartVT &&
"Part type doesn't match vector breakdown!");
372 "Part type sizes don't match!");
376 if (NumIntermediates == NumParts) {
379 for (
unsigned i = 0; i != NumParts; ++i)
381 V, InChain, CallConv);
382 }
else if (NumParts > 0) {
385 assert(NumParts % NumIntermediates == 0 &&
386 "Must expand into a divisible number of parts!");
387 unsigned Factor = NumParts / NumIntermediates;
388 for (
unsigned i = 0; i != NumIntermediates; ++i)
390 IntermediateVT, V, InChain, CallConv);
405 DL, BuiltVectorTy,
Ops);
411 if (PartEVT == ValueVT)
427 "Cannot narrow, it would be a lossy transformation");
433 if (PartEVT == ValueVT)
458 }
else if (ValueVT.
bitsLT(PartEVT)) {
467 *DAG.
getContext(), V,
"non-trivial scalar-to-vector conversion");
498 std::optional<CallingConv::ID> CallConv);
505 unsigned NumParts,
MVT PartVT,
const Value *V,
506 std::optional<CallingConv::ID> CallConv = std::nullopt,
520 unsigned OrigNumParts = NumParts;
522 "Copying to an illegal type!");
528 EVT PartEVT = PartVT;
529 if (PartEVT == ValueVT) {
530 assert(NumParts == 1 &&
"No-op copy with multiple parts!");
539 assert(NumParts == 1 &&
"Do not know what to promote to!");
550 "Unknown mismatch!");
552 Val = DAG.
getNode(ExtendKind,
DL, ValueVT, Val);
553 if (PartVT == MVT::x86mmx)
558 assert(NumParts == 1 && PartEVT != ValueVT);
564 "Unknown mismatch!");
567 if (PartVT == MVT::x86mmx)
574 "Failed to tile the value with PartVT!");
577 if (PartEVT != ValueVT) {
579 "scalar-to-vector conversion failed");
588 if (NumParts & (NumParts - 1)) {
591 "Do not know what to expand to!");
593 unsigned RoundBits = RoundParts * PartBits;
594 unsigned OddParts = NumParts - RoundParts;
603 std::reverse(Parts + RoundParts, Parts + NumParts);
605 NumParts = RoundParts;
617 for (
unsigned StepSize = NumParts; StepSize > 1; StepSize /= 2) {
618 for (
unsigned i = 0; i < NumParts; i += StepSize) {
619 unsigned ThisBits = StepSize * PartBits / 2;
622 SDValue &Part1 = Parts[i+StepSize/2];
629 if (ThisBits == PartBits && ThisVT != PartVT) {
637 std::reverse(Parts, Parts + OrigNumParts);
659 if (ValueEVT == MVT::bf16 && PartEVT == MVT::f16) {
661 "Cannot widen to illegal type");
665 }
else if (PartEVT != ValueEVT) {
680 Ops.append((PartNumElts - ValueNumElts).getFixedValue(), EltUndef);
691 std::optional<CallingConv::ID> CallConv) {
695 const bool IsABIRegCopy = CallConv.has_value();
698 EVT PartEVT = PartVT;
699 if (PartEVT == ValueVT) {
745 "lossy conversion of vector to scalar type");
760 unsigned NumIntermediates;
764 *DAG.
getContext(), *CallConv, ValueVT, IntermediateVT, NumIntermediates,
769 NumIntermediates, RegisterVT);
772 assert(NumRegs == NumParts &&
"Part count doesn't match vector breakdown!");
774 assert(RegisterVT == PartVT &&
"Part type doesn't match vector breakdown!");
777 "Mixing scalable and fixed vectors when copying in parts");
779 std::optional<ElementCount> DestEltCnt;
789 if (ValueVT == BuiltVectorTy) {
813 for (
unsigned i = 0; i != NumIntermediates; ++i) {
828 if (NumParts == NumIntermediates) {
831 for (
unsigned i = 0; i != NumParts; ++i)
833 }
else if (NumParts > 0) {
836 assert(NumIntermediates != 0 &&
"division by zero");
837 assert(NumParts % NumIntermediates == 0 &&
838 "Must expand into a divisible number of parts!");
839 unsigned Factor = NumParts / NumIntermediates;
840 for (
unsigned i = 0; i != NumIntermediates; ++i)
848 if (
I.hasOperandBundlesOtherThan(AllowedBundles)) {
852 for (
unsigned i = 0, e =
I.getNumOperandBundles(); i != e; ++i) {
855 OS << LS << U.getTagName();
858 Twine(
"cannot lower ", Name)
864 EVT valuevt, std::optional<CallingConv::ID> CC)
870 std::optional<CallingConv::ID> CC) {
884 for (
unsigned i = 0; i != NumRegs; ++i)
885 Regs.push_back(Reg + i);
886 RegVTs.push_back(RegisterVT);
888 Reg = Reg.id() + NumRegs;
915 for (
unsigned i = 0; i != NumRegs; ++i) {
921 *Glue =
P.getValue(2);
924 Chain =
P.getValue(1);
952 EVT FromVT(MVT::Other);
956 }
else if (NumSignBits > 1) {
964 assert(FromVT != MVT::Other);
970 RegisterVT, ValueVT, V, Chain,
CallConv);
986 unsigned NumRegs =
Regs.size();
1001 NumParts, RegisterVT, V,
CallConv, ExtendKind);
1007 for (
unsigned i = 0; i != NumRegs; ++i) {
1019 if (NumRegs == 1 || Glue)
1030 Chain = Chains[NumRegs-1];
1036 unsigned MatchingIdx,
const SDLoc &dl,
1038 std::vector<SDValue> &
Ops)
const {
1043 Flag.setMatchingOp(MatchingIdx);
1044 else if (!
Regs.empty() &&
Regs.front().isVirtual()) {
1052 Flag.setRegClass(RC->
getID());
1063 "No 1:1 mapping from clobbers to regs?");
1066 for (
unsigned I = 0, E =
ValueVTs.size();
I != E; ++
I) {
1071 "If we clobbered the stack pointer, MFI should know about it.");
1080 for (
unsigned i = 0; i != NumRegs; ++i) {
1081 assert(Reg <
Regs.size() &&
"Mismatch in # registers expected");
1093 unsigned RegCount = std::get<0>(CountAndVT);
1094 MVT RegisterVT = std::get<1>(CountAndVT);
1112 SL->init(
DAG.getTargetLoweringInfo(), TM,
DAG.getDataLayout());
1114 *
DAG.getMachineFunction().getFunction().getParent());
1119 UnusedArgNodeMap.clear();
1121 PendingExports.clear();
1122 PendingConstrainedFP.clear();
1123 PendingConstrainedFPStrict.clear();
1131 DanglingDebugInfoMap.clear();
1138 if (Pending.
empty())
1144 unsigned i = 0, e = Pending.
size();
1145 for (; i != e; ++i) {
1147 if (Pending[i].
getNode()->getOperand(0) == Root)
1155 if (Pending.
size() == 1)
1182 if (!PendingConstrainedFPStrict.empty()) {
1183 assert(PendingConstrainedFP.empty());
1184 updateRoot(PendingConstrainedFPStrict);
1197 if (!PendingConstrainedFP.empty()) {
1198 assert(PendingConstrainedFPStrict.empty());
1199 updateRoot(PendingConstrainedFP);
1203 return DAG.getRoot();
1211 PendingConstrainedFP.size() +
1212 PendingConstrainedFPStrict.size());
1214 PendingConstrainedFP.end());
1215 PendingLoads.append(PendingConstrainedFPStrict.begin(),
1216 PendingConstrainedFPStrict.end());
1217 PendingConstrainedFP.clear();
1218 PendingConstrainedFPStrict.clear();
1225 PendingExports.append(PendingConstrainedFPStrict.begin(),
1226 PendingConstrainedFPStrict.end());
1227 PendingConstrainedFPStrict.clear();
1228 return updateRoot(PendingExports);
1235 assert(Variable &&
"Missing variable");
1242 <<
"dbg_declare: Dropping debug info (bad/undef/unused-arg address)\n");
1258 if (IsParameter && FINode) {
1260 SDV =
DAG.getFrameIndexDbgValue(Variable,
Expression, FINode->getIndex(),
1261 true,
DL, SDNodeOrder);
1266 FuncArgumentDbgValueKind::Declare,
N);
1269 SDV =
DAG.getDbgValue(Variable,
Expression,
N.getNode(),
N.getResNo(),
1270 true,
DL, SDNodeOrder);
1272 DAG.AddDbgValue(SDV, IsParameter);
1277 FuncArgumentDbgValueKind::Declare,
N)) {
1279 <<
" (could not emit func-arg dbg_value)\n");
1290 for (
auto It = FnVarLocs->locs_begin(&
I), End = FnVarLocs->locs_end(&
I);
1292 auto *Var = FnVarLocs->getDILocalVariable(It->VariableID);
1294 if (It->Values.isKillLocation(It->Expr)) {
1300 It->Values.hasArgList())) {
1303 FnVarLocs->getDILocalVariable(It->VariableID),
1304 It->Expr, Vals.
size() > 1, It->DL, SDNodeOrder);
1317 bool SkipDbgVariableRecords =
DAG.getFunctionVarLocs();
1320 for (
DbgRecord &DR :
I.getDbgRecordRange()) {
1322 assert(DLR->getLabel() &&
"Missing label");
1324 DAG.getDbgLabel(DLR->getLabel(), DLR->getDebugLoc(), SDNodeOrder);
1325 DAG.AddDbgLabel(SDV);
1329 if (SkipDbgVariableRecords)
1337 if (
FuncInfo.PreprocessedDVRDeclares.contains(&DVR))
1339 LLVM_DEBUG(
dbgs() <<
"SelectionDAG visiting dbg_declare: " << DVR
1365 SDNodeOrder, IsVariadic)) {
1376 if (
I.isTerminator()) {
1377 HandlePHINodesInSuccessorBlocks(
I.getParent());
1384 bool NodeInserted =
false;
1385 std::unique_ptr<SelectionDAG::DAGNodeInsertedListener> InsertedListener;
1386 MDNode *PCSectionsMD =
I.getMetadata(LLVMContext::MD_pcsections);
1387 MDNode *MMRA =
I.getMetadata(LLVMContext::MD_mmra);
1388 if (PCSectionsMD || MMRA) {
1389 InsertedListener = std::make_unique<SelectionDAG::DAGNodeInsertedListener>(
1390 DAG, [&](
SDNode *) { NodeInserted =
true; });
1400 if (PCSectionsMD || MMRA) {
1401 auto It = NodeMap.find(&
I);
1402 if (It != NodeMap.end()) {
1404 DAG.addPCSections(It->second.getNode(), PCSectionsMD);
1406 DAG.addMMRAMetadata(It->second.getNode(), MMRA);
1407 }
else if (NodeInserted) {
1410 errs() <<
"warning: loosing !pcsections and/or !mmra metadata ["
1411 <<
I.getModule()->getName() <<
"]\n";
1420void SelectionDAGBuilder::visitPHI(
const PHINode &) {
1430#define HANDLE_INST(NUM, OPCODE, CLASS) \
1431 case Instruction::OPCODE: visit##OPCODE((const CLASS&)I); break;
1432#include "llvm/IR/Instruction.def"
1469 DanglingDebugInfoMap[
Values[0]].emplace_back(Var, Expr,
DL, Order);
1474 auto isMatchingDbgValue = [&](DanglingDebugInfo &DDI) {
1475 DIVariable *DanglingVariable = DDI.getVariable();
1477 if (DanglingVariable == Variable && Expr->
fragmentsOverlap(DanglingExpr)) {
1479 << printDDI(
nullptr, DDI) <<
"\n");
1485 for (
auto &DDIMI : DanglingDebugInfoMap) {
1486 DanglingDebugInfoVector &DDIV = DDIMI.second;
1490 for (
auto &DDI : DDIV)
1491 if (isMatchingDbgValue(DDI))
1494 erase_if(DDIV, isMatchingDbgValue);
1502 auto DanglingDbgInfoIt = DanglingDebugInfoMap.find(V);
1503 if (DanglingDbgInfoIt == DanglingDebugInfoMap.end())
1506 DanglingDebugInfoVector &DDIV = DanglingDbgInfoIt->second;
1507 for (
auto &DDI : DDIV) {
1509 unsigned DbgSDNodeOrder = DDI.getSDNodeOrder();
1512 assert(Variable->isValidLocationForIntrinsic(
DL) &&
1513 "Expected inlined-at fields to agree");
1523 if (!EmitFuncArgumentDbgValue(V, Variable, Expr,
DL,
1524 FuncArgumentDbgValueKind::Value, Val)) {
1526 << printDDI(V, DDI) <<
"\n");
1533 <<
"changing SDNodeOrder from " << DbgSDNodeOrder <<
" to "
1534 << ValSDNodeOrder <<
"\n");
1535 SDV = getDbgValue(Val, Variable, Expr,
DL,
1536 std::max(DbgSDNodeOrder, ValSDNodeOrder));
1537 DAG.AddDbgValue(SDV,
false);
1541 <<
" in EmitFuncArgumentDbgValue\n");
1543 LLVM_DEBUG(
dbgs() <<
"Dropping debug info for " << printDDI(V, DDI)
1547 DAG.getConstantDbgValue(Variable, Expr,
Poison,
DL, DbgSDNodeOrder);
1548 DAG.AddDbgValue(SDV,
false);
1555 DanglingDebugInfo &DDI) {
1560 const Value *OrigV = V;
1564 unsigned SDOrder = DDI.getSDNodeOrder();
1568 bool StackValue =
true;
1593 if (!AdditionalValues.
empty())
1603 dbgs() <<
"Salvaged debug location info for:\n " << *Var <<
"\n"
1604 << *OrigV <<
"\nBy stripping back to:\n " << *V <<
"\n");
1612 assert(OrigV &&
"V shouldn't be null");
1614 auto *SDV =
DAG.getConstantDbgValue(Var, Expr,
Poison,
DL, SDNodeOrder);
1615 DAG.AddDbgValue(SDV,
false);
1617 << printDDI(OrigV, DDI) <<
"\n");
1634 unsigned Order,
bool IsVariadic) {
1639 if (visitEntryValueDbgValue(
Values, Var, Expr, DbgLoc))
1644 for (
const Value *V :
Values) {
1654 if (CE->getOpcode() == Instruction::IntToPtr) {
1673 N = UnusedArgNodeMap[V];
1678 EmitFuncArgumentDbgValue(V, Var, Expr, DbgLoc,
1679 FuncArgumentDbgValueKind::Value,
N))
1706 bool IsParamOfFunc =
1714 auto VMI =
FuncInfo.ValueMap.find(V);
1715 if (VMI !=
FuncInfo.ValueMap.end()) {
1720 V->getType(), std::nullopt);
1726 unsigned BitsToDescribe = 0;
1728 BitsToDescribe = *VarSize;
1730 BitsToDescribe = Fragment->SizeInBits;
1733 if (
Offset >= BitsToDescribe)
1736 unsigned RegisterSize = RegAndSize.second;
1737 unsigned FragmentSize = (
Offset + RegisterSize > BitsToDescribe)
1738 ? BitsToDescribe -
Offset
1741 Expr,
Offset, FragmentSize);
1745 Var, *FragmentExpr, RegAndSize.first,
false, DbgLoc, Order);
1746 DAG.AddDbgValue(SDV,
false);
1762 DAG.getDbgValueList(Var, Expr, LocationOps, Dependencies,
1763 false, DbgLoc, Order, IsVariadic);
1764 DAG.AddDbgValue(SDV,
false);
1770 for (
auto &Pair : DanglingDebugInfoMap)
1771 for (
auto &DDI : Pair.second)
1779 auto It =
FuncInfo.ValueMap.find(V);
1782 if (It !=
FuncInfo.ValueMap.end()) {
1786 DAG.getDataLayout(), InReg, Ty,
1803 if (
N.getNode())
return N;
1863 return DAG.getSplatBuildVector(
1866 return DAG.getConstant(*CI,
DL, VT);
1878 getValue(CPA->getAddrDiscriminator()),
1879 getValue(CPA->getDiscriminator()));
1895 visit(CE->getOpcode(), *CE);
1897 assert(N1.
getNode() &&
"visit didn't populate the NodeMap!");
1903 for (
const Use &U :
C->operands()) {
1909 for (
unsigned i = 0, e = Val->
getNumValues(); i != e; ++i)
1910 Constants.push_back(
SDValue(Val, i));
1919 for (
uint64_t i = 0, e = CDS->getNumElements(); i != e; ++i) {
1923 for (
unsigned i = 0, e = Val->
getNumValues(); i != e; ++i)
1932 if (
C->getType()->isStructTy() ||
C->getType()->isArrayTy()) {
1934 "Unknown struct or array constant!");
1938 unsigned NumElts = ValueVTs.
size();
1942 for (
unsigned i = 0; i != NumElts; ++i) {
1943 EVT EltVT = ValueVTs[i];
1945 Constants[i] =
DAG.getUNDEF(EltVT);
1956 return DAG.getBlockAddress(BA, VT);
1959 return getValue(Equiv->getGlobalValue());
1964 if (VT == MVT::aarch64svcount) {
1965 assert(
C->isNullValue() &&
"Can only zero this target type!");
1971 assert(
C->isNullValue() &&
"Can only zero this target type!");
1981 if (VT == MVT::externref || VT == MVT::funcref) {
1982 assert(
C->isNullValue() &&
"Can only zero this target type!");
1985 Intrinsic::ID IID = VT == MVT::externref ? Intrinsic::wasm_ref_null_extern
1986 : Intrinsic::wasm_ref_null_func;
1998 for (
unsigned i = 0; i != NumElements; ++i)
2025 return DAG.getFrameIndex(
2033 Inst->getType(), std::nullopt);
2047void SelectionDAGBuilder::visitCatchPad(
const CatchPadInst &
I) {
2060 if (IsMSVCCXX || IsCoreCLR)
2066 MachineBasicBlock *TargetMBB =
FuncInfo.getMBB(
I.getSuccessor());
2067 FuncInfo.MBB->addSuccessor(TargetMBB);
2074 if (TargetMBB != NextBlock(
FuncInfo.MBB) ||
2083 DAG.getMachineFunction().setHasEHContTarget(
true);
2089 Value *ParentPad =
I.getCatchSwitchParentPad();
2092 SuccessorColor = &
FuncInfo.Fn->getEntryBlock();
2095 assert(SuccessorColor &&
"No parent funclet for catchret!");
2096 MachineBasicBlock *SuccessorColorMBB =
FuncInfo.getMBB(SuccessorColor);
2097 assert(SuccessorColorMBB &&
"No MBB for SuccessorColor!");
2102 DAG.getBasicBlock(SuccessorColorMBB));
2106void SelectionDAGBuilder::visitCleanupPad(
const CleanupPadInst &CPI) {
2112 FuncInfo.MBB->setIsEHFuncletEntry();
2113 FuncInfo.MBB->setIsCleanupFuncletEntry();
2142 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2148 UnwindDests.emplace_back(FuncInfo.
getMBB(EHPadBB), Prob);
2149 UnwindDests.back().first->setIsEHScopeEntry();
2152 UnwindDests.back().first->setIsEHFuncletEntry();
2156 for (
const BasicBlock *CatchPadBB : CatchSwitch->handlers()) {
2157 UnwindDests.emplace_back(FuncInfo.
getMBB(CatchPadBB), Prob);
2159 if (IsMSVCCXX || IsCoreCLR)
2160 UnwindDests.back().first->setIsEHFuncletEntry();
2162 UnwindDests.back().first->setIsEHScopeEntry();
2164 NewEHPadBB = CatchSwitch->getUnwindDest();
2170 if (BPI && NewEHPadBB)
2172 EHPadBB = NewEHPadBB;
2179 auto UnwindDest =
I.getUnwindDest();
2180 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
2181 BranchProbability UnwindDestProb =
2186 for (
auto &UnwindDest : UnwindDests) {
2187 UnwindDest.first->setIsEHPad();
2188 addSuccessorWithProb(
FuncInfo.MBB, UnwindDest.first, UnwindDest.second);
2190 FuncInfo.MBB->normalizeSuccProbs();
2193 MachineBasicBlock *CleanupPadMBB =
2194 FuncInfo.getMBB(
I.getCleanupPad()->getParent());
2200void SelectionDAGBuilder::visitCatchSwitch(
const CatchSwitchInst &CSI) {
2204void SelectionDAGBuilder::visitRet(
const ReturnInst &
I) {
2205 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
2206 auto &
DL =
DAG.getDataLayout();
2218 if (
I.getParent()->getTerminatingDeoptimizeCall()) {
2235 SmallVector<uint64_t, 4>
Offsets;
2238 unsigned NumValues = ValueVTs.
size();
2241 Align BaseAlign =
DL.getPrefTypeAlign(
I.getOperand(0)->getType());
2242 for (
unsigned i = 0; i != NumValues; ++i) {
2249 if (MemVTs[i] != ValueVTs[i])
2251 Chains[i] =
DAG.getStore(
2259 MVT::Other, Chains);
2260 }
else if (
I.getNumOperands() != 0) {
2263 unsigned NumValues =
Types.size();
2267 const Function *
F =
I.getParent()->getParent();
2270 I.getOperand(0)->getType(),
F->getCallingConv(),
2274 if (
F->getAttributes().hasRetAttr(Attribute::SExt))
2276 else if (
F->getAttributes().hasRetAttr(Attribute::ZExt))
2279 LLVMContext &
Context =
F->getContext();
2280 bool RetInReg =
F->getAttributes().hasRetAttr(Attribute::InReg);
2282 for (
unsigned j = 0;
j != NumValues; ++
j) {
2295 &Parts[0], NumParts, PartVT, &
I, CC, ExtendKind);
2298 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2302 if (
I.getOperand(0)->getType()->isPointerTy()) {
2304 Flags.setPointerAddrSpace(
2308 if (NeedsRegBlock) {
2309 Flags.setInConsecutiveRegs();
2310 if (j == NumValues - 1)
2311 Flags.setInConsecutiveRegsLast();
2319 else if (
F->getAttributes().hasRetAttr(Attribute::NoExt))
2322 for (
unsigned i = 0; i < NumParts; ++i) {
2325 VT, Types[j], 0, 0));
2335 const Function *
F =
I.getParent()->getParent();
2337 F->getAttributes().hasAttrSomewhere(Attribute::SwiftError)) {
2339 ISD::ArgFlagsTy
Flags = ISD::ArgFlagsTy();
2340 Flags.setSwiftError();
2352 bool isVarArg =
DAG.getMachineFunction().getFunction().isVarArg();
2354 DAG.getMachineFunction().getFunction().getCallingConv();
2355 Chain =
DAG.getTargetLoweringInfo().LowerReturn(
2360 "LowerReturn didn't return a valid chain!");
2371 if (V->getType()->isEmptyTy())
2374 auto VMI =
FuncInfo.ValueMap.find(V);
2375 if (VMI !=
FuncInfo.ValueMap.end()) {
2377 "Unused value assigned virtual registers!");
2390 if (
FuncInfo.isExportedInst(V))
return;
2402 if (VI->getParent() == FromBB)
2428 const BasicBlock *SrcBB = Src->getBasicBlock();
2429 const BasicBlock *DstBB = Dst->getBasicBlock();
2433 auto SuccSize = std::max<uint32_t>(
succ_size(SrcBB), 1);
2443 Src->addSuccessorWithoutProb(Dst);
2446 Prob = getEdgeProbability(Src, Dst);
2447 Src->addSuccessor(Dst, Prob);
2453 return I->getParent() == BB;
2477 if (CurBB == SwitchBB ||
2483 InvertCond ? IC->getInversePredicate() : IC->getPredicate();
2488 InvertCond ? FC->getInversePredicate() : FC->getPredicate();
2490 if (FC->hasNoNaNs() ||
2498 CaseBlock CB(Condition, BOp->getOperand(0), BOp->getOperand(1),
nullptr,
2500 SL->SwitchCases.push_back(CB);
2509 SL->SwitchCases.push_back(CB);
2517 unsigned Depth = 0) {
2526 if (Necessary !=
nullptr) {
2529 if (Necessary->contains(
I))
2557 if (BPI !=
nullptr) {
2563 std::optional<bool> Likely;
2566 else if (BPI->
isEdgeHot(
I.getParent(), IfFalse))
2570 if (
Opc == (*Likely ? Instruction::And : Instruction::Or))
2582 if (CostThresh <= 0)
2603 Value *BrCond =
I.getCondition();
2604 auto ShouldCountInsn = [&RhsDeps, &BrCond](
const Instruction *Ins) {
2605 for (
const auto *U : Ins->users()) {
2608 if (UIns != BrCond && !RhsDeps.
contains(UIns))
2621 for (
unsigned PruneIters = 0; PruneIters < MaxPruneIters; ++PruneIters) {
2623 for (
const auto &InsPair : RhsDeps) {
2624 if (!ShouldCountInsn(InsPair.first)) {
2625 ToDrop = InsPair.first;
2629 if (ToDrop ==
nullptr)
2631 RhsDeps.erase(ToDrop);
2634 for (
const auto &InsPair : RhsDeps) {
2639 CostOfIncluding +=
TTI->getInstructionCost(
2642 if (CostOfIncluding > CostThresh)
2668 const Value *BOpOp0, *BOpOp1;
2682 if (BOpc == Instruction::And)
2683 BOpc = Instruction::Or;
2684 else if (BOpc == Instruction::Or)
2685 BOpc = Instruction::And;
2691 bool BOpIsInOrAndTree = BOpc && BOpc ==
Opc && BOp->
hasOneUse();
2696 TProb, FProb, InvertCond);
2706 if (
Opc == Instruction::Or) {
2727 auto NewTrueProb = TProb / 2;
2728 auto NewFalseProb = TProb / 2 + FProb;
2731 NewFalseProb, InvertCond);
2738 Probs[1], InvertCond);
2740 assert(
Opc == Instruction::And &&
"Unknown merge op!");
2760 auto NewTrueProb = TProb + FProb / 2;
2761 auto NewFalseProb = FProb / 2;
2764 NewFalseProb, InvertCond);
2771 Probs[1], InvertCond);
2780 if (Cases.size() != 2)
return true;
2784 if ((Cases[0].CmpLHS == Cases[1].CmpLHS &&
2785 Cases[0].CmpRHS == Cases[1].CmpRHS) ||
2786 (Cases[0].CmpRHS == Cases[1].CmpLHS &&
2787 Cases[0].CmpLHS == Cases[1].CmpRHS)) {
2793 if (Cases[0].CmpRHS == Cases[1].CmpRHS &&
2794 Cases[0].CC == Cases[1].CC &&
2797 if (Cases[0].CC ==
ISD::SETEQ && Cases[0].TrueBB == Cases[1].ThisBB)
2799 if (Cases[0].CC ==
ISD::SETNE && Cases[0].FalseBB == Cases[1].ThisBB)
2806void SelectionDAGBuilder::visitUncondBr(
const UncondBrInst &
I) {
2816 if (Succ0MBB != NextBlock(BrMBB) ||
2825void SelectionDAGBuilder::visitCondBr(
const CondBrInst &
I) {
2826 MachineBasicBlock *BrMBB =
FuncInfo.MBB;
2828 MachineBasicBlock *Succ0MBB =
FuncInfo.getMBB(
I.getSuccessor(0));
2832 const Value *CondVal =
I.getCondition();
2833 MachineBasicBlock *Succ1MBB =
FuncInfo.getMBB(
I.getSuccessor(1));
2852 bool IsUnpredictable =
I.hasMetadata(LLVMContext::MD_unpredictable);
2854 if (!
DAG.getTargetLoweringInfo().isJumpExpensive() && BOp &&
2857 const Value *BOp0, *BOp1;
2860 Opcode = Instruction::And;
2862 Opcode = Instruction::Or;
2869 DAG.getTargetLoweringInfo().getJumpConditionMergingParams(
2870 Opcode, BOp0, BOp1))) {
2872 getEdgeProbability(BrMBB, Succ0MBB),
2873 getEdgeProbability(BrMBB, Succ1MBB),
2878 assert(
SL->SwitchCases[0].ThisBB == BrMBB &&
"Unexpected lowering!");
2882 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i) {
2889 SL->SwitchCases.erase(
SL->SwitchCases.begin());
2895 for (
unsigned i = 1, e =
SL->SwitchCases.size(); i != e; ++i)
2896 FuncInfo.MF->erase(
SL->SwitchCases[i].ThisBB);
2898 SL->SwitchCases.clear();
2904 nullptr, Succ0MBB, Succ1MBB, BrMBB,
getCurSDLoc(),
2925 if (CB.
TrueBB != NextBlock(SwitchBB)) {
2932 auto &TLI =
DAG.getTargetLoweringInfo();
2956 Cond =
DAG.getSetCC(dl, MVT::i1, CondLHS, CondRHS, CB.
CC);
2968 Cond =
DAG.getSetCC(dl, MVT::i1, CmpOp,
DAG.getConstant(
High, dl, VT),
2972 VT, CmpOp,
DAG.getConstant(
Low, dl, VT));
2973 Cond =
DAG.getSetCC(dl, MVT::i1, SUB,
2988 if (CB.
TrueBB == NextBlock(SwitchBB)) {
3004 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
3007 DAG.setRoot(BrCond);
3013 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3014 assert(JT.
Reg &&
"Should lower JT Header first!");
3015 EVT PTy =
DAG.getTargetLoweringInfo().getJumpTableRegTy(
DAG.getDataLayout());
3019 Index.getValue(1), Table, Index);
3020 DAG.setRoot(BrJumpTable);
3028 assert(JT.
SL &&
"Should set SDLoc for SelectionDAG!");
3035 DAG.getConstant(JTH.
First, dl, VT));
3050 JT.
Reg = JumpTableReg;
3058 Sub.getValueType()),
3062 MVT::Other, CopyTo, CMP,
3066 if (JT.
MBB != NextBlock(SwitchBB))
3067 BrCond =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrCond,
3068 DAG.getBasicBlock(JT.
MBB));
3070 DAG.setRoot(BrCond);
3073 if (JT.
MBB != NextBlock(SwitchBB))
3075 DAG.getBasicBlock(JT.
MBB)));
3077 DAG.setRoot(CopyTo);
3101 if (PtrTy != PtrMemTy)
3117 auto &
DL =
DAG.getDataLayout();
3126 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3133 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3151 assert(GuardCheckFn &&
"Guard check function is null");
3162 Entry.IsInReg =
true;
3163 Args.push_back(Entry);
3169 getValue(GuardCheckFn), std::move(Args));
3171 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
3172 DAG.setRoot(Result.second);
3184 Guard =
DAG.getLoad(PtrMemTy, dl, Chain, GuardPtr,
3190 Guard =
DAG.getPOISON(PtrMemTy);
3199 Guard =
DAG.getLoad(PtrMemTy, dl, Chain, GuardPtr,
3205 Guard =
DAG.getPOISON(PtrMemTy);
3251 auto &
DL =
DAG.getDataLayout();
3259 SDValue StackSlotPtr =
DAG.getFrameIndex(FI, PtrTy);
3265 PtrMemTy, dl,
DAG.getEntryNode(), StackSlotPtr,
3280 if (GuardCheckFn->hasParamAttribute(0, Attribute::AttrKind::InReg))
3281 Entry.IsInReg =
true;
3282 Args.push_back(Entry);
3288 getValue(GuardCheckFn), std::move(Args));
3294 Chain = TLI.
makeLibCall(
DAG, RTLIB::STACKPROTECTOR_CHECK_FAIL, MVT::isVoid,
3317 DAG.getNode(
ISD::SUB, dl, VT, SwitchOp,
DAG.getConstant(
B.First, dl, VT));
3321 bool UsePtrType =
false;
3345 if (!
B.FallthroughUnreachable)
3346 addSuccessorWithProb(SwitchBB,
B.Default,
B.DefaultProb);
3347 addSuccessorWithProb(SwitchBB,
MBB,
B.Prob);
3351 if (!
B.FallthroughUnreachable) {
3360 DAG.getBasicBlock(
B.Default));
3364 if (
MBB != NextBlock(SwitchBB))
3382 if (PopCount == 1) {
3389 }
else if (PopCount == BB.
Range) {
3397 DAG.getConstant(1, dl, VT), ShiftOp);
3401 VT, SwitchVal,
DAG.getConstant(
B.Mask, dl, VT));
3408 addSuccessorWithProb(SwitchBB,
B.TargetBB,
B.ExtraProb);
3410 addSuccessorWithProb(SwitchBB, NextMBB, BranchProbToNext);
3418 Cmp,
DAG.getBasicBlock(
B.TargetBB));
3421 if (NextMBB != NextBlock(SwitchBB))
3422 BrAnd =
DAG.getNode(
ISD::BR, dl, MVT::Other, BrAnd,
3423 DAG.getBasicBlock(NextMBB));
3428void SelectionDAGBuilder::visitInvoke(
const InvokeInst &
I) {
3446 const Value *Callee(
I.getCalledOperand());
3449 visitInlineAsm(
I, EHPadBB);
3454 case Intrinsic::donothing:
3456 case Intrinsic::seh_try_begin:
3457 case Intrinsic::seh_scope_begin:
3458 case Intrinsic::seh_try_end:
3459 case Intrinsic::seh_scope_end:
3465 case Intrinsic::experimental_patchpoint_void:
3466 case Intrinsic::experimental_patchpoint:
3467 visitPatchpoint(
I, EHPadBB);
3469 case Intrinsic::experimental_gc_statepoint:
3475 case Intrinsic::wasm_throw: {
3477 std::array<SDValue, 4>
Ops = {
3488 case Intrinsic::wasm_rethrow: {
3489 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3490 std::array<SDValue, 2>
Ops = {
3499 }
else if (
I.hasDeoptState()) {
3520 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
3521 BranchProbability EHPadBBProb =
3527 addSuccessorWithProb(InvokeMBB, Return);
3528 for (
auto &UnwindDest : UnwindDests) {
3529 UnwindDest.first->setIsEHPad();
3530 addSuccessorWithProb(InvokeMBB, UnwindDest.first, UnwindDest.second);
3536 DAG.getBasicBlock(Return)));
3545void SelectionDAGBuilder::visitCallBrIntrinsic(
const CallBrInst &
I) {
3548 DAG.getTargetLoweringInfo().getTgtMemIntrinsic(
3549 Infos,
I,
DAG.getMachineFunction(),
I.getIntrinsicID());
3550 assert(Infos.
empty() &&
"Intrinsic touches memory");
3553 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
3556 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad);
3557 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
3561 getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
3562 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
3567void SelectionDAGBuilder::visitCallBr(
const CallBrInst &
I) {
3568 MachineBasicBlock *CallBrMBB =
FuncInfo.MBB;
3570 if (
I.isInlineAsm()) {
3577 assert(!
I.hasOperandBundles() &&
3578 "Can't have operand bundles for intrinsics");
3579 visitCallBrIntrinsic(
I);
3584 SmallPtrSet<BasicBlock *, 8> Dests;
3585 Dests.
insert(
I.getDefaultDest());
3595 if (
I.isInlineAsm()) {
3596 for (BasicBlock *Dest :
I.getIndirectDests()) {
3598 Target->setIsInlineAsmBrIndirectTarget();
3604 Target->setLabelMustBeEmitted();
3606 if (Dests.
insert(Dest).second)
3615 DAG.getBasicBlock(Return)));
3618void SelectionDAGBuilder::visitResume(
const ResumeInst &RI) {
3619 llvm_unreachable(
"SelectionDAGBuilder shouldn't visit resume instructions!");
3622void SelectionDAGBuilder::visitLandingPad(
const LandingPadInst &LP) {
3624 "Call to landingpad not in landing pad!");
3628 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
3644 assert(ValueVTs.
size() == 2 &&
"Only two-valued landingpads are supported");
3649 if (
FuncInfo.ExceptionPointerVirtReg) {
3650 Ops[0] =
DAG.getZExtOrTrunc(
3651 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3658 Ops[1] =
DAG.getZExtOrTrunc(
3659 DAG.getCopyFromReg(
DAG.getEntryNode(), dl,
3666 DAG.getVTList(ValueVTs),
Ops);
3674 if (JTB.first.HeaderBB ==
First)
3675 JTB.first.HeaderBB =
Last;
3688 for (
unsigned i = 0, e =
I.getNumSuccessors(); i != e; ++i) {
3690 bool Inserted =
Done.insert(BB).second;
3695 addSuccessorWithProb(IndirectBrMBB, Succ);
3705 if (!
I.shouldLowerToTrap(
DAG.getTarget().Options.TrapUnreachable,
3706 DAG.getTarget().Options.NoTrapAfterNoreturn))
3712void SelectionDAGBuilder::visitUnary(
const User &
I,
unsigned Opcode) {
3715 Flags.copyFMF(*FPOp);
3723void SelectionDAGBuilder::visitBinary(
const User &
I,
unsigned Opcode) {
3726 Flags.setNoSignedWrap(OFBinOp->hasNoSignedWrap());
3727 Flags.setNoUnsignedWrap(OFBinOp->hasNoUnsignedWrap());
3730 Flags.setExact(ExactOp->isExact());
3732 Flags.setDisjoint(DisjointOp->isDisjoint());
3734 Flags.copyFMF(*FPOp);
3743void SelectionDAGBuilder::visitShift(
const User &
I,
unsigned Opcode) {
3747 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
3752 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
3754 "Unexpected shift type");
3764 if (
const OverflowingBinaryOperator *OFBinOp =
3766 nuw = OFBinOp->hasNoUnsignedWrap();
3767 nsw = OFBinOp->hasNoSignedWrap();
3769 if (
const PossiblyExactOperator *ExactOp =
3771 exact = ExactOp->isExact();
3774 Flags.setExact(exact);
3775 Flags.setNoSignedWrap(nsw);
3776 Flags.setNoUnsignedWrap(nuw);
3782void SelectionDAGBuilder::visitSDiv(
const User &
I) {
3793void SelectionDAGBuilder::visitICmp(
const ICmpInst &
I) {
3799 auto &TLI =
DAG.getTargetLoweringInfo();
3812 Flags.setSameSign(
I.hasSameSign());
3814 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3820void SelectionDAGBuilder::visitFCmp(
const FCmpInst &
I) {
3827 if (FPMO->hasNoNaNs() ||
3828 (
DAG.isKnownNeverNaN(Op1) &&
DAG.isKnownNeverNaN(Op2)))
3832 Flags.copyFMF(*FPMO);
3834 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
3844 return isa<SelectInst>(V);
3848void SelectionDAGBuilder::visitSelect(
const User &
I) {
3852 unsigned NumValues = ValueVTs.
size();
3853 if (NumValues == 0)
return;
3863 bool IsUnaryAbs =
false;
3864 bool Negate =
false;
3868 Flags.copyFMF(*FPOp);
3870 Flags.setUnpredictable(
3875 EVT VT = ValueVTs[0];
3876 LLVMContext &Ctx = *
DAG.getContext();
3877 auto &TLI =
DAG.getTargetLoweringInfo();
3887 bool UseScalarMinMax = VT.
isVector() &&
3896 switch (SPR.Flavor) {
3905 switch (SPR.NaNBehavior) {
3910 Flags.setNoSignedZeros(
true);
3924 switch (SPR.NaNBehavior) {
3929 Flags.setNoSignedZeros(
true);
3971 for (
unsigned i = 0; i != NumValues; ++i) {
3980 for (
unsigned i = 0; i != NumValues; ++i) {
3994void SelectionDAGBuilder::visitTrunc(
const User &
I) {
3997 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4001 Flags.setNoSignedWrap(Trunc->hasNoSignedWrap());
4002 Flags.setNoUnsignedWrap(Trunc->hasNoUnsignedWrap());
4008void SelectionDAGBuilder::visitZExt(
const User &
I) {
4012 auto &TLI =
DAG.getTargetLoweringInfo();
4017 Flags.setNonNeg(PNI->hasNonNeg());
4022 if (
Flags.hasNonNeg() &&
4031void SelectionDAGBuilder::visitSExt(
const User &
I) {
4035 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4040void SelectionDAGBuilder::visitFPTrunc(
const User &
I) {
4046 Flags.copyFMF(*FPOp);
4047 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4050 DAG.getTargetConstant(
4055void SelectionDAGBuilder::visitFPExt(
const User &
I) {
4058 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4062 Flags.copyFMF(*FPOp);
4066void SelectionDAGBuilder::visitFPToUI(
const User &
I) {
4069 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4074void SelectionDAGBuilder::visitFPToSI(
const User &
I) {
4077 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4082void SelectionDAGBuilder::visitUIToFP(
const User &
I) {
4085 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4094void SelectionDAGBuilder::visitSIToFP(
const User &
I) {
4097 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4105void SelectionDAGBuilder::visitPtrToAddr(
const User &
I) {
4108 const auto &TLI =
DAG.getTargetLoweringInfo();
4116void SelectionDAGBuilder::visitPtrToInt(
const User &
I) {
4120 auto &TLI =
DAG.getTargetLoweringInfo();
4121 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4130void SelectionDAGBuilder::visitIntToPtr(
const User &
I) {
4134 auto &TLI =
DAG.getTargetLoweringInfo();
4142void SelectionDAGBuilder::visitBitCast(
const User &
I) {
4145 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
4150 if (DestVT !=
N.getValueType())
4158 setValue(&
I,
DAG.getConstant(
C->getValue(), dl, DestVT,
false,
4164void SelectionDAGBuilder::visitAddrSpaceCast(
const User &
I) {
4165 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4166 const Value *SV =
I.getOperand(0);
4171 unsigned DestAS =
I.getType()->getPointerAddressSpace();
4173 if (!TM.isNoopAddrSpaceCast(SrcAS, DestAS))
4179void SelectionDAGBuilder::visitInsertElement(
const User &
I) {
4180 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4187 InVec, InVal, InIdx));
4190void SelectionDAGBuilder::visitExtractElement(
const User &
I) {
4191 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4200void SelectionDAGBuilder::visitShuffleVector(
const User &
I) {
4205 Mask = SVI->getShuffleMask();
4209 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4217 DAG.getVectorIdxConstant(0,
DL));
4228 unsigned MaskNumElts =
Mask.size();
4230 if (SrcNumElts == MaskNumElts) {
4236 if (SrcNumElts < MaskNumElts) {
4240 if (MaskNumElts % SrcNumElts == 0) {
4244 unsigned NumConcat = MaskNumElts / SrcNumElts;
4245 bool IsConcat =
true;
4246 SmallVector<int, 8> ConcatSrcs(NumConcat, -1);
4247 for (
unsigned i = 0; i != MaskNumElts; ++i) {
4253 if ((Idx % SrcNumElts != (i % SrcNumElts)) ||
4254 (ConcatSrcs[i / SrcNumElts] >= 0 &&
4255 ConcatSrcs[i / SrcNumElts] != (
int)(Idx / SrcNumElts))) {
4260 ConcatSrcs[i / SrcNumElts] = Idx / SrcNumElts;
4267 for (
auto Src : ConcatSrcs) {
4280 unsigned PaddedMaskNumElts =
alignTo(MaskNumElts, SrcNumElts);
4281 unsigned NumConcat = PaddedMaskNumElts / SrcNumElts;
4297 SmallVector<int, 8> MappedOps(PaddedMaskNumElts, -1);
4298 for (
unsigned i = 0; i != MaskNumElts; ++i) {
4300 if (Idx >= (
int)SrcNumElts)
4301 Idx -= SrcNumElts - PaddedMaskNumElts;
4309 if (MaskNumElts != PaddedMaskNumElts)
4311 DAG.getVectorIdxConstant(0,
DL));
4317 assert(SrcNumElts > MaskNumElts);
4321 int StartIdx[2] = {-1, -1};
4322 bool CanExtract =
true;
4323 for (
int Idx : Mask) {
4328 if (Idx >= (
int)SrcNumElts) {
4336 int NewStartIdx =
alignDown(Idx, MaskNumElts);
4337 if (NewStartIdx + MaskNumElts > SrcNumElts ||
4338 (StartIdx[Input] >= 0 && StartIdx[Input] != NewStartIdx))
4342 StartIdx[Input] = NewStartIdx;
4345 if (StartIdx[0] < 0 && StartIdx[1] < 0) {
4351 for (
unsigned Input = 0; Input < 2; ++Input) {
4352 SDValue &Src = Input == 0 ? Src1 : Src2;
4353 if (StartIdx[Input] < 0)
4354 Src =
DAG.getUNDEF(VT);
4357 DAG.getVectorIdxConstant(StartIdx[Input],
DL));
4362 SmallVector<int, 8> MappedOps(Mask);
4363 for (
int &Idx : MappedOps) {
4364 if (Idx >= (
int)SrcNumElts)
4365 Idx -= SrcNumElts + StartIdx[1] - MaskNumElts;
4370 setValue(&
I,
DAG.getVectorShuffle(VT,
DL, Src1, Src2, MappedOps));
4379 for (
int Idx : Mask) {
4383 Res =
DAG.getUNDEF(EltVT);
4385 SDValue &Src = Idx < (int)SrcNumElts ? Src1 : Src2;
4386 if (Idx >= (
int)SrcNumElts) Idx -= SrcNumElts;
4389 DAG.getVectorIdxConstant(Idx,
DL));
4399 ArrayRef<unsigned> Indices =
I.getIndices();
4400 const Value *Op0 =
I.getOperand(0);
4402 Type *AggTy =
I.getType();
4409 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4415 unsigned NumAggValues = AggValueVTs.
size();
4416 unsigned NumValValues = ValValueVTs.
size();
4420 if (!NumAggValues) {
4428 for (; i != LinearIndex; ++i)
4429 Values[i] = IntoUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4434 for (; i != LinearIndex + NumValValues; ++i)
4435 Values[i] = FromUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4439 for (; i != NumAggValues; ++i)
4440 Values[i] = IntoUndef ?
DAG.getUNDEF(AggValueVTs[i]) :
4448 ArrayRef<unsigned> Indices =
I.getIndices();
4449 const Value *Op0 =
I.getOperand(0);
4451 Type *ValTy =
I.getType();
4456 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4460 unsigned NumValValues = ValValueVTs.
size();
4463 if (!NumValValues) {
4472 for (
unsigned i = LinearIndex; i != LinearIndex + NumValValues; ++i)
4473 Values[i - LinearIndex] =
4482void SelectionDAGBuilder::visitGetElementPtr(
const User &
I) {
4483 Value *Op0 =
I.getOperand(0);
4489 auto &TLI =
DAG.getTargetLoweringInfo();
4494 bool IsVectorGEP =
I.getType()->isVectorTy();
4495 ElementCount VectorElementCount =
4501 const Value *Idx = GTI.getOperand();
4502 if (StructType *StTy = GTI.getStructTypeOrNull()) {
4507 DAG.getDataLayout().getStructLayout(StTy)->getElementOffset(
Field);
4517 N =
DAG.getMemBasePlusOffset(
4518 N,
DAG.getConstant(
Offset, dl,
N.getValueType()), dl, Flags);
4524 unsigned IdxSize =
DAG.getDataLayout().getIndexSizeInBits(AS);
4526 TypeSize ElementSize =
4527 GTI.getSequentialElementStride(
DAG.getDataLayout());
4532 bool ElementScalable = ElementSize.
isScalable();
4538 C =
C->getSplatValue();
4541 if (CI && CI->isZero())
4543 if (CI && !ElementScalable) {
4544 APInt Offs = ElementMul * CI->getValue().sextOrTrunc(IdxSize);
4547 if (
N.getValueType().isVector())
4548 OffsVal =
DAG.getConstant(
4551 OffsVal =
DAG.getConstant(Offs, dl, IdxTy);
4558 Flags.setNoUnsignedWrap(
true);
4561 OffsVal =
DAG.getSExtOrTrunc(OffsVal, dl,
N.getValueType());
4563 N =
DAG.getMemBasePlusOffset(
N, OffsVal, dl, Flags);
4571 if (
N.getValueType().isVector()) {
4573 VectorElementCount);
4574 IdxN =
DAG.getSplat(VT, dl, IdxN);
4578 N =
DAG.getSplat(VT, dl,
N);
4584 IdxN =
DAG.getSExtOrTrunc(IdxN, dl,
N.getValueType());
4586 SDNodeFlags ScaleFlags;
4595 if (ElementScalable) {
4596 EVT VScaleTy =
N.getValueType().getScalarType();
4599 DAG.getConstant(ElementMul.getZExtValue(), dl, VScaleTy));
4600 if (
N.getValueType().isVector())
4601 VScale =
DAG.getSplatVector(
N.getValueType(), dl, VScale);
4602 IdxN =
DAG.getNode(
ISD::MUL, dl,
N.getValueType(), IdxN, VScale,
4607 if (ElementMul != 1) {
4608 if (ElementMul.isPowerOf2()) {
4609 unsigned Amt = ElementMul.logBase2();
4612 DAG.getShiftAmountConstant(Amt,
N.getValueType(), dl),
4615 SDValue Scale =
DAG.getConstant(ElementMul.getZExtValue(), dl,
4617 IdxN =
DAG.getNode(
ISD::MUL, dl,
N.getValueType(), IdxN, Scale,
4627 SDNodeFlags AddFlags;
4631 N =
DAG.getMemBasePlusOffset(
N, IdxN, dl, AddFlags);
4635 if (IsVectorGEP && !
N.getValueType().isVector()) {
4637 N =
DAG.getSplat(VT, dl,
N);
4648 N =
DAG.getPtrExtendInReg(
N, dl, PtrMemTy);
4653void SelectionDAGBuilder::visitAlloca(
const AllocaInst &
I) {
4660 Type *Ty =
I.getAllocatedType();
4661 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4662 auto &
DL =
DAG.getDataLayout();
4663 TypeSize TySize =
DL.getTypeAllocSize(Ty);
4664 MaybeAlign Alignment =
I.getAlign();
4670 AllocSize =
DAG.getZExtOrTrunc(AllocSize, dl, IntPtr);
4672 AllocSize =
DAG.getNode(
4674 DAG.getZExtOrTrunc(
DAG.getTypeSize(dl, MVT::i64, TySize), dl, IntPtr));
4679 Align StackAlign =
DAG.getSubtarget().getFrameLowering()->getStackAlign();
4680 if (*Alignment <= StackAlign)
4681 Alignment = std::nullopt;
4683 const uint64_t StackAlignMask = StackAlign.
value() - 1U;
4688 DAG.getConstant(StackAlignMask, dl, IntPtr),
4693 DAG.getSignedConstant(~StackAlignMask, dl, IntPtr));
4697 DAG.getConstant(Alignment ? Alignment->value() : 0, dl, IntPtr)};
4707 return I.getMetadata(LLVMContext::MD_range);
4712 if (std::optional<ConstantRange> CR = CB->getRange())
4716 return std::nullopt;
4721 return CB->getRetNoFPClass();
4725void SelectionDAGBuilder::visitLoad(
const LoadInst &
I) {
4727 return visitAtomicLoad(
I);
4729 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4730 const Value *SV =
I.getOperand(0);
4735 if (Arg->hasSwiftErrorAttr())
4736 return visitLoadFromSwiftError(
I);
4740 if (Alloca->isSwiftError())
4741 return visitLoadFromSwiftError(
I);
4747 Type *Ty =
I.getType();
4751 unsigned NumValues = ValueVTs.
size();
4755 Align Alignment =
I.getAlign();
4756 AAMDNodes AAInfo =
I.getAAMetadata();
4758 bool isVolatile =
I.isVolatile();
4763 bool ConstantMemory =
false;
4770 BatchAA->pointsToConstantMemory(MemoryLocation(
4775 Root =
DAG.getEntryNode();
4776 ConstantMemory =
true;
4780 Root =
DAG.getRoot();
4791 unsigned ChainI = 0;
4792 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4808 MachinePointerInfo PtrInfo =
4810 ? MachinePointerInfo(SV, Offsets[i].getKnownMinValue())
4811 : MachinePointerInfo();
4813 SDValue A =
DAG.getObjectPtrOffset(dl, Ptr, Offsets[i]);
4814 SDValue L =
DAG.getLoad(MemVTs[i], dl, Root,
A, PtrInfo, Alignment,
4815 MMOFlags, AAInfo, Ranges);
4816 Chains[ChainI] =
L.getValue(1);
4818 if (MemVTs[i] != ValueVTs[i])
4819 L =
DAG.getPtrExtOrTrunc(L, dl, ValueVTs[i]);
4821 if (MDNode *NoFPClassMD =
I.getMetadata(LLVMContext::MD_nofpclass)) {
4822 uint64_t FPTestInt =
4824 cast<ConstantAsMetadata>(NoFPClassMD->getOperand(0))->getValue())
4826 if (FPTestInt != fcNone) {
4827 SDValue FPTestConst =
4828 DAG.getTargetConstant(FPTestInt, SDLoc(), MVT::i32);
4829 L = DAG.getNode(ISD::AssertNoFPClass, dl, L.getValueType(), L,
4836 if (!ConstantMemory) {
4842 PendingLoads.push_back(Chain);
4846 DAG.getVTList(ValueVTs),
Values));
4849void SelectionDAGBuilder::visitStoreToSwiftError(
const StoreInst &
I) {
4850 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4851 "call visitStoreToSwiftError when backend supports swifterror");
4854 SmallVector<uint64_t, 4>
Offsets;
4855 const Value *SrcV =
I.getOperand(0);
4857 SrcV->
getType(), ValueVTs,
nullptr, &Offsets, 0);
4858 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4859 "expect a single EVT for swifterror");
4868 SDValue(Src.getNode(), Src.getResNo()));
4869 DAG.setRoot(CopyNode);
4872void SelectionDAGBuilder::visitLoadFromSwiftError(
const LoadInst &
I) {
4873 assert(
DAG.getTargetLoweringInfo().supportSwiftError() &&
4874 "call visitLoadFromSwiftError when backend supports swifterror");
4877 !
I.hasMetadata(LLVMContext::MD_nontemporal) &&
4878 !
I.hasMetadata(LLVMContext::MD_invariant_load) &&
4879 "Support volatile, non temporal, invariant for load_from_swift_error");
4881 const Value *SV =
I.getOperand(0);
4882 Type *Ty =
I.getType();
4885 !
BatchAA->pointsToConstantMemory(MemoryLocation(
4887 I.getAAMetadata()))) &&
4888 "load_from_swift_error should not be constant memory");
4891 SmallVector<uint64_t, 4>
Offsets;
4893 ValueVTs,
nullptr, &Offsets, 0);
4894 assert(ValueVTs.
size() == 1 && Offsets[0] == 0 &&
4895 "expect a single EVT for swifterror");
4905void SelectionDAGBuilder::visitStore(
const StoreInst &
I) {
4907 return visitAtomicStore(
I);
4909 const Value *SrcV =
I.getOperand(0);
4910 const Value *PtrV =
I.getOperand(1);
4912 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
4917 if (Arg->hasSwiftErrorAttr())
4918 return visitStoreToSwiftError(
I);
4922 if (Alloca->isSwiftError())
4923 return visitStoreToSwiftError(
I);
4930 SrcV->
getType(), ValueVTs, &MemVTs, &Offsets);
4931 unsigned NumValues = ValueVTs.
size();
4944 Align Alignment =
I.getAlign();
4945 AAMDNodes AAInfo =
I.getAAMetadata();
4949 unsigned ChainI = 0;
4950 for (
unsigned i = 0; i != NumValues; ++i, ++ChainI) {
4960 MachinePointerInfo PtrInfo =
4962 ? MachinePointerInfo(PtrV, Offsets[i].getKnownMinValue())
4963 : MachinePointerInfo();
4967 if (MemVTs[i] != ValueVTs[i])
4968 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVTs[i]);
4970 DAG.getStore(Root, dl, Val,
Add, PtrInfo, Alignment, MMOFlags, AAInfo);
4971 Chains[ChainI] = St;
4977 DAG.setRoot(StoreNode);
4980void SelectionDAGBuilder::visitMaskedStore(
const CallInst &
I,
4981 bool IsCompressing) {
4984 Value *Src0Operand =
I.getArgOperand(0);
4985 Value *PtrOperand =
I.getArgOperand(1);
4986 Value *MaskOperand =
I.getArgOperand(2);
4987 Align Alignment =
I.getParamAlign(1).valueOrOne();
4997 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
5000 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5001 MachinePointerInfo(PtrOperand), MMOFlags,
5005 const auto &TLI =
DAG.getTargetLoweringInfo();
5008 !IsCompressing &&
TTI->hasConditionalLoadStoreForType(
5009 I.getArgOperand(0)->getType(),
true)
5015 DAG.setRoot(StoreNode);
5045 C =
C->getSplatValue();
5059 if (!
GEP ||
GEP->getParent() != CurBB)
5062 if (
GEP->getNumOperands() != 2)
5065 const Value *BasePtr =
GEP->getPointerOperand();
5066 const Value *IndexVal =
GEP->getOperand(
GEP->getNumOperands() - 1);
5072 TypeSize ScaleVal =
DL.getTypeAllocSize(
GEP->getResultElementType());
5077 if (ScaleVal != 1 &&
5089void SelectionDAGBuilder::visitMaskedScatter(
const CallInst &
I) {
5093 const Value *Ptr =
I.getArgOperand(1);
5097 Align Alignment =
I.getParamAlign(1).valueOrOne();
5098 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5107 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5117 EVT IdxVT =
Index.getValueType();
5125 SDValue Scatter =
DAG.getMaskedScatter(
DAG.getVTList(MVT::Other), VT, sdl,
5127 DAG.setRoot(Scatter);
5131void SelectionDAGBuilder::visitMaskedLoad(
const CallInst &
I,
bool IsExpanding) {
5134 Value *PtrOperand =
I.getArgOperand(0);
5135 Value *MaskOperand =
I.getArgOperand(1);
5136 Value *Src0Operand =
I.getArgOperand(2);
5137 Align Alignment =
I.getParamAlign(0).valueOrOne();
5145 AAMDNodes AAInfo =
I.getAAMetadata();
5152 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
5155 if (
I.hasMetadata(LLVMContext::MD_nontemporal))
5157 if (
I.hasMetadata(LLVMContext::MD_invariant_load))
5160 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5161 MachinePointerInfo(PtrOperand), MMOFlags,
5164 const auto &TLI =
DAG.getTargetLoweringInfo();
5171 TTI->hasConditionalLoadStoreForType(Src0Operand->
getType(),
5176 DAG.getMaskedLoad(VT, sdl, InChain, Ptr,
Offset, Mask, Src0, VT, MMO,
5183void SelectionDAGBuilder::visitMaskedGather(
const CallInst &
I) {
5187 const Value *Ptr =
I.getArgOperand(0);
5191 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5193 Align Alignment =
I.getParamAlign(0).valueOrOne();
5204 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5216 EVT IdxVT =
Index.getValueType();
5225 DAG.getMaskedGather(
DAG.getVTList(VT, MVT::Other), VT, sdl,
Ops, MMO,
5241 SDVTList VTs =
DAG.getVTList(MemVT, MVT::i1, MVT::Other);
5243 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5246 MachineFunction &MF =
DAG.getMachineFunction();
5247 MachineMemOperand *MMO =
5250 nullptr, SSID, SuccessOrdering, FailureOrdering);
5253 dl, MemVT, VTs, InChain,
5261 DAG.setRoot(OutChain);
5264void SelectionDAGBuilder::visitAtomicRMW(
const AtomicRMWInst &
I) {
5267 switch (
I.getOperation()) {
5315 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5318 MachineFunction &MF =
DAG.getMachineFunction();
5320 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5321 I.getAlign(), AAMDNodes(),
nullptr, SSID, Ordering);
5324 DAG.getAtomic(NT, dl, MemVT, InChain,
5331 DAG.setRoot(OutChain);
5334void SelectionDAGBuilder::visitFence(
const FenceInst &
I) {
5336 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5339 Ops[1] =
DAG.getTargetConstant((
unsigned)
I.getOrdering(), dl,
5341 Ops[2] =
DAG.getTargetConstant(
I.getSyncScopeID(), dl,
5348void SelectionDAGBuilder::visitAtomicLoad(
const LoadInst &
I) {
5355 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5366 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
5367 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5368 I.getAlign(), AAMDNodes(), Ranges, SSID, Order);
5378 L =
DAG.getPtrExtOrTrunc(L, dl, VT);
5381 DAG.setRoot(OutChain);
5384void SelectionDAGBuilder::visitAtomicStore(
const StoreInst &
I) {
5392 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5402 MachineFunction &MF =
DAG.getMachineFunction();
5404 MachinePointerInfo(
I.getPointerOperand()), Flags, MemVT.
getStoreSize(),
5405 I.getAlign(), AAMDNodes(),
nullptr, SSID, Ordering);
5409 Val =
DAG.getPtrExtOrTrunc(Val, dl, MemVT);
5416 DAG.setRoot(OutChain);
5424std::pair<bool, bool>
5425SelectionDAGBuilder::getTargetIntrinsicCallProperties(
const CallBase &
I) {
5427 bool HasChain = !
F->doesNotAccessMemory();
5429 HasChain &&
F->onlyReadsMemory() &&
F->willReturn() &&
F->doesNotThrow();
5431 return {HasChain, OnlyLoad};
5435 const CallBase &
I,
bool HasChain,
bool OnlyLoad,
5437 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5444 Ops.push_back(
DAG.getRoot());
5457 for (
unsigned i = 0, e =
I.arg_size(); i != e; ++i) {
5458 const Value *Arg =
I.getArgOperand(i);
5459 if (!
I.paramHasAttr(i, Attribute::ImmArg)) {
5467 assert(CI->getBitWidth() <= 64 &&
5468 "large intrinsic immediates not handled");
5469 Ops.push_back(
DAG.getTargetConstant(*CI, SDLoc(), VT));
5476 if (std::optional<OperandBundleUse> Bundle =
5478 auto *Sym = Bundle->Inputs[0].get();
5481 Ops.push_back(SDSym);
5484 if (std::optional<OperandBundleUse> Bundle =
5486 Value *Token = Bundle->Inputs[0].get();
5488 assert(
Ops.back().getValueType() != MVT::Glue &&
5489 "Did not expect another glue node here.");
5492 Ops.push_back(ConvControlToken);
5500 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5508 return DAG.getVTList(ValueVTs);
5512SDValue SelectionDAGBuilder::getTargetNonMemIntrinsicNode(
5535 if (
I.getType()->isVoidTy())
5550void SelectionDAGBuilder::visitTargetIntrinsic(
const CallInst &
I,
5552 auto [HasChain, OnlyLoad] = getTargetIntrinsicCallProperties(
I);
5555 if (!
DAG.getMachineFunction().getSubtarget().isIntrinsicSupported(
5558 DAG.getContext()->diagnose(DiagnosticInfoUnsupportedTargetIntrinsic(
5559 *
I.getFunction(), IntrinsicID,
DL.getDebugLoc()));
5564 if (HasChain && !OnlyLoad)
5567 if (!
I.getType()->isVoidTy()) {
5570 I.getType(), ValueVTs);
5572 for (EVT VT : ValueVTs)
5581 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
5584 TargetLowering::IntrinsicInfo *
Info = !Infos.
empty() ? &Infos[0] :
nullptr;
5587 getTargetIntrinsicOperands(
I, HasChain, OnlyLoad, Info);
5588 SDVTList VTs = getTargetIntrinsicVTList(
I, HasChain);
5593 Flags.copyFMF(*FPMO);
5594 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
5601 if (!Infos.
empty()) {
5604 MachineFunction &MF =
DAG.getMachineFunction();
5606 for (
const auto &Info : Infos) {
5609 MachinePointerInfo MPI;
5611 MPI = MachinePointerInfo(
Info.ptrVal,
Info.offset);
5612 else if (
Info.fallbackAddressSpace)
5613 MPI = MachinePointerInfo(*
Info.fallbackAddressSpace);
5614 EVT MemVT =
Info.memVT;
5616 if (
Size.hasValue() && !
Size.getValue())
5618 Align Alignment =
Info.align.value_or(
DAG.getEVTAlign(MemVT));
5620 MPI,
Info.flags,
Size, Alignment,
I.getAAMetadata(),
5628 Result = getTargetNonMemIntrinsicNode(*
I.getType(), HasChain,
Ops, VTs);
5631 Result = handleTargetIntrinsicRet(
I, HasChain, OnlyLoad, Result);
5688 SDValue TwoToFractionalPartOfX;
5765 if (
Op.getValueType() == MVT::f32 &&
5789 if (
Op.getValueType() == MVT::f32 &&
5888 if (
Op.getValueType() == MVT::f32 &&
5972 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log2ofMantissa);
5985 if (
Op.getValueType() == MVT::f32 &&
6062 return DAG.
getNode(
ISD::FADD, dl, MVT::f32, LogOfExponent, Log10ofMantissa);
6073 if (
Op.getValueType() == MVT::f32 &&
6086 bool IsExp10 =
false;
6087 if (
LHS.getValueType() == MVT::f32 &&
RHS.getValueType() == MVT::f32 &&
6091 IsExp10 = LHSC->isExactlyValue(Ten);
6118 unsigned Val = RHSC->getSExtValue();
6147 CurSquare, CurSquare);
6152 if (RHSC->getSExtValue() < 0)
6166 EVT VT =
LHS.getValueType();
6189 if ((ScaleInt > 0 || (Saturating &&
Signed)) &&
6193 Opcode, VT, ScaleInt);
6228 switch (
N.getOpcode()) {
6232 Op.getValueType().getSizeInBits());
6257bool SelectionDAGBuilder::EmitFuncArgumentDbgValue(
6264 MachineFunction &MF =
DAG.getMachineFunction();
6265 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
6269 auto MakeVRegDbgValue = [&](
Register Reg, DIExpression *FragExpr,
6274 auto &Inst =
TII->get(TargetOpcode::DBG_INSTR_REF);
6281 auto *NewDIExpr = FragExpr;
6288 return BuildMI(MF,
DL, Inst,
false, MOs, Variable, NewDIExpr);
6291 auto &Inst =
TII->get(TargetOpcode::DBG_VALUE);
6292 return BuildMI(MF,
DL, Inst, Indirect,
Reg, Variable, FragExpr);
6296 if (Kind == FuncArgumentDbgValueKind::Value) {
6301 if (!IsInEntryBlock)
6317 bool VariableIsFunctionInputArg =
Variable->isParameter() &&
6318 !
DL->getInlinedAt();
6320 if (!IsInPrologue && !VariableIsFunctionInputArg)
6354 if (VariableIsFunctionInputArg) {
6356 if (ArgNo >=
FuncInfo.DescribedArgs.size())
6357 FuncInfo.DescribedArgs.resize(ArgNo + 1,
false);
6358 else if (!IsInPrologue &&
FuncInfo.DescribedArgs.test(ArgNo))
6359 return !NodeMap[
V].getNode();
6364 bool IsIndirect =
false;
6365 std::optional<MachineOperand>
Op;
6367 int FI =
FuncInfo.getArgumentFrameIndex(Arg);
6368 if (FI != std::numeric_limits<int>::max())
6372 if (!
Op &&
N.getNode()) {
6375 if (ArgRegsAndSizes.
size() == 1)
6376 Reg = ArgRegsAndSizes.
front().first;
6379 MachineRegisterInfo &RegInfo = MF.
getRegInfo();
6386 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6390 if (!
Op &&
N.getNode()) {
6394 if (FrameIndexSDNode *FINode =
6401 auto splitMultiRegDbgValue =
6414 uint64_t ExprFragmentSizeInBits = ExprFragmentInfo->SizeInBits;
6417 if (
Offset >= ExprFragmentSizeInBits)
6421 if (
Offset + RegFragmentSizeInBits > ExprFragmentSizeInBits) {
6422 RegFragmentSizeInBits = ExprFragmentSizeInBits -
Offset;
6427 Expr,
Offset, RegFragmentSizeInBits);
6431 if (!FragmentExpr) {
6432 SDDbgValue *SDV =
DAG.getConstantDbgValue(
6434 DAG.AddDbgValue(SDV,
false);
6437 MachineInstr *NewMI = MakeVRegDbgValue(
6438 Reg, *FragmentExpr, Kind != FuncArgumentDbgValueKind::Value);
6439 FuncInfo.ArgDbgValues.push_back(NewMI);
6448 if (VMI !=
FuncInfo.ValueMap.end()) {
6449 const auto &TLI =
DAG.getTargetLoweringInfo();
6450 RegsForValue RFV(
V->getContext(), TLI,
DAG.getDataLayout(), VMI->second,
6451 V->getType(), std::nullopt);
6452 if (RFV.occupiesMultipleRegs())
6453 return splitMultiRegDbgValue(RFV.getRegsAndSizes());
6456 IsIndirect =
Kind != FuncArgumentDbgValueKind::Value;
6457 }
else if (ArgRegsAndSizes.
size() > 1) {
6460 return splitMultiRegDbgValue(ArgRegsAndSizes);
6468 "Expected inlined-at fields to agree");
6469 MachineInstr *NewMI =
nullptr;
6472 NewMI = MakeVRegDbgValue(
Op->getReg(), Expr, IsIndirect);
6474 NewMI =
BuildMI(MF,
DL,
TII->get(TargetOpcode::DBG_VALUE),
true, *
Op,
6478 FuncInfo.ArgDbgValues.push_back(NewMI);
6487 unsigned DbgSDNodeOrder) {
6499 return DAG.getFrameIndexDbgValue(Variable, Expr, FISDN->getIndex(),
6500 false, dl, DbgSDNodeOrder);
6502 return DAG.getDbgValue(Variable, Expr,
N.getNode(),
N.getResNo(),
6503 false, dl, DbgSDNodeOrder);
6508 case Intrinsic::smul_fix:
6510 case Intrinsic::umul_fix:
6512 case Intrinsic::smul_fix_sat:
6514 case Intrinsic::umul_fix_sat:
6516 case Intrinsic::sdiv_fix:
6518 case Intrinsic::udiv_fix:
6520 case Intrinsic::sdiv_fix_sat:
6522 case Intrinsic::udiv_fix_sat:
6535 "expected call_preallocated_setup Value");
6536 for (
const auto *U : PreallocatedSetup->
users()) {
6538 const Function *Fn = UseCall->getCalledFunction();
6539 if (!Fn || Fn->
getIntrinsicID() != Intrinsic::call_preallocated_arg) {
6549bool SelectionDAGBuilder::visitEntryValueDbgValue(
6559 auto ArgIt =
FuncInfo.ValueMap.find(Arg);
6560 if (ArgIt ==
FuncInfo.ValueMap.end()) {
6562 dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6563 "couldn't find an associated register for the Argument\n");
6566 Register ArgVReg = ArgIt->getSecond();
6568 for (
auto [PhysReg, VirtReg] :
FuncInfo.RegInfo->liveins())
6569 if (ArgVReg == VirtReg || ArgVReg == PhysReg) {
6570 SDDbgValue *SDV =
DAG.getVRegDbgValue(
6571 Variable, Expr, PhysReg,
false , DbgLoc, SDNodeOrder);
6572 DAG.AddDbgValue(SDV,
false );
6575 LLVM_DEBUG(
dbgs() <<
"Dropping dbg.value: expression is entry_value but "
6576 "couldn't find a physical register\n");
6581void SelectionDAGBuilder::visitConvergenceControl(
const CallInst &
I,
6584 switch (Intrinsic) {
6585 case Intrinsic::experimental_convergence_anchor:
6588 case Intrinsic::experimental_convergence_entry:
6591 case Intrinsic::experimental_convergence_loop: {
6593 auto *Token = Bundle->Inputs[0].get();
6601void SelectionDAGBuilder::visitVectorHistogram(
const CallInst &
I,
6602 unsigned IntrinsicID) {
6605 assert(IntrinsicID == Intrinsic::experimental_vector_histogram_add &&
6606 "Tried to lower unsupported histogram type");
6612 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6613 DataLayout TargetDL =
DAG.getDataLayout();
6615 Align Alignment =
DAG.getEVTAlign(VT);
6628 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
6629 MachinePointerInfo(AS),
6640 EVT IdxVT =
Index.getValueType();
6651 SDValue ID =
DAG.getTargetConstant(IntrinsicID, sdl, MVT::i32);
6654 SDValue Histogram =
DAG.getMaskedHistogram(
DAG.getVTList(MVT::Other), VT, sdl,
6658 DAG.setRoot(Histogram);
6661void SelectionDAGBuilder::visitVectorExtractLastActive(
const CallInst &
I,
6663 assert(Intrinsic == Intrinsic::experimental_vector_extract_last_active &&
6664 "Tried lowering invalid vector extract last");
6666 const DataLayout &Layout =
DAG.getDataLayout();
6670 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6680 EVT BoolVT =
Mask.getValueType().getScalarType();
6682 Result =
DAG.getSelect(sdl, ResVT, AnyActive, Result, PassThru);
6689void SelectionDAGBuilder::visitIntrinsicCall(
const CallInst &
I,
6691 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
6698 Flags.copyFMF(*FPOp);
6700 switch (Intrinsic) {
6703 visitTargetIntrinsic(
I, Intrinsic);
6705 case Intrinsic::vscale: {
6710 case Intrinsic::vastart: visitVAStart(
I);
return;
6711 case Intrinsic::vaend: visitVAEnd(
I);
return;
6712 case Intrinsic::vacopy: visitVACopy(
I);
return;
6713 case Intrinsic::returnaddress:
6718 case Intrinsic::addressofreturnaddress:
6723 case Intrinsic::sponentry:
6728 case Intrinsic::frameaddress:
6733 case Intrinsic::read_volatile_register:
6734 case Intrinsic::read_register: {
6735 Value *
Reg =
I.getArgOperand(0);
6741 DAG.getVTList(VT, MVT::Other), Chain,
RegName);
6746 case Intrinsic::write_register: {
6747 Value *
Reg =
I.getArgOperand(0);
6748 Value *RegValue =
I.getArgOperand(1);
6756 case Intrinsic::write_volatile_register: {
6757 Value *
Reg =
I.getArgOperand(0);
6758 Value *RegValue =
I.getArgOperand(1);
6771 const MachineFunction &MF =
DAG.getMachineFunction();
6775 const TargetRegisterInfo *TRI = MF.getSubtarget().getRegisterInfo();
6776 const TargetRegisterClass *RC = TRI->getMinimalPhysRegClass(PhysReg);
6777 MVT RegVT = *TRI->legalclasstypes_begin(*RC);
6778 DAG.setRoot(DAG.getNode(ISD::FAKE_USE, sdl, MVT::Other,
6779 {WriteChain, DAG.getRegister(PhysReg, RegVT)}));
6781 DAG.setRoot(WriteChain);
6785 case Intrinsic::memcpy:
6786 case Intrinsic::memcpy_inline: {
6792 "memcpy_inline needs constant size");
6794 Align DstAlign = MCI.getDestAlign().valueOrOne();
6795 Align SrcAlign = MCI.getSourceAlign().valueOrOne();
6796 bool isVol = MCI.isVolatile();
6798 SDValue MC =
DAG.getMemcpy(Root, sdl, Dst, Src,
Size, DstAlign, SrcAlign,
6799 isVol, MCI.isForceInlined(), &
I, std::nullopt,
6800 MachinePointerInfo(
I.getArgOperand(0)),
6801 MachinePointerInfo(
I.getArgOperand(1)),
6803 updateDAGForMaybeTailCall(MC);
6806 case Intrinsic::memset:
6807 case Intrinsic::memset_inline: {
6813 "memset_inline needs constant size");
6815 Align DstAlign = MSII.getDestAlign().valueOrOne();
6816 bool isVol = MSII.isVolatile();
6819 Root, sdl, Dst, Value,
Size, DstAlign, isVol, MSII.isForceInlined(),
6820 &
I, MachinePointerInfo(
I.getArgOperand(0)),
I.getAAMetadata());
6821 updateDAGForMaybeTailCall(MC);
6824 case Intrinsic::memmove: {
6830 Align DstAlign = MMI.getDestAlign().valueOrOne();
6831 Align SrcAlign = MMI.getSourceAlign().valueOrOne();
6832 bool isVol = MMI.isVolatile();
6835 Root, sdl, Op1, Op2, Op3, DstAlign, SrcAlign, isVol, &
I,
6837 MachinePointerInfo(
I.getArgOperand(0)),
6838 MachinePointerInfo(
I.getArgOperand(1)),
I.getAAMetadata(),
BatchAA);
6839 updateDAGForMaybeTailCall(MM);
6842 case Intrinsic::memcpy_element_unordered_atomic: {
6848 Type *LengthTy =
MI.getLength()->getType();
6849 unsigned ElemSz =
MI.getElementSizeInBytes();
6853 isTC, MachinePointerInfo(
MI.getRawDest()),
6854 MachinePointerInfo(
MI.getRawSource()));
6855 updateDAGForMaybeTailCall(MC);
6858 case Intrinsic::memmove_element_unordered_atomic: {
6864 Type *LengthTy =
MI.getLength()->getType();
6865 unsigned ElemSz =
MI.getElementSizeInBytes();
6869 isTC, MachinePointerInfo(
MI.getRawDest()),
6870 MachinePointerInfo(
MI.getRawSource()));
6871 updateDAGForMaybeTailCall(MC);
6874 case Intrinsic::memset_element_unordered_atomic: {
6880 Type *LengthTy =
MI.getLength()->getType();
6881 unsigned ElemSz =
MI.getElementSizeInBytes();
6885 isTC, MachinePointerInfo(
MI.getRawDest()));
6886 updateDAGForMaybeTailCall(MC);
6889 case Intrinsic::call_preallocated_setup: {
6891 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6898 case Intrinsic::call_preallocated_arg: {
6900 SDValue SrcValue =
DAG.getSrcValue(PreallocatedCall);
6914 case Intrinsic::eh_typeid_for: {
6917 unsigned TypeID =
DAG.getMachineFunction().getTypeIDFor(GV);
6918 Res =
DAG.getConstant(
TypeID, sdl, MVT::i32);
6923 case Intrinsic::eh_return_i32:
6924 case Intrinsic::eh_return_i64:
6925 DAG.getMachineFunction().setCallsEHReturn(
true);
6932 case Intrinsic::eh_unwind_init:
6933 DAG.getMachineFunction().setCallsUnwindInit(
true);
6935 case Intrinsic::eh_dwarf_cfa:
6940 case Intrinsic::eh_sjlj_callsite: {
6942 assert(
FuncInfo.getCurrentCallSite() == 0 &&
"Overlapping call sites!");
6947 case Intrinsic::eh_sjlj_functioncontext: {
6949 MachineFrameInfo &MFI =
DAG.getMachineFunction().getFrameInfo();
6952 int FI =
FuncInfo.StaticAllocaMap[FnCtx];
6956 case Intrinsic::eh_sjlj_setjmp: {
6961 DAG.getVTList(MVT::i32, MVT::Other),
Ops);
6963 DAG.setRoot(
Op.getValue(1));
6966 case Intrinsic::eh_sjlj_longjmp:
6970 case Intrinsic::eh_sjlj_setup_dispatch:
6974 case Intrinsic::masked_gather:
6975 visitMaskedGather(
I);
6977 case Intrinsic::masked_load:
6980 case Intrinsic::masked_scatter:
6981 visitMaskedScatter(
I);
6983 case Intrinsic::masked_store:
6984 visitMaskedStore(
I);
6986 case Intrinsic::masked_expandload:
6987 visitMaskedLoad(
I,
true );
6989 case Intrinsic::masked_compressstore:
6990 visitMaskedStore(
I,
true );
6992 case Intrinsic::powi:
6996 case Intrinsic::log:
6999 case Intrinsic::log2:
7003 case Intrinsic::log10:
7007 case Intrinsic::exp:
7010 case Intrinsic::exp2:
7014 case Intrinsic::pow:
7018 case Intrinsic::sqrt:
7019 case Intrinsic::fabs:
7020 case Intrinsic::sin:
7021 case Intrinsic::cos:
7022 case Intrinsic::tan:
7023 case Intrinsic::asin:
7024 case Intrinsic::acos:
7025 case Intrinsic::atan:
7026 case Intrinsic::sinh:
7027 case Intrinsic::cosh:
7028 case Intrinsic::tanh:
7029 case Intrinsic::exp10:
7030 case Intrinsic::floor:
7031 case Intrinsic::ceil:
7032 case Intrinsic::trunc:
7033 case Intrinsic::rint:
7034 case Intrinsic::nearbyint:
7035 case Intrinsic::round:
7036 case Intrinsic::roundeven:
7037 case Intrinsic::canonicalize: {
7040 switch (Intrinsic) {
7042 case Intrinsic::sqrt: Opcode =
ISD::FSQRT;
break;
7043 case Intrinsic::fabs: Opcode =
ISD::FABS;
break;
7044 case Intrinsic::sin: Opcode =
ISD::FSIN;
break;
7045 case Intrinsic::cos: Opcode =
ISD::FCOS;
break;
7046 case Intrinsic::tan: Opcode =
ISD::FTAN;
break;
7047 case Intrinsic::asin: Opcode =
ISD::FASIN;
break;
7048 case Intrinsic::acos: Opcode =
ISD::FACOS;
break;
7049 case Intrinsic::atan: Opcode =
ISD::FATAN;
break;
7050 case Intrinsic::sinh: Opcode =
ISD::FSINH;
break;
7051 case Intrinsic::cosh: Opcode =
ISD::FCOSH;
break;
7052 case Intrinsic::tanh: Opcode =
ISD::FTANH;
break;
7053 case Intrinsic::exp10: Opcode =
ISD::FEXP10;
break;
7054 case Intrinsic::floor: Opcode =
ISD::FFLOOR;
break;
7055 case Intrinsic::ceil: Opcode =
ISD::FCEIL;
break;
7056 case Intrinsic::trunc: Opcode =
ISD::FTRUNC;
break;
7057 case Intrinsic::rint: Opcode =
ISD::FRINT;
break;
7059 case Intrinsic::round: Opcode =
ISD::FROUND;
break;
7066 getValue(
I.getArgOperand(0)).getValueType(),
7070 case Intrinsic::atan2:
7072 getValue(
I.getArgOperand(0)).getValueType(),
7076 case Intrinsic::lround:
7077 case Intrinsic::llround:
7078 case Intrinsic::lrint:
7079 case Intrinsic::llrint: {
7082 switch (Intrinsic) {
7084 case Intrinsic::lround: Opcode =
ISD::LROUND;
break;
7086 case Intrinsic::lrint: Opcode =
ISD::LRINT;
break;
7087 case Intrinsic::llrint: Opcode =
ISD::LLRINT;
break;
7096 case Intrinsic::minnum:
7098 getValue(
I.getArgOperand(0)).getValueType(),
7102 case Intrinsic::maxnum:
7104 getValue(
I.getArgOperand(0)).getValueType(),
7108 case Intrinsic::minimum:
7110 getValue(
I.getArgOperand(0)).getValueType(),
7114 case Intrinsic::maximum:
7116 getValue(
I.getArgOperand(0)).getValueType(),
7120 case Intrinsic::minimumnum:
7122 getValue(
I.getArgOperand(0)).getValueType(),
7126 case Intrinsic::maximumnum:
7128 getValue(
I.getArgOperand(0)).getValueType(),
7132 case Intrinsic::copysign:
7134 getValue(
I.getArgOperand(0)).getValueType(),
7138 case Intrinsic::ldexp:
7140 getValue(
I.getArgOperand(0)).getValueType(),
7144 case Intrinsic::modf:
7145 case Intrinsic::sincos:
7146 case Intrinsic::sincospi:
7147 case Intrinsic::frexp: {
7149 switch (Intrinsic) {
7152 case Intrinsic::sincos:
7155 case Intrinsic::sincospi:
7158 case Intrinsic::modf:
7161 case Intrinsic::frexp:
7167 SDVTList VTs =
DAG.getVTList(ValueVTs);
7169 &
I,
DAG.getNode(Opcode, sdl, VTs,
getValue(
I.getArgOperand(0)), Flags));
7172 case Intrinsic::arithmetic_fence: {
7174 getValue(
I.getArgOperand(0)).getValueType(),
7178 case Intrinsic::fma:
7184#define INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC) \
7185 case Intrinsic::INTRINSIC:
7186#include "llvm/IR/ConstrainedOps.def"
7189#define BEGIN_REGISTER_VP_INTRINSIC(VPID, ...) case Intrinsic::VPID:
7190#include "llvm/IR/VPIntrinsics.def"
7193 case Intrinsic::fptrunc_round: {
7197 std::optional<RoundingMode> RoundMode =
7205 SelectionDAG::FlagInserter FlagsInserter(
DAG, Flags);
7210 DAG.getTargetConstant((
int)*RoundMode, sdl, MVT::i32));
7215 case Intrinsic::fmuladd: {
7220 getValue(
I.getArgOperand(0)).getValueType(),
7227 getValue(
I.getArgOperand(0)).getValueType(),
7243 case Intrinsic::fptosi_sat: {
7250 case Intrinsic::fptoui_sat: {
7257 case Intrinsic::convert_from_arbitrary_fp: {
7262 const fltSemantics *SrcSem =
7265 DAG.getContext()->emitError(
7266 "convert_from_arbitrary_fp: not implemented format '" + FormatStr +
7277 DAG.getTargetConstant(
static_cast<int>(SemEnum), sdl, MVT::i32);
7282 case Intrinsic::convert_to_arbitrary_fp: {
7287 const fltSemantics *DstSem =
7290 DAG.getContext()->emitError(
7291 "convert_to_arbitrary_fp: not implemented format '" + FormatStr +
7303 "Dynamic rounding mode should have been rejected by the verifier");
7311 DAG.getTargetConstant(
static_cast<int>(SemEnum), sdl, MVT::i32);
7313 DAG.getTargetConstant(
static_cast<int>(*RoundMode), sdl, MVT::i32);
7314 SDValue SatConst =
DAG.getTargetConstant(Saturate, sdl, MVT::i32);
7316 SemConst, RoundConst, SatConst));
7319 case Intrinsic::set_rounding:
7325 case Intrinsic::is_fpclass: {
7326 const DataLayout DLayout =
DAG.getDataLayout();
7328 EVT ArgVT = TLI.
getValueType(DLayout,
I.getArgOperand(0)->getType());
7331 MachineFunction &MF =
DAG.getMachineFunction();
7335 Flags.setNoFPExcept(
7336 !
F.getAttributes().hasFnAttr(llvm::Attribute::StrictFP));
7352 case Intrinsic::get_fpenv: {
7353 const DataLayout DLayout =
DAG.getDataLayout();
7355 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7370 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7373 Chain =
DAG.getGetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7374 Res =
DAG.getLoad(EnvVT, sdl, Chain, Temp, MPI);
7380 case Intrinsic::set_fpenv: {
7381 const DataLayout DLayout =
DAG.getDataLayout();
7384 Align TempAlign =
DAG.getEVTAlign(EnvVT);
7397 Chain =
DAG.getStore(Chain, sdl, Env, Temp, MPI, TempAlign,
7399 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
7402 Chain =
DAG.getSetFPEnv(Chain, sdl, Temp, EnvVT, MMO);
7407 case Intrinsic::reset_fpenv:
7410 case Intrinsic::get_fpmode:
7419 case Intrinsic::set_fpmode:
7424 case Intrinsic::reset_fpmode: {
7429 case Intrinsic::pcmarker: {
7434 case Intrinsic::readcyclecounter: {
7437 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7442 case Intrinsic::readsteadycounter: {
7445 DAG.getVTList(MVT::i64, MVT::Other),
Op);
7450 case Intrinsic::bitreverse:
7452 getValue(
I.getArgOperand(0)).getValueType(),
7455 case Intrinsic::bswap:
7457 getValue(
I.getArgOperand(0)).getValueType(),
7460 case Intrinsic::cttz: {
7468 case Intrinsic::ctlz: {
7476 case Intrinsic::ctpop: {
7482 case Intrinsic::fshl:
7483 case Intrinsic::fshr: {
7484 bool IsFSHL =
Intrinsic == Intrinsic::fshl;
7488 EVT VT =
X.getValueType();
7499 case Intrinsic::clmul: {
7505 case Intrinsic::pext: {
7511 case Intrinsic::pdep: {
7517 case Intrinsic::sadd_sat: {
7523 case Intrinsic::uadd_sat: {
7529 case Intrinsic::ssub_sat: {
7535 case Intrinsic::usub_sat: {
7541 case Intrinsic::sshl_sat:
7542 case Intrinsic::ushl_sat: {
7546 EVT ShiftTy =
DAG.getTargetLoweringInfo().getShiftAmountTy(
7551 if (!
I.getType()->isVectorTy() && Op2.
getValueType() != ShiftTy) {
7554 "Unexpected shift type");
7563 case Intrinsic::smul_fix:
7564 case Intrinsic::umul_fix:
7565 case Intrinsic::smul_fix_sat:
7566 case Intrinsic::umul_fix_sat: {
7574 case Intrinsic::sdiv_fix:
7575 case Intrinsic::udiv_fix:
7576 case Intrinsic::sdiv_fix_sat:
7577 case Intrinsic::udiv_fix_sat: {
7582 Op1, Op2, Op3,
DAG, TLI));
7585 case Intrinsic::smax: {
7591 case Intrinsic::smin: {
7597 case Intrinsic::umax: {
7603 case Intrinsic::umin: {
7609 case Intrinsic::abs: {
7616 case Intrinsic::scmp: {
7623 case Intrinsic::ucmp: {
7630 case Intrinsic::stackaddress:
7631 case Intrinsic::stacksave: {
7636 Res =
DAG.getNode(SDOpcode, sdl,
DAG.getVTList(VT, MVT::Other),
Op);
7641 case Intrinsic::stackrestore:
7645 case Intrinsic::get_dynamic_area_offset: {
7654 case Intrinsic::stackguard: {
7655 MachineFunction &MF =
DAG.getMachineFunction();
7661 Res =
DAG.getPtrExtOrTrunc(Res, sdl, PtrTy);
7665 LLVMContext &Ctx = *
DAG.getContext();
7666 Ctx.
diagnose(DiagnosticInfoGeneric(
"unable to lower stackguard"));
7673 MachinePointerInfo(
Global, 0), Align,
7685 case Intrinsic::stackprotector: {
7687 MachineFunction &MF =
DAG.getMachineFunction();
7707 Chain, sdl, Src, FIN,
7714 case Intrinsic::objectsize:
7717 case Intrinsic::is_constant:
7720 case Intrinsic::annotation:
7721 case Intrinsic::ptr_annotation:
7722 case Intrinsic::launder_invariant_group:
7723 case Intrinsic::strip_invariant_group:
7728 case Intrinsic::type_test:
7729 case Intrinsic::public_type_test:
7731 "LowerTypeTests pass before code generation");
7734 case Intrinsic::assume:
7735 case Intrinsic::experimental_noalias_scope_decl:
7736 case Intrinsic::var_annotation:
7737 case Intrinsic::sideeffect:
7742 case Intrinsic::codeview_annotation: {
7744 MachineFunction &MF =
DAG.getMachineFunction();
7753 case Intrinsic::init_trampoline: {
7761 Ops[4] =
DAG.getSrcValue(
I.getArgOperand(0));
7769 case Intrinsic::adjust_trampoline:
7774 case Intrinsic::gcroot: {
7775 assert(
DAG.getMachineFunction().getFunction().hasGC() &&
7776 "only valid in functions with gc specified, enforced by Verifier");
7778 const Value *Alloca =
I.getArgOperand(0)->stripPointerCasts();
7785 case Intrinsic::gcread:
7786 case Intrinsic::gcwrite:
7788 case Intrinsic::get_rounding:
7794 case Intrinsic::expect:
7795 case Intrinsic::expect_with_probability:
7801 case Intrinsic::ubsantrap:
7802 case Intrinsic::debugtrap:
7803 case Intrinsic::trap: {
7804 StringRef TrapFuncName =
7805 I.getAttributes().getFnAttr(
"trap-func-name").getValueAsString();
7806 if (TrapFuncName.
empty()) {
7807 switch (Intrinsic) {
7808 case Intrinsic::trap:
7811 case Intrinsic::debugtrap:
7814 case Intrinsic::ubsantrap:
7817 DAG.getTargetConstant(
7823 DAG.addNoMergeSiteInfo(
DAG.getRoot().getNode(),
7824 I.hasFnAttr(Attribute::NoMerge));
7828 if (Intrinsic == Intrinsic::ubsantrap) {
7829 Value *Arg =
I.getArgOperand(0);
7833 TargetLowering::CallLoweringInfo CLI(
DAG);
7834 CLI.setDebugLoc(sdl).setChain(
getRoot()).setLibCallee(
7836 DAG.getExternalSymbol(TrapFuncName.
data(),
7839 CLI.NoMerge =
I.hasFnAttr(Attribute::NoMerge);
7845 case Intrinsic::allow_runtime_check:
7846 case Intrinsic::allow_ubsan_check:
7850 case Intrinsic::uadd_with_overflow:
7851 case Intrinsic::sadd_with_overflow:
7852 case Intrinsic::usub_with_overflow:
7853 case Intrinsic::ssub_with_overflow:
7854 case Intrinsic::umul_with_overflow:
7855 case Intrinsic::smul_with_overflow: {
7857 switch (Intrinsic) {
7859 case Intrinsic::uadd_with_overflow:
Op =
ISD::UADDO;
break;
7860 case Intrinsic::sadd_with_overflow:
Op =
ISD::SADDO;
break;
7861 case Intrinsic::usub_with_overflow:
Op =
ISD::USUBO;
break;
7862 case Intrinsic::ssub_with_overflow:
Op =
ISD::SSUBO;
break;
7863 case Intrinsic::umul_with_overflow:
Op =
ISD::UMULO;
break;
7864 case Intrinsic::smul_with_overflow:
Op =
ISD::SMULO;
break;
7872 SDVTList VTs =
DAG.getVTList(ResultVT, OverflowVT);
7876 case Intrinsic::prefetch: {
7891 std::nullopt, Flags);
7897 DAG.setRoot(Result);
7900 case Intrinsic::lifetime_start:
7901 case Intrinsic::lifetime_end: {
7902 bool IsStart = (
Intrinsic == Intrinsic::lifetime_start);
7908 if (!LifetimeObject)
7913 auto SI =
FuncInfo.StaticAllocaMap.find(LifetimeObject);
7914 if (SI ==
FuncInfo.StaticAllocaMap.end())
7918 Res =
DAG.getLifetimeNode(IsStart, sdl,
getRoot(), FrameIndex);
7922 case Intrinsic::pseudoprobe: {
7930 case Intrinsic::invariant_start:
7935 case Intrinsic::invariant_end:
7938 case Intrinsic::clear_cache: {
7943 {InputChain, StartVal, EndVal});
7948 case Intrinsic::donothing:
7949 case Intrinsic::seh_try_begin:
7950 case Intrinsic::seh_scope_begin:
7951 case Intrinsic::seh_try_end:
7952 case Intrinsic::seh_scope_end:
7955 case Intrinsic::experimental_stackmap:
7958 case Intrinsic::experimental_patchpoint_void:
7959 case Intrinsic::experimental_patchpoint:
7962 case Intrinsic::experimental_gc_statepoint:
7965 case Intrinsic::experimental_gc_result:
7968 case Intrinsic::experimental_gc_relocate:
7971 case Intrinsic::instrprof_cover:
7973 case Intrinsic::instrprof_increment:
7975 case Intrinsic::instrprof_timestamp:
7977 case Intrinsic::instrprof_value_profile:
7979 case Intrinsic::instrprof_mcdc_parameters:
7981 case Intrinsic::instrprof_mcdc_tvbitmap_update:
7983 case Intrinsic::localescape: {
7984 MachineFunction &MF =
DAG.getMachineFunction();
7985 const TargetInstrInfo *
TII =
DAG.getSubtarget().getInstrInfo();
7989 for (
unsigned Idx = 0,
E =
I.arg_size(); Idx <
E; ++Idx) {
7995 "can only escape static allocas");
8000 TII->get(TargetOpcode::LOCAL_ESCAPE))
8008 case Intrinsic::localrecover: {
8010 MachineFunction &MF =
DAG.getMachineFunction();
8016 unsigned(Idx->getLimitedValue(std::numeric_limits<int>::max()));
8020 Value *
FP =
I.getArgOperand(1);
8026 SDValue OffsetSym =
DAG.getMCSymbol(FrameAllocSym, PtrVT);
8031 SDValue Add =
DAG.getMemBasePlusOffset(FPVal, OffsetVal, sdl);
8037 case Intrinsic::fake_use: {
8038 Value *
V =
I.getArgOperand(0);
8043 auto FakeUseValue = [&]() ->
SDValue {
8057 if (!FakeUseValue || FakeUseValue.isUndef())
8060 Ops[1] = FakeUseValue;
8069 case Intrinsic::reloc_none: {
8074 DAG.getTargetExternalSymbol(
8080 case Intrinsic::cond_loop: {
8090 case Intrinsic::eh_exceptionpointer:
8091 case Intrinsic::eh_exceptioncode: {
8097 SDValue N =
DAG.getCopyFromReg(
DAG.getEntryNode(), sdl, VReg, PtrVT);
8098 if (Intrinsic == Intrinsic::eh_exceptioncode)
8099 N =
DAG.getZExtOrTrunc(
N, sdl, MVT::i32);
8103 case Intrinsic::xray_customevent: {
8106 const auto &Triple =
DAG.getTarget().getTargetTriple();
8107 if (!Triple.isAArch64(64) && Triple.getArch() !=
Triple::x86_64 &&
8116 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
8118 Ops.push_back(LogEntryVal);
8119 Ops.push_back(StrSizeVal);
8120 Ops.push_back(Chain);
8126 MachineSDNode *MN =
DAG.getMachineNode(TargetOpcode::PATCHABLE_EVENT_CALL,
8129 DAG.setRoot(patchableNode);
8133 case Intrinsic::xray_typedevent: {
8136 const auto &Triple =
DAG.getTarget().getTargetTriple();
8137 if (!Triple.isAArch64(64) && Triple.getArch() !=
Triple::x86_64 &&
8149 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
8151 Ops.push_back(LogTypeId);
8152 Ops.push_back(LogEntryVal);
8153 Ops.push_back(StrSizeVal);
8154 Ops.push_back(Chain);
8160 MachineSDNode *MN =
DAG.getMachineNode(
8161 TargetOpcode::PATCHABLE_TYPED_EVENT_CALL, sdl, NodeTys,
Ops);
8163 DAG.setRoot(patchableNode);
8167 case Intrinsic::experimental_deoptimize:
8170 case Intrinsic::stepvector:
8173 case Intrinsic::vector_reduce_fadd:
8174 case Intrinsic::vector_reduce_fmul:
8175 case Intrinsic::vector_reduce_add:
8176 case Intrinsic::vector_reduce_mul:
8177 case Intrinsic::vector_reduce_and:
8178 case Intrinsic::vector_reduce_or:
8179 case Intrinsic::vector_reduce_xor:
8180 case Intrinsic::vector_reduce_smax:
8181 case Intrinsic::vector_reduce_smin:
8182 case Intrinsic::vector_reduce_umax:
8183 case Intrinsic::vector_reduce_umin:
8184 case Intrinsic::vector_reduce_fmax:
8185 case Intrinsic::vector_reduce_fmin:
8186 case Intrinsic::vector_reduce_fmaximum:
8187 case Intrinsic::vector_reduce_fminimum:
8188 visitVectorReduce(
I, Intrinsic);
8191 case Intrinsic::icall_branch_funnel: {
8197 I.getArgOperand(1),
Offset,
DAG.getDataLayout()));
8200 "llvm.icall.branch.funnel operand must be a GlobalValue");
8201 Ops.push_back(
DAG.getTargetGlobalAddress(
Base, sdl, MVT::i64, 0));
8203 struct BranchFunnelTarget {
8209 for (
unsigned Op = 1,
N =
I.arg_size();
Op !=
N;
Op += 2) {
8212 if (ElemBase !=
Base)
8214 "to the same GlobalValue");
8220 "llvm.icall.branch.funnel operand must be a GlobalValue");
8226 [](
const BranchFunnelTarget &
T1,
const BranchFunnelTarget &T2) {
8227 return T1.Offset < T2.Offset;
8230 for (
auto &
T : Targets) {
8231 Ops.push_back(
DAG.getTargetConstant(
T.Offset, sdl, MVT::i32));
8232 Ops.push_back(
T.Target);
8235 Ops.push_back(
DAG.getRoot());
8236 SDValue N(
DAG.getMachineNode(TargetOpcode::ICALL_BRANCH_FUNNEL, sdl,
8245 case Intrinsic::wasm_landingpad_index:
8251 case Intrinsic::aarch64_settag:
8252 case Intrinsic::aarch64_settag_zero: {
8253 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
8254 bool ZeroMemory =
Intrinsic == Intrinsic::aarch64_settag_zero;
8257 getValue(
I.getArgOperand(1)), MachinePointerInfo(
I.getArgOperand(0)),
8263 case Intrinsic::amdgcn_cs_chain: {
8268 Type *RetTy =
I.getType();
8278 for (
unsigned Idx : {2, 3, 1}) {
8279 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8281 Arg.setAttributes(&
I, Idx);
8282 Args.push_back(Arg);
8285 assert(Args[0].IsInReg &&
"SGPR args should be marked inreg");
8286 assert(!Args[1].IsInReg &&
"VGPR args should not be marked inreg");
8287 Args[2].IsInReg =
true;
8290 for (
unsigned Idx = 4; Idx <
I.arg_size(); ++Idx) {
8291 TargetLowering::ArgListEntry Arg(
getValue(
I.getOperand(Idx)),
8293 Arg.setAttributes(&
I, Idx);
8294 Args.push_back(Arg);
8297 TargetLowering::CallLoweringInfo CLI(
DAG);
8300 .setCallee(CC, RetTy, Callee, std::move(Args))
8303 .setConvergent(
I.isConvergent());
8305 std::pair<SDValue, SDValue>
Result =
8309 "Should've lowered as tail call");
8314 case Intrinsic::amdgcn_call_whole_wave: {
8316 bool isTailCall =
I.isTailCall();
8319 for (
unsigned Idx = 1; Idx <
I.arg_size(); ++Idx) {
8320 TargetLowering::ArgListEntry Arg(
getValue(
I.getArgOperand(Idx)),
8321 I.getArgOperand(Idx)->getType());
8322 Arg.setAttributes(&
I, Idx);
8329 Args.push_back(Arg);
8334 auto *Token = Bundle->Inputs[0].get();
8335 ConvControlToken =
getValue(Token);
8338 TargetLowering::CallLoweringInfo CLI(
DAG);
8342 getValue(
I.getArgOperand(0)), std::move(Args))
8346 .setConvergent(
I.isConvergent())
8347 .setConvergenceControlToken(ConvControlToken);
8350 std::pair<SDValue, SDValue>
Result =
8353 if (
Result.first.getNode())
8357 case Intrinsic::ptrmask: {
8373 auto HighOnes =
DAG.getNode(
8374 ISD::SHL, sdl, PtrVT,
DAG.getAllOnesConstant(sdl, PtrVT),
8375 DAG.getShiftAmountConstant(
Mask.getValueType().getFixedSizeInBits(),
8378 DAG.getZExtOrTrunc(Mask, sdl, PtrVT), HighOnes);
8379 }
else if (
Mask.getValueType() != PtrVT)
8380 Mask =
DAG.getPtrExtOrTrunc(Mask, sdl, PtrVT);
8386 case Intrinsic::threadlocal_address: {
8390 case Intrinsic::get_active_lane_mask: {
8394 EVT ElementVT =
Index.getValueType();
8405 SDValue VectorIndex =
DAG.getSplat(VecTy, sdl, Index);
8406 SDValue VectorTripCount =
DAG.getSplat(VecTy, sdl, TripCount);
8407 SDValue VectorStep =
DAG.getStepVector(sdl, VecTy);
8410 SDValue SetCC =
DAG.getSetCC(sdl, CCVT, VectorInduction,
8415 case Intrinsic::experimental_get_vector_length: {
8417 "Expected positive VF");
8422 EVT CountVT =
Count.getValueType();
8425 visitTargetIntrinsic(
I, Intrinsic);
8434 if (CountVT.
bitsLT(VT)) {
8439 SDValue MaxEVL =
DAG.getElementCount(sdl, CountVT,
8449 case Intrinsic::vector_partial_reduce_add: {
8457 case Intrinsic::vector_partial_reduce_fadd: {
8465 case Intrinsic::experimental_cttz_elts: {
8467 EVT OpVT =
Op.getValueType();
8474 SDValue AllZero =
DAG.getConstant(0, sdl, OpVT);
8483 case Intrinsic::vector_insert: {
8491 if (
Index.getValueType() != VectorIdxTy)
8492 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8499 case Intrinsic::vector_extract: {
8507 if (
Index.getValueType() != VectorIdxTy)
8508 Index =
DAG.getVectorIdxConstant(
Index->getAsZExtVal(), sdl);
8514 case Intrinsic::experimental_vector_match: {
8520 EVT ResVT =
Mask.getValueType();
8526 visitTargetIntrinsic(
I, Intrinsic);
8530 SDValue Ret =
DAG.getConstant(0, sdl, ResVT);
8532 for (
unsigned i = 0; i < SearchSize; ++i) {
8535 DAG.getVectorIdxConstant(i, sdl));
8538 Ret =
DAG.getNode(
ISD::OR, sdl, ResVT, Ret, Cmp);
8544 case Intrinsic::vector_reverse:
8545 visitVectorReverse(
I);
8547 case Intrinsic::vector_splice_left:
8548 case Intrinsic::vector_splice_right:
8549 visitVectorSplice(
I);
8551 case Intrinsic::callbr_landingpad:
8552 visitCallBrLandingPad(
I);
8554 case Intrinsic::vector_interleave2:
8555 visitVectorInterleave(
I, 2);
8557 case Intrinsic::vector_interleave3:
8558 visitVectorInterleave(
I, 3);
8560 case Intrinsic::vector_interleave4:
8561 visitVectorInterleave(
I, 4);
8563 case Intrinsic::vector_interleave5:
8564 visitVectorInterleave(
I, 5);
8566 case Intrinsic::vector_interleave6:
8567 visitVectorInterleave(
I, 6);
8569 case Intrinsic::vector_interleave7:
8570 visitVectorInterleave(
I, 7);
8572 case Intrinsic::vector_interleave8:
8573 visitVectorInterleave(
I, 8);
8575 case Intrinsic::vector_deinterleave2:
8576 visitVectorDeinterleave(
I, 2);
8578 case Intrinsic::vector_deinterleave3:
8579 visitVectorDeinterleave(
I, 3);
8581 case Intrinsic::vector_deinterleave4:
8582 visitVectorDeinterleave(
I, 4);
8584 case Intrinsic::vector_deinterleave5:
8585 visitVectorDeinterleave(
I, 5);
8587 case Intrinsic::vector_deinterleave6:
8588 visitVectorDeinterleave(
I, 6);
8590 case Intrinsic::vector_deinterleave7:
8591 visitVectorDeinterleave(
I, 7);
8593 case Intrinsic::vector_deinterleave8:
8594 visitVectorDeinterleave(
I, 8);
8596 case Intrinsic::experimental_vector_compress:
8598 getValue(
I.getArgOperand(0)).getValueType(),
8603 case Intrinsic::experimental_convergence_anchor:
8604 case Intrinsic::experimental_convergence_entry:
8605 case Intrinsic::experimental_convergence_loop:
8606 visitConvergenceControl(
I, Intrinsic);
8608 case Intrinsic::experimental_vector_histogram_add: {
8609 visitVectorHistogram(
I, Intrinsic);
8612 case Intrinsic::experimental_vector_extract_last_active: {
8613 visitVectorExtractLastActive(
I, Intrinsic);
8616 case Intrinsic::loop_dependence_war_mask:
8621 DAG.getConstant(0, sdl, MVT::i64)));
8623 case Intrinsic::loop_dependence_raw_mask:
8628 DAG.getConstant(0, sdl, MVT::i64)));
8630 case Intrinsic::masked_udiv:
8636 case Intrinsic::masked_sdiv:
8642 case Intrinsic::masked_urem:
8648 case Intrinsic::masked_srem:
8657void SelectionDAGBuilder::pushFPOpOutChain(
SDValue Result,
8673 PendingConstrainedFP.push_back(OutChain);
8676 PendingConstrainedFPStrict.push_back(OutChain);
8681void SelectionDAGBuilder::visitConstrainedFPIntrinsic(
8695 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8697 SDVTList VTs =
DAG.getVTList(VT, MVT::Other);
8701 Flags.setNoFPExcept(
true);
8704 Flags.copyFMF(*FPOp);
8709#define DAG_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) \
8710 case Intrinsic::INTRINSIC: \
8711 Opcode = ISD::STRICT_##DAGN; \
8713#include "llvm/IR/ConstrainedOps.def"
8714 case Intrinsic::experimental_constrained_fmuladd: {
8721 pushFPOpOutChain(
Mul, EB);
8744 if (
DAG.isKnownNeverNaN(Opers[1]) &&
DAG.isKnownNeverNaN(Opers[2]))
8752 pushFPOpOutChain(Result, EB);
8759 std::optional<unsigned> ResOPC;
8761 case Intrinsic::vp_ctlz: {
8763 ResOPC = IsZeroUndef ? ISD::VP_CTLZ_ZERO_POISON : ISD::VP_CTLZ;
8766 case Intrinsic::vp_cttz: {
8768 ResOPC = IsZeroUndef ? ISD::VP_CTTZ_ZERO_POISON : ISD::VP_CTTZ;
8771 case Intrinsic::vp_cttz_elts: {
8773 ResOPC = IsZeroPoison ? ISD::VP_CTTZ_ELTS_ZERO_POISON : ISD::VP_CTTZ_ELTS;
8776#define HELPER_MAP_VPID_TO_VPSD(VPID, VPSD) \
8777 case Intrinsic::VPID: \
8778 ResOPC = ISD::VPSD; \
8780#include "llvm/IR/VPIntrinsics.def"
8785 "Inconsistency: no SDNode available for this VPIntrinsic!");
8787 if (*ResOPC == ISD::VP_REDUCE_SEQ_FADD ||
8788 *ResOPC == ISD::VP_REDUCE_SEQ_FMUL) {
8790 return *ResOPC == ISD::VP_REDUCE_SEQ_FADD ? ISD::VP_REDUCE_FADD
8791 : ISD::VP_REDUCE_FMUL;
8797void SelectionDAGBuilder::visitVPLoad(
8809 Alignment =
DAG.getEVTAlign(VT);
8812 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8813 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8816 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8817 MachinePointerInfo(PtrOperand), MMOFlags,
8819 LD =
DAG.getLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8826void SelectionDAGBuilder::visitVPLoadFF(
8829 assert(OpValues.
size() == 3 &&
"Unexpected number of operands");
8839 Alignment =
DAG.getEVTAlign(VT);
8842 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8843 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8846 LD =
DAG.getLoadFFVP(VT,
DL, InChain, OpValues[0], OpValues[1], OpValues[2],
8851 setValue(&VPIntrin,
DAG.getMergeValues({LD.getValue(0), Trunc},
DL));
8854void SelectionDAGBuilder::visitVPGather(
8858 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8870 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8872 *Alignment, AAInfo, Ranges);
8882 EVT IdxVT =
Index.getValueType();
8888 LD =
DAG.getGatherVP(
8889 DAG.getVTList(VT, MVT::Other), VT,
DL,
8890 {DAG.getRoot(), Base, Index, Scale, OpValues[1], OpValues[2]}, MMO,
8896void SelectionDAGBuilder::visitVPStore(
8900 EVT VT = OpValues[0].getValueType();
8905 Alignment =
DAG.getEVTAlign(VT);
8908 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8911 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8912 MachinePointerInfo(PtrOperand), MMOFlags,
8921void SelectionDAGBuilder::visitVPScatter(
8924 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8926 EVT VT = OpValues[0].getValueType();
8936 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8938 *Alignment, AAInfo);
8948 EVT IdxVT =
Index.getValueType();
8954 ST =
DAG.getScatterVP(
DAG.getVTList(MVT::Other), VT,
DL,
8955 {getMemoryRoot(), OpValues[0], Base, Index, Scale,
8956 OpValues[2], OpValues[3]},
8962void SelectionDAGBuilder::visitVPStridedLoad(
8974 SDValue InChain = AddToChain ?
DAG.getRoot() :
DAG.getEntryNode();
8976 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
8979 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
8981 *Alignment, AAInfo, Ranges);
8983 SDValue LD =
DAG.getStridedLoadVP(VT,
DL, InChain, OpValues[0], OpValues[1],
8984 OpValues[2], OpValues[3], MMO,
8992void SelectionDAGBuilder::visitVPStridedStore(
8996 EVT VT = OpValues[0].getValueType();
9002 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9005 MachineMemOperand *MMO =
DAG.getMachineFunction().getMachineMemOperand(
9007 *Alignment, AAInfo);
9011 DAG.getUNDEF(OpValues[1].getValueType()), OpValues[2], OpValues[3],
9019void SelectionDAGBuilder::visitVPCmp(
const VPCmpIntrinsic &VPIntrin) {
9020 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9033 "Unexpected target EVL type");
9038 SimplifyQuery SQ(
DAG.getDataLayout(), &VPIntrin);
9045 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9048 Condition, MaskOp, EVL));
9051void SelectionDAGBuilder::visitVectorPredicationIntrinsic(
9059 return visitVPCmp(*CmpI);
9062 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9064 SDVTList VTs =
DAG.getVTList(ValueVTs);
9070 "Unexpected target EVL type");
9074 for (
unsigned I = 0;
I < VPIntrin.
arg_size(); ++
I) {
9076 if (
I == EVLParamPos)
9083 SDNodeFlags SDFlags;
9091 visitVPLoad(VPIntrin, ValueVTs[0], OpValues);
9093 case ISD::VP_LOAD_FF:
9094 visitVPLoadFF(VPIntrin, ValueVTs[0], ValueVTs[1], OpValues);
9096 case ISD::VP_GATHER:
9097 visitVPGather(VPIntrin, ValueVTs[0], OpValues);
9099 case ISD::EXPERIMENTAL_VP_STRIDED_LOAD:
9100 visitVPStridedLoad(VPIntrin, ValueVTs[0], OpValues);
9103 visitVPStore(VPIntrin, OpValues);
9105 case ISD::VP_SCATTER:
9106 visitVPScatter(VPIntrin, OpValues);
9108 case ISD::EXPERIMENTAL_VP_STRIDED_STORE:
9109 visitVPStridedStore(VPIntrin, OpValues);
9111 case ISD::VP_FMULADD: {
9112 assert(OpValues.
size() == 5 &&
"Unexpected number of operands");
9113 SDNodeFlags SDFlags;
9118 setValue(&VPIntrin,
DAG.getNode(ISD::VP_FMA,
DL, VTs, OpValues, SDFlags));
9121 ISD::VP_FMUL,
DL, VTs,
9122 {OpValues[0], OpValues[1], OpValues[3], OpValues[4]}, SDFlags);
9124 DAG.getNode(ISD::VP_FADD,
DL, VTs,
9125 {
Mul, OpValues[2], OpValues[3], OpValues[4]}, SDFlags);
9130 case ISD::VP_IS_FPCLASS: {
9131 const DataLayout DLayout =
DAG.getDataLayout();
9133 auto Constant = OpValues[1]->getAsZExtVal();
9136 {OpValues[0],
Check, OpValues[2], OpValues[3]});
9140 case ISD::VP_INTTOPTR: {
9151 case ISD::VP_PTRTOINT: {
9153 EVT DestVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9166 case ISD::VP_CTLZ_ZERO_POISON:
9168 case ISD::VP_CTTZ_ZERO_POISON:
9169 case ISD::VP_CTTZ_ELTS_ZERO_POISON:
9170 case ISD::VP_CTTZ_ELTS: {
9172 DAG.getNode(Opcode,
DL, VTs, {OpValues[0], OpValues[2], OpValues[3]});
9190 unsigned CallSiteIndex =
FuncInfo.getCurrentCallSite();
9191 if (CallSiteIndex) {
9205 assert(BeginLabel &&
"BeginLabel should've been set");
9219 assert(
II &&
"II should've been set");
9230std::pair<SDValue, SDValue>
9244 std::pair<SDValue, SDValue> Result = TLI.
LowerCallTo(CLI);
9247 "Non-null chain expected with non-tail call!");
9248 assert((Result.second.getNode() || !Result.first.getNode()) &&
9249 "Null value expected with tail call!");
9251 if (!Result.second.getNode()) {
9258 PendingExports.clear();
9260 DAG.setRoot(Result.second);
9278 if (!isMustTailCall &&
9279 Caller->getFnAttribute(
"disable-tail-calls").getValueAsBool())
9285 if (
DAG.getTargetLoweringInfo().supportSwiftError() &&
9286 Caller->getAttributes().hasAttrSomewhere(Attribute::SwiftError))
9295 bool isTailCall,
bool isMustTailCall,
9298 auto &
DL =
DAG.getDataLayout();
9305 const Value *SwiftErrorVal =
nullptr;
9312 const Value *V = *
I;
9315 if (V->getType()->isEmptyTy())
9320 Entry.setAttributes(&CB,
I - CB.
arg_begin());
9332 Args.push_back(Entry);
9343 Value *V = Bundle->Inputs[0];
9345 Entry.IsCFGuardTarget =
true;
9346 Args.push_back(Entry);
9359 "Target doesn't support calls with kcfi operand bundles.");
9367 auto *Token = Bundle->Inputs[0].get();
9368 ConvControlToken =
getValue(Token);
9379 .
setCallee(RetTy, FTy, Callee, std::move(Args), CB)
9392 "This target doesn't support calls with ptrauth operand bundles.");
9396 std::pair<SDValue, SDValue> Result =
lowerInvokable(CLI, EHPadBB);
9398 if (Result.first.getNode()) {
9413 DAG.setRoot(CopyNode);
9429 LoadTy, Builder.DAG.getDataLayout()))
9430 return Builder.getValue(LoadCst);
9436 bool ConstantMemory =
false;
9439 if (Builder.BatchAA && Builder.BatchAA->pointsToConstantMemory(PtrVal)) {
9440 Root = Builder.DAG.getEntryNode();
9441 ConstantMemory =
true;
9444 Root = Builder.DAG.getRoot();
9449 Builder.DAG.getLoad(LoadVT, Builder.getCurSDLoc(), Root, Ptr,
9452 if (!ConstantMemory)
9453 Builder.PendingLoads.push_back(LoadVal.
getValue(1));
9459void SelectionDAGBuilder::processIntegerCallValue(
const Instruction &
I,
9462 EVT VT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9473bool SelectionDAGBuilder::visitMemCmpBCmpCall(
const CallInst &
I) {
9474 const Value *
LHS =
I.getArgOperand(0), *
RHS =
I.getArgOperand(1);
9475 const Value *
Size =
I.getArgOperand(2);
9478 EVT CallVT =
DAG.getTargetLoweringInfo().getValueType(
DAG.getDataLayout(),
9484 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9488 if (Res.first.getNode()) {
9489 processIntegerCallValue(
I, Res.first,
true);
9503 auto hasFastLoadsAndCompare = [&](
unsigned NumBits) {
9504 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
9526 switch (NumBitsToCompare) {
9538 LoadVT = hasFastLoadsAndCompare(NumBitsToCompare);
9551 LoadL =
DAG.getBitcast(CmpVT, LoadL);
9552 LoadR =
DAG.getBitcast(CmpVT, LoadR);
9556 processIntegerCallValue(
I, Cmp,
false);
9565bool SelectionDAGBuilder::visitMemChrCall(
const CallInst &
I) {
9566 const Value *Src =
I.getArgOperand(0);
9567 const Value *
Char =
I.getArgOperand(1);
9568 const Value *
Length =
I.getArgOperand(2);
9570 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9571 std::pair<SDValue, SDValue> Res =
9574 MachinePointerInfo(Src));
9575 if (Res.first.getNode()) {
9589bool SelectionDAGBuilder::visitMemCCpyCall(
const CallInst &
I) {
9590 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9597 processIntegerCallValue(
I, Res.first,
true);
9609bool SelectionDAGBuilder::visitMemPCpyCall(
const CallInst &
I) {
9614 Align DstAlign =
DAG.InferPtrAlign(Dst).valueOrOne();
9615 Align SrcAlign =
DAG.InferPtrAlign(Src).valueOrOne();
9624 Root, sdl, Dst, Src,
Size, DstAlign, SrcAlign,
false,
false,
9625 nullptr, std::nullopt, MachinePointerInfo(
I.getArgOperand(0)),
9626 MachinePointerInfo(
I.getArgOperand(1)),
I.getAAMetadata());
9628 "** memcpy should not be lowered as TailCall in mempcpy context **");
9632 Size =
DAG.getSExtOrTrunc(
Size, sdl, Dst.getValueType());
9645bool SelectionDAGBuilder::visitStrCpyCall(
const CallInst &
I,
bool isStpcpy) {
9646 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9648 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9651 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), isStpcpy, &
I);
9652 if (Res.first.getNode()) {
9654 DAG.setRoot(Res.second);
9666bool SelectionDAGBuilder::visitStrCmpCall(
const CallInst &
I) {
9667 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9669 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9672 MachinePointerInfo(Arg0), MachinePointerInfo(Arg1), &
I);
9673 if (Res.first.getNode()) {
9674 processIntegerCallValue(
I, Res.first,
true);
9687bool SelectionDAGBuilder::visitStrLenCall(
const CallInst &
I) {
9688 const Value *Arg0 =
I.getArgOperand(0);
9690 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9693 if (Res.first.getNode()) {
9694 processIntegerCallValue(
I, Res.first,
false);
9707bool SelectionDAGBuilder::visitStrNLenCall(
const CallInst &
I) {
9708 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9710 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9711 std::pair<SDValue, SDValue> Res =
9714 MachinePointerInfo(Arg0));
9715 if (Res.first.getNode()) {
9716 processIntegerCallValue(
I, Res.first,
false);
9729bool SelectionDAGBuilder::visitStrstrCall(
const CallInst &
I) {
9730 const SelectionDAGTargetInfo &TSI =
DAG.getSelectionDAGInfo();
9731 const Value *Arg0 =
I.getArgOperand(0), *Arg1 =
I.getArgOperand(1);
9735 processIntegerCallValue(
I, Res.first,
false);
9747bool SelectionDAGBuilder::visitUnaryFloatCall(
const CallInst &
I,
9752 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9769bool SelectionDAGBuilder::visitBinaryFloatCall(
const CallInst &
I,
9774 if (!
I.onlyReadsMemory() ||
I.isStrictFP())
9787void SelectionDAGBuilder::visitCall(
const CallInst &
I) {
9789 if (
I.isInlineAsm()) {
9796 if (Function *
F =
I.getCalledFunction()) {
9797 if (
F->isDeclaration()) {
9799 if (
unsigned IID =
F->getIntrinsicID()) {
9800 visitIntrinsicCall(
I, IID);
9811 if (!
I.isNoBuiltin() && !
F->hasLocalLinkage() &&
F->hasName() &&
9812 LibInfo->getLibFunc(*
F, Func) &&
LibInfo->hasOptimizedCodeGen(Func)) {
9816 if (visitMemCmpBCmpCall(
I))
9819 case LibFunc_copysign:
9820 case LibFunc_copysignf:
9821 case LibFunc_copysignl:
9824 if (
I.onlyReadsMemory()) {
9869 case LibFunc_atan2f:
9870 case LibFunc_atan2l:
9895 case LibFunc_sqrt_finite:
9896 case LibFunc_sqrtf_finite:
9897 case LibFunc_sqrtl_finite:
9914 case LibFunc_exp10f:
9915 case LibFunc_exp10l:
9920 case LibFunc_ldexpf:
9921 case LibFunc_ldexpl:
9925 case LibFunc_strstr:
9926 if (visitStrstrCall(
I))
9929 case LibFunc_memcmp:
9930 if (visitMemCmpBCmpCall(
I))
9933 case LibFunc_memccpy:
9934 if (visitMemCCpyCall(
I))
9937 case LibFunc_mempcpy:
9938 if (visitMemPCpyCall(
I))
9941 case LibFunc_memchr:
9942 if (visitMemChrCall(
I))
9945 case LibFunc_strcpy:
9946 if (visitStrCpyCall(
I,
false))
9949 case LibFunc_stpcpy:
9950 if (visitStrCpyCall(
I,
true))
9953 case LibFunc_strcmp:
9954 if (visitStrCmpCall(
I))
9957 case LibFunc_strlen:
9958 if (visitStrLenCall(
I))
9961 case LibFunc_strnlen:
9962 if (visitStrNLenCall(
I))
9986 if (
I.hasDeoptState())
10003 const Value *Discriminator = PAB->Inputs[1];
10005 assert(
Key->getType()->isIntegerTy(32) &&
"Invalid ptrauth key");
10006 assert(Discriminator->getType()->isIntegerTy(64) &&
10007 "Invalid ptrauth discriminator");
10012 if (CalleeCPA->isKnownCompatibleWith(
Key, Discriminator,
10013 DAG.getDataLayout()))
10053 for (
const auto &Code : Codes)
10068 SDISelAsmOperandInfo &MatchingOpInfo,
10070 if (OpInfo.ConstraintVT == MatchingOpInfo.ConstraintVT)
10076 std::pair<unsigned, const TargetRegisterClass *> MatchRC =
10078 OpInfo.ConstraintVT);
10079 std::pair<unsigned, const TargetRegisterClass *> InputRC =
10081 MatchingOpInfo.ConstraintVT);
10082 const bool OutOpIsIntOrFP =
10083 OpInfo.ConstraintVT.isInteger() || OpInfo.ConstraintVT.isFloatingPoint();
10084 const bool InOpIsIntOrFP = MatchingOpInfo.ConstraintVT.isInteger() ||
10085 MatchingOpInfo.ConstraintVT.isFloatingPoint();
10086 if ((OutOpIsIntOrFP != InOpIsIntOrFP) || (MatchRC.second != InputRC.second)) {
10089 " with a matching output constraint of"
10090 " incompatible type!");
10092 MatchingOpInfo.ConstraintVT = OpInfo.ConstraintVT;
10099 SDISelAsmOperandInfo &OpInfo,
10112 const Value *OpVal = OpInfo.CallOperandVal;
10130 DL.getPrefTypeAlign(Ty),
false,
10133 Chain = DAG.
getTruncStore(Chain, Location, OpInfo.CallOperand, StackSlot,
10136 OpInfo.CallOperand = StackSlot;
10149static std::optional<unsigned>
10151 SDISelAsmOperandInfo &OpInfo,
10152 SDISelAsmOperandInfo &RefOpInfo) {
10163 return std::nullopt;
10167 unsigned AssignedReg;
10170 &
TRI, RefOpInfo.ConstraintCode, RefOpInfo.ConstraintVT);
10173 return std::nullopt;
10178 const MVT RegVT = *
TRI.legalclasstypes_begin(*RC);
10180 if (OpInfo.ConstraintVT != MVT::Other && RegVT != MVT::Untyped) {
10189 !
TRI.isTypeLegalForClass(*RC, OpInfo.ConstraintVT)) {
10194 if (RegVT.
getSizeInBits() == OpInfo.ConstraintVT.getSizeInBits()) {
10199 OpInfo.CallOperand =
10201 OpInfo.ConstraintVT = RegVT;
10205 }
else if (RegVT.
isInteger() && OpInfo.ConstraintVT.isFloatingPoint()) {
10208 OpInfo.CallOperand =
10210 OpInfo.ConstraintVT = VT;
10217 if (OpInfo.isMatchingInputConstraint())
10218 return std::nullopt;
10220 EVT ValueVT = OpInfo.ConstraintVT;
10221 if (OpInfo.ConstraintVT == MVT::Other)
10225 unsigned NumRegs = 1;
10226 if (OpInfo.ConstraintVT != MVT::Other)
10241 I = std::find(
I, RC->
end(), AssignedReg);
10242 if (
I == RC->
end()) {
10245 return {AssignedReg};
10249 for (; NumRegs; --NumRegs, ++
I) {
10250 assert(
I != RC->
end() &&
"Ran out of registers to allocate!");
10255 OpInfo.AssignedRegs =
RegsForValue(Regs, RegVT, ValueVT);
10256 return std::nullopt;
10261 const std::vector<SDValue> &AsmNodeOperands) {
10264 for (; OperandNo; --OperandNo) {
10266 unsigned OpFlag = AsmNodeOperands[CurOp]->getAsZExtVal();
10269 (
F.isRegDefKind() ||
F.isRegDefEarlyClobberKind() ||
F.isMemKind()) &&
10270 "Skipped past definitions?");
10271 CurOp +=
F.getNumOperandRegisters() + 1;
10279 unsigned Flags = 0;
10282 explicit ExtraFlags(
const CallBase &
Call) {
10284 if (
IA->hasSideEffects())
10286 if (
IA->isAlignStack())
10288 if (
IA->canThrow())
10295 void update(
const TargetLowering::AsmOperandInfo &OpInfo) {
10311 unsigned get()
const {
return Flags; }
10335struct ConstraintDecisionInfo {
10337 std::vector<SDValue> AsmNodeOperands;
10339 bool HasSideEffect =
false;
10342 SmallVector<char> Buffer;
10343 raw_svector_ostream ErrorMsg;
10345 ConstraintDecisionInfo() : ErrorMsg(Buffer) {}
10355 ExtraFlags &ExtraInfo) {
10356 for (
auto &
T : TargetConstraints) {
10357 Info.ConstraintOperands.push_back(SDISelAsmOperandInfo(
T));
10358 SDISelAsmOperandInfo &OpInfo = Info.ConstraintOperands.back();
10360 if (OpInfo.CallOperandVal)
10361 OpInfo.CallOperand = Builder.getValue(OpInfo.CallOperandVal);
10363 if (!Info.HasSideEffect)
10364 Info.HasSideEffect = OpInfo.hasMemory(TLI);
10376 Info.ErrorMsg <<
"constraint '" <<
T.ConstraintCode
10377 <<
"' expects an integer constant expression";
10381 ExtraInfo.update(
T);
10395 IA->collectAsmStrs(AsmStrs);
10398 for (SDISelAsmOperandInfo &OpInfo : Info.ConstraintOperands) {
10406 if (OpInfo.hasMatchingInput()) {
10407 SDISelAsmOperandInfo &
Input =
10408 Info.ConstraintOperands[OpInfo.MatchingInput];
10439 if (OpInfo.isIndirect &&
isFunction(OpInfo.CallOperand) &&
10442 OpInfo.isIndirect =
false;
10449 !OpInfo.isIndirect) {
10450 assert((OpInfo.isMultipleAlternative ||
10452 "Can only indirectify direct input operands!");
10459 OpInfo.CallOperandVal =
nullptr;
10462 OpInfo.isIndirect =
true;
10474 SDLoc DL = Builder.getCurSDLoc();
10475 for (SDISelAsmOperandInfo &OpInfo : Info.ConstraintOperands) {
10477 SDISelAsmOperandInfo &RefOpInfo =
10478 OpInfo.isMatchingInputConstraint()
10479 ? Info.ConstraintOperands[OpInfo.getMatchedOperand()]
10485 const char *
RegName =
TRI.getName(*RegError);
10486 Info.ErrorMsg <<
"register '" <<
RegName <<
"' allocated for constraint '"
10487 << OpInfo.ConstraintCode
10488 <<
"' does not match required type";
10492 auto DetectWriteToReservedRegister = [&]() {
10497 if (
Reg.isPhysical() &&
TRI.isInlineAsmReadOnlyReg(MF,
Reg)) {
10498 Info.ErrorMsg <<
"write to reserved register '"
10499 <<
TRI.getRegAsmName(
Reg) <<
"'";
10508 !OpInfo.isMatchingInputConstraint())) &&
10509 "Only address as input operand is allowed.");
10511 switch (OpInfo.Type) {
10517 "Failed to convert memory constraint code to constraint id.");
10522 Info.AsmNodeOperands.push_back(
10524 Info.AsmNodeOperands.push_back(OpInfo.CallOperand);
10529 if (OpInfo.AssignedRegs.Regs.empty()) {
10530 Info.ErrorMsg <<
"could not allocate output register for "
10531 <<
"constraint '" << OpInfo.ConstraintCode <<
"'";
10535 if (DetectWriteToReservedRegister())
10540 OpInfo.AssignedRegs.AddInlineAsmOperands(
10543 false, 0,
DL, DAG, Info.AsmNodeOperands);
10549 SDValue InOperandVal = OpInfo.CallOperand;
10551 if (OpInfo.isMatchingInputConstraint()) {
10555 Info.AsmNodeOperands);
10557 if (Flag.isRegDefKind() || Flag.isRegDefEarlyClobberKind()) {
10558 if (OpInfo.isIndirect) {
10560 Info.ErrorMsg <<
"inline asm not supported yet: cannot handle "
10561 <<
"tied indirect register inputs";
10571 MVT RegVT = R->getSimpleValueType(0);
10575 :
TRI.getMinimalPhysRegClass(TiedReg);
10576 for (
unsigned I = 0,
E = Flag.getNumOperandRegisters();
I !=
E; ++
I)
10583 &Info.Glue, &
Call);
10585 OpInfo.getMatchedOperand(),
DL, DAG,
10586 Info.AsmNodeOperands);
10590 assert(Flag.isMemKind() &&
"Unknown matching constraint!");
10591 assert(Flag.getNumOperandRegisters() == 1 &&
10592 "Unexpected number of operands");
10596 Flag.clearMemConstraint();
10597 Flag.setMatchingOp(OpInfo.getMatchedOperand());
10600 Info.AsmNodeOperands.push_back(Info.AsmNodeOperands[CurOp + 1]);
10611 std::vector<SDValue>
Ops;
10617 Info.ErrorMsg <<
"value out of range for constraint '"
10618 << OpInfo.ConstraintCode <<
"'";
10622 Info.ErrorMsg <<
"invalid operand for inline asm constraint '"
10623 << OpInfo.ConstraintCode <<
"'";
10636 assert((OpInfo.isIndirect ||
10638 "Operand must be indirect to be a mem!");
10641 "Memory operands expect pointer values");
10646 "Failed to convert memory constraint code to constraint id.");
10651 Info.AsmNodeOperands.push_back(
10653 Info.AsmNodeOperands.push_back(InOperandVal);
10661 "Failed to convert memory constraint code to constraint id.");
10665 SDValue AsmOp = InOperandVal;
10677 Info.AsmNodeOperands.push_back(
10679 Info.AsmNodeOperands.push_back(AsmOp);
10685 Info.ErrorMsg <<
"unknown asm constraint '" << OpInfo.ConstraintCode
10691 if (OpInfo.isIndirect) {
10692 Info.ErrorMsg <<
"cannot handle indirect register inputs yet for "
10693 <<
"constraint '" << OpInfo.ConstraintCode <<
"'";
10698 if (OpInfo.AssignedRegs.Regs.empty()) {
10699 Info.ErrorMsg <<
"could not allocate input reg for constraint '"
10700 << OpInfo.ConstraintCode <<
"'";
10704 if (DetectWriteToReservedRegister())
10707 OpInfo.AssignedRegs.getCopyToRegs(InOperandVal, DAG,
DL, Info.Chain,
10708 &Info.Glue, &
Call);
10709 OpInfo.AssignedRegs.AddInlineAsmOperands(
10717 if (!OpInfo.AssignedRegs.Regs.empty())
10718 OpInfo.AssignedRegs.AddInlineAsmOperands(
10735 ExtraFlags ExtraInfo(
Call);
10738 Info.HasSideEffect = IA->hasSideEffects();
10744 Info.Chain = Info.HasSideEffect ? Builder.getRoot() : DAG.
getRoot();
10748 if (IsCallBr || EmitEHLabels)
10752 Info.Chain = Builder.getControlRoot();
10755 Info.Chain = Builder.lowerStartEH(Info.Chain, EHPadBB, Info.BeginLabel);
10761 Info.AsmNodeOperands.push_back(
SDValue());
10768 const MDNode *SrcLoc =
Call.getMetadata(
"srcloc");
10769 Info.AsmNodeOperands.push_back(DAG.
getMDNode(SrcLoc));
10773 Info.AsmNodeOperands.push_back(
10782void SelectionDAGBuilder::visitInlineAsm(
const CallBase &
Call,
10784 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10786 DAG.getDataLayout(),
DAG.getSubtarget().getRegisterInfo(),
Call);
10789 "InvokeInst must have an EHPadBB");
10791 ConstraintDecisionInfo
Info;
10794 return emitInlineAsmError(
Call,
Info.ErrorMsg.str());
10802 Info.AsmNodeOperands.push_back(Glue);
10808 Info.AsmNodeOperands);
10820 ResultTypes = StructResult->elements();
10821 else if (!CallResultType->
isVoidTy())
10822 ResultTypes =
ArrayRef(CallResultType);
10824 auto CurResultType = ResultTypes.
begin();
10825 auto handleRegAssign = [&](
SDValue V) {
10826 assert(CurResultType != ResultTypes.
end() &&
"Unexpected value");
10827 assert((*CurResultType)->isSized() &&
"Unexpected unsized type");
10828 EVT ResultVT = TLI.
getValueType(
DAG.getDataLayout(), *CurResultType);
10840 if (ResultVT !=
V.getValueType() &&
10843 else if (ResultVT !=
V.getValueType() && ResultVT.
isInteger() &&
10844 V.getValueType().isInteger()) {
10850 assert(ResultVT ==
V.getValueType() &&
"Asm result value mismatch!");
10856 for (SDISelAsmOperandInfo &OpInfo :
Info.ConstraintOperands) {
10860 if (OpInfo.AssignedRegs.
Regs.empty())
10863 switch (OpInfo.ConstraintType) {
10867 Chain, &Glue, &
Call);
10879 assert(
false &&
"Unexpected unknown constraint");
10883 if (OpInfo.isIndirect) {
10884 const Value *Ptr = OpInfo.CallOperandVal;
10885 assert(Ptr &&
"Expected value CallOperandVal for indirect asm operand");
10887 MachinePointerInfo(Ptr));
10894 handleRegAssign(V);
10896 handleRegAssign(Val);
10902 if (!ResultValues.
empty()) {
10903 assert(CurResultType == ResultTypes.
end() &&
10904 "Mismatch in number of ResultTypes");
10906 "Mismatch in number of output operands in asm result");
10909 DAG.getVTList(ResultVTs), ResultValues);
10914 if (!OutChains.
empty())
10918 Chain = lowerEndEH(Chain,
II, EHPadBB,
Info.BeginLabel);
10921 if (ResultValues.
empty() ||
Info.HasSideEffect || !OutChains.
empty() ||
10923 DAG.setRoot(Chain);
10926void SelectionDAGBuilder::emitInlineAsmError(
const CallBase &
Call,
10927 const Twine &Message) {
10928 LLVMContext &Ctx = *
DAG.getContext();
10932 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10936 if (ValueVTs.
empty())
10940 for (
const EVT &VT : ValueVTs)
10941 Ops.push_back(
DAG.getUNDEF(VT));
10946void SelectionDAGBuilder::visitVAStart(
const CallInst &
I) {
10950 DAG.getSrcValue(
I.getArgOperand(0))));
10953void SelectionDAGBuilder::visitVAArg(
const VAArgInst &
I) {
10954 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
10955 const DataLayout &
DL =
DAG.getDataLayout();
10959 DL.getABITypeAlign(
I.getType()).value());
10960 DAG.setRoot(
V.getValue(1));
10962 if (
I.getType()->isPointerTy())
10963 V =
DAG.getPtrExtOrTrunc(
10968void SelectionDAGBuilder::visitVAEnd(
const CallInst &
I) {
10972 DAG.getSrcValue(
I.getArgOperand(0))));
10975void SelectionDAGBuilder::visitVACopy(
const CallInst &
I) {
10980 DAG.getSrcValue(
I.getArgOperand(0)),
10981 DAG.getSrcValue(
I.getArgOperand(1))));
10987 std::optional<ConstantRange> CR =
getRange(
I);
10989 if (!CR || CR->isFullSet() || CR->isEmptySet() || CR->isUpperWrapped())
10992 APInt Hi = CR->getUnsignedMax();
10993 unsigned Bits = std::max(
Hi.getActiveBits(),
11001 DAG.getValueType(SmallVT));
11002 unsigned NumVals =
Op.getNode()->getNumValues();
11008 Ops.push_back(ZExt);
11009 for (
unsigned I = 1;
I != NumVals; ++
I)
11010 Ops.push_back(
Op.getValue(
I));
11012 return DAG.getMergeValues(
Ops,
SL);
11022 SDValue TestConst =
DAG.getTargetConstant(Classes,
SDLoc(), MVT::i32);
11030 for (
unsigned I = 0, E =
Ops.size();
I != E; ++
I) {
11033 MergeOp, TestConst);
11036 return DAG.getMergeValues(
Ops,
SL);
11047 unsigned ArgIdx,
unsigned NumArgs,
SDValue Callee,
Type *ReturnTy,
11050 Args.reserve(NumArgs);
11054 for (
unsigned ArgI = ArgIdx, ArgE = ArgIdx + NumArgs;
11055 ArgI != ArgE; ++ArgI) {
11056 const Value *V =
Call->getOperand(ArgI);
11058 assert(!V->getType()->isEmptyTy() &&
"Empty type passed to intrinsic.");
11061 Entry.setAttributes(
Call, ArgI);
11062 Args.push_back(Entry);
11067 .
setCallee(
Call->getCallingConv(), ReturnTy, Callee, std::move(Args),
11096 for (
unsigned I = StartIdx;
I <
Call.arg_size();
I++) {
11105 Ops.push_back(Builder.getValue(
Call.getArgOperand(
I)));
11111void SelectionDAGBuilder::visitStackmap(
const CallInst &CI) {
11137 Ops.push_back(Chain);
11138 Ops.push_back(InGlue);
11145 assert(
ID.getValueType() == MVT::i64);
11147 DAG.getTargetConstant(
ID->getAsZExtVal(),
DL,
ID.getValueType());
11148 Ops.push_back(IDConst);
11154 Ops.push_back(ShadConst);
11160 SDVTList NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
11164 Chain =
DAG.getCALLSEQ_END(Chain, 0, 0, InGlue,
DL);
11169 DAG.setRoot(Chain);
11172 FuncInfo.MF->getFrameInfo().setHasStackMap();
11176void SelectionDAGBuilder::visitPatchpoint(
const CallBase &CB,
11193 Callee =
DAG.getIntPtrConstant(ConstCallee->getZExtValue(), dl,
11196 Callee =
DAG.getTargetGlobalAddress(SymbolicCallee->getGlobal(),
11197 SDLoc(SymbolicCallee),
11198 SymbolicCallee->getValueType(0));
11208 "Not enough arguments provided to the patchpoint intrinsic");
11211 unsigned NumCallArgs = IsAnyRegCC ? 0 : NumArgs;
11215 TargetLowering::CallLoweringInfo CLI(
DAG);
11220 SDNode *CallEnd =
Result.second.getNode();
11229 "Expected a callseq node.");
11231 bool HasGlue =
Call->getGluedNode();
11256 Ops.push_back(Callee);
11262 NumCallRegArgs = IsAnyRegCC ? NumArgs : NumCallRegArgs;
11263 Ops.push_back(
DAG.getTargetConstant(NumCallRegArgs, dl, MVT::i32));
11266 Ops.push_back(
DAG.getTargetConstant((
unsigned)CC, dl, MVT::i32));
11271 for (
unsigned i = NumMetaOpers, e = NumMetaOpers + NumArgs; i !=
e; ++i)
11282 if (IsAnyRegCC && HasDef) {
11284 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11287 assert(ValueVTs.
size() == 1 &&
"Expected only one return value type.");
11292 NodeTys =
DAG.getVTList(ValueVTs);
11294 NodeTys =
DAG.getVTList(MVT::Other, MVT::Glue);
11311 if (IsAnyRegCC && HasDef) {
11314 DAG.ReplaceAllUsesOfValuesWith(From, To, 2);
11320 FuncInfo.MF->getFrameInfo().setHasPatchPoint();
11323void SelectionDAGBuilder::visitVectorReduce(
const CallInst &
I,
11325 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
11328 if (
I.arg_size() > 1)
11333 SDNodeFlags SDFlags;
11337 switch (Intrinsic) {
11338 case Intrinsic::vector_reduce_fadd:
11346 case Intrinsic::vector_reduce_fmul:
11354 case Intrinsic::vector_reduce_add:
11357 case Intrinsic::vector_reduce_mul:
11360 case Intrinsic::vector_reduce_and:
11363 case Intrinsic::vector_reduce_or:
11366 case Intrinsic::vector_reduce_xor:
11369 case Intrinsic::vector_reduce_smax:
11372 case Intrinsic::vector_reduce_smin:
11375 case Intrinsic::vector_reduce_umax:
11378 case Intrinsic::vector_reduce_umin:
11381 case Intrinsic::vector_reduce_fmax:
11384 case Intrinsic::vector_reduce_fmin:
11387 case Intrinsic::vector_reduce_fmaximum:
11390 case Intrinsic::vector_reduce_fminimum:
11404 Attrs.push_back(Attribute::SExt);
11406 Attrs.push_back(Attribute::ZExt);
11408 Attrs.push_back(Attribute::InReg);
11410 return AttributeList::get(CLI.
RetTy->
getContext(), AttributeList::ReturnIndex,
11418std::pair<SDValue, SDValue>
11432 "Only supported for non-aggregate returns");
11435 for (
Type *Ty : RetOrigTys)
11444 RetOrigTys.
swap(OldRetOrigTys);
11445 RetVTs.
swap(OldRetVTs);
11446 Offsets.swap(OldOffsets);
11448 for (
size_t i = 0, e = OldRetVTs.
size(); i != e; ++i) {
11449 EVT RetVT = OldRetVTs[i];
11453 unsigned RegisterVTByteSZ = RegisterVT.
getSizeInBits() / 8;
11454 RetOrigTys.
append(NumRegs, OldRetOrigTys[i]);
11455 RetVTs.
append(NumRegs, RegisterVT);
11456 for (
unsigned j = 0; j != NumRegs; ++j)
11469 int DemoteStackIdx = -100;
11482 ArgListEntry Entry(DemoteStackSlot, StackSlotPtrType);
11483 Entry.IsSRet =
true;
11484 Entry.Alignment = Alignment;
11496 for (
unsigned I = 0, E = RetVTs.
size();
I != E; ++
I) {
11498 if (NeedsRegBlock) {
11499 Flags.setInConsecutiveRegs();
11500 if (
I == RetVTs.
size() - 1)
11501 Flags.setInConsecutiveRegsLast();
11503 EVT VT = RetVTs[
I];
11507 for (
unsigned i = 0; i != NumRegs; ++i) {
11521 CLI.
Ins.push_back(Ret);
11530 if (Arg.IsSwiftError) {
11536 CLI.
Ins.push_back(Ret);
11544 for (
unsigned i = 0, e = Args.size(); i != e; ++i) {
11548 Type *FinalType = Args[i].Ty;
11549 if (Args[i].IsByVal)
11550 FinalType = Args[i].IndirectType;
11553 for (
unsigned Value = 0, NumValues = OrigArgTys.
size();
Value != NumValues;
11556 Type *ArgTy = OrigArgTy;
11557 if (Args[i].Ty != Args[i].OrigTy) {
11558 assert(
Value == 0 &&
"Only supported for non-aggregate arguments");
11559 ArgTy = Args[i].Ty;
11564 Args[i].Node.getResNo() +
Value);
11571 Flags.setOrigAlign(OriginalAlignment);
11576 Flags.setPointer();
11579 if (Args[i].IsZExt)
11581 if (Args[i].IsSExt)
11583 if (Args[i].IsNoExt)
11585 if (Args[i].IsInReg) {
11592 Flags.setHvaStart();
11598 if (Args[i].IsSRet)
11600 if (Args[i].IsSwiftSelf)
11601 Flags.setSwiftSelf();
11602 if (Args[i].IsSwiftAsync)
11603 Flags.setSwiftAsync();
11604 if (Args[i].IsSwiftError)
11605 Flags.setSwiftError();
11606 if (Args[i].IsCFGuardTarget)
11607 Flags.setCFGuardTarget();
11608 if (Args[i].IsByVal)
11610 if (Args[i].IsByRef)
11612 if (Args[i].IsPreallocated) {
11613 Flags.setPreallocated();
11621 if (Args[i].IsInAlloca) {
11622 Flags.setInAlloca();
11631 if (Args[i].IsByVal || Args[i].IsInAlloca || Args[i].IsPreallocated) {
11632 unsigned FrameSize =
DL.getTypeAllocSize(Args[i].IndirectType);
11633 Flags.setByValSize(FrameSize);
11636 if (
auto MA = Args[i].Alignment)
11640 }
else if (
auto MA = Args[i].Alignment) {
11643 MemAlign = OriginalAlignment;
11645 Flags.setMemAlign(MemAlign);
11646 if (Args[i].IsNest)
11649 Flags.setInConsecutiveRegs();
11652 unsigned NumParts =
11657 if (Args[i].IsSExt)
11659 else if (Args[i].IsZExt)
11664 if (Args[i].IsReturned && !
Op.getValueType().isVector() &&
11669 Args[i].Ty->getPointerAddressSpace())) &&
11670 RetVTs.
size() == NumValues &&
"unexpected use of 'returned'");
11683 CLI.
RetZExt == Args[i].IsZExt))
11684 Flags.setReturned();
11690 for (
unsigned j = 0; j != NumParts; ++j) {
11696 j * Parts[j].
getValueType().getStoreSize().getKnownMinValue());
11697 if (NumParts > 1 && j == 0)
11701 if (j == NumParts - 1)
11705 CLI.
Outs.push_back(MyFlags);
11706 CLI.
OutVals.push_back(Parts[j]);
11709 if (NeedsRegBlock &&
Value == NumValues - 1)
11710 CLI.
Outs[CLI.
Outs.size() - 1].Flags.setInConsecutiveRegsLast();
11722 "LowerCall didn't return a valid chain!");
11724 "LowerCall emitted a return value for a tail call!");
11726 "LowerCall didn't emit the correct number of values!");
11738 for (
unsigned i = 0, e = CLI.
Ins.size(); i != e; ++i) {
11739 assert(InVals[i].
getNode() &&
"LowerCall emitted a null value!");
11740 assert(
EVT(CLI.
Ins[i].VT) == InVals[i].getValueType() &&
11741 "LowerCall emitted a value with the wrong type!");
11751 unsigned NumValues = RetVTs.
size();
11752 ReturnValues.
resize(NumValues);
11759 for (
unsigned i = 0; i < NumValues; ++i) {
11766 DemoteStackIdx, Offsets[i]),
11768 ReturnValues[i] = L;
11769 Chains[i] = L.getValue(1);
11776 std::optional<ISD::NodeType> AssertOp;
11781 unsigned CurReg = 0;
11782 for (
EVT VT : RetVTs) {
11788 CLI.
DAG, CLI.
DL, &InVals[CurReg], NumRegs, RegisterVT, VT,
nullptr,
11796 if (ReturnValues.
empty())
11802 return std::make_pair(Res, CLI.
Chain);
11819 if (
N->getNumValues() == 1) {
11827 "Lowering returned the wrong number of results!");
11830 for (
unsigned I = 0, E =
N->getNumValues();
I != E; ++
I)
11844 "Copy from a reg to the same reg!");
11845 assert(!Reg.isPhysical() &&
"Is a physreg");
11851 RegsForValue RFV(V->getContext(), TLI,
DAG.getDataLayout(), Reg, V->getType(),
11856 auto PreferredExtendIt =
FuncInfo.PreferredExtendType.find(V);
11857 if (PreferredExtendIt !=
FuncInfo.PreferredExtendType.end())
11858 ExtendType = PreferredExtendIt->second;
11861 PendingExports.push_back(Chain);
11873 return A->use_empty();
11875 const BasicBlock &Entry =
A->getParent()->front();
11876 for (
const User *U :
A->users())
11885 std::pair<const AllocaInst *, const StoreInst *>>;
11897 enum StaticAllocaInfo {
Unknown, Clobbered, Elidable };
11899 unsigned NumArgs = FuncInfo->
Fn->
arg_size();
11900 StaticAllocas.
reserve(NumArgs * 2);
11902 auto GetInfoIfStaticAlloca = [&](
const Value *V) -> StaticAllocaInfo * {
11905 V = V->stripPointerCasts();
11907 if (!AI || !AI->isStaticAlloca() || !FuncInfo->
StaticAllocaMap.count(AI))
11910 return &Iter.first->second;
11927 if (
I.isDebugOrPseudoInst())
11931 for (
const Use &U :
I.operands()) {
11932 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(U))
11933 *Info = StaticAllocaInfo::Clobbered;
11939 if (StaticAllocaInfo *Info = GetInfoIfStaticAlloca(
SI->getValueOperand()))
11940 *Info = StaticAllocaInfo::Clobbered;
11943 const Value *Dst =
SI->getPointerOperand()->stripPointerCasts();
11944 StaticAllocaInfo *Info = GetInfoIfStaticAlloca(Dst);
11950 if (*Info != StaticAllocaInfo::Unknown)
11958 const Value *Val =
SI->getValueOperand()->stripPointerCasts();
11961 if (!Arg || Arg->hasPassPointeeByValueCopyAttr() ||
11963 DL.getTypeStoreSize(Arg->
getType()) != *AllocaSize ||
11964 !
DL.typeSizeEqualsStoreSize(Arg->
getType()) ||
11965 ArgCopyElisionCandidates.count(Arg)) {
11966 *Info = StaticAllocaInfo::Clobbered;
11970 LLVM_DEBUG(
dbgs() <<
"Found argument copy elision candidate: " << *AI
11974 *Info = StaticAllocaInfo::Elidable;
11975 ArgCopyElisionCandidates.insert({Arg, {AI,
SI}});
11980 if (ArgCopyElisionCandidates.size() == NumArgs)
12004 auto ArgCopyIter = ArgCopyElisionCandidates.find(&Arg);
12005 assert(ArgCopyIter != ArgCopyElisionCandidates.end());
12006 const AllocaInst *AI = ArgCopyIter->second.first;
12007 int FixedIndex = FINode->getIndex();
12009 int OldIndex = AllocaIndex;
12013 dbgs() <<
" argument copy elision failed due to bad fixed stack "
12019 LLVM_DEBUG(
dbgs() <<
" argument copy elision failed: alignment of alloca "
12020 "greater than stack argument alignment ("
12021 <<
DebugStr(RequiredAlignment) <<
" vs "
12029 dbgs() <<
"Eliding argument copy from " << Arg <<
" to " << *AI <<
'\n'
12030 <<
" Replacing frame index " << OldIndex <<
" with " << FixedIndex
12036 AllocaIndex = FixedIndex;
12037 ArgCopyElisionFrameIndexMap.
insert({OldIndex, FixedIndex});
12038 for (
SDValue ArgVal : ArgVals)
12042 const StoreInst *
SI = ArgCopyIter->second.second;
12055void SelectionDAGISel::LowerArguments(
const Function &
F) {
12056 SelectionDAG &DAG =
SDB->DAG;
12057 SDLoc dl =
SDB->getCurSDLoc();
12062 if (
F.hasFnAttribute(Attribute::Naked))
12067 MVT ValueVT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
12069 ISD::ArgFlagsTy
Flags;
12071 MVT RegisterVT =
TLI->getRegisterType(*DAG.
getContext(), ValueVT);
12072 ISD::InputArg RetArg(Flags, RegisterVT, ValueVT,
F.getReturnType(),
true,
12082 ArgCopyElisionCandidates);
12085 for (
const Argument &Arg :
F.args()) {
12086 unsigned ArgNo = Arg.getArgNo();
12089 bool isArgValueUsed = !Arg.
use_empty();
12091 if (Arg.hasAttribute(Attribute::ByVal))
12092 FinalType = Arg.getParamByValType();
12093 bool NeedsRegBlock =
TLI->functionArgumentNeedsConsecutiveRegisters(
12094 FinalType,
F.getCallingConv(),
F.isVarArg(),
DL);
12095 for (
unsigned Value = 0, NumValues =
Types.size();
Value != NumValues;
12098 EVT VT =
TLI->getValueType(
DL, ArgTy);
12099 ISD::ArgFlagsTy
Flags;
12102 Flags.setPointer();
12105 if (Arg.hasAttribute(Attribute::ZExt))
12107 if (Arg.hasAttribute(Attribute::SExt))
12109 if (Arg.hasAttribute(Attribute::InReg)) {
12116 Flags.setHvaStart();
12122 if (Arg.hasAttribute(Attribute::StructRet))
12124 if (Arg.hasAttribute(Attribute::SwiftSelf))
12125 Flags.setSwiftSelf();
12126 if (Arg.hasAttribute(Attribute::SwiftAsync))
12127 Flags.setSwiftAsync();
12128 if (Arg.hasAttribute(Attribute::SwiftError))
12129 Flags.setSwiftError();
12130 if (Arg.hasAttribute(Attribute::ByVal))
12132 if (Arg.hasAttribute(Attribute::ByRef))
12134 if (Arg.hasAttribute(Attribute::InAlloca)) {
12135 Flags.setInAlloca();
12143 if (Arg.hasAttribute(Attribute::Preallocated)) {
12144 Flags.setPreallocated();
12156 const Align OriginalAlignment(
12157 TLI->getABIAlignmentForCallingConv(ArgTy,
DL));
12158 Flags.setOrigAlign(OriginalAlignment);
12161 Type *ArgMemTy =
nullptr;
12162 if (
Flags.isByVal() ||
Flags.isInAlloca() ||
Flags.isPreallocated() ||
12165 ArgMemTy = Arg.getPointeeInMemoryValueType();
12167 uint64_t MemSize =
DL.getTypeAllocSize(ArgMemTy);
12172 if (
auto ParamAlign = Arg.getParamStackAlign())
12173 MemAlign = *ParamAlign;
12174 else if ((ParamAlign = Arg.getParamAlign()))
12175 MemAlign = *ParamAlign;
12177 MemAlign =
TLI->getByValTypeAlignment(ArgMemTy,
DL);
12178 if (
Flags.isByRef())
12179 Flags.setByRefSize(MemSize);
12181 Flags.setByValSize(MemSize);
12182 }
else if (
auto ParamAlign = Arg.getParamStackAlign()) {
12183 MemAlign = *ParamAlign;
12185 MemAlign = OriginalAlignment;
12187 Flags.setMemAlign(MemAlign);
12189 if (Arg.hasAttribute(Attribute::Nest))
12192 Flags.setInConsecutiveRegs();
12193 if (ArgCopyElisionCandidates.count(&Arg))
12194 Flags.setCopyElisionCandidate();
12195 if (Arg.hasAttribute(Attribute::Returned))
12196 Flags.setReturned();
12198 MVT RegisterVT =
TLI->getRegisterTypeForCallingConv(
12199 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12200 unsigned NumRegs =
TLI->getNumRegistersForCallingConv(
12201 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12202 for (
unsigned i = 0; i != NumRegs; ++i) {
12206 ISD::InputArg MyFlags(
12207 Flags, RegisterVT, VT, ArgTy, isArgValueUsed, ArgNo,
12209 if (NumRegs > 1 && i == 0)
12210 MyFlags.Flags.setSplit();
12213 MyFlags.Flags.setOrigAlign(
Align(1));
12214 if (i == NumRegs - 1)
12215 MyFlags.Flags.setSplitEnd();
12219 if (NeedsRegBlock &&
Value == NumValues - 1)
12220 Ins[Ins.
size() - 1].Flags.setInConsecutiveRegsLast();
12226 SDValue NewRoot =
TLI->LowerFormalArguments(
12227 DAG.
getRoot(),
F.getCallingConv(),
F.isVarArg(), Ins, dl, DAG, InVals);
12231 "LowerFormalArguments didn't return a valid chain!");
12233 "LowerFormalArguments didn't emit the correct number of values!");
12235 "LowerFormalArguments emitted a null value!");
12245 MVT VT =
TLI->getPointerTy(
DL,
DL.getAllocaAddrSpace());
12246 MVT RegVT =
TLI->getRegisterType(*
CurDAG->getContext(), VT);
12247 std::optional<ISD::NodeType> AssertOp;
12250 F.getCallingConv(), AssertOp);
12252 MachineFunction&
MF =
SDB->DAG.getMachineFunction();
12253 MachineRegisterInfo&
RegInfo =
MF.getRegInfo();
12255 RegInfo.createVirtualRegister(
TLI->getRegClassFor(RegVT));
12256 FuncInfo->DemoteRegister = SRetReg;
12258 SDB->DAG.getCopyToReg(NewRoot,
SDB->getCurSDLoc(), SRetReg, ArgValue);
12266 DenseMap<int, int> ArgCopyElisionFrameIndexMap;
12267 for (
const Argument &Arg :
F.args()) {
12271 unsigned NumValues = ValueVTs.
size();
12272 if (NumValues == 0)
12279 if (Ins[i].
Flags.isCopyElisionCandidate()) {
12280 unsigned NumParts = 0;
12281 for (EVT VT : ValueVTs)
12282 NumParts +=
TLI->getNumRegistersForCallingConv(*
CurDAG->getContext(),
12283 F.getCallingConv(), VT);
12287 ArrayRef(&InVals[i], NumParts), ArgHasUses);
12292 bool isSwiftErrorArg =
12293 TLI->supportSwiftError() &&
12294 Arg.hasAttribute(Attribute::SwiftError);
12295 if (!ArgHasUses && !isSwiftErrorArg) {
12296 SDB->setUnusedArgValue(&Arg, InVals[i]);
12299 if (FrameIndexSDNode *FI =
12301 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12304 for (
unsigned Val = 0; Val != NumValues; ++Val) {
12305 EVT VT = ValueVTs[Val];
12306 MVT PartVT =
TLI->getRegisterTypeForCallingConv(*
CurDAG->getContext(),
12307 F.getCallingConv(), VT);
12308 unsigned NumParts =
TLI->getNumRegistersForCallingConv(
12309 *
CurDAG->getContext(),
F.getCallingConv(), VT);
12314 if (ArgHasUses || isSwiftErrorArg) {
12315 std::optional<ISD::NodeType> AssertOp;
12316 if (Arg.hasAttribute(Attribute::SExt))
12318 else if (Arg.hasAttribute(Attribute::ZExt))
12323 NewRoot,
F.getCallingConv(), AssertOp);
12326 if (NoFPClass !=
fcNone) {
12328 static_cast<uint64_t
>(NoFPClass), dl, MVT::i32);
12330 OutVal, SDNoFPClass);
12339 if (ArgValues.
empty())
12343 if (FrameIndexSDNode *FI =
12345 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12348 SDB->getCurSDLoc());
12350 SDB->setValue(&Arg, Res);
12360 if (LoadSDNode *LNode =
12362 if (FrameIndexSDNode *FI =
12364 FuncInfo->setArgumentFrameIndex(&Arg, FI->getIndex());
12392 FuncInfo->InitializeRegForValue(&Arg);
12393 SDB->CopyToExportRegsIfNeeded(&Arg);
12397 if (!Chains.
empty()) {
12404 assert(i == InVals.
size() &&
"Argument register count mismatch!");
12408 if (!ArgCopyElisionFrameIndexMap.
empty()) {
12409 for (MachineFunction::VariableDbgInfo &VI :
12410 MF->getInStackSlotVariableDbgInfo()) {
12411 auto I = ArgCopyElisionFrameIndexMap.
find(
VI.getStackSlot());
12412 if (
I != ArgCopyElisionFrameIndexMap.
end())
12413 VI.updateStackSlot(
I->second);
12428SelectionDAGBuilder::HandlePHINodesInSuccessorBlocks(
const BasicBlock *LLVMBB) {
12429 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
12431 SmallPtrSet<MachineBasicBlock *, 4> SuccsHandled;
12437 MachineBasicBlock *SuccMBB =
FuncInfo.getMBB(SuccBB);
12441 if (!SuccsHandled.
insert(SuccMBB).second)
12449 for (
const PHINode &PN : SuccBB->phis()) {
12451 if (PN.use_empty())
12455 if (PN.getType()->isEmptyTy())
12459 const Value *PHIOp = PN.getIncomingValueForBlock(LLVMBB);
12464 RegOut =
FuncInfo.CreateRegs(&PN);
12475 auto I =
FuncInfo.ValueMap.find(PHIOp);
12481 "Didn't codegen value into a register!??");
12491 for (EVT VT : ValueVTs) {
12493 for (
unsigned i = 0; i != NumRegisters; ++i)
12495 Reg += NumRegisters;
12515void SelectionDAGBuilder::updateDAGForMaybeTailCall(
SDValue MaybeTC) {
12517 if (MaybeTC.
getNode() !=
nullptr)
12518 DAG.setRoot(MaybeTC);
12523void SelectionDAGBuilder::lowerWorkItem(SwitchWorkListItem W,
Value *
Cond,
12526 MachineFunction *CurMF =
FuncInfo.MF;
12527 MachineBasicBlock *NextMBB =
nullptr;
12532 unsigned Size =
W.LastCluster -
W.FirstCluster + 1;
12534 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12536 if (
Size == 2 &&
W.MBB == SwitchMBB) {
12544 CaseCluster &
Small = *
W.FirstCluster;
12545 CaseCluster &
Big = *
W.LastCluster;
12549 const APInt &SmallValue =
Small.Low->getValue();
12550 const APInt &BigValue =
Big.Low->getValue();
12553 APInt CommonBit = BigValue ^ SmallValue;
12560 DAG.getConstant(CommonBit,
DL, VT));
12562 DL, MVT::i1,
Or,
DAG.getConstant(BigValue | SmallValue,
DL, VT),
12568 addSuccessorWithProb(SwitchMBB,
Small.MBB,
Small.Prob +
Big.Prob);
12570 addSuccessorWithProb(
12571 SwitchMBB, DefaultMBB,
12575 addSuccessorWithProb(SwitchMBB, DefaultMBB);
12583 DAG.getBasicBlock(DefaultMBB));
12585 DAG.setRoot(BrCond);
12597 [](
const CaseCluster &a,
const CaseCluster &b) {
12598 return a.Prob != b.Prob ?
12600 a.Low->getValue().slt(b.Low->getValue());
12607 if (
I->Prob >
W.LastCluster->Prob)
12609 if (
I->Kind ==
CC_Range &&
I->MBB == NextMBB) {
12617 BranchProbability DefaultProb =
W.DefaultProb;
12618 BranchProbability UnhandledProbs = DefaultProb;
12620 UnhandledProbs +=
I->Prob;
12622 MachineBasicBlock *CurMBB =
W.MBB;
12624 bool FallthroughUnreachable =
false;
12625 MachineBasicBlock *Fallthrough;
12626 if (
I ==
W.LastCluster) {
12628 Fallthrough = DefaultMBB;
12633 CurMF->
insert(BBI, Fallthrough);
12637 UnhandledProbs -=
I->Prob;
12642 JumpTableHeader *JTH = &
SL->JTCases[
I->JTCasesIndex].first;
12643 SwitchCG::JumpTable *JT = &
SL->JTCases[
I->JTCasesIndex].second;
12646 MachineBasicBlock *JumpMBB = JT->
MBB;
12647 CurMF->
insert(BBI, JumpMBB);
12649 auto JumpProb =
I->Prob;
12650 auto FallthroughProb = UnhandledProbs;
12658 if (*SI == DefaultMBB) {
12659 JumpProb += DefaultProb / 2;
12660 FallthroughProb -= DefaultProb / 2;
12678 if (FallthroughUnreachable) {
12685 addSuccessorWithProb(CurMBB, Fallthrough, FallthroughProb);
12686 addSuccessorWithProb(CurMBB, JumpMBB, JumpProb);
12695 if (CurMBB == SwitchMBB) {
12703 BitTestBlock *BTB = &
SL->BitTestCases[
I->BTCasesIndex];
12706 for (BitTestCase &BTC : BTB->
Cases)
12718 BTB->
Prob += DefaultProb / 2;
12722 if (FallthroughUnreachable)
12726 if (CurMBB == SwitchMBB) {
12733 const Value *
RHS, *
LHS, *MHS;
12735 if (
I->Low ==
I->High) {
12750 if (FallthroughUnreachable)
12754 CaseBlock CB(CC,
LHS,
RHS, MHS,
I->MBB, Fallthrough, CurMBB,
12757 if (CurMBB == SwitchMBB)
12760 SL->SwitchCases.push_back(CB);
12765 CurMBB = Fallthrough;
12769void SelectionDAGBuilder::splitWorkItem(
SwitchWorkList &WorkList,
12770 const SwitchWorkListItem &W,
12773 assert(
W.FirstCluster->Low->getValue().slt(
W.LastCluster->Low->getValue()) &&
12774 "Clusters not sorted?");
12775 assert(
W.LastCluster -
W.FirstCluster + 1 >= 2 &&
"Too small to split!");
12777 auto [LastLeft, FirstRight, LeftProb, RightProb] =
12778 SL->computeSplitWorkItemInfo(W);
12783 assert(PivotCluster >
W.FirstCluster);
12784 assert(PivotCluster <=
W.LastCluster);
12789 const ConstantInt *Pivot = PivotCluster->Low;
12798 MachineBasicBlock *LeftMBB;
12799 if (FirstLeft == LastLeft && FirstLeft->Kind ==
CC_Range &&
12800 FirstLeft->Low ==
W.GE &&
12801 (FirstLeft->High->getValue() + 1LL) == Pivot->
getValue()) {
12802 LeftMBB = FirstLeft->MBB;
12804 LeftMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12805 FuncInfo.MF->insert(BBI, LeftMBB);
12807 {LeftMBB, FirstLeft, LastLeft,
W.GE, Pivot,
W.DefaultProb / 2});
12815 MachineBasicBlock *RightMBB;
12816 if (FirstRight == LastRight && FirstRight->Kind ==
CC_Range &&
12817 W.LT && (FirstRight->High->getValue() + 1ULL) ==
W.LT->getValue()) {
12818 RightMBB = FirstRight->MBB;
12820 RightMBB =
FuncInfo.MF->CreateMachineBasicBlock(
W.MBB->getBasicBlock());
12821 FuncInfo.MF->insert(BBI, RightMBB);
12823 {RightMBB, FirstRight, LastRight, Pivot,
W.LT,
W.DefaultProb / 2});
12829 CaseBlock CB(
ISD::SETLT,
Cond, Pivot,
nullptr, LeftMBB, RightMBB,
W.MBB,
12832 if (
W.MBB == SwitchMBB)
12835 SL->SwitchCases.push_back(CB);
12860 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12868 unsigned PeeledCaseIndex = 0;
12869 bool SwitchPeeled =
false;
12870 for (
unsigned Index = 0;
Index < Clusters.size(); ++
Index) {
12871 CaseCluster &CC = Clusters[
Index];
12872 if (CC.
Prob < TopCaseProb)
12874 TopCaseProb = CC.
Prob;
12875 PeeledCaseIndex =
Index;
12876 SwitchPeeled =
true;
12881 LLVM_DEBUG(
dbgs() <<
"Peeled one top case in switch stmt, prob: "
12882 << TopCaseProb <<
"\n");
12887 MachineBasicBlock *PeeledSwitchMBB =
12889 FuncInfo.MF->insert(BBI, PeeledSwitchMBB);
12892 auto PeeledCaseIt = Clusters.begin() + PeeledCaseIndex;
12893 SwitchWorkListItem
W = {SwitchMBB, PeeledCaseIt, PeeledCaseIt,
12894 nullptr,
nullptr, TopCaseProb.
getCompl()};
12895 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, PeeledSwitchMBB);
12897 Clusters.erase(PeeledCaseIt);
12898 for (CaseCluster &CC : Clusters) {
12900 dbgs() <<
"Scale the probablity for one cluster, before scaling: "
12901 << CC.
Prob <<
"\n");
12905 PeeledCaseProb = TopCaseProb;
12906 return PeeledSwitchMBB;
12909void SelectionDAGBuilder::visitSwitch(
const SwitchInst &
SI) {
12911 BranchProbabilityInfo *BPI =
FuncInfo.BPI;
12913 Clusters.reserve(
SI.getNumCases());
12914 for (
auto I :
SI.cases()) {
12915 MachineBasicBlock *Succ =
FuncInfo.getMBB(
I.getCaseSuccessor());
12916 const ConstantInt *CaseVal =
I.getCaseValue();
12917 BranchProbability Prob =
12919 : BranchProbability(1,
SI.getNumCases() + 1);
12923 MachineBasicBlock *DefaultMBB =
FuncInfo.getMBB(
SI.getDefaultDest());
12932 MachineBasicBlock *PeeledSwitchMBB =
12933 peelDominantCaseCluster(SI, Clusters, PeeledCaseProb);
12936 MachineBasicBlock *SwitchMBB =
FuncInfo.MBB;
12937 if (Clusters.empty()) {
12938 assert(PeeledSwitchMBB == SwitchMBB);
12940 if (DefaultMBB != NextBlock(SwitchMBB)) {
12947 SL->findJumpTables(Clusters, &SI,
getCurSDLoc(), DefaultMBB,
DAG.getPSI(),
12949 SL->findBitTestClusters(Clusters, &SI);
12952 dbgs() <<
"Case clusters: ";
12953 for (
const CaseCluster &
C : Clusters) {
12959 C.Low->getValue().print(
dbgs(),
true);
12960 if (
C.Low !=
C.High) {
12962 C.High->getValue().print(
dbgs(),
true);
12969 assert(!Clusters.empty());
12973 auto DefaultProb = getEdgeProbability(PeeledSwitchMBB, DefaultMBB);
12977 DefaultMBB ==
FuncInfo.getMBB(
SI.getDefaultDest()))
12980 {PeeledSwitchMBB,
First,
Last,
nullptr,
nullptr, DefaultProb});
12982 while (!WorkList.
empty()) {
12984 unsigned NumClusters =
W.LastCluster -
W.FirstCluster + 1;
12989 splitWorkItem(WorkList, W,
SI.getCondition(), SwitchMBB);
12993 lowerWorkItem(W,
SI.getCondition(), SwitchMBB, DefaultMBB);
12997void SelectionDAGBuilder::visitStepVector(
const CallInst &
I) {
12998 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13004void SelectionDAGBuilder::visitVectorReverse(
const CallInst &
I) {
13005 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13010 assert(VT ==
V.getValueType() &&
"Malformed vector.reverse!");
13019 SmallVector<int, 8>
Mask;
13021 for (
unsigned i = 0; i != NumElts; ++i)
13022 Mask.push_back(NumElts - 1 - i);
13027void SelectionDAGBuilder::visitVectorDeinterleave(
const CallInst &
I,
13036 EVT OutVT = ValueVTs[0];
13040 for (
unsigned i = 0; i != Factor; ++i) {
13041 assert(ValueVTs[i] == OutVT &&
"Expected VTs to be the same");
13043 DAG.getVectorIdxConstant(OutNumElts * i,
DL));
13049 SDValue Even =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
13051 SDValue Odd =
DAG.getVectorShuffle(OutVT,
DL, SubVecs[0], SubVecs[1],
13059 DAG.getVTList(ValueVTs), SubVecs);
13063void SelectionDAGBuilder::visitVectorInterleave(
const CallInst &
I,
13066 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13071 for (
unsigned i = 0; i < Factor; ++i) {
13074 "Expected VTs to be the same");
13092 for (
unsigned i = 0; i < Factor; ++i)
13099void SelectionDAGBuilder::visitFreeze(
const FreezeInst &
I) {
13103 unsigned NumValues = ValueVTs.
size();
13104 if (NumValues == 0)
return;
13109 for (
unsigned i = 0; i != NumValues; ++i)
13117void SelectionDAGBuilder::visitVectorSplice(
const CallInst &
I) {
13118 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13124 const bool IsLeft =
I.getIntrinsicID() == Intrinsic::vector_splice_left;
13139 uint64_t Idx = IsLeft ?
Imm : NumElts -
Imm;
13142 SmallVector<int, 8>
Mask;
13143 for (
unsigned i = 0; i < NumElts; ++i)
13144 Mask.push_back(Idx + i);
13172 assert(
MI->getOpcode() == TargetOpcode::COPY &&
13173 "start of copy chain MUST be COPY");
13174 Reg =
MI->getOperand(1).getReg();
13177 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
13181 if (
MI->getOpcode() == TargetOpcode::COPY) {
13182 assert(
Reg.isVirtual() &&
"expected COPY of virtual register");
13183 Reg =
MI->getOperand(1).getReg();
13184 assert(
Reg.isPhysical() &&
"expected COPY of physical register");
13187 assert(
MI->getOpcode() == TargetOpcode::INLINEASM_BR &&
13188 "end of copy chain MUST be INLINEASM_BR");
13198void SelectionDAGBuilder::visitCallBrLandingPad(
const CallInst &
I) {
13204 const TargetLowering &TLI =
DAG.getTargetLoweringInfo();
13205 const TargetRegisterInfo *
TRI =
DAG.getSubtarget().getRegisterInfo();
13206 MachineRegisterInfo &MRI =
DAG.getMachineFunction().getRegInfo();
13214 for (
auto &
T : TargetConstraints) {
13215 SDISelAsmOperandInfo OpInfo(
T);
13223 switch (OpInfo.ConstraintType) {
13234 FuncInfo.MBB->addLiveIn(OriginalDef);
13242 ResultVTs.
push_back(OpInfo.ConstraintVT);
13251 ResultVTs.
push_back(OpInfo.ConstraintVT);
13259 DAG.getVTList(ResultVTs), ResultValues);
static unsigned getIntrinsicID(const SDNode *N)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static msgpack::DocNode getNode(msgpack::DocNode DN, msgpack::Type Type, MCValue Val)
This file declares a class to represent arbitrary precision floating point values and provide a varie...
This file implements a class to represent arbitrary precision integral constant values and operations...
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
MachineBasicBlock MachineBasicBlock::iterator MBBI
Function Alias Analysis Results
Atomic ordering constants.
This file contains the simple types necessary to represent the attributes associated with functions a...
static const Function * getParent(const Value *V)
This file implements the BitVector class.
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
This file contains the declarations for the subclasses of Constant, which represent the different fla...
static AttributeList getReturnAttrs(FastISel::CallLoweringInfo &CLI)
Returns an AttributeList representing the attributes applied to the return value of the given call.
static Value * getCondition(Instruction *I)
const HexagonInstrInfo * TII
This file provides various utilities for inspecting and working with the control flow graph in LLVM I...
Module.h This file contains the declarations for the Module class.
static void getRegistersForValue(MachineFunction &MF, MachineIRBuilder &MIRBuilder, GISelAsmOperandInfo &OpInfo, GISelAsmOperandInfo &RefOpInfo)
Assign virtual/physical registers for the specified register operand.
static void computeConstraintToUse(const TargetLowering *TLI, TargetLowering::AsmOperandInfo &OpInfo)
This file defines an InstructionCost class that is used when calculating the cost of an instruction,...
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
Machine Check Debug Module
static bool isUndef(const MachineInstr &MI)
Register const TargetRegisterInfo * TRI
Promote Memory to Register
static const Function * getCalledFunction(const Value *V)
This file provides utility analysis objects describing memory locations.
This file provides utility for Memory Model Relaxation Annotations (MMRAs).
static MCRegister getReg(const MCDisassembler *D, unsigned RC, unsigned RegNo)
static unsigned getAddressSpace(const Value *V, unsigned MaxLookup)
MachineInstr unsigned OpIdx
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
uint64_t IntrinsicInst * II
OptimizedStructLayoutField Field
const SmallVectorImpl< MachineOperand > MachineBasicBlock * TBB
const SmallVectorImpl< MachineOperand > & Cond
static Type * getValueType(Value *V, bool LookThroughCmp=false)
Returns the "element type" of the given value/instruction V.
static bool hasOnlySelectUsers(const Value *Cond)
static SDValue getLoadStackGuard(SelectionDAG &DAG, const SDLoc &DL, SDValue &Chain)
Create a LOAD_STACK_GUARD node, and let it carry the target specific global variable if there exists ...
static bool getUniformBase(const Value *Ptr, SDValue &Base, SDValue &Index, SDValue &Scale, SelectionDAGBuilder *SDB, const BasicBlock *CurBB, uint64_t ElemSize)
static void failForInvalidBundles(const CallBase &I, StringRef Name, ArrayRef< uint32_t > AllowedBundles)
static void addStackMapLiveVars(const CallBase &Call, unsigned StartIdx, const SDLoc &DL, SmallVectorImpl< SDValue > &Ops, SelectionDAGBuilder &Builder)
Add a stack map intrinsic call's live variable operands to a stackmap or patchpoint target node's ope...
static const unsigned MaxParallelChains
static SDValue expandPow(const SDLoc &dl, SDValue LHS, SDValue RHS, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
visitPow - Lower a pow intrinsic.
static const CallBase * FindPreallocatedCall(const Value *PreallocatedSetup)
Given a @llvm.call.preallocated.setup, return the corresponding preallocated call.
static cl::opt< unsigned > SwitchPeelThreshold("switch-peel-threshold", cl::Hidden, cl::init(66), cl::desc("Set the case probability threshold for peeling the case from a " "switch statement. A value greater than 100 will void this " "optimization"))
static cl::opt< bool > InsertAssertAlign("insert-assert-align", cl::init(true), cl::desc("Insert the experimental `assertalign` node."), cl::ReallyHidden)
static unsigned getISDForVPIntrinsic(const VPIntrinsic &VPIntrin)
static bool handleDanglingVariadicDebugInfo(SelectionDAG &DAG, DILocalVariable *Variable, DebugLoc DL, unsigned Order, SmallVectorImpl< Value * > &Values, DIExpression *Expression)
static bool prepareDAGLevelOperands(ConstraintDecisionInfo &Info, const CallBase &Call, SelectionDAGBuilder &Builder, const TargetLowering &TLI, SelectionDAG &DAG)
Prepare DAG-level operands.
static unsigned findMatchingInlineAsmOperand(unsigned OperandNo, const std::vector< SDValue > &AsmNodeOperands)
static void patchMatchingInput(const SDISelAsmOperandInfo &OpInfo, SDISelAsmOperandInfo &MatchingOpInfo, SelectionDAG &DAG)
Make sure that the output operand OpInfo and its corresponding input operand MatchingOpInfo have comp...
static void findUnwindDestinations(FunctionLoweringInfo &FuncInfo, const BasicBlock *EHPadBB, BranchProbability Prob, SmallVectorImpl< std::pair< MachineBasicBlock *, BranchProbability > > &UnwindDests)
When an invoke or a cleanupret unwinds to the next EH pad, there are many places it could ultimately ...
static unsigned FixedPointIntrinsicToOpcode(unsigned Intrinsic)
static BranchProbability scaleCaseProbality(BranchProbability CaseProb, BranchProbability PeeledCaseProb)
static SDValue expandExp2(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandExp2 - Lower an exp2 intrinsic.
static SDValue expandDivFix(unsigned Opcode, const SDLoc &DL, SDValue LHS, SDValue RHS, SDValue Scale, SelectionDAG &DAG, const TargetLowering &TLI)
static SDValue getF32Constant(SelectionDAG &DAG, unsigned Flt, const SDLoc &dl)
getF32Constant - Get 32-bit floating point constant.
static SDValue widenVectorToPartType(SelectionDAG &DAG, SDValue Val, const SDLoc &DL, EVT PartVT)
static SDValue expandLog10(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandLog10 - Lower a log10 intrinsic.
DenseMap< const Argument *, std::pair< const AllocaInst *, const StoreInst * > > ArgCopyElisionMapTy
static void getCopyToPartsVector(SelectionDAG &DAG, const SDLoc &dl, SDValue Val, SDValue *Parts, unsigned NumParts, MVT PartVT, const Value *V, std::optional< CallingConv::ID > CallConv)
getCopyToPartsVector - Create a series of nodes that contain the specified value split into legal par...
static void getUnderlyingArgRegs(SmallVectorImpl< std::pair< Register, TypeSize > > &Regs, const SDValue &N)
static void getCopyToParts(SelectionDAG &DAG, const SDLoc &DL, SDValue Val, SDValue *Parts, unsigned NumParts, MVT PartVT, const Value *V, std::optional< CallingConv::ID > CallConv=std::nullopt, ISD::NodeType ExtendKind=ISD::ANY_EXTEND)
getCopyToParts - Create a series of nodes that contain the specified value split into legal parts.
static SDValue getMemCmpLoad(const Value *PtrVal, MVT LoadVT, SelectionDAGBuilder &Builder)
static SDValue expandLog2(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandLog2 - Lower a log2 intrinsic.
static SDValue getAddressForMemoryInput(SDValue Chain, const SDLoc &Location, SDISelAsmOperandInfo &OpInfo, SelectionDAG &DAG)
Get a direct memory input to behave well as an indirect operand.
static bool isOnlyUsedInEntryBlock(const Argument *A, bool FastISel)
isOnlyUsedInEntryBlock - If the specified argument is only used in the entry block,...
static void diagnosePossiblyInvalidConstraint(LLVMContext &Ctx, const Value *V, const Twine &ErrMsg)
static bool collectInstructionDeps(SmallMapVector< const Instruction *, bool, 8 > *Deps, const Value *V, SmallMapVector< const Instruction *, bool, 8 > *Necessary=nullptr, unsigned Depth=0)
static void findArgumentCopyElisionCandidates(const DataLayout &DL, FunctionLoweringInfo *FuncInfo, ArgCopyElisionMapTy &ArgCopyElisionCandidates)
Scan the entry block of the function in FuncInfo for arguments that look like copies into a local all...
static bool isFunction(SDValue Op)
static SDValue GetExponent(SelectionDAG &DAG, SDValue Op, const TargetLowering &TLI, const SDLoc &dl)
GetExponent - Get the exponent:
static Register FollowCopyChain(MachineRegisterInfo &MRI, Register Reg)
static SDValue ExpandPowI(const SDLoc &DL, SDValue LHS, SDValue RHS, SelectionDAG &DAG)
ExpandPowI - Expand a llvm.powi intrinsic.
static SDValue expandLog(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandLog - Lower a log intrinsic.
static SDValue getCopyFromParts(SelectionDAG &DAG, const SDLoc &DL, const SDValue *Parts, unsigned NumParts, MVT PartVT, EVT ValueVT, const Value *V, SDValue InChain, std::optional< CallingConv::ID > CC=std::nullopt, std::optional< ISD::NodeType > AssertOp=std::nullopt)
getCopyFromParts - Create a value that contains the specified legal parts combined into the value the...
static SDValue getLimitedPrecisionExp2(SDValue t0, const SDLoc &dl, SelectionDAG &DAG)
static bool determineConstraints(ConstraintDecisionInfo &Info, TargetLowering::AsmOperandInfoVector &TargetConstraints, const CallBase &Call, SelectionDAGBuilder &Builder, const TargetLowering &TLI, const TargetMachine &TM, SelectionDAG &DAG, const BasicBlock *EHPadBB)
DetermineConstraints - Find the constraints to use for inline asm operands.
static bool constructOperandInfo(ConstraintDecisionInfo &Info, TargetLowering::AsmOperandInfoVector &TargetConstraints, SelectionDAGBuilder &Builder, const TargetLowering &TLI, ExtraFlags &ExtraInfo)
Construct operand info objects.
static SDValue GetSignificand(SelectionDAG &DAG, SDValue Op, const SDLoc &dl)
GetSignificand - Get the significand and build it into a floating-point number with exponent of 1:
static SDValue expandExp(const SDLoc &dl, SDValue Op, SelectionDAG &DAG, const TargetLowering &TLI, SDNodeFlags Flags)
expandExp - Lower an exp intrinsic.
static const MDNode * getRangeMetadata(const Instruction &I)
static cl::opt< unsigned, true > LimitFPPrecision("limit-float-precision", cl::desc("Generate low-precision inline sequences " "for some float libcalls"), cl::location(LimitFloatPrecision), cl::Hidden, cl::init(0))
static void tryToElideArgumentCopy(FunctionLoweringInfo &FuncInfo, SmallVectorImpl< SDValue > &Chains, DenseMap< int, int > &ArgCopyElisionFrameIndexMap, SmallPtrSetImpl< const Instruction * > &ElidedArgCopyInstrs, ArgCopyElisionMapTy &ArgCopyElisionCandidates, const Argument &Arg, ArrayRef< SDValue > ArgVals, bool &ArgHasUses)
Try to elide argument copies from memory into a local alloca.
static unsigned LimitFloatPrecision
LimitFloatPrecision - Generate low-precision inline sequences for some float libcalls (6,...
static SDValue getCopyFromPartsVector(SelectionDAG &DAG, const SDLoc &DL, const SDValue *Parts, unsigned NumParts, MVT PartVT, EVT ValueVT, const Value *V, SDValue InChain, std::optional< CallingConv::ID > CC)
getCopyFromPartsVector - Create a value that contains the specified legal parts combined into the val...
static bool InBlock(const Value *V, const BasicBlock *BB)
static FPClassTest getNoFPClass(const Instruction &I)
static LLVM_ATTRIBUTE_ALWAYS_INLINE MVT::SimpleValueType getSimpleVT(const uint8_t *MatcherTable, size_t &MatcherIndex)
getSimpleVT - Decode a value in MatcherTable, if it's a VBR encoded value, use GetVBR to decode it.
This file defines the SmallPtrSet class.
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static SymbolRef::Type getType(const Symbol *Sym)
uint16_t RegSizeInBits(const MCRegisterInfo &MRI, MCRegister RegNo)
static const fltSemantics & IEEEsingle()
static LLVM_ABI Semantics SemanticsToEnum(const llvm::fltSemantics &Sem)
static LLVM_ABI const fltSemantics * getArbitraryFPSemantics(StringRef Format)
Returns the fltSemantics for a given arbitrary FP format string, or nullptr if invalid.
Class for arbitrary precision integers.
bool isNonNegative() const
Determine if this APInt Value is non-negative (>= 0)
bool isPowerOf2() const
Check if this APInt's value is a power of two greater than zero.
an instruction to allocate memory on the stack
Align getAlign() const
Return the alignment of the memory that is being allocated by the instruction.
LLVM_ABI std::optional< TypeSize > getAllocationSize(const DataLayout &DL) const
Get allocation size in bytes.
This class represents an incoming formal argument to a Function.
LLVM_ABI bool hasAttribute(Attribute::AttrKind Kind) const
Check if an argument has a given attribute.
unsigned getArgNo() const
Return the index of this formal argument in its containing function.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
Get the array size.
A cache of @llvm.assume calls within a function.
An instruction that atomically checks whether a specified value is in a memory location,...
an instruction that atomically reads a memory location, combines it with another value,...
@ USubCond
Subtract only if no unsigned overflow.
@ FMinimum
*p = minimum(old, v) minimum matches the behavior of llvm.minimum.
@ Min
*p = old <signed v ? old : v
@ USubSat
*p = usub.sat(old, v) usub.sat matches the behavior of llvm.usub.sat.
@ FMaximum
*p = maximum(old, v) maximum matches the behavior of llvm.maximum.
@ UIncWrap
Increment one up to a maximum value.
@ Max
*p = old >signed v ? old : v
@ UMin
*p = old <unsigned v ? old : v
@ FMin
*p = minnum(old, v) minnum matches the behavior of llvm.minnum.
@ UMax
*p = old >unsigned v ? old : v
@ FMaximumNum
*p = maximumnum(old, v) maximumnum matches the behavior of llvm.maximumnum.
@ FMax
*p = maxnum(old, v) maxnum matches the behavior of llvm.maxnum.
@ UDecWrap
Decrement one until a minimum value or zero.
@ FMinimumNum
*p = minimumnum(old, v) minimumnum matches the behavior of llvm.minimumnum.
This class holds the attributes for a particular argument, parameter, function, or return value.
LLVM Basic Block Representation.
const Function * getParent() const
Return the enclosing method, or null if none.
LLVM_ABI InstListType::const_iterator getFirstNonPHIIt() const
Returns an iterator to the first instruction in this block that is not a PHINode instruction.
InstListType::const_iterator const_iterator
LLVM_ABI bool isEntryBlock() const
Return true if this is the entry block of the containing function.
LLVM_ABI InstListType::const_iterator getFirstNonPHIOrDbg(bool SkipPseudoOp=true) const
Returns a pointer to the first instruction in this block that is not a PHINode or a debug intrinsic,...
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction; assumes that the block is well-formed.
This class is a wrapper over an AAResults, and it is intended to be used only when there are no IR ch...
This class represents a no-op cast from one type to another.
The address of a basic block.
Analysis providing branch probability information.
LLVM_ABI BranchProbability getEdgeProbability(const BasicBlock *Src, unsigned IndexInSuccessors) const
Get an edge's probability, relative to other out-edges of the Src.
LLVM_ABI bool isEdgeHot(const BasicBlock *Src, const BasicBlock *Dst) const
Test if an edge is hot relative to other out-edges of the Src.
static uint32_t getDenominator()
static BranchProbability getOne()
static BranchProbability getUnknown()
uint32_t getNumerator() const
LLVM_ABI uint64_t scale(uint64_t Num) const
Scale a large integer.
BranchProbability getCompl() const
static BranchProbability getZero()
static void normalizeProbabilities(ProbabilityIter Begin, ProbabilityIter End)
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
std::optional< OperandBundleUse > getOperandBundle(StringRef Name) const
Return an operand bundle by name, if present.
CallingConv::ID getCallingConv() const
User::op_iterator arg_begin()
Return the iterator pointing to the beginning of the argument list.
LLVM_ABI bool isMustTailCall() const
Tests if this call site must be tail call optimized.
LLVM_ABI bool isIndirectCall() const
Return true if the callsite is an indirect call.
unsigned countOperandBundlesOfType(StringRef Name) const
Return the number of operand bundles with the tag Name attached to this instruction.
Value * getCalledOperand() const
Value * getArgOperand(unsigned i) const
User::op_iterator arg_end()
Return the iterator pointing to the end of the argument list.
bool isConvergent() const
Determine if the invoke is convergent.
FunctionType * getFunctionType() const
unsigned arg_size() const
AttributeList getAttributes() const
Return the attributes for this call.
LLVM_ABI bool isTailCall() const
Tests if this call site is marked as a tail call.
CallBr instruction, tracking function calls that may not return control but instead transfer it to a ...
This class represents a function call, abstracting a target machine's calling convention.
This class is the base class for the comparison instructions.
Predicate
This enumeration lists the possible predicates for CmpInst subclasses.
Conditional Branch instruction.
Class for constant bytes.
ConstantDataSequential - A vector or array constant whose element type is a simple 1/2/4/8-byte integ...
A constant value that is initialized with an expression using other constant values.
ConstantFP - Floating Point Values [float, double].
This is the shared class of boolean and integer constants.
static LLVM_ABI ConstantInt * getTrue(LLVMContext &Context)
bool isZero() const
This is just a convenience method to make client code smaller for a common code.
static LLVM_ABI ConstantInt * getFalse(LLVMContext &Context)
uint64_t getZExtValue() const
Return the constant as a 64-bit unsigned integer value after it has been zero extended as appropriate...
const APInt & getValue() const
Return the constant as an APInt value reference.
A signed pointer, in the ptrauth sense.
uint64_t getZExtValue() const
Constant Vector Declarations.
This is an important base class in LLVM.
This is the common base class for constrained floating point intrinsics.
LLVM_ABI std::optional< fp::ExceptionBehavior > getExceptionBehavior() const
LLVM_ABI unsigned getNonMetadataArgCount() const
LLVM_ABI bool isEntryValue() const
Check if the expression consists of exactly one entry value operand.
static bool fragmentsOverlap(const FragmentInfo &A, const FragmentInfo &B)
Check if fragments overlap between a pair of FragmentInfos.
static LLVM_ABI DIExpression * appendOpsToArg(const DIExpression *Expr, ArrayRef< uint64_t > Ops, unsigned ArgNo, bool StackValue=false)
Create a copy of Expr by appending the given list of Ops to each instance of the operand DW_OP_LLVM_a...
static LLVM_ABI std::optional< FragmentInfo > getFragmentInfo(expr_op_iterator Start, expr_op_iterator End)
Retrieve the details of this fragment expression.
LLVM_ABI uint64_t getNumLocationOperands() const
Return the number of unique location operands referred to (via DW_OP_LLVM_arg) in this expression; th...
static LLVM_ABI std::optional< DIExpression * > createFragmentExpression(const DIExpression *Expr, unsigned OffsetInBits, unsigned SizeInBits)
Create a DIExpression to describe one part of an aggregate variable that is fragmented across multipl...
static LLVM_ABI const DIExpression * convertToUndefExpression(const DIExpression *Expr)
Removes all elements from Expr that do not apply to an undef debug value, which includes every operat...
static LLVM_ABI DIExpression * prepend(const DIExpression *Expr, uint8_t Flags, int64_t Offset=0)
Prepend DIExpr with a deref and offset operation and optionally turn it into a stack value or/and an ...
static LLVM_ABI DIExpression * prependOpcodes(const DIExpression *Expr, SmallVectorImpl< uint64_t > &Ops, bool StackValue=false, bool EntryValue=false)
Prepend DIExpr with the given opcodes and optionally turn it into a stack value.
Base class for variables.
LLVM_ABI std::optional< uint64_t > getSizeInBits() const
Determines the size of the variable's type.
A parsed version of the target data layout string in and methods for querying it.
Records a position in IR for a source label (DILabel).
Base class for non-instruction debug metadata records that have positions within IR.
DebugLoc getDebugLoc() const
Record of a variable value-assignment, aka a non instruction representation of the dbg....
LocationType getType() const
LLVM_ABI Value * getVariableLocationOp(unsigned OpIdx) const
DIExpression * getExpression() const
DILocalVariable * getVariable() const
LLVM_ABI iterator_range< location_op_iterator > location_ops() const
Get the locations corresponding to the variable referenced by the debug info intrinsic.
LLVM_ABI DILocation * getInlinedAt() const
iterator find(const_arg_type_t< KeyT > Val)
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT, true > const_iterator
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
void reserve(size_type NumEntries)
Grow the densemap so that it can contain at least NumEntries items before resizing again.
Diagnostic information for inline asm reporting.
static constexpr ElementCount getFixed(ScalarTy MinVal)
static constexpr ElementCount get(ScalarTy MinVal, bool Scalable)
constexpr bool isScalar() const
Exactly one element.
Lightweight error class with error context and mandatory checking.
Class representing an expression and its matching format.
This instruction compares its operands according to the predicate given to the constructor.
This is a fast-path instruction selection class that generates poor code and doesn't support illegal ...
bool allowReassoc() const
Flag queries.
An instruction for ordering other memory operations.
static LLVM_ABI FixedVectorType * get(Type *ElementType, unsigned NumElts)
This class represents a freeze function that returns random concrete value if an operand is either a ...
FunctionLoweringInfo - This contains information that is global to a function that is used when lower...
BranchProbabilityInfo * BPI
MachineBasicBlock * getMBB(const BasicBlock *BB) const
DenseMap< const AllocaInst *, int > StaticAllocaMap
StaticAllocaMap - Keep track of frame indices for fixed sized allocas in the entry block.
const LiveOutInfo * GetLiveOutRegInfo(Register Reg)
GetLiveOutRegInfo - Gets LiveOutInfo for a register, returning NULL if the register is a PHI destinat...
MachineBasicBlock * MBB
MBB - The current block.
Class to represent function types.
unsigned getNumParams() const
Return the number of fixed parameters this function type requires.
Type * getParamType(unsigned i) const
Parameter type accessors.
Type * getReturnType() const
Data structure describing the variable locations in a function.
const BasicBlock & getEntryBlock() const
FunctionType * getFunctionType() const
Returns the FunctionType for me.
Intrinsic::ID getIntrinsicID() const LLVM_READONLY
getIntrinsicID - This method returns the ID number of the specified function, or Intrinsic::not_intri...
bool hasMinSize() const
Optimize this function for minimum size (-Oz).
bool hasParamAttribute(unsigned ArgNo, Attribute::AttrKind Kind) const
check if an attributes is in the list of attributes.
CallingConv::ID getCallingConv() const
getCallingConv()/setCallingConv(CC) - These method get and set the calling convention of this functio...
Constant * getPersonalityFn() const
Get the personality function associated with this function.
AttributeList getAttributes() const
Return the attribute list for this Function.
bool isIntrinsic() const
isIntrinsic - Returns true if the function's name starts with "llvm.".
bool hasFnAttribute(Attribute::AttrKind Kind) const
Return true if the function has the attribute.
Garbage collection metadata for a single function.
bool hasNoUnsignedSignedWrap() const
bool hasNoUnsignedWrap() const
an instruction for type-safe pointer arithmetic to access elements of arrays and structs
static StringRef dropLLVMManglingEscape(StringRef Name)
If the given string begins with the GlobalValue name mangling escape character '\1',...
bool hasDLLImportStorageClass() const
Module * getParent()
Get the module that this global value is contained inside of...
This instruction compares its operands according to the predicate given to the constructor.
Indirect Branch Instruction.
void setMemConstraint(ConstraintCode C)
setMemConstraint - Augment an existing flag with the constraint code for a memory constraint.
This instruction inserts a struct field of array element value into an aggregate value.
MDNode * getMetadata(unsigned KindID) const
Get the metadata of given kind attached to this Instruction.
LLVM_ABI FastMathFlags getFastMathFlags() const LLVM_READONLY
Convenience function for getting all the fast-math flags, which must be an operator which supports th...
LLVM_ABI AAMDNodes getAAMetadata() const
Returns the AA metadata for this instruction.
@ MIN_INT_BITS
Minimum number of bits that can be specified.
Intrinsic::ID getIntrinsicID() const
Return the intrinsic ID of this intrinsic.
This is an important class for using LLVM in a threaded context.
@ OB_clang_arc_attachedcall
LLVM_ABI void diagnose(const DiagnosticInfo &DI)
Report a message to the currently installed diagnostic handler.
The landingpad instruction holds all of the information necessary to generate correct exception handl...
A helper class to return the specified delimiter string after the first invocation of operator String...
An instruction for reading from memory.
static LocationSize precise(uint64_t Value)
static constexpr LocationSize beforeOrAfterPointer()
Any location before or after the base pointer (but still within the underlying object).
static LocationSize upperBound(uint64_t Value)
LLVM_ABI MCSymbol * createTempSymbol()
Create a temporary symbol with a unique name.
LLVM_ABI MCSymbol * getOrCreateFrameAllocSymbol(const Twine &FuncName, unsigned Idx)
Gets a symbol that will be defined to the final stack offset of a local variable after codegen.
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
const MDOperand & getOperand(unsigned I) const
LLVM_ABI StringRef getString() const
@ INVALID_SIMPLE_VALUE_TYPE
uint64_t getScalarSizeInBits() const
unsigned getVectorNumElements() const
bool isVector() const
Return true if this is a vector value type.
bool isInteger() const
Return true if this is an integer or a vector integer type.
TypeSize getSizeInBits() const
Returns the size of the specified MVT in bits.
uint64_t getFixedSizeInBits() const
Return the size of the specified fixed width value type in bits.
ElementCount getVectorElementCount() const
TypeSize getStoreSize() const
Return the number of bytes overwritten by a store of the specified value type.
bool bitsGE(MVT VT) const
Return true if this has no less bits than VT.
bool isScalarInteger() const
Return true if this is an integer, not including vectors.
static MVT getVectorVT(MVT VT, unsigned NumElements)
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
static MVT getIntegerVT(unsigned BitWidth)
void normalizeSuccProbs()
Normalize probabilities of all successors so that the sum of them becomes one.
const BasicBlock * getBasicBlock() const
Return the LLVM basic block that this instance corresponded to originally.
LLVM_ABI void setSuccProbability(succ_iterator I, BranchProbability Prob)
Set successor probability of a given iterator.
succ_iterator succ_begin()
LLVM_ABI void addSuccessor(MachineBasicBlock *Succ, BranchProbability Prob=BranchProbability::getUnknown())
Add Succ as a successor of this MachineBasicBlock.
SmallVectorImpl< MachineBasicBlock * >::iterator succ_iterator
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
void setIsEHContTarget(bool V=true)
Indicates if this is a target of Windows EH Continuation Guard.
void setIsEHFuncletEntry(bool V=true)
Indicates if this is the entry block of an EH funclet.
MachineInstrBundleIterator< MachineInstr > iterator
void setIsEHScopeEntry(bool V=true)
Indicates if this is the entry block of an EH scope, i.e., the block that that used to have a catchpa...
void setMachineBlockAddressTaken()
Set this block to indicate that its address is used as something other than the target of a terminato...
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
void setIsImmutableObjectIndex(int ObjectIdx, bool IsImmutable)
Marks the immutability of an object.
LLVM_ABI int CreateStackObject(uint64_t Size, Align Alignment, bool isSpillSlot, const AllocaInst *Alloca=nullptr, uint8_t ID=0)
Create a new statically sized stack object, returning a nonnegative identifier to represent it.
bool hasOpaqueSPAdjustment() const
Returns true if the function contains opaque dynamic stack adjustments.
int getStackProtectorIndex() const
Return the index for the stack protector object.
void setStackProtectorIndex(int I)
void setIsAliasedObjectIndex(int ObjectIdx, bool IsAliased)
Set "maybe pointed to by an LLVM IR value" for an object.
Align getObjectAlign(int ObjectIdx) const
Return the alignment of the specified stack object.
int64_t getObjectSize(int ObjectIdx) const
Return the size of the specified object.
void RemoveStackObject(int ObjectIdx)
Remove or mark dead a statically sized stack object.
void setFunctionContextIndex(int I)
const WinEHFuncInfo * getWinEHFuncInfo() const
getWinEHFuncInfo - Return information about how the current function uses Windows exception handling.
bool useDebugInstrRef() const
Returns true if the function's variable locations are tracked with instruction referencing.
void setCallSiteBeginLabel(MCSymbol *BeginLabel, unsigned Site)
Map the begin label for a call site.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineMemOperand * getMachineMemOperand(MachinePointerInfo PtrInfo, MachineMemOperand::Flags f, LLT MemTy, Align base_alignment, const AAMDNodes &AAInfo=AAMDNodes(), const MDNode *Ranges=nullptr, SyncScope::ID SSID=SyncScope::System, AtomicOrdering Ordering=AtomicOrdering::NotAtomic, AtomicOrdering FailureOrdering=AtomicOrdering::NotAtomic)
getMachineMemOperand - Allocate a new MachineMemOperand.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MCContext & getContext() const
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
void addCodeViewAnnotation(MCSymbol *Label, MDNode *MD)
Record annotations associated with a particular label.
Function & getFunction()
Return the LLVM function that this machine code represents.
BasicBlockListType::iterator iterator
bool hasEHFunclets() const
void setHasEHContTarget(bool V)
void addInvoke(MachineBasicBlock *LandingPad, MCSymbol *BeginLabel, MCSymbol *EndLabel)
Provide the begin and end labels of an invoke style call and associate it with a try landing pad bloc...
MachineBasicBlock * CreateMachineBasicBlock(const BasicBlock *BB=nullptr, std::optional< UniqueBBID > BBID=std::nullopt)
CreateMachineInstr - Allocate a new MachineInstr.
void insert(iterator MBBI, MachineBasicBlock *MBB)
const MachineInstrBuilder & addSym(MCSymbol *Sym, unsigned char TargetFlags=0) const
const MachineInstrBuilder & addFrameIndex(int Idx) const
Representation of each machine instruction.
A description of a memory reference used in the backend.
Flags
Flags values. These may be or'd together.
@ MOVolatile
The memory access is volatile.
@ MODereferenceable
The memory access is dereferenceable (i.e., doesn't trap).
@ MOLoad
The memory access reads data.
@ MONonTemporal
The memory access is non-temporal.
@ MOInvariant
The memory access always returns the same value (or traps).
@ MOStore
The memory access writes data.
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
static MachineOperand CreateReg(Register Reg, bool isDef, bool isImp=false, bool isKill=false, bool isDead=false, bool isUndef=false, bool isEarlyClobber=false, unsigned SubReg=0, bool isDebug=false, bool isInternalRead=false, bool isRenamable=false)
static MachineOperand CreateFI(int Idx)
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
const TargetRegisterClass * getRegClass(Register Reg) const
Return the register class of the specified virtual register.
def_iterator def_begin(Register RegNo) const
LLVM_ABI Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
LLVM_ABI MCRegister getLiveInPhysReg(Register VReg) const
getLiveInPhysReg - If VReg is a live-in virtual register, return the corresponding live-in physical r...
An SDNode that represents everything that will be needed to construct a MachineInstr.
std::pair< iterator, bool > try_emplace(const KeyT &Key, Ts &&...Args)
bool contains(const KeyT &Key) const
static MemoryLocation getAfter(const Value *Ptr, const AAMDNodes &AATags=AAMDNodes())
Return a location that may access any location after Ptr, while remaining within the underlying objec...
A Module instance is used to store all the information related to an LLVM module.
static PointerType * getUnqual(Type *ElementType)
This constructs a pointer to an object of the specified type in the default address space (address sp...
static LLVM_ABI PointerType * get(Type *ElementType, unsigned AddressSpace)
This constructs a pointer to an object of the specified type in a numbered address space.
static LLVM_ABI PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified type.
Wrapper class representing virtual and physical registers.
constexpr bool isValid() const
constexpr bool isVirtual() const
Return true if the specified register number is in the virtual register namespace.
constexpr bool isPhysical() const
Return true if the specified register number is in the physical register namespace.
Resume the propagation of an exception.
Return a value (possibly void), from a function.
Holds the information from a dbg_label node through SDISel.
static SDDbgOperand fromNode(SDNode *Node, unsigned ResNo)
static SDDbgOperand fromFrameIdx(unsigned FrameIdx)
static SDDbgOperand fromVReg(Register VReg)
static SDDbgOperand fromConst(const Value *Const)
Holds the information from a dbg_value node through SDISel.
Wrapper class for IR location info (IR ordering and DebugLoc) to be passed into SDNode creation funct...
Represents one node in the SelectionDAG.
unsigned getOpcode() const
Return the SelectionDAG opcode value for this node.
iterator_range< value_op_iterator > op_values() const
unsigned getIROrder() const
Return the node ordering.
uint64_t getAsZExtVal() const
Helper method returns the zero-extended integer value of a ConstantSDNode.
unsigned getNumValues() const
Return the number of values defined/returned by this operator.
const SDValue & getOperand(unsigned Num) const
EVT getValueType(unsigned ResNo) const
Return the type of a specified result.
Unlike LLVM values, Selection DAG nodes may return multiple values as the result of a computation.
SDNode * getNode() const
get the SDNode which holds the desired result
SDValue getValue(unsigned R) const
EVT getValueType() const
Return the ValueType of the referenced return value.
TypeSize getValueSizeInBits() const
Returns the size of the value in bits.
const SDValue & getOperand(unsigned i) const
unsigned getResNo() const
get the index which selects a specific result in the SDNode
MVT getSimpleValueType() const
Return the simple ValueType of the referenced return value.
unsigned getOpcode() const
SelectionDAGBuilder - This is the common target-independent lowering implementation that is parameter...
SDValue getValue(const Value *V)
getValue - Return an SDValue for the given Value.
bool shouldKeepJumpConditionsTogether(const FunctionLoweringInfo &FuncInfo, const CondBrInst &I, Instruction::BinaryOps Opc, const Value *Lhs, const Value *Rhs, TargetLoweringBase::CondMergingParams Params) const
DenseMap< const Constant *, Register > ConstantsOut
void addDanglingDebugInfo(SmallVectorImpl< Value * > &Values, DILocalVariable *Var, DIExpression *Expr, bool IsVariadic, DebugLoc DL, unsigned Order)
Register a dbg_value which relies on a Value which we have not yet seen.
void visitDbgInfo(const Instruction &I)
void clearDanglingDebugInfo()
Clear the dangling debug information map.
SDValue lowerStartEH(SDValue Chain, const BasicBlock *EHPadBB, MCSymbol *&BeginLabel)
void LowerCallTo(const CallBase &CB, SDValue Callee, bool IsTailCall, bool IsMustTailCall, const BasicBlock *EHPadBB=nullptr, const TargetLowering::PtrAuthInfo *PAI=nullptr)
void clear()
Clear out the current SelectionDAG and the associated state and prepare this SelectionDAGBuilder obje...
void visitBitTestHeader(SwitchCG::BitTestBlock &B, MachineBasicBlock *SwitchBB)
visitBitTestHeader - This function emits necessary code to produce value suitable for "bit tests"
void LowerStatepoint(const GCStatepointInst &I, const BasicBlock *EHPadBB=nullptr)
std::unique_ptr< SDAGSwitchLowering > SL
SDValue lowerRangeToAssertZExt(SelectionDAG &DAG, const Instruction &I, SDValue Op)
bool HasTailCall
This is set to true if a call in the current block has been translated as a tail call.
bool ShouldEmitAsBranches(const std::vector< SwitchCG::CaseBlock > &Cases)
If the set of cases should be emitted as a series of branches, return true.
void EmitBranchForMergedCondition(const Value *Cond, MachineBasicBlock *TBB, MachineBasicBlock *FBB, MachineBasicBlock *CurBB, MachineBasicBlock *SwitchBB, BranchProbability TProb, BranchProbability FProb, bool InvertCond)
EmitBranchForMergedCondition - Helper method for FindMergedConditions.
void LowerDeoptimizeCall(const CallInst *CI)
void LowerCallSiteWithDeoptBundle(const CallBase *Call, SDValue Callee, const BasicBlock *EHPadBB)
SwiftErrorValueTracking & SwiftError
Information about the swifterror values used throughout the function.
SDValue getNonRegisterValue(const Value *V)
getNonRegisterValue - Return an SDValue for the given Value, but don't look in FuncInfo....
const TargetTransformInfo * TTI
DenseMap< MachineBasicBlock *, SmallVector< unsigned, 4 > > LPadToCallSiteMap
Map a landing pad to the call site indexes.
SDValue lowerNoFPClassToAssertNoFPClass(SelectionDAG &DAG, const Instruction &I, SDValue Op)
void handleDebugDeclare(Value *Address, DILocalVariable *Variable, DIExpression *Expression, DebugLoc DL)
StatepointLoweringState StatepointLowering
State used while lowering a statepoint sequence (gc_statepoint, gc_relocate, and gc_result).
void visitBitTestCase(SwitchCG::BitTestBlock &BB, MachineBasicBlock *NextMBB, BranchProbability BranchProbToNext, Register Reg, SwitchCG::BitTestCase &B, MachineBasicBlock *SwitchBB)
visitBitTestCase - this function produces one "bit test"
bool canTailCall(const CallBase &CB) const
void populateCallLoweringInfo(TargetLowering::CallLoweringInfo &CLI, const CallBase *Call, unsigned ArgIdx, unsigned NumArgs, SDValue Callee, Type *ReturnTy, AttributeSet RetAttrs, bool IsPatchPoint)
Populate a CallLowerinInfo (into CLI) based on the properties of the call being lowered.
void CopyValueToVirtualRegister(const Value *V, Register Reg, ISD::NodeType ExtendType=ISD::ANY_EXTEND)
void salvageUnresolvedDbgValue(const Value *V, DanglingDebugInfo &DDI)
For the given dangling debuginfo record, perform last-ditch efforts to resolve the debuginfo to somet...
SmallVector< SDValue, 8 > PendingLoads
Loads are not emitted to the program immediately.
GCFunctionInfo * GFI
Garbage collection metadata for the function.
void init(GCFunctionInfo *gfi, BatchAAResults *BatchAA, AssumptionCache *AC, const TargetLibraryInfo *li, const TargetTransformInfo &TTI)
SDValue getRoot()
Similar to getMemoryRoot, but also flushes PendingConstrainedFP(Strict) items.
void ExportFromCurrentBlock(const Value *V)
ExportFromCurrentBlock - If this condition isn't known to be exported from the current basic block,...
DebugLoc getCurDebugLoc() const
void resolveOrClearDbgInfo()
Evict any dangling debug information, attempting to salvage it first.
std::pair< SDValue, SDValue > lowerInvokable(TargetLowering::CallLoweringInfo &CLI, const BasicBlock *EHPadBB=nullptr)
SDValue getMemoryRoot()
Return the current virtual root of the Selection DAG, flushing any PendingLoad items.
void resolveDanglingDebugInfo(const Value *V, SDValue Val)
If we saw an earlier dbg_value referring to V, generate the debug data structures now that we've seen...
SDLoc getCurSDLoc() const
void visit(const Instruction &I)
void dropDanglingDebugInfo(const DILocalVariable *Variable, const DIExpression *Expr)
If we have dangling debug info that describes Variable, or an overlapping part of variable considerin...
SDValue getCopyFromRegs(const Value *V, Type *Ty)
If there was virtual register allocated for the value V emit CopyFromReg of the specified type Ty.
void CopyToExportRegsIfNeeded(const Value *V)
CopyToExportRegsIfNeeded - If the given value has virtual registers created for it,...
void handleKillDebugValue(DILocalVariable *Var, DIExpression *Expr, DebugLoc DbgLoc, unsigned Order)
Create a record for a kill location debug intrinsic.
void visitJumpTable(SwitchCG::JumpTable &JT)
visitJumpTable - Emit JumpTable node in the current MBB
SDValue getFPOperationRoot(fp::ExceptionBehavior EB)
Return the current virtual root of the Selection DAG, flushing PendingConstrainedFP or PendingConstra...
void visitJumpTableHeader(SwitchCG::JumpTable &JT, SwitchCG::JumpTableHeader &JTH, MachineBasicBlock *SwitchBB)
visitJumpTableHeader - This function emits necessary code to produce index in the JumpTable from swit...
void LowerCallSiteWithPtrAuthBundle(const CallBase &CB, const BasicBlock *EHPadBB)
static const unsigned LowestSDNodeOrder
Lowest valid SDNodeOrder.
void LowerDeoptimizingReturn()
FunctionLoweringInfo & FuncInfo
Information about the function as a whole.
void setValue(const Value *V, SDValue NewN)
void FindMergedConditions(const Value *Cond, MachineBasicBlock *TBB, MachineBasicBlock *FBB, MachineBasicBlock *CurBB, MachineBasicBlock *SwitchBB, Instruction::BinaryOps Opc, BranchProbability TProb, BranchProbability FProb, bool InvertCond)
const TargetLibraryInfo * LibInfo
bool isExportableFromCurrentBlock(const Value *V, const BasicBlock *FromBB)
void visitSPDescriptorParent(StackProtectorDescriptor &SPD, MachineBasicBlock *ParentBB)
Codegen a new tail for a stack protector check ParentMBB which has had its tail spliced into a stack ...
bool handleDebugValue(ArrayRef< const Value * > Values, DILocalVariable *Var, DIExpression *Expr, DebugLoc DbgLoc, unsigned Order, bool IsVariadic)
For a given list of Values, attempt to create and record a SDDbgValue in the SelectionDAG.
SDValue getControlRoot()
Similar to getRoot, but instead of flushing all the PendingLoad items, flush all the PendingExports (...
void UpdateSplitBlock(MachineBasicBlock *First, MachineBasicBlock *Last)
When an MBB was split during scheduling, update the references that need to refer to the last resulti...
SDValue getValueImpl(const Value *V)
getValueImpl - Helper function for getValue and getNonRegisterValue.
void visitSwitchCase(SwitchCG::CaseBlock &CB, MachineBasicBlock *SwitchBB)
visitSwitchCase - Emits the necessary code to represent a single node in the binary search tree resul...
void visitSPDescriptorFailure(StackProtectorDescriptor &SPD)
Codegen the failure basic block for a stack protector check.
std::unique_ptr< FunctionLoweringInfo > FuncInfo
SmallPtrSet< const Instruction *, 4 > ElidedArgCopyInstrs
const TargetLowering * TLI
MachineRegisterInfo * RegInfo
std::unique_ptr< SwiftErrorValueTracking > SwiftError
virtual void emitFunctionEntryCode()
std::unique_ptr< SelectionDAGBuilder > SDB
virtual std::pair< SDValue, SDValue > EmitTargetCodeForMemccpy(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Dst, SDValue Src, SDValue C, SDValue Size, const CallInst *CI) const
Emit target-specific code that performs a memccpy, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrnlen(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, SDValue Src, SDValue MaxLength, MachinePointerInfo SrcPtrInfo) const
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrlen(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, SDValue Src, const CallInst *CI) const
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrstr(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Op1, SDValue Op2, const CallInst *CI) const
Emit target-specific code that performs a strstr, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForMemchr(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Src, SDValue Char, SDValue Length, MachinePointerInfo SrcPtrInfo) const
Emit target-specific code that performs a memchr, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrcmp(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Op1, SDValue Op2, MachinePointerInfo Op1PtrInfo, MachinePointerInfo Op2PtrInfo, const CallInst *CI) const
Emit target-specific code that performs a strcmp, in cases where that is faster than a libcall.
virtual std::pair< SDValue, SDValue > EmitTargetCodeForMemcmp(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Op1, SDValue Op2, SDValue Op3, const CallInst *CI) const
Emit target-specific code that performs a memcmp/bcmp, in cases where that is faster than a libcall.
virtual SDValue EmitTargetCodeForSetTag(SelectionDAG &DAG, const SDLoc &dl, SDValue Chain, SDValue Addr, SDValue Size, MachinePointerInfo DstPtrInfo, bool ZeroData) const
virtual std::pair< SDValue, SDValue > EmitTargetCodeForStrcpy(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, SDValue Dest, SDValue Src, MachinePointerInfo DestPtrInfo, MachinePointerInfo SrcPtrInfo, bool isStpcpy, const CallInst *CI) const
Emit target-specific code that performs a strcpy or stpcpy, in cases where that is faster than a libc...
This is used to represent a portion of an LLVM function in a low-level Data Dependence DAG representa...
SDValue getTargetGlobalAddress(const GlobalValue *GV, const SDLoc &DL, EVT VT, int64_t offset=0, unsigned TargetFlags=0)
SDValue getExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT, unsigned Opcode)
Convert Op, which must be of integer type, to the integer type VT, by either any/sign/zero-extending ...
const SDValue & getRoot() const
Return the root tag of the SelectionDAG.
const TargetSubtargetInfo & getSubtarget() const
SDValue getCopyToReg(SDValue Chain, const SDLoc &dl, Register Reg, SDValue N)
LLVM_ABI SDValue getMergeValues(ArrayRef< SDValue > Ops, const SDLoc &dl)
Create a MERGE_VALUES node from the given operands.
LLVM_ABI SDVTList getVTList(EVT VT)
Return an SDVTList that represents the list of values specified.
LLVM_ABI SDValue getShiftAmountConstant(uint64_t Val, EVT VT, const SDLoc &DL)
LLVM_ABI MachineSDNode * getMachineNode(unsigned Opcode, const SDLoc &dl, EVT VT)
These are used for target selectors to create a new node with specified return type(s),...
LLVM_ABI void ExtractVectorElements(SDValue Op, SmallVectorImpl< SDValue > &Args, unsigned Start=0, unsigned Count=0, EVT EltVT=EVT())
Append the extracted elements from Start to Count out of the vector Op in Args.
LLVM_ABI SDValue getConstantPool(const Constant *C, EVT VT, MaybeAlign Align=std::nullopt, int Offs=0, bool isT=false, unsigned TargetFlags=0)
LLVM_ABI SDValue getConstantFP(double Val, const SDLoc &DL, EVT VT, bool isTarget=false)
Create a ConstantFPSDNode wrapping a constant value.
LLVM_ABI SDValue getRegister(Register Reg, EVT VT)
LLVM_ABI SDValue getLoad(EVT VT, const SDLoc &dl, SDValue Chain, SDValue Ptr, MachinePointerInfo PtrInfo, MaybeAlign Alignment=MaybeAlign(), MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes(), const MDNode *Ranges=nullptr)
Loads are not normal binary operators: their result type is not determined by their operands,...
LLVM_ABI Align getEVTAlign(EVT MemoryVT) const
Compute the default alignment value for the given type.
LLVM_ABI bool shouldOptForSize() const
const TargetLowering & getTargetLoweringInfo() const
static constexpr unsigned MaxRecursionDepth
LLVM_ABI void AddDbgValue(SDDbgValue *DB, bool isParameter)
Add a dbg_value SDNode.
SDValue getUNDEF(EVT VT)
Return an UNDEF node. UNDEF does not have a useful SDLoc.
SDValue getBuildVector(EVT VT, const SDLoc &DL, ArrayRef< SDValue > Ops)
Return an ISD::BUILD_VECTOR node.
LLVM_ABI SDValue getBitcast(EVT VT, SDValue V)
Return a bitcast using the SDLoc of the value operand, and casting to the provided type.
LLVM_ABI SDDbgValue * getDbgValueList(DIVariable *Var, DIExpression *Expr, ArrayRef< SDDbgOperand > Locs, ArrayRef< SDNode * > Dependencies, bool IsIndirect, const DebugLoc &DL, unsigned O, bool IsVariadic)
Creates a SDDbgValue node from a list of locations.
SDValue getCopyFromReg(SDValue Chain, const SDLoc &dl, Register Reg, EVT VT)
LLVM_ABI void setNodeMemRefs(MachineSDNode *N, ArrayRef< MachineMemOperand * > NewMemRefs)
Mutate the specified machine node's memory references to the provided list.
const DataLayout & getDataLayout() const
SDValue getTargetFrameIndex(int FI, EVT VT)
LLVM_ABI SDValue getConstant(uint64_t Val, const SDLoc &DL, EVT VT, bool isTarget=false, bool isOpaque=false)
Create a ConstantSDNode wrapping a constant value.
LLVM_ABI SDValue getMemBasePlusOffset(SDValue Base, TypeSize Offset, const SDLoc &DL, const SDNodeFlags Flags=SDNodeFlags())
Returns sum of the base pointer and offset.
LLVM_ABI SDValue getTruncStore(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Ptr, MachinePointerInfo PtrInfo, EVT SVT, Align Alignment, MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes())
LLVM_ABI SDValue getMDNode(const MDNode *MD)
Return an MDNodeSDNode which holds an MDNode.
LLVM_ABI SDValue getBasicBlock(MachineBasicBlock *MBB)
LLVM_ABI SDValue getEHLabel(const SDLoc &dl, SDValue Root, MCSymbol *Label)
LLVM_ABI SDValue getPtrExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either truncating it or perform...
LLVM_ABI SDValue getAnyExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either any-extending or truncat...
const LibcallLoweringInfo & getLibcalls() const
LLVM_ABI SDValue getIntPtrConstant(uint64_t Val, const SDLoc &DL, bool isTarget=false)
LLVM_ABI SDValue getValueType(EVT)
LLVM_ABI SDValue getNode(unsigned Opcode, const SDLoc &DL, EVT VT, ArrayRef< SDUse > Ops)
Gets or creates the specified node.
LLVM_ABI SDValue getFPExtendOrRound(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of float type, to the float type VT, by either extending or rounding (by tr...
SDValue getTargetConstant(uint64_t Val, const SDLoc &DL, EVT VT, bool isOpaque=false)
LLVM_ABI SDValue getVectorIdxConstant(uint64_t Val, const SDLoc &DL, bool isTarget=false)
MachineFunction & getMachineFunction() const
LLVM_ABI SDValue getFrameIndex(int FI, EVT VT, bool isTarget=false)
LLVM_ABI SDValue getZExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either zero-extending or trunca...
LLVMContext * getContext() const
const SDValue & setRoot(SDValue N)
Set the current root tag of the SelectionDAG.
LLVM_ABI SDValue getTargetExternalSymbol(const char *Sym, EVT VT, unsigned TargetFlags=0)
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
void swap(SmallVectorImpl &RHS)
void push_back(const T &Elt)
pointer data()
Return a pointer to the vector's buffer, even if empty().
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Encapsulates all of the information needed to generate a stack protector check, and signals to isel w...
MachineBasicBlock * getSuccessMBB()
MachineBasicBlock * getFailureMBB()
MachineBasicBlock * getParentMBB()
bool shouldEmitFunctionBasedCheckStackProtector() const
An instruction for storing to memory.
Represent a constant reference to a string, i.e.
constexpr bool empty() const
Check if the string is empty.
constexpr const char * data() const
Get a pointer to the start of the string (which may not be null terminated).
Information about stack frame layout on the target.
virtual TargetStackID::Value getStackIDForScalableVectors() const
Returns the StackID that scalable vectors should be associated with.
Provides information about what library functions are available for the current target.
virtual Align getByValTypeAlignment(Type *Ty, const DataLayout &DL) const
Returns the desired alignment for ByVal or InAlloca aggregate function arguments in the caller parame...
virtual bool isFMAFasterThanFMulAndFAdd(const MachineFunction &MF, EVT) const
Return true if an FMA operation is faster than a pair of fmul and fadd instructions.
EVT getMemValueType(const DataLayout &DL, Type *Ty, bool AllowUnknown=false) const
Function * getSSPStackGuardCheck(const Module &M, const LibcallLoweringInfo &Libcalls) const
If the target has a standard stack protection check function that performs validation and error handl...
EVT getValueType(const DataLayout &DL, Type *Ty, bool AllowUnknown=false) const
Return the EVT corresponding to this LLVM type.
LegalizeAction
This enum indicates whether operations are valid for a target, and if not, what action should be used...
virtual 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 bool useStackGuardMixFP() const
If this function returns true, stack protection checks should mix the frame pointer (or whichever poi...
virtual unsigned getNumRegisters(LLVMContext &Context, EVT VT, std::optional< MVT > RegisterVT=std::nullopt) const
Return the number of registers that this ValueType will eventually require.
MachineMemOperand::Flags getLoadMemOperandFlags(const LoadInst &LI, const DataLayout &DL, AssumptionCache *AC=nullptr, const TargetLibraryInfo *LibInfo=nullptr, CodeGenOptLevel OptLevel=CodeGenOptLevel::Default) const
virtual bool shouldExtendGSIndex(EVT VT, EVT &EltTy) const
Returns true if the index type for a masked gather/scatter requires extending.
virtual unsigned getVectorTypeBreakdownForCallingConv(LLVMContext &Context, CallingConv::ID CC, EVT VT, EVT &IntermediateVT, unsigned &NumIntermediates, MVT &RegisterVT) const
Certain targets such as MIPS require that some types such as vectors are always broken down into scal...
Register getStackPointerRegisterToSaveRestore() const
If a physical register, this specifies the register that llvm.savestack/llvm.restorestack should save...
LegalizeAction getFixedPointOperationAction(unsigned Op, EVT VT, unsigned Scale) const
Some fixed point operations may be natively supported by the target but only for specific scales.
MachineMemOperand::Flags getAtomicMemOperandFlags(const Instruction &AI, const DataLayout &DL) const
virtual bool allowsMisalignedMemoryAccesses(EVT, unsigned AddrSpace=0, Align Alignment=Align(1), MachineMemOperand::Flags Flags=MachineMemOperand::MONone, unsigned *=nullptr) const
Determine if the target supports unaligned memory accesses.
bool isOperationCustom(unsigned Op, EVT VT) const
Return true if the operation uses custom lowering, regardless of whether the type is legal or not.
bool hasBigEndianPartOrdering(EVT VT, const DataLayout &DL) const
When splitting a value of the specified type into parts, does the Lo or Hi part come first?
EVT getShiftAmountTy(EVT LHSTy, const DataLayout &DL) const
Returns the type for the shift amount of a shift opcode.
virtual Align getABIAlignmentForCallingConv(Type *ArgTy, const DataLayout &DL) const
Certain targets have context sensitive alignment requirements, where one type has the alignment requi...
MachineMemOperand::Flags getVPIntrinsicMemOperandFlags(const VPIntrinsic &VPIntrin) const
virtual bool shouldExpandGetActiveLaneMask(EVT VT, EVT OpVT) const
Return true if the @llvm.get.active.lane.mask intrinsic should be expanded using generic code in Sele...
virtual EVT getSetCCResultType(const DataLayout &DL, LLVMContext &Context, EVT VT) const
Return the ValueType of the result of SETCC operations.
virtual EVT getTypeToTransformTo(LLVMContext &Context, EVT VT) const
For types supported by the target, this is an identity function.
bool isTypeLegal(EVT VT) const
Return true if the target has native support for the specified value type.
MVT getProgramPointerTy(const DataLayout &DL) const
Return the type for code pointers, which is determined by the program address space specified through...
virtual MVT getPointerTy(const DataLayout &DL, uint32_t AS=0) const
Return the pointer type for the given address space, defaults to the pointer type from the data layou...
bool isOperationLegal(unsigned Op, EVT VT) const
Return true if the specified operation is legal on this target.
virtual bool shouldExpandVectorMatch(EVT VT, unsigned SearchSize) const
Return true if the @llvm.experimental.vector.match intrinsic should be expanded for vector type ‘VT’ ...
virtual bool isProfitableToCombineMinNumMaxNum(EVT VT) const
virtual MVT getFenceOperandTy(const DataLayout &DL) const
Return the type for operands of fence.
virtual bool shouldExpandGetVectorLength(EVT CountVT, unsigned VF, bool IsScalable) const
bool isOperationLegalOrCustom(unsigned Op, EVT VT, bool LegalOnly=false) const
Return true if the specified operation is legal on this target or can be made legal with custom lower...
virtual MVT hasFastEqualityCompare(unsigned NumBits) const
Return the preferred operand type if the target has a quick way to compare integer values of the give...
MachineMemOperand::Flags getStoreMemOperandFlags(const StoreInst &SI, const DataLayout &DL) const
virtual void getTgtMemIntrinsic(SmallVectorImpl< IntrinsicInfo > &Infos, const CallBase &I, MachineFunction &MF, unsigned Intrinsic) const
Given an intrinsic, checks if on the target the intrinsic will need to map to a MemIntrinsicNode (tou...
virtual bool signExtendConstant(const ConstantInt *C) const
Return true if this constant should be sign extended when promoting to a larger type.
virtual Value * getSDagStackGuard(const Module &M, const LibcallLoweringInfo &Libcalls) const
Return the variable that's previously inserted by insertSSPDeclarations, if any, otherwise return nul...
LegalizeTypeAction getTypeAction(LLVMContext &Context, EVT VT) const
Return how we should legalize values of this type, either it is already legal (return 'Legal') or we ...
virtual Register getExceptionPointerRegister(const Constant *PersonalityFn) const
If a physical register, this returns the register that receives the exception address on entry to an ...
bool supportsUnalignedAtomics() const
Whether the target supports unaligned atomic operations.
std::vector< ArgListEntry > ArgListTy
bool isBeneficialToExpandPowI(int64_t Exponent, bool OptForSize) const
Return true if it is beneficial to expand an @llvm.powi.
MVT getFrameIndexTy(const DataLayout &DL) const
Return the type for frame index, which is determined by the alloca address space specified through th...
virtual Register getExceptionSelectorRegister(const Constant *PersonalityFn) const
If a physical register, this returns the register that receives the exception typeid on entry to a la...
virtual MVT getPointerMemTy(const DataLayout &DL, uint32_t AS=0) const
Return the in-memory pointer type for the given address space, defaults to the pointer type from the ...
MVT getRegisterType(MVT VT) const
Return the type of registers that this ValueType will eventually require.
unsigned getVectorTypeBreakdown(LLVMContext &Context, EVT VT, EVT &IntermediateVT, unsigned &NumIntermediates, MVT &RegisterVT) const
Vector types are broken down into some number of legal first class types.
virtual MVT getVPExplicitVectorLengthTy() const
Returns the type to be used for the EVL/AVL operand of VP nodes: ISD::VP_ADD, ISD::VP_SUB,...
This class defines information used to lower LLVM code to legal SelectionDAG operators that the targe...
virtual bool supportKCFIBundles() const
Return true if the target supports kcfi operand bundles.
virtual bool supportPtrAuthBundles() const
Return true if the target supports ptrauth operand bundles.
virtual bool supportSwiftError() const
Return true if the target supports swifterror attribute.
virtual SDValue visitMaskedLoad(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, MachineMemOperand *MMO, SDValue &NewLoad, SDValue Ptr, SDValue PassThru, SDValue Mask) const
virtual EVT getTypeForExtReturn(LLVMContext &Context, EVT VT, ISD::NodeType) const
Return the type that should be used to zero or sign extend a zeroext/signext integer return value.
virtual Register getRegisterByName(const char *RegName, LLT Ty, const MachineFunction &MF) const
Return the register ID of the name passed in.
virtual InlineAsm::ConstraintCode getInlineAsmMemConstraint(StringRef ConstraintCode) const
std::vector< AsmOperandInfo > AsmOperandInfoVector
SDValue expandIS_FPCLASS(EVT ResultVT, SDValue Op, FPClassTest Test, SDNodeFlags Flags, const SDLoc &DL, SelectionDAG &DAG) const
Expand check for floating point class.
virtual SDValue prepareVolatileOrAtomicLoad(SDValue Chain, const SDLoc &DL, SelectionDAG &DAG) const
This callback is used to prepare for a volatile or atomic load.
virtual SDValue emitStackGuardMixFP(SelectionDAG &DAG, SDValue Val, const SDLoc &DL) const
virtual ConstraintType getConstraintType(StringRef Constraint) const
Given a constraint, return the type of constraint it is for this target.
virtual bool splitValueIntoRegisterParts(SelectionDAG &DAG, const SDLoc &DL, SDValue Val, SDValue *Parts, unsigned NumParts, MVT PartVT, std::optional< CallingConv::ID > CC) const
Target-specific splitting of values into parts that fit a register storing a legal type.
virtual SDValue joinRegisterPartsIntoValue(SelectionDAG &DAG, const SDLoc &DL, const SDValue *Parts, unsigned NumParts, MVT PartVT, EVT ValueVT, std::optional< CallingConv::ID > CC) const
Target-specific combining of register parts into its original value.
virtual SDValue LowerCall(CallLoweringInfo &, SmallVectorImpl< SDValue > &) const
This hook must be implemented to lower calls into the specified DAG.
std::pair< SDValue, SDValue > LowerCallTo(CallLoweringInfo &CLI) const
This function lowers an abstract call to a function into an actual call.
virtual SDValue LowerAsmOutputForConstraint(SDValue &Chain, SDValue &Glue, const SDLoc &DL, const AsmOperandInfo &OpInfo, SelectionDAG &DAG) const
virtual std::pair< unsigned, const TargetRegisterClass * > getRegForInlineAsmConstraint(const TargetRegisterInfo *TRI, StringRef Constraint, MVT VT) const
Given a physical register constraint (e.g.
virtual AsmOperandInfoVector ParseConstraints(const DataLayout &DL, const TargetRegisterInfo *TRI, const CallBase &Call) const
Split up the constraint string from the inline assembly value into the specific constraints and their...
virtual SDValue LowerOperation(SDValue Op, SelectionDAG &DAG) const
This callback is invoked for operations that are unsupported by the target, which are registered to u...
virtual bool functionArgumentNeedsConsecutiveRegisters(Type *Ty, CallingConv::ID CallConv, bool isVarArg, const DataLayout &DL) const
For some targets, an LLVM struct type must be broken down into multiple simple types,...
virtual void ComputeConstraintToUse(AsmOperandInfo &OpInfo, SDValue Op, SelectionDAG *DAG=nullptr) const
Determines the constraint code and constraint type to use for the specific AsmOperandInfo,...
virtual void CollectTargetIntrinsicOperands(const CallInst &I, SmallVectorImpl< SDValue > &Ops, SelectionDAG &DAG) const
virtual SDValue visitMaskedStore(SelectionDAG &DAG, const SDLoc &DL, SDValue Chain, MachineMemOperand *MMO, SDValue Ptr, SDValue Val, SDValue Mask) const
virtual bool useLoadStackGuardNode(const Module &M) const
If this function returns true, SelectionDAGBuilder emits a LOAD_STACK_GUARD node when it is lowering ...
virtual void LowerAsmOperandForConstraint(SDValue Op, StringRef Constraint, std::vector< SDValue > &Ops, SelectionDAG &DAG) const
Lower the specified operand into the Ops vector.
std::pair< SDValue, SDValue > makeLibCall(SelectionDAG &DAG, RTLIB::LibcallImpl LibcallImpl, EVT RetVT, ArrayRef< SDValue > Ops, MakeLibCallOptions CallOptions, const SDLoc &dl, SDValue Chain=SDValue()) const
Returns a pair of (return value, chain).
virtual void LowerOperationWrapper(SDNode *N, SmallVectorImpl< SDValue > &Results, SelectionDAG &DAG) const
This callback is invoked by the type legalizer to legalize nodes with an illegal operand type but leg...
virtual bool isInlineAsmTargetBranch(const SmallVectorImpl< StringRef > &AsmStrs, unsigned OpNo) const
On x86, return true if the operand with index OpNo is a CALL or JUMP instruction, which can use eithe...
virtual MVT getJumpTableRegTy(const DataLayout &DL) const
virtual bool CanLowerReturn(CallingConv::ID, MachineFunction &, bool, const SmallVectorImpl< ISD::OutputArg > &, LLVMContext &, const Type *RetTy) const
This hook should be implemented to check whether the return values described by the Outs array can fi...
Primary interface to the complete machine description for the target machine.
CodeGenOptLevel getOptLevel() const
Returns the optimization level: None, Less, Default, or Aggressive.
CodeModel::Model getCodeModel() const
Returns the code model.
unsigned NoTrapAfterNoreturn
Do not emit a trap instruction for 'unreachable' IR instructions behind noreturn calls,...
unsigned TrapUnreachable
Emit target-specific trap instruction for 'unreachable' IR instructions.
unsigned getID() const
Return the register class ID number.
const MCPhysReg * iterator
iterator begin() const
begin/end - Return all of the registers in this class.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
virtual const TargetFrameLowering * getFrameLowering() const
virtual const TargetRegisterInfo * getRegisterInfo() const =0
Return the target's register information.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
static constexpr TypeSize getFixed(ScalarTy ExactSize)
The instances of the Type class are immutable: once they are created, they are never changed.
LLVM_ABI bool isEmptyTy() const
Return true if this type is empty, that is, it has no elements or all of its elements are empty.
bool isVectorTy() const
True if this is an instance of VectorType.
bool isPointerTy() const
True if this is an instance of PointerType.
LLVM_ABI unsigned getPointerAddressSpace() const
Get the address space of this pointer or pointer vector type.
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
Type * getScalarType() const
If this is a vector type, return the element type, otherwise return 'this'.
LLVMContext & getContext() const
Return the LLVMContext in which this type was uniqued.
static LLVM_ABI IntegerType * getInt1Ty(LLVMContext &C)
bool isIntegerTy() const
True if this is an instance of IntegerType.
bool isTokenTy() const
Return true if this is 'token'.
static LLVM_ABI IntegerType * getIntNTy(LLVMContext &C, unsigned N)
bool isFPOrFPVectorTy() const
Return true if this is a FP type or a vector of FP.
bool isVoidTy() const
Return true if this is 'void'.
Unconditional Branch instruction.
This function has undefined behavior.
A Use represents the edge between a Value definition and its users.
Value * getOperand(unsigned i) const
unsigned getNumOperands() const
This class represents the va_arg llvm instruction, which returns an argument of the specified type gi...
LLVM_ABI CmpInst::Predicate getPredicate() const
This is the common base class for vector predication intrinsics.
static LLVM_ABI std::optional< unsigned > getVectorLengthParamPos(Intrinsic::ID IntrinsicID)
LLVM_ABI MaybeAlign getPointerAlignment() const
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
bool hasOneUse() const
Return true if there is exactly one use of this value.
LLVMContext & getContext() const
All values hold a context through their type.
iterator_range< user_iterator > users()
LLVM_ABI const Value * stripPointerCasts() const
Strip off pointer casts, all-zero GEPs and address space casts.
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
Base class of all SIMD vector types.
Type * getElementType() const
constexpr ScalarTy getFixedValue() const
static constexpr bool isKnownLE(const FixedOrScalableQuantity &LHS, const FixedOrScalableQuantity &RHS)
constexpr bool isScalable() const
Returns whether the quantity is scaled by a runtime quantity (vscale).
constexpr ScalarTy getKnownMinValue() const
Returns the minimum value this quantity can represent.
const ParentTy * getParent() const
A raw_ostream that writes to an std::string.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
constexpr char Align[]
Key for Kernel::Arg::Metadata::mAlign.
constexpr char Args[]
Key for Kernel::Metadata::mArgs.
constexpr char SymbolName[]
Key for Kernel::Metadata::mSymbolName.
constexpr std::underlying_type_t< E > Mask()
Get a bitmask with 1s in all places up to the high-order bit of E's largest value.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ AnyReg
OBSOLETED - Used for stack based JavaScript calls.
@ AMDGPU_CS_Chain
Used on AMDGPUs to give the middle-end more control over argument placement.
@ X86_VectorCall
MSVC calling convention that passes vectors and vector aggregates in SSE registers.
@ C
The default llvm calling convention, compatible with C.
NodeType
ISD::NodeType enum - This enum defines the target-independent operators for a SelectionDAG.
@ MERGE_VALUES
MERGE_VALUES - This node takes multiple discrete operands and returns them all as its individual resu...
@ STACKRESTORE
STACKRESTORE has two operands, an input chain and a pointer to restore to it returns an output chain.
@ STACKSAVE
STACKSAVE - STACKSAVE has one operand, an input chain.
@ CONVERGENCECTRL_ANCHOR
The llvm.experimental.convergence.* intrinsics.
@ STRICT_FSETCC
STRICT_FSETCC/STRICT_FSETCCS - Constrained versions of SETCC, used for floating-point operands only.
@ DELETED_NODE
DELETED_NODE - This is an illegal value that is used to catch errors.
@ SET_FPENV
Sets the current floating-point environment.
@ ATOMIC_LOAD_FMINIMUMNUM
@ LOOP_DEPENDENCE_RAW_MASK
@ VECREDUCE_SEQ_FADD
Generic reduction nodes.
@ COND_LOOP
COND_LOOP is a conditional branch to self, used for implementing efficient conditional traps.
@ EH_SJLJ_LONGJMP
OUTCHAIN = EH_SJLJ_LONGJMP(INCHAIN, buffer) This corresponds to the eh.sjlj.longjmp intrinsic.
@ INSERT_SUBVECTOR
INSERT_SUBVECTOR(VECTOR1, VECTOR2, IDX) - Returns a vector with VECTOR2 inserted into VECTOR1.
@ STACKADDRESS
STACKADDRESS - Represents the llvm.stackaddress intrinsic.
@ BSWAP
Byte Swap and Counting operators.
@ SMULFIX
RESULT = [US]MULFIX(LHS, RHS, SCALE) - Perform fixed point multiplication on 2 integers with the same...
@ VAEND
VAEND, VASTART - VAEND and VASTART have three operands: an input chain, pointer, and a SRCVALUE.
@ ATOMIC_STORE
OUTCHAIN = ATOMIC_STORE(INCHAIN, val, ptr) This corresponds to "store atomic" instruction.
@ RESET_FPENV
Set floating-point environment to default state.
@ ADD
Simple integer binary arithmetic operators.
@ SMULFIXSAT
Same as the corresponding unsaturated fixed point instructions, but the result is clamped between the...
@ SET_FPMODE
Sets the current dynamic floating-point control modes.
@ ANY_EXTEND
ANY_EXTEND - Used for integer types. The high bits are undefined.
@ CTTZ_ELTS
Returns the number of number of trailing (least significant) zero elements in a vector.
@ FMA
FMA - Perform a * b + c with no intermediate rounding step.
@ VECTOR_FIND_LAST_ACTIVE
Finds the index of the last active mask element Operands: Mask.
@ FMODF
FMODF - Decomposes the operand into integral and fractional parts, each having the same type and sign...
@ FATAN2
FATAN2 - atan2, inspired by libm.
@ FSINCOSPI
FSINCOSPI - Compute both the sine and cosine times pi more accurately than FSINCOS(pi*x),...
@ INTRINSIC_VOID
OUTCHAIN = INTRINSIC_VOID(INCHAIN, INTRINSICID, arg1, arg2, ...) This node represents a target intrin...
@ EH_SJLJ_SETUP_DISPATCH
OUTCHAIN = EH_SJLJ_SETUP_DISPATCH(INCHAIN) The target initializes the dispatch table here.
@ ATOMIC_CMP_SWAP_WITH_SUCCESS
Val, Success, OUTCHAIN = ATOMIC_CMP_SWAP_WITH_SUCCESS(INCHAIN, ptr, cmp, swap) N.b.
@ SINT_TO_FP
[SU]INT_TO_FP - These operators convert integers (whose interpreted sign depends on the first letter)...
@ CONCAT_VECTORS
CONCAT_VECTORS(VECTOR0, VECTOR1, ...) - Given a number of values of vector type with the same length ...
@ VECREDUCE_FMAX
FMIN/FMAX nodes can have flags, for NaN/NoNaN variants.
@ FADD
Simple binary floating point operators.
@ VECREDUCE_FMAXIMUM
FMINIMUM/FMAXIMUM nodes propatate NaNs and signed zeroes using the llvm.minimum and llvm....
@ ABS
ABS - Determine the unsigned absolute value of a signed integer value of the same bitwidth.
@ ATOMIC_FENCE
OUTCHAIN = ATOMIC_FENCE(INCHAIN, ordering, scope) This corresponds to the fence instruction.
@ RESET_FPMODE
Sets default dynamic floating-point control modes.
@ FMULADD
FMULADD - Performs a * b + c, with, or without, intermediate rounding.
@ FPTRUNC_ROUND
FPTRUNC_ROUND - This corresponds to the fptrunc_round intrinsic.
@ FAKE_USE
FAKE_USE represents a use of the operand but does not do anything.
@ BITCAST
BITCAST - This operator converts between integer, vector and FP values, as if the value was stored to...
@ BUILD_PAIR
BUILD_PAIR - This is the opposite of EXTRACT_ELEMENT in some ways.
@ CLMUL
Carry-less multiplication operations.
@ INIT_TRAMPOLINE
INIT_TRAMPOLINE - This corresponds to the init_trampoline intrinsic.
@ FLDEXP
FLDEXP - ldexp, inspired by libm (op0 * 2**op1).
@ SDIVFIX
RESULT = [US]DIVFIX(LHS, RHS, SCALE) - Perform fixed point division on 2 integers with the same width...
@ CONVERT_FROM_ARBITRARY_FP
CONVERT_FROM_ARBITRARY_FP - This operator converts from an arbitrary floating-point represented as an...
@ EH_LABEL
EH_LABEL - Represents a label in mid basic block used to track locations needed for debug and excepti...
@ EH_RETURN
OUTCHAIN = EH_RETURN(INCHAIN, OFFSET, HANDLER) - This node represents 'eh_return' gcc dwarf builtin,...
@ ANNOTATION_LABEL
ANNOTATION_LABEL - Represents a mid basic block label used by annotations.
@ SET_ROUNDING
Set rounding mode.
@ CONVERGENCECTRL_GLUE
This does not correspond to any convergence control intrinsic.
@ SIGN_EXTEND
Conversion operators.
@ PREALLOCATED_SETUP
PREALLOCATED_SETUP - This has 2 operands: an input chain and a SRCVALUE with the preallocated call Va...
@ READSTEADYCOUNTER
READSTEADYCOUNTER - This corresponds to the readfixedcounter intrinsic.
@ ADDROFRETURNADDR
ADDROFRETURNADDR - Represents the llvm.addressofreturnaddress intrinsic.
@ BR
Control flow instructions. These all have token chains.
@ VECREDUCE_FADD
These reductions have relaxed evaluation order semantics, and have a single vector operand.
@ PREFETCH
PREFETCH - This corresponds to a prefetch intrinsic.
@ FSINCOS
FSINCOS - Compute both fsin and fcos as a single operation.
@ SSUBO
Same for subtraction.
@ PREALLOCATED_ARG
PREALLOCATED_ARG - This has 3 operands: an input chain, a SRCVALUE with the preallocated call Value,...
@ BRIND
BRIND - Indirect branch.
@ BR_JT
BR_JT - Jumptable branch.
@ VECTOR_INTERLEAVE
VECTOR_INTERLEAVE(VEC1, VEC2, ...) - Returns N vectors from N input vectors, where N is the factor to...
@ FCANONICALIZE
Returns platform specific canonical encoding of a floating point number.
@ IS_FPCLASS
Performs a check of floating point class property, defined by IEEE-754.
@ SSUBSAT
RESULT = [US]SUBSAT(LHS, RHS) - Perform saturation subtraction on 2 integers with the same bit width ...
@ SELECT
Select(COND, TRUEVAL, FALSEVAL).
@ EXTRACT_ELEMENT
EXTRACT_ELEMENT - This is used to get the lower or upper (determined by a Constant,...
@ SPLAT_VECTOR
SPLAT_VECTOR(VAL) - Returns a vector with the scalar value VAL duplicated in all lanes.
@ VACOPY
VACOPY - VACOPY has 5 operands: an input chain, a destination pointer, a source pointer,...
@ GET_ACTIVE_LANE_MASK
GET_ACTIVE_LANE_MASK - this corrosponds to the llvm.get.active.lane.mask intrinsic.
@ BasicBlock
Various leaf nodes.
@ CopyFromReg
CopyFromReg - This node indicates that the input value is a virtual or physical register that is defi...
@ SADDO
RESULT, BOOL = [SU]ADDO(LHS, RHS) - Overflow-aware nodes for addition.
@ ARITH_FENCE
ARITH_FENCE - This corresponds to a arithmetic fence intrinsic.
@ VECREDUCE_ADD
Integer reductions may have a result type larger than the vector element type.
@ GET_ROUNDING
Returns current rounding mode: -1 Undefined 0 Round to 0 1 Round to nearest, ties to even 2 Round to ...
@ CLEANUPRET
CLEANUPRET - Represents a return from a cleanup block funclet.
@ GET_FPMODE
Reads the current dynamic floating-point control modes.
@ GET_FPENV
Gets the current floating-point environment.
@ SHL
Shift and rotation operations.
@ AssertNoFPClass
AssertNoFPClass - These nodes record if a register contains a float value that is known to be not som...
@ PtrAuthGlobalAddress
A ptrauth constant.
@ EXTRACT_SUBVECTOR
EXTRACT_SUBVECTOR(VECTOR, IDX) - Returns a subvector from VECTOR.
@ EntryToken
EntryToken - This is the marker used to indicate the start of a region.
@ READ_REGISTER
READ_REGISTER, WRITE_REGISTER - This node represents llvm.register on the DAG, which implements the n...
@ EXTRACT_VECTOR_ELT
EXTRACT_VECTOR_ELT(VECTOR, IDX) - Returns a single element from VECTOR identified by the (potentially...
@ ZERO_EXTEND
ZERO_EXTEND - Used for integer types, zeroing the new bits.
@ DEBUGTRAP
DEBUGTRAP - Trap intended to get the attention of a debugger.
@ VSCALE
VSCALE(IMM) - Returns the runtime scaling factor used to calculate the number of elements within a sc...
@ LOCAL_RECOVER
LOCAL_RECOVER - Represents the llvm.localrecover intrinsic.
@ FMINNUM
FMINNUM/FMAXNUM - Perform floating-point minimum maximum on two values, following IEEE-754 definition...
@ UBSANTRAP
UBSANTRAP - Trap with an immediate describing the kind of sanitizer failure.
@ SSHLSAT
RESULT = [US]SHLSAT(LHS, RHS) - Perform saturation left shift.
@ PATCHPOINT
The llvm.experimental.patchpoint.
@ SMULO
Same for multiplication.
@ DYNAMIC_STACKALLOC
DYNAMIC_STACKALLOC - Allocate some number of bytes on the stack aligned to a specified boundary.
@ VECTOR_SPLICE_LEFT
VECTOR_SPLICE_LEFT(VEC1, VEC2, OFFSET) - Shifts CONCAT_VECTORS(VEC1, VEC2) left by OFFSET elements an...
@ SMIN
[US]{MIN/MAX} - Binary minimum or maximum of signed or unsigned integers.
@ MASKED_UDIV
Masked vector arithmetic that returns poison on disabled lanes.
@ VECTOR_REVERSE
VECTOR_REVERSE(VECTOR) - Returns a vector, of the same type as VECTOR, whose elements are shuffled us...
@ SDIVFIXSAT
Same as the corresponding unsaturated fixed point instructions, but the result is clamped between the...
@ FP_EXTEND
X = FP_EXTEND(Y) - Extend a smaller FP type into a larger FP type.
@ VSELECT
Select with a vector condition (op #0) and two vector operands (ops #1 and #2), returning a vector re...
@ PCMARKER
PCMARKER - This corresponds to the pcmarker intrinsic.
@ INLINEASM_BR
INLINEASM_BR - Branching version of inline asm. Used by asm-goto.
@ ATOMIC_LOAD_FMAXIMUMNUM
@ EH_DWARF_CFA
EH_DWARF_CFA - This node represents the pointer to the DWARF Canonical Frame Address (CFA),...
@ FRAMEADDR
FRAMEADDR, RETURNADDR - These nodes represent llvm.frameaddress and llvm.returnaddress on the DAG.
@ PEXT
Parallel bit extract (compress) and parallel bit deposit (expand).
@ STRICT_FP_ROUND
X = STRICT_FP_ROUND(Y, TRUNC) - Rounding 'Y' from a larger floating point type down to the precision ...
@ FMINIMUM
FMINIMUM/FMAXIMUM - NaN-propagating minimum/maximum that also treat -0.0 as less than 0....
@ FP_TO_SINT
FP_TO_[US]INT - Convert a floating point value to a signed or unsigned integer.
@ READCYCLECOUNTER
READCYCLECOUNTER - This corresponds to the readcyclecounter intrinsic.
@ RELOC_NONE
Issue a no-op relocation against a given symbol at the current location.
@ AND
Bitwise operators - logical and, logical or, logical xor.
@ TRAP
TRAP - Trapping instruction.
@ INTRINSIC_WO_CHAIN
RESULT = INTRINSIC_WO_CHAIN(INTRINSICID, arg1, arg2, ...) This node represents a target intrinsic fun...
@ SCMP
[US]CMP - 3-way comparison of signed or unsigned integers.
@ VECTOR_SPLICE_RIGHT
VECTOR_SPLICE_RIGHT(VEC1, VEC2, OFFSET) - Shifts CONCAT_VECTORS(VEC1,VEC2) right by OFFSET elements a...
@ STRICT_FADD
Constrained versions of the binary floating point operators.
@ STACKMAP
The llvm.experimental.stackmap intrinsic.
@ FREEZE
FREEZE - FREEZE(VAL) returns an arbitrary value if VAL is UNDEF (or is evaluated to UNDEF),...
@ INSERT_VECTOR_ELT
INSERT_VECTOR_ELT(VECTOR, VAL, IDX) - Returns VECTOR with the element at IDX replaced with VAL.
@ TokenFactor
TokenFactor - This node takes multiple tokens as input and produces a single token result.
@ ATOMIC_SWAP
Val, OUTCHAIN = ATOMIC_SWAP(INCHAIN, ptr, amt) Val, OUTCHAIN = ATOMIC_LOAD_[OpName](INCHAIN,...
@ CTTZ_ZERO_POISON
Bit counting operators with a poisoned result for zero inputs.
@ FFREXP
FFREXP - frexp, extract fractional and exponent component of a floating-point value.
@ FP_ROUND
X = FP_ROUND(Y, TRUNC) - Rounding 'Y' from a larger floating point type down to the precision of the ...
@ VECTOR_COMPRESS
VECTOR_COMPRESS(Vec, Mask, Passthru) consecutively place vector elements based on mask e....
@ SPONENTRY
SPONENTRY - Represents the llvm.sponentry intrinsic.
@ CLEAR_CACHE
llvm.clear_cache intrinsic Operands: Input Chain, Start Addres, End Address Outputs: Output Chain
@ INLINEASM
INLINEASM - Represents an inline asm block.
@ FP_TO_SINT_SAT
FP_TO_[US]INT_SAT - Convert floating point value in operand 0 to a signed or unsigned scalar integer ...
@ EH_SJLJ_SETJMP
RESULT, OUTCHAIN = EH_SJLJ_SETJMP(INCHAIN, buffer) This corresponds to the eh.sjlj....
@ TRUNCATE
TRUNCATE - Completely drop the high bits.
@ BRCOND
BRCOND - Conditional branch.
@ CONVERT_TO_ARBITRARY_FP
CONVERT_TO_ARBITRARY_FP - Converts a native FP value to an arbitrary floating-point format,...
@ CATCHRET
CATCHRET - Represents a return from a catch block funclet.
@ AssertSext
AssertSext, AssertZext - These nodes record if a register contains a value that has already been zero...
@ FCOPYSIGN
FCOPYSIGN(X, Y) - Return the value of X with the sign of Y.
@ SADDSAT
RESULT = [US]ADDSAT(LHS, RHS) - Perform saturation addition on 2 integers with the same bit width (W)...
@ VECTOR_DEINTERLEAVE
VECTOR_DEINTERLEAVE(VEC1, VEC2, ...) - Returns N vectors from N input vectors, where N is the factor ...
@ GET_DYNAMIC_AREA_OFFSET
GET_DYNAMIC_AREA_OFFSET - get offset from native SP to the address of the most recent dynamic alloca.
@ FMINIMUMNUM
FMINIMUMNUM/FMAXIMUMNUM - minimumnum/maximumnum that is same with FMINNUM_IEEE and FMAXNUM_IEEE besid...
@ ADJUST_TRAMPOLINE
ADJUST_TRAMPOLINE - This corresponds to the adjust_trampoline intrinsic.
@ INTRINSIC_W_CHAIN
RESULT,OUTCHAIN = INTRINSIC_W_CHAIN(INCHAIN, INTRINSICID, arg1, ...) This node represents a target in...
@ ABS_MIN_POISON
ABS with a poison result for INT_MIN.
@ BUILD_VECTOR
BUILD_VECTOR(ELT0, ELT1, ELT2, ELT3,...) - Return a fixed-width vector with the specified,...
@ LOOP_DEPENDENCE_WAR_MASK
The llvm.loop.dependence.
CondCode
ISD::CondCode enum - These are ordered carefully to make the bitfields below work out,...
This namespace contains an enum with a value for every intrinsic/builtin function known by LLVM.
Flag
These should be considered private to the implementation of the MCInstrDesc class.
BinaryOp_match< SrcTy, SpecificConstantMatch, TargetOpcode::G_XOR, true > m_Not(const SrcTy &&Src)
Matches a register not-ed by a G_XOR.
OneUse_match< SubPat > m_OneUse(const SubPat &SP)
bool match(Val *V, const Pattern &P)
specificval_ty m_Specific(const Value *V)
Match if we have a specific specified value.
TwoOps_match< Val_t, Idx_t, Instruction::ExtractElement > m_ExtractElt(const Val_t &Val, const Idx_t &Idx)
Matches ExtractElementInst.
IntrinsicID_match m_VScale()
Matches a call to llvm.vscale().
auto m_Value()
Match an arbitrary value and ignore it.
auto m_LogicalOr()
Matches L || R where L and R are arbitrary values.
auto m_LogicalAnd()
Matches L && R where L and R are arbitrary values.
std::pair< JumpTableHeader, JumpTable > JumpTableBlock
LLVM_ABI void sortAndRangeify(CaseClusterVector &Clusters)
Sort Clusters and merge adjacent cases.
std::vector< CaseCluster > CaseClusterVector
@ CC_Range
A cluster of adjacent case labels with the same destination, or just one case.
@ CC_JumpTable
A cluster of cases suitable for jump table lowering.
@ CC_BitTests
A cluster of cases suitable for bit test lowering.
SmallVector< SwitchWorkListItem, 4 > SwitchWorkList
CaseClusterVector::iterator CaseClusterIt
initializer< Ty > init(const Ty &Val)
LocationClass< Ty > location(Ty &L)
@ DW_OP_LLVM_arg
Only used in LLVM metadata.
ExceptionBehavior
Exception behavior used for floating point operations.
@ ebStrict
This corresponds to "fpexcept.strict".
@ ebMayTrap
This corresponds to "fpexcept.maytrap".
@ ebIgnore
This corresponds to "fpexcept.ignore".
NodeAddr< FuncNode * > Func
friend class Instruction
Iterator for Instructions in a `BasicBlock.
This is an optimization pass for GlobalISel generic memory operations.
@ Low
Lower the current thread's priority such that it does not affect foreground tasks significantly.
unsigned Log2_32_Ceil(uint32_t Value)
Return the ceil log base 2 of the specified value, 32 if the value is zero.
LLVM_ABI ISD::CondCode getICmpCondCode(ICmpInst::Predicate Pred)
getICmpCondCode - Return the ISD condition code corresponding to the given LLVM IR integer condition ...
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
SDValue peekThroughFreeze(SDValue V)
Return the non-frozen source operand of V if it exists.
LLVM_ABI void GetReturnInfo(CallingConv::ID CC, Type *ReturnType, AttributeList attr, SmallVectorImpl< ISD::OutputArg > &Outs, const TargetLowering &TLI, const DataLayout &DL)
Given an LLVM IR type and return type attributes, compute the return value EVTs and flags,...
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
RelativeUniformCounterPtr Values
LLVM_ABI bool isOnlyUsedInZeroEqualityComparison(const Instruction *CxtI)
LLVM_ABI void ComputeValueVTs(const TargetLowering &TLI, const DataLayout &DL, Type *Ty, SmallVectorImpl< EVT > &ValueVTs, SmallVectorImpl< EVT > *MemVTs=nullptr, SmallVectorImpl< TypeSize > *Offsets=nullptr, TypeSize StartingOffset=TypeSize::getZero())
ComputeValueVTs - Given an LLVM IR type, compute a sequence of EVTs that represent all the individual...
LLVM_ABI SDValue peekThroughBitcasts(SDValue V)
Return the non-bitcasted source operand of V if it exists.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
int countr_one(T Value)
Count the number of ones from the least significant bit to the first zero bit.
LLVM_ABI void diagnoseDontCall(const CallInst &CI)
auto successors(const MachineBasicBlock *BB)
bool isIntOrFPConstant(SDValue V)
Return true if V is either a integer or FP constant.
static ConstantRange getRange(Value *Op, SCCPSolver &Solver, const SmallPtrSetImpl< Value * > &InsertedValues)
Helper for getting ranges from Solver.
void append_range(Container &C, Range &&R)
Wrapper function to append range R to container C.
Value * GetPointerBaseWithConstantOffset(Value *Ptr, int64_t &Offset, const DataLayout &DL, bool AllowNonInbounds=true)
Analyze the specified pointer to see if it can be expressed as a base pointer plus a constant offset.
constexpr bool isUIntN(unsigned N, uint64_t x)
Checks if an unsigned integer fits into the given (dynamic) bit width.
auto cast_or_null(const Y &Val)
constexpr T alignDown(U Value, V Align, W Skew=0)
Returns the largest unsigned integer less than or equal to Value and is Skew mod Align.
gep_type_iterator gep_type_end(const User *GEP)
LLVM_ABI LLT getLLTForMVT(MVT Ty)
Get a rough equivalent of an LLT for a given MVT.
constexpr auto equal_to(T &&Arg)
Functor variant of std::equal_to that can be used as a UnaryPredicate in functional algorithms like a...
constexpr int popcount(T Value) noexcept
Count the number of set bits in a value.
LLVM_ABI ConstantRange getConstantRangeFromMetadata(const MDNode &RangeMD)
Parse out a conservative ConstantRange from !range metadata.
detail::concat_range< ValueT, RangeTs... > concat(RangeTs &&...Ranges)
Returns a concatenated range across two or more ranges.
bool isScopedEHPersonality(EHPersonality Pers)
Returns true if this personality uses scope-style EH IR instructions: catchswitch,...
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr Value
int countr_zero(T Val)
Count number of 0's from the least significant bit to the most stopping at the first 1.
LLVM_ABI void ComputeValueTypes(const DataLayout &DL, Type *Ty, SmallVectorImpl< Type * > &Types, SmallVectorImpl< TypeSize > *Offsets=nullptr, TypeSize StartingOffset=TypeSize::getZero())
Given an LLVM IR type, compute non-aggregate subtypes.
auto dyn_cast_or_null(const Y &Val)
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI llvm::SmallVector< int, 16 > createStrideMask(unsigned Start, unsigned Stride, unsigned VF)
Create a stride shuffle mask.
@ SPF_ABS
Floating point maxnum.
@ SPF_NABS
Absolute value.
@ SPF_FMAXNUM
Floating point minnum.
@ SPF_UMIN
Signed minimum.
@ SPF_UMAX
Signed maximum.
@ SPF_SMAX
Unsigned minimum.
@ SPF_FMINNUM
Unsigned maximum.
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
detail::zippy< detail::zip_first, T, U, Args... > zip_first(T &&t, U &&u, Args &&...args)
zip iterator that, for the sake of efficiency, assumes the first iteratee to be the shortest.
void sort(IteratorTy Start, IteratorTy End)
FPClassTest
Floating-point class tests, supported by 'is_fpclass' intrinsic.
LLVM_ABI SelectPatternResult matchSelectPattern(Value *V, Value *&LHS, Value *&RHS, Instruction::CastOps *CastOp=nullptr, unsigned Depth=0)
Pattern match integer [SU]MIN, [SU]MAX and ABS idioms, returning the kind and providing the out param...
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
constexpr uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
generic_gep_type_iterator<> gep_type_iterator
auto succ_size(const MachineBasicBlock *BB)
bool hasSingleElement(ContainerTy &&C)
Returns true if the given container only contains a single element.
LLVM_ABI ISD::CondCode getFCmpCondCode(FCmpInst::Predicate Pred)
getFCmpCondCode - Return the ISD condition code corresponding to the given LLVM IR floating-point con...
LLVM_ABI EHPersonality classifyEHPersonality(const Value *Pers)
See if the given exception handling personality function is one that we understand.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
LLVM_ATTRIBUTE_VISIBILITY_DEFAULT AnalysisKey InnerAnalysisManagerProxy< AnalysisManagerT, IRUnitT, ExtraArgTs... >::Key
LLVM_ABI Value * salvageDebugInfoImpl(Instruction &I, uint64_t CurrentLocOps, SmallVectorImpl< uint64_t > &Ops, SmallVectorImpl< Value * > &AdditionalValues)
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ Global
Append to llvm.global_dtors.
AtomicOrdering
Atomic ordering for LLVM's memory model.
@ First
Helpers to iterate all locations in the MemoryEffectsBase class.
bool isFuncletEHPersonality(EHPersonality Pers)
Returns true if this is a personality function that invokes handler funclets (which must return to it...
LLVM_ABI bool isAssignmentTrackingEnabled(const Module &M)
Return true if assignment tracking is enabled for module M.
LLVM_ABI llvm::SmallVector< int, 16 > createInterleaveMask(unsigned VF, unsigned NumVecs)
Create an interleave shuffle mask.
@ UMin
Unsigned integer min implemented in terms of select(cmp()).
@ Or
Bitwise or logical OR of integers.
@ Mul
Product of integers.
@ Sub
Subtraction of integers.
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr Count
@ SPNB_RETURNS_NAN
NaN behavior not applicable.
@ SPNB_RETURNS_OTHER
Given one NaN input, returns the NaN.
@ SPNB_RETURNS_ANY
Given one NaN input, returns the non-NaN.
LLVM_ABI bool isInTailCallPosition(const CallBase &Call, const TargetMachine &TM, bool ReturnsFirstArg=false)
Test if the given instruction is in a position to be optimized with a tail-call.
DWARFExpression::Operation Op
@ Dynamic
Denotes mode unknown at compile time.
LLVM_ABI ISD::CondCode getFCmpCodeWithoutNaN(ISD::CondCode CC)
getFCmpCodeWithoutNaN - Given an ISD condition code comparing floats, return the equivalent code if w...
ArrayRef(const T &OneElt) -> ArrayRef< T >
bool isAsynchronousEHPersonality(EHPersonality Pers)
Returns true if this personality function catches asynchronous exceptions.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI bool isKnownNeverNaN(const Value *V, const SimplifyQuery &SQ, unsigned Depth=0)
Return true if the floating-point scalar value is not a NaN or if the floating-point vector value has...
LLVM_ABI std::optional< RoundingMode > convertStrToRoundingMode(StringRef)
Returns a valid RoundingMode enumerator when given a string that is valid as input in constrained int...
gep_type_iterator gep_type_begin(const User *GEP)
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
LLVM_ABI GlobalValue * ExtractTypeInfo(Value *V)
ExtractTypeInfo - Returns the type info, possibly bitcast, encoded in V.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
Align commonAlignment(Align A, uint64_t Offset)
Returns the alignment that satisfies both alignments.
bool all_equal(std::initializer_list< T > Values)
Returns true if all Values in the initializer lists are equal or the list.
LLVM_ABI Constant * ConstantFoldLoadFromConstPtr(Constant *C, Type *Ty, APInt Offset, const DataLayout &DL)
Return the value that a load from C with offset Offset would produce if it is constant and determinab...
LLVM_ABI unsigned ComputeLinearIndex(Type *Ty, const unsigned *Indices, const unsigned *IndicesEnd, unsigned CurIndex=0)
Compute the linearized index of a member in a nested aggregate/struct/array.
T bit_floor(T Value)
Returns the largest integral power of two no greater than Value if Value is nonzero.
@ Default
The result value is uniform if and only if all operands are uniform.
LLVM_ABI void reportFatalUsageError(Error Err)
Report a fatal error that does not indicate a bug in LLVM.
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.
This struct is a compact representation of a valid (non-zero power of two) alignment.
constexpr uint64_t value() const
This is a hole in the type system and should not be abused.
TypeSize getStoreSize() const
Return the number of bytes overwritten by a store of the specified value type.
bool isSimple() const
Test if the given EVT is simple (as opposed to being extended).
static EVT getVectorVT(LLVMContext &Context, EVT VT, unsigned NumElements, bool IsScalable=false)
Returns the EVT that represents a vector NumElements in length, where each element is of type VT.
uint64_t getScalarStoreSize() const
bool bitsGT(EVT VT) const
Return true if this has more bits than VT.
bool bitsLT(EVT VT) const
Return true if this has less bits than VT.
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
ElementCount getVectorElementCount() const
TypeSize getSizeInBits() const
Return the size of the specified value type in bits.
unsigned getVectorMinNumElements() const
Given a vector type, return the minimum number of elements it contains.
uint64_t getScalarSizeInBits() const
static LLVM_ABI EVT getEVT(Type *Ty, bool HandleUnknown=false)
Return the value type corresponding to the specified type.
EVT changeVectorElementType(LLVMContext &Context, EVT EltVT) const
Return a VT for a vector type whose attributes match ourselves with the exception of the element type...
MVT getSimpleVT() const
Return the SimpleValueType held in the specified simple EVT.
static EVT getIntegerVT(LLVMContext &Context, unsigned BitWidth)
Returns the EVT that represents an integer with the given number of bits.
bool isRISCVVectorTuple() const
Return true if this is a vector value type.
uint64_t getFixedSizeInBits() const
Return the size of the specified fixed width value type in bits.
bool isFixedLengthVector() const
bool isVector() const
Return true if this is a vector value type.
EVT getScalarType() const
If this is a vector type, return the element type, otherwise return this.
bool bitsGE(EVT VT) const
Return true if this has no less bits than VT.
bool isScalableVector() const
Return true if this is a vector type where the runtime length is machine dependent.
EVT getVectorElementType() const
Given a vector type, return the type of each element.
EVT changeElementType(LLVMContext &Context, EVT EltVT) const
Return a VT for a type whose attributes match ourselves with the exception of the element type that i...
bool isScalarInteger() const
Return true if this is an integer, but not a vector.
unsigned getVectorNumElements() const
Given a vector type, return the number of elements it contains.
bool isInteger() const
Return true if this is an integer or a vector integer type.
void setPointerAddrSpace(unsigned AS)
void setOrigAlign(Align A)
OutputArg - This struct carries flags and a value for a single outgoing (actual) argument or outgoing...
ConstraintPrefix Type
Type - The basic type of the constraint: input/output/clobber/label.
unsigned countMinLeadingZeros() const
Returns the minimum number of leading zero bits.
This class contains a discriminated union of information about pointers in memory operands,...
static LLVM_ABI MachinePointerInfo getUnknownStack(MachineFunction &MF)
Stack memory without other information.
static LLVM_ABI MachinePointerInfo getFixedStack(MachineFunction &MF, int FI, int64_t Offset=0)
Return a MachinePointerInfo record that refers to the specified FrameIndex.
A lightweight accessor for an operand bundle meant to be passed around by value.
This struct represents the registers (physical or virtual) that a particular set of values is assigne...
SmallVector< std::pair< Register, TypeSize >, 4 > getRegsAndSizes() const
Return a list of registers and their sizes.
SmallVector< unsigned, 4 > RegCount
This list holds the number of registers for each value.
bool isABIMangled() const
SmallVector< EVT, 4 > ValueVTs
The value types of the values, which may not be legal, and may need be promoted or synthesized from o...
SmallVector< Register, 4 > Regs
This list holds the registers assigned to the values.
void AddInlineAsmOperands(InlineAsm::Kind Code, bool HasMatching, unsigned MatchingIdx, const SDLoc &dl, SelectionDAG &DAG, std::vector< SDValue > &Ops) const
Add this value to the specified inlineasm node operand list.
SDValue getCopyFromRegs(SelectionDAG &DAG, FunctionLoweringInfo &FuncInfo, const SDLoc &dl, SDValue &Chain, SDValue *Glue, const Value *V=nullptr) const
Emit a series of CopyFromReg nodes that copies from this value and returns the result as a ValueVTs v...
SmallVector< MVT, 4 > RegVTs
The value types of the registers.
void getCopyToRegs(SDValue Val, SelectionDAG &DAG, const SDLoc &dl, SDValue &Chain, SDValue *Glue, const Value *V=nullptr, ISD::NodeType PreferredExtendType=ISD::ANY_EXTEND) const
Emit a series of CopyToReg nodes that copies the specified value into the registers specified by this...
std::optional< CallingConv::ID > CallConv
Records if this value needs to be treated in an ABI dependant manner, different to normal type legali...
bool occupiesMultipleRegs() const
Check if the total RegCount is greater than one.
These are IR-level optimization flags that may be propagated to SDNodes.
void copyFMF(const FPMathOperator &FPMO)
Propagate the fast-math-flags from an IR FPMathOperator.
void setUnpredictable(bool b)
bool hasAllowReassociation() const
void setNoUnsignedWrap(bool b)
void setNoSignedWrap(bool b)
This represents a list of ValueType's that has been intern'd by a SelectionDAG.
A MapVector that performs no allocations if smaller than a certain size.
MachineBasicBlock * Default
BranchProbability DefaultProb
MachineBasicBlock * Parent
bool FallthroughUnreachable
MachineBasicBlock * ThisBB
This structure is used to communicate between SelectionDAGBuilder and SDISel for the code generation ...
BranchProbability TrueProb
BranchProbability FalseProb
MachineBasicBlock * TrueBB
MachineBasicBlock * FalseBB
SDLoc DL
The debug location of the instruction this CaseBlock was produced from.
static CaseCluster range(const ConstantInt *Low, const ConstantInt *High, MachineBasicBlock *MBB, BranchProbability Prob)
Register Reg
The virtual register containing the index of the jump table entry to jump to.
MachineBasicBlock * Default
The MBB of the default bb, which is a successor of the range check MBB.
unsigned JTI
The JumpTableIndex for this jump table in the function.
MachineBasicBlock * MBB
The MBB into which to emit the code for the indirect jump.
std::optional< SDLoc > SL
The debug location of the instruction this JumpTable was produced from.
This contains information for each constraint that we are lowering.
TargetLowering::ConstraintType ConstraintType
Information about the constraint code, e.g.
This structure contains all information that is necessary for lowering calls.
CallLoweringInfo & setConvergent(bool Value=true)
CallLoweringInfo & setDeactivationSymbol(GlobalValue *Sym)
CallLoweringInfo & setCFIType(const ConstantInt *Type)
SmallVector< ISD::InputArg, 32 > Ins
bool IsPostTypeLegalization
SmallVector< SDValue, 4 > InVals
Type * OrigRetTy
Original unlegalized return type.
CallLoweringInfo & setDiscardResult(bool Value=true)
CallLoweringInfo & setIsPatchPoint(bool Value=true)
CallLoweringInfo & setDebugLoc(const SDLoc &dl)
CallLoweringInfo & setTailCall(bool Value=true)
CallLoweringInfo & setIsPreallocated(bool Value=true)
CallLoweringInfo & setConvergenceControlToken(SDValue Token)
SmallVector< ISD::OutputArg, 32 > Outs
SmallVector< SDValue, 32 > OutVals
Type * RetTy
Same as OrigRetTy, or partially legalized for soft float libcalls.
CallLoweringInfo & setChain(SDValue InChain)
CallLoweringInfo & setPtrAuth(PtrAuthInfo Value)
CallLoweringInfo & setCallee(CallingConv::ID CC, Type *ResultType, SDValue Target, ArgListTy &&ArgsList, AttributeSet ResultAttrs={})
This structure is used to pass arguments to makeLibCall function.
MakeLibCallOptions & setDiscardResult(bool Value=true)
This structure contains the information necessary for lowering pointer-authenticating indirect calls.
LLVM_ABI void addIPToStateRange(const InvokeInst *II, MCSymbol *InvokeBegin, MCSymbol *InvokeEnd)