| //===----- ABIInfo.h - ABI information access & encapsulation ---*- C++ -*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef CLANG_CODEGEN_ABIINFO_H |
| #define CLANG_CODEGEN_ABIINFO_H |
| |
| #include "clang/AST/Type.h" |
| #include "llvm/IR/Type.h" |
| #include "llvm/IR/CallingConv.h" |
| |
| namespace llvm { |
| class Value; |
| class LLVMContext; |
| class DataLayout; |
| } |
| |
| namespace clang { |
| class ASTContext; |
| |
| namespace CodeGen { |
| class CGFunctionInfo; |
| class CodeGenFunction; |
| class CodeGenTypes; |
| } |
| |
| // FIXME: All of this stuff should be part of the target interface |
| // somehow. It is currently here because it is not clear how to factor |
| // the targets to support this, since the Targets currently live in a |
| // layer below types n'stuff. |
| |
| /// ABIArgInfo - Helper class to encapsulate information about how a |
| /// specific C type should be passed to or returned from a function. |
| class ABIArgInfo { |
| public: |
| enum Kind { |
| /// Direct - Pass the argument directly using the normal converted LLVM |
| /// type, or by coercing to another specified type stored in |
| /// 'CoerceToType'). If an offset is specified (in UIntData), then the |
| /// argument passed is offset by some number of bytes in the memory |
| /// representation. A dummy argument is emitted before the real argument |
| /// if the specified type stored in "PaddingType" is not zero. |
| Direct, |
| |
| /// Extend - Valid only for integer argument types. Same as 'direct' |
| /// but also emit a zero/sign extension attribute. |
| Extend, |
| |
| /// Indirect - Pass the argument indirectly via a hidden pointer |
| /// with the specified alignment (0 indicates default alignment). |
| Indirect, |
| |
| /// Ignore - Ignore the argument (treat as void). Useful for void and |
| /// empty structs. |
| Ignore, |
| |
| /// Expand - Only valid for aggregate argument types. The structure should |
| /// be expanded into consecutive arguments for its constituent fields. |
| /// Currently expand is only allowed on structures whose fields |
| /// are all scalar types or are themselves expandable types. |
| Expand, |
| |
| KindFirst=Direct, KindLast=Expand |
| }; |
| |
| private: |
| Kind TheKind; |
| llvm::Type *TypeData; |
| llvm::Type *PaddingType; |
| unsigned UIntData; |
| bool BoolData0; |
| bool BoolData1; |
| bool InReg; |
| bool PaddingInReg; |
| |
| ABIArgInfo(Kind K, llvm::Type *TD, unsigned UI, bool B0, bool B1, bool IR, |
| bool PIR, llvm::Type* P) |
| : TheKind(K), TypeData(TD), PaddingType(P), UIntData(UI), BoolData0(B0), |
| BoolData1(B1), InReg(IR), PaddingInReg(PIR) {} |
| |
| public: |
| ABIArgInfo() : TheKind(Direct), TypeData(0), UIntData(0) {} |
| |
| static ABIArgInfo getDirect(llvm::Type *T = 0, unsigned Offset = 0, |
| llvm::Type *Padding = 0) { |
| return ABIArgInfo(Direct, T, Offset, false, false, false, false, Padding); |
| } |
| static ABIArgInfo getDirectInReg(llvm::Type *T = 0) { |
| return ABIArgInfo(Direct, T, 0, false, false, true, false, 0); |
| } |
| static ABIArgInfo getExtend(llvm::Type *T = 0) { |
| return ABIArgInfo(Extend, T, 0, false, false, false, false, 0); |
| } |
| static ABIArgInfo getExtendInReg(llvm::Type *T = 0) { |
| return ABIArgInfo(Extend, T, 0, false, false, true, false, 0); |
| } |
| static ABIArgInfo getIgnore() { |
| return ABIArgInfo(Ignore, 0, 0, false, false, false, false, 0); |
| } |
| static ABIArgInfo getIndirect(unsigned Alignment, bool ByVal = true |
| , bool Realign = false |
| , llvm::Type *Padding = 0) { |
| return ABIArgInfo(Indirect, 0, Alignment, ByVal, Realign, false, false, |
| Padding); |
| } |
| static ABIArgInfo getIndirectInReg(unsigned Alignment, bool ByVal = true |
| , bool Realign = false) { |
| return ABIArgInfo(Indirect, 0, Alignment, ByVal, Realign, true, false, 0); |
| } |
| static ABIArgInfo getExpand() { |
| return ABIArgInfo(Expand, 0, 0, false, false, false, false, 0); |
| } |
| static ABIArgInfo getExpandWithPadding(bool PaddingInReg, |
| llvm::Type *Padding) { |
| return ABIArgInfo(Expand, 0, 0, false, false, false, PaddingInReg, |
| Padding); |
| } |
| |
| Kind getKind() const { return TheKind; } |
| bool isDirect() const { return TheKind == Direct; } |
| bool isExtend() const { return TheKind == Extend; } |
| bool isIgnore() const { return TheKind == Ignore; } |
| bool isIndirect() const { return TheKind == Indirect; } |
| bool isExpand() const { return TheKind == Expand; } |
| |
| bool canHaveCoerceToType() const { |
| return TheKind == Direct || TheKind == Extend; |
| } |
| |
| // Direct/Extend accessors |
| unsigned getDirectOffset() const { |
| assert((isDirect() || isExtend()) && "Not a direct or extend kind"); |
| return UIntData; |
| } |
| |
| llvm::Type *getPaddingType() const { |
| return PaddingType; |
| } |
| |
| bool getPaddingInReg() const { |
| return PaddingInReg; |
| } |
| |
| llvm::Type *getCoerceToType() const { |
| assert(canHaveCoerceToType() && "Invalid kind!"); |
| return TypeData; |
| } |
| |
| void setCoerceToType(llvm::Type *T) { |
| assert(canHaveCoerceToType() && "Invalid kind!"); |
| TypeData = T; |
| } |
| |
| bool getInReg() const { |
| assert((isDirect() || isExtend() || isIndirect()) && "Invalid kind!"); |
| return InReg; |
| } |
| |
| // Indirect accessors |
| unsigned getIndirectAlign() const { |
| assert(TheKind == Indirect && "Invalid kind!"); |
| return UIntData; |
| } |
| |
| bool getIndirectByVal() const { |
| assert(TheKind == Indirect && "Invalid kind!"); |
| return BoolData0; |
| } |
| |
| bool getIndirectRealign() const { |
| assert(TheKind == Indirect && "Invalid kind!"); |
| return BoolData1; |
| } |
| |
| void dump() const; |
| }; |
| |
| /// ABIInfo - Target specific hooks for defining how a type should be |
| /// passed or returned from functions. |
| class ABIInfo { |
| public: |
| CodeGen::CodeGenTypes &CGT; |
| protected: |
| llvm::CallingConv::ID RuntimeCC; |
| public: |
| ABIInfo(CodeGen::CodeGenTypes &cgt) |
| : CGT(cgt), RuntimeCC(llvm::CallingConv::C) {} |
| |
| virtual ~ABIInfo(); |
| |
| ASTContext &getContext() const; |
| llvm::LLVMContext &getVMContext() const; |
| const llvm::DataLayout &getDataLayout() const; |
| |
| /// Return the calling convention to use for system runtime |
| /// functions. |
| llvm::CallingConv::ID getRuntimeCC() const { |
| return RuntimeCC; |
| } |
| |
| virtual void computeInfo(CodeGen::CGFunctionInfo &FI) const = 0; |
| |
| /// EmitVAArg - Emit the target dependent code to load a value of |
| /// \arg Ty from the va_list pointed to by \arg VAListAddr. |
| |
| // FIXME: This is a gaping layering violation if we wanted to drop |
| // the ABI information any lower than CodeGen. Of course, for |
| // VAArg handling it has to be at this level; there is no way to |
| // abstract this out. |
| virtual llvm::Value *EmitVAArg(llvm::Value *VAListAddr, QualType Ty, |
| CodeGen::CodeGenFunction &CGF) const = 0; |
| }; |
| } // end namespace clang |
| |
| #endif |