forked from facebook/hhvm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharg-group.h
301 lines (255 loc) · 8.38 KB
/
arg-group.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
/*
+----------------------------------------------------------------------+
| HipHop for PHP |
+----------------------------------------------------------------------+
| Copyright (c) 2010-present Facebook, Inc. (http://www.facebook.com) |
+----------------------------------------------------------------------+
| This source file is subject to version 3.01 of the PHP license, |
| that is bundled with this package in the file LICENSE, and is |
| available through the world-wide-web at the following url: |
| http://www.php.net/license/3_01.txt |
| If you did not receive a copy of the PHP license and are unable to |
| obtain it through the world-wide-web, please send a note to |
| [email protected] so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
*/
#pragma once
#include "hphp/runtime/vm/jit/containers.h"
#include "hphp/runtime/vm/jit/reg-alloc.h"
#include "hphp/runtime/vm/jit/vasm-reg.h"
#include "hphp/util/arch.h"
namespace HPHP::jit {
struct SSATmp;
struct IRInstruction;
namespace NativeCalls { struct CallInfo; }
//////////////////////////////////////////////////////////////////////
/*
* CallDest is the destination specification for a cgCallHelper
* invocation.
*
* The DestType describes the return type of native helper calls,
* particularly register assignments.
*
* These are created using the callDest() member functions.
*/
//////////////////////////////////////////////////////////////////////
enum class DestType : uint8_t {
None, // return void (no valid registers)
Indirect, // return struct/object to the address in the first arg
SSA, // return an SSA value in 1 or 2 integer registers
Byte, // return a single-byte register value
TV, // return a TypedValue packed in two registers
Dbl, // return scalar double in a single FP register
SIMD, // return a TypedValue in one SIMD register
};
const char* destTypeName(DestType);
struct CallDest {
CallDest(DestType t, Type vt, Vreg r0 = Vreg{}, Vreg r1 = Vreg{})
: valueType{vt}, reg0{r0}, reg1{r1}, type{t}
{}
CallDest(DestType t, Vreg r0 = Vreg{}, Vreg r1 = Vreg{})
: CallDest(t, TTop, r0, r1)
{}
Type valueType;
Vreg reg0, reg1;
DestType type;
};
UNUSED const CallDest kVoidDest { DestType::None };
UNUSED const CallDest kIndirectDest { DestType::Indirect };
struct ArgDesc {
enum class Kind {
Reg, // Normal register
Imm, // 64-bit Immediate
TypeImm, // DataType Immediate
Addr, // Address (register plus 32-bit displacement)
DataPtr, // Pointer to data section
IndRet, // Indirect Return Address (register plus 32-bit displacement)
SpilledTV, // Address of TypedValue pushed onto the stack
};
Vreg srcReg() const { return m_srcReg; }
Kind kind() const { return m_kind; }
Immed64 imm() const {
assertx(m_kind == Kind::Imm || m_kind == Kind::DataPtr);
return m_imm64;
}
DataType typeImm() const {
assertx(m_kind == Kind::TypeImm);
return m_typeImm;
}
Immed disp() const {
assertx(m_kind == Kind::Addr ||
m_kind == Kind::IndRet);
return m_disp32;
}
Vreg srcReg2() const {
assertx(m_kind == Kind::SpilledTV);
return m_srcReg2;
}
bool isZeroExtend() const { return m_zeroExtend; }
Optional<AuxUnion> aux() const { return m_aux; }
private: // These should be created using ArgGroup.
friend struct ArgGroup;
explicit ArgDesc(Kind kind, Immed64 imm)
: m_kind(kind)
, m_imm64(imm)
{}
explicit ArgDesc(Kind kind, Vreg srcReg, Immed disp)
: m_kind(kind)
, m_srcReg(srcReg)
, m_disp32(disp)
{}
explicit ArgDesc(Kind kind)
: m_kind(kind)
{}
explicit ArgDesc(Kind kind, Vreg srcReg1, Vreg srcReg2)
: m_kind(kind)
, m_srcReg(srcReg1)
, m_srcReg2(srcReg2)
{}
explicit ArgDesc(SSATmp* tmp,
Vloc,
bool val = true,
Optional<AuxUnion> aux = std::nullopt);
private:
Kind m_kind;
Vreg m_srcReg;
union {
Immed64 m_imm64; // 64-bit plain immediate
Immed m_disp32; // 32-bit displacement
DataType m_typeImm;
Vreg m_srcReg2;
};
Optional<AuxUnion> m_aux;
bool m_zeroExtend{false};
};
//////////////////////////////////////////////////////////////////////
/*
* Bag of ArgDesc for use with cgCallHelper.
*
* You can create this using function chaining. Example:
*
* ArgGroup args;
* args.imm(0)
* .reg(rax)
* .immPtr(makeStaticString("Yo"))
* ;
* assertx(args.size() == 3);
*/
struct ArgGroup {
using ArgVec = jit::vector<ArgDesc>;
explicit ArgGroup(const IRInstruction* inst,
const StateVector<SSATmp,Vloc>& locs)
: m_inst(inst), m_locs(locs)
{}
size_t numGpArgs() const { return m_gpArgs.size(); }
size_t numSimdArgs() const { return m_simdArgs.size(); }
size_t numStackArgs() const { return m_stkArgs.size(); }
size_t numIndRetArgs() const { return m_indRetArgs.size(); }
const std::vector<Type>& argTypes() const {
return m_argTypes;
}
ArgDesc& gpArg(size_t i) {
assertx(i < m_gpArgs.size());
return m_gpArgs[i];
}
const ArgDesc& gpArg(size_t i) const {
return const_cast<ArgGroup*>(this)->gpArg(i);
}
const ArgDesc& simdArg(size_t i) const {
assertx(i < m_simdArgs.size());
return m_simdArgs[i];
}
const ArgDesc& stkArg(size_t i) const {
assertx(i < m_stkArgs.size());
return m_stkArgs[i];
}
const ArgDesc& indRetArg(size_t i) const {
assertx(i < m_indRetArgs.size());
return m_indRetArgs[i];
}
ArgDesc& operator[](size_t i) = delete;
ArgGroup& imm(Immed64 imm) {
push_arg(ArgDesc(ArgDesc::Kind::Imm, imm));
return *this;
}
template<class T> ArgGroup& immPtr(const T* ptr) {
return imm(uintptr_t(ptr));
}
ArgGroup& immPtr(std::nullptr_t) { return imm(0); }
template<class T> ArgGroup& dataPtr(const T* ptr) {
push_arg(ArgDesc{ArgDesc::Kind::DataPtr, ptr});
return *this;
}
ArgGroup& reg(Vreg reg) {
push_arg(ArgDesc(ArgDesc::Kind::Reg, reg, -1));
return *this;
}
ArgGroup& addr(Vreg base, Immed off) {
push_arg(ArgDesc(ArgDesc::Kind::Addr, base, off));
return *this;
}
/*
* indRet args are similar to simple addr args, but are used specifically to
* pass the address that the native call will use for indirect returns. If a
* platform has no dedicated registers for indirect returns, then it uses
* the first general purpose argument register.
*/
ArgGroup& indRet(Vreg base, Immed off) {
push_arg(ArgDesc(ArgDesc::Kind::IndRet, base, off));
return *this;
}
ArgGroup& ssa(int i, bool allowFP = true);
/*
* Pass tmp as a TypedValue passed by value.
*/
ArgGroup& typedValue(int i, Optional<AuxUnion> aux = std::nullopt);
ArgGroup& memberKeyIS(int i) {
return memberKeyImpl(i, true);
}
ArgGroup& memberKeyS(int i) {
return memberKeyImpl(i, false);
}
/*
* Push a TypedValue onto the stack and pass its address. This is
* needed, for example, if you want to pass a tv_lval to a function
* expecting a const Variant&.
*/
ArgGroup& constPtrToTV(Vreg type, Vreg data) {
push_arg(ArgDesc(ArgDesc::Kind::SpilledTV, type, data));
return *this;
}
const IRInstruction* inst() const {
return m_inst;
}
private:
void push_arg(const ArgDesc& arg, Type t = TBottom);
void push_SIMDarg(const ArgDesc& arg, Type t = TBottom);
/*
* For passing the m_type field of a TypedValue.
*/
ArgGroup& type(int i, Optional<AuxUnion> aux) {
auto s = m_inst->src(i);
push_arg(ArgDesc(s, m_locs[s], false, aux));
return *this;
}
ArgGroup& memberKeyImpl(int i, bool allowInt) {
auto key = m_inst->src(i);
if (key->isA(TStr) || (allowInt && key->isA(TInt))) {
return ssa(i);
}
return typedValue(i);
}
private:
const IRInstruction* m_inst;
const StateVector<SSATmp,Vloc>& m_locs;
ArgVec* m_override{nullptr}; // force args to go into a specific ArgVec
ArgVec m_indRetArgs; // Indirect Return Address
ArgVec m_gpArgs; // INTEGER class args
ArgVec m_simdArgs; // SSE class args
ArgVec m_stkArgs; // Overflow
jit::vector<Type> m_argTypes;
};
ArgGroup toArgGroup(const NativeCalls::CallInfo&,
const StateVector<SSATmp,Vloc>& locs,
const IRInstruction*);
}