|  | //===---- 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 |