blob: e20a80ae18cd00073957bca69c45fc7027f9b883 [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_METHOD_COMPILER_H_
18#define ART_SRC_COMPILER_LLVM_METHOD_COMPILER_H_
19
Logan Chienfca7e872011-12-20 20:08:22 +080020#include "backend_types.h"
Logan Chienc670a8d2011-12-20 21:25:56 +080021#include "dalvik_reg.h"
Shih-wei Liaod1fec812012-02-13 09:51:10 -080022#include "dex_file.h"
23#include "dex_instruction.h"
Elliott Hughes0f3c5532012-03-30 14:51:51 -070024#include "invoke_type.h"
Shih-wei Liaod1fec812012-02-13 09:51:10 -080025#include "object_utils.h"
26
27#include <llvm/Support/IRBuilder.h>
28
29#include <vector>
30
31#include <stdint.h>
32
33
34namespace art {
35 class ClassLinker;
36 class ClassLoader;
37 class CompiledMethod;
38 class Compiler;
39 class DexCache;
Logan Chien438c4b62012-01-17 16:06:00 +080040 class Field;
Logan Chien4dd96f52012-02-29 01:26:58 +080041 class OatCompilationUnit;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080042}
43
44
45namespace llvm {
46 class AllocaInst;
47 class BasicBlock;
48 class Function;
49 class FunctionType;
50 class LLVMContext;
51 class Module;
52 class Type;
53}
54
55
56namespace art {
57namespace compiler_llvm {
58
Logan Chien8b977d32012-02-21 19:14:55 +080059class CompilationUnit;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080060class CompilerLLVM;
61class IRBuilder;
62
63class MethodCompiler {
Shih-wei Liaod1fec812012-02-13 09:51:10 -080064 public:
Logan Chien8b977d32012-02-21 19:14:55 +080065 MethodCompiler(CompilationUnit* cunit,
Logan Chien83426162011-12-09 09:29:50 +080066 Compiler* compiler,
Logan Chien4dd96f52012-02-29 01:26:58 +080067 OatCompilationUnit* oat_compilation_unit);
Shih-wei Liaod1fec812012-02-13 09:51:10 -080068
69 ~MethodCompiler();
70
Logan Chien83426162011-12-09 09:29:50 +080071 CompiledMethod* Compile();
Shih-wei Liaod1fec812012-02-13 09:51:10 -080072
Logan Chienc670a8d2011-12-20 21:25:56 +080073
74 // Code generation helper function
75
76 IRBuilder& GetIRBuilder() const {
77 return irb_;
78 }
79
80
81 // Register helper function
82
83 llvm::Value* AllocDalvikLocalVarReg(RegCategory cat, uint32_t reg_idx);
84
85 llvm::Value* AllocDalvikRetValReg(RegCategory cat);
86
87
Shih-wei Liaod1fec812012-02-13 09:51:10 -080088 private:
89 void CreateFunction();
Shih-wei Liaod1fec812012-02-13 09:51:10 -080090 void EmitPrologue();
TDYa1274165a832012-04-03 17:47:16 -070091 void EmitStackOverflowCheck();
Logan Chienc670a8d2011-12-20 21:25:56 +080092 void EmitPrologueLastBranch();
Logan Chien8dfcbea2012-02-17 18:50:32 +080093 void EmitPrologueAllocShadowFrame();
Logan Chiend6ececa2011-12-27 16:20:15 +080094 void EmitPrologueAssignArgRegister();
Shih-wei Liaod1fec812012-02-13 09:51:10 -080095 void EmitInstructions();
Logan Chien83426162011-12-09 09:29:50 +080096 void EmitInstruction(uint32_t dex_pc, Instruction const* insn);
Shih-wei Liaod1fec812012-02-13 09:51:10 -080097
Logan Chien70f94b42011-12-27 17:49:11 +080098 enum CondBranchKind {
99 kCondBranch_EQ,
100 kCondBranch_NE,
101 kCondBranch_LT,
102 kCondBranch_GE,
103 kCondBranch_GT,
104 kCondBranch_LE,
105 };
106
107 enum IntArithmKind {
108 kIntArithm_Add,
109 kIntArithm_Sub,
110 kIntArithm_Mul,
111 kIntArithm_Div,
112 kIntArithm_Rem,
113 kIntArithm_And,
114 kIntArithm_Or,
115 kIntArithm_Xor,
Logan Chien5539ad02012-04-02 14:36:55 +0800116 };
117
118 enum IntShiftArithmKind {
Logan Chien70f94b42011-12-27 17:49:11 +0800119 kIntArithm_Shl,
120 kIntArithm_Shr,
121 kIntArithm_UShr,
122 };
123
124 enum FPArithmKind {
125 kFPArithm_Add,
126 kFPArithm_Sub,
127 kFPArithm_Mul,
128 kFPArithm_Div,
129 kFPArithm_Rem,
130 };
131
132#define GEN_INSN_ARGS uint32_t dex_pc, Instruction const* insn
133
134 // NOP, PAYLOAD (unreachable) instructions
135 void EmitInsn_Nop(GEN_INSN_ARGS);
136
137 // MOVE, MOVE_RESULT instructions
138 void EmitInsn_Move(GEN_INSN_ARGS, JType jty);
139 void EmitInsn_MoveResult(GEN_INSN_ARGS, JType jty);
140
141 // MOVE_EXCEPTION, THROW instructions
142 void EmitInsn_MoveException(GEN_INSN_ARGS);
143 void EmitInsn_ThrowException(GEN_INSN_ARGS);
Logan Chien9e5f5c12012-04-10 13:51:45 +0800144 void EmitInsn_ThrowVerificationError(GEN_INSN_ARGS);
Logan Chien70f94b42011-12-27 17:49:11 +0800145
146 // RETURN instructions
147 void EmitInsn_ReturnVoid(GEN_INSN_ARGS);
148 void EmitInsn_Return(GEN_INSN_ARGS);
149
150 // CONST, CONST_CLASS, CONST_STRING instructions
151 void EmitInsn_LoadConstant(GEN_INSN_ARGS, JType imm_jty);
152 void EmitInsn_LoadConstantString(GEN_INSN_ARGS);
153 void EmitInsn_LoadConstantClass(GEN_INSN_ARGS);
154
155 // MONITOR_ENTER, MONITOR_EXIT instructions
156 void EmitInsn_MonitorEnter(GEN_INSN_ARGS);
157 void EmitInsn_MonitorExit(GEN_INSN_ARGS);
158
159 // CHECK_CAST, INSTANCE_OF instructions
160 void EmitInsn_CheckCast(GEN_INSN_ARGS);
161 void EmitInsn_InstanceOf(GEN_INSN_ARGS);
162
163 // NEW_INSTANCE instructions
164 void EmitInsn_NewInstance(GEN_INSN_ARGS);
165
166 // ARRAY_LEN, NEW_ARRAY, FILLED_NEW_ARRAY, FILL_ARRAY_DATA instructions
167 void EmitInsn_ArrayLength(GEN_INSN_ARGS);
168 void EmitInsn_NewArray(GEN_INSN_ARGS);
169 void EmitInsn_FilledNewArray(GEN_INSN_ARGS, bool is_range);
170 void EmitInsn_FillArrayData(GEN_INSN_ARGS);
171
172 // GOTO, IF_TEST, IF_TESTZ instructions
173 void EmitInsn_UnconditionalBranch(GEN_INSN_ARGS);
174 void EmitInsn_BinaryConditionalBranch(GEN_INSN_ARGS, CondBranchKind cond);
175 void EmitInsn_UnaryConditionalBranch(GEN_INSN_ARGS, CondBranchKind cond);
176
177 // PACKED_SWITCH, SPARSE_SWITCH instrutions
178 void EmitInsn_PackedSwitch(GEN_INSN_ARGS);
179 void EmitInsn_SparseSwitch(GEN_INSN_ARGS);
180
181 // CMPX_FLOAT, CMPX_DOUBLE, CMP_LONG instructions
182 void EmitInsn_FPCompare(GEN_INSN_ARGS, JType fp_jty, bool gt_bias);
183 void EmitInsn_LongCompare(GEN_INSN_ARGS);
184
185 // AGET, APUT instrutions
186 void EmitInsn_AGet(GEN_INSN_ARGS, JType elem_jty);
187 void EmitInsn_APut(GEN_INSN_ARGS, JType elem_jty);
188
189 // IGET, IPUT instructions
190 void EmitInsn_IGet(GEN_INSN_ARGS, JType field_jty);
191 void EmitInsn_IPut(GEN_INSN_ARGS, JType field_jty);
192
193 // SGET, SPUT instructions
194 void EmitInsn_SGet(GEN_INSN_ARGS, JType field_jty);
195 void EmitInsn_SPut(GEN_INSN_ARGS, JType field_jty);
196
197 // INVOKE instructions
TDYa1270b686e52012-04-09 22:43:35 -0700198 llvm::Value* EmitFixStub(llvm::Value* callee_method_object_addr,
199 uint32_t method_idx,
200 bool is_static);
TDYa12785321912012-04-01 15:24:56 -0700201 llvm::Value* EmitEnsureResolved(llvm::Value* callee,
202 llvm::Value* caller,
203 uint32_t dex_method_idx,
TDYa1270b686e52012-04-09 22:43:35 -0700204 bool is_virtual);
TDYa12785321912012-04-01 15:24:56 -0700205
Logan Chien70f94b42011-12-27 17:49:11 +0800206 void EmitInsn_InvokeVirtual(GEN_INSN_ARGS, bool is_range);
207 void EmitInsn_InvokeSuper(GEN_INSN_ARGS, bool is_range);
Shih-wei Liao399ed3f2012-03-08 01:27:04 -0800208 void EmitInsn_InvokeVirtualSuperSlow(uint32_t dex_pc,
209 DecodedInstruction &dec_insn,
210 bool is_range,
211 uint32_t callee_method_idx,
212 bool is_virtual);
Logan Chien1a121b92012-02-15 22:23:42 +0800213 void EmitInsn_InvokeStaticDirect(GEN_INSN_ARGS,
Logan Chien61c65dc2012-02-29 03:22:30 +0800214 InvokeType invoke_type,
215 bool is_range);
Logan Chien70f94b42011-12-27 17:49:11 +0800216 void EmitInsn_InvokeInterface(GEN_INSN_ARGS, bool is_range);
217
218 // Unary instructions
219 void EmitInsn_Neg(GEN_INSN_ARGS, JType op_jty);
220 void EmitInsn_Not(GEN_INSN_ARGS, JType op_jty);
221 void EmitInsn_SExt(GEN_INSN_ARGS);
222 void EmitInsn_Trunc(GEN_INSN_ARGS);
223 void EmitInsn_TruncAndSExt(GEN_INSN_ARGS, unsigned N);
224 void EmitInsn_TruncAndZExt(GEN_INSN_ARGS, unsigned N);
225
226 void EmitInsn_FNeg(GEN_INSN_ARGS, JType op_jty);
227 void EmitInsn_IntToFP(GEN_INSN_ARGS, JType src_jty, JType dest_jty);
228 void EmitInsn_FPToInt(GEN_INSN_ARGS, JType src_jty, JType dest_jty);
229 void EmitInsn_FExt(GEN_INSN_ARGS);
230 void EmitInsn_FTrunc(GEN_INSN_ARGS);
231
232 // Integer binary arithmetic instructions
233 void EmitInsn_IntArithm(GEN_INSN_ARGS, IntArithmKind arithm,
234 JType op_jty, bool is_2addr);
235
236 void EmitInsn_IntArithmImmediate(GEN_INSN_ARGS, IntArithmKind arithm);
237
Logan Chien5539ad02012-04-02 14:36:55 +0800238 void EmitInsn_IntShiftArithm(GEN_INSN_ARGS, IntShiftArithmKind arithm,
239 JType op_jty, bool is_2addr);
240
241 void EmitInsn_IntShiftArithmImmediate(GEN_INSN_ARGS,
242 IntShiftArithmKind arithm);
243
Logan Chien70f94b42011-12-27 17:49:11 +0800244 void EmitInsn_RSubImmediate(GEN_INSN_ARGS);
245
246
247 // Floating-point binary arithmetic instructions
248 void EmitInsn_FPArithm(GEN_INSN_ARGS, FPArithmKind arithm,
249 JType op_jty, bool is_2addr);
250
251#undef GEN_INSN_ARGS
252
Logan Chien0b827102011-12-20 19:46:14 +0800253
Logan Chien8dfcbea2012-02-17 18:50:32 +0800254 // Shadow frame helper function
255 void EmitPopShadowFrame();
256 void EmitUpdateLineNum(int32_t line_number);
257 void EmitUpdateLineNumFromDexPC(uint32_t dex_pc);
258
259
Logan Chienbb4d12a2012-02-17 14:10:01 +0800260 // Dex cache code generation helper function
261 llvm::Value* EmitLoadDexCacheAddr(MemberOffset dex_cache_offset);
262
Logan Chienbb4d12a2012-02-17 14:10:01 +0800263 llvm::Value* EmitLoadDexCacheStaticStorageFieldAddr(uint32_t type_idx);
264
265 llvm::Value* EmitLoadDexCacheResolvedTypeFieldAddr(uint32_t type_idx);
266
Logan Chien61c65dc2012-02-29 03:22:30 +0800267 llvm::Value* EmitLoadDexCacheResolvedMethodFieldAddr(uint32_t method_idx);
268
Logan Chienbb4d12a2012-02-17 14:10:01 +0800269 llvm::Value* EmitLoadDexCacheStringFieldAddr(uint32_t string_idx);
270
271
Logan Chien0b827102011-12-20 19:46:14 +0800272 // Code generation helper function
273
274 llvm::Value* EmitLoadMethodObjectAddr();
275
276 llvm::FunctionType* GetFunctionType(uint32_t method_idx, bool is_static);
277
Logan Chien5bcc04e2012-01-30 14:15:12 +0800278 void EmitGuard_ExceptionLandingPad(uint32_t dex_pc);
279
280 void EmitBranchExceptionLandingPad(uint32_t dex_pc);
281
Logan Chien70f94b42011-12-27 17:49:11 +0800282 void EmitGuard_GarbageCollectionSuspend(uint32_t dex_pc);
Logan Chien924072f2012-01-30 15:07:24 +0800283
Logan Chien2c37e8e2011-12-27 17:58:46 +0800284 llvm::Value* EmitCompareResultSelection(llvm::Value* cmp_eq,
285 llvm::Value* cmp_lt);
286
Logan Chiena78e3c82011-12-27 17:59:35 +0800287 llvm::Value* EmitConditionResult(llvm::Value* lhs,
288 llvm::Value* rhs,
289 CondBranchKind cond);
290
Logan Chienc3f7d962011-12-27 18:13:18 +0800291 llvm::Value* EmitIntArithmResultComputation(uint32_t dex_pc,
292 llvm::Value* lhs,
293 llvm::Value* rhs,
294 IntArithmKind arithm,
295 JType op_jty);
296
TDYa127f8641ce2012-04-02 06:40:40 -0700297 llvm::Value* EmitIntDivRemResultComputation(uint32_t dex_pc,
298 llvm::Value* dividend,
299 llvm::Value* divisor,
300 IntArithmKind arithm,
301 JType op_jty);
302
Logan Chien5539ad02012-04-02 14:36:55 +0800303 llvm::Value* EmitIntShiftArithmResultComputation(uint32_t dex_pc,
304 llvm::Value* lhs,
305 llvm::Value* rhs,
306 IntShiftArithmKind arithm,
307 JType op_jty);
308
Logan Chien76e1c792011-12-27 18:15:01 +0800309 llvm::Value* EmitFPArithmResultComputation(uint32_t dex_pc,
310 llvm::Value* lhs,
311 llvm::Value* rhs,
312 FPArithmKind arithm);
313
Logan Chiena2cc6a32012-01-16 10:38:41 +0800314 llvm::Value* EmitAllocNewArray(uint32_t dex_pc,
315 int32_t length,
316 uint32_t type_idx,
317 bool is_filled_new_array);
318
Logan Chien46fbb412012-02-15 22:29:08 +0800319 llvm::Value* EmitLoadClassObjectAddr(llvm::Value* this_addr);
320
321 llvm::Value* EmitLoadVTableAddr(llvm::Value* class_object_addr);
322
323 llvm::Value* EmitLoadMethodObjectAddrFromVTable(llvm::Value* vtable_addr,
324 uint16_t vtable_index);
325
326 llvm::Value* EmitLoadCodeAddr(llvm::Value* method_object_addr,
327 uint32_t method_idx,
328 bool is_static);
329
Logan Chien61bb6142012-02-03 15:34:53 +0800330 llvm::Value* EmitLoadArrayLength(llvm::Value* array);
331
Logan Chiene27fdbb2012-01-02 23:27:26 +0800332 llvm::Value* EmitArrayGEP(llvm::Value* array_addr,
333 llvm::Value* index_value,
Ian Rogers04ec04e2012-02-28 16:15:33 -0800334 llvm::Type* elem_type,
335 JType elem_jty);
Logan Chiene27fdbb2012-01-02 23:27:26 +0800336
Logan Chien27b30252012-01-14 03:43:35 +0800337 llvm::Value* EmitLoadConstantClass(uint32_t dex_pc, uint32_t type_idx);
338
Logan Chien438c4b62012-01-17 16:06:00 +0800339 llvm::Value* EmitLoadStaticStorage(uint32_t dex_pc, uint32_t type_idx);
340
Elliott Hughesadb8c672012-03-06 16:49:32 -0800341 llvm::Value* EmitLoadCalleeThis(DecodedInstruction const& di, bool is_range);
Logan Chien1a121b92012-02-15 22:23:42 +0800342
343 void EmitLoadActualParameters(std::vector<llvm::Value*>& args,
344 uint32_t callee_method_idx,
Elliott Hughesadb8c672012-03-06 16:49:32 -0800345 DecodedInstruction const& di,
Logan Chien1a121b92012-02-15 22:23:42 +0800346 bool is_range,
347 bool is_static);
348
Logan Chienc3f7d962011-12-27 18:13:18 +0800349 void EmitGuard_DivZeroException(uint32_t dex_pc,
350 llvm::Value* denominator,
351 JType op_jty);
352
Logan Chien61bb6142012-02-03 15:34:53 +0800353 void EmitGuard_NullPointerException(uint32_t dex_pc,
354 llvm::Value* object);
355
Logan Chiene27fdbb2012-01-02 23:27:26 +0800356 void EmitGuard_ArrayIndexOutOfBoundsException(uint32_t dex_pc,
357 llvm::Value* array,
358 llvm::Value* index);
359
360 void EmitGuard_ArrayException(uint32_t dex_pc,
361 llvm::Value* array,
362 llvm::Value* index);
363
Logan Chiena78e3c82011-12-27 17:59:35 +0800364 RegCategory GetInferredRegCategory(uint32_t dex_pc, uint16_t reg);
365
Logan Chien2aeca0b2012-02-03 22:55:46 +0800366 Method* ResolveMethod(uint32_t method_idx);
367
Logan Chien438c4b62012-01-17 16:06:00 +0800368 Field* ResolveField(uint32_t field_idx);
369
370 Field* FindFieldAndDeclaringTypeIdx(uint32_t field_idx,
371 uint32_t &resolved_type_idx);
372
Logan Chiend6c239a2011-12-23 15:11:45 +0800373
Logan Chien48f1d2a2012-01-02 22:49:53 +0800374 // Diagnostics helper function
375 void PrintUnresolvedFieldWarning(int32_t field_idx);
376
377
Logan Chiend6c239a2011-12-23 15:11:45 +0800378 // Basic block helper functions
379 llvm::BasicBlock* GetBasicBlock(uint32_t dex_pc);
380
381 llvm::BasicBlock* GetNextBasicBlock(uint32_t dex_pc);
382
383 llvm::BasicBlock* CreateBasicBlockWithDexPC(uint32_t dex_pc,
384 char const* postfix = NULL);
385
Logan Chien5bcc04e2012-01-30 14:15:12 +0800386 int32_t GetTryItemOffset(uint32_t dex_pc);
387
388 llvm::BasicBlock* GetLandingPadBasicBlock(uint32_t dex_pc);
389
390 llvm::BasicBlock* GetUnwindBasicBlock();
391
Logan Chienc670a8d2011-12-20 21:25:56 +0800392
393 // Register helper function
394
395 llvm::Value* EmitLoadDalvikReg(uint32_t reg_idx, JType jty,
396 JTypeSpace space) {
397 return regs_[reg_idx]->GetValue(jty, space);
398 }
399
400 llvm::Value* EmitLoadDalvikReg(uint32_t reg_idx, char shorty,
401 JTypeSpace space) {
402 return EmitLoadDalvikReg(reg_idx, GetJTypeFromShorty(shorty), space);
403 }
404
405 void EmitStoreDalvikReg(uint32_t reg_idx, JType jty,
406 JTypeSpace space, llvm::Value* new_value) {
407 regs_[reg_idx]->SetValue(jty, space, new_value);
408 }
409
410 void EmitStoreDalvikReg(uint32_t reg_idx, char shorty,
411 JTypeSpace space, llvm::Value* new_value) {
412 EmitStoreDalvikReg(reg_idx, GetJTypeFromShorty(shorty), space, new_value);
413 }
414
415 llvm::Value* EmitLoadDalvikRetValReg(JType jty, JTypeSpace space) {
416 return retval_reg_->GetValue(jty, space);
417 }
418
419 llvm::Value* EmitLoadDalvikRetValReg(char shorty, JTypeSpace space) {
420 return EmitLoadDalvikRetValReg(GetJTypeFromShorty(shorty), space);
421 }
422
423 void EmitStoreDalvikRetValReg(JType jty, JTypeSpace space,
424 llvm::Value* new_value) {
425 retval_reg_->SetValue(jty, space, new_value);
426 }
427
428 void EmitStoreDalvikRetValReg(char shorty, JTypeSpace space,
429 llvm::Value* new_value) {
430 EmitStoreDalvikRetValReg(GetJTypeFromShorty(shorty), space, new_value);
431 }
Logan Chien50aaf812012-03-01 21:03:24 +0800432
433
434 private:
435 CompilationUnit* cunit_;
436 Compiler* compiler_;
437
438 ClassLinker* class_linker_;
439 ClassLoader const* class_loader_;
440
441 DexFile const* dex_file_;
442 DexCache* dex_cache_;
443 DexFile::CodeItem const* code_item_;
444
445 OatCompilationUnit* oat_compilation_unit_;
446 Method* method_;
447 MethodHelper method_helper_;
448
449 uint32_t method_idx_;
450 uint32_t access_flags_;
451
452 llvm::Module* module_;
453 llvm::LLVMContext* context_;
454 IRBuilder& irb_;
455 llvm::Function* func_;
456
457 std::vector<DalvikReg*> regs_;
458 UniquePtr<DalvikReg> retval_reg_;
459
TDYa1274165a832012-04-03 17:47:16 -0700460 llvm::BasicBlock* basic_block_stack_overflow_;
Logan Chien50aaf812012-03-01 21:03:24 +0800461 llvm::BasicBlock* basic_block_reg_alloca_;
462 llvm::BasicBlock* basic_block_shadow_frame_alloca_;
463 llvm::BasicBlock* basic_block_reg_zero_init_;
464 llvm::BasicBlock* basic_block_reg_arg_init_;
465 std::vector<llvm::BasicBlock*> basic_blocks_;
466
467 std::vector<llvm::BasicBlock*> basic_block_landing_pads_;
468 llvm::BasicBlock* basic_block_unwind_;
469 llvm::BasicBlock* basic_block_unreachable_;
470
471 llvm::AllocaInst* shadow_frame_;
Logan Chien937105a2012-04-02 02:37:37 +0800472
473 uint16_t elf_func_idx_;
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800474};
475
476
477} // namespace compiler_llvm
478} // namespace art
479
480#endif // ART_SRC_COMPILER_LLVM_METHOD_COMPILER_H_