mirror of
https://github.com/RPCS3/llvm.git
synced 2025-01-14 13:57:51 +00:00
7271ac2c03
SystemZ wants normal register scavenging slots, as close to the stack or frame pointer as possible. The only reason it was using custom code was because PrologEpilogInserter assumed an x86-like layout, where the frame pointer is at the opposite end of the frame from the stack pointer. This meant that when frame pointer elimination was disabled, the slots ended up being as close as possible to the incoming stack pointer, which is the opposite of what we want on SystemZ. This patch adds a new knob to say which layout is used and converts SystemZ to use target-independent scavenging slots. It's one of the pieces needed to support frame-to-frame MVCs, where two slots might be required. The ABI requires us to allocate 160 bytes for calls, so one approach would be to use that area as temporary spill space instead. It would need some surgery to make sure that the slot isn't live across a call though. I stuck to the "isFPCloseToIncomingSP - ..." style comment on the "do what the surrounding code does" principle. The FP case is already covered by several Systemz/frame-* tests, which fail without the PrologueEpilogueInserter change, so no new ones are needed. No behavioural change intended. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@185696 91177308-0d34-0410-b5e6-96231b3b80d8
79 lines
3.0 KiB
C++
79 lines
3.0 KiB
C++
//===-- SystemZFrameLowering.h - Frame lowering for SystemZ -----*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SYSTEMZFRAMELOWERING_H
|
|
#define SYSTEMZFRAMELOWERING_H
|
|
|
|
#include "SystemZSubtarget.h"
|
|
#include "llvm/ADT/IndexedMap.h"
|
|
#include "llvm/Target/TargetFrameLowering.h"
|
|
|
|
namespace llvm {
|
|
class SystemZTargetMachine;
|
|
class SystemZSubtarget;
|
|
|
|
class SystemZFrameLowering : public TargetFrameLowering {
|
|
IndexedMap<unsigned> RegSpillOffsets;
|
|
|
|
protected:
|
|
const SystemZTargetMachine &TM;
|
|
const SystemZSubtarget &STI;
|
|
|
|
public:
|
|
SystemZFrameLowering(const SystemZTargetMachine &tm,
|
|
const SystemZSubtarget &sti);
|
|
|
|
// Override TargetFrameLowering.
|
|
virtual bool isFPCloseToIncomingSP() const LLVM_OVERRIDE { return false; }
|
|
virtual const SpillSlot *getCalleeSavedSpillSlots(unsigned &NumEntries) const
|
|
LLVM_OVERRIDE;
|
|
virtual void
|
|
processFunctionBeforeCalleeSavedScan(MachineFunction &MF,
|
|
RegScavenger *RS) const LLVM_OVERRIDE;
|
|
virtual bool
|
|
spillCalleeSavedRegisters(MachineBasicBlock &MBB,
|
|
MachineBasicBlock::iterator MBBI,
|
|
const std::vector<CalleeSavedInfo> &CSI,
|
|
const TargetRegisterInfo *TRI) const
|
|
LLVM_OVERRIDE;
|
|
virtual bool
|
|
restoreCalleeSavedRegisters(MachineBasicBlock &MBB,
|
|
MachineBasicBlock::iterator MBBII,
|
|
const std::vector<CalleeSavedInfo> &CSI,
|
|
const TargetRegisterInfo *TRI) const
|
|
LLVM_OVERRIDE;
|
|
virtual void processFunctionBeforeFrameFinalized(MachineFunction &MF,
|
|
RegScavenger *RS) const;
|
|
virtual void emitPrologue(MachineFunction &MF) const LLVM_OVERRIDE;
|
|
virtual void emitEpilogue(MachineFunction &MF,
|
|
MachineBasicBlock &MBB) const LLVM_OVERRIDE;
|
|
virtual bool hasFP(const MachineFunction &MF) const LLVM_OVERRIDE;
|
|
virtual int getFrameIndexOffset(const MachineFunction &MF,
|
|
int FI) const LLVM_OVERRIDE;
|
|
virtual bool hasReservedCallFrame(const MachineFunction &MF) const
|
|
LLVM_OVERRIDE;
|
|
virtual void
|
|
eliminateCallFramePseudoInstr(MachineFunction &MF,
|
|
MachineBasicBlock &MBB,
|
|
MachineBasicBlock::iterator MI) const
|
|
LLVM_OVERRIDE;
|
|
|
|
// Return the number of bytes in the callee-allocated part of the frame.
|
|
uint64_t getAllocatedStackSize(const MachineFunction &MF) const;
|
|
|
|
// Return the byte offset from the incoming stack pointer of Reg's
|
|
// ABI-defined save slot. Return 0 if no slot is defined for Reg.
|
|
unsigned getRegSpillOffset(unsigned Reg) const {
|
|
return RegSpillOffsets[Reg];
|
|
}
|
|
};
|
|
} // end namespace llvm
|
|
|
|
#endif
|