10b57cec5SDimitry Andric //===-- WebAssemblyFastISel.cpp - WebAssembly FastISel implementation -----===// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric /// 90b57cec5SDimitry Andric /// \file 100b57cec5SDimitry Andric /// This file defines the WebAssembly-specific support for the FastISel 110b57cec5SDimitry Andric /// class. Some of the target-specific code is generated by tablegen in the file 120b57cec5SDimitry Andric /// WebAssemblyGenFastISel.inc, which is #included here. 130b57cec5SDimitry Andric /// 140b57cec5SDimitry Andric /// TODO: kill flags 150b57cec5SDimitry Andric /// 160b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 170b57cec5SDimitry Andric 180b57cec5SDimitry Andric #include "MCTargetDesc/WebAssemblyMCTargetDesc.h" 191fd87a68SDimitry Andric #include "Utils/WebAssemblyTypeUtilities.h" 20fe6060f1SDimitry Andric #include "Utils/WebAssemblyUtilities.h" 210b57cec5SDimitry Andric #include "WebAssembly.h" 220b57cec5SDimitry Andric #include "WebAssemblyMachineFunctionInfo.h" 230b57cec5SDimitry Andric #include "WebAssemblySubtarget.h" 240b57cec5SDimitry Andric #include "WebAssemblyTargetMachine.h" 250b57cec5SDimitry Andric #include "llvm/Analysis/BranchProbabilityInfo.h" 260b57cec5SDimitry Andric #include "llvm/CodeGen/FastISel.h" 270b57cec5SDimitry Andric #include "llvm/CodeGen/FunctionLoweringInfo.h" 280b57cec5SDimitry Andric #include "llvm/CodeGen/MachineConstantPool.h" 290b57cec5SDimitry Andric #include "llvm/CodeGen/MachineFrameInfo.h" 300b57cec5SDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h" 31e8d8bef9SDimitry Andric #include "llvm/CodeGen/MachineModuleInfo.h" 320b57cec5SDimitry Andric #include "llvm/CodeGen/MachineRegisterInfo.h" 330b57cec5SDimitry Andric #include "llvm/IR/DataLayout.h" 340b57cec5SDimitry Andric #include "llvm/IR/DerivedTypes.h" 350b57cec5SDimitry Andric #include "llvm/IR/Function.h" 360b57cec5SDimitry Andric #include "llvm/IR/GetElementPtrTypeIterator.h" 370b57cec5SDimitry Andric #include "llvm/IR/GlobalAlias.h" 380b57cec5SDimitry Andric #include "llvm/IR/GlobalVariable.h" 390b57cec5SDimitry Andric #include "llvm/IR/Instructions.h" 400b57cec5SDimitry Andric #include "llvm/IR/IntrinsicInst.h" 410b57cec5SDimitry Andric #include "llvm/IR/Operator.h" 420b57cec5SDimitry Andric #include "llvm/IR/PatternMatch.h" 430b57cec5SDimitry Andric 440b57cec5SDimitry Andric using namespace llvm; 450b57cec5SDimitry Andric using namespace PatternMatch; 460b57cec5SDimitry Andric 470b57cec5SDimitry Andric #define DEBUG_TYPE "wasm-fastisel" 480b57cec5SDimitry Andric 490b57cec5SDimitry Andric namespace { 500b57cec5SDimitry Andric 510b57cec5SDimitry Andric class WebAssemblyFastISel final : public FastISel { 520b57cec5SDimitry Andric // All possible address modes. 530b57cec5SDimitry Andric class Address { 540b57cec5SDimitry Andric public: 550b57cec5SDimitry Andric using BaseKind = enum { RegBase, FrameIndexBase }; 560b57cec5SDimitry Andric 570b57cec5SDimitry Andric private: 580b57cec5SDimitry Andric BaseKind Kind = RegBase; 590b57cec5SDimitry Andric union { 600b57cec5SDimitry Andric unsigned Reg; 610b57cec5SDimitry Andric int FI; 620b57cec5SDimitry Andric } Base; 630b57cec5SDimitry Andric 64eaeb601bSDimitry Andric // Whether the base has been determined yet 65eaeb601bSDimitry Andric bool IsBaseSet = false; 66eaeb601bSDimitry Andric 670b57cec5SDimitry Andric int64_t Offset = 0; 680b57cec5SDimitry Andric 690b57cec5SDimitry Andric const GlobalValue *GV = nullptr; 700b57cec5SDimitry Andric 710b57cec5SDimitry Andric public: 720b57cec5SDimitry Andric // Innocuous defaults for our address. 730b57cec5SDimitry Andric Address() { Base.Reg = 0; } 740b57cec5SDimitry Andric void setKind(BaseKind K) { 750b57cec5SDimitry Andric assert(!isSet() && "Can't change kind with non-zero base"); 760b57cec5SDimitry Andric Kind = K; 770b57cec5SDimitry Andric } 780b57cec5SDimitry Andric BaseKind getKind() const { return Kind; } 790b57cec5SDimitry Andric bool isRegBase() const { return Kind == RegBase; } 800b57cec5SDimitry Andric bool isFIBase() const { return Kind == FrameIndexBase; } 810b57cec5SDimitry Andric void setReg(unsigned Reg) { 820b57cec5SDimitry Andric assert(isRegBase() && "Invalid base register access!"); 83eaeb601bSDimitry Andric assert(!IsBaseSet && "Base cannot be reset"); 840b57cec5SDimitry Andric Base.Reg = Reg; 85eaeb601bSDimitry Andric IsBaseSet = true; 860b57cec5SDimitry Andric } 870b57cec5SDimitry Andric unsigned getReg() const { 880b57cec5SDimitry Andric assert(isRegBase() && "Invalid base register access!"); 890b57cec5SDimitry Andric return Base.Reg; 900b57cec5SDimitry Andric } 910b57cec5SDimitry Andric void setFI(unsigned FI) { 920b57cec5SDimitry Andric assert(isFIBase() && "Invalid base frame index access!"); 93eaeb601bSDimitry Andric assert(!IsBaseSet && "Base cannot be reset"); 940b57cec5SDimitry Andric Base.FI = FI; 95eaeb601bSDimitry Andric IsBaseSet = true; 960b57cec5SDimitry Andric } 970b57cec5SDimitry Andric unsigned getFI() const { 980b57cec5SDimitry Andric assert(isFIBase() && "Invalid base frame index access!"); 990b57cec5SDimitry Andric return Base.FI; 1000b57cec5SDimitry Andric } 1010b57cec5SDimitry Andric 1020b57cec5SDimitry Andric void setOffset(int64_t NewOffset) { 1030b57cec5SDimitry Andric assert(NewOffset >= 0 && "Offsets must be non-negative"); 1040b57cec5SDimitry Andric Offset = NewOffset; 1050b57cec5SDimitry Andric } 1060b57cec5SDimitry Andric int64_t getOffset() const { return Offset; } 1070b57cec5SDimitry Andric void setGlobalValue(const GlobalValue *G) { GV = G; } 1080b57cec5SDimitry Andric const GlobalValue *getGlobalValue() const { return GV; } 109eaeb601bSDimitry Andric bool isSet() const { return IsBaseSet; } 1100b57cec5SDimitry Andric }; 1110b57cec5SDimitry Andric 1120b57cec5SDimitry Andric /// Keep a pointer to the WebAssemblySubtarget around so that we can make the 1130b57cec5SDimitry Andric /// right decision when generating code for different targets. 1140b57cec5SDimitry Andric const WebAssemblySubtarget *Subtarget; 1150b57cec5SDimitry Andric LLVMContext *Context; 1160b57cec5SDimitry Andric 1170b57cec5SDimitry Andric private: 1180b57cec5SDimitry Andric // Utility helper routines 1190b57cec5SDimitry Andric MVT::SimpleValueType getSimpleType(Type *Ty) { 1200b57cec5SDimitry Andric EVT VT = TLI.getValueType(DL, Ty, /*AllowUnknown=*/true); 1210b57cec5SDimitry Andric return VT.isSimple() ? VT.getSimpleVT().SimpleTy 1220b57cec5SDimitry Andric : MVT::INVALID_SIMPLE_VALUE_TYPE; 1230b57cec5SDimitry Andric } 1240b57cec5SDimitry Andric MVT::SimpleValueType getLegalType(MVT::SimpleValueType VT) { 1250b57cec5SDimitry Andric switch (VT) { 1260b57cec5SDimitry Andric case MVT::i1: 1270b57cec5SDimitry Andric case MVT::i8: 1280b57cec5SDimitry Andric case MVT::i16: 1290b57cec5SDimitry Andric return MVT::i32; 1300b57cec5SDimitry Andric case MVT::i32: 1310b57cec5SDimitry Andric case MVT::i64: 1320b57cec5SDimitry Andric case MVT::f32: 1330b57cec5SDimitry Andric case MVT::f64: 134fe6060f1SDimitry Andric return VT; 135e8d8bef9SDimitry Andric case MVT::funcref: 136e8d8bef9SDimitry Andric case MVT::externref: 137fe6060f1SDimitry Andric if (Subtarget->hasReferenceTypes()) 1380b57cec5SDimitry Andric return VT; 139fe6060f1SDimitry Andric break; 1400b57cec5SDimitry Andric case MVT::f16: 1410b57cec5SDimitry Andric return MVT::f32; 1420b57cec5SDimitry Andric case MVT::v16i8: 1430b57cec5SDimitry Andric case MVT::v8i16: 1440b57cec5SDimitry Andric case MVT::v4i32: 1450b57cec5SDimitry Andric case MVT::v4f32: 1460b57cec5SDimitry Andric case MVT::v2i64: 1470b57cec5SDimitry Andric case MVT::v2f64: 148fe6060f1SDimitry Andric if (Subtarget->hasSIMD128()) 1490b57cec5SDimitry Andric return VT; 1500b57cec5SDimitry Andric break; 1510b57cec5SDimitry Andric default: 1520b57cec5SDimitry Andric break; 1530b57cec5SDimitry Andric } 1540b57cec5SDimitry Andric return MVT::INVALID_SIMPLE_VALUE_TYPE; 1550b57cec5SDimitry Andric } 1560b57cec5SDimitry Andric bool computeAddress(const Value *Obj, Address &Addr); 1570b57cec5SDimitry Andric void materializeLoadStoreOperands(Address &Addr); 1580b57cec5SDimitry Andric void addLoadStoreOperands(const Address &Addr, const MachineInstrBuilder &MIB, 1590b57cec5SDimitry Andric MachineMemOperand *MMO); 1600b57cec5SDimitry Andric unsigned maskI1Value(unsigned Reg, const Value *V); 16169ade1e0SDimitry Andric unsigned getRegForI1Value(const Value *V, const BasicBlock *BB, bool &Not); 1620b57cec5SDimitry Andric unsigned zeroExtendToI32(unsigned Reg, const Value *V, 1630b57cec5SDimitry Andric MVT::SimpleValueType From); 1640b57cec5SDimitry Andric unsigned signExtendToI32(unsigned Reg, const Value *V, 1650b57cec5SDimitry Andric MVT::SimpleValueType From); 1660b57cec5SDimitry Andric unsigned zeroExtend(unsigned Reg, const Value *V, MVT::SimpleValueType From, 1670b57cec5SDimitry Andric MVT::SimpleValueType To); 1680b57cec5SDimitry Andric unsigned signExtend(unsigned Reg, const Value *V, MVT::SimpleValueType From, 1690b57cec5SDimitry Andric MVT::SimpleValueType To); 1700b57cec5SDimitry Andric unsigned getRegForUnsignedValue(const Value *V); 1710b57cec5SDimitry Andric unsigned getRegForSignedValue(const Value *V); 1720b57cec5SDimitry Andric unsigned getRegForPromotedValue(const Value *V, bool IsSigned); 1730b57cec5SDimitry Andric unsigned notValue(unsigned Reg); 1740b57cec5SDimitry Andric unsigned copyValue(unsigned Reg); 1750b57cec5SDimitry Andric 1760b57cec5SDimitry Andric // Backend specific FastISel code. 1770b57cec5SDimitry Andric unsigned fastMaterializeAlloca(const AllocaInst *AI) override; 1780b57cec5SDimitry Andric unsigned fastMaterializeConstant(const Constant *C) override; 1790b57cec5SDimitry Andric bool fastLowerArguments() override; 1800b57cec5SDimitry Andric 1810b57cec5SDimitry Andric // Selection routines. 1820b57cec5SDimitry Andric bool selectCall(const Instruction *I); 1830b57cec5SDimitry Andric bool selectSelect(const Instruction *I); 1840b57cec5SDimitry Andric bool selectTrunc(const Instruction *I); 1850b57cec5SDimitry Andric bool selectZExt(const Instruction *I); 1860b57cec5SDimitry Andric bool selectSExt(const Instruction *I); 1870b57cec5SDimitry Andric bool selectICmp(const Instruction *I); 1880b57cec5SDimitry Andric bool selectFCmp(const Instruction *I); 1890b57cec5SDimitry Andric bool selectBitCast(const Instruction *I); 1900b57cec5SDimitry Andric bool selectLoad(const Instruction *I); 1910b57cec5SDimitry Andric bool selectStore(const Instruction *I); 1920b57cec5SDimitry Andric bool selectBr(const Instruction *I); 1930b57cec5SDimitry Andric bool selectRet(const Instruction *I); 1940b57cec5SDimitry Andric bool selectUnreachable(const Instruction *I); 1950b57cec5SDimitry Andric 1960b57cec5SDimitry Andric public: 1970b57cec5SDimitry Andric // Backend specific FastISel code. 1980b57cec5SDimitry Andric WebAssemblyFastISel(FunctionLoweringInfo &FuncInfo, 1990b57cec5SDimitry Andric const TargetLibraryInfo *LibInfo) 2000b57cec5SDimitry Andric : FastISel(FuncInfo, LibInfo, /*SkipTargetIndependentISel=*/true) { 2010b57cec5SDimitry Andric Subtarget = &FuncInfo.MF->getSubtarget<WebAssemblySubtarget>(); 2020b57cec5SDimitry Andric Context = &FuncInfo.Fn->getContext(); 2030b57cec5SDimitry Andric } 2040b57cec5SDimitry Andric 2050b57cec5SDimitry Andric bool fastSelectInstruction(const Instruction *I) override; 2060b57cec5SDimitry Andric 2070b57cec5SDimitry Andric #include "WebAssemblyGenFastISel.inc" 2080b57cec5SDimitry Andric }; 2090b57cec5SDimitry Andric 2100b57cec5SDimitry Andric } // end anonymous namespace 2110b57cec5SDimitry Andric 2120b57cec5SDimitry Andric bool WebAssemblyFastISel::computeAddress(const Value *Obj, Address &Addr) { 2130b57cec5SDimitry Andric const User *U = nullptr; 2140b57cec5SDimitry Andric unsigned Opcode = Instruction::UserOp1; 2150b57cec5SDimitry Andric if (const auto *I = dyn_cast<Instruction>(Obj)) { 2160b57cec5SDimitry Andric // Don't walk into other basic blocks unless the object is an alloca from 2170b57cec5SDimitry Andric // another block, otherwise it may not have a virtual register assigned. 2180b57cec5SDimitry Andric if (FuncInfo.StaticAllocaMap.count(static_cast<const AllocaInst *>(Obj)) || 2190b57cec5SDimitry Andric FuncInfo.MBBMap[I->getParent()] == FuncInfo.MBB) { 2200b57cec5SDimitry Andric Opcode = I->getOpcode(); 2210b57cec5SDimitry Andric U = I; 2220b57cec5SDimitry Andric } 2230b57cec5SDimitry Andric } else if (const auto *C = dyn_cast<ConstantExpr>(Obj)) { 2240b57cec5SDimitry Andric Opcode = C->getOpcode(); 2250b57cec5SDimitry Andric U = C; 2260b57cec5SDimitry Andric } 2270b57cec5SDimitry Andric 2280b57cec5SDimitry Andric if (auto *Ty = dyn_cast<PointerType>(Obj->getType())) 2290b57cec5SDimitry Andric if (Ty->getAddressSpace() > 255) 2300b57cec5SDimitry Andric // Fast instruction selection doesn't support the special 2310b57cec5SDimitry Andric // address spaces. 2320b57cec5SDimitry Andric return false; 2330b57cec5SDimitry Andric 2340b57cec5SDimitry Andric if (const auto *GV = dyn_cast<GlobalValue>(Obj)) { 2350b57cec5SDimitry Andric if (TLI.isPositionIndependent()) 2360b57cec5SDimitry Andric return false; 2370b57cec5SDimitry Andric if (Addr.getGlobalValue()) 2380b57cec5SDimitry Andric return false; 2390b57cec5SDimitry Andric if (GV->isThreadLocal()) 2400b57cec5SDimitry Andric return false; 2410b57cec5SDimitry Andric Addr.setGlobalValue(GV); 2420b57cec5SDimitry Andric return true; 2430b57cec5SDimitry Andric } 2440b57cec5SDimitry Andric 2450b57cec5SDimitry Andric switch (Opcode) { 2460b57cec5SDimitry Andric default: 2470b57cec5SDimitry Andric break; 2480b57cec5SDimitry Andric case Instruction::BitCast: { 2490b57cec5SDimitry Andric // Look through bitcasts. 2500b57cec5SDimitry Andric return computeAddress(U->getOperand(0), Addr); 2510b57cec5SDimitry Andric } 2520b57cec5SDimitry Andric case Instruction::IntToPtr: { 2530b57cec5SDimitry Andric // Look past no-op inttoptrs. 2540b57cec5SDimitry Andric if (TLI.getValueType(DL, U->getOperand(0)->getType()) == 2550b57cec5SDimitry Andric TLI.getPointerTy(DL)) 2560b57cec5SDimitry Andric return computeAddress(U->getOperand(0), Addr); 2570b57cec5SDimitry Andric break; 2580b57cec5SDimitry Andric } 2590b57cec5SDimitry Andric case Instruction::PtrToInt: { 2600b57cec5SDimitry Andric // Look past no-op ptrtoints. 2610b57cec5SDimitry Andric if (TLI.getValueType(DL, U->getType()) == TLI.getPointerTy(DL)) 2620b57cec5SDimitry Andric return computeAddress(U->getOperand(0), Addr); 2630b57cec5SDimitry Andric break; 2640b57cec5SDimitry Andric } 2650b57cec5SDimitry Andric case Instruction::GetElementPtr: { 2660b57cec5SDimitry Andric Address SavedAddr = Addr; 2670b57cec5SDimitry Andric uint64_t TmpOffset = Addr.getOffset(); 2680b57cec5SDimitry Andric // Non-inbounds geps can wrap; wasm's offsets can't. 2690b57cec5SDimitry Andric if (!cast<GEPOperator>(U)->isInBounds()) 2700b57cec5SDimitry Andric goto unsupported_gep; 2710b57cec5SDimitry Andric // Iterate through the GEP folding the constants into offsets where 2720b57cec5SDimitry Andric // we can. 2730b57cec5SDimitry Andric for (gep_type_iterator GTI = gep_type_begin(U), E = gep_type_end(U); 2740b57cec5SDimitry Andric GTI != E; ++GTI) { 2750b57cec5SDimitry Andric const Value *Op = GTI.getOperand(); 2760b57cec5SDimitry Andric if (StructType *STy = GTI.getStructTypeOrNull()) { 2770b57cec5SDimitry Andric const StructLayout *SL = DL.getStructLayout(STy); 2780b57cec5SDimitry Andric unsigned Idx = cast<ConstantInt>(Op)->getZExtValue(); 2790b57cec5SDimitry Andric TmpOffset += SL->getElementOffset(Idx); 2800b57cec5SDimitry Andric } else { 2810b57cec5SDimitry Andric uint64_t S = DL.getTypeAllocSize(GTI.getIndexedType()); 2820b57cec5SDimitry Andric for (;;) { 2830b57cec5SDimitry Andric if (const auto *CI = dyn_cast<ConstantInt>(Op)) { 2840b57cec5SDimitry Andric // Constant-offset addressing. 2850b57cec5SDimitry Andric TmpOffset += CI->getSExtValue() * S; 2860b57cec5SDimitry Andric break; 2870b57cec5SDimitry Andric } 2880b57cec5SDimitry Andric if (S == 1 && Addr.isRegBase() && Addr.getReg() == 0) { 2890b57cec5SDimitry Andric // An unscaled add of a register. Set it as the new base. 29004eeddc0SDimitry Andric Register Reg = getRegForValue(Op); 2910b57cec5SDimitry Andric if (Reg == 0) 2920b57cec5SDimitry Andric return false; 2930b57cec5SDimitry Andric Addr.setReg(Reg); 2940b57cec5SDimitry Andric break; 2950b57cec5SDimitry Andric } 2960b57cec5SDimitry Andric if (canFoldAddIntoGEP(U, Op)) { 2970b57cec5SDimitry Andric // A compatible add with a constant operand. Fold the constant. 2980b57cec5SDimitry Andric auto *CI = cast<ConstantInt>(cast<AddOperator>(Op)->getOperand(1)); 2990b57cec5SDimitry Andric TmpOffset += CI->getSExtValue() * S; 3000b57cec5SDimitry Andric // Iterate on the other operand. 3010b57cec5SDimitry Andric Op = cast<AddOperator>(Op)->getOperand(0); 3020b57cec5SDimitry Andric continue; 3030b57cec5SDimitry Andric } 3040b57cec5SDimitry Andric // Unsupported 3050b57cec5SDimitry Andric goto unsupported_gep; 3060b57cec5SDimitry Andric } 3070b57cec5SDimitry Andric } 3080b57cec5SDimitry Andric } 3090b57cec5SDimitry Andric // Don't fold in negative offsets. 3100b57cec5SDimitry Andric if (int64_t(TmpOffset) >= 0) { 3110b57cec5SDimitry Andric // Try to grab the base operand now. 3120b57cec5SDimitry Andric Addr.setOffset(TmpOffset); 3130b57cec5SDimitry Andric if (computeAddress(U->getOperand(0), Addr)) 3140b57cec5SDimitry Andric return true; 3150b57cec5SDimitry Andric } 3160b57cec5SDimitry Andric // We failed, restore everything and try the other options. 3170b57cec5SDimitry Andric Addr = SavedAddr; 3180b57cec5SDimitry Andric unsupported_gep: 3190b57cec5SDimitry Andric break; 3200b57cec5SDimitry Andric } 3210b57cec5SDimitry Andric case Instruction::Alloca: { 3220b57cec5SDimitry Andric const auto *AI = cast<AllocaInst>(Obj); 3230b57cec5SDimitry Andric DenseMap<const AllocaInst *, int>::iterator SI = 3240b57cec5SDimitry Andric FuncInfo.StaticAllocaMap.find(AI); 3250b57cec5SDimitry Andric if (SI != FuncInfo.StaticAllocaMap.end()) { 3260b57cec5SDimitry Andric if (Addr.isSet()) { 3270b57cec5SDimitry Andric return false; 3280b57cec5SDimitry Andric } 3290b57cec5SDimitry Andric Addr.setKind(Address::FrameIndexBase); 3300b57cec5SDimitry Andric Addr.setFI(SI->second); 3310b57cec5SDimitry Andric return true; 3320b57cec5SDimitry Andric } 3330b57cec5SDimitry Andric break; 3340b57cec5SDimitry Andric } 3350b57cec5SDimitry Andric case Instruction::Add: { 3360b57cec5SDimitry Andric // Adds of constants are common and easy enough. 3370b57cec5SDimitry Andric const Value *LHS = U->getOperand(0); 3380b57cec5SDimitry Andric const Value *RHS = U->getOperand(1); 3390b57cec5SDimitry Andric 3400b57cec5SDimitry Andric if (isa<ConstantInt>(LHS)) 3410b57cec5SDimitry Andric std::swap(LHS, RHS); 3420b57cec5SDimitry Andric 3430b57cec5SDimitry Andric if (const auto *CI = dyn_cast<ConstantInt>(RHS)) { 3440b57cec5SDimitry Andric uint64_t TmpOffset = Addr.getOffset() + CI->getSExtValue(); 3450b57cec5SDimitry Andric if (int64_t(TmpOffset) >= 0) { 3460b57cec5SDimitry Andric Addr.setOffset(TmpOffset); 3470b57cec5SDimitry Andric return computeAddress(LHS, Addr); 3480b57cec5SDimitry Andric } 3490b57cec5SDimitry Andric } 3500b57cec5SDimitry Andric 3510b57cec5SDimitry Andric Address Backup = Addr; 3520b57cec5SDimitry Andric if (computeAddress(LHS, Addr) && computeAddress(RHS, Addr)) 3530b57cec5SDimitry Andric return true; 3540b57cec5SDimitry Andric Addr = Backup; 3550b57cec5SDimitry Andric 3560b57cec5SDimitry Andric break; 3570b57cec5SDimitry Andric } 3580b57cec5SDimitry Andric case Instruction::Sub: { 3590b57cec5SDimitry Andric // Subs of constants are common and easy enough. 3600b57cec5SDimitry Andric const Value *LHS = U->getOperand(0); 3610b57cec5SDimitry Andric const Value *RHS = U->getOperand(1); 3620b57cec5SDimitry Andric 3630b57cec5SDimitry Andric if (const auto *CI = dyn_cast<ConstantInt>(RHS)) { 3640b57cec5SDimitry Andric int64_t TmpOffset = Addr.getOffset() - CI->getSExtValue(); 3650b57cec5SDimitry Andric if (TmpOffset >= 0) { 3660b57cec5SDimitry Andric Addr.setOffset(TmpOffset); 3670b57cec5SDimitry Andric return computeAddress(LHS, Addr); 3680b57cec5SDimitry Andric } 3690b57cec5SDimitry Andric } 3700b57cec5SDimitry Andric break; 3710b57cec5SDimitry Andric } 3720b57cec5SDimitry Andric } 3730b57cec5SDimitry Andric if (Addr.isSet()) { 3740b57cec5SDimitry Andric return false; 3750b57cec5SDimitry Andric } 37604eeddc0SDimitry Andric Register Reg = getRegForValue(Obj); 3770b57cec5SDimitry Andric if (Reg == 0) 3780b57cec5SDimitry Andric return false; 3790b57cec5SDimitry Andric Addr.setReg(Reg); 3800b57cec5SDimitry Andric return Addr.getReg() != 0; 3810b57cec5SDimitry Andric } 3820b57cec5SDimitry Andric 3830b57cec5SDimitry Andric void WebAssemblyFastISel::materializeLoadStoreOperands(Address &Addr) { 3840b57cec5SDimitry Andric if (Addr.isRegBase()) { 3850b57cec5SDimitry Andric unsigned Reg = Addr.getReg(); 3860b57cec5SDimitry Andric if (Reg == 0) { 3870b57cec5SDimitry Andric Reg = createResultReg(Subtarget->hasAddr64() ? &WebAssembly::I64RegClass 3880b57cec5SDimitry Andric : &WebAssembly::I32RegClass); 3890b57cec5SDimitry Andric unsigned Opc = Subtarget->hasAddr64() ? WebAssembly::CONST_I64 3900b57cec5SDimitry Andric : WebAssembly::CONST_I32; 391*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), Reg) 3920b57cec5SDimitry Andric .addImm(0); 3930b57cec5SDimitry Andric Addr.setReg(Reg); 3940b57cec5SDimitry Andric } 3950b57cec5SDimitry Andric } 3960b57cec5SDimitry Andric } 3970b57cec5SDimitry Andric 3980b57cec5SDimitry Andric void WebAssemblyFastISel::addLoadStoreOperands(const Address &Addr, 3990b57cec5SDimitry Andric const MachineInstrBuilder &MIB, 4000b57cec5SDimitry Andric MachineMemOperand *MMO) { 4010b57cec5SDimitry Andric // Set the alignment operand (this is rewritten in SetP2AlignOperands). 4020b57cec5SDimitry Andric // TODO: Disable SetP2AlignOperands for FastISel and just do it here. 4030b57cec5SDimitry Andric MIB.addImm(0); 4040b57cec5SDimitry Andric 4050b57cec5SDimitry Andric if (const GlobalValue *GV = Addr.getGlobalValue()) 4060b57cec5SDimitry Andric MIB.addGlobalAddress(GV, Addr.getOffset()); 4070b57cec5SDimitry Andric else 4080b57cec5SDimitry Andric MIB.addImm(Addr.getOffset()); 4090b57cec5SDimitry Andric 4100b57cec5SDimitry Andric if (Addr.isRegBase()) 4110b57cec5SDimitry Andric MIB.addReg(Addr.getReg()); 4120b57cec5SDimitry Andric else 4130b57cec5SDimitry Andric MIB.addFrameIndex(Addr.getFI()); 4140b57cec5SDimitry Andric 4150b57cec5SDimitry Andric MIB.addMemOperand(MMO); 4160b57cec5SDimitry Andric } 4170b57cec5SDimitry Andric 4180b57cec5SDimitry Andric unsigned WebAssemblyFastISel::maskI1Value(unsigned Reg, const Value *V) { 4190b57cec5SDimitry Andric return zeroExtendToI32(Reg, V, MVT::i1); 4200b57cec5SDimitry Andric } 4210b57cec5SDimitry Andric 42269ade1e0SDimitry Andric unsigned WebAssemblyFastISel::getRegForI1Value(const Value *V, 42369ade1e0SDimitry Andric const BasicBlock *BB, 42469ade1e0SDimitry Andric bool &Not) { 4250b57cec5SDimitry Andric if (const auto *ICmp = dyn_cast<ICmpInst>(V)) 4260b57cec5SDimitry Andric if (const ConstantInt *C = dyn_cast<ConstantInt>(ICmp->getOperand(1))) 42769ade1e0SDimitry Andric if (ICmp->isEquality() && C->isZero() && C->getType()->isIntegerTy(32) && 42869ade1e0SDimitry Andric ICmp->getParent() == BB) { 4290b57cec5SDimitry Andric Not = ICmp->isTrueWhenEqual(); 4300b57cec5SDimitry Andric return getRegForValue(ICmp->getOperand(0)); 4310b57cec5SDimitry Andric } 4320b57cec5SDimitry Andric 4330b57cec5SDimitry Andric Not = false; 43404eeddc0SDimitry Andric Register Reg = getRegForValue(V); 4350b57cec5SDimitry Andric if (Reg == 0) 4360b57cec5SDimitry Andric return 0; 4370b57cec5SDimitry Andric return maskI1Value(Reg, V); 4380b57cec5SDimitry Andric } 4390b57cec5SDimitry Andric 4400b57cec5SDimitry Andric unsigned WebAssemblyFastISel::zeroExtendToI32(unsigned Reg, const Value *V, 4410b57cec5SDimitry Andric MVT::SimpleValueType From) { 4420b57cec5SDimitry Andric if (Reg == 0) 4430b57cec5SDimitry Andric return 0; 4440b57cec5SDimitry Andric 4450b57cec5SDimitry Andric switch (From) { 4460b57cec5SDimitry Andric case MVT::i1: 4470b57cec5SDimitry Andric // If the value is naturally an i1, we don't need to mask it. We only know 4480b57cec5SDimitry Andric // if a value is naturally an i1 if it is definitely lowered by FastISel, 4490b57cec5SDimitry Andric // not a DAG ISel fallback. 4500b57cec5SDimitry Andric if (V != nullptr && isa<Argument>(V) && cast<Argument>(V)->hasZExtAttr()) 4510b57cec5SDimitry Andric return copyValue(Reg); 4520b57cec5SDimitry Andric break; 4530b57cec5SDimitry Andric case MVT::i8: 4540b57cec5SDimitry Andric case MVT::i16: 4550b57cec5SDimitry Andric break; 4560b57cec5SDimitry Andric case MVT::i32: 4570b57cec5SDimitry Andric return copyValue(Reg); 4580b57cec5SDimitry Andric default: 4590b57cec5SDimitry Andric return 0; 4600b57cec5SDimitry Andric } 4610b57cec5SDimitry Andric 46204eeddc0SDimitry Andric Register Imm = createResultReg(&WebAssembly::I32RegClass); 463*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 4640b57cec5SDimitry Andric TII.get(WebAssembly::CONST_I32), Imm) 4650b57cec5SDimitry Andric .addImm(~(~uint64_t(0) << MVT(From).getSizeInBits())); 4660b57cec5SDimitry Andric 46704eeddc0SDimitry Andric Register Result = createResultReg(&WebAssembly::I32RegClass); 468*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 4690b57cec5SDimitry Andric TII.get(WebAssembly::AND_I32), Result) 4700b57cec5SDimitry Andric .addReg(Reg) 4710b57cec5SDimitry Andric .addReg(Imm); 4720b57cec5SDimitry Andric 4730b57cec5SDimitry Andric return Result; 4740b57cec5SDimitry Andric } 4750b57cec5SDimitry Andric 4760b57cec5SDimitry Andric unsigned WebAssemblyFastISel::signExtendToI32(unsigned Reg, const Value *V, 4770b57cec5SDimitry Andric MVT::SimpleValueType From) { 4780b57cec5SDimitry Andric if (Reg == 0) 4790b57cec5SDimitry Andric return 0; 4800b57cec5SDimitry Andric 4810b57cec5SDimitry Andric switch (From) { 4820b57cec5SDimitry Andric case MVT::i1: 4830b57cec5SDimitry Andric case MVT::i8: 4840b57cec5SDimitry Andric case MVT::i16: 4850b57cec5SDimitry Andric break; 4860b57cec5SDimitry Andric case MVT::i32: 4870b57cec5SDimitry Andric return copyValue(Reg); 4880b57cec5SDimitry Andric default: 4890b57cec5SDimitry Andric return 0; 4900b57cec5SDimitry Andric } 4910b57cec5SDimitry Andric 49204eeddc0SDimitry Andric Register Imm = createResultReg(&WebAssembly::I32RegClass); 493*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 4940b57cec5SDimitry Andric TII.get(WebAssembly::CONST_I32), Imm) 4950b57cec5SDimitry Andric .addImm(32 - MVT(From).getSizeInBits()); 4960b57cec5SDimitry Andric 49704eeddc0SDimitry Andric Register Left = createResultReg(&WebAssembly::I32RegClass); 498*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 4990b57cec5SDimitry Andric TII.get(WebAssembly::SHL_I32), Left) 5000b57cec5SDimitry Andric .addReg(Reg) 5010b57cec5SDimitry Andric .addReg(Imm); 5020b57cec5SDimitry Andric 50304eeddc0SDimitry Andric Register Right = createResultReg(&WebAssembly::I32RegClass); 504*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 5050b57cec5SDimitry Andric TII.get(WebAssembly::SHR_S_I32), Right) 5060b57cec5SDimitry Andric .addReg(Left) 5070b57cec5SDimitry Andric .addReg(Imm); 5080b57cec5SDimitry Andric 5090b57cec5SDimitry Andric return Right; 5100b57cec5SDimitry Andric } 5110b57cec5SDimitry Andric 5120b57cec5SDimitry Andric unsigned WebAssemblyFastISel::zeroExtend(unsigned Reg, const Value *V, 5130b57cec5SDimitry Andric MVT::SimpleValueType From, 5140b57cec5SDimitry Andric MVT::SimpleValueType To) { 5150b57cec5SDimitry Andric if (To == MVT::i64) { 5160b57cec5SDimitry Andric if (From == MVT::i64) 5170b57cec5SDimitry Andric return copyValue(Reg); 5180b57cec5SDimitry Andric 5190b57cec5SDimitry Andric Reg = zeroExtendToI32(Reg, V, From); 5200b57cec5SDimitry Andric 52104eeddc0SDimitry Andric Register Result = createResultReg(&WebAssembly::I64RegClass); 522*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 5230b57cec5SDimitry Andric TII.get(WebAssembly::I64_EXTEND_U_I32), Result) 5240b57cec5SDimitry Andric .addReg(Reg); 5250b57cec5SDimitry Andric return Result; 5260b57cec5SDimitry Andric } 5270b57cec5SDimitry Andric 5280b57cec5SDimitry Andric if (To == MVT::i32) 5290b57cec5SDimitry Andric return zeroExtendToI32(Reg, V, From); 5300b57cec5SDimitry Andric 5310b57cec5SDimitry Andric return 0; 5320b57cec5SDimitry Andric } 5330b57cec5SDimitry Andric 5340b57cec5SDimitry Andric unsigned WebAssemblyFastISel::signExtend(unsigned Reg, const Value *V, 5350b57cec5SDimitry Andric MVT::SimpleValueType From, 5360b57cec5SDimitry Andric MVT::SimpleValueType To) { 5370b57cec5SDimitry Andric if (To == MVT::i64) { 5380b57cec5SDimitry Andric if (From == MVT::i64) 5390b57cec5SDimitry Andric return copyValue(Reg); 5400b57cec5SDimitry Andric 5410b57cec5SDimitry Andric Reg = signExtendToI32(Reg, V, From); 5420b57cec5SDimitry Andric 54304eeddc0SDimitry Andric Register Result = createResultReg(&WebAssembly::I64RegClass); 544*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 5450b57cec5SDimitry Andric TII.get(WebAssembly::I64_EXTEND_S_I32), Result) 5460b57cec5SDimitry Andric .addReg(Reg); 5470b57cec5SDimitry Andric return Result; 5480b57cec5SDimitry Andric } 5490b57cec5SDimitry Andric 5500b57cec5SDimitry Andric if (To == MVT::i32) 5510b57cec5SDimitry Andric return signExtendToI32(Reg, V, From); 5520b57cec5SDimitry Andric 5530b57cec5SDimitry Andric return 0; 5540b57cec5SDimitry Andric } 5550b57cec5SDimitry Andric 5560b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForUnsignedValue(const Value *V) { 5570b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(V->getType()); 5580b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(From); 55904eeddc0SDimitry Andric Register VReg = getRegForValue(V); 5600b57cec5SDimitry Andric if (VReg == 0) 5610b57cec5SDimitry Andric return 0; 5620b57cec5SDimitry Andric return zeroExtend(VReg, V, From, To); 5630b57cec5SDimitry Andric } 5640b57cec5SDimitry Andric 5650b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForSignedValue(const Value *V) { 5660b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(V->getType()); 5670b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(From); 56804eeddc0SDimitry Andric Register VReg = getRegForValue(V); 5690b57cec5SDimitry Andric if (VReg == 0) 5700b57cec5SDimitry Andric return 0; 5710b57cec5SDimitry Andric return signExtend(VReg, V, From, To); 5720b57cec5SDimitry Andric } 5730b57cec5SDimitry Andric 5740b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForPromotedValue(const Value *V, 5750b57cec5SDimitry Andric bool IsSigned) { 5760b57cec5SDimitry Andric return IsSigned ? getRegForSignedValue(V) : getRegForUnsignedValue(V); 5770b57cec5SDimitry Andric } 5780b57cec5SDimitry Andric 5790b57cec5SDimitry Andric unsigned WebAssemblyFastISel::notValue(unsigned Reg) { 5800b57cec5SDimitry Andric assert(MRI.getRegClass(Reg) == &WebAssembly::I32RegClass); 5810b57cec5SDimitry Andric 58204eeddc0SDimitry Andric Register NotReg = createResultReg(&WebAssembly::I32RegClass); 583*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 5840b57cec5SDimitry Andric TII.get(WebAssembly::EQZ_I32), NotReg) 5850b57cec5SDimitry Andric .addReg(Reg); 5860b57cec5SDimitry Andric return NotReg; 5870b57cec5SDimitry Andric } 5880b57cec5SDimitry Andric 5890b57cec5SDimitry Andric unsigned WebAssemblyFastISel::copyValue(unsigned Reg) { 59004eeddc0SDimitry Andric Register ResultReg = createResultReg(MRI.getRegClass(Reg)); 591*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(WebAssembly::COPY), 5920b57cec5SDimitry Andric ResultReg) 5930b57cec5SDimitry Andric .addReg(Reg); 5940b57cec5SDimitry Andric return ResultReg; 5950b57cec5SDimitry Andric } 5960b57cec5SDimitry Andric 5970b57cec5SDimitry Andric unsigned WebAssemblyFastISel::fastMaterializeAlloca(const AllocaInst *AI) { 5980b57cec5SDimitry Andric DenseMap<const AllocaInst *, int>::iterator SI = 5990b57cec5SDimitry Andric FuncInfo.StaticAllocaMap.find(AI); 6000b57cec5SDimitry Andric 6010b57cec5SDimitry Andric if (SI != FuncInfo.StaticAllocaMap.end()) { 60204eeddc0SDimitry Andric Register ResultReg = 6030b57cec5SDimitry Andric createResultReg(Subtarget->hasAddr64() ? &WebAssembly::I64RegClass 6040b57cec5SDimitry Andric : &WebAssembly::I32RegClass); 6050b57cec5SDimitry Andric unsigned Opc = 6060b57cec5SDimitry Andric Subtarget->hasAddr64() ? WebAssembly::COPY_I64 : WebAssembly::COPY_I32; 607*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), ResultReg) 6080b57cec5SDimitry Andric .addFrameIndex(SI->second); 6090b57cec5SDimitry Andric return ResultReg; 6100b57cec5SDimitry Andric } 6110b57cec5SDimitry Andric 6120b57cec5SDimitry Andric return 0; 6130b57cec5SDimitry Andric } 6140b57cec5SDimitry Andric 6150b57cec5SDimitry Andric unsigned WebAssemblyFastISel::fastMaterializeConstant(const Constant *C) { 6160b57cec5SDimitry Andric if (const GlobalValue *GV = dyn_cast<GlobalValue>(C)) { 6170b57cec5SDimitry Andric if (TLI.isPositionIndependent()) 6180b57cec5SDimitry Andric return 0; 6190b57cec5SDimitry Andric if (GV->isThreadLocal()) 6200b57cec5SDimitry Andric return 0; 62104eeddc0SDimitry Andric Register ResultReg = 6220b57cec5SDimitry Andric createResultReg(Subtarget->hasAddr64() ? &WebAssembly::I64RegClass 6230b57cec5SDimitry Andric : &WebAssembly::I32RegClass); 6240b57cec5SDimitry Andric unsigned Opc = Subtarget->hasAddr64() ? WebAssembly::CONST_I64 6250b57cec5SDimitry Andric : WebAssembly::CONST_I32; 626*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), ResultReg) 6270b57cec5SDimitry Andric .addGlobalAddress(GV); 6280b57cec5SDimitry Andric return ResultReg; 6290b57cec5SDimitry Andric } 6300b57cec5SDimitry Andric 6310b57cec5SDimitry Andric // Let target-independent code handle it. 6320b57cec5SDimitry Andric return 0; 6330b57cec5SDimitry Andric } 6340b57cec5SDimitry Andric 6350b57cec5SDimitry Andric bool WebAssemblyFastISel::fastLowerArguments() { 6360b57cec5SDimitry Andric if (!FuncInfo.CanLowerReturn) 6370b57cec5SDimitry Andric return false; 6380b57cec5SDimitry Andric 6390b57cec5SDimitry Andric const Function *F = FuncInfo.Fn; 6400b57cec5SDimitry Andric if (F->isVarArg()) 6410b57cec5SDimitry Andric return false; 6420b57cec5SDimitry Andric 6435ffd83dbSDimitry Andric if (FuncInfo.Fn->getCallingConv() == CallingConv::Swift) 6445ffd83dbSDimitry Andric return false; 6455ffd83dbSDimitry Andric 6460b57cec5SDimitry Andric unsigned I = 0; 6470b57cec5SDimitry Andric for (auto const &Arg : F->args()) { 6480b57cec5SDimitry Andric const AttributeList &Attrs = F->getAttributes(); 649349cc55cSDimitry Andric if (Attrs.hasParamAttr(I, Attribute::ByVal) || 650349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::SwiftSelf) || 651349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::SwiftError) || 652349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::InAlloca) || 653349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::Nest)) 6540b57cec5SDimitry Andric return false; 6550b57cec5SDimitry Andric 6560b57cec5SDimitry Andric Type *ArgTy = Arg.getType(); 6570b57cec5SDimitry Andric if (ArgTy->isStructTy() || ArgTy->isArrayTy()) 6580b57cec5SDimitry Andric return false; 6590b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && ArgTy->isVectorTy()) 6600b57cec5SDimitry Andric return false; 6610b57cec5SDimitry Andric 6620b57cec5SDimitry Andric unsigned Opc; 6630b57cec5SDimitry Andric const TargetRegisterClass *RC; 6640b57cec5SDimitry Andric switch (getSimpleType(ArgTy)) { 6650b57cec5SDimitry Andric case MVT::i1: 6660b57cec5SDimitry Andric case MVT::i8: 6670b57cec5SDimitry Andric case MVT::i16: 6680b57cec5SDimitry Andric case MVT::i32: 6690b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_i32; 6700b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 6710b57cec5SDimitry Andric break; 6720b57cec5SDimitry Andric case MVT::i64: 6730b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_i64; 6740b57cec5SDimitry Andric RC = &WebAssembly::I64RegClass; 6750b57cec5SDimitry Andric break; 6760b57cec5SDimitry Andric case MVT::f32: 6770b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_f32; 6780b57cec5SDimitry Andric RC = &WebAssembly::F32RegClass; 6790b57cec5SDimitry Andric break; 6800b57cec5SDimitry Andric case MVT::f64: 6810b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_f64; 6820b57cec5SDimitry Andric RC = &WebAssembly::F64RegClass; 6830b57cec5SDimitry Andric break; 6840b57cec5SDimitry Andric case MVT::v16i8: 6850b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v16i8; 6860b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 6870b57cec5SDimitry Andric break; 6880b57cec5SDimitry Andric case MVT::v8i16: 6890b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v8i16; 6900b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 6910b57cec5SDimitry Andric break; 6920b57cec5SDimitry Andric case MVT::v4i32: 6930b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v4i32; 6940b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 6950b57cec5SDimitry Andric break; 6960b57cec5SDimitry Andric case MVT::v2i64: 6970b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v2i64; 6980b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 6990b57cec5SDimitry Andric break; 7000b57cec5SDimitry Andric case MVT::v4f32: 7010b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v4f32; 7020b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 7030b57cec5SDimitry Andric break; 7040b57cec5SDimitry Andric case MVT::v2f64: 7050b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v2f64; 7060b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 7070b57cec5SDimitry Andric break; 708e8d8bef9SDimitry Andric case MVT::funcref: 709e8d8bef9SDimitry Andric Opc = WebAssembly::ARGUMENT_funcref; 710e8d8bef9SDimitry Andric RC = &WebAssembly::FUNCREFRegClass; 711e8d8bef9SDimitry Andric break; 712e8d8bef9SDimitry Andric case MVT::externref: 713e8d8bef9SDimitry Andric Opc = WebAssembly::ARGUMENT_externref; 714e8d8bef9SDimitry Andric RC = &WebAssembly::EXTERNREFRegClass; 7150b57cec5SDimitry Andric break; 7160b57cec5SDimitry Andric default: 7170b57cec5SDimitry Andric return false; 7180b57cec5SDimitry Andric } 71904eeddc0SDimitry Andric Register ResultReg = createResultReg(RC); 720*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), ResultReg) 7210b57cec5SDimitry Andric .addImm(I); 7220b57cec5SDimitry Andric updateValueMap(&Arg, ResultReg); 7230b57cec5SDimitry Andric 7240b57cec5SDimitry Andric ++I; 7250b57cec5SDimitry Andric } 7260b57cec5SDimitry Andric 7270b57cec5SDimitry Andric MRI.addLiveIn(WebAssembly::ARGUMENTS); 7280b57cec5SDimitry Andric 7290b57cec5SDimitry Andric auto *MFI = MF->getInfo<WebAssemblyFunctionInfo>(); 7300b57cec5SDimitry Andric for (auto const &Arg : F->args()) { 7310b57cec5SDimitry Andric MVT::SimpleValueType ArgTy = getLegalType(getSimpleType(Arg.getType())); 7320b57cec5SDimitry Andric if (ArgTy == MVT::INVALID_SIMPLE_VALUE_TYPE) { 7330b57cec5SDimitry Andric MFI->clearParamsAndResults(); 7340b57cec5SDimitry Andric return false; 7350b57cec5SDimitry Andric } 7360b57cec5SDimitry Andric MFI->addParam(ArgTy); 7370b57cec5SDimitry Andric } 7380b57cec5SDimitry Andric 7390b57cec5SDimitry Andric if (!F->getReturnType()->isVoidTy()) { 7400b57cec5SDimitry Andric MVT::SimpleValueType RetTy = 7410b57cec5SDimitry Andric getLegalType(getSimpleType(F->getReturnType())); 7420b57cec5SDimitry Andric if (RetTy == MVT::INVALID_SIMPLE_VALUE_TYPE) { 7430b57cec5SDimitry Andric MFI->clearParamsAndResults(); 7440b57cec5SDimitry Andric return false; 7450b57cec5SDimitry Andric } 7460b57cec5SDimitry Andric MFI->addResult(RetTy); 7470b57cec5SDimitry Andric } 7480b57cec5SDimitry Andric 7490b57cec5SDimitry Andric return true; 7500b57cec5SDimitry Andric } 7510b57cec5SDimitry Andric 7520b57cec5SDimitry Andric bool WebAssemblyFastISel::selectCall(const Instruction *I) { 7530b57cec5SDimitry Andric const auto *Call = cast<CallInst>(I); 7540b57cec5SDimitry Andric 7550b57cec5SDimitry Andric // TODO: Support tail calls in FastISel 7560b57cec5SDimitry Andric if (Call->isMustTailCall() || Call->isInlineAsm() || 7570b57cec5SDimitry Andric Call->getFunctionType()->isVarArg()) 7580b57cec5SDimitry Andric return false; 7590b57cec5SDimitry Andric 7600b57cec5SDimitry Andric Function *Func = Call->getCalledFunction(); 7610b57cec5SDimitry Andric if (Func && Func->isIntrinsic()) 7620b57cec5SDimitry Andric return false; 7630b57cec5SDimitry Andric 7645ffd83dbSDimitry Andric if (Call->getCallingConv() == CallingConv::Swift) 7655ffd83dbSDimitry Andric return false; 7665ffd83dbSDimitry Andric 7670b57cec5SDimitry Andric bool IsDirect = Func != nullptr; 7685ffd83dbSDimitry Andric if (!IsDirect && isa<ConstantExpr>(Call->getCalledOperand())) 7690b57cec5SDimitry Andric return false; 7700b57cec5SDimitry Andric 7710b57cec5SDimitry Andric FunctionType *FuncTy = Call->getFunctionType(); 7725ffd83dbSDimitry Andric unsigned Opc = IsDirect ? WebAssembly::CALL : WebAssembly::CALL_INDIRECT; 7730b57cec5SDimitry Andric bool IsVoid = FuncTy->getReturnType()->isVoidTy(); 7740b57cec5SDimitry Andric unsigned ResultReg; 7755ffd83dbSDimitry Andric if (!IsVoid) { 7760b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && Call->getType()->isVectorTy()) 7770b57cec5SDimitry Andric return false; 7780b57cec5SDimitry Andric 7790b57cec5SDimitry Andric MVT::SimpleValueType RetTy = getSimpleType(Call->getType()); 7800b57cec5SDimitry Andric switch (RetTy) { 7810b57cec5SDimitry Andric case MVT::i1: 7820b57cec5SDimitry Andric case MVT::i8: 7830b57cec5SDimitry Andric case MVT::i16: 7840b57cec5SDimitry Andric case MVT::i32: 7850b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::I32RegClass); 7860b57cec5SDimitry Andric break; 7870b57cec5SDimitry Andric case MVT::i64: 7880b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::I64RegClass); 7890b57cec5SDimitry Andric break; 7900b57cec5SDimitry Andric case MVT::f32: 7910b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::F32RegClass); 7920b57cec5SDimitry Andric break; 7930b57cec5SDimitry Andric case MVT::f64: 7940b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::F64RegClass); 7950b57cec5SDimitry Andric break; 7960b57cec5SDimitry Andric case MVT::v16i8: 7970b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 7980b57cec5SDimitry Andric break; 7990b57cec5SDimitry Andric case MVT::v8i16: 8000b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 8010b57cec5SDimitry Andric break; 8020b57cec5SDimitry Andric case MVT::v4i32: 8030b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 8040b57cec5SDimitry Andric break; 8050b57cec5SDimitry Andric case MVT::v2i64: 8060b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 8070b57cec5SDimitry Andric break; 8080b57cec5SDimitry Andric case MVT::v4f32: 8090b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 8100b57cec5SDimitry Andric break; 8110b57cec5SDimitry Andric case MVT::v2f64: 8120b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 8130b57cec5SDimitry Andric break; 814e8d8bef9SDimitry Andric case MVT::funcref: 815e8d8bef9SDimitry Andric ResultReg = createResultReg(&WebAssembly::FUNCREFRegClass); 816e8d8bef9SDimitry Andric break; 817e8d8bef9SDimitry Andric case MVT::externref: 818e8d8bef9SDimitry Andric ResultReg = createResultReg(&WebAssembly::EXTERNREFRegClass); 8190b57cec5SDimitry Andric break; 8200b57cec5SDimitry Andric default: 8210b57cec5SDimitry Andric return false; 8220b57cec5SDimitry Andric } 8230b57cec5SDimitry Andric } 8240b57cec5SDimitry Andric 8250b57cec5SDimitry Andric SmallVector<unsigned, 8> Args; 826349cc55cSDimitry Andric for (unsigned I = 0, E = Call->arg_size(); I < E; ++I) { 8270b57cec5SDimitry Andric Value *V = Call->getArgOperand(I); 8280b57cec5SDimitry Andric MVT::SimpleValueType ArgTy = getSimpleType(V->getType()); 8290b57cec5SDimitry Andric if (ArgTy == MVT::INVALID_SIMPLE_VALUE_TYPE) 8300b57cec5SDimitry Andric return false; 8310b57cec5SDimitry Andric 8320b57cec5SDimitry Andric const AttributeList &Attrs = Call->getAttributes(); 833349cc55cSDimitry Andric if (Attrs.hasParamAttr(I, Attribute::ByVal) || 834349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::SwiftSelf) || 835349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::SwiftError) || 836349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::InAlloca) || 837349cc55cSDimitry Andric Attrs.hasParamAttr(I, Attribute::Nest)) 8380b57cec5SDimitry Andric return false; 8390b57cec5SDimitry Andric 8400b57cec5SDimitry Andric unsigned Reg; 8410b57cec5SDimitry Andric 842349cc55cSDimitry Andric if (Attrs.hasParamAttr(I, Attribute::SExt)) 8430b57cec5SDimitry Andric Reg = getRegForSignedValue(V); 844349cc55cSDimitry Andric else if (Attrs.hasParamAttr(I, Attribute::ZExt)) 8450b57cec5SDimitry Andric Reg = getRegForUnsignedValue(V); 8460b57cec5SDimitry Andric else 8470b57cec5SDimitry Andric Reg = getRegForValue(V); 8480b57cec5SDimitry Andric 8490b57cec5SDimitry Andric if (Reg == 0) 8500b57cec5SDimitry Andric return false; 8510b57cec5SDimitry Andric 8520b57cec5SDimitry Andric Args.push_back(Reg); 8530b57cec5SDimitry Andric } 8540b57cec5SDimitry Andric 8550b57cec5SDimitry Andric unsigned CalleeReg = 0; 8560b57cec5SDimitry Andric if (!IsDirect) { 8575ffd83dbSDimitry Andric CalleeReg = getRegForValue(Call->getCalledOperand()); 8580b57cec5SDimitry Andric if (!CalleeReg) 8590b57cec5SDimitry Andric return false; 8600b57cec5SDimitry Andric } 8610b57cec5SDimitry Andric 862*bdd1243dSDimitry Andric auto MIB = BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc)); 8630b57cec5SDimitry Andric 8640b57cec5SDimitry Andric if (!IsVoid) 8650b57cec5SDimitry Andric MIB.addReg(ResultReg, RegState::Define); 8660b57cec5SDimitry Andric 8675ffd83dbSDimitry Andric if (IsDirect) { 8680b57cec5SDimitry Andric MIB.addGlobalAddress(Func); 8695ffd83dbSDimitry Andric } else { 870fe6060f1SDimitry Andric // Placeholder for the type index. 8715ffd83dbSDimitry Andric MIB.addImm(0); 872fe6060f1SDimitry Andric // The table into which this call_indirect indexes. 873fe6060f1SDimitry Andric MCSymbolWasm *Table = WebAssembly::getOrCreateFunctionTableSymbol( 874fe6060f1SDimitry Andric MF->getMMI().getContext(), Subtarget); 875fe6060f1SDimitry Andric if (Subtarget->hasReferenceTypes()) { 876fe6060f1SDimitry Andric MIB.addSym(Table); 877fe6060f1SDimitry Andric } else { 878fe6060f1SDimitry Andric // Otherwise for the MVP there is at most one table whose number is 0, but 879fe6060f1SDimitry Andric // we can't write a table symbol or issue relocations. Instead we just 880fe6060f1SDimitry Andric // ensure the table is live. 881fe6060f1SDimitry Andric Table->setNoStrip(); 8825ffd83dbSDimitry Andric MIB.addImm(0); 883fe6060f1SDimitry Andric } 884fe6060f1SDimitry Andric // See if we must truncate the function pointer. 885fe6060f1SDimitry Andric // CALL_INDIRECT takes an i32, but in wasm64 we represent function pointers 886fe6060f1SDimitry Andric // as 64-bit for uniformity with other pointer types. 887fe6060f1SDimitry Andric // See also: WebAssemblyISelLowering.cpp: LowerCallResults 888fe6060f1SDimitry Andric if (Subtarget->hasAddr64()) { 889*bdd1243dSDimitry Andric auto Wrap = BuildMI(*FuncInfo.MBB, std::prev(FuncInfo.InsertPt), MIMD, 890fe6060f1SDimitry Andric TII.get(WebAssembly::I32_WRAP_I64)); 89104eeddc0SDimitry Andric Register Reg32 = createResultReg(&WebAssembly::I32RegClass); 892fe6060f1SDimitry Andric Wrap.addReg(Reg32, RegState::Define); 893fe6060f1SDimitry Andric Wrap.addReg(CalleeReg); 894fe6060f1SDimitry Andric CalleeReg = Reg32; 895fe6060f1SDimitry Andric } 8965ffd83dbSDimitry Andric } 8970b57cec5SDimitry Andric 8980b57cec5SDimitry Andric for (unsigned ArgReg : Args) 8990b57cec5SDimitry Andric MIB.addReg(ArgReg); 9000b57cec5SDimitry Andric 9015ffd83dbSDimitry Andric if (!IsDirect) 9025ffd83dbSDimitry Andric MIB.addReg(CalleeReg); 9035ffd83dbSDimitry Andric 9040b57cec5SDimitry Andric if (!IsVoid) 9050b57cec5SDimitry Andric updateValueMap(Call, ResultReg); 9060b57cec5SDimitry Andric return true; 9070b57cec5SDimitry Andric } 9080b57cec5SDimitry Andric 9090b57cec5SDimitry Andric bool WebAssemblyFastISel::selectSelect(const Instruction *I) { 9100b57cec5SDimitry Andric const auto *Select = cast<SelectInst>(I); 9110b57cec5SDimitry Andric 9120b57cec5SDimitry Andric bool Not; 91369ade1e0SDimitry Andric unsigned CondReg = 91469ade1e0SDimitry Andric getRegForI1Value(Select->getCondition(), I->getParent(), Not); 9150b57cec5SDimitry Andric if (CondReg == 0) 9160b57cec5SDimitry Andric return false; 9170b57cec5SDimitry Andric 91804eeddc0SDimitry Andric Register TrueReg = getRegForValue(Select->getTrueValue()); 9190b57cec5SDimitry Andric if (TrueReg == 0) 9200b57cec5SDimitry Andric return false; 9210b57cec5SDimitry Andric 92204eeddc0SDimitry Andric Register FalseReg = getRegForValue(Select->getFalseValue()); 9230b57cec5SDimitry Andric if (FalseReg == 0) 9240b57cec5SDimitry Andric return false; 9250b57cec5SDimitry Andric 9260b57cec5SDimitry Andric if (Not) 9270b57cec5SDimitry Andric std::swap(TrueReg, FalseReg); 9280b57cec5SDimitry Andric 9290b57cec5SDimitry Andric unsigned Opc; 9300b57cec5SDimitry Andric const TargetRegisterClass *RC; 9310b57cec5SDimitry Andric switch (getSimpleType(Select->getType())) { 9320b57cec5SDimitry Andric case MVT::i1: 9330b57cec5SDimitry Andric case MVT::i8: 9340b57cec5SDimitry Andric case MVT::i16: 9350b57cec5SDimitry Andric case MVT::i32: 9360b57cec5SDimitry Andric Opc = WebAssembly::SELECT_I32; 9370b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 9380b57cec5SDimitry Andric break; 9390b57cec5SDimitry Andric case MVT::i64: 9400b57cec5SDimitry Andric Opc = WebAssembly::SELECT_I64; 9410b57cec5SDimitry Andric RC = &WebAssembly::I64RegClass; 9420b57cec5SDimitry Andric break; 9430b57cec5SDimitry Andric case MVT::f32: 9440b57cec5SDimitry Andric Opc = WebAssembly::SELECT_F32; 9450b57cec5SDimitry Andric RC = &WebAssembly::F32RegClass; 9460b57cec5SDimitry Andric break; 9470b57cec5SDimitry Andric case MVT::f64: 9480b57cec5SDimitry Andric Opc = WebAssembly::SELECT_F64; 9490b57cec5SDimitry Andric RC = &WebAssembly::F64RegClass; 9500b57cec5SDimitry Andric break; 951e8d8bef9SDimitry Andric case MVT::funcref: 952e8d8bef9SDimitry Andric Opc = WebAssembly::SELECT_FUNCREF; 953e8d8bef9SDimitry Andric RC = &WebAssembly::FUNCREFRegClass; 954e8d8bef9SDimitry Andric break; 955e8d8bef9SDimitry Andric case MVT::externref: 956e8d8bef9SDimitry Andric Opc = WebAssembly::SELECT_EXTERNREF; 957e8d8bef9SDimitry Andric RC = &WebAssembly::EXTERNREFRegClass; 9580b57cec5SDimitry Andric break; 9590b57cec5SDimitry Andric default: 9600b57cec5SDimitry Andric return false; 9610b57cec5SDimitry Andric } 9620b57cec5SDimitry Andric 96304eeddc0SDimitry Andric Register ResultReg = createResultReg(RC); 964*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), ResultReg) 9650b57cec5SDimitry Andric .addReg(TrueReg) 9660b57cec5SDimitry Andric .addReg(FalseReg) 9670b57cec5SDimitry Andric .addReg(CondReg); 9680b57cec5SDimitry Andric 9690b57cec5SDimitry Andric updateValueMap(Select, ResultReg); 9700b57cec5SDimitry Andric return true; 9710b57cec5SDimitry Andric } 9720b57cec5SDimitry Andric 9730b57cec5SDimitry Andric bool WebAssemblyFastISel::selectTrunc(const Instruction *I) { 9740b57cec5SDimitry Andric const auto *Trunc = cast<TruncInst>(I); 9750b57cec5SDimitry Andric 97604eeddc0SDimitry Andric Register Reg = getRegForValue(Trunc->getOperand(0)); 9770b57cec5SDimitry Andric if (Reg == 0) 9780b57cec5SDimitry Andric return false; 9790b57cec5SDimitry Andric 9800b57cec5SDimitry Andric if (Trunc->getOperand(0)->getType()->isIntegerTy(64)) { 98104eeddc0SDimitry Andric Register Result = createResultReg(&WebAssembly::I32RegClass); 982*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 9830b57cec5SDimitry Andric TII.get(WebAssembly::I32_WRAP_I64), Result) 9840b57cec5SDimitry Andric .addReg(Reg); 9850b57cec5SDimitry Andric Reg = Result; 9860b57cec5SDimitry Andric } 9870b57cec5SDimitry Andric 9880b57cec5SDimitry Andric updateValueMap(Trunc, Reg); 9890b57cec5SDimitry Andric return true; 9900b57cec5SDimitry Andric } 9910b57cec5SDimitry Andric 9920b57cec5SDimitry Andric bool WebAssemblyFastISel::selectZExt(const Instruction *I) { 9930b57cec5SDimitry Andric const auto *ZExt = cast<ZExtInst>(I); 9940b57cec5SDimitry Andric 9950b57cec5SDimitry Andric const Value *Op = ZExt->getOperand(0); 9960b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(Op->getType()); 9970b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(getSimpleType(ZExt->getType())); 99804eeddc0SDimitry Andric Register In = getRegForValue(Op); 9990b57cec5SDimitry Andric if (In == 0) 10000b57cec5SDimitry Andric return false; 10010b57cec5SDimitry Andric unsigned Reg = zeroExtend(In, Op, From, To); 10020b57cec5SDimitry Andric if (Reg == 0) 10030b57cec5SDimitry Andric return false; 10040b57cec5SDimitry Andric 10050b57cec5SDimitry Andric updateValueMap(ZExt, Reg); 10060b57cec5SDimitry Andric return true; 10070b57cec5SDimitry Andric } 10080b57cec5SDimitry Andric 10090b57cec5SDimitry Andric bool WebAssemblyFastISel::selectSExt(const Instruction *I) { 10100b57cec5SDimitry Andric const auto *SExt = cast<SExtInst>(I); 10110b57cec5SDimitry Andric 10120b57cec5SDimitry Andric const Value *Op = SExt->getOperand(0); 10130b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(Op->getType()); 10140b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(getSimpleType(SExt->getType())); 101504eeddc0SDimitry Andric Register In = getRegForValue(Op); 10160b57cec5SDimitry Andric if (In == 0) 10170b57cec5SDimitry Andric return false; 10180b57cec5SDimitry Andric unsigned Reg = signExtend(In, Op, From, To); 10190b57cec5SDimitry Andric if (Reg == 0) 10200b57cec5SDimitry Andric return false; 10210b57cec5SDimitry Andric 10220b57cec5SDimitry Andric updateValueMap(SExt, Reg); 10230b57cec5SDimitry Andric return true; 10240b57cec5SDimitry Andric } 10250b57cec5SDimitry Andric 10260b57cec5SDimitry Andric bool WebAssemblyFastISel::selectICmp(const Instruction *I) { 10270b57cec5SDimitry Andric const auto *ICmp = cast<ICmpInst>(I); 10280b57cec5SDimitry Andric 10290b57cec5SDimitry Andric bool I32 = getSimpleType(ICmp->getOperand(0)->getType()) != MVT::i64; 10300b57cec5SDimitry Andric unsigned Opc; 10310b57cec5SDimitry Andric bool IsSigned = false; 10320b57cec5SDimitry Andric switch (ICmp->getPredicate()) { 10330b57cec5SDimitry Andric case ICmpInst::ICMP_EQ: 10340b57cec5SDimitry Andric Opc = I32 ? WebAssembly::EQ_I32 : WebAssembly::EQ_I64; 10350b57cec5SDimitry Andric break; 10360b57cec5SDimitry Andric case ICmpInst::ICMP_NE: 10370b57cec5SDimitry Andric Opc = I32 ? WebAssembly::NE_I32 : WebAssembly::NE_I64; 10380b57cec5SDimitry Andric break; 10390b57cec5SDimitry Andric case ICmpInst::ICMP_UGT: 10400b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GT_U_I32 : WebAssembly::GT_U_I64; 10410b57cec5SDimitry Andric break; 10420b57cec5SDimitry Andric case ICmpInst::ICMP_UGE: 10430b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GE_U_I32 : WebAssembly::GE_U_I64; 10440b57cec5SDimitry Andric break; 10450b57cec5SDimitry Andric case ICmpInst::ICMP_ULT: 10460b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LT_U_I32 : WebAssembly::LT_U_I64; 10470b57cec5SDimitry Andric break; 10480b57cec5SDimitry Andric case ICmpInst::ICMP_ULE: 10490b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LE_U_I32 : WebAssembly::LE_U_I64; 10500b57cec5SDimitry Andric break; 10510b57cec5SDimitry Andric case ICmpInst::ICMP_SGT: 10520b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GT_S_I32 : WebAssembly::GT_S_I64; 10530b57cec5SDimitry Andric IsSigned = true; 10540b57cec5SDimitry Andric break; 10550b57cec5SDimitry Andric case ICmpInst::ICMP_SGE: 10560b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GE_S_I32 : WebAssembly::GE_S_I64; 10570b57cec5SDimitry Andric IsSigned = true; 10580b57cec5SDimitry Andric break; 10590b57cec5SDimitry Andric case ICmpInst::ICMP_SLT: 10600b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LT_S_I32 : WebAssembly::LT_S_I64; 10610b57cec5SDimitry Andric IsSigned = true; 10620b57cec5SDimitry Andric break; 10630b57cec5SDimitry Andric case ICmpInst::ICMP_SLE: 10640b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LE_S_I32 : WebAssembly::LE_S_I64; 10650b57cec5SDimitry Andric IsSigned = true; 10660b57cec5SDimitry Andric break; 10670b57cec5SDimitry Andric default: 10680b57cec5SDimitry Andric return false; 10690b57cec5SDimitry Andric } 10700b57cec5SDimitry Andric 10710b57cec5SDimitry Andric unsigned LHS = getRegForPromotedValue(ICmp->getOperand(0), IsSigned); 10720b57cec5SDimitry Andric if (LHS == 0) 10730b57cec5SDimitry Andric return false; 10740b57cec5SDimitry Andric 10750b57cec5SDimitry Andric unsigned RHS = getRegForPromotedValue(ICmp->getOperand(1), IsSigned); 10760b57cec5SDimitry Andric if (RHS == 0) 10770b57cec5SDimitry Andric return false; 10780b57cec5SDimitry Andric 107904eeddc0SDimitry Andric Register ResultReg = createResultReg(&WebAssembly::I32RegClass); 1080*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), ResultReg) 10810b57cec5SDimitry Andric .addReg(LHS) 10820b57cec5SDimitry Andric .addReg(RHS); 10830b57cec5SDimitry Andric updateValueMap(ICmp, ResultReg); 10840b57cec5SDimitry Andric return true; 10850b57cec5SDimitry Andric } 10860b57cec5SDimitry Andric 10870b57cec5SDimitry Andric bool WebAssemblyFastISel::selectFCmp(const Instruction *I) { 10880b57cec5SDimitry Andric const auto *FCmp = cast<FCmpInst>(I); 10890b57cec5SDimitry Andric 109004eeddc0SDimitry Andric Register LHS = getRegForValue(FCmp->getOperand(0)); 10910b57cec5SDimitry Andric if (LHS == 0) 10920b57cec5SDimitry Andric return false; 10930b57cec5SDimitry Andric 109404eeddc0SDimitry Andric Register RHS = getRegForValue(FCmp->getOperand(1)); 10950b57cec5SDimitry Andric if (RHS == 0) 10960b57cec5SDimitry Andric return false; 10970b57cec5SDimitry Andric 10980b57cec5SDimitry Andric bool F32 = getSimpleType(FCmp->getOperand(0)->getType()) != MVT::f64; 10990b57cec5SDimitry Andric unsigned Opc; 11000b57cec5SDimitry Andric bool Not = false; 11010b57cec5SDimitry Andric switch (FCmp->getPredicate()) { 11020b57cec5SDimitry Andric case FCmpInst::FCMP_OEQ: 11030b57cec5SDimitry Andric Opc = F32 ? WebAssembly::EQ_F32 : WebAssembly::EQ_F64; 11040b57cec5SDimitry Andric break; 11050b57cec5SDimitry Andric case FCmpInst::FCMP_UNE: 11060b57cec5SDimitry Andric Opc = F32 ? WebAssembly::NE_F32 : WebAssembly::NE_F64; 11070b57cec5SDimitry Andric break; 11080b57cec5SDimitry Andric case FCmpInst::FCMP_OGT: 11090b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GT_F32 : WebAssembly::GT_F64; 11100b57cec5SDimitry Andric break; 11110b57cec5SDimitry Andric case FCmpInst::FCMP_OGE: 11120b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GE_F32 : WebAssembly::GE_F64; 11130b57cec5SDimitry Andric break; 11140b57cec5SDimitry Andric case FCmpInst::FCMP_OLT: 11150b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LT_F32 : WebAssembly::LT_F64; 11160b57cec5SDimitry Andric break; 11170b57cec5SDimitry Andric case FCmpInst::FCMP_OLE: 11180b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LE_F32 : WebAssembly::LE_F64; 11190b57cec5SDimitry Andric break; 11200b57cec5SDimitry Andric case FCmpInst::FCMP_UGT: 11210b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LE_F32 : WebAssembly::LE_F64; 11220b57cec5SDimitry Andric Not = true; 11230b57cec5SDimitry Andric break; 11240b57cec5SDimitry Andric case FCmpInst::FCMP_UGE: 11250b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LT_F32 : WebAssembly::LT_F64; 11260b57cec5SDimitry Andric Not = true; 11270b57cec5SDimitry Andric break; 11280b57cec5SDimitry Andric case FCmpInst::FCMP_ULT: 11290b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GE_F32 : WebAssembly::GE_F64; 11300b57cec5SDimitry Andric Not = true; 11310b57cec5SDimitry Andric break; 11320b57cec5SDimitry Andric case FCmpInst::FCMP_ULE: 11330b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GT_F32 : WebAssembly::GT_F64; 11340b57cec5SDimitry Andric Not = true; 11350b57cec5SDimitry Andric break; 11360b57cec5SDimitry Andric default: 11370b57cec5SDimitry Andric return false; 11380b57cec5SDimitry Andric } 11390b57cec5SDimitry Andric 114004eeddc0SDimitry Andric Register ResultReg = createResultReg(&WebAssembly::I32RegClass); 1141*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), ResultReg) 11420b57cec5SDimitry Andric .addReg(LHS) 11430b57cec5SDimitry Andric .addReg(RHS); 11440b57cec5SDimitry Andric 11450b57cec5SDimitry Andric if (Not) 11460b57cec5SDimitry Andric ResultReg = notValue(ResultReg); 11470b57cec5SDimitry Andric 11480b57cec5SDimitry Andric updateValueMap(FCmp, ResultReg); 11490b57cec5SDimitry Andric return true; 11500b57cec5SDimitry Andric } 11510b57cec5SDimitry Andric 11520b57cec5SDimitry Andric bool WebAssemblyFastISel::selectBitCast(const Instruction *I) { 11530b57cec5SDimitry Andric // Target-independent code can handle this, except it doesn't set the dead 11540b57cec5SDimitry Andric // flag on the ARGUMENTS clobber, so we have to do that manually in order 11550b57cec5SDimitry Andric // to satisfy code that expects this of isBitcast() instructions. 11560b57cec5SDimitry Andric EVT VT = TLI.getValueType(DL, I->getOperand(0)->getType()); 11570b57cec5SDimitry Andric EVT RetVT = TLI.getValueType(DL, I->getType()); 11580b57cec5SDimitry Andric if (!VT.isSimple() || !RetVT.isSimple()) 11590b57cec5SDimitry Andric return false; 11600b57cec5SDimitry Andric 116104eeddc0SDimitry Andric Register In = getRegForValue(I->getOperand(0)); 11620b57cec5SDimitry Andric if (In == 0) 11630b57cec5SDimitry Andric return false; 11640b57cec5SDimitry Andric 11650b57cec5SDimitry Andric if (VT == RetVT) { 11660b57cec5SDimitry Andric // No-op bitcast. 11670b57cec5SDimitry Andric updateValueMap(I, In); 11680b57cec5SDimitry Andric return true; 11690b57cec5SDimitry Andric } 11700b57cec5SDimitry Andric 11718bcb0991SDimitry Andric Register Reg = fastEmit_ISD_BITCAST_r(VT.getSimpleVT(), RetVT.getSimpleVT(), 1172fe6060f1SDimitry Andric In); 11730b57cec5SDimitry Andric if (!Reg) 11740b57cec5SDimitry Andric return false; 11750b57cec5SDimitry Andric MachineBasicBlock::iterator Iter = FuncInfo.InsertPt; 11760b57cec5SDimitry Andric --Iter; 11770b57cec5SDimitry Andric assert(Iter->isBitcast()); 11788bcb0991SDimitry Andric Iter->setPhysRegsDeadExcept(ArrayRef<Register>(), TRI); 11790b57cec5SDimitry Andric updateValueMap(I, Reg); 11800b57cec5SDimitry Andric return true; 11810b57cec5SDimitry Andric } 11820b57cec5SDimitry Andric 11830b57cec5SDimitry Andric bool WebAssemblyFastISel::selectLoad(const Instruction *I) { 11840b57cec5SDimitry Andric const auto *Load = cast<LoadInst>(I); 11850b57cec5SDimitry Andric if (Load->isAtomic()) 11860b57cec5SDimitry Andric return false; 1187fe6060f1SDimitry Andric if (!WebAssembly::isDefaultAddressSpace(Load->getPointerAddressSpace())) 1188fe6060f1SDimitry Andric return false; 11890b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && Load->getType()->isVectorTy()) 11900b57cec5SDimitry Andric return false; 11910b57cec5SDimitry Andric 11920b57cec5SDimitry Andric Address Addr; 11930b57cec5SDimitry Andric if (!computeAddress(Load->getPointerOperand(), Addr)) 11940b57cec5SDimitry Andric return false; 11950b57cec5SDimitry Andric 11960b57cec5SDimitry Andric // TODO: Fold a following sign-/zero-extend into the load instruction. 11970b57cec5SDimitry Andric 11980b57cec5SDimitry Andric unsigned Opc; 11990b57cec5SDimitry Andric const TargetRegisterClass *RC; 12005ffd83dbSDimitry Andric bool A64 = Subtarget->hasAddr64(); 12010b57cec5SDimitry Andric switch (getSimpleType(Load->getType())) { 12020b57cec5SDimitry Andric case MVT::i1: 12030b57cec5SDimitry Andric case MVT::i8: 12045ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::LOAD8_U_I32_A64 : WebAssembly::LOAD8_U_I32_A32; 12050b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 12060b57cec5SDimitry Andric break; 12070b57cec5SDimitry Andric case MVT::i16: 12085ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::LOAD16_U_I32_A64 : WebAssembly::LOAD16_U_I32_A32; 12090b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 12100b57cec5SDimitry Andric break; 12110b57cec5SDimitry Andric case MVT::i32: 12125ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::LOAD_I32_A64 : WebAssembly::LOAD_I32_A32; 12130b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 12140b57cec5SDimitry Andric break; 12150b57cec5SDimitry Andric case MVT::i64: 12165ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::LOAD_I64_A64 : WebAssembly::LOAD_I64_A32; 12170b57cec5SDimitry Andric RC = &WebAssembly::I64RegClass; 12180b57cec5SDimitry Andric break; 12190b57cec5SDimitry Andric case MVT::f32: 12205ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::LOAD_F32_A64 : WebAssembly::LOAD_F32_A32; 12210b57cec5SDimitry Andric RC = &WebAssembly::F32RegClass; 12220b57cec5SDimitry Andric break; 12230b57cec5SDimitry Andric case MVT::f64: 12245ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::LOAD_F64_A64 : WebAssembly::LOAD_F64_A32; 12250b57cec5SDimitry Andric RC = &WebAssembly::F64RegClass; 12260b57cec5SDimitry Andric break; 12270b57cec5SDimitry Andric default: 12280b57cec5SDimitry Andric return false; 12290b57cec5SDimitry Andric } 12300b57cec5SDimitry Andric 12310b57cec5SDimitry Andric materializeLoadStoreOperands(Addr); 12320b57cec5SDimitry Andric 123304eeddc0SDimitry Andric Register ResultReg = createResultReg(RC); 1234*bdd1243dSDimitry Andric auto MIB = BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc), 12350b57cec5SDimitry Andric ResultReg); 12360b57cec5SDimitry Andric 12370b57cec5SDimitry Andric addLoadStoreOperands(Addr, MIB, createMachineMemOperandFor(Load)); 12380b57cec5SDimitry Andric 12390b57cec5SDimitry Andric updateValueMap(Load, ResultReg); 12400b57cec5SDimitry Andric return true; 12410b57cec5SDimitry Andric } 12420b57cec5SDimitry Andric 12430b57cec5SDimitry Andric bool WebAssemblyFastISel::selectStore(const Instruction *I) { 12440b57cec5SDimitry Andric const auto *Store = cast<StoreInst>(I); 12450b57cec5SDimitry Andric if (Store->isAtomic()) 12460b57cec5SDimitry Andric return false; 1247fe6060f1SDimitry Andric if (!WebAssembly::isDefaultAddressSpace(Store->getPointerAddressSpace())) 1248fe6060f1SDimitry Andric return false; 12490b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && 12500b57cec5SDimitry Andric Store->getValueOperand()->getType()->isVectorTy()) 12510b57cec5SDimitry Andric return false; 12520b57cec5SDimitry Andric 12530b57cec5SDimitry Andric Address Addr; 12540b57cec5SDimitry Andric if (!computeAddress(Store->getPointerOperand(), Addr)) 12550b57cec5SDimitry Andric return false; 12560b57cec5SDimitry Andric 12570b57cec5SDimitry Andric unsigned Opc; 12580b57cec5SDimitry Andric bool VTIsi1 = false; 12595ffd83dbSDimitry Andric bool A64 = Subtarget->hasAddr64(); 12600b57cec5SDimitry Andric switch (getSimpleType(Store->getValueOperand()->getType())) { 12610b57cec5SDimitry Andric case MVT::i1: 12620b57cec5SDimitry Andric VTIsi1 = true; 1263*bdd1243dSDimitry Andric [[fallthrough]]; 12640b57cec5SDimitry Andric case MVT::i8: 12655ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::STORE8_I32_A64 : WebAssembly::STORE8_I32_A32; 12660b57cec5SDimitry Andric break; 12670b57cec5SDimitry Andric case MVT::i16: 12685ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::STORE16_I32_A64 : WebAssembly::STORE16_I32_A32; 12690b57cec5SDimitry Andric break; 12700b57cec5SDimitry Andric case MVT::i32: 12715ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::STORE_I32_A64 : WebAssembly::STORE_I32_A32; 12720b57cec5SDimitry Andric break; 12730b57cec5SDimitry Andric case MVT::i64: 12745ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::STORE_I64_A64 : WebAssembly::STORE_I64_A32; 12750b57cec5SDimitry Andric break; 12760b57cec5SDimitry Andric case MVT::f32: 12775ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::STORE_F32_A64 : WebAssembly::STORE_F32_A32; 12780b57cec5SDimitry Andric break; 12790b57cec5SDimitry Andric case MVT::f64: 12805ffd83dbSDimitry Andric Opc = A64 ? WebAssembly::STORE_F64_A64 : WebAssembly::STORE_F64_A32; 12810b57cec5SDimitry Andric break; 12820b57cec5SDimitry Andric default: 12830b57cec5SDimitry Andric return false; 12840b57cec5SDimitry Andric } 12850b57cec5SDimitry Andric 12860b57cec5SDimitry Andric materializeLoadStoreOperands(Addr); 12870b57cec5SDimitry Andric 128804eeddc0SDimitry Andric Register ValueReg = getRegForValue(Store->getValueOperand()); 12890b57cec5SDimitry Andric if (ValueReg == 0) 12900b57cec5SDimitry Andric return false; 12910b57cec5SDimitry Andric if (VTIsi1) 12920b57cec5SDimitry Andric ValueReg = maskI1Value(ValueReg, Store->getValueOperand()); 12930b57cec5SDimitry Andric 1294*bdd1243dSDimitry Andric auto MIB = BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc)); 12950b57cec5SDimitry Andric 12960b57cec5SDimitry Andric addLoadStoreOperands(Addr, MIB, createMachineMemOperandFor(Store)); 12970b57cec5SDimitry Andric 12980b57cec5SDimitry Andric MIB.addReg(ValueReg); 12990b57cec5SDimitry Andric return true; 13000b57cec5SDimitry Andric } 13010b57cec5SDimitry Andric 13020b57cec5SDimitry Andric bool WebAssemblyFastISel::selectBr(const Instruction *I) { 13030b57cec5SDimitry Andric const auto *Br = cast<BranchInst>(I); 13040b57cec5SDimitry Andric if (Br->isUnconditional()) { 13050b57cec5SDimitry Andric MachineBasicBlock *MSucc = FuncInfo.MBBMap[Br->getSuccessor(0)]; 13060b57cec5SDimitry Andric fastEmitBranch(MSucc, Br->getDebugLoc()); 13070b57cec5SDimitry Andric return true; 13080b57cec5SDimitry Andric } 13090b57cec5SDimitry Andric 13100b57cec5SDimitry Andric MachineBasicBlock *TBB = FuncInfo.MBBMap[Br->getSuccessor(0)]; 13110b57cec5SDimitry Andric MachineBasicBlock *FBB = FuncInfo.MBBMap[Br->getSuccessor(1)]; 13120b57cec5SDimitry Andric 13130b57cec5SDimitry Andric bool Not; 131469ade1e0SDimitry Andric unsigned CondReg = getRegForI1Value(Br->getCondition(), Br->getParent(), Not); 13150b57cec5SDimitry Andric if (CondReg == 0) 13160b57cec5SDimitry Andric return false; 13170b57cec5SDimitry Andric 13180b57cec5SDimitry Andric unsigned Opc = WebAssembly::BR_IF; 13190b57cec5SDimitry Andric if (Not) 13200b57cec5SDimitry Andric Opc = WebAssembly::BR_UNLESS; 13210b57cec5SDimitry Andric 1322*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, TII.get(Opc)) 13230b57cec5SDimitry Andric .addMBB(TBB) 13240b57cec5SDimitry Andric .addReg(CondReg); 13250b57cec5SDimitry Andric 13260b57cec5SDimitry Andric finishCondBranch(Br->getParent(), TBB, FBB); 13270b57cec5SDimitry Andric return true; 13280b57cec5SDimitry Andric } 13290b57cec5SDimitry Andric 13300b57cec5SDimitry Andric bool WebAssemblyFastISel::selectRet(const Instruction *I) { 13310b57cec5SDimitry Andric if (!FuncInfo.CanLowerReturn) 13320b57cec5SDimitry Andric return false; 13330b57cec5SDimitry Andric 13340b57cec5SDimitry Andric const auto *Ret = cast<ReturnInst>(I); 13350b57cec5SDimitry Andric 13360b57cec5SDimitry Andric if (Ret->getNumOperands() == 0) { 1337*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 13388bcb0991SDimitry Andric TII.get(WebAssembly::RETURN)); 13390b57cec5SDimitry Andric return true; 13400b57cec5SDimitry Andric } 13410b57cec5SDimitry Andric 13428bcb0991SDimitry Andric // TODO: support multiple return in FastISel 13438bcb0991SDimitry Andric if (Ret->getNumOperands() > 1) 13448bcb0991SDimitry Andric return false; 13458bcb0991SDimitry Andric 13460b57cec5SDimitry Andric Value *RV = Ret->getOperand(0); 13470b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && RV->getType()->isVectorTy()) 13480b57cec5SDimitry Andric return false; 13490b57cec5SDimitry Andric 13500b57cec5SDimitry Andric switch (getSimpleType(RV->getType())) { 13510b57cec5SDimitry Andric case MVT::i1: 13520b57cec5SDimitry Andric case MVT::i8: 13530b57cec5SDimitry Andric case MVT::i16: 13540b57cec5SDimitry Andric case MVT::i32: 13550b57cec5SDimitry Andric case MVT::i64: 13560b57cec5SDimitry Andric case MVT::f32: 13570b57cec5SDimitry Andric case MVT::f64: 13580b57cec5SDimitry Andric case MVT::v16i8: 13590b57cec5SDimitry Andric case MVT::v8i16: 13600b57cec5SDimitry Andric case MVT::v4i32: 13610b57cec5SDimitry Andric case MVT::v2i64: 13620b57cec5SDimitry Andric case MVT::v4f32: 13630b57cec5SDimitry Andric case MVT::v2f64: 1364e8d8bef9SDimitry Andric case MVT::funcref: 1365e8d8bef9SDimitry Andric case MVT::externref: 13660b57cec5SDimitry Andric break; 13670b57cec5SDimitry Andric default: 13680b57cec5SDimitry Andric return false; 13690b57cec5SDimitry Andric } 13700b57cec5SDimitry Andric 13710b57cec5SDimitry Andric unsigned Reg; 1372349cc55cSDimitry Andric if (FuncInfo.Fn->getAttributes().hasRetAttr(Attribute::SExt)) 13730b57cec5SDimitry Andric Reg = getRegForSignedValue(RV); 1374349cc55cSDimitry Andric else if (FuncInfo.Fn->getAttributes().hasRetAttr(Attribute::ZExt)) 13750b57cec5SDimitry Andric Reg = getRegForUnsignedValue(RV); 13760b57cec5SDimitry Andric else 13770b57cec5SDimitry Andric Reg = getRegForValue(RV); 13780b57cec5SDimitry Andric 13790b57cec5SDimitry Andric if (Reg == 0) 13800b57cec5SDimitry Andric return false; 13810b57cec5SDimitry Andric 1382*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 13838bcb0991SDimitry Andric TII.get(WebAssembly::RETURN)) 13848bcb0991SDimitry Andric .addReg(Reg); 13850b57cec5SDimitry Andric return true; 13860b57cec5SDimitry Andric } 13870b57cec5SDimitry Andric 13880b57cec5SDimitry Andric bool WebAssemblyFastISel::selectUnreachable(const Instruction *I) { 1389*bdd1243dSDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, MIMD, 13900b57cec5SDimitry Andric TII.get(WebAssembly::UNREACHABLE)); 13910b57cec5SDimitry Andric return true; 13920b57cec5SDimitry Andric } 13930b57cec5SDimitry Andric 13940b57cec5SDimitry Andric bool WebAssemblyFastISel::fastSelectInstruction(const Instruction *I) { 13950b57cec5SDimitry Andric switch (I->getOpcode()) { 13960b57cec5SDimitry Andric case Instruction::Call: 13970b57cec5SDimitry Andric if (selectCall(I)) 13980b57cec5SDimitry Andric return true; 13990b57cec5SDimitry Andric break; 14000b57cec5SDimitry Andric case Instruction::Select: 14010b57cec5SDimitry Andric return selectSelect(I); 14020b57cec5SDimitry Andric case Instruction::Trunc: 14030b57cec5SDimitry Andric return selectTrunc(I); 14040b57cec5SDimitry Andric case Instruction::ZExt: 14050b57cec5SDimitry Andric return selectZExt(I); 14060b57cec5SDimitry Andric case Instruction::SExt: 14070b57cec5SDimitry Andric return selectSExt(I); 14080b57cec5SDimitry Andric case Instruction::ICmp: 14090b57cec5SDimitry Andric return selectICmp(I); 14100b57cec5SDimitry Andric case Instruction::FCmp: 14110b57cec5SDimitry Andric return selectFCmp(I); 14120b57cec5SDimitry Andric case Instruction::BitCast: 14130b57cec5SDimitry Andric return selectBitCast(I); 14140b57cec5SDimitry Andric case Instruction::Load: 14150b57cec5SDimitry Andric return selectLoad(I); 14160b57cec5SDimitry Andric case Instruction::Store: 14170b57cec5SDimitry Andric return selectStore(I); 14180b57cec5SDimitry Andric case Instruction::Br: 14190b57cec5SDimitry Andric return selectBr(I); 14200b57cec5SDimitry Andric case Instruction::Ret: 14210b57cec5SDimitry Andric return selectRet(I); 14220b57cec5SDimitry Andric case Instruction::Unreachable: 14230b57cec5SDimitry Andric return selectUnreachable(I); 14240b57cec5SDimitry Andric default: 14250b57cec5SDimitry Andric break; 14260b57cec5SDimitry Andric } 14270b57cec5SDimitry Andric 14280b57cec5SDimitry Andric // Fall back to target-independent instruction selection. 14290b57cec5SDimitry Andric return selectOperator(I, I->getOpcode()); 14300b57cec5SDimitry Andric } 14310b57cec5SDimitry Andric 14320b57cec5SDimitry Andric FastISel *WebAssembly::createFastISel(FunctionLoweringInfo &FuncInfo, 14330b57cec5SDimitry Andric const TargetLibraryInfo *LibInfo) { 14340b57cec5SDimitry Andric return new WebAssemblyFastISel(FuncInfo, LibInfo); 14350b57cec5SDimitry Andric } 1436