|  | //===- subzero/src/IceTargetLoweringX8632.h - x86-32 lowering ---*- C++ -*-===// | 
|  | // | 
|  | //                        The Subzero Code Generator | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file declares the TargetLoweringX8632 class, which | 
|  | // implements the TargetLowering interface for the x86-32 | 
|  | // architecture. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #ifndef SUBZERO_SRC_ICETARGETLOWERINGX8632_H | 
|  | #define SUBZERO_SRC_ICETARGETLOWERINGX8632_H | 
|  |  | 
|  | #include "IceDefs.h" | 
|  | #include "IceTargetLowering.h" | 
|  | #include "IceInstX8632.h" | 
|  |  | 
|  | namespace Ice { | 
|  |  | 
|  | class TargetX8632 : public TargetLowering { | 
|  | public: | 
|  | static TargetX8632 *create(Cfg *Func) { return new TargetX8632(Func); } | 
|  |  | 
|  | virtual void translateOm1(); | 
|  | virtual void translateO2(); | 
|  |  | 
|  | virtual Variable *getPhysicalRegister(SizeT RegNum); | 
|  | virtual IceString getRegName(SizeT RegNum, Type Ty) const; | 
|  | virtual llvm::SmallBitVector getRegisterSet(RegSetMask Include, | 
|  | RegSetMask Exclude) const; | 
|  | virtual const llvm::SmallBitVector &getRegisterSetForType(Type Ty) const { | 
|  | return TypeToRegisterSet[Ty]; | 
|  | } | 
|  | virtual bool hasFramePointer() const { return IsEbpBasedFrame; } | 
|  | virtual SizeT getFrameOrStackReg() const { | 
|  | return IsEbpBasedFrame ? Reg_ebp : Reg_esp; | 
|  | } | 
|  | virtual size_t typeWidthInBytesOnStack(Type Ty) const { | 
|  | // Round up to the next multiple of 4 bytes.  In particular, i1, | 
|  | // i8, and i16 are rounded up to 4 bytes. | 
|  | return (typeWidthInBytes(Ty) + 3) & ~3; | 
|  | } | 
|  | virtual SizeT getBundleAlignLog2Bytes() const { return 5; } | 
|  | virtual llvm::ArrayRef<uint8_t> getNonExecBundlePadding() const { | 
|  | static const uint8_t Padding[] = { 0xF4 }; | 
|  | return llvm::ArrayRef<uint8_t>(Padding, 1); | 
|  | } | 
|  | virtual void emitVariable(const Variable *Var, const Cfg *Func) const; | 
|  | virtual void lowerArguments(); | 
|  | virtual void addProlog(CfgNode *Node); | 
|  | virtual void addEpilog(CfgNode *Node); | 
|  | virtual void emitConstants() const; | 
|  | SizeT makeNextLabelNumber() { return NextLabelNumber++; } | 
|  | // Ensure that a 64-bit Variable has been split into 2 32-bit | 
|  | // Variables, creating them if necessary.  This is needed for all | 
|  | // I64 operations, and it is needed for pushing F64 arguments for | 
|  | // function calls using the 32-bit push instruction (though the | 
|  | // latter could be done by directly writing to the stack). | 
|  | void split64(Variable *Var); | 
|  | void finishArgumentLowering(Variable *Arg, Variable *FramePtr, | 
|  | size_t BasicFrameOffset, size_t &InArgsSizeBytes); | 
|  | Operand *loOperand(Operand *Operand); | 
|  | Operand *hiOperand(Operand *Operand); | 
|  |  | 
|  | enum Registers { | 
|  | #define X(val, init, name, name16, name8, scratch, preserved, stackptr,        \ | 
|  | frameptr, isI8, isInt, isFP)                                         \ | 
|  | val init, | 
|  | REGX8632_TABLE | 
|  | #undef X | 
|  | Reg_NUM | 
|  | }; | 
|  |  | 
|  | enum X86InstructionSet { | 
|  | // SSE2 is the PNaCl baseline instruction set. | 
|  | SSE2, | 
|  | SSE4_1 | 
|  | }; | 
|  |  | 
|  | X86InstructionSet getInstructionSet() const { return InstructionSet; } | 
|  |  | 
|  | protected: | 
|  | TargetX8632(Cfg *Func); | 
|  |  | 
|  | virtual void postLower(); | 
|  |  | 
|  | virtual void lowerAlloca(const InstAlloca *Inst); | 
|  | virtual void lowerArithmetic(const InstArithmetic *Inst); | 
|  | virtual void lowerAssign(const InstAssign *Inst); | 
|  | virtual void lowerBr(const InstBr *Inst); | 
|  | virtual void lowerCall(const InstCall *Inst); | 
|  | virtual void lowerCast(const InstCast *Inst); | 
|  | virtual void lowerExtractElement(const InstExtractElement *Inst); | 
|  | virtual void lowerFcmp(const InstFcmp *Inst); | 
|  | virtual void lowerIcmp(const InstIcmp *Inst); | 
|  | virtual void lowerIntrinsicCall(const InstIntrinsicCall *Inst); | 
|  | virtual void lowerInsertElement(const InstInsertElement *Inst); | 
|  | virtual void lowerLoad(const InstLoad *Inst); | 
|  | virtual void lowerPhi(const InstPhi *Inst); | 
|  | virtual void lowerRet(const InstRet *Inst); | 
|  | virtual void lowerSelect(const InstSelect *Inst); | 
|  | virtual void lowerStore(const InstStore *Inst); | 
|  | virtual void lowerSwitch(const InstSwitch *Inst); | 
|  | virtual void lowerUnreachable(const InstUnreachable *Inst); | 
|  | virtual void doAddressOptLoad(); | 
|  | virtual void doAddressOptStore(); | 
|  | virtual void randomlyInsertNop(float Probability); | 
|  |  | 
|  | // Naive lowering of cmpxchg. | 
|  | void lowerAtomicCmpxchg(Variable *DestPrev, Operand *Ptr, Operand *Expected, | 
|  | Operand *Desired); | 
|  | // Attempt a more optimized lowering of cmpxchg. Returns true if optimized. | 
|  | bool tryOptimizedCmpxchgCmpBr(Variable *DestPrev, Operand *Ptr, | 
|  | Operand *Expected, Operand *Desired); | 
|  | void lowerAtomicRMW(Variable *Dest, uint32_t Operation, Operand *Ptr, | 
|  | Operand *Val); | 
|  | void lowerCountZeros(bool Cttz, Type Ty, Variable *Dest, Operand *FirstVal, | 
|  | Operand *SecondVal); | 
|  |  | 
|  | typedef void (TargetX8632::*LowerBinOp)(Variable *, Operand *); | 
|  | void expandAtomicRMWAsCmpxchg(LowerBinOp op_lo, LowerBinOp op_hi, | 
|  | Variable *Dest, Operand *Ptr, Operand *Val); | 
|  |  | 
|  | void eliminateNextVectorSextInstruction(Variable *SignExtendedResult); | 
|  |  | 
|  | void scalarizeArithmetic(InstArithmetic::OpKind K, Variable *Dest, | 
|  | Operand *Src0, Operand *Src1); | 
|  |  | 
|  | void sortByAlignment(VarList &Dest, const VarList &Source) const; | 
|  |  | 
|  | // Operand legalization helpers.  To deal with address mode | 
|  | // constraints, the helpers will create a new Operand and emit | 
|  | // instructions that guarantee that the Operand kind is one of those | 
|  | // indicated by the LegalMask (a bitmask of allowed kinds).  If the | 
|  | // input Operand is known to already meet the constraints, it may be | 
|  | // simply returned as the result, without creating any new | 
|  | // instructions or operands. | 
|  | enum OperandLegalization { | 
|  | Legal_None = 0, | 
|  | Legal_Reg = 1 << 0, // physical register, not stack location | 
|  | Legal_Imm = 1 << 1, | 
|  | Legal_Mem = 1 << 2, // includes [eax+4*ecx] as well as [esp+12] | 
|  | // TODO(stichnot): LEAHACK: remove Legal_Reloc once a proper | 
|  | // emitter is used. | 
|  | Legal_Reloc = 1 << 3, | 
|  | Legal_All = ~Legal_None | 
|  | }; | 
|  | typedef uint32_t LegalMask; | 
|  | Operand *legalize(Operand *From, LegalMask Allowed = Legal_All & ~Legal_Reloc, | 
|  | bool AllowOverlap = false, | 
|  | int32_t RegNum = Variable::NoRegister); | 
|  | Variable *legalizeToVar(Operand *From, bool AllowOverlap = false, | 
|  | int32_t RegNum = Variable::NoRegister); | 
|  | // Turn a pointer operand into a memory operand that can be | 
|  | // used by a real load/store operation. Legalizes the operand as well. | 
|  | // This is a nop if the operand is already a legal memory operand. | 
|  | OperandX8632Mem *FormMemoryOperand(Operand *Ptr, Type Ty); | 
|  |  | 
|  | Variable *makeReg(Type Ty, int32_t RegNum = Variable::NoRegister); | 
|  | InstCall *makeHelperCall(const IceString &Name, Variable *Dest, | 
|  | SizeT MaxSrcs) { | 
|  | bool SuppressMangling = true; | 
|  | const Type FunctionPointerType = IceType_i32; | 
|  | Constant *CallTarget = | 
|  | Ctx->getConstantSym(FunctionPointerType, 0, Name, SuppressMangling); | 
|  | InstCall *Call = InstCall::create(Func, MaxSrcs, Dest, CallTarget); | 
|  | return Call; | 
|  | } | 
|  | static Type stackSlotType(); | 
|  |  | 
|  | Variable *copyToReg(Operand *Src, int32_t RegNum = Variable::NoRegister); | 
|  |  | 
|  | // Returns a vector in a register with the given constant entries. | 
|  | Variable *makeVectorOfZeros(Type Ty, int32_t RegNum = Variable::NoRegister); | 
|  | Variable *makeVectorOfOnes(Type Ty, int32_t RegNum = Variable::NoRegister); | 
|  | Variable *makeVectorOfMinusOnes(Type Ty, | 
|  | int32_t RegNum = Variable::NoRegister); | 
|  | Variable *makeVectorOfHighOrderBits(Type Ty, | 
|  | int32_t RegNum = Variable::NoRegister); | 
|  |  | 
|  | // Return a memory operand corresponding to a stack allocated Variable. | 
|  | OperandX8632Mem *getMemoryOperandForStackSlot(Type Ty, Variable *Slot, | 
|  | uint32_t Offset = 0); | 
|  |  | 
|  | // The following are helpers that insert lowered x86 instructions | 
|  | // with minimal syntactic overhead, so that the lowering code can | 
|  | // look as close to assembly as practical. | 
|  | void _adc(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Adc::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _add(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Add::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _adjust_stack(int32_t Amount) { | 
|  | Context.insert(InstX8632AdjustStack::create(Func, Amount)); | 
|  | } | 
|  | void _addps(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Addps::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _addss(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Addss::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _and(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632And::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _blendvps(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Blendvps::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _br(InstX8632::BrCond Condition, CfgNode *TargetTrue, | 
|  | CfgNode *TargetFalse) { | 
|  | Context.insert( | 
|  | InstX8632Br::create(Func, TargetTrue, TargetFalse, Condition)); | 
|  | } | 
|  | void _br(CfgNode *Target) { | 
|  | Context.insert(InstX8632Br::create(Func, Target)); | 
|  | } | 
|  | void _br(InstX8632::BrCond Condition, CfgNode *Target) { | 
|  | Context.insert(InstX8632Br::create(Func, Target, Condition)); | 
|  | } | 
|  | void _br(InstX8632::BrCond Condition, InstX8632Label *Label) { | 
|  | Context.insert(InstX8632Br::create(Func, Label, Condition)); | 
|  | } | 
|  | void _bsf(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Bsf::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _bsr(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Bsr::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _bswap(Variable *SrcDest) { | 
|  | Context.insert(InstX8632Bswap::create(Func, SrcDest)); | 
|  | } | 
|  | void _cbwdq(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Cbwdq::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _cmov(Variable *Dest, Operand *Src0, InstX8632::BrCond Condition) { | 
|  | Context.insert(InstX8632Cmov::create(Func, Dest, Src0, Condition)); | 
|  | } | 
|  | void _cmp(Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Icmp::create(Func, Src0, Src1)); | 
|  | } | 
|  | void _cmpps(Variable *Dest, Operand *Src0, | 
|  | InstX8632Cmpps::CmppsCond Condition) { | 
|  | Context.insert(InstX8632Cmpps::create(Func, Dest, Src0, Condition)); | 
|  | } | 
|  | void _cmpxchg(Operand *DestOrAddr, Variable *Eax, Variable *Desired, | 
|  | bool Locked) { | 
|  | Context.insert( | 
|  | InstX8632Cmpxchg::create(Func, DestOrAddr, Eax, Desired, Locked)); | 
|  | // Mark eax as possibly modified by cmpxchg. | 
|  | Context.insert( | 
|  | InstFakeDef::create(Func, Eax, llvm::dyn_cast<Variable>(DestOrAddr))); | 
|  | } | 
|  | void _cmpxchg8b(OperandX8632 *Addr, Variable *Edx, Variable *Eax, | 
|  | Variable *Ecx, Variable *Ebx, bool Locked) { | 
|  | Context.insert( | 
|  | InstX8632Cmpxchg8b::create(Func, Addr, Edx, Eax, Ecx, Ebx, Locked)); | 
|  | // Mark edx, and eax as possibly modified by cmpxchg8b. | 
|  | Context.insert(InstFakeDef::create(Func, Edx)); | 
|  | Context.insert(InstFakeDef::create(Func, Eax)); | 
|  | } | 
|  | void _cvt(Variable *Dest, Operand *Src0) { | 
|  | const bool Trunc = false; | 
|  | Context.insert(InstX8632Cvt::create(Func, Dest, Src0, Trunc)); | 
|  | } | 
|  | void _cvtt(Variable *Dest, Operand *Src0) { | 
|  | const bool Trunc = true; | 
|  | Context.insert(InstX8632Cvt::create(Func, Dest, Src0, Trunc)); | 
|  | } | 
|  | void _div(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Div::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _divps(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Divps::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _divss(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Divss::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _fld(Operand *Src0) { Context.insert(InstX8632Fld::create(Func, Src0)); } | 
|  | void _fstp(Variable *Dest) { | 
|  | Context.insert(InstX8632Fstp::create(Func, Dest)); | 
|  | } | 
|  | void _idiv(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Idiv::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _imul(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Imul::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _insertps(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Insertps::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _lea(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Lea::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _mfence() { Context.insert(InstX8632Mfence::create(Func)); } | 
|  | // If Dest=NULL is passed in, then a new variable is created, marked | 
|  | // as infinite register allocation weight, and returned through the | 
|  | // in/out Dest argument. | 
|  | void _mov(Variable *&Dest, Operand *Src0, | 
|  | int32_t RegNum = Variable::NoRegister) { | 
|  | if (Dest == NULL) { | 
|  | Dest = legalizeToVar(Src0, false, RegNum); | 
|  | } else { | 
|  | Context.insert(InstX8632Mov::create(Func, Dest, Src0)); | 
|  | } | 
|  | } | 
|  | void _movd(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Movd::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _movp(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Movp::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _movq(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Movq::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _movss(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Movss::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _movsx(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Movsx::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _movzx(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Movzx::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _mul(Variable *Dest, Variable *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Mul::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _mulps(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Mulps::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _mulss(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Mulss::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _neg(Variable *SrcDest) { | 
|  | Context.insert(InstX8632Neg::create(Func, SrcDest)); | 
|  | } | 
|  | void _nop(SizeT Variant) { | 
|  | Context.insert(InstX8632Nop::create(Func, Variant)); | 
|  | } | 
|  | void _or(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Or::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _padd(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Padd::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pand(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pand::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pandn(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pandn::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pblendvb(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Pblendvb::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _pcmpeq(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pcmpeq::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pcmpgt(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pcmpgt::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pextr(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Pextr::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _pinsr(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Pinsr::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _pmull(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pmull::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pmuludq(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pmuludq::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pop(Variable *Dest) { | 
|  | Context.insert(InstX8632Pop::create(Func, Dest)); | 
|  | } | 
|  | void _por(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Por::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _pshufd(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Pshufd::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _psll(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Psll::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _psra(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Psra::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _psub(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Psub::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _push(Operand *Src0, bool SuppressStackAdjustment = false) { | 
|  | Context.insert(InstX8632Push::create(Func, Src0, SuppressStackAdjustment)); | 
|  | } | 
|  | void _pxor(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Pxor::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _ret(Variable *Src0 = NULL) { | 
|  | Context.insert(InstX8632Ret::create(Func, Src0)); | 
|  | } | 
|  | void _rol(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Rol::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _sar(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Sar::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _sbb(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Sbb::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _shl(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Shl::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _shld(Variable *Dest, Variable *Src0, Variable *Src1) { | 
|  | Context.insert(InstX8632Shld::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _shr(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Shr::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _shrd(Variable *Dest, Variable *Src0, Variable *Src1) { | 
|  | Context.insert(InstX8632Shrd::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _shufps(Variable *Dest, Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Shufps::create(Func, Dest, Src0, Src1)); | 
|  | } | 
|  | void _sqrtss(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Sqrtss::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _store(Operand *Value, OperandX8632 *Mem) { | 
|  | Context.insert(InstX8632Store::create(Func, Value, Mem)); | 
|  | } | 
|  | void _storep(Operand *Value, OperandX8632 *Mem) { | 
|  | Context.insert(InstX8632StoreP::create(Func, Value, Mem)); | 
|  | } | 
|  | void _storeq(Operand *Value, OperandX8632 *Mem) { | 
|  | Context.insert(InstX8632StoreQ::create(Func, Value, Mem)); | 
|  | } | 
|  | void _sub(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Sub::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _subps(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Subps::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _subss(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Subss::create(Func, Dest, Src0)); | 
|  | } | 
|  | void _test(Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Test::create(Func, Src0, Src1)); | 
|  | } | 
|  | void _ucomiss(Operand *Src0, Operand *Src1) { | 
|  | Context.insert(InstX8632Ucomiss::create(Func, Src0, Src1)); | 
|  | } | 
|  | void _ud2() { Context.insert(InstX8632UD2::create(Func)); } | 
|  | void _xadd(Operand *Dest, Variable *Src, bool Locked) { | 
|  | Context.insert(InstX8632Xadd::create(Func, Dest, Src, Locked)); | 
|  | // The xadd exchanges Dest and Src (modifying Src). | 
|  | // Model that update with a FakeDef. | 
|  | Context.insert( | 
|  | InstFakeDef::create(Func, Src, llvm::dyn_cast<Variable>(Dest))); | 
|  | } | 
|  | void _xchg(Operand *Dest, Variable *Src) { | 
|  | Context.insert(InstX8632Xchg::create(Func, Dest, Src)); | 
|  | // The xchg modifies Dest and Src -- model that update with a FakeDef. | 
|  | Context.insert( | 
|  | InstFakeDef::create(Func, Src, llvm::dyn_cast<Variable>(Dest))); | 
|  | } | 
|  | void _xor(Variable *Dest, Operand *Src0) { | 
|  | Context.insert(InstX8632Xor::create(Func, Dest, Src0)); | 
|  | } | 
|  |  | 
|  | const X86InstructionSet InstructionSet; | 
|  | bool IsEbpBasedFrame; | 
|  | bool NeedsStackAlignment; | 
|  | size_t FrameSizeLocals; | 
|  | size_t SpillAreaSizeBytes; | 
|  | llvm::SmallBitVector TypeToRegisterSet[IceType_NUM]; | 
|  | llvm::SmallBitVector ScratchRegs; | 
|  | llvm::SmallBitVector RegsUsed; | 
|  | SizeT NextLabelNumber; | 
|  | bool ComputedLiveRanges; | 
|  | VarList PhysicalRegisters; | 
|  | static IceString RegNames[]; | 
|  |  | 
|  | private: | 
|  | TargetX8632(const TargetX8632 &) LLVM_DELETED_FUNCTION; | 
|  | TargetX8632 &operator=(const TargetX8632 &) LLVM_DELETED_FUNCTION; | 
|  | virtual ~TargetX8632() {} | 
|  | template <typename T> void emitConstantPool() const; | 
|  | }; | 
|  |  | 
|  | class TargetGlobalInitX8632 : public TargetGlobalInitLowering { | 
|  | public: | 
|  | static TargetGlobalInitLowering *create(GlobalContext *Ctx) { | 
|  | return new TargetGlobalInitX8632(Ctx); | 
|  | } | 
|  | virtual void lower(const IceString &Name, SizeT Align, bool IsInternal, | 
|  | bool IsConst, bool IsZeroInitializer, SizeT Size, | 
|  | const char *Data, bool DisableTranslation); | 
|  |  | 
|  | protected: | 
|  | TargetGlobalInitX8632(GlobalContext *Ctx); | 
|  |  | 
|  | private: | 
|  | TargetGlobalInitX8632(const TargetGlobalInitX8632 &) LLVM_DELETED_FUNCTION; | 
|  | TargetGlobalInitX8632 & | 
|  | operator=(const TargetGlobalInitX8632 &) LLVM_DELETED_FUNCTION; | 
|  | virtual ~TargetGlobalInitX8632() {} | 
|  | }; | 
|  |  | 
|  | template <> void ConstantInteger::emit(GlobalContext *Ctx) const; | 
|  | template <> void ConstantFloat::emit(GlobalContext *Ctx) const; | 
|  | template <> void ConstantDouble::emit(GlobalContext *Ctx) const; | 
|  |  | 
|  | } // end of namespace Ice | 
|  |  | 
|  | #endif // SUBZERO_SRC_ICETARGETLOWERINGX8632_H |