blob: e2b828267515880bf8479cc011208e3b3a82dc93 [file] [log] [blame]
buzbee31a4a6f2012-02-28 15:36:15 -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
Ian Rogers57b86d42012-03-27 16:05:41 -070017#include "oat/runtime/oat_support_entrypoints.h"
18
buzbee31a4a6f2012-02-28 15:36:15 -080019namespace art {
20
21/*
22 * This source files contains "gen" codegen routines that should
23 * be applicable to most targets. Only mid-level support utilities
24 * and "op" calls may be used here.
25 */
buzbeefc9e6fa2012-03-23 15:14:29 -070026void genInvoke(CompilationUnit* cUnit, BasicBlock* bb, MIR* mir,
27 InvokeType type, bool isRange);
buzbee31a4a6f2012-02-28 15:36:15 -080028#if defined(TARGET_ARM)
buzbee82488f52012-03-02 08:20:26 -080029LIR* opIT(CompilationUnit* cUnit, ArmConditionCode cond, const char* guide);
buzbeef3aac972012-04-11 16:33:36 -070030bool smallLiteralDivide(CompilationUnit* cUnit, Instruction::Code dalvikOpcode,
31 RegLocation rlSrc, RegLocation rlDest, int lit);
buzbee31a4a6f2012-02-28 15:36:15 -080032#endif
33
Ian Rogersab2b55d2012-03-18 00:06:11 -070034void callRuntimeHelperImm(CompilationUnit* cUnit, int helperOffset, int arg0) {
35#if !defined(TARGET_X86)
36 int rTgt = loadHelper(cUnit, helperOffset);
37#endif
38 loadConstant(cUnit, rARG0, arg0);
buzbee31a4a6f2012-02-28 15:36:15 -080039 oatClobberCalleeSave(cUnit);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -070040#if !defined(TARGET_X86)
Ian Rogersab2b55d2012-03-18 00:06:11 -070041 opReg(cUnit, kOpBlx, rTgt);
42 oatFreeTemp(cUnit, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -070043#else
Ian Rogersab2b55d2012-03-18 00:06:11 -070044 opThreadMem(cUnit, kOpBlx, helperOffset);
45#endif
46}
47
Ian Rogers7caad772012-03-30 01:07:54 -070048void callRuntimeHelperReg(CompilationUnit* cUnit, int helperOffset, int arg0) {
49#if !defined(TARGET_X86)
50 int rTgt = loadHelper(cUnit, helperOffset);
51#endif
52 opRegCopy(cUnit, rARG0, arg0);
53 oatClobberCalleeSave(cUnit);
54#if !defined(TARGET_X86)
55 opReg(cUnit, kOpBlx, rTgt);
56 oatFreeTemp(cUnit, rTgt);
57#else
58 opThreadMem(cUnit, kOpBlx, helperOffset);
59#endif
60}
61
Ian Rogersab2b55d2012-03-18 00:06:11 -070062void callRuntimeHelperRegLocation(CompilationUnit* cUnit, int helperOffset,
63 RegLocation arg0) {
64#if !defined(TARGET_X86)
65 int rTgt = loadHelper(cUnit, helperOffset);
66#endif
67 if (arg0.wide == 0) {
68 loadValueDirectFixed(cUnit, arg0, rARG0);
69 } else {
70 loadValueDirectWideFixed(cUnit, arg0, rARG0, rARG1);
71 }
72 oatClobberCalleeSave(cUnit);
73#if !defined(TARGET_X86)
74 opReg(cUnit, kOpBlx, rTgt);
75 oatFreeTemp(cUnit, rTgt);
76#else
77 opThreadMem(cUnit, kOpBlx, helperOffset);
78#endif
79}
80
81void callRuntimeHelperImmImm(CompilationUnit* cUnit, int helperOffset,
82 int arg0, int arg1) {
83#if !defined(TARGET_X86)
84 int rTgt = loadHelper(cUnit, helperOffset);
85#endif
86 loadConstant(cUnit, rARG0, arg0);
87 loadConstant(cUnit, rARG1, arg1);
88 oatClobberCalleeSave(cUnit);
89#if !defined(TARGET_X86)
90 opReg(cUnit, kOpBlx, rTgt);
91 oatFreeTemp(cUnit, rTgt);
92#else
93 opThreadMem(cUnit, kOpBlx, helperOffset);
94#endif
95}
96
97void callRuntimeHelperImmRegLocation(CompilationUnit* cUnit, int helperOffset,
98 int arg0, RegLocation arg1) {
99#if !defined(TARGET_X86)
100 int rTgt = loadHelper(cUnit, helperOffset);
101#endif
102 if (arg1.wide == 0) {
103 loadValueDirectFixed(cUnit, arg1, rARG1);
104 } else {
105 loadValueDirectWideFixed(cUnit, arg1, rARG1, rARG2);
106 }
107 loadConstant(cUnit, rARG0, arg0);
108 oatClobberCalleeSave(cUnit);
109#if !defined(TARGET_X86)
110 opReg(cUnit, kOpBlx, rTgt);
111 oatFreeTemp(cUnit, rTgt);
112#else
113 opThreadMem(cUnit, kOpBlx, helperOffset);
114#endif
115}
116
117void callRuntimeHelperRegLocationImm(CompilationUnit* cUnit, int helperOffset,
118 RegLocation arg0, int arg1) {
119#if !defined(TARGET_X86)
120 int rTgt = loadHelper(cUnit, helperOffset);
121#endif
122 loadValueDirectFixed(cUnit, arg0, rARG0);
123 loadConstant(cUnit, rARG1, arg1);
124 oatClobberCalleeSave(cUnit);
125#if !defined(TARGET_X86)
126 opReg(cUnit, kOpBlx, rTgt);
127 oatFreeTemp(cUnit, rTgt);
128#else
129 opThreadMem(cUnit, kOpBlx, helperOffset);
130#endif
131}
132
133void callRuntimeHelperImmReg(CompilationUnit* cUnit, int helperOffset,
134 int arg0, int arg1) {
135#if !defined(TARGET_X86)
136 int rTgt = loadHelper(cUnit, helperOffset);
137#endif
138 opRegCopy(cUnit, rARG1, arg1);
139 loadConstant(cUnit, rARG0, arg0);
140 oatClobberCalleeSave(cUnit);
141#if !defined(TARGET_X86)
142 opReg(cUnit, kOpBlx, rTgt);
143 oatFreeTemp(cUnit, rTgt);
144#else
145 opThreadMem(cUnit, kOpBlx, helperOffset);
146#endif
147}
148
149void callRuntimeHelperRegImm(CompilationUnit* cUnit, int helperOffset,
150 int arg0, int arg1) {
151#if !defined(TARGET_X86)
152 int rTgt = loadHelper(cUnit, helperOffset);
153#endif
154 opRegCopy(cUnit, rARG0, arg0);
155 loadConstant(cUnit, rARG1, arg1);
156 oatClobberCalleeSave(cUnit);
157#if !defined(TARGET_X86)
158 opReg(cUnit, kOpBlx, rTgt);
159 oatFreeTemp(cUnit, rTgt);
160#else
161 opThreadMem(cUnit, kOpBlx, helperOffset);
162#endif
163}
164
165void callRuntimeHelperImmMethod(CompilationUnit* cUnit, int helperOffset,
166 int arg0) {
167#if !defined(TARGET_X86)
168 int rTgt = loadHelper(cUnit, helperOffset);
169#endif
170 loadCurrMethodDirect(cUnit, rARG1);
171 loadConstant(cUnit, rARG0, arg0);
172 oatClobberCalleeSave(cUnit);
173#if !defined(TARGET_X86)
174 opReg(cUnit, kOpBlx, rTgt);
175 oatFreeTemp(cUnit, rTgt);
176#else
177 opThreadMem(cUnit, kOpBlx, helperOffset);
178#endif
179}
180
181void callRuntimeHelperRegLocationRegLocation(CompilationUnit* cUnit,
182 int helperOffset,
183 RegLocation arg0,
184 RegLocation arg1) {
185#if !defined(TARGET_X86)
186 int rTgt = loadHelper(cUnit, helperOffset);
187#endif
188 if (arg0.wide == 0) {
189 loadValueDirectFixed(cUnit, arg0, rARG0);
190 if (arg1.wide == 0) {
191 loadValueDirectFixed(cUnit, arg1, rARG1);
192 } else {
193 loadValueDirectWideFixed(cUnit, arg1, rARG1, rARG2);
194 }
195 } else {
196 loadValueDirectWideFixed(cUnit, arg0, rARG0, rARG1);
197 if (arg1.wide == 0) {
198 loadValueDirectFixed(cUnit, arg1, rARG2);
199 } else {
Ian Rogersab2b55d2012-03-18 00:06:11 -0700200 loadValueDirectWideFixed(cUnit, arg1, rARG2, rARG3);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700201 }
202 }
203 oatClobberCalleeSave(cUnit);
204#if !defined(TARGET_X86)
205 opReg(cUnit, kOpBlx, rTgt);
206 oatFreeTemp(cUnit, rTgt);
207#else
208 opThreadMem(cUnit, kOpBlx, helperOffset);
209#endif
210}
211
212void callRuntimeHelperRegReg(CompilationUnit* cUnit, int helperOffset,
213 int arg0, int arg1) {
214#if !defined(TARGET_X86)
215 int rTgt = loadHelper(cUnit, helperOffset);
216#endif
217 DCHECK_NE((int)rARG0, arg1); // check copy into arg0 won't clobber arg1
218 opRegCopy(cUnit, rARG0, arg0);
219 opRegCopy(cUnit, rARG1, arg1);
220 oatClobberCalleeSave(cUnit);
221#if !defined(TARGET_X86)
222 opReg(cUnit, kOpBlx, rTgt);
223 oatFreeTemp(cUnit, rTgt);
224#else
225 opThreadMem(cUnit, kOpBlx, helperOffset);
226#endif
227}
228
229void callRuntimeHelperRegRegImm(CompilationUnit* cUnit, int helperOffset,
230 int arg0, int arg1, int arg2) {
231#if !defined(TARGET_X86)
232 int rTgt = loadHelper(cUnit, helperOffset);
233#endif
234 DCHECK_NE((int)rARG0, arg1); // check copy into arg0 won't clobber arg1
235 opRegCopy(cUnit, rARG0, arg0);
236 opRegCopy(cUnit, rARG1, arg1);
237 loadConstant(cUnit, rARG2, arg2);
238 oatClobberCalleeSave(cUnit);
239#if !defined(TARGET_X86)
240 opReg(cUnit, kOpBlx, rTgt);
241 oatFreeTemp(cUnit, rTgt);
242#else
243 opThreadMem(cUnit, kOpBlx, helperOffset);
244#endif
245}
246
247void callRuntimeHelperImmMethodRegLocation(CompilationUnit* cUnit, int helperOffset,
248 int arg0, RegLocation arg2) {
249#if !defined(TARGET_X86)
250 int rTgt = loadHelper(cUnit, helperOffset);
251#endif
252 loadValueDirectFixed(cUnit, arg2, rARG2);
253 loadCurrMethodDirect(cUnit, rARG1);
254 loadConstant(cUnit, rARG0, arg0);
255 oatClobberCalleeSave(cUnit);
256#if !defined(TARGET_X86)
257 opReg(cUnit, kOpBlx, rTgt);
258 oatFreeTemp(cUnit, rTgt);
259#else
260 opThreadMem(cUnit, kOpBlx, helperOffset);
261#endif
262}
263
264void callRuntimeHelperImmMethodImm(CompilationUnit* cUnit, int helperOffset,
265 int arg0, int arg2) {
266#if !defined(TARGET_X86)
267 int rTgt = loadHelper(cUnit, helperOffset);
268#endif
269 loadCurrMethodDirect(cUnit, rARG1);
270 loadConstant(cUnit, rARG2, arg2);
271 loadConstant(cUnit, rARG0, arg0);
272 oatClobberCalleeSave(cUnit);
273#if !defined(TARGET_X86)
274 opReg(cUnit, kOpBlx, rTgt);
275 oatFreeTemp(cUnit, rTgt);
276#else
277 opThreadMem(cUnit, kOpBlx, helperOffset);
278#endif
279}
280
281void callRuntimeHelperImmRegLocationRegLocation(CompilationUnit* cUnit,
282 int helperOffset,
283 int arg0, RegLocation arg1,
284 RegLocation arg2) {
285#if !defined(TARGET_X86)
286 int rTgt = loadHelper(cUnit, helperOffset);
287#endif
288 loadValueDirectFixed(cUnit, arg1, rARG1);
289 if (arg2.wide == 0) {
290 loadValueDirectFixed(cUnit, arg2, rARG2);
291 } else {
Ian Rogersab2b55d2012-03-18 00:06:11 -0700292 loadValueDirectWideFixed(cUnit, arg2, rARG2, rARG3);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700293 }
294 loadConstant(cUnit, rARG0, arg0);
295 oatClobberCalleeSave(cUnit);
296#if !defined(TARGET_X86)
297 opReg(cUnit, kOpBlx, rTgt);
298 oatFreeTemp(cUnit, rTgt);
299#else
300 opThreadMem(cUnit, kOpBlx, helperOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700301#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800302}
303
304/*
305 * Generate an kPseudoBarrier marker to indicate the boundary of special
306 * blocks.
307 */
308void genBarrier(CompilationUnit* cUnit)
309{
310 LIR* barrier = newLIR0(cUnit, kPseudoBarrier);
311 /* Mark all resources as being clobbered */
312 barrier->defMask = -1;
313}
314
buzbee31a4a6f2012-02-28 15:36:15 -0800315
316/* Generate unconditional branch instructions */
buzbee82488f52012-03-02 08:20:26 -0800317LIR* opUnconditionalBranch(CompilationUnit* cUnit, LIR* target)
buzbee31a4a6f2012-02-28 15:36:15 -0800318{
Ian Rogers680b1bd2012-03-07 20:18:49 -0800319 LIR* branch = opBranchUnconditional(cUnit, kOpUncondBr);
buzbee31a4a6f2012-02-28 15:36:15 -0800320 branch->target = (LIR*) target;
321 return branch;
322}
323
buzbee5de34942012-03-01 14:51:57 -0800324// FIXME: need to do some work to split out targets with
325// condition codes and those without
326#if defined(TARGET_ARM) || defined(TARGET_X86)
buzbee31a4a6f2012-02-28 15:36:15 -0800327LIR* genCheck(CompilationUnit* cUnit, ConditionCode cCode, MIR* mir,
328 ThrowKind kind)
329{
buzbeea2ebdd72012-03-04 14:57:06 -0800330 LIR* tgt = rawLIR(cUnit, 0, kPseudoThrowTarget, kind,
331 mir ? mir->offset : 0);
buzbee82488f52012-03-02 08:20:26 -0800332 LIR* branch = opCondBranch(cUnit, cCode, tgt);
buzbee31a4a6f2012-02-28 15:36:15 -0800333 // Remember branch target - will process later
334 oatInsertGrowableList(cUnit, &cUnit->throwLaunchpads, (intptr_t)tgt);
335 return branch;
336}
buzbee5de34942012-03-01 14:51:57 -0800337#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800338
339LIR* genImmedCheck(CompilationUnit* cUnit, ConditionCode cCode,
340 int reg, int immVal, MIR* mir, ThrowKind kind)
341{
buzbeea2ebdd72012-03-04 14:57:06 -0800342 LIR* tgt = rawLIR(cUnit, 0, kPseudoThrowTarget, kind, mir->offset);
buzbee31a4a6f2012-02-28 15:36:15 -0800343 LIR* branch;
344 if (cCode == kCondAl) {
buzbee82488f52012-03-02 08:20:26 -0800345 branch = opUnconditionalBranch(cUnit, tgt);
buzbee31a4a6f2012-02-28 15:36:15 -0800346 } else {
buzbee82488f52012-03-02 08:20:26 -0800347 branch = opCmpImmBranch(cUnit, cCode, reg, immVal, tgt);
buzbee31a4a6f2012-02-28 15:36:15 -0800348 }
349 // Remember branch target - will process later
350 oatInsertGrowableList(cUnit, &cUnit->throwLaunchpads, (intptr_t)tgt);
351 return branch;
352}
353
354/* Perform null-check on a register. */
355LIR* genNullCheck(CompilationUnit* cUnit, int sReg, int mReg, MIR* mir)
356{
357 if (!(cUnit->disableOpt & (1 << kNullCheckElimination)) &&
358 mir->optimizationFlags & MIR_IGNORE_NULL_CHECK) {
359 return NULL;
360 }
361 return genImmedCheck(cUnit, kCondEq, mReg, 0, mir, kThrowNullPointer);
362}
363
364/* Perform check on two registers */
365LIR* genRegRegCheck(CompilationUnit* cUnit, ConditionCode cCode,
Ian Rogersb5d09b22012-03-06 22:14:17 -0800366 int reg1, int reg2, MIR* mir, ThrowKind kind)
buzbee31a4a6f2012-02-28 15:36:15 -0800367{
buzbeea2ebdd72012-03-04 14:57:06 -0800368 LIR* tgt = rawLIR(cUnit, 0, kPseudoThrowTarget, kind,
369 mir ? mir->offset : 0, reg1, reg2);
buzbee5de34942012-03-01 14:51:57 -0800370#if defined(TARGET_MIPS)
buzbee82488f52012-03-02 08:20:26 -0800371 LIR* branch = opCmpBranch(cUnit, cCode, reg1, reg2, tgt);
buzbee5de34942012-03-01 14:51:57 -0800372#else
buzbee31a4a6f2012-02-28 15:36:15 -0800373 opRegReg(cUnit, kOpCmp, reg1, reg2);
buzbee82488f52012-03-02 08:20:26 -0800374 LIR* branch = opCondBranch(cUnit, cCode, tgt);
buzbee5de34942012-03-01 14:51:57 -0800375#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800376 // Remember branch target - will process later
377 oatInsertGrowableList(cUnit, &cUnit->throwLaunchpads, (intptr_t)tgt);
378 return branch;
379}
380
381void genCompareAndBranch(CompilationUnit* cUnit, BasicBlock* bb, MIR* mir,
382 RegLocation rlSrc1, RegLocation rlSrc2, LIR* labelList)
383{
384 ConditionCode cond;
385 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
386 rlSrc2 = loadValue(cUnit, rlSrc2, kCoreReg);
Elliott Hughesadb8c672012-03-06 16:49:32 -0800387 Instruction::Code opcode = mir->dalvikInsn.opcode;
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700388 switch (opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800389 case Instruction::IF_EQ:
buzbee31a4a6f2012-02-28 15:36:15 -0800390 cond = kCondEq;
391 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800392 case Instruction::IF_NE:
buzbee31a4a6f2012-02-28 15:36:15 -0800393 cond = kCondNe;
394 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800395 case Instruction::IF_LT:
buzbee31a4a6f2012-02-28 15:36:15 -0800396 cond = kCondLt;
397 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800398 case Instruction::IF_GE:
buzbee31a4a6f2012-02-28 15:36:15 -0800399 cond = kCondGe;
400 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800401 case Instruction::IF_GT:
buzbee31a4a6f2012-02-28 15:36:15 -0800402 cond = kCondGt;
403 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800404 case Instruction::IF_LE:
buzbee31a4a6f2012-02-28 15:36:15 -0800405 cond = kCondLe;
406 break;
407 default:
408 cond = (ConditionCode)0;
409 LOG(FATAL) << "Unexpected opcode " << (int)opcode;
410 }
buzbee5de34942012-03-01 14:51:57 -0800411#if defined(TARGET_MIPS)
buzbee82488f52012-03-02 08:20:26 -0800412 opCmpBranch(cUnit, cond, rlSrc1.lowReg, rlSrc2.lowReg,
413 &labelList[bb->taken->id]);
buzbee5de34942012-03-01 14:51:57 -0800414#else
415 opRegReg(cUnit, kOpCmp, rlSrc1.lowReg, rlSrc2.lowReg);
buzbee82488f52012-03-02 08:20:26 -0800416 opCondBranch(cUnit, cond, &labelList[bb->taken->id]);
buzbee5de34942012-03-01 14:51:57 -0800417#endif
buzbee82488f52012-03-02 08:20:26 -0800418 opUnconditionalBranch(cUnit, &labelList[bb->fallThrough->id]);
buzbee31a4a6f2012-02-28 15:36:15 -0800419}
420
421void genCompareZeroAndBranch(CompilationUnit* cUnit, BasicBlock* bb, MIR* mir,
422 RegLocation rlSrc, LIR* labelList)
423{
424 ConditionCode cond;
425 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
Elliott Hughesadb8c672012-03-06 16:49:32 -0800426 Instruction::Code opcode = mir->dalvikInsn.opcode;
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700427 switch (opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800428 case Instruction::IF_EQZ:
buzbee31a4a6f2012-02-28 15:36:15 -0800429 cond = kCondEq;
430 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800431 case Instruction::IF_NEZ:
buzbee31a4a6f2012-02-28 15:36:15 -0800432 cond = kCondNe;
433 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800434 case Instruction::IF_LTZ:
buzbee31a4a6f2012-02-28 15:36:15 -0800435 cond = kCondLt;
436 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800437 case Instruction::IF_GEZ:
buzbee31a4a6f2012-02-28 15:36:15 -0800438 cond = kCondGe;
439 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800440 case Instruction::IF_GTZ:
buzbee31a4a6f2012-02-28 15:36:15 -0800441 cond = kCondGt;
442 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800443 case Instruction::IF_LEZ:
buzbee31a4a6f2012-02-28 15:36:15 -0800444 cond = kCondLe;
445 break;
446 default:
447 cond = (ConditionCode)0;
448 LOG(FATAL) << "Unexpected opcode " << (int)opcode;
449 }
Ian Rogers7caad772012-03-30 01:07:54 -0700450#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbee82488f52012-03-02 08:20:26 -0800451 opCmpImmBranch(cUnit, cond, rlSrc.lowReg, 0, &labelList[bb->taken->id]);
buzbee5de34942012-03-01 14:51:57 -0800452#else
453 opRegImm(cUnit, kOpCmp, rlSrc.lowReg, 0);
buzbee82488f52012-03-02 08:20:26 -0800454 opCondBranch(cUnit, cond, &labelList[bb->taken->id]);
buzbee5de34942012-03-01 14:51:57 -0800455#endif
buzbee82488f52012-03-02 08:20:26 -0800456 opUnconditionalBranch(cUnit, &labelList[bb->fallThrough->id]);
buzbee31a4a6f2012-02-28 15:36:15 -0800457}
458
459void genIntToLong(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
460 RegLocation rlSrc)
461{
462 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
463 if (rlSrc.location == kLocPhysReg) {
buzbee82488f52012-03-02 08:20:26 -0800464 opRegCopy(cUnit, rlResult.lowReg, rlSrc.lowReg);
buzbee31a4a6f2012-02-28 15:36:15 -0800465 } else {
466 loadValueDirect(cUnit, rlSrc, rlResult.lowReg);
467 }
468 opRegRegImm(cUnit, kOpAsr, rlResult.highReg,
469 rlResult.lowReg, 31);
470 storeValueWide(cUnit, rlDest, rlResult);
471}
472
473void genIntNarrowing(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
474 RegLocation rlSrc)
475{
476 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
477 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
478 OpKind op = kOpInvalid;
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700479 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800480 case Instruction::INT_TO_BYTE:
buzbee31a4a6f2012-02-28 15:36:15 -0800481 op = kOp2Byte;
482 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800483 case Instruction::INT_TO_SHORT:
buzbee31a4a6f2012-02-28 15:36:15 -0800484 op = kOp2Short;
485 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800486 case Instruction::INT_TO_CHAR:
buzbee31a4a6f2012-02-28 15:36:15 -0800487 op = kOp2Char;
488 break;
489 default:
490 LOG(ERROR) << "Bad int conversion type";
491 }
492 opRegReg(cUnit, op, rlResult.lowReg, rlSrc.lowReg);
493 storeValue(cUnit, rlDest, rlResult);
494}
495
496/*
497 * Let helper function take care of everything. Will call
498 * Array::AllocFromCode(type_idx, method, count);
499 * Note: AllocFromCode will handle checks for errNegativeArraySize.
500 */
501void genNewArray(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
502 RegLocation rlSrc)
503{
504 oatFlushAllRegs(cUnit); /* Everything to home location */
505 uint32_t type_idx = mir->dalvikInsn.vC;
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700506 int funcOffset;
buzbee31a4a6f2012-02-28 15:36:15 -0800507 if (cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
508 cUnit->dex_cache,
509 *cUnit->dex_file,
510 type_idx)) {
Ian Rogers57b86d42012-03-27 16:05:41 -0700511 funcOffset = ENTRYPOINT_OFFSET(pAllocArrayFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800512 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -0700513 funcOffset= ENTRYPOINT_OFFSET(pAllocArrayFromCodeWithAccessCheck);
buzbee31a4a6f2012-02-28 15:36:15 -0800514 }
Ian Rogersab2b55d2012-03-18 00:06:11 -0700515 callRuntimeHelperImmMethodRegLocation(cUnit, funcOffset, type_idx, rlSrc);
Ian Rogersf7d9ad32012-03-13 18:45:39 -0700516 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -0800517 storeValue(cUnit, rlDest, rlResult);
518}
519
520/*
521 * Similar to genNewArray, but with post-allocation initialization.
522 * Verifier guarantees we're dealing with an array class. Current
523 * code throws runtime exception "bad Filled array req" for 'D' and 'J'.
524 * Current code also throws internal unimp if not 'L', '[' or 'I'.
525 */
526void genFilledNewArray(CompilationUnit* cUnit, MIR* mir, bool isRange)
527{
528 DecodedInstruction* dInsn = &mir->dalvikInsn;
529 int elems = dInsn->vA;
Ian Rogersab2b55d2012-03-18 00:06:11 -0700530 int typeIdx = dInsn->vB;
buzbee31a4a6f2012-02-28 15:36:15 -0800531 oatFlushAllRegs(cUnit); /* Everything to home location */
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700532 int funcOffset;
buzbee31a4a6f2012-02-28 15:36:15 -0800533 if (cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
534 cUnit->dex_cache,
535 *cUnit->dex_file,
Ian Rogersab2b55d2012-03-18 00:06:11 -0700536 typeIdx)) {
Ian Rogers57b86d42012-03-27 16:05:41 -0700537 funcOffset = ENTRYPOINT_OFFSET(pCheckAndAllocArrayFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800538 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -0700539 funcOffset = ENTRYPOINT_OFFSET(pCheckAndAllocArrayFromCodeWithAccessCheck);
buzbee31a4a6f2012-02-28 15:36:15 -0800540 }
Ian Rogersab2b55d2012-03-18 00:06:11 -0700541 callRuntimeHelperImmMethodImm(cUnit, funcOffset, typeIdx, elems);
buzbeee1965672012-03-11 18:39:19 -0700542 oatFreeTemp(cUnit, rARG2);
543 oatFreeTemp(cUnit, rARG1);
buzbee31a4a6f2012-02-28 15:36:15 -0800544 /*
Elliott Hughesadb8c672012-03-06 16:49:32 -0800545 * NOTE: the implicit target for Instruction::FILLED_NEW_ARRAY is the
buzbee31a4a6f2012-02-28 15:36:15 -0800546 * return region. Because AllocFromCode placed the new array
547 * in rRET0, we'll just lock it into place. When debugger support is
548 * added, it may be necessary to additionally copy all return
549 * values to a home location in thread-local storage
550 */
551 oatLockTemp(cUnit, rRET0);
552
553 // TODO: use the correct component size, currently all supported types
554 // share array alignment with ints (see comment at head of function)
555 size_t component_size = sizeof(int32_t);
556
557 // Having a range of 0 is legal
558 if (isRange && (dInsn->vA > 0)) {
559 /*
560 * Bit of ugliness here. We're going generate a mem copy loop
561 * on the register range, but it is possible that some regs
562 * in the range have been promoted. This is unlikely, but
563 * before generating the copy, we'll just force a flush
564 * of any regs in the source range that have been promoted to
565 * home location.
566 */
567 for (unsigned int i = 0; i < dInsn->vA; i++) {
568 RegLocation loc = oatUpdateLoc(cUnit,
569 oatGetSrc(cUnit, mir, i));
570 if (loc.location == kLocPhysReg) {
571 storeBaseDisp(cUnit, rSP, oatSRegOffset(cUnit, loc.sRegLow),
572 loc.lowReg, kWord);
573 }
574 }
575 /*
576 * TUNING note: generated code here could be much improved, but
577 * this is an uncommon operation and isn't especially performance
578 * critical.
579 */
580 int rSrc = oatAllocTemp(cUnit);
581 int rDst = oatAllocTemp(cUnit);
582 int rIdx = oatAllocTemp(cUnit);
buzbee5de34942012-03-01 14:51:57 -0800583#if defined(TARGET_ARM)
buzbee31a4a6f2012-02-28 15:36:15 -0800584 int rVal = rLR; // Using a lot of temps, rLR is known free here
jeffhao174651d2012-04-19 15:27:22 -0700585#elif defined(TARGET_X86)
586 int rVal = rSrc;
buzbee5de34942012-03-01 14:51:57 -0800587#else
588 int rVal = oatAllocTemp(cUnit);
589#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800590 // Set up source pointer
591 RegLocation rlFirst = oatGetSrc(cUnit, mir, 0);
592 opRegRegImm(cUnit, kOpAdd, rSrc, rSP,
593 oatSRegOffset(cUnit, rlFirst.sRegLow));
594 // Set up the target pointer
595 opRegRegImm(cUnit, kOpAdd, rDst, rRET0,
596 Array::DataOffset(component_size).Int32Value());
597 // Set up the loop counter (known to be > 0)
598 loadConstant(cUnit, rIdx, dInsn->vA - 1);
599 // Generate the copy loop. Going backwards for convenience
600 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800601 // Copy next element
602 loadBaseIndexed(cUnit, rSrc, rIdx, rVal, 2, kWord);
603 storeBaseIndexed(cUnit, rDst, rIdx, rVal, 2, kWord);
604#if defined(TARGET_ARM)
605 // Combine sub & test using sub setflags encoding here
606 newLIR3(cUnit, kThumb2SubsRRI12, rIdx, rIdx, 1);
buzbee82488f52012-03-02 08:20:26 -0800607 opCondBranch(cUnit, kCondGe, target);
buzbee31a4a6f2012-02-28 15:36:15 -0800608#else
buzbee5de34942012-03-01 14:51:57 -0800609 oatFreeTemp(cUnit, rVal);
buzbee31a4a6f2012-02-28 15:36:15 -0800610 opRegImm(cUnit, kOpSub, rIdx, 1);
buzbee82488f52012-03-02 08:20:26 -0800611 opCmpImmBranch(cUnit, kCondGe, rIdx, 0, target);
buzbee31a4a6f2012-02-28 15:36:15 -0800612#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800613 } else if (!isRange) {
614 // TUNING: interleave
615 for (unsigned int i = 0; i < dInsn->vA; i++) {
616 RegLocation rlArg = loadValue(cUnit,
617 oatGetSrc(cUnit, mir, i), kCoreReg);
618 storeBaseDisp(cUnit, rRET0,
619 Array::DataOffset(component_size).Int32Value() +
620 i * 4, rlArg.lowReg, kWord);
621 // If the loadValue caused a temp to be allocated, free it
622 if (oatIsTemp(cUnit, rlArg.lowReg)) {
623 oatFreeTemp(cUnit, rlArg.lowReg);
624 }
625 }
626 }
627}
628
629void genSput(CompilationUnit* cUnit, MIR* mir, RegLocation rlSrc,
630 bool isLongOrDouble, bool isObject)
631{
632 int fieldOffset;
633 int ssbIndex;
634 bool isVolatile;
635 bool isReferrersClass;
636 uint32_t fieldIdx = mir->dalvikInsn.vB;
637
638 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
639 *cUnit->dex_file, *cUnit->dex_cache,
640 cUnit->code_item, cUnit->method_idx,
641 cUnit->access_flags);
642
643 bool fastPath =
644 cUnit->compiler->ComputeStaticFieldInfo(fieldIdx, &mUnit,
645 fieldOffset, ssbIndex,
646 isReferrersClass, isVolatile, true);
647 if (fastPath && !SLOW_FIELD_PATH) {
648 DCHECK_GE(fieldOffset, 0);
649 int rBase;
buzbee31a4a6f2012-02-28 15:36:15 -0800650 if (isReferrersClass) {
651 // Fast path, static storage base is this method's class
buzbeee1965672012-03-11 18:39:19 -0700652 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800653 rBase = oatAllocTemp(cUnit);
buzbeee1965672012-03-11 18:39:19 -0700654 loadWordDisp(cUnit, rlMethod.lowReg,
buzbee31a4a6f2012-02-28 15:36:15 -0800655 Method::DeclaringClassOffset().Int32Value(), rBase);
Ian Rogersc6f3bb82012-03-21 20:40:33 -0700656 if (oatIsTemp(cUnit, rlMethod.lowReg)) {
657 oatFreeTemp(cUnit, rlMethod.lowReg);
658 }
buzbee31a4a6f2012-02-28 15:36:15 -0800659 } else {
660 // Medium path, static storage base in a different class which
661 // requires checks that the other class is initialized.
662 DCHECK_GE(ssbIndex, 0);
663 // May do runtime call so everything to home locations.
664 oatFlushAllRegs(cUnit);
665 // Using fixed register to sync with possible call to runtime
666 // support.
buzbeee1965672012-03-11 18:39:19 -0700667 int rMethod = rARG1;
buzbee31a4a6f2012-02-28 15:36:15 -0800668 oatLockTemp(cUnit, rMethod);
669 loadCurrMethodDirect(cUnit, rMethod);
670 rBase = rARG0;
671 oatLockTemp(cUnit, rBase);
672 loadWordDisp(cUnit, rMethod,
673 Method::DexCacheInitializedStaticStorageOffset().Int32Value(),
674 rBase);
675 loadWordDisp(cUnit, rBase,
676 Array::DataOffset(sizeof(Object*)).Int32Value() + sizeof(int32_t*) *
677 ssbIndex, rBase);
678 // rBase now points at appropriate static storage base (Class*)
679 // or NULL if not initialized. Check for NULL and call helper if NULL.
680 // TUNING: fast path should fall through
buzbee82488f52012-03-02 08:20:26 -0800681 LIR* branchOver = opCmpImmBranch(cUnit, kCondNe, rBase, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -0800682 loadConstant(cUnit, rARG0, ssbIndex);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700683 callRuntimeHelperImm(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -0700684 ENTRYPOINT_OFFSET(pInitializeStaticStorage),
Ian Rogersab2b55d2012-03-18 00:06:11 -0700685 ssbIndex);
buzbee31a4a6f2012-02-28 15:36:15 -0800686#if defined(TARGET_MIPS)
687 // For Arm, rRET0 = rARG0 = rBASE, for Mips, we need to copy
buzbee82488f52012-03-02 08:20:26 -0800688 opRegCopy(cUnit, rBase, rRET0);
buzbee31a4a6f2012-02-28 15:36:15 -0800689#endif
690 LIR* skipTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800691 branchOver->target = (LIR*)skipTarget;
buzbeee1965672012-03-11 18:39:19 -0700692 oatFreeTemp(cUnit, rMethod);
buzbee31a4a6f2012-02-28 15:36:15 -0800693 }
694 // rBase now holds static storage base
buzbee31a4a6f2012-02-28 15:36:15 -0800695 if (isLongOrDouble) {
696 rlSrc = oatGetSrcWide(cUnit, mir, 0, 1);
697 rlSrc = loadValueWide(cUnit, rlSrc, kAnyReg);
698 } else {
699 rlSrc = oatGetSrc(cUnit, mir, 0);
700 rlSrc = loadValue(cUnit, rlSrc, kAnyReg);
701 }
702//FIXME: need to generalize the barrier call
703 if (isVolatile) {
704 oatGenMemBarrier(cUnit, kST);
705 }
706 if (isLongOrDouble) {
707 storeBaseDispWide(cUnit, rBase, fieldOffset, rlSrc.lowReg,
708 rlSrc.highReg);
709 } else {
710 storeWordDisp(cUnit, rBase, fieldOffset, rlSrc.lowReg);
711 }
712 if (isVolatile) {
713 oatGenMemBarrier(cUnit, kSY);
714 }
715 if (isObject) {
716 markGCCard(cUnit, rlSrc.lowReg, rBase);
717 }
718 oatFreeTemp(cUnit, rBase);
719 } else {
720 oatFlushAllRegs(cUnit); // Everything to home locations
Ian Rogers57b86d42012-03-27 16:05:41 -0700721 int setterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pSet64Static) :
722 (isObject ? ENTRYPOINT_OFFSET(pSetObjStatic)
723 : ENTRYPOINT_OFFSET(pSet32Static));
Ian Rogersab2b55d2012-03-18 00:06:11 -0700724 callRuntimeHelperImmRegLocation(cUnit, setterOffset, fieldIdx, rlSrc);
buzbee31a4a6f2012-02-28 15:36:15 -0800725 }
726}
727
728void genSget(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
729 bool isLongOrDouble, bool isObject)
730{
731 int fieldOffset;
732 int ssbIndex;
733 bool isVolatile;
734 bool isReferrersClass;
735 uint32_t fieldIdx = mir->dalvikInsn.vB;
736
737 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
738 *cUnit->dex_file, *cUnit->dex_cache,
739 cUnit->code_item, cUnit->method_idx,
740 cUnit->access_flags);
741
742 bool fastPath =
743 cUnit->compiler->ComputeStaticFieldInfo(fieldIdx, &mUnit,
744 fieldOffset, ssbIndex,
745 isReferrersClass, isVolatile,
746 false);
747 if (fastPath && !SLOW_FIELD_PATH) {
748 DCHECK_GE(fieldOffset, 0);
749 int rBase;
buzbee31a4a6f2012-02-28 15:36:15 -0800750 if (isReferrersClass) {
751 // Fast path, static storage base is this method's class
buzbeee1965672012-03-11 18:39:19 -0700752 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800753 rBase = oatAllocTemp(cUnit);
buzbeee1965672012-03-11 18:39:19 -0700754 loadWordDisp(cUnit, rlMethod.lowReg,
buzbee31a4a6f2012-02-28 15:36:15 -0800755 Method::DeclaringClassOffset().Int32Value(), rBase);
756 } else {
757 // Medium path, static storage base in a different class which
758 // requires checks that the other class is initialized
759 DCHECK_GE(ssbIndex, 0);
760 // May do runtime call so everything to home locations.
761 oatFlushAllRegs(cUnit);
762 // Using fixed register to sync with possible call to runtime
763 // support
buzbeee1965672012-03-11 18:39:19 -0700764 int rMethod = rARG1;
buzbee31a4a6f2012-02-28 15:36:15 -0800765 oatLockTemp(cUnit, rMethod);
766 loadCurrMethodDirect(cUnit, rMethod);
767 rBase = rARG0;
768 oatLockTemp(cUnit, rBase);
769 loadWordDisp(cUnit, rMethod,
770 Method::DexCacheInitializedStaticStorageOffset().Int32Value(),
771 rBase);
772 loadWordDisp(cUnit, rBase,
773 Array::DataOffset(sizeof(Object*)).Int32Value() +
774 sizeof(int32_t*) * ssbIndex,
775 rBase);
776 // rBase now points at appropriate static storage base (Class*)
777 // or NULL if not initialized. Check for NULL and call helper if NULL.
778 // TUNING: fast path should fall through
buzbee82488f52012-03-02 08:20:26 -0800779 LIR* branchOver = opCmpImmBranch(cUnit, kCondNe, rBase, 0, NULL);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700780 callRuntimeHelperImm(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -0700781 ENTRYPOINT_OFFSET(pInitializeStaticStorage),
Ian Rogersab2b55d2012-03-18 00:06:11 -0700782 ssbIndex);
buzbee31a4a6f2012-02-28 15:36:15 -0800783#if defined(TARGET_MIPS)
784 // For Arm, rRET0 = rARG0 = rBASE, for Mips, we need to copy
buzbee82488f52012-03-02 08:20:26 -0800785 opRegCopy(cUnit, rBase, rRET0);
buzbee31a4a6f2012-02-28 15:36:15 -0800786#endif
787 LIR* skipTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800788 branchOver->target = (LIR*)skipTarget;
buzbeee1965672012-03-11 18:39:19 -0700789 oatFreeTemp(cUnit, rMethod);
buzbee31a4a6f2012-02-28 15:36:15 -0800790 }
791 // rBase now holds static storage base
buzbee31a4a6f2012-02-28 15:36:15 -0800792 rlDest = isLongOrDouble ? oatGetDestWide(cUnit, mir, 0, 1)
793 : oatGetDest(cUnit, mir, 0);
794 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kAnyReg, true);
795 if (isVolatile) {
796 oatGenMemBarrier(cUnit, kSY);
797 }
798 if (isLongOrDouble) {
799 loadBaseDispWide(cUnit, NULL, rBase, fieldOffset, rlResult.lowReg,
800 rlResult.highReg, INVALID_SREG);
801 } else {
802 loadWordDisp(cUnit, rBase, fieldOffset, rlResult.lowReg);
803 }
804 oatFreeTemp(cUnit, rBase);
805 if (isLongOrDouble) {
806 storeValueWide(cUnit, rlDest, rlResult);
807 } else {
808 storeValue(cUnit, rlDest, rlResult);
809 }
810 } else {
811 oatFlushAllRegs(cUnit); // Everything to home locations
Ian Rogers57b86d42012-03-27 16:05:41 -0700812 int getterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pGet64Static) :
813 (isObject ? ENTRYPOINT_OFFSET(pGetObjStatic)
814 : ENTRYPOINT_OFFSET(pGet32Static));
Ian Rogersab2b55d2012-03-18 00:06:11 -0700815 callRuntimeHelperImm(cUnit, getterOffset, fieldIdx);
buzbee31a4a6f2012-02-28 15:36:15 -0800816 if (isLongOrDouble) {
Ian Rogersf7d9ad32012-03-13 18:45:39 -0700817 RegLocation rlResult = oatGetReturnWide(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -0800818 storeValueWide(cUnit, rlDest, rlResult);
819 } else {
Ian Rogersf7d9ad32012-03-13 18:45:39 -0700820 RegLocation rlResult = oatGetReturn(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -0800821 storeValue(cUnit, rlDest, rlResult);
822 }
823 }
824}
825
826
827// Debugging routine - if null target, branch to DebugMe
828void genShowTarget(CompilationUnit* cUnit)
829{
buzbeea7678db2012-03-05 15:35:46 -0800830#if defined(TARGET_X86)
831 UNIMPLEMENTED(WARNING) << "genShowTarget";
832#else
buzbee0398c422012-03-02 15:22:47 -0800833 LIR* branchOver = opCmpImmBranch(cUnit, kCondNe, rINVOKE_TGT, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -0800834 loadWordDisp(cUnit, rSELF,
Ian Rogers57b86d42012-03-27 16:05:41 -0700835 ENTRYPOINT_OFFSET(pDebugMe), rINVOKE_TGT);
buzbee31a4a6f2012-02-28 15:36:15 -0800836 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800837 branchOver->target = (LIR*)target;
buzbeea7678db2012-03-05 15:35:46 -0800838#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800839}
840
841void genThrowVerificationError(CompilationUnit* cUnit, MIR* mir)
842{
Ian Rogers57b86d42012-03-27 16:05:41 -0700843 callRuntimeHelperImmImm(cUnit, ENTRYPOINT_OFFSET(pThrowVerificationErrorFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -0700844 mir->dalvikInsn.vA, mir->dalvikInsn.vB);
buzbee31a4a6f2012-02-28 15:36:15 -0800845}
846
847void handleSuspendLaunchpads(CompilationUnit *cUnit)
848{
Ian Rogersab2b55d2012-03-18 00:06:11 -0700849 LIR** suspendLabel = (LIR **)cUnit->suspendLaunchpads.elemList;
buzbee31a4a6f2012-02-28 15:36:15 -0800850 int numElems = cUnit->suspendLaunchpads.numUsed;
buzbee31a4a6f2012-02-28 15:36:15 -0800851 for (int i = 0; i < numElems; i++) {
buzbeec5159d52012-03-03 11:48:39 -0800852 oatResetRegPool(cUnit);
buzbeefc9e6fa2012-03-23 15:14:29 -0700853 oatResetDefTracking(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800854 LIR* lab = suspendLabel[i];
855 LIR* resumeLab = (LIR*)lab->operands[0];
856 cUnit->currentDalvikOffset = lab->operands[1];
Ian Rogersab2b55d2012-03-18 00:06:11 -0700857 oatAppendLIR(cUnit, lab);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700858#if defined(TARGET_X86)
859 opThreadMem(cUnit, kOpBlx,
Ian Rogers57b86d42012-03-27 16:05:41 -0700860 ENTRYPOINT_OFFSET(pTestSuspendFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700861#else
Ian Rogers57b86d42012-03-27 16:05:41 -0700862 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pTestSuspendFromCode));
buzbee31a4a6f2012-02-28 15:36:15 -0800863 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700864#endif
buzbee82488f52012-03-02 08:20:26 -0800865 opUnconditionalBranch(cUnit, resumeLab);
buzbee31a4a6f2012-02-28 15:36:15 -0800866 }
867}
868
buzbeefc9e6fa2012-03-23 15:14:29 -0700869void handleIntrinsicLaunchpads(CompilationUnit *cUnit)
870{
871 LIR** intrinsicLabel = (LIR **)cUnit->intrinsicLaunchpads.elemList;
872 int numElems = cUnit->intrinsicLaunchpads.numUsed;
873 for (int i = 0; i < numElems; i++) {
874 oatResetRegPool(cUnit);
875 oatResetDefTracking(cUnit);
876 LIR* lab = intrinsicLabel[i];
877 MIR* mir = (MIR*)lab->operands[0];
878 InvokeType type = (InvokeType)lab->operands[1];
879 BasicBlock* bb = (BasicBlock*)lab->operands[3];
880 cUnit->currentDalvikOffset = mir->offset;
881 oatAppendLIR(cUnit, lab);
882 genInvoke(cUnit, bb, mir, type, false /* isRange */);
883 LIR* resumeLab = (LIR*)lab->operands[2];
884 if (resumeLab != NULL) {
885 opUnconditionalBranch(cUnit, resumeLab);
886 }
887 }
888}
889
buzbee31a4a6f2012-02-28 15:36:15 -0800890void handleThrowLaunchpads(CompilationUnit *cUnit)
891{
Ian Rogersab2b55d2012-03-18 00:06:11 -0700892 LIR** throwLabel = (LIR **)cUnit->throwLaunchpads.elemList;
buzbee31a4a6f2012-02-28 15:36:15 -0800893 int numElems = cUnit->throwLaunchpads.numUsed;
Ian Rogersab2b55d2012-03-18 00:06:11 -0700894 for (int i = 0; i < numElems; i++) {
buzbeec5159d52012-03-03 11:48:39 -0800895 oatResetRegPool(cUnit);
buzbeefc9e6fa2012-03-23 15:14:29 -0700896 oatResetDefTracking(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800897 LIR* lab = throwLabel[i];
898 cUnit->currentDalvikOffset = lab->operands[1];
Ian Rogersab2b55d2012-03-18 00:06:11 -0700899 oatAppendLIR(cUnit, lab);
buzbee31a4a6f2012-02-28 15:36:15 -0800900 int funcOffset = 0;
901 int v1 = lab->operands[2];
902 int v2 = lab->operands[3];
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700903 switch (lab->operands[0]) {
buzbee31a4a6f2012-02-28 15:36:15 -0800904 case kThrowNullPointer:
Ian Rogers57b86d42012-03-27 16:05:41 -0700905 funcOffset = ENTRYPOINT_OFFSET(pThrowNullPointerFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800906 break;
907 case kThrowArrayBounds:
buzbee5de34942012-03-01 14:51:57 -0800908 if (v2 != rARG0) {
buzbee82488f52012-03-02 08:20:26 -0800909 opRegCopy(cUnit, rARG0, v1);
910 opRegCopy(cUnit, rARG1, v2);
buzbee31a4a6f2012-02-28 15:36:15 -0800911 } else {
buzbee5de34942012-03-01 14:51:57 -0800912 if (v1 == rARG1) {
buzbee31a4a6f2012-02-28 15:36:15 -0800913#if defined(TARGET_ARM)
914 int rTmp = r12;
915#else
916 int rTmp = oatAllocTemp(cUnit);
917#endif
buzbee82488f52012-03-02 08:20:26 -0800918 opRegCopy(cUnit, rTmp, v1);
919 opRegCopy(cUnit, rARG1, v2);
920 opRegCopy(cUnit, rARG0, rTmp);
buzbee31a4a6f2012-02-28 15:36:15 -0800921 } else {
buzbee82488f52012-03-02 08:20:26 -0800922 opRegCopy(cUnit, rARG1, v2);
923 opRegCopy(cUnit, rARG0, v1);
buzbee31a4a6f2012-02-28 15:36:15 -0800924 }
925 }
Ian Rogers57b86d42012-03-27 16:05:41 -0700926 funcOffset = ENTRYPOINT_OFFSET(pThrowArrayBoundsFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800927 break;
928 case kThrowDivZero:
Ian Rogers57b86d42012-03-27 16:05:41 -0700929 funcOffset = ENTRYPOINT_OFFSET(pThrowDivZeroFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800930 break;
931 case kThrowVerificationError:
932 loadConstant(cUnit, rARG0, v1);
933 loadConstant(cUnit, rARG1, v2);
934 funcOffset =
Ian Rogers57b86d42012-03-27 16:05:41 -0700935 ENTRYPOINT_OFFSET(pThrowVerificationErrorFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800936 break;
937 case kThrowNoSuchMethod:
buzbee82488f52012-03-02 08:20:26 -0800938 opRegCopy(cUnit, rARG0, v1);
buzbee31a4a6f2012-02-28 15:36:15 -0800939 funcOffset =
Ian Rogers57b86d42012-03-27 16:05:41 -0700940 ENTRYPOINT_OFFSET(pThrowNoSuchMethodFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800941 break;
942 case kThrowStackOverflow:
Elliott Hughes36ecb782012-04-17 16:55:45 -0700943 funcOffset = ENTRYPOINT_OFFSET(pThrowStackOverflowFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800944 // Restore stack alignment
Elliott Hughes36ecb782012-04-17 16:55:45 -0700945#if !defined(TARGET_X86)
946 opRegImm(cUnit, kOpAdd, rSP, (cUnit->numCoreSpills + cUnit->numFPSpills) * 4);
947#else
948 opRegImm(cUnit, kOpAdd, rSP, cUnit->frameSize);
949#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800950 break;
951 default:
952 LOG(FATAL) << "Unexpected throw kind: " << lab->operands[0];
953 }
Ian Rogersab2b55d2012-03-18 00:06:11 -0700954 oatClobberCalleeSave(cUnit);
955#if !defined(TARGET_X86)
buzbee31a4a6f2012-02-28 15:36:15 -0800956 int rTgt = loadHelper(cUnit, funcOffset);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700957 opReg(cUnit, kOpBlx, rTgt);
buzbee0398c422012-03-02 15:22:47 -0800958 oatFreeTemp(cUnit, rTgt);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700959#else
960 opThreadMem(cUnit, kOpBlx, funcOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700961#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800962 }
963}
964
965/* Needed by the Assembler */
966void oatSetupResourceMasks(LIR* lir)
967{
968 setupResourceMasks(lir);
969}
970
buzbee16da88c2012-03-20 10:38:17 -0700971bool fastInstance(CompilationUnit* cUnit, uint32_t fieldIdx,
972 int& fieldOffset, bool& isVolatile, bool isPut)
973{
974 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
975 *cUnit->dex_file, *cUnit->dex_cache,
976 cUnit->code_item, cUnit->method_idx,
977 cUnit->access_flags);
978 return cUnit->compiler->ComputeInstanceFieldInfo(fieldIdx, &mUnit,
979 fieldOffset, isVolatile, isPut);
980}
981
buzbee31a4a6f2012-02-28 15:36:15 -0800982void genIGet(CompilationUnit* cUnit, MIR* mir, OpSize size,
983 RegLocation rlDest, RegLocation rlObj,
984 bool isLongOrDouble, bool isObject)
985{
986 int fieldOffset;
987 bool isVolatile;
988 uint32_t fieldIdx = mir->dalvikInsn.vC;
989
buzbee16da88c2012-03-20 10:38:17 -0700990 bool fastPath = fastInstance(cUnit, fieldIdx, fieldOffset, isVolatile,
991 false);
buzbee31a4a6f2012-02-28 15:36:15 -0800992
993 if (fastPath && !SLOW_FIELD_PATH) {
994 RegLocation rlResult;
995 RegisterClass regClass = oatRegClassBySize(size);
996 DCHECK_GE(fieldOffset, 0);
997 rlObj = loadValue(cUnit, rlObj, kCoreReg);
998 if (isLongOrDouble) {
999 DCHECK(rlDest.wide);
1000 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
Ian Rogersb5d09b22012-03-06 22:14:17 -08001001#if defined(TARGET_X86)
1002 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1003 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1004 loadBaseDispWide(cUnit, mir, rlObj.lowReg, fieldOffset, rlResult.lowReg,
1005 rlResult.highReg, rlObj.sRegLow);
1006 if (isVolatile) {
1007 oatGenMemBarrier(cUnit, kSY);
1008 }
1009#else
buzbee31a4a6f2012-02-28 15:36:15 -08001010 int regPtr = oatAllocTemp(cUnit);
1011 opRegRegImm(cUnit, kOpAdd, regPtr, rlObj.lowReg, fieldOffset);
1012 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1013 loadPair(cUnit, regPtr, rlResult.lowReg, rlResult.highReg);
1014 if (isVolatile) {
1015 oatGenMemBarrier(cUnit, kSY);
1016 }
1017 oatFreeTemp(cUnit, regPtr);
Ian Rogersb5d09b22012-03-06 22:14:17 -08001018#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001019 storeValueWide(cUnit, rlDest, rlResult);
1020 } else {
1021 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1022 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1023 loadBaseDisp(cUnit, mir, rlObj.lowReg, fieldOffset, rlResult.lowReg,
1024 kWord, rlObj.sRegLow);
1025 if (isVolatile) {
1026 oatGenMemBarrier(cUnit, kSY);
1027 }
1028 storeValue(cUnit, rlDest, rlResult);
1029 }
1030 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -07001031 int getterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pGet64Instance) :
1032 (isObject ? ENTRYPOINT_OFFSET(pGetObjInstance)
1033 : ENTRYPOINT_OFFSET(pGet32Instance));
Ian Rogersab2b55d2012-03-18 00:06:11 -07001034 callRuntimeHelperImmRegLocation(cUnit, getterOffset, fieldIdx, rlObj);
buzbee31a4a6f2012-02-28 15:36:15 -08001035 if (isLongOrDouble) {
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001036 RegLocation rlResult = oatGetReturnWide(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08001037 storeValueWide(cUnit, rlDest, rlResult);
1038 } else {
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001039 RegLocation rlResult = oatGetReturn(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08001040 storeValue(cUnit, rlDest, rlResult);
1041 }
1042 }
1043}
1044
1045void genIPut(CompilationUnit* cUnit, MIR* mir, OpSize size, RegLocation rlSrc,
1046 RegLocation rlObj, bool isLongOrDouble, bool isObject)
1047{
1048 int fieldOffset;
1049 bool isVolatile;
1050 uint32_t fieldIdx = mir->dalvikInsn.vC;
1051
buzbee16da88c2012-03-20 10:38:17 -07001052 bool fastPath = fastInstance(cUnit, fieldIdx, fieldOffset, isVolatile,
1053 true);
buzbee31a4a6f2012-02-28 15:36:15 -08001054 if (fastPath && !SLOW_FIELD_PATH) {
1055 RegisterClass regClass = oatRegClassBySize(size);
1056 DCHECK_GE(fieldOffset, 0);
1057 rlObj = loadValue(cUnit, rlObj, kCoreReg);
1058 if (isLongOrDouble) {
1059 int regPtr;
1060 rlSrc = loadValueWide(cUnit, rlSrc, kAnyReg);
1061 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1062 regPtr = oatAllocTemp(cUnit);
1063 opRegRegImm(cUnit, kOpAdd, regPtr, rlObj.lowReg, fieldOffset);
1064 if (isVolatile) {
1065 oatGenMemBarrier(cUnit, kST);
1066 }
1067 storePair(cUnit, regPtr, rlSrc.lowReg, rlSrc.highReg);
1068 if (isVolatile) {
1069 oatGenMemBarrier(cUnit, kSY);
1070 }
1071 oatFreeTemp(cUnit, regPtr);
1072 } else {
1073 rlSrc = loadValue(cUnit, rlSrc, regClass);
1074 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1075 if (isVolatile) {
1076 oatGenMemBarrier(cUnit, kST);
1077 }
1078 storeBaseDisp(cUnit, rlObj.lowReg, fieldOffset, rlSrc.lowReg, kWord);
1079 if (isVolatile) {
1080 oatGenMemBarrier(cUnit, kSY);
1081 }
buzbeea7c12682012-03-19 13:13:53 -07001082 if (isObject) {
1083 markGCCard(cUnit, rlSrc.lowReg, rlObj.lowReg);
1084 }
buzbee31a4a6f2012-02-28 15:36:15 -08001085 }
1086 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -07001087 int setterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pSet64Instance) :
1088 (isObject ? ENTRYPOINT_OFFSET(pSetObjInstance)
1089 : ENTRYPOINT_OFFSET(pSet32Instance));
Ian Rogersab2b55d2012-03-18 00:06:11 -07001090 callRuntimeHelperImmRegLocationRegLocation(cUnit, setterOffset,
1091 fieldIdx, rlObj, rlSrc);
buzbee31a4a6f2012-02-28 15:36:15 -08001092 }
1093}
1094
1095void genConstClass(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1096 RegLocation rlSrc)
1097{
1098 uint32_t type_idx = mir->dalvikInsn.vB;
buzbeee1965672012-03-11 18:39:19 -07001099 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001100 int resReg = oatAllocTemp(cUnit);
1101 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1102 if (!cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
1103 cUnit->dex_cache,
1104 *cUnit->dex_file,
1105 type_idx)) {
1106 // Call out to helper which resolves type and verifies access.
1107 // Resolved type returned in rRET0.
Ian Rogersab2b55d2012-03-18 00:06:11 -07001108 callRuntimeHelperImmReg(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001109 ENTRYPOINT_OFFSET(pInitializeTypeAndVerifyAccessFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001110 type_idx, rlMethod.lowReg);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001111 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001112 storeValue(cUnit, rlDest, rlResult);
1113 } else {
1114 // We're don't need access checks, load type from dex cache
1115 int32_t dex_cache_offset =
1116 Method::DexCacheResolvedTypesOffset().Int32Value();
buzbeee1965672012-03-11 18:39:19 -07001117 loadWordDisp(cUnit, rlMethod.lowReg, dex_cache_offset, resReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001118 int32_t offset_of_type =
1119 Array::DataOffset(sizeof(Class*)).Int32Value() + (sizeof(Class*)
1120 * type_idx);
1121 loadWordDisp(cUnit, resReg, offset_of_type, rlResult.lowReg);
1122 if (!cUnit->compiler->CanAssumeTypeIsPresentInDexCache(cUnit->dex_cache,
1123 type_idx) || SLOW_TYPE_PATH) {
1124 // Slow path, at runtime test if type is null and if so initialize
1125 oatFlushAllRegs(cUnit);
buzbee82488f52012-03-02 08:20:26 -08001126 LIR* branch1 = opCmpImmBranch(cUnit, kCondEq, rlResult.lowReg, 0,
1127 NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08001128 // Resolved, store and hop over following code
1129 storeValue(cUnit, rlDest, rlResult);
buzbee3d661942012-03-14 17:37:27 -07001130 /*
1131 * Because we have stores of the target value on two paths,
1132 * clobber temp tracking for the destination using the ssa name
1133 */
1134 oatClobberSReg(cUnit, rlDest.sRegLow);
buzbee82488f52012-03-02 08:20:26 -08001135 LIR* branch2 = opUnconditionalBranch(cUnit,0);
buzbee31a4a6f2012-02-28 15:36:15 -08001136 // TUNING: move slow path to end & remove unconditional branch
1137 LIR* target1 = newLIR0(cUnit, kPseudoTargetLabel);
buzbee5de34942012-03-01 14:51:57 -08001138 // Call out to helper, which will return resolved type in rARG0
Ian Rogers57b86d42012-03-27 16:05:41 -07001139 callRuntimeHelperImmReg(cUnit, ENTRYPOINT_OFFSET(pInitializeTypeFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001140 type_idx, rlMethod.lowReg);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001141 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001142 storeValue(cUnit, rlDest, rlResult);
buzbee3d661942012-03-14 17:37:27 -07001143 /*
1144 * Because we have stores of the target value on two paths,
1145 * clobber temp tracking for the destination using the ssa name
1146 */
1147 oatClobberSReg(cUnit, rlDest.sRegLow);
buzbee31a4a6f2012-02-28 15:36:15 -08001148 // Rejoin code paths
1149 LIR* target2 = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001150 branch1->target = (LIR*)target1;
1151 branch2->target = (LIR*)target2;
1152 } else {
1153 // Fast path, we're done - just store result
1154 storeValue(cUnit, rlDest, rlResult);
1155 }
1156 }
1157}
Ian Rogersab2b55d2012-03-18 00:06:11 -07001158
buzbee31a4a6f2012-02-28 15:36:15 -08001159void genConstString(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1160 RegLocation rlSrc)
1161{
1162 /* NOTE: Most strings should be available at compile time */
1163 uint32_t string_idx = mir->dalvikInsn.vB;
1164 int32_t offset_of_string = Array::DataOffset(sizeof(String*)).Int32Value() +
1165 (sizeof(String*) * string_idx);
1166 if (!cUnit->compiler->CanAssumeStringIsPresentInDexCache(
1167 cUnit->dex_cache, string_idx) || SLOW_STRING_PATH) {
1168 // slow path, resolve string if not in dex cache
1169 oatFlushAllRegs(cUnit);
1170 oatLockCallTemps(cUnit); // Using explicit registers
1171 loadCurrMethodDirect(cUnit, rARG2);
1172 loadWordDisp(cUnit, rARG2,
1173 Method::DexCacheStringsOffset().Int32Value(), rARG0);
1174 // Might call out to helper, which will return resolved string in rRET0
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001175#if !defined(TARGET_X86)
Ian Rogers57b86d42012-03-27 16:05:41 -07001176 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pResolveStringFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001177#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001178 loadWordDisp(cUnit, rRET0, offset_of_string, rARG0);
1179 loadConstant(cUnit, rARG1, string_idx);
1180#if defined(TARGET_ARM)
1181 opRegImm(cUnit, kOpCmp, rRET0, 0); // Is resolved?
1182 genBarrier(cUnit);
1183 // For testing, always force through helper
1184 if (!EXERCISE_SLOWEST_STRING_PATH) {
buzbee82488f52012-03-02 08:20:26 -08001185 opIT(cUnit, kArmCondEq, "T");
buzbee31a4a6f2012-02-28 15:36:15 -08001186 }
buzbee82488f52012-03-02 08:20:26 -08001187 opRegCopy(cUnit, rARG0, rARG2); // .eq
buzbee31a4a6f2012-02-28 15:36:15 -08001188 opReg(cUnit, kOpBlx, rTgt); // .eq, helper(Method*, string_idx)
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001189 oatFreeTemp(cUnit, rTgt);
1190#elif defined(TARGET_MIPS)
buzbee82488f52012-03-02 08:20:26 -08001191 LIR* branch = opCmpImmBranch(cUnit, kCondNe, rRET0, 0, NULL);
1192 opRegCopy(cUnit, rARG0, rARG2); // .eq
buzbee31a4a6f2012-02-28 15:36:15 -08001193 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001194 oatFreeTemp(cUnit, rTgt);
buzbee31a4a6f2012-02-28 15:36:15 -08001195 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001196 branch->target = target;
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001197#else
Ian Rogers57b86d42012-03-27 16:05:41 -07001198 callRuntimeHelperRegReg(cUnit, ENTRYPOINT_OFFSET(pResolveStringFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001199 rARG2, rARG1);
buzbee31a4a6f2012-02-28 15:36:15 -08001200#endif
1201 genBarrier(cUnit);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001202 storeValue(cUnit, rlDest, oatGetReturn(cUnit, false));
buzbee31a4a6f2012-02-28 15:36:15 -08001203 } else {
buzbeee1965672012-03-11 18:39:19 -07001204 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001205 int resReg = oatAllocTemp(cUnit);
1206 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
buzbeee1965672012-03-11 18:39:19 -07001207 loadWordDisp(cUnit, rlMethod.lowReg,
buzbee31a4a6f2012-02-28 15:36:15 -08001208 Method::DexCacheStringsOffset().Int32Value(), resReg);
1209 loadWordDisp(cUnit, resReg, offset_of_string, rlResult.lowReg);
1210 storeValue(cUnit, rlDest, rlResult);
1211 }
1212}
1213
1214/*
1215 * Let helper function take care of everything. Will
1216 * call Class::NewInstanceFromCode(type_idx, method);
1217 */
1218void genNewInstance(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest)
1219{
1220 oatFlushAllRegs(cUnit); /* Everything to home location */
1221 uint32_t type_idx = mir->dalvikInsn.vB;
1222 // alloc will always check for resolution, do we also need to verify
1223 // access because the verifier was unable to?
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001224 int funcOffset;
buzbee31a4a6f2012-02-28 15:36:15 -08001225 if (cUnit->compiler->CanAccessInstantiableTypeWithoutChecks(
1226 cUnit->method_idx, cUnit->dex_cache, *cUnit->dex_file, type_idx)) {
Ian Rogers57b86d42012-03-27 16:05:41 -07001227 funcOffset = ENTRYPOINT_OFFSET(pAllocObjectFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -08001228 } else {
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001229 funcOffset =
Ian Rogers57b86d42012-03-27 16:05:41 -07001230 ENTRYPOINT_OFFSET(pAllocObjectFromCodeWithAccessCheck);
buzbee31a4a6f2012-02-28 15:36:15 -08001231 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07001232 callRuntimeHelperImmMethod(cUnit, funcOffset, type_idx);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001233 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001234 storeValue(cUnit, rlDest, rlResult);
1235}
1236
Ian Rogersab2b55d2012-03-18 00:06:11 -07001237void genThrow(CompilationUnit* cUnit, MIR* mir, RegLocation rlSrc)
1238{
1239 oatFlushAllRegs(cUnit);
Ian Rogers57b86d42012-03-27 16:05:41 -07001240 callRuntimeHelperRegLocation(cUnit, ENTRYPOINT_OFFSET(pDeliverException), rlSrc);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001241}
1242
buzbee31a4a6f2012-02-28 15:36:15 -08001243void genInstanceof(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1244 RegLocation rlSrc)
1245{
1246 oatFlushAllRegs(cUnit);
1247 // May generate a call - use explicit registers
1248 oatLockCallTemps(cUnit);
1249 uint32_t type_idx = mir->dalvikInsn.vC;
buzbee5de34942012-03-01 14:51:57 -08001250 loadCurrMethodDirect(cUnit, rARG1); // rARG1 <= current Method*
buzbee31a4a6f2012-02-28 15:36:15 -08001251 int classReg = rARG2; // rARG2 will hold the Class*
1252 if (!cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
1253 cUnit->dex_cache,
1254 *cUnit->dex_file,
1255 type_idx)) {
1256 // Check we have access to type_idx and if not throw IllegalAccessError,
buzbee5de34942012-03-01 14:51:57 -08001257 // returns Class* in rARG0
Ian Rogersab2b55d2012-03-18 00:06:11 -07001258 callRuntimeHelperImm(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001259 ENTRYPOINT_OFFSET(pInitializeTypeAndVerifyAccessFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001260 type_idx);
buzbee82488f52012-03-02 08:20:26 -08001261 opRegCopy(cUnit, classReg, rRET0); // Align usage with fast path
buzbee5de34942012-03-01 14:51:57 -08001262 loadValueDirectFixed(cUnit, rlSrc, rARG0); // rARG0 <= ref
buzbee31a4a6f2012-02-28 15:36:15 -08001263 } else {
buzbee5de34942012-03-01 14:51:57 -08001264 // Load dex cache entry into classReg (rARG2)
buzbee31a4a6f2012-02-28 15:36:15 -08001265 loadValueDirectFixed(cUnit, rlSrc, rARG0); // rARG0 <= ref
1266 loadWordDisp(cUnit, rARG1,
1267 Method::DexCacheResolvedTypesOffset().Int32Value(),
1268 classReg);
1269 int32_t offset_of_type =
1270 Array::DataOffset(sizeof(Class*)).Int32Value() + (sizeof(Class*)
1271 * type_idx);
1272 loadWordDisp(cUnit, classReg, offset_of_type, classReg);
1273 if (!cUnit->compiler->CanAssumeTypeIsPresentInDexCache(
1274 cUnit->dex_cache, type_idx)) {
1275 // Need to test presence of type in dex cache at runtime
buzbee82488f52012-03-02 08:20:26 -08001276 LIR* hopBranch = opCmpImmBranch(cUnit, kCondNe, classReg, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08001277 // Not resolved
1278 // Call out to helper, which will return resolved type in rRET0
Ian Rogers57b86d42012-03-27 16:05:41 -07001279 callRuntimeHelperImm(cUnit, ENTRYPOINT_OFFSET(pInitializeTypeFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001280 type_idx);
buzbee82488f52012-03-02 08:20:26 -08001281 opRegCopy(cUnit, rARG2, rRET0); // Align usage with fast path
buzbee31a4a6f2012-02-28 15:36:15 -08001282 loadValueDirectFixed(cUnit, rlSrc, rARG0); /* reload Ref */
1283 // Rejoin code paths
1284 LIR* hopTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001285 hopBranch->target = (LIR*)hopTarget;
1286 }
1287 }
1288 /* rARG0 is ref, rARG2 is class. If ref==null, use directly as bool result */
buzbee82488f52012-03-02 08:20:26 -08001289 LIR* branch1 = opCmpImmBranch(cUnit, kCondEq, rARG0, 0, NULL);
Elliott Hughese84278b2012-03-22 10:06:53 -07001290 /* load object->klass_ */
buzbee31a4a6f2012-02-28 15:36:15 -08001291 DCHECK_EQ(Object::ClassOffset().Int32Value(), 0);
1292 loadWordDisp(cUnit, rARG0, Object::ClassOffset().Int32Value(), rARG1);
Elliott Hughese84278b2012-03-22 10:06:53 -07001293 /* rARG0 is ref, rARG1 is ref->klass_, rARG2 is class */
buzbee0398c422012-03-02 15:22:47 -08001294#if defined(TARGET_ARM)
1295 /* Uses conditional nullification */
Ian Rogers57b86d42012-03-27 16:05:41 -07001296 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pInstanceofNonTrivialFromCode));
buzbee31a4a6f2012-02-28 15:36:15 -08001297 opRegReg(cUnit, kOpCmp, rARG1, rARG2); // Same?
buzbee82488f52012-03-02 08:20:26 -08001298 opIT(cUnit, kArmCondEq, "EE"); // if-convert the test
buzbee31a4a6f2012-02-28 15:36:15 -08001299 loadConstant(cUnit, rARG0, 1); // .eq case - load true
buzbee82488f52012-03-02 08:20:26 -08001300 opRegCopy(cUnit, rARG0, rARG2); // .ne case - arg0 <= class
buzbee31a4a6f2012-02-28 15:36:15 -08001301 opReg(cUnit, kOpBlx, rTgt); // .ne case: helper(class, ref->class)
Ian Rogersab2b55d2012-03-18 00:06:11 -07001302 oatFreeTemp(cUnit, rTgt);
buzbee31a4a6f2012-02-28 15:36:15 -08001303#else
buzbee0398c422012-03-02 15:22:47 -08001304 /* Uses branchovers */
1305 loadConstant(cUnit, rARG0, 1); // assume true
1306 LIR* branchover = opCmpBranch(cUnit, kCondEq, rARG1, rARG2, NULL);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001307#if !defined(TARGET_X86)
Ian Rogers57b86d42012-03-27 16:05:41 -07001308 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pInstanceofNonTrivialFromCode));
buzbee0398c422012-03-02 15:22:47 -08001309 opRegCopy(cUnit, rARG0, rARG2); // .ne case - arg0 <= class
1310 opReg(cUnit, kOpBlx, rTgt); // .ne case: helper(class, ref->class)
Ian Rogersab2b55d2012-03-18 00:06:11 -07001311 oatFreeTemp(cUnit, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001312#else
1313 opRegCopy(cUnit, rARG0, rARG2);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001314 opThreadMem(cUnit, kOpBlx,
Ian Rogers57b86d42012-03-27 16:05:41 -07001315 ENTRYPOINT_OFFSET(pInstanceofNonTrivialFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001316#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001317#endif
buzbee0398c422012-03-02 15:22:47 -08001318 oatClobberCalleeSave(cUnit);
1319 /* branch targets here */
buzbee31a4a6f2012-02-28 15:36:15 -08001320 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001321 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001322 storeValue(cUnit, rlDest, rlResult);
buzbee0398c422012-03-02 15:22:47 -08001323 branch1->target = target;
1324#if !defined(TARGET_ARM)
1325 branchover->target = target;
1326#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001327}
1328
1329void genCheckCast(CompilationUnit* cUnit, MIR* mir, RegLocation rlSrc)
1330{
1331 oatFlushAllRegs(cUnit);
1332 // May generate a call - use explicit registers
1333 oatLockCallTemps(cUnit);
1334 uint32_t type_idx = mir->dalvikInsn.vB;
1335 loadCurrMethodDirect(cUnit, rARG1); // rARG1 <= current Method*
1336 int classReg = rARG2; // rARG2 will hold the Class*
1337 if (!cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
1338 cUnit->dex_cache,
1339 *cUnit->dex_file,
1340 type_idx)) {
1341 // Check we have access to type_idx and if not throw IllegalAccessError,
1342 // returns Class* in rRET0
Ian Rogersab2b55d2012-03-18 00:06:11 -07001343 // InitializeTypeAndVerifyAccess(idx, method)
1344 callRuntimeHelperImmReg(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001345 ENTRYPOINT_OFFSET(pInitializeTypeAndVerifyAccessFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001346 type_idx, rARG1);
buzbee82488f52012-03-02 08:20:26 -08001347 opRegCopy(cUnit, classReg, rRET0); // Align usage with fast path
buzbee31a4a6f2012-02-28 15:36:15 -08001348 } else {
1349 // Load dex cache entry into classReg (rARG2)
1350 loadWordDisp(cUnit, rARG1,
1351 Method::DexCacheResolvedTypesOffset().Int32Value(),
1352 classReg);
1353 int32_t offset_of_type =
1354 Array::DataOffset(sizeof(Class*)).Int32Value() +
1355 (sizeof(Class*) * type_idx);
1356 loadWordDisp(cUnit, classReg, offset_of_type, classReg);
1357 if (!cUnit->compiler->CanAssumeTypeIsPresentInDexCache(
1358 cUnit->dex_cache, type_idx)) {
1359 // Need to test presence of type in dex cache at runtime
buzbee82488f52012-03-02 08:20:26 -08001360 LIR* hopBranch = opCmpImmBranch(cUnit, kCondNe, classReg, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08001361 // Not resolved
buzbee5de34942012-03-01 14:51:57 -08001362 // Call out to helper, which will return resolved type in rARG0
Ian Rogersab2b55d2012-03-18 00:06:11 -07001363 // InitializeTypeFromCode(idx, method)
1364 callRuntimeHelperImmReg(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001365 ENTRYPOINT_OFFSET(pInitializeTypeFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001366 type_idx, rARG1);
buzbee82488f52012-03-02 08:20:26 -08001367 opRegCopy(cUnit, classReg, rARG0); // Align usage with fast path
buzbee31a4a6f2012-02-28 15:36:15 -08001368 // Rejoin code paths
1369 LIR* hopTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001370 hopBranch->target = (LIR*)hopTarget;
1371 }
1372 }
buzbee5de34942012-03-01 14:51:57 -08001373 // At this point, classReg (rARG2) has class
buzbee31a4a6f2012-02-28 15:36:15 -08001374 loadValueDirectFixed(cUnit, rlSrc, rARG0); // rARG0 <= ref
1375 /* Null is OK - continue */
buzbee82488f52012-03-02 08:20:26 -08001376 LIR* branch1 = opCmpImmBranch(cUnit, kCondEq, rARG0, 0, NULL);
Elliott Hughese84278b2012-03-22 10:06:53 -07001377 /* load object->klass_ */
buzbee31a4a6f2012-02-28 15:36:15 -08001378 DCHECK_EQ(Object::ClassOffset().Int32Value(), 0);
1379 loadWordDisp(cUnit, rARG0, Object::ClassOffset().Int32Value(), rARG1);
Elliott Hughese84278b2012-03-22 10:06:53 -07001380 /* rARG1 now contains object->klass_ */
Ian Rogersab2b55d2012-03-18 00:06:11 -07001381#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbee82488f52012-03-02 08:20:26 -08001382 LIR* branch2 = opCmpBranch(cUnit, kCondEq, rARG1, classReg, NULL);
Ian Rogers57b86d42012-03-27 16:05:41 -07001383 callRuntimeHelperRegReg(cUnit, ENTRYPOINT_OFFSET(pCheckCastFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001384 rARG1, rARG2);
1385#else // defined(TARGET_ARM)
Ian Rogers57b86d42012-03-27 16:05:41 -07001386 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pCheckCastFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001387 opRegReg(cUnit, kOpCmp, rARG1, classReg);
1388 LIR* branch2 = opCondBranch(cUnit, kCondEq, NULL); /* If eq, trivial yes */
1389 opRegCopy(cUnit, rARG0, rARG1);
1390 opRegCopy(cUnit, rARG1, rARG2);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001391 oatClobberCalleeSave(cUnit);
1392 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001393 oatFreeTemp(cUnit, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001394#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001395 /* branch target here */
1396 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
Ian Rogersd36c52e2012-04-09 16:29:25 -07001397 branch1->target = target;
1398 branch2->target = target;
buzbee31a4a6f2012-02-28 15:36:15 -08001399}
1400
buzbee31a4a6f2012-02-28 15:36:15 -08001401/*
1402 * Generate array store
1403 *
1404 */
1405void genArrayObjPut(CompilationUnit* cUnit, MIR* mir, RegLocation rlArray,
1406 RegLocation rlIndex, RegLocation rlSrc, int scale)
1407{
buzbee31a4a6f2012-02-28 15:36:15 -08001408 int lenOffset = Array::LengthOffset().Int32Value();
1409 int dataOffset = Array::DataOffset(sizeof(Object*)).Int32Value();
1410
Ian Rogersd36c52e2012-04-09 16:29:25 -07001411 oatFlushAllRegs(cUnit); // Use explicit registers
1412 oatLockCallTemps(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001413
Ian Rogersd36c52e2012-04-09 16:29:25 -07001414 int rValue = rARG0; // Register holding value
1415 int rArrayClass = rARG1; // Register holding array's Class
1416 int rArray = rARG2; // Register holding array
1417 int rIndex = rARG3; // Register holding index into array
1418
1419 loadValueDirectFixed(cUnit, rlArray, rArray); // Grab array
1420 loadValueDirectFixed(cUnit, rlSrc, rValue); // Grab value
1421 loadValueDirectFixed(cUnit, rlIndex, rIndex); // Grab index
1422
1423 genNullCheck(cUnit, rlArray.sRegLow, rArray, mir); // NPE?
1424
1425 // Store of null?
1426 LIR* null_value_check = opCmpImmBranch(cUnit, kCondEq, rValue, 0, NULL);
1427
1428 // Get the array's class.
1429 loadWordDisp(cUnit, rArray, Object::ClassOffset().Int32Value(), rArrayClass);
Ian Rogers57b86d42012-03-27 16:05:41 -07001430 callRuntimeHelperRegReg(cUnit, ENTRYPOINT_OFFSET(pCanPutArrayElementFromCode),
Ian Rogersd36c52e2012-04-09 16:29:25 -07001431 rValue, rArrayClass);
1432 // Redo loadValues in case they didn't survive the call.
1433 loadValueDirectFixed(cUnit, rlArray, rArray); // Reload array
1434 loadValueDirectFixed(cUnit, rlIndex, rIndex); // Reload index
1435 loadValueDirectFixed(cUnit, rlSrc, rValue); // Reload value
1436 rArrayClass = INVALID_REG;
buzbee31a4a6f2012-02-28 15:36:15 -08001437
Ian Rogersd36c52e2012-04-09 16:29:25 -07001438 // Branch here if value to be stored == null
1439 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
1440 null_value_check->target = target;
buzbee31a4a6f2012-02-28 15:36:15 -08001441
Ian Rogersb41b33b2012-03-20 14:22:54 -07001442#if defined(TARGET_X86)
Ian Rogersd36c52e2012-04-09 16:29:25 -07001443 // make an extra temp available for card mark below
1444 oatFreeTemp(cUnit, rARG1);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001445 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1446 /* if (rlIndex >= [rlArray + lenOffset]) goto kThrowArrayBounds */
Ian Rogersd36c52e2012-04-09 16:29:25 -07001447 genRegMemCheck(cUnit, kCondUge, rIndex, rArray,
Ian Rogersb41b33b2012-03-20 14:22:54 -07001448 lenOffset, mir, kThrowArrayBounds);
1449 }
Ian Rogersd36c52e2012-04-09 16:29:25 -07001450 storeBaseIndexedDisp(cUnit, NULL, rArray, rIndex, scale,
1451 dataOffset, rValue, INVALID_REG, kWord,
Ian Rogersb41b33b2012-03-20 14:22:54 -07001452 INVALID_SREG);
1453#else
buzbee239c4e72012-03-16 08:42:29 -07001454 bool needsRangeCheck = (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK));
1455 int regLen = INVALID_REG;
1456 if (needsRangeCheck) {
Ian Rogersd36c52e2012-04-09 16:29:25 -07001457 regLen = rARG1;
1458 loadWordDisp(cUnit, rlArray.lowReg, lenOffset, regLen); // Get len
buzbee239c4e72012-03-16 08:42:29 -07001459 }
Ian Rogersd36c52e2012-04-09 16:29:25 -07001460 /* rPtr -> array data */
1461 int rPtr = oatAllocTemp(cUnit);
1462 opRegRegImm(cUnit, kOpAdd, rPtr, rArray, dataOffset);
buzbee239c4e72012-03-16 08:42:29 -07001463 if (needsRangeCheck) {
Ian Rogersd36c52e2012-04-09 16:29:25 -07001464 genRegRegCheck(cUnit, kCondCs, rIndex, regLen, mir,
buzbee31a4a6f2012-02-28 15:36:15 -08001465 kThrowArrayBounds);
buzbee31a4a6f2012-02-28 15:36:15 -08001466 }
Ian Rogersd36c52e2012-04-09 16:29:25 -07001467 storeBaseIndexed(cUnit, rPtr, rIndex, rValue, scale, kWord);
1468 oatFreeTemp(cUnit, rPtr);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001469#endif
Ian Rogersd36c52e2012-04-09 16:29:25 -07001470 oatFreeTemp(cUnit, rIndex);
1471 markGCCard(cUnit, rValue, rArray);
buzbee31a4a6f2012-02-28 15:36:15 -08001472}
1473
1474/*
1475 * Generate array load
1476 */
1477void genArrayGet(CompilationUnit* cUnit, MIR* mir, OpSize size,
1478 RegLocation rlArray, RegLocation rlIndex,
1479 RegLocation rlDest, int scale)
1480{
1481 RegisterClass regClass = oatRegClassBySize(size);
1482 int lenOffset = Array::LengthOffset().Int32Value();
1483 int dataOffset;
1484 RegLocation rlResult;
1485 rlArray = loadValue(cUnit, rlArray, kCoreReg);
1486 rlIndex = loadValue(cUnit, rlIndex, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001487
1488 if (size == kLong || size == kDouble) {
1489 dataOffset = Array::DataOffset(sizeof(int64_t)).Int32Value();
1490 } else {
1491 dataOffset = Array::DataOffset(sizeof(int32_t)).Int32Value();
1492 }
1493
1494 /* null object? */
1495 genNullCheck(cUnit, rlArray.sRegLow, rlArray.lowReg, mir);
1496
Ian Rogersb5d09b22012-03-06 22:14:17 -08001497#if defined(TARGET_X86)
1498 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1499 /* if (rlIndex >= [rlArray + lenOffset]) goto kThrowArrayBounds */
1500 genRegMemCheck(cUnit, kCondUge, rlIndex.lowReg, rlArray.lowReg,
1501 lenOffset, mir, kThrowArrayBounds);
1502 }
1503 if ((size == kLong) || (size == kDouble)) {
1504 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1505 loadBaseIndexedDisp(cUnit, NULL, rlArray.lowReg, rlIndex.lowReg, scale, dataOffset,
1506 rlResult.lowReg, rlResult.highReg, size, INVALID_SREG);
buzbee31a4a6f2012-02-28 15:36:15 -08001507
Ian Rogersb5d09b22012-03-06 22:14:17 -08001508 storeValueWide(cUnit, rlDest, rlResult);
1509 } else {
1510 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1511
1512 loadBaseIndexedDisp(cUnit, NULL, rlArray.lowReg, rlIndex.lowReg, scale, dataOffset,
1513 rlResult.lowReg, INVALID_REG, size, INVALID_SREG);
1514
1515 storeValue(cUnit, rlDest, rlResult);
1516 }
1517#else
1518 int regPtr = oatAllocTemp(cUnit);
buzbee239c4e72012-03-16 08:42:29 -07001519 bool needsRangeCheck = (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK));
1520 int regLen = INVALID_REG;
1521 if (needsRangeCheck) {
1522 regLen = oatAllocTemp(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001523 /* Get len */
1524 loadWordDisp(cUnit, rlArray.lowReg, lenOffset, regLen);
buzbee31a4a6f2012-02-28 15:36:15 -08001525 }
buzbee239c4e72012-03-16 08:42:29 -07001526 /* regPtr -> array data */
1527 opRegRegImm(cUnit, kOpAdd, regPtr, rlArray.lowReg, dataOffset);
buzbee31a4a6f2012-02-28 15:36:15 -08001528 oatFreeTemp(cUnit, rlArray.lowReg);
1529 if ((size == kLong) || (size == kDouble)) {
1530 if (scale) {
1531 int rNewIndex = oatAllocTemp(cUnit);
1532 opRegRegImm(cUnit, kOpLsl, rNewIndex, rlIndex.lowReg, scale);
1533 opRegReg(cUnit, kOpAdd, regPtr, rNewIndex);
1534 oatFreeTemp(cUnit, rNewIndex);
1535 } else {
1536 opRegReg(cUnit, kOpAdd, regPtr, rlIndex.lowReg);
1537 }
1538 oatFreeTemp(cUnit, rlIndex.lowReg);
1539 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1540
buzbee239c4e72012-03-16 08:42:29 -07001541 if (needsRangeCheck) {
1542 // TODO: change kCondCS to a more meaningful name, is the sense of
1543 // carry-set/clear flipped?
1544 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1545 kThrowArrayBounds);
1546 oatFreeTemp(cUnit, regLen);
1547 }
buzbee31a4a6f2012-02-28 15:36:15 -08001548 loadPair(cUnit, regPtr, rlResult.lowReg, rlResult.highReg);
1549
1550 oatFreeTemp(cUnit, regPtr);
1551 storeValueWide(cUnit, rlDest, rlResult);
1552 } else {
1553 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1554
buzbee239c4e72012-03-16 08:42:29 -07001555 if (needsRangeCheck) {
1556 // TODO: change kCondCS to a more meaningful name, is the sense of
1557 // carry-set/clear flipped?
1558 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1559 kThrowArrayBounds);
1560 oatFreeTemp(cUnit, regLen);
1561 }
buzbee31a4a6f2012-02-28 15:36:15 -08001562 loadBaseIndexed(cUnit, regPtr, rlIndex.lowReg, rlResult.lowReg,
1563 scale, size);
1564
1565 oatFreeTemp(cUnit, regPtr);
1566 storeValue(cUnit, rlDest, rlResult);
1567 }
Ian Rogersb5d09b22012-03-06 22:14:17 -08001568#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001569}
1570
1571/*
1572 * Generate array store
1573 *
1574 */
1575void genArrayPut(CompilationUnit* cUnit, MIR* mir, OpSize size,
1576 RegLocation rlArray, RegLocation rlIndex,
1577 RegLocation rlSrc, int scale)
1578{
1579 RegisterClass regClass = oatRegClassBySize(size);
1580 int lenOffset = Array::LengthOffset().Int32Value();
1581 int dataOffset;
1582
1583 if (size == kLong || size == kDouble) {
1584 dataOffset = Array::DataOffset(sizeof(int64_t)).Int32Value();
1585 } else {
1586 dataOffset = Array::DataOffset(sizeof(int32_t)).Int32Value();
1587 }
1588
buzbee31a4a6f2012-02-28 15:36:15 -08001589 rlArray = loadValue(cUnit, rlArray, kCoreReg);
1590 rlIndex = loadValue(cUnit, rlIndex, kCoreReg);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001591#if !defined(TARGET_X86)
1592 int regPtr;
buzbee31a4a6f2012-02-28 15:36:15 -08001593 if (oatIsTemp(cUnit, rlArray.lowReg)) {
1594 oatClobber(cUnit, rlArray.lowReg);
1595 regPtr = rlArray.lowReg;
1596 } else {
1597 regPtr = oatAllocTemp(cUnit);
buzbee82488f52012-03-02 08:20:26 -08001598 opRegCopy(cUnit, regPtr, rlArray.lowReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001599 }
Ian Rogersb41b33b2012-03-20 14:22:54 -07001600#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001601
1602 /* null object? */
1603 genNullCheck(cUnit, rlArray.sRegLow, rlArray.lowReg, mir);
1604
Ian Rogersb41b33b2012-03-20 14:22:54 -07001605#if defined(TARGET_X86)
1606 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1607 /* if (rlIndex >= [rlArray + lenOffset]) goto kThrowArrayBounds */
1608 genRegMemCheck(cUnit, kCondUge, rlIndex.lowReg, rlArray.lowReg,
1609 lenOffset, mir, kThrowArrayBounds);
1610 }
Ian Rogersc6f3bb82012-03-21 20:40:33 -07001611 if ((size == kLong) || (size == kDouble)) {
1612 rlSrc = loadValueWide(cUnit, rlSrc, regClass);
1613 } else {
1614 rlSrc = loadValue(cUnit, rlSrc, regClass);
1615 }
Ian Rogersb41b33b2012-03-20 14:22:54 -07001616 storeBaseIndexedDisp(cUnit, NULL, rlArray.lowReg, rlIndex.lowReg, scale, dataOffset,
1617 rlSrc.lowReg, rlSrc.highReg, size, INVALID_SREG);
1618#else
buzbee239c4e72012-03-16 08:42:29 -07001619 bool needsRangeCheck = (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK));
1620 int regLen = INVALID_REG;
1621 if (needsRangeCheck) {
1622 regLen = oatAllocTemp(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001623 //NOTE: max live temps(4) here.
1624 /* Get len */
1625 loadWordDisp(cUnit, rlArray.lowReg, lenOffset, regLen);
buzbee31a4a6f2012-02-28 15:36:15 -08001626 }
buzbee239c4e72012-03-16 08:42:29 -07001627 /* regPtr -> array data */
1628 opRegImm(cUnit, kOpAdd, regPtr, dataOffset);
buzbee31a4a6f2012-02-28 15:36:15 -08001629 /* at this point, regPtr points to array, 2 live temps */
1630 if ((size == kLong) || (size == kDouble)) {
1631 //TUNING: specific wide routine that can handle fp regs
1632 if (scale) {
1633 int rNewIndex = oatAllocTemp(cUnit);
1634 opRegRegImm(cUnit, kOpLsl, rNewIndex, rlIndex.lowReg, scale);
1635 opRegReg(cUnit, kOpAdd, regPtr, rNewIndex);
1636 oatFreeTemp(cUnit, rNewIndex);
1637 } else {
1638 opRegReg(cUnit, kOpAdd, regPtr, rlIndex.lowReg);
1639 }
1640 rlSrc = loadValueWide(cUnit, rlSrc, regClass);
1641
buzbee239c4e72012-03-16 08:42:29 -07001642 if (needsRangeCheck) {
1643 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1644 kThrowArrayBounds);
1645 oatFreeTemp(cUnit, regLen);
1646 }
1647
buzbee31a4a6f2012-02-28 15:36:15 -08001648 storePair(cUnit, regPtr, rlSrc.lowReg, rlSrc.highReg);
1649
1650 oatFreeTemp(cUnit, regPtr);
1651 } else {
1652 rlSrc = loadValue(cUnit, rlSrc, regClass);
buzbee239c4e72012-03-16 08:42:29 -07001653 if (needsRangeCheck) {
1654 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1655 kThrowArrayBounds);
1656 oatFreeTemp(cUnit, regLen);
1657 }
buzbee31a4a6f2012-02-28 15:36:15 -08001658 storeBaseIndexed(cUnit, regPtr, rlIndex.lowReg, rlSrc.lowReg,
1659 scale, size);
1660 }
Ian Rogersb41b33b2012-03-20 14:22:54 -07001661#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001662}
1663
1664void genLong3Addr(CompilationUnit* cUnit, MIR* mir, OpKind firstOp,
1665 OpKind secondOp, RegLocation rlDest,
1666 RegLocation rlSrc1, RegLocation rlSrc2)
1667{
1668 RegLocation rlResult;
1669#if defined(TARGET_ARM)
1670 /*
1671 * NOTE: This is the one place in the code in which we might have
1672 * as many as six live temporary registers. There are 5 in the normal
1673 * set for Arm. Until we have spill capabilities, temporarily add
1674 * lr to the temp set. It is safe to do this locally, but note that
1675 * lr is used explicitly elsewhere in the code generator and cannot
1676 * normally be used as a general temp register.
1677 */
1678 oatMarkTemp(cUnit, rLR); // Add lr to the temp pool
1679 oatFreeTemp(cUnit, rLR); // and make it available
1680#endif
1681 rlSrc1 = loadValueWide(cUnit, rlSrc1, kCoreReg);
1682 rlSrc2 = loadValueWide(cUnit, rlSrc2, kCoreReg);
1683 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1684 // The longs may overlap - use intermediate temp if so
1685 if (rlResult.lowReg == rlSrc1.highReg) {
1686 int tReg = oatAllocTemp(cUnit);
buzbee82488f52012-03-02 08:20:26 -08001687 opRegCopy(cUnit, tReg, rlSrc1.highReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001688 opRegRegReg(cUnit, firstOp, rlResult.lowReg, rlSrc1.lowReg,
1689 rlSrc2.lowReg);
1690 opRegRegReg(cUnit, secondOp, rlResult.highReg, tReg,
1691 rlSrc2.highReg);
1692 oatFreeTemp(cUnit, tReg);
1693 } else {
1694 opRegRegReg(cUnit, firstOp, rlResult.lowReg, rlSrc1.lowReg,
1695 rlSrc2.lowReg);
1696 opRegRegReg(cUnit, secondOp, rlResult.highReg, rlSrc1.highReg,
1697 rlSrc2.highReg);
1698 }
1699 /*
1700 * NOTE: If rlDest refers to a frame variable in a large frame, the
1701 * following storeValueWide might need to allocate a temp register.
1702 * To further work around the lack of a spill capability, explicitly
1703 * free any temps from rlSrc1 & rlSrc2 that aren't still live in rlResult.
1704 * Remove when spill is functional.
1705 */
1706 freeRegLocTemps(cUnit, rlResult, rlSrc1);
1707 freeRegLocTemps(cUnit, rlResult, rlSrc2);
1708 storeValueWide(cUnit, rlDest, rlResult);
1709#if defined(TARGET_ARM)
1710 oatClobber(cUnit, rLR);
1711 oatUnmarkTemp(cUnit, rLR); // Remove lr from the temp pool
1712#endif
1713}
1714
1715
1716bool genShiftOpLong(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1717 RegLocation rlSrc1, RegLocation rlShift)
1718{
1719 int funcOffset;
1720
Elliott Hughesb25c3f62012-03-26 16:35:06 -07001721 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001722 case Instruction::SHL_LONG:
1723 case Instruction::SHL_LONG_2ADDR:
Ian Rogers57b86d42012-03-27 16:05:41 -07001724 funcOffset = ENTRYPOINT_OFFSET(pShlLong);
buzbee31a4a6f2012-02-28 15:36:15 -08001725 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001726 case Instruction::SHR_LONG:
1727 case Instruction::SHR_LONG_2ADDR:
Ian Rogers57b86d42012-03-27 16:05:41 -07001728 funcOffset = ENTRYPOINT_OFFSET(pShrLong);
buzbee31a4a6f2012-02-28 15:36:15 -08001729 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001730 case Instruction::USHR_LONG:
1731 case Instruction::USHR_LONG_2ADDR:
Ian Rogers57b86d42012-03-27 16:05:41 -07001732 funcOffset = ENTRYPOINT_OFFSET(pUshrLong);
buzbee31a4a6f2012-02-28 15:36:15 -08001733 break;
1734 default:
1735 LOG(FATAL) << "Unexpected case";
1736 return true;
1737 }
1738 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogersab2b55d2012-03-18 00:06:11 -07001739 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset, rlSrc1, rlShift);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001740 RegLocation rlResult = oatGetReturnWide(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001741 storeValueWide(cUnit, rlDest, rlResult);
1742 return false;
1743}
1744
1745
1746bool genArithOpInt(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1747 RegLocation rlSrc1, RegLocation rlSrc2)
1748{
1749 OpKind op = kOpBkpt;
1750 bool callOut = false;
1751 bool checkZero = false;
1752 bool unary = false;
buzbee31a4a6f2012-02-28 15:36:15 -08001753 RegLocation rlResult;
1754 bool shiftOp = false;
Ian Rogersb5d09b22012-03-06 22:14:17 -08001755 int funcOffset;
1756 int retReg = rRET0;
buzbee31a4a6f2012-02-28 15:36:15 -08001757 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001758 case Instruction::NEG_INT:
buzbee31a4a6f2012-02-28 15:36:15 -08001759 op = kOpNeg;
1760 unary = true;
1761 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001762 case Instruction::NOT_INT:
buzbee31a4a6f2012-02-28 15:36:15 -08001763 op = kOpMvn;
1764 unary = true;
1765 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001766 case Instruction::ADD_INT:
1767 case Instruction::ADD_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001768 op = kOpAdd;
1769 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001770 case Instruction::SUB_INT:
1771 case Instruction::SUB_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001772 op = kOpSub;
1773 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001774 case Instruction::MUL_INT:
1775 case Instruction::MUL_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001776 op = kOpMul;
1777 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001778 case Instruction::DIV_INT:
1779 case Instruction::DIV_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001780 checkZero = true;
Ian Rogersb5d09b22012-03-06 22:14:17 -08001781 op = kOpDiv;
1782 callOut = true;
jeffhao174651d2012-04-19 15:27:22 -07001783 funcOffset = ENTRYPOINT_OFFSET(pIdivmod);
buzbee31a4a6f2012-02-28 15:36:15 -08001784 retReg = rRET0;
1785 break;
buzbee5de34942012-03-01 14:51:57 -08001786 /* NOTE: returns in rARG1 */
Elliott Hughesadb8c672012-03-06 16:49:32 -08001787 case Instruction::REM_INT:
1788 case Instruction::REM_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001789 checkZero = true;
Ian Rogersb5d09b22012-03-06 22:14:17 -08001790 op = kOpRem;
1791 callOut = true;
Ian Rogers57b86d42012-03-27 16:05:41 -07001792 funcOffset = ENTRYPOINT_OFFSET(pIdivmod);
buzbee31a4a6f2012-02-28 15:36:15 -08001793 retReg = rRET1;
1794 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001795 case Instruction::AND_INT:
1796 case Instruction::AND_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001797 op = kOpAnd;
1798 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001799 case Instruction::OR_INT:
1800 case Instruction::OR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001801 op = kOpOr;
1802 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001803 case Instruction::XOR_INT:
1804 case Instruction::XOR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001805 op = kOpXor;
1806 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001807 case Instruction::SHL_INT:
1808 case Instruction::SHL_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001809 shiftOp = true;
1810 op = kOpLsl;
1811 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001812 case Instruction::SHR_INT:
1813 case Instruction::SHR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001814 shiftOp = true;
1815 op = kOpAsr;
1816 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001817 case Instruction::USHR_INT:
1818 case Instruction::USHR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001819 shiftOp = true;
1820 op = kOpLsr;
1821 break;
1822 default:
1823 LOG(FATAL) << "Invalid word arith op: " <<
1824 (int)mir->dalvikInsn.opcode;
1825 }
1826 if (!callOut) {
buzbee31a4a6f2012-02-28 15:36:15 -08001827 if (unary) {
Ian Rogers7caad772012-03-30 01:07:54 -07001828 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001829 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1830 opRegReg(cUnit, op, rlResult.lowReg,
1831 rlSrc1.lowReg);
1832 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08001833 if (shiftOp) {
Ian Rogers7caad772012-03-30 01:07:54 -07001834#if !defined(TARGET_X86)
1835 rlSrc2 = loadValue(cUnit, rlSrc2, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001836 int tReg = oatAllocTemp(cUnit);
1837 opRegRegImm(cUnit, kOpAnd, tReg, rlSrc2.lowReg, 31);
Ian Rogers7caad772012-03-30 01:07:54 -07001838#else
1839 // X86 doesn't require masking and must use ECX
1840 loadValueDirectFixed(cUnit, rlSrc2, rCX);
1841 int tReg = rCX;
1842#endif
1843 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001844 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1845 opRegRegReg(cUnit, op, rlResult.lowReg,
1846 rlSrc1.lowReg, tReg);
1847 oatFreeTemp(cUnit, tReg);
1848 } else {
Ian Rogers7caad772012-03-30 01:07:54 -07001849 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
1850 rlSrc2 = loadValue(cUnit, rlSrc2, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001851 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1852 opRegRegReg(cUnit, op, rlResult.lowReg,
1853 rlSrc1.lowReg, rlSrc2.lowReg);
1854 }
1855 }
1856 storeValue(cUnit, rlDest, rlResult);
1857 } else {
1858 RegLocation rlResult;
1859 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogers55bd45f2012-04-04 17:31:20 -07001860 loadValueDirectFixed(cUnit, rlSrc2, rARG1);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001861#if !defined(TARGET_X86)
buzbee31a4a6f2012-02-28 15:36:15 -08001862 int rTgt = loadHelper(cUnit, funcOffset);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001863#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001864 loadValueDirectFixed(cUnit, rlSrc1, rARG0);
1865 if (checkZero) {
1866 genImmedCheck(cUnit, kCondEq, rARG1, 0, mir, kThrowDivZero);
1867 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07001868#if !defined(TARGET_X86)
1869 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001870 oatFreeTemp(cUnit, rTgt);
1871#else
Ian Rogersab2b55d2012-03-18 00:06:11 -07001872 opThreadMem(cUnit, kOpBlx, funcOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001873#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001874 if (retReg == rRET0)
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001875 rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001876 else
1877 rlResult = oatGetReturnAlt(cUnit);
1878 storeValue(cUnit, rlDest, rlResult);
1879 }
1880 return false;
1881}
1882
1883/*
1884 * The following are the first-level codegen routines that analyze the format
1885 * of each bytecode then either dispatch special purpose codegen routines
1886 * or produce corresponding Thumb instructions directly.
1887 */
1888
1889bool isPowerOfTwo(int x)
1890{
1891 return (x & (x - 1)) == 0;
1892}
1893
1894// Returns true if no more than two bits are set in 'x'.
1895bool isPopCountLE2(unsigned int x)
1896{
1897 x &= x - 1;
1898 return (x & (x - 1)) == 0;
1899}
1900
1901// Returns the index of the lowest set bit in 'x'.
1902int lowestSetBit(unsigned int x) {
1903 int bit_posn = 0;
1904 while ((x & 0xf) == 0) {
1905 bit_posn += 4;
1906 x >>= 4;
1907 }
1908 while ((x & 1) == 0) {
1909 bit_posn++;
1910 x >>= 1;
1911 }
1912 return bit_posn;
1913}
1914
1915// Returns true if it added instructions to 'cUnit' to divide 'rlSrc' by 'lit'
1916// and store the result in 'rlDest'.
Elliott Hughesadb8c672012-03-06 16:49:32 -08001917bool handleEasyDivide(CompilationUnit* cUnit, Instruction::Code dalvikOpcode,
buzbee31a4a6f2012-02-28 15:36:15 -08001918 RegLocation rlSrc, RegLocation rlDest, int lit)
1919{
buzbeef3aac972012-04-11 16:33:36 -07001920#if defined(TARGET_ARM)
1921 // No divide instruction for Arm, so check for more special cases
1922 if (lit < 2) {
1923 return false;
1924 }
1925 if (!isPowerOfTwo(lit)) {
1926 return smallLiteralDivide(cUnit, dalvikOpcode, rlSrc, rlDest, lit);
1927 }
1928#else
buzbee31a4a6f2012-02-28 15:36:15 -08001929 if (lit < 2 || !isPowerOfTwo(lit)) {
1930 return false;
1931 }
buzbeef3aac972012-04-11 16:33:36 -07001932#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001933 int k = lowestSetBit(lit);
1934 if (k >= 30) {
1935 // Avoid special cases.
1936 return false;
1937 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08001938 bool div = (dalvikOpcode == Instruction::DIV_INT_LIT8 ||
1939 dalvikOpcode == Instruction::DIV_INT_LIT16);
buzbee31a4a6f2012-02-28 15:36:15 -08001940 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
1941 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1942 if (div) {
1943 int tReg = oatAllocTemp(cUnit);
1944 if (lit == 2) {
1945 // Division by 2 is by far the most common division by constant.
1946 opRegRegImm(cUnit, kOpLsr, tReg, rlSrc.lowReg, 32 - k);
1947 opRegRegReg(cUnit, kOpAdd, tReg, tReg, rlSrc.lowReg);
1948 opRegRegImm(cUnit, kOpAsr, rlResult.lowReg, tReg, k);
1949 } else {
1950 opRegRegImm(cUnit, kOpAsr, tReg, rlSrc.lowReg, 31);
1951 opRegRegImm(cUnit, kOpLsr, tReg, tReg, 32 - k);
1952 opRegRegReg(cUnit, kOpAdd, tReg, tReg, rlSrc.lowReg);
1953 opRegRegImm(cUnit, kOpAsr, rlResult.lowReg, tReg, k);
1954 }
1955 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08001956 int tReg1 = oatAllocTemp(cUnit);
1957 int tReg2 = oatAllocTemp(cUnit);
1958 if (lit == 2) {
1959 opRegRegImm(cUnit, kOpLsr, tReg1, rlSrc.lowReg, 32 - k);
1960 opRegRegReg(cUnit, kOpAdd, tReg2, tReg1, rlSrc.lowReg);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001961 opRegRegImm(cUnit, kOpAnd, tReg2, tReg2, lit -1);
buzbee31a4a6f2012-02-28 15:36:15 -08001962 opRegRegReg(cUnit, kOpSub, rlResult.lowReg, tReg2, tReg1);
1963 } else {
1964 opRegRegImm(cUnit, kOpAsr, tReg1, rlSrc.lowReg, 31);
1965 opRegRegImm(cUnit, kOpLsr, tReg1, tReg1, 32 - k);
1966 opRegRegReg(cUnit, kOpAdd, tReg2, tReg1, rlSrc.lowReg);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001967 opRegRegImm(cUnit, kOpAnd, tReg2, tReg2, lit - 1);
buzbee31a4a6f2012-02-28 15:36:15 -08001968 opRegRegReg(cUnit, kOpSub, rlResult.lowReg, tReg2, tReg1);
1969 }
1970 }
1971 storeValue(cUnit, rlDest, rlResult);
1972 return true;
1973}
1974
1975void genMultiplyByTwoBitMultiplier(CompilationUnit* cUnit, RegLocation rlSrc,
1976 RegLocation rlResult, int lit,
1977 int firstBit, int secondBit)
1978{
buzbee0398c422012-03-02 15:22:47 -08001979#if defined(TARGET_ARM)
buzbee31a4a6f2012-02-28 15:36:15 -08001980 opRegRegRegShift(cUnit, kOpAdd, rlResult.lowReg, rlSrc.lowReg, rlSrc.lowReg,
1981 encodeShift(kArmLsl, secondBit - firstBit));
buzbee0398c422012-03-02 15:22:47 -08001982#else
1983 int tReg = oatAllocTemp(cUnit);
1984 opRegRegImm(cUnit, kOpLsl, tReg, rlSrc.lowReg, secondBit - firstBit);
1985 opRegRegReg(cUnit, kOpAdd, rlResult.lowReg, rlSrc.lowReg, tReg);
1986 oatFreeTemp(cUnit, tReg);
buzbee5de34942012-03-01 14:51:57 -08001987#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001988 if (firstBit != 0) {
1989 opRegRegImm(cUnit, kOpLsl, rlResult.lowReg, rlResult.lowReg, firstBit);
1990 }
1991}
1992
1993// Returns true if it added instructions to 'cUnit' to multiply 'rlSrc' by 'lit'
1994// and store the result in 'rlDest'.
1995bool handleEasyMultiply(CompilationUnit* cUnit, RegLocation rlSrc,
1996 RegLocation rlDest, int lit)
1997{
1998 // Can we simplify this multiplication?
1999 bool powerOfTwo = false;
2000 bool popCountLE2 = false;
2001 bool powerOfTwoMinusOne = false;
2002 if (lit < 2) {
2003 // Avoid special cases.
2004 return false;
2005 } else if (isPowerOfTwo(lit)) {
2006 powerOfTwo = true;
2007 } else if (isPopCountLE2(lit)) {
2008 popCountLE2 = true;
2009 } else if (isPowerOfTwo(lit + 1)) {
2010 powerOfTwoMinusOne = true;
2011 } else {
2012 return false;
2013 }
2014 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
2015 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2016 if (powerOfTwo) {
2017 // Shift.
2018 opRegRegImm(cUnit, kOpLsl, rlResult.lowReg, rlSrc.lowReg,
2019 lowestSetBit(lit));
2020 } else if (popCountLE2) {
2021 // Shift and add and shift.
2022 int firstBit = lowestSetBit(lit);
2023 int secondBit = lowestSetBit(lit ^ (1 << firstBit));
2024 genMultiplyByTwoBitMultiplier(cUnit, rlSrc, rlResult, lit,
2025 firstBit, secondBit);
2026 } else {
2027 // Reverse subtract: (src << (shift + 1)) - src.
2028 DCHECK(powerOfTwoMinusOne);
2029 // TUNING: rsb dst, src, src lsl#lowestSetBit(lit + 1)
2030 int tReg = oatAllocTemp(cUnit);
2031 opRegRegImm(cUnit, kOpLsl, tReg, rlSrc.lowReg, lowestSetBit(lit + 1));
2032 opRegRegReg(cUnit, kOpSub, rlResult.lowReg, tReg, rlSrc.lowReg);
2033 }
2034 storeValue(cUnit, rlDest, rlResult);
2035 return true;
2036}
2037
2038bool genArithOpIntLit(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
2039 RegLocation rlSrc, int lit)
2040{
Elliott Hughesadb8c672012-03-06 16:49:32 -08002041 Instruction::Code dalvikOpcode = mir->dalvikInsn.opcode;
buzbee31a4a6f2012-02-28 15:36:15 -08002042 RegLocation rlResult;
2043 OpKind op = (OpKind)0; /* Make gcc happy */
2044 int shiftOp = false;
2045 bool isDiv = false;
2046 int funcOffset;
buzbee31a4a6f2012-02-28 15:36:15 -08002047
2048 switch (dalvikOpcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002049 case Instruction::RSUB_INT_LIT8:
2050 case Instruction::RSUB_INT: {
buzbee31a4a6f2012-02-28 15:36:15 -08002051 int tReg;
2052 //TUNING: add support for use of Arm rsub op
2053 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
2054 tReg = oatAllocTemp(cUnit);
2055 loadConstant(cUnit, tReg, lit);
2056 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2057 opRegRegReg(cUnit, kOpSub, rlResult.lowReg,
2058 tReg, rlSrc.lowReg);
2059 storeValue(cUnit, rlDest, rlResult);
2060 return false;
2061 break;
2062 }
2063
Elliott Hughesadb8c672012-03-06 16:49:32 -08002064 case Instruction::ADD_INT_LIT8:
2065 case Instruction::ADD_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002066 op = kOpAdd;
2067 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002068 case Instruction::MUL_INT_LIT8:
2069 case Instruction::MUL_INT_LIT16: {
buzbee31a4a6f2012-02-28 15:36:15 -08002070 if (handleEasyMultiply(cUnit, rlSrc, rlDest, lit)) {
2071 return false;
2072 }
2073 op = kOpMul;
2074 break;
2075 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08002076 case Instruction::AND_INT_LIT8:
2077 case Instruction::AND_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002078 op = kOpAnd;
2079 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002080 case Instruction::OR_INT_LIT8:
2081 case Instruction::OR_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002082 op = kOpOr;
2083 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002084 case Instruction::XOR_INT_LIT8:
2085 case Instruction::XOR_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002086 op = kOpXor;
2087 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002088 case Instruction::SHL_INT_LIT8:
buzbee31a4a6f2012-02-28 15:36:15 -08002089 lit &= 31;
2090 shiftOp = true;
2091 op = kOpLsl;
2092 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002093 case Instruction::SHR_INT_LIT8:
buzbee31a4a6f2012-02-28 15:36:15 -08002094 lit &= 31;
2095 shiftOp = true;
2096 op = kOpAsr;
2097 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002098 case Instruction::USHR_INT_LIT8:
buzbee31a4a6f2012-02-28 15:36:15 -08002099 lit &= 31;
2100 shiftOp = true;
2101 op = kOpLsr;
2102 break;
2103
Elliott Hughesadb8c672012-03-06 16:49:32 -08002104 case Instruction::DIV_INT_LIT8:
2105 case Instruction::DIV_INT_LIT16:
2106 case Instruction::REM_INT_LIT8:
2107 case Instruction::REM_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002108 if (lit == 0) {
2109 genImmedCheck(cUnit, kCondAl, 0, 0, mir, kThrowDivZero);
2110 return false;
2111 }
2112 if (handleEasyDivide(cUnit, dalvikOpcode, rlSrc, rlDest, lit)) {
2113 return false;
2114 }
2115 oatFlushAllRegs(cUnit); /* Everything to home location */
2116 loadValueDirectFixed(cUnit, rlSrc, rARG0);
2117 oatClobber(cUnit, rARG0);
jeffhao174651d2012-04-19 15:27:22 -07002118 funcOffset = ENTRYPOINT_OFFSET(pIdivmod);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002119 if ((dalvikOpcode == Instruction::DIV_INT_LIT8) ||
2120 (dalvikOpcode == Instruction::DIV_INT_LIT16)) {
buzbee31a4a6f2012-02-28 15:36:15 -08002121 isDiv = true;
2122 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08002123 isDiv = false;
2124 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07002125 callRuntimeHelperRegImm(cUnit, funcOffset, rARG0, lit);
buzbee31a4a6f2012-02-28 15:36:15 -08002126 if (isDiv)
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002127 rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08002128 else
2129 rlResult = oatGetReturnAlt(cUnit);
2130 storeValue(cUnit, rlDest, rlResult);
2131 return false;
2132 break;
2133 default:
2134 return true;
2135 }
2136 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
2137 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2138 // Avoid shifts by literal 0 - no support in Thumb. Change to copy
2139 if (shiftOp && (lit == 0)) {
buzbee82488f52012-03-02 08:20:26 -08002140 opRegCopy(cUnit, rlResult.lowReg, rlSrc.lowReg);
buzbee31a4a6f2012-02-28 15:36:15 -08002141 } else {
2142 opRegRegImm(cUnit, op, rlResult.lowReg, rlSrc.lowReg, lit);
2143 }
2144 storeValue(cUnit, rlDest, rlResult);
2145 return false;
2146}
2147
2148bool genArithOpLong(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
2149 RegLocation rlSrc1, RegLocation rlSrc2)
2150{
2151 RegLocation rlResult;
2152 OpKind firstOp = kOpBkpt;
2153 OpKind secondOp = kOpBkpt;
2154 bool callOut = false;
2155 bool checkZero = false;
2156 int funcOffset;
2157 int retReg = rRET0;
2158
2159 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002160 case Instruction::NOT_LONG:
buzbee31a4a6f2012-02-28 15:36:15 -08002161 rlSrc2 = loadValueWide(cUnit, rlSrc2, kCoreReg);
2162 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2163 // Check for destructive overlap
2164 if (rlResult.lowReg == rlSrc2.highReg) {
2165 int tReg = oatAllocTemp(cUnit);
buzbee82488f52012-03-02 08:20:26 -08002166 opRegCopy(cUnit, tReg, rlSrc2.highReg);
buzbee31a4a6f2012-02-28 15:36:15 -08002167 opRegReg(cUnit, kOpMvn, rlResult.lowReg, rlSrc2.lowReg);
2168 opRegReg(cUnit, kOpMvn, rlResult.highReg, tReg);
2169 oatFreeTemp(cUnit, tReg);
2170 } else {
2171 opRegReg(cUnit, kOpMvn, rlResult.lowReg, rlSrc2.lowReg);
2172 opRegReg(cUnit, kOpMvn, rlResult.highReg, rlSrc2.highReg);
2173 }
2174 storeValueWide(cUnit, rlDest, rlResult);
2175 return false;
2176 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002177 case Instruction::ADD_LONG:
2178 case Instruction::ADD_LONG_2ADDR:
Ian Rogers7caad772012-03-30 01:07:54 -07002179#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbeec5159d52012-03-03 11:48:39 -08002180 return genAddLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2181#else
buzbee31a4a6f2012-02-28 15:36:15 -08002182 firstOp = kOpAdd;
2183 secondOp = kOpAdc;
2184 break;
buzbeec5159d52012-03-03 11:48:39 -08002185#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002186 case Instruction::SUB_LONG:
2187 case Instruction::SUB_LONG_2ADDR:
Ian Rogers7caad772012-03-30 01:07:54 -07002188#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbeec5159d52012-03-03 11:48:39 -08002189 return genSubLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
Ian Rogers7caad772012-03-30 01:07:54 -07002190#else
buzbee31a4a6f2012-02-28 15:36:15 -08002191 firstOp = kOpSub;
2192 secondOp = kOpSbc;
2193 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002194#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002195 case Instruction::MUL_LONG:
2196 case Instruction::MUL_LONG_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08002197 callOut = true;
2198 retReg = rRET0;
Ian Rogers57b86d42012-03-27 16:05:41 -07002199 funcOffset = ENTRYPOINT_OFFSET(pLmul);
buzbee31a4a6f2012-02-28 15:36:15 -08002200 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002201 case Instruction::DIV_LONG:
2202 case Instruction::DIV_LONG_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08002203 callOut = true;
2204 checkZero = true;
2205 retReg = rRET0;
Ian Rogers57b86d42012-03-27 16:05:41 -07002206 funcOffset = ENTRYPOINT_OFFSET(pLdivmod);
buzbee31a4a6f2012-02-28 15:36:15 -08002207 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002208 case Instruction::REM_LONG:
2209 case Instruction::REM_LONG_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08002210 callOut = true;
2211 checkZero = true;
Ian Rogers55bd45f2012-04-04 17:31:20 -07002212 funcOffset = ENTRYPOINT_OFFSET(pLdiv);
2213#if defined(TARGET_ARM)
2214 /* NOTE - result is in rARG2/rARG3 instead of rRET0/rRET1 */
buzbee31a4a6f2012-02-28 15:36:15 -08002215 retReg = rARG2;
Ian Rogers55bd45f2012-04-04 17:31:20 -07002216#else
2217 retReg = rRET0;
2218#endif
buzbee31a4a6f2012-02-28 15:36:15 -08002219 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002220 case Instruction::AND_LONG_2ADDR:
2221 case Instruction::AND_LONG:
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002222#if defined(TARGET_X86)
Ian Rogers7caad772012-03-30 01:07:54 -07002223 return genAndLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2224#else
buzbee31a4a6f2012-02-28 15:36:15 -08002225 firstOp = kOpAnd;
2226 secondOp = kOpAnd;
2227 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002228#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002229 case Instruction::OR_LONG:
2230 case Instruction::OR_LONG_2ADDR:
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002231#if defined(TARGET_X86)
Ian Rogers7caad772012-03-30 01:07:54 -07002232 return genOrLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2233#else
buzbee31a4a6f2012-02-28 15:36:15 -08002234 firstOp = kOpOr;
2235 secondOp = kOpOr;
2236 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002237#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002238 case Instruction::XOR_LONG:
2239 case Instruction::XOR_LONG_2ADDR:
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002240#if defined(TARGET_X86)
Ian Rogers7caad772012-03-30 01:07:54 -07002241 return genXorLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2242#else
buzbee31a4a6f2012-02-28 15:36:15 -08002243 firstOp = kOpXor;
2244 secondOp = kOpXor;
2245 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002246#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002247 case Instruction::NEG_LONG: {
buzbeec5159d52012-03-03 11:48:39 -08002248 return genNegLong(cUnit, mir, rlDest, rlSrc2);
buzbee31a4a6f2012-02-28 15:36:15 -08002249 }
2250 default:
2251 LOG(FATAL) << "Invalid long arith op";
2252 }
2253 if (!callOut) {
2254 genLong3Addr(cUnit, mir, firstOp, secondOp, rlDest, rlSrc1, rlSrc2);
2255 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08002256 oatFlushAllRegs(cUnit); /* Send everything to home location */
2257 if (checkZero) {
2258 loadValueDirectWideFixed(cUnit, rlSrc2, rARG2, rARG3);
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002259#if !defined(TARGET_X86)
Ian Rogersab2b55d2012-03-18 00:06:11 -07002260 int rTgt = loadHelper(cUnit, funcOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07002261#endif
buzbee31a4a6f2012-02-28 15:36:15 -08002262 int tReg = oatAllocTemp(cUnit);
2263#if defined(TARGET_ARM)
2264 newLIR4(cUnit, kThumb2OrrRRRs, tReg, rARG2, rARG3, 0);
2265 oatFreeTemp(cUnit, tReg);
2266 genCheck(cUnit, kCondEq, mir, kThrowDivZero);
2267#else
2268 opRegRegReg(cUnit, kOpOr, tReg, rARG2, rARG3);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002269#endif
buzbee5de34942012-03-01 14:51:57 -08002270 genImmedCheck(cUnit, kCondEq, tReg, 0, mir, kThrowDivZero);
buzbee31a4a6f2012-02-28 15:36:15 -08002271 oatFreeTemp(cUnit, tReg);
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002272 loadValueDirectWideFixed(cUnit, rlSrc1, rARG0, rARG1);
Ian Rogersab2b55d2012-03-18 00:06:11 -07002273#if !defined(TARGET_X86)
2274 opReg(cUnit, kOpBlx, rTgt);
2275 oatFreeTemp(cUnit, rTgt);
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002276#else
2277 opThreadMem(cUnit, kOpBlx, funcOffset);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002278#endif
Ian Rogersab2b55d2012-03-18 00:06:11 -07002279 } else {
2280 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset,
2281 rlSrc1, rlSrc2);
buzbee31a4a6f2012-02-28 15:36:15 -08002282 }
buzbee31a4a6f2012-02-28 15:36:15 -08002283 // Adjust return regs in to handle case of rem returning rARG2/rARG3
2284 if (retReg == rRET0)
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002285 rlResult = oatGetReturnWide(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08002286 else
2287 rlResult = oatGetReturnWideAlt(cUnit);
2288 storeValueWide(cUnit, rlDest, rlResult);
2289 }
2290 return false;
2291}
2292
2293bool genConversionCall(CompilationUnit* cUnit, MIR* mir, int funcOffset,
2294 int srcSize, int tgtSize)
2295{
2296 /*
2297 * Don't optimize the register usage since it calls out to support
2298 * functions
2299 */
2300 RegLocation rlSrc;
2301 RegLocation rlDest;
2302 oatFlushAllRegs(cUnit); /* Send everything to home location */
buzbee31a4a6f2012-02-28 15:36:15 -08002303 if (srcSize == 1) {
2304 rlSrc = oatGetSrc(cUnit, mir, 0);
2305 loadValueDirectFixed(cUnit, rlSrc, rARG0);
2306 } else {
2307 rlSrc = oatGetSrcWide(cUnit, mir, 0, 1);
2308 loadValueDirectWideFixed(cUnit, rlSrc, rARG0, rARG1);
2309 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07002310 callRuntimeHelperRegLocation(cUnit, funcOffset, rlSrc);
buzbee31a4a6f2012-02-28 15:36:15 -08002311 if (tgtSize == 1) {
2312 RegLocation rlResult;
2313 rlDest = oatGetDest(cUnit, mir, 0);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002314 rlResult = oatGetReturn(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08002315 storeValue(cUnit, rlDest, rlResult);
2316 } else {
2317 RegLocation rlResult;
2318 rlDest = oatGetDestWide(cUnit, mir, 0, 1);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002319 rlResult = oatGetReturnWide(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08002320 storeValueWide(cUnit, rlDest, rlResult);
2321 }
2322 return false;
2323}
2324
2325void genNegFloat(CompilationUnit* cUnit, RegLocation rlDest, RegLocation rlSrc);
2326bool genArithOpFloatPortable(CompilationUnit* cUnit, MIR* mir,
2327 RegLocation rlDest, RegLocation rlSrc1,
2328 RegLocation rlSrc2)
2329{
2330 RegLocation rlResult;
2331 int funcOffset;
2332
2333 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002334 case Instruction::ADD_FLOAT_2ADDR:
2335 case Instruction::ADD_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002336 funcOffset = ENTRYPOINT_OFFSET(pFadd);
buzbee31a4a6f2012-02-28 15:36:15 -08002337 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002338 case Instruction::SUB_FLOAT_2ADDR:
2339 case Instruction::SUB_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002340 funcOffset = ENTRYPOINT_OFFSET(pFsub);
buzbee31a4a6f2012-02-28 15:36:15 -08002341 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002342 case Instruction::DIV_FLOAT_2ADDR:
2343 case Instruction::DIV_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002344 funcOffset = ENTRYPOINT_OFFSET(pFdiv);
buzbee31a4a6f2012-02-28 15:36:15 -08002345 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002346 case Instruction::MUL_FLOAT_2ADDR:
2347 case Instruction::MUL_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002348 funcOffset = ENTRYPOINT_OFFSET(pFmul);
buzbee31a4a6f2012-02-28 15:36:15 -08002349 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002350 case Instruction::REM_FLOAT_2ADDR:
2351 case Instruction::REM_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002352 funcOffset = ENTRYPOINT_OFFSET(pFmodf);
buzbee31a4a6f2012-02-28 15:36:15 -08002353 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002354 case Instruction::NEG_FLOAT: {
buzbee31a4a6f2012-02-28 15:36:15 -08002355 genNegFloat(cUnit, rlDest, rlSrc1);
2356 return false;
2357 }
2358 default:
2359 return true;
2360 }
2361 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogersab2b55d2012-03-18 00:06:11 -07002362 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset, rlSrc1, rlSrc2);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002363 rlResult = oatGetReturn(cUnit, true);
buzbee31a4a6f2012-02-28 15:36:15 -08002364 storeValue(cUnit, rlDest, rlResult);
2365 return false;
2366}
2367
2368void genNegDouble(CompilationUnit* cUnit, RegLocation rlDst, RegLocation rlSrc);
2369bool genArithOpDoublePortable(CompilationUnit* cUnit, MIR* mir,
2370 RegLocation rlDest, RegLocation rlSrc1,
2371 RegLocation rlSrc2)
2372{
2373 RegLocation rlResult;
2374 int funcOffset;
2375
2376 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002377 case Instruction::ADD_DOUBLE_2ADDR:
2378 case Instruction::ADD_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002379 funcOffset = ENTRYPOINT_OFFSET(pDadd);
buzbee31a4a6f2012-02-28 15:36:15 -08002380 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002381 case Instruction::SUB_DOUBLE_2ADDR:
2382 case Instruction::SUB_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002383 funcOffset = ENTRYPOINT_OFFSET(pDsub);
buzbee31a4a6f2012-02-28 15:36:15 -08002384 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002385 case Instruction::DIV_DOUBLE_2ADDR:
2386 case Instruction::DIV_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002387 funcOffset = ENTRYPOINT_OFFSET(pDdiv);
buzbee31a4a6f2012-02-28 15:36:15 -08002388 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002389 case Instruction::MUL_DOUBLE_2ADDR:
2390 case Instruction::MUL_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002391 funcOffset = ENTRYPOINT_OFFSET(pDmul);
buzbee31a4a6f2012-02-28 15:36:15 -08002392 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002393 case Instruction::REM_DOUBLE_2ADDR:
2394 case Instruction::REM_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002395 funcOffset = ENTRYPOINT_OFFSET(pFmod);
buzbee31a4a6f2012-02-28 15:36:15 -08002396 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002397 case Instruction::NEG_DOUBLE: {
buzbee31a4a6f2012-02-28 15:36:15 -08002398 genNegDouble(cUnit, rlDest, rlSrc1);
2399 return false;
2400 }
2401 default:
2402 return true;
2403 }
2404 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogersab2b55d2012-03-18 00:06:11 -07002405 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset, rlSrc1, rlSrc2);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002406 rlResult = oatGetReturnWide(cUnit, true);
buzbee31a4a6f2012-02-28 15:36:15 -08002407 storeValueWide(cUnit, rlDest, rlResult);
2408 return false;
2409}
2410
2411bool genConversionPortable(CompilationUnit* cUnit, MIR* mir)
2412{
Elliott Hughesadb8c672012-03-06 16:49:32 -08002413 Instruction::Code opcode = mir->dalvikInsn.opcode;
buzbee31a4a6f2012-02-28 15:36:15 -08002414
2415 switch (opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002416 case Instruction::INT_TO_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002417 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pI2f),
buzbee31a4a6f2012-02-28 15:36:15 -08002418 1, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002419 case Instruction::FLOAT_TO_INT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002420 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pF2iz),
buzbee31a4a6f2012-02-28 15:36:15 -08002421 1, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002422 case Instruction::DOUBLE_TO_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002423 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pD2f),
buzbee31a4a6f2012-02-28 15:36:15 -08002424 2, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002425 case Instruction::FLOAT_TO_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002426 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pF2d),
buzbee31a4a6f2012-02-28 15:36:15 -08002427 1, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002428 case Instruction::INT_TO_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002429 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pI2d),
buzbee31a4a6f2012-02-28 15:36:15 -08002430 1, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002431 case Instruction::DOUBLE_TO_INT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002432 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pD2iz),
buzbee31a4a6f2012-02-28 15:36:15 -08002433 2, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002434 case Instruction::FLOAT_TO_LONG:
Ian Rogers57b86d42012-03-27 16:05:41 -07002435 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pF2l),
2436 1, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002437 case Instruction::LONG_TO_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002438 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pL2f),
buzbee31a4a6f2012-02-28 15:36:15 -08002439 2, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002440 case Instruction::DOUBLE_TO_LONG:
Ian Rogers57b86d42012-03-27 16:05:41 -07002441 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pD2l),
2442 2, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002443 case Instruction::LONG_TO_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002444 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pL2d),
buzbee31a4a6f2012-02-28 15:36:15 -08002445 2, 2);
2446 default:
2447 return true;
2448 }
2449 return false;
2450}
2451
2452/*
2453 * Generate callout to updateDebugger. Note that we're overloading
2454 * the use of rSUSPEND here. When the debugger is active, this
2455 * register holds the address of the update function. So, if it's
2456 * non-null, we call out to it.
2457 *
2458 * Note also that rRET0 and rRET1 must be preserved across this
2459 * code. This must be handled by the stub.
2460 */
2461void genDebuggerUpdate(CompilationUnit* cUnit, int32_t offset)
2462{
2463 // Following DCHECK verifies that dPC is in range of single load immediate
2464 DCHECK((offset == DEBUGGER_METHOD_ENTRY) ||
2465 (offset == DEBUGGER_METHOD_EXIT) || ((offset & 0xffff) == offset));
2466 oatClobberCalleeSave(cUnit);
2467#if defined(TARGET_ARM)
2468 opRegImm(cUnit, kOpCmp, rSUSPEND, 0);
buzbee82488f52012-03-02 08:20:26 -08002469 opIT(cUnit, kArmCondNe, "T");
buzbee31a4a6f2012-02-28 15:36:15 -08002470 loadConstant(cUnit, rARG2, offset); // arg2 <- Entry code
2471 opReg(cUnit, kOpBlx, rSUSPEND);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002472#elif defined(TARGET_X86)
2473 UNIMPLEMENTED(FATAL);
buzbee31a4a6f2012-02-28 15:36:15 -08002474#else
buzbee82488f52012-03-02 08:20:26 -08002475 LIR* branch = opCmpImmBranch(cUnit, kCondEq, rSUSPEND, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08002476 loadConstant(cUnit, rARG2, offset);
2477 opReg(cUnit, kOpBlx, rSUSPEND);
2478 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08002479 branch->target = (LIR*)target;
2480#endif
2481 oatFreeTemp(cUnit, rARG2);
2482}
2483
2484/* Check if we need to check for pending suspend request */
2485void genSuspendTest(CompilationUnit* cUnit, MIR* mir)
2486{
2487 if (NO_SUSPEND || (mir->optimizationFlags & MIR_IGNORE_SUSPEND_CHECK)) {
2488 return;
2489 }
2490 oatFlushAllRegs(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08002491 if (cUnit->genDebugger) {
2492 // If generating code for the debugger, always check for suspension
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07002493#if defined(TARGET_X86)
2494 UNIMPLEMENTED(FATAL);
2495#else
Ian Rogers57b86d42012-03-27 16:05:41 -07002496 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pTestSuspendFromCode));
buzbee86a4bce2012-03-06 18:15:00 -08002497 opReg(cUnit, kOpBlx, rTgt);
2498 // Refresh rSUSPEND
2499 loadWordDisp(cUnit, rSELF,
Ian Rogers57b86d42012-03-27 16:05:41 -07002500 ENTRYPOINT_OFFSET(pUpdateDebuggerFromCode),
buzbee86a4bce2012-03-06 18:15:00 -08002501 rSUSPEND);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002502#endif
buzbee31a4a6f2012-02-28 15:36:15 -08002503 } else {
Ian Rogersb5d09b22012-03-06 22:14:17 -08002504 LIR* branch = NULL;
buzbee31a4a6f2012-02-28 15:36:15 -08002505#if defined(TARGET_ARM)
2506 // In non-debug case, only check periodically
2507 newLIR2(cUnit, kThumbSubRI8, rSUSPEND, 1);
buzbee82488f52012-03-02 08:20:26 -08002508 branch = opCondBranch(cUnit, kCondEq, NULL);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002509#elif defined(TARGET_X86)
Elliott Hughescd4935f2012-04-10 16:15:59 -07002510 newLIR2(cUnit, kX86Cmp32TI8, Thread::SuspendCountOffset().Int32Value(), 0);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002511 branch = opCondBranch(cUnit, kCondNe, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08002512#else
2513 opRegImm(cUnit, kOpSub, rSUSPEND, 1);
buzbee82488f52012-03-02 08:20:26 -08002514 branch = opCmpImmBranch(cUnit, kCondEq, rSUSPEND, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08002515#endif
buzbee86a4bce2012-03-06 18:15:00 -08002516 LIR* retLab = newLIR0(cUnit, kPseudoTargetLabel);
2517 LIR* target = rawLIR(cUnit, cUnit->currentDalvikOffset,
2518 kPseudoSuspendTarget, (intptr_t)retLab, mir->offset);
2519 branch->target = (LIR*)target;
2520 oatInsertGrowableList(cUnit, &cUnit->suspendLaunchpads, (intptr_t)target);
buzbee31a4a6f2012-02-28 15:36:15 -08002521 }
buzbee31a4a6f2012-02-28 15:36:15 -08002522}
2523
buzbeefead2932012-03-30 14:02:01 -07002524/* Check if we need to check for pending suspend request */
2525void genSuspendTestAndBranch(CompilationUnit* cUnit, MIR* mir, LIR* target)
2526{
2527 if (NO_SUSPEND || (mir->optimizationFlags & MIR_IGNORE_SUSPEND_CHECK)) {
2528 opUnconditionalBranch(cUnit, target);
2529 return;
2530 }
2531 if (cUnit->genDebugger) {
2532 genSuspendTest(cUnit, mir);
2533 opUnconditionalBranch(cUnit, target);
2534 } else {
2535#if defined(TARGET_ARM)
2536 // In non-debug case, only check periodically
2537 newLIR2(cUnit, kThumbSubRI8, rSUSPEND, 1);
2538 opCondBranch(cUnit, kCondNe, target);
2539#elif defined(TARGET_X86)
Elliott Hughescd4935f2012-04-10 16:15:59 -07002540 newLIR2(cUnit, kX86Cmp32TI8, Thread::SuspendCountOffset().Int32Value(), 0);
buzbeefead2932012-03-30 14:02:01 -07002541 opCondBranch(cUnit, kCondEq, target);
2542#else
2543 opRegImm(cUnit, kOpSub, rSUSPEND, 1);
2544 opCmpImmBranch(cUnit, kCondNe, rSUSPEND, 0, target);
2545#endif
2546 LIR* launchPad = rawLIR(cUnit, cUnit->currentDalvikOffset,
2547 kPseudoSuspendTarget, (intptr_t)target, mir->offset);
2548 oatFlushAllRegs(cUnit);
2549 opUnconditionalBranch(cUnit, launchPad);
2550 oatInsertGrowableList(cUnit, &cUnit->suspendLaunchpads, (intptr_t)launchPad);
2551 }
2552}
2553
buzbee31a4a6f2012-02-28 15:36:15 -08002554} // namespace art