mirror of
https://github.com/RPCS3/llvm.git
synced 2025-05-22 21:35:57 +00:00

This patch makes one change to GOT handling and two changes to N64's relocation model handling. Furthermore, the jumptable encodings have been corrected for static N64. Big GOT handling is now done via a new SDNode MipsGotHi - this node is unconditionally lowered to an lui instruction. The first change to N64's relocation handling is the lifting of the restriction that N64 always uses PIC. Now it is possible to target static environments. The second change adds support for 64 bit symbols and enables them by default. Previously N64 had patterns for sym32 mode only. In this mode all symbols are assumed to have 32 bit addresses. sym32 mode support is selectable with attribute 'sym32'. A follow on patch for clang will add the necessary frontend parameter. This partially resolves PR/23485. Thanks to Brooks Davis for reporting the issue! This version corrects a "Conditional jump or move depends on uninitialised value(s)" error detected by valgrind present in the original commit. Reviewers: dsanders, seanbruno, zoran.jovanovic, vkalintiris Differential Revision: https://reviews.llvm.org/D23652 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@293279 91177308-0d34-0410-b5e6-96231b3b80d8
178 lines
6.5 KiB
C++
178 lines
6.5 KiB
C++
//===-- MipsSubtarget.cpp - Mips Subtarget Information --------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements the Mips specific subclass of TargetSubtargetInfo.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "MipsMachineFunction.h"
|
|
#include "Mips.h"
|
|
#include "MipsRegisterInfo.h"
|
|
#include "MipsSubtarget.h"
|
|
#include "MipsTargetMachine.h"
|
|
#include "llvm/IR/Attributes.h"
|
|
#include "llvm/IR/Function.h"
|
|
#include "llvm/Support/CommandLine.h"
|
|
#include "llvm/Support/Debug.h"
|
|
#include "llvm/Support/TargetRegistry.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
using namespace llvm;
|
|
|
|
#define DEBUG_TYPE "mips-subtarget"
|
|
|
|
#define GET_SUBTARGETINFO_TARGET_DESC
|
|
#define GET_SUBTARGETINFO_CTOR
|
|
#include "MipsGenSubtargetInfo.inc"
|
|
|
|
// FIXME: Maybe this should be on by default when Mips16 is specified
|
|
//
|
|
static cl::opt<bool>
|
|
Mixed16_32("mips-mixed-16-32", cl::init(false),
|
|
cl::desc("Allow for a mixture of Mips16 "
|
|
"and Mips32 code in a single output file"),
|
|
cl::Hidden);
|
|
|
|
static cl::opt<bool> Mips_Os16("mips-os16", cl::init(false),
|
|
cl::desc("Compile all functions that don't use "
|
|
"floating point as Mips 16"),
|
|
cl::Hidden);
|
|
|
|
static cl::opt<bool> Mips16HardFloat("mips16-hard-float", cl::NotHidden,
|
|
cl::desc("Enable mips16 hard float."),
|
|
cl::init(false));
|
|
|
|
static cl::opt<bool>
|
|
Mips16ConstantIslands("mips16-constant-islands", cl::NotHidden,
|
|
cl::desc("Enable mips16 constant islands."),
|
|
cl::init(true));
|
|
|
|
static cl::opt<bool>
|
|
GPOpt("mgpopt", cl::Hidden,
|
|
cl::desc("Enable gp-relative addressing of mips small data items"));
|
|
|
|
void MipsSubtarget::anchor() { }
|
|
|
|
MipsSubtarget::MipsSubtarget(const Triple &TT, const std::string &CPU,
|
|
const std::string &FS, bool little,
|
|
const MipsTargetMachine &TM)
|
|
: MipsGenSubtargetInfo(TT, CPU, FS), MipsArchVersion(MipsDefault),
|
|
IsLittle(little), IsSoftFloat(false), IsSingleFloat(false), IsFPXX(false),
|
|
NoABICalls(false), IsFP64bit(false), UseOddSPReg(true),
|
|
IsNaN2008bit(false), IsGP64bit(false), HasVFPU(false), HasCnMips(false),
|
|
HasMips3_32(false), HasMips3_32r2(false), HasMips4_32(false),
|
|
HasMips4_32r2(false), HasMips5_32r2(false), InMips16Mode(false),
|
|
InMips16HardFloat(Mips16HardFloat), InMicroMipsMode(false), HasDSP(false),
|
|
HasDSPR2(false), HasDSPR3(false), AllowMixed16_32(Mixed16_32 | Mips_Os16),
|
|
Os16(Mips_Os16), HasMSA(false), UseTCCInDIV(false), HasSym32(false),
|
|
HasEVA(false), TM(TM), TargetTriple(TT), TSInfo(),
|
|
InstrInfo(
|
|
MipsInstrInfo::create(initializeSubtargetDependencies(CPU, FS, TM))),
|
|
FrameLowering(MipsFrameLowering::create(*this)),
|
|
TLInfo(MipsTargetLowering::create(TM, *this)) {
|
|
|
|
PreviousInMips16Mode = InMips16Mode;
|
|
|
|
if (MipsArchVersion == MipsDefault)
|
|
MipsArchVersion = Mips32;
|
|
|
|
// Don't even attempt to generate code for MIPS-I and MIPS-V. They have not
|
|
// been tested and currently exist for the integrated assembler only.
|
|
if (MipsArchVersion == Mips1)
|
|
report_fatal_error("Code generation for MIPS-I is not implemented", false);
|
|
if (MipsArchVersion == Mips5)
|
|
report_fatal_error("Code generation for MIPS-V is not implemented", false);
|
|
|
|
// Check if Architecture and ABI are compatible.
|
|
assert(((!isGP64bit() && isABI_O32()) ||
|
|
(isGP64bit() && (isABI_N32() || isABI_N64()))) &&
|
|
"Invalid Arch & ABI pair.");
|
|
|
|
if (hasMSA() && !isFP64bit())
|
|
report_fatal_error("MSA requires a 64-bit FPU register file (FR=1 mode). "
|
|
"See -mattr=+fp64.",
|
|
false);
|
|
|
|
if (!isABI_O32() && !useOddSPReg())
|
|
report_fatal_error("-mattr=+nooddspreg requires the O32 ABI.", false);
|
|
|
|
if (IsFPXX && (isABI_N32() || isABI_N64()))
|
|
report_fatal_error("FPXX is not permitted for the N32/N64 ABI's.", false);
|
|
|
|
if (hasMips32r6()) {
|
|
StringRef ISA = hasMips64r6() ? "MIPS64r6" : "MIPS32r6";
|
|
|
|
assert(isFP64bit());
|
|
assert(isNaN2008());
|
|
if (hasDSP())
|
|
report_fatal_error(ISA + " is not compatible with the DSP ASE", false);
|
|
}
|
|
|
|
if (NoABICalls && TM.isPositionIndependent())
|
|
report_fatal_error("position-independent code requires '-mabicalls'");
|
|
|
|
if (isABI_N64() && !TM.isPositionIndependent() && !hasSym32())
|
|
NoABICalls = true;
|
|
|
|
// Set UseSmallSection.
|
|
UseSmallSection = GPOpt;
|
|
if (!NoABICalls && GPOpt) {
|
|
errs() << "warning: cannot use small-data accesses for '-mabicalls'"
|
|
<< "\n";
|
|
UseSmallSection = false;
|
|
}
|
|
}
|
|
|
|
bool MipsSubtarget::isPositionIndependent() const {
|
|
return TM.isPositionIndependent();
|
|
}
|
|
|
|
/// This overrides the PostRAScheduler bit in the SchedModel for any CPU.
|
|
bool MipsSubtarget::enablePostRAScheduler() const { return true; }
|
|
|
|
void MipsSubtarget::getCriticalPathRCs(RegClassVector &CriticalPathRCs) const {
|
|
CriticalPathRCs.clear();
|
|
CriticalPathRCs.push_back(isGP64bit() ?
|
|
&Mips::GPR64RegClass : &Mips::GPR32RegClass);
|
|
}
|
|
|
|
CodeGenOpt::Level MipsSubtarget::getOptLevelToEnablePostRAScheduler() const {
|
|
return CodeGenOpt::Aggressive;
|
|
}
|
|
|
|
MipsSubtarget &
|
|
MipsSubtarget::initializeSubtargetDependencies(StringRef CPU, StringRef FS,
|
|
const TargetMachine &TM) {
|
|
std::string CPUName = MIPS_MC::selectMipsCPU(TM.getTargetTriple(), CPU);
|
|
|
|
// Parse features string.
|
|
ParseSubtargetFeatures(CPUName, FS);
|
|
// Initialize scheduling itinerary for the specified CPU.
|
|
InstrItins = getInstrItineraryForCPU(CPUName);
|
|
|
|
if (InMips16Mode && !IsSoftFloat)
|
|
InMips16HardFloat = true;
|
|
|
|
return *this;
|
|
}
|
|
|
|
bool MipsSubtarget::useConstantIslands() {
|
|
DEBUG(dbgs() << "use constant islands " << Mips16ConstantIslands << "\n");
|
|
return Mips16ConstantIslands;
|
|
}
|
|
|
|
Reloc::Model MipsSubtarget::getRelocationModel() const {
|
|
return TM.getRelocationModel();
|
|
}
|
|
|
|
bool MipsSubtarget::isABI_N64() const { return getABI().IsN64(); }
|
|
bool MipsSubtarget::isABI_N32() const { return getABI().IsN32(); }
|
|
bool MipsSubtarget::isABI_O32() const { return getABI().IsO32(); }
|
|
const MipsABIInfo &MipsSubtarget::getABI() const { return TM.getABI(); }
|