FTL should support typed array GetByVal and related ops
[WebKit-https.git] / Source / JavaScriptCore / ftl / FTLAbbreviations.h
1 /*
2  * Copyright (C) 2013 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #ifndef FTLAbbreviations_h
27 #define FTLAbbreviations_h
28
29 #include <wtf/Platform.h>
30
31 #if ENABLE(FTL_JIT)
32
33 #include "FTLAbbreviatedTypes.h"
34 #include "FTLSwitchCase.h"
35 #include "FTLValueFromBlock.h"
36 #include <cstring>
37
38 namespace JSC { namespace FTL {
39
40 // This file contains short-form calls into the LLVM C API. It is meant to
41 // save typing and make the lowering code clearer. If we ever call an LLVM C API
42 // function more than once in the FTL lowering code, we should add a shortcut for
43 // it here.
44
45 #if USE(JSVALUE32_64)
46 #error "The FTL backend assumes that pointers are 64-bit."
47 #endif
48
49 static inline LType voidType(LContext context) { return LLVMVoidTypeInContext(context); }
50 static inline LType int1Type(LContext context) { return LLVMInt1TypeInContext(context); }
51 static inline LType int8Type(LContext context) { return LLVMInt8TypeInContext(context); }
52 static inline LType int16Type(LContext context) { return LLVMInt16TypeInContext(context); }
53 static inline LType int32Type(LContext context) { return LLVMInt32TypeInContext(context); }
54 static inline LType int64Type(LContext context) { return LLVMInt64TypeInContext(context); }
55 static inline LType intPtrType(LContext context) { return LLVMInt64TypeInContext(context); }
56 static inline LType floatType(LContext context) { return LLVMFloatTypeInContext(context); }
57 static inline LType doubleType(LContext context) { return LLVMDoubleTypeInContext(context); }
58
59 static inline LType pointerType(LType type) { return LLVMPointerType(type, 0); }
60
61 enum PackingMode { NotPacked, Packed };
62 static inline LType structType(LContext context, LType* elementTypes, unsigned elementCount, PackingMode packing = NotPacked)
63 {
64     return LLVMStructTypeInContext(context, elementTypes, elementCount, packing == Packed);
65 }
66 static inline LType structType(LContext context, PackingMode packing = NotPacked)
67 {
68     return structType(context, 0, 0, packing);
69 }
70 static inline LType structType(LContext context, LType element1, PackingMode packing = NotPacked)
71 {
72     return structType(context, &element1, 1, packing);
73 }
74 static inline LType structType(LContext context, LType element1, LType element2, PackingMode packing = NotPacked)
75 {
76     LType elements[] = { element1, element2 };
77     return structType(context, elements, 2, packing);
78 }
79
80 enum Variadicity { NotVariadic, Variadic };
81 static inline LType functionType(LType returnType, const LType* paramTypes, unsigned paramCount, Variadicity variadicity)
82 {
83     return LLVMFunctionType(returnType, const_cast<LType*>(paramTypes), paramCount, variadicity == Variadic);
84 }
85 template<typename VectorType>
86 inline LType functionType(LType returnType, const VectorType& vector, Variadicity variadicity = NotVariadic)
87 {
88     return functionType(returnType, vector.begin(), vector.size(), variadicity);
89 }
90 static inline LType functionType(LType returnType, Variadicity variadicity = NotVariadic)
91 {
92     return functionType(returnType, 0, 0, variadicity);
93 }
94 static inline LType functionType(LType returnType, LType param1, Variadicity variadicity = NotVariadic)
95 {
96     return functionType(returnType, &param1, 1, variadicity);
97 }
98 static inline LType functionType(LType returnType, LType param1, LType param2, Variadicity variadicity = NotVariadic)
99 {
100     LType paramTypes[] = { param1, param2 };
101     return functionType(returnType, paramTypes, 2, variadicity);
102 }
103 static inline LType functionType(LType returnType, LType param1, LType param2, LType param3, Variadicity variadicity = NotVariadic)
104 {
105     LType paramTypes[] = { param1, param2, param3 };
106     return functionType(returnType, paramTypes, 3, variadicity);
107 }
108 static inline LType functionType(LType returnType, LType param1, LType param2, LType param3, LType param4, Variadicity variadicity = NotVariadic)
109 {
110     LType paramTypes[] = { param1, param2, param3, param4 };
111     return functionType(returnType, paramTypes, 4, variadicity);
112 }
113
114 static inline LType typeOf(LValue value) { return LLVMTypeOf(value); }
115
116 static inline unsigned mdKindID(LContext context, const char* string) { return LLVMGetMDKindIDInContext(context, string, std::strlen(string)); }
117 static inline LValue mdString(LContext context, const char* string, unsigned length) { return LLVMMDStringInContext(context, string, length); }
118 static inline LValue mdString(LContext context, const char* string) { return mdString(context, string, std::strlen(string)); }
119 static inline LValue mdNode(LContext context, LValue* args, unsigned numArgs) { return LLVMMDNodeInContext(context, args, numArgs); }
120 static inline LValue mdNode(LContext context) { return mdNode(context, 0, 0); }
121 static inline LValue mdNode(LContext context, LValue arg1) { return mdNode(context, &arg1, 1); }
122 static inline LValue mdNode(LContext context, LValue arg1, LValue arg2)
123 {
124     LValue args[] = { arg1, arg2 };
125     return mdNode(context, args, 2);
126 }
127
128 static inline void setMetadata(LValue instruction, unsigned kind, LValue metadata) { LLVMSetMetadata(instruction, kind, metadata); }
129
130 static inline LValue addFunction(LModule module, const char* name, LType type) { return LLVMAddFunction(module, name, type); }
131 static inline void setLinkage(LValue global, LLinkage linkage) { LLVMSetLinkage(global, linkage); }
132 static inline void setFunctionCallingConv(LValue function, LCallConv convention) { LLVMSetFunctionCallConv(function, convention); }
133
134 static inline LValue addExternFunction(LModule module, const char* name, LType type)
135 {
136     LValue result = addFunction(module, name, type);
137     setLinkage(result, LLVMExternalLinkage);
138     return result;
139 }
140
141 static inline LValue getParam(LValue function, unsigned index) { return LLVMGetParam(function, index); }
142
143 enum BitExtension { ZeroExtend, SignExtend };
144 static inline LValue constInt(LType type, unsigned long long value, BitExtension extension = ZeroExtend) { return LLVMConstInt(type, value, extension == SignExtend); }
145 static inline LValue constReal(LType type, double value) { return LLVMConstReal(type, value); }
146 static inline LValue constIntToPtr(LValue value, LType type) { return LLVMConstIntToPtr(value, type); }
147 static inline LValue constBitCast(LValue value, LType type) { return LLVMConstBitCast(value, type); }
148
149 static inline LBasicBlock appendBasicBlock(LContext context, LValue function, const char* name = "") { return LLVMAppendBasicBlockInContext(context, function, name); }
150 static inline LBasicBlock insertBasicBlock(LContext context, LBasicBlock beforeBasicBlock, const char* name = "") { return LLVMInsertBasicBlockInContext(context, beforeBasicBlock, name); }
151
152 static inline LValue buildPhi(LBuilder builder, LType type) { return LLVMBuildPhi(builder, type, ""); }
153 static inline void addIncoming(LValue phi, const LValue* values, const LBasicBlock* blocks, unsigned numPredecessors)
154 {
155     LLVMAddIncoming(phi, const_cast<LValue*>(values), const_cast<LBasicBlock*>(blocks), numPredecessors);
156 }
157 static inline void addIncoming(LValue phi, ValueFromBlock value1)
158 {
159     LValue value = value1.value();
160     LBasicBlock block = value1.block();
161     addIncoming(phi, &value, &block, 1);
162 }
163 static inline void addIncoming(LValue phi, ValueFromBlock value1, ValueFromBlock value2)
164 {
165     LValue values[] = { value1.value(), value2.value() };
166     LBasicBlock blocks[] = { value1.block(), value2.block() };
167     addIncoming(phi, values, blocks, 2);
168 }
169 static inline LValue buildPhi(LBuilder builder, LType type, ValueFromBlock value1)
170 {
171     LValue result = buildPhi(builder, type);
172     addIncoming(result, value1);
173     return result;
174 }
175 static inline LValue buildPhi(
176     LBuilder builder, LType type, ValueFromBlock value1, ValueFromBlock value2)
177 {
178     LValue result = buildPhi(builder, type);
179     addIncoming(result, value1, value2);
180     return result;
181 }
182
183 static inline LValue buildAlloca(LBuilder builder, LType type) { return LLVMBuildAlloca(builder, type, ""); }
184 static inline LValue buildAdd(LBuilder builder, LValue left, LValue right) { return LLVMBuildAdd(builder, left, right, ""); }
185 static inline LValue buildSub(LBuilder builder, LValue left, LValue right) { return LLVMBuildSub(builder, left, right, ""); }
186 static inline LValue buildMul(LBuilder builder, LValue left, LValue right) { return LLVMBuildMul(builder, left, right, ""); }
187 static inline LValue buildDiv(LBuilder builder, LValue left, LValue right) { return LLVMBuildSDiv(builder, left, right, ""); }
188 static inline LValue buildRem(LBuilder builder, LValue left, LValue right) { return LLVMBuildSRem(builder, left, right, ""); }
189 static inline LValue buildNeg(LBuilder builder, LValue value) { return LLVMBuildNeg(builder, value, ""); }
190 static inline LValue buildFAdd(LBuilder builder, LValue left, LValue right) { return LLVMBuildFAdd(builder, left, right, ""); }
191 static inline LValue buildFSub(LBuilder builder, LValue left, LValue right) { return LLVMBuildFSub(builder, left, right, ""); }
192 static inline LValue buildFMul(LBuilder builder, LValue left, LValue right) { return LLVMBuildFMul(builder, left, right, ""); }
193 static inline LValue buildFDiv(LBuilder builder, LValue left, LValue right) { return LLVMBuildFDiv(builder, left, right, ""); }
194 static inline LValue buildFRem(LBuilder builder, LValue left, LValue right) { return LLVMBuildFRem(builder, left, right, ""); }
195 static inline LValue buildFNeg(LBuilder builder, LValue value) { return LLVMBuildFNeg(builder, value, ""); }
196 static inline LValue buildAnd(LBuilder builder, LValue left, LValue right) { return LLVMBuildAnd(builder, left, right, ""); }
197 static inline LValue buildOr(LBuilder builder, LValue left, LValue right) { return LLVMBuildOr(builder, left, right, ""); }
198 static inline LValue buildXor(LBuilder builder, LValue left, LValue right) { return LLVMBuildXor(builder, left, right, ""); }
199 static inline LValue buildShl(LBuilder builder, LValue left, LValue right) { return LLVMBuildShl(builder, left, right, ""); }
200 static inline LValue buildAShr(LBuilder builder, LValue left, LValue right) { return LLVMBuildAShr(builder, left, right, ""); }
201 static inline LValue buildLShr(LBuilder builder, LValue left, LValue right) { return LLVMBuildLShr(builder, left, right, ""); }
202 static inline LValue buildNot(LBuilder builder, LValue value) { return LLVMBuildNot(builder, value, ""); }
203 static inline LValue buildLoad(LBuilder builder, LValue pointer) { return LLVMBuildLoad(builder, pointer, ""); }
204 static inline LValue buildStore(LBuilder builder, LValue value, LValue pointer) { return LLVMBuildStore(builder, value, pointer); }
205 static inline LValue buildSExt(LBuilder builder, LValue value, LType type) { return LLVMBuildSExt(builder, value, type, ""); }
206 static inline LValue buildZExt(LBuilder builder, LValue value, LType type) { return LLVMBuildZExt(builder, value, type, ""); }
207 static inline LValue buildFPToSI(LBuilder builder, LValue value, LType type) { return LLVMBuildFPToSI(builder, value, type, ""); }
208 static inline LValue buildSIToFP(LBuilder builder, LValue value, LType type) { return LLVMBuildSIToFP(builder, value, type, ""); }
209 static inline LValue buildUIToFP(LBuilder builder, LValue value, LType type) { return LLVMBuildUIToFP(builder, value, type, ""); }
210 static inline LValue buildIntCast(LBuilder builder, LValue value, LType type) { return LLVMBuildIntCast(builder, value, type, ""); }
211 static inline LValue buildFPCast(LBuilder builder, LValue value, LType type) { return LLVMBuildFPCast(builder, value, type, ""); }
212 static inline LValue buildIntToPtr(LBuilder builder, LValue value, LType type) { return LLVMBuildIntToPtr(builder, value, type, ""); }
213 static inline LValue buildPtrToInt(LBuilder builder, LValue value, LType type) { return LLVMBuildPtrToInt(builder, value, type, ""); }
214 static inline LValue buildBitCast(LBuilder builder, LValue value, LType type) { return LLVMBuildBitCast(builder, value, type, ""); }
215 static inline LValue buildICmp(LBuilder builder, LIntPredicate cond, LValue left, LValue right) { return LLVMBuildICmp(builder, cond, left, right, ""); }
216 static inline LValue buildFCmp(LBuilder builder, LRealPredicate cond, LValue left, LValue right) { return LLVMBuildFCmp(builder, cond, left, right, ""); }
217 static inline LValue buildCall(LBuilder builder, LValue function, const LValue* args, unsigned numArgs)
218 {
219     return LLVMBuildCall(builder, function, const_cast<LValue*>(args), numArgs, "");
220 }
221 template<typename VectorType>
222 inline LValue buildCall(LBuilder builder, LValue function, const VectorType& vector)
223 {
224     return buildCall(builder, function, vector.begin(), vector.size());
225 }
226 static inline LValue buildCall(LBuilder builder, LValue function)
227 {
228     return buildCall(builder, function, 0, 0);
229 }
230 static inline LValue buildCall(LBuilder builder, LValue function, LValue arg1)
231 {
232     return buildCall(builder, function, &arg1, 1);
233 }
234 static inline LValue buildCall(LBuilder builder, LValue function, LValue arg1, LValue arg2)
235 {
236     LValue args[] = { arg1, arg2 };
237     return buildCall(builder, function, args, 2);
238 }
239 static inline LValue buildCall(LBuilder builder, LValue function, LValue arg1, LValue arg2, LValue arg3)
240 {
241     LValue args[] = { arg1, arg2, arg3 };
242     return buildCall(builder, function, args, 3);
243 }
244 static inline LValue buildCall(LBuilder builder, LValue function, LValue arg1, LValue arg2, LValue arg3, LValue arg4)
245 {
246     LValue args[] = { arg1, arg2, arg3, arg4 };
247     return buildCall(builder, function, args, 4);
248 }
249 enum TailCallMode { IsNotTailCall, IsTailCall };
250 static inline void setTailCall(LValue call, TailCallMode mode) { LLVMSetTailCall(call, mode == IsTailCall); }
251 static inline LValue buildExtractValue(LBuilder builder, LValue aggVal, unsigned index) { return LLVMBuildExtractValue(builder, aggVal, index, ""); }
252 static inline LValue buildSelect(LBuilder builder, LValue condition, LValue taken, LValue notTaken) { return LLVMBuildSelect(builder, condition, taken, notTaken, ""); }
253 static inline LValue buildBr(LBuilder builder, LBasicBlock destination) { return LLVMBuildBr(builder, destination); }
254 static inline LValue buildCondBr(LBuilder builder, LValue condition, LBasicBlock taken, LBasicBlock notTaken) { return LLVMBuildCondBr(builder, condition, taken, notTaken); }
255 static inline LValue buildSwitch(LBuilder builder, LValue value, LBasicBlock fallThrough, unsigned numCases) { return LLVMBuildSwitch(builder, value, fallThrough, numCases); }
256 static inline void addCase(LValue switchInst, LValue value, LBasicBlock target) { LLVMAddCase(switchInst, value, target); }
257 template<typename VectorType>
258 static inline LValue buildSwitch(LBuilder builder, LValue value, const VectorType& cases, LBasicBlock fallThrough)
259 {
260     LValue result = buildSwitch(builder, value, fallThrough, cases.size());
261     for (unsigned i = 0; i < cases.size(); ++i)
262         addCase(result, cases[i].value(), cases[i].target());
263     return result;
264 }
265 static inline LValue buildRet(LBuilder builder, LValue value) { return LLVMBuildRet(builder, value); }
266 static inline LValue buildUnreachable(LBuilder builder) { return LLVMBuildUnreachable(builder); }
267
268 static inline void dumpModule(LModule module) { LLVMDumpModule(module); }
269 static inline void verifyModule(LModule module)
270 {
271     char* error = 0;
272     LLVMVerifyModule(module, LLVMAbortProcessAction, &error);
273     LLVMDisposeMessage(error);
274 }
275
276 } } // namespace JSC::FTL
277
278 #endif // ENABLE(FTL_JIT)
279
280 #endif // FTLAbbreviations_h
281