Nick Lewycky | c9e935c | 2011-12-15 22:58:58 +0000 | [diff] [blame] | 1 | //===----- TargetFrameLoweringImpl.cpp - Implement target frame interface --==// |
Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 2 | // |
Misha Brukman | 505a083 | 2004-03-11 23:52:43 +0000 | [diff] [blame] | 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
Chris Lattner | f3ebc3f | 2007-12-29 20:36:04 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 7 | // |
Misha Brukman | 505a083 | 2004-03-11 23:52:43 +0000 | [diff] [blame] | 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
| 10 | // Implements the layout of a stack frame on the target machine. |
| 11 | // |
| 12 | //===----------------------------------------------------------------------===// |
| 13 | |
Matthias Braun | 0256486 | 2015-07-14 17:17:13 +0000 | [diff] [blame] | 14 | #include "llvm/ADT/BitVector.h" |
Chandler Carruth | ed0881b | 2012-12-03 16:50:05 +0000 | [diff] [blame] | 15 | #include "llvm/Target/TargetFrameLowering.h" |
Anton Korobeynikov | 36590fc | 2010-11-20 16:14:57 +0000 | [diff] [blame] | 16 | #include "llvm/CodeGen/MachineFrameInfo.h" |
| 17 | #include "llvm/CodeGen/MachineFunction.h" |
Matthias Braun | 0256486 | 2015-07-14 17:17:13 +0000 | [diff] [blame] | 18 | #include "llvm/CodeGen/MachineModuleInfo.h" |
| 19 | #include "llvm/CodeGen/MachineRegisterInfo.h" |
Akira Hatanaka | ddf76aa | 2015-05-23 01:14:08 +0000 | [diff] [blame] | 20 | #include "llvm/IR/Function.h" |
Anton Korobeynikov | 4687778 | 2010-11-20 15:59:32 +0000 | [diff] [blame] | 21 | #include "llvm/Target/TargetRegisterInfo.h" |
Eric Christopher | d913448 | 2014-08-04 21:25:23 +0000 | [diff] [blame] | 22 | #include "llvm/Target/TargetSubtargetInfo.h" |
Misha Brukman | 505a083 | 2004-03-11 23:52:43 +0000 | [diff] [blame] | 23 | #include <cstdlib> |
Misha Brukman | 505a083 | 2004-03-11 23:52:43 +0000 | [diff] [blame] | 24 | using namespace llvm; |
| 25 | |
Anton Korobeynikov | 2f93128 | 2011-01-10 12:39:04 +0000 | [diff] [blame] | 26 | TargetFrameLowering::~TargetFrameLowering() { |
Reid Spencer | ff7b16c | 2005-04-25 02:55:55 +0000 | [diff] [blame] | 27 | } |
Anton Korobeynikov | 14ee344 | 2010-11-18 23:25:52 +0000 | [diff] [blame] | 28 | |
Akira Hatanaka | ddf76aa | 2015-05-23 01:14:08 +0000 | [diff] [blame] | 29 | /// The default implementation just looks at attribute "no-frame-pointer-elim". |
| 30 | bool TargetFrameLowering::noFramePointerElim(const MachineFunction &MF) const { |
| 31 | auto Attr = MF.getFunction()->getFnAttribute("no-frame-pointer-elim"); |
| 32 | return Attr.getValueAsString() == "true"; |
| 33 | } |
| 34 | |
Anton Korobeynikov | 4687778 | 2010-11-20 15:59:32 +0000 | [diff] [blame] | 35 | /// getFrameIndexOffset - Returns the displacement from the frame register to |
| 36 | /// the stack frame of the specified index. This is the default implementation |
| 37 | /// which is overridden for some targets. |
Anton Korobeynikov | 2f93128 | 2011-01-10 12:39:04 +0000 | [diff] [blame] | 38 | int TargetFrameLowering::getFrameIndexOffset(const MachineFunction &MF, |
Eric Christopher | 307c2cb | 2014-10-14 07:22:08 +0000 | [diff] [blame] | 39 | int FI) const { |
Anton Korobeynikov | 4687778 | 2010-11-20 15:59:32 +0000 | [diff] [blame] | 40 | const MachineFrameInfo *MFI = MF.getFrameInfo(); |
| 41 | return MFI->getObjectOffset(FI) + MFI->getStackSize() - |
| 42 | getOffsetOfLocalArea() + MFI->getOffsetAdjustment(); |
| 43 | } |
| 44 | |
Anton Korobeynikov | 2f93128 | 2011-01-10 12:39:04 +0000 | [diff] [blame] | 45 | int TargetFrameLowering::getFrameIndexReference(const MachineFunction &MF, |
| 46 | int FI, unsigned &FrameReg) const { |
Eric Christopher | fc6de42 | 2014-08-05 02:39:49 +0000 | [diff] [blame] | 47 | const TargetRegisterInfo *RI = MF.getSubtarget().getRegisterInfo(); |
Anton Korobeynikov | 4687778 | 2010-11-20 15:59:32 +0000 | [diff] [blame] | 48 | |
| 49 | // By default, assume all frame indices are referenced via whatever |
| 50 | // getFrameRegister() says. The target can override this if it's doing |
| 51 | // something different. |
| 52 | FrameReg = RI->getFrameRegister(MF); |
| 53 | return getFrameIndexOffset(MF, FI); |
| 54 | } |
Michael Kuperstein | 13fbd45 | 2015-02-01 16:56:04 +0000 | [diff] [blame] | 55 | |
| 56 | bool TargetFrameLowering::needsFrameIndexResolution( |
| 57 | const MachineFunction &MF) const { |
| 58 | return MF.getFrameInfo()->hasStackObjects(); |
| 59 | } |
Matthias Braun | 0256486 | 2015-07-14 17:17:13 +0000 | [diff] [blame] | 60 | |
| 61 | void TargetFrameLowering::determineCalleeSaves(MachineFunction &MF, |
| 62 | BitVector &SavedRegs, |
| 63 | RegScavenger *RS) const { |
| 64 | // Get the callee saved register list... |
| 65 | const TargetRegisterInfo &TRI = *MF.getSubtarget().getRegisterInfo(); |
| 66 | const MCPhysReg *CSRegs = TRI.getCalleeSavedRegs(&MF); |
| 67 | |
| 68 | // Early exit if there are no callee saved registers. |
| 69 | if (!CSRegs || CSRegs[0] == 0) |
| 70 | return; |
| 71 | |
| 72 | SavedRegs.resize(TRI.getNumRegs()); |
| 73 | |
| 74 | // In Naked functions we aren't going to save any registers. |
| 75 | if (MF.getFunction()->hasFnAttribute(Attribute::Naked)) |
| 76 | return; |
| 77 | |
| 78 | // Functions which call __builtin_unwind_init get all their registers saved. |
| 79 | bool CallsUnwindInit = MF.getMMI().callsUnwindInit(); |
| 80 | const MachineRegisterInfo &MRI = MF.getRegInfo(); |
| 81 | for (unsigned i = 0; CSRegs[i]; ++i) { |
| 82 | unsigned Reg = CSRegs[i]; |
| 83 | if (CallsUnwindInit || MRI.isPhysRegModified(Reg)) |
| 84 | SavedRegs.set(Reg); |
| 85 | } |
| 86 | } |