blob: f89d6488dbbc6bf91a6f6f4cc7d06dd3805cad14 [file] [log] [blame]
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001/*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "interpreter.h"
18
19#include <math.h>
20
Elliott Hughes07ed66b2012-12-12 18:34:25 -080021#include "base/logging.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070022#include "common_throws.h"
jeffhao373c52f2012-11-20 16:11:52 -080023#include "debugger.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070024#include "dex_instruction.h"
25#include "invoke_arg_array_builder.h"
Ian Rogers64b6d142012-10-29 16:34:15 -070026#include "nth_caller_visitor.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070027#include "object.h"
28#include "object_utils.h"
29#include "runtime_support.h"
30#include "ScopedLocalRef.h"
31#include "scoped_thread_state_change.h"
32#include "thread.h"
33
34namespace art {
35namespace interpreter {
36
jeffhao87a6edd2012-11-20 18:05:18 -080037static const int32_t kMaxInt = std::numeric_limits<int32_t>::max();
38static const int32_t kMinInt = std::numeric_limits<int32_t>::min();
39static const int64_t kMaxLong = std::numeric_limits<int64_t>::max();
40static const int64_t kMinLong = std::numeric_limits<int64_t>::min();
41
Ian Rogers64b6d142012-10-29 16:34:15 -070042static void UnstartedRuntimeInvoke(Thread* self, AbstractMethod* target_method,
43 Object* receiver, JValue* args, JValue* result)
44 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
45 // In a runtime that's not started we intercept certain methods to avoid complicated dependency
46 // problems in core libraries.
47 std::string name(PrettyMethod(target_method));
48 if (name == "java.lang.Class java.lang.Class.forName(java.lang.String)") {
49 std::string descriptor(DotToDescriptor(args[0].GetL()->AsString()->ToModifiedUtf8().c_str()));
50 ClassLoader* class_loader = NULL; // shadow_frame.GetMethod()->GetDeclaringClass()->GetClassLoader();
51 Class* found = Runtime::Current()->GetClassLinker()->FindClass(descriptor.c_str(),
52 class_loader);
53 CHECK(found != NULL) << "Class.forName failed in un-started runtime for class: "
54 << PrettyDescriptor(descriptor);
55 result->SetL(found);
56 } else if (name == "java.lang.Object java.lang.Class.newInstance()") {
57 Class* klass = receiver->AsClass();
58 AbstractMethod* c = klass->FindDeclaredDirectMethod("<init>", "()V");
59 CHECK(c != NULL);
60 Object* obj = klass->AllocObject(self);
61 CHECK(obj != NULL);
62 EnterInterpreterFromInvoke(self, c, obj, NULL, NULL);
63 result->SetL(obj);
64 } else if (name == "java.lang.reflect.Field java.lang.Class.getDeclaredField(java.lang.String)") {
65 // Special managed code cut-out to allow field lookup in a un-started runtime that'd fail
66 // going the reflective Dex way.
67 Class* klass = receiver->AsClass();
68 String* name = args[0].GetL()->AsString();
69 Field* found = NULL;
70 FieldHelper fh;
71 ObjectArray<Field>* fields = klass->GetIFields();
72 for (int32_t i = 0; i < fields->GetLength() && found == NULL; ++i) {
73 Field* f = fields->Get(i);
74 fh.ChangeField(f);
75 if (name->Equals(fh.GetName())) {
76 found = f;
77 }
78 }
79 if (found == NULL) {
80 fields = klass->GetSFields();
81 for (int32_t i = 0; i < fields->GetLength() && found == NULL; ++i) {
82 Field* f = fields->Get(i);
83 fh.ChangeField(f);
84 if (name->Equals(fh.GetName())) {
85 found = f;
86 }
87 }
88 }
89 CHECK(found != NULL)
90 << "Failed to find field in Class.getDeclaredField in un-started runtime. name="
91 << name->ToModifiedUtf8() << " class=" << PrettyDescriptor(klass);
92 // TODO: getDeclaredField calls GetType once the field is found to ensure a
93 // NoClassDefFoundError is thrown if the field's type cannot be resolved.
94 result->SetL(found);
95 } else if (name == "void java.lang.System.arraycopy(java.lang.Object, int, java.lang.Object, int, int)") {
96 // Special case array copying without initializing System.
97 Class* ctype = args[0].GetL()->GetClass()->GetComponentType();
98 jint srcPos = args[1].GetI();
99 jint dstPos = args[3].GetI();
100 jint length = args[4].GetI();
101 if (!ctype->IsPrimitive()) {
102 ObjectArray<Object>* src = args[0].GetL()->AsObjectArray<Object>();
103 ObjectArray<Object>* dst = args[2].GetL()->AsObjectArray<Object>();
104 for (jint i = 0; i < length; ++i) {
105 dst->Set(dstPos + i, src->Get(srcPos + i));
106 }
107 } else if (ctype->IsPrimitiveChar()) {
108 CharArray* src = args[0].GetL()->AsCharArray();
109 CharArray* dst = args[2].GetL()->AsCharArray();
110 for (jint i = 0; i < length; ++i) {
111 dst->Set(dstPos + i, src->Get(srcPos + i));
112 }
113 } else if (ctype->IsPrimitiveInt()) {
114 IntArray* src = args[0].GetL()->AsIntArray();
115 IntArray* dst = args[2].GetL()->AsIntArray();
116 for (jint i = 0; i < length; ++i) {
117 dst->Set(dstPos + i, src->Get(srcPos + i));
118 }
119 } else {
120 UNIMPLEMENTED(FATAL) << "System.arraycopy of unexpected type: " << PrettyDescriptor(ctype);
121 }
122 } else {
123 // Not special, continue with regular interpreter execution.
124 EnterInterpreterFromInvoke(self, target_method, receiver, args, result);
125 }
126}
127
128// Hand select a number of methods to be run in a not yet started runtime without using JNI.
129static void UnstartedRuntimeJni(Thread* self, AbstractMethod* method,
130 Object* receiver, JValue* args, JValue* result)
131 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
132 std::string name(PrettyMethod(method));
133 if (name == "java.lang.ClassLoader dalvik.system.VMStack.getCallingClassLoader()") {
134 result->SetL(NULL);
135 } else if (name == "java.lang.Class dalvik.system.VMStack.getStackClass2()") {
136 NthCallerVisitor visitor(self->GetManagedStack(), NULL, 3);
137 visitor.WalkStack();
138 result->SetL(visitor.caller->GetDeclaringClass());
139 } else if (name == "double java.lang.Math.log(double)") {
140 result->SetD(log(args[0].GetD()));
141 } else if (name == "java.lang.String java.lang.Class.getNameNative()") {
142 result->SetL(receiver->AsClass()->ComputeName());
143 } else if (name == "int java.lang.Float.floatToRawIntBits(float)") {
144 result->SetI(args[0].GetI());
145 } else if (name == "float java.lang.Float.intBitsToFloat(int)") {
146 result->SetF(args[0].GetF());
147 } else if (name == "double java.lang.Math.exp(double)") {
148 result->SetD(exp(args[0].GetD()));
149 } else if (name == "java.lang.Object java.lang.Object.internalClone()") {
150 result->SetL(receiver->Clone(self));
151 } else if (name == "void java.lang.Object.notifyAll()") {
152 receiver->NotifyAll();
153 } else if (name == "int java.lang.String.compareTo(java.lang.String)") {
154 String* rhs = args[0].GetL()->AsString();
155 CHECK(rhs != NULL);
156 result->SetI(receiver->AsString()->CompareTo(rhs));
157 } else if (name == "java.lang.String java.lang.String.intern()") {
158 result->SetL(receiver->AsString()->Intern());
159 } else if (name == "int java.lang.String.fastIndexOf(int, int)") {
160 result->SetI(receiver->AsString()->FastIndexOf(args[0].GetI(), args[1].GetI()));
161 } else if (name == "java.lang.Object java.lang.reflect.Array.createMultiArray(java.lang.Class, int[])") {
162 result->SetL(Array::CreateMultiArray(self, args[0].GetL()->AsClass(), args[1].GetL()->AsIntArray()));
163 } else if (name == "java.lang.Object java.lang.Throwable.nativeFillInStackTrace()") {
164 ScopedObjectAccessUnchecked soa(self);
165 result->SetL(soa.Decode<Object*>(self->CreateInternalStackTrace(soa)));
166 } else if (name == "boolean java.nio.ByteOrder.isLittleEndian()") {
167 result->SetJ(JNI_TRUE);
168 } else if (name == "boolean sun.misc.Unsafe.compareAndSwapInt(java.lang.Object, long, int, int)") {
169 Object* obj = args[0].GetL();
170 jlong offset = args[1].GetJ();
171 jint expectedValue = args[2].GetI();
172 jint newValue = args[3].GetI();
173 byte* raw_addr = reinterpret_cast<byte*>(obj) + offset;
174 volatile int32_t* address = reinterpret_cast<volatile int32_t*>(raw_addr);
175 // Note: android_atomic_release_cas() returns 0 on success, not failure.
176 int r = android_atomic_release_cas(expectedValue, newValue, address);
177 result->SetZ(r == 0);
178 } else if (name == "void sun.misc.Unsafe.putObject(java.lang.Object, long, java.lang.Object)") {
179 Object* obj = args[0].GetL();
180 Object* newValue = args[2].GetL();
181 obj->SetFieldObject(MemberOffset(args[1].GetJ()), newValue, false);
182 } else {
183 LOG(FATAL) << "Attempt to invoke native method in non-started runtime: " << name;
184 }
185}
186
187static void InterpreterJni(Thread* self, AbstractMethod* method, StringPiece shorty,
188 Object* receiver, JValue* args, JValue* result)
189 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
190 // TODO: The following enters JNI code using a typedef-ed function rather than the JNI compiler,
191 // it should be removed and JNI compiled stubs used instead.
192 ScopedObjectAccessUnchecked soa(self);
193 if (method->IsStatic()) {
194 if (shorty == "L") {
195 typedef jobject (fnptr)(JNIEnv*, jclass);
196 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
197 ScopedLocalRef<jclass> klass(soa.Env(),
198 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
Ian Rogers556d6372012-11-20 12:19:36 -0800199 jobject jresult;
200 {
201 ScopedThreadStateChange tsc(self, kNative);
202 jresult = fn(soa.Env(), klass.get());
203 }
204 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700205 } else if (shorty == "V") {
206 typedef void (fnptr)(JNIEnv*, jclass);
207 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
208 ScopedLocalRef<jclass> klass(soa.Env(),
209 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
210 ScopedThreadStateChange tsc(self, kNative);
211 fn(soa.Env(), klass.get());
212 } else if (shorty == "Z") {
213 typedef jboolean (fnptr)(JNIEnv*, jclass);
214 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
215 ScopedLocalRef<jclass> klass(soa.Env(),
216 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
217 ScopedThreadStateChange tsc(self, kNative);
218 result->SetZ(fn(soa.Env(), klass.get()));
219 } else if (shorty == "BI") {
220 typedef jbyte (fnptr)(JNIEnv*, jclass, jint);
221 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
222 ScopedLocalRef<jclass> klass(soa.Env(),
223 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
224 ScopedThreadStateChange tsc(self, kNative);
225 result->SetB(fn(soa.Env(), klass.get(), args[0].GetI()));
226 } else if (shorty == "II") {
227 typedef jint (fnptr)(JNIEnv*, jclass, jint);
228 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
229 ScopedLocalRef<jclass> klass(soa.Env(),
230 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
231 ScopedThreadStateChange tsc(self, kNative);
232 result->SetI(fn(soa.Env(), klass.get(), args[0].GetI()));
233 } else if (shorty == "LL") {
234 typedef jobject (fnptr)(JNIEnv*, jclass, jobject);
235 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
236 ScopedLocalRef<jclass> klass(soa.Env(),
237 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
238 ScopedLocalRef<jobject> arg0(soa.Env(),
239 soa.AddLocalReference<jobject>(args[0].GetL()));
Ian Rogers556d6372012-11-20 12:19:36 -0800240 jobject jresult;
241 {
242 ScopedThreadStateChange tsc(self, kNative);
243 jresult = fn(soa.Env(), klass.get(), arg0.get());
244 }
245 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700246 } else if (shorty == "IIZ") {
247 typedef jint (fnptr)(JNIEnv*, jclass, jint, jboolean);
248 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
249 ScopedLocalRef<jclass> klass(soa.Env(),
250 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
251 ScopedThreadStateChange tsc(self, kNative);
252 result->SetI(fn(soa.Env(), klass.get(), args[0].GetI(), args[1].GetZ()));
253 } else if (shorty == "ILI") {
254 typedef jint (fnptr)(JNIEnv*, jclass, jobject, jint);
255 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
256 ScopedLocalRef<jclass> klass(soa.Env(),
257 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
258 ScopedLocalRef<jobject> arg0(soa.Env(),
259 soa.AddLocalReference<jobject>(args[0].GetL()));
260 ScopedThreadStateChange tsc(self, kNative);
261 result->SetI(fn(soa.Env(), klass.get(), arg0.get(), args[1].GetI()));
262 } else if (shorty == "SIZ") {
263 typedef jshort (fnptr)(JNIEnv*, jclass, jint, jboolean);
264 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
265 ScopedLocalRef<jclass> klass(soa.Env(),
266 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
267 ScopedThreadStateChange tsc(self, kNative);
268 result->SetS(fn(soa.Env(), klass.get(), args[0].GetI(), args[1].GetZ()));
269 } else if (shorty == "VIZ") {
270 typedef void (fnptr)(JNIEnv*, jclass, jint, jboolean);
271 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
272 ScopedLocalRef<jclass> klass(soa.Env(),
273 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
274 ScopedThreadStateChange tsc(self, kNative);
275 fn(soa.Env(), klass.get(), args[0].GetI(), args[1].GetZ());
276 } else if (shorty == "ZLL") {
277 typedef jboolean (fnptr)(JNIEnv*, jclass, jobject, jobject);
278 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
279 ScopedLocalRef<jclass> klass(soa.Env(),
280 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
281 ScopedLocalRef<jobject> arg0(soa.Env(),
282 soa.AddLocalReference<jobject>(args[0].GetL()));
283 ScopedLocalRef<jobject> arg1(soa.Env(),
284 soa.AddLocalReference<jobject>(args[1].GetL()));
285 ScopedThreadStateChange tsc(self, kNative);
286 result->SetZ(fn(soa.Env(), klass.get(), arg0.get(), arg1.get()));
287 } else if (shorty == "ZILL") {
288 typedef jboolean (fnptr)(JNIEnv*, jclass, jint, jobject, jobject);
289 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
290 ScopedLocalRef<jclass> klass(soa.Env(),
291 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
292 ScopedLocalRef<jobject> arg1(soa.Env(),
293 soa.AddLocalReference<jobject>(args[1].GetL()));
294 ScopedLocalRef<jobject> arg2(soa.Env(),
295 soa.AddLocalReference<jobject>(args[2].GetL()));
296 ScopedThreadStateChange tsc(self, kNative);
297 result->SetZ(fn(soa.Env(), klass.get(), args[0].GetI(), arg1.get(), arg2.get()));
298 } else if (shorty == "VILII") {
299 typedef void (fnptr)(JNIEnv*, jclass, jint, jobject, jint, jint);
300 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
301 ScopedLocalRef<jclass> klass(soa.Env(),
302 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
303 ScopedLocalRef<jobject> arg1(soa.Env(),
304 soa.AddLocalReference<jobject>(args[1].GetL()));
305 ScopedThreadStateChange tsc(self, kNative);
306 fn(soa.Env(), klass.get(), args[0].GetI(), arg1.get(), args[2].GetI(), args[3].GetI());
307 } else if (shorty == "VLILII") {
308 typedef void (fnptr)(JNIEnv*, jclass, jobject, jint, jobject, jint, jint);
309 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
310 ScopedLocalRef<jclass> klass(soa.Env(),
311 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
312 ScopedLocalRef<jobject> arg0(soa.Env(),
313 soa.AddLocalReference<jobject>(args[0].GetL()));
314 ScopedLocalRef<jobject> arg2(soa.Env(),
315 soa.AddLocalReference<jobject>(args[2].GetL()));
316 ScopedThreadStateChange tsc(self, kNative);
317 fn(soa.Env(), klass.get(), arg0.get(), args[1].GetI(), arg2.get(), args[3].GetI(),
318 args[4].GetI());
319 } else {
320 LOG(FATAL) << "Do something with static native method: " << PrettyMethod(method)
321 << " shorty: " << shorty;
322 }
323 } else {
324 if (shorty == "L") {
325 typedef jobject (fnptr)(JNIEnv*, jobject);
326 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
327 ScopedLocalRef<jobject> rcvr(soa.Env(),
328 soa.AddLocalReference<jobject>(receiver));
Ian Rogers556d6372012-11-20 12:19:36 -0800329 jobject jresult;
330 {
331 ScopedThreadStateChange tsc(self, kNative);
332 jresult = fn(soa.Env(), rcvr.get());
333 }
334 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700335 } else if (shorty == "LL") {
336 typedef jobject (fnptr)(JNIEnv*, jobject, jobject);
337 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
338 ScopedLocalRef<jobject> rcvr(soa.Env(),
339 soa.AddLocalReference<jobject>(receiver));
340 ScopedLocalRef<jobject> arg0(soa.Env(),
341 soa.AddLocalReference<jobject>(args[0].GetL()));
Ian Rogers556d6372012-11-20 12:19:36 -0800342 jobject jresult;
343 {
344 ScopedThreadStateChange tsc(self, kNative);
345 jresult = fn(soa.Env(), rcvr.get(), arg0.get());
346
347 }
348 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700349 ScopedThreadStateChange tsc(self, kNative);
Ian Rogers64b6d142012-10-29 16:34:15 -0700350 } else if (shorty == "III") {
351 typedef jint (fnptr)(JNIEnv*, jobject, jint, jint);
352 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
353 ScopedLocalRef<jobject> rcvr(soa.Env(),
354 soa.AddLocalReference<jobject>(receiver));
355 ScopedThreadStateChange tsc(self, kNative);
356 result->SetI(fn(soa.Env(), rcvr.get(), args[0].GetI(), args[1].GetI()));
357 } else {
358 LOG(FATAL) << "Do something with native method: " << PrettyMethod(method)
359 << " shorty: " << shorty;
360 }
361 }
362}
363
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700364static void DoMonitorEnter(Thread* self, Object* ref) NO_THREAD_SAFETY_ANALYSIS {
365 ref->MonitorEnter(self);
366}
367
368static void DoMonitorExit(Thread* self, Object* ref) NO_THREAD_SAFETY_ANALYSIS {
369 ref->MonitorExit(self);
370}
371
372static void DoInvoke(Thread* self, MethodHelper& mh, ShadowFrame& shadow_frame,
373 const DecodedInstruction& dec_insn, InvokeType type, bool is_range,
374 JValue* result)
375 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
376 Object* receiver;
377 if (type == kStatic) {
378 receiver = NULL;
379 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800380 receiver = shadow_frame.GetVRegReference(dec_insn.vC);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700381 }
382 uint32_t method_idx = dec_insn.vB;
383 AbstractMethod* target_method = FindMethodFromCode(method_idx, receiver,
384 shadow_frame.GetMethod(), self, true,
385 type);
386 if (UNLIKELY(target_method == NULL)) {
387 CHECK(self->IsExceptionPending());
388 result->SetJ(0);
389 return;
390 }
391 mh.ChangeMethod(target_method);
392 ArgArray arg_array(mh.GetShorty(), mh.GetShortyLength());
393 if (is_range) {
394 arg_array.BuildArgArray(shadow_frame, dec_insn.vC + (type != kStatic ? 1 : 0));
395 } else {
396 arg_array.BuildArgArray(shadow_frame, dec_insn.arg + (type != kStatic ? 1 : 0));
397 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700398 if (LIKELY(Runtime::Current()->IsStarted())) {
399 target_method->Invoke(self, receiver, arg_array.get(), result);
400 } else {
401 UnstartedRuntimeInvoke(self, target_method, receiver, arg_array.get(), result);
402 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700403 mh.ChangeMethod(shadow_frame.GetMethod());
404}
405
406static void DoFieldGet(Thread* self, ShadowFrame& shadow_frame,
407 const DecodedInstruction& dec_insn, FindFieldType find_type,
408 Primitive::Type field_type)
409 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
410 bool is_static = (find_type == StaticObjectRead) || (find_type == StaticPrimitiveRead);
411 uint32_t field_idx = is_static ? dec_insn.vB : dec_insn.vC;
412 Field* f = FindFieldFromCode(field_idx, shadow_frame.GetMethod(), self,
413 find_type, Primitive::FieldSize(field_type));
414 if (LIKELY(f != NULL)) {
415 Object* obj;
416 if (is_static) {
417 obj = f->GetDeclaringClass();
418 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800419 obj = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700420 if (UNLIKELY(obj == NULL)) {
421 ThrowNullPointerExceptionForFieldAccess(f, true);
Ian Rogers689d9f02012-11-20 16:30:29 -0800422 return;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700423 }
424 }
425 switch (field_type) {
426 case Primitive::kPrimBoolean:
427 shadow_frame.SetVReg(dec_insn.vA, f->GetBoolean(obj));
428 break;
429 case Primitive::kPrimByte:
430 shadow_frame.SetVReg(dec_insn.vA, f->GetByte(obj));
431 break;
432 case Primitive::kPrimChar:
433 shadow_frame.SetVReg(dec_insn.vA, f->GetChar(obj));
434 break;
435 case Primitive::kPrimShort:
436 shadow_frame.SetVReg(dec_insn.vA, f->GetShort(obj));
437 break;
438 case Primitive::kPrimInt:
439 shadow_frame.SetVReg(dec_insn.vA, f->GetInt(obj));
440 break;
441 case Primitive::kPrimLong:
442 shadow_frame.SetVRegLong(dec_insn.vA, f->GetLong(obj));
443 break;
444 case Primitive::kPrimNot:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800445 shadow_frame.SetVRegReference(dec_insn.vA, f->GetObject(obj));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700446 break;
447 default:
448 LOG(FATAL) << "Unreachable: " << field_type;
449 }
450 }
451}
452
453static void DoFieldPut(Thread* self, ShadowFrame& shadow_frame,
454 const DecodedInstruction& dec_insn, FindFieldType find_type,
455 Primitive::Type field_type)
456 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
457 bool is_static = (find_type == StaticObjectWrite) || (find_type == StaticPrimitiveWrite);
458 uint32_t field_idx = is_static ? dec_insn.vB : dec_insn.vC;
459 Field* f = FindFieldFromCode(field_idx, shadow_frame.GetMethod(), self,
460 find_type, Primitive::FieldSize(field_type));
461 if (LIKELY(f != NULL)) {
462 Object* obj;
463 if (is_static) {
464 obj = f->GetDeclaringClass();
465 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800466 obj = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700467 if (UNLIKELY(obj == NULL)) {
468 ThrowNullPointerExceptionForFieldAccess(f, false);
Ian Rogers689d9f02012-11-20 16:30:29 -0800469 return;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700470 }
471 }
472 switch (field_type) {
473 case Primitive::kPrimBoolean:
474 f->SetBoolean(obj, shadow_frame.GetVReg(dec_insn.vA));
475 break;
476 case Primitive::kPrimByte:
477 f->SetByte(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700478 break;
479 case Primitive::kPrimChar:
480 f->SetChar(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700481 break;
482 case Primitive::kPrimShort:
483 f->SetShort(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700484 break;
485 case Primitive::kPrimInt:
486 f->SetInt(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700487 break;
488 case Primitive::kPrimLong:
489 f->SetLong(obj, shadow_frame.GetVRegLong(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700490 break;
491 case Primitive::kPrimNot:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800492 f->SetObj(obj, shadow_frame.GetVRegReference(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700493 break;
494 default:
495 LOG(FATAL) << "Unreachable: " << field_type;
496 }
497 }
498}
499
jeffhaod91398c2012-11-20 17:17:33 -0800500static void DoIntDivide(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
501 int32_t dividend, int32_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800502 if (UNLIKELY(divisor == 0)) {
503 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
504 } else if (UNLIKELY(dividend == kMinInt && divisor == -1)) {
505 shadow_frame.SetVReg(result_reg, kMinInt);
506 } else {
507 shadow_frame.SetVReg(result_reg, dividend / divisor);
508 }
509}
510
511static void DoIntRemainder(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
512 int32_t dividend, int32_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800513 if (UNLIKELY(divisor == 0)) {
514 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
515 } else if (UNLIKELY(dividend == kMinInt && divisor == -1)) {
516 shadow_frame.SetVReg(result_reg, 0);
517 } else {
518 shadow_frame.SetVReg(result_reg, dividend % divisor);
519 }
520}
521
522static void DoLongDivide(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
523 int64_t dividend, int64_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800524 if (UNLIKELY(divisor == 0)) {
525 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
526 } else if (UNLIKELY(dividend == kMinLong && divisor == -1)) {
527 shadow_frame.SetVRegLong(result_reg, kMinLong);
528 } else {
529 shadow_frame.SetVRegLong(result_reg, dividend / divisor);
530 }
531}
532
533static void DoLongRemainder(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
534 int64_t dividend, int64_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800535 if (UNLIKELY(divisor == 0)) {
536 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
537 } else if (UNLIKELY(dividend == kMinLong && divisor == -1)) {
538 shadow_frame.SetVRegLong(result_reg, 0);
539 } else {
540 shadow_frame.SetVRegLong(result_reg, dividend % divisor);
541 }
542}
543
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700544static JValue Execute(Thread* self, MethodHelper& mh, const DexFile::CodeItem* code_item,
545 ShadowFrame& shadow_frame) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
546 const uint16_t* insns = code_item->insns_;
547 const Instruction* inst = Instruction::At(insns + shadow_frame.GetDexPC());
jeffhao14f0db92012-12-14 17:50:42 -0800548 bool entry = (inst->GetDexPc(insns) == 0);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700549 JValue result_register;
550 while (true) {
jeffhao373c52f2012-11-20 16:11:52 -0800551 CheckSuspend(self);
552 uint32_t dex_pc = inst->GetDexPc(insns);
553 shadow_frame.SetDexPC(dex_pc);
jeffhao14f0db92012-12-14 17:50:42 -0800554 if (entry) {
555 Dbg::UpdateDebugger(-1, self);
556 }
557 entry = false;
jeffhao373c52f2012-11-20 16:11:52 -0800558 Dbg::UpdateDebugger(dex_pc, self);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700559 DecodedInstruction dec_insn(inst);
Ian Rogers64b6d142012-10-29 16:34:15 -0700560 const bool kTracing = false;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700561 if (kTracing) {
562 LOG(INFO) << PrettyMethod(shadow_frame.GetMethod())
563 << StringPrintf("\n0x%x: %s\nReferences:",
564 inst->GetDexPc(insns), inst->DumpString(&mh.GetDexFile()).c_str());
TDYa127ce4cc0d2012-11-18 16:59:53 -0800565 for (size_t i = 0; i < shadow_frame.NumberOfVRegs(); ++i) {
566 Object* o = shadow_frame.GetVRegReference(i);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700567 if (o != NULL) {
568 if (o->GetClass()->IsStringClass() && o->AsString()->GetCharArray() != NULL) {
569 LOG(INFO) << i << ": java.lang.String " << static_cast<void*>(o)
570 << " \"" << o->AsString()->ToModifiedUtf8() << "\"";
571 } else {
572 LOG(INFO) << i << ": " << PrettyTypeOf(o) << " " << static_cast<void*>(o);
573 }
574 } else {
575 LOG(INFO) << i << ": null";
576 }
577 }
578 LOG(INFO) << "vregs:";
TDYa127ce4cc0d2012-11-18 16:59:53 -0800579 for (size_t i = 0; i < shadow_frame.NumberOfVRegs(); ++i) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700580 LOG(INFO) << StringPrintf("%d: %08x", i, shadow_frame.GetVReg(i));
581 }
582 }
583 const Instruction* next_inst = inst->Next();
584 switch (dec_insn.opcode) {
585 case Instruction::NOP:
586 break;
587 case Instruction::MOVE:
588 case Instruction::MOVE_FROM16:
589 case Instruction::MOVE_16:
590 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
591 break;
592 case Instruction::MOVE_WIDE:
593 case Instruction::MOVE_WIDE_FROM16:
594 case Instruction::MOVE_WIDE_16:
595 shadow_frame.SetVRegLong(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
596 break;
597 case Instruction::MOVE_OBJECT:
598 case Instruction::MOVE_OBJECT_FROM16:
599 case Instruction::MOVE_OBJECT_16:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800600 shadow_frame.SetVRegReference(dec_insn.vA, shadow_frame.GetVRegReference(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700601 break;
602 case Instruction::MOVE_RESULT:
603 shadow_frame.SetVReg(dec_insn.vA, result_register.GetI());
604 break;
605 case Instruction::MOVE_RESULT_WIDE:
606 shadow_frame.SetVRegLong(dec_insn.vA, result_register.GetJ());
607 break;
608 case Instruction::MOVE_RESULT_OBJECT:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800609 shadow_frame.SetVRegReference(dec_insn.vA, result_register.GetL());
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700610 break;
611 case Instruction::MOVE_EXCEPTION: {
612 Throwable* exception = self->GetException();
613 self->ClearException();
TDYa127ce4cc0d2012-11-18 16:59:53 -0800614 shadow_frame.SetVRegReference(dec_insn.vA, exception);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700615 break;
616 }
617 case Instruction::RETURN_VOID: {
618 JValue result;
619 result.SetJ(0);
620 return result;
621 }
622 case Instruction::RETURN: {
623 JValue result;
624 result.SetJ(0);
625 result.SetI(shadow_frame.GetVReg(dec_insn.vA));
626 return result;
627 }
628 case Instruction::RETURN_WIDE: {
629 JValue result;
630 result.SetJ(shadow_frame.GetVRegLong(dec_insn.vA));
631 return result;
632 }
633 case Instruction::RETURN_OBJECT: {
634 JValue result;
635 result.SetJ(0);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800636 result.SetL(shadow_frame.GetVRegReference(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700637 return result;
638 }
639 case Instruction::CONST_4: {
Ian Rogers64b6d142012-10-29 16:34:15 -0700640 int32_t val = static_cast<int32_t>(dec_insn.vB << 28) >> 28;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700641 shadow_frame.SetVReg(dec_insn.vA, val);
642 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800643 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700644 }
645 break;
646 }
647 case Instruction::CONST_16: {
648 int32_t val = static_cast<int16_t>(dec_insn.vB);
649 shadow_frame.SetVReg(dec_insn.vA, val);
650 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800651 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700652 }
653 break;
654 }
655 case Instruction::CONST: {
656 int32_t val = dec_insn.vB;
657 shadow_frame.SetVReg(dec_insn.vA, val);
658 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800659 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700660 }
661 break;
662 }
663 case Instruction::CONST_HIGH16: {
664 int32_t val = dec_insn.vB << 16;
665 shadow_frame.SetVReg(dec_insn.vA, val);
666 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800667 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700668 }
669 break;
670 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700671 case Instruction::CONST_WIDE_16:
672 shadow_frame.SetVRegLong(dec_insn.vA, static_cast<int16_t>(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700673 break;
Ian Rogers64b6d142012-10-29 16:34:15 -0700674 case Instruction::CONST_WIDE_32:
675 shadow_frame.SetVRegLong(dec_insn.vA, static_cast<int32_t>(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700676 break;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700677 case Instruction::CONST_WIDE:
Ian Rogers64b6d142012-10-29 16:34:15 -0700678 shadow_frame.SetVRegLong(dec_insn.vA, dec_insn.vB_wide);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700679 break;
680 case Instruction::CONST_WIDE_HIGH16:
Ian Rogers64b6d142012-10-29 16:34:15 -0700681 shadow_frame.SetVRegLong(dec_insn.vA, static_cast<uint64_t>(dec_insn.vB) << 48);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700682 break;
683 case Instruction::CONST_STRING:
684 case Instruction::CONST_STRING_JUMBO: {
685 if (UNLIKELY(!String::GetJavaLangString()->IsInitialized())) {
686 Runtime::Current()->GetClassLinker()->EnsureInitialized(String::GetJavaLangString(),
687 true, true);
688 }
689 String* s = mh.ResolveString(dec_insn.vB);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800690 shadow_frame.SetVRegReference(dec_insn.vA, s);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700691 break;
692 }
jeffhao0a9bb732012-11-26 12:28:49 -0800693 case Instruction::CONST_CLASS: {
694 Class* c = ResolveVerifyAndClinit(dec_insn.vB, shadow_frame.GetMethod(), self, false, true);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800695 shadow_frame.SetVRegReference(dec_insn.vA, c);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700696 break;
jeffhao0a9bb732012-11-26 12:28:49 -0800697 }
jeffhao87a6edd2012-11-20 18:05:18 -0800698 case Instruction::MONITOR_ENTER: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800699 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao87a6edd2012-11-20 18:05:18 -0800700 if (UNLIKELY(obj == NULL)) {
701 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
702 } else {
703 DoMonitorEnter(self, obj);
704 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700705 break;
jeffhao87a6edd2012-11-20 18:05:18 -0800706 }
707 case Instruction::MONITOR_EXIT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800708 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao87a6edd2012-11-20 18:05:18 -0800709 if (UNLIKELY(obj == NULL)) {
710 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
711 } else {
712 DoMonitorExit(self, obj);
713 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700714 break;
jeffhao87a6edd2012-11-20 18:05:18 -0800715 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700716 case Instruction::CHECK_CAST: {
jeffhao0a9bb732012-11-26 12:28:49 -0800717 Class* c = ResolveVerifyAndClinit(dec_insn.vB, shadow_frame.GetMethod(), self, false, true);
jeffhao87a6edd2012-11-20 18:05:18 -0800718 if (UNLIKELY(c == NULL)) {
719 CHECK(self->IsExceptionPending());
720 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800721 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao87a6edd2012-11-20 18:05:18 -0800722 if (UNLIKELY(obj != NULL && !obj->InstanceOf(c))) {
723 self->ThrowNewExceptionF("Ljava/lang/ClassCastException;",
724 "%s cannot be cast to %s",
725 PrettyDescriptor(obj->GetClass()).c_str(),
726 PrettyDescriptor(c).c_str());
727 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700728 }
729 break;
730 }
731 case Instruction::INSTANCE_OF: {
jeffhao0a9bb732012-11-26 12:28:49 -0800732 Class* c = ResolveVerifyAndClinit(dec_insn.vC, shadow_frame.GetMethod(), self, false, true);
jeffhao87a6edd2012-11-20 18:05:18 -0800733 if (UNLIKELY(c == NULL)) {
734 CHECK(self->IsExceptionPending());
735 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800736 Object* obj = shadow_frame.GetVRegReference(dec_insn.vB);
jeffhao87a6edd2012-11-20 18:05:18 -0800737 shadow_frame.SetVReg(dec_insn.vA, (obj != NULL && obj->InstanceOf(c)) ? 1 : 0);
738 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700739 break;
740 }
741 case Instruction::ARRAY_LENGTH: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800742 Object* array = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700743 if (UNLIKELY(array == NULL)) {
744 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
745 break;
746 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700747 shadow_frame.SetVReg(dec_insn.vA, array->AsArray()->GetLength());
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700748 break;
749 }
750 case Instruction::NEW_INSTANCE: {
751 Object* obj = AllocObjectFromCode(dec_insn.vB, shadow_frame.GetMethod(), self, true);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800752 shadow_frame.SetVRegReference(dec_insn.vA, obj);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700753 break;
754 }
755 case Instruction::NEW_ARRAY: {
756 int32_t length = shadow_frame.GetVReg(dec_insn.vB);
757 Object* obj = AllocArrayFromCode(dec_insn.vC, shadow_frame.GetMethod(), length, self, true);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800758 shadow_frame.SetVRegReference(dec_insn.vA, obj);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700759 break;
760 }
761 case Instruction::FILLED_NEW_ARRAY:
Ian Rogers64b6d142012-10-29 16:34:15 -0700762 case Instruction::FILLED_NEW_ARRAY_RANGE: {
763 bool is_range = (dec_insn.opcode == Instruction::FILLED_NEW_ARRAY_RANGE);
764 int32_t length = dec_insn.vA;
765 CHECK(is_range || length <= 5);
jeffhao42b4dc42012-12-10 10:25:43 -0800766 if (UNLIKELY(length < 0)) {
767 self->ThrowNewExceptionF("Ljava/lang/NegativeArraySizeException;", "%d", length);
768 break;
769 }
jeffhao0a9bb732012-11-26 12:28:49 -0800770 Class* arrayClass = ResolveVerifyAndClinit(dec_insn.vB, shadow_frame.GetMethod(), self, false, true);
jeffhao42b4dc42012-12-10 10:25:43 -0800771 if (UNLIKELY(arrayClass == NULL)) {
772 CHECK(self->IsExceptionPending());
773 break;
774 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700775 CHECK(arrayClass->IsArrayClass());
jeffhao42b4dc42012-12-10 10:25:43 -0800776 Class* componentClass = arrayClass->GetComponentType();
777 if (UNLIKELY(componentClass->IsPrimitive() && !componentClass->IsPrimitiveInt())) {
778 if (componentClass->IsPrimitiveLong() || componentClass->IsPrimitiveDouble()) {
779 self->ThrowNewExceptionF("Ljava/lang/RuntimeException;",
780 "Bad filled array request for type %s",
781 PrettyDescriptor(componentClass).c_str());
782 } else {
783 self->ThrowNewExceptionF("Ljava/lang/InternalError;",
784 "Found type %s; filled-new-array not implemented for anything but \'int\'",
785 PrettyDescriptor(componentClass).c_str());
786 }
787 break;
788 }
789 Object* newArray = Array::Alloc(self, arrayClass, length);
790 if (newArray != NULL) {
791 for (int32_t i = 0; i < length; ++i) {
792 if (is_range) {
793 if (componentClass->IsPrimitiveInt()) {
794 newArray->AsIntArray()->Set(i, shadow_frame.GetVReg(dec_insn.vC + i));
Ian Rogers64b6d142012-10-29 16:34:15 -0700795 } else {
jeffhao42b4dc42012-12-10 10:25:43 -0800796 newArray->AsObjectArray<Object>()->Set(i, shadow_frame.GetVRegReference(dec_insn.vC + i));
797 }
798 } else {
799 if (componentClass->IsPrimitiveInt()) {
800 newArray->AsIntArray()->Set(i, shadow_frame.GetVReg(dec_insn.arg[i]));
801 } else {
802 newArray->AsObjectArray<Object>()->Set(i, shadow_frame.GetVRegReference(dec_insn.arg[i]));
Ian Rogers64b6d142012-10-29 16:34:15 -0700803 }
804 }
805 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700806 }
jeffhao42b4dc42012-12-10 10:25:43 -0800807 result_register.SetL(newArray);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700808 break;
Ian Rogers64b6d142012-10-29 16:34:15 -0700809 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700810 case Instruction::CMPL_FLOAT: {
811 float val1 = shadow_frame.GetVRegFloat(dec_insn.vB);
812 float val2 = shadow_frame.GetVRegFloat(dec_insn.vC);
813 int32_t result;
814 if (val1 == val2) {
815 result = 0;
816 } else if (val1 > val2) {
817 result = 1;
818 } else {
819 result = -1;
820 }
821 shadow_frame.SetVReg(dec_insn.vA, result);
822 break;
823 }
824 case Instruction::CMPG_FLOAT: {
825 float val1 = shadow_frame.GetVRegFloat(dec_insn.vB);
826 float val2 = shadow_frame.GetVRegFloat(dec_insn.vC);
827 int32_t result;
828 if (val1 == val2) {
829 result = 0;
830 } else if (val1 < val2) {
831 result = -1;
832 } else {
833 result = 1;
834 }
835 shadow_frame.SetVReg(dec_insn.vA, result);
836 break;
837 }
838 case Instruction::CMPL_DOUBLE: {
839 double val1 = shadow_frame.GetVRegDouble(dec_insn.vB);
840 double val2 = shadow_frame.GetVRegDouble(dec_insn.vC);
841 int32_t result;
842 if (val1 == val2) {
843 result = 0;
Ian Rogers58bf0c62012-11-20 16:24:12 -0800844 } else if (val1 > val2) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700845 result = 1;
846 } else {
847 result = -1;
848 }
849 shadow_frame.SetVReg(dec_insn.vA, result);
850 break;
851 }
852
853 case Instruction::CMPG_DOUBLE: {
854 double val1 = shadow_frame.GetVRegDouble(dec_insn.vB);
855 double val2 = shadow_frame.GetVRegDouble(dec_insn.vC);
856 int32_t result;
857 if (val1 == val2) {
858 result = 0;
Ian Rogers58bf0c62012-11-20 16:24:12 -0800859 } else if (val1 < val2) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700860 result = -1;
861 } else {
862 result = 1;
863 }
864 shadow_frame.SetVReg(dec_insn.vA, result);
865 break;
866 }
867 case Instruction::CMP_LONG: {
868 int64_t val1 = shadow_frame.GetVRegLong(dec_insn.vB);
869 int64_t val2 = shadow_frame.GetVRegLong(dec_insn.vC);
870 int32_t result;
jeffhao87a6edd2012-11-20 18:05:18 -0800871 if (val1 > val2) {
Ian Rogers64b6d142012-10-29 16:34:15 -0700872 result = 1;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700873 } else if (val1 == val2) {
874 result = 0;
875 } else {
Ian Rogers64b6d142012-10-29 16:34:15 -0700876 result = -1;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700877 }
878 shadow_frame.SetVReg(dec_insn.vA, result);
879 break;
880 }
881 case Instruction::THROW: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800882 Throwable* t = shadow_frame.GetVRegReference(dec_insn.vA)->AsThrowable();
jeffhao94d6df42012-11-26 16:02:12 -0800883 self->DeliverException(t);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700884 break;
885 }
886 case Instruction::GOTO:
887 case Instruction::GOTO_16:
888 case Instruction::GOTO_32: {
889 uint32_t dex_pc = inst->GetDexPc(insns);
890 next_inst = Instruction::At(insns + dex_pc + dec_insn.vA);
891 break;
892 }
Ian Rogers556d6372012-11-20 12:19:36 -0800893 case Instruction::PACKED_SWITCH: {
894 uint32_t dex_pc = inst->GetDexPc(insns);
895 const uint16_t* switch_data = insns + dex_pc + dec_insn.vB;
896 int32_t test_val = shadow_frame.GetVReg(dec_insn.vA);
897 CHECK_EQ(switch_data[0], static_cast<uint16_t>(Instruction::kPackedSwitchSignature));
898 uint16_t size = switch_data[1];
899 CHECK_GT(size, 0);
900 const int32_t* keys = reinterpret_cast<const int32_t*>(&switch_data[2]);
901 CHECK(IsAligned<4>(keys));
902 int32_t first_key = keys[0];
903 const int32_t* targets = reinterpret_cast<const int32_t*>(&switch_data[4]);
904 CHECK(IsAligned<4>(targets));
905 int32_t index = test_val - first_key;
906 if (index >= 0 && index < size) {
907 next_inst = Instruction::At(insns + dex_pc + targets[index]);
908 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700909 break;
Ian Rogers556d6372012-11-20 12:19:36 -0800910 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700911 case Instruction::SPARSE_SWITCH: {
912 uint32_t dex_pc = inst->GetDexPc(insns);
Ian Rogers556d6372012-11-20 12:19:36 -0800913 const uint16_t* switch_data = insns + dex_pc + dec_insn.vB;
914 int32_t test_val = shadow_frame.GetVReg(dec_insn.vA);
915 CHECK_EQ(switch_data[0], static_cast<uint16_t>(Instruction::kSparseSwitchSignature));
916 uint16_t size = switch_data[1];
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700917 CHECK_GT(size, 0);
Ian Rogers556d6372012-11-20 12:19:36 -0800918 const int32_t* keys = reinterpret_cast<const int32_t*>(&switch_data[2]);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700919 CHECK(IsAligned<4>(keys));
920 const int32_t* entries = keys + size;
921 CHECK(IsAligned<4>(entries));
922 int lo = 0;
923 int hi = size - 1;
924 while (lo <= hi) {
925 int mid = (lo + hi) / 2;
926 int32_t foundVal = keys[mid];
Ian Rogers556d6372012-11-20 12:19:36 -0800927 if (test_val < foundVal) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700928 hi = mid - 1;
Ian Rogers556d6372012-11-20 12:19:36 -0800929 } else if (test_val > foundVal) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700930 lo = mid + 1;
931 } else {
932 next_inst = Instruction::At(insns + dex_pc + entries[mid]);
933 break;
934 }
935 }
936 break;
937 }
938 case Instruction::FILL_ARRAY_DATA: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800939 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao94d6df42012-11-26 16:02:12 -0800940 if (UNLIKELY(obj == NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700941 Thread::Current()->ThrowNewExceptionF("Ljava/lang/NullPointerException;",
942 "null array in FILL_ARRAY_DATA");
943 break;
944 }
jeffhao94d6df42012-11-26 16:02:12 -0800945 Array* array = obj->AsArray();
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700946 DCHECK(array->IsArrayInstance() && !array->IsObjectArray());
947 uint32_t dex_pc = inst->GetDexPc(insns);
948 const Instruction::ArrayDataPayload* payload =
949 reinterpret_cast<const Instruction::ArrayDataPayload*>(insns + dex_pc + dec_insn.vB);
950 if (UNLIKELY(static_cast<int32_t>(payload->element_count) > array->GetLength())) {
951 Thread::Current()->ThrowNewExceptionF("Ljava/lang/ArrayIndexOutOfBoundsException;",
952 "failed FILL_ARRAY_DATA; length=%d, index=%d",
953 array->GetLength(), payload->element_count);
954 break;
955 }
956 uint32_t size_in_bytes = payload->element_count * payload->element_width;
957 memcpy(array->GetRawData(payload->element_width), payload->data, size_in_bytes);
958 break;
959 }
960 case Instruction::IF_EQ: {
961 if (shadow_frame.GetVReg(dec_insn.vA) == shadow_frame.GetVReg(dec_insn.vB)) {
962 uint32_t dex_pc = inst->GetDexPc(insns);
963 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
964 }
965 break;
966 }
967 case Instruction::IF_NE: {
968 if (shadow_frame.GetVReg(dec_insn.vA) != shadow_frame.GetVReg(dec_insn.vB)) {
969 uint32_t dex_pc = inst->GetDexPc(insns);
970 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
971 }
972 break;
973 }
974 case Instruction::IF_LT: {
975 if (shadow_frame.GetVReg(dec_insn.vA) < shadow_frame.GetVReg(dec_insn.vB)) {
976 uint32_t dex_pc = inst->GetDexPc(insns);
977 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
978 }
979 break;
980 }
981 case Instruction::IF_GE: {
982 if (shadow_frame.GetVReg(dec_insn.vA) >= shadow_frame.GetVReg(dec_insn.vB)) {
983 uint32_t dex_pc = inst->GetDexPc(insns);
984 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
985 }
986 break;
987 }
988 case Instruction::IF_GT: {
989 if (shadow_frame.GetVReg(dec_insn.vA) > shadow_frame.GetVReg(dec_insn.vB)) {
990 uint32_t dex_pc = inst->GetDexPc(insns);
991 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
992 }
993 break;
994 }
995 case Instruction::IF_LE: {
996 if (shadow_frame.GetVReg(dec_insn.vA) <= shadow_frame.GetVReg(dec_insn.vB)) {
997 uint32_t dex_pc = inst->GetDexPc(insns);
998 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
999 }
1000 break;
1001 }
1002 case Instruction::IF_EQZ: {
1003 if (shadow_frame.GetVReg(dec_insn.vA) == 0) {
1004 uint32_t dex_pc = inst->GetDexPc(insns);
1005 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1006 }
1007 break;
1008 }
1009 case Instruction::IF_NEZ: {
1010 if (shadow_frame.GetVReg(dec_insn.vA) != 0) {
1011 uint32_t dex_pc = inst->GetDexPc(insns);
1012 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1013 }
1014 break;
1015 }
1016 case Instruction::IF_LTZ: {
1017 if (shadow_frame.GetVReg(dec_insn.vA) < 0) {
1018 uint32_t dex_pc = inst->GetDexPc(insns);
1019 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1020 }
1021 break;
1022 }
1023 case Instruction::IF_GEZ: {
1024 if (shadow_frame.GetVReg(dec_insn.vA) >= 0) {
1025 uint32_t dex_pc = inst->GetDexPc(insns);
1026 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1027 }
1028 break;
1029 }
1030 case Instruction::IF_GTZ: {
1031 if (shadow_frame.GetVReg(dec_insn.vA) > 0) {
1032 uint32_t dex_pc = inst->GetDexPc(insns);
1033 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1034 }
1035 break;
1036 }
1037 case Instruction::IF_LEZ: {
1038 if (shadow_frame.GetVReg(dec_insn.vA) <= 0) {
1039 uint32_t dex_pc = inst->GetDexPc(insns);
1040 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1041 }
1042 break;
1043 }
1044 case Instruction::AGET_BOOLEAN: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001045 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001046 if (UNLIKELY(a == NULL)) {
1047 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1048 break;
1049 }
1050 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001051 shadow_frame.SetVReg(dec_insn.vA, a->AsBooleanArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001052 break;
1053 }
1054 case Instruction::AGET_BYTE: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001055 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001056 if (UNLIKELY(a == NULL)) {
1057 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1058 break;
1059 }
1060 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001061 shadow_frame.SetVReg(dec_insn.vA, a->AsByteArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001062 break;
1063 }
1064 case Instruction::AGET_CHAR: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001065 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001066 if (UNLIKELY(a == NULL)) {
1067 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1068 break;
1069 }
1070 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001071 shadow_frame.SetVReg(dec_insn.vA, a->AsCharArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001072 break;
1073 }
1074 case Instruction::AGET_SHORT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001075 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001076 if (UNLIKELY(a == NULL)) {
1077 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1078 break;
1079 }
1080 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001081 shadow_frame.SetVReg(dec_insn.vA, a->AsShortArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001082 break;
1083 }
1084 case Instruction::AGET: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001085 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001086 if (UNLIKELY(a == NULL)) {
1087 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1088 break;
1089 }
1090 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001091 shadow_frame.SetVReg(dec_insn.vA, a->AsIntArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001092 break;
1093 }
1094 case Instruction::AGET_WIDE: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001095 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001096 if (UNLIKELY(a == NULL)) {
1097 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1098 break;
1099 }
1100 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001101 shadow_frame.SetVRegLong(dec_insn.vA, a->AsLongArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001102 break;
1103 }
1104 case Instruction::AGET_OBJECT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001105 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001106 if (UNLIKELY(a == NULL)) {
1107 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1108 break;
1109 }
1110 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001111 shadow_frame.SetVRegReference(dec_insn.vA, a->AsObjectArray<Object>()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001112 break;
1113 }
1114 case Instruction::APUT_BOOLEAN: {
1115 uint8_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001116 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001117 if (UNLIKELY(a == NULL)) {
1118 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1119 break;
1120 }
1121 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001122 a->AsBooleanArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001123 break;
1124 }
1125 case Instruction::APUT_BYTE: {
1126 int8_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001127 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001128 if (UNLIKELY(a == NULL)) {
1129 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1130 break;
1131 }
1132 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001133 a->AsByteArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001134 break;
1135 }
1136 case Instruction::APUT_CHAR: {
1137 uint16_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001138 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001139 if (UNLIKELY(a == NULL)) {
1140 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1141 break;
1142 }
1143 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001144 a->AsCharArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001145 break;
1146 }
1147 case Instruction::APUT_SHORT: {
1148 int16_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001149 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001150 if (UNLIKELY(a == NULL)) {
1151 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1152 break;
1153 }
1154 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001155 a->AsShortArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001156 break;
1157 }
1158 case Instruction::APUT: {
1159 int32_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001160 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001161 if (UNLIKELY(a == NULL)) {
1162 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1163 break;
1164 }
1165 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001166 a->AsIntArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001167 break;
1168 }
1169 case Instruction::APUT_WIDE: {
1170 int64_t val = shadow_frame.GetVRegLong(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001171 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001172 if (UNLIKELY(a == NULL)) {
1173 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1174 break;
1175 }
1176 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001177 a->AsLongArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001178 break;
1179 }
1180 case Instruction::APUT_OBJECT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001181 Object* val = shadow_frame.GetVRegReference(dec_insn.vA);
1182 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001183 if (UNLIKELY(a == NULL)) {
1184 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1185 break;
1186 }
1187 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001188 a->AsObjectArray<Object>()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001189 break;
1190 }
1191 case Instruction::IGET_BOOLEAN:
1192 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimBoolean);
1193 break;
1194 case Instruction::IGET_BYTE:
1195 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimByte);
1196 break;
1197 case Instruction::IGET_CHAR:
1198 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimChar);
1199 break;
1200 case Instruction::IGET_SHORT:
1201 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimShort);
1202 break;
1203 case Instruction::IGET:
1204 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimInt);
1205 break;
1206 case Instruction::IGET_WIDE:
1207 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimLong);
1208 break;
1209 case Instruction::IGET_OBJECT:
1210 DoFieldGet(self, shadow_frame, dec_insn, InstanceObjectRead, Primitive::kPrimNot);
1211 break;
1212 case Instruction::SGET_BOOLEAN:
1213 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimBoolean);
1214 break;
1215 case Instruction::SGET_BYTE:
1216 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimByte);
1217 break;
1218 case Instruction::SGET_CHAR:
1219 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimChar);
1220 break;
1221 case Instruction::SGET_SHORT:
1222 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimShort);
1223 break;
1224 case Instruction::SGET:
1225 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimInt);
1226 break;
1227 case Instruction::SGET_WIDE:
1228 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimLong);
1229 break;
1230 case Instruction::SGET_OBJECT:
1231 DoFieldGet(self, shadow_frame, dec_insn, StaticObjectRead, Primitive::kPrimNot);
1232 break;
1233 case Instruction::IPUT_BOOLEAN:
1234 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimBoolean);
1235 break;
1236 case Instruction::IPUT_BYTE:
1237 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimByte);
1238 break;
1239 case Instruction::IPUT_CHAR:
1240 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimChar);
1241 break;
1242 case Instruction::IPUT_SHORT:
1243 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimShort);
1244 break;
1245 case Instruction::IPUT:
1246 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimInt);
1247 break;
1248 case Instruction::IPUT_WIDE:
1249 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimLong);
1250 break;
1251 case Instruction::IPUT_OBJECT:
1252 DoFieldPut(self, shadow_frame, dec_insn, InstanceObjectWrite, Primitive::kPrimNot);
1253 break;
1254 case Instruction::SPUT_BOOLEAN:
1255 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimBoolean);
1256 break;
1257 case Instruction::SPUT_BYTE:
1258 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimByte);
1259 break;
1260 case Instruction::SPUT_CHAR:
1261 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimChar);
1262 break;
1263 case Instruction::SPUT_SHORT:
1264 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimShort);
1265 break;
1266 case Instruction::SPUT:
1267 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimInt);
1268 break;
1269 case Instruction::SPUT_WIDE:
1270 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimLong);
1271 break;
1272 case Instruction::SPUT_OBJECT:
1273 DoFieldPut(self, shadow_frame, dec_insn, StaticObjectWrite, Primitive::kPrimNot);
1274 break;
1275 case Instruction::INVOKE_VIRTUAL:
1276 DoInvoke(self, mh, shadow_frame, dec_insn, kVirtual, false, &result_register);
1277 break;
1278 case Instruction::INVOKE_VIRTUAL_RANGE:
1279 DoInvoke(self, mh, shadow_frame, dec_insn, kVirtual, true, &result_register);
1280 break;
1281 case Instruction::INVOKE_SUPER:
1282 DoInvoke(self, mh, shadow_frame, dec_insn, kSuper, false, &result_register);
1283 break;
1284 case Instruction::INVOKE_SUPER_RANGE:
1285 DoInvoke(self, mh, shadow_frame, dec_insn, kSuper, true, &result_register);
1286 break;
1287 case Instruction::INVOKE_DIRECT:
1288 DoInvoke(self, mh, shadow_frame, dec_insn, kDirect, false, &result_register);
1289 break;
1290 case Instruction::INVOKE_DIRECT_RANGE:
1291 DoInvoke(self, mh, shadow_frame, dec_insn, kDirect, true, &result_register);
1292 break;
1293 case Instruction::INVOKE_INTERFACE:
1294 DoInvoke(self, mh, shadow_frame, dec_insn, kInterface, false, &result_register);
1295 break;
1296 case Instruction::INVOKE_INTERFACE_RANGE:
1297 DoInvoke(self, mh, shadow_frame, dec_insn, kInterface, true, &result_register);
1298 break;
1299 case Instruction::INVOKE_STATIC:
1300 DoInvoke(self, mh, shadow_frame, dec_insn, kStatic, false, &result_register);
1301 break;
1302 case Instruction::INVOKE_STATIC_RANGE:
1303 DoInvoke(self, mh, shadow_frame, dec_insn, kStatic, true, &result_register);
1304 break;
1305 case Instruction::NEG_INT:
1306 shadow_frame.SetVReg(dec_insn.vA, -shadow_frame.GetVReg(dec_insn.vB));
1307 break;
1308 case Instruction::NOT_INT:
jeffhao42b4dc42012-12-10 10:25:43 -08001309 shadow_frame.SetVReg(dec_insn.vA, ~shadow_frame.GetVReg(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001310 break;
1311 case Instruction::NEG_LONG:
1312 shadow_frame.SetVRegLong(dec_insn.vA, -shadow_frame.GetVRegLong(dec_insn.vB));
1313 break;
1314 case Instruction::NOT_LONG:
jeffhao42b4dc42012-12-10 10:25:43 -08001315 shadow_frame.SetVRegLong(dec_insn.vA, ~shadow_frame.GetVRegLong(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001316 break;
1317 case Instruction::NEG_FLOAT:
1318 shadow_frame.SetVRegFloat(dec_insn.vA, -shadow_frame.GetVRegFloat(dec_insn.vB));
1319 break;
1320 case Instruction::NEG_DOUBLE:
1321 shadow_frame.SetVRegDouble(dec_insn.vA, -shadow_frame.GetVRegDouble(dec_insn.vB));
1322 break;
1323 case Instruction::INT_TO_LONG:
1324 shadow_frame.SetVRegLong(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
1325 break;
1326 case Instruction::INT_TO_FLOAT:
1327 shadow_frame.SetVRegFloat(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
1328 break;
1329 case Instruction::INT_TO_DOUBLE:
1330 shadow_frame.SetVRegDouble(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
1331 break;
1332 case Instruction::LONG_TO_INT:
1333 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
1334 break;
1335 case Instruction::LONG_TO_FLOAT:
1336 shadow_frame.SetVRegFloat(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
1337 break;
1338 case Instruction::LONG_TO_DOUBLE:
1339 shadow_frame.SetVRegDouble(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
1340 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001341 case Instruction::FLOAT_TO_INT: {
1342 float val = shadow_frame.GetVRegFloat(dec_insn.vB);
1343 if (val != val) {
1344 shadow_frame.SetVReg(dec_insn.vA, 0);
1345 } else if (val > static_cast<float>(kMaxInt)) {
1346 shadow_frame.SetVReg(dec_insn.vA, kMaxInt);
1347 } else if (val < static_cast<float>(kMinInt)) {
1348 shadow_frame.SetVReg(dec_insn.vA, kMinInt);
1349 } else {
1350 shadow_frame.SetVReg(dec_insn.vA, val);
1351 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001352 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001353 }
1354 case Instruction::FLOAT_TO_LONG: {
1355 float val = shadow_frame.GetVRegFloat(dec_insn.vB);
1356 if (val != val) {
1357 shadow_frame.SetVRegLong(dec_insn.vA, 0);
1358 } else if (val > static_cast<float>(kMaxLong)) {
1359 shadow_frame.SetVRegLong(dec_insn.vA, kMaxLong);
1360 } else if (val < static_cast<float>(kMinLong)) {
1361 shadow_frame.SetVRegLong(dec_insn.vA, kMinLong);
1362 } else {
1363 shadow_frame.SetVRegLong(dec_insn.vA, val);
1364 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001365 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001366 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001367 case Instruction::FLOAT_TO_DOUBLE:
1368 shadow_frame.SetVRegDouble(dec_insn.vA, shadow_frame.GetVRegFloat(dec_insn.vB));
1369 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001370 case Instruction::DOUBLE_TO_INT: {
1371 double val = shadow_frame.GetVRegDouble(dec_insn.vB);
1372 if (val != val) {
1373 shadow_frame.SetVReg(dec_insn.vA, 0);
1374 } else if (val > static_cast<double>(kMaxInt)) {
1375 shadow_frame.SetVReg(dec_insn.vA, kMaxInt);
1376 } else if (val < static_cast<double>(kMinInt)) {
1377 shadow_frame.SetVReg(dec_insn.vA, kMinInt);
1378 } else {
1379 shadow_frame.SetVReg(dec_insn.vA, val);
1380 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001381 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001382 }
1383 case Instruction::DOUBLE_TO_LONG: {
1384 double val = shadow_frame.GetVRegDouble(dec_insn.vB);
1385 if (val != val) {
1386 shadow_frame.SetVRegLong(dec_insn.vA, 0);
1387 } else if (val > static_cast<double>(kMaxLong)) {
1388 shadow_frame.SetVRegLong(dec_insn.vA, kMaxLong);
1389 } else if (val < static_cast<double>(kMinLong)) {
1390 shadow_frame.SetVRegLong(dec_insn.vA, kMinLong);
1391 } else {
1392 shadow_frame.SetVRegLong(dec_insn.vA, val);
1393 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001394 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001395 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001396 case Instruction::DOUBLE_TO_FLOAT:
1397 shadow_frame.SetVRegFloat(dec_insn.vA, shadow_frame.GetVRegDouble(dec_insn.vB));
1398 break;
1399 case Instruction::INT_TO_BYTE:
1400 shadow_frame.SetVReg(dec_insn.vA, static_cast<int8_t>(shadow_frame.GetVReg(dec_insn.vB)));
1401 break;
1402 case Instruction::INT_TO_CHAR:
1403 shadow_frame.SetVReg(dec_insn.vA, static_cast<uint16_t>(shadow_frame.GetVReg(dec_insn.vB)));
1404 break;
1405 case Instruction::INT_TO_SHORT:
1406 shadow_frame.SetVReg(dec_insn.vA, static_cast<int16_t>(shadow_frame.GetVReg(dec_insn.vB)));
1407 break;
1408 case Instruction::ADD_INT:
1409 shadow_frame.SetVReg(dec_insn.vA,
1410 shadow_frame.GetVReg(dec_insn.vB) + shadow_frame.GetVReg(dec_insn.vC));
1411 break;
1412 case Instruction::SUB_INT:
1413 shadow_frame.SetVReg(dec_insn.vA,
1414 shadow_frame.GetVReg(dec_insn.vB) - shadow_frame.GetVReg(dec_insn.vC));
1415 break;
1416 case Instruction::MUL_INT:
1417 shadow_frame.SetVReg(dec_insn.vA,
1418 shadow_frame.GetVReg(dec_insn.vB) * shadow_frame.GetVReg(dec_insn.vC));
1419 break;
1420 case Instruction::REM_INT:
jeffhaod91398c2012-11-20 17:17:33 -08001421 DoIntRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1422 shadow_frame.GetVReg(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001423 break;
1424 case Instruction::DIV_INT:
jeffhaod91398c2012-11-20 17:17:33 -08001425 DoIntDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1426 shadow_frame.GetVReg(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001427 break;
1428 case Instruction::SHL_INT:
jeffhao42b4dc42012-12-10 10:25:43 -08001429 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) <<
1430 (shadow_frame.GetVReg(dec_insn.vC) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001431 break;
1432 case Instruction::SHR_INT:
jeffhao42b4dc42012-12-10 10:25:43 -08001433 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) >>
1434 (shadow_frame.GetVReg(dec_insn.vC) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001435 break;
1436 case Instruction::USHR_INT:
1437 shadow_frame.SetVReg(dec_insn.vA,
1438 static_cast<uint32_t>(shadow_frame.GetVReg(dec_insn.vB)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001439 (shadow_frame.GetVReg(dec_insn.vC) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001440 break;
1441 case Instruction::AND_INT:
1442 shadow_frame.SetVReg(dec_insn.vA,
1443 shadow_frame.GetVReg(dec_insn.vB) & shadow_frame.GetVReg(dec_insn.vC));
1444 break;
1445 case Instruction::OR_INT:
1446 shadow_frame.SetVReg(dec_insn.vA,
1447 shadow_frame.GetVReg(dec_insn.vB) | shadow_frame.GetVReg(dec_insn.vC));
1448 break;
1449 case Instruction::XOR_INT:
1450 shadow_frame.SetVReg(dec_insn.vA,
1451 shadow_frame.GetVReg(dec_insn.vB) ^ shadow_frame.GetVReg(dec_insn.vC));
1452 break;
1453 case Instruction::ADD_LONG:
1454 shadow_frame.SetVRegLong(dec_insn.vA,
1455 shadow_frame.GetVRegLong(dec_insn.vB) +
1456 shadow_frame.GetVRegLong(dec_insn.vC));
1457 break;
1458 case Instruction::SUB_LONG:
1459 shadow_frame.SetVRegLong(dec_insn.vA,
1460 shadow_frame.GetVRegLong(dec_insn.vB) -
1461 shadow_frame.GetVRegLong(dec_insn.vC));
1462 break;
1463 case Instruction::MUL_LONG:
1464 shadow_frame.SetVRegLong(dec_insn.vA,
1465 shadow_frame.GetVRegLong(dec_insn.vB) *
1466 shadow_frame.GetVRegLong(dec_insn.vC));
1467 break;
1468 case Instruction::DIV_LONG:
jeffhaod91398c2012-11-20 17:17:33 -08001469 DoLongDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB),
1470 shadow_frame.GetVRegLong(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001471 break;
1472 case Instruction::REM_LONG:
jeffhaod91398c2012-11-20 17:17:33 -08001473 DoLongRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB),
1474 shadow_frame.GetVRegLong(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001475 break;
1476 case Instruction::AND_LONG:
1477 shadow_frame.SetVRegLong(dec_insn.vA,
1478 shadow_frame.GetVRegLong(dec_insn.vB) &
1479 shadow_frame.GetVRegLong(dec_insn.vC));
1480 break;
1481 case Instruction::OR_LONG:
1482 shadow_frame.SetVRegLong(dec_insn.vA,
1483 shadow_frame.GetVRegLong(dec_insn.vB) |
1484 shadow_frame.GetVRegLong(dec_insn.vC));
1485 break;
1486 case Instruction::XOR_LONG:
1487 shadow_frame.SetVRegLong(dec_insn.vA,
1488 shadow_frame.GetVRegLong(dec_insn.vB) ^
1489 shadow_frame.GetVRegLong(dec_insn.vC));
1490 break;
1491 case Instruction::SHL_LONG:
1492 shadow_frame.SetVRegLong(dec_insn.vA,
1493 shadow_frame.GetVRegLong(dec_insn.vB) <<
jeffhao42b4dc42012-12-10 10:25:43 -08001494 (shadow_frame.GetVReg(dec_insn.vC) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001495 break;
1496 case Instruction::SHR_LONG:
1497 shadow_frame.SetVRegLong(dec_insn.vA,
1498 shadow_frame.GetVRegLong(dec_insn.vB) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001499 (shadow_frame.GetVReg(dec_insn.vC) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001500 break;
1501 case Instruction::USHR_LONG:
1502 shadow_frame.SetVRegLong(dec_insn.vA,
1503 static_cast<uint64_t>(shadow_frame.GetVRegLong(dec_insn.vB)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001504 (shadow_frame.GetVReg(dec_insn.vC) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001505 break;
1506 case Instruction::ADD_FLOAT:
1507 shadow_frame.SetVRegFloat(dec_insn.vA,
1508 shadow_frame.GetVRegFloat(dec_insn.vB) +
1509 shadow_frame.GetVRegFloat(dec_insn.vC));
1510 break;
1511 case Instruction::SUB_FLOAT:
1512 shadow_frame.SetVRegFloat(dec_insn.vA,
1513 shadow_frame.GetVRegFloat(dec_insn.vB) -
1514 shadow_frame.GetVRegFloat(dec_insn.vC));
1515 break;
1516 case Instruction::MUL_FLOAT:
1517 shadow_frame.SetVRegFloat(dec_insn.vA,
1518 shadow_frame.GetVRegFloat(dec_insn.vB) *
1519 shadow_frame.GetVRegFloat(dec_insn.vC));
1520 break;
1521 case Instruction::DIV_FLOAT:
1522 shadow_frame.SetVRegFloat(dec_insn.vA,
1523 shadow_frame.GetVRegFloat(dec_insn.vB) /
1524 shadow_frame.GetVRegFloat(dec_insn.vC));
1525 break;
1526 case Instruction::REM_FLOAT:
1527 shadow_frame.SetVRegFloat(dec_insn.vA,
1528 fmodf(shadow_frame.GetVRegFloat(dec_insn.vB),
1529 shadow_frame.GetVRegFloat(dec_insn.vC)));
1530 break;
1531 case Instruction::ADD_DOUBLE:
1532 shadow_frame.SetVRegDouble(dec_insn.vA,
1533 shadow_frame.GetVRegDouble(dec_insn.vB) +
1534 shadow_frame.GetVRegDouble(dec_insn.vC));
1535 break;
1536 case Instruction::SUB_DOUBLE:
1537 shadow_frame.SetVRegDouble(dec_insn.vA,
1538 shadow_frame.GetVRegDouble(dec_insn.vB) -
1539 shadow_frame.GetVRegDouble(dec_insn.vC));
1540 break;
1541 case Instruction::MUL_DOUBLE:
1542 shadow_frame.SetVRegDouble(dec_insn.vA,
1543 shadow_frame.GetVRegDouble(dec_insn.vB) *
1544 shadow_frame.GetVRegDouble(dec_insn.vC));
1545 break;
1546 case Instruction::DIV_DOUBLE:
1547 shadow_frame.SetVRegDouble(dec_insn.vA,
1548 shadow_frame.GetVRegDouble(dec_insn.vB) /
1549 shadow_frame.GetVRegDouble(dec_insn.vC));
1550 break;
1551 case Instruction::REM_DOUBLE:
1552 shadow_frame.SetVRegDouble(dec_insn.vA,
1553 fmod(shadow_frame.GetVRegDouble(dec_insn.vB),
1554 shadow_frame.GetVRegDouble(dec_insn.vC)));
1555 break;
1556 case Instruction::ADD_INT_2ADDR:
1557 shadow_frame.SetVReg(dec_insn.vA,
1558 shadow_frame.GetVReg(dec_insn.vA) + shadow_frame.GetVReg(dec_insn.vB));
1559 break;
1560 case Instruction::SUB_INT_2ADDR:
1561 shadow_frame.SetVReg(dec_insn.vA,
1562 shadow_frame.GetVReg(dec_insn.vA) - shadow_frame.GetVReg(dec_insn.vB));
1563 break;
1564 case Instruction::MUL_INT_2ADDR:
1565 shadow_frame.SetVReg(dec_insn.vA,
1566 shadow_frame.GetVReg(dec_insn.vA) * shadow_frame.GetVReg(dec_insn.vB));
1567 break;
1568 case Instruction::REM_INT_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001569 DoIntRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA),
1570 shadow_frame.GetVReg(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001571 break;
1572 case Instruction::SHL_INT_2ADDR:
jeffhao42b4dc42012-12-10 10:25:43 -08001573 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA) <<
1574 (shadow_frame.GetVReg(dec_insn.vB) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001575 break;
1576 case Instruction::SHR_INT_2ADDR:
jeffhao42b4dc42012-12-10 10:25:43 -08001577 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA) >>
1578 (shadow_frame.GetVReg(dec_insn.vB) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001579 break;
1580 case Instruction::USHR_INT_2ADDR:
1581 shadow_frame.SetVReg(dec_insn.vA,
1582 static_cast<uint32_t>(shadow_frame.GetVReg(dec_insn.vA)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001583 (shadow_frame.GetVReg(dec_insn.vB) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001584 break;
1585 case Instruction::AND_INT_2ADDR:
1586 shadow_frame.SetVReg(dec_insn.vA,
1587 shadow_frame.GetVReg(dec_insn.vA) & shadow_frame.GetVReg(dec_insn.vB));
1588 break;
1589 case Instruction::OR_INT_2ADDR:
1590 shadow_frame.SetVReg(dec_insn.vA,
1591 shadow_frame.GetVReg(dec_insn.vA) | shadow_frame.GetVReg(dec_insn.vB));
1592 break;
1593 case Instruction::XOR_INT_2ADDR:
1594 shadow_frame.SetVReg(dec_insn.vA,
1595 shadow_frame.GetVReg(dec_insn.vA) ^ shadow_frame.GetVReg(dec_insn.vB));
1596 break;
1597 case Instruction::DIV_INT_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001598 DoIntDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA),
1599 shadow_frame.GetVReg(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001600 break;
1601 case Instruction::ADD_LONG_2ADDR:
1602 shadow_frame.SetVRegLong(dec_insn.vA,
1603 shadow_frame.GetVRegLong(dec_insn.vA) +
1604 shadow_frame.GetVRegLong(dec_insn.vB));
1605 break;
1606 case Instruction::SUB_LONG_2ADDR:
1607 shadow_frame.SetVRegLong(dec_insn.vA,
1608 shadow_frame.GetVRegLong(dec_insn.vA) -
1609 shadow_frame.GetVRegLong(dec_insn.vB));
1610 break;
1611 case Instruction::MUL_LONG_2ADDR:
1612 shadow_frame.SetVRegLong(dec_insn.vA,
Ian Rogers64b6d142012-10-29 16:34:15 -07001613 shadow_frame.GetVRegLong(dec_insn.vA) *
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001614 shadow_frame.GetVRegLong(dec_insn.vB));
1615 break;
1616 case Instruction::DIV_LONG_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001617 DoLongDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vA),
1618 shadow_frame.GetVRegLong(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001619 break;
1620 case Instruction::REM_LONG_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001621 DoLongRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vA),
1622 shadow_frame.GetVRegLong(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001623 break;
1624 case Instruction::AND_LONG_2ADDR:
1625 shadow_frame.SetVRegLong(dec_insn.vA,
1626 shadow_frame.GetVRegLong(dec_insn.vA) &
1627 shadow_frame.GetVRegLong(dec_insn.vB));
1628 break;
1629 case Instruction::OR_LONG_2ADDR:
1630 shadow_frame.SetVRegLong(dec_insn.vA,
1631 shadow_frame.GetVRegLong(dec_insn.vA) |
1632 shadow_frame.GetVRegLong(dec_insn.vB));
1633 break;
1634 case Instruction::XOR_LONG_2ADDR:
1635 shadow_frame.SetVRegLong(dec_insn.vA,
1636 shadow_frame.GetVRegLong(dec_insn.vA) ^
1637 shadow_frame.GetVRegLong(dec_insn.vB));
1638 break;
1639 case Instruction::SHL_LONG_2ADDR:
1640 shadow_frame.SetVRegLong(dec_insn.vA,
1641 shadow_frame.GetVRegLong(dec_insn.vA) <<
jeffhao42b4dc42012-12-10 10:25:43 -08001642 (shadow_frame.GetVReg(dec_insn.vB) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001643 break;
1644 case Instruction::SHR_LONG_2ADDR:
1645 shadow_frame.SetVRegLong(dec_insn.vA,
1646 shadow_frame.GetVRegLong(dec_insn.vA) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001647 (shadow_frame.GetVReg(dec_insn.vB) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001648 break;
1649 case Instruction::USHR_LONG_2ADDR:
1650 shadow_frame.SetVRegLong(dec_insn.vA,
1651 static_cast<uint64_t>(shadow_frame.GetVRegLong(dec_insn.vA)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001652 (shadow_frame.GetVReg(dec_insn.vB) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001653 break;
1654 case Instruction::ADD_FLOAT_2ADDR:
1655 shadow_frame.SetVRegFloat(dec_insn.vA,
1656 shadow_frame.GetVRegFloat(dec_insn.vA) +
1657 shadow_frame.GetVRegFloat(dec_insn.vB));
1658 break;
1659 case Instruction::SUB_FLOAT_2ADDR:
1660 shadow_frame.SetVRegFloat(dec_insn.vA,
1661 shadow_frame.GetVRegFloat(dec_insn.vA) -
1662 shadow_frame.GetVRegFloat(dec_insn.vB));
1663 break;
1664 case Instruction::MUL_FLOAT_2ADDR:
1665 shadow_frame.SetVRegFloat(dec_insn.vA,
1666 shadow_frame.GetVRegFloat(dec_insn.vA) *
1667 shadow_frame.GetVRegFloat(dec_insn.vB));
1668 break;
1669 case Instruction::DIV_FLOAT_2ADDR:
1670 shadow_frame.SetVRegFloat(dec_insn.vA,
1671 shadow_frame.GetVRegFloat(dec_insn.vA) /
1672 shadow_frame.GetVRegFloat(dec_insn.vB));
1673 break;
1674 case Instruction::REM_FLOAT_2ADDR:
1675 shadow_frame.SetVRegFloat(dec_insn.vA,
1676 fmodf(shadow_frame.GetVRegFloat(dec_insn.vA),
1677 shadow_frame.GetVRegFloat(dec_insn.vB)));
1678 break;
1679 case Instruction::ADD_DOUBLE_2ADDR:
1680 shadow_frame.SetVRegDouble(dec_insn.vA,
1681 shadow_frame.GetVRegDouble(dec_insn.vA) +
1682 shadow_frame.GetVRegDouble(dec_insn.vB));
1683 break;
1684 case Instruction::SUB_DOUBLE_2ADDR:
1685 shadow_frame.SetVRegDouble(dec_insn.vA,
1686 shadow_frame.GetVRegDouble(dec_insn.vA) -
1687 shadow_frame.GetVRegDouble(dec_insn.vB));
1688 break;
1689 case Instruction::MUL_DOUBLE_2ADDR:
1690 shadow_frame.SetVRegDouble(dec_insn.vA,
1691 shadow_frame.GetVRegDouble(dec_insn.vA) *
1692 shadow_frame.GetVRegDouble(dec_insn.vB));
1693 break;
1694 case Instruction::DIV_DOUBLE_2ADDR:
1695 shadow_frame.SetVRegDouble(dec_insn.vA,
1696 shadow_frame.GetVRegDouble(dec_insn.vA) /
1697 shadow_frame.GetVRegDouble(dec_insn.vB));
1698 break;
1699 case Instruction::REM_DOUBLE_2ADDR:
1700 shadow_frame.SetVRegDouble(dec_insn.vA,
1701 fmod(shadow_frame.GetVRegDouble(dec_insn.vA),
1702 shadow_frame.GetVRegDouble(dec_insn.vB)));
1703 break;
1704 case Instruction::ADD_INT_LIT16:
1705 case Instruction::ADD_INT_LIT8:
1706 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) + dec_insn.vC);
1707 break;
1708 case Instruction::RSUB_INT:
1709 case Instruction::RSUB_INT_LIT8:
1710 shadow_frame.SetVReg(dec_insn.vA, dec_insn.vC - shadow_frame.GetVReg(dec_insn.vB));
1711 break;
1712 case Instruction::MUL_INT_LIT16:
1713 case Instruction::MUL_INT_LIT8:
1714 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) * dec_insn.vC);
1715 break;
1716 case Instruction::DIV_INT_LIT16:
1717 case Instruction::DIV_INT_LIT8:
jeffhaod91398c2012-11-20 17:17:33 -08001718 DoIntDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1719 dec_insn.vC);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001720 break;
1721 case Instruction::REM_INT_LIT16:
1722 case Instruction::REM_INT_LIT8:
jeffhaod91398c2012-11-20 17:17:33 -08001723 DoIntRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1724 dec_insn.vC);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001725 break;
1726 case Instruction::AND_INT_LIT16:
1727 case Instruction::AND_INT_LIT8:
1728 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) & dec_insn.vC);
1729 break;
1730 case Instruction::OR_INT_LIT16:
1731 case Instruction::OR_INT_LIT8:
1732 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) | dec_insn.vC);
1733 break;
1734 case Instruction::XOR_INT_LIT16:
1735 case Instruction::XOR_INT_LIT8:
1736 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) ^ dec_insn.vC);
1737 break;
1738 case Instruction::SHL_INT_LIT8:
jeffhao42b4dc42012-12-10 10:25:43 -08001739 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) <<
1740 (dec_insn.vC & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001741 break;
1742 case Instruction::SHR_INT_LIT8:
jeffhao42b4dc42012-12-10 10:25:43 -08001743 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) >>
1744 (dec_insn.vC & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001745 break;
1746 case Instruction::USHR_INT_LIT8:
1747 shadow_frame.SetVReg(dec_insn.vA,
1748 static_cast<uint32_t>(shadow_frame.GetVReg(dec_insn.vB)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001749 (dec_insn.vC & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001750 break;
1751 default:
1752 LOG(FATAL) << "Unexpected instruction: " << inst->DumpString(&mh.GetDexFile());
1753 break;
1754 }
1755 if (UNLIKELY(self->IsExceptionPending())) {
1756 uint32_t found_dex_pc =
1757 shadow_frame.GetMethod()->FindCatchBlock(self->GetException()->GetClass(),
1758 inst->GetDexPc(insns));
1759 if (found_dex_pc == DexFile::kDexNoIndex) {
1760 JValue result;
1761 result.SetJ(0);
1762 return result; // Handler in caller.
1763 } else {
1764 next_inst = Instruction::At(insns + found_dex_pc);
1765 }
1766 }
1767 inst = next_inst;
1768 }
1769}
1770
1771void EnterInterpreterFromInvoke(Thread* self, AbstractMethod* method, Object* receiver,
1772 JValue* args, JValue* result) {
Ian Rogers64b6d142012-10-29 16:34:15 -07001773 DCHECK_EQ(self, Thread::Current());
jeffhaod7521322012-11-21 15:38:24 -08001774 if (__builtin_frame_address(0) < self->GetStackEnd()) {
1775 ThrowStackOverflowError(self);
1776 return;
1777 }
1778
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001779 MethodHelper mh(method);
1780 const DexFile::CodeItem* code_item = mh.GetCodeItem();
1781 uint16_t num_regs;
1782 uint16_t num_ins;
1783 if (code_item != NULL) {
1784 num_regs = code_item->registers_size_;
1785 num_ins = code_item->ins_size_;
jeffhao0a9bb732012-11-26 12:28:49 -08001786 } else if (method->IsAbstract()) {
1787 self->ThrowNewExceptionF("Ljava/lang/AbstractMethodError;", "abstract method \"%s\"",
1788 PrettyMethod(method).c_str());
1789 return;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001790 } else {
1791 DCHECK(method->IsNative());
1792 num_regs = num_ins = AbstractMethod::NumArgRegisters(mh.GetShorty());
1793 if (!method->IsStatic()) {
1794 num_regs++;
1795 num_ins++;
1796 }
1797 }
1798 // Set up shadow frame with matching number of reference slots to vregs.
1799 ShadowFrame* last_shadow_frame = self->GetManagedStack()->GetTopShadowFrame();
TDYa127ce4cc0d2012-11-18 16:59:53 -08001800 UniquePtr<ShadowFrame> shadow_frame(ShadowFrame::Create(num_regs,
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001801 (last_shadow_frame == NULL) ? NULL : last_shadow_frame->GetLink(),
1802 method, 0));
1803 self->PushShadowFrame(shadow_frame.get());
1804 size_t cur_reg = num_regs - num_ins;
1805 if (!method->IsStatic()) {
1806 CHECK(receiver != NULL);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001807 shadow_frame->SetVRegReference(cur_reg, receiver);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001808 ++cur_reg;
1809 } else if (!method->GetDeclaringClass()->IsInitializing()) {
jeffhao94d6df42012-11-26 16:02:12 -08001810 if (!Runtime::Current()->GetClassLinker()->EnsureInitialized(method->GetDeclaringClass(),
1811 true, true)) {
1812 DCHECK(Thread::Current()->IsExceptionPending());
1813 return;
1814 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001815 CHECK(method->GetDeclaringClass()->IsInitializing());
1816 }
Ian Rogers64b6d142012-10-29 16:34:15 -07001817 const char* shorty = mh.GetShorty();
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001818 size_t arg_pos = 0;
1819 for (; cur_reg < num_regs; ++cur_reg, ++arg_pos) {
1820 DCHECK_LT(arg_pos + 1, mh.GetShortyLength());
1821 switch (shorty[arg_pos + 1]) {
1822 case 'L': {
1823 Object* o = args[arg_pos].GetL();
TDYa127ce4cc0d2012-11-18 16:59:53 -08001824 shadow_frame->SetVRegReference(cur_reg, o);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001825 break;
1826 }
1827 case 'J': case 'D':
1828 shadow_frame->SetVRegLong(cur_reg, args[arg_pos].GetJ());
1829 cur_reg++;
1830 break;
1831 default:
1832 shadow_frame->SetVReg(cur_reg, args[arg_pos].GetI());
1833 break;
1834 }
1835 }
Ian Rogers64b6d142012-10-29 16:34:15 -07001836 if (LIKELY(!method->IsNative())) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001837 JValue r = Execute(self, mh, code_item, *shadow_frame.get());
1838 if (result != NULL) {
1839 *result = r;
1840 }
1841 } else {
Ian Rogers64b6d142012-10-29 16:34:15 -07001842 // We don't expect to be asked to interpret native code (which is entered via a JNI compiler
1843 // generated stub) except during testing and image writing.
1844 if (!Runtime::Current()->IsStarted()) {
1845 UnstartedRuntimeJni(self, method, receiver, args, result);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001846 } else {
Ian Rogers64b6d142012-10-29 16:34:15 -07001847 InterpreterJni(self, method, shorty, receiver, args, result);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001848 }
1849 }
1850 self->PopShadowFrame();
1851}
1852
1853} // namespace interpreter
1854} // namespace art