blob: d3ab0393253bc9f5fc5a965059acf0f50d8b0bdc [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
buzbee5de34942012-03-01 14:51:57 -0800585#else
586 int rVal = oatAllocTemp(cUnit);
587#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800588 // Set up source pointer
589 RegLocation rlFirst = oatGetSrc(cUnit, mir, 0);
Ian Rogersb5d09b22012-03-06 22:14:17 -0800590#if defined(TARGET_X86)
591 UNIMPLEMENTED(FATAL);
592#else
buzbee31a4a6f2012-02-28 15:36:15 -0800593 opRegRegImm(cUnit, kOpAdd, rSrc, rSP,
594 oatSRegOffset(cUnit, rlFirst.sRegLow));
595 // Set up the target pointer
596 opRegRegImm(cUnit, kOpAdd, rDst, rRET0,
597 Array::DataOffset(component_size).Int32Value());
Ian Rogersb5d09b22012-03-06 22:14:17 -0800598#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800599 // Set up the loop counter (known to be > 0)
600 loadConstant(cUnit, rIdx, dInsn->vA - 1);
601 // Generate the copy loop. Going backwards for convenience
602 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800603 // Copy next element
604 loadBaseIndexed(cUnit, rSrc, rIdx, rVal, 2, kWord);
605 storeBaseIndexed(cUnit, rDst, rIdx, rVal, 2, kWord);
606#if defined(TARGET_ARM)
607 // Combine sub & test using sub setflags encoding here
608 newLIR3(cUnit, kThumb2SubsRRI12, rIdx, rIdx, 1);
buzbee82488f52012-03-02 08:20:26 -0800609 opCondBranch(cUnit, kCondGe, target);
buzbee31a4a6f2012-02-28 15:36:15 -0800610#else
buzbee5de34942012-03-01 14:51:57 -0800611 oatFreeTemp(cUnit, rVal);
buzbee31a4a6f2012-02-28 15:36:15 -0800612 opRegImm(cUnit, kOpSub, rIdx, 1);
buzbee82488f52012-03-02 08:20:26 -0800613 opCmpImmBranch(cUnit, kCondGe, rIdx, 0, target);
buzbee31a4a6f2012-02-28 15:36:15 -0800614#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800615 } else if (!isRange) {
616 // TUNING: interleave
617 for (unsigned int i = 0; i < dInsn->vA; i++) {
618 RegLocation rlArg = loadValue(cUnit,
619 oatGetSrc(cUnit, mir, i), kCoreReg);
620 storeBaseDisp(cUnit, rRET0,
621 Array::DataOffset(component_size).Int32Value() +
622 i * 4, rlArg.lowReg, kWord);
623 // If the loadValue caused a temp to be allocated, free it
624 if (oatIsTemp(cUnit, rlArg.lowReg)) {
625 oatFreeTemp(cUnit, rlArg.lowReg);
626 }
627 }
628 }
629}
630
631void genSput(CompilationUnit* cUnit, MIR* mir, RegLocation rlSrc,
632 bool isLongOrDouble, bool isObject)
633{
634 int fieldOffset;
635 int ssbIndex;
636 bool isVolatile;
637 bool isReferrersClass;
638 uint32_t fieldIdx = mir->dalvikInsn.vB;
639
640 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
641 *cUnit->dex_file, *cUnit->dex_cache,
642 cUnit->code_item, cUnit->method_idx,
643 cUnit->access_flags);
644
645 bool fastPath =
646 cUnit->compiler->ComputeStaticFieldInfo(fieldIdx, &mUnit,
647 fieldOffset, ssbIndex,
648 isReferrersClass, isVolatile, true);
649 if (fastPath && !SLOW_FIELD_PATH) {
650 DCHECK_GE(fieldOffset, 0);
651 int rBase;
buzbee31a4a6f2012-02-28 15:36:15 -0800652 if (isReferrersClass) {
653 // Fast path, static storage base is this method's class
buzbeee1965672012-03-11 18:39:19 -0700654 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800655 rBase = oatAllocTemp(cUnit);
buzbeee1965672012-03-11 18:39:19 -0700656 loadWordDisp(cUnit, rlMethod.lowReg,
buzbee31a4a6f2012-02-28 15:36:15 -0800657 Method::DeclaringClassOffset().Int32Value(), rBase);
Ian Rogersc6f3bb82012-03-21 20:40:33 -0700658 if (oatIsTemp(cUnit, rlMethod.lowReg)) {
659 oatFreeTemp(cUnit, rlMethod.lowReg);
660 }
buzbee31a4a6f2012-02-28 15:36:15 -0800661 } else {
662 // Medium path, static storage base in a different class which
663 // requires checks that the other class is initialized.
664 DCHECK_GE(ssbIndex, 0);
665 // May do runtime call so everything to home locations.
666 oatFlushAllRegs(cUnit);
667 // Using fixed register to sync with possible call to runtime
668 // support.
buzbeee1965672012-03-11 18:39:19 -0700669 int rMethod = rARG1;
buzbee31a4a6f2012-02-28 15:36:15 -0800670 oatLockTemp(cUnit, rMethod);
671 loadCurrMethodDirect(cUnit, rMethod);
672 rBase = rARG0;
673 oatLockTemp(cUnit, rBase);
674 loadWordDisp(cUnit, rMethod,
675 Method::DexCacheInitializedStaticStorageOffset().Int32Value(),
676 rBase);
677 loadWordDisp(cUnit, rBase,
678 Array::DataOffset(sizeof(Object*)).Int32Value() + sizeof(int32_t*) *
679 ssbIndex, rBase);
680 // rBase now points at appropriate static storage base (Class*)
681 // or NULL if not initialized. Check for NULL and call helper if NULL.
682 // TUNING: fast path should fall through
buzbee82488f52012-03-02 08:20:26 -0800683 LIR* branchOver = opCmpImmBranch(cUnit, kCondNe, rBase, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -0800684 loadConstant(cUnit, rARG0, ssbIndex);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700685 callRuntimeHelperImm(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -0700686 ENTRYPOINT_OFFSET(pInitializeStaticStorage),
Ian Rogersab2b55d2012-03-18 00:06:11 -0700687 ssbIndex);
buzbee31a4a6f2012-02-28 15:36:15 -0800688#if defined(TARGET_MIPS)
689 // For Arm, rRET0 = rARG0 = rBASE, for Mips, we need to copy
buzbee82488f52012-03-02 08:20:26 -0800690 opRegCopy(cUnit, rBase, rRET0);
buzbee31a4a6f2012-02-28 15:36:15 -0800691#endif
692 LIR* skipTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800693 branchOver->target = (LIR*)skipTarget;
buzbeee1965672012-03-11 18:39:19 -0700694 oatFreeTemp(cUnit, rMethod);
buzbee31a4a6f2012-02-28 15:36:15 -0800695 }
696 // rBase now holds static storage base
buzbee31a4a6f2012-02-28 15:36:15 -0800697 if (isLongOrDouble) {
698 rlSrc = oatGetSrcWide(cUnit, mir, 0, 1);
699 rlSrc = loadValueWide(cUnit, rlSrc, kAnyReg);
700 } else {
701 rlSrc = oatGetSrc(cUnit, mir, 0);
702 rlSrc = loadValue(cUnit, rlSrc, kAnyReg);
703 }
704//FIXME: need to generalize the barrier call
705 if (isVolatile) {
706 oatGenMemBarrier(cUnit, kST);
707 }
708 if (isLongOrDouble) {
709 storeBaseDispWide(cUnit, rBase, fieldOffset, rlSrc.lowReg,
710 rlSrc.highReg);
711 } else {
712 storeWordDisp(cUnit, rBase, fieldOffset, rlSrc.lowReg);
713 }
714 if (isVolatile) {
715 oatGenMemBarrier(cUnit, kSY);
716 }
717 if (isObject) {
718 markGCCard(cUnit, rlSrc.lowReg, rBase);
719 }
720 oatFreeTemp(cUnit, rBase);
721 } else {
722 oatFlushAllRegs(cUnit); // Everything to home locations
Ian Rogers57b86d42012-03-27 16:05:41 -0700723 int setterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pSet64Static) :
724 (isObject ? ENTRYPOINT_OFFSET(pSetObjStatic)
725 : ENTRYPOINT_OFFSET(pSet32Static));
Ian Rogersab2b55d2012-03-18 00:06:11 -0700726 callRuntimeHelperImmRegLocation(cUnit, setterOffset, fieldIdx, rlSrc);
buzbee31a4a6f2012-02-28 15:36:15 -0800727 }
728}
729
730void genSget(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
731 bool isLongOrDouble, bool isObject)
732{
733 int fieldOffset;
734 int ssbIndex;
735 bool isVolatile;
736 bool isReferrersClass;
737 uint32_t fieldIdx = mir->dalvikInsn.vB;
738
739 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
740 *cUnit->dex_file, *cUnit->dex_cache,
741 cUnit->code_item, cUnit->method_idx,
742 cUnit->access_flags);
743
744 bool fastPath =
745 cUnit->compiler->ComputeStaticFieldInfo(fieldIdx, &mUnit,
746 fieldOffset, ssbIndex,
747 isReferrersClass, isVolatile,
748 false);
749 if (fastPath && !SLOW_FIELD_PATH) {
750 DCHECK_GE(fieldOffset, 0);
751 int rBase;
buzbee31a4a6f2012-02-28 15:36:15 -0800752 if (isReferrersClass) {
753 // Fast path, static storage base is this method's class
buzbeee1965672012-03-11 18:39:19 -0700754 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800755 rBase = oatAllocTemp(cUnit);
buzbeee1965672012-03-11 18:39:19 -0700756 loadWordDisp(cUnit, rlMethod.lowReg,
buzbee31a4a6f2012-02-28 15:36:15 -0800757 Method::DeclaringClassOffset().Int32Value(), rBase);
758 } else {
759 // Medium path, static storage base in a different class which
760 // requires checks that the other class is initialized
761 DCHECK_GE(ssbIndex, 0);
762 // May do runtime call so everything to home locations.
763 oatFlushAllRegs(cUnit);
764 // Using fixed register to sync with possible call to runtime
765 // support
buzbeee1965672012-03-11 18:39:19 -0700766 int rMethod = rARG1;
buzbee31a4a6f2012-02-28 15:36:15 -0800767 oatLockTemp(cUnit, rMethod);
768 loadCurrMethodDirect(cUnit, rMethod);
769 rBase = rARG0;
770 oatLockTemp(cUnit, rBase);
771 loadWordDisp(cUnit, rMethod,
772 Method::DexCacheInitializedStaticStorageOffset().Int32Value(),
773 rBase);
774 loadWordDisp(cUnit, rBase,
775 Array::DataOffset(sizeof(Object*)).Int32Value() +
776 sizeof(int32_t*) * ssbIndex,
777 rBase);
778 // rBase now points at appropriate static storage base (Class*)
779 // or NULL if not initialized. Check for NULL and call helper if NULL.
780 // TUNING: fast path should fall through
buzbee82488f52012-03-02 08:20:26 -0800781 LIR* branchOver = opCmpImmBranch(cUnit, kCondNe, rBase, 0, NULL);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700782 callRuntimeHelperImm(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -0700783 ENTRYPOINT_OFFSET(pInitializeStaticStorage),
Ian Rogersab2b55d2012-03-18 00:06:11 -0700784 ssbIndex);
buzbee31a4a6f2012-02-28 15:36:15 -0800785#if defined(TARGET_MIPS)
786 // For Arm, rRET0 = rARG0 = rBASE, for Mips, we need to copy
buzbee82488f52012-03-02 08:20:26 -0800787 opRegCopy(cUnit, rBase, rRET0);
buzbee31a4a6f2012-02-28 15:36:15 -0800788#endif
789 LIR* skipTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800790 branchOver->target = (LIR*)skipTarget;
buzbeee1965672012-03-11 18:39:19 -0700791 oatFreeTemp(cUnit, rMethod);
buzbee31a4a6f2012-02-28 15:36:15 -0800792 }
793 // rBase now holds static storage base
buzbee31a4a6f2012-02-28 15:36:15 -0800794 rlDest = isLongOrDouble ? oatGetDestWide(cUnit, mir, 0, 1)
795 : oatGetDest(cUnit, mir, 0);
796 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kAnyReg, true);
797 if (isVolatile) {
798 oatGenMemBarrier(cUnit, kSY);
799 }
800 if (isLongOrDouble) {
801 loadBaseDispWide(cUnit, NULL, rBase, fieldOffset, rlResult.lowReg,
802 rlResult.highReg, INVALID_SREG);
803 } else {
804 loadWordDisp(cUnit, rBase, fieldOffset, rlResult.lowReg);
805 }
806 oatFreeTemp(cUnit, rBase);
807 if (isLongOrDouble) {
808 storeValueWide(cUnit, rlDest, rlResult);
809 } else {
810 storeValue(cUnit, rlDest, rlResult);
811 }
812 } else {
813 oatFlushAllRegs(cUnit); // Everything to home locations
Ian Rogers57b86d42012-03-27 16:05:41 -0700814 int getterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pGet64Static) :
815 (isObject ? ENTRYPOINT_OFFSET(pGetObjStatic)
816 : ENTRYPOINT_OFFSET(pGet32Static));
Ian Rogersab2b55d2012-03-18 00:06:11 -0700817 callRuntimeHelperImm(cUnit, getterOffset, fieldIdx);
buzbee31a4a6f2012-02-28 15:36:15 -0800818 if (isLongOrDouble) {
Ian Rogersf7d9ad32012-03-13 18:45:39 -0700819 RegLocation rlResult = oatGetReturnWide(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -0800820 storeValueWide(cUnit, rlDest, rlResult);
821 } else {
Ian Rogersf7d9ad32012-03-13 18:45:39 -0700822 RegLocation rlResult = oatGetReturn(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -0800823 storeValue(cUnit, rlDest, rlResult);
824 }
825 }
826}
827
828
829// Debugging routine - if null target, branch to DebugMe
830void genShowTarget(CompilationUnit* cUnit)
831{
buzbeea7678db2012-03-05 15:35:46 -0800832#if defined(TARGET_X86)
833 UNIMPLEMENTED(WARNING) << "genShowTarget";
834#else
buzbee0398c422012-03-02 15:22:47 -0800835 LIR* branchOver = opCmpImmBranch(cUnit, kCondNe, rINVOKE_TGT, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -0800836 loadWordDisp(cUnit, rSELF,
Ian Rogers57b86d42012-03-27 16:05:41 -0700837 ENTRYPOINT_OFFSET(pDebugMe), rINVOKE_TGT);
buzbee31a4a6f2012-02-28 15:36:15 -0800838 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -0800839 branchOver->target = (LIR*)target;
buzbeea7678db2012-03-05 15:35:46 -0800840#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800841}
842
843void genThrowVerificationError(CompilationUnit* cUnit, MIR* mir)
844{
Ian Rogers57b86d42012-03-27 16:05:41 -0700845 callRuntimeHelperImmImm(cUnit, ENTRYPOINT_OFFSET(pThrowVerificationErrorFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -0700846 mir->dalvikInsn.vA, mir->dalvikInsn.vB);
buzbee31a4a6f2012-02-28 15:36:15 -0800847}
848
849void handleSuspendLaunchpads(CompilationUnit *cUnit)
850{
Ian Rogersab2b55d2012-03-18 00:06:11 -0700851 LIR** suspendLabel = (LIR **)cUnit->suspendLaunchpads.elemList;
buzbee31a4a6f2012-02-28 15:36:15 -0800852 int numElems = cUnit->suspendLaunchpads.numUsed;
buzbee31a4a6f2012-02-28 15:36:15 -0800853 for (int i = 0; i < numElems; i++) {
buzbeec5159d52012-03-03 11:48:39 -0800854 oatResetRegPool(cUnit);
buzbeefc9e6fa2012-03-23 15:14:29 -0700855 oatResetDefTracking(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800856 LIR* lab = suspendLabel[i];
857 LIR* resumeLab = (LIR*)lab->operands[0];
858 cUnit->currentDalvikOffset = lab->operands[1];
Ian Rogersab2b55d2012-03-18 00:06:11 -0700859 oatAppendLIR(cUnit, lab);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700860#if defined(TARGET_X86)
861 opThreadMem(cUnit, kOpBlx,
Ian Rogers57b86d42012-03-27 16:05:41 -0700862 ENTRYPOINT_OFFSET(pTestSuspendFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700863#else
Ian Rogers57b86d42012-03-27 16:05:41 -0700864 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pTestSuspendFromCode));
buzbee31a4a6f2012-02-28 15:36:15 -0800865 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700866#endif
buzbee82488f52012-03-02 08:20:26 -0800867 opUnconditionalBranch(cUnit, resumeLab);
buzbee31a4a6f2012-02-28 15:36:15 -0800868 }
869}
870
buzbeefc9e6fa2012-03-23 15:14:29 -0700871void handleIntrinsicLaunchpads(CompilationUnit *cUnit)
872{
873 LIR** intrinsicLabel = (LIR **)cUnit->intrinsicLaunchpads.elemList;
874 int numElems = cUnit->intrinsicLaunchpads.numUsed;
875 for (int i = 0; i < numElems; i++) {
876 oatResetRegPool(cUnit);
877 oatResetDefTracking(cUnit);
878 LIR* lab = intrinsicLabel[i];
879 MIR* mir = (MIR*)lab->operands[0];
880 InvokeType type = (InvokeType)lab->operands[1];
881 BasicBlock* bb = (BasicBlock*)lab->operands[3];
882 cUnit->currentDalvikOffset = mir->offset;
883 oatAppendLIR(cUnit, lab);
884 genInvoke(cUnit, bb, mir, type, false /* isRange */);
885 LIR* resumeLab = (LIR*)lab->operands[2];
886 if (resumeLab != NULL) {
887 opUnconditionalBranch(cUnit, resumeLab);
888 }
889 }
890}
891
buzbee31a4a6f2012-02-28 15:36:15 -0800892void handleThrowLaunchpads(CompilationUnit *cUnit)
893{
Ian Rogersab2b55d2012-03-18 00:06:11 -0700894 LIR** throwLabel = (LIR **)cUnit->throwLaunchpads.elemList;
buzbee31a4a6f2012-02-28 15:36:15 -0800895 int numElems = cUnit->throwLaunchpads.numUsed;
Ian Rogersab2b55d2012-03-18 00:06:11 -0700896 for (int i = 0; i < numElems; i++) {
buzbeec5159d52012-03-03 11:48:39 -0800897 oatResetRegPool(cUnit);
buzbeefc9e6fa2012-03-23 15:14:29 -0700898 oatResetDefTracking(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -0800899 LIR* lab = throwLabel[i];
900 cUnit->currentDalvikOffset = lab->operands[1];
Ian Rogersab2b55d2012-03-18 00:06:11 -0700901 oatAppendLIR(cUnit, lab);
buzbee31a4a6f2012-02-28 15:36:15 -0800902 int funcOffset = 0;
903 int v1 = lab->operands[2];
904 int v2 = lab->operands[3];
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700905 switch (lab->operands[0]) {
buzbee31a4a6f2012-02-28 15:36:15 -0800906 case kThrowNullPointer:
Ian Rogers57b86d42012-03-27 16:05:41 -0700907 funcOffset = ENTRYPOINT_OFFSET(pThrowNullPointerFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800908 break;
909 case kThrowArrayBounds:
buzbee5de34942012-03-01 14:51:57 -0800910 if (v2 != rARG0) {
buzbee82488f52012-03-02 08:20:26 -0800911 opRegCopy(cUnit, rARG0, v1);
912 opRegCopy(cUnit, rARG1, v2);
buzbee31a4a6f2012-02-28 15:36:15 -0800913 } else {
buzbee5de34942012-03-01 14:51:57 -0800914 if (v1 == rARG1) {
buzbee31a4a6f2012-02-28 15:36:15 -0800915#if defined(TARGET_ARM)
916 int rTmp = r12;
917#else
918 int rTmp = oatAllocTemp(cUnit);
919#endif
buzbee82488f52012-03-02 08:20:26 -0800920 opRegCopy(cUnit, rTmp, v1);
921 opRegCopy(cUnit, rARG1, v2);
922 opRegCopy(cUnit, rARG0, rTmp);
buzbee31a4a6f2012-02-28 15:36:15 -0800923 } else {
buzbee82488f52012-03-02 08:20:26 -0800924 opRegCopy(cUnit, rARG1, v2);
925 opRegCopy(cUnit, rARG0, v1);
buzbee31a4a6f2012-02-28 15:36:15 -0800926 }
927 }
Ian Rogers57b86d42012-03-27 16:05:41 -0700928 funcOffset = ENTRYPOINT_OFFSET(pThrowArrayBoundsFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800929 break;
930 case kThrowDivZero:
Ian Rogers57b86d42012-03-27 16:05:41 -0700931 funcOffset = ENTRYPOINT_OFFSET(pThrowDivZeroFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800932 break;
933 case kThrowVerificationError:
934 loadConstant(cUnit, rARG0, v1);
935 loadConstant(cUnit, rARG1, v2);
936 funcOffset =
Ian Rogers57b86d42012-03-27 16:05:41 -0700937 ENTRYPOINT_OFFSET(pThrowVerificationErrorFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800938 break;
939 case kThrowNoSuchMethod:
buzbee82488f52012-03-02 08:20:26 -0800940 opRegCopy(cUnit, rARG0, v1);
buzbee31a4a6f2012-02-28 15:36:15 -0800941 funcOffset =
Ian Rogers57b86d42012-03-27 16:05:41 -0700942 ENTRYPOINT_OFFSET(pThrowNoSuchMethodFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800943 break;
944 case kThrowStackOverflow:
Elliott Hughes36ecb782012-04-17 16:55:45 -0700945 funcOffset = ENTRYPOINT_OFFSET(pThrowStackOverflowFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -0800946 // Restore stack alignment
Elliott Hughes36ecb782012-04-17 16:55:45 -0700947#if !defined(TARGET_X86)
948 opRegImm(cUnit, kOpAdd, rSP, (cUnit->numCoreSpills + cUnit->numFPSpills) * 4);
949#else
950 opRegImm(cUnit, kOpAdd, rSP, cUnit->frameSize);
951#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800952 break;
953 default:
954 LOG(FATAL) << "Unexpected throw kind: " << lab->operands[0];
955 }
Ian Rogersab2b55d2012-03-18 00:06:11 -0700956 oatClobberCalleeSave(cUnit);
957#if !defined(TARGET_X86)
buzbee31a4a6f2012-02-28 15:36:15 -0800958 int rTgt = loadHelper(cUnit, funcOffset);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700959 opReg(cUnit, kOpBlx, rTgt);
buzbee0398c422012-03-02 15:22:47 -0800960 oatFreeTemp(cUnit, rTgt);
Ian Rogersab2b55d2012-03-18 00:06:11 -0700961#else
962 opThreadMem(cUnit, kOpBlx, funcOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -0700963#endif
buzbee31a4a6f2012-02-28 15:36:15 -0800964 }
965}
966
967/* Needed by the Assembler */
968void oatSetupResourceMasks(LIR* lir)
969{
970 setupResourceMasks(lir);
971}
972
buzbee16da88c2012-03-20 10:38:17 -0700973bool fastInstance(CompilationUnit* cUnit, uint32_t fieldIdx,
974 int& fieldOffset, bool& isVolatile, bool isPut)
975{
976 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
977 *cUnit->dex_file, *cUnit->dex_cache,
978 cUnit->code_item, cUnit->method_idx,
979 cUnit->access_flags);
980 return cUnit->compiler->ComputeInstanceFieldInfo(fieldIdx, &mUnit,
981 fieldOffset, isVolatile, isPut);
982}
983
buzbee31a4a6f2012-02-28 15:36:15 -0800984void genIGet(CompilationUnit* cUnit, MIR* mir, OpSize size,
985 RegLocation rlDest, RegLocation rlObj,
986 bool isLongOrDouble, bool isObject)
987{
988 int fieldOffset;
989 bool isVolatile;
990 uint32_t fieldIdx = mir->dalvikInsn.vC;
991
buzbee16da88c2012-03-20 10:38:17 -0700992 bool fastPath = fastInstance(cUnit, fieldIdx, fieldOffset, isVolatile,
993 false);
buzbee31a4a6f2012-02-28 15:36:15 -0800994
995 if (fastPath && !SLOW_FIELD_PATH) {
996 RegLocation rlResult;
997 RegisterClass regClass = oatRegClassBySize(size);
998 DCHECK_GE(fieldOffset, 0);
999 rlObj = loadValue(cUnit, rlObj, kCoreReg);
1000 if (isLongOrDouble) {
1001 DCHECK(rlDest.wide);
1002 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
Ian Rogersb5d09b22012-03-06 22:14:17 -08001003#if defined(TARGET_X86)
1004 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1005 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1006 loadBaseDispWide(cUnit, mir, rlObj.lowReg, fieldOffset, rlResult.lowReg,
1007 rlResult.highReg, rlObj.sRegLow);
1008 if (isVolatile) {
1009 oatGenMemBarrier(cUnit, kSY);
1010 }
1011#else
buzbee31a4a6f2012-02-28 15:36:15 -08001012 int regPtr = oatAllocTemp(cUnit);
1013 opRegRegImm(cUnit, kOpAdd, regPtr, rlObj.lowReg, fieldOffset);
1014 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1015 loadPair(cUnit, regPtr, rlResult.lowReg, rlResult.highReg);
1016 if (isVolatile) {
1017 oatGenMemBarrier(cUnit, kSY);
1018 }
1019 oatFreeTemp(cUnit, regPtr);
Ian Rogersb5d09b22012-03-06 22:14:17 -08001020#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001021 storeValueWide(cUnit, rlDest, rlResult);
1022 } else {
1023 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1024 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1025 loadBaseDisp(cUnit, mir, rlObj.lowReg, fieldOffset, rlResult.lowReg,
1026 kWord, rlObj.sRegLow);
1027 if (isVolatile) {
1028 oatGenMemBarrier(cUnit, kSY);
1029 }
1030 storeValue(cUnit, rlDest, rlResult);
1031 }
1032 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -07001033 int getterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pGet64Instance) :
1034 (isObject ? ENTRYPOINT_OFFSET(pGetObjInstance)
1035 : ENTRYPOINT_OFFSET(pGet32Instance));
Ian Rogersab2b55d2012-03-18 00:06:11 -07001036 callRuntimeHelperImmRegLocation(cUnit, getterOffset, fieldIdx, rlObj);
buzbee31a4a6f2012-02-28 15:36:15 -08001037 if (isLongOrDouble) {
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001038 RegLocation rlResult = oatGetReturnWide(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08001039 storeValueWide(cUnit, rlDest, rlResult);
1040 } else {
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001041 RegLocation rlResult = oatGetReturn(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08001042 storeValue(cUnit, rlDest, rlResult);
1043 }
1044 }
1045}
1046
1047void genIPut(CompilationUnit* cUnit, MIR* mir, OpSize size, RegLocation rlSrc,
1048 RegLocation rlObj, bool isLongOrDouble, bool isObject)
1049{
1050 int fieldOffset;
1051 bool isVolatile;
1052 uint32_t fieldIdx = mir->dalvikInsn.vC;
1053
buzbee16da88c2012-03-20 10:38:17 -07001054 bool fastPath = fastInstance(cUnit, fieldIdx, fieldOffset, isVolatile,
1055 true);
buzbee31a4a6f2012-02-28 15:36:15 -08001056 if (fastPath && !SLOW_FIELD_PATH) {
1057 RegisterClass regClass = oatRegClassBySize(size);
1058 DCHECK_GE(fieldOffset, 0);
1059 rlObj = loadValue(cUnit, rlObj, kCoreReg);
1060 if (isLongOrDouble) {
1061 int regPtr;
1062 rlSrc = loadValueWide(cUnit, rlSrc, kAnyReg);
1063 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1064 regPtr = oatAllocTemp(cUnit);
1065 opRegRegImm(cUnit, kOpAdd, regPtr, rlObj.lowReg, fieldOffset);
1066 if (isVolatile) {
1067 oatGenMemBarrier(cUnit, kST);
1068 }
1069 storePair(cUnit, regPtr, rlSrc.lowReg, rlSrc.highReg);
1070 if (isVolatile) {
1071 oatGenMemBarrier(cUnit, kSY);
1072 }
1073 oatFreeTemp(cUnit, regPtr);
1074 } else {
1075 rlSrc = loadValue(cUnit, rlSrc, regClass);
1076 genNullCheck(cUnit, rlObj.sRegLow, rlObj.lowReg, mir);/* null? */
1077 if (isVolatile) {
1078 oatGenMemBarrier(cUnit, kST);
1079 }
1080 storeBaseDisp(cUnit, rlObj.lowReg, fieldOffset, rlSrc.lowReg, kWord);
1081 if (isVolatile) {
1082 oatGenMemBarrier(cUnit, kSY);
1083 }
buzbeea7c12682012-03-19 13:13:53 -07001084 if (isObject) {
1085 markGCCard(cUnit, rlSrc.lowReg, rlObj.lowReg);
1086 }
buzbee31a4a6f2012-02-28 15:36:15 -08001087 }
1088 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -07001089 int setterOffset = isLongOrDouble ? ENTRYPOINT_OFFSET(pSet64Instance) :
1090 (isObject ? ENTRYPOINT_OFFSET(pSetObjInstance)
1091 : ENTRYPOINT_OFFSET(pSet32Instance));
Ian Rogersab2b55d2012-03-18 00:06:11 -07001092 callRuntimeHelperImmRegLocationRegLocation(cUnit, setterOffset,
1093 fieldIdx, rlObj, rlSrc);
buzbee31a4a6f2012-02-28 15:36:15 -08001094 }
1095}
1096
1097void genConstClass(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1098 RegLocation rlSrc)
1099{
1100 uint32_t type_idx = mir->dalvikInsn.vB;
buzbeee1965672012-03-11 18:39:19 -07001101 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001102 int resReg = oatAllocTemp(cUnit);
1103 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1104 if (!cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
1105 cUnit->dex_cache,
1106 *cUnit->dex_file,
1107 type_idx)) {
1108 // Call out to helper which resolves type and verifies access.
1109 // Resolved type returned in rRET0.
Ian Rogersab2b55d2012-03-18 00:06:11 -07001110 callRuntimeHelperImmReg(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001111 ENTRYPOINT_OFFSET(pInitializeTypeAndVerifyAccessFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001112 type_idx, rlMethod.lowReg);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001113 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001114 storeValue(cUnit, rlDest, rlResult);
1115 } else {
1116 // We're don't need access checks, load type from dex cache
1117 int32_t dex_cache_offset =
1118 Method::DexCacheResolvedTypesOffset().Int32Value();
buzbeee1965672012-03-11 18:39:19 -07001119 loadWordDisp(cUnit, rlMethod.lowReg, dex_cache_offset, resReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001120 int32_t offset_of_type =
1121 Array::DataOffset(sizeof(Class*)).Int32Value() + (sizeof(Class*)
1122 * type_idx);
1123 loadWordDisp(cUnit, resReg, offset_of_type, rlResult.lowReg);
1124 if (!cUnit->compiler->CanAssumeTypeIsPresentInDexCache(cUnit->dex_cache,
1125 type_idx) || SLOW_TYPE_PATH) {
1126 // Slow path, at runtime test if type is null and if so initialize
1127 oatFlushAllRegs(cUnit);
buzbee82488f52012-03-02 08:20:26 -08001128 LIR* branch1 = opCmpImmBranch(cUnit, kCondEq, rlResult.lowReg, 0,
1129 NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08001130 // Resolved, store and hop over following code
1131 storeValue(cUnit, rlDest, rlResult);
buzbee3d661942012-03-14 17:37:27 -07001132 /*
1133 * Because we have stores of the target value on two paths,
1134 * clobber temp tracking for the destination using the ssa name
1135 */
1136 oatClobberSReg(cUnit, rlDest.sRegLow);
buzbee82488f52012-03-02 08:20:26 -08001137 LIR* branch2 = opUnconditionalBranch(cUnit,0);
buzbee31a4a6f2012-02-28 15:36:15 -08001138 // TUNING: move slow path to end & remove unconditional branch
1139 LIR* target1 = newLIR0(cUnit, kPseudoTargetLabel);
buzbee5de34942012-03-01 14:51:57 -08001140 // Call out to helper, which will return resolved type in rARG0
Ian Rogers57b86d42012-03-27 16:05:41 -07001141 callRuntimeHelperImmReg(cUnit, ENTRYPOINT_OFFSET(pInitializeTypeFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001142 type_idx, rlMethod.lowReg);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001143 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001144 storeValue(cUnit, rlDest, rlResult);
buzbee3d661942012-03-14 17:37:27 -07001145 /*
1146 * Because we have stores of the target value on two paths,
1147 * clobber temp tracking for the destination using the ssa name
1148 */
1149 oatClobberSReg(cUnit, rlDest.sRegLow);
buzbee31a4a6f2012-02-28 15:36:15 -08001150 // Rejoin code paths
1151 LIR* target2 = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001152 branch1->target = (LIR*)target1;
1153 branch2->target = (LIR*)target2;
1154 } else {
1155 // Fast path, we're done - just store result
1156 storeValue(cUnit, rlDest, rlResult);
1157 }
1158 }
1159}
Ian Rogersab2b55d2012-03-18 00:06:11 -07001160
buzbee31a4a6f2012-02-28 15:36:15 -08001161void genConstString(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1162 RegLocation rlSrc)
1163{
1164 /* NOTE: Most strings should be available at compile time */
1165 uint32_t string_idx = mir->dalvikInsn.vB;
1166 int32_t offset_of_string = Array::DataOffset(sizeof(String*)).Int32Value() +
1167 (sizeof(String*) * string_idx);
1168 if (!cUnit->compiler->CanAssumeStringIsPresentInDexCache(
1169 cUnit->dex_cache, string_idx) || SLOW_STRING_PATH) {
1170 // slow path, resolve string if not in dex cache
1171 oatFlushAllRegs(cUnit);
1172 oatLockCallTemps(cUnit); // Using explicit registers
1173 loadCurrMethodDirect(cUnit, rARG2);
1174 loadWordDisp(cUnit, rARG2,
1175 Method::DexCacheStringsOffset().Int32Value(), rARG0);
1176 // Might call out to helper, which will return resolved string in rRET0
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001177#if !defined(TARGET_X86)
Ian Rogers57b86d42012-03-27 16:05:41 -07001178 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pResolveStringFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001179#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001180 loadWordDisp(cUnit, rRET0, offset_of_string, rARG0);
1181 loadConstant(cUnit, rARG1, string_idx);
1182#if defined(TARGET_ARM)
1183 opRegImm(cUnit, kOpCmp, rRET0, 0); // Is resolved?
1184 genBarrier(cUnit);
1185 // For testing, always force through helper
1186 if (!EXERCISE_SLOWEST_STRING_PATH) {
buzbee82488f52012-03-02 08:20:26 -08001187 opIT(cUnit, kArmCondEq, "T");
buzbee31a4a6f2012-02-28 15:36:15 -08001188 }
buzbee82488f52012-03-02 08:20:26 -08001189 opRegCopy(cUnit, rARG0, rARG2); // .eq
buzbee31a4a6f2012-02-28 15:36:15 -08001190 opReg(cUnit, kOpBlx, rTgt); // .eq, helper(Method*, string_idx)
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001191 oatFreeTemp(cUnit, rTgt);
1192#elif defined(TARGET_MIPS)
buzbee82488f52012-03-02 08:20:26 -08001193 LIR* branch = opCmpImmBranch(cUnit, kCondNe, rRET0, 0, NULL);
1194 opRegCopy(cUnit, rARG0, rARG2); // .eq
buzbee31a4a6f2012-02-28 15:36:15 -08001195 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001196 oatFreeTemp(cUnit, rTgt);
buzbee31a4a6f2012-02-28 15:36:15 -08001197 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001198 branch->target = target;
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001199#else
Ian Rogers57b86d42012-03-27 16:05:41 -07001200 callRuntimeHelperRegReg(cUnit, ENTRYPOINT_OFFSET(pResolveStringFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001201 rARG2, rARG1);
buzbee31a4a6f2012-02-28 15:36:15 -08001202#endif
1203 genBarrier(cUnit);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001204 storeValue(cUnit, rlDest, oatGetReturn(cUnit, false));
buzbee31a4a6f2012-02-28 15:36:15 -08001205 } else {
buzbeee1965672012-03-11 18:39:19 -07001206 RegLocation rlMethod = loadCurrMethod(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001207 int resReg = oatAllocTemp(cUnit);
1208 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
buzbeee1965672012-03-11 18:39:19 -07001209 loadWordDisp(cUnit, rlMethod.lowReg,
buzbee31a4a6f2012-02-28 15:36:15 -08001210 Method::DexCacheStringsOffset().Int32Value(), resReg);
1211 loadWordDisp(cUnit, resReg, offset_of_string, rlResult.lowReg);
1212 storeValue(cUnit, rlDest, rlResult);
1213 }
1214}
1215
1216/*
1217 * Let helper function take care of everything. Will
1218 * call Class::NewInstanceFromCode(type_idx, method);
1219 */
1220void genNewInstance(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest)
1221{
1222 oatFlushAllRegs(cUnit); /* Everything to home location */
1223 uint32_t type_idx = mir->dalvikInsn.vB;
1224 // alloc will always check for resolution, do we also need to verify
1225 // access because the verifier was unable to?
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001226 int funcOffset;
buzbee31a4a6f2012-02-28 15:36:15 -08001227 if (cUnit->compiler->CanAccessInstantiableTypeWithoutChecks(
1228 cUnit->method_idx, cUnit->dex_cache, *cUnit->dex_file, type_idx)) {
Ian Rogers57b86d42012-03-27 16:05:41 -07001229 funcOffset = ENTRYPOINT_OFFSET(pAllocObjectFromCode);
buzbee31a4a6f2012-02-28 15:36:15 -08001230 } else {
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001231 funcOffset =
Ian Rogers57b86d42012-03-27 16:05:41 -07001232 ENTRYPOINT_OFFSET(pAllocObjectFromCodeWithAccessCheck);
buzbee31a4a6f2012-02-28 15:36:15 -08001233 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07001234 callRuntimeHelperImmMethod(cUnit, funcOffset, type_idx);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001235 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001236 storeValue(cUnit, rlDest, rlResult);
1237}
1238
Ian Rogersab2b55d2012-03-18 00:06:11 -07001239void genThrow(CompilationUnit* cUnit, MIR* mir, RegLocation rlSrc)
1240{
1241 oatFlushAllRegs(cUnit);
Ian Rogers57b86d42012-03-27 16:05:41 -07001242 callRuntimeHelperRegLocation(cUnit, ENTRYPOINT_OFFSET(pDeliverException), rlSrc);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001243}
1244
buzbee31a4a6f2012-02-28 15:36:15 -08001245void genInstanceof(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1246 RegLocation rlSrc)
1247{
1248 oatFlushAllRegs(cUnit);
1249 // May generate a call - use explicit registers
1250 oatLockCallTemps(cUnit);
1251 uint32_t type_idx = mir->dalvikInsn.vC;
buzbee5de34942012-03-01 14:51:57 -08001252 loadCurrMethodDirect(cUnit, rARG1); // rARG1 <= current Method*
buzbee31a4a6f2012-02-28 15:36:15 -08001253 int classReg = rARG2; // rARG2 will hold the Class*
1254 if (!cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
1255 cUnit->dex_cache,
1256 *cUnit->dex_file,
1257 type_idx)) {
1258 // Check we have access to type_idx and if not throw IllegalAccessError,
buzbee5de34942012-03-01 14:51:57 -08001259 // returns Class* in rARG0
Ian Rogersab2b55d2012-03-18 00:06:11 -07001260 callRuntimeHelperImm(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001261 ENTRYPOINT_OFFSET(pInitializeTypeAndVerifyAccessFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001262 type_idx);
buzbee82488f52012-03-02 08:20:26 -08001263 opRegCopy(cUnit, classReg, rRET0); // Align usage with fast path
buzbee5de34942012-03-01 14:51:57 -08001264 loadValueDirectFixed(cUnit, rlSrc, rARG0); // rARG0 <= ref
buzbee31a4a6f2012-02-28 15:36:15 -08001265 } else {
buzbee5de34942012-03-01 14:51:57 -08001266 // Load dex cache entry into classReg (rARG2)
buzbee31a4a6f2012-02-28 15:36:15 -08001267 loadValueDirectFixed(cUnit, rlSrc, rARG0); // rARG0 <= ref
1268 loadWordDisp(cUnit, rARG1,
1269 Method::DexCacheResolvedTypesOffset().Int32Value(),
1270 classReg);
1271 int32_t offset_of_type =
1272 Array::DataOffset(sizeof(Class*)).Int32Value() + (sizeof(Class*)
1273 * type_idx);
1274 loadWordDisp(cUnit, classReg, offset_of_type, classReg);
1275 if (!cUnit->compiler->CanAssumeTypeIsPresentInDexCache(
1276 cUnit->dex_cache, type_idx)) {
1277 // Need to test presence of type in dex cache at runtime
buzbee82488f52012-03-02 08:20:26 -08001278 LIR* hopBranch = opCmpImmBranch(cUnit, kCondNe, classReg, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08001279 // Not resolved
1280 // Call out to helper, which will return resolved type in rRET0
Ian Rogers57b86d42012-03-27 16:05:41 -07001281 callRuntimeHelperImm(cUnit, ENTRYPOINT_OFFSET(pInitializeTypeFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001282 type_idx);
buzbee82488f52012-03-02 08:20:26 -08001283 opRegCopy(cUnit, rARG2, rRET0); // Align usage with fast path
buzbee31a4a6f2012-02-28 15:36:15 -08001284 loadValueDirectFixed(cUnit, rlSrc, rARG0); /* reload Ref */
1285 // Rejoin code paths
1286 LIR* hopTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001287 hopBranch->target = (LIR*)hopTarget;
1288 }
1289 }
1290 /* rARG0 is ref, rARG2 is class. If ref==null, use directly as bool result */
buzbee82488f52012-03-02 08:20:26 -08001291 LIR* branch1 = opCmpImmBranch(cUnit, kCondEq, rARG0, 0, NULL);
Elliott Hughese84278b2012-03-22 10:06:53 -07001292 /* load object->klass_ */
buzbee31a4a6f2012-02-28 15:36:15 -08001293 DCHECK_EQ(Object::ClassOffset().Int32Value(), 0);
1294 loadWordDisp(cUnit, rARG0, Object::ClassOffset().Int32Value(), rARG1);
Elliott Hughese84278b2012-03-22 10:06:53 -07001295 /* rARG0 is ref, rARG1 is ref->klass_, rARG2 is class */
buzbee0398c422012-03-02 15:22:47 -08001296#if defined(TARGET_ARM)
1297 /* Uses conditional nullification */
Ian Rogers57b86d42012-03-27 16:05:41 -07001298 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pInstanceofNonTrivialFromCode));
buzbee31a4a6f2012-02-28 15:36:15 -08001299 opRegReg(cUnit, kOpCmp, rARG1, rARG2); // Same?
buzbee82488f52012-03-02 08:20:26 -08001300 opIT(cUnit, kArmCondEq, "EE"); // if-convert the test
buzbee31a4a6f2012-02-28 15:36:15 -08001301 loadConstant(cUnit, rARG0, 1); // .eq case - load true
buzbee82488f52012-03-02 08:20:26 -08001302 opRegCopy(cUnit, rARG0, rARG2); // .ne case - arg0 <= class
buzbee31a4a6f2012-02-28 15:36:15 -08001303 opReg(cUnit, kOpBlx, rTgt); // .ne case: helper(class, ref->class)
Ian Rogersab2b55d2012-03-18 00:06:11 -07001304 oatFreeTemp(cUnit, rTgt);
buzbee31a4a6f2012-02-28 15:36:15 -08001305#else
buzbee0398c422012-03-02 15:22:47 -08001306 /* Uses branchovers */
1307 loadConstant(cUnit, rARG0, 1); // assume true
1308 LIR* branchover = opCmpBranch(cUnit, kCondEq, rARG1, rARG2, NULL);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001309#if !defined(TARGET_X86)
Ian Rogers57b86d42012-03-27 16:05:41 -07001310 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pInstanceofNonTrivialFromCode));
buzbee0398c422012-03-02 15:22:47 -08001311 opRegCopy(cUnit, rARG0, rARG2); // .ne case - arg0 <= class
1312 opReg(cUnit, kOpBlx, rTgt); // .ne case: helper(class, ref->class)
Ian Rogersab2b55d2012-03-18 00:06:11 -07001313 oatFreeTemp(cUnit, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001314#else
1315 opRegCopy(cUnit, rARG0, rARG2);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001316 opThreadMem(cUnit, kOpBlx,
Ian Rogers57b86d42012-03-27 16:05:41 -07001317 ENTRYPOINT_OFFSET(pInstanceofNonTrivialFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001318#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001319#endif
buzbee0398c422012-03-02 15:22:47 -08001320 oatClobberCalleeSave(cUnit);
1321 /* branch targets here */
buzbee31a4a6f2012-02-28 15:36:15 -08001322 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001323 RegLocation rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001324 storeValue(cUnit, rlDest, rlResult);
buzbee0398c422012-03-02 15:22:47 -08001325 branch1->target = target;
1326#if !defined(TARGET_ARM)
1327 branchover->target = target;
1328#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001329}
1330
1331void genCheckCast(CompilationUnit* cUnit, MIR* mir, RegLocation rlSrc)
1332{
1333 oatFlushAllRegs(cUnit);
1334 // May generate a call - use explicit registers
1335 oatLockCallTemps(cUnit);
1336 uint32_t type_idx = mir->dalvikInsn.vB;
1337 loadCurrMethodDirect(cUnit, rARG1); // rARG1 <= current Method*
1338 int classReg = rARG2; // rARG2 will hold the Class*
1339 if (!cUnit->compiler->CanAccessTypeWithoutChecks(cUnit->method_idx,
1340 cUnit->dex_cache,
1341 *cUnit->dex_file,
1342 type_idx)) {
1343 // Check we have access to type_idx and if not throw IllegalAccessError,
1344 // returns Class* in rRET0
Ian Rogersab2b55d2012-03-18 00:06:11 -07001345 // InitializeTypeAndVerifyAccess(idx, method)
1346 callRuntimeHelperImmReg(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001347 ENTRYPOINT_OFFSET(pInitializeTypeAndVerifyAccessFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001348 type_idx, rARG1);
buzbee82488f52012-03-02 08:20:26 -08001349 opRegCopy(cUnit, classReg, rRET0); // Align usage with fast path
buzbee31a4a6f2012-02-28 15:36:15 -08001350 } else {
1351 // Load dex cache entry into classReg (rARG2)
1352 loadWordDisp(cUnit, rARG1,
1353 Method::DexCacheResolvedTypesOffset().Int32Value(),
1354 classReg);
1355 int32_t offset_of_type =
1356 Array::DataOffset(sizeof(Class*)).Int32Value() +
1357 (sizeof(Class*) * type_idx);
1358 loadWordDisp(cUnit, classReg, offset_of_type, classReg);
1359 if (!cUnit->compiler->CanAssumeTypeIsPresentInDexCache(
1360 cUnit->dex_cache, type_idx)) {
1361 // Need to test presence of type in dex cache at runtime
buzbee82488f52012-03-02 08:20:26 -08001362 LIR* hopBranch = opCmpImmBranch(cUnit, kCondNe, classReg, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08001363 // Not resolved
buzbee5de34942012-03-01 14:51:57 -08001364 // Call out to helper, which will return resolved type in rARG0
Ian Rogersab2b55d2012-03-18 00:06:11 -07001365 // InitializeTypeFromCode(idx, method)
1366 callRuntimeHelperImmReg(cUnit,
Ian Rogers57b86d42012-03-27 16:05:41 -07001367 ENTRYPOINT_OFFSET(pInitializeTypeFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001368 type_idx, rARG1);
buzbee82488f52012-03-02 08:20:26 -08001369 opRegCopy(cUnit, classReg, rARG0); // Align usage with fast path
buzbee31a4a6f2012-02-28 15:36:15 -08001370 // Rejoin code paths
1371 LIR* hopTarget = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08001372 hopBranch->target = (LIR*)hopTarget;
1373 }
1374 }
buzbee5de34942012-03-01 14:51:57 -08001375 // At this point, classReg (rARG2) has class
buzbee31a4a6f2012-02-28 15:36:15 -08001376 loadValueDirectFixed(cUnit, rlSrc, rARG0); // rARG0 <= ref
1377 /* Null is OK - continue */
buzbee82488f52012-03-02 08:20:26 -08001378 LIR* branch1 = opCmpImmBranch(cUnit, kCondEq, rARG0, 0, NULL);
Elliott Hughese84278b2012-03-22 10:06:53 -07001379 /* load object->klass_ */
buzbee31a4a6f2012-02-28 15:36:15 -08001380 DCHECK_EQ(Object::ClassOffset().Int32Value(), 0);
1381 loadWordDisp(cUnit, rARG0, Object::ClassOffset().Int32Value(), rARG1);
Elliott Hughese84278b2012-03-22 10:06:53 -07001382 /* rARG1 now contains object->klass_ */
Ian Rogersab2b55d2012-03-18 00:06:11 -07001383#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbee82488f52012-03-02 08:20:26 -08001384 LIR* branch2 = opCmpBranch(cUnit, kCondEq, rARG1, classReg, NULL);
Ian Rogers57b86d42012-03-27 16:05:41 -07001385 callRuntimeHelperRegReg(cUnit, ENTRYPOINT_OFFSET(pCheckCastFromCode),
Ian Rogersab2b55d2012-03-18 00:06:11 -07001386 rARG1, rARG2);
1387#else // defined(TARGET_ARM)
Ian Rogers57b86d42012-03-27 16:05:41 -07001388 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pCheckCastFromCode));
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001389 opRegReg(cUnit, kOpCmp, rARG1, classReg);
1390 LIR* branch2 = opCondBranch(cUnit, kCondEq, NULL); /* If eq, trivial yes */
1391 opRegCopy(cUnit, rARG0, rARG1);
1392 opRegCopy(cUnit, rARG1, rARG2);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001393 oatClobberCalleeSave(cUnit);
1394 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001395 oatFreeTemp(cUnit, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001396#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001397 /* branch target here */
1398 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
Ian Rogersd36c52e2012-04-09 16:29:25 -07001399 branch1->target = target;
1400 branch2->target = target;
buzbee31a4a6f2012-02-28 15:36:15 -08001401}
1402
buzbee31a4a6f2012-02-28 15:36:15 -08001403/*
1404 * Generate array store
1405 *
1406 */
1407void genArrayObjPut(CompilationUnit* cUnit, MIR* mir, RegLocation rlArray,
1408 RegLocation rlIndex, RegLocation rlSrc, int scale)
1409{
buzbee31a4a6f2012-02-28 15:36:15 -08001410 int lenOffset = Array::LengthOffset().Int32Value();
1411 int dataOffset = Array::DataOffset(sizeof(Object*)).Int32Value();
1412
Ian Rogersd36c52e2012-04-09 16:29:25 -07001413 oatFlushAllRegs(cUnit); // Use explicit registers
1414 oatLockCallTemps(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001415
Ian Rogersd36c52e2012-04-09 16:29:25 -07001416 int rValue = rARG0; // Register holding value
1417 int rArrayClass = rARG1; // Register holding array's Class
1418 int rArray = rARG2; // Register holding array
1419 int rIndex = rARG3; // Register holding index into array
1420
1421 loadValueDirectFixed(cUnit, rlArray, rArray); // Grab array
1422 loadValueDirectFixed(cUnit, rlSrc, rValue); // Grab value
1423 loadValueDirectFixed(cUnit, rlIndex, rIndex); // Grab index
1424
1425 genNullCheck(cUnit, rlArray.sRegLow, rArray, mir); // NPE?
1426
1427 // Store of null?
1428 LIR* null_value_check = opCmpImmBranch(cUnit, kCondEq, rValue, 0, NULL);
1429
1430 // Get the array's class.
1431 loadWordDisp(cUnit, rArray, Object::ClassOffset().Int32Value(), rArrayClass);
Ian Rogers57b86d42012-03-27 16:05:41 -07001432 callRuntimeHelperRegReg(cUnit, ENTRYPOINT_OFFSET(pCanPutArrayElementFromCode),
Ian Rogersd36c52e2012-04-09 16:29:25 -07001433 rValue, rArrayClass);
1434 // Redo loadValues in case they didn't survive the call.
1435 loadValueDirectFixed(cUnit, rlArray, rArray); // Reload array
1436 loadValueDirectFixed(cUnit, rlIndex, rIndex); // Reload index
1437 loadValueDirectFixed(cUnit, rlSrc, rValue); // Reload value
1438 rArrayClass = INVALID_REG;
buzbee31a4a6f2012-02-28 15:36:15 -08001439
Ian Rogersd36c52e2012-04-09 16:29:25 -07001440 // Branch here if value to be stored == null
1441 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
1442 null_value_check->target = target;
buzbee31a4a6f2012-02-28 15:36:15 -08001443
Ian Rogersb41b33b2012-03-20 14:22:54 -07001444#if defined(TARGET_X86)
Ian Rogersd36c52e2012-04-09 16:29:25 -07001445 // make an extra temp available for card mark below
1446 oatFreeTemp(cUnit, rARG1);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001447 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1448 /* if (rlIndex >= [rlArray + lenOffset]) goto kThrowArrayBounds */
Ian Rogersd36c52e2012-04-09 16:29:25 -07001449 genRegMemCheck(cUnit, kCondUge, rIndex, rArray,
Ian Rogersb41b33b2012-03-20 14:22:54 -07001450 lenOffset, mir, kThrowArrayBounds);
1451 }
Ian Rogersd36c52e2012-04-09 16:29:25 -07001452 storeBaseIndexedDisp(cUnit, NULL, rArray, rIndex, scale,
1453 dataOffset, rValue, INVALID_REG, kWord,
Ian Rogersb41b33b2012-03-20 14:22:54 -07001454 INVALID_SREG);
1455#else
buzbee239c4e72012-03-16 08:42:29 -07001456 bool needsRangeCheck = (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK));
1457 int regLen = INVALID_REG;
1458 if (needsRangeCheck) {
Ian Rogersd36c52e2012-04-09 16:29:25 -07001459 regLen = rARG1;
1460 loadWordDisp(cUnit, rlArray.lowReg, lenOffset, regLen); // Get len
buzbee239c4e72012-03-16 08:42:29 -07001461 }
Ian Rogersd36c52e2012-04-09 16:29:25 -07001462 /* rPtr -> array data */
1463 int rPtr = oatAllocTemp(cUnit);
1464 opRegRegImm(cUnit, kOpAdd, rPtr, rArray, dataOffset);
buzbee239c4e72012-03-16 08:42:29 -07001465 if (needsRangeCheck) {
Ian Rogersd36c52e2012-04-09 16:29:25 -07001466 genRegRegCheck(cUnit, kCondCs, rIndex, regLen, mir,
buzbee31a4a6f2012-02-28 15:36:15 -08001467 kThrowArrayBounds);
buzbee31a4a6f2012-02-28 15:36:15 -08001468 }
Ian Rogersd36c52e2012-04-09 16:29:25 -07001469 storeBaseIndexed(cUnit, rPtr, rIndex, rValue, scale, kWord);
1470 oatFreeTemp(cUnit, rPtr);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001471#endif
Ian Rogersd36c52e2012-04-09 16:29:25 -07001472 oatFreeTemp(cUnit, rIndex);
1473 markGCCard(cUnit, rValue, rArray);
buzbee31a4a6f2012-02-28 15:36:15 -08001474}
1475
1476/*
1477 * Generate array load
1478 */
1479void genArrayGet(CompilationUnit* cUnit, MIR* mir, OpSize size,
1480 RegLocation rlArray, RegLocation rlIndex,
1481 RegLocation rlDest, int scale)
1482{
1483 RegisterClass regClass = oatRegClassBySize(size);
1484 int lenOffset = Array::LengthOffset().Int32Value();
1485 int dataOffset;
1486 RegLocation rlResult;
1487 rlArray = loadValue(cUnit, rlArray, kCoreReg);
1488 rlIndex = loadValue(cUnit, rlIndex, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001489
1490 if (size == kLong || size == kDouble) {
1491 dataOffset = Array::DataOffset(sizeof(int64_t)).Int32Value();
1492 } else {
1493 dataOffset = Array::DataOffset(sizeof(int32_t)).Int32Value();
1494 }
1495
1496 /* null object? */
1497 genNullCheck(cUnit, rlArray.sRegLow, rlArray.lowReg, mir);
1498
Ian Rogersb5d09b22012-03-06 22:14:17 -08001499#if defined(TARGET_X86)
1500 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1501 /* if (rlIndex >= [rlArray + lenOffset]) goto kThrowArrayBounds */
1502 genRegMemCheck(cUnit, kCondUge, rlIndex.lowReg, rlArray.lowReg,
1503 lenOffset, mir, kThrowArrayBounds);
1504 }
1505 if ((size == kLong) || (size == kDouble)) {
1506 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1507 loadBaseIndexedDisp(cUnit, NULL, rlArray.lowReg, rlIndex.lowReg, scale, dataOffset,
1508 rlResult.lowReg, rlResult.highReg, size, INVALID_SREG);
buzbee31a4a6f2012-02-28 15:36:15 -08001509
Ian Rogersb5d09b22012-03-06 22:14:17 -08001510 storeValueWide(cUnit, rlDest, rlResult);
1511 } else {
1512 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1513
1514 loadBaseIndexedDisp(cUnit, NULL, rlArray.lowReg, rlIndex.lowReg, scale, dataOffset,
1515 rlResult.lowReg, INVALID_REG, size, INVALID_SREG);
1516
1517 storeValue(cUnit, rlDest, rlResult);
1518 }
1519#else
1520 int regPtr = oatAllocTemp(cUnit);
buzbee239c4e72012-03-16 08:42:29 -07001521 bool needsRangeCheck = (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK));
1522 int regLen = INVALID_REG;
1523 if (needsRangeCheck) {
1524 regLen = oatAllocTemp(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001525 /* Get len */
1526 loadWordDisp(cUnit, rlArray.lowReg, lenOffset, regLen);
buzbee31a4a6f2012-02-28 15:36:15 -08001527 }
buzbee239c4e72012-03-16 08:42:29 -07001528 /* regPtr -> array data */
1529 opRegRegImm(cUnit, kOpAdd, regPtr, rlArray.lowReg, dataOffset);
buzbee31a4a6f2012-02-28 15:36:15 -08001530 oatFreeTemp(cUnit, rlArray.lowReg);
1531 if ((size == kLong) || (size == kDouble)) {
1532 if (scale) {
1533 int rNewIndex = oatAllocTemp(cUnit);
1534 opRegRegImm(cUnit, kOpLsl, rNewIndex, rlIndex.lowReg, scale);
1535 opRegReg(cUnit, kOpAdd, regPtr, rNewIndex);
1536 oatFreeTemp(cUnit, rNewIndex);
1537 } else {
1538 opRegReg(cUnit, kOpAdd, regPtr, rlIndex.lowReg);
1539 }
1540 oatFreeTemp(cUnit, rlIndex.lowReg);
1541 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1542
buzbee239c4e72012-03-16 08:42:29 -07001543 if (needsRangeCheck) {
1544 // TODO: change kCondCS to a more meaningful name, is the sense of
1545 // carry-set/clear flipped?
1546 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1547 kThrowArrayBounds);
1548 oatFreeTemp(cUnit, regLen);
1549 }
buzbee31a4a6f2012-02-28 15:36:15 -08001550 loadPair(cUnit, regPtr, rlResult.lowReg, rlResult.highReg);
1551
1552 oatFreeTemp(cUnit, regPtr);
1553 storeValueWide(cUnit, rlDest, rlResult);
1554 } else {
1555 rlResult = oatEvalLoc(cUnit, rlDest, regClass, true);
1556
buzbee239c4e72012-03-16 08:42:29 -07001557 if (needsRangeCheck) {
1558 // TODO: change kCondCS to a more meaningful name, is the sense of
1559 // carry-set/clear flipped?
1560 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1561 kThrowArrayBounds);
1562 oatFreeTemp(cUnit, regLen);
1563 }
buzbee31a4a6f2012-02-28 15:36:15 -08001564 loadBaseIndexed(cUnit, regPtr, rlIndex.lowReg, rlResult.lowReg,
1565 scale, size);
1566
1567 oatFreeTemp(cUnit, regPtr);
1568 storeValue(cUnit, rlDest, rlResult);
1569 }
Ian Rogersb5d09b22012-03-06 22:14:17 -08001570#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001571}
1572
1573/*
1574 * Generate array store
1575 *
1576 */
1577void genArrayPut(CompilationUnit* cUnit, MIR* mir, OpSize size,
1578 RegLocation rlArray, RegLocation rlIndex,
1579 RegLocation rlSrc, int scale)
1580{
1581 RegisterClass regClass = oatRegClassBySize(size);
1582 int lenOffset = Array::LengthOffset().Int32Value();
1583 int dataOffset;
1584
1585 if (size == kLong || size == kDouble) {
1586 dataOffset = Array::DataOffset(sizeof(int64_t)).Int32Value();
1587 } else {
1588 dataOffset = Array::DataOffset(sizeof(int32_t)).Int32Value();
1589 }
1590
buzbee31a4a6f2012-02-28 15:36:15 -08001591 rlArray = loadValue(cUnit, rlArray, kCoreReg);
1592 rlIndex = loadValue(cUnit, rlIndex, kCoreReg);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001593#if !defined(TARGET_X86)
1594 int regPtr;
buzbee31a4a6f2012-02-28 15:36:15 -08001595 if (oatIsTemp(cUnit, rlArray.lowReg)) {
1596 oatClobber(cUnit, rlArray.lowReg);
1597 regPtr = rlArray.lowReg;
1598 } else {
1599 regPtr = oatAllocTemp(cUnit);
buzbee82488f52012-03-02 08:20:26 -08001600 opRegCopy(cUnit, regPtr, rlArray.lowReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001601 }
Ian Rogersb41b33b2012-03-20 14:22:54 -07001602#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001603
1604 /* null object? */
1605 genNullCheck(cUnit, rlArray.sRegLow, rlArray.lowReg, mir);
1606
Ian Rogersb41b33b2012-03-20 14:22:54 -07001607#if defined(TARGET_X86)
1608 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1609 /* if (rlIndex >= [rlArray + lenOffset]) goto kThrowArrayBounds */
1610 genRegMemCheck(cUnit, kCondUge, rlIndex.lowReg, rlArray.lowReg,
1611 lenOffset, mir, kThrowArrayBounds);
1612 }
Ian Rogersc6f3bb82012-03-21 20:40:33 -07001613 if ((size == kLong) || (size == kDouble)) {
1614 rlSrc = loadValueWide(cUnit, rlSrc, regClass);
1615 } else {
1616 rlSrc = loadValue(cUnit, rlSrc, regClass);
1617 }
Ian Rogersb41b33b2012-03-20 14:22:54 -07001618 storeBaseIndexedDisp(cUnit, NULL, rlArray.lowReg, rlIndex.lowReg, scale, dataOffset,
1619 rlSrc.lowReg, rlSrc.highReg, size, INVALID_SREG);
1620#else
buzbee239c4e72012-03-16 08:42:29 -07001621 bool needsRangeCheck = (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK));
1622 int regLen = INVALID_REG;
1623 if (needsRangeCheck) {
1624 regLen = oatAllocTemp(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08001625 //NOTE: max live temps(4) here.
1626 /* Get len */
1627 loadWordDisp(cUnit, rlArray.lowReg, lenOffset, regLen);
buzbee31a4a6f2012-02-28 15:36:15 -08001628 }
buzbee239c4e72012-03-16 08:42:29 -07001629 /* regPtr -> array data */
1630 opRegImm(cUnit, kOpAdd, regPtr, dataOffset);
buzbee31a4a6f2012-02-28 15:36:15 -08001631 /* at this point, regPtr points to array, 2 live temps */
1632 if ((size == kLong) || (size == kDouble)) {
1633 //TUNING: specific wide routine that can handle fp regs
1634 if (scale) {
1635 int rNewIndex = oatAllocTemp(cUnit);
1636 opRegRegImm(cUnit, kOpLsl, rNewIndex, rlIndex.lowReg, scale);
1637 opRegReg(cUnit, kOpAdd, regPtr, rNewIndex);
1638 oatFreeTemp(cUnit, rNewIndex);
1639 } else {
1640 opRegReg(cUnit, kOpAdd, regPtr, rlIndex.lowReg);
1641 }
1642 rlSrc = loadValueWide(cUnit, rlSrc, regClass);
1643
buzbee239c4e72012-03-16 08:42:29 -07001644 if (needsRangeCheck) {
1645 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1646 kThrowArrayBounds);
1647 oatFreeTemp(cUnit, regLen);
1648 }
1649
buzbee31a4a6f2012-02-28 15:36:15 -08001650 storePair(cUnit, regPtr, rlSrc.lowReg, rlSrc.highReg);
1651
1652 oatFreeTemp(cUnit, regPtr);
1653 } else {
1654 rlSrc = loadValue(cUnit, rlSrc, regClass);
buzbee239c4e72012-03-16 08:42:29 -07001655 if (needsRangeCheck) {
1656 genRegRegCheck(cUnit, kCondCs, rlIndex.lowReg, regLen, mir,
1657 kThrowArrayBounds);
1658 oatFreeTemp(cUnit, regLen);
1659 }
buzbee31a4a6f2012-02-28 15:36:15 -08001660 storeBaseIndexed(cUnit, regPtr, rlIndex.lowReg, rlSrc.lowReg,
1661 scale, size);
1662 }
Ian Rogersb41b33b2012-03-20 14:22:54 -07001663#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001664}
1665
1666void genLong3Addr(CompilationUnit* cUnit, MIR* mir, OpKind firstOp,
1667 OpKind secondOp, RegLocation rlDest,
1668 RegLocation rlSrc1, RegLocation rlSrc2)
1669{
1670 RegLocation rlResult;
1671#if defined(TARGET_ARM)
1672 /*
1673 * NOTE: This is the one place in the code in which we might have
1674 * as many as six live temporary registers. There are 5 in the normal
1675 * set for Arm. Until we have spill capabilities, temporarily add
1676 * lr to the temp set. It is safe to do this locally, but note that
1677 * lr is used explicitly elsewhere in the code generator and cannot
1678 * normally be used as a general temp register.
1679 */
1680 oatMarkTemp(cUnit, rLR); // Add lr to the temp pool
1681 oatFreeTemp(cUnit, rLR); // and make it available
1682#endif
1683 rlSrc1 = loadValueWide(cUnit, rlSrc1, kCoreReg);
1684 rlSrc2 = loadValueWide(cUnit, rlSrc2, kCoreReg);
1685 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1686 // The longs may overlap - use intermediate temp if so
1687 if (rlResult.lowReg == rlSrc1.highReg) {
1688 int tReg = oatAllocTemp(cUnit);
buzbee82488f52012-03-02 08:20:26 -08001689 opRegCopy(cUnit, tReg, rlSrc1.highReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001690 opRegRegReg(cUnit, firstOp, rlResult.lowReg, rlSrc1.lowReg,
1691 rlSrc2.lowReg);
1692 opRegRegReg(cUnit, secondOp, rlResult.highReg, tReg,
1693 rlSrc2.highReg);
1694 oatFreeTemp(cUnit, tReg);
1695 } else {
1696 opRegRegReg(cUnit, firstOp, rlResult.lowReg, rlSrc1.lowReg,
1697 rlSrc2.lowReg);
1698 opRegRegReg(cUnit, secondOp, rlResult.highReg, rlSrc1.highReg,
1699 rlSrc2.highReg);
1700 }
1701 /*
1702 * NOTE: If rlDest refers to a frame variable in a large frame, the
1703 * following storeValueWide might need to allocate a temp register.
1704 * To further work around the lack of a spill capability, explicitly
1705 * free any temps from rlSrc1 & rlSrc2 that aren't still live in rlResult.
1706 * Remove when spill is functional.
1707 */
1708 freeRegLocTemps(cUnit, rlResult, rlSrc1);
1709 freeRegLocTemps(cUnit, rlResult, rlSrc2);
1710 storeValueWide(cUnit, rlDest, rlResult);
1711#if defined(TARGET_ARM)
1712 oatClobber(cUnit, rLR);
1713 oatUnmarkTemp(cUnit, rLR); // Remove lr from the temp pool
1714#endif
1715}
1716
1717
1718bool genShiftOpLong(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1719 RegLocation rlSrc1, RegLocation rlShift)
1720{
1721 int funcOffset;
1722
Elliott Hughesb25c3f62012-03-26 16:35:06 -07001723 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001724 case Instruction::SHL_LONG:
1725 case Instruction::SHL_LONG_2ADDR:
Ian Rogers57b86d42012-03-27 16:05:41 -07001726 funcOffset = ENTRYPOINT_OFFSET(pShlLong);
buzbee31a4a6f2012-02-28 15:36:15 -08001727 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001728 case Instruction::SHR_LONG:
1729 case Instruction::SHR_LONG_2ADDR:
Ian Rogers57b86d42012-03-27 16:05:41 -07001730 funcOffset = ENTRYPOINT_OFFSET(pShrLong);
buzbee31a4a6f2012-02-28 15:36:15 -08001731 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001732 case Instruction::USHR_LONG:
1733 case Instruction::USHR_LONG_2ADDR:
Ian Rogers57b86d42012-03-27 16:05:41 -07001734 funcOffset = ENTRYPOINT_OFFSET(pUshrLong);
buzbee31a4a6f2012-02-28 15:36:15 -08001735 break;
1736 default:
1737 LOG(FATAL) << "Unexpected case";
1738 return true;
1739 }
1740 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogersab2b55d2012-03-18 00:06:11 -07001741 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset, rlSrc1, rlShift);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001742 RegLocation rlResult = oatGetReturnWide(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001743 storeValueWide(cUnit, rlDest, rlResult);
1744 return false;
1745}
1746
1747
1748bool genArithOpInt(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
1749 RegLocation rlSrc1, RegLocation rlSrc2)
1750{
1751 OpKind op = kOpBkpt;
1752 bool callOut = false;
1753 bool checkZero = false;
1754 bool unary = false;
buzbee31a4a6f2012-02-28 15:36:15 -08001755 RegLocation rlResult;
1756 bool shiftOp = false;
Ian Rogersb5d09b22012-03-06 22:14:17 -08001757 int funcOffset;
1758 int retReg = rRET0;
buzbee31a4a6f2012-02-28 15:36:15 -08001759 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001760 case Instruction::NEG_INT:
buzbee31a4a6f2012-02-28 15:36:15 -08001761 op = kOpNeg;
1762 unary = true;
1763 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001764 case Instruction::NOT_INT:
buzbee31a4a6f2012-02-28 15:36:15 -08001765 op = kOpMvn;
1766 unary = true;
1767 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001768 case Instruction::ADD_INT:
1769 case Instruction::ADD_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001770 op = kOpAdd;
1771 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001772 case Instruction::SUB_INT:
1773 case Instruction::SUB_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001774 op = kOpSub;
1775 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001776 case Instruction::MUL_INT:
1777 case Instruction::MUL_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001778 op = kOpMul;
1779 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001780 case Instruction::DIV_INT:
1781 case Instruction::DIV_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001782 checkZero = true;
Ian Rogersb5d09b22012-03-06 22:14:17 -08001783 op = kOpDiv;
1784 callOut = true;
Ian Rogers57b86d42012-03-27 16:05:41 -07001785 funcOffset = ENTRYPOINT_OFFSET(pIdiv);
buzbee31a4a6f2012-02-28 15:36:15 -08001786 retReg = rRET0;
1787 break;
buzbee5de34942012-03-01 14:51:57 -08001788 /* NOTE: returns in rARG1 */
Elliott Hughesadb8c672012-03-06 16:49:32 -08001789 case Instruction::REM_INT:
1790 case Instruction::REM_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001791 checkZero = true;
Ian Rogersb5d09b22012-03-06 22:14:17 -08001792 op = kOpRem;
1793 callOut = true;
Ian Rogers57b86d42012-03-27 16:05:41 -07001794 funcOffset = ENTRYPOINT_OFFSET(pIdivmod);
buzbee31a4a6f2012-02-28 15:36:15 -08001795 retReg = rRET1;
1796 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001797 case Instruction::AND_INT:
1798 case Instruction::AND_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001799 op = kOpAnd;
1800 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001801 case Instruction::OR_INT:
1802 case Instruction::OR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001803 op = kOpOr;
1804 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001805 case Instruction::XOR_INT:
1806 case Instruction::XOR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001807 op = kOpXor;
1808 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001809 case Instruction::SHL_INT:
1810 case Instruction::SHL_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001811 shiftOp = true;
1812 op = kOpLsl;
1813 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001814 case Instruction::SHR_INT:
1815 case Instruction::SHR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001816 shiftOp = true;
1817 op = kOpAsr;
1818 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001819 case Instruction::USHR_INT:
1820 case Instruction::USHR_INT_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08001821 shiftOp = true;
1822 op = kOpLsr;
1823 break;
1824 default:
1825 LOG(FATAL) << "Invalid word arith op: " <<
1826 (int)mir->dalvikInsn.opcode;
1827 }
1828 if (!callOut) {
buzbee31a4a6f2012-02-28 15:36:15 -08001829 if (unary) {
Ian Rogers7caad772012-03-30 01:07:54 -07001830 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001831 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1832 opRegReg(cUnit, op, rlResult.lowReg,
1833 rlSrc1.lowReg);
1834 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08001835 if (shiftOp) {
Ian Rogers7caad772012-03-30 01:07:54 -07001836#if !defined(TARGET_X86)
1837 rlSrc2 = loadValue(cUnit, rlSrc2, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001838 int tReg = oatAllocTemp(cUnit);
1839 opRegRegImm(cUnit, kOpAnd, tReg, rlSrc2.lowReg, 31);
Ian Rogers7caad772012-03-30 01:07:54 -07001840#else
1841 // X86 doesn't require masking and must use ECX
1842 loadValueDirectFixed(cUnit, rlSrc2, rCX);
1843 int tReg = rCX;
1844#endif
1845 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001846 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1847 opRegRegReg(cUnit, op, rlResult.lowReg,
1848 rlSrc1.lowReg, tReg);
1849 oatFreeTemp(cUnit, tReg);
1850 } else {
Ian Rogers7caad772012-03-30 01:07:54 -07001851 rlSrc1 = loadValue(cUnit, rlSrc1, kCoreReg);
1852 rlSrc2 = loadValue(cUnit, rlSrc2, kCoreReg);
buzbee31a4a6f2012-02-28 15:36:15 -08001853 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1854 opRegRegReg(cUnit, op, rlResult.lowReg,
1855 rlSrc1.lowReg, rlSrc2.lowReg);
1856 }
1857 }
1858 storeValue(cUnit, rlDest, rlResult);
1859 } else {
1860 RegLocation rlResult;
1861 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogers55bd45f2012-04-04 17:31:20 -07001862 loadValueDirectFixed(cUnit, rlSrc2, rARG1);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001863#if !defined(TARGET_X86)
buzbee31a4a6f2012-02-28 15:36:15 -08001864 int rTgt = loadHelper(cUnit, funcOffset);
Ian Rogersab2b55d2012-03-18 00:06:11 -07001865#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001866 loadValueDirectFixed(cUnit, rlSrc1, rARG0);
1867 if (checkZero) {
1868 genImmedCheck(cUnit, kCondEq, rARG1, 0, mir, kThrowDivZero);
1869 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07001870#if !defined(TARGET_X86)
1871 opReg(cUnit, kOpBlx, rTgt);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001872 oatFreeTemp(cUnit, rTgt);
1873#else
Ian Rogersab2b55d2012-03-18 00:06:11 -07001874 opThreadMem(cUnit, kOpBlx, funcOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07001875#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001876 if (retReg == rRET0)
Ian Rogersf7d9ad32012-03-13 18:45:39 -07001877 rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08001878 else
1879 rlResult = oatGetReturnAlt(cUnit);
1880 storeValue(cUnit, rlDest, rlResult);
1881 }
1882 return false;
1883}
1884
1885/*
1886 * The following are the first-level codegen routines that analyze the format
1887 * of each bytecode then either dispatch special purpose codegen routines
1888 * or produce corresponding Thumb instructions directly.
1889 */
1890
1891bool isPowerOfTwo(int x)
1892{
1893 return (x & (x - 1)) == 0;
1894}
1895
1896// Returns true if no more than two bits are set in 'x'.
1897bool isPopCountLE2(unsigned int x)
1898{
1899 x &= x - 1;
1900 return (x & (x - 1)) == 0;
1901}
1902
1903// Returns the index of the lowest set bit in 'x'.
1904int lowestSetBit(unsigned int x) {
1905 int bit_posn = 0;
1906 while ((x & 0xf) == 0) {
1907 bit_posn += 4;
1908 x >>= 4;
1909 }
1910 while ((x & 1) == 0) {
1911 bit_posn++;
1912 x >>= 1;
1913 }
1914 return bit_posn;
1915}
1916
1917// Returns true if it added instructions to 'cUnit' to divide 'rlSrc' by 'lit'
1918// and store the result in 'rlDest'.
Elliott Hughesadb8c672012-03-06 16:49:32 -08001919bool handleEasyDivide(CompilationUnit* cUnit, Instruction::Code dalvikOpcode,
buzbee31a4a6f2012-02-28 15:36:15 -08001920 RegLocation rlSrc, RegLocation rlDest, int lit)
1921{
buzbeef3aac972012-04-11 16:33:36 -07001922#if defined(TARGET_ARM)
1923 // No divide instruction for Arm, so check for more special cases
1924 if (lit < 2) {
1925 return false;
1926 }
1927 if (!isPowerOfTwo(lit)) {
1928 return smallLiteralDivide(cUnit, dalvikOpcode, rlSrc, rlDest, lit);
1929 }
1930#else
buzbee31a4a6f2012-02-28 15:36:15 -08001931 if (lit < 2 || !isPowerOfTwo(lit)) {
1932 return false;
1933 }
buzbeef3aac972012-04-11 16:33:36 -07001934#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001935 int k = lowestSetBit(lit);
1936 if (k >= 30) {
1937 // Avoid special cases.
1938 return false;
1939 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08001940 bool div = (dalvikOpcode == Instruction::DIV_INT_LIT8 ||
1941 dalvikOpcode == Instruction::DIV_INT_LIT16);
buzbee31a4a6f2012-02-28 15:36:15 -08001942 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
1943 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
1944 if (div) {
1945 int tReg = oatAllocTemp(cUnit);
1946 if (lit == 2) {
1947 // Division by 2 is by far the most common division by constant.
1948 opRegRegImm(cUnit, kOpLsr, tReg, rlSrc.lowReg, 32 - k);
1949 opRegRegReg(cUnit, kOpAdd, tReg, tReg, rlSrc.lowReg);
1950 opRegRegImm(cUnit, kOpAsr, rlResult.lowReg, tReg, k);
1951 } else {
1952 opRegRegImm(cUnit, kOpAsr, tReg, rlSrc.lowReg, 31);
1953 opRegRegImm(cUnit, kOpLsr, tReg, tReg, 32 - k);
1954 opRegRegReg(cUnit, kOpAdd, tReg, tReg, rlSrc.lowReg);
1955 opRegRegImm(cUnit, kOpAsr, rlResult.lowReg, tReg, k);
1956 }
1957 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08001958 int tReg1 = oatAllocTemp(cUnit);
1959 int tReg2 = oatAllocTemp(cUnit);
1960 if (lit == 2) {
1961 opRegRegImm(cUnit, kOpLsr, tReg1, rlSrc.lowReg, 32 - k);
1962 opRegRegReg(cUnit, kOpAdd, tReg2, tReg1, rlSrc.lowReg);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001963 opRegRegImm(cUnit, kOpAnd, tReg2, tReg2, lit -1);
buzbee31a4a6f2012-02-28 15:36:15 -08001964 opRegRegReg(cUnit, kOpSub, rlResult.lowReg, tReg2, tReg1);
1965 } else {
1966 opRegRegImm(cUnit, kOpAsr, tReg1, rlSrc.lowReg, 31);
1967 opRegRegImm(cUnit, kOpLsr, tReg1, tReg1, 32 - k);
1968 opRegRegReg(cUnit, kOpAdd, tReg2, tReg1, rlSrc.lowReg);
Ian Rogersb41b33b2012-03-20 14:22:54 -07001969 opRegRegImm(cUnit, kOpAnd, tReg2, tReg2, lit - 1);
buzbee31a4a6f2012-02-28 15:36:15 -08001970 opRegRegReg(cUnit, kOpSub, rlResult.lowReg, tReg2, tReg1);
1971 }
1972 }
1973 storeValue(cUnit, rlDest, rlResult);
1974 return true;
1975}
1976
1977void genMultiplyByTwoBitMultiplier(CompilationUnit* cUnit, RegLocation rlSrc,
1978 RegLocation rlResult, int lit,
1979 int firstBit, int secondBit)
1980{
buzbee0398c422012-03-02 15:22:47 -08001981#if defined(TARGET_ARM)
buzbee31a4a6f2012-02-28 15:36:15 -08001982 opRegRegRegShift(cUnit, kOpAdd, rlResult.lowReg, rlSrc.lowReg, rlSrc.lowReg,
1983 encodeShift(kArmLsl, secondBit - firstBit));
buzbee0398c422012-03-02 15:22:47 -08001984#else
1985 int tReg = oatAllocTemp(cUnit);
1986 opRegRegImm(cUnit, kOpLsl, tReg, rlSrc.lowReg, secondBit - firstBit);
1987 opRegRegReg(cUnit, kOpAdd, rlResult.lowReg, rlSrc.lowReg, tReg);
1988 oatFreeTemp(cUnit, tReg);
buzbee5de34942012-03-01 14:51:57 -08001989#endif
buzbee31a4a6f2012-02-28 15:36:15 -08001990 if (firstBit != 0) {
1991 opRegRegImm(cUnit, kOpLsl, rlResult.lowReg, rlResult.lowReg, firstBit);
1992 }
1993}
1994
1995// Returns true if it added instructions to 'cUnit' to multiply 'rlSrc' by 'lit'
1996// and store the result in 'rlDest'.
1997bool handleEasyMultiply(CompilationUnit* cUnit, RegLocation rlSrc,
1998 RegLocation rlDest, int lit)
1999{
2000 // Can we simplify this multiplication?
2001 bool powerOfTwo = false;
2002 bool popCountLE2 = false;
2003 bool powerOfTwoMinusOne = false;
2004 if (lit < 2) {
2005 // Avoid special cases.
2006 return false;
2007 } else if (isPowerOfTwo(lit)) {
2008 powerOfTwo = true;
2009 } else if (isPopCountLE2(lit)) {
2010 popCountLE2 = true;
2011 } else if (isPowerOfTwo(lit + 1)) {
2012 powerOfTwoMinusOne = true;
2013 } else {
2014 return false;
2015 }
2016 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
2017 RegLocation rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2018 if (powerOfTwo) {
2019 // Shift.
2020 opRegRegImm(cUnit, kOpLsl, rlResult.lowReg, rlSrc.lowReg,
2021 lowestSetBit(lit));
2022 } else if (popCountLE2) {
2023 // Shift and add and shift.
2024 int firstBit = lowestSetBit(lit);
2025 int secondBit = lowestSetBit(lit ^ (1 << firstBit));
2026 genMultiplyByTwoBitMultiplier(cUnit, rlSrc, rlResult, lit,
2027 firstBit, secondBit);
2028 } else {
2029 // Reverse subtract: (src << (shift + 1)) - src.
2030 DCHECK(powerOfTwoMinusOne);
2031 // TUNING: rsb dst, src, src lsl#lowestSetBit(lit + 1)
2032 int tReg = oatAllocTemp(cUnit);
2033 opRegRegImm(cUnit, kOpLsl, tReg, rlSrc.lowReg, lowestSetBit(lit + 1));
2034 opRegRegReg(cUnit, kOpSub, rlResult.lowReg, tReg, rlSrc.lowReg);
2035 }
2036 storeValue(cUnit, rlDest, rlResult);
2037 return true;
2038}
2039
2040bool genArithOpIntLit(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
2041 RegLocation rlSrc, int lit)
2042{
Elliott Hughesadb8c672012-03-06 16:49:32 -08002043 Instruction::Code dalvikOpcode = mir->dalvikInsn.opcode;
buzbee31a4a6f2012-02-28 15:36:15 -08002044 RegLocation rlResult;
2045 OpKind op = (OpKind)0; /* Make gcc happy */
2046 int shiftOp = false;
2047 bool isDiv = false;
2048 int funcOffset;
buzbee31a4a6f2012-02-28 15:36:15 -08002049
2050 switch (dalvikOpcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002051 case Instruction::RSUB_INT_LIT8:
2052 case Instruction::RSUB_INT: {
buzbee31a4a6f2012-02-28 15:36:15 -08002053 int tReg;
2054 //TUNING: add support for use of Arm rsub op
2055 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
2056 tReg = oatAllocTemp(cUnit);
2057 loadConstant(cUnit, tReg, lit);
2058 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2059 opRegRegReg(cUnit, kOpSub, rlResult.lowReg,
2060 tReg, rlSrc.lowReg);
2061 storeValue(cUnit, rlDest, rlResult);
2062 return false;
2063 break;
2064 }
2065
Elliott Hughesadb8c672012-03-06 16:49:32 -08002066 case Instruction::ADD_INT_LIT8:
2067 case Instruction::ADD_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002068 op = kOpAdd;
2069 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002070 case Instruction::MUL_INT_LIT8:
2071 case Instruction::MUL_INT_LIT16: {
buzbee31a4a6f2012-02-28 15:36:15 -08002072 if (handleEasyMultiply(cUnit, rlSrc, rlDest, lit)) {
2073 return false;
2074 }
2075 op = kOpMul;
2076 break;
2077 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08002078 case Instruction::AND_INT_LIT8:
2079 case Instruction::AND_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002080 op = kOpAnd;
2081 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002082 case Instruction::OR_INT_LIT8:
2083 case Instruction::OR_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002084 op = kOpOr;
2085 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002086 case Instruction::XOR_INT_LIT8:
2087 case Instruction::XOR_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002088 op = kOpXor;
2089 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002090 case Instruction::SHL_INT_LIT8:
buzbee31a4a6f2012-02-28 15:36:15 -08002091 lit &= 31;
2092 shiftOp = true;
2093 op = kOpLsl;
2094 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002095 case Instruction::SHR_INT_LIT8:
buzbee31a4a6f2012-02-28 15:36:15 -08002096 lit &= 31;
2097 shiftOp = true;
2098 op = kOpAsr;
2099 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002100 case Instruction::USHR_INT_LIT8:
buzbee31a4a6f2012-02-28 15:36:15 -08002101 lit &= 31;
2102 shiftOp = true;
2103 op = kOpLsr;
2104 break;
2105
Elliott Hughesadb8c672012-03-06 16:49:32 -08002106 case Instruction::DIV_INT_LIT8:
2107 case Instruction::DIV_INT_LIT16:
2108 case Instruction::REM_INT_LIT8:
2109 case Instruction::REM_INT_LIT16:
buzbee31a4a6f2012-02-28 15:36:15 -08002110 if (lit == 0) {
2111 genImmedCheck(cUnit, kCondAl, 0, 0, mir, kThrowDivZero);
2112 return false;
2113 }
2114 if (handleEasyDivide(cUnit, dalvikOpcode, rlSrc, rlDest, lit)) {
2115 return false;
2116 }
2117 oatFlushAllRegs(cUnit); /* Everything to home location */
2118 loadValueDirectFixed(cUnit, rlSrc, rARG0);
2119 oatClobber(cUnit, rARG0);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002120 if ((dalvikOpcode == Instruction::DIV_INT_LIT8) ||
2121 (dalvikOpcode == Instruction::DIV_INT_LIT16)) {
Ian Rogers57b86d42012-03-27 16:05:41 -07002122 funcOffset = ENTRYPOINT_OFFSET(pIdiv);
buzbee31a4a6f2012-02-28 15:36:15 -08002123 isDiv = true;
2124 } else {
Ian Rogers57b86d42012-03-27 16:05:41 -07002125 funcOffset = ENTRYPOINT_OFFSET(pIdivmod);
buzbee31a4a6f2012-02-28 15:36:15 -08002126 isDiv = false;
2127 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07002128 callRuntimeHelperRegImm(cUnit, funcOffset, rARG0, lit);
buzbee31a4a6f2012-02-28 15:36:15 -08002129 if (isDiv)
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002130 rlResult = oatGetReturn(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08002131 else
2132 rlResult = oatGetReturnAlt(cUnit);
2133 storeValue(cUnit, rlDest, rlResult);
2134 return false;
2135 break;
2136 default:
2137 return true;
2138 }
2139 rlSrc = loadValue(cUnit, rlSrc, kCoreReg);
2140 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2141 // Avoid shifts by literal 0 - no support in Thumb. Change to copy
2142 if (shiftOp && (lit == 0)) {
buzbee82488f52012-03-02 08:20:26 -08002143 opRegCopy(cUnit, rlResult.lowReg, rlSrc.lowReg);
buzbee31a4a6f2012-02-28 15:36:15 -08002144 } else {
2145 opRegRegImm(cUnit, op, rlResult.lowReg, rlSrc.lowReg, lit);
2146 }
2147 storeValue(cUnit, rlDest, rlResult);
2148 return false;
2149}
2150
2151bool genArithOpLong(CompilationUnit* cUnit, MIR* mir, RegLocation rlDest,
2152 RegLocation rlSrc1, RegLocation rlSrc2)
2153{
2154 RegLocation rlResult;
2155 OpKind firstOp = kOpBkpt;
2156 OpKind secondOp = kOpBkpt;
2157 bool callOut = false;
2158 bool checkZero = false;
2159 int funcOffset;
2160 int retReg = rRET0;
2161
2162 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002163 case Instruction::NOT_LONG:
buzbee31a4a6f2012-02-28 15:36:15 -08002164 rlSrc2 = loadValueWide(cUnit, rlSrc2, kCoreReg);
2165 rlResult = oatEvalLoc(cUnit, rlDest, kCoreReg, true);
2166 // Check for destructive overlap
2167 if (rlResult.lowReg == rlSrc2.highReg) {
2168 int tReg = oatAllocTemp(cUnit);
buzbee82488f52012-03-02 08:20:26 -08002169 opRegCopy(cUnit, tReg, rlSrc2.highReg);
buzbee31a4a6f2012-02-28 15:36:15 -08002170 opRegReg(cUnit, kOpMvn, rlResult.lowReg, rlSrc2.lowReg);
2171 opRegReg(cUnit, kOpMvn, rlResult.highReg, tReg);
2172 oatFreeTemp(cUnit, tReg);
2173 } else {
2174 opRegReg(cUnit, kOpMvn, rlResult.lowReg, rlSrc2.lowReg);
2175 opRegReg(cUnit, kOpMvn, rlResult.highReg, rlSrc2.highReg);
2176 }
2177 storeValueWide(cUnit, rlDest, rlResult);
2178 return false;
2179 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002180 case Instruction::ADD_LONG:
2181 case Instruction::ADD_LONG_2ADDR:
Ian Rogers7caad772012-03-30 01:07:54 -07002182#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbeec5159d52012-03-03 11:48:39 -08002183 return genAddLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2184#else
buzbee31a4a6f2012-02-28 15:36:15 -08002185 firstOp = kOpAdd;
2186 secondOp = kOpAdc;
2187 break;
buzbeec5159d52012-03-03 11:48:39 -08002188#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002189 case Instruction::SUB_LONG:
2190 case Instruction::SUB_LONG_2ADDR:
Ian Rogers7caad772012-03-30 01:07:54 -07002191#if defined(TARGET_MIPS) || defined(TARGET_X86)
buzbeec5159d52012-03-03 11:48:39 -08002192 return genSubLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
Ian Rogers7caad772012-03-30 01:07:54 -07002193#else
buzbee31a4a6f2012-02-28 15:36:15 -08002194 firstOp = kOpSub;
2195 secondOp = kOpSbc;
2196 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002197#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002198 case Instruction::MUL_LONG:
2199 case Instruction::MUL_LONG_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08002200 callOut = true;
2201 retReg = rRET0;
Ian Rogers57b86d42012-03-27 16:05:41 -07002202 funcOffset = ENTRYPOINT_OFFSET(pLmul);
buzbee31a4a6f2012-02-28 15:36:15 -08002203 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002204 case Instruction::DIV_LONG:
2205 case Instruction::DIV_LONG_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08002206 callOut = true;
2207 checkZero = true;
2208 retReg = rRET0;
Ian Rogers57b86d42012-03-27 16:05:41 -07002209 funcOffset = ENTRYPOINT_OFFSET(pLdivmod);
buzbee31a4a6f2012-02-28 15:36:15 -08002210 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002211 case Instruction::REM_LONG:
2212 case Instruction::REM_LONG_2ADDR:
buzbee31a4a6f2012-02-28 15:36:15 -08002213 callOut = true;
2214 checkZero = true;
Ian Rogers55bd45f2012-04-04 17:31:20 -07002215 funcOffset = ENTRYPOINT_OFFSET(pLdiv);
2216#if defined(TARGET_ARM)
2217 /* NOTE - result is in rARG2/rARG3 instead of rRET0/rRET1 */
buzbee31a4a6f2012-02-28 15:36:15 -08002218 retReg = rARG2;
Ian Rogers55bd45f2012-04-04 17:31:20 -07002219#else
2220 retReg = rRET0;
2221#endif
buzbee31a4a6f2012-02-28 15:36:15 -08002222 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002223 case Instruction::AND_LONG_2ADDR:
2224 case Instruction::AND_LONG:
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002225#if defined(TARGET_X86)
Ian Rogers7caad772012-03-30 01:07:54 -07002226 return genAndLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2227#else
buzbee31a4a6f2012-02-28 15:36:15 -08002228 firstOp = kOpAnd;
2229 secondOp = kOpAnd;
2230 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002231#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002232 case Instruction::OR_LONG:
2233 case Instruction::OR_LONG_2ADDR:
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002234#if defined(TARGET_X86)
Ian Rogers7caad772012-03-30 01:07:54 -07002235 return genOrLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2236#else
buzbee31a4a6f2012-02-28 15:36:15 -08002237 firstOp = kOpOr;
2238 secondOp = kOpOr;
2239 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002240#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002241 case Instruction::XOR_LONG:
2242 case Instruction::XOR_LONG_2ADDR:
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002243#if defined(TARGET_X86)
Ian Rogers7caad772012-03-30 01:07:54 -07002244 return genXorLong(cUnit, mir, rlDest, rlSrc1, rlSrc2);
2245#else
buzbee31a4a6f2012-02-28 15:36:15 -08002246 firstOp = kOpXor;
2247 secondOp = kOpXor;
2248 break;
Ian Rogers7caad772012-03-30 01:07:54 -07002249#endif
Elliott Hughesadb8c672012-03-06 16:49:32 -08002250 case Instruction::NEG_LONG: {
buzbeec5159d52012-03-03 11:48:39 -08002251 return genNegLong(cUnit, mir, rlDest, rlSrc2);
buzbee31a4a6f2012-02-28 15:36:15 -08002252 }
2253 default:
2254 LOG(FATAL) << "Invalid long arith op";
2255 }
2256 if (!callOut) {
2257 genLong3Addr(cUnit, mir, firstOp, secondOp, rlDest, rlSrc1, rlSrc2);
2258 } else {
buzbee31a4a6f2012-02-28 15:36:15 -08002259 oatFlushAllRegs(cUnit); /* Send everything to home location */
2260 if (checkZero) {
2261 loadValueDirectWideFixed(cUnit, rlSrc2, rARG2, rARG3);
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002262#if !defined(TARGET_X86)
Ian Rogersab2b55d2012-03-18 00:06:11 -07002263 int rTgt = loadHelper(cUnit, funcOffset);
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07002264#endif
buzbee31a4a6f2012-02-28 15:36:15 -08002265 int tReg = oatAllocTemp(cUnit);
2266#if defined(TARGET_ARM)
2267 newLIR4(cUnit, kThumb2OrrRRRs, tReg, rARG2, rARG3, 0);
2268 oatFreeTemp(cUnit, tReg);
2269 genCheck(cUnit, kCondEq, mir, kThrowDivZero);
2270#else
2271 opRegRegReg(cUnit, kOpOr, tReg, rARG2, rARG3);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002272#endif
buzbee5de34942012-03-01 14:51:57 -08002273 genImmedCheck(cUnit, kCondEq, tReg, 0, mir, kThrowDivZero);
buzbee31a4a6f2012-02-28 15:36:15 -08002274 oatFreeTemp(cUnit, tReg);
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002275 loadValueDirectWideFixed(cUnit, rlSrc1, rARG0, rARG1);
Ian Rogersab2b55d2012-03-18 00:06:11 -07002276#if !defined(TARGET_X86)
2277 opReg(cUnit, kOpBlx, rTgt);
2278 oatFreeTemp(cUnit, rTgt);
Ian Rogersc6f3bb82012-03-21 20:40:33 -07002279#else
2280 opThreadMem(cUnit, kOpBlx, funcOffset);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002281#endif
Ian Rogersab2b55d2012-03-18 00:06:11 -07002282 } else {
2283 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset,
2284 rlSrc1, rlSrc2);
buzbee31a4a6f2012-02-28 15:36:15 -08002285 }
buzbee31a4a6f2012-02-28 15:36:15 -08002286 // Adjust return regs in to handle case of rem returning rARG2/rARG3
2287 if (retReg == rRET0)
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002288 rlResult = oatGetReturnWide(cUnit, false);
buzbee31a4a6f2012-02-28 15:36:15 -08002289 else
2290 rlResult = oatGetReturnWideAlt(cUnit);
2291 storeValueWide(cUnit, rlDest, rlResult);
2292 }
2293 return false;
2294}
2295
2296bool genConversionCall(CompilationUnit* cUnit, MIR* mir, int funcOffset,
2297 int srcSize, int tgtSize)
2298{
2299 /*
2300 * Don't optimize the register usage since it calls out to support
2301 * functions
2302 */
2303 RegLocation rlSrc;
2304 RegLocation rlDest;
2305 oatFlushAllRegs(cUnit); /* Send everything to home location */
buzbee31a4a6f2012-02-28 15:36:15 -08002306 if (srcSize == 1) {
2307 rlSrc = oatGetSrc(cUnit, mir, 0);
2308 loadValueDirectFixed(cUnit, rlSrc, rARG0);
2309 } else {
2310 rlSrc = oatGetSrcWide(cUnit, mir, 0, 1);
2311 loadValueDirectWideFixed(cUnit, rlSrc, rARG0, rARG1);
2312 }
Ian Rogersab2b55d2012-03-18 00:06:11 -07002313 callRuntimeHelperRegLocation(cUnit, funcOffset, rlSrc);
buzbee31a4a6f2012-02-28 15:36:15 -08002314 if (tgtSize == 1) {
2315 RegLocation rlResult;
2316 rlDest = oatGetDest(cUnit, mir, 0);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002317 rlResult = oatGetReturn(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08002318 storeValue(cUnit, rlDest, rlResult);
2319 } else {
2320 RegLocation rlResult;
2321 rlDest = oatGetDestWide(cUnit, mir, 0, 1);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002322 rlResult = oatGetReturnWide(cUnit, rlDest.fp);
buzbee31a4a6f2012-02-28 15:36:15 -08002323 storeValueWide(cUnit, rlDest, rlResult);
2324 }
2325 return false;
2326}
2327
2328void genNegFloat(CompilationUnit* cUnit, RegLocation rlDest, RegLocation rlSrc);
2329bool genArithOpFloatPortable(CompilationUnit* cUnit, MIR* mir,
2330 RegLocation rlDest, RegLocation rlSrc1,
2331 RegLocation rlSrc2)
2332{
2333 RegLocation rlResult;
2334 int funcOffset;
2335
2336 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002337 case Instruction::ADD_FLOAT_2ADDR:
2338 case Instruction::ADD_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002339 funcOffset = ENTRYPOINT_OFFSET(pFadd);
buzbee31a4a6f2012-02-28 15:36:15 -08002340 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002341 case Instruction::SUB_FLOAT_2ADDR:
2342 case Instruction::SUB_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002343 funcOffset = ENTRYPOINT_OFFSET(pFsub);
buzbee31a4a6f2012-02-28 15:36:15 -08002344 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002345 case Instruction::DIV_FLOAT_2ADDR:
2346 case Instruction::DIV_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002347 funcOffset = ENTRYPOINT_OFFSET(pFdiv);
buzbee31a4a6f2012-02-28 15:36:15 -08002348 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002349 case Instruction::MUL_FLOAT_2ADDR:
2350 case Instruction::MUL_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002351 funcOffset = ENTRYPOINT_OFFSET(pFmul);
buzbee31a4a6f2012-02-28 15:36:15 -08002352 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002353 case Instruction::REM_FLOAT_2ADDR:
2354 case Instruction::REM_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002355 funcOffset = ENTRYPOINT_OFFSET(pFmodf);
buzbee31a4a6f2012-02-28 15:36:15 -08002356 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002357 case Instruction::NEG_FLOAT: {
buzbee31a4a6f2012-02-28 15:36:15 -08002358 genNegFloat(cUnit, rlDest, rlSrc1);
2359 return false;
2360 }
2361 default:
2362 return true;
2363 }
2364 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogersab2b55d2012-03-18 00:06:11 -07002365 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset, rlSrc1, rlSrc2);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002366 rlResult = oatGetReturn(cUnit, true);
buzbee31a4a6f2012-02-28 15:36:15 -08002367 storeValue(cUnit, rlDest, rlResult);
2368 return false;
2369}
2370
2371void genNegDouble(CompilationUnit* cUnit, RegLocation rlDst, RegLocation rlSrc);
2372bool genArithOpDoublePortable(CompilationUnit* cUnit, MIR* mir,
2373 RegLocation rlDest, RegLocation rlSrc1,
2374 RegLocation rlSrc2)
2375{
2376 RegLocation rlResult;
2377 int funcOffset;
2378
2379 switch (mir->dalvikInsn.opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002380 case Instruction::ADD_DOUBLE_2ADDR:
2381 case Instruction::ADD_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002382 funcOffset = ENTRYPOINT_OFFSET(pDadd);
buzbee31a4a6f2012-02-28 15:36:15 -08002383 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002384 case Instruction::SUB_DOUBLE_2ADDR:
2385 case Instruction::SUB_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002386 funcOffset = ENTRYPOINT_OFFSET(pDsub);
buzbee31a4a6f2012-02-28 15:36:15 -08002387 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002388 case Instruction::DIV_DOUBLE_2ADDR:
2389 case Instruction::DIV_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002390 funcOffset = ENTRYPOINT_OFFSET(pDdiv);
buzbee31a4a6f2012-02-28 15:36:15 -08002391 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002392 case Instruction::MUL_DOUBLE_2ADDR:
2393 case Instruction::MUL_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002394 funcOffset = ENTRYPOINT_OFFSET(pDmul);
buzbee31a4a6f2012-02-28 15:36:15 -08002395 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002396 case Instruction::REM_DOUBLE_2ADDR:
2397 case Instruction::REM_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002398 funcOffset = ENTRYPOINT_OFFSET(pFmod);
buzbee31a4a6f2012-02-28 15:36:15 -08002399 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002400 case Instruction::NEG_DOUBLE: {
buzbee31a4a6f2012-02-28 15:36:15 -08002401 genNegDouble(cUnit, rlDest, rlSrc1);
2402 return false;
2403 }
2404 default:
2405 return true;
2406 }
2407 oatFlushAllRegs(cUnit); /* Send everything to home location */
Ian Rogersab2b55d2012-03-18 00:06:11 -07002408 callRuntimeHelperRegLocationRegLocation(cUnit, funcOffset, rlSrc1, rlSrc2);
Ian Rogersf7d9ad32012-03-13 18:45:39 -07002409 rlResult = oatGetReturnWide(cUnit, true);
buzbee31a4a6f2012-02-28 15:36:15 -08002410 storeValueWide(cUnit, rlDest, rlResult);
2411 return false;
2412}
2413
2414bool genConversionPortable(CompilationUnit* cUnit, MIR* mir)
2415{
Elliott Hughesadb8c672012-03-06 16:49:32 -08002416 Instruction::Code opcode = mir->dalvikInsn.opcode;
buzbee31a4a6f2012-02-28 15:36:15 -08002417
2418 switch (opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08002419 case Instruction::INT_TO_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002420 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pI2f),
buzbee31a4a6f2012-02-28 15:36:15 -08002421 1, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002422 case Instruction::FLOAT_TO_INT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002423 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pF2iz),
buzbee31a4a6f2012-02-28 15:36:15 -08002424 1, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002425 case Instruction::DOUBLE_TO_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002426 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pD2f),
buzbee31a4a6f2012-02-28 15:36:15 -08002427 2, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002428 case Instruction::FLOAT_TO_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002429 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pF2d),
buzbee31a4a6f2012-02-28 15:36:15 -08002430 1, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002431 case Instruction::INT_TO_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002432 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pI2d),
buzbee31a4a6f2012-02-28 15:36:15 -08002433 1, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002434 case Instruction::DOUBLE_TO_INT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002435 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pD2iz),
buzbee31a4a6f2012-02-28 15:36:15 -08002436 2, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002437 case Instruction::FLOAT_TO_LONG:
Ian Rogers57b86d42012-03-27 16:05:41 -07002438 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pF2l),
2439 1, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002440 case Instruction::LONG_TO_FLOAT:
Ian Rogers57b86d42012-03-27 16:05:41 -07002441 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pL2f),
buzbee31a4a6f2012-02-28 15:36:15 -08002442 2, 1);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002443 case Instruction::DOUBLE_TO_LONG:
Ian Rogers57b86d42012-03-27 16:05:41 -07002444 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pD2l),
2445 2, 2);
Elliott Hughesadb8c672012-03-06 16:49:32 -08002446 case Instruction::LONG_TO_DOUBLE:
Ian Rogers57b86d42012-03-27 16:05:41 -07002447 return genConversionCall(cUnit, mir, ENTRYPOINT_OFFSET(pL2d),
buzbee31a4a6f2012-02-28 15:36:15 -08002448 2, 2);
2449 default:
2450 return true;
2451 }
2452 return false;
2453}
2454
2455/*
2456 * Generate callout to updateDebugger. Note that we're overloading
2457 * the use of rSUSPEND here. When the debugger is active, this
2458 * register holds the address of the update function. So, if it's
2459 * non-null, we call out to it.
2460 *
2461 * Note also that rRET0 and rRET1 must be preserved across this
2462 * code. This must be handled by the stub.
2463 */
2464void genDebuggerUpdate(CompilationUnit* cUnit, int32_t offset)
2465{
2466 // Following DCHECK verifies that dPC is in range of single load immediate
2467 DCHECK((offset == DEBUGGER_METHOD_ENTRY) ||
2468 (offset == DEBUGGER_METHOD_EXIT) || ((offset & 0xffff) == offset));
2469 oatClobberCalleeSave(cUnit);
2470#if defined(TARGET_ARM)
2471 opRegImm(cUnit, kOpCmp, rSUSPEND, 0);
buzbee82488f52012-03-02 08:20:26 -08002472 opIT(cUnit, kArmCondNe, "T");
buzbee31a4a6f2012-02-28 15:36:15 -08002473 loadConstant(cUnit, rARG2, offset); // arg2 <- Entry code
2474 opReg(cUnit, kOpBlx, rSUSPEND);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002475#elif defined(TARGET_X86)
2476 UNIMPLEMENTED(FATAL);
buzbee31a4a6f2012-02-28 15:36:15 -08002477#else
buzbee82488f52012-03-02 08:20:26 -08002478 LIR* branch = opCmpImmBranch(cUnit, kCondEq, rSUSPEND, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08002479 loadConstant(cUnit, rARG2, offset);
2480 opReg(cUnit, kOpBlx, rSUSPEND);
2481 LIR* target = newLIR0(cUnit, kPseudoTargetLabel);
buzbee31a4a6f2012-02-28 15:36:15 -08002482 branch->target = (LIR*)target;
2483#endif
2484 oatFreeTemp(cUnit, rARG2);
2485}
2486
2487/* Check if we need to check for pending suspend request */
2488void genSuspendTest(CompilationUnit* cUnit, MIR* mir)
2489{
2490 if (NO_SUSPEND || (mir->optimizationFlags & MIR_IGNORE_SUSPEND_CHECK)) {
2491 return;
2492 }
2493 oatFlushAllRegs(cUnit);
buzbee31a4a6f2012-02-28 15:36:15 -08002494 if (cUnit->genDebugger) {
2495 // If generating code for the debugger, always check for suspension
Ian Rogers6cbb2bd2012-03-16 13:45:30 -07002496#if defined(TARGET_X86)
2497 UNIMPLEMENTED(FATAL);
2498#else
Ian Rogers57b86d42012-03-27 16:05:41 -07002499 int rTgt = loadHelper(cUnit, ENTRYPOINT_OFFSET(pTestSuspendFromCode));
buzbee86a4bce2012-03-06 18:15:00 -08002500 opReg(cUnit, kOpBlx, rTgt);
2501 // Refresh rSUSPEND
2502 loadWordDisp(cUnit, rSELF,
Ian Rogers57b86d42012-03-27 16:05:41 -07002503 ENTRYPOINT_OFFSET(pUpdateDebuggerFromCode),
buzbee86a4bce2012-03-06 18:15:00 -08002504 rSUSPEND);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002505#endif
buzbee31a4a6f2012-02-28 15:36:15 -08002506 } else {
Ian Rogersb5d09b22012-03-06 22:14:17 -08002507 LIR* branch = NULL;
buzbee31a4a6f2012-02-28 15:36:15 -08002508#if defined(TARGET_ARM)
2509 // In non-debug case, only check periodically
2510 newLIR2(cUnit, kThumbSubRI8, rSUSPEND, 1);
buzbee82488f52012-03-02 08:20:26 -08002511 branch = opCondBranch(cUnit, kCondEq, NULL);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002512#elif defined(TARGET_X86)
Elliott Hughescd4935f2012-04-10 16:15:59 -07002513 newLIR2(cUnit, kX86Cmp32TI8, Thread::SuspendCountOffset().Int32Value(), 0);
Ian Rogersb5d09b22012-03-06 22:14:17 -08002514 branch = opCondBranch(cUnit, kCondNe, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08002515#else
2516 opRegImm(cUnit, kOpSub, rSUSPEND, 1);
buzbee82488f52012-03-02 08:20:26 -08002517 branch = opCmpImmBranch(cUnit, kCondEq, rSUSPEND, 0, NULL);
buzbee31a4a6f2012-02-28 15:36:15 -08002518#endif
buzbee86a4bce2012-03-06 18:15:00 -08002519 LIR* retLab = newLIR0(cUnit, kPseudoTargetLabel);
2520 LIR* target = rawLIR(cUnit, cUnit->currentDalvikOffset,
2521 kPseudoSuspendTarget, (intptr_t)retLab, mir->offset);
2522 branch->target = (LIR*)target;
2523 oatInsertGrowableList(cUnit, &cUnit->suspendLaunchpads, (intptr_t)target);
buzbee31a4a6f2012-02-28 15:36:15 -08002524 }
buzbee31a4a6f2012-02-28 15:36:15 -08002525}
2526
buzbeefead2932012-03-30 14:02:01 -07002527/* Check if we need to check for pending suspend request */
2528void genSuspendTestAndBranch(CompilationUnit* cUnit, MIR* mir, LIR* target)
2529{
2530 if (NO_SUSPEND || (mir->optimizationFlags & MIR_IGNORE_SUSPEND_CHECK)) {
2531 opUnconditionalBranch(cUnit, target);
2532 return;
2533 }
2534 if (cUnit->genDebugger) {
2535 genSuspendTest(cUnit, mir);
2536 opUnconditionalBranch(cUnit, target);
2537 } else {
2538#if defined(TARGET_ARM)
2539 // In non-debug case, only check periodically
2540 newLIR2(cUnit, kThumbSubRI8, rSUSPEND, 1);
2541 opCondBranch(cUnit, kCondNe, target);
2542#elif defined(TARGET_X86)
Elliott Hughescd4935f2012-04-10 16:15:59 -07002543 newLIR2(cUnit, kX86Cmp32TI8, Thread::SuspendCountOffset().Int32Value(), 0);
buzbeefead2932012-03-30 14:02:01 -07002544 opCondBranch(cUnit, kCondEq, target);
2545#else
2546 opRegImm(cUnit, kOpSub, rSUSPEND, 1);
2547 opCmpImmBranch(cUnit, kCondNe, rSUSPEND, 0, target);
2548#endif
2549 LIR* launchPad = rawLIR(cUnit, cUnit->currentDalvikOffset,
2550 kPseudoSuspendTarget, (intptr_t)target, mir->offset);
2551 oatFlushAllRegs(cUnit);
2552 opUnconditionalBranch(cUnit, launchPad);
2553 oatInsertGrowableList(cUnit, &cUnit->suspendLaunchpads, (intptr_t)launchPad);
2554 }
2555}
2556
buzbee31a4a6f2012-02-28 15:36:15 -08002557} // namespace art