
This also fixes some missing implicit uses on call instructions, adds missing G_ASSERT_SEXT/ZEXT annotations, and some missing outgoing sext/zexts. This also fixes not respecting tablegen requested type promotions. This starts treating f64 passed in i32 GPRs as a type of custom assignment, which restores some previously XFAILed tests. This is due to getNumRegistersForCallingConv returns a static value, but in this case it is context dependent on other arguments. Most of the ugliness is reproducing a hack CC_MipsO32 uses in SelectionDAG. CC_MipsO32 depends on a bunch of vectors populated from the original IR argument types in MipsCCState. The way this ends up working in GlobalISel is it only ends up inspecting the most recently added vector element. I'm pretty sure there are cleaner ways to do this, but this seemed easier than fixing up the current DAG handling. This is another case where it would be easier of the CCAssignFns were passed the original type instead of only the pre-legalized ones. There's still a lot of junk here that shouldn't be necessary. This also likely breaks big endian handling, but it wasn't complete/tested anyway since the IRTranslator gives up on big endian targets.
210 lines
7.8 KiB
C++
210 lines
7.8 KiB
C++
//===---- MipsCCState.h - CCState with Mips specific extensions -----------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef MIPSCCSTATE_H
|
|
#define MIPSCCSTATE_H
|
|
|
|
#include "MipsISelLowering.h"
|
|
#include "llvm/ADT/SmallVector.h"
|
|
#include "llvm/CodeGen/CallingConvLower.h"
|
|
|
|
namespace llvm {
|
|
class SDNode;
|
|
class MipsSubtarget;
|
|
|
|
class MipsCCState : public CCState {
|
|
public:
|
|
enum SpecialCallingConvType { Mips16RetHelperConv, NoSpecialCallingConv };
|
|
|
|
/// Determine the SpecialCallingConvType for the given callee
|
|
static SpecialCallingConvType
|
|
getSpecialCallingConvForCallee(const SDNode *Callee,
|
|
const MipsSubtarget &Subtarget);
|
|
|
|
/// This function returns true if CallSym is a long double emulation routine.
|
|
///
|
|
/// FIXME: Changing the ABI based on the callee name is unsound. The lib func
|
|
/// address could be captured.
|
|
static bool isF128SoftLibCall(const char *CallSym);
|
|
|
|
static bool originalTypeIsF128(const Type *Ty, const char *Func);
|
|
static bool originalEVTTypeIsVectorFloat(EVT Ty);
|
|
static bool originalTypeIsVectorFloat(const Type *Ty);
|
|
|
|
void PreAnalyzeCallOperand(const Type *ArgTy, bool IsFixed, const char *Func);
|
|
|
|
void PreAnalyzeFormalArgument(const Type *ArgTy, ISD::ArgFlagsTy Flags);
|
|
void PreAnalyzeReturnValue(EVT ArgVT);
|
|
|
|
private:
|
|
/// Identify lowered values that originated from f128 arguments and record
|
|
/// this for use by RetCC_MipsN.
|
|
void PreAnalyzeCallResultForF128(const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
const Type *RetTy, const char * Func);
|
|
|
|
/// Identify lowered values that originated from f128 arguments and record
|
|
/// this for use by RetCC_MipsN.
|
|
void PreAnalyzeReturnForF128(const SmallVectorImpl<ISD::OutputArg> &Outs);
|
|
|
|
/// Identify lowered values that originated from f128 arguments and record
|
|
/// this.
|
|
void
|
|
PreAnalyzeCallOperands(const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
std::vector<TargetLowering::ArgListEntry> &FuncArgs,
|
|
const char *Func);
|
|
|
|
/// Identify lowered values that originated from f128 arguments and record
|
|
/// this for use by RetCC_MipsN.
|
|
void
|
|
PreAnalyzeFormalArgumentsForF128(const SmallVectorImpl<ISD::InputArg> &Ins);
|
|
|
|
void
|
|
PreAnalyzeCallResultForVectorFloat(const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
const Type *RetTy);
|
|
|
|
void PreAnalyzeFormalArgumentsForVectorFloat(
|
|
const SmallVectorImpl<ISD::InputArg> &Ins);
|
|
|
|
void
|
|
PreAnalyzeReturnForVectorFloat(const SmallVectorImpl<ISD::OutputArg> &Outs);
|
|
|
|
/// Records whether the value has been lowered from an f128.
|
|
SmallVector<bool, 4> OriginalArgWasF128;
|
|
|
|
/// Records whether the value has been lowered from float.
|
|
SmallVector<bool, 4> OriginalArgWasFloat;
|
|
|
|
/// Records whether the value has been lowered from a floating point vector.
|
|
SmallVector<bool, 4> OriginalArgWasFloatVector;
|
|
|
|
/// Records whether the return value has been lowered from a floating point
|
|
/// vector.
|
|
SmallVector<bool, 4> OriginalRetWasFloatVector;
|
|
|
|
/// Records whether the value was a fixed argument.
|
|
/// See ISD::OutputArg::IsFixed,
|
|
SmallVector<bool, 4> CallOperandIsFixed;
|
|
|
|
// Used to handle MIPS16-specific calling convention tweaks.
|
|
// FIXME: This should probably be a fully fledged calling convention.
|
|
SpecialCallingConvType SpecialCallingConv;
|
|
|
|
public:
|
|
MipsCCState(CallingConv::ID CC, bool isVarArg, MachineFunction &MF,
|
|
SmallVectorImpl<CCValAssign> &locs, LLVMContext &C,
|
|
SpecialCallingConvType SpecialCC = NoSpecialCallingConv)
|
|
: CCState(CC, isVarArg, MF, locs, C), SpecialCallingConv(SpecialCC) {}
|
|
|
|
void PreAnalyzeCallOperands(
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs, CCAssignFn Fn,
|
|
std::vector<TargetLowering::ArgListEntry> &FuncArgs, const char *Func) {
|
|
OriginalArgWasF128.clear();
|
|
OriginalArgWasFloat.clear();
|
|
OriginalArgWasFloatVector.clear();
|
|
CallOperandIsFixed.clear();
|
|
PreAnalyzeCallOperands(Outs, FuncArgs, Func);
|
|
}
|
|
|
|
void
|
|
AnalyzeCallOperands(const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
CCAssignFn Fn,
|
|
std::vector<TargetLowering::ArgListEntry> &FuncArgs,
|
|
const char *Func) {
|
|
PreAnalyzeCallOperands(Outs, Fn, FuncArgs, Func);
|
|
CCState::AnalyzeCallOperands(Outs, Fn);
|
|
}
|
|
|
|
// The AnalyzeCallOperands in the base class is not usable since we must
|
|
// provide a means of accessing ArgListEntry::IsFixed. Delete them from this
|
|
// class. This doesn't stop them being used via the base class though.
|
|
void AnalyzeCallOperands(const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
CCAssignFn Fn) = delete;
|
|
void AnalyzeCallOperands(const SmallVectorImpl<MVT> &Outs,
|
|
SmallVectorImpl<ISD::ArgFlagsTy> &Flags,
|
|
CCAssignFn Fn) = delete;
|
|
|
|
void PreAnalyzeFormalArguments(const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
CCAssignFn Fn) {
|
|
OriginalArgWasFloat.clear();
|
|
OriginalArgWasF128.clear();
|
|
OriginalArgWasFloatVector.clear();
|
|
PreAnalyzeFormalArgumentsForF128(Ins);
|
|
}
|
|
|
|
void AnalyzeFormalArguments(const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
CCAssignFn Fn) {
|
|
PreAnalyzeFormalArguments(Ins, Fn);
|
|
CCState::AnalyzeFormalArguments(Ins, Fn);
|
|
}
|
|
|
|
void PreAnalyzeCallResult(const Type *RetTy, const char *Func) {
|
|
OriginalArgWasF128.push_back(originalTypeIsF128(RetTy, Func));
|
|
OriginalArgWasFloat.push_back(RetTy->isFloatingPointTy());
|
|
OriginalRetWasFloatVector.push_back(originalTypeIsVectorFloat(RetTy));
|
|
}
|
|
|
|
void PreAnalyzeCallResult(const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
CCAssignFn Fn, const Type *RetTy,
|
|
const char *Func) {
|
|
OriginalArgWasFloat.clear();
|
|
OriginalArgWasF128.clear();
|
|
OriginalArgWasFloatVector.clear();
|
|
PreAnalyzeCallResultForF128(Ins, RetTy, Func);
|
|
PreAnalyzeCallResultForVectorFloat(Ins, RetTy);
|
|
}
|
|
|
|
void AnalyzeCallResult(const SmallVectorImpl<ISD::InputArg> &Ins,
|
|
CCAssignFn Fn, const Type *RetTy,
|
|
const char *Func) {
|
|
PreAnalyzeCallResult(Ins, Fn, RetTy, Func);
|
|
CCState::AnalyzeCallResult(Ins, Fn);
|
|
}
|
|
|
|
void PreAnalyzeReturn(const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
CCAssignFn Fn) {
|
|
OriginalArgWasFloat.clear();
|
|
OriginalArgWasF128.clear();
|
|
OriginalArgWasFloatVector.clear();
|
|
PreAnalyzeReturnForF128(Outs);
|
|
PreAnalyzeReturnForVectorFloat(Outs);
|
|
}
|
|
|
|
void AnalyzeReturn(const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
CCAssignFn Fn) {
|
|
PreAnalyzeReturn(Outs, Fn);
|
|
CCState::AnalyzeReturn(Outs, Fn);
|
|
}
|
|
|
|
bool CheckReturn(const SmallVectorImpl<ISD::OutputArg> &ArgsFlags,
|
|
CCAssignFn Fn) {
|
|
PreAnalyzeReturnForF128(ArgsFlags);
|
|
PreAnalyzeReturnForVectorFloat(ArgsFlags);
|
|
bool Return = CCState::CheckReturn(ArgsFlags, Fn);
|
|
OriginalArgWasFloat.clear();
|
|
OriginalArgWasF128.clear();
|
|
OriginalArgWasFloatVector.clear();
|
|
return Return;
|
|
}
|
|
|
|
bool WasOriginalArgF128(unsigned ValNo) { return OriginalArgWasF128[ValNo]; }
|
|
bool WasOriginalArgFloat(unsigned ValNo) {
|
|
return OriginalArgWasFloat[ValNo];
|
|
}
|
|
bool WasOriginalArgVectorFloat(unsigned ValNo) const {
|
|
return OriginalArgWasFloatVector[ValNo];
|
|
}
|
|
bool WasOriginalRetVectorFloat(unsigned ValNo) const {
|
|
return OriginalRetWasFloatVector[ValNo];
|
|
}
|
|
bool IsCallOperandFixed(unsigned ValNo) { return CallOperandIsFixed[ValNo]; }
|
|
SpecialCallingConvType getSpecialCallingConv() { return SpecialCallingConv; }
|
|
};
|
|
}
|
|
|
|
#endif
|