blob: d199b7660459084c377205a7e3080449e12b1f49 [file] [log] [blame]
Shih-wei Liaod1fec812012-02-13 09:51:10 -08001/*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef ART_SRC_COMPILER_LLVM_IR_BUILDER_H_
18#define ART_SRC_COMPILER_LLVM_IR_BUILDER_H_
19
20#include "backend_types.h"
Logan Chien42e0e152012-01-13 15:42:36 +080021#include "runtime_support_func.h"
Shih-wei Liaod1fec812012-02-13 09:51:10 -080022
23#include <llvm/Constants.h>
24#include <llvm/DerivedTypes.h>
25#include <llvm/Support/IRBuilder.h>
26#include <llvm/Type.h>
27
28#include <stdint.h>
29
30
31namespace art {
32namespace compiler_llvm {
33
34
35typedef llvm::IRBuilder<> LLVMIRBuilder;
36// NOTE: Here we define our own LLVMIRBuilder type alias, so that we can
37// switch "preserveNames" template parameter easily.
38
39
40class IRBuilder : public LLVMIRBuilder {
41 public:
42 //--------------------------------------------------------------------------
43 // General
44 //--------------------------------------------------------------------------
45
46 IRBuilder(llvm::LLVMContext& context, llvm::Module& module);
47
48
49 //--------------------------------------------------------------------------
Shih-wei Liao4c1f4252012-02-13 09:57:20 -080050 // Pointer Arithmetic Helper Function
51 //--------------------------------------------------------------------------
52
53 llvm::IntegerType* getPtrEquivIntTy() {
54 return getInt32Ty();
55 }
56
57 size_t getSizeOfPtrEquivInt() {
58 return 4;
59 }
60
61 llvm::ConstantInt* getSizeOfPtrEquivIntValue() {
62 return getPtrEquivInt(getSizeOfPtrEquivInt());
63 }
64
65 llvm::ConstantInt* getPtrEquivInt(uint64_t i) {
66 return llvm::ConstantInt::get(getPtrEquivIntTy(), i);
67 }
68
69 llvm::Value* CreatePtrDisp(llvm::Value* base,
70 llvm::Value* offset,
71 llvm::PointerType* ret_ty) {
72
73 llvm::Value* base_int = CreatePtrToInt(base, getPtrEquivIntTy());
74 llvm::Value* result_int = CreateAdd(base_int, offset);
75 llvm::Value* result = CreateIntToPtr(result_int, ret_ty);
76
77 return result;
78 }
79
80 llvm::Value* CreatePtrDisp(llvm::Value* base,
81 llvm::Value* bs,
82 llvm::Value* count,
83 llvm::Value* offset,
84 llvm::PointerType* ret_ty) {
85
86 llvm::Value* block_offset = CreateMul(bs, count);
87 llvm::Value* total_offset = CreateAdd(block_offset, offset);
88
89 return CreatePtrDisp(base, total_offset, ret_ty);
90 }
91
TDYa1275bb86012012-04-11 05:57:28 -070092 llvm::Value* LoadFromObjectOffset(llvm::Value* object_addr, int32_t offset, llvm::Type* type) {
93 // Convert offset to llvm::value
94 llvm::Value* llvm_offset = getPtrEquivInt(offset);
95 // Calculate the value's address
96 llvm::Value* value_addr = CreatePtrDisp(object_addr, llvm_offset, type->getPointerTo());
97 // Load
98 return CreateLoad(value_addr);
99 }
100
101 void StoreToObjectOffset(llvm::Value* object_addr, int32_t offset, llvm::Value* new_value) {
102 // Convert offset to llvm::value
103 llvm::Value* llvm_offset = getPtrEquivInt(offset);
104 // Calculate the value's address
105 llvm::Value* value_addr = CreatePtrDisp(object_addr,
106 llvm_offset,
107 new_value->getType()->getPointerTo());
108 // Store
109 CreateStore(new_value, value_addr);
110 }
111
Shih-wei Liao4c1f4252012-02-13 09:57:20 -0800112
113 //--------------------------------------------------------------------------
Logan Chien42e0e152012-01-13 15:42:36 +0800114 // Runtime Helper Function
115 //--------------------------------------------------------------------------
116
117 llvm::Function* GetRuntime(runtime_support::RuntimeId rt) const;
118
119
120 //--------------------------------------------------------------------------
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800121 // Type Helper Function
122 //--------------------------------------------------------------------------
123
124 llvm::Type* getJType(char shorty_jty, JTypeSpace space) {
125 return getJType(GetJTypeFromShorty(shorty_jty), space);
126 }
127
128 llvm::Type* getJType(JType jty, JTypeSpace space) {
129 switch (space) {
130 case kAccurate:
131 return getJTypeInAccurateSpace(jty);
132
133 case kReg:
134 case kField: // Currently field space is equivalent to register space.
135 return getJTypeInRegSpace(jty);
136
137 case kArray:
138 return getJTypeInArraySpace(jty);
139 }
140
Logan Chien83426162011-12-09 09:29:50 +0800141 LOG(FATAL) << "Unknown type space: " << space;
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800142 return NULL;
143 }
144
145 llvm::Type* getJVoidTy() {
146 return getVoidTy();
147 }
148
149 llvm::IntegerType* getJBooleanTy() {
150 return getInt1Ty();
151 }
152
153 llvm::IntegerType* getJByteTy() {
154 return getInt8Ty();
155 }
156
157 llvm::IntegerType* getJCharTy() {
158 return getInt16Ty();
159 }
160
161 llvm::IntegerType* getJShortTy() {
162 return getInt16Ty();
163 }
164
165 llvm::IntegerType* getJIntTy() {
166 return getInt32Ty();
167 }
168
169 llvm::IntegerType* getJLongTy() {
170 return getInt64Ty();
171 }
172
173 llvm::Type* getJFloatTy() {
174 return getFloatTy();
175 }
176
177 llvm::Type* getJDoubleTy() {
178 return getDoubleTy();
179 }
180
181 llvm::PointerType* getJObjectTy() {
182 return jobject_type_;
183 }
184
Logan Chienf04364f2012-02-10 12:01:39 +0800185 llvm::PointerType* getJEnvTy() {
186 return jenv_type_;
187 }
188
189 llvm::Type* getJValueTy() {
190 // NOTE: JValue is an union type, which may contains boolean, byte, char,
191 // short, int, long, float, double, Object. However, LLVM itself does
192 // not support union type, so we have to return a type with biggest size,
193 // then bitcast it before we use it.
194 return getJLongTy();
195 }
196
Logan Chien8dfcbea2012-02-17 18:50:32 +0800197 llvm::StructType* getShadowFrameTy(uint32_t sirt_size);
198
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800199
200 //--------------------------------------------------------------------------
201 // Constant Value Helper Function
202 //--------------------------------------------------------------------------
203
204 llvm::ConstantInt* getJBoolean(bool is_true) {
205 return (is_true) ? getTrue() : getFalse();
206 }
207
208 llvm::ConstantInt* getJByte(int8_t i) {
209 return llvm::ConstantInt::getSigned(getJByteTy(), i);
210 }
211
212 llvm::ConstantInt* getJChar(int16_t i) {
213 return llvm::ConstantInt::getSigned(getJCharTy(), i);
214 }
215
216 llvm::ConstantInt* getJShort(int16_t i) {
217 return llvm::ConstantInt::getSigned(getJShortTy(), i);
218 }
219
220 llvm::ConstantInt* getJInt(int32_t i) {
221 return llvm::ConstantInt::getSigned(getJIntTy(), i);
222 }
223
224 llvm::ConstantInt* getJLong(int64_t i) {
225 return llvm::ConstantInt::getSigned(getJLongTy(), i);
226 }
227
228 llvm::Constant* getJFloat(float f) {
229 return llvm::ConstantFP::get(getJFloatTy(), f);
230 }
231
232 llvm::Constant* getJDouble(double d) {
233 return llvm::ConstantFP::get(getJDoubleTy(), d);
234 }
235
236 llvm::ConstantPointerNull* getJNull() {
237 return llvm::ConstantPointerNull::get(getJObjectTy());
238 }
239
240 llvm::Constant* getJZero(char shorty_jty) {
241 return getJZero(GetJTypeFromShorty(shorty_jty));
242 }
243
244 llvm::Constant* getJZero(JType jty) {
245 switch (jty) {
246 case kVoid:
Logan Chien83426162011-12-09 09:29:50 +0800247 LOG(FATAL) << "Zero is not a value of void type";
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800248 return NULL;
249
250 case kBoolean:
251 return getJBoolean(false);
252
253 case kByte:
254 return getJByte(0);
255
256 case kChar:
257 return getJChar(0);
258
259 case kShort:
260 return getJShort(0);
261
262 case kInt:
263 return getJInt(0);
264
265 case kLong:
266 return getJLong(0);
267
268 case kFloat:
269 return getJFloat(0.0f);
270
271 case kDouble:
272 return getJDouble(0.0);
273
274 case kObject:
275 return getJNull();
276 }
277
278 LOG(FATAL) << "Unknown java type: " << jty;
279 return NULL;
280 }
281
282
283 private:
284 //--------------------------------------------------------------------------
Logan Chien42e0e152012-01-13 15:42:36 +0800285 // Runtime Helper Function (Private)
286 //--------------------------------------------------------------------------
287
Logan Chien6a917992012-02-17 18:43:48 +0800288 void InitRuntimeSupportFuncDecl();
Logan Chien42e0e152012-01-13 15:42:36 +0800289
290
291 //--------------------------------------------------------------------------
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800292 // Type Helper Function (Private)
293 //--------------------------------------------------------------------------
294
295 llvm::Type* getJTypeInAccurateSpace(JType jty);
296 llvm::Type* getJTypeInRegSpace(JType jty);
297 llvm::Type* getJTypeInArraySpace(JType jty);
298
299
300 private:
Logan Chien6a917992012-02-17 18:43:48 +0800301 llvm::Module* module_;
302
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800303 llvm::PointerType* jobject_type_;
304
Logan Chienf04364f2012-02-10 12:01:39 +0800305 llvm::PointerType* jenv_type_;
306
Logan Chien8dfcbea2012-02-17 18:50:32 +0800307 llvm::StructType* art_frame_type_;
308
Logan Chien42e0e152012-01-13 15:42:36 +0800309 llvm::Function* runtime_support_func_decls_[runtime_support::MAX_ID];
310
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800311};
312
313
314} // namespace compiler_llvm
315} // namespace art
316
317#endif // ART_SRC_COMPILER_LLVM_IR_BUILDER_H_