44#define DEBUG_TYPE "M68k-isel"
58 auto *RegInfo = Subtarget.getRegisterInfo();
82 if (Subtarget.atLeastM68020())
111 for (
auto VT : {MVT::i8, MVT::i16, MVT::i32}) {
119 for (
auto VT : {MVT::i8, MVT::i16, MVT::i32}) {
129 for (
auto VT : {MVT::i8, MVT::i16, MVT::i32}) {
137 for (
auto VT : {MVT::i8, MVT::i16, MVT::i32}) {
188 {MVT::i8, MVT::i16, MVT::i32},
LibCall);
195 return Subtarget.atLeastM68020()
228 return Ty.getSimpleVT();
233#include "M68kGenCallingConv.inc"
271 Align Alignment = Flags.getNonZeroByValAlign();
273 return DAG.
getMemcpy(Chain,
DL, Dst, Src, SizeNode, Alignment, Alignment,
335 if (!Flags.isByVal()) {
336 if (!
TII->isLoadFromStackSlot(*Def, FI))
339 unsigned Opcode = Def->getOpcode();
340 if ((Opcode == M68k::LEA32p || Opcode == M68k::LEA32f) &&
341 Def->getOperand(1).isFI()) {
342 FI = Def->getOperand(1).getIndex();
343 Bytes = Flags.getByValSize();
355 SDValue Ptr = Ld->getBasePtr();
363 Bytes = Flags.getByValSize();
387M68kTargetLowering::getReturnAddressFrameIndex(
SelectionDAG &DAG)
const {
389 M68kMachineFunctionInfo *FuncInfo = MF.
getInfo<M68kMachineFunctionInfo>();
392 if (ReturnAddrIndex == 0) {
394 unsigned SlotSize = Subtarget.getSlotSize();
396 SlotSize, -(int64_t)SlotSize,
false);
406 bool IsTailCall,
int FPDiff,
409 OutRetAddr = getReturnAddressFrameIndex(DAG);
412 OutRetAddr = DAG.
getLoad(VT,
DL, Chain, OutRetAddr, MachinePointerInfo());
416SDValue M68kTargetLowering::EmitTailCallStoreRetAddr(
418 EVT PtrVT,
unsigned SlotSize,
int FPDiff,
const SDLoc &
DL)
const {
424 SlotSize, (int64_t)FPDiff - SlotSize,
false);
429 Chain,
DL, RetFI, NewFI,
440 unsigned ArgIdx)
const {
442 ISD::ArgFlagsTy
Flags = Ins[ArgIdx].Flags;
457 }
else if (VA.
getValVT() == MVT::i16) {
471 bool IsImmutable = !AlwaysUseMutable && !
Flags.isByVal();
473 if (
Flags.isByVal()) {
474 unsigned Bytes =
Flags.getByValSize();
497 ValVT,
DL, Chain, FIN,
517 Chain,
DL, Arg, PtrOff,
527 SelectionDAG &DAG = CLI.
DAG;
529 SmallVectorImpl<ISD::OutputArg> &Outs = CLI.
Outs;
530 SmallVectorImpl<SDValue> &OutVals = CLI.
OutVals;
531 SmallVectorImpl<ISD::InputArg> &Ins = CLI.
Ins;
540 bool IsSibcall =
false;
541 M68kMachineFunctionInfo *MFI = MF.
getInfo<M68kMachineFunctionInfo>();
548 if (Attr.getValueAsBool())
559 }
else if (IsTailCall) {
561 IsTailCall = IsEligibleForTailCallOptimization(
576 "Var args not supported with calling convention fastcc");
581 for (
const auto &Arg : CLI.
getArgs())
583 M68kCCState CCInfo(ArgTypes, CallConv, IsVarArg, MF, ArgLocs,
585 CCInfo.AnalyzeCallOperands(Outs, CC_M68k);
588 unsigned NumBytes = CCInfo.getAlignedCallFrameSize();
595 NumBytes = GetAlignedArgumentStackSize(NumBytes, DAG);
599 if (IsTailCall && !IsSibcall && !IsMustTail) {
603 FPDiff = NumBytesCallerPushed - NumBytes;
607 if (FPDiff < MFI->getTCReturnAddrDelta())
611 unsigned NumBytesToPush = NumBytes;
612 unsigned NumBytesToPop = NumBytes;
617 if (!Outs.
empty() && Outs.
back().Flags.isInAlloca()) {
619 if (!ArgLocs.
back().isMemLoc())
622 if (ArgLocs.
back().getLocMemOffset() != 0)
624 "the only memory argument");
629 NumBytes - NumBytesToPush,
DL);
633 if (IsTailCall && FPDiff)
634 Chain = EmitTailCallLoadRetAddr(DAG, RetFI, Chain, IsTailCall, FPDiff,
DL);
642 const M68kRegisterInfo *RegInfo = Subtarget.getRegisterInfo();
643 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i) {
644 ISD::ArgFlagsTy
Flags = Outs[i].Flags;
647 if (
Flags.isInAlloca())
650 CCValAssign &VA = ArgLocs[i];
653 bool IsByVal =
Flags.isByVal();
678 Chain,
DL, Arg, SpillSlot,
687 }
else if (!IsSibcall && (!IsTailCall || IsByVal)) {
694 LowerMemOpCallTo(Chain, StackPtr, Arg,
DL, DAG, VA, Flags));
698 if (!MemOpChains.
empty())
705 if (IsVarArg && IsMustTail) {
707 for (
const auto &
F : Forwards) {
709 RegsToPass.
push_back(std::make_pair(
unsigned(
F.PReg), Val));
716 if (!IsSibcall && IsTailCall) {
728 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i) {
729 CCValAssign &VA = ArgLocs[i];
734 ISD::ArgFlagsTy
Flags = Outs[i].Flags;
736 if (
Flags.isInAlloca())
740 uint32_t OpSize = (VA.
getLocVT().getSizeInBits() + 7) / 8;
744 if (
Flags.isByVal()) {
759 ArgChain,
DL, Arg, FIN,
764 if (!MemOpChains2.
empty())
768 Chain = EmitTailCallStoreRetAddr(DAG, MF, Chain, RetFI,
770 Subtarget.getSlotSize(), FPDiff,
DL);
776 for (
unsigned i = 0, e = RegsToPass.
size(); i != e; ++i) {
778 RegsToPass[i].second, InGlue);
790 const GlobalValue *GV =
G->getGlobal();
792 unsigned char OpFlags = Subtarget.classifyGlobalFunctionReference(GV);
811 unsigned char OpFlags =
812 Subtarget.classifyGlobalFunctionReference(
nullptr, *
Mod);
820 if (!IsSibcall && IsTailCall) {
825 Ops.push_back(Chain);
826 Ops.push_back(Callee);
833 for (
unsigned i = 0, e = RegsToPass.
size(); i != e; ++i)
835 RegsToPass[i].second.getValueType()));
839 assert(Mask &&
"Missing call preserved mask for calling convention");
844 Ops.push_back(InGlue);
848 return DAG.
getNode(M68kISD::TC_RETURN,
DL, MVT::Other,
Ops);
852 Chain = DAG.
getNode(M68kISD::CALL,
DL, {MVT::Other, MVT::Glue},
Ops);
856 unsigned NumBytesForCalleeToPop;
859 NumBytesForCalleeToPop = NumBytes;
863 NumBytesForCalleeToPop = 4;
865 NumBytesForCalleeToPop = 0;
871 NumBytesForCalleeToPop = NumBytes;
876 Chain = DAG.
getCALLSEQ_END(Chain, NumBytesToPop, NumBytesForCalleeToPop,
883 return LowerCallResult(Chain, InGlue, CallConv, IsVarArg, Ins,
DL, DAG,
887SDValue M68kTargetLowering::LowerCallResult(
896 CCInfo.AnalyzeCallResult(Ins, RetCC_M68k);
899 for (
unsigned i = 0, e = RVLocs.
size(); i != e; ++i) {
900 CCValAssign &VA = RVLocs[i];
922SDValue M68kTargetLowering::LowerFormalArguments(
927 M68kMachineFunctionInfo *MMFI = MF.
getInfo<M68kMachineFunctionInfo>();
937 M68kCCState CCInfo(ArgTypes, CCID, IsVarArg, MF, ArgLocs, *DAG.
getContext());
939 CCInfo.AnalyzeFormalArguments(Ins, CC_M68k);
941 unsigned LastVal = ~0
U;
943 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i) {
944 CCValAssign &VA = ArgLocs[i];
945 assert(VA.
getValNo() != LastVal &&
"Same value in different locations");
952 const TargetRegisterClass *RC;
953 if (RegVT == MVT::i32)
954 RC = &M68k::XR32RegClass;
979 ArgValue = LowerMemArgument(Chain, CCID, Ins,
DL, DAG, VA, MFI, i);
991 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i) {
1000 if (Ins[i].
Flags.isSRet()) {
1013 unsigned StackSize = CCInfo.getStackSize();
1016 StackSize = GetAlignedArgumentStackSize(StackSize, DAG);
1021 if (MFI.hasVAStart()) {
1025 if (IsVarArg && MFI.hasMustTailInVarArgFunc()) {
1028 MVT IntVT = MVT::i32;
1033 SmallVectorImpl<ForwardedRegister> &Forwards =
1035 CCInfo.analyzeMustTailForwardedRegisters(Forwards, RegParmTypes, CC_M68k);
1038 for (ForwardedRegister &
F : Forwards) {
1066bool M68kTargetLowering::CanLowerReturn(
1069 const Type *RetTy)
const {
1071 CCState CCInfo(CCID, IsVarArg, MF, RVLocs,
Context);
1072 return CCInfo.CheckReturn(Outs, RetCC_M68k);
1082 M68kMachineFunctionInfo *MFI = MF.
getInfo<M68kMachineFunctionInfo>();
1085 CCState CCInfo(CCID, IsVarArg, MF, RVLocs, *DAG.
getContext());
1086 CCInfo.AnalyzeReturn(Outs, RetCC_M68k);
1097 for (
unsigned i = 0, e = RVLocs.size(); i != e; ++i) {
1098 CCValAssign &VA = RVLocs[i];
1100 SDValue ValToCopy = OutVals[i];
1158 unsigned RetValReg = M68k::D0;
1172 return DAG.
getNode(M68kISD::RET,
DL, MVT::Other, RetOps);
1208M68kTargetLowering::GetAlignedArgumentStackSize(
unsigned StackSize,
1210 const TargetFrameLowering &TFI = *Subtarget.getFrameLowering();
1212 uint64_t AlignMask = StackAlignment - 1;
1213 int64_t
Offset = StackSize;
1214 unsigned SlotSize = Subtarget.getSlotSize();
1215 if ((
Offset & AlignMask) <= (StackAlignment - SlotSize)) {
1217 Offset += ((StackAlignment - SlotSize) - (
Offset & AlignMask));
1221 ((~AlignMask) &
Offset) + StackAlignment + (StackAlignment - SlotSize);
1228bool M68kTargetLowering::IsEligibleForTailCallOptimization(
1230 bool IsCalleeStructRet,
bool IsCallerStructRet,
Type *RetTy,
1242 bool CCMatch = CallerCC == CalleeCC;
1255 const M68kRegisterInfo *RegInfo = Subtarget.getRegisterInfo();
1256 if (RegInfo->hasStackRealignment(MF))
1261 if (IsCalleeStructRet || IsCallerStructRet)
1267 if (IsVarArg && !Outs.
empty()) {
1270 CCState CCInfo(CalleeCC, IsVarArg, MF, ArgLocs,
C);
1272 CCInfo.AnalyzeCallOperands(Outs, CC_M68k);
1273 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i)
1274 if (!ArgLocs[i].isRegLoc())
1284 const M68kRegisterInfo *
TRI = Subtarget.getRegisterInfo();
1285 const uint32_t *CallerPreserved =
TRI->getCallPreservedMask(MF, CallerCC);
1287 const uint32_t *CalleePreserved =
TRI->getCallPreservedMask(MF, CalleeCC);
1288 if (!
TRI->regmaskSubsetEqual(CallerPreserved, CalleePreserved))
1292 unsigned StackArgsSize = 0;
1296 if (!Outs.
empty()) {
1300 CCState CCInfo(CalleeCC, IsVarArg, MF, ArgLocs,
C);
1302 CCInfo.AnalyzeCallOperands(Outs, CC_M68k);
1303 StackArgsSize = CCInfo.getStackSize();
1305 if (StackArgsSize) {
1309 const MachineRegisterInfo *MRI = &MF.
getRegInfo();
1310 const M68kInstrInfo *
TII = Subtarget.getInstrInfo();
1311 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i) {
1312 CCValAssign &VA = ArgLocs[i];
1314 ISD::ArgFlagsTy
Flags = Outs[i].Flags;
1333 PositionIndependent) {
1334 unsigned NumInRegs = 0;
1337 unsigned MaxInRegs = PositionIndependent ? 1 : 2;
1339 for (
unsigned i = 0, e = ArgLocs.
size(); i != e; ++i) {
1340 CCValAssign &VA = ArgLocs[i];
1349 if (++NumInRegs == MaxInRegs)
1356 const MachineRegisterInfo &MRI = MF.
getRegInfo();
1364 if (
unsigned BytesToPop =
1365 MF.
getInfo<M68kMachineFunctionInfo>()->getBytesToPopOnReturn()) {
1367 bool CalleePopMatches = CalleeWillPop && BytesToPop == StackArgsSize;
1368 if (!CalleePopMatches)
1370 }
else if (CalleeWillPop && StackArgsSize > 0) {
1384 switch (
Op.getOpcode()) {
1393 return LowerXALUO(
Op, DAG);
1395 return LowerSETCC(
Op, DAG);
1397 return LowerSETCCCARRY(
Op, DAG);
1399 return LowerSELECT(
Op, DAG);
1401 return LowerBRCOND(
Op, DAG);
1406 return LowerADDC_ADDE_SUBC_SUBE(
Op, DAG);
1408 return LowerConstantPool(
Op, DAG);
1410 return LowerGlobalAddress(
Op, DAG);
1412 return LowerExternalSymbol(
Op, DAG);
1414 return LowerBlockAddress(
Op, DAG);
1416 return LowerJumpTable(
Op, DAG);
1418 return LowerVASTART(
Op, DAG);
1420 return LowerDYNAMIC_STACKALLOC(
Op, DAG);
1422 return LowerShiftLeftParts(
Op, DAG);
1424 return LowerShiftRightParts(
Op, DAG,
true);
1426 return LowerShiftRightParts(
Op, DAG,
false);
1428 return LowerATOMICFENCE(
Op, DAG);
1430 return LowerGlobalTLSAddress(
Op, DAG);
1437 ArgListTy &&ArgList)
const {
1439 CallLoweringInfo CLI(DAG);
1445 std::move(ArgList));
1451 unsigned TargetFlags)
const {
1460 Args.emplace_back(Arg, PtrTy);
1461 return LowerExternalSymbolCall(DAG,
SDLoc(GA),
"__tls_get_addr",
1466 return LowerExternalSymbolCall(DAG, Loc,
"__m68k_read_tp",
ArgListTy());
1486 SDValue Tp = getM68kReadTp(SDLoc(GA), DAG);
1500 SDValue Tp = getM68kReadTp(SDLoc(GA), DAG);
1509 assert(Subtarget.isTargetELF());
1514 switch (AccessModel) {
1516 return LowerTLSGeneralDynamic(GA, DAG);
1518 return LowerTLSLocalDynamic(GA, DAG);
1520 return LowerTLSInitialExec(GA, DAG);
1522 return LowerTLSLocalExec(GA, DAG);
1535 return VT.
bitsLE(MVT::i32) || Subtarget.atLeastM68020();
1556 EVT VT =
N->getValueType(0);
1561 unsigned TruncOp = 0;
1562 auto PromoteMULO = [&](
unsigned ExtOp) {
1567 if (VT == MVT::i8) {
1575 bool NoOverflow =
false;
1576 unsigned BaseOp = 0;
1577 switch (
Op.getOpcode()) {
1581 BaseOp = M68kISD::ADD;
1585 BaseOp = M68kISD::ADD;
1589 BaseOp = M68kISD::SUB;
1593 BaseOp = M68kISD::SUB;
1598 NoOverflow = VT != MVT::i32;
1604 NoOverflow = VT != MVT::i32;
1621 Result = DAG.
getNode(TruncOp,
DL, MVT::i8, Arith);
1630 SDNode *
N =
Op.getNode();
1646 Overflow = DAG.
getNode(M68kISD::SETCC,
DL,
N->getValueType(1),
1660 if (Src.getValueType() == MVT::i8 || Src.getValueType() == MVT::i16)
1672 return DAG.
getNode(M68kISD::SETCC,
DL, MVT::i8,
1694 unsigned AndBitWidth =
And.getValueSizeInBits();
1697 if (Known.countMinLeadingZeros() <
BitWidth - AndBitWidth)
1704 uint64_t AndRHSVal = AndRHS->getZExtValue();
1726 switch (SetCCOpcode) {
1760 if (SetCCOpcode ==
ISD::SETGT && RHSC->isAllOnes()) {
1765 if (SetCCOpcode ==
ISD::SETLT && RHSC->isZero()) {
1769 if (SetCCOpcode ==
ISD::SETLT && RHSC->getZExtValue() == 1) {
1783 SetCCOpcode = getSetCCSwappedOperands(SetCCOpcode);
1787 switch (SetCCOpcode) {
1804 switch (SetCCOpcode) {
1840 "Expected TRUNCATE to i1 node");
1842 if (
Op.getOperand(0).getOpcode() !=
ISD::SRL)
1855 unsigned UOpNo = UI->getOperandNo();
1874 bool NeedCF =
false;
1875 bool NeedOF =
false;
1894 switch (
Op->getOpcode()) {
1899 if (
Op.getNode()->getFlags().hasNoSignedWrap())
1913 if (
Op.getResNo() != 0 || NeedOF || NeedCF) {
1915 return DAG.
getNode(M68kISD::CMP,
DL, MVT::i8,
1918 unsigned Opcode = 0;
1919 unsigned NumOperands = 0;
1924 bool NeedTruncation =
false;
1938 NeedTruncation =
true;
1949 Opcode = M68kISD::ADD;
1960 EVT VT =
Op.getValueType();
1962 unsigned ShAmt =
Op->getConstantOperandVal(1);
1968 if (!
Mask.isSignedIntN(32))
1983 bool IsLegalAndnType = VT == MVT::i32 || VT == MVT::i64;
1987 if ( !IsAndn || !IsLegalAndnType)
1996 for (
const auto *U :
Op.getNode()->users())
2005 Opcode = M68kISD::SUB;
2008 Opcode = M68kISD::XOR;
2011 Opcode = M68kISD::AND;
2014 Opcode = M68kISD::OR;
2033 if (NeedTruncation) {
2034 EVT VT =
Op.getValueType();
2037 unsigned ConvertedOp = 0;
2045 ConvertedOp = M68kISD::ADD;
2048 ConvertedOp = M68kISD::SUB;
2051 ConvertedOp = M68kISD::AND;
2054 ConvertedOp = M68kISD::OR;
2057 ConvertedOp = M68kISD::XOR;
2073 return DAG.
getNode(M68kISD::CMP,
DL, MVT::i8,
2076 SDVTList VTs = DAG.
getVTList(
Op.getValueType(), MVT::i8);
2107 return EmitTest(Op0, M68kCC,
DL, DAG);
2110 "Unexpected comparison operation for MVT::i1 operands");
2121 Op0 = DAG.
getNode(ExtendOp,
DL, MVT::i32, Op0);
2122 Op1 = DAG.
getNode(ExtendOp,
DL, MVT::i32, Op1);
2129 return DAG.
getNode(M68kISD::CMP,
DL, MVT::i8, Op0, Op1);
2145 MVT VT =
Op.getSimpleValueType();
2146 assert(VT == MVT::i8 &&
"SetCC type must be 8-bit integer");
2160 if (
SDValue NewSetCC = LowerToBTST(Op0, CC,
DL, DAG)) {
2174 if (Op0.
getOpcode() == M68kISD::SETCC) {
2182 DAG.
getNode(M68kISD::SETCC,
DL, MVT::i8,
2205 SDValue CCR = EmitCmp(Op0, Op1, M68kCC,
DL, DAG);
2206 return DAG.
getNode(M68kISD::SETCC,
DL, MVT::i8,
2218 assert(
LHS.getSimpleValueType().isInteger() &&
"SETCCCARRY is integer only.");
2226 SDVTList VTs = DAG.
getVTList(
LHS.getValueType(), MVT::i32);
2230 return DAG.
getNode(M68kISD::SETCC,
DL, MVT::i8,
2236 unsigned Opc =
Op.getNode()->getOpcode();
2237 if (
Opc == M68kISD::CMP)
2239 if (
Op.getResNo() == 1 &&
2240 (
Opc == M68kISD::ADD ||
Opc == M68kISD::SUB ||
Opc == M68kISD::ADDX ||
2241 Opc == M68kISD::SUBX ||
Opc == M68kISD::SMUL ||
Opc == M68kISD::UMUL ||
2242 Opc == M68kISD::OR ||
Opc == M68kISD::XOR ||
Opc == M68kISD::AND))
2245 if (
Op.getResNo() == 2 &&
Opc == M68kISD::UMUL)
2257 unsigned Bits = V.getValueSizeInBits();
2263 bool addTest =
true;
2279 if (
Cond.getOpcode() == M68kISD::SETCC &&
2280 Cond.getOperand(1).getOpcode() == M68kISD::CMP &&
2313 DAG.
getNode(M68kISD::SETCC_CARRY,
DL,
Op.getValueType(),
2327 Cond.getOperand(0).getOpcode() == M68kISD::SETCC_CARRY &&
2333 unsigned CondOpcode =
Cond.getOpcode();
2334 if (CondOpcode == M68kISD::SETCC || CondOpcode == M68kISD::SETCC_CARRY) {
2338 unsigned Opc =
Cmp.getOpcode();
2340 bool IllegalFPCMov =
false;
2380 if (
Cond.getOpcode() == M68kISD::SUB) {
2387 DAG.
getNode(M68kISD::SETCC_CARRY,
DL,
Op.getValueType(),
2401 if (
T1.getValueType() == T2.getValueType() &&
2415 const APInt &
C =
Const->getAPIntValue();
2416 if (
C.countr_zero() >= 5)
2418 else if (
C.countr_one() >= 5)
2431 Opc =
Op.getOpcode();
2435 Op.getOperand(0).hasOneUse() &&
2437 Op.getOperand(1).hasOneUse());
2446 return Op.getOperand(0).getOpcode() == M68kISD::SETCC &&
2447 Op.getOperand(0).hasOneUse();
2452 bool AddTest =
true;
2458 bool Inverted =
false;
2464 Cond.getOperand(0).getResNo() == 1 &&
2479 Cond.getOperand(0).getOpcode() == M68kISD::SETCC_CARRY &&
2485 unsigned CondOpcode =
Cond.getOpcode();
2486 if (CondOpcode == M68kISD::SETCC || CondOpcode == M68kISD::SETCC_CARRY) {
2490 unsigned Opc =
Cmp.getOpcode();
2503 Cond =
Cond.getNode()->getOperand(1);
2509 CondOpcode =
Cond.getOpcode();
2530 Chain = DAG.
getNode(M68kISD::BRCOND,
DL,
Op.getValueType(), Chain,
2543 Op.getNode()->hasOneUse()) {
2548 SDNode *
User = *
Op.getNode()->user_begin();
2560 Chain = DAG.
getNode(M68kISD::BRCOND,
DL,
Op.getValueType(), Chain,
2590 if (
Cond.hasOneUse()) {
2604 return DAG.
getNode(M68kISD::BRCOND,
DL,
Op.getValueType(), Chain, Dest, CC,
2610 MVT VT =
Op.getNode()->getSimpleValueType(0);
2616 SDVTList VTs = DAG.
getVTList(VT, MVT::i8);
2619 bool ExtraOp =
false;
2620 switch (
Op.getOpcode()) {
2627 Opc = M68kISD::ADDX;
2634 Opc = M68kISD::SUBX;
2640 return DAG.
getNode(
Opc, SDLoc(
Op), VTs,
Op.getOperand(0),
Op.getOperand(1));
2641 return DAG.
getNode(
Opc, SDLoc(
Op), VTs,
Op.getOperand(0),
Op.getOperand(1),
2657 unsigned char OpFlag = Subtarget.classifyLocalReference(
nullptr);
2659 unsigned WrapperKind = M68kISD::Wrapper;
2661 WrapperKind = M68kISD::WrapperPC;
2674 DAG.
getNode(M68kISD::GLOBAL_BASE_REG, SDLoc(), PtrVT),
2688 unsigned char OpFlag = Subtarget.classifyExternalReference(*
Mod);
2690 unsigned WrapperKind = M68kISD::Wrapper;
2692 WrapperKind = M68kISD::WrapperPC;
2704 DAG.
getNode(M68kISD::GLOBAL_BASE_REG, SDLoc(), PtrVT),
2720 unsigned char OpFlags = Subtarget.classifyBlockAddressReference();
2739 DAG.
getNode(M68kISD::GLOBAL_BASE_REG,
DL, PtrVT), Result);
2748 unsigned char OpFlags = Subtarget.classifyGlobalReference(GV);
2770 DAG.
getNode(M68kISD::GLOBAL_BASE_REG,
DL, PtrVT), Result);
2794 return LowerGlobalAddress(GV, SDLoc(
Op),
Offset, DAG);
2807 unsigned char OpFlag = Subtarget.classifyLocalReference(
nullptr);
2809 unsigned WrapperKind = M68kISD::Wrapper;
2811 WrapperKind = M68kISD::WrapperPC;
2822 DAG.
getNode(M68kISD::GLOBAL_BASE_REG, SDLoc(), PtrVT),
2830 return Subtarget.getJumpTableEncoding();
2857 if (Constraint.
size() > 0) {
2858 switch (Constraint[0]) {
2872 if (Constraint.
size() == 2)
2873 switch (Constraint[1]) {
2895 std::vector<SDValue> &
Ops,
2899 if (Constraint.
size() == 1) {
2901 switch (Constraint[0]) {
2914 int64_t Val =
C->getSExtValue();
2915 switch (Constraint[0]) {
2917 if (Val > 0 && Val <= 8)
2925 if (Val < -0x80 || Val >= 0x80)
2929 if (Val < 0 && Val >= -8)
2933 if (Val < -0x100 || Val >= 0x100)
2937 if (Val >= 24 && Val <= 31)
2945 if (Val >= 8 && Val <= 15)
2960 if (Constraint.
size() == 2) {
2961 switch (Constraint[0]) {
2964 switch (Constraint[1]) {
2972 int64_t Val =
C->getSExtValue();
2973 switch (Constraint[1]) {
3000 if (Result.getNode()) {
3001 Ops.push_back(Result);
3008std::pair<unsigned, const TargetRegisterClass *>
3012 if (Constraint.
size() == 1) {
3013 switch (Constraint[0]) {
3018 return std::make_pair(0U, &M68k::DR8RegClass);
3020 return std::make_pair(0U, &M68k::DR16RegClass);
3022 return std::make_pair(0U, &M68k::DR32RegClass);
3030 return std::make_pair(0U, &M68k::AR16RegClass);
3032 return std::make_pair(0U, &M68k::AR32RegClass);
3055 switch (
MI.getOpcode()) {
3086 if (miI == BB->
end())
3088 if (SBB->isLiveIn(M68k::CCR))
3093 SelectItr->addRegisterKilled(M68k::CCR,
TRI);
3100 const TargetInstrInfo *
TII = Subtarget.getInstrInfo();
3116 MachineBasicBlock *ThisMBB =
MBB;
3155 MachineInstr *CascadedCMOV =
nullptr;
3156 MachineInstr *LastCMOV = &
MI;
3169 (NextMIIt->getOperand(3).getImm() == CC ||
3170 NextMIIt->getOperand(3).getImm() == OppCC)) {
3171 LastCMOV = &*NextMIIt;
3178 if (LastCMOV == &
MI && NextMIIt !=
MBB->
end() &&
3179 NextMIIt->getOpcode() ==
MI.getOpcode() &&
3180 NextMIIt->getOperand(2).getReg() ==
MI.getOperand(2).getReg() &&
3181 NextMIIt->getOperand(1).getReg() ==
MI.getOperand(0).getReg() &&
3182 NextMIIt->getOperand(1).isKill()) {
3183 CascadedCMOV = &*NextMIIt;
3186 MachineBasicBlock *Jcc1MBB =
nullptr;
3191 Jcc1MBB =
F->CreateMachineBasicBlock(BB);
3196 MachineBasicBlock *Copy0MBB =
F->CreateMachineBasicBlock(BB);
3197 MachineBasicBlock *SinkMBB =
F->CreateMachineBasicBlock(BB);
3198 F->insert(It, Copy0MBB);
3199 F->insert(It, SinkMBB);
3202 unsigned CallFrameSize =
TII->getCallFrameSizeAt(
MI);
3208 const TargetRegisterInfo *
TRI = Subtarget.getRegisterInfo();
3210 MachineInstr *LastCCRSUser = CascadedCMOV ? CascadedCMOV : LastCMOV;
3260 DenseMap<unsigned, std::pair<unsigned, unsigned>> RegRewriteTable;
3261 MachineInstrBuilder MIB;
3271 Register DestReg = MIIt->getOperand(0).getReg();
3272 Register Op1Reg = MIIt->getOperand(1).getReg();
3273 Register Op2Reg = MIIt->getOperand(2).getReg();
3278 if (MIIt->getOperand(3).getImm() == OppCC)
3281 if (RegRewriteTable.
find(Op1Reg) != RegRewriteTable.
end())
3282 Op1Reg = RegRewriteTable[Op1Reg].first;
3284 if (RegRewriteTable.
find(Op2Reg) != RegRewriteTable.
end())
3285 Op2Reg = RegRewriteTable[Op2Reg].second;
3288 BuildMI(*SinkMBB, SinkInsertionPoint,
DL,
TII->get(M68k::PHI), DestReg)
3295 RegRewriteTable[DestReg] = std::make_pair(Op1Reg, Op2Reg);
3304 DL,
TII->get(TargetOpcode::COPY),
3306 .
addReg(
MI.getOperand(0).getReg());
3312 (MIIt++)->eraseFromParent();
3320 llvm_unreachable(
"Cannot lower Segmented Stack Alloca with stack-split on");
3326 switch (
MI.getOpcode()) {
3332 return EmitLoweredSelect(
MI, BB);
3334 return EmitLoweredSegAlloca(
MI, BB);
3361 const SDValue AsmOperands[4] = {
3372 DAG.
getVTList(MVT::Other, MVT::Glue), AsmOperands);
3388 SDNode *
Node =
Op.getNode();
3391 unsigned Align =
Op.getConstantOperandVal(2);
3392 EVT VT =
Node->getValueType(0);
3410 assert(
SPReg &&
"Target cannot require DYNAMIC_STACKALLOC expansion and"
3411 " not tell us which reg is the stack pointer!");
3414 Chain =
SP.getValue(1);
3415 const TargetFrameLowering &TFI = *Subtarget.getFrameLowering();
3418 if (Align > StackAlign)
3436 EVT VT =
Lo.getValueType();
3449 SDValue ShamtMinusRegisterSize =
3451 SDValue RegisterSizeMinus1Shamt =
3477 EVT VT =
Lo.getValueType();
3501 SDValue ShamtMinusRegisterSize =
3503 SDValue RegisterSizeMinus1Shamt =
3513 DAG.
getNode(ShiftRightOp,
DL, VT,
Hi, ShamtMinusRegisterSize);
3532 return DAG.
getNode(M68kISD::SETCC, dl, MVT::i8,
3549 if (Carry.
getOpcode() == M68kISD::SETCC ||
3550 Carry.
getOpcode() == M68kISD::SETCC_CARRY) {
3597 return DAG.
getNode(M68kISD::BRCOND,
DL,
N->getVTList(),
N->getOperand(0),
3598 N->getOperand(1),
Cond, Flags);
3606 MVT VT =
N->getSimpleValueType(0);
3608 return DAG.
getNode(M68kISD::SUBX,
SDLoc(
N), VTs,
N->getOperand(0),
3609 N->getOperand(1), Flags);
3619 MVT VT =
N->getSimpleValueType(0);
3621 return DAG.
getNode(M68kISD::ADDX,
SDLoc(
N), VTs,
N->getOperand(0),
3622 N->getOperand(1), Flags);
3629 DAGCombinerInfo &DCI)
const {
3630 SelectionDAG &DAG = DCI.DAG;
3631 switch (
N->getOpcode()) {
3636 case M68kISD::SETCC:
3638 case M68kISD::BRCOND:
3646 bool IsVarArg)
const {
3648 return RetCC_M68k_C;
static SDValue getSETCC(AArch64CC::CondCode CC, SDValue NZCV, const SDLoc &DL, SelectionDAG &DAG)
Helper function to create 'CSET', which is equivalent to 'CSINC <Wd>, WZR, WZR, invert(<cond>)'.
static bool canGuaranteeTCO(CallingConv::ID CC, bool GuaranteeTailCalls)
Return true if the calling convention is one that we can guarantee TCO for.
static bool mayTailCallThisCC(CallingConv::ID CC)
Return true if we might ever do TCO for calls with this calling convention.
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
const HexagonInstrInfo * TII
static SDValue CreateCopyOfByValArgument(SDValue Src, SDValue Dst, SDValue Chain, ISD::ArgFlagsTy Flags, SelectionDAG &DAG, const SDLoc &dl)
CreateCopyOfByValArgument - Make a copy of an aggregate at address specified by "Src" to address "Dst...
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
This file contains the custom routines for the M68k Calling Convention that aren't done by tablegen.
static SDValue LowerTruncateToBTST(SDValue Op, ISD::CondCode CC, const SDLoc &DL, SelectionDAG &DAG)
static void lowerOverflowArithmetic(SDValue Op, SelectionDAG &DAG, SDValue &Result, SDValue &CCR, unsigned &CC)
static SDValue combineADDX(SDNode *N, SelectionDAG &DAG, TargetLowering::DAGCombinerInfo &DCI)
static bool isAndOrOfSetCCs(SDValue Op, unsigned &Opc)
Return true if node is an ISD::AND or ISD::OR of two M68k::SETcc nodes each of which has no other use...
static bool hasNonFlagsUse(SDValue Op)
return true if Op has a use that doesn't just read flags.
static bool isM68kCCUnsigned(unsigned M68kCC)
Return true if the condition is an unsigned comparison operation.
static StructReturnType callIsStructReturn(const SmallVectorImpl< ISD::OutputArg > &Outs)
static bool isXor1OfSetCC(SDValue Op)
Return true if node is an ISD::XOR of a M68kISD::SETCC and 1 and that the SETCC node has a single use...
static SDValue LowerAndToBTST(SDValue And, ISD::CondCode CC, const SDLoc &DL, SelectionDAG &DAG)
Result of 'and' is compared against zero. Change to a BTST node if possible.
static SDValue combineM68kBrCond(SDNode *N, SelectionDAG &DAG, const M68kSubtarget &Subtarget)
static M68k::CondCode TranslateIntegerM68kCC(ISD::CondCode SetCCOpcode)
static StructReturnType argsAreStructReturn(const SmallVectorImpl< ISD::InputArg > &Ins)
Determines whether a function uses struct return semantics.
static bool isCMOVPseudo(MachineInstr &MI)
static bool shouldGuaranteeTCO(CallingConv::ID CC, bool GuaranteedTailCallOpt)
Return true if the function is being made into a tailcall target by changing its ABI.
static bool isM68kLogicalCmp(SDValue Op)
Return true if opcode is a M68k logical comparison.
static SDValue combineM68kSetCC(SDNode *N, SelectionDAG &DAG, const M68kSubtarget &Subtarget)
static SDValue combineSetCCCCR(SDValue CCR, M68k::CondCode &CC, SelectionDAG &DAG, const M68kSubtarget &Subtarget)
Optimize a CCR definition used according to the condition code CC into a simpler CCR value,...
static SDValue combineCarryThroughADD(SDValue CCR)
static bool isOverflowArithmetic(unsigned Opcode)
static bool MatchingStackOffset(SDValue Arg, unsigned Offset, ISD::ArgFlagsTy Flags, MachineFrameInfo &MFI, const MachineRegisterInfo *MRI, const M68kInstrInfo *TII, const CCValAssign &VA)
Return true if the given stack call argument is already available in the same position (relatively) o...
static SDValue getBitTestCondition(SDValue Src, SDValue BitNo, ISD::CondCode CC, const SDLoc &DL, SelectionDAG &DAG)
Create a BTST (Bit Test) node - Test bit BitNo in Src and set condition according to equal/not-equal ...
static bool isTruncWithZeroHighBitsInput(SDValue V, SelectionDAG &DAG)
static bool checkAndUpdateCCRKill(MachineBasicBlock::iterator SelectItr, MachineBasicBlock *BB, const TargetRegisterInfo *TRI)
static SDValue combineSUBX(SDNode *N, SelectionDAG &DAG)
static unsigned TranslateM68kCC(ISD::CondCode SetCCOpcode, const SDLoc &DL, bool IsFP, SDValue &LHS, SDValue &RHS, SelectionDAG &DAG)
Do a one-to-one translation of a ISD::CondCode to the M68k-specific condition code,...
This file defines the interfaces that M68k uses to lower LLVM code into a selection DAG.
This file contains the declarations of the M68k MCAsmInfo properties.
This file declares the M68k specific subclass of MachineFunctionInfo.
This file declares the M68k specific subclass of TargetSubtargetInfo.
This file declares the M68k specific subclass of TargetMachine.
This file contains declarations for M68k ELF object file lowering.
Machine Check Debug Module
Register const TargetRegisterInfo * TRI
Promote Memory to Register
static constexpr MCPhysReg SPReg
const SmallVectorImpl< MachineOperand > & Cond
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static APInt getAllOnes(unsigned numBits)
Return an APInt of a specified width with all bits set.
static APInt getHighBitsSet(unsigned numBits, unsigned hiBitsSet)
Constructs an APInt value that has the top hiBitsSet bits set.
an instruction that atomically reads a memory location, combines it with another value,...
static LLVM_ABI bool resultsCompatible(CallingConv::ID CalleeCC, CallingConv::ID CallerCC, MachineFunction &MF, LLVMContext &C, const SmallVectorImpl< ISD::InputArg > &Ins, CCAssignFn CalleeFn, CCAssignFn CallerFn)
Returns true if the results of the two calling conventions are compatible.
CCValAssign - Represent assignment of one arg/retval to a location.
Register getLocReg() const
LocInfo getLocInfo() const
int64_t getLocMemOffset() const
unsigned getValNo() const
LLVM_ABI bool isMustTailCall() const
Tests if this call site must be tail call optimized.
const Constant * getConstVal() const
This is an important base class in LLVM.
A parsed version of the target data layout string in and methods for querying it.
iterator find(const_arg_type_t< KeyT > Val)
iterator_range< arg_iterator > args()
Attribute getFnAttribute(Attribute::AttrKind Kind) const
Return the attribute for the given attribute kind.
bool hasMinSize() const
Optimize this function for minimum size (-Oz).
bool hasStructRetAttr() const
Determine if the function returns a structure through first or second pointer argument.
int64_t getOffset() const
const GlobalValue * getGlobal() const
bool hasDLLImportStorageClass() const
Module * getParent()
Get the module that this global value is contained inside of...
This is an important class for using LLVM in a threaded context.
void setVarArgsFrameIndex(int Index)
void setSRetReturnReg(unsigned Reg)
SmallVectorImpl< ForwardedRegister > & getForwardedMustTailRegParms()
void setBytesToPopOnReturn(unsigned bytes)
unsigned getBytesToPopOnReturn() const
unsigned getSRetReturnReg() const
void setRAIndex(int Index)
int getVarArgsFrameIndex() const
void setArgumentStackSize(unsigned size)
void setTCReturnAddrDelta(int delta)
const uint32_t * getCallPreservedMask(const MachineFunction &MF, CallingConv::ID) const override
unsigned getStackRegister() const
const M68kRegisterInfo * getRegisterInfo() const override
ConstraintType getConstraintType(StringRef ConstraintStr) const override
Given a constraint, return the type of constraint it is for this target.
void LowerAsmOperandForConstraint(SDValue Op, StringRef Constraint, std::vector< SDValue > &Ops, SelectionDAG &DAG) const override
Lower the specified operand into the Ops vector.
MachineBasicBlock * EmitInstrWithCustomInserter(MachineInstr &MI, MachineBasicBlock *MBB) const override
This method should be implemented by targets that mark instructions with the 'usesCustomInserter' fla...
AtomicExpansionKind shouldExpandAtomicRMWInIR(const AtomicRMWInst *RMW) const override
Returns how the IR-level AtomicExpand pass should expand the given AtomicRMW, if at all.
virtual MVT getScalarShiftAmountTy(const DataLayout &, EVT) const override
EVT is not used in-tree, but is used by out-of-tree target.
const MCExpr * LowerCustomJumpTableEntry(const MachineJumpTableInfo *MJTI, const MachineBasicBlock *MBB, unsigned uid, MCContext &Ctx) const override
SDValue getPICJumpTableRelocBase(SDValue Table, SelectionDAG &DAG) const override
Returns relocation base for the given PIC jumptable.
const MCExpr * getPICJumpTableRelocBaseExpr(const MachineFunction *MF, unsigned JTI, MCContext &Ctx) const override
This returns the relocation base for the given PIC jumptable, the same as getPICJumpTableRelocBase,...
Register getExceptionPointerRegister(const Constant *PersonalityFn) const override
If a physical register, this returns the register that receives the exception address on entry to an ...
CCAssignFn * getCCAssignFn(CallingConv::ID CC, bool Return, bool IsVarArg) const
M68kTargetLowering(const M68kTargetMachine &TM, const M68kSubtarget &STI)
InlineAsm::ConstraintCode getInlineAsmMemConstraint(StringRef ConstraintCode) const override
SDValue LowerOperation(SDValue Op, SelectionDAG &DAG) const override
Provide custom lowering hooks for some operations.
EVT getSetCCResultType(const DataLayout &DL, LLVMContext &Context, EVT VT) const override
Return the value type to use for ISD::SETCC.
unsigned getJumpTableEncoding() const override
Return the entry encoding for a jump table in the current function.
std::pair< unsigned, const TargetRegisterClass * > getRegForInlineAsmConstraint(const TargetRegisterInfo *TRI, StringRef Constraint, MVT VT) const override
Given a physical register constraint (e.g.
Register getExceptionSelectorRegister(const Constant *PersonalityFn) const override
If a physical register, this returns the register that receives the exception typeid on entry to a la...
Context object for machine code objects.
Base class for the full range of assembler expressions which are needed for parsing.
static const MCSymbolRefExpr * create(const MCSymbol *Symbol, MCContext &Ctx, SMLoc Loc=SMLoc())
static MDTuple * get(LLVMContext &Context, ArrayRef< Metadata * > MDs)
static auto integer_valuetypes()
TypeSize getSizeInBits() const
Returns the size of the specified MVT in bits.
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
static MVT getIntegerVT(unsigned BitWidth)
MVT getScalarType() const
If this is a vector, return the element type, otherwise return this.
LLVM_ABI void transferSuccessorsAndUpdatePHIs(MachineBasicBlock *FromMBB)
Transfers all the successors, as in transferSuccessors, and update PHI operands in the successor bloc...
LLVM_ABI instr_iterator insert(instr_iterator I, MachineInstr *M)
Insert MI into the instruction list before I, possibly inside a bundle.
void setCallFrameSize(unsigned N)
Set the call frame size on entry to this basic block.
const BasicBlock * getBasicBlock() const
Return the LLVM basic block that this instance corresponded to originally.
LLVM_ABI void addSuccessor(MachineBasicBlock *Succ, BranchProbability Prob=BranchProbability::getUnknown())
Add Succ as a successor of this MachineBasicBlock.
void addLiveIn(MCRegister PhysReg, LaneBitmask LaneMask=LaneBitmask::getAll())
Adds the specified register as a live in.
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
iterator_range< succ_iterator > successors()
void splice(iterator Where, MachineBasicBlock *Other, iterator From)
Take an instruction from MBB 'Other' at the position From, and insert it into this MBB right before '...
MachineInstrBundleIterator< MachineInstr > iterator
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
LLVM_ABI int CreateFixedObject(uint64_t Size, int64_t SPOffset, bool IsImmutable, bool isAliased=false)
Create a new object at a fixed location on the stack.
void setObjectZExt(int ObjectIdx, bool IsZExt)
void setObjectSExt(int ObjectIdx, bool IsSExt)
void setHasTailCall(bool V=true)
bool isObjectZExt(int ObjectIdx) const
int64_t getObjectSize(int ObjectIdx) const
Return the size of the specified object.
bool isObjectSExt(int ObjectIdx) const
int64_t getObjectOffset(int ObjectIdx) const
Return the assigned stack offset of the specified object from the incoming stack pointer.
bool isFixedObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a fixed stack object.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MCSymbol * getJTISymbol(unsigned JTI, MCContext &Ctx, bool isLinkerPrivate=false) const
getJTISymbol - Return the MCSymbol for the specified non-empty jump table.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
const DataLayout & getDataLayout() const
Return the DataLayout attached to the Module associated to this MF.
Function & getFunction()
Return the LLVM function that this machine code represents.
BasicBlockListType::iterator iterator
bool shouldSplitStack() const
Should we be emitting segmented stack stuff for the function.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
Register addLiveIn(MCRegister PReg, const TargetRegisterClass *RC)
addLiveIn - Add the specified physical register as a live-in value and create a corresponding virtual...
const TargetMachine & getTarget() const
getTarget - Return the target machine this machine code is compiled with
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & addMBB(MachineBasicBlock *MBB, unsigned TargetFlags=0) const
MachineInstr * getInstr() const
If conversion operators fail, use this method to get the MachineInstr explicitly.
Representation of each machine instruction.
bool readsRegister(Register Reg, const TargetRegisterInfo *TRI) const
Return true if the MachineInstr reads the specified register.
bool killsRegister(Register Reg, const TargetRegisterInfo *TRI) const
Return true if the MachineInstr kills the specified register.
bool definesRegister(Register Reg, const TargetRegisterInfo *TRI) const
Return true if the MachineInstr fully defines the specified register.
const MachineOperand & getOperand(unsigned i) const
LLVM_ABI MachineInstrBundleIterator< MachineInstr > eraseFromParent()
Unlink 'this' from the containing basic block and delete it.
@ EK_Custom32
EK_Custom32 - Each entry is a 32-bit value that is custom lowered by the TargetLowering::LowerCustomJ...
Register getReg() const
getReg - Returns the register number.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
LLVM_ABI MachineInstr * getVRegDef(Register Reg) const
getVRegDef - Return the machine instr that defines the specified virtual register or null if none is ...
LLVM_ABI Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
Class to represent pointers.
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.
Wrapper class representing virtual and physical registers.
static constexpr bool isVirtualRegister(unsigned Reg)
Return true if the specified register number is in the virtual register namespace.
Wrapper class for IR location info (IR ordering and DebugLoc) to be passed into SDNode creation funct...
This class provides iterator support for SDUse operands that use a specific SDNode.
Represents one node in the SelectionDAG.
bool hasOneUse() const
Return true if there is exactly one use of this node.
uint64_t getAsZExtVal() const
Helper method returns the zero-extended integer value of a ConstantSDNode.
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
bool hasOneUse() const
Return true if there is exactly one node using value ResNo of Node.
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
uint64_t getConstantOperandVal(unsigned i) const
MVT getSimpleValueType() const
Return the simple ValueType of the referenced return value.
unsigned getOpcode() const
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)
LLVM_ABI SDValue getStackArgumentTokenFactor(SDValue Chain)
Compute a TokenFactor to force all the incoming stack arguments to be loaded from the stack.
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 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,...
SDValue getGLOBAL_OFFSET_TABLE(EVT VT)
Return a GLOBAL_OFFSET_TABLE node. This does not have a useful SDLoc.
SDValue getSetCC(const SDLoc &DL, EVT VT, SDValue LHS, SDValue RHS, ISD::CondCode Cond, SDValue Chain=SDValue(), bool IsSignaling=false, SDNodeFlags Flags={})
Helper function to make it easier to build SetCC's if you just have an ISD::CondCode instead of an SD...
LLVM_ABI SDValue getNOT(const SDLoc &DL, SDValue Val, EVT VT)
Create a bitwise NOT operation as (XOR Val, -1).
LLVM_ABI SDValue getMemcpy(SDValue Chain, const SDLoc &dl, SDValue Dst, SDValue Src, SDValue Size, Align DstAlign, Align SrcAlign, bool isVol, bool AlwaysInline, const CallInst *CI, std::optional< bool > OverrideTailCall, MachinePointerInfo DstPtrInfo, MachinePointerInfo SrcPtrInfo, const AAMDNodes &AAInfo=AAMDNodes(), BatchAAResults *BatchAA=nullptr)
const TargetLowering & getTargetLoweringInfo() const
SDValue getTargetJumpTable(int JTI, EVT VT, unsigned TargetFlags=0)
SDValue getCALLSEQ_END(SDValue Chain, SDValue Op1, SDValue Op2, SDValue InGlue, const SDLoc &DL)
Return a new CALLSEQ_END node, which always must have a glue result (to ensure it's not CSE'd).
LLVM_ABI SDValue getBitcast(EVT VT, SDValue V)
Return a bitcast using the SDLoc of the value operand, and casting to the provided type.
SDValue getCopyFromReg(SDValue Chain, const SDLoc &dl, Register Reg, EVT VT)
const DataLayout & getDataLayout() const
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.
SDValue getSignedTargetConstant(int64_t Val, const SDLoc &DL, EVT VT, bool isOpaque=false)
LLVM_ABI SDValue getMDNode(const MDNode *MD)
Return an MDNodeSDNode which holds an MDNode.
LLVM_ABI void ReplaceAllUsesWith(SDValue From, SDValue To)
Modify anything using 'From' to use 'To' instead.
LLVM_ABI SDValue getStore(SDValue Chain, const SDLoc &dl, SDValue Val, SDValue Ptr, MachinePointerInfo PtrInfo, Align Alignment, MachineMemOperand::Flags MMOFlags=MachineMemOperand::MONone, const AAMDNodes &AAInfo=AAMDNodes())
Helper function to build ISD::STORE nodes.
LLVM_ABI SDValue getSignedConstant(int64_t Val, const SDLoc &DL, EVT VT, bool isTarget=false, bool isOpaque=false)
SDValue getCALLSEQ_START(SDValue Chain, uint64_t InSize, uint64_t OutSize, const SDLoc &DL)
Return a new CALLSEQ_START node, that starts new call frame, in which InSize bytes are set up inside ...
LLVM_ABI SDValue getExternalSymbol(const char *Sym, EVT VT)
const TargetMachine & getTarget() 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.
SDValue getTargetConstant(uint64_t Val, const SDLoc &DL, EVT VT, bool isOpaque=false)
SDValue getTargetBlockAddress(const BlockAddress *BA, EVT VT, int64_t Offset=0, unsigned TargetFlags=0)
MachineFunction & getMachineFunction() const
LLVM_ABI SDValue getFrameIndex(int FI, EVT VT, bool isTarget=false)
LLVM_ABI KnownBits computeKnownBits(SDValue Op, unsigned Depth=0) const
Determine which bits of Op are known to be either zero or one and return them in Known.
LLVM_ABI SDValue getRegisterMask(const uint32_t *RegMask)
LLVM_ABI bool MaskedValueIsZero(SDValue Op, const APInt &Mask, unsigned Depth=0) const
Return true if 'Op & Mask' is known to be zero.
LLVMContext * getContext() const
LLVM_ABI SDValue getTargetExternalSymbol(const char *Sym, EVT VT, unsigned TargetFlags=0)
LLVM_ABI SDValue CreateStackTemporary(TypeSize Bytes, Align Alignment)
Create a stack temporary based on the size in bytes and the alignment.
LLVM_ABI SDNode * UpdateNodeOperands(SDNode *N, SDValue Op)
Mutate the specified node in-place to have the specified operands.
SDValue getTargetConstantPool(const Constant *C, EVT VT, MaybeAlign Align=std::nullopt, int Offset=0, unsigned TargetFlags=0)
SDValue getEntryNode() const
Return the token chain corresponding to the entry of the function.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void push_back(const T &Elt)
Represent a constant reference to a string, i.e.
constexpr size_t size() const
Get the string size.
A switch()-like statement whose cases are string literals.
StringSwitch & Case(StringLiteral S, T Value)
unsigned getStackAlignment() const
getStackAlignment - This method returns the number of bytes to which the stack pointer must be aligne...
void setOperationAction(unsigned Op, MVT VT, LegalizeAction Action)
Indicate that the specified operation does not work with the specified type and indicate what to do a...
virtual const TargetRegisterClass * getRegClassFor(MVT VT, bool isDivergent=false) const
Return the register class that should be used for the specified value type.
const TargetMachine & getTargetMachine() const
void setMaxAtomicSizeInBitsSupported(unsigned SizeInBits)
Set the maximum atomic operation size supported by the backend.
Register getStackPointerRegisterToSaveRestore() const
If a physical register, this specifies the register that llvm.savestack/llvm.restorestack should save...
void setMinFunctionAlignment(Align Alignment)
Set the target's minimum function alignment.
void setBooleanContents(BooleanContent Ty)
Specify how the target extends the result of integer and floating point boolean values from i1 to a w...
void computeRegisterProperties(const TargetRegisterInfo *TRI)
Once all of the register classes are added, this allows us to compute derived properties we expose.
void addRegisterClass(MVT VT, const TargetRegisterClass *RC)
Add the specified register class as an available regclass for the specified value type.
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.
void setTruncStoreAction(MVT ValVT, MVT MemVT, LegalizeAction Action)
Indicate that the specified truncating store does not work with the specified type and indicate what ...
@ ZeroOrNegativeOneBooleanContent
void setStackPointerRegisterToSaveRestore(Register R)
If set to a physical register, this specifies the register that llvm.savestack/llvm....
AtomicExpansionKind
Enum that specifies what an atomic load/AtomicRMWInst is expanded to, if at all.
void setLoadExtAction(unsigned ExtType, MVT ValVT, MVT MemVT, LegalizeAction Action)
Indicate that the specified load with extension does not work with the specified type and indicate wh...
std::vector< ArgListEntry > ArgListTy
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 ...
This class defines information used to lower LLVM code to legal SelectionDAG operators that the targe...
virtual InlineAsm::ConstraintCode getInlineAsmMemConstraint(StringRef ConstraintCode) const
virtual ConstraintType getConstraintType(StringRef Constraint) const
Given a constraint, return the type of constraint it is for this target.
bool parametersInCSRMatch(const MachineRegisterInfo &MRI, const uint32_t *CallerPreservedMask, const SmallVectorImpl< CCValAssign > &ArgLocs, const SmallVectorImpl< SDValue > &OutVals) const
Check whether parameters to a call that are passed in callee saved registers are the same as from the...
std::pair< SDValue, SDValue > LowerCallTo(CallLoweringInfo &CLI) const
This function lowers an abstract call to a function into an actual call.
bool isPositionIndependent() const
virtual std::pair< unsigned, const TargetRegisterClass * > getRegForInlineAsmConstraint(const TargetRegisterInfo *TRI, StringRef Constraint, MVT VT) const
Given a physical register constraint (e.g.
TargetLowering(const TargetLowering &)=delete
virtual void LowerAsmOperandForConstraint(SDValue Op, StringRef Constraint, std::vector< SDValue > &Ops, SelectionDAG &DAG) const
Lower the specified operand into the Ops vector.
TLSModel::Model getTLSModel(const GlobalValue *GV) const
Returns the TLS model which should be used for the given global variable.
unsigned GuaranteedTailCallOpt
GuaranteedTailCallOpt - This flag is enabled when -tailcallopt is specified on the commandline.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
The instances of the Type class are immutable: once they are created, they are never changed.
LLVM_ABI unsigned getOperandNo() const
Return the operand # of this use in its User.
User * getUser() const
Returns the User that contains this Use.
LLVM Value Representation.
bool hasOneUse() const
Return true if there is exactly one use of this value.
self_iterator getIterator()
#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 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.
@ M68k_INTR
Used for M68k interrupt routines.
@ Swift
Calling convention for Swift.
@ M68k_RTD
Used for M68k rtd-based CC (similar to X86's stdcall).
@ C
The default llvm calling convention, compatible with C.
LLVM_ABI CondCode getSetCCInverse(CondCode Operation, bool isIntegerLike)
Return the operation corresponding to !(X op Y), where 'op' is a valid SetCC operation.
bool isNON_EXTLoad(const SDNode *N)
Returns true if the specified node is a non-extending load.
@ SETCC
SetCC operator - This evaluates to a true value iff the condition is true.
@ MERGE_VALUES
MERGE_VALUES - This node takes multiple discrete operands and returns them all as its individual resu...
@ 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.
@ SMUL_LOHI
SMUL_LOHI/UMUL_LOHI - Multiply two integers of type iN, producing a signed/unsigned value of type i[2...
@ BSWAP
Byte Swap and Counting operators.
@ VAEND
VAEND, VASTART - VAEND and VASTART have three operands: an input chain, pointer, and a SRCVALUE.
@ ADDC
Carry-setting nodes for multiple precision addition and subtraction.
@ ADD
Simple integer binary arithmetic operators.
@ ANY_EXTEND
ANY_EXTEND - Used for integer types. The high bits are undefined.
@ ATOMIC_FENCE
OUTCHAIN = ATOMIC_FENCE(INCHAIN, ordering, scope) This corresponds to the fence instruction.
@ SDIVREM
SDIVREM/UDIVREM - Divide two integers and produce both a quotient and remainder result.
@ BITCAST
BITCAST - This operator converts between integer, vector and FP values, as if the value was stored to...
@ SIGN_EXTEND
Conversion operators.
@ BR
Control flow instructions. These all have token chains.
@ SETCCCARRY
Like SetCC, ops #0 and #1 are the LHS and RHS operands to compare, but op #2 is a boolean indicating ...
@ BR_CC
BR_CC - Conditional branch.
@ SSUBO
Same for subtraction.
@ BR_JT
BR_JT - Jumptable branch.
@ SELECT
Select(COND, TRUEVAL, FALSEVAL).
@ VACOPY
VACOPY - VACOPY has 5 operands: an input chain, a destination pointer, a source pointer,...
@ 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.
@ MULHU
MULHU/MULHS - Multiply high - Multiply two integers of type iN, producing an unsigned/signed value of...
@ SHL
Shift and rotation operations.
@ ZERO_EXTEND
ZERO_EXTEND - Used for integer types, zeroing the new bits.
@ SELECT_CC
Select with condition operator - This selects between a true value and a false value (ops #2 and #3) ...
@ ATOMIC_CMP_SWAP
Val, OUTCHAIN = ATOMIC_CMP_SWAP(INCHAIN, ptr, cmp, swap) For double-word atomic operations: ValLo,...
@ SMULO
Same for multiplication.
@ DYNAMIC_STACKALLOC
DYNAMIC_STACKALLOC - Allocate some number of bytes on the stack aligned to a specified boundary.
@ AND
Bitwise operators - logical and, logical or, logical xor.
@ ADDE
Carry-using nodes for multiple precision addition and subtraction.
@ 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,...
@ INLINEASM
INLINEASM - Represents an inline asm block.
@ TRUNCATE
TRUNCATE - Completely drop the high bits.
@ VAARG
VAARG - VAARG has four operands: an input chain, a pointer, a SRCVALUE, and the alignment.
@ BRCOND
BRCOND - Conditional branch.
@ SHL_PARTS
SHL_PARTS/SRA_PARTS/SRL_PARTS - These operators are used for expanded integer shift operations.
@ AssertSext
AssertSext, AssertZext - These nodes record if a register contains a value that has already been zero...
CondCode
ISD::CondCode enum - These are ordered carefully to make the bitfields below work out,...
static bool isPCRelBlockReference(unsigned char Flag)
Return True if the Block is referenced using PC.
static bool isGlobalRelativeToPICBase(unsigned char TargetFlag)
Return true if the specified global value reference is relative to a 32-bit PIC base (M68kISD::GLOBAL...
static bool isGlobalStubReference(unsigned char TargetFlag)
Return true if the specified TargetFlag operand is a reference to a stub for a global,...
static bool isPCRelGlobalReference(unsigned char Flag)
Return True if the specified GlobalValue requires PC addressing mode.
@ MO_TLSLDM
On a symbol operand, this indicates that the immediate is the offset to the slot in GOT which stores ...
@ MO_TLSLE
On a symbol operand, this indicates that the immediate is the offset to the variable within in the th...
@ MO_TLSGD
On a symbol operand, this indicates that the immediate is the offset to the slot in GOT which stores ...
@ MO_GOTPCREL
On a symbol operand this indicates that the immediate is offset to the GOT entry for the symbol name ...
@ MO_TLSIE
On a symbol operand, this indicates that the immediate is the offset to the variable within the threa...
@ MO_TLSLD
On a symbol operand, this indicates that the immediate is the offset to variable within the thread lo...
static bool isDirectGlobalReference(unsigned char Flag)
Return True if the specified GlobalValue is a direct reference for a symbol.
static bool IsSETCC(unsigned SETCC)
static unsigned GetCondBranchFromCond(M68k::CondCode CC)
bool isCalleePop(CallingConv::ID CallingConv, bool IsVarArg, bool GuaranteeTCO)
Determines whether the callee is required to pop its own arguments.
static M68k::CondCode GetOppositeBranchCondition(M68k::CondCode CC)
@ User
could "use" a pointer
NodeAddr< NodeBase * > Node
This is an optimization pass for GlobalISel generic memory operations.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
constexpr bool isInt(int64_t x)
Checks if an integer fits into the given bit width.
LLVM_ABI bool isNullConstant(SDValue V)
Returns true if V is a constant integer zero.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
unsigned Log2_64_Ceil(uint64_t Value)
Return the ceil log base 2 of the specified value, 64 if the value is zero.
bool CCAssignFn(unsigned ValNo, MVT ValVT, MVT LocVT, CCValAssign::LocInfo LocInfo, ISD::ArgFlagsTy ArgFlags, Type *OrigTy, CCState &State)
CCAssignFn - This function assigns a location for Val, updating State to reflect the change.
constexpr bool isPowerOf2_64(uint64_t Value)
Return true if the argument is a power of two > 0 (64 bit edition.)
LLVM_ABI bool isBitwiseNot(SDValue V, bool AllowUndefs=false)
Returns true if V is a bitwise not operation.
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
constexpr bool isUInt(uint64_t x)
Checks if an unsigned integer fits into the given bit width.
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...
@ Mod
The access may modify the value stored in memory.
@ Xor
Bitwise or logical XOR of integers.
@ Sub
Subtraction of integers.
DWARFExpression::Operation Op
constexpr unsigned BitWidth
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI bool isOneConstant(SDValue V)
Returns true if V is a constant integer one.
LLVM_ABI bool isAllOnesConstant(SDValue V)
Returns true if V is an integer constant with all bits set.
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.
TypeSize getSizeInBits() const
Return the size of the specified value type in bits.
uint64_t getScalarSizeInBits() const
bool isVectorOf(EVT EltVT) const
Return true if this is a vector with matching element type.
bool bitsLE(EVT VT) const
Return true if this has no more bits than VT.
This class contains a discriminated union of information about pointers in memory operands,...
static LLVM_ABI MachinePointerInfo getStack(MachineFunction &MF, int64_t Offset, uint8_t ID=0)
Stack pointer relative access.
static LLVM_ABI MachinePointerInfo getGOT(MachineFunction &MF)
Return a MachinePointerInfo record that refers to a GOT entry.
static LLVM_ABI MachinePointerInfo getFixedStack(MachineFunction &MF, int FI, int64_t Offset=0)
Return a MachinePointerInfo record that refers to the specified FrameIndex.
This represents a list of ValueType's that has been intern'd by a SelectionDAG.
This structure contains all information that is necessary for lowering calls.
CallLoweringInfo & setLibCallee(CallingConv::ID CC, Type *ResultType, SDValue Target, ArgListTy &&ArgsList)
SmallVector< ISD::InputArg, 32 > Ins
CallLoweringInfo & setDebugLoc(const SDLoc &dl)
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)