blob: dd10bc8d42f3bd8576682a93eac99c9e1c2ded20 [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"
Shih-wei Liaod1fec812012-02-13 09:51:10 -080021#include "constants.h"
Logan Chienc670a8d2011-12-20 21:25:56 +080022#include "dalvik_reg.h"
Shih-wei Liaod1fec812012-02-13 09:51:10 -080023#include "dex_file.h"
24#include "dex_instruction.h"
25#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;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080041}
42
43
44namespace llvm {
45 class AllocaInst;
46 class BasicBlock;
47 class Function;
48 class FunctionType;
49 class LLVMContext;
50 class Module;
51 class Type;
52}
53
54
55namespace art {
56namespace compiler_llvm {
57
58class CompilerLLVM;
59class IRBuilder;
60
61class MethodCompiler {
62 private:
Logan Chien83426162011-12-09 09:29:50 +080063 InstructionSet insn_set_;
64 Compiler* compiler_;
65 compiler_llvm::CompilerLLVM* compiler_llvm_;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080066
Logan Chien83426162011-12-09 09:29:50 +080067 ClassLinker* class_linker_;
68 ClassLoader const* class_loader_;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080069
Logan Chien83426162011-12-09 09:29:50 +080070 DexFile const* dex_file_;
71 DexCache* dex_cache_;
72 DexFile::CodeItem const* code_item_;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080073
Logan Chien83426162011-12-09 09:29:50 +080074 Method* method_;
75 MethodHelper method_helper_;
Shih-wei Liaod1fec812012-02-13 09:51:10 -080076
77 uint32_t method_idx_;
78 uint32_t access_flags_;
79
80 llvm::Module* module_;
81 llvm::LLVMContext* context_;
82 IRBuilder& irb_;
83 llvm::Function* func_;
84
Logan Chienc670a8d2011-12-20 21:25:56 +080085 std::vector<DalvikReg*> regs_;
86 UniquePtr<DalvikReg> retval_reg_;
Logan Chiend6c239a2011-12-23 15:11:45 +080087
Logan Chienc670a8d2011-12-20 21:25:56 +080088 llvm::BasicBlock* basic_block_reg_alloca_;
Logan Chien8dfcbea2012-02-17 18:50:32 +080089 llvm::BasicBlock* basic_block_shadow_frame_alloca_;
Logan Chienc670a8d2011-12-20 21:25:56 +080090 llvm::BasicBlock* basic_block_reg_zero_init_;
Logan Chiend6ececa2011-12-27 16:20:15 +080091 llvm::BasicBlock* basic_block_reg_arg_init_;
Logan Chiend6c239a2011-12-23 15:11:45 +080092 std::vector<llvm::BasicBlock*> basic_blocks_;
93
Logan Chien5bcc04e2012-01-30 14:15:12 +080094 std::vector<llvm::BasicBlock*> basic_block_landing_pads_;
95 llvm::BasicBlock* basic_block_unwind_;
96 llvm::BasicBlock* basic_block_unreachable_;
97
Logan Chien8dfcbea2012-02-17 18:50:32 +080098 llvm::AllocaInst* shadow_frame_;
99
Logan Chiend6c239a2011-12-23 15:11:45 +0800100
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800101 public:
Logan Chien83426162011-12-09 09:29:50 +0800102 MethodCompiler(InstructionSet insn_set,
103 Compiler* compiler,
104 ClassLinker* class_linker,
105 ClassLoader const* class_loader,
106 DexFile const* dex_file,
107 DexCache* dex_cache,
108 DexFile::CodeItem const* code_item,
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800109 uint32_t method_idx,
110 uint32_t access_flags);
111
112 ~MethodCompiler();
113
Logan Chien83426162011-12-09 09:29:50 +0800114 CompiledMethod* Compile();
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800115
Logan Chienc670a8d2011-12-20 21:25:56 +0800116
117 // Code generation helper function
118
119 IRBuilder& GetIRBuilder() const {
120 return irb_;
121 }
122
123
124 // Register helper function
125
126 llvm::Value* AllocDalvikLocalVarReg(RegCategory cat, uint32_t reg_idx);
127
128 llvm::Value* AllocDalvikRetValReg(RegCategory cat);
129
130
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800131 private:
132 void CreateFunction();
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800133 void EmitPrologue();
Logan Chienc670a8d2011-12-20 21:25:56 +0800134 void EmitPrologueLastBranch();
Logan Chien8dfcbea2012-02-17 18:50:32 +0800135 void EmitPrologueAllocShadowFrame();
Logan Chiend6ececa2011-12-27 16:20:15 +0800136 void EmitPrologueAssignArgRegister();
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800137 void EmitInstructions();
Logan Chien83426162011-12-09 09:29:50 +0800138 void EmitInstruction(uint32_t dex_pc, Instruction const* insn);
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800139
Logan Chien70f94b42011-12-27 17:49:11 +0800140 enum CondBranchKind {
141 kCondBranch_EQ,
142 kCondBranch_NE,
143 kCondBranch_LT,
144 kCondBranch_GE,
145 kCondBranch_GT,
146 kCondBranch_LE,
147 };
148
149 enum IntArithmKind {
150 kIntArithm_Add,
151 kIntArithm_Sub,
152 kIntArithm_Mul,
153 kIntArithm_Div,
154 kIntArithm_Rem,
155 kIntArithm_And,
156 kIntArithm_Or,
157 kIntArithm_Xor,
158 kIntArithm_Shl,
159 kIntArithm_Shr,
160 kIntArithm_UShr,
161 };
162
163 enum FPArithmKind {
164 kFPArithm_Add,
165 kFPArithm_Sub,
166 kFPArithm_Mul,
167 kFPArithm_Div,
168 kFPArithm_Rem,
169 };
170
171#define GEN_INSN_ARGS uint32_t dex_pc, Instruction const* insn
172
173 // NOP, PAYLOAD (unreachable) instructions
174 void EmitInsn_Nop(GEN_INSN_ARGS);
175
176 // MOVE, MOVE_RESULT instructions
177 void EmitInsn_Move(GEN_INSN_ARGS, JType jty);
178 void EmitInsn_MoveResult(GEN_INSN_ARGS, JType jty);
179
180 // MOVE_EXCEPTION, THROW instructions
181 void EmitInsn_MoveException(GEN_INSN_ARGS);
182 void EmitInsn_ThrowException(GEN_INSN_ARGS);
183
184 // RETURN instructions
185 void EmitInsn_ReturnVoid(GEN_INSN_ARGS);
186 void EmitInsn_Return(GEN_INSN_ARGS);
187
188 // CONST, CONST_CLASS, CONST_STRING instructions
189 void EmitInsn_LoadConstant(GEN_INSN_ARGS, JType imm_jty);
190 void EmitInsn_LoadConstantString(GEN_INSN_ARGS);
191 void EmitInsn_LoadConstantClass(GEN_INSN_ARGS);
192
193 // MONITOR_ENTER, MONITOR_EXIT instructions
194 void EmitInsn_MonitorEnter(GEN_INSN_ARGS);
195 void EmitInsn_MonitorExit(GEN_INSN_ARGS);
196
197 // CHECK_CAST, INSTANCE_OF instructions
198 void EmitInsn_CheckCast(GEN_INSN_ARGS);
199 void EmitInsn_InstanceOf(GEN_INSN_ARGS);
200
201 // NEW_INSTANCE instructions
202 void EmitInsn_NewInstance(GEN_INSN_ARGS);
203
204 // ARRAY_LEN, NEW_ARRAY, FILLED_NEW_ARRAY, FILL_ARRAY_DATA instructions
205 void EmitInsn_ArrayLength(GEN_INSN_ARGS);
206 void EmitInsn_NewArray(GEN_INSN_ARGS);
207 void EmitInsn_FilledNewArray(GEN_INSN_ARGS, bool is_range);
208 void EmitInsn_FillArrayData(GEN_INSN_ARGS);
209
210 // GOTO, IF_TEST, IF_TESTZ instructions
211 void EmitInsn_UnconditionalBranch(GEN_INSN_ARGS);
212 void EmitInsn_BinaryConditionalBranch(GEN_INSN_ARGS, CondBranchKind cond);
213 void EmitInsn_UnaryConditionalBranch(GEN_INSN_ARGS, CondBranchKind cond);
214
215 // PACKED_SWITCH, SPARSE_SWITCH instrutions
216 void EmitInsn_PackedSwitch(GEN_INSN_ARGS);
217 void EmitInsn_SparseSwitch(GEN_INSN_ARGS);
218
219 // CMPX_FLOAT, CMPX_DOUBLE, CMP_LONG instructions
220 void EmitInsn_FPCompare(GEN_INSN_ARGS, JType fp_jty, bool gt_bias);
221 void EmitInsn_LongCompare(GEN_INSN_ARGS);
222
223 // AGET, APUT instrutions
224 void EmitInsn_AGet(GEN_INSN_ARGS, JType elem_jty);
225 void EmitInsn_APut(GEN_INSN_ARGS, JType elem_jty);
226
227 // IGET, IPUT instructions
228 void EmitInsn_IGet(GEN_INSN_ARGS, JType field_jty);
229 void EmitInsn_IPut(GEN_INSN_ARGS, JType field_jty);
230
231 // SGET, SPUT instructions
232 void EmitInsn_SGet(GEN_INSN_ARGS, JType field_jty);
233 void EmitInsn_SPut(GEN_INSN_ARGS, JType field_jty);
234
235 // INVOKE instructions
236 void EmitInsn_InvokeVirtual(GEN_INSN_ARGS, bool is_range);
237 void EmitInsn_InvokeSuper(GEN_INSN_ARGS, bool is_range);
Logan Chien1a121b92012-02-15 22:23:42 +0800238 void EmitInsn_InvokeStaticDirect(GEN_INSN_ARGS,
239 bool is_range,
240 bool is_static);
Logan Chien70f94b42011-12-27 17:49:11 +0800241 void EmitInsn_InvokeInterface(GEN_INSN_ARGS, bool is_range);
242
243 // Unary instructions
244 void EmitInsn_Neg(GEN_INSN_ARGS, JType op_jty);
245 void EmitInsn_Not(GEN_INSN_ARGS, JType op_jty);
246 void EmitInsn_SExt(GEN_INSN_ARGS);
247 void EmitInsn_Trunc(GEN_INSN_ARGS);
248 void EmitInsn_TruncAndSExt(GEN_INSN_ARGS, unsigned N);
249 void EmitInsn_TruncAndZExt(GEN_INSN_ARGS, unsigned N);
250
251 void EmitInsn_FNeg(GEN_INSN_ARGS, JType op_jty);
252 void EmitInsn_IntToFP(GEN_INSN_ARGS, JType src_jty, JType dest_jty);
253 void EmitInsn_FPToInt(GEN_INSN_ARGS, JType src_jty, JType dest_jty);
254 void EmitInsn_FExt(GEN_INSN_ARGS);
255 void EmitInsn_FTrunc(GEN_INSN_ARGS);
256
257 // Integer binary arithmetic instructions
258 void EmitInsn_IntArithm(GEN_INSN_ARGS, IntArithmKind arithm,
259 JType op_jty, bool is_2addr);
260
261 void EmitInsn_IntArithmImmediate(GEN_INSN_ARGS, IntArithmKind arithm);
262
263 void EmitInsn_RSubImmediate(GEN_INSN_ARGS);
264
265
266 // Floating-point binary arithmetic instructions
267 void EmitInsn_FPArithm(GEN_INSN_ARGS, FPArithmKind arithm,
268 JType op_jty, bool is_2addr);
269
270#undef GEN_INSN_ARGS
271
Logan Chien0b827102011-12-20 19:46:14 +0800272
Logan Chien8dfcbea2012-02-17 18:50:32 +0800273 // Shadow frame helper function
274 void EmitPopShadowFrame();
275 void EmitUpdateLineNum(int32_t line_number);
276 void EmitUpdateLineNumFromDexPC(uint32_t dex_pc);
277
278
Logan Chienbb4d12a2012-02-17 14:10:01 +0800279 // Dex cache code generation helper function
280 llvm::Value* EmitLoadDexCacheAddr(MemberOffset dex_cache_offset);
281
282 void EmitLoadDexCacheCodeAndDirectMethodFieldAddr(
283 llvm::Value*& code_addr_field_addr,
284 llvm::Value*& method_field_addr,
285 uint32_t method_idx);
286
287 llvm::Value* EmitLoadDexCacheStaticStorageFieldAddr(uint32_t type_idx);
288
289 llvm::Value* EmitLoadDexCacheResolvedTypeFieldAddr(uint32_t type_idx);
290
291 llvm::Value* EmitLoadDexCacheStringFieldAddr(uint32_t string_idx);
292
293
Logan Chien0b827102011-12-20 19:46:14 +0800294 // Code generation helper function
295
296 llvm::Value* EmitLoadMethodObjectAddr();
297
298 llvm::FunctionType* GetFunctionType(uint32_t method_idx, bool is_static);
299
Logan Chien5bcc04e2012-01-30 14:15:12 +0800300 void EmitGuard_ExceptionLandingPad(uint32_t dex_pc);
301
302 void EmitBranchExceptionLandingPad(uint32_t dex_pc);
303
Logan Chien70f94b42011-12-27 17:49:11 +0800304 void EmitGuard_GarbageCollectionSuspend(uint32_t dex_pc);
Logan Chien924072f2012-01-30 15:07:24 +0800305
Logan Chien2c37e8e2011-12-27 17:58:46 +0800306 llvm::Value* EmitCompareResultSelection(llvm::Value* cmp_eq,
307 llvm::Value* cmp_lt);
308
Logan Chiena78e3c82011-12-27 17:59:35 +0800309 llvm::Value* EmitConditionResult(llvm::Value* lhs,
310 llvm::Value* rhs,
311 CondBranchKind cond);
312
Logan Chienc3f7d962011-12-27 18:13:18 +0800313 llvm::Value* EmitIntArithmResultComputation(uint32_t dex_pc,
314 llvm::Value* lhs,
315 llvm::Value* rhs,
316 IntArithmKind arithm,
317 JType op_jty);
318
Logan Chien76e1c792011-12-27 18:15:01 +0800319 llvm::Value* EmitFPArithmResultComputation(uint32_t dex_pc,
320 llvm::Value* lhs,
321 llvm::Value* rhs,
322 FPArithmKind arithm);
323
Logan Chiena2cc6a32012-01-16 10:38:41 +0800324 llvm::Value* EmitAllocNewArray(uint32_t dex_pc,
325 int32_t length,
326 uint32_t type_idx,
327 bool is_filled_new_array);
328
Logan Chien46fbb412012-02-15 22:29:08 +0800329 llvm::Value* EmitLoadClassObjectAddr(llvm::Value* this_addr);
330
331 llvm::Value* EmitLoadVTableAddr(llvm::Value* class_object_addr);
332
333 llvm::Value* EmitLoadMethodObjectAddrFromVTable(llvm::Value* vtable_addr,
334 uint16_t vtable_index);
335
336 llvm::Value* EmitLoadCodeAddr(llvm::Value* method_object_addr,
337 uint32_t method_idx,
338 bool is_static);
339
Logan Chien61bb6142012-02-03 15:34:53 +0800340 llvm::Value* EmitLoadArrayLength(llvm::Value* array);
341
Logan Chiene27fdbb2012-01-02 23:27:26 +0800342 llvm::Value* EmitArrayGEP(llvm::Value* array_addr,
343 llvm::Value* index_value,
344 llvm::Type* elem_type);
345
Logan Chien27b30252012-01-14 03:43:35 +0800346 llvm::Value* EmitLoadConstantClass(uint32_t dex_pc, uint32_t type_idx);
347
Logan Chien438c4b62012-01-17 16:06:00 +0800348 llvm::Value* EmitLoadStaticStorage(uint32_t dex_pc, uint32_t type_idx);
349
Logan Chien1a121b92012-02-15 22:23:42 +0800350 llvm::Value* EmitLoadCalleeThis(Instruction::DecodedInstruction const& di,
351 bool is_range);
352
353 void EmitLoadActualParameters(std::vector<llvm::Value*>& args,
354 uint32_t callee_method_idx,
355 Instruction::DecodedInstruction const& di,
356 bool is_range,
357 bool is_static);
358
Logan Chienc3f7d962011-12-27 18:13:18 +0800359 void EmitGuard_DivZeroException(uint32_t dex_pc,
360 llvm::Value* denominator,
361 JType op_jty);
362
Logan Chien61bb6142012-02-03 15:34:53 +0800363 void EmitGuard_NullPointerException(uint32_t dex_pc,
364 llvm::Value* object);
365
Logan Chiene27fdbb2012-01-02 23:27:26 +0800366 void EmitGuard_ArrayIndexOutOfBoundsException(uint32_t dex_pc,
367 llvm::Value* array,
368 llvm::Value* index);
369
370 void EmitGuard_ArrayException(uint32_t dex_pc,
371 llvm::Value* array,
372 llvm::Value* index);
373
Logan Chiena78e3c82011-12-27 17:59:35 +0800374 RegCategory GetInferredRegCategory(uint32_t dex_pc, uint16_t reg);
375
Logan Chien2aeca0b2012-02-03 22:55:46 +0800376 Method* ResolveMethod(uint32_t method_idx);
377
Logan Chien438c4b62012-01-17 16:06:00 +0800378 Field* ResolveField(uint32_t field_idx);
379
380 Field* FindFieldAndDeclaringTypeIdx(uint32_t field_idx,
381 uint32_t &resolved_type_idx);
382
Logan Chiend6c239a2011-12-23 15:11:45 +0800383
Logan Chien48f1d2a2012-01-02 22:49:53 +0800384 // Diagnostics helper function
385 void PrintUnresolvedFieldWarning(int32_t field_idx);
386
387
Logan Chiend6c239a2011-12-23 15:11:45 +0800388 // Basic block helper functions
389 llvm::BasicBlock* GetBasicBlock(uint32_t dex_pc);
390
391 llvm::BasicBlock* GetNextBasicBlock(uint32_t dex_pc);
392
393 llvm::BasicBlock* CreateBasicBlockWithDexPC(uint32_t dex_pc,
394 char const* postfix = NULL);
395
Logan Chien5bcc04e2012-01-30 14:15:12 +0800396 int32_t GetTryItemOffset(uint32_t dex_pc);
397
398 llvm::BasicBlock* GetLandingPadBasicBlock(uint32_t dex_pc);
399
400 llvm::BasicBlock* GetUnwindBasicBlock();
401
Logan Chienc670a8d2011-12-20 21:25:56 +0800402
403 // Register helper function
404
405 llvm::Value* EmitLoadDalvikReg(uint32_t reg_idx, JType jty,
406 JTypeSpace space) {
407 return regs_[reg_idx]->GetValue(jty, space);
408 }
409
410 llvm::Value* EmitLoadDalvikReg(uint32_t reg_idx, char shorty,
411 JTypeSpace space) {
412 return EmitLoadDalvikReg(reg_idx, GetJTypeFromShorty(shorty), space);
413 }
414
415 void EmitStoreDalvikReg(uint32_t reg_idx, JType jty,
416 JTypeSpace space, llvm::Value* new_value) {
417 regs_[reg_idx]->SetValue(jty, space, new_value);
418 }
419
420 void EmitStoreDalvikReg(uint32_t reg_idx, char shorty,
421 JTypeSpace space, llvm::Value* new_value) {
422 EmitStoreDalvikReg(reg_idx, GetJTypeFromShorty(shorty), space, new_value);
423 }
424
425 llvm::Value* EmitLoadDalvikRetValReg(JType jty, JTypeSpace space) {
426 return retval_reg_->GetValue(jty, space);
427 }
428
429 llvm::Value* EmitLoadDalvikRetValReg(char shorty, JTypeSpace space) {
430 return EmitLoadDalvikRetValReg(GetJTypeFromShorty(shorty), space);
431 }
432
433 void EmitStoreDalvikRetValReg(JType jty, JTypeSpace space,
434 llvm::Value* new_value) {
435 retval_reg_->SetValue(jty, space, new_value);
436 }
437
438 void EmitStoreDalvikRetValReg(char shorty, JTypeSpace space,
439 llvm::Value* new_value) {
440 EmitStoreDalvikRetValReg(GetJTypeFromShorty(shorty), space, new_value);
441 }
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800442};
443
444
445} // namespace compiler_llvm
446} // namespace art
447
448#endif // ART_SRC_COMPILER_LLVM_METHOD_COMPILER_H_