| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 1 | //===-- Mips16FrameLowering.cpp - Mips16 Frame Information ----------------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | // This file contains the Mips16 implementation of TargetFrameLowering class. | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
|  | 14 | #include "Mips16FrameLowering.h" | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 15 | #include "MCTargetDesc/MipsBaseInfo.h" | 
| Chandler Carruth | be81023 | 2013-01-02 10:22:59 +0000 | [diff] [blame] | 16 | #include "Mips16InstrInfo.h" | 
| Chandler Carruth | ed0881b | 2012-12-03 16:50:05 +0000 | [diff] [blame] | 17 | #include "MipsInstrInfo.h" | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 18 | #include "llvm/CodeGen/MachineFrameInfo.h" | 
|  | 19 | #include "llvm/CodeGen/MachineFunction.h" | 
|  | 20 | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
|  | 21 | #include "llvm/CodeGen/MachineModuleInfo.h" | 
|  | 22 | #include "llvm/CodeGen/MachineRegisterInfo.h" | 
| Chandler Carruth | 9fb823b | 2013-01-02 11:36:10 +0000 | [diff] [blame] | 23 | #include "llvm/IR/DataLayout.h" | 
|  | 24 | #include "llvm/IR/Function.h" | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 25 | #include "llvm/Support/CommandLine.h" | 
| Chandler Carruth | ed0881b | 2012-12-03 16:50:05 +0000 | [diff] [blame] | 26 | #include "llvm/Target/TargetOptions.h" | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 27 |  | 
|  | 28 | using namespace llvm; | 
|  | 29 |  | 
|  | 30 | void Mips16FrameLowering::emitPrologue(MachineFunction &MF) const { | 
|  | 31 | MachineBasicBlock &MBB = MF.front(); | 
|  | 32 | MachineFrameInfo *MFI = MF.getFrameInfo(); | 
| Reed Kotler | d019dbf | 2012-12-20 04:07:42 +0000 | [diff] [blame] | 33 | const Mips16InstrInfo &TII = | 
|  | 34 | *static_cast<const Mips16InstrInfo*>(MF.getTarget().getInstrInfo()); | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 35 | MachineBasicBlock::iterator MBBI = MBB.begin(); | 
|  | 36 | DebugLoc dl = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc(); | 
|  | 37 | uint64_t StackSize = MFI->getStackSize(); | 
|  | 38 |  | 
|  | 39 | // No need to allocate space on the stack. | 
|  | 40 | if (StackSize == 0 && !MFI->adjustsStack()) return; | 
|  | 41 |  | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 42 | MachineModuleInfo &MMI = MF.getMMI(); | 
| Rafael Espindola | b08d2c2 | 2013-05-16 21:02:15 +0000 | [diff] [blame] | 43 | const MCRegisterInfo &MRI = MMI.getContext().getRegisterInfo(); | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 44 | MachineLocation DstML, SrcML; | 
|  | 45 |  | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 46 | // Adjust stack. | 
| Reed Kotler | d019dbf | 2012-12-20 04:07:42 +0000 | [diff] [blame] | 47 | TII.makeFrame(Mips::SP, StackSize, MBB, MBBI); | 
| Reed Kotler | 3589dd7 | 2012-10-28 06:02:37 +0000 | [diff] [blame] | 48 |  | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 49 | // emit ".cfi_def_cfa_offset StackSize" | 
|  | 50 | MCSymbol *AdjustSPLabel = MMI.getContext().CreateTempSymbol(); | 
|  | 51 | BuildMI(MBB, MBBI, dl, | 
|  | 52 | TII.get(TargetOpcode::PROLOG_LABEL)).addSym(AdjustSPLabel); | 
| Rafael Espindola | b08d2c2 | 2013-05-16 21:02:15 +0000 | [diff] [blame] | 53 | MMI.addFrameInst( | 
|  | 54 | MCCFIInstruction::createDefCfaOffset(AdjustSPLabel, -StackSize)); | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 55 |  | 
|  | 56 | MCSymbol *CSLabel = MMI.getContext().CreateTempSymbol(); | 
|  | 57 | BuildMI(MBB, MBBI, dl, | 
|  | 58 | TII.get(TargetOpcode::PROLOG_LABEL)).addSym(CSLabel); | 
| Rafael Espindola | b08d2c2 | 2013-05-16 21:02:15 +0000 | [diff] [blame] | 59 | unsigned S1 = MRI.getDwarfRegNum(Mips::S1, true); | 
|  | 60 | MMI.addFrameInst(MCCFIInstruction::createOffset(CSLabel, S1, -8)); | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 61 |  | 
| Rafael Espindola | b08d2c2 | 2013-05-16 21:02:15 +0000 | [diff] [blame] | 62 | unsigned S0 = MRI.getDwarfRegNum(Mips::S0, true); | 
|  | 63 | MMI.addFrameInst(MCCFIInstruction::createOffset(CSLabel, S0, -12)); | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 64 |  | 
| Rafael Espindola | b08d2c2 | 2013-05-16 21:02:15 +0000 | [diff] [blame] | 65 | unsigned RA = MRI.getDwarfRegNum(Mips::RA, true); | 
|  | 66 | MMI.addFrameInst(MCCFIInstruction::createOffset(CSLabel, RA, -4)); | 
| Reed Kotler | d11acc7 | 2012-12-20 06:59:37 +0000 | [diff] [blame] | 67 |  | 
| Reed Kotler | 3589dd7 | 2012-10-28 06:02:37 +0000 | [diff] [blame] | 68 | if (hasFP(MF)) | 
|  | 69 | BuildMI(MBB, MBBI, dl, TII.get(Mips::MoveR3216), Mips::S0) | 
|  | 70 | .addReg(Mips::SP); | 
|  | 71 |  | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 72 | } | 
|  | 73 |  | 
|  | 74 | void Mips16FrameLowering::emitEpilogue(MachineFunction &MF, | 
|  | 75 | MachineBasicBlock &MBB) const { | 
|  | 76 | MachineBasicBlock::iterator MBBI = MBB.getLastNonDebugInstr(); | 
|  | 77 | MachineFrameInfo *MFI = MF.getFrameInfo(); | 
| Reed Kotler | d019dbf | 2012-12-20 04:07:42 +0000 | [diff] [blame] | 78 | const Mips16InstrInfo &TII = | 
|  | 79 | *static_cast<const Mips16InstrInfo*>(MF.getTarget().getInstrInfo()); | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 80 | DebugLoc dl = MBBI->getDebugLoc(); | 
|  | 81 | uint64_t StackSize = MFI->getStackSize(); | 
|  | 82 |  | 
|  | 83 | if (!StackSize) | 
|  | 84 | return; | 
|  | 85 |  | 
| Reed Kotler | 3589dd7 | 2012-10-28 06:02:37 +0000 | [diff] [blame] | 86 | if (hasFP(MF)) | 
|  | 87 | BuildMI(MBB, MBBI, dl, TII.get(Mips::Move32R16), Mips::SP) | 
|  | 88 | .addReg(Mips::S0); | 
|  | 89 |  | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 90 | // Adjust stack. | 
| Reed Kotler | d019dbf | 2012-12-20 04:07:42 +0000 | [diff] [blame] | 91 | // assumes stacksize multiple of 8 | 
|  | 92 | TII.restoreFrame(Mips::SP, StackSize, MBB, MBBI); | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 93 | } | 
|  | 94 |  | 
|  | 95 | bool Mips16FrameLowering:: | 
|  | 96 | spillCalleeSavedRegisters(MachineBasicBlock &MBB, | 
|  | 97 | MachineBasicBlock::iterator MI, | 
|  | 98 | const std::vector<CalleeSavedInfo> &CSI, | 
|  | 99 | const TargetRegisterInfo *TRI) const { | 
| Akira Hatanaka | cd04e2b | 2012-09-21 01:08:16 +0000 | [diff] [blame] | 100 | MachineFunction *MF = MBB.getParent(); | 
|  | 101 | MachineBasicBlock *EntryBlock = MF->begin(); | 
| Akira Hatanaka | cd04e2b | 2012-09-21 01:08:16 +0000 | [diff] [blame] | 102 |  | 
|  | 103 | // | 
|  | 104 | // Registers RA, S0,S1 are the callee saved registers and they | 
|  | 105 | // will be saved with the "save" instruction | 
|  | 106 | // during emitPrologue | 
|  | 107 | // | 
|  | 108 | for (unsigned i = 0, e = CSI.size(); i != e; ++i) { | 
|  | 109 | // Add the callee-saved register as live-in. Do not add if the register is | 
|  | 110 | // RA and return address is taken, because it has already been added in | 
|  | 111 | // method MipsTargetLowering::LowerRETURNADDR. | 
|  | 112 | // It's killed at the spill, unless the register is RA and return address | 
|  | 113 | // is taken. | 
|  | 114 | unsigned Reg = CSI[i].getReg(); | 
|  | 115 | bool IsRAAndRetAddrIsTaken = (Reg == Mips::RA) | 
|  | 116 | && MF->getFrameInfo()->isReturnAddressTaken(); | 
|  | 117 | if (!IsRAAndRetAddrIsTaken) | 
|  | 118 | EntryBlock->addLiveIn(Reg); | 
|  | 119 | } | 
|  | 120 |  | 
|  | 121 | return true; | 
|  | 122 | } | 
|  | 123 |  | 
|  | 124 | bool Mips16FrameLowering::restoreCalleeSavedRegisters(MachineBasicBlock &MBB, | 
|  | 125 | MachineBasicBlock::iterator MI, | 
|  | 126 | const std::vector<CalleeSavedInfo> &CSI, | 
|  | 127 | const TargetRegisterInfo *TRI) const { | 
|  | 128 | // | 
|  | 129 | // Registers RA,S0,S1 are the callee saved registers and they will be restored | 
|  | 130 | // with the restore instruction during emitEpilogue. | 
|  | 131 | // We need to override this virtual function, otherwise llvm will try and | 
|  | 132 | // restore the registers on it's on from the stack. | 
|  | 133 | // | 
|  | 134 |  | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 135 | return true; | 
|  | 136 | } | 
|  | 137 |  | 
| Eli Bendersky | 8da8716 | 2013-02-21 20:05:00 +0000 | [diff] [blame] | 138 | // Eliminate ADJCALLSTACKDOWN, ADJCALLSTACKUP pseudo instructions | 
|  | 139 | void Mips16FrameLowering:: | 
|  | 140 | eliminateCallFramePseudoInstr(MachineFunction &MF, MachineBasicBlock &MBB, | 
|  | 141 | MachineBasicBlock::iterator I) const { | 
|  | 142 | if (!hasReservedCallFrame(MF)) { | 
|  | 143 | int64_t Amount = I->getOperand(0).getImm(); | 
|  | 144 |  | 
|  | 145 | if (I->getOpcode() == Mips::ADJCALLSTACKDOWN) | 
|  | 146 | Amount = -Amount; | 
|  | 147 |  | 
|  | 148 | const Mips16InstrInfo &TII = | 
|  | 149 | *static_cast<const Mips16InstrInfo*>(MF.getTarget().getInstrInfo()); | 
|  | 150 |  | 
|  | 151 | TII.adjustStackPtr(Mips::SP, Amount, MBB, I); | 
|  | 152 | } | 
|  | 153 |  | 
|  | 154 | MBB.erase(I); | 
|  | 155 | } | 
|  | 156 |  | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 157 | bool | 
|  | 158 | Mips16FrameLowering::hasReservedCallFrame(const MachineFunction &MF) const { | 
| Reed Kotler | 27a7229 | 2012-10-31 05:21:10 +0000 | [diff] [blame] | 159 | const MachineFrameInfo *MFI = MF.getFrameInfo(); | 
|  | 160 | // Reserve call frame if the size of the maximum call frame fits into 15-bit | 
|  | 161 | // immediate field and there are no variable sized objects on the stack. | 
|  | 162 | return isInt<15>(MFI->getMaxCallFrameSize()) && !MFI->hasVarSizedObjects(); | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 163 | } | 
|  | 164 |  | 
|  | 165 | void Mips16FrameLowering:: | 
|  | 166 | processFunctionBeforeCalleeSavedScan(MachineFunction &MF, | 
|  | 167 | RegScavenger *RS) const { | 
| Akira Hatanaka | cd04e2b | 2012-09-21 01:08:16 +0000 | [diff] [blame] | 168 | MF.getRegInfo().setPhysRegUsed(Mips::RA); | 
|  | 169 | MF.getRegInfo().setPhysRegUsed(Mips::S0); | 
|  | 170 | MF.getRegInfo().setPhysRegUsed(Mips::S1); | 
| Akira Hatanaka | d1c43ce | 2012-07-31 22:50:19 +0000 | [diff] [blame] | 171 | } | 
| Akira Hatanaka | fab8929 | 2012-08-02 18:21:47 +0000 | [diff] [blame] | 172 |  | 
|  | 173 | const MipsFrameLowering * | 
|  | 174 | llvm::createMips16FrameLowering(const MipsSubtarget &ST) { | 
|  | 175 | return new Mips16FrameLowering(ST); | 
|  | 176 | } |