mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-09-27 00:21:03 +00:00
Revert r221056 and others, "[mips] Move F128 argument handling into MipsCCState as we did for returns. NFC."
r221056 "[mips] Move F128 argument handling into MipsCCState as we did for returns. NFC." r221058 "[mips] Fix unused variable warning introduced in r221056" r221059 "[mips] Move all ByVal handling into CCState and tablegen-erated code. NFC." r221061 "Renamed CCState members that appear to misspell 'Processed' as 'Proceed'. NFC." It cuased an undefined behavior in LLVM :: CodeGen/Mips/return-vector.ll. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@221081 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -21,7 +21,6 @@ add_llvm_target(MipsCodeGen
|
|||||||
Mips16ISelDAGToDAG.cpp
|
Mips16ISelDAGToDAG.cpp
|
||||||
Mips16ISelLowering.cpp
|
Mips16ISelLowering.cpp
|
||||||
Mips16RegisterInfo.cpp
|
Mips16RegisterInfo.cpp
|
||||||
MipsABIInfo.cpp
|
|
||||||
MipsAnalyzeImmediate.cpp
|
MipsAnalyzeImmediate.cpp
|
||||||
MipsAsmPrinter.cpp
|
MipsAsmPrinter.cpp
|
||||||
MipsConstantIslandPass.cpp
|
MipsConstantIslandPass.cpp
|
||||||
|
@@ -244,8 +244,9 @@ Mips16TargetLowering::EmitInstrWithCustomInserter(MachineInstr *MI,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mips16TargetLowering::isEligibleForTailCallOptimization(
|
bool Mips16TargetLowering::
|
||||||
const CCState &CCInfo, unsigned NextStackOffset,
|
isEligibleForTailCallOptimization(const MipsCC &MipsCCInfo,
|
||||||
|
unsigned NextStackOffset,
|
||||||
const MipsFunctionInfo& FI) const {
|
const MipsFunctionInfo& FI) const {
|
||||||
// No tail call optimization for mips16.
|
// No tail call optimization for mips16.
|
||||||
return false;
|
return false;
|
||||||
|
@@ -31,8 +31,8 @@ namespace llvm {
|
|||||||
MachineBasicBlock *MBB) const override;
|
MachineBasicBlock *MBB) const override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool isEligibleForTailCallOptimization(
|
bool isEligibleForTailCallOptimization(const MipsCC &MipsCCInfo,
|
||||||
const CCState &CCInfo, unsigned NextStackOffset,
|
unsigned NextStackOffset,
|
||||||
const MipsFunctionInfo& FI) const override;
|
const MipsFunctionInfo& FI) const override;
|
||||||
|
|
||||||
void setMips16HardFloatLibCalls();
|
void setMips16HardFloatLibCalls();
|
||||||
|
@@ -1,29 +0,0 @@
|
|||||||
//===---- MipsABIInfo.cpp - Information about MIPS ABI's ------------------===//
|
|
||||||
//
|
|
||||||
// The LLVM Compiler Infrastructure
|
|
||||||
//
|
|
||||||
// This file is distributed under the University of Illinois Open Source
|
|
||||||
// License. See LICENSE.TXT for details.
|
|
||||||
//
|
|
||||||
//===----------------------------------------------------------------------===//
|
|
||||||
|
|
||||||
#include "MipsABIInfo.h"
|
|
||||||
#include "MipsRegisterInfo.h"
|
|
||||||
|
|
||||||
using namespace llvm;
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
static const MCPhysReg O32IntRegs[4] = {Mips::A0, Mips::A1, Mips::A2, Mips::A3};
|
|
||||||
|
|
||||||
static const MCPhysReg Mips64IntRegs[8] = {
|
|
||||||
Mips::A0_64, Mips::A1_64, Mips::A2_64, Mips::A3_64,
|
|
||||||
Mips::T0_64, Mips::T1_64, Mips::T2_64, Mips::T3_64};
|
|
||||||
}
|
|
||||||
|
|
||||||
const ArrayRef<MCPhysReg> MipsABIInfo::GetByValArgRegs() const {
|
|
||||||
if (IsO32())
|
|
||||||
return makeArrayRef(O32IntRegs);
|
|
||||||
if (IsN32() || IsN64())
|
|
||||||
return makeArrayRef(Mips64IntRegs);
|
|
||||||
llvm_unreachable("Unhandled ABI");
|
|
||||||
}
|
|
@@ -10,11 +10,7 @@
|
|||||||
#ifndef MIPSABIINFO_H
|
#ifndef MIPSABIINFO_H
|
||||||
#define MIPSABIINFO_H
|
#define MIPSABIINFO_H
|
||||||
|
|
||||||
#include "llvm/ADT/ArrayRef.h"
|
|
||||||
#include "llvm/MC/MCRegisterInfo.h"
|
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
class MipsABIInfo {
|
class MipsABIInfo {
|
||||||
public:
|
public:
|
||||||
enum class ABI { Unknown, O32, N32, N64, EABI };
|
enum class ABI { Unknown, O32, N32, N64, EABI };
|
||||||
@@ -38,8 +34,6 @@ public:
|
|||||||
bool IsEABI() const { return ThisABI == ABI::EABI; }
|
bool IsEABI() const { return ThisABI == ABI::EABI; }
|
||||||
ABI GetEnumValue() const { return ThisABI; }
|
ABI GetEnumValue() const { return ThisABI; }
|
||||||
|
|
||||||
const ArrayRef<MCPhysReg> GetByValArgRegs() const;
|
|
||||||
|
|
||||||
/// Ordering of ABI's
|
/// Ordering of ABI's
|
||||||
/// MipsGenSubtargetInfo.inc will use this to resolve conflicts when given
|
/// MipsGenSubtargetInfo.inc will use this to resolve conflicts when given
|
||||||
/// multiple ABI options.
|
/// multiple ABI options.
|
||||||
|
@@ -279,6 +279,13 @@ def CC_Mips_FastCC : CallingConv<[
|
|||||||
CCDelegateTo<CC_MipsN_FastCC>
|
CCDelegateTo<CC_MipsN_FastCC>
|
||||||
]>;
|
]>;
|
||||||
|
|
||||||
|
//==
|
||||||
|
|
||||||
|
def CC_Mips16RetHelper : CallingConv<[
|
||||||
|
// Integer arguments are passed in integer registers.
|
||||||
|
CCIfType<[i32], CCAssignToReg<[V0, V1, A0, A1]>>
|
||||||
|
]>;
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// Mips Calling Convention Dispatch
|
// Mips Calling Convention Dispatch
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
@@ -290,29 +297,7 @@ def RetCC_Mips : CallingConv<[
|
|||||||
CCDelegateTo<RetCC_MipsO32>
|
CCDelegateTo<RetCC_MipsO32>
|
||||||
]>;
|
]>;
|
||||||
|
|
||||||
def CC_Mips_ByVal : CallingConv<[
|
|
||||||
CCIfSubtarget<"isABI_O32()", CCIfByVal<CCPassByVal<4, 4>>>,
|
|
||||||
CCIfByVal<CCPassByVal<8, 8>>
|
|
||||||
]>;
|
|
||||||
|
|
||||||
def CC_Mips_FixedArg : CallingConv<[
|
def CC_Mips_FixedArg : CallingConv<[
|
||||||
CCIfByVal<CCDelegateTo<CC_Mips_ByVal>>,
|
|
||||||
|
|
||||||
// f128 needs to be handled similarly to f32 and f64 on hard-float. However,
|
|
||||||
// f128 is not legal and is lowered to i128 which is further lowered to a pair
|
|
||||||
// of i64's.
|
|
||||||
// This presents us with a problem for the calling convention since hard-float
|
|
||||||
// still needs to pass them in FPU registers. We therefore resort to a
|
|
||||||
// pre-analyze (see PreAnalyzeFormalArgsForF128()) step to pass information on
|
|
||||||
// whether the argument was originally an f128 into the tablegen-erated code.
|
|
||||||
//
|
|
||||||
// f128 should only occur for the N64 ABI where long double is 128-bit. On
|
|
||||||
// N32, long double is equivalent to double.
|
|
||||||
CCIfType<[i64],
|
|
||||||
CCIfSubtargetNot<"abiUsesSoftFloat()",
|
|
||||||
CCIf<"static_cast<MipsCCState *>(&State)->WasOriginalArgF128(ValNo)",
|
|
||||||
CCBitConvertToType<f64>>>>,
|
|
||||||
|
|
||||||
CCIfCC<"CallingConv::Fast", CCDelegateTo<CC_Mips_FastCC>>,
|
CCIfCC<"CallingConv::Fast", CCDelegateTo<CC_Mips_FastCC>>,
|
||||||
|
|
||||||
// FIXME: There wasn't an EABI case in the original code and it seems unlikely
|
// FIXME: There wasn't an EABI case in the original code and it seems unlikely
|
||||||
@@ -322,23 +307,12 @@ def CC_Mips_FixedArg : CallingConv<[
|
|||||||
]>;
|
]>;
|
||||||
|
|
||||||
def CC_Mips_VarArg : CallingConv<[
|
def CC_Mips_VarArg : CallingConv<[
|
||||||
CCIfByVal<CCDelegateTo<CC_Mips_ByVal>>,
|
|
||||||
|
|
||||||
// FIXME: There wasn't an EABI case in the original code and it seems unlikely
|
// FIXME: There wasn't an EABI case in the original code and it seems unlikely
|
||||||
// that it's the same as CC_MipsN_VarArg
|
// that it's the same as CC_MipsN_VarArg
|
||||||
CCIfSubtarget<"isABI_O32()", CCDelegateTo<CC_MipsO32_FP>>,
|
CCIfSubtarget<"isABI_O32()", CCDelegateTo<CC_MipsO32_FP>>,
|
||||||
CCDelegateTo<CC_MipsN_VarArg>
|
CCDelegateTo<CC_MipsN_VarArg>
|
||||||
]>;
|
]>;
|
||||||
|
|
||||||
//==
|
|
||||||
|
|
||||||
def CC_Mips16RetHelper : CallingConv<[
|
|
||||||
CCIfByVal<CCDelegateTo<CC_Mips_ByVal>>,
|
|
||||||
|
|
||||||
// Integer arguments are passed in integer registers.
|
|
||||||
CCIfType<[i32], CCAssignToReg<[V0, V1, A0, A1]>>
|
|
||||||
]>;
|
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// Callee-saved register lists.
|
// Callee-saved register lists.
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@@ -95,51 +95,10 @@ private:
|
|||||||
originalTypeIsF128(MF.getFunction()->getReturnType(), nullptr));
|
originalTypeIsF128(MF.getFunction()->getReturnType(), nullptr));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Identify lowered values that originated from f128 arguments and record
|
|
||||||
/// this.
|
|
||||||
void PreAnalyzeCallOperandsForF128(
|
|
||||||
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
||||||
std::vector<TargetLowering::ArgListEntry> &FuncArgs, SDNode *CallNode) {
|
|
||||||
for (unsigned i = 0; i < Outs.size(); ++i)
|
|
||||||
OriginalArgWasF128.push_back(
|
|
||||||
originalTypeIsF128(FuncArgs[Outs[i].OrigArgIndex].Ty, CallNode));
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Identify lowered values that originated from f128 arguments and record
|
|
||||||
/// this.
|
|
||||||
void
|
|
||||||
PreAnalyzeFormalArgumentsForF128(const SmallVectorImpl<ISD::InputArg> &Ins) {
|
|
||||||
const MachineFunction &MF = getMachineFunction();
|
|
||||||
for (unsigned i = 0; i < Ins.size(); ++i) {
|
|
||||||
Function::const_arg_iterator FuncArg = MF.getFunction()->arg_begin();
|
|
||||||
std::advance(FuncArg, Ins[i].OrigArgIndex);
|
|
||||||
|
|
||||||
OriginalArgWasF128.push_back(
|
|
||||||
originalTypeIsF128(FuncArg->getType(), nullptr));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Records whether the value has been lowered from an f128.
|
/// Records whether the value has been lowered from an f128.
|
||||||
SmallVector<bool, 4> OriginalArgWasF128;
|
SmallVector<bool, 4> OriginalArgWasF128;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// FIXME: Remove this from a public inteface ASAP. It's a temporary trap door
|
|
||||||
// to allow analyzeCallOperands to be removed incrementally.
|
|
||||||
void PreAnalyzeCallOperandsForF128_(
|
|
||||||
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
||||||
std::vector<TargetLowering::ArgListEntry> &FuncArgs, SDNode *CallNode) {
|
|
||||||
PreAnalyzeCallOperandsForF128(Outs, FuncArgs, CallNode);
|
|
||||||
}
|
|
||||||
// FIXME: Remove this from a public inteface ASAP. It's a temporary trap door
|
|
||||||
// to allow analyzeFormalArguments to be removed incrementally.
|
|
||||||
void
|
|
||||||
PreAnalyzeFormalArgumentsForF128_(const SmallVectorImpl<ISD::InputArg> &Ins) {
|
|
||||||
PreAnalyzeFormalArgumentsForF128(Ins);
|
|
||||||
}
|
|
||||||
// FIXME: Remove this from a public inteface ASAP. It's a temporary trap door
|
|
||||||
// to clean up after the above functions.
|
|
||||||
void ClearOriginalArgWasF128() { OriginalArgWasF128.clear(); }
|
|
||||||
|
|
||||||
MipsCCState(CallingConv::ID CC, bool isVarArg, MachineFunction &MF,
|
MipsCCState(CallingConv::ID CC, bool isVarArg, MachineFunction &MF,
|
||||||
SmallVectorImpl<CCValAssign> &locs, LLVMContext &C)
|
SmallVectorImpl<CCValAssign> &locs, LLVMContext &C)
|
||||||
: CCState(CC, isVarArg, MF, locs, C) {}
|
: CCState(CC, isVarArg, MF, locs, C) {}
|
||||||
@@ -2587,22 +2546,21 @@ MipsTargetLowering::LowerCall(TargetLowering::CallLoweringInfo &CLI,
|
|||||||
|
|
||||||
// Analyze operands of the call, assigning locations to each operand.
|
// Analyze operands of the call, assigning locations to each operand.
|
||||||
SmallVector<CCValAssign, 16> ArgLocs;
|
SmallVector<CCValAssign, 16> ArgLocs;
|
||||||
MipsCCState CCInfo(CallConv, IsVarArg, DAG.getMachineFunction(), ArgLocs,
|
CCState CCInfo(CallConv, IsVarArg, DAG.getMachineFunction(), ArgLocs,
|
||||||
*DAG.getContext());
|
*DAG.getContext());
|
||||||
MipsCC MipsCCInfo(CallConv, Subtarget, CCInfo);
|
MipsCC MipsCCInfo(CallConv, Subtarget, CCInfo);
|
||||||
|
|
||||||
CCInfo.PreAnalyzeCallOperandsForF128_(Outs, CLI.getArgs(), Callee.getNode());
|
|
||||||
MipsCCInfo.analyzeCallOperands(Outs, IsVarArg, Subtarget.abiUsesSoftFloat(),
|
MipsCCInfo.analyzeCallOperands(Outs, IsVarArg, Subtarget.abiUsesSoftFloat(),
|
||||||
Callee.getNode(), CLI.getArgs(), CCInfo);
|
Callee.getNode(), CLI.getArgs(), CCInfo);
|
||||||
CCInfo.ClearOriginalArgWasF128();
|
|
||||||
|
|
||||||
// Get a count of how many bytes are to be pushed on the stack.
|
// Get a count of how many bytes are to be pushed on the stack.
|
||||||
unsigned NextStackOffset = CCInfo.getNextStackOffset();
|
unsigned NextStackOffset = CCInfo.getNextStackOffset();
|
||||||
|
|
||||||
// Check if it's really possible to do a tail call.
|
// Check if it's really possible to do a tail call.
|
||||||
if (IsTailCall)
|
if (IsTailCall)
|
||||||
IsTailCall = isEligibleForTailCallOptimization(
|
IsTailCall =
|
||||||
CCInfo, NextStackOffset, *MF.getInfo<MipsFunctionInfo>());
|
isEligibleForTailCallOptimization(MipsCCInfo, NextStackOffset,
|
||||||
|
*MF.getInfo<MipsFunctionInfo>());
|
||||||
|
|
||||||
if (!IsTailCall && CLI.CS && CLI.CS->isMustTailCall())
|
if (!IsTailCall && CLI.CS && CLI.CS->isMustTailCall())
|
||||||
report_fatal_error("failed to perform tail call elimination on a call "
|
report_fatal_error("failed to perform tail call elimination on a call "
|
||||||
@@ -2628,8 +2586,7 @@ MipsTargetLowering::LowerCall(TargetLowering::CallLoweringInfo &CLI,
|
|||||||
// With EABI is it possible to have 16 args on registers.
|
// With EABI is it possible to have 16 args on registers.
|
||||||
std::deque< std::pair<unsigned, SDValue> > RegsToPass;
|
std::deque< std::pair<unsigned, SDValue> > RegsToPass;
|
||||||
SmallVector<SDValue, 8> MemOpChains;
|
SmallVector<SDValue, 8> MemOpChains;
|
||||||
|
MipsCC::byval_iterator ByValArg = MipsCCInfo.byval_begin();
|
||||||
CCInfo.rewindByValRegsInfo();
|
|
||||||
|
|
||||||
// Walk the register/memloc assignments, inserting copies/loads.
|
// Walk the register/memloc assignments, inserting copies/loads.
|
||||||
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
||||||
@@ -2640,19 +2597,14 @@ MipsTargetLowering::LowerCall(TargetLowering::CallLoweringInfo &CLI,
|
|||||||
|
|
||||||
// ByVal Arg.
|
// ByVal Arg.
|
||||||
if (Flags.isByVal()) {
|
if (Flags.isByVal()) {
|
||||||
unsigned FirstByValReg, LastByValReg;
|
|
||||||
unsigned ByValIdx = CCInfo.getInRegsParamsProcessed();
|
|
||||||
CCInfo.getInRegsParamInfo(ByValIdx, FirstByValReg, LastByValReg);
|
|
||||||
|
|
||||||
assert(Flags.getByValSize() &&
|
assert(Flags.getByValSize() &&
|
||||||
"ByVal args of size 0 should have been ignored by front-end.");
|
"ByVal args of size 0 should have been ignored by front-end.");
|
||||||
assert(ByValIdx < CCInfo.getInRegsParamsCount());
|
assert(ByValArg != MipsCCInfo.byval_end());
|
||||||
assert(!IsTailCall &&
|
assert(!IsTailCall &&
|
||||||
"Do not tail-call optimize if there is a byval argument.");
|
"Do not tail-call optimize if there is a byval argument.");
|
||||||
passByValArg(Chain, DL, RegsToPass, MemOpChains, StackPtr, MFI, DAG, Arg,
|
passByValArg(Chain, DL, RegsToPass, MemOpChains, StackPtr, MFI, DAG, Arg,
|
||||||
MipsCCInfo, FirstByValReg, LastByValReg, Flags,
|
MipsCCInfo, *ByValArg, Flags, Subtarget.isLittle(), VA);
|
||||||
Subtarget.isLittle(), VA);
|
++ByValArg;
|
||||||
CCInfo.nextInRegsParam();
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2680,9 +2632,6 @@ MipsTargetLowering::LowerCall(TargetLowering::CallLoweringInfo &CLI,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case CCValAssign::BCvt:
|
|
||||||
Arg = DAG.getNode(ISD::BITCAST, DL, LocVT, Arg);
|
|
||||||
break;
|
|
||||||
case CCValAssign::SExt:
|
case CCValAssign::SExt:
|
||||||
Arg = DAG.getNode(ISD::SIGN_EXTEND, DL, LocVT, Arg);
|
Arg = DAG.getNode(ISD::SIGN_EXTEND, DL, LocVT, Arg);
|
||||||
break;
|
break;
|
||||||
@@ -2880,21 +2829,19 @@ MipsTargetLowering::LowerFormalArguments(SDValue Chain,
|
|||||||
|
|
||||||
// Assign locations to all of the incoming arguments.
|
// Assign locations to all of the incoming arguments.
|
||||||
SmallVector<CCValAssign, 16> ArgLocs;
|
SmallVector<CCValAssign, 16> ArgLocs;
|
||||||
MipsCCState CCInfo(CallConv, IsVarArg, DAG.getMachineFunction(), ArgLocs,
|
CCState CCInfo(CallConv, IsVarArg, DAG.getMachineFunction(), ArgLocs,
|
||||||
*DAG.getContext());
|
*DAG.getContext());
|
||||||
MipsCC MipsCCInfo(CallConv, Subtarget, CCInfo);
|
MipsCC MipsCCInfo(CallConv, Subtarget, CCInfo);
|
||||||
Function::const_arg_iterator FuncArg =
|
Function::const_arg_iterator FuncArg =
|
||||||
DAG.getMachineFunction().getFunction()->arg_begin();
|
DAG.getMachineFunction().getFunction()->arg_begin();
|
||||||
bool UseSoftFloat = Subtarget.abiUsesSoftFloat();
|
bool UseSoftFloat = Subtarget.abiUsesSoftFloat();
|
||||||
|
|
||||||
CCInfo.PreAnalyzeFormalArgumentsForF128_(Ins);
|
MipsCCInfo.analyzeFormalArguments(Ins, UseSoftFloat, FuncArg, CCInfo);
|
||||||
MipsCCInfo.analyzeFormalArguments(Ins, UseSoftFloat, CCInfo);
|
|
||||||
CCInfo.ClearOriginalArgWasF128();
|
|
||||||
MipsFI->setFormalArgInfo(CCInfo.getNextStackOffset(),
|
MipsFI->setFormalArgInfo(CCInfo.getNextStackOffset(),
|
||||||
CCInfo.getInRegsParamsCount() > 0);
|
MipsCCInfo.hasByValArg());
|
||||||
|
|
||||||
unsigned CurArgIdx = 0;
|
unsigned CurArgIdx = 0;
|
||||||
CCInfo.rewindByValRegsInfo();
|
MipsCC::byval_iterator ByValArg = MipsCCInfo.byval_begin();
|
||||||
|
|
||||||
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
||||||
CCValAssign &VA = ArgLocs[i];
|
CCValAssign &VA = ArgLocs[i];
|
||||||
@@ -2905,16 +2852,12 @@ MipsTargetLowering::LowerFormalArguments(SDValue Chain,
|
|||||||
bool IsRegLoc = VA.isRegLoc();
|
bool IsRegLoc = VA.isRegLoc();
|
||||||
|
|
||||||
if (Flags.isByVal()) {
|
if (Flags.isByVal()) {
|
||||||
unsigned FirstByValReg, LastByValReg;
|
|
||||||
unsigned ByValIdx = CCInfo.getInRegsParamsProcessed();
|
|
||||||
CCInfo.getInRegsParamInfo(ByValIdx, FirstByValReg, LastByValReg);
|
|
||||||
|
|
||||||
assert(Flags.getByValSize() &&
|
assert(Flags.getByValSize() &&
|
||||||
"ByVal args of size 0 should have been ignored by front-end.");
|
"ByVal args of size 0 should have been ignored by front-end.");
|
||||||
assert(ByValIdx < CCInfo.getInRegsParamsCount());
|
assert(ByValArg != MipsCCInfo.byval_end());
|
||||||
copyByValRegs(Chain, DL, OutChains, DAG, Flags, InVals, &*FuncArg,
|
copyByValRegs(Chain, DL, OutChains, DAG, Flags, InVals, &*FuncArg,
|
||||||
MipsCCInfo, FirstByValReg, LastByValReg, VA);
|
MipsCCInfo, *ByValArg, VA);
|
||||||
CCInfo.nextInRegsParam();
|
++ByValArg;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2932,24 +2875,16 @@ MipsTargetLowering::LowerFormalArguments(SDValue Chain,
|
|||||||
// If this is an 8 or 16-bit value, it has been passed promoted
|
// If this is an 8 or 16-bit value, it has been passed promoted
|
||||||
// to 32 bits. Insert an assert[sz]ext to capture this, then
|
// to 32 bits. Insert an assert[sz]ext to capture this, then
|
||||||
// truncate to the right size.
|
// truncate to the right size.
|
||||||
switch (VA.getLocInfo()) {
|
if (VA.getLocInfo() != CCValAssign::Full) {
|
||||||
default:
|
unsigned Opcode = 0;
|
||||||
llvm_unreachable("Unknown loc info!");
|
if (VA.getLocInfo() == CCValAssign::SExt)
|
||||||
case CCValAssign::Full:
|
Opcode = ISD::AssertSext;
|
||||||
break;
|
else if (VA.getLocInfo() == CCValAssign::ZExt)
|
||||||
case CCValAssign::SExt:
|
Opcode = ISD::AssertZext;
|
||||||
ArgValue = DAG.getNode(ISD::AssertSext, DL, RegVT, ArgValue,
|
if (Opcode)
|
||||||
|
ArgValue = DAG.getNode(Opcode, DL, RegVT, ArgValue,
|
||||||
DAG.getValueType(ValVT));
|
DAG.getValueType(ValVT));
|
||||||
ArgValue = DAG.getNode(ISD::TRUNCATE, DL, ValVT, ArgValue);
|
ArgValue = DAG.getNode(ISD::TRUNCATE, DL, ValVT, ArgValue);
|
||||||
break;
|
|
||||||
case CCValAssign::ZExt:
|
|
||||||
ArgValue = DAG.getNode(ISD::AssertZext, DL, RegVT, ArgValue,
|
|
||||||
DAG.getValueType(ValVT));
|
|
||||||
ArgValue = DAG.getNode(ISD::TRUNCATE, DL, ValVT, ArgValue);
|
|
||||||
break;
|
|
||||||
case CCValAssign::BCvt:
|
|
||||||
ArgValue = DAG.getNode(ISD::BITCAST, DL, ValVT, ArgValue);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle floating point arguments passed in integer registers and
|
// Handle floating point arguments passed in integer registers and
|
||||||
@@ -3621,10 +3556,18 @@ void MipsTargetLowering::MipsCC::analyzeCallOperands(
|
|||||||
ISD::ArgFlagsTy ArgFlags = Args[I].Flags;
|
ISD::ArgFlagsTy ArgFlags = Args[I].Flags;
|
||||||
bool R;
|
bool R;
|
||||||
|
|
||||||
|
if (ArgFlags.isByVal()) {
|
||||||
|
handleByValArg(I, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, State);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (IsVarArg && !Args[I].IsFixed)
|
if (IsVarArg && !Args[I].IsFixed)
|
||||||
R = CC_Mips_VarArg(I, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, State);
|
R = CC_Mips_VarArg(I, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, State);
|
||||||
else
|
else {
|
||||||
R = FixedFn(I, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, State);
|
MVT RegVT = getRegVT(ArgVT, FuncArgs[Args[I].OrigArgIndex].Ty, CallNode,
|
||||||
|
IsSoftFloat);
|
||||||
|
R = FixedFn(I, ArgVT, RegVT, CCValAssign::Full, ArgFlags, State);
|
||||||
|
}
|
||||||
|
|
||||||
if (R) {
|
if (R) {
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
@@ -3638,14 +3581,24 @@ void MipsTargetLowering::MipsCC::analyzeCallOperands(
|
|||||||
|
|
||||||
void MipsTargetLowering::MipsCC::analyzeFormalArguments(
|
void MipsTargetLowering::MipsCC::analyzeFormalArguments(
|
||||||
const SmallVectorImpl<ISD::InputArg> &Args, bool IsSoftFloat,
|
const SmallVectorImpl<ISD::InputArg> &Args, bool IsSoftFloat,
|
||||||
CCState &State) {
|
Function::const_arg_iterator FuncArg, CCState &State) {
|
||||||
unsigned NumArgs = Args.size();
|
unsigned NumArgs = Args.size();
|
||||||
|
unsigned CurArgIdx = 0;
|
||||||
|
|
||||||
for (unsigned I = 0; I != NumArgs; ++I) {
|
for (unsigned I = 0; I != NumArgs; ++I) {
|
||||||
MVT ArgVT = Args[I].VT;
|
MVT ArgVT = Args[I].VT;
|
||||||
ISD::ArgFlagsTy ArgFlags = Args[I].Flags;
|
ISD::ArgFlagsTy ArgFlags = Args[I].Flags;
|
||||||
|
std::advance(FuncArg, Args[I].OrigArgIndex - CurArgIdx);
|
||||||
|
CurArgIdx = Args[I].OrigArgIndex;
|
||||||
|
|
||||||
if (!CC_Mips_FixedArg(I, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, State))
|
if (ArgFlags.isByVal()) {
|
||||||
|
handleByValArg(I, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, State);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
MVT RegVT = getRegVT(ArgVT, FuncArg->getType(), nullptr, IsSoftFloat);
|
||||||
|
|
||||||
|
if (!CC_Mips_FixedArg(I, ArgVT, RegVT, CCValAssign::Full, ArgFlags, State))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
@@ -3656,6 +3609,30 @@ void MipsTargetLowering::MipsCC::analyzeFormalArguments(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MipsTargetLowering::MipsCC::handleByValArg(unsigned ValNo, MVT ValVT,
|
||||||
|
MVT LocVT,
|
||||||
|
CCValAssign::LocInfo LocInfo,
|
||||||
|
ISD::ArgFlagsTy ArgFlags,
|
||||||
|
CCState &State) {
|
||||||
|
assert(ArgFlags.getByValSize() && "Byval argument's size shouldn't be 0.");
|
||||||
|
|
||||||
|
struct ByValArgInfo ByVal;
|
||||||
|
unsigned RegSizeInBytes = Subtarget.getGPRSizeInBytes();
|
||||||
|
unsigned ByValSize =
|
||||||
|
RoundUpToAlignment(ArgFlags.getByValSize(), RegSizeInBytes);
|
||||||
|
unsigned Align = std::min(std::max(ArgFlags.getByValAlign(), RegSizeInBytes),
|
||||||
|
RegSizeInBytes * 2);
|
||||||
|
|
||||||
|
if (useRegsForByval())
|
||||||
|
allocateRegs(ByVal, ByValSize, Align, State);
|
||||||
|
|
||||||
|
// Allocate space on caller's stack.
|
||||||
|
unsigned Offset =
|
||||||
|
State.AllocateStack(ByValSize - RegSizeInBytes * ByVal.NumRegs, Align);
|
||||||
|
State.addLoc(CCValAssign::getMem(ValNo, ValVT, Offset, LocVT, LocInfo));
|
||||||
|
ByValArgs.push_back(ByVal);
|
||||||
|
}
|
||||||
|
|
||||||
unsigned MipsTargetLowering::MipsCC::reservedArgArea() const {
|
unsigned MipsTargetLowering::MipsCC::reservedArgArea() const {
|
||||||
return (Subtarget.isABI_O32() && (CallConv != CallingConv::Fast)) ? 16 : 0;
|
return (Subtarget.isABI_O32() && (CallConv != CallingConv::Fast)) ? 16 : 0;
|
||||||
}
|
}
|
||||||
@@ -3666,6 +3643,35 @@ const ArrayRef<MCPhysReg> MipsTargetLowering::MipsCC::intArgRegs() const {
|
|||||||
return makeArrayRef(Mips64IntRegs);
|
return makeArrayRef(Mips64IntRegs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const MCPhysReg *MipsTargetLowering::MipsCC::shadowRegs() const {
|
||||||
|
return Subtarget.isABI_O32() ? O32IntRegs : Mips64DPRegs;
|
||||||
|
}
|
||||||
|
|
||||||
|
void MipsTargetLowering::MipsCC::allocateRegs(ByValArgInfo &ByVal,
|
||||||
|
unsigned ByValSize,
|
||||||
|
unsigned Align, CCState &State) {
|
||||||
|
unsigned RegSizeInBytes = Subtarget.getGPRSizeInBytes();
|
||||||
|
const ArrayRef<MCPhysReg> IntArgRegs = intArgRegs();
|
||||||
|
const MCPhysReg *ShadowRegs = shadowRegs();
|
||||||
|
assert(!(ByValSize % RegSizeInBytes) && !(Align % RegSizeInBytes) &&
|
||||||
|
"Byval argument's size and alignment should be a multiple of"
|
||||||
|
"RegSizeInBytes.");
|
||||||
|
|
||||||
|
ByVal.FirstIdx =
|
||||||
|
State.getFirstUnallocated(IntArgRegs.data(), IntArgRegs.size());
|
||||||
|
|
||||||
|
// If Align > RegSizeInBytes, the first arg register must be even.
|
||||||
|
if ((Align > RegSizeInBytes) && (ByVal.FirstIdx % 2)) {
|
||||||
|
State.AllocateReg(IntArgRegs[ByVal.FirstIdx], ShadowRegs[ByVal.FirstIdx]);
|
||||||
|
++ByVal.FirstIdx;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mark the registers allocated.
|
||||||
|
for (unsigned I = ByVal.FirstIdx; ByValSize && (I < IntArgRegs.size());
|
||||||
|
ByValSize -= RegSizeInBytes, ++I, ++ByVal.NumRegs)
|
||||||
|
State.AllocateReg(IntArgRegs[I], ShadowRegs[I]);
|
||||||
|
}
|
||||||
|
|
||||||
MVT MipsTargetLowering::MipsCC::getRegVT(MVT VT, const Type *OrigTy,
|
MVT MipsTargetLowering::MipsCC::getRegVT(MVT VT, const Type *OrigTy,
|
||||||
const SDNode *CallNode,
|
const SDNode *CallNode,
|
||||||
bool IsSoftFloat) const {
|
bool IsSoftFloat) const {
|
||||||
@@ -3684,20 +3690,19 @@ MVT MipsTargetLowering::MipsCC::getRegVT(MVT VT, const Type *OrigTy,
|
|||||||
void MipsTargetLowering::copyByValRegs(
|
void MipsTargetLowering::copyByValRegs(
|
||||||
SDValue Chain, SDLoc DL, std::vector<SDValue> &OutChains, SelectionDAG &DAG,
|
SDValue Chain, SDLoc DL, std::vector<SDValue> &OutChains, SelectionDAG &DAG,
|
||||||
const ISD::ArgFlagsTy &Flags, SmallVectorImpl<SDValue> &InVals,
|
const ISD::ArgFlagsTy &Flags, SmallVectorImpl<SDValue> &InVals,
|
||||||
const Argument *FuncArg, const MipsCC &CC, unsigned FirstReg,
|
const Argument *FuncArg, const MipsCC &CC, const ByValArgInfo &ByVal,
|
||||||
unsigned LastReg, const CCValAssign &VA) const {
|
const CCValAssign &VA) const {
|
||||||
MachineFunction &MF = DAG.getMachineFunction();
|
MachineFunction &MF = DAG.getMachineFunction();
|
||||||
MachineFrameInfo *MFI = MF.getFrameInfo();
|
MachineFrameInfo *MFI = MF.getFrameInfo();
|
||||||
unsigned GPRSizeInBytes = Subtarget.getGPRSizeInBytes();
|
unsigned GPRSizeInBytes = Subtarget.getGPRSizeInBytes();
|
||||||
unsigned NumRegs = LastReg - FirstReg;
|
unsigned RegAreaSize = ByVal.NumRegs * GPRSizeInBytes;
|
||||||
unsigned RegAreaSize = NumRegs * GPRSizeInBytes;
|
|
||||||
unsigned FrameObjSize = std::max(Flags.getByValSize(), RegAreaSize);
|
unsigned FrameObjSize = std::max(Flags.getByValSize(), RegAreaSize);
|
||||||
int FrameObjOffset;
|
int FrameObjOffset;
|
||||||
|
|
||||||
if (RegAreaSize)
|
if (RegAreaSize)
|
||||||
FrameObjOffset =
|
FrameObjOffset =
|
||||||
(int)CC.reservedArgArea() -
|
(int)CC.reservedArgArea() -
|
||||||
(int)((CC.intArgRegs().size() - FirstReg) * GPRSizeInBytes);
|
(int)((CC.intArgRegs().size() - ByVal.FirstIdx) * GPRSizeInBytes);
|
||||||
else
|
else
|
||||||
FrameObjOffset = VA.getLocMemOffset();
|
FrameObjOffset = VA.getLocMemOffset();
|
||||||
|
|
||||||
@@ -3707,15 +3712,15 @@ void MipsTargetLowering::copyByValRegs(
|
|||||||
SDValue FIN = DAG.getFrameIndex(FI, PtrTy);
|
SDValue FIN = DAG.getFrameIndex(FI, PtrTy);
|
||||||
InVals.push_back(FIN);
|
InVals.push_back(FIN);
|
||||||
|
|
||||||
if (!NumRegs)
|
if (!ByVal.NumRegs)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Copy arg registers.
|
// Copy arg registers.
|
||||||
MVT RegTy = MVT::getIntegerVT(GPRSizeInBytes * 8);
|
MVT RegTy = MVT::getIntegerVT(GPRSizeInBytes * 8);
|
||||||
const TargetRegisterClass *RC = getRegClassFor(RegTy);
|
const TargetRegisterClass *RC = getRegClassFor(RegTy);
|
||||||
|
|
||||||
for (unsigned I = 0; I < NumRegs; ++I) {
|
for (unsigned I = 0; I < ByVal.NumRegs; ++I) {
|
||||||
unsigned ArgReg = CC.intArgRegs()[FirstReg + I];
|
unsigned ArgReg = CC.intArgRegs()[ByVal.FirstIdx + I];
|
||||||
unsigned VReg = addLiveIn(MF, ArgReg, RC);
|
unsigned VReg = addLiveIn(MF, ArgReg, RC);
|
||||||
unsigned Offset = I * GPRSizeInBytes;
|
unsigned Offset = I * GPRSizeInBytes;
|
||||||
SDValue StorePtr = DAG.getNode(ISD::ADD, DL, PtrTy, FIN,
|
SDValue StorePtr = DAG.getNode(ISD::ADD, DL, PtrTy, FIN,
|
||||||
@@ -3733,29 +3738,29 @@ void MipsTargetLowering::passByValArg(
|
|||||||
std::deque<std::pair<unsigned, SDValue>> &RegsToPass,
|
std::deque<std::pair<unsigned, SDValue>> &RegsToPass,
|
||||||
SmallVectorImpl<SDValue> &MemOpChains, SDValue StackPtr,
|
SmallVectorImpl<SDValue> &MemOpChains, SDValue StackPtr,
|
||||||
MachineFrameInfo *MFI, SelectionDAG &DAG, SDValue Arg, const MipsCC &CC,
|
MachineFrameInfo *MFI, SelectionDAG &DAG, SDValue Arg, const MipsCC &CC,
|
||||||
unsigned FirstReg, unsigned LastReg, const ISD::ArgFlagsTy &Flags,
|
const ByValArgInfo &ByVal, const ISD::ArgFlagsTy &Flags, bool isLittle,
|
||||||
bool isLittle, const CCValAssign &VA) const {
|
const CCValAssign &VA) const {
|
||||||
unsigned ByValSizeInBytes = Flags.getByValSize();
|
unsigned ByValSizeInBytes = Flags.getByValSize();
|
||||||
unsigned OffsetInBytes = 0; // From beginning of struct
|
unsigned OffsetInBytes = 0; // From beginning of struct
|
||||||
unsigned RegSizeInBytes = Subtarget.getGPRSizeInBytes();
|
unsigned RegSizeInBytes = Subtarget.getGPRSizeInBytes();
|
||||||
unsigned Alignment = std::min(Flags.getByValAlign(), RegSizeInBytes);
|
unsigned Alignment = std::min(Flags.getByValAlign(), RegSizeInBytes);
|
||||||
EVT PtrTy = getPointerTy(), RegTy = MVT::getIntegerVT(RegSizeInBytes * 8);
|
EVT PtrTy = getPointerTy(), RegTy = MVT::getIntegerVT(RegSizeInBytes * 8);
|
||||||
unsigned NumRegs = LastReg - FirstReg;
|
|
||||||
|
|
||||||
if (NumRegs) {
|
if (ByVal.NumRegs) {
|
||||||
const ArrayRef<MCPhysReg> ArgRegs = CC.intArgRegs();
|
const ArrayRef<MCPhysReg> ArgRegs = CC.intArgRegs();
|
||||||
bool LeftoverBytes = (NumRegs * RegSizeInBytes > ByValSizeInBytes);
|
bool LeftoverBytes = (ByVal.NumRegs * RegSizeInBytes > ByValSizeInBytes);
|
||||||
unsigned I = 0;
|
unsigned I = 0;
|
||||||
|
|
||||||
// Copy words to registers.
|
// Copy words to registers.
|
||||||
for (; I < NumRegs - LeftoverBytes; ++I, OffsetInBytes += RegSizeInBytes) {
|
for (; I < ByVal.NumRegs - LeftoverBytes;
|
||||||
|
++I, OffsetInBytes += RegSizeInBytes) {
|
||||||
SDValue LoadPtr = DAG.getNode(ISD::ADD, DL, PtrTy, Arg,
|
SDValue LoadPtr = DAG.getNode(ISD::ADD, DL, PtrTy, Arg,
|
||||||
DAG.getConstant(OffsetInBytes, PtrTy));
|
DAG.getConstant(OffsetInBytes, PtrTy));
|
||||||
SDValue LoadVal = DAG.getLoad(RegTy, DL, Chain, LoadPtr,
|
SDValue LoadVal = DAG.getLoad(RegTy, DL, Chain, LoadPtr,
|
||||||
MachinePointerInfo(), false, false, false,
|
MachinePointerInfo(), false, false, false,
|
||||||
Alignment);
|
Alignment);
|
||||||
MemOpChains.push_back(LoadVal.getValue(1));
|
MemOpChains.push_back(LoadVal.getValue(1));
|
||||||
unsigned ArgReg = ArgRegs[FirstReg + I];
|
unsigned ArgReg = ArgRegs[ByVal.FirstIdx + I];
|
||||||
RegsToPass.push_back(std::make_pair(ArgReg, LoadVal));
|
RegsToPass.push_back(std::make_pair(ArgReg, LoadVal));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -3765,6 +3770,9 @@ void MipsTargetLowering::passByValArg(
|
|||||||
|
|
||||||
// Copy the remainder of the byval argument with sub-word loads and shifts.
|
// Copy the remainder of the byval argument with sub-word loads and shifts.
|
||||||
if (LeftoverBytes) {
|
if (LeftoverBytes) {
|
||||||
|
assert((ByValSizeInBytes > OffsetInBytes) &&
|
||||||
|
(ByValSizeInBytes < OffsetInBytes + RegSizeInBytes) &&
|
||||||
|
"Size of the remainder should be smaller than RegSizeInBytes.");
|
||||||
SDValue Val;
|
SDValue Val;
|
||||||
|
|
||||||
for (unsigned LoadSizeInBytes = RegSizeInBytes / 2, TotalBytesLoaded = 0;
|
for (unsigned LoadSizeInBytes = RegSizeInBytes / 2, TotalBytesLoaded = 0;
|
||||||
@@ -3804,7 +3812,7 @@ void MipsTargetLowering::passByValArg(
|
|||||||
Alignment = std::min(Alignment, LoadSizeInBytes);
|
Alignment = std::min(Alignment, LoadSizeInBytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned ArgReg = ArgRegs[FirstReg + I];
|
unsigned ArgReg = ArgRegs[ByVal.FirstIdx + I];
|
||||||
RegsToPass.push_back(std::make_pair(ArgReg, Val));
|
RegsToPass.push_back(std::make_pair(ArgReg, Val));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -3867,49 +3875,3 @@ void MipsTargetLowering::writeVarArgRegs(std::vector<SDValue> &OutChains,
|
|||||||
OutChains.push_back(Store);
|
OutChains.push_back(Store);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MipsTargetLowering::HandleByVal(CCState *State, unsigned &Size,
|
|
||||||
unsigned Align) const {
|
|
||||||
MachineFunction &MF = State->getMachineFunction();
|
|
||||||
const TargetFrameLowering *TFL = MF.getSubtarget().getFrameLowering();
|
|
||||||
|
|
||||||
assert(Size && "Byval argument's size shouldn't be 0.");
|
|
||||||
|
|
||||||
Align = std::min(Align, TFL->getStackAlignment());
|
|
||||||
|
|
||||||
unsigned FirstReg = 0;
|
|
||||||
unsigned NumRegs = 0;
|
|
||||||
|
|
||||||
if (State->getCallingConv() != CallingConv::Fast) {
|
|
||||||
unsigned RegSizeInBytes = Subtarget.getGPRSizeInBytes();
|
|
||||||
const ArrayRef<MCPhysReg> IntArgRegs = Subtarget.getABI().GetByValArgRegs();
|
|
||||||
// FIXME: The O32 case actually describes no shadow registers.
|
|
||||||
const MCPhysReg *ShadowRegs =
|
|
||||||
Subtarget.isABI_O32() ? IntArgRegs.data() : Mips64DPRegs;
|
|
||||||
|
|
||||||
// We used to check the size as well but we can't do that anymore since
|
|
||||||
// CCState::HandleByVal() rounds up the size after calling this function.
|
|
||||||
assert(!(Align % RegSizeInBytes) &&
|
|
||||||
"Byval argument's alignment should be a multiple of"
|
|
||||||
"RegSizeInBytes.");
|
|
||||||
|
|
||||||
FirstReg = State->getFirstUnallocated(IntArgRegs.data(), IntArgRegs.size());
|
|
||||||
|
|
||||||
// If Align > RegSizeInBytes, the first arg register must be even.
|
|
||||||
// FIXME: This condition happens to do the right thing but it's not the
|
|
||||||
// right way to test it. We want to check that the stack frame offset
|
|
||||||
// of the register is aligned.
|
|
||||||
if ((Align > RegSizeInBytes) && (FirstReg % 2)) {
|
|
||||||
State->AllocateReg(IntArgRegs[FirstReg], ShadowRegs[FirstReg]);
|
|
||||||
++FirstReg;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Mark the registers allocated.
|
|
||||||
Size = RoundUpToAlignment(Size, RegSizeInBytes);
|
|
||||||
for (unsigned I = FirstReg; Size > 0 && (I < IntArgRegs.size());
|
|
||||||
Size -= RegSizeInBytes, ++I, ++NumRegs)
|
|
||||||
State->AllocateReg(IntArgRegs[I], ShadowRegs[I]);
|
|
||||||
}
|
|
||||||
|
|
||||||
State->addInRegsParamInfo(FirstReg, FirstReg + NumRegs);
|
|
||||||
}
|
|
||||||
|
@@ -259,8 +259,6 @@ namespace llvm {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
void HandleByVal(CCState *, unsigned &, unsigned) const override;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
SDValue getGlobalReg(SelectionDAG &DAG, EVT Ty) const;
|
SDValue getGlobalReg(SelectionDAG &DAG, EVT Ty) const;
|
||||||
|
|
||||||
@@ -341,6 +339,14 @@ namespace llvm {
|
|||||||
bool IsCallReloc, CallLoweringInfo &CLI, SDValue Callee,
|
bool IsCallReloc, CallLoweringInfo &CLI, SDValue Callee,
|
||||||
SDValue Chain) const;
|
SDValue Chain) const;
|
||||||
|
|
||||||
|
/// ByValArgInfo - Byval argument information.
|
||||||
|
struct ByValArgInfo {
|
||||||
|
unsigned FirstIdx; // Index of the first register used.
|
||||||
|
unsigned NumRegs; // Number of registers used for this argument.
|
||||||
|
|
||||||
|
ByValArgInfo() : FirstIdx(0), NumRegs(0) {}
|
||||||
|
};
|
||||||
|
|
||||||
/// MipsCC - This class provides methods used to analyze formal and call
|
/// MipsCC - This class provides methods used to analyze formal and call
|
||||||
/// arguments and inquire about calling convention information.
|
/// arguments and inquire about calling convention information.
|
||||||
class MipsCC {
|
class MipsCC {
|
||||||
@@ -359,8 +365,12 @@ namespace llvm {
|
|||||||
CCState &State);
|
CCState &State);
|
||||||
void analyzeFormalArguments(const SmallVectorImpl<ISD::InputArg> &Ins,
|
void analyzeFormalArguments(const SmallVectorImpl<ISD::InputArg> &Ins,
|
||||||
bool IsSoftFloat,
|
bool IsSoftFloat,
|
||||||
|
Function::const_arg_iterator FuncArg,
|
||||||
CCState &State);
|
CCState &State);
|
||||||
|
|
||||||
|
/// hasByValArg - Returns true if function has byval arguments.
|
||||||
|
bool hasByValArg() const { return !ByValArgs.empty(); }
|
||||||
|
|
||||||
/// reservedArgArea - The size of the area the caller reserves for
|
/// reservedArgArea - The size of the area the caller reserves for
|
||||||
/// register arguments. This is 16-byte if ABI is O32.
|
/// register arguments. This is 16-byte if ABI is O32.
|
||||||
unsigned reservedArgArea() const;
|
unsigned reservedArgArea() const;
|
||||||
@@ -368,7 +378,24 @@ namespace llvm {
|
|||||||
/// Return pointer to array of integer argument registers.
|
/// Return pointer to array of integer argument registers.
|
||||||
const ArrayRef<MCPhysReg> intArgRegs() const;
|
const ArrayRef<MCPhysReg> intArgRegs() const;
|
||||||
|
|
||||||
|
typedef SmallVectorImpl<ByValArgInfo>::const_iterator byval_iterator;
|
||||||
|
byval_iterator byval_begin() const { return ByValArgs.begin(); }
|
||||||
|
byval_iterator byval_end() const { return ByValArgs.end(); }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
void handleByValArg(unsigned ValNo, MVT ValVT, MVT LocVT,
|
||||||
|
CCValAssign::LocInfo LocInfo,
|
||||||
|
ISD::ArgFlagsTy ArgFlags, CCState &State);
|
||||||
|
|
||||||
|
/// useRegsForByval - Returns true if the calling convention allows the
|
||||||
|
/// use of registers to pass byval arguments.
|
||||||
|
bool useRegsForByval() const { return CallConv != CallingConv::Fast; }
|
||||||
|
|
||||||
|
const MCPhysReg *shadowRegs() const;
|
||||||
|
|
||||||
|
void allocateRegs(ByValArgInfo &ByVal, unsigned ByValSize, unsigned Align,
|
||||||
|
CCState &State);
|
||||||
|
|
||||||
/// Return the type of the register which is used to pass an argument or
|
/// Return the type of the register which is used to pass an argument or
|
||||||
/// return a value. This function returns f64 if the argument is an i64
|
/// return a value. This function returns f64 if the argument is an i64
|
||||||
/// value which has been generated as a result of softening an f128 value.
|
/// value which has been generated as a result of softening an f128 value.
|
||||||
@@ -384,6 +411,7 @@ namespace llvm {
|
|||||||
|
|
||||||
CallingConv::ID CallConv;
|
CallingConv::ID CallConv;
|
||||||
const MipsSubtarget &Subtarget;
|
const MipsSubtarget &Subtarget;
|
||||||
|
SmallVector<ByValArgInfo, 2> ByValArgs;
|
||||||
};
|
};
|
||||||
protected:
|
protected:
|
||||||
SDValue lowerLOAD(SDValue Op, SelectionDAG &DAG) const;
|
SDValue lowerLOAD(SDValue Op, SelectionDAG &DAG) const;
|
||||||
@@ -448,7 +476,7 @@ namespace llvm {
|
|||||||
/// isEligibleForTailCallOptimization - Check whether the call is eligible
|
/// isEligibleForTailCallOptimization - Check whether the call is eligible
|
||||||
/// for tail call optimization.
|
/// for tail call optimization.
|
||||||
virtual bool
|
virtual bool
|
||||||
isEligibleForTailCallOptimization(const CCState &CCInfo,
|
isEligibleForTailCallOptimization(const MipsCC &MipsCCInfo,
|
||||||
unsigned NextStackOffset,
|
unsigned NextStackOffset,
|
||||||
const MipsFunctionInfo& FI) const = 0;
|
const MipsFunctionInfo& FI) const = 0;
|
||||||
|
|
||||||
@@ -459,15 +487,14 @@ namespace llvm {
|
|||||||
SelectionDAG &DAG, const ISD::ArgFlagsTy &Flags,
|
SelectionDAG &DAG, const ISD::ArgFlagsTy &Flags,
|
||||||
SmallVectorImpl<SDValue> &InVals,
|
SmallVectorImpl<SDValue> &InVals,
|
||||||
const Argument *FuncArg, const MipsCC &CC,
|
const Argument *FuncArg, const MipsCC &CC,
|
||||||
unsigned FirstReg, unsigned LastReg,
|
const ByValArgInfo &ByVal, const CCValAssign &VA) const;
|
||||||
const CCValAssign &VA) const;
|
|
||||||
|
|
||||||
/// passByValArg - Pass a byval argument in registers or on stack.
|
/// passByValArg - Pass a byval argument in registers or on stack.
|
||||||
void passByValArg(SDValue Chain, SDLoc DL,
|
void passByValArg(SDValue Chain, SDLoc DL,
|
||||||
std::deque<std::pair<unsigned, SDValue>> &RegsToPass,
|
std::deque<std::pair<unsigned, SDValue>> &RegsToPass,
|
||||||
SmallVectorImpl<SDValue> &MemOpChains, SDValue StackPtr,
|
SmallVectorImpl<SDValue> &MemOpChains, SDValue StackPtr,
|
||||||
MachineFrameInfo *MFI, SelectionDAG &DAG, SDValue Arg,
|
MachineFrameInfo *MFI, SelectionDAG &DAG, SDValue Arg,
|
||||||
const MipsCC &CC, unsigned FirstReg, unsigned LastReg,
|
const MipsCC &CC, const ByValArgInfo &ByVal,
|
||||||
const ISD::ArgFlagsTy &Flags, bool isLittle,
|
const ISD::ArgFlagsTy &Flags, bool isLittle,
|
||||||
const CCValAssign &VA) const;
|
const CCValAssign &VA) const;
|
||||||
|
|
||||||
|
@@ -1167,14 +1167,15 @@ MipsSETargetLowering::EmitInstrWithCustomInserter(MachineInstr *MI,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MipsSETargetLowering::isEligibleForTailCallOptimization(
|
bool MipsSETargetLowering::
|
||||||
const CCState &CCInfo, unsigned NextStackOffset,
|
isEligibleForTailCallOptimization(const MipsCC &MipsCCInfo,
|
||||||
|
unsigned NextStackOffset,
|
||||||
const MipsFunctionInfo& FI) const {
|
const MipsFunctionInfo& FI) const {
|
||||||
if (!EnableMipsTailCalls)
|
if (!EnableMipsTailCalls)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Return false if either the callee or caller has a byval argument.
|
// Return false if either the callee or caller has a byval argument.
|
||||||
if (CCInfo.getInRegsParamsCount() > 0 || FI.hasByvalArg())
|
if (MipsCCInfo.hasByValArg() || FI.hasByvalArg())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Return true if the callee's argument area is no larger than the
|
// Return true if the callee's argument area is no larger than the
|
||||||
|
@@ -51,8 +51,8 @@ namespace llvm {
|
|||||||
const TargetRegisterClass *getRepRegClassFor(MVT VT) const override;
|
const TargetRegisterClass *getRepRegClassFor(MVT VT) const override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool isEligibleForTailCallOptimization(
|
bool isEligibleForTailCallOptimization(const MipsCC &MipsCCInfo,
|
||||||
const CCState &CCInfo, unsigned NextStackOffset,
|
unsigned NextStackOffset,
|
||||||
const MipsFunctionInfo& FI) const override;
|
const MipsFunctionInfo& FI) const override;
|
||||||
|
|
||||||
void
|
void
|
||||||
|
Reference in New Issue
Block a user