| //===-- MipsISelDAGToDAG.cpp - A Dag to Dag Inst Selector for Mips --------===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file defines an instruction selector for the MIPS target. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #define DEBUG_TYPE "mips-isel" |
| #include "MipsISelDAGToDAG.h" |
| #include "Mips16ISelDAGToDAG.h" |
| #include "MipsSEISelDAGToDAG.h" |
| #include "Mips.h" |
| #include "MCTargetDesc/MipsBaseInfo.h" |
| #include "MipsAnalyzeImmediate.h" |
| #include "MipsMachineFunction.h" |
| #include "MipsRegisterInfo.h" |
| #include "llvm/CodeGen/MachineConstantPool.h" |
| #include "llvm/CodeGen/MachineFrameInfo.h" |
| #include "llvm/CodeGen/MachineFunction.h" |
| #include "llvm/CodeGen/MachineInstrBuilder.h" |
| #include "llvm/CodeGen/MachineRegisterInfo.h" |
| #include "llvm/CodeGen/SelectionDAGNodes.h" |
| #include "llvm/IR/GlobalValue.h" |
| #include "llvm/IR/Instructions.h" |
| #include "llvm/IR/Intrinsics.h" |
| #include "llvm/IR/Type.h" |
| #include "llvm/Support/CFG.h" |
| #include "llvm/Support/Debug.h" |
| #include "llvm/Support/ErrorHandling.h" |
| #include "llvm/Support/raw_ostream.h" |
| #include "llvm/Target/TargetMachine.h" |
| using namespace llvm; |
| |
| //===----------------------------------------------------------------------===// |
| // Instruction Selector Implementation |
| //===----------------------------------------------------------------------===// |
| |
| //===----------------------------------------------------------------------===// |
| // MipsDAGToDAGISel - MIPS specific code to select MIPS machine |
| // instructions for SelectionDAG operations. |
| //===----------------------------------------------------------------------===// |
| |
| bool MipsDAGToDAGISel::runOnMachineFunction(MachineFunction &MF) { |
| bool Ret = SelectionDAGISel::runOnMachineFunction(MF); |
| |
| processFunctionAfterISel(MF); |
| |
| return Ret; |
| } |
| |
| /// getGlobalBaseReg - Output the instructions required to put the |
| /// GOT address into a register. |
| SDNode *MipsDAGToDAGISel::getGlobalBaseReg() { |
| unsigned GlobalBaseReg = MF->getInfo<MipsFunctionInfo>()->getGlobalBaseReg(); |
| return CurDAG->getRegister(GlobalBaseReg, TLI.getPointerTy()).getNode(); |
| } |
| |
| /// ComplexPattern used on MipsInstrInfo |
| /// Used on Mips Load/Store instructions |
| bool MipsDAGToDAGISel::selectAddrRegImm(SDValue Addr, SDValue &Base, |
| SDValue &Offset) const { |
| llvm_unreachable("Unimplemented function."); |
| return false; |
| } |
| |
| bool MipsDAGToDAGISel::selectAddrDefault(SDValue Addr, SDValue &Base, |
| SDValue &Offset) const { |
| llvm_unreachable("Unimplemented function."); |
| return false; |
| } |
| |
| bool MipsDAGToDAGISel::selectIntAddr(SDValue Addr, SDValue &Base, |
| SDValue &Offset) const { |
| llvm_unreachable("Unimplemented function."); |
| return false; |
| } |
| |
| bool MipsDAGToDAGISel::selectAddr16(SDNode *Parent, SDValue N, SDValue &Base, |
| SDValue &Offset, SDValue &Alias) { |
| llvm_unreachable("Unimplemented function."); |
| return false; |
| } |
| |
| /// Select instructions not customized! Used for |
| /// expanded, promoted and normal instructions |
| SDNode* MipsDAGToDAGISel::Select(SDNode *Node) { |
| unsigned Opcode = Node->getOpcode(); |
| DebugLoc DL = Node->getDebugLoc(); |
| EVT NodeTy = Node->getValueType(0); |
| |
| // Dump information about the Node being selected |
| DEBUG(errs() << "Selecting: "; Node->dump(CurDAG); errs() << "\n"); |
| |
| // If we have a custom node, we already have selected! |
| if (Node->isMachineOpcode()) { |
| DEBUG(errs() << "== "; Node->dump(CurDAG); errs() << "\n"); |
| return NULL; |
| } |
| |
| // See if subclasses can handle this node. |
| std::pair<bool, SDNode*> Ret = selectNode(Node); |
| |
| if (Ret.first) |
| return Ret.second; |
| |
| switch(Opcode) { |
| default: break; |
| |
| // Get target GOT address. |
| case ISD::GLOBAL_OFFSET_TABLE: |
| return getGlobalBaseReg(); |
| |
| #ifndef NDEBUG |
| case ISD::LOAD: |
| case ISD::STORE: |
| assert(cast<MemSDNode>(Node)->getMemoryVT().getSizeInBits() / 8 <= |
| cast<MemSDNode>(Node)->getAlignment() && |
| "Unexpected unaligned loads/stores."); |
| break; |
| #endif |
| } |
| |
| // Select the default instruction |
| SDNode *ResNode = SelectCode(Node); |
| |
| DEBUG(errs() << "=> "); |
| if (ResNode == NULL || ResNode == Node) |
| DEBUG(Node->dump(CurDAG)); |
| else |
| DEBUG(ResNode->dump(CurDAG)); |
| DEBUG(errs() << "\n"); |
| return ResNode; |
| } |
| |
| bool MipsDAGToDAGISel:: |
| SelectInlineAsmMemoryOperand(const SDValue &Op, char ConstraintCode, |
| std::vector<SDValue> &OutOps) { |
| assert(ConstraintCode == 'm' && "unexpected asm memory constraint"); |
| OutOps.push_back(Op); |
| return false; |
| } |
| |
| /// createMipsISelDag - This pass converts a legalized DAG into a |
| /// MIPS-specific DAG, ready for instruction scheduling. |
| FunctionPass *llvm::createMipsISelDag(MipsTargetMachine &TM) { |
| if (TM.getSubtargetImpl()->inMips16Mode()) |
| return llvm::createMips16ISelDag(TM); |
| |
| return llvm::createMipsSEISelDag(TM); |
| } |