35#include "llvm/IR/IntrinsicsWebAssembly.h"
42#define DEBUG_TYPE "wasm-lower"
47 auto MVTPtr = Subtarget->hasAddr64() ? MVT::i64 : MVT::i32;
61 Subtarget->hasAddr64() ? WebAssembly::SP64 : WebAssembly::SP32);
67 if (Subtarget->hasSIMD128()) {
75 if (Subtarget->hasFP16()) {
78 if (Subtarget->hasReferenceTypes()) {
81 if (Subtarget->hasExceptionHandling()) {
90 for (
auto T : {MVT::i32, MVT::i64, MVT::f32, MVT::f64}) {
94 if (Subtarget->hasSIMD128()) {
95 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v4f32, MVT::v2i64,
101 if (Subtarget->hasFP16()) {
105 if (Subtarget->hasReferenceTypes()) {
108 for (
auto T : {MVT::externref, MVT::funcref, MVT::Other}) {
129 for (
auto T : {MVT::f32, MVT::f64, MVT::v4f32, MVT::v2f64, MVT::v8f16}) {
130 if (!Subtarget->hasFP16() &&
T == MVT::v8f16) {
143 if (
MVT(
T).isVector())
157 if (
T != MVT::v8f16) {
161 if (Subtarget->hasFP16() &&
T == MVT::f32) {
175 for (
auto T : {MVT::i32, MVT::i64})
177 if (Subtarget->hasSIMD128())
178 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v2i64})
182 if (Subtarget->hasWideArithmetic()) {
190 if (Subtarget->hasNontrappingFPToInt())
192 for (
auto T : {MVT::i32, MVT::i64})
195 if (Subtarget->hasRelaxedSIMD()) {
198 {MVT::v4f32, MVT::v2f64},
Custom);
201 if (Subtarget->hasSIMD128()) {
236 for (
auto T : {MVT::v16i8, MVT::v8i16})
240 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v2i64})
244 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v4f32, MVT::v2i64,
248 if (Subtarget->hasFP16()) {
254 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v4f32, MVT::v2i64,
258 if (Subtarget->hasFP16())
262 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v4f32, MVT::v2i64,
270 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v2i64})
275 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v4f32, MVT::v2i64,
285 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v2i64})
290 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32})
300 for (
auto T : {MVT::v8i16, MVT::v4i32, MVT::v2i64})
306 for (
auto T : {MVT::v4f32, MVT::v2f64})
316 for (
auto T : {MVT::v2i64, MVT::v2f64})
322 if (Subtarget->hasFP16()) {
334 if (Subtarget->hasFP16()) {
338 if (Subtarget->hasRelaxedSIMD()) {
353 if (!Subtarget->hasSignExt()) {
355 auto Action = Subtarget->hasSIMD128() ?
Custom :
Expand;
356 for (
auto T : {MVT::i8, MVT::i16, MVT::i32})
372 for (
auto T : {MVT::i32, MVT::i64, MVT::f32, MVT::f64})
376 if (Subtarget->hasReferenceTypes())
378 for (
auto T : {MVT::externref, MVT::funcref})
383 {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v4f32, MVT::v2i64, MVT::v2f64})
399 if (Subtarget->hasSIMD128()) {
400 for (
auto T : {MVT::v16i8, MVT::v8i16, MVT::v4i32, MVT::v2i64, MVT::v4f32,
403 if (
MVT(
T) != MemT) {
442 return MVT::externref;
451 return MVT::externref;
458WebAssemblyTargetLowering::shouldExpandAtomicRMWInIR(
475bool WebAssemblyTargetLowering::shouldScalarizeBinop(
SDValue VecOp)
const {
495FastISel *WebAssemblyTargetLowering::createFastISel(
501MVT WebAssemblyTargetLowering::getScalarShiftAmountTy(
const DataLayout & ,
512 "32-bit shift counts ought to be enough for anyone");
517 "Unable to represent scalar shift amount type");
527 bool IsUnsigned,
bool Int64,
528 bool Float64,
unsigned LoweredOpcode) {
534 unsigned Abs = Float64 ? WebAssembly::ABS_F64 : WebAssembly::ABS_F32;
535 unsigned FConst = Float64 ? WebAssembly::CONST_F64 : WebAssembly::CONST_F32;
536 unsigned LT = Float64 ? WebAssembly::LT_F64 : WebAssembly::LT_F32;
537 unsigned GE = Float64 ? WebAssembly::GE_F64 : WebAssembly::GE_F32;
538 unsigned IConst = Int64 ? WebAssembly::CONST_I64 : WebAssembly::CONST_I32;
539 unsigned Eqz = WebAssembly::EQZ_I32;
540 unsigned And = WebAssembly::AND_I32;
541 int64_t Limit = Int64 ?
INT64_MIN : INT32_MIN;
542 int64_t Substitute = IsUnsigned ? 0 : Limit;
543 double CmpVal = IsUnsigned ? -(double)Limit * 2.0 : -(double)Limit;
554 F->insert(It, FalseMBB);
555 F->insert(It, TrueMBB);
556 F->insert(It, DoneMBB);
559 DoneMBB->
splice(DoneMBB->
begin(), BB, std::next(
MI.getIterator()), BB->
end());
567 unsigned Tmp0, Tmp1, CmpReg, EqzReg, FalseReg, TrueReg;
575 MI.eraseFromParent();
632 if (Def->getOpcode() == WebAssembly::CONST_I32 ||
633 Def->getOpcode() == WebAssembly::CONST_I64) {
634 if (Def->getOperand(1).getImm() == 0) {
636 MI.eraseFromParent();
640 unsigned MemoryCopy =
641 Int64 ? WebAssembly::MEMORY_COPY_A64 : WebAssembly::MEMORY_COPY_A32;
648 MI.eraseFromParent();
659 unsigned Eqz = Int64 ? WebAssembly::EQZ_I64 : WebAssembly::EQZ_I32;
660 unsigned MemoryCopy =
661 Int64 ? WebAssembly::MEMORY_COPY_A64 : WebAssembly::MEMORY_COPY_A32;
672 F->insert(It, TrueMBB);
673 F->insert(It, DoneMBB);
676 DoneMBB->
splice(DoneMBB->
begin(), BB, std::next(
MI.getIterator()), BB->
end());
689 MI.eraseFromParent();
723 if (Def->getOpcode() == WebAssembly::CONST_I32 ||
724 Def->getOpcode() == WebAssembly::CONST_I64) {
725 if (Def->getOperand(1).getImm() == 0) {
727 MI.eraseFromParent();
731 unsigned MemoryFill =
732 Int64 ? WebAssembly::MEMORY_FILL_A64 : WebAssembly::MEMORY_FILL_A32;
738 MI.eraseFromParent();
749 unsigned Eqz = Int64 ? WebAssembly::EQZ_I64 : WebAssembly::EQZ_I32;
750 unsigned MemoryFill =
751 Int64 ? WebAssembly::MEMORY_FILL_A64 : WebAssembly::MEMORY_FILL_A32;
762 F->insert(It, TrueMBB);
763 F->insert(It, DoneMBB);
766 DoneMBB->
splice(DoneMBB->
begin(), BB, std::next(
MI.getIterator()), BB->
end());
779 MI.eraseFromParent();
801 CallResults.
getOpcode() == WebAssembly::RET_CALL_RESULTS);
805 bool IsRetCall = CallResults.
getOpcode() == WebAssembly::RET_CALL_RESULTS;
807 bool IsFuncrefCall =
false;
813 IsFuncrefCall = (TRC == &WebAssembly::FUNCREFRegClass);
818 if (IsIndirect && IsRetCall) {
819 CallOp = WebAssembly::RET_CALL_INDIRECT;
820 }
else if (IsIndirect) {
821 CallOp = WebAssembly::CALL_INDIRECT;
822 }
else if (IsRetCall) {
823 CallOp = WebAssembly::RET_CALL;
825 CallOp = WebAssembly::CALL;
854 for (
auto Def : CallResults.
defs())
878 for (
auto Use : CallParams.
uses())
894 if (IsIndirect && IsFuncrefCall) {
906 BuildMI(MF,
DL,
TII.get(WebAssembly::REF_NULL_FUNCREF), RegFuncref);
910 BuildMI(MF,
DL,
TII.get(WebAssembly::TABLE_SET_FUNCREF))
922 const TargetInstrInfo &
TII = *Subtarget->getInstrInfo();
925 switch (
MI.getOpcode()) {
928 case WebAssembly::FP_TO_SINT_I32_F32:
930 WebAssembly::I32_TRUNC_S_F32);
931 case WebAssembly::FP_TO_UINT_I32_F32:
933 WebAssembly::I32_TRUNC_U_F32);
934 case WebAssembly::FP_TO_SINT_I64_F32:
936 WebAssembly::I64_TRUNC_S_F32);
937 case WebAssembly::FP_TO_UINT_I64_F32:
939 WebAssembly::I64_TRUNC_U_F32);
940 case WebAssembly::FP_TO_SINT_I32_F64:
942 WebAssembly::I32_TRUNC_S_F64);
943 case WebAssembly::FP_TO_UINT_I32_F64:
945 WebAssembly::I32_TRUNC_U_F64);
946 case WebAssembly::FP_TO_SINT_I64_F64:
948 WebAssembly::I64_TRUNC_S_F64);
949 case WebAssembly::FP_TO_UINT_I64_F64:
951 WebAssembly::I64_TRUNC_U_F64);
952 case WebAssembly::MEMCPY_A32:
954 case WebAssembly::MEMCPY_A64:
956 case WebAssembly::MEMSET_A32:
958 case WebAssembly::MEMSET_A64:
960 case WebAssembly::CALL_RESULTS:
961 case WebAssembly::RET_CALL_RESULTS:
966std::pair<unsigned, const TargetRegisterClass *>
967WebAssemblyTargetLowering::getRegForInlineAsmConstraint(
971 if (Constraint.
size() == 1) {
972 switch (Constraint[0]) {
974 assert(VT != MVT::iPTR &&
"Pointer MVT not expected here");
975 if (Subtarget->hasSIMD128() && VT.
isVector()) {
977 return std::make_pair(0U, &WebAssembly::V128RegClass);
981 return std::make_pair(0U, &WebAssembly::I32RegClass);
983 return std::make_pair(0U, &WebAssembly::I64RegClass);
988 return std::make_pair(0U, &WebAssembly::F32RegClass);
990 return std::make_pair(0U, &WebAssembly::F64RegClass);
1004bool WebAssemblyTargetLowering::isCheapToSpeculateCttz(
Type *Ty)
const {
1009bool WebAssemblyTargetLowering::isCheapToSpeculateCtlz(
Type *Ty)
const {
1014bool WebAssemblyTargetLowering::isLegalAddressingMode(
const DataLayout &
DL,
1016 Type *Ty,
unsigned AS,
1021 if (AM.BaseOffs < 0)
1032bool WebAssemblyTargetLowering::allowsMisalignedMemoryAccesses(
1046bool WebAssemblyTargetLowering::isIntDivCheap(
EVT VT,
1047 AttributeList Attr)
const {
1053bool WebAssemblyTargetLowering::isVectorLoadExtDesirable(
SDValue ExtVal)
const {
1059 EVT MemT =
Load->getValueType(0);
1060 return (ExtT == MVT::v8i16 && MemT == MVT::v8i8) ||
1061 (ExtT == MVT::v4i32 && MemT == MVT::v4i16) ||
1062 (ExtT == MVT::v2i64 && MemT == MVT::v2i32);
1065bool WebAssemblyTargetLowering::isOffsetFoldingLegal(
1068 const GlobalValue *GV = GA->
getGlobal();
1072EVT WebAssemblyTargetLowering::getSetCCResultType(
const DataLayout &
DL,
1089void WebAssemblyTargetLowering::getTgtMemIntrinsic(
1093 switch (Intrinsic) {
1094 case Intrinsic::wasm_memory_atomic_notify:
1096 Info.memVT = MVT::i32;
1097 Info.ptrVal =
I.getArgOperand(0);
1109 case Intrinsic::wasm_memory_atomic_wait32:
1111 Info.memVT = MVT::i32;
1112 Info.ptrVal =
I.getArgOperand(0);
1118 case Intrinsic::wasm_memory_atomic_wait64:
1120 Info.memVT = MVT::i64;
1121 Info.ptrVal =
I.getArgOperand(0);
1127 case Intrinsic::wasm_loadf16_f32:
1129 Info.memVT = MVT::f16;
1130 Info.ptrVal =
I.getArgOperand(0);
1136 case Intrinsic::wasm_storef16_f32:
1138 Info.memVT = MVT::f16;
1139 Info.ptrVal =
I.getArgOperand(1);
1150void WebAssemblyTargetLowering::computeKnownBitsForTargetNode(
1153 switch (
Op.getOpcode()) {
1157 unsigned IntNo =
Op.getConstantOperandVal(0);
1161 case Intrinsic::wasm_bitmask: {
1163 EVT VT =
Op.getOperand(1).getSimpleValueType();
1166 Known.
Zero |= ZeroMask;
1172 case WebAssemblyISD::EXTEND_LOW_U:
1173 case WebAssemblyISD::EXTEND_HIGH_U: {
1178 if (VT == MVT::v8i8 || VT == MVT::v16i8) {
1182 }
else if (VT == MVT::v4i16 || VT == MVT::v8i16) {
1186 }
else if (VT == MVT::v2i32 || VT == MVT::v4i32) {
1196 case WebAssemblyISD::I64_ADD128:
1197 if (
Op.getResNo() == 1) {
1208WebAssemblyTargetLowering::getPreferredVectorAction(
MVT VT)
const {
1214 if (EltVT == MVT::i8 || EltVT == MVT::i16 || EltVT == MVT::i32 ||
1215 EltVT == MVT::i64 || EltVT == MVT::f32 || EltVT == MVT::f64)
1222bool WebAssemblyTargetLowering::isFMAFasterThanFMulAndFAdd(
1224 if (!Subtarget->hasFP16() || !VT.
isVector())
1234bool WebAssemblyTargetLowering::shouldSimplifyDemandedVectorElts(
1235 SDValue Op,
const TargetLoweringOpt &TLO)
const {
1288WebAssemblyTargetLowering::LowerCall(CallLoweringInfo &CLI,
1290 SelectionDAG &DAG = CLI.DAG;
1300 "WebAssembly doesn't support language-specific or target-specific "
1301 "calling conventions yet");
1302 if (CLI.IsPatchPoint)
1303 fail(
DL, DAG,
"WebAssembly doesn't support patch point yet");
1305 if (CLI.IsTailCall) {
1306 auto NoTail = [&](
const char *Msg) {
1307 if (CLI.CB && CLI.CB->isMustTailCall())
1309 CLI.IsTailCall =
false;
1312 if (!Subtarget->hasTailCall())
1313 NoTail(
"WebAssembly 'tail-call' feature not enabled");
1317 NoTail(
"WebAssembly does not support varargs tail calls");
1322 Type *RetTy =
F.getReturnType();
1327 bool TypesMatch = CallerRetTys.
size() == CalleeRetTys.
size() &&
1328 std::equal(CallerRetTys.
begin(), CallerRetTys.
end(),
1329 CalleeRetTys.
begin());
1331 NoTail(
"WebAssembly tail call requires caller and callee return types to "
1336 for (
auto &Arg : CLI.CB->args()) {
1337 Value *Val = Arg.get();
1342 Src =
GEP->getPointerOperand();
1349 "WebAssembly does not support tail calling with stack arguments");
1356 SmallVectorImpl<ISD::InputArg> &Ins = CLI.Ins;
1357 SmallVectorImpl<ISD::OutputArg> &Outs = CLI.Outs;
1358 SmallVectorImpl<SDValue> &OutVals = CLI.OutVals;
1364 Outs[0].Flags.isSRet()) {
1369 bool HasSwiftSelfArg =
false;
1370 bool HasSwiftErrorArg =
false;
1371 bool HasSwiftAsyncArg =
false;
1372 unsigned NumFixedArgs = 0;
1373 for (
unsigned I = 0;
I < Outs.
size(); ++
I) {
1374 const ISD::OutputArg &Out = Outs[
I];
1380 fail(
DL, DAG,
"WebAssembly hasn't implemented nest arguments");
1382 fail(
DL, DAG,
"WebAssembly hasn't implemented inalloca arguments");
1384 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs arguments");
1386 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs last arguments");
1396 Chain = DAG.
getMemcpy(Chain,
DL, FINode, OutVal, SizeNode, Alignment,
1399 nullptr, std::nullopt, MachinePointerInfo(),
1400 MachinePointerInfo());
1407 bool IsVarArg = CLI.IsVarArg;
1416 if (!HasSwiftSelfArg) {
1418 ISD::ArgFlagsTy
Flags;
1419 Flags.setSwiftSelf();
1420 ISD::OutputArg Arg(Flags, PtrVT, EVT(PtrVT), PtrTy, 0, 0);
1421 CLI.Outs.push_back(Arg);
1423 CLI.OutVals.push_back(ArgVal);
1425 if (!HasSwiftErrorArg) {
1427 ISD::ArgFlagsTy
Flags;
1428 Flags.setSwiftError();
1429 ISD::OutputArg Arg(Flags, PtrVT, EVT(PtrVT), PtrTy, 0, 0);
1430 CLI.Outs.push_back(Arg);
1432 CLI.OutVals.push_back(ArgVal);
1436 ISD::ArgFlagsTy
Flags;
1437 Flags.setSwiftAsync();
1438 ISD::OutputArg Arg(Flags, PtrVT, EVT(PtrVT), PtrTy, 0, 0);
1439 CLI.Outs.push_back(Arg);
1441 CLI.OutVals.push_back(ArgVal);
1447 CCState CCInfo(CallConv, IsVarArg, MF, ArgLocs, *DAG.
getContext());
1452 for (
unsigned I = NumFixedArgs;
I < Outs.
size(); ++
I) {
1453 const ISD::OutputArg &Out = Outs[
I];
1456 assert(VT != MVT::iPTR &&
"Legalized args should be concrete");
1461 CCInfo.AllocateStack(Layout.getTypeAllocSize(Ty), Alignment);
1468 unsigned NumBytes = CCInfo.getAlignedCallFrameSize();
1471 if (IsVarArg && NumBytes) {
1474 MaybeAlign StackAlign = Layout.getStackAlignment();
1475 assert(StackAlign &&
"data layout string is missing stack alignment");
1481 assert(ArgLocs[ValNo].getValNo() == ValNo &&
1482 "ArgLocs should remain in order and only hold varargs args");
1483 unsigned Offset = ArgLocs[ValNo++].getLocMemOffset();
1491 if (!Chains.
empty())
1493 }
else if (IsVarArg) {
1511 Ops.push_back(Chain);
1512 Ops.push_back(Callee);
1517 IsVarArg ? OutVals.
begin() + NumFixedArgs : OutVals.
end());
1520 Ops.push_back(FINode);
1523 for (
const auto &In : Ins) {
1524 assert(!
In.Flags.isByVal() &&
"byval is not valid for return values");
1525 assert(!
In.Flags.isNest() &&
"nest is not valid for return values");
1526 if (
In.Flags.isInAlloca())
1527 fail(
DL, DAG,
"WebAssembly hasn't implemented inalloca return values");
1528 if (
In.Flags.isInConsecutiveRegs())
1529 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs return values");
1530 if (
In.Flags.isInConsecutiveRegsLast())
1532 "WebAssembly hasn't implemented cons regs last return values");
1541 CLI.CB->getCalledOperand()->getType())) {
1556 WebAssemblyISD::TABLE_SET,
DL, DAG.
getVTList(MVT::Other), TableSetOps,
1561 CLI.CB->getCalledOperand()->getPointerAlignment(DAG.
getDataLayout()),
1567 if (CLI.IsTailCall) {
1569 SDVTList NodeTys = DAG.
getVTList(MVT::Other, MVT::Glue);
1574 SDVTList InTyList = DAG.
getVTList(InTys);
1577 for (
size_t I = 0;
I < Ins.size(); ++
I)
1584bool WebAssemblyTargetLowering::CanLowerReturn(
1587 const Type *RetTy)
const {
1592SDValue WebAssemblyTargetLowering::LowerReturn(
1598 "MVP WebAssembly can only return up to one value");
1600 fail(
DL, DAG,
"WebAssembly doesn't support non-C calling conventions");
1603 RetOps.append(OutVals.
begin(), OutVals.
end());
1604 Chain = DAG.
getNode(WebAssemblyISD::RETURN,
DL, MVT::Other, RetOps);
1607 for (
const ISD::OutputArg &Out : Outs) {
1612 fail(
DL, DAG,
"WebAssembly hasn't implemented inalloca results");
1614 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs results");
1616 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs last results");
1622SDValue WebAssemblyTargetLowering::LowerFormalArguments(
1627 fail(
DL, DAG,
"WebAssembly doesn't support non-C calling conventions");
1630 auto *MFI = MF.
getInfo<WebAssemblyFunctionInfo>();
1636 bool HasSwiftErrorArg =
false;
1637 bool HasSwiftSelfArg =
false;
1638 bool HasSwiftAsyncArg =
false;
1639 for (
const ISD::InputArg &In : Ins) {
1640 HasSwiftSelfArg |=
In.Flags.isSwiftSelf();
1641 HasSwiftErrorArg |=
In.Flags.isSwiftError();
1642 HasSwiftAsyncArg |=
In.Flags.isSwiftAsync();
1643 if (
In.Flags.isInAlloca())
1644 fail(
DL, DAG,
"WebAssembly hasn't implemented inalloca arguments");
1645 if (
In.Flags.isNest())
1646 fail(
DL, DAG,
"WebAssembly hasn't implemented nest arguments");
1647 if (
In.Flags.isInConsecutiveRegs())
1648 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs arguments");
1649 if (
In.Flags.isInConsecutiveRegsLast())
1650 fail(
DL, DAG,
"WebAssembly hasn't implemented cons regs last arguments");
1659 MFI->addParam(
In.VT);
1668 if (!HasSwiftSelfArg) {
1669 MFI->addParam(PtrVT);
1671 if (!HasSwiftErrorArg) {
1672 MFI->addParam(PtrVT);
1675 MFI->addParam(PtrVT);
1684 MFI->setVarargBufferVreg(VarargVreg);
1686 Chain,
DL, VarargVreg,
1687 DAG.
getNode(WebAssemblyISD::ARGUMENT,
DL, PtrVT,
1689 MFI->addParam(PtrVT);
1701 assert(MFI->getParams().size() == Params.
size() &&
1702 std::equal(MFI->getParams().begin(), MFI->getParams().end(),
1708void WebAssemblyTargetLowering::ReplaceNodeResults(
1710 switch (
N->getOpcode()) {
1724 EVT VT =
N->getValueType(0);
1726 if (VT == MVT::v4f16 && Src.getValueType() == MVT::v4f32) {
1728 DAG.
getNode(WebAssemblyISD::DEMOTE_ZERO, SDLoc(
N), MVT::v8f16, Src));
1734 Results.push_back(Replace128Op(
N, DAG));
1738 "ReplaceNodeResults not implemented for this op for WebAssembly!");
1749 switch (
Op.getOpcode()) {
1754 return LowerFrameIndex(
Op, DAG);
1756 return LowerGlobalAddress(
Op, DAG);
1758 return LowerGlobalTLSAddress(
Op, DAG);
1760 return LowerExternalSymbol(
Op, DAG);
1762 return LowerJumpTable(
Op, DAG);
1764 return LowerBR_JT(
Op, DAG);
1766 return LowerVASTART(
Op, DAG);
1769 fail(
DL, DAG,
"WebAssembly hasn't implemented computed gotos");
1772 return LowerRETURNADDR(
Op, DAG);
1774 return LowerFRAMEADDR(
Op, DAG);
1776 return LowerCopyToReg(
Op, DAG);
1779 return LowerAccessVectorElement(
Op, DAG);
1783 return LowerIntrinsic(
Op, DAG);
1785 return LowerSIGN_EXTEND_INREG(
Op, DAG);
1789 return LowerEXTEND_VECTOR_INREG(
Op, DAG);
1791 return LowerBUILD_VECTOR(
Op, DAG);
1793 return LowerVECTOR_SHUFFLE(
Op, DAG);
1795 return LowerSETCC(
Op, DAG);
1799 return LowerShift(
Op, DAG);
1802 return LowerFP_TO_INT_SAT(
Op, DAG);
1805 return LowerFMIN(
Op, DAG);
1808 return LowerFMAX(
Op, DAG);
1810 return LowerLoad(
Op, DAG);
1812 return LowerStore(
Op, DAG);
1821 return LowerMUL_LOHI(
Op, DAG);
1823 return LowerUADDO(
Op, DAG);
1838 return std::nullopt;
1857 SDVTList Tys = DAG.
getVTList(MVT::Other);
1869 SDVTList Tys = DAG.
getVTList(MVT::Other);
1871 return DAG.
getNode(WebAssemblyISD::LOCAL_SET,
DL, Tys,
Ops);
1876 "Encountered an unlowerable store to the wasm_var address space",
1892 "unexpected offset when loading from webassembly global",
false);
1903 "unexpected offset when loading from webassembly local",
false);
1907 return DAG.
getNode(WebAssemblyISD::LOCAL_GET,
DL, {LocalVT, MVT::Other},
1913 "Encountered an unlowerable load from the wasm_var address space",
1921 assert(Subtarget->hasWideArithmetic());
1922 assert(
Op.getValueType() == MVT::i64);
1925 switch (
Op.getOpcode()) {
1927 Opcode = WebAssemblyISD::I64_MUL_WIDE_U;
1930 Opcode = WebAssemblyISD::I64_MUL_WIDE_S;
1951 assert(Subtarget->hasWideArithmetic());
1952 assert(
Op.getValueType() == MVT::i64);
1959 DAG.
getNode(WebAssemblyISD::I64_ADD128,
DL,
1969 assert(Subtarget->hasWideArithmetic());
1970 assert(
N->getValueType(0) == MVT::i128);
1973 switch (
N->getOpcode()) {
1975 Opcode = WebAssemblyISD::I64_ADD128;
1978 Opcode = WebAssemblyISD::I64_SUB128;
1993 LHS_0, LHS_1, RHS_0, RHS_1);
2010 EVT VT = Src.getValueType();
2012 : WebAssembly::COPY_I64,
2015 return Op.getNode()->getNumValues() == 1
2034 if (!Subtarget->getTargetTriple().isOSEmscripten()) {
2036 "Non-Emscripten WebAssembly hasn't implemented "
2037 "__builtin_return_address");
2041 unsigned Depth =
Op.getConstantOperandVal(0);
2043 return makeLibCall(DAG, RTLIB::RETURN_ADDRESS,
Op.getValueType(),
2044 {DAG.getConstant(Depth, DL, MVT::i32)}, CallOptions,
DL)
2053 if (
Op.getConstantOperandVal(0) > 0)
2057 EVT VT =
Op.getValueType();
2064WebAssemblyTargetLowering::LowerGlobalTLSAddress(
SDValue Op,
2070 if (!MF.
getSubtarget<WebAssemblySubtarget>().hasBulkMemory())
2074 const GlobalValue *GV = GA->
getGlobal();
2079 auto model = Subtarget->getTargetTriple().isOSEmscripten()
2100 DAG.
getNode(WebAssemblyISD::WrapperREL,
DL, PtrVT, TLSOffset);
2107 EVT VT =
Op.getValueType();
2108 return DAG.
getNode(WebAssemblyISD::Wrapper,
DL, VT,
2118 EVT VT =
Op.getValueType();
2120 "Unexpected target flags on generic GlobalAddressSDNode");
2122 fail(
DL, DAG,
"Invalid address space for WebAssembly target");
2125 const GlobalValue *GV = GA->
getGlobal();
2133 const char *BaseName;
2142 DAG.
getNode(WebAssemblyISD::Wrapper,
DL, PtrVT,
2146 WebAssemblyISD::WrapperREL,
DL, VT,
2155 return DAG.
getNode(WebAssemblyISD::Wrapper,
DL, VT,
2161WebAssemblyTargetLowering::LowerExternalSymbol(
SDValue Op,
2165 EVT VT =
Op.getValueType();
2166 assert(ES->getTargetFlags() == 0 &&
2167 "Unexpected target flags on generic ExternalSymbolSDNode");
2168 return DAG.
getNode(WebAssemblyISD::Wrapper,
DL, VT,
2191 Ops.push_back(Chain);
2192 Ops.push_back(Index);
2198 for (
auto *
MBB : MBBs)
2205 return DAG.
getNode(WebAssemblyISD::BR_TABLE,
DL, MVT::Other,
Ops);
2217 MFI->getVarargBufferVreg(), PtrVT);
2218 return DAG.
getStore(
Op.getOperand(0),
DL, ArgN,
Op.getOperand(1),
2219 MachinePointerInfo(SV));
2226 switch (
Op.getOpcode()) {
2229 IntNo =
Op.getConstantOperandVal(1);
2232 IntNo =
Op.getConstantOperandVal(0);
2243 case Intrinsic::wasm_lsda: {
2252 DAG.
getNode(WebAssemblyISD::Wrapper,
DL, PtrVT,
2255 DAG.
getNode(WebAssemblyISD::WrapperREL,
DL, PtrVT, Node);
2259 return DAG.
getNode(WebAssemblyISD::Wrapper,
DL, PtrVT, Node);
2262 case Intrinsic::wasm_shuffle: {
2268 while (
OpIdx < 18) {
2277 return DAG.
getNode(WebAssemblyISD::SHUFFLE,
DL,
Op.getValueType(),
Ops);
2280 case Intrinsic::thread_pointer: {
2287WebAssemblyTargetLowering::LowerSIGN_EXTEND_INREG(
SDValue Op,
2297 assert(!Subtarget->hasSignExt() && Subtarget->hasSIMD128());
2301 const SDValue &Extract =
Op.getOperand(0);
2305 MVT ExtractedLaneT =
2309 if (ExtractedVecT == VecT)
2316 unsigned IndexVal =
Index->getAsZExtVal();
2335 assert((UserOpc == WebAssemblyISD::EXTEND_LOW_U ||
2336 UserOpc == WebAssemblyISD::EXTEND_LOW_S) &&
2337 "expected extend_low");
2342 size_t FirstIdx = Mask.size() / 2;
2343 for (
size_t i = 0; i < Mask.size() / 2; ++i) {
2344 if (Mask[i] !=
static_cast<int>(FirstIdx + i)) {
2350 unsigned Opc = UserOpc == WebAssemblyISD::EXTEND_LOW_S
2351 ? WebAssemblyISD::EXTEND_HIGH_S
2352 : WebAssemblyISD::EXTEND_HIGH_U;
2355 ShuffleSrc = DAG.
getBitcast(
Op.getValueType(), ShuffleSrc);
2361WebAssemblyTargetLowering::LowerEXTEND_VECTOR_INREG(
SDValue Op,
2364 EVT VT =
Op.getValueType();
2366 EVT SrcVT = Src.getValueType();
2373 "Unexpected extension factor.");
2376 if (Scale != 2 && Scale != 4 && Scale != 8)
2380 switch (
Op.getOpcode()) {
2385 Ext = WebAssemblyISD::EXTEND_LOW_U;
2388 Ext = WebAssemblyISD::EXTEND_LOW_S;
2399 while (Scale != 1) {
2413 if (
Op.getValueType() != MVT::v2f64 &&
Op.getValueType() != MVT::v4f32)
2417 unsigned &Index) ->
bool {
2418 switch (
Op.getOpcode()) {
2420 Opcode = WebAssemblyISD::CONVERT_LOW_S;
2423 Opcode = WebAssemblyISD::CONVERT_LOW_U;
2427 Opcode = WebAssemblyISD::PROMOTE_LOW;
2433 auto ExtractVector =
Op.getOperand(0);
2440 SrcVec = ExtractVector.getOperand(0);
2441 Index = ExtractVector.getConstantOperandVal(1);
2445 unsigned NumLanes =
Op.getValueType() == MVT::v2f64 ? 2 : 4;
2446 unsigned FirstOpcode = 0, SecondOpcode = 0, ThirdOpcode = 0, FourthOpcode = 0;
2447 unsigned FirstIndex = 0, SecondIndex = 0, ThirdIndex = 0, FourthIndex = 0;
2448 SDValue FirstSrcVec, SecondSrcVec, ThirdSrcVec, FourthSrcVec;
2450 if (!GetConvertedLane(
Op.getOperand(0), FirstOpcode, FirstSrcVec,
2452 !GetConvertedLane(
Op.getOperand(1), SecondOpcode, SecondSrcVec,
2457 if (NumLanes == 4 && (!GetConvertedLane(
Op.getOperand(2), ThirdOpcode,
2458 ThirdSrcVec, ThirdIndex) ||
2459 !GetConvertedLane(
Op.getOperand(3), FourthOpcode,
2460 FourthSrcVec, FourthIndex)))
2463 if (FirstOpcode != SecondOpcode)
2469 if (NumLanes == 4 &&
2470 (FirstOpcode != ThirdOpcode || FirstOpcode != FourthOpcode ||
2471 FirstSrcVec != SecondSrcVec || FirstSrcVec != ThirdSrcVec ||
2472 FirstSrcVec != FourthSrcVec || FirstIndex != 0 || SecondIndex != 1 ||
2473 ThirdIndex != 2 || FourthIndex != 3))
2477 switch (FirstOpcode) {
2478 case WebAssemblyISD::CONVERT_LOW_S:
2479 case WebAssemblyISD::CONVERT_LOW_U:
2480 ExpectedSrcVT = MVT::v4i32;
2482 case WebAssemblyISD::PROMOTE_LOW:
2483 ExpectedSrcVT = NumLanes == 2 ? MVT::v4f32 : MVT::v8i16;
2489 auto Src = FirstSrcVec;
2490 if (NumLanes == 2 &&
2491 (FirstIndex != 0 || SecondIndex != 1 || FirstSrcVec != SecondSrcVec)) {
2494 {
static_cast<int>(FirstIndex),
2495 static_cast<int>(SecondIndex) + 4, -1, -1});
2497 return DAG.
getNode(FirstOpcode,
DL, NumLanes == 2 ? MVT::v2f64 : MVT::v4f32,
2503 MVT VT =
Op.getSimpleValueType();
2504 if (VT == MVT::v8f16) {
2519 const EVT VecT =
Op.getValueType();
2520 const EVT LaneT =
Op.getOperand(0).getValueType();
2522 bool CanSwizzle = VecT == MVT::v16i8;
2543 auto GetSwizzleSrcs = [](
size_t I,
const SDValue &Lane) {
2547 const SDValue &SwizzleSrc = Lane->getOperand(0);
2548 const SDValue &IndexExt = Lane->getOperand(1);
2558 Index->getConstantOperandVal(1) !=
I)
2560 return std::make_pair(SwizzleSrc, SwizzleIndices);
2567 auto GetShuffleSrc = [&](
const SDValue &Lane) {
2572 if (Lane->getOperand(0).getValueType().getVectorNumElements() >
2575 return Lane->getOperand(0);
2578 using ValueEntry = std::pair<SDValue, size_t>;
2581 using SwizzleEntry = std::pair<std::pair<SDValue, SDValue>,
size_t>;
2584 using ShuffleEntry = std::pair<SDValue, size_t>;
2587 auto AddCount = [](
auto &Counts,
const auto &Val) {
2590 if (CountIt == Counts.end()) {
2591 Counts.emplace_back(Val, 1);
2597 auto GetMostCommon = [](
auto &Counts) {
2599 assert(CommonIt != Counts.end() &&
"Unexpected all-undef build_vector");
2603 size_t NumConstantLanes = 0;
2606 for (
size_t I = 0;
I < Lanes; ++
I) {
2611 AddCount(SplatValueCounts, Lane);
2615 if (
auto ShuffleSrc = GetShuffleSrc(Lane))
2616 AddCount(ShuffleCounts, ShuffleSrc);
2618 auto SwizzleSrcs = GetSwizzleSrcs(
I, Lane);
2619 if (SwizzleSrcs.first)
2620 AddCount(SwizzleCounts, SwizzleSrcs);
2625 size_t NumSplatLanes;
2626 std::tie(SplatValue, NumSplatLanes) = GetMostCommon(SplatValueCounts);
2630 size_t NumSwizzleLanes = 0;
2631 if (SwizzleCounts.
size())
2632 std::forward_as_tuple(std::tie(SwizzleSrc, SwizzleIndices),
2633 NumSwizzleLanes) = GetMostCommon(SwizzleCounts);
2637 SDValue ShuffleSrc1, ShuffleSrc2;
2638 size_t NumShuffleLanes = 0;
2639 if (ShuffleCounts.
size()) {
2640 std::tie(ShuffleSrc1, NumShuffleLanes) = GetMostCommon(ShuffleCounts);
2642 [&](
const auto &Pair) {
return Pair.first == ShuffleSrc1; });
2644 if (ShuffleCounts.
size()) {
2645 size_t AdditionalShuffleLanes;
2646 std::tie(ShuffleSrc2, AdditionalShuffleLanes) =
2647 GetMostCommon(ShuffleCounts);
2648 NumShuffleLanes += AdditionalShuffleLanes;
2653 std::function<bool(
size_t,
const SDValue &)> IsLaneConstructed;
2656 if (NumSwizzleLanes >= NumShuffleLanes &&
2657 NumSwizzleLanes >= NumConstantLanes && NumSwizzleLanes >= NumSplatLanes) {
2660 auto Swizzled = std::make_pair(SwizzleSrc, SwizzleIndices);
2661 IsLaneConstructed = [&, Swizzled](
size_t I,
const SDValue &Lane) {
2662 return Swizzled == GetSwizzleSrcs(
I, Lane);
2664 }
else if (NumShuffleLanes >= NumConstantLanes &&
2665 NumShuffleLanes >= NumSplatLanes) {
2675 assert(LaneSize > DestLaneSize);
2676 Scale1 = LaneSize / DestLaneSize;
2682 assert(LaneSize > DestLaneSize);
2683 Scale2 = LaneSize / DestLaneSize;
2688 assert(DestLaneCount <= 16);
2689 for (
size_t I = 0;
I < DestLaneCount; ++
I) {
2691 SDValue Src = GetShuffleSrc(Lane);
2692 if (Src == ShuffleSrc1) {
2694 }
else if (Src && Src == ShuffleSrc2) {
2700 ArrayRef<int> MaskRef(Mask, DestLaneCount);
2702 IsLaneConstructed = [&](size_t,
const SDValue &Lane) {
2703 auto Src = GetShuffleSrc(Lane);
2704 return Src == ShuffleSrc1 || (Src && Src == ShuffleSrc2);
2706 }
else if (NumConstantLanes >= NumSplatLanes) {
2708 for (
const SDValue &Lane :
Op->op_values()) {
2714 uint64_t LaneBits = 128 / Lanes;
2717 Const->getAPIntValue().trunc(LaneBits).getZExtValue(),
2718 SDLoc(Lane), LaneT));
2734 if (NumSplatLanes == 1 &&
Op->getOperand(0) == SplatValue &&
2735 (DestLaneSize == 32 || DestLaneSize == 64)) {
2742 IsLaneConstructed = [&SplatValue](
size_t _,
const SDValue &Lane) {
2743 return Lane == SplatValue;
2748 assert(IsLaneConstructed);
2751 for (
size_t I = 0;
I < Lanes; ++
I) {
2753 if (!Lane.
isUndef() && !IsLaneConstructed(
I, Lane))
2762WebAssemblyTargetLowering::LowerVECTOR_SHUFFLE(
SDValue Op,
2766 MVT VecType =
Op.getOperand(0).getSimpleValueType();
2777 for (
int M : Mask) {
2778 for (
size_t J = 0; J < LaneBytes; ++J) {
2782 uint64_t ByteIndex =
M == -1 ? J : (uint64_t)M * LaneBytes + J;
2787 return DAG.
getNode(WebAssemblyISD::SHUFFLE,
DL,
Op.getValueType(),
Ops);
2795 assert(
Op->getOperand(0)->getSimpleValueType(0) == MVT::v2i64);
2800 auto MakeLane = [&](
unsigned I) {
2806 {MakeLane(0), MakeLane(1)});
2810WebAssemblyTargetLowering::LowerAccessVectorElement(
SDValue Op,
2827 EVT LaneT =
Op.getSimpleValueType().getVectorElementType();
2829 if (LaneT.
bitsGE(MVT::i32))
2833 size_t NumLanes =
Op.getSimpleValueType().getVectorNumElements();
2835 unsigned ShiftOpcode =
Op.getOpcode();
2841 for (
size_t i = 0; i < NumLanes; ++i) {
2844 SDValue ShiftedValue = ShiftedElements[i];
2849 DAG.
getNode(ShiftOpcode,
DL, MVT::i32, ShiftedValue, MaskedShiftValue));
2858 assert(
Op.getSimpleValueType().isVector());
2860 uint64_t LaneBits =
Op.getValueType().getScalarSizeInBits();
2861 auto ShiftVal =
Op.getOperand(1);
2864 auto SkipImpliedMask = [](
SDValue MaskOp, uint64_t MaskBits) {
2875 MaskVal == MaskBits)
2882 if (ConstantRHS && ConstantRHS->getAPIntValue() == MaskBits)
2890 ShiftVal = SkipImpliedMask(ShiftVal, LaneBits - 1);
2896 ShiftVal = SkipImpliedMask(ShiftVal, LaneBits - 1);
2901 switch (
Op.getOpcode()) {
2903 Opcode = WebAssemblyISD::VEC_SHL;
2906 Opcode = WebAssemblyISD::VEC_SHR_S;
2909 Opcode = WebAssemblyISD::VEC_SHR_U;
2915 return DAG.
getNode(Opcode,
DL,
Op.getValueType(),
Op.getOperand(0), ShiftVal);
2920 EVT ResT =
Op.getValueType();
2923 if ((ResT == MVT::i32 || ResT == MVT::i64) &&
2924 (SatVT == MVT::i32 || SatVT == MVT::i64))
2927 if (ResT == MVT::v4i32 && SatVT == MVT::i32)
2930 if (ResT == MVT::v8i16 && SatVT == MVT::i16)
2937 return (
Op->getFlags().hasNoNaNs() ||
2940 (
Op->getFlags().hasNoSignedZeros() ||
2948 return DAG.
getNode(WebAssemblyISD::RELAXED_FMIN, SDLoc(
Op),
2949 Op.getValueType(),
Op.getOperand(0),
Op.getOperand(1));
2957 return DAG.
getNode(WebAssemblyISD::RELAXED_FMAX, SDLoc(
Op),
2958 Op.getValueType(),
Op.getOperand(0),
Op.getOperand(1));
2968 auto &DAG = DCI.
DAG;
2975 SDValue Bitcast =
N->getOperand(0);
2978 if (!
N->getOperand(1).isUndef())
2980 SDValue CastOp = Bitcast.getOperand(0);
2982 EVT DstType = Bitcast.getValueType();
2983 if (!SrcType.is128BitVector() ||
2984 SrcType.getVectorNumElements() != DstType.getVectorNumElements())
2987 SrcType,
SDLoc(
N), CastOp, DAG.
getUNDEF(SrcType), Shuffle->getMask());
2997 auto &DAG = DCI.
DAG;
3001 EVT InVT =
N->getOperand(0)->getValueType(0);
3002 EVT ResVT =
N->getValueType(0);
3004 if (ResVT == MVT::v4f32 && (InVT == MVT::v4i16 || InVT == MVT::v4i8))
3006 else if (ResVT == MVT::v2f64 && (InVT == MVT::v2i16 || InVT == MVT::v2i8))
3020 auto &DAG = DCI.
DAG;
3024 EVT VT =
N->getValueType(0);
3038 auto &DAG = DCI.
DAG;
3042 EVT ResVT =
N->getValueType(0);
3046 if (ResVT == MVT::v16i32 &&
N->getOperand(0)->getValueType(0) == MVT::v16i8) {
3050 IsSext ? WebAssemblyISD::EXTEND_LOW_S : WebAssemblyISD::EXTEND_LOW_U;
3052 IsSext ? WebAssemblyISD::EXTEND_HIGH_S : WebAssemblyISD::EXTEND_HIGH_U;
3057 DAG.
getNode(LowOp,
DL, MVT::v4i32, LowHalf),
3058 DAG.
getNode(HighOp,
DL, MVT::v4i32, LowHalf),
3059 DAG.
getNode(LowOp,
DL, MVT::v4i32, HighHalf),
3060 DAG.
getNode(HighOp,
DL, MVT::v4i32, HighHalf),
3067 auto Extract =
N->getOperand(0);
3072 if (IndexNode ==
nullptr)
3074 auto Index = IndexNode->getZExtValue();
3078 if (ResVT == MVT::v8i16) {
3080 Source.getValueType() != MVT::v16i8 || (Index != 0 && Index != 8))
3082 }
else if (ResVT == MVT::v4i32) {
3084 Source.getValueType() != MVT::v8i16 || (Index != 0 && Index != 4))
3086 }
else if (ResVT == MVT::v2i64) {
3088 Source.getValueType() != MVT::v4i32 || (Index != 0 && Index != 2))
3094 bool IsLow = Index == 0;
3096 unsigned Op = IsSext ? (IsLow ? WebAssemblyISD::EXTEND_LOW_S
3097 : WebAssemblyISD::EXTEND_HIGH_S)
3098 : (IsLow ? WebAssemblyISD::EXTEND_LOW_U
3099 : WebAssemblyISD::EXTEND_HIGH_U);
3106 auto &DAG = DCI.
DAG;
3108 auto GetWasmConversionOp = [](
unsigned Op) {
3111 return WebAssemblyISD::TRUNC_SAT_ZERO_S;
3113 return WebAssemblyISD::TRUNC_SAT_ZERO_U;
3115 return WebAssemblyISD::DEMOTE_ZERO;
3120 auto IsZeroSplat = [](
SDValue SplatVal) {
3122 APInt SplatValue, SplatUndef;
3123 unsigned SplatBitSize;
3128 Splat->isConstantSplat(SplatValue, SplatUndef, SplatBitSize,
3147 EVT ExpectedConversionType;
3150 switch (ConversionOp) {
3154 ExpectedConversionType = MVT::v2i32;
3157 if (
Conversion.getValueType() == MVT::v2f32) {
3159 ExpectedConversionType = MVT::v2f32;
3160 }
else if (
Conversion.getValueType() == MVT::v4f16) {
3162 ExpectedConversionType = MVT::v4f16;
3171 if (
N->getValueType(0) != ResVT)
3174 if (
Conversion.getValueType() != ExpectedConversionType)
3178 if (!((Source.getValueType() == MVT::v2f64 && ResVT == MVT::v4f32) ||
3179 (Source.getValueType() == MVT::v2f64 && ResVT == MVT::v4i32) ||
3180 (Source.getValueType() == MVT::v4f32 && ResVT == MVT::v8f16)))
3183 if (!IsZeroSplat(
N->getOperand(1)) ||
3184 N->getOperand(1).getValueType() != ExpectedConversionType)
3187 unsigned Op = GetWasmConversionOp(ConversionOp);
3204 auto ConversionOp =
N->getOpcode();
3205 switch (ConversionOp) {
3211 ResVT =
N->getValueType(0);
3217 if (
N->getValueType(0) != ResVT)
3220 auto Concat =
N->getOperand(0);
3224 EVT SourceVT =
Concat.getOperand(0).getValueType();
3226 if (!IsZeroSplat(
Concat.getOperand(1)))
3231 ConcatVT == MVT::v4f64 && SourceVT == MVT::v2f64 && ResVT == MVT::v4f32;
3233 ConcatVT == MVT::v8f32 && SourceVT == MVT::v4f32 && ResVT == MVT::v8f16;
3234 if (!(IsF64ToF32 || IsF32ToF16))
3237 if (ConcatVT != MVT::v4f64 || SourceVT != MVT::v2f64 || ResVT != MVT::v4i32)
3241 unsigned Op = GetWasmConversionOp(ConversionOp);
3247 const SDLoc &
DL,
unsigned VectorWidth) {
3255 unsigned ElemsPerChunk = VectorWidth / ElVT.
getSizeInBits();
3260 IdxVal &= ~(ElemsPerChunk - 1);
3265 Vec->
ops().slice(IdxVal, ElemsPerChunk));
3277 EVT SrcVT = In.getValueType();
3295 EVT InVT = MVT::i16, OutVT = MVT::i8;
3300 unsigned SubSizeInBits = SrcSizeInBits / 2;
3302 OutVT =
EVT::getVectorVT(Ctx, OutVT, SubSizeInBits / OutVT.getSizeInBits());
3328 auto &DAG = DCI.
DAG;
3331 EVT InVT = In.getValueType();
3335 EVT OutVT =
N->getValueType(0);
3342 if (!((InSVT == MVT::i16 || InSVT == MVT::i32 || InSVT == MVT::i64) &&
3343 (OutSVT == MVT::i8 || OutSVT == MVT::i16) && OutVT.
is128BitVector()))
3356 auto &DAG = DCI.
DAG;
3359 EVT VT =
N->getValueType(0);
3360 EVT SrcVT = Src.getValueType();
3371 if (NumElts == 2 || NumElts == 4 || NumElts == 8 || NumElts == 16) {
3374 {DAG.getConstant(Intrinsic::wasm_bitmask, DL, MVT::i32),
3375 DAG.getSExtOrTrunc(N->getOperand(0), DL,
3376 SrcVT.changeVectorElementType(
3377 *DAG.getContext(), Width))}),
3382 if (NumElts == 32 || NumElts == 64) {
3395 EVT ConcatOperandVT =
Concat.getOperand(0).getValueType();
3398 EVT ConcatOperandMaskVT =
3401 EVT ConcatOperandBitmaskVT =
3403 EVT ReturnVT =
N->getValueType(0);
3413 "concat_vectors operands must have the same type");
3417 if (!SetCCVectorOperand ||
3427 DL, ConcatOperandMaskVT, ConcatOperand, SetCCVectorOperand, SetCond);
3428 SDValue ConcatOperandBitmask =
3429 DAG.
getBitcast(ConcatOperandBitmaskVT, ConcatOperandMask);
3430 SDValue ExtendedConcatOperandBitmask =
3435 ReconstructedBitmask = DAG.
getNode(
3436 ISD::SHL,
DL, ReturnVT, ReconstructedBitmask,
3441 ReconstructedBitmask =
3443 ExtendedConcatOperandBitmask);
3446 return ReconstructedBitmask;
3457 if (
N->getConstantOperandVal(0) != Intrinsic::wasm_bitmask)
3468 {DAG.getConstant(Intrinsic::wasm_bitmask, DL, MVT::i32), LHS});
3480 if (
N->getNumOperands() < 2 ||
3484 EVT LT =
LHS.getValueType();
3485 if (LT.getScalarSizeInBits() > 128 / LT.getVectorNumElements())
3488 auto CombineSetCC = [&
N, &DAG](Intrinsic::WASMIntrinsics InPre,
3490 Intrinsic::WASMIntrinsics InPost) {
3491 if (
N->getConstantOperandVal(0) != InPre)
3509 Intrinsic::wasm_alltrue))
3512 Intrinsic::wasm_anytrue))
3515 Intrinsic::wasm_anytrue))
3518 Intrinsic::wasm_alltrue))
3537 "mask reduction should be widened to a 128-bit vector");
3540 SDValue Mask =
N->getOperand(0)->getOperand(0);
3554 assert((NumElts == 32 || NumElts == 64) &&
3555 "combineWideMaskReduction is only for wide masks");
3559 unsigned ChunkElts = 16;
3580 for (
unsigned I = 0;
I < NumElts;
I += ChunkElts) {
3590 SDValue Acc = ChunkResults[0];
3591 for (
unsigned I = 1;
I < ChunkResults.
size(); ++
I)
3593 DAG.
getNode(Info.WideCombineOpcode,
DL, MVT::i32, Acc, ChunkResults[
I]);
3605 return std::nullopt;
3625 return std::nullopt;
3637 EVT VT =
N->getValueType(0);
3638 EVT OpVT =
X.getValueType();
3642 Attribute::NoImplicitFloat))
3648 !Subtarget->
hasSIMD128() || !isIntEqualitySetCC(CC))
3652 auto IsVectorBitCastCheap = [](
SDValue X) {
3657 if (!IsVectorBitCastCheap(
X) || !IsVectorBitCastCheap(
Y))
3667 : Intrinsic::wasm_anytrue,
3681 EVT VT =
N->getValueType(0);
3692 EVT FromVT =
LHS->getOperand(0).getValueType();
3701 auto &DAG = DCI.
DAG;
3702 if (NumElts == 2 || NumElts == 4 || NumElts == 8 || NumElts == 16)
3705 if (NumElts == 32 || NumElts == 64)
3713 EVT VT =
N->getValueType(0);
3714 if (VT != MVT::v8i32 && VT != MVT::v16i32)
3720 if (
LHS.getOpcode() !=
RHS.getOpcode())
3727 if (
LHS->getOperand(0).getValueType() !=
RHS->getOperand(0).getValueType())
3730 EVT FromVT =
LHS->getOperand(0).getValueType();
3732 if (EltTy != MVT::i8)
3760 unsigned ExtendLowOpc =
3761 IsSigned ? WebAssemblyISD::EXTEND_LOW_S : WebAssemblyISD::EXTEND_LOW_U;
3762 unsigned ExtendHighOpc =
3763 IsSigned ? WebAssemblyISD::EXTEND_HIGH_S : WebAssemblyISD::EXTEND_HIGH_U;
3765 auto GetExtendLow = [&DAG, &
DL, &ExtendLowOpc](
EVT VT,
SDValue Op) {
3772 if (NumElts == 16) {
3773 SDValue LowLHS = GetExtendLow(MVT::v8i16, ExtendInLHS);
3774 SDValue LowRHS = GetExtendLow(MVT::v8i16, ExtendInRHS);
3780 GetExtendLow(MVT::v4i32, MulLow),
3782 GetExtendLow(MVT::v4i32, MulHigh),
3791 SDValue Lo = GetExtendLow(MVT::v4i32, MulLow);
3801 EVT VT =
N->getValueType(0);
3810 if (VT != MVT::v8i8 && VT != MVT::v16i8)
3817 EVT MulVT = MVT::v8i16;
3819 if (VT == MVT::v8i8) {
3825 DAG.
getNode(WebAssemblyISD::EXTEND_LOW_U,
DL, MulVT, PromotedLHS);
3827 DAG.
getNode(WebAssemblyISD::EXTEND_LOW_U,
DL, MulVT, PromotedRHS);
3832 MVT::v16i8,
DL, MulLow, DAG.
getUNDEF(MVT::v16i8),
3833 {0, 2, 4, 6, 8, 10, 12, 14, -1, -1, -1, -1, -1, -1, -1, -1});
3836 assert(VT == MVT::v16i8 &&
"Expected v16i8");
3840 DAG.
getNode(WebAssemblyISD::EXTEND_HIGH_U,
DL, MulVT,
LHS);
3842 DAG.
getNode(WebAssemblyISD::EXTEND_HIGH_U,
DL, MulVT,
RHS);
3851 VT,
DL, MulLow, MulHigh,
3852 {0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30});
3860 EVT InVT = In.getValueType();
3865 if (NumElems < RequiredNumElems) {
3872 EVT OutVT =
N->getValueType(0);
3877 if (OutElTy != MVT::i8 && OutElTy != MVT::i16)
3884 EVT FPVT =
N->getOperand(0)->getValueType(0);
3902 EVT NarrowedVT = OutElTy == MVT::i8 ? MVT::v16i8 : MVT::v8i16;
3930 EVT VT =
N->getValueType(0);
3931 if (VT != MVT::v8i32)
3936 unsigned ExtOpc =
LHS.getOpcode();
3946 if (FromVT != MVT::v8i16)
3954 for (
unsigned I = 0;
I < NumElts; ++
I) {
3959 const APInt &ShiftAmt =
C->getAPIntValue();
3960 if (ShiftAmt.
uge(MaxValidShift))
3969 unsigned ExtLowOpc =
3970 IsSigned ? WebAssemblyISD::EXTEND_LOW_S : WebAssemblyISD::EXTEND_LOW_U;
3971 unsigned ExtHighOpc =
3972 IsSigned ? WebAssemblyISD::EXTEND_HIGH_S : WebAssemblyISD::EXTEND_HIGH_U;
3974 EVT HalfVT = MVT::v4i32;
3985WebAssemblyTargetLowering::PerformDAGCombine(
SDNode *
N,
3986 DAGCombinerInfo &DCI)
const {
3987 switch (
N->getOpcode()) {
static SDValue performMulCombine(SDNode *N, SelectionDAG &DAG, TargetLowering::DAGCombinerInfo &DCI, const AArch64Subtarget *Subtarget)
static SDValue performTruncateCombine(SDNode *N, SelectionDAG &DAG, TargetLowering::DAGCombinerInfo &DCI)
static SDValue performSETCCCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI, SelectionDAG &DAG)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
Function Alias Analysis false
Function Alias Analysis Results
static void fail(const SDLoc &DL, SelectionDAG &DAG, const Twine &Msg, SDValue Val={})
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
const HexagonInstrInfo * TII
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
Register const TargetRegisterInfo * TRI
Promote Memory to Register
MachineInstr unsigned OpIdx
static SDValue performVECTOR_SHUFFLECombine(SDNode *N, SelectionDAG &DAG, const RISCVSubtarget &Subtarget, const RISCVTargetLowering &TLI)
static SDValue combineVectorSizedSetCCEquality(EVT VT, SDValue X, SDValue Y, ISD::CondCode CC, const SDLoc &DL, SelectionDAG &DAG, const RISCVSubtarget &Subtarget)
Try to map an integer comparison with size > XLEN to vector instructions before type legalization spl...
Contains matchers for matching SelectionDAG nodes and values.
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static bool callingConvSupported(CallingConv::ID CallConv)
static MachineBasicBlock * LowerFPToInt(MachineInstr &MI, DebugLoc DL, MachineBasicBlock *BB, const TargetInstrInfo &TII, bool IsUnsigned, bool Int64, bool Float64, unsigned LoweredOpcode)
static SDValue TryWideExtMulCombine(SDNode *N, SelectionDAG &DAG)
static MachineBasicBlock * LowerMemcpy(MachineInstr &MI, DebugLoc DL, MachineBasicBlock *BB, const TargetInstrInfo &TII, bool Int64)
static std::optional< unsigned > IsWebAssemblyLocal(SDValue Op, SelectionDAG &DAG)
static SDValue performVectorExtendCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI)
static SDValue performVectorNonNegToFPCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI)
static SDValue unrollVectorShift(SDValue Op, SelectionDAG &DAG)
static SDValue performAnyAllCombine(SDNode *N, SelectionDAG &DAG)
static MachineBasicBlock * LowerCallResults(MachineInstr &CallResults, DebugLoc DL, MachineBasicBlock *BB, const WebAssemblySubtarget *Subtarget, const TargetInstrInfo &TII)
static std::optional< MaskReduceInfo > classifyMaskReduction(SDNode *N)
static SDValue GetExtendHigh(SDValue Op, unsigned UserOpc, EVT VT, SelectionDAG &DAG)
SDValue performConvertFPCombine(SDNode *N, SelectionDAG &DAG)
static SDValue performBitmaskCombine(SDNode *N, SelectionDAG &DAG)
static SDValue performVectorTruncZeroCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI)
static bool IsWebAssemblyGlobal(SDValue Op)
static SDValue combineSmallMaskReduction(SDNode *N, EVT FromVT, unsigned NumElts, const MaskReduceInfo &Info, SelectionDAG &DAG)
static MachineBasicBlock * LowerMemset(MachineInstr &MI, DebugLoc DL, MachineBasicBlock *BB, const TargetInstrInfo &TII, bool Int64)
static bool HasNoSignedZerosOrNaNs(SDValue Op, SelectionDAG &DAG)
SDValue DoubleVectorWidth(SDValue In, unsigned RequiredNumElems, SelectionDAG &DAG)
static SDValue performVectorExtendToFPCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI)
Convert ({u,s}itofp vec) --> ({u,s}itofp ({s,z}ext vec)) so it doesn't get split up into scalar instr...
static SDValue performShiftCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI)
static SDValue LowerConvertLow(SDValue Op, SelectionDAG &DAG)
static SDValue extractSubVector(SDValue Vec, unsigned IdxVal, SelectionDAG &DAG, const SDLoc &DL, unsigned VectorWidth)
static SDValue performBitcastCombine(SDNode *N, TargetLowering::DAGCombinerInfo &DCI)
static SDValue truncateVectorWithNARROW(EVT DstVT, SDValue In, const SDLoc &DL, SelectionDAG &DAG)
static SDValue combineWideMaskReduction(SDNode *N, SDValue Mask, EVT MaskVT, unsigned NumElts, const MaskReduceInfo &Info, SelectionDAG &DAG)
This file defines the interfaces that WebAssembly uses to lower LLVM code into a selection DAG.
This file provides WebAssembly-specific target descriptions.
This file declares WebAssembly-specific per-machine-function information.
This file declares the WebAssembly-specific subclass of TargetSubtarget.
This file declares the WebAssembly-specific subclass of TargetMachine.
This file contains the declaration of the WebAssembly-specific type parsing utility functions.
This file contains the declaration of the WebAssembly-specific utility functions.
static constexpr int Concat[]
Class for arbitrary precision integers.
uint64_t getZExtValue() const
Get zero extended value.
void setBitsFrom(unsigned loBit)
Set the top bits starting from loBit.
static APInt getLowBitsSet(unsigned numBits, unsigned loBitsSet)
Constructs an APInt value that has the bottom loBitsSet bits set.
static APInt getHighBitsSet(unsigned numBits, unsigned hiBitsSet)
Constructs an APInt value that has the top hiBitsSet bits set.
static APInt getOneBitSet(unsigned numBits, unsigned BitNo)
Return an APInt with exactly one bit set in the result.
bool uge(const APInt &RHS) const
Unsigned greater or equal comparison.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
an instruction that atomically reads a memory location, combines it with another value,...
BinOp getOperation() const
LLVM Basic Block Representation.
static CCValAssign getMem(unsigned ValNo, MVT ValVT, int64_t Offset, MVT LocVT, LocInfo HTP, bool IsCustom=false)
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
uint64_t getNumOperands() const
A parsed version of the target data layout string in and methods for querying it.
Diagnostic information for unsupported feature in backend.
static constexpr ElementCount getFixed(ScalarTy MinVal)
This is a fast-path instruction selection class that generates poor code and doesn't support illegal ...
FunctionLoweringInfo - This contains information that is global to a function that is used when lower...
FunctionType * getFunctionType() const
Returns the FunctionType for me.
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
bool hasFnAttribute(Attribute::AttrKind Kind) const
Return true if the function has the attribute.
int64_t getOffset() const
LLVM_ABI unsigned getAddressSpace() const
unsigned getTargetFlags() const
const GlobalValue * getGlobal() const
ThreadLocalMode getThreadLocalMode() const
Type * getValueType() const
unsigned getTargetFlags() const
This is an important class for using LLVM in a threaded context.
LLVM_ABI void diagnose(const DiagnosticInfo &DI)
Report a message to the currently installed diagnostic handler.
Tracks which library functions to use for a particular subtarget.
const SDValue & getBasePtr() const
const SDValue & getOffset() const
Describe properties that are true of each instruction in the target description file.
bool is128BitVector() const
Return true if this is a 128-bit vector type.
@ INVALID_SIMPLE_VALUE_TYPE
static auto integer_fixedlen_vector_valuetypes()
MVT changeVectorElementType(MVT EltVT) const
Return a VT for a vector type whose attributes match ourselves with the exception of the element type...
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.
static auto integer_valuetypes()
TypeSize getSizeInBits() const
Returns the size of the specified MVT in bits.
static auto fixedlen_vector_valuetypes()
bool isFixedLengthVector() const
static MVT getVectorVT(MVT VT, unsigned NumElements)
MVT getVectorElementType() const
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
static MVT getIntegerVT(unsigned BitWidth)
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.
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.
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
iterator insertAfter(iterator I, MachineInstr *MI)
Insert MI into the instruction list after I.
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 '...
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.
void setFrameAddressIsTaken(bool T)
unsigned getFunctionNumber() const
getFunctionNumber - Return a unique ID for the current function.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
const char * createExternalSymbolName(StringRef Name)
Allocate a string and populate it with the given external symbol name.
MCContext & getContext() const
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
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const MachineJumpTableInfo * getJumpTableInfo() const
getJumpTableInfo - Return the jump table info object for the current function.
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & add(const MachineOperand &MO) const
const MachineInstrBuilder & addSym(MCSymbol *Sym, unsigned char TargetFlags=0) const
const MachineInstrBuilder & addFPImm(const ConstantFP *Val) const
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.
mop_range defs()
Returns all explicit operands that are register definitions.
unsigned getOpcode() const
Returns the opcode of this MachineInstr.
LLVM_ABI void addOperand(MachineFunction &MF, const MachineOperand &Op)
Add the specified operand to the instruction.
mop_range uses()
Returns all operands which may be register uses.
LLVM_ABI void removeOperand(unsigned OpNo)
Erase an operand from an instruction, leaving it with one fewer operand than it started with.
const MachineOperand & getOperand(unsigned i) const
LLVM_ABI MachineInstrBundleIterator< MachineInstr > eraseFromParent()
Unlink 'this' from the containing basic block and delete it.
const std::vector< MachineJumpTableEntry > & getJumpTables() const
Flags
Flags values. These may be or'd together.
@ MOVolatile
The memory access is volatile.
@ MOLoad
The memory access reads data.
@ MOStore
The memory access writes data.
MachineOperand class - Representation of each machine instruction operand.
bool isReg() const
isReg - Tests if this is a MO_Register operand.
void setIsKill(bool Val=true)
Register getReg() const
getReg - Returns the register number.
bool isFI() const
isFI - Tests if this is a MO_FrameIndex operand.
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.
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...
void addLiveIn(MCRegister Reg, Register vreg=Register())
addLiveIn - Add the specified register as a live-in.
unsigned getAddressSpace() const
Return the address space for the associated pointer.
MachineMemOperand * getMemOperand() const
Return the unique MachineMemOperand object describing the memory reference performed by operation.
const SDValue & getChain() const
EVT getMemoryVT() const
Return the type of the in-memory value.
static PointerType * getUnqual(Type *ElementType)
This constructs a pointer to an object of the specified type in the default address space (address sp...
Wrapper class representing virtual and physical registers.
Wrapper class for IR location info (IR ordering and DebugLoc) to be passed into SDNode creation funct...
Represents one node in the SelectionDAG.
ArrayRef< SDUse > ops() const
unsigned getOpcode() const
Return the SelectionDAG opcode value for this node.
uint64_t getAsZExtVal() const
Helper method returns the zero-extended integer value of a ConstantSDNode.
const SDValue & getOperand(unsigned Num) const
uint64_t getConstantOperandVal(unsigned Num) const
Helper method returns the integer value of a ConstantSDNode operand.
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.
const SDValue & getOperand(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...
LLVM_ABI bool isKnownNeverLogicalZero(SDValue Op, const APInt &DemandedElts, unsigned Depth=0) const
Test whether the given floating point SDValue (or all elements of it, if it is a vector) is known to ...
SDValue getTargetGlobalAddress(const GlobalValue *GV, const SDLoc &DL, EVT VT, int64_t offset=0, unsigned TargetFlags=0)
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 SDValue getSplatValue(SDValue V, bool LegalTypes=false)
If V is a splat vector, return its scalar source operand by extracting that element from the source v...
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 UnrollVectorOp(SDNode *N, unsigned ResNE=0)
Utility function used by legalize and lowering to "unroll" a vector operation by splitting out the sc...
LLVM_ABI SDValue getConstantFP(double Val, const SDLoc &DL, EVT VT, bool isTarget=false)
Create a ConstantFPSDNode wrapping a constant value.
LLVM_ABI SDValue getMemIntrinsicNode(unsigned Opcode, const SDLoc &dl, SDVTList VTList, ArrayRef< SDValue > Ops, EVT MemVT, MachinePointerInfo PtrInfo, Align Alignment, MachineMemOperand::Flags Flags=MachineMemOperand::MOLoad|MachineMemOperand::MOStore, LocationSize Size=LocationSize::precise(0), const AAMDNodes &AAInfo=AAMDNodes())
Creates a MemIntrinsicNode that may produce a result and takes a list of operands.
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 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)
SDValue getTargetJumpTable(int JTI, EVT VT, unsigned TargetFlags=0)
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.
SDValue getCopyFromReg(SDValue Chain, const SDLoc &dl, Register Reg, EVT VT)
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 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 bool SignBitIsZero(SDValue Op, unsigned Depth=0) const
Return true if the sign bit of Op is known to be zero.
LLVM_ABI SDValue getBasicBlock(MachineBasicBlock *MBB)
LLVM_ABI SDValue getSExtOrTrunc(SDValue Op, const SDLoc &DL, EVT VT)
Convert Op, which must be of integer type, to the integer type VT, by either sign-extending or trunca...
const TargetMachine & getTarget() const
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...
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 bool isKnownNeverNaN(SDValue Op, const APInt &DemandedElts, bool SNaN=false, unsigned Depth=0) const
Test whether the given SDValue (or all elements of it, if it is a vector) is known to never be NaN in...
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
SDValue getPOISON(EVT VT)
Return a POISON node. POISON does not have a useful SDLoc.
SDValue getSplatBuildVector(EVT VT, const SDLoc &DL, SDValue Op)
Return a splat ISD::BUILD_VECTOR node, consisting of Op splatted to all elements.
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
LLVM_ABI SDValue getTargetExternalSymbol(const char *Sym, EVT VT, unsigned TargetFlags=0)
LLVM_ABI SDValue getMCSymbol(MCSymbol *Sym, EVT VT)
SDValue getEntryNode() const
Return the token chain corresponding to the entry of the function.
LLVM_ABI SDValue getVectorShuffle(EVT VT, const SDLoc &dl, SDValue N1, SDValue N2, ArrayRef< int > Mask)
Return an ISD::VECTOR_SHUFFLE node.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
const SDValue & getBasePtr() const
const SDValue & getOffset() const
const SDValue & getValue() const
Represent a constant reference to a string, i.e.
constexpr size_t size() const
Get the string size.
TargetInstrInfo - Interface to description of machine instruction set.
Provides information about what library functions are available for the current target.
void setBooleanVectorContents(BooleanContent Ty)
Specify how the target extends the result of a vector boolean value from a vector of i1 to a wider ty...
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
unsigned MaxLoadsPerMemcmp
Specify maximum number of load instructions per memcmp call.
LegalizeTypeAction
This enum indicates whether a types are legal for a target, and if not, what action should be used to...
void setMaxAtomicSizeInBitsSupported(unsigned SizeInBits)
Set the maximum atomic operation size supported by the backend.
virtual TargetLoweringBase::LegalizeTypeAction getPreferredVectorAction(MVT VT) const
Return the preferred vector type legalization action.
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.
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...
void setMinimumJumpTableEntries(unsigned Val)
Indicate the minimum number of blocks to generate jump tables.
void setPartialReduceMLAAction(unsigned Opc, MVT AccVT, MVT InputVT, LegalizeAction Action)
Indicate how a PARTIAL_REDUCE_U/SMLA node with Acc type AccVT and Input type InputVT should be treate...
void setTruncStoreAction(MVT ValVT, MVT MemVT, LegalizeAction Action)
Indicate that the specified truncating store does not work with the specified type and indicate what ...
@ ZeroOrOneBooleanContent
@ ZeroOrNegativeOneBooleanContent
unsigned MaxLoadsPerMemcmpOptSize
Likewise for functions with the OptSize attribute.
virtual bool isBinOp(unsigned Opcode) const
Return true if the node is a math/logic binary operator.
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 setCondCodeAction(ArrayRef< ISD::CondCode > CCs, MVT VT, LegalizeAction Action)
Indicate that the specified condition code is or isn't supported on the target and indicate what to d...
void setTargetDAGCombine(ArrayRef< ISD::NodeType > NTs)
Targets should invoke this method for each target independent node that they want to provide a custom...
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...
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 ...
void setSchedulingPreference(Sched::Preference Pref)
Specify the target scheduling preference.
bool isOperationLegalOrCustomOrPromote(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...
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 bool isOffsetFoldingLegal(const GlobalAddressSDNode *GA) const
Return true if folding a constant offset with the given GlobalAddress is legal.
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).
Primary interface to the complete machine description for the target machine.
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.
bool isFunctionTy() const
True if this is an instance of FunctionType.
static LLVM_ABI Type * getDoubleTy(LLVMContext &C)
static LLVM_ABI Type * getFloatTy(LLVMContext &C)
A Use represents the edge between a Value definition and its users.
LLVM_ABI const Value * stripPointerCastsAndAliases() const
Strip off pointer casts, all-zero GEPs, address space casts, and aliases.
static std::optional< unsigned > getLocalForStackObject(MachineFunction &MF, int FrameIndex)
bool hasCallIndirectOverlong() const
bool hasReferenceTypes() const
WebAssemblyTargetLowering(const TargetMachine &TM, const WebAssemblySubtarget &STI)
MVT getPointerTy(const DataLayout &DL, uint32_t AS=0) const override
Return the pointer type for the given address space, defaults to the pointer type from the data layou...
MVT getPointerMemTy(const DataLayout &DL, uint32_t AS=0) const override
Return the in-memory pointer type for the given address space, defaults to the pointer type from the ...
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.
@ Swift
Calling convention for Swift.
@ PreserveMost
Used for runtime calls that preserves most registers.
@ CXX_FAST_TLS
Used for access functions.
@ WASM_EmscriptenInvoke
For emscripten __invoke_* functions.
@ Cold
Attempts to make code in the caller as efficient as possible under the assumption that the call is no...
@ PreserveAll
Used for runtime calls that preserves (almost) all registers.
@ Fast
Attempts to make calls as fast as possible (e.g.
@ SwiftTail
This follows the Swift calling convention in how arguments are passed but guarantees tail calls will ...
@ C
The default llvm calling convention, compatible with C.
@ SETCC
SetCC operator - This evaluates to a true value iff the condition is true.
@ 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.
@ PARTIAL_REDUCE_SMLA
PARTIAL_REDUCE_[U|S]MLA(Accumulator, Input1, Input2) The partial reduction nodes sign or zero extend ...
@ 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.
@ LOAD
LOAD and STORE have token chains as their first operand, then the same operands as an LLVM load/store...
@ FMA
FMA - Perform a * b + c with no intermediate rounding step.
@ INTRINSIC_VOID
OUTCHAIN = INTRINSIC_VOID(INCHAIN, INTRINSICID, arg1, arg2, ...) This node represents a target intrin...
@ 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 ...
@ ABS
ABS - Determine the unsigned absolute value of a signed integer value of the same bitwidth.
@ SIGN_EXTEND_VECTOR_INREG
SIGN_EXTEND_VECTOR_INREG(Vector) - This operator represents an in-register sign-extension of the low ...
@ SDIVREM
SDIVREM/UDIVREM - Divide two integers and produce both a quotient and remainder result.
@ FP16_TO_FP
FP16_TO_FP, FP_TO_FP16 - These operators are used to perform promotions and truncation for half-preci...
@ FMULADD
FMULADD - Performs a * b + c, with, or without, intermediate rounding.
@ 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.
@ BUILTIN_OP_END
BUILTIN_OP_END - This must be the last enum value in this list.
@ SIGN_EXTEND
Conversion operators.
@ SCALAR_TO_VECTOR
SCALAR_TO_VECTOR(VAL) - This represents the operation of loading a scalar value into element 0 of the...
@ FSINCOS
FSINCOS - Compute both fsin and fcos as a single operation.
@ BR_CC
BR_CC - Conditional branch.
@ BRIND
BRIND - Indirect branch.
@ BR_JT
BR_JT - Jumptable branch.
@ SSUBSAT
RESULT = [US]SUBSAT(LHS, RHS) - Perform saturation subtraction on 2 integers with the same bit width ...
@ 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,...
@ MULHU
MULHU/MULHS - Multiply high - Multiply two integers of type iN, producing an unsigned/signed value of...
@ SHL
Shift and rotation operations.
@ VECTOR_SHUFFLE
VECTOR_SHUFFLE(VEC1, VEC2) - Returns a vector, of the same type as VEC1/VEC2.
@ EXTRACT_SUBVECTOR
EXTRACT_SUBVECTOR(VECTOR, IDX) - Returns a subvector from VECTOR.
@ EXTRACT_VECTOR_ELT
EXTRACT_VECTOR_ELT(VECTOR, IDX) - Returns a single element from VECTOR identified by the (potentially...
@ CopyToReg
CopyToReg - This node has three operands: a chain, a register number to set to this value,...
@ ZERO_EXTEND
ZERO_EXTEND - Used for integer types, zeroing the new bits.
@ DEBUGTRAP
DEBUGTRAP - Trap intended to get the attention of a debugger.
@ SELECT_CC
Select with condition operator - This selects between a true value and a false value (ops #2 and #3) ...
@ FMINNUM
FMINNUM/FMAXNUM - Perform floating-point minimum maximum on two values, following IEEE-754 definition...
@ DYNAMIC_STACKALLOC
DYNAMIC_STACKALLOC - Allocate some number of bytes on the stack aligned to a specified boundary.
@ ANY_EXTEND_VECTOR_INREG
ANY_EXTEND_VECTOR_INREG(Vector) - This operator represents an in-register any-extension of the low la...
@ SIGN_EXTEND_INREG
SIGN_EXTEND_INREG - This operator atomically performs a SHL/SRA pair to sign extend a small value in ...
@ SMIN
[US]{MIN/MAX} - Binary minimum or maximum of signed or unsigned integers.
@ FP_EXTEND
X = FP_EXTEND(Y) - Extend a smaller FP type into a larger FP type.
@ FRAMEADDR
FRAMEADDR, RETURNADDR - These nodes represent llvm.frameaddress and llvm.returnaddress on the DAG.
@ 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.
@ TargetConstant
TargetConstant* - Like Constant*, but the DAG does not do any folding, simplification,...
@ 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...
@ ADDE
Carry-using nodes for multiple precision addition and subtraction.
@ 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.
@ FP_ROUND
X = FP_ROUND(Y, TRUNC) - Rounding 'Y' from a larger floating point type down to the precision of the ...
@ CLEAR_CACHE
llvm.clear_cache intrinsic Operands: Input Chain, Start Addres, End Address Outputs: Output Chain
@ ZERO_EXTEND_VECTOR_INREG
ZERO_EXTEND_VECTOR_INREG(Vector) - This operator represents an in-register zero-extension of the low ...
@ FP_TO_SINT_SAT
FP_TO_[US]INT_SAT - Convert floating point value in operand 0 to a signed or unsigned scalar integer ...
@ TRUNCATE
TRUNCATE - Completely drop the high bits.
@ VAARG
VAARG - VAARG has four operands: an input chain, a pointer, a SRCVALUE, and the alignment.
@ SHL_PARTS
SHL_PARTS/SRA_PARTS/SRL_PARTS - These operators are used for expanded integer shift operations.
@ 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)...
@ FMINIMUMNUM
FMINIMUMNUM/FMAXIMUMNUM - minimumnum/maximumnum that is same with FMINNUM_IEEE and FMAXNUM_IEEE besid...
@ INTRINSIC_W_CHAIN
RESULT,OUTCHAIN = INTRINSIC_W_CHAIN(INCHAIN, INTRINSICID, arg1, ...) This node represents a target in...
@ BUILD_VECTOR
BUILD_VECTOR(ELT0, ELT1, ELT2, ELT3,...) - Return a fixed-width vector with the specified,...
LLVM_ABI bool isConstantSplatVector(const SDNode *N, APInt &SplatValue)
Node predicates.
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.
OperandFlags
These are flags set on operands, but should be considered private, all access should go through the M...
auto m_Value()
Match an arbitrary value and ignore it.
CastOperator_match< OpTy, Instruction::BitCast > m_BitCast(const OpTy &Op)
Matches BitCast.
is_zero m_Zero()
Match any null constant or a vector with all elements equal to 0.
bool sd_match(SDNode *N, const SelectionDAG *DAG, Pattern &&P)
CondCode_match m_SpecificCondCode(ISD::CondCode CC)
Match a conditional code SDNode with a specific ISD::CondCode.
CondCode_match m_CondCode()
Match any conditional code SDNode.
TernaryOpc_match< T0_P, T1_P, T2_P, true, false > m_c_SetCC(const T0_P &LHS, const T1_P &RHS, const T2_P &CC)
MCSymbolWasm * getOrCreateFunctionTableSymbol(MCContext &Ctx, const WebAssemblySubtarget *Subtarget)
Returns the __indirect_function_table, for use in call_indirect and in function bitcasts.
@ WASM_ADDRESS_SPACE_EXTERNREF
@ WASM_ADDRESS_SPACE_FUNCREF
bool isWebAssemblyFuncrefType(const Type *Ty)
Return true if this is a WebAssembly Funcref Type.
bool isWebAssemblyTableType(const Type *Ty)
Return true if the table represents a WebAssembly table type.
MCSymbolWasm * getOrCreateFuncrefCallTableSymbol(MCContext &Ctx, const WebAssemblySubtarget *Subtarget)
Returns the __funcref_call_table, for use in funcref calls when lowered to table.set + call_indirect.
bool isValidAddressSpace(unsigned AS)
FastISel * createFastISel(FunctionLoweringInfo &funcInfo, const TargetLibraryInfo *libInfo, const LibcallLoweringInfo *libcallLowering)
bool canLowerReturn(size_t ResultSize, const WebAssemblySubtarget *Subtarget)
Returns true if the function's return value(s) can be lowered directly, i.e., not indirectly via a po...
MachineSDNode * getTLSBase(SelectionDAG &DAG, const SDLoc &DL, const WebAssemblySubtarget *Subtarget, const SDValue Chain=SDValue())
bool isWasmVarAddressSpace(unsigned AS)
NodeAddr< NodeBase * > Node
This is an optimization pass for GlobalISel generic memory operations.
auto drop_begin(T &&RangeOrContainer, size_t N=1)
Return a range covering RangeOrContainer with the first N elements excluded.
unsigned Log2_32_Ceil(uint32_t Value)
Return the ceil log base 2 of the specified value, 32 if the value is zero.
FunctionAddr VTableAddr Value
void computeSignatureVTs(const FunctionType *Ty, const Function *TargetFunc, const Function &ContextFunc, const TargetMachine &TM, SmallVectorImpl< MVT > &Params, SmallVectorImpl< MVT > &Results)
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.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
LLVM_ABI bool isNullConstant(SDValue V)
Returns true if V is a constant integer zero.
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.
constexpr bool isPowerOf2_32(uint32_t Value)
Return true if the argument is a power of two > 0.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
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...
DWARFExpression::Operation Op
auto max_element(R &&Range)
Provide wrappers to std::max_element which take ranges instead of having to pass begin/end explicitly...
constexpr unsigned BitWidth
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
void computeLegalValueVTs(const WebAssemblyTargetLowering &TLI, LLVMContext &Ctx, const DataLayout &DL, Type *Ty, SmallVectorImpl< MVT > &ValueVTs)
constexpr uint64_t NextPowerOf2(uint64_t A)
Returns the next power of two (in 64-bits) that is strictly greater than A.
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.
unsigned WideCombineOpcode
This struct is a compact representation of a valid (non-zero power of two) alignment.
EVT changeVectorElementTypeToInteger() const
Return a vector with the same number of elements as this vector, but with the element type converted ...
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.
bool isFloatingPoint() const
Return true if this is a FP or a vector FP type.
TypeSize getSizeInBits() const
Return the size of the specified value type in bits.
bool isByteSized() const
Return true if the bit size is a multiple of 8.
uint64_t getScalarSizeInBits() const
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.
bool is128BitVector() const
Return true if this is a 128-bit vector type.
static EVT getIntegerVT(LLVMContext &Context, unsigned BitWidth)
Returns the EVT that represents an integer with the given number of bits.
uint64_t getFixedSizeInBits() const
Return the size of the specified fixed width value type in bits.
EVT widenIntegerVectorElementType(LLVMContext &Context) const
Return a VT for an integer vector type with the size of the elements doubled.
bool isFixedLengthVector() const
bool isFixedLengthVectorOf(EVT EltVT) const
Return true if this is a fixed length vector with matching element type.
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 is256BitVector() const
Return true if this is a 256-bit vector type.
LLVM_ABI Type * getTypeForEVT(LLVMContext &Context) const
This method returns an LLVM type corresponding to the specified EVT.
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.
EVT getHalfNumVectorElementsVT(LLVMContext &Context) const
bool isInConsecutiveRegs() const
Align getNonZeroOrigAlign() const
bool isSwiftError() const
unsigned getByValSize() const
bool isInConsecutiveRegsLast() const
bool isSwiftAsync() const
Align getNonZeroByValAlign() const
unsigned getBitWidth() const
Get the bit width of this value.
static LLVM_ABI MachinePointerInfo getFixedStack(MachineFunction &MF, int FI, int64_t Offset=0)
Return a MachinePointerInfo record that refers to the specified FrameIndex.
These are IR-level optimization flags that may be propagated to SDNodes.
bool isBeforeLegalize() const
This structure is used to pass arguments to makeLibCall function.