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