1*0b57cec5SDimitry Andric //===-- WebAssemblyFastISel.cpp - WebAssembly FastISel implementation -----===// 2*0b57cec5SDimitry Andric // 3*0b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*0b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5*0b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*0b57cec5SDimitry Andric // 7*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 8*0b57cec5SDimitry Andric /// 9*0b57cec5SDimitry Andric /// \file 10*0b57cec5SDimitry Andric /// This file defines the WebAssembly-specific support for the FastISel 11*0b57cec5SDimitry Andric /// class. Some of the target-specific code is generated by tablegen in the file 12*0b57cec5SDimitry Andric /// WebAssemblyGenFastISel.inc, which is #included here. 13*0b57cec5SDimitry Andric /// 14*0b57cec5SDimitry Andric /// TODO: kill flags 15*0b57cec5SDimitry Andric /// 16*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 17*0b57cec5SDimitry Andric 18*0b57cec5SDimitry Andric #include "MCTargetDesc/WebAssemblyMCTargetDesc.h" 19*0b57cec5SDimitry Andric #include "WebAssembly.h" 20*0b57cec5SDimitry Andric #include "WebAssemblyMachineFunctionInfo.h" 21*0b57cec5SDimitry Andric #include "WebAssemblySubtarget.h" 22*0b57cec5SDimitry Andric #include "WebAssemblyTargetMachine.h" 23*0b57cec5SDimitry Andric #include "llvm/Analysis/BranchProbabilityInfo.h" 24*0b57cec5SDimitry Andric #include "llvm/CodeGen/FastISel.h" 25*0b57cec5SDimitry Andric #include "llvm/CodeGen/FunctionLoweringInfo.h" 26*0b57cec5SDimitry Andric #include "llvm/CodeGen/MachineConstantPool.h" 27*0b57cec5SDimitry Andric #include "llvm/CodeGen/MachineFrameInfo.h" 28*0b57cec5SDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h" 29*0b57cec5SDimitry Andric #include "llvm/CodeGen/MachineRegisterInfo.h" 30*0b57cec5SDimitry Andric #include "llvm/IR/DataLayout.h" 31*0b57cec5SDimitry Andric #include "llvm/IR/DerivedTypes.h" 32*0b57cec5SDimitry Andric #include "llvm/IR/Function.h" 33*0b57cec5SDimitry Andric #include "llvm/IR/GetElementPtrTypeIterator.h" 34*0b57cec5SDimitry Andric #include "llvm/IR/GlobalAlias.h" 35*0b57cec5SDimitry Andric #include "llvm/IR/GlobalVariable.h" 36*0b57cec5SDimitry Andric #include "llvm/IR/Instructions.h" 37*0b57cec5SDimitry Andric #include "llvm/IR/IntrinsicInst.h" 38*0b57cec5SDimitry Andric #include "llvm/IR/Operator.h" 39*0b57cec5SDimitry Andric #include "llvm/IR/PatternMatch.h" 40*0b57cec5SDimitry Andric 41*0b57cec5SDimitry Andric using namespace llvm; 42*0b57cec5SDimitry Andric using namespace PatternMatch; 43*0b57cec5SDimitry Andric 44*0b57cec5SDimitry Andric #define DEBUG_TYPE "wasm-fastisel" 45*0b57cec5SDimitry Andric 46*0b57cec5SDimitry Andric namespace { 47*0b57cec5SDimitry Andric 48*0b57cec5SDimitry Andric class WebAssemblyFastISel final : public FastISel { 49*0b57cec5SDimitry Andric // All possible address modes. 50*0b57cec5SDimitry Andric class Address { 51*0b57cec5SDimitry Andric public: 52*0b57cec5SDimitry Andric using BaseKind = enum { RegBase, FrameIndexBase }; 53*0b57cec5SDimitry Andric 54*0b57cec5SDimitry Andric private: 55*0b57cec5SDimitry Andric BaseKind Kind = RegBase; 56*0b57cec5SDimitry Andric union { 57*0b57cec5SDimitry Andric unsigned Reg; 58*0b57cec5SDimitry Andric int FI; 59*0b57cec5SDimitry Andric } Base; 60*0b57cec5SDimitry Andric 61*0b57cec5SDimitry Andric int64_t Offset = 0; 62*0b57cec5SDimitry Andric 63*0b57cec5SDimitry Andric const GlobalValue *GV = nullptr; 64*0b57cec5SDimitry Andric 65*0b57cec5SDimitry Andric public: 66*0b57cec5SDimitry Andric // Innocuous defaults for our address. 67*0b57cec5SDimitry Andric Address() { Base.Reg = 0; } 68*0b57cec5SDimitry Andric void setKind(BaseKind K) { 69*0b57cec5SDimitry Andric assert(!isSet() && "Can't change kind with non-zero base"); 70*0b57cec5SDimitry Andric Kind = K; 71*0b57cec5SDimitry Andric } 72*0b57cec5SDimitry Andric BaseKind getKind() const { return Kind; } 73*0b57cec5SDimitry Andric bool isRegBase() const { return Kind == RegBase; } 74*0b57cec5SDimitry Andric bool isFIBase() const { return Kind == FrameIndexBase; } 75*0b57cec5SDimitry Andric void setReg(unsigned Reg) { 76*0b57cec5SDimitry Andric assert(isRegBase() && "Invalid base register access!"); 77*0b57cec5SDimitry Andric assert(Base.Reg == 0 && "Overwriting non-zero register"); 78*0b57cec5SDimitry Andric Base.Reg = Reg; 79*0b57cec5SDimitry Andric } 80*0b57cec5SDimitry Andric unsigned getReg() const { 81*0b57cec5SDimitry Andric assert(isRegBase() && "Invalid base register access!"); 82*0b57cec5SDimitry Andric return Base.Reg; 83*0b57cec5SDimitry Andric } 84*0b57cec5SDimitry Andric void setFI(unsigned FI) { 85*0b57cec5SDimitry Andric assert(isFIBase() && "Invalid base frame index access!"); 86*0b57cec5SDimitry Andric assert(Base.FI == 0 && "Overwriting non-zero frame index"); 87*0b57cec5SDimitry Andric Base.FI = FI; 88*0b57cec5SDimitry Andric } 89*0b57cec5SDimitry Andric unsigned getFI() const { 90*0b57cec5SDimitry Andric assert(isFIBase() && "Invalid base frame index access!"); 91*0b57cec5SDimitry Andric return Base.FI; 92*0b57cec5SDimitry Andric } 93*0b57cec5SDimitry Andric 94*0b57cec5SDimitry Andric void setOffset(int64_t NewOffset) { 95*0b57cec5SDimitry Andric assert(NewOffset >= 0 && "Offsets must be non-negative"); 96*0b57cec5SDimitry Andric Offset = NewOffset; 97*0b57cec5SDimitry Andric } 98*0b57cec5SDimitry Andric int64_t getOffset() const { return Offset; } 99*0b57cec5SDimitry Andric void setGlobalValue(const GlobalValue *G) { GV = G; } 100*0b57cec5SDimitry Andric const GlobalValue *getGlobalValue() const { return GV; } 101*0b57cec5SDimitry Andric bool isSet() const { 102*0b57cec5SDimitry Andric if (isRegBase()) { 103*0b57cec5SDimitry Andric return Base.Reg != 0; 104*0b57cec5SDimitry Andric } else { 105*0b57cec5SDimitry Andric return Base.FI != 0; 106*0b57cec5SDimitry Andric } 107*0b57cec5SDimitry Andric } 108*0b57cec5SDimitry Andric }; 109*0b57cec5SDimitry Andric 110*0b57cec5SDimitry Andric /// Keep a pointer to the WebAssemblySubtarget around so that we can make the 111*0b57cec5SDimitry Andric /// right decision when generating code for different targets. 112*0b57cec5SDimitry Andric const WebAssemblySubtarget *Subtarget; 113*0b57cec5SDimitry Andric LLVMContext *Context; 114*0b57cec5SDimitry Andric 115*0b57cec5SDimitry Andric private: 116*0b57cec5SDimitry Andric // Utility helper routines 117*0b57cec5SDimitry Andric MVT::SimpleValueType getSimpleType(Type *Ty) { 118*0b57cec5SDimitry Andric EVT VT = TLI.getValueType(DL, Ty, /*AllowUnknown=*/true); 119*0b57cec5SDimitry Andric return VT.isSimple() ? VT.getSimpleVT().SimpleTy 120*0b57cec5SDimitry Andric : MVT::INVALID_SIMPLE_VALUE_TYPE; 121*0b57cec5SDimitry Andric } 122*0b57cec5SDimitry Andric MVT::SimpleValueType getLegalType(MVT::SimpleValueType VT) { 123*0b57cec5SDimitry Andric switch (VT) { 124*0b57cec5SDimitry Andric case MVT::i1: 125*0b57cec5SDimitry Andric case MVT::i8: 126*0b57cec5SDimitry Andric case MVT::i16: 127*0b57cec5SDimitry Andric return MVT::i32; 128*0b57cec5SDimitry Andric case MVT::i32: 129*0b57cec5SDimitry Andric case MVT::i64: 130*0b57cec5SDimitry Andric case MVT::f32: 131*0b57cec5SDimitry Andric case MVT::f64: 132*0b57cec5SDimitry Andric case MVT::exnref: 133*0b57cec5SDimitry Andric return VT; 134*0b57cec5SDimitry Andric case MVT::f16: 135*0b57cec5SDimitry Andric return MVT::f32; 136*0b57cec5SDimitry Andric case MVT::v16i8: 137*0b57cec5SDimitry Andric case MVT::v8i16: 138*0b57cec5SDimitry Andric case MVT::v4i32: 139*0b57cec5SDimitry Andric case MVT::v4f32: 140*0b57cec5SDimitry Andric if (Subtarget->hasSIMD128()) 141*0b57cec5SDimitry Andric return VT; 142*0b57cec5SDimitry Andric break; 143*0b57cec5SDimitry Andric case MVT::v2i64: 144*0b57cec5SDimitry Andric case MVT::v2f64: 145*0b57cec5SDimitry Andric if (Subtarget->hasUnimplementedSIMD128()) 146*0b57cec5SDimitry Andric return VT; 147*0b57cec5SDimitry Andric break; 148*0b57cec5SDimitry Andric default: 149*0b57cec5SDimitry Andric break; 150*0b57cec5SDimitry Andric } 151*0b57cec5SDimitry Andric return MVT::INVALID_SIMPLE_VALUE_TYPE; 152*0b57cec5SDimitry Andric } 153*0b57cec5SDimitry Andric bool computeAddress(const Value *Obj, Address &Addr); 154*0b57cec5SDimitry Andric void materializeLoadStoreOperands(Address &Addr); 155*0b57cec5SDimitry Andric void addLoadStoreOperands(const Address &Addr, const MachineInstrBuilder &MIB, 156*0b57cec5SDimitry Andric MachineMemOperand *MMO); 157*0b57cec5SDimitry Andric unsigned maskI1Value(unsigned Reg, const Value *V); 158*0b57cec5SDimitry Andric unsigned getRegForI1Value(const Value *V, bool &Not); 159*0b57cec5SDimitry Andric unsigned zeroExtendToI32(unsigned Reg, const Value *V, 160*0b57cec5SDimitry Andric MVT::SimpleValueType From); 161*0b57cec5SDimitry Andric unsigned signExtendToI32(unsigned Reg, const Value *V, 162*0b57cec5SDimitry Andric MVT::SimpleValueType From); 163*0b57cec5SDimitry Andric unsigned zeroExtend(unsigned Reg, const Value *V, MVT::SimpleValueType From, 164*0b57cec5SDimitry Andric MVT::SimpleValueType To); 165*0b57cec5SDimitry Andric unsigned signExtend(unsigned Reg, const Value *V, MVT::SimpleValueType From, 166*0b57cec5SDimitry Andric MVT::SimpleValueType To); 167*0b57cec5SDimitry Andric unsigned getRegForUnsignedValue(const Value *V); 168*0b57cec5SDimitry Andric unsigned getRegForSignedValue(const Value *V); 169*0b57cec5SDimitry Andric unsigned getRegForPromotedValue(const Value *V, bool IsSigned); 170*0b57cec5SDimitry Andric unsigned notValue(unsigned Reg); 171*0b57cec5SDimitry Andric unsigned copyValue(unsigned Reg); 172*0b57cec5SDimitry Andric 173*0b57cec5SDimitry Andric // Backend specific FastISel code. 174*0b57cec5SDimitry Andric unsigned fastMaterializeAlloca(const AllocaInst *AI) override; 175*0b57cec5SDimitry Andric unsigned fastMaterializeConstant(const Constant *C) override; 176*0b57cec5SDimitry Andric bool fastLowerArguments() override; 177*0b57cec5SDimitry Andric 178*0b57cec5SDimitry Andric // Selection routines. 179*0b57cec5SDimitry Andric bool selectCall(const Instruction *I); 180*0b57cec5SDimitry Andric bool selectSelect(const Instruction *I); 181*0b57cec5SDimitry Andric bool selectTrunc(const Instruction *I); 182*0b57cec5SDimitry Andric bool selectZExt(const Instruction *I); 183*0b57cec5SDimitry Andric bool selectSExt(const Instruction *I); 184*0b57cec5SDimitry Andric bool selectICmp(const Instruction *I); 185*0b57cec5SDimitry Andric bool selectFCmp(const Instruction *I); 186*0b57cec5SDimitry Andric bool selectBitCast(const Instruction *I); 187*0b57cec5SDimitry Andric bool selectLoad(const Instruction *I); 188*0b57cec5SDimitry Andric bool selectStore(const Instruction *I); 189*0b57cec5SDimitry Andric bool selectBr(const Instruction *I); 190*0b57cec5SDimitry Andric bool selectRet(const Instruction *I); 191*0b57cec5SDimitry Andric bool selectUnreachable(const Instruction *I); 192*0b57cec5SDimitry Andric 193*0b57cec5SDimitry Andric public: 194*0b57cec5SDimitry Andric // Backend specific FastISel code. 195*0b57cec5SDimitry Andric WebAssemblyFastISel(FunctionLoweringInfo &FuncInfo, 196*0b57cec5SDimitry Andric const TargetLibraryInfo *LibInfo) 197*0b57cec5SDimitry Andric : FastISel(FuncInfo, LibInfo, /*SkipTargetIndependentISel=*/true) { 198*0b57cec5SDimitry Andric Subtarget = &FuncInfo.MF->getSubtarget<WebAssemblySubtarget>(); 199*0b57cec5SDimitry Andric Context = &FuncInfo.Fn->getContext(); 200*0b57cec5SDimitry Andric } 201*0b57cec5SDimitry Andric 202*0b57cec5SDimitry Andric bool fastSelectInstruction(const Instruction *I) override; 203*0b57cec5SDimitry Andric 204*0b57cec5SDimitry Andric #include "WebAssemblyGenFastISel.inc" 205*0b57cec5SDimitry Andric }; 206*0b57cec5SDimitry Andric 207*0b57cec5SDimitry Andric } // end anonymous namespace 208*0b57cec5SDimitry Andric 209*0b57cec5SDimitry Andric bool WebAssemblyFastISel::computeAddress(const Value *Obj, Address &Addr) { 210*0b57cec5SDimitry Andric const User *U = nullptr; 211*0b57cec5SDimitry Andric unsigned Opcode = Instruction::UserOp1; 212*0b57cec5SDimitry Andric if (const auto *I = dyn_cast<Instruction>(Obj)) { 213*0b57cec5SDimitry Andric // Don't walk into other basic blocks unless the object is an alloca from 214*0b57cec5SDimitry Andric // another block, otherwise it may not have a virtual register assigned. 215*0b57cec5SDimitry Andric if (FuncInfo.StaticAllocaMap.count(static_cast<const AllocaInst *>(Obj)) || 216*0b57cec5SDimitry Andric FuncInfo.MBBMap[I->getParent()] == FuncInfo.MBB) { 217*0b57cec5SDimitry Andric Opcode = I->getOpcode(); 218*0b57cec5SDimitry Andric U = I; 219*0b57cec5SDimitry Andric } 220*0b57cec5SDimitry Andric } else if (const auto *C = dyn_cast<ConstantExpr>(Obj)) { 221*0b57cec5SDimitry Andric Opcode = C->getOpcode(); 222*0b57cec5SDimitry Andric U = C; 223*0b57cec5SDimitry Andric } 224*0b57cec5SDimitry Andric 225*0b57cec5SDimitry Andric if (auto *Ty = dyn_cast<PointerType>(Obj->getType())) 226*0b57cec5SDimitry Andric if (Ty->getAddressSpace() > 255) 227*0b57cec5SDimitry Andric // Fast instruction selection doesn't support the special 228*0b57cec5SDimitry Andric // address spaces. 229*0b57cec5SDimitry Andric return false; 230*0b57cec5SDimitry Andric 231*0b57cec5SDimitry Andric if (const auto *GV = dyn_cast<GlobalValue>(Obj)) { 232*0b57cec5SDimitry Andric if (TLI.isPositionIndependent()) 233*0b57cec5SDimitry Andric return false; 234*0b57cec5SDimitry Andric if (Addr.getGlobalValue()) 235*0b57cec5SDimitry Andric return false; 236*0b57cec5SDimitry Andric if (GV->isThreadLocal()) 237*0b57cec5SDimitry Andric return false; 238*0b57cec5SDimitry Andric Addr.setGlobalValue(GV); 239*0b57cec5SDimitry Andric return true; 240*0b57cec5SDimitry Andric } 241*0b57cec5SDimitry Andric 242*0b57cec5SDimitry Andric switch (Opcode) { 243*0b57cec5SDimitry Andric default: 244*0b57cec5SDimitry Andric break; 245*0b57cec5SDimitry Andric case Instruction::BitCast: { 246*0b57cec5SDimitry Andric // Look through bitcasts. 247*0b57cec5SDimitry Andric return computeAddress(U->getOperand(0), Addr); 248*0b57cec5SDimitry Andric } 249*0b57cec5SDimitry Andric case Instruction::IntToPtr: { 250*0b57cec5SDimitry Andric // Look past no-op inttoptrs. 251*0b57cec5SDimitry Andric if (TLI.getValueType(DL, U->getOperand(0)->getType()) == 252*0b57cec5SDimitry Andric TLI.getPointerTy(DL)) 253*0b57cec5SDimitry Andric return computeAddress(U->getOperand(0), Addr); 254*0b57cec5SDimitry Andric break; 255*0b57cec5SDimitry Andric } 256*0b57cec5SDimitry Andric case Instruction::PtrToInt: { 257*0b57cec5SDimitry Andric // Look past no-op ptrtoints. 258*0b57cec5SDimitry Andric if (TLI.getValueType(DL, U->getType()) == TLI.getPointerTy(DL)) 259*0b57cec5SDimitry Andric return computeAddress(U->getOperand(0), Addr); 260*0b57cec5SDimitry Andric break; 261*0b57cec5SDimitry Andric } 262*0b57cec5SDimitry Andric case Instruction::GetElementPtr: { 263*0b57cec5SDimitry Andric Address SavedAddr = Addr; 264*0b57cec5SDimitry Andric uint64_t TmpOffset = Addr.getOffset(); 265*0b57cec5SDimitry Andric // Non-inbounds geps can wrap; wasm's offsets can't. 266*0b57cec5SDimitry Andric if (!cast<GEPOperator>(U)->isInBounds()) 267*0b57cec5SDimitry Andric goto unsupported_gep; 268*0b57cec5SDimitry Andric // Iterate through the GEP folding the constants into offsets where 269*0b57cec5SDimitry Andric // we can. 270*0b57cec5SDimitry Andric for (gep_type_iterator GTI = gep_type_begin(U), E = gep_type_end(U); 271*0b57cec5SDimitry Andric GTI != E; ++GTI) { 272*0b57cec5SDimitry Andric const Value *Op = GTI.getOperand(); 273*0b57cec5SDimitry Andric if (StructType *STy = GTI.getStructTypeOrNull()) { 274*0b57cec5SDimitry Andric const StructLayout *SL = DL.getStructLayout(STy); 275*0b57cec5SDimitry Andric unsigned Idx = cast<ConstantInt>(Op)->getZExtValue(); 276*0b57cec5SDimitry Andric TmpOffset += SL->getElementOffset(Idx); 277*0b57cec5SDimitry Andric } else { 278*0b57cec5SDimitry Andric uint64_t S = DL.getTypeAllocSize(GTI.getIndexedType()); 279*0b57cec5SDimitry Andric for (;;) { 280*0b57cec5SDimitry Andric if (const auto *CI = dyn_cast<ConstantInt>(Op)) { 281*0b57cec5SDimitry Andric // Constant-offset addressing. 282*0b57cec5SDimitry Andric TmpOffset += CI->getSExtValue() * S; 283*0b57cec5SDimitry Andric break; 284*0b57cec5SDimitry Andric } 285*0b57cec5SDimitry Andric if (S == 1 && Addr.isRegBase() && Addr.getReg() == 0) { 286*0b57cec5SDimitry Andric // An unscaled add of a register. Set it as the new base. 287*0b57cec5SDimitry Andric unsigned Reg = getRegForValue(Op); 288*0b57cec5SDimitry Andric if (Reg == 0) 289*0b57cec5SDimitry Andric return false; 290*0b57cec5SDimitry Andric Addr.setReg(Reg); 291*0b57cec5SDimitry Andric break; 292*0b57cec5SDimitry Andric } 293*0b57cec5SDimitry Andric if (canFoldAddIntoGEP(U, Op)) { 294*0b57cec5SDimitry Andric // A compatible add with a constant operand. Fold the constant. 295*0b57cec5SDimitry Andric auto *CI = cast<ConstantInt>(cast<AddOperator>(Op)->getOperand(1)); 296*0b57cec5SDimitry Andric TmpOffset += CI->getSExtValue() * S; 297*0b57cec5SDimitry Andric // Iterate on the other operand. 298*0b57cec5SDimitry Andric Op = cast<AddOperator>(Op)->getOperand(0); 299*0b57cec5SDimitry Andric continue; 300*0b57cec5SDimitry Andric } 301*0b57cec5SDimitry Andric // Unsupported 302*0b57cec5SDimitry Andric goto unsupported_gep; 303*0b57cec5SDimitry Andric } 304*0b57cec5SDimitry Andric } 305*0b57cec5SDimitry Andric } 306*0b57cec5SDimitry Andric // Don't fold in negative offsets. 307*0b57cec5SDimitry Andric if (int64_t(TmpOffset) >= 0) { 308*0b57cec5SDimitry Andric // Try to grab the base operand now. 309*0b57cec5SDimitry Andric Addr.setOffset(TmpOffset); 310*0b57cec5SDimitry Andric if (computeAddress(U->getOperand(0), Addr)) 311*0b57cec5SDimitry Andric return true; 312*0b57cec5SDimitry Andric } 313*0b57cec5SDimitry Andric // We failed, restore everything and try the other options. 314*0b57cec5SDimitry Andric Addr = SavedAddr; 315*0b57cec5SDimitry Andric unsupported_gep: 316*0b57cec5SDimitry Andric break; 317*0b57cec5SDimitry Andric } 318*0b57cec5SDimitry Andric case Instruction::Alloca: { 319*0b57cec5SDimitry Andric const auto *AI = cast<AllocaInst>(Obj); 320*0b57cec5SDimitry Andric DenseMap<const AllocaInst *, int>::iterator SI = 321*0b57cec5SDimitry Andric FuncInfo.StaticAllocaMap.find(AI); 322*0b57cec5SDimitry Andric if (SI != FuncInfo.StaticAllocaMap.end()) { 323*0b57cec5SDimitry Andric if (Addr.isSet()) { 324*0b57cec5SDimitry Andric return false; 325*0b57cec5SDimitry Andric } 326*0b57cec5SDimitry Andric Addr.setKind(Address::FrameIndexBase); 327*0b57cec5SDimitry Andric Addr.setFI(SI->second); 328*0b57cec5SDimitry Andric return true; 329*0b57cec5SDimitry Andric } 330*0b57cec5SDimitry Andric break; 331*0b57cec5SDimitry Andric } 332*0b57cec5SDimitry Andric case Instruction::Add: { 333*0b57cec5SDimitry Andric // Adds of constants are common and easy enough. 334*0b57cec5SDimitry Andric const Value *LHS = U->getOperand(0); 335*0b57cec5SDimitry Andric const Value *RHS = U->getOperand(1); 336*0b57cec5SDimitry Andric 337*0b57cec5SDimitry Andric if (isa<ConstantInt>(LHS)) 338*0b57cec5SDimitry Andric std::swap(LHS, RHS); 339*0b57cec5SDimitry Andric 340*0b57cec5SDimitry Andric if (const auto *CI = dyn_cast<ConstantInt>(RHS)) { 341*0b57cec5SDimitry Andric uint64_t TmpOffset = Addr.getOffset() + CI->getSExtValue(); 342*0b57cec5SDimitry Andric if (int64_t(TmpOffset) >= 0) { 343*0b57cec5SDimitry Andric Addr.setOffset(TmpOffset); 344*0b57cec5SDimitry Andric return computeAddress(LHS, Addr); 345*0b57cec5SDimitry Andric } 346*0b57cec5SDimitry Andric } 347*0b57cec5SDimitry Andric 348*0b57cec5SDimitry Andric Address Backup = Addr; 349*0b57cec5SDimitry Andric if (computeAddress(LHS, Addr) && computeAddress(RHS, Addr)) 350*0b57cec5SDimitry Andric return true; 351*0b57cec5SDimitry Andric Addr = Backup; 352*0b57cec5SDimitry Andric 353*0b57cec5SDimitry Andric break; 354*0b57cec5SDimitry Andric } 355*0b57cec5SDimitry Andric case Instruction::Sub: { 356*0b57cec5SDimitry Andric // Subs of constants are common and easy enough. 357*0b57cec5SDimitry Andric const Value *LHS = U->getOperand(0); 358*0b57cec5SDimitry Andric const Value *RHS = U->getOperand(1); 359*0b57cec5SDimitry Andric 360*0b57cec5SDimitry Andric if (const auto *CI = dyn_cast<ConstantInt>(RHS)) { 361*0b57cec5SDimitry Andric int64_t TmpOffset = Addr.getOffset() - CI->getSExtValue(); 362*0b57cec5SDimitry Andric if (TmpOffset >= 0) { 363*0b57cec5SDimitry Andric Addr.setOffset(TmpOffset); 364*0b57cec5SDimitry Andric return computeAddress(LHS, Addr); 365*0b57cec5SDimitry Andric } 366*0b57cec5SDimitry Andric } 367*0b57cec5SDimitry Andric break; 368*0b57cec5SDimitry Andric } 369*0b57cec5SDimitry Andric } 370*0b57cec5SDimitry Andric if (Addr.isSet()) { 371*0b57cec5SDimitry Andric return false; 372*0b57cec5SDimitry Andric } 373*0b57cec5SDimitry Andric unsigned Reg = getRegForValue(Obj); 374*0b57cec5SDimitry Andric if (Reg == 0) 375*0b57cec5SDimitry Andric return false; 376*0b57cec5SDimitry Andric Addr.setReg(Reg); 377*0b57cec5SDimitry Andric return Addr.getReg() != 0; 378*0b57cec5SDimitry Andric } 379*0b57cec5SDimitry Andric 380*0b57cec5SDimitry Andric void WebAssemblyFastISel::materializeLoadStoreOperands(Address &Addr) { 381*0b57cec5SDimitry Andric if (Addr.isRegBase()) { 382*0b57cec5SDimitry Andric unsigned Reg = Addr.getReg(); 383*0b57cec5SDimitry Andric if (Reg == 0) { 384*0b57cec5SDimitry Andric Reg = createResultReg(Subtarget->hasAddr64() ? &WebAssembly::I64RegClass 385*0b57cec5SDimitry Andric : &WebAssembly::I32RegClass); 386*0b57cec5SDimitry Andric unsigned Opc = Subtarget->hasAddr64() ? WebAssembly::CONST_I64 387*0b57cec5SDimitry Andric : WebAssembly::CONST_I32; 388*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), Reg) 389*0b57cec5SDimitry Andric .addImm(0); 390*0b57cec5SDimitry Andric Addr.setReg(Reg); 391*0b57cec5SDimitry Andric } 392*0b57cec5SDimitry Andric } 393*0b57cec5SDimitry Andric } 394*0b57cec5SDimitry Andric 395*0b57cec5SDimitry Andric void WebAssemblyFastISel::addLoadStoreOperands(const Address &Addr, 396*0b57cec5SDimitry Andric const MachineInstrBuilder &MIB, 397*0b57cec5SDimitry Andric MachineMemOperand *MMO) { 398*0b57cec5SDimitry Andric // Set the alignment operand (this is rewritten in SetP2AlignOperands). 399*0b57cec5SDimitry Andric // TODO: Disable SetP2AlignOperands for FastISel and just do it here. 400*0b57cec5SDimitry Andric MIB.addImm(0); 401*0b57cec5SDimitry Andric 402*0b57cec5SDimitry Andric if (const GlobalValue *GV = Addr.getGlobalValue()) 403*0b57cec5SDimitry Andric MIB.addGlobalAddress(GV, Addr.getOffset()); 404*0b57cec5SDimitry Andric else 405*0b57cec5SDimitry Andric MIB.addImm(Addr.getOffset()); 406*0b57cec5SDimitry Andric 407*0b57cec5SDimitry Andric if (Addr.isRegBase()) 408*0b57cec5SDimitry Andric MIB.addReg(Addr.getReg()); 409*0b57cec5SDimitry Andric else 410*0b57cec5SDimitry Andric MIB.addFrameIndex(Addr.getFI()); 411*0b57cec5SDimitry Andric 412*0b57cec5SDimitry Andric MIB.addMemOperand(MMO); 413*0b57cec5SDimitry Andric } 414*0b57cec5SDimitry Andric 415*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::maskI1Value(unsigned Reg, const Value *V) { 416*0b57cec5SDimitry Andric return zeroExtendToI32(Reg, V, MVT::i1); 417*0b57cec5SDimitry Andric } 418*0b57cec5SDimitry Andric 419*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForI1Value(const Value *V, bool &Not) { 420*0b57cec5SDimitry Andric if (const auto *ICmp = dyn_cast<ICmpInst>(V)) 421*0b57cec5SDimitry Andric if (const ConstantInt *C = dyn_cast<ConstantInt>(ICmp->getOperand(1))) 422*0b57cec5SDimitry Andric if (ICmp->isEquality() && C->isZero() && C->getType()->isIntegerTy(32)) { 423*0b57cec5SDimitry Andric Not = ICmp->isTrueWhenEqual(); 424*0b57cec5SDimitry Andric return getRegForValue(ICmp->getOperand(0)); 425*0b57cec5SDimitry Andric } 426*0b57cec5SDimitry Andric 427*0b57cec5SDimitry Andric Value *NotV; 428*0b57cec5SDimitry Andric if (match(V, m_Not(m_Value(NotV))) && V->getType()->isIntegerTy(32)) { 429*0b57cec5SDimitry Andric Not = true; 430*0b57cec5SDimitry Andric return getRegForValue(NotV); 431*0b57cec5SDimitry Andric } 432*0b57cec5SDimitry Andric 433*0b57cec5SDimitry Andric Not = false; 434*0b57cec5SDimitry Andric unsigned Reg = getRegForValue(V); 435*0b57cec5SDimitry Andric if (Reg == 0) 436*0b57cec5SDimitry Andric return 0; 437*0b57cec5SDimitry Andric return maskI1Value(Reg, V); 438*0b57cec5SDimitry Andric } 439*0b57cec5SDimitry Andric 440*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::zeroExtendToI32(unsigned Reg, const Value *V, 441*0b57cec5SDimitry Andric MVT::SimpleValueType From) { 442*0b57cec5SDimitry Andric if (Reg == 0) 443*0b57cec5SDimitry Andric return 0; 444*0b57cec5SDimitry Andric 445*0b57cec5SDimitry Andric switch (From) { 446*0b57cec5SDimitry Andric case MVT::i1: 447*0b57cec5SDimitry Andric // If the value is naturally an i1, we don't need to mask it. We only know 448*0b57cec5SDimitry Andric // if a value is naturally an i1 if it is definitely lowered by FastISel, 449*0b57cec5SDimitry Andric // not a DAG ISel fallback. 450*0b57cec5SDimitry Andric if (V != nullptr && isa<Argument>(V) && cast<Argument>(V)->hasZExtAttr()) 451*0b57cec5SDimitry Andric return copyValue(Reg); 452*0b57cec5SDimitry Andric break; 453*0b57cec5SDimitry Andric case MVT::i8: 454*0b57cec5SDimitry Andric case MVT::i16: 455*0b57cec5SDimitry Andric break; 456*0b57cec5SDimitry Andric case MVT::i32: 457*0b57cec5SDimitry Andric return copyValue(Reg); 458*0b57cec5SDimitry Andric default: 459*0b57cec5SDimitry Andric return 0; 460*0b57cec5SDimitry Andric } 461*0b57cec5SDimitry Andric 462*0b57cec5SDimitry Andric unsigned Imm = createResultReg(&WebAssembly::I32RegClass); 463*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 464*0b57cec5SDimitry Andric TII.get(WebAssembly::CONST_I32), Imm) 465*0b57cec5SDimitry Andric .addImm(~(~uint64_t(0) << MVT(From).getSizeInBits())); 466*0b57cec5SDimitry Andric 467*0b57cec5SDimitry Andric unsigned Result = createResultReg(&WebAssembly::I32RegClass); 468*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 469*0b57cec5SDimitry Andric TII.get(WebAssembly::AND_I32), Result) 470*0b57cec5SDimitry Andric .addReg(Reg) 471*0b57cec5SDimitry Andric .addReg(Imm); 472*0b57cec5SDimitry Andric 473*0b57cec5SDimitry Andric return Result; 474*0b57cec5SDimitry Andric } 475*0b57cec5SDimitry Andric 476*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::signExtendToI32(unsigned Reg, const Value *V, 477*0b57cec5SDimitry Andric MVT::SimpleValueType From) { 478*0b57cec5SDimitry Andric if (Reg == 0) 479*0b57cec5SDimitry Andric return 0; 480*0b57cec5SDimitry Andric 481*0b57cec5SDimitry Andric switch (From) { 482*0b57cec5SDimitry Andric case MVT::i1: 483*0b57cec5SDimitry Andric case MVT::i8: 484*0b57cec5SDimitry Andric case MVT::i16: 485*0b57cec5SDimitry Andric break; 486*0b57cec5SDimitry Andric case MVT::i32: 487*0b57cec5SDimitry Andric return copyValue(Reg); 488*0b57cec5SDimitry Andric default: 489*0b57cec5SDimitry Andric return 0; 490*0b57cec5SDimitry Andric } 491*0b57cec5SDimitry Andric 492*0b57cec5SDimitry Andric unsigned Imm = createResultReg(&WebAssembly::I32RegClass); 493*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 494*0b57cec5SDimitry Andric TII.get(WebAssembly::CONST_I32), Imm) 495*0b57cec5SDimitry Andric .addImm(32 - MVT(From).getSizeInBits()); 496*0b57cec5SDimitry Andric 497*0b57cec5SDimitry Andric unsigned Left = createResultReg(&WebAssembly::I32RegClass); 498*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 499*0b57cec5SDimitry Andric TII.get(WebAssembly::SHL_I32), Left) 500*0b57cec5SDimitry Andric .addReg(Reg) 501*0b57cec5SDimitry Andric .addReg(Imm); 502*0b57cec5SDimitry Andric 503*0b57cec5SDimitry Andric unsigned Right = createResultReg(&WebAssembly::I32RegClass); 504*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 505*0b57cec5SDimitry Andric TII.get(WebAssembly::SHR_S_I32), Right) 506*0b57cec5SDimitry Andric .addReg(Left) 507*0b57cec5SDimitry Andric .addReg(Imm); 508*0b57cec5SDimitry Andric 509*0b57cec5SDimitry Andric return Right; 510*0b57cec5SDimitry Andric } 511*0b57cec5SDimitry Andric 512*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::zeroExtend(unsigned Reg, const Value *V, 513*0b57cec5SDimitry Andric MVT::SimpleValueType From, 514*0b57cec5SDimitry Andric MVT::SimpleValueType To) { 515*0b57cec5SDimitry Andric if (To == MVT::i64) { 516*0b57cec5SDimitry Andric if (From == MVT::i64) 517*0b57cec5SDimitry Andric return copyValue(Reg); 518*0b57cec5SDimitry Andric 519*0b57cec5SDimitry Andric Reg = zeroExtendToI32(Reg, V, From); 520*0b57cec5SDimitry Andric 521*0b57cec5SDimitry Andric unsigned Result = createResultReg(&WebAssembly::I64RegClass); 522*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 523*0b57cec5SDimitry Andric TII.get(WebAssembly::I64_EXTEND_U_I32), Result) 524*0b57cec5SDimitry Andric .addReg(Reg); 525*0b57cec5SDimitry Andric return Result; 526*0b57cec5SDimitry Andric } 527*0b57cec5SDimitry Andric 528*0b57cec5SDimitry Andric if (To == MVT::i32) 529*0b57cec5SDimitry Andric return zeroExtendToI32(Reg, V, From); 530*0b57cec5SDimitry Andric 531*0b57cec5SDimitry Andric return 0; 532*0b57cec5SDimitry Andric } 533*0b57cec5SDimitry Andric 534*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::signExtend(unsigned Reg, const Value *V, 535*0b57cec5SDimitry Andric MVT::SimpleValueType From, 536*0b57cec5SDimitry Andric MVT::SimpleValueType To) { 537*0b57cec5SDimitry Andric if (To == MVT::i64) { 538*0b57cec5SDimitry Andric if (From == MVT::i64) 539*0b57cec5SDimitry Andric return copyValue(Reg); 540*0b57cec5SDimitry Andric 541*0b57cec5SDimitry Andric Reg = signExtendToI32(Reg, V, From); 542*0b57cec5SDimitry Andric 543*0b57cec5SDimitry Andric unsigned Result = createResultReg(&WebAssembly::I64RegClass); 544*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 545*0b57cec5SDimitry Andric TII.get(WebAssembly::I64_EXTEND_S_I32), Result) 546*0b57cec5SDimitry Andric .addReg(Reg); 547*0b57cec5SDimitry Andric return Result; 548*0b57cec5SDimitry Andric } 549*0b57cec5SDimitry Andric 550*0b57cec5SDimitry Andric if (To == MVT::i32) 551*0b57cec5SDimitry Andric return signExtendToI32(Reg, V, From); 552*0b57cec5SDimitry Andric 553*0b57cec5SDimitry Andric return 0; 554*0b57cec5SDimitry Andric } 555*0b57cec5SDimitry Andric 556*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForUnsignedValue(const Value *V) { 557*0b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(V->getType()); 558*0b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(From); 559*0b57cec5SDimitry Andric unsigned VReg = getRegForValue(V); 560*0b57cec5SDimitry Andric if (VReg == 0) 561*0b57cec5SDimitry Andric return 0; 562*0b57cec5SDimitry Andric return zeroExtend(VReg, V, From, To); 563*0b57cec5SDimitry Andric } 564*0b57cec5SDimitry Andric 565*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForSignedValue(const Value *V) { 566*0b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(V->getType()); 567*0b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(From); 568*0b57cec5SDimitry Andric unsigned VReg = getRegForValue(V); 569*0b57cec5SDimitry Andric if (VReg == 0) 570*0b57cec5SDimitry Andric return 0; 571*0b57cec5SDimitry Andric return signExtend(VReg, V, From, To); 572*0b57cec5SDimitry Andric } 573*0b57cec5SDimitry Andric 574*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::getRegForPromotedValue(const Value *V, 575*0b57cec5SDimitry Andric bool IsSigned) { 576*0b57cec5SDimitry Andric return IsSigned ? getRegForSignedValue(V) : getRegForUnsignedValue(V); 577*0b57cec5SDimitry Andric } 578*0b57cec5SDimitry Andric 579*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::notValue(unsigned Reg) { 580*0b57cec5SDimitry Andric assert(MRI.getRegClass(Reg) == &WebAssembly::I32RegClass); 581*0b57cec5SDimitry Andric 582*0b57cec5SDimitry Andric unsigned NotReg = createResultReg(&WebAssembly::I32RegClass); 583*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 584*0b57cec5SDimitry Andric TII.get(WebAssembly::EQZ_I32), NotReg) 585*0b57cec5SDimitry Andric .addReg(Reg); 586*0b57cec5SDimitry Andric return NotReg; 587*0b57cec5SDimitry Andric } 588*0b57cec5SDimitry Andric 589*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::copyValue(unsigned Reg) { 590*0b57cec5SDimitry Andric unsigned ResultReg = createResultReg(MRI.getRegClass(Reg)); 591*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(WebAssembly::COPY), 592*0b57cec5SDimitry Andric ResultReg) 593*0b57cec5SDimitry Andric .addReg(Reg); 594*0b57cec5SDimitry Andric return ResultReg; 595*0b57cec5SDimitry Andric } 596*0b57cec5SDimitry Andric 597*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::fastMaterializeAlloca(const AllocaInst *AI) { 598*0b57cec5SDimitry Andric DenseMap<const AllocaInst *, int>::iterator SI = 599*0b57cec5SDimitry Andric FuncInfo.StaticAllocaMap.find(AI); 600*0b57cec5SDimitry Andric 601*0b57cec5SDimitry Andric if (SI != FuncInfo.StaticAllocaMap.end()) { 602*0b57cec5SDimitry Andric unsigned ResultReg = 603*0b57cec5SDimitry Andric createResultReg(Subtarget->hasAddr64() ? &WebAssembly::I64RegClass 604*0b57cec5SDimitry Andric : &WebAssembly::I32RegClass); 605*0b57cec5SDimitry Andric unsigned Opc = 606*0b57cec5SDimitry Andric Subtarget->hasAddr64() ? WebAssembly::COPY_I64 : WebAssembly::COPY_I32; 607*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ResultReg) 608*0b57cec5SDimitry Andric .addFrameIndex(SI->second); 609*0b57cec5SDimitry Andric return ResultReg; 610*0b57cec5SDimitry Andric } 611*0b57cec5SDimitry Andric 612*0b57cec5SDimitry Andric return 0; 613*0b57cec5SDimitry Andric } 614*0b57cec5SDimitry Andric 615*0b57cec5SDimitry Andric unsigned WebAssemblyFastISel::fastMaterializeConstant(const Constant *C) { 616*0b57cec5SDimitry Andric if (const GlobalValue *GV = dyn_cast<GlobalValue>(C)) { 617*0b57cec5SDimitry Andric if (TLI.isPositionIndependent()) 618*0b57cec5SDimitry Andric return 0; 619*0b57cec5SDimitry Andric if (GV->isThreadLocal()) 620*0b57cec5SDimitry Andric return 0; 621*0b57cec5SDimitry Andric unsigned ResultReg = 622*0b57cec5SDimitry Andric createResultReg(Subtarget->hasAddr64() ? &WebAssembly::I64RegClass 623*0b57cec5SDimitry Andric : &WebAssembly::I32RegClass); 624*0b57cec5SDimitry Andric unsigned Opc = Subtarget->hasAddr64() ? WebAssembly::CONST_I64 625*0b57cec5SDimitry Andric : WebAssembly::CONST_I32; 626*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ResultReg) 627*0b57cec5SDimitry Andric .addGlobalAddress(GV); 628*0b57cec5SDimitry Andric return ResultReg; 629*0b57cec5SDimitry Andric } 630*0b57cec5SDimitry Andric 631*0b57cec5SDimitry Andric // Let target-independent code handle it. 632*0b57cec5SDimitry Andric return 0; 633*0b57cec5SDimitry Andric } 634*0b57cec5SDimitry Andric 635*0b57cec5SDimitry Andric bool WebAssemblyFastISel::fastLowerArguments() { 636*0b57cec5SDimitry Andric if (!FuncInfo.CanLowerReturn) 637*0b57cec5SDimitry Andric return false; 638*0b57cec5SDimitry Andric 639*0b57cec5SDimitry Andric const Function *F = FuncInfo.Fn; 640*0b57cec5SDimitry Andric if (F->isVarArg()) 641*0b57cec5SDimitry Andric return false; 642*0b57cec5SDimitry Andric 643*0b57cec5SDimitry Andric unsigned I = 0; 644*0b57cec5SDimitry Andric for (auto const &Arg : F->args()) { 645*0b57cec5SDimitry Andric const AttributeList &Attrs = F->getAttributes(); 646*0b57cec5SDimitry Andric if (Attrs.hasParamAttribute(I, Attribute::ByVal) || 647*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::SwiftSelf) || 648*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::SwiftError) || 649*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::InAlloca) || 650*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::Nest)) 651*0b57cec5SDimitry Andric return false; 652*0b57cec5SDimitry Andric 653*0b57cec5SDimitry Andric Type *ArgTy = Arg.getType(); 654*0b57cec5SDimitry Andric if (ArgTy->isStructTy() || ArgTy->isArrayTy()) 655*0b57cec5SDimitry Andric return false; 656*0b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && ArgTy->isVectorTy()) 657*0b57cec5SDimitry Andric return false; 658*0b57cec5SDimitry Andric 659*0b57cec5SDimitry Andric unsigned Opc; 660*0b57cec5SDimitry Andric const TargetRegisterClass *RC; 661*0b57cec5SDimitry Andric switch (getSimpleType(ArgTy)) { 662*0b57cec5SDimitry Andric case MVT::i1: 663*0b57cec5SDimitry Andric case MVT::i8: 664*0b57cec5SDimitry Andric case MVT::i16: 665*0b57cec5SDimitry Andric case MVT::i32: 666*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_i32; 667*0b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 668*0b57cec5SDimitry Andric break; 669*0b57cec5SDimitry Andric case MVT::i64: 670*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_i64; 671*0b57cec5SDimitry Andric RC = &WebAssembly::I64RegClass; 672*0b57cec5SDimitry Andric break; 673*0b57cec5SDimitry Andric case MVT::f32: 674*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_f32; 675*0b57cec5SDimitry Andric RC = &WebAssembly::F32RegClass; 676*0b57cec5SDimitry Andric break; 677*0b57cec5SDimitry Andric case MVT::f64: 678*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_f64; 679*0b57cec5SDimitry Andric RC = &WebAssembly::F64RegClass; 680*0b57cec5SDimitry Andric break; 681*0b57cec5SDimitry Andric case MVT::v16i8: 682*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v16i8; 683*0b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 684*0b57cec5SDimitry Andric break; 685*0b57cec5SDimitry Andric case MVT::v8i16: 686*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v8i16; 687*0b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 688*0b57cec5SDimitry Andric break; 689*0b57cec5SDimitry Andric case MVT::v4i32: 690*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v4i32; 691*0b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 692*0b57cec5SDimitry Andric break; 693*0b57cec5SDimitry Andric case MVT::v2i64: 694*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v2i64; 695*0b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 696*0b57cec5SDimitry Andric break; 697*0b57cec5SDimitry Andric case MVT::v4f32: 698*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v4f32; 699*0b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 700*0b57cec5SDimitry Andric break; 701*0b57cec5SDimitry Andric case MVT::v2f64: 702*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_v2f64; 703*0b57cec5SDimitry Andric RC = &WebAssembly::V128RegClass; 704*0b57cec5SDimitry Andric break; 705*0b57cec5SDimitry Andric case MVT::exnref: 706*0b57cec5SDimitry Andric Opc = WebAssembly::ARGUMENT_exnref; 707*0b57cec5SDimitry Andric RC = &WebAssembly::EXNREFRegClass; 708*0b57cec5SDimitry Andric break; 709*0b57cec5SDimitry Andric default: 710*0b57cec5SDimitry Andric return false; 711*0b57cec5SDimitry Andric } 712*0b57cec5SDimitry Andric unsigned ResultReg = createResultReg(RC); 713*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ResultReg) 714*0b57cec5SDimitry Andric .addImm(I); 715*0b57cec5SDimitry Andric updateValueMap(&Arg, ResultReg); 716*0b57cec5SDimitry Andric 717*0b57cec5SDimitry Andric ++I; 718*0b57cec5SDimitry Andric } 719*0b57cec5SDimitry Andric 720*0b57cec5SDimitry Andric MRI.addLiveIn(WebAssembly::ARGUMENTS); 721*0b57cec5SDimitry Andric 722*0b57cec5SDimitry Andric auto *MFI = MF->getInfo<WebAssemblyFunctionInfo>(); 723*0b57cec5SDimitry Andric for (auto const &Arg : F->args()) { 724*0b57cec5SDimitry Andric MVT::SimpleValueType ArgTy = getLegalType(getSimpleType(Arg.getType())); 725*0b57cec5SDimitry Andric if (ArgTy == MVT::INVALID_SIMPLE_VALUE_TYPE) { 726*0b57cec5SDimitry Andric MFI->clearParamsAndResults(); 727*0b57cec5SDimitry Andric return false; 728*0b57cec5SDimitry Andric } 729*0b57cec5SDimitry Andric MFI->addParam(ArgTy); 730*0b57cec5SDimitry Andric } 731*0b57cec5SDimitry Andric 732*0b57cec5SDimitry Andric if (!F->getReturnType()->isVoidTy()) { 733*0b57cec5SDimitry Andric MVT::SimpleValueType RetTy = 734*0b57cec5SDimitry Andric getLegalType(getSimpleType(F->getReturnType())); 735*0b57cec5SDimitry Andric if (RetTy == MVT::INVALID_SIMPLE_VALUE_TYPE) { 736*0b57cec5SDimitry Andric MFI->clearParamsAndResults(); 737*0b57cec5SDimitry Andric return false; 738*0b57cec5SDimitry Andric } 739*0b57cec5SDimitry Andric MFI->addResult(RetTy); 740*0b57cec5SDimitry Andric } 741*0b57cec5SDimitry Andric 742*0b57cec5SDimitry Andric return true; 743*0b57cec5SDimitry Andric } 744*0b57cec5SDimitry Andric 745*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectCall(const Instruction *I) { 746*0b57cec5SDimitry Andric const auto *Call = cast<CallInst>(I); 747*0b57cec5SDimitry Andric 748*0b57cec5SDimitry Andric // TODO: Support tail calls in FastISel 749*0b57cec5SDimitry Andric if (Call->isMustTailCall() || Call->isInlineAsm() || 750*0b57cec5SDimitry Andric Call->getFunctionType()->isVarArg()) 751*0b57cec5SDimitry Andric return false; 752*0b57cec5SDimitry Andric 753*0b57cec5SDimitry Andric Function *Func = Call->getCalledFunction(); 754*0b57cec5SDimitry Andric if (Func && Func->isIntrinsic()) 755*0b57cec5SDimitry Andric return false; 756*0b57cec5SDimitry Andric 757*0b57cec5SDimitry Andric bool IsDirect = Func != nullptr; 758*0b57cec5SDimitry Andric if (!IsDirect && isa<ConstantExpr>(Call->getCalledValue())) 759*0b57cec5SDimitry Andric return false; 760*0b57cec5SDimitry Andric 761*0b57cec5SDimitry Andric FunctionType *FuncTy = Call->getFunctionType(); 762*0b57cec5SDimitry Andric unsigned Opc; 763*0b57cec5SDimitry Andric bool IsVoid = FuncTy->getReturnType()->isVoidTy(); 764*0b57cec5SDimitry Andric unsigned ResultReg; 765*0b57cec5SDimitry Andric if (IsVoid) { 766*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_VOID : WebAssembly::PCALL_INDIRECT_VOID; 767*0b57cec5SDimitry Andric } else { 768*0b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && Call->getType()->isVectorTy()) 769*0b57cec5SDimitry Andric return false; 770*0b57cec5SDimitry Andric 771*0b57cec5SDimitry Andric MVT::SimpleValueType RetTy = getSimpleType(Call->getType()); 772*0b57cec5SDimitry Andric switch (RetTy) { 773*0b57cec5SDimitry Andric case MVT::i1: 774*0b57cec5SDimitry Andric case MVT::i8: 775*0b57cec5SDimitry Andric case MVT::i16: 776*0b57cec5SDimitry Andric case MVT::i32: 777*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_i32 : WebAssembly::PCALL_INDIRECT_i32; 778*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::I32RegClass); 779*0b57cec5SDimitry Andric break; 780*0b57cec5SDimitry Andric case MVT::i64: 781*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_i64 : WebAssembly::PCALL_INDIRECT_i64; 782*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::I64RegClass); 783*0b57cec5SDimitry Andric break; 784*0b57cec5SDimitry Andric case MVT::f32: 785*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_f32 : WebAssembly::PCALL_INDIRECT_f32; 786*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::F32RegClass); 787*0b57cec5SDimitry Andric break; 788*0b57cec5SDimitry Andric case MVT::f64: 789*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_f64 : WebAssembly::PCALL_INDIRECT_f64; 790*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::F64RegClass); 791*0b57cec5SDimitry Andric break; 792*0b57cec5SDimitry Andric case MVT::v16i8: 793*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_v16i8 794*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_v16i8; 795*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 796*0b57cec5SDimitry Andric break; 797*0b57cec5SDimitry Andric case MVT::v8i16: 798*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_v8i16 799*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_v8i16; 800*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 801*0b57cec5SDimitry Andric break; 802*0b57cec5SDimitry Andric case MVT::v4i32: 803*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_v4i32 804*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_v4i32; 805*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 806*0b57cec5SDimitry Andric break; 807*0b57cec5SDimitry Andric case MVT::v2i64: 808*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_v2i64 809*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_v2i64; 810*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 811*0b57cec5SDimitry Andric break; 812*0b57cec5SDimitry Andric case MVT::v4f32: 813*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_v4f32 814*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_v4f32; 815*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 816*0b57cec5SDimitry Andric break; 817*0b57cec5SDimitry Andric case MVT::v2f64: 818*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_v2f64 819*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_v2f64; 820*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::V128RegClass); 821*0b57cec5SDimitry Andric break; 822*0b57cec5SDimitry Andric case MVT::exnref: 823*0b57cec5SDimitry Andric Opc = IsDirect ? WebAssembly::CALL_exnref 824*0b57cec5SDimitry Andric : WebAssembly::PCALL_INDIRECT_exnref; 825*0b57cec5SDimitry Andric ResultReg = createResultReg(&WebAssembly::EXNREFRegClass); 826*0b57cec5SDimitry Andric break; 827*0b57cec5SDimitry Andric default: 828*0b57cec5SDimitry Andric return false; 829*0b57cec5SDimitry Andric } 830*0b57cec5SDimitry Andric } 831*0b57cec5SDimitry Andric 832*0b57cec5SDimitry Andric SmallVector<unsigned, 8> Args; 833*0b57cec5SDimitry Andric for (unsigned I = 0, E = Call->getNumArgOperands(); I < E; ++I) { 834*0b57cec5SDimitry Andric Value *V = Call->getArgOperand(I); 835*0b57cec5SDimitry Andric MVT::SimpleValueType ArgTy = getSimpleType(V->getType()); 836*0b57cec5SDimitry Andric if (ArgTy == MVT::INVALID_SIMPLE_VALUE_TYPE) 837*0b57cec5SDimitry Andric return false; 838*0b57cec5SDimitry Andric 839*0b57cec5SDimitry Andric const AttributeList &Attrs = Call->getAttributes(); 840*0b57cec5SDimitry Andric if (Attrs.hasParamAttribute(I, Attribute::ByVal) || 841*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::SwiftSelf) || 842*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::SwiftError) || 843*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::InAlloca) || 844*0b57cec5SDimitry Andric Attrs.hasParamAttribute(I, Attribute::Nest)) 845*0b57cec5SDimitry Andric return false; 846*0b57cec5SDimitry Andric 847*0b57cec5SDimitry Andric unsigned Reg; 848*0b57cec5SDimitry Andric 849*0b57cec5SDimitry Andric if (Attrs.hasParamAttribute(I, Attribute::SExt)) 850*0b57cec5SDimitry Andric Reg = getRegForSignedValue(V); 851*0b57cec5SDimitry Andric else if (Attrs.hasParamAttribute(I, Attribute::ZExt)) 852*0b57cec5SDimitry Andric Reg = getRegForUnsignedValue(V); 853*0b57cec5SDimitry Andric else 854*0b57cec5SDimitry Andric Reg = getRegForValue(V); 855*0b57cec5SDimitry Andric 856*0b57cec5SDimitry Andric if (Reg == 0) 857*0b57cec5SDimitry Andric return false; 858*0b57cec5SDimitry Andric 859*0b57cec5SDimitry Andric Args.push_back(Reg); 860*0b57cec5SDimitry Andric } 861*0b57cec5SDimitry Andric 862*0b57cec5SDimitry Andric unsigned CalleeReg = 0; 863*0b57cec5SDimitry Andric if (!IsDirect) { 864*0b57cec5SDimitry Andric CalleeReg = getRegForValue(Call->getCalledValue()); 865*0b57cec5SDimitry Andric if (!CalleeReg) 866*0b57cec5SDimitry Andric return false; 867*0b57cec5SDimitry Andric } 868*0b57cec5SDimitry Andric 869*0b57cec5SDimitry Andric auto MIB = BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc)); 870*0b57cec5SDimitry Andric 871*0b57cec5SDimitry Andric if (!IsVoid) 872*0b57cec5SDimitry Andric MIB.addReg(ResultReg, RegState::Define); 873*0b57cec5SDimitry Andric 874*0b57cec5SDimitry Andric if (IsDirect) 875*0b57cec5SDimitry Andric MIB.addGlobalAddress(Func); 876*0b57cec5SDimitry Andric else 877*0b57cec5SDimitry Andric MIB.addReg(CalleeReg); 878*0b57cec5SDimitry Andric 879*0b57cec5SDimitry Andric for (unsigned ArgReg : Args) 880*0b57cec5SDimitry Andric MIB.addReg(ArgReg); 881*0b57cec5SDimitry Andric 882*0b57cec5SDimitry Andric if (!IsVoid) 883*0b57cec5SDimitry Andric updateValueMap(Call, ResultReg); 884*0b57cec5SDimitry Andric return true; 885*0b57cec5SDimitry Andric } 886*0b57cec5SDimitry Andric 887*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectSelect(const Instruction *I) { 888*0b57cec5SDimitry Andric const auto *Select = cast<SelectInst>(I); 889*0b57cec5SDimitry Andric 890*0b57cec5SDimitry Andric bool Not; 891*0b57cec5SDimitry Andric unsigned CondReg = getRegForI1Value(Select->getCondition(), Not); 892*0b57cec5SDimitry Andric if (CondReg == 0) 893*0b57cec5SDimitry Andric return false; 894*0b57cec5SDimitry Andric 895*0b57cec5SDimitry Andric unsigned TrueReg = getRegForValue(Select->getTrueValue()); 896*0b57cec5SDimitry Andric if (TrueReg == 0) 897*0b57cec5SDimitry Andric return false; 898*0b57cec5SDimitry Andric 899*0b57cec5SDimitry Andric unsigned FalseReg = getRegForValue(Select->getFalseValue()); 900*0b57cec5SDimitry Andric if (FalseReg == 0) 901*0b57cec5SDimitry Andric return false; 902*0b57cec5SDimitry Andric 903*0b57cec5SDimitry Andric if (Not) 904*0b57cec5SDimitry Andric std::swap(TrueReg, FalseReg); 905*0b57cec5SDimitry Andric 906*0b57cec5SDimitry Andric unsigned Opc; 907*0b57cec5SDimitry Andric const TargetRegisterClass *RC; 908*0b57cec5SDimitry Andric switch (getSimpleType(Select->getType())) { 909*0b57cec5SDimitry Andric case MVT::i1: 910*0b57cec5SDimitry Andric case MVT::i8: 911*0b57cec5SDimitry Andric case MVT::i16: 912*0b57cec5SDimitry Andric case MVT::i32: 913*0b57cec5SDimitry Andric Opc = WebAssembly::SELECT_I32; 914*0b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 915*0b57cec5SDimitry Andric break; 916*0b57cec5SDimitry Andric case MVT::i64: 917*0b57cec5SDimitry Andric Opc = WebAssembly::SELECT_I64; 918*0b57cec5SDimitry Andric RC = &WebAssembly::I64RegClass; 919*0b57cec5SDimitry Andric break; 920*0b57cec5SDimitry Andric case MVT::f32: 921*0b57cec5SDimitry Andric Opc = WebAssembly::SELECT_F32; 922*0b57cec5SDimitry Andric RC = &WebAssembly::F32RegClass; 923*0b57cec5SDimitry Andric break; 924*0b57cec5SDimitry Andric case MVT::f64: 925*0b57cec5SDimitry Andric Opc = WebAssembly::SELECT_F64; 926*0b57cec5SDimitry Andric RC = &WebAssembly::F64RegClass; 927*0b57cec5SDimitry Andric break; 928*0b57cec5SDimitry Andric case MVT::exnref: 929*0b57cec5SDimitry Andric Opc = WebAssembly::SELECT_EXNREF; 930*0b57cec5SDimitry Andric RC = &WebAssembly::EXNREFRegClass; 931*0b57cec5SDimitry Andric break; 932*0b57cec5SDimitry Andric default: 933*0b57cec5SDimitry Andric return false; 934*0b57cec5SDimitry Andric } 935*0b57cec5SDimitry Andric 936*0b57cec5SDimitry Andric unsigned ResultReg = createResultReg(RC); 937*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ResultReg) 938*0b57cec5SDimitry Andric .addReg(TrueReg) 939*0b57cec5SDimitry Andric .addReg(FalseReg) 940*0b57cec5SDimitry Andric .addReg(CondReg); 941*0b57cec5SDimitry Andric 942*0b57cec5SDimitry Andric updateValueMap(Select, ResultReg); 943*0b57cec5SDimitry Andric return true; 944*0b57cec5SDimitry Andric } 945*0b57cec5SDimitry Andric 946*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectTrunc(const Instruction *I) { 947*0b57cec5SDimitry Andric const auto *Trunc = cast<TruncInst>(I); 948*0b57cec5SDimitry Andric 949*0b57cec5SDimitry Andric unsigned Reg = getRegForValue(Trunc->getOperand(0)); 950*0b57cec5SDimitry Andric if (Reg == 0) 951*0b57cec5SDimitry Andric return false; 952*0b57cec5SDimitry Andric 953*0b57cec5SDimitry Andric if (Trunc->getOperand(0)->getType()->isIntegerTy(64)) { 954*0b57cec5SDimitry Andric unsigned Result = createResultReg(&WebAssembly::I32RegClass); 955*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 956*0b57cec5SDimitry Andric TII.get(WebAssembly::I32_WRAP_I64), Result) 957*0b57cec5SDimitry Andric .addReg(Reg); 958*0b57cec5SDimitry Andric Reg = Result; 959*0b57cec5SDimitry Andric } 960*0b57cec5SDimitry Andric 961*0b57cec5SDimitry Andric updateValueMap(Trunc, Reg); 962*0b57cec5SDimitry Andric return true; 963*0b57cec5SDimitry Andric } 964*0b57cec5SDimitry Andric 965*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectZExt(const Instruction *I) { 966*0b57cec5SDimitry Andric const auto *ZExt = cast<ZExtInst>(I); 967*0b57cec5SDimitry Andric 968*0b57cec5SDimitry Andric const Value *Op = ZExt->getOperand(0); 969*0b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(Op->getType()); 970*0b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(getSimpleType(ZExt->getType())); 971*0b57cec5SDimitry Andric unsigned In = getRegForValue(Op); 972*0b57cec5SDimitry Andric if (In == 0) 973*0b57cec5SDimitry Andric return false; 974*0b57cec5SDimitry Andric unsigned Reg = zeroExtend(In, Op, From, To); 975*0b57cec5SDimitry Andric if (Reg == 0) 976*0b57cec5SDimitry Andric return false; 977*0b57cec5SDimitry Andric 978*0b57cec5SDimitry Andric updateValueMap(ZExt, Reg); 979*0b57cec5SDimitry Andric return true; 980*0b57cec5SDimitry Andric } 981*0b57cec5SDimitry Andric 982*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectSExt(const Instruction *I) { 983*0b57cec5SDimitry Andric const auto *SExt = cast<SExtInst>(I); 984*0b57cec5SDimitry Andric 985*0b57cec5SDimitry Andric const Value *Op = SExt->getOperand(0); 986*0b57cec5SDimitry Andric MVT::SimpleValueType From = getSimpleType(Op->getType()); 987*0b57cec5SDimitry Andric MVT::SimpleValueType To = getLegalType(getSimpleType(SExt->getType())); 988*0b57cec5SDimitry Andric unsigned In = getRegForValue(Op); 989*0b57cec5SDimitry Andric if (In == 0) 990*0b57cec5SDimitry Andric return false; 991*0b57cec5SDimitry Andric unsigned Reg = signExtend(In, Op, From, To); 992*0b57cec5SDimitry Andric if (Reg == 0) 993*0b57cec5SDimitry Andric return false; 994*0b57cec5SDimitry Andric 995*0b57cec5SDimitry Andric updateValueMap(SExt, Reg); 996*0b57cec5SDimitry Andric return true; 997*0b57cec5SDimitry Andric } 998*0b57cec5SDimitry Andric 999*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectICmp(const Instruction *I) { 1000*0b57cec5SDimitry Andric const auto *ICmp = cast<ICmpInst>(I); 1001*0b57cec5SDimitry Andric 1002*0b57cec5SDimitry Andric bool I32 = getSimpleType(ICmp->getOperand(0)->getType()) != MVT::i64; 1003*0b57cec5SDimitry Andric unsigned Opc; 1004*0b57cec5SDimitry Andric bool IsSigned = false; 1005*0b57cec5SDimitry Andric switch (ICmp->getPredicate()) { 1006*0b57cec5SDimitry Andric case ICmpInst::ICMP_EQ: 1007*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::EQ_I32 : WebAssembly::EQ_I64; 1008*0b57cec5SDimitry Andric break; 1009*0b57cec5SDimitry Andric case ICmpInst::ICMP_NE: 1010*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::NE_I32 : WebAssembly::NE_I64; 1011*0b57cec5SDimitry Andric break; 1012*0b57cec5SDimitry Andric case ICmpInst::ICMP_UGT: 1013*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GT_U_I32 : WebAssembly::GT_U_I64; 1014*0b57cec5SDimitry Andric break; 1015*0b57cec5SDimitry Andric case ICmpInst::ICMP_UGE: 1016*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GE_U_I32 : WebAssembly::GE_U_I64; 1017*0b57cec5SDimitry Andric break; 1018*0b57cec5SDimitry Andric case ICmpInst::ICMP_ULT: 1019*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LT_U_I32 : WebAssembly::LT_U_I64; 1020*0b57cec5SDimitry Andric break; 1021*0b57cec5SDimitry Andric case ICmpInst::ICMP_ULE: 1022*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LE_U_I32 : WebAssembly::LE_U_I64; 1023*0b57cec5SDimitry Andric break; 1024*0b57cec5SDimitry Andric case ICmpInst::ICMP_SGT: 1025*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GT_S_I32 : WebAssembly::GT_S_I64; 1026*0b57cec5SDimitry Andric IsSigned = true; 1027*0b57cec5SDimitry Andric break; 1028*0b57cec5SDimitry Andric case ICmpInst::ICMP_SGE: 1029*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::GE_S_I32 : WebAssembly::GE_S_I64; 1030*0b57cec5SDimitry Andric IsSigned = true; 1031*0b57cec5SDimitry Andric break; 1032*0b57cec5SDimitry Andric case ICmpInst::ICMP_SLT: 1033*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LT_S_I32 : WebAssembly::LT_S_I64; 1034*0b57cec5SDimitry Andric IsSigned = true; 1035*0b57cec5SDimitry Andric break; 1036*0b57cec5SDimitry Andric case ICmpInst::ICMP_SLE: 1037*0b57cec5SDimitry Andric Opc = I32 ? WebAssembly::LE_S_I32 : WebAssembly::LE_S_I64; 1038*0b57cec5SDimitry Andric IsSigned = true; 1039*0b57cec5SDimitry Andric break; 1040*0b57cec5SDimitry Andric default: 1041*0b57cec5SDimitry Andric return false; 1042*0b57cec5SDimitry Andric } 1043*0b57cec5SDimitry Andric 1044*0b57cec5SDimitry Andric unsigned LHS = getRegForPromotedValue(ICmp->getOperand(0), IsSigned); 1045*0b57cec5SDimitry Andric if (LHS == 0) 1046*0b57cec5SDimitry Andric return false; 1047*0b57cec5SDimitry Andric 1048*0b57cec5SDimitry Andric unsigned RHS = getRegForPromotedValue(ICmp->getOperand(1), IsSigned); 1049*0b57cec5SDimitry Andric if (RHS == 0) 1050*0b57cec5SDimitry Andric return false; 1051*0b57cec5SDimitry Andric 1052*0b57cec5SDimitry Andric unsigned ResultReg = createResultReg(&WebAssembly::I32RegClass); 1053*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ResultReg) 1054*0b57cec5SDimitry Andric .addReg(LHS) 1055*0b57cec5SDimitry Andric .addReg(RHS); 1056*0b57cec5SDimitry Andric updateValueMap(ICmp, ResultReg); 1057*0b57cec5SDimitry Andric return true; 1058*0b57cec5SDimitry Andric } 1059*0b57cec5SDimitry Andric 1060*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectFCmp(const Instruction *I) { 1061*0b57cec5SDimitry Andric const auto *FCmp = cast<FCmpInst>(I); 1062*0b57cec5SDimitry Andric 1063*0b57cec5SDimitry Andric unsigned LHS = getRegForValue(FCmp->getOperand(0)); 1064*0b57cec5SDimitry Andric if (LHS == 0) 1065*0b57cec5SDimitry Andric return false; 1066*0b57cec5SDimitry Andric 1067*0b57cec5SDimitry Andric unsigned RHS = getRegForValue(FCmp->getOperand(1)); 1068*0b57cec5SDimitry Andric if (RHS == 0) 1069*0b57cec5SDimitry Andric return false; 1070*0b57cec5SDimitry Andric 1071*0b57cec5SDimitry Andric bool F32 = getSimpleType(FCmp->getOperand(0)->getType()) != MVT::f64; 1072*0b57cec5SDimitry Andric unsigned Opc; 1073*0b57cec5SDimitry Andric bool Not = false; 1074*0b57cec5SDimitry Andric switch (FCmp->getPredicate()) { 1075*0b57cec5SDimitry Andric case FCmpInst::FCMP_OEQ: 1076*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::EQ_F32 : WebAssembly::EQ_F64; 1077*0b57cec5SDimitry Andric break; 1078*0b57cec5SDimitry Andric case FCmpInst::FCMP_UNE: 1079*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::NE_F32 : WebAssembly::NE_F64; 1080*0b57cec5SDimitry Andric break; 1081*0b57cec5SDimitry Andric case FCmpInst::FCMP_OGT: 1082*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GT_F32 : WebAssembly::GT_F64; 1083*0b57cec5SDimitry Andric break; 1084*0b57cec5SDimitry Andric case FCmpInst::FCMP_OGE: 1085*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GE_F32 : WebAssembly::GE_F64; 1086*0b57cec5SDimitry Andric break; 1087*0b57cec5SDimitry Andric case FCmpInst::FCMP_OLT: 1088*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LT_F32 : WebAssembly::LT_F64; 1089*0b57cec5SDimitry Andric break; 1090*0b57cec5SDimitry Andric case FCmpInst::FCMP_OLE: 1091*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LE_F32 : WebAssembly::LE_F64; 1092*0b57cec5SDimitry Andric break; 1093*0b57cec5SDimitry Andric case FCmpInst::FCMP_UGT: 1094*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LE_F32 : WebAssembly::LE_F64; 1095*0b57cec5SDimitry Andric Not = true; 1096*0b57cec5SDimitry Andric break; 1097*0b57cec5SDimitry Andric case FCmpInst::FCMP_UGE: 1098*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::LT_F32 : WebAssembly::LT_F64; 1099*0b57cec5SDimitry Andric Not = true; 1100*0b57cec5SDimitry Andric break; 1101*0b57cec5SDimitry Andric case FCmpInst::FCMP_ULT: 1102*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GE_F32 : WebAssembly::GE_F64; 1103*0b57cec5SDimitry Andric Not = true; 1104*0b57cec5SDimitry Andric break; 1105*0b57cec5SDimitry Andric case FCmpInst::FCMP_ULE: 1106*0b57cec5SDimitry Andric Opc = F32 ? WebAssembly::GT_F32 : WebAssembly::GT_F64; 1107*0b57cec5SDimitry Andric Not = true; 1108*0b57cec5SDimitry Andric break; 1109*0b57cec5SDimitry Andric default: 1110*0b57cec5SDimitry Andric return false; 1111*0b57cec5SDimitry Andric } 1112*0b57cec5SDimitry Andric 1113*0b57cec5SDimitry Andric unsigned ResultReg = createResultReg(&WebAssembly::I32RegClass); 1114*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), ResultReg) 1115*0b57cec5SDimitry Andric .addReg(LHS) 1116*0b57cec5SDimitry Andric .addReg(RHS); 1117*0b57cec5SDimitry Andric 1118*0b57cec5SDimitry Andric if (Not) 1119*0b57cec5SDimitry Andric ResultReg = notValue(ResultReg); 1120*0b57cec5SDimitry Andric 1121*0b57cec5SDimitry Andric updateValueMap(FCmp, ResultReg); 1122*0b57cec5SDimitry Andric return true; 1123*0b57cec5SDimitry Andric } 1124*0b57cec5SDimitry Andric 1125*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectBitCast(const Instruction *I) { 1126*0b57cec5SDimitry Andric // Target-independent code can handle this, except it doesn't set the dead 1127*0b57cec5SDimitry Andric // flag on the ARGUMENTS clobber, so we have to do that manually in order 1128*0b57cec5SDimitry Andric // to satisfy code that expects this of isBitcast() instructions. 1129*0b57cec5SDimitry Andric EVT VT = TLI.getValueType(DL, I->getOperand(0)->getType()); 1130*0b57cec5SDimitry Andric EVT RetVT = TLI.getValueType(DL, I->getType()); 1131*0b57cec5SDimitry Andric if (!VT.isSimple() || !RetVT.isSimple()) 1132*0b57cec5SDimitry Andric return false; 1133*0b57cec5SDimitry Andric 1134*0b57cec5SDimitry Andric unsigned In = getRegForValue(I->getOperand(0)); 1135*0b57cec5SDimitry Andric if (In == 0) 1136*0b57cec5SDimitry Andric return false; 1137*0b57cec5SDimitry Andric 1138*0b57cec5SDimitry Andric if (VT == RetVT) { 1139*0b57cec5SDimitry Andric // No-op bitcast. 1140*0b57cec5SDimitry Andric updateValueMap(I, In); 1141*0b57cec5SDimitry Andric return true; 1142*0b57cec5SDimitry Andric } 1143*0b57cec5SDimitry Andric 1144*0b57cec5SDimitry Andric unsigned Reg = fastEmit_ISD_BITCAST_r(VT.getSimpleVT(), RetVT.getSimpleVT(), 1145*0b57cec5SDimitry Andric In, I->getOperand(0)->hasOneUse()); 1146*0b57cec5SDimitry Andric if (!Reg) 1147*0b57cec5SDimitry Andric return false; 1148*0b57cec5SDimitry Andric MachineBasicBlock::iterator Iter = FuncInfo.InsertPt; 1149*0b57cec5SDimitry Andric --Iter; 1150*0b57cec5SDimitry Andric assert(Iter->isBitcast()); 1151*0b57cec5SDimitry Andric Iter->setPhysRegsDeadExcept(ArrayRef<unsigned>(), TRI); 1152*0b57cec5SDimitry Andric updateValueMap(I, Reg); 1153*0b57cec5SDimitry Andric return true; 1154*0b57cec5SDimitry Andric } 1155*0b57cec5SDimitry Andric 1156*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectLoad(const Instruction *I) { 1157*0b57cec5SDimitry Andric const auto *Load = cast<LoadInst>(I); 1158*0b57cec5SDimitry Andric if (Load->isAtomic()) 1159*0b57cec5SDimitry Andric return false; 1160*0b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && Load->getType()->isVectorTy()) 1161*0b57cec5SDimitry Andric return false; 1162*0b57cec5SDimitry Andric 1163*0b57cec5SDimitry Andric Address Addr; 1164*0b57cec5SDimitry Andric if (!computeAddress(Load->getPointerOperand(), Addr)) 1165*0b57cec5SDimitry Andric return false; 1166*0b57cec5SDimitry Andric 1167*0b57cec5SDimitry Andric // TODO: Fold a following sign-/zero-extend into the load instruction. 1168*0b57cec5SDimitry Andric 1169*0b57cec5SDimitry Andric unsigned Opc; 1170*0b57cec5SDimitry Andric const TargetRegisterClass *RC; 1171*0b57cec5SDimitry Andric switch (getSimpleType(Load->getType())) { 1172*0b57cec5SDimitry Andric case MVT::i1: 1173*0b57cec5SDimitry Andric case MVT::i8: 1174*0b57cec5SDimitry Andric Opc = WebAssembly::LOAD8_U_I32; 1175*0b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 1176*0b57cec5SDimitry Andric break; 1177*0b57cec5SDimitry Andric case MVT::i16: 1178*0b57cec5SDimitry Andric Opc = WebAssembly::LOAD16_U_I32; 1179*0b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 1180*0b57cec5SDimitry Andric break; 1181*0b57cec5SDimitry Andric case MVT::i32: 1182*0b57cec5SDimitry Andric Opc = WebAssembly::LOAD_I32; 1183*0b57cec5SDimitry Andric RC = &WebAssembly::I32RegClass; 1184*0b57cec5SDimitry Andric break; 1185*0b57cec5SDimitry Andric case MVT::i64: 1186*0b57cec5SDimitry Andric Opc = WebAssembly::LOAD_I64; 1187*0b57cec5SDimitry Andric RC = &WebAssembly::I64RegClass; 1188*0b57cec5SDimitry Andric break; 1189*0b57cec5SDimitry Andric case MVT::f32: 1190*0b57cec5SDimitry Andric Opc = WebAssembly::LOAD_F32; 1191*0b57cec5SDimitry Andric RC = &WebAssembly::F32RegClass; 1192*0b57cec5SDimitry Andric break; 1193*0b57cec5SDimitry Andric case MVT::f64: 1194*0b57cec5SDimitry Andric Opc = WebAssembly::LOAD_F64; 1195*0b57cec5SDimitry Andric RC = &WebAssembly::F64RegClass; 1196*0b57cec5SDimitry Andric break; 1197*0b57cec5SDimitry Andric default: 1198*0b57cec5SDimitry Andric return false; 1199*0b57cec5SDimitry Andric } 1200*0b57cec5SDimitry Andric 1201*0b57cec5SDimitry Andric materializeLoadStoreOperands(Addr); 1202*0b57cec5SDimitry Andric 1203*0b57cec5SDimitry Andric unsigned ResultReg = createResultReg(RC); 1204*0b57cec5SDimitry Andric auto MIB = BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc), 1205*0b57cec5SDimitry Andric ResultReg); 1206*0b57cec5SDimitry Andric 1207*0b57cec5SDimitry Andric addLoadStoreOperands(Addr, MIB, createMachineMemOperandFor(Load)); 1208*0b57cec5SDimitry Andric 1209*0b57cec5SDimitry Andric updateValueMap(Load, ResultReg); 1210*0b57cec5SDimitry Andric return true; 1211*0b57cec5SDimitry Andric } 1212*0b57cec5SDimitry Andric 1213*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectStore(const Instruction *I) { 1214*0b57cec5SDimitry Andric const auto *Store = cast<StoreInst>(I); 1215*0b57cec5SDimitry Andric if (Store->isAtomic()) 1216*0b57cec5SDimitry Andric return false; 1217*0b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && 1218*0b57cec5SDimitry Andric Store->getValueOperand()->getType()->isVectorTy()) 1219*0b57cec5SDimitry Andric return false; 1220*0b57cec5SDimitry Andric 1221*0b57cec5SDimitry Andric Address Addr; 1222*0b57cec5SDimitry Andric if (!computeAddress(Store->getPointerOperand(), Addr)) 1223*0b57cec5SDimitry Andric return false; 1224*0b57cec5SDimitry Andric 1225*0b57cec5SDimitry Andric unsigned Opc; 1226*0b57cec5SDimitry Andric bool VTIsi1 = false; 1227*0b57cec5SDimitry Andric switch (getSimpleType(Store->getValueOperand()->getType())) { 1228*0b57cec5SDimitry Andric case MVT::i1: 1229*0b57cec5SDimitry Andric VTIsi1 = true; 1230*0b57cec5SDimitry Andric LLVM_FALLTHROUGH; 1231*0b57cec5SDimitry Andric case MVT::i8: 1232*0b57cec5SDimitry Andric Opc = WebAssembly::STORE8_I32; 1233*0b57cec5SDimitry Andric break; 1234*0b57cec5SDimitry Andric case MVT::i16: 1235*0b57cec5SDimitry Andric Opc = WebAssembly::STORE16_I32; 1236*0b57cec5SDimitry Andric break; 1237*0b57cec5SDimitry Andric case MVT::i32: 1238*0b57cec5SDimitry Andric Opc = WebAssembly::STORE_I32; 1239*0b57cec5SDimitry Andric break; 1240*0b57cec5SDimitry Andric case MVT::i64: 1241*0b57cec5SDimitry Andric Opc = WebAssembly::STORE_I64; 1242*0b57cec5SDimitry Andric break; 1243*0b57cec5SDimitry Andric case MVT::f32: 1244*0b57cec5SDimitry Andric Opc = WebAssembly::STORE_F32; 1245*0b57cec5SDimitry Andric break; 1246*0b57cec5SDimitry Andric case MVT::f64: 1247*0b57cec5SDimitry Andric Opc = WebAssembly::STORE_F64; 1248*0b57cec5SDimitry Andric break; 1249*0b57cec5SDimitry Andric default: 1250*0b57cec5SDimitry Andric return false; 1251*0b57cec5SDimitry Andric } 1252*0b57cec5SDimitry Andric 1253*0b57cec5SDimitry Andric materializeLoadStoreOperands(Addr); 1254*0b57cec5SDimitry Andric 1255*0b57cec5SDimitry Andric unsigned ValueReg = getRegForValue(Store->getValueOperand()); 1256*0b57cec5SDimitry Andric if (ValueReg == 0) 1257*0b57cec5SDimitry Andric return false; 1258*0b57cec5SDimitry Andric if (VTIsi1) 1259*0b57cec5SDimitry Andric ValueReg = maskI1Value(ValueReg, Store->getValueOperand()); 1260*0b57cec5SDimitry Andric 1261*0b57cec5SDimitry Andric auto MIB = BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc)); 1262*0b57cec5SDimitry Andric 1263*0b57cec5SDimitry Andric addLoadStoreOperands(Addr, MIB, createMachineMemOperandFor(Store)); 1264*0b57cec5SDimitry Andric 1265*0b57cec5SDimitry Andric MIB.addReg(ValueReg); 1266*0b57cec5SDimitry Andric return true; 1267*0b57cec5SDimitry Andric } 1268*0b57cec5SDimitry Andric 1269*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectBr(const Instruction *I) { 1270*0b57cec5SDimitry Andric const auto *Br = cast<BranchInst>(I); 1271*0b57cec5SDimitry Andric if (Br->isUnconditional()) { 1272*0b57cec5SDimitry Andric MachineBasicBlock *MSucc = FuncInfo.MBBMap[Br->getSuccessor(0)]; 1273*0b57cec5SDimitry Andric fastEmitBranch(MSucc, Br->getDebugLoc()); 1274*0b57cec5SDimitry Andric return true; 1275*0b57cec5SDimitry Andric } 1276*0b57cec5SDimitry Andric 1277*0b57cec5SDimitry Andric MachineBasicBlock *TBB = FuncInfo.MBBMap[Br->getSuccessor(0)]; 1278*0b57cec5SDimitry Andric MachineBasicBlock *FBB = FuncInfo.MBBMap[Br->getSuccessor(1)]; 1279*0b57cec5SDimitry Andric 1280*0b57cec5SDimitry Andric bool Not; 1281*0b57cec5SDimitry Andric unsigned CondReg = getRegForI1Value(Br->getCondition(), Not); 1282*0b57cec5SDimitry Andric if (CondReg == 0) 1283*0b57cec5SDimitry Andric return false; 1284*0b57cec5SDimitry Andric 1285*0b57cec5SDimitry Andric unsigned Opc = WebAssembly::BR_IF; 1286*0b57cec5SDimitry Andric if (Not) 1287*0b57cec5SDimitry Andric Opc = WebAssembly::BR_UNLESS; 1288*0b57cec5SDimitry Andric 1289*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc)) 1290*0b57cec5SDimitry Andric .addMBB(TBB) 1291*0b57cec5SDimitry Andric .addReg(CondReg); 1292*0b57cec5SDimitry Andric 1293*0b57cec5SDimitry Andric finishCondBranch(Br->getParent(), TBB, FBB); 1294*0b57cec5SDimitry Andric return true; 1295*0b57cec5SDimitry Andric } 1296*0b57cec5SDimitry Andric 1297*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectRet(const Instruction *I) { 1298*0b57cec5SDimitry Andric if (!FuncInfo.CanLowerReturn) 1299*0b57cec5SDimitry Andric return false; 1300*0b57cec5SDimitry Andric 1301*0b57cec5SDimitry Andric const auto *Ret = cast<ReturnInst>(I); 1302*0b57cec5SDimitry Andric 1303*0b57cec5SDimitry Andric if (Ret->getNumOperands() == 0) { 1304*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 1305*0b57cec5SDimitry Andric TII.get(WebAssembly::RETURN_VOID)); 1306*0b57cec5SDimitry Andric return true; 1307*0b57cec5SDimitry Andric } 1308*0b57cec5SDimitry Andric 1309*0b57cec5SDimitry Andric Value *RV = Ret->getOperand(0); 1310*0b57cec5SDimitry Andric if (!Subtarget->hasSIMD128() && RV->getType()->isVectorTy()) 1311*0b57cec5SDimitry Andric return false; 1312*0b57cec5SDimitry Andric 1313*0b57cec5SDimitry Andric unsigned Opc; 1314*0b57cec5SDimitry Andric switch (getSimpleType(RV->getType())) { 1315*0b57cec5SDimitry Andric case MVT::i1: 1316*0b57cec5SDimitry Andric case MVT::i8: 1317*0b57cec5SDimitry Andric case MVT::i16: 1318*0b57cec5SDimitry Andric case MVT::i32: 1319*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_I32; 1320*0b57cec5SDimitry Andric break; 1321*0b57cec5SDimitry Andric case MVT::i64: 1322*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_I64; 1323*0b57cec5SDimitry Andric break; 1324*0b57cec5SDimitry Andric case MVT::f32: 1325*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_F32; 1326*0b57cec5SDimitry Andric break; 1327*0b57cec5SDimitry Andric case MVT::f64: 1328*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_F64; 1329*0b57cec5SDimitry Andric break; 1330*0b57cec5SDimitry Andric case MVT::v16i8: 1331*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_v16i8; 1332*0b57cec5SDimitry Andric break; 1333*0b57cec5SDimitry Andric case MVT::v8i16: 1334*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_v8i16; 1335*0b57cec5SDimitry Andric break; 1336*0b57cec5SDimitry Andric case MVT::v4i32: 1337*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_v4i32; 1338*0b57cec5SDimitry Andric break; 1339*0b57cec5SDimitry Andric case MVT::v2i64: 1340*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_v2i64; 1341*0b57cec5SDimitry Andric break; 1342*0b57cec5SDimitry Andric case MVT::v4f32: 1343*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_v4f32; 1344*0b57cec5SDimitry Andric break; 1345*0b57cec5SDimitry Andric case MVT::v2f64: 1346*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_v2f64; 1347*0b57cec5SDimitry Andric break; 1348*0b57cec5SDimitry Andric case MVT::exnref: 1349*0b57cec5SDimitry Andric Opc = WebAssembly::RETURN_EXNREF; 1350*0b57cec5SDimitry Andric break; 1351*0b57cec5SDimitry Andric default: 1352*0b57cec5SDimitry Andric return false; 1353*0b57cec5SDimitry Andric } 1354*0b57cec5SDimitry Andric 1355*0b57cec5SDimitry Andric unsigned Reg; 1356*0b57cec5SDimitry Andric if (FuncInfo.Fn->getAttributes().hasAttribute(0, Attribute::SExt)) 1357*0b57cec5SDimitry Andric Reg = getRegForSignedValue(RV); 1358*0b57cec5SDimitry Andric else if (FuncInfo.Fn->getAttributes().hasAttribute(0, Attribute::ZExt)) 1359*0b57cec5SDimitry Andric Reg = getRegForUnsignedValue(RV); 1360*0b57cec5SDimitry Andric else 1361*0b57cec5SDimitry Andric Reg = getRegForValue(RV); 1362*0b57cec5SDimitry Andric 1363*0b57cec5SDimitry Andric if (Reg == 0) 1364*0b57cec5SDimitry Andric return false; 1365*0b57cec5SDimitry Andric 1366*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, TII.get(Opc)).addReg(Reg); 1367*0b57cec5SDimitry Andric return true; 1368*0b57cec5SDimitry Andric } 1369*0b57cec5SDimitry Andric 1370*0b57cec5SDimitry Andric bool WebAssemblyFastISel::selectUnreachable(const Instruction *I) { 1371*0b57cec5SDimitry Andric BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, 1372*0b57cec5SDimitry Andric TII.get(WebAssembly::UNREACHABLE)); 1373*0b57cec5SDimitry Andric return true; 1374*0b57cec5SDimitry Andric } 1375*0b57cec5SDimitry Andric 1376*0b57cec5SDimitry Andric bool WebAssemblyFastISel::fastSelectInstruction(const Instruction *I) { 1377*0b57cec5SDimitry Andric switch (I->getOpcode()) { 1378*0b57cec5SDimitry Andric case Instruction::Call: 1379*0b57cec5SDimitry Andric if (selectCall(I)) 1380*0b57cec5SDimitry Andric return true; 1381*0b57cec5SDimitry Andric break; 1382*0b57cec5SDimitry Andric case Instruction::Select: 1383*0b57cec5SDimitry Andric return selectSelect(I); 1384*0b57cec5SDimitry Andric case Instruction::Trunc: 1385*0b57cec5SDimitry Andric return selectTrunc(I); 1386*0b57cec5SDimitry Andric case Instruction::ZExt: 1387*0b57cec5SDimitry Andric return selectZExt(I); 1388*0b57cec5SDimitry Andric case Instruction::SExt: 1389*0b57cec5SDimitry Andric return selectSExt(I); 1390*0b57cec5SDimitry Andric case Instruction::ICmp: 1391*0b57cec5SDimitry Andric return selectICmp(I); 1392*0b57cec5SDimitry Andric case Instruction::FCmp: 1393*0b57cec5SDimitry Andric return selectFCmp(I); 1394*0b57cec5SDimitry Andric case Instruction::BitCast: 1395*0b57cec5SDimitry Andric return selectBitCast(I); 1396*0b57cec5SDimitry Andric case Instruction::Load: 1397*0b57cec5SDimitry Andric return selectLoad(I); 1398*0b57cec5SDimitry Andric case Instruction::Store: 1399*0b57cec5SDimitry Andric return selectStore(I); 1400*0b57cec5SDimitry Andric case Instruction::Br: 1401*0b57cec5SDimitry Andric return selectBr(I); 1402*0b57cec5SDimitry Andric case Instruction::Ret: 1403*0b57cec5SDimitry Andric return selectRet(I); 1404*0b57cec5SDimitry Andric case Instruction::Unreachable: 1405*0b57cec5SDimitry Andric return selectUnreachable(I); 1406*0b57cec5SDimitry Andric default: 1407*0b57cec5SDimitry Andric break; 1408*0b57cec5SDimitry Andric } 1409*0b57cec5SDimitry Andric 1410*0b57cec5SDimitry Andric // Fall back to target-independent instruction selection. 1411*0b57cec5SDimitry Andric return selectOperator(I, I->getOpcode()); 1412*0b57cec5SDimitry Andric } 1413*0b57cec5SDimitry Andric 1414*0b57cec5SDimitry Andric FastISel *WebAssembly::createFastISel(FunctionLoweringInfo &FuncInfo, 1415*0b57cec5SDimitry Andric const TargetLibraryInfo *LibInfo) { 1416*0b57cec5SDimitry Andric return new WebAssemblyFastISel(FuncInfo, LibInfo); 1417*0b57cec5SDimitry Andric } 1418