blob: 2853bc80a5d43536eaaa9e143230bea77fb7ea6f [file] [log] [blame]
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +00001//===----- ABIInfo.h - ABI information access & encapsulation ---*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#ifndef CLANG_CODEGEN_ABIINFO_H
11#define CLANG_CODEGEN_ABIINFO_H
12
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000013#include "clang/AST/Type.h"
Chris Lattner958c53c2010-06-29 19:21:36 +000014#include "llvm/Type.h"
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000015
Daniel Dunbar88c2fa92009-02-03 05:31:23 +000016namespace llvm {
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000017 class Value;
Benjamin Kramerf21efe92009-08-11 17:46:57 +000018 class LLVMContext;
Chris Lattnerea044322010-07-29 02:01:43 +000019 class TargetData;
Daniel Dunbar88c2fa92009-02-03 05:31:23 +000020}
21
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000022namespace clang {
Daniel Dunbar6bad2652009-02-03 06:51:18 +000023 class ASTContext;
24
Daniel Dunbar6bad2652009-02-03 06:51:18 +000025 namespace CodeGen {
26 class CGFunctionInfo;
Daniel Dunbarb53e3e72009-02-10 21:44:36 +000027 class CodeGenFunction;
Chris Lattnerea044322010-07-29 02:01:43 +000028 class CodeGenTypes;
Daniel Dunbar6bad2652009-02-03 06:51:18 +000029 }
30
Chris Lattner2eb9cdd2010-07-28 23:46:15 +000031 // FIXME: All of this stuff should be part of the target interface
32 // somehow. It is currently here because it is not clear how to factor
33 // the targets to support this, since the Targets currently live in a
34 // layer below types n'stuff.
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000035
36 /// ABIArgInfo - Helper class to encapsulate information about how a
37 /// specific C type should be passed to or returned from a function.
38 class ABIArgInfo {
39 public:
40 enum Kind {
Chris Lattner117e3f42010-07-30 04:02:24 +000041 /// Direct - Pass the argument directly using the normal converted LLVM
42 /// type, or by coercing to another specified type stored in
43 /// 'CoerceToType'). If an offset is specified (in UIntData), then the
44 /// argument passed is offset by some number of bytes in the memory
Akira Hatanakaf0cc2082012-01-07 00:25:33 +000045 /// representation. A dummy argument is emitted before the real argument
46 /// if the specified type stored in "PaddingType" is not zero.
Chris Lattner117e3f42010-07-30 04:02:24 +000047 Direct,
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000048
Chris Lattner117e3f42010-07-30 04:02:24 +000049 /// Extend - Valid only for integer argument types. Same as 'direct'
50 /// but also emit a zero/sign extension attribute.
51 Extend,
Anton Korobeynikovcc6fa882009-06-06 09:36:29 +000052
Chris Lattner117e3f42010-07-30 04:02:24 +000053 /// Indirect - Pass the argument indirectly via a hidden pointer
54 /// with the specified alignment (0 indicates default alignment).
55 Indirect,
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000056
Chris Lattner117e3f42010-07-30 04:02:24 +000057 /// Ignore - Ignore the argument (treat as void). Useful for void and
58 /// empty structs.
59 Ignore,
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000060
Chris Lattner117e3f42010-07-30 04:02:24 +000061 /// Expand - Only valid for aggregate argument types. The structure should
62 /// be expanded into consecutive arguments for its constituent fields.
63 /// Currently expand is only allowed on structures whose fields
64 /// are all scalar types or are themselves expandable types.
65 Expand,
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000066
Daniel Dunbar0bcc5212009-02-03 06:30:17 +000067 KindFirst=Direct, KindLast=Expand
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000068 };
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000069
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000070 private:
71 Kind TheKind;
Chris Lattner9cbe4f02011-07-09 17:41:47 +000072 llvm::Type *TypeData;
Akira Hatanakaf0cc2082012-01-07 00:25:33 +000073 llvm::Type *PaddingType; // Currently allowed only for Direct.
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000074 unsigned UIntData;
Daniel Dunbarcf3b6f22010-09-16 20:42:02 +000075 bool BoolData0;
76 bool BoolData1;
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +000077
Akira Hatanakaf0cc2082012-01-07 00:25:33 +000078 ABIArgInfo(Kind K, llvm::Type *TD=0, unsigned UI=0,
79 bool B0 = false, bool B1 = false, llvm::Type* P = 0)
80 : TheKind(K), TypeData(TD), PaddingType(P), UIntData(UI), BoolData0(B0),
81 BoolData1(B1) {}
Anders Carlsson0a8f8472009-09-16 15:53:40 +000082
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000083 public:
Daniel Dunbar0bcc5212009-02-03 06:30:17 +000084 ABIArgInfo() : TheKind(Direct), TypeData(0), UIntData(0) {}
Daniel Dunbar88c2fa92009-02-03 05:31:23 +000085
Akira Hatanakaf0cc2082012-01-07 00:25:33 +000086 static ABIArgInfo getDirect(llvm::Type *T = 0, unsigned Offset = 0,
87 llvm::Type *Padding = 0) {
88 return ABIArgInfo(Direct, T, Offset, false, false, Padding);
Daniel Dunbar46327aa2009-02-03 06:17:37 +000089 }
Chris Lattner9cbe4f02011-07-09 17:41:47 +000090 static ABIArgInfo getExtend(llvm::Type *T = 0) {
Chris Lattner117e3f42010-07-30 04:02:24 +000091 return ABIArgInfo(Extend, T, 0);
Anton Korobeynikovcc6fa882009-06-06 09:36:29 +000092 }
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000093 static ABIArgInfo getIgnore() {
94 return ABIArgInfo(Ignore);
95 }
Daniel Dunbarcf3b6f22010-09-16 20:42:02 +000096 static ABIArgInfo getIndirect(unsigned Alignment, bool ByVal = true
97 , bool Realign = false) {
98 return ABIArgInfo(Indirect, 0, Alignment, ByVal, Realign);
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +000099 }
100 static ABIArgInfo getExpand() {
101 return ABIArgInfo(Expand);
102 }
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +0000103
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000104 Kind getKind() const { return TheKind; }
Daniel Dunbar46327aa2009-02-03 06:17:37 +0000105 bool isDirect() const { return TheKind == Direct; }
Anton Korobeynikovcc6fa882009-06-06 09:36:29 +0000106 bool isExtend() const { return TheKind == Extend; }
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000107 bool isIgnore() const { return TheKind == Ignore; }
Daniel Dunbar11e383a2009-02-05 08:00:50 +0000108 bool isIndirect() const { return TheKind == Indirect; }
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000109 bool isExpand() const { return TheKind == Expand; }
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +0000110
Chris Lattner800588f2010-07-29 06:26:06 +0000111 bool canHaveCoerceToType() const {
112 return TheKind == Direct || TheKind == Extend;
113 }
Michael J. Spencer9cac4942010-10-19 06:39:39 +0000114
Chris Lattner800588f2010-07-29 06:26:06 +0000115 // Direct/Extend accessors
Chris Lattner117e3f42010-07-30 04:02:24 +0000116 unsigned getDirectOffset() const {
117 assert((isDirect() || isExtend()) && "Not a direct or extend kind");
118 return UIntData;
119 }
Akira Hatanakaf0cc2082012-01-07 00:25:33 +0000120
121 llvm::Type *getPaddingType() const {
122 return PaddingType;
123 }
124
Chris Lattner9cbe4f02011-07-09 17:41:47 +0000125 llvm::Type *getCoerceToType() const {
Chris Lattner800588f2010-07-29 06:26:06 +0000126 assert(canHaveCoerceToType() && "Invalid kind!");
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000127 return TypeData;
128 }
Michael J. Spencer9cac4942010-10-19 06:39:39 +0000129
Chris Lattner9cbe4f02011-07-09 17:41:47 +0000130 void setCoerceToType(llvm::Type *T) {
Chris Lattner800588f2010-07-29 06:26:06 +0000131 assert(canHaveCoerceToType() && "Invalid kind!");
132 TypeData = T;
133 }
Michael J. Spencer9cac4942010-10-19 06:39:39 +0000134
Anders Carlsson0a8f8472009-09-16 15:53:40 +0000135 // Indirect accessors
Daniel Dunbar11e383a2009-02-05 08:00:50 +0000136 unsigned getIndirectAlign() const {
137 assert(TheKind == Indirect && "Invalid kind!");
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000138 return UIntData;
139 }
Daniel Dunbar6f7279b2009-02-04 23:24:38 +0000140
Anders Carlsson0a8f8472009-09-16 15:53:40 +0000141 bool getIndirectByVal() const {
142 assert(TheKind == Indirect && "Invalid kind!");
Daniel Dunbarcf3b6f22010-09-16 20:42:02 +0000143 return BoolData0;
144 }
145
146 bool getIndirectRealign() const {
147 assert(TheKind == Indirect && "Invalid kind!");
148 return BoolData1;
Anders Carlsson0a8f8472009-09-16 15:53:40 +0000149 }
Michael J. Spencer9cac4942010-10-19 06:39:39 +0000150
Daniel Dunbar6f7279b2009-02-04 23:24:38 +0000151 void dump() const;
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000152 };
153
154 /// ABIInfo - Target specific hooks for defining how a type should be
155 /// passed or returned from functions.
156 class ABIInfo {
157 public:
Chris Lattnerea044322010-07-29 02:01:43 +0000158 CodeGen::CodeGenTypes &CGT;
Michael J. Spencer9cac4942010-10-19 06:39:39 +0000159
Chris Lattnerea044322010-07-29 02:01:43 +0000160 ABIInfo(CodeGen::CodeGenTypes &cgt) : CGT(cgt) {}
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000161 virtual ~ABIInfo();
Michael J. Spencer9cac4942010-10-19 06:39:39 +0000162
Chris Lattnerea044322010-07-29 02:01:43 +0000163 ASTContext &getContext() const;
164 llvm::LLVMContext &getVMContext() const;
165 const llvm::TargetData &getTargetData() const;
Daniel Dunbar6bad2652009-02-03 06:51:18 +0000166
Chris Lattneree5dcd02010-07-29 02:31:05 +0000167 virtual void computeInfo(CodeGen::CGFunctionInfo &FI) const = 0;
Daniel Dunbarb53e3e72009-02-10 21:44:36 +0000168
169 /// EmitVAArg - Emit the target dependent code to load a value of
170 /// \arg Ty from the va_list pointed to by \arg VAListAddr.
Anton Korobeynikovc4a59eb2009-06-05 22:08:42 +0000171
Daniel Dunbarb53e3e72009-02-10 21:44:36 +0000172 // FIXME: This is a gaping layering violation if we wanted to drop
173 // the ABI information any lower than CodeGen. Of course, for
174 // VAArg handling it has to be at this level; there is no way to
175 // abstract this out.
176 virtual llvm::Value *EmitVAArg(llvm::Value *VAListAddr, QualType Ty,
177 CodeGen::CodeGenFunction &CGF) const = 0;
Daniel Dunbar9eb5c6d2009-02-03 01:05:53 +0000178 };
179} // end namespace clang
180
181#endif