blob: 195549a3a04a94dec68e193c43e6d0de57c6c731 [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 Rogers2dd0e2c2013-01-24 12:42:14 -080022#include "class_linker-inl.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070023#include "common_throws.h"
jeffhao373c52f2012-11-20 16:11:52 -080024#include "debugger.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070025#include "dex_instruction.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080026#include "gc/card_table-inl.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070027#include "invoke_arg_array_builder.h"
Ian Rogers64b6d142012-10-29 16:34:15 -070028#include "nth_caller_visitor.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080029#include "mirror/class.h"
30#include "mirror/class-inl.h"
31#include "mirror/field-inl.h"
32#include "mirror/abstract_method.h"
33#include "mirror/abstract_method-inl.h"
34#include "mirror/object-inl.h"
35#include "mirror/object_array-inl.h"
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070036#include "object_utils.h"
37#include "runtime_support.h"
38#include "ScopedLocalRef.h"
39#include "scoped_thread_state_change.h"
40#include "thread.h"
41
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080042using namespace art::mirror;
43
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070044namespace art {
45namespace interpreter {
46
jeffhao87a6edd2012-11-20 18:05:18 -080047static const int32_t kMaxInt = std::numeric_limits<int32_t>::max();
48static const int32_t kMinInt = std::numeric_limits<int32_t>::min();
49static const int64_t kMaxLong = std::numeric_limits<int64_t>::max();
50static const int64_t kMinLong = std::numeric_limits<int64_t>::min();
51
jeffhao9b5aa6f2012-12-18 11:47:11 -080052static JDWP::FrameId throw_frame_id_ = 0;
53static AbstractMethod* throw_method_ = NULL;
54static uint32_t throw_dex_pc_ = 0;
55
Ian Rogers64b6d142012-10-29 16:34:15 -070056static void UnstartedRuntimeInvoke(Thread* self, AbstractMethod* target_method,
57 Object* receiver, JValue* args, JValue* result)
58 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
59 // In a runtime that's not started we intercept certain methods to avoid complicated dependency
60 // problems in core libraries.
61 std::string name(PrettyMethod(target_method));
62 if (name == "java.lang.Class java.lang.Class.forName(java.lang.String)") {
63 std::string descriptor(DotToDescriptor(args[0].GetL()->AsString()->ToModifiedUtf8().c_str()));
64 ClassLoader* class_loader = NULL; // shadow_frame.GetMethod()->GetDeclaringClass()->GetClassLoader();
65 Class* found = Runtime::Current()->GetClassLinker()->FindClass(descriptor.c_str(),
66 class_loader);
67 CHECK(found != NULL) << "Class.forName failed in un-started runtime for class: "
68 << PrettyDescriptor(descriptor);
69 result->SetL(found);
70 } else if (name == "java.lang.Object java.lang.Class.newInstance()") {
71 Class* klass = receiver->AsClass();
72 AbstractMethod* c = klass->FindDeclaredDirectMethod("<init>", "()V");
73 CHECK(c != NULL);
74 Object* obj = klass->AllocObject(self);
75 CHECK(obj != NULL);
76 EnterInterpreterFromInvoke(self, c, obj, NULL, NULL);
77 result->SetL(obj);
78 } else if (name == "java.lang.reflect.Field java.lang.Class.getDeclaredField(java.lang.String)") {
79 // Special managed code cut-out to allow field lookup in a un-started runtime that'd fail
80 // going the reflective Dex way.
81 Class* klass = receiver->AsClass();
82 String* name = args[0].GetL()->AsString();
83 Field* found = NULL;
84 FieldHelper fh;
85 ObjectArray<Field>* fields = klass->GetIFields();
86 for (int32_t i = 0; i < fields->GetLength() && found == NULL; ++i) {
87 Field* f = fields->Get(i);
88 fh.ChangeField(f);
89 if (name->Equals(fh.GetName())) {
90 found = f;
91 }
92 }
93 if (found == NULL) {
94 fields = klass->GetSFields();
95 for (int32_t i = 0; i < fields->GetLength() && found == NULL; ++i) {
96 Field* f = fields->Get(i);
97 fh.ChangeField(f);
98 if (name->Equals(fh.GetName())) {
99 found = f;
100 }
101 }
102 }
103 CHECK(found != NULL)
104 << "Failed to find field in Class.getDeclaredField in un-started runtime. name="
105 << name->ToModifiedUtf8() << " class=" << PrettyDescriptor(klass);
106 // TODO: getDeclaredField calls GetType once the field is found to ensure a
107 // NoClassDefFoundError is thrown if the field's type cannot be resolved.
108 result->SetL(found);
109 } else if (name == "void java.lang.System.arraycopy(java.lang.Object, int, java.lang.Object, int, int)") {
110 // Special case array copying without initializing System.
111 Class* ctype = args[0].GetL()->GetClass()->GetComponentType();
112 jint srcPos = args[1].GetI();
113 jint dstPos = args[3].GetI();
114 jint length = args[4].GetI();
115 if (!ctype->IsPrimitive()) {
116 ObjectArray<Object>* src = args[0].GetL()->AsObjectArray<Object>();
117 ObjectArray<Object>* dst = args[2].GetL()->AsObjectArray<Object>();
118 for (jint i = 0; i < length; ++i) {
119 dst->Set(dstPos + i, src->Get(srcPos + i));
120 }
121 } else if (ctype->IsPrimitiveChar()) {
122 CharArray* src = args[0].GetL()->AsCharArray();
123 CharArray* dst = args[2].GetL()->AsCharArray();
124 for (jint i = 0; i < length; ++i) {
125 dst->Set(dstPos + i, src->Get(srcPos + i));
126 }
127 } else if (ctype->IsPrimitiveInt()) {
128 IntArray* src = args[0].GetL()->AsIntArray();
129 IntArray* dst = args[2].GetL()->AsIntArray();
130 for (jint i = 0; i < length; ++i) {
131 dst->Set(dstPos + i, src->Get(srcPos + i));
132 }
133 } else {
134 UNIMPLEMENTED(FATAL) << "System.arraycopy of unexpected type: " << PrettyDescriptor(ctype);
135 }
136 } else {
137 // Not special, continue with regular interpreter execution.
138 EnterInterpreterFromInvoke(self, target_method, receiver, args, result);
139 }
140}
141
142// Hand select a number of methods to be run in a not yet started runtime without using JNI.
143static void UnstartedRuntimeJni(Thread* self, AbstractMethod* method,
144 Object* receiver, JValue* args, JValue* result)
145 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
146 std::string name(PrettyMethod(method));
147 if (name == "java.lang.ClassLoader dalvik.system.VMStack.getCallingClassLoader()") {
148 result->SetL(NULL);
149 } else if (name == "java.lang.Class dalvik.system.VMStack.getStackClass2()") {
Ian Rogers7a22fa62013-01-23 12:16:16 -0800150 NthCallerVisitor visitor(self, 3);
Ian Rogers64b6d142012-10-29 16:34:15 -0700151 visitor.WalkStack();
152 result->SetL(visitor.caller->GetDeclaringClass());
153 } else if (name == "double java.lang.Math.log(double)") {
154 result->SetD(log(args[0].GetD()));
155 } else if (name == "java.lang.String java.lang.Class.getNameNative()") {
156 result->SetL(receiver->AsClass()->ComputeName());
157 } else if (name == "int java.lang.Float.floatToRawIntBits(float)") {
158 result->SetI(args[0].GetI());
159 } else if (name == "float java.lang.Float.intBitsToFloat(int)") {
160 result->SetF(args[0].GetF());
161 } else if (name == "double java.lang.Math.exp(double)") {
162 result->SetD(exp(args[0].GetD()));
163 } else if (name == "java.lang.Object java.lang.Object.internalClone()") {
164 result->SetL(receiver->Clone(self));
165 } else if (name == "void java.lang.Object.notifyAll()") {
Ian Rogers05f30572013-02-20 12:13:11 -0800166 receiver->NotifyAll(self);
Ian Rogers64b6d142012-10-29 16:34:15 -0700167 } else if (name == "int java.lang.String.compareTo(java.lang.String)") {
168 String* rhs = args[0].GetL()->AsString();
169 CHECK(rhs != NULL);
170 result->SetI(receiver->AsString()->CompareTo(rhs));
171 } else if (name == "java.lang.String java.lang.String.intern()") {
172 result->SetL(receiver->AsString()->Intern());
173 } else if (name == "int java.lang.String.fastIndexOf(int, int)") {
174 result->SetI(receiver->AsString()->FastIndexOf(args[0].GetI(), args[1].GetI()));
175 } else if (name == "java.lang.Object java.lang.reflect.Array.createMultiArray(java.lang.Class, int[])") {
176 result->SetL(Array::CreateMultiArray(self, args[0].GetL()->AsClass(), args[1].GetL()->AsIntArray()));
177 } else if (name == "java.lang.Object java.lang.Throwable.nativeFillInStackTrace()") {
178 ScopedObjectAccessUnchecked soa(self);
179 result->SetL(soa.Decode<Object*>(self->CreateInternalStackTrace(soa)));
180 } else if (name == "boolean java.nio.ByteOrder.isLittleEndian()") {
181 result->SetJ(JNI_TRUE);
182 } else if (name == "boolean sun.misc.Unsafe.compareAndSwapInt(java.lang.Object, long, int, int)") {
183 Object* obj = args[0].GetL();
184 jlong offset = args[1].GetJ();
185 jint expectedValue = args[2].GetI();
186 jint newValue = args[3].GetI();
187 byte* raw_addr = reinterpret_cast<byte*>(obj) + offset;
188 volatile int32_t* address = reinterpret_cast<volatile int32_t*>(raw_addr);
189 // Note: android_atomic_release_cas() returns 0 on success, not failure.
190 int r = android_atomic_release_cas(expectedValue, newValue, address);
191 result->SetZ(r == 0);
192 } else if (name == "void sun.misc.Unsafe.putObject(java.lang.Object, long, java.lang.Object)") {
193 Object* obj = args[0].GetL();
194 Object* newValue = args[2].GetL();
195 obj->SetFieldObject(MemberOffset(args[1].GetJ()), newValue, false);
196 } else {
197 LOG(FATAL) << "Attempt to invoke native method in non-started runtime: " << name;
198 }
199}
200
201static void InterpreterJni(Thread* self, AbstractMethod* method, StringPiece shorty,
202 Object* receiver, JValue* args, JValue* result)
203 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
204 // TODO: The following enters JNI code using a typedef-ed function rather than the JNI compiler,
205 // it should be removed and JNI compiled stubs used instead.
206 ScopedObjectAccessUnchecked soa(self);
207 if (method->IsStatic()) {
208 if (shorty == "L") {
209 typedef jobject (fnptr)(JNIEnv*, jclass);
210 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
211 ScopedLocalRef<jclass> klass(soa.Env(),
212 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
Ian Rogers556d6372012-11-20 12:19:36 -0800213 jobject jresult;
214 {
215 ScopedThreadStateChange tsc(self, kNative);
216 jresult = fn(soa.Env(), klass.get());
217 }
218 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700219 } else if (shorty == "V") {
220 typedef void (fnptr)(JNIEnv*, jclass);
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 fn(soa.Env(), klass.get());
226 } else if (shorty == "Z") {
227 typedef jboolean (fnptr)(JNIEnv*, jclass);
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->SetZ(fn(soa.Env(), klass.get()));
233 } else if (shorty == "BI") {
234 typedef jbyte (fnptr)(JNIEnv*, jclass, jint);
235 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
236 ScopedLocalRef<jclass> klass(soa.Env(),
237 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
238 ScopedThreadStateChange tsc(self, kNative);
239 result->SetB(fn(soa.Env(), klass.get(), args[0].GetI()));
240 } else if (shorty == "II") {
241 typedef jint (fnptr)(JNIEnv*, jclass, jint);
242 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
243 ScopedLocalRef<jclass> klass(soa.Env(),
244 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
245 ScopedThreadStateChange tsc(self, kNative);
246 result->SetI(fn(soa.Env(), klass.get(), args[0].GetI()));
247 } else if (shorty == "LL") {
248 typedef jobject (fnptr)(JNIEnv*, jclass, jobject);
249 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
250 ScopedLocalRef<jclass> klass(soa.Env(),
251 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
252 ScopedLocalRef<jobject> arg0(soa.Env(),
253 soa.AddLocalReference<jobject>(args[0].GetL()));
Ian Rogers556d6372012-11-20 12:19:36 -0800254 jobject jresult;
255 {
256 ScopedThreadStateChange tsc(self, kNative);
257 jresult = fn(soa.Env(), klass.get(), arg0.get());
258 }
259 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700260 } else if (shorty == "IIZ") {
261 typedef jint (fnptr)(JNIEnv*, jclass, jint, jboolean);
262 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
263 ScopedLocalRef<jclass> klass(soa.Env(),
264 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
265 ScopedThreadStateChange tsc(self, kNative);
266 result->SetI(fn(soa.Env(), klass.get(), args[0].GetI(), args[1].GetZ()));
267 } else if (shorty == "ILI") {
268 typedef jint (fnptr)(JNIEnv*, jclass, jobject, jint);
269 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
270 ScopedLocalRef<jclass> klass(soa.Env(),
271 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
272 ScopedLocalRef<jobject> arg0(soa.Env(),
273 soa.AddLocalReference<jobject>(args[0].GetL()));
274 ScopedThreadStateChange tsc(self, kNative);
275 result->SetI(fn(soa.Env(), klass.get(), arg0.get(), args[1].GetI()));
276 } else if (shorty == "SIZ") {
277 typedef jshort (fnptr)(JNIEnv*, jclass, jint, jboolean);
278 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
279 ScopedLocalRef<jclass> klass(soa.Env(),
280 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
281 ScopedThreadStateChange tsc(self, kNative);
282 result->SetS(fn(soa.Env(), klass.get(), args[0].GetI(), args[1].GetZ()));
283 } else if (shorty == "VIZ") {
284 typedef void (fnptr)(JNIEnv*, jclass, jint, jboolean);
285 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
286 ScopedLocalRef<jclass> klass(soa.Env(),
287 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
288 ScopedThreadStateChange tsc(self, kNative);
289 fn(soa.Env(), klass.get(), args[0].GetI(), args[1].GetZ());
290 } else if (shorty == "ZLL") {
291 typedef jboolean (fnptr)(JNIEnv*, jclass, jobject, jobject);
292 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
293 ScopedLocalRef<jclass> klass(soa.Env(),
294 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
295 ScopedLocalRef<jobject> arg0(soa.Env(),
296 soa.AddLocalReference<jobject>(args[0].GetL()));
297 ScopedLocalRef<jobject> arg1(soa.Env(),
298 soa.AddLocalReference<jobject>(args[1].GetL()));
299 ScopedThreadStateChange tsc(self, kNative);
300 result->SetZ(fn(soa.Env(), klass.get(), arg0.get(), arg1.get()));
301 } else if (shorty == "ZILL") {
302 typedef jboolean (fnptr)(JNIEnv*, jclass, jint, jobject, jobject);
303 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
304 ScopedLocalRef<jclass> klass(soa.Env(),
305 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
306 ScopedLocalRef<jobject> arg1(soa.Env(),
307 soa.AddLocalReference<jobject>(args[1].GetL()));
308 ScopedLocalRef<jobject> arg2(soa.Env(),
309 soa.AddLocalReference<jobject>(args[2].GetL()));
310 ScopedThreadStateChange tsc(self, kNative);
311 result->SetZ(fn(soa.Env(), klass.get(), args[0].GetI(), arg1.get(), arg2.get()));
312 } else if (shorty == "VILII") {
313 typedef void (fnptr)(JNIEnv*, jclass, jint, jobject, jint, jint);
314 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
315 ScopedLocalRef<jclass> klass(soa.Env(),
316 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
317 ScopedLocalRef<jobject> arg1(soa.Env(),
318 soa.AddLocalReference<jobject>(args[1].GetL()));
319 ScopedThreadStateChange tsc(self, kNative);
320 fn(soa.Env(), klass.get(), args[0].GetI(), arg1.get(), args[2].GetI(), args[3].GetI());
321 } else if (shorty == "VLILII") {
322 typedef void (fnptr)(JNIEnv*, jclass, jobject, jint, jobject, jint, jint);
323 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
324 ScopedLocalRef<jclass> klass(soa.Env(),
325 soa.AddLocalReference<jclass>(method->GetDeclaringClass()));
326 ScopedLocalRef<jobject> arg0(soa.Env(),
327 soa.AddLocalReference<jobject>(args[0].GetL()));
328 ScopedLocalRef<jobject> arg2(soa.Env(),
329 soa.AddLocalReference<jobject>(args[2].GetL()));
330 ScopedThreadStateChange tsc(self, kNative);
331 fn(soa.Env(), klass.get(), arg0.get(), args[1].GetI(), arg2.get(), args[3].GetI(),
332 args[4].GetI());
333 } else {
334 LOG(FATAL) << "Do something with static native method: " << PrettyMethod(method)
335 << " shorty: " << shorty;
336 }
337 } else {
338 if (shorty == "L") {
339 typedef jobject (fnptr)(JNIEnv*, jobject);
340 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
341 ScopedLocalRef<jobject> rcvr(soa.Env(),
342 soa.AddLocalReference<jobject>(receiver));
Ian Rogers556d6372012-11-20 12:19:36 -0800343 jobject jresult;
344 {
345 ScopedThreadStateChange tsc(self, kNative);
346 jresult = fn(soa.Env(), rcvr.get());
347 }
348 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700349 } else if (shorty == "LL") {
350 typedef jobject (fnptr)(JNIEnv*, jobject, jobject);
351 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
352 ScopedLocalRef<jobject> rcvr(soa.Env(),
353 soa.AddLocalReference<jobject>(receiver));
354 ScopedLocalRef<jobject> arg0(soa.Env(),
355 soa.AddLocalReference<jobject>(args[0].GetL()));
Ian Rogers556d6372012-11-20 12:19:36 -0800356 jobject jresult;
357 {
358 ScopedThreadStateChange tsc(self, kNative);
359 jresult = fn(soa.Env(), rcvr.get(), arg0.get());
360
361 }
362 result->SetL(soa.Decode<Object*>(jresult));
Ian Rogers64b6d142012-10-29 16:34:15 -0700363 ScopedThreadStateChange tsc(self, kNative);
Ian Rogers64b6d142012-10-29 16:34:15 -0700364 } else if (shorty == "III") {
365 typedef jint (fnptr)(JNIEnv*, jobject, jint, jint);
366 fnptr* fn = reinterpret_cast<fnptr*>(method->GetNativeMethod());
367 ScopedLocalRef<jobject> rcvr(soa.Env(),
368 soa.AddLocalReference<jobject>(receiver));
369 ScopedThreadStateChange tsc(self, kNative);
370 result->SetI(fn(soa.Env(), rcvr.get(), args[0].GetI(), args[1].GetI()));
371 } else {
372 LOG(FATAL) << "Do something with native method: " << PrettyMethod(method)
373 << " shorty: " << shorty;
374 }
375 }
376}
377
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700378static void DoMonitorEnter(Thread* self, Object* ref) NO_THREAD_SAFETY_ANALYSIS {
379 ref->MonitorEnter(self);
380}
381
382static void DoMonitorExit(Thread* self, Object* ref) NO_THREAD_SAFETY_ANALYSIS {
383 ref->MonitorExit(self);
384}
385
386static void DoInvoke(Thread* self, MethodHelper& mh, ShadowFrame& shadow_frame,
387 const DecodedInstruction& dec_insn, InvokeType type, bool is_range,
388 JValue* result)
389 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
390 Object* receiver;
391 if (type == kStatic) {
392 receiver = NULL;
393 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800394 receiver = shadow_frame.GetVRegReference(dec_insn.vC);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700395 }
396 uint32_t method_idx = dec_insn.vB;
397 AbstractMethod* target_method = FindMethodFromCode(method_idx, receiver,
398 shadow_frame.GetMethod(), self, true,
399 type);
400 if (UNLIKELY(target_method == NULL)) {
401 CHECK(self->IsExceptionPending());
402 result->SetJ(0);
403 return;
404 }
405 mh.ChangeMethod(target_method);
406 ArgArray arg_array(mh.GetShorty(), mh.GetShortyLength());
407 if (is_range) {
408 arg_array.BuildArgArray(shadow_frame, dec_insn.vC + (type != kStatic ? 1 : 0));
409 } else {
410 arg_array.BuildArgArray(shadow_frame, dec_insn.arg + (type != kStatic ? 1 : 0));
411 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700412 if (LIKELY(Runtime::Current()->IsStarted())) {
413 target_method->Invoke(self, receiver, arg_array.get(), result);
414 } else {
415 UnstartedRuntimeInvoke(self, target_method, receiver, arg_array.get(), result);
416 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700417 mh.ChangeMethod(shadow_frame.GetMethod());
418}
419
420static void DoFieldGet(Thread* self, ShadowFrame& shadow_frame,
421 const DecodedInstruction& dec_insn, FindFieldType find_type,
422 Primitive::Type field_type)
423 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
424 bool is_static = (find_type == StaticObjectRead) || (find_type == StaticPrimitiveRead);
425 uint32_t field_idx = is_static ? dec_insn.vB : dec_insn.vC;
426 Field* f = FindFieldFromCode(field_idx, shadow_frame.GetMethod(), self,
427 find_type, Primitive::FieldSize(field_type));
428 if (LIKELY(f != NULL)) {
429 Object* obj;
430 if (is_static) {
431 obj = f->GetDeclaringClass();
432 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800433 obj = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700434 if (UNLIKELY(obj == NULL)) {
435 ThrowNullPointerExceptionForFieldAccess(f, true);
Ian Rogers689d9f02012-11-20 16:30:29 -0800436 return;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700437 }
438 }
439 switch (field_type) {
440 case Primitive::kPrimBoolean:
441 shadow_frame.SetVReg(dec_insn.vA, f->GetBoolean(obj));
442 break;
443 case Primitive::kPrimByte:
444 shadow_frame.SetVReg(dec_insn.vA, f->GetByte(obj));
445 break;
446 case Primitive::kPrimChar:
447 shadow_frame.SetVReg(dec_insn.vA, f->GetChar(obj));
448 break;
449 case Primitive::kPrimShort:
450 shadow_frame.SetVReg(dec_insn.vA, f->GetShort(obj));
451 break;
452 case Primitive::kPrimInt:
453 shadow_frame.SetVReg(dec_insn.vA, f->GetInt(obj));
454 break;
455 case Primitive::kPrimLong:
456 shadow_frame.SetVRegLong(dec_insn.vA, f->GetLong(obj));
457 break;
458 case Primitive::kPrimNot:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800459 shadow_frame.SetVRegReference(dec_insn.vA, f->GetObject(obj));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700460 break;
461 default:
462 LOG(FATAL) << "Unreachable: " << field_type;
463 }
464 }
465}
466
467static void DoFieldPut(Thread* self, ShadowFrame& shadow_frame,
468 const DecodedInstruction& dec_insn, FindFieldType find_type,
469 Primitive::Type field_type)
470 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
471 bool is_static = (find_type == StaticObjectWrite) || (find_type == StaticPrimitiveWrite);
472 uint32_t field_idx = is_static ? dec_insn.vB : dec_insn.vC;
473 Field* f = FindFieldFromCode(field_idx, shadow_frame.GetMethod(), self,
474 find_type, Primitive::FieldSize(field_type));
475 if (LIKELY(f != NULL)) {
476 Object* obj;
477 if (is_static) {
478 obj = f->GetDeclaringClass();
479 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800480 obj = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700481 if (UNLIKELY(obj == NULL)) {
482 ThrowNullPointerExceptionForFieldAccess(f, false);
Ian Rogers689d9f02012-11-20 16:30:29 -0800483 return;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700484 }
485 }
486 switch (field_type) {
487 case Primitive::kPrimBoolean:
488 f->SetBoolean(obj, shadow_frame.GetVReg(dec_insn.vA));
489 break;
490 case Primitive::kPrimByte:
491 f->SetByte(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700492 break;
493 case Primitive::kPrimChar:
494 f->SetChar(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700495 break;
496 case Primitive::kPrimShort:
497 f->SetShort(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700498 break;
499 case Primitive::kPrimInt:
500 f->SetInt(obj, shadow_frame.GetVReg(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700501 break;
502 case Primitive::kPrimLong:
503 f->SetLong(obj, shadow_frame.GetVRegLong(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700504 break;
505 case Primitive::kPrimNot:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800506 f->SetObj(obj, shadow_frame.GetVRegReference(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700507 break;
508 default:
509 LOG(FATAL) << "Unreachable: " << field_type;
510 }
511 }
512}
513
jeffhaod91398c2012-11-20 17:17:33 -0800514static void DoIntDivide(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
515 int32_t dividend, int32_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800516 if (UNLIKELY(divisor == 0)) {
517 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
518 } else if (UNLIKELY(dividend == kMinInt && divisor == -1)) {
519 shadow_frame.SetVReg(result_reg, kMinInt);
520 } else {
521 shadow_frame.SetVReg(result_reg, dividend / divisor);
522 }
523}
524
525static void DoIntRemainder(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
526 int32_t dividend, int32_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800527 if (UNLIKELY(divisor == 0)) {
528 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
529 } else if (UNLIKELY(dividend == kMinInt && divisor == -1)) {
530 shadow_frame.SetVReg(result_reg, 0);
531 } else {
532 shadow_frame.SetVReg(result_reg, dividend % divisor);
533 }
534}
535
536static void DoLongDivide(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
537 int64_t dividend, int64_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800538 if (UNLIKELY(divisor == 0)) {
539 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
540 } else if (UNLIKELY(dividend == kMinLong && divisor == -1)) {
541 shadow_frame.SetVRegLong(result_reg, kMinLong);
542 } else {
543 shadow_frame.SetVRegLong(result_reg, dividend / divisor);
544 }
545}
546
547static void DoLongRemainder(Thread* self, ShadowFrame& shadow_frame, size_t result_reg,
548 int64_t dividend, int64_t divisor) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
jeffhaod91398c2012-11-20 17:17:33 -0800549 if (UNLIKELY(divisor == 0)) {
550 self->ThrowNewException("Ljava/lang/ArithmeticException;", "divide by zero");
551 } else if (UNLIKELY(dividend == kMinLong && divisor == -1)) {
552 shadow_frame.SetVRegLong(result_reg, 0);
553 } else {
554 shadow_frame.SetVRegLong(result_reg, dividend % divisor);
555 }
556}
557
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700558static JValue Execute(Thread* self, MethodHelper& mh, const DexFile::CodeItem* code_item,
Ian Rogers306057f2012-11-26 12:45:53 -0800559 ShadowFrame& shadow_frame, JValue result_register)
560 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700561 const uint16_t* insns = code_item->insns_;
562 const Instruction* inst = Instruction::At(insns + shadow_frame.GetDexPC());
jeffhao14f0db92012-12-14 17:50:42 -0800563 bool entry = (inst->GetDexPc(insns) == 0);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700564 while (true) {
jeffhao373c52f2012-11-20 16:11:52 -0800565 CheckSuspend(self);
566 uint32_t dex_pc = inst->GetDexPc(insns);
567 shadow_frame.SetDexPC(dex_pc);
jeffhao14f0db92012-12-14 17:50:42 -0800568 if (entry) {
569 Dbg::UpdateDebugger(-1, self);
570 }
571 entry = false;
jeffhao373c52f2012-11-20 16:11:52 -0800572 Dbg::UpdateDebugger(dex_pc, self);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700573 DecodedInstruction dec_insn(inst);
Ian Rogers64b6d142012-10-29 16:34:15 -0700574 const bool kTracing = false;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700575 if (kTracing) {
576 LOG(INFO) << PrettyMethod(shadow_frame.GetMethod())
577 << StringPrintf("\n0x%x: %s\nReferences:",
578 inst->GetDexPc(insns), inst->DumpString(&mh.GetDexFile()).c_str());
TDYa127ce4cc0d2012-11-18 16:59:53 -0800579 for (size_t i = 0; i < shadow_frame.NumberOfVRegs(); ++i) {
580 Object* o = shadow_frame.GetVRegReference(i);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700581 if (o != NULL) {
582 if (o->GetClass()->IsStringClass() && o->AsString()->GetCharArray() != NULL) {
583 LOG(INFO) << i << ": java.lang.String " << static_cast<void*>(o)
584 << " \"" << o->AsString()->ToModifiedUtf8() << "\"";
585 } else {
586 LOG(INFO) << i << ": " << PrettyTypeOf(o) << " " << static_cast<void*>(o);
587 }
588 } else {
589 LOG(INFO) << i << ": null";
590 }
591 }
592 LOG(INFO) << "vregs:";
TDYa127ce4cc0d2012-11-18 16:59:53 -0800593 for (size_t i = 0; i < shadow_frame.NumberOfVRegs(); ++i) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700594 LOG(INFO) << StringPrintf("%d: %08x", i, shadow_frame.GetVReg(i));
595 }
596 }
597 const Instruction* next_inst = inst->Next();
598 switch (dec_insn.opcode) {
599 case Instruction::NOP:
600 break;
601 case Instruction::MOVE:
602 case Instruction::MOVE_FROM16:
603 case Instruction::MOVE_16:
604 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
605 break;
606 case Instruction::MOVE_WIDE:
607 case Instruction::MOVE_WIDE_FROM16:
608 case Instruction::MOVE_WIDE_16:
609 shadow_frame.SetVRegLong(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
610 break;
611 case Instruction::MOVE_OBJECT:
612 case Instruction::MOVE_OBJECT_FROM16:
613 case Instruction::MOVE_OBJECT_16:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800614 shadow_frame.SetVRegReference(dec_insn.vA, shadow_frame.GetVRegReference(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700615 break;
616 case Instruction::MOVE_RESULT:
617 shadow_frame.SetVReg(dec_insn.vA, result_register.GetI());
618 break;
619 case Instruction::MOVE_RESULT_WIDE:
620 shadow_frame.SetVRegLong(dec_insn.vA, result_register.GetJ());
621 break;
622 case Instruction::MOVE_RESULT_OBJECT:
TDYa127ce4cc0d2012-11-18 16:59:53 -0800623 shadow_frame.SetVRegReference(dec_insn.vA, result_register.GetL());
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700624 break;
625 case Instruction::MOVE_EXCEPTION: {
626 Throwable* exception = self->GetException();
627 self->ClearException();
TDYa127ce4cc0d2012-11-18 16:59:53 -0800628 shadow_frame.SetVRegReference(dec_insn.vA, exception);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700629 break;
630 }
631 case Instruction::RETURN_VOID: {
632 JValue result;
633 result.SetJ(0);
634 return result;
635 }
636 case Instruction::RETURN: {
637 JValue result;
638 result.SetJ(0);
639 result.SetI(shadow_frame.GetVReg(dec_insn.vA));
640 return result;
641 }
642 case Instruction::RETURN_WIDE: {
643 JValue result;
644 result.SetJ(shadow_frame.GetVRegLong(dec_insn.vA));
645 return result;
646 }
647 case Instruction::RETURN_OBJECT: {
648 JValue result;
649 result.SetJ(0);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800650 result.SetL(shadow_frame.GetVRegReference(dec_insn.vA));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700651 return result;
652 }
653 case Instruction::CONST_4: {
Ian Rogers64b6d142012-10-29 16:34:15 -0700654 int32_t val = static_cast<int32_t>(dec_insn.vB << 28) >> 28;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700655 shadow_frame.SetVReg(dec_insn.vA, val);
656 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800657 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700658 }
659 break;
660 }
661 case Instruction::CONST_16: {
662 int32_t val = static_cast<int16_t>(dec_insn.vB);
663 shadow_frame.SetVReg(dec_insn.vA, val);
664 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800665 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700666 }
667 break;
668 }
669 case Instruction::CONST: {
670 int32_t val = dec_insn.vB;
671 shadow_frame.SetVReg(dec_insn.vA, val);
672 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800673 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700674 }
675 break;
676 }
677 case Instruction::CONST_HIGH16: {
678 int32_t val = dec_insn.vB << 16;
679 shadow_frame.SetVReg(dec_insn.vA, val);
680 if (val == 0) {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800681 shadow_frame.SetVRegReference(dec_insn.vA, NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700682 }
683 break;
684 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700685 case Instruction::CONST_WIDE_16:
686 shadow_frame.SetVRegLong(dec_insn.vA, static_cast<int16_t>(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700687 break;
Ian Rogers64b6d142012-10-29 16:34:15 -0700688 case Instruction::CONST_WIDE_32:
689 shadow_frame.SetVRegLong(dec_insn.vA, static_cast<int32_t>(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700690 break;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700691 case Instruction::CONST_WIDE:
Ian Rogers64b6d142012-10-29 16:34:15 -0700692 shadow_frame.SetVRegLong(dec_insn.vA, dec_insn.vB_wide);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700693 break;
694 case Instruction::CONST_WIDE_HIGH16:
Ian Rogers64b6d142012-10-29 16:34:15 -0700695 shadow_frame.SetVRegLong(dec_insn.vA, static_cast<uint64_t>(dec_insn.vB) << 48);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700696 break;
697 case Instruction::CONST_STRING:
698 case Instruction::CONST_STRING_JUMBO: {
699 if (UNLIKELY(!String::GetJavaLangString()->IsInitialized())) {
700 Runtime::Current()->GetClassLinker()->EnsureInitialized(String::GetJavaLangString(),
701 true, true);
702 }
703 String* s = mh.ResolveString(dec_insn.vB);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800704 shadow_frame.SetVRegReference(dec_insn.vA, s);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700705 break;
706 }
jeffhao0a9bb732012-11-26 12:28:49 -0800707 case Instruction::CONST_CLASS: {
708 Class* c = ResolveVerifyAndClinit(dec_insn.vB, shadow_frame.GetMethod(), self, false, true);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800709 shadow_frame.SetVRegReference(dec_insn.vA, c);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700710 break;
jeffhao0a9bb732012-11-26 12:28:49 -0800711 }
jeffhao87a6edd2012-11-20 18:05:18 -0800712 case Instruction::MONITOR_ENTER: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800713 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao87a6edd2012-11-20 18:05:18 -0800714 if (UNLIKELY(obj == NULL)) {
715 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
716 } else {
717 DoMonitorEnter(self, obj);
718 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700719 break;
jeffhao87a6edd2012-11-20 18:05:18 -0800720 }
721 case Instruction::MONITOR_EXIT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800722 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao87a6edd2012-11-20 18:05:18 -0800723 if (UNLIKELY(obj == NULL)) {
724 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
725 } else {
726 DoMonitorExit(self, obj);
727 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700728 break;
jeffhao87a6edd2012-11-20 18:05:18 -0800729 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700730 case Instruction::CHECK_CAST: {
jeffhao0a9bb732012-11-26 12:28:49 -0800731 Class* c = ResolveVerifyAndClinit(dec_insn.vB, shadow_frame.GetMethod(), self, false, true);
jeffhao87a6edd2012-11-20 18:05:18 -0800732 if (UNLIKELY(c == NULL)) {
733 CHECK(self->IsExceptionPending());
734 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800735 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao87a6edd2012-11-20 18:05:18 -0800736 if (UNLIKELY(obj != NULL && !obj->InstanceOf(c))) {
737 self->ThrowNewExceptionF("Ljava/lang/ClassCastException;",
738 "%s cannot be cast to %s",
739 PrettyDescriptor(obj->GetClass()).c_str(),
740 PrettyDescriptor(c).c_str());
741 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700742 }
743 break;
744 }
745 case Instruction::INSTANCE_OF: {
jeffhao0a9bb732012-11-26 12:28:49 -0800746 Class* c = ResolveVerifyAndClinit(dec_insn.vC, shadow_frame.GetMethod(), self, false, true);
jeffhao87a6edd2012-11-20 18:05:18 -0800747 if (UNLIKELY(c == NULL)) {
748 CHECK(self->IsExceptionPending());
749 } else {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800750 Object* obj = shadow_frame.GetVRegReference(dec_insn.vB);
jeffhao87a6edd2012-11-20 18:05:18 -0800751 shadow_frame.SetVReg(dec_insn.vA, (obj != NULL && obj->InstanceOf(c)) ? 1 : 0);
752 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700753 break;
754 }
755 case Instruction::ARRAY_LENGTH: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800756 Object* array = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700757 if (UNLIKELY(array == NULL)) {
758 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
759 break;
760 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700761 shadow_frame.SetVReg(dec_insn.vA, array->AsArray()->GetLength());
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700762 break;
763 }
764 case Instruction::NEW_INSTANCE: {
765 Object* obj = AllocObjectFromCode(dec_insn.vB, shadow_frame.GetMethod(), self, true);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800766 shadow_frame.SetVRegReference(dec_insn.vA, obj);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700767 break;
768 }
769 case Instruction::NEW_ARRAY: {
770 int32_t length = shadow_frame.GetVReg(dec_insn.vB);
771 Object* obj = AllocArrayFromCode(dec_insn.vC, shadow_frame.GetMethod(), length, self, true);
TDYa127ce4cc0d2012-11-18 16:59:53 -0800772 shadow_frame.SetVRegReference(dec_insn.vA, obj);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700773 break;
774 }
775 case Instruction::FILLED_NEW_ARRAY:
Ian Rogers64b6d142012-10-29 16:34:15 -0700776 case Instruction::FILLED_NEW_ARRAY_RANGE: {
777 bool is_range = (dec_insn.opcode == Instruction::FILLED_NEW_ARRAY_RANGE);
778 int32_t length = dec_insn.vA;
779 CHECK(is_range || length <= 5);
jeffhao42b4dc42012-12-10 10:25:43 -0800780 if (UNLIKELY(length < 0)) {
781 self->ThrowNewExceptionF("Ljava/lang/NegativeArraySizeException;", "%d", length);
782 break;
783 }
jeffhao0a9bb732012-11-26 12:28:49 -0800784 Class* arrayClass = ResolveVerifyAndClinit(dec_insn.vB, shadow_frame.GetMethod(), self, false, true);
jeffhao42b4dc42012-12-10 10:25:43 -0800785 if (UNLIKELY(arrayClass == NULL)) {
786 CHECK(self->IsExceptionPending());
787 break;
788 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700789 CHECK(arrayClass->IsArrayClass());
jeffhao42b4dc42012-12-10 10:25:43 -0800790 Class* componentClass = arrayClass->GetComponentType();
791 if (UNLIKELY(componentClass->IsPrimitive() && !componentClass->IsPrimitiveInt())) {
792 if (componentClass->IsPrimitiveLong() || componentClass->IsPrimitiveDouble()) {
793 self->ThrowNewExceptionF("Ljava/lang/RuntimeException;",
794 "Bad filled array request for type %s",
795 PrettyDescriptor(componentClass).c_str());
796 } else {
797 self->ThrowNewExceptionF("Ljava/lang/InternalError;",
798 "Found type %s; filled-new-array not implemented for anything but \'int\'",
799 PrettyDescriptor(componentClass).c_str());
800 }
801 break;
802 }
803 Object* newArray = Array::Alloc(self, arrayClass, length);
804 if (newArray != NULL) {
805 for (int32_t i = 0; i < length; ++i) {
806 if (is_range) {
807 if (componentClass->IsPrimitiveInt()) {
808 newArray->AsIntArray()->Set(i, shadow_frame.GetVReg(dec_insn.vC + i));
Ian Rogers64b6d142012-10-29 16:34:15 -0700809 } else {
jeffhao42b4dc42012-12-10 10:25:43 -0800810 newArray->AsObjectArray<Object>()->Set(i, shadow_frame.GetVRegReference(dec_insn.vC + i));
811 }
812 } else {
813 if (componentClass->IsPrimitiveInt()) {
814 newArray->AsIntArray()->Set(i, shadow_frame.GetVReg(dec_insn.arg[i]));
815 } else {
816 newArray->AsObjectArray<Object>()->Set(i, shadow_frame.GetVRegReference(dec_insn.arg[i]));
Ian Rogers64b6d142012-10-29 16:34:15 -0700817 }
818 }
819 }
Ian Rogers64b6d142012-10-29 16:34:15 -0700820 }
jeffhao42b4dc42012-12-10 10:25:43 -0800821 result_register.SetL(newArray);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700822 break;
Ian Rogers64b6d142012-10-29 16:34:15 -0700823 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700824 case Instruction::CMPL_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::CMPG_FLOAT: {
839 float val1 = shadow_frame.GetVRegFloat(dec_insn.vB);
840 float val2 = shadow_frame.GetVRegFloat(dec_insn.vC);
841 int32_t result;
842 if (val1 == val2) {
843 result = 0;
844 } else if (val1 < val2) {
845 result = -1;
846 } else {
847 result = 1;
848 }
849 shadow_frame.SetVReg(dec_insn.vA, result);
850 break;
851 }
852 case Instruction::CMPL_DOUBLE: {
853 double val1 = shadow_frame.GetVRegDouble(dec_insn.vB);
854 double val2 = shadow_frame.GetVRegDouble(dec_insn.vC);
855 int32_t result;
856 if (val1 == val2) {
857 result = 0;
Ian Rogers58bf0c62012-11-20 16:24:12 -0800858 } else if (val1 > val2) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700859 result = 1;
860 } else {
861 result = -1;
862 }
863 shadow_frame.SetVReg(dec_insn.vA, result);
864 break;
865 }
866
867 case Instruction::CMPG_DOUBLE: {
868 double val1 = shadow_frame.GetVRegDouble(dec_insn.vB);
869 double val2 = shadow_frame.GetVRegDouble(dec_insn.vC);
870 int32_t result;
871 if (val1 == val2) {
872 result = 0;
Ian Rogers58bf0c62012-11-20 16:24:12 -0800873 } else if (val1 < val2) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700874 result = -1;
875 } else {
876 result = 1;
877 }
878 shadow_frame.SetVReg(dec_insn.vA, result);
879 break;
880 }
881 case Instruction::CMP_LONG: {
882 int64_t val1 = shadow_frame.GetVRegLong(dec_insn.vB);
883 int64_t val2 = shadow_frame.GetVRegLong(dec_insn.vC);
884 int32_t result;
jeffhao87a6edd2012-11-20 18:05:18 -0800885 if (val1 > val2) {
Ian Rogers64b6d142012-10-29 16:34:15 -0700886 result = 1;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700887 } else if (val1 == val2) {
888 result = 0;
889 } else {
Ian Rogers64b6d142012-10-29 16:34:15 -0700890 result = -1;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700891 }
892 shadow_frame.SetVReg(dec_insn.vA, result);
893 break;
894 }
895 case Instruction::THROW: {
Ian Rogers5f548e02013-01-16 22:12:55 -0800896 Object* o = shadow_frame.GetVRegReference(dec_insn.vA);
897 Throwable* t = (o == NULL) ? NULL : o->AsThrowable();
jeffhao94d6df42012-11-26 16:02:12 -0800898 self->DeliverException(t);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700899 break;
900 }
901 case Instruction::GOTO:
902 case Instruction::GOTO_16:
903 case Instruction::GOTO_32: {
904 uint32_t dex_pc = inst->GetDexPc(insns);
905 next_inst = Instruction::At(insns + dex_pc + dec_insn.vA);
906 break;
907 }
Ian Rogers556d6372012-11-20 12:19:36 -0800908 case Instruction::PACKED_SWITCH: {
909 uint32_t dex_pc = inst->GetDexPc(insns);
910 const uint16_t* switch_data = insns + dex_pc + dec_insn.vB;
911 int32_t test_val = shadow_frame.GetVReg(dec_insn.vA);
912 CHECK_EQ(switch_data[0], static_cast<uint16_t>(Instruction::kPackedSwitchSignature));
913 uint16_t size = switch_data[1];
914 CHECK_GT(size, 0);
915 const int32_t* keys = reinterpret_cast<const int32_t*>(&switch_data[2]);
916 CHECK(IsAligned<4>(keys));
917 int32_t first_key = keys[0];
918 const int32_t* targets = reinterpret_cast<const int32_t*>(&switch_data[4]);
919 CHECK(IsAligned<4>(targets));
920 int32_t index = test_val - first_key;
921 if (index >= 0 && index < size) {
922 next_inst = Instruction::At(insns + dex_pc + targets[index]);
923 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700924 break;
Ian Rogers556d6372012-11-20 12:19:36 -0800925 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700926 case Instruction::SPARSE_SWITCH: {
927 uint32_t dex_pc = inst->GetDexPc(insns);
Ian Rogers556d6372012-11-20 12:19:36 -0800928 const uint16_t* switch_data = insns + dex_pc + dec_insn.vB;
929 int32_t test_val = shadow_frame.GetVReg(dec_insn.vA);
930 CHECK_EQ(switch_data[0], static_cast<uint16_t>(Instruction::kSparseSwitchSignature));
931 uint16_t size = switch_data[1];
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700932 CHECK_GT(size, 0);
Ian Rogers556d6372012-11-20 12:19:36 -0800933 const int32_t* keys = reinterpret_cast<const int32_t*>(&switch_data[2]);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700934 CHECK(IsAligned<4>(keys));
935 const int32_t* entries = keys + size;
936 CHECK(IsAligned<4>(entries));
937 int lo = 0;
938 int hi = size - 1;
939 while (lo <= hi) {
940 int mid = (lo + hi) / 2;
941 int32_t foundVal = keys[mid];
Ian Rogers556d6372012-11-20 12:19:36 -0800942 if (test_val < foundVal) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700943 hi = mid - 1;
Ian Rogers556d6372012-11-20 12:19:36 -0800944 } else if (test_val > foundVal) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700945 lo = mid + 1;
946 } else {
947 next_inst = Instruction::At(insns + dex_pc + entries[mid]);
948 break;
949 }
950 }
951 break;
952 }
953 case Instruction::FILL_ARRAY_DATA: {
TDYa127ce4cc0d2012-11-18 16:59:53 -0800954 Object* obj = shadow_frame.GetVRegReference(dec_insn.vA);
jeffhao94d6df42012-11-26 16:02:12 -0800955 if (UNLIKELY(obj == NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700956 Thread::Current()->ThrowNewExceptionF("Ljava/lang/NullPointerException;",
957 "null array in FILL_ARRAY_DATA");
958 break;
959 }
jeffhao94d6df42012-11-26 16:02:12 -0800960 Array* array = obj->AsArray();
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700961 DCHECK(array->IsArrayInstance() && !array->IsObjectArray());
962 uint32_t dex_pc = inst->GetDexPc(insns);
963 const Instruction::ArrayDataPayload* payload =
964 reinterpret_cast<const Instruction::ArrayDataPayload*>(insns + dex_pc + dec_insn.vB);
965 if (UNLIKELY(static_cast<int32_t>(payload->element_count) > array->GetLength())) {
966 Thread::Current()->ThrowNewExceptionF("Ljava/lang/ArrayIndexOutOfBoundsException;",
967 "failed FILL_ARRAY_DATA; length=%d, index=%d",
968 array->GetLength(), payload->element_count);
969 break;
970 }
971 uint32_t size_in_bytes = payload->element_count * payload->element_width;
972 memcpy(array->GetRawData(payload->element_width), payload->data, size_in_bytes);
973 break;
974 }
975 case Instruction::IF_EQ: {
976 if (shadow_frame.GetVReg(dec_insn.vA) == shadow_frame.GetVReg(dec_insn.vB)) {
977 uint32_t dex_pc = inst->GetDexPc(insns);
978 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
979 }
980 break;
981 }
982 case Instruction::IF_NE: {
983 if (shadow_frame.GetVReg(dec_insn.vA) != shadow_frame.GetVReg(dec_insn.vB)) {
984 uint32_t dex_pc = inst->GetDexPc(insns);
985 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
986 }
987 break;
988 }
989 case Instruction::IF_LT: {
990 if (shadow_frame.GetVReg(dec_insn.vA) < shadow_frame.GetVReg(dec_insn.vB)) {
991 uint32_t dex_pc = inst->GetDexPc(insns);
992 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
993 }
994 break;
995 }
996 case Instruction::IF_GE: {
997 if (shadow_frame.GetVReg(dec_insn.vA) >= shadow_frame.GetVReg(dec_insn.vB)) {
998 uint32_t dex_pc = inst->GetDexPc(insns);
999 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
1000 }
1001 break;
1002 }
1003 case Instruction::IF_GT: {
1004 if (shadow_frame.GetVReg(dec_insn.vA) > shadow_frame.GetVReg(dec_insn.vB)) {
1005 uint32_t dex_pc = inst->GetDexPc(insns);
1006 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
1007 }
1008 break;
1009 }
1010 case Instruction::IF_LE: {
1011 if (shadow_frame.GetVReg(dec_insn.vA) <= shadow_frame.GetVReg(dec_insn.vB)) {
1012 uint32_t dex_pc = inst->GetDexPc(insns);
1013 next_inst = Instruction::At(insns + dex_pc + dec_insn.vC);
1014 }
1015 break;
1016 }
1017 case Instruction::IF_EQZ: {
1018 if (shadow_frame.GetVReg(dec_insn.vA) == 0) {
1019 uint32_t dex_pc = inst->GetDexPc(insns);
1020 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1021 }
1022 break;
1023 }
1024 case Instruction::IF_NEZ: {
1025 if (shadow_frame.GetVReg(dec_insn.vA) != 0) {
1026 uint32_t dex_pc = inst->GetDexPc(insns);
1027 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1028 }
1029 break;
1030 }
1031 case Instruction::IF_LTZ: {
1032 if (shadow_frame.GetVReg(dec_insn.vA) < 0) {
1033 uint32_t dex_pc = inst->GetDexPc(insns);
1034 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1035 }
1036 break;
1037 }
1038 case Instruction::IF_GEZ: {
1039 if (shadow_frame.GetVReg(dec_insn.vA) >= 0) {
1040 uint32_t dex_pc = inst->GetDexPc(insns);
1041 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1042 }
1043 break;
1044 }
1045 case Instruction::IF_GTZ: {
1046 if (shadow_frame.GetVReg(dec_insn.vA) > 0) {
1047 uint32_t dex_pc = inst->GetDexPc(insns);
1048 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1049 }
1050 break;
1051 }
1052 case Instruction::IF_LEZ: {
1053 if (shadow_frame.GetVReg(dec_insn.vA) <= 0) {
1054 uint32_t dex_pc = inst->GetDexPc(insns);
1055 next_inst = Instruction::At(insns + dex_pc + dec_insn.vB);
1056 }
1057 break;
1058 }
1059 case Instruction::AGET_BOOLEAN: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001060 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001061 if (UNLIKELY(a == NULL)) {
1062 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1063 break;
1064 }
1065 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001066 shadow_frame.SetVReg(dec_insn.vA, a->AsBooleanArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001067 break;
1068 }
1069 case Instruction::AGET_BYTE: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001070 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001071 if (UNLIKELY(a == NULL)) {
1072 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1073 break;
1074 }
1075 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001076 shadow_frame.SetVReg(dec_insn.vA, a->AsByteArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001077 break;
1078 }
1079 case Instruction::AGET_CHAR: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001080 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001081 if (UNLIKELY(a == NULL)) {
1082 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1083 break;
1084 }
1085 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001086 shadow_frame.SetVReg(dec_insn.vA, a->AsCharArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001087 break;
1088 }
1089 case Instruction::AGET_SHORT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001090 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001091 if (UNLIKELY(a == NULL)) {
1092 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1093 break;
1094 }
1095 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001096 shadow_frame.SetVReg(dec_insn.vA, a->AsShortArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001097 break;
1098 }
1099 case Instruction::AGET: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001100 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001101 if (UNLIKELY(a == NULL)) {
1102 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1103 break;
1104 }
1105 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001106 shadow_frame.SetVReg(dec_insn.vA, a->AsIntArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001107 break;
1108 }
1109 case Instruction::AGET_WIDE: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001110 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001111 if (UNLIKELY(a == NULL)) {
1112 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1113 break;
1114 }
1115 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001116 shadow_frame.SetVRegLong(dec_insn.vA, a->AsLongArray()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001117 break;
1118 }
1119 case Instruction::AGET_OBJECT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001120 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001121 if (UNLIKELY(a == NULL)) {
1122 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1123 break;
1124 }
1125 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001126 shadow_frame.SetVRegReference(dec_insn.vA, a->AsObjectArray<Object>()->Get(index));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001127 break;
1128 }
1129 case Instruction::APUT_BOOLEAN: {
1130 uint8_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001131 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001132 if (UNLIKELY(a == NULL)) {
1133 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1134 break;
1135 }
1136 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001137 a->AsBooleanArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001138 break;
1139 }
1140 case Instruction::APUT_BYTE: {
1141 int8_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001142 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001143 if (UNLIKELY(a == NULL)) {
1144 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1145 break;
1146 }
1147 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001148 a->AsByteArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001149 break;
1150 }
1151 case Instruction::APUT_CHAR: {
1152 uint16_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001153 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001154 if (UNLIKELY(a == NULL)) {
1155 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1156 break;
1157 }
1158 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001159 a->AsCharArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001160 break;
1161 }
1162 case Instruction::APUT_SHORT: {
1163 int16_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001164 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001165 if (UNLIKELY(a == NULL)) {
1166 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1167 break;
1168 }
1169 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001170 a->AsShortArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001171 break;
1172 }
1173 case Instruction::APUT: {
1174 int32_t val = shadow_frame.GetVReg(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001175 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001176 if (UNLIKELY(a == NULL)) {
1177 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1178 break;
1179 }
1180 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001181 a->AsIntArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001182 break;
1183 }
1184 case Instruction::APUT_WIDE: {
1185 int64_t val = shadow_frame.GetVRegLong(dec_insn.vA);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001186 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001187 if (UNLIKELY(a == NULL)) {
1188 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1189 break;
1190 }
1191 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001192 a->AsLongArray()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001193 break;
1194 }
1195 case Instruction::APUT_OBJECT: {
TDYa127ce4cc0d2012-11-18 16:59:53 -08001196 Object* val = shadow_frame.GetVRegReference(dec_insn.vA);
1197 Object* a = shadow_frame.GetVRegReference(dec_insn.vB);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001198 if (UNLIKELY(a == NULL)) {
1199 ThrowNullPointerExceptionFromDexPC(shadow_frame.GetMethod(), inst->GetDexPc(insns));
1200 break;
1201 }
1202 int32_t index = shadow_frame.GetVReg(dec_insn.vC);
jeffhao94d6df42012-11-26 16:02:12 -08001203 a->AsObjectArray<Object>()->Set(index, val);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001204 break;
1205 }
1206 case Instruction::IGET_BOOLEAN:
1207 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimBoolean);
1208 break;
1209 case Instruction::IGET_BYTE:
1210 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimByte);
1211 break;
1212 case Instruction::IGET_CHAR:
1213 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimChar);
1214 break;
1215 case Instruction::IGET_SHORT:
1216 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimShort);
1217 break;
1218 case Instruction::IGET:
1219 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimInt);
1220 break;
1221 case Instruction::IGET_WIDE:
1222 DoFieldGet(self, shadow_frame, dec_insn, InstancePrimitiveRead, Primitive::kPrimLong);
1223 break;
1224 case Instruction::IGET_OBJECT:
1225 DoFieldGet(self, shadow_frame, dec_insn, InstanceObjectRead, Primitive::kPrimNot);
1226 break;
1227 case Instruction::SGET_BOOLEAN:
1228 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimBoolean);
1229 break;
1230 case Instruction::SGET_BYTE:
1231 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimByte);
1232 break;
1233 case Instruction::SGET_CHAR:
1234 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimChar);
1235 break;
1236 case Instruction::SGET_SHORT:
1237 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimShort);
1238 break;
1239 case Instruction::SGET:
1240 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimInt);
1241 break;
1242 case Instruction::SGET_WIDE:
1243 DoFieldGet(self, shadow_frame, dec_insn, StaticPrimitiveRead, Primitive::kPrimLong);
1244 break;
1245 case Instruction::SGET_OBJECT:
1246 DoFieldGet(self, shadow_frame, dec_insn, StaticObjectRead, Primitive::kPrimNot);
1247 break;
1248 case Instruction::IPUT_BOOLEAN:
1249 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimBoolean);
1250 break;
1251 case Instruction::IPUT_BYTE:
1252 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimByte);
1253 break;
1254 case Instruction::IPUT_CHAR:
1255 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimChar);
1256 break;
1257 case Instruction::IPUT_SHORT:
1258 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimShort);
1259 break;
1260 case Instruction::IPUT:
1261 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimInt);
1262 break;
1263 case Instruction::IPUT_WIDE:
1264 DoFieldPut(self, shadow_frame, dec_insn, InstancePrimitiveWrite, Primitive::kPrimLong);
1265 break;
1266 case Instruction::IPUT_OBJECT:
1267 DoFieldPut(self, shadow_frame, dec_insn, InstanceObjectWrite, Primitive::kPrimNot);
1268 break;
1269 case Instruction::SPUT_BOOLEAN:
1270 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimBoolean);
1271 break;
1272 case Instruction::SPUT_BYTE:
1273 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimByte);
1274 break;
1275 case Instruction::SPUT_CHAR:
1276 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimChar);
1277 break;
1278 case Instruction::SPUT_SHORT:
1279 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimShort);
1280 break;
1281 case Instruction::SPUT:
1282 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimInt);
1283 break;
1284 case Instruction::SPUT_WIDE:
1285 DoFieldPut(self, shadow_frame, dec_insn, StaticPrimitiveWrite, Primitive::kPrimLong);
1286 break;
1287 case Instruction::SPUT_OBJECT:
1288 DoFieldPut(self, shadow_frame, dec_insn, StaticObjectWrite, Primitive::kPrimNot);
1289 break;
1290 case Instruction::INVOKE_VIRTUAL:
1291 DoInvoke(self, mh, shadow_frame, dec_insn, kVirtual, false, &result_register);
1292 break;
1293 case Instruction::INVOKE_VIRTUAL_RANGE:
1294 DoInvoke(self, mh, shadow_frame, dec_insn, kVirtual, true, &result_register);
1295 break;
1296 case Instruction::INVOKE_SUPER:
1297 DoInvoke(self, mh, shadow_frame, dec_insn, kSuper, false, &result_register);
1298 break;
1299 case Instruction::INVOKE_SUPER_RANGE:
1300 DoInvoke(self, mh, shadow_frame, dec_insn, kSuper, true, &result_register);
1301 break;
1302 case Instruction::INVOKE_DIRECT:
1303 DoInvoke(self, mh, shadow_frame, dec_insn, kDirect, false, &result_register);
1304 break;
1305 case Instruction::INVOKE_DIRECT_RANGE:
1306 DoInvoke(self, mh, shadow_frame, dec_insn, kDirect, true, &result_register);
1307 break;
1308 case Instruction::INVOKE_INTERFACE:
1309 DoInvoke(self, mh, shadow_frame, dec_insn, kInterface, false, &result_register);
1310 break;
1311 case Instruction::INVOKE_INTERFACE_RANGE:
1312 DoInvoke(self, mh, shadow_frame, dec_insn, kInterface, true, &result_register);
1313 break;
1314 case Instruction::INVOKE_STATIC:
1315 DoInvoke(self, mh, shadow_frame, dec_insn, kStatic, false, &result_register);
1316 break;
1317 case Instruction::INVOKE_STATIC_RANGE:
1318 DoInvoke(self, mh, shadow_frame, dec_insn, kStatic, true, &result_register);
1319 break;
1320 case Instruction::NEG_INT:
1321 shadow_frame.SetVReg(dec_insn.vA, -shadow_frame.GetVReg(dec_insn.vB));
1322 break;
1323 case Instruction::NOT_INT:
jeffhao42b4dc42012-12-10 10:25:43 -08001324 shadow_frame.SetVReg(dec_insn.vA, ~shadow_frame.GetVReg(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001325 break;
1326 case Instruction::NEG_LONG:
1327 shadow_frame.SetVRegLong(dec_insn.vA, -shadow_frame.GetVRegLong(dec_insn.vB));
1328 break;
1329 case Instruction::NOT_LONG:
jeffhao42b4dc42012-12-10 10:25:43 -08001330 shadow_frame.SetVRegLong(dec_insn.vA, ~shadow_frame.GetVRegLong(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001331 break;
1332 case Instruction::NEG_FLOAT:
1333 shadow_frame.SetVRegFloat(dec_insn.vA, -shadow_frame.GetVRegFloat(dec_insn.vB));
1334 break;
1335 case Instruction::NEG_DOUBLE:
1336 shadow_frame.SetVRegDouble(dec_insn.vA, -shadow_frame.GetVRegDouble(dec_insn.vB));
1337 break;
1338 case Instruction::INT_TO_LONG:
1339 shadow_frame.SetVRegLong(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
1340 break;
1341 case Instruction::INT_TO_FLOAT:
1342 shadow_frame.SetVRegFloat(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
1343 break;
1344 case Instruction::INT_TO_DOUBLE:
1345 shadow_frame.SetVRegDouble(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB));
1346 break;
1347 case Instruction::LONG_TO_INT:
1348 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
1349 break;
1350 case Instruction::LONG_TO_FLOAT:
1351 shadow_frame.SetVRegFloat(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
1352 break;
1353 case Instruction::LONG_TO_DOUBLE:
1354 shadow_frame.SetVRegDouble(dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB));
1355 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001356 case Instruction::FLOAT_TO_INT: {
1357 float val = shadow_frame.GetVRegFloat(dec_insn.vB);
1358 if (val != val) {
1359 shadow_frame.SetVReg(dec_insn.vA, 0);
1360 } else if (val > static_cast<float>(kMaxInt)) {
1361 shadow_frame.SetVReg(dec_insn.vA, kMaxInt);
1362 } else if (val < static_cast<float>(kMinInt)) {
1363 shadow_frame.SetVReg(dec_insn.vA, kMinInt);
1364 } else {
1365 shadow_frame.SetVReg(dec_insn.vA, val);
1366 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001367 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001368 }
1369 case Instruction::FLOAT_TO_LONG: {
1370 float val = shadow_frame.GetVRegFloat(dec_insn.vB);
1371 if (val != val) {
1372 shadow_frame.SetVRegLong(dec_insn.vA, 0);
1373 } else if (val > static_cast<float>(kMaxLong)) {
1374 shadow_frame.SetVRegLong(dec_insn.vA, kMaxLong);
1375 } else if (val < static_cast<float>(kMinLong)) {
1376 shadow_frame.SetVRegLong(dec_insn.vA, kMinLong);
1377 } else {
1378 shadow_frame.SetVRegLong(dec_insn.vA, val);
1379 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001380 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001381 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001382 case Instruction::FLOAT_TO_DOUBLE:
1383 shadow_frame.SetVRegDouble(dec_insn.vA, shadow_frame.GetVRegFloat(dec_insn.vB));
1384 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001385 case Instruction::DOUBLE_TO_INT: {
1386 double val = shadow_frame.GetVRegDouble(dec_insn.vB);
1387 if (val != val) {
1388 shadow_frame.SetVReg(dec_insn.vA, 0);
1389 } else if (val > static_cast<double>(kMaxInt)) {
1390 shadow_frame.SetVReg(dec_insn.vA, kMaxInt);
1391 } else if (val < static_cast<double>(kMinInt)) {
1392 shadow_frame.SetVReg(dec_insn.vA, kMinInt);
1393 } else {
1394 shadow_frame.SetVReg(dec_insn.vA, val);
1395 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001396 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001397 }
1398 case Instruction::DOUBLE_TO_LONG: {
1399 double val = shadow_frame.GetVRegDouble(dec_insn.vB);
1400 if (val != val) {
1401 shadow_frame.SetVRegLong(dec_insn.vA, 0);
1402 } else if (val > static_cast<double>(kMaxLong)) {
1403 shadow_frame.SetVRegLong(dec_insn.vA, kMaxLong);
1404 } else if (val < static_cast<double>(kMinLong)) {
1405 shadow_frame.SetVRegLong(dec_insn.vA, kMinLong);
1406 } else {
1407 shadow_frame.SetVRegLong(dec_insn.vA, val);
1408 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001409 break;
jeffhao87a6edd2012-11-20 18:05:18 -08001410 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001411 case Instruction::DOUBLE_TO_FLOAT:
1412 shadow_frame.SetVRegFloat(dec_insn.vA, shadow_frame.GetVRegDouble(dec_insn.vB));
1413 break;
1414 case Instruction::INT_TO_BYTE:
1415 shadow_frame.SetVReg(dec_insn.vA, static_cast<int8_t>(shadow_frame.GetVReg(dec_insn.vB)));
1416 break;
1417 case Instruction::INT_TO_CHAR:
1418 shadow_frame.SetVReg(dec_insn.vA, static_cast<uint16_t>(shadow_frame.GetVReg(dec_insn.vB)));
1419 break;
1420 case Instruction::INT_TO_SHORT:
1421 shadow_frame.SetVReg(dec_insn.vA, static_cast<int16_t>(shadow_frame.GetVReg(dec_insn.vB)));
1422 break;
1423 case Instruction::ADD_INT:
1424 shadow_frame.SetVReg(dec_insn.vA,
1425 shadow_frame.GetVReg(dec_insn.vB) + shadow_frame.GetVReg(dec_insn.vC));
1426 break;
1427 case Instruction::SUB_INT:
1428 shadow_frame.SetVReg(dec_insn.vA,
1429 shadow_frame.GetVReg(dec_insn.vB) - shadow_frame.GetVReg(dec_insn.vC));
1430 break;
1431 case Instruction::MUL_INT:
1432 shadow_frame.SetVReg(dec_insn.vA,
1433 shadow_frame.GetVReg(dec_insn.vB) * shadow_frame.GetVReg(dec_insn.vC));
1434 break;
1435 case Instruction::REM_INT:
jeffhaod91398c2012-11-20 17:17:33 -08001436 DoIntRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1437 shadow_frame.GetVReg(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001438 break;
1439 case Instruction::DIV_INT:
jeffhaod91398c2012-11-20 17:17:33 -08001440 DoIntDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1441 shadow_frame.GetVReg(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001442 break;
1443 case Instruction::SHL_INT:
jeffhao42b4dc42012-12-10 10:25:43 -08001444 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) <<
1445 (shadow_frame.GetVReg(dec_insn.vC) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001446 break;
1447 case Instruction::SHR_INT:
jeffhao42b4dc42012-12-10 10:25:43 -08001448 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) >>
1449 (shadow_frame.GetVReg(dec_insn.vC) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001450 break;
1451 case Instruction::USHR_INT:
1452 shadow_frame.SetVReg(dec_insn.vA,
1453 static_cast<uint32_t>(shadow_frame.GetVReg(dec_insn.vB)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001454 (shadow_frame.GetVReg(dec_insn.vC) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001455 break;
1456 case Instruction::AND_INT:
1457 shadow_frame.SetVReg(dec_insn.vA,
1458 shadow_frame.GetVReg(dec_insn.vB) & shadow_frame.GetVReg(dec_insn.vC));
1459 break;
1460 case Instruction::OR_INT:
1461 shadow_frame.SetVReg(dec_insn.vA,
1462 shadow_frame.GetVReg(dec_insn.vB) | shadow_frame.GetVReg(dec_insn.vC));
1463 break;
1464 case Instruction::XOR_INT:
1465 shadow_frame.SetVReg(dec_insn.vA,
1466 shadow_frame.GetVReg(dec_insn.vB) ^ shadow_frame.GetVReg(dec_insn.vC));
1467 break;
1468 case Instruction::ADD_LONG:
1469 shadow_frame.SetVRegLong(dec_insn.vA,
1470 shadow_frame.GetVRegLong(dec_insn.vB) +
1471 shadow_frame.GetVRegLong(dec_insn.vC));
1472 break;
1473 case Instruction::SUB_LONG:
1474 shadow_frame.SetVRegLong(dec_insn.vA,
1475 shadow_frame.GetVRegLong(dec_insn.vB) -
1476 shadow_frame.GetVRegLong(dec_insn.vC));
1477 break;
1478 case Instruction::MUL_LONG:
1479 shadow_frame.SetVRegLong(dec_insn.vA,
1480 shadow_frame.GetVRegLong(dec_insn.vB) *
1481 shadow_frame.GetVRegLong(dec_insn.vC));
1482 break;
1483 case Instruction::DIV_LONG:
jeffhaod91398c2012-11-20 17:17:33 -08001484 DoLongDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB),
1485 shadow_frame.GetVRegLong(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001486 break;
1487 case Instruction::REM_LONG:
jeffhaod91398c2012-11-20 17:17:33 -08001488 DoLongRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vB),
1489 shadow_frame.GetVRegLong(dec_insn.vC));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001490 break;
1491 case Instruction::AND_LONG:
1492 shadow_frame.SetVRegLong(dec_insn.vA,
1493 shadow_frame.GetVRegLong(dec_insn.vB) &
1494 shadow_frame.GetVRegLong(dec_insn.vC));
1495 break;
1496 case Instruction::OR_LONG:
1497 shadow_frame.SetVRegLong(dec_insn.vA,
1498 shadow_frame.GetVRegLong(dec_insn.vB) |
1499 shadow_frame.GetVRegLong(dec_insn.vC));
1500 break;
1501 case Instruction::XOR_LONG:
1502 shadow_frame.SetVRegLong(dec_insn.vA,
1503 shadow_frame.GetVRegLong(dec_insn.vB) ^
1504 shadow_frame.GetVRegLong(dec_insn.vC));
1505 break;
1506 case Instruction::SHL_LONG:
1507 shadow_frame.SetVRegLong(dec_insn.vA,
1508 shadow_frame.GetVRegLong(dec_insn.vB) <<
jeffhao42b4dc42012-12-10 10:25:43 -08001509 (shadow_frame.GetVReg(dec_insn.vC) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001510 break;
1511 case Instruction::SHR_LONG:
1512 shadow_frame.SetVRegLong(dec_insn.vA,
1513 shadow_frame.GetVRegLong(dec_insn.vB) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001514 (shadow_frame.GetVReg(dec_insn.vC) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001515 break;
1516 case Instruction::USHR_LONG:
1517 shadow_frame.SetVRegLong(dec_insn.vA,
1518 static_cast<uint64_t>(shadow_frame.GetVRegLong(dec_insn.vB)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001519 (shadow_frame.GetVReg(dec_insn.vC) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001520 break;
1521 case Instruction::ADD_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::SUB_FLOAT:
1527 shadow_frame.SetVRegFloat(dec_insn.vA,
1528 shadow_frame.GetVRegFloat(dec_insn.vB) -
1529 shadow_frame.GetVRegFloat(dec_insn.vC));
1530 break;
1531 case Instruction::MUL_FLOAT:
1532 shadow_frame.SetVRegFloat(dec_insn.vA,
1533 shadow_frame.GetVRegFloat(dec_insn.vB) *
1534 shadow_frame.GetVRegFloat(dec_insn.vC));
1535 break;
1536 case Instruction::DIV_FLOAT:
1537 shadow_frame.SetVRegFloat(dec_insn.vA,
1538 shadow_frame.GetVRegFloat(dec_insn.vB) /
1539 shadow_frame.GetVRegFloat(dec_insn.vC));
1540 break;
1541 case Instruction::REM_FLOAT:
1542 shadow_frame.SetVRegFloat(dec_insn.vA,
1543 fmodf(shadow_frame.GetVRegFloat(dec_insn.vB),
1544 shadow_frame.GetVRegFloat(dec_insn.vC)));
1545 break;
1546 case Instruction::ADD_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::SUB_DOUBLE:
1552 shadow_frame.SetVRegDouble(dec_insn.vA,
1553 shadow_frame.GetVRegDouble(dec_insn.vB) -
1554 shadow_frame.GetVRegDouble(dec_insn.vC));
1555 break;
1556 case Instruction::MUL_DOUBLE:
1557 shadow_frame.SetVRegDouble(dec_insn.vA,
1558 shadow_frame.GetVRegDouble(dec_insn.vB) *
1559 shadow_frame.GetVRegDouble(dec_insn.vC));
1560 break;
1561 case Instruction::DIV_DOUBLE:
1562 shadow_frame.SetVRegDouble(dec_insn.vA,
1563 shadow_frame.GetVRegDouble(dec_insn.vB) /
1564 shadow_frame.GetVRegDouble(dec_insn.vC));
1565 break;
1566 case Instruction::REM_DOUBLE:
1567 shadow_frame.SetVRegDouble(dec_insn.vA,
1568 fmod(shadow_frame.GetVRegDouble(dec_insn.vB),
1569 shadow_frame.GetVRegDouble(dec_insn.vC)));
1570 break;
1571 case Instruction::ADD_INT_2ADDR:
1572 shadow_frame.SetVReg(dec_insn.vA,
1573 shadow_frame.GetVReg(dec_insn.vA) + shadow_frame.GetVReg(dec_insn.vB));
1574 break;
1575 case Instruction::SUB_INT_2ADDR:
1576 shadow_frame.SetVReg(dec_insn.vA,
1577 shadow_frame.GetVReg(dec_insn.vA) - shadow_frame.GetVReg(dec_insn.vB));
1578 break;
1579 case Instruction::MUL_INT_2ADDR:
1580 shadow_frame.SetVReg(dec_insn.vA,
1581 shadow_frame.GetVReg(dec_insn.vA) * shadow_frame.GetVReg(dec_insn.vB));
1582 break;
1583 case Instruction::REM_INT_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001584 DoIntRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA),
1585 shadow_frame.GetVReg(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001586 break;
1587 case Instruction::SHL_INT_2ADDR:
jeffhao42b4dc42012-12-10 10:25:43 -08001588 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA) <<
1589 (shadow_frame.GetVReg(dec_insn.vB) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001590 break;
1591 case Instruction::SHR_INT_2ADDR:
jeffhao42b4dc42012-12-10 10:25:43 -08001592 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA) >>
1593 (shadow_frame.GetVReg(dec_insn.vB) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001594 break;
1595 case Instruction::USHR_INT_2ADDR:
1596 shadow_frame.SetVReg(dec_insn.vA,
1597 static_cast<uint32_t>(shadow_frame.GetVReg(dec_insn.vA)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001598 (shadow_frame.GetVReg(dec_insn.vB) & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001599 break;
1600 case Instruction::AND_INT_2ADDR:
1601 shadow_frame.SetVReg(dec_insn.vA,
1602 shadow_frame.GetVReg(dec_insn.vA) & shadow_frame.GetVReg(dec_insn.vB));
1603 break;
1604 case Instruction::OR_INT_2ADDR:
1605 shadow_frame.SetVReg(dec_insn.vA,
1606 shadow_frame.GetVReg(dec_insn.vA) | shadow_frame.GetVReg(dec_insn.vB));
1607 break;
1608 case Instruction::XOR_INT_2ADDR:
1609 shadow_frame.SetVReg(dec_insn.vA,
1610 shadow_frame.GetVReg(dec_insn.vA) ^ shadow_frame.GetVReg(dec_insn.vB));
1611 break;
1612 case Instruction::DIV_INT_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001613 DoIntDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vA),
1614 shadow_frame.GetVReg(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001615 break;
1616 case Instruction::ADD_LONG_2ADDR:
1617 shadow_frame.SetVRegLong(dec_insn.vA,
1618 shadow_frame.GetVRegLong(dec_insn.vA) +
1619 shadow_frame.GetVRegLong(dec_insn.vB));
1620 break;
1621 case Instruction::SUB_LONG_2ADDR:
1622 shadow_frame.SetVRegLong(dec_insn.vA,
1623 shadow_frame.GetVRegLong(dec_insn.vA) -
1624 shadow_frame.GetVRegLong(dec_insn.vB));
1625 break;
1626 case Instruction::MUL_LONG_2ADDR:
1627 shadow_frame.SetVRegLong(dec_insn.vA,
Ian Rogers64b6d142012-10-29 16:34:15 -07001628 shadow_frame.GetVRegLong(dec_insn.vA) *
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001629 shadow_frame.GetVRegLong(dec_insn.vB));
1630 break;
1631 case Instruction::DIV_LONG_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001632 DoLongDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vA),
1633 shadow_frame.GetVRegLong(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001634 break;
1635 case Instruction::REM_LONG_2ADDR:
jeffhaod91398c2012-11-20 17:17:33 -08001636 DoLongRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVRegLong(dec_insn.vA),
1637 shadow_frame.GetVRegLong(dec_insn.vB));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001638 break;
1639 case Instruction::AND_LONG_2ADDR:
1640 shadow_frame.SetVRegLong(dec_insn.vA,
1641 shadow_frame.GetVRegLong(dec_insn.vA) &
1642 shadow_frame.GetVRegLong(dec_insn.vB));
1643 break;
1644 case Instruction::OR_LONG_2ADDR:
1645 shadow_frame.SetVRegLong(dec_insn.vA,
1646 shadow_frame.GetVRegLong(dec_insn.vA) |
1647 shadow_frame.GetVRegLong(dec_insn.vB));
1648 break;
1649 case Instruction::XOR_LONG_2ADDR:
1650 shadow_frame.SetVRegLong(dec_insn.vA,
1651 shadow_frame.GetVRegLong(dec_insn.vA) ^
1652 shadow_frame.GetVRegLong(dec_insn.vB));
1653 break;
1654 case Instruction::SHL_LONG_2ADDR:
1655 shadow_frame.SetVRegLong(dec_insn.vA,
1656 shadow_frame.GetVRegLong(dec_insn.vA) <<
jeffhao42b4dc42012-12-10 10:25:43 -08001657 (shadow_frame.GetVReg(dec_insn.vB) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001658 break;
1659 case Instruction::SHR_LONG_2ADDR:
1660 shadow_frame.SetVRegLong(dec_insn.vA,
1661 shadow_frame.GetVRegLong(dec_insn.vA) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001662 (shadow_frame.GetVReg(dec_insn.vB) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001663 break;
1664 case Instruction::USHR_LONG_2ADDR:
1665 shadow_frame.SetVRegLong(dec_insn.vA,
1666 static_cast<uint64_t>(shadow_frame.GetVRegLong(dec_insn.vA)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001667 (shadow_frame.GetVReg(dec_insn.vB) & 0x3f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001668 break;
1669 case Instruction::ADD_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::SUB_FLOAT_2ADDR:
1675 shadow_frame.SetVRegFloat(dec_insn.vA,
1676 shadow_frame.GetVRegFloat(dec_insn.vA) -
1677 shadow_frame.GetVRegFloat(dec_insn.vB));
1678 break;
1679 case Instruction::MUL_FLOAT_2ADDR:
1680 shadow_frame.SetVRegFloat(dec_insn.vA,
1681 shadow_frame.GetVRegFloat(dec_insn.vA) *
1682 shadow_frame.GetVRegFloat(dec_insn.vB));
1683 break;
1684 case Instruction::DIV_FLOAT_2ADDR:
1685 shadow_frame.SetVRegFloat(dec_insn.vA,
1686 shadow_frame.GetVRegFloat(dec_insn.vA) /
1687 shadow_frame.GetVRegFloat(dec_insn.vB));
1688 break;
1689 case Instruction::REM_FLOAT_2ADDR:
1690 shadow_frame.SetVRegFloat(dec_insn.vA,
1691 fmodf(shadow_frame.GetVRegFloat(dec_insn.vA),
1692 shadow_frame.GetVRegFloat(dec_insn.vB)));
1693 break;
1694 case Instruction::ADD_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::SUB_DOUBLE_2ADDR:
1700 shadow_frame.SetVRegDouble(dec_insn.vA,
1701 shadow_frame.GetVRegDouble(dec_insn.vA) -
1702 shadow_frame.GetVRegDouble(dec_insn.vB));
1703 break;
1704 case Instruction::MUL_DOUBLE_2ADDR:
1705 shadow_frame.SetVRegDouble(dec_insn.vA,
1706 shadow_frame.GetVRegDouble(dec_insn.vA) *
1707 shadow_frame.GetVRegDouble(dec_insn.vB));
1708 break;
1709 case Instruction::DIV_DOUBLE_2ADDR:
1710 shadow_frame.SetVRegDouble(dec_insn.vA,
1711 shadow_frame.GetVRegDouble(dec_insn.vA) /
1712 shadow_frame.GetVRegDouble(dec_insn.vB));
1713 break;
1714 case Instruction::REM_DOUBLE_2ADDR:
1715 shadow_frame.SetVRegDouble(dec_insn.vA,
1716 fmod(shadow_frame.GetVRegDouble(dec_insn.vA),
1717 shadow_frame.GetVRegDouble(dec_insn.vB)));
1718 break;
1719 case Instruction::ADD_INT_LIT16:
1720 case Instruction::ADD_INT_LIT8:
1721 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) + dec_insn.vC);
1722 break;
1723 case Instruction::RSUB_INT:
1724 case Instruction::RSUB_INT_LIT8:
1725 shadow_frame.SetVReg(dec_insn.vA, dec_insn.vC - shadow_frame.GetVReg(dec_insn.vB));
1726 break;
1727 case Instruction::MUL_INT_LIT16:
1728 case Instruction::MUL_INT_LIT8:
1729 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) * dec_insn.vC);
1730 break;
1731 case Instruction::DIV_INT_LIT16:
1732 case Instruction::DIV_INT_LIT8:
jeffhaod91398c2012-11-20 17:17:33 -08001733 DoIntDivide(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1734 dec_insn.vC);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001735 break;
1736 case Instruction::REM_INT_LIT16:
1737 case Instruction::REM_INT_LIT8:
jeffhaod91398c2012-11-20 17:17:33 -08001738 DoIntRemainder(self, shadow_frame, dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB),
1739 dec_insn.vC);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001740 break;
1741 case Instruction::AND_INT_LIT16:
1742 case Instruction::AND_INT_LIT8:
1743 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) & dec_insn.vC);
1744 break;
1745 case Instruction::OR_INT_LIT16:
1746 case Instruction::OR_INT_LIT8:
1747 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) | dec_insn.vC);
1748 break;
1749 case Instruction::XOR_INT_LIT16:
1750 case Instruction::XOR_INT_LIT8:
1751 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) ^ dec_insn.vC);
1752 break;
1753 case Instruction::SHL_INT_LIT8:
jeffhao42b4dc42012-12-10 10:25:43 -08001754 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) <<
1755 (dec_insn.vC & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001756 break;
1757 case Instruction::SHR_INT_LIT8:
jeffhao42b4dc42012-12-10 10:25:43 -08001758 shadow_frame.SetVReg(dec_insn.vA, shadow_frame.GetVReg(dec_insn.vB) >>
1759 (dec_insn.vC & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001760 break;
1761 case Instruction::USHR_INT_LIT8:
1762 shadow_frame.SetVReg(dec_insn.vA,
1763 static_cast<uint32_t>(shadow_frame.GetVReg(dec_insn.vB)) >>
jeffhao42b4dc42012-12-10 10:25:43 -08001764 (dec_insn.vC & 0x1f));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001765 break;
1766 default:
1767 LOG(FATAL) << "Unexpected instruction: " << inst->DumpString(&mh.GetDexFile());
1768 break;
1769 }
1770 if (UNLIKELY(self->IsExceptionPending())) {
jeffhao9b5aa6f2012-12-18 11:47:11 -08001771 if (throw_frame_id_ == 0) {
1772 throw_method_ = shadow_frame.GetMethod();
1773 throw_dex_pc_ = dex_pc;
1774 }
1775 throw_frame_id_++;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001776 uint32_t found_dex_pc =
1777 shadow_frame.GetMethod()->FindCatchBlock(self->GetException()->GetClass(),
1778 inst->GetDexPc(insns));
1779 if (found_dex_pc == DexFile::kDexNoIndex) {
1780 JValue result;
1781 result.SetJ(0);
1782 return result; // Handler in caller.
1783 } else {
jeffhao9b5aa6f2012-12-18 11:47:11 -08001784 Dbg::PostException(self, throw_frame_id_, throw_method_, throw_dex_pc_,
1785 shadow_frame.GetMethod(), found_dex_pc, self->GetException());
1786 throw_frame_id_ = 0;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001787 next_inst = Instruction::At(insns + found_dex_pc);
1788 }
1789 }
1790 inst = next_inst;
1791 }
1792}
1793
1794void EnterInterpreterFromInvoke(Thread* self, AbstractMethod* method, Object* receiver,
1795 JValue* args, JValue* result) {
Ian Rogers64b6d142012-10-29 16:34:15 -07001796 DCHECK_EQ(self, Thread::Current());
jeffhaod7521322012-11-21 15:38:24 -08001797 if (__builtin_frame_address(0) < self->GetStackEnd()) {
1798 ThrowStackOverflowError(self);
1799 return;
1800 }
1801
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001802 MethodHelper mh(method);
1803 const DexFile::CodeItem* code_item = mh.GetCodeItem();
1804 uint16_t num_regs;
1805 uint16_t num_ins;
1806 if (code_item != NULL) {
1807 num_regs = code_item->registers_size_;
1808 num_ins = code_item->ins_size_;
jeffhao0a9bb732012-11-26 12:28:49 -08001809 } else if (method->IsAbstract()) {
1810 self->ThrowNewExceptionF("Ljava/lang/AbstractMethodError;", "abstract method \"%s\"",
1811 PrettyMethod(method).c_str());
1812 return;
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001813 } else {
1814 DCHECK(method->IsNative());
1815 num_regs = num_ins = AbstractMethod::NumArgRegisters(mh.GetShorty());
1816 if (!method->IsStatic()) {
1817 num_regs++;
1818 num_ins++;
1819 }
1820 }
1821 // Set up shadow frame with matching number of reference slots to vregs.
1822 ShadowFrame* last_shadow_frame = self->GetManagedStack()->GetTopShadowFrame();
TDYa127ce4cc0d2012-11-18 16:59:53 -08001823 UniquePtr<ShadowFrame> shadow_frame(ShadowFrame::Create(num_regs,
Ian Rogers7db619b2013-01-16 18:35:48 -08001824 last_shadow_frame,
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001825 method, 0));
1826 self->PushShadowFrame(shadow_frame.get());
1827 size_t cur_reg = num_regs - num_ins;
1828 if (!method->IsStatic()) {
1829 CHECK(receiver != NULL);
TDYa127ce4cc0d2012-11-18 16:59:53 -08001830 shadow_frame->SetVRegReference(cur_reg, receiver);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001831 ++cur_reg;
1832 } else if (!method->GetDeclaringClass()->IsInitializing()) {
jeffhao94d6df42012-11-26 16:02:12 -08001833 if (!Runtime::Current()->GetClassLinker()->EnsureInitialized(method->GetDeclaringClass(),
1834 true, true)) {
1835 DCHECK(Thread::Current()->IsExceptionPending());
1836 return;
1837 }
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001838 CHECK(method->GetDeclaringClass()->IsInitializing());
1839 }
Ian Rogers64b6d142012-10-29 16:34:15 -07001840 const char* shorty = mh.GetShorty();
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001841 size_t arg_pos = 0;
1842 for (; cur_reg < num_regs; ++cur_reg, ++arg_pos) {
1843 DCHECK_LT(arg_pos + 1, mh.GetShortyLength());
1844 switch (shorty[arg_pos + 1]) {
1845 case 'L': {
1846 Object* o = args[arg_pos].GetL();
TDYa127ce4cc0d2012-11-18 16:59:53 -08001847 shadow_frame->SetVRegReference(cur_reg, o);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001848 break;
1849 }
1850 case 'J': case 'D':
1851 shadow_frame->SetVRegLong(cur_reg, args[arg_pos].GetJ());
1852 cur_reg++;
1853 break;
1854 default:
1855 shadow_frame->SetVReg(cur_reg, args[arg_pos].GetI());
1856 break;
1857 }
1858 }
Ian Rogers64b6d142012-10-29 16:34:15 -07001859 if (LIKELY(!method->IsNative())) {
Ian Rogers306057f2012-11-26 12:45:53 -08001860 JValue r = Execute(self, mh, code_item, *shadow_frame.get(), JValue());
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001861 if (result != NULL) {
1862 *result = r;
1863 }
1864 } else {
Ian Rogers64b6d142012-10-29 16:34:15 -07001865 // We don't expect to be asked to interpret native code (which is entered via a JNI compiler
1866 // generated stub) except during testing and image writing.
1867 if (!Runtime::Current()->IsStarted()) {
1868 UnstartedRuntimeJni(self, method, receiver, args, result);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001869 } else {
Ian Rogers64b6d142012-10-29 16:34:15 -07001870 InterpreterJni(self, method, shorty, receiver, args, result);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001871 }
1872 }
1873 self->PopShadowFrame();
1874}
1875
Ian Rogers306057f2012-11-26 12:45:53 -08001876JValue EnterInterpreterFromDeoptimize(Thread* self, ShadowFrame& shadow_frame, JValue ret_val)
1877 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
1878 MethodHelper mh(shadow_frame.GetMethod());
1879 const DexFile::CodeItem* code_item = mh.GetCodeItem();
1880 return Execute(self, mh, code_item, shadow_frame, ret_val);
1881}
1882
Jeff Hao11ffc2d2013-02-01 11:52:17 -08001883void EnterInterpreterFromLLVM(Thread* self, ShadowFrame* shadow_frame, JValue* ret_val)
1884 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
1885 JValue value;
1886 MethodHelper mh(shadow_frame->GetMethod());
1887 const DexFile::CodeItem* code_item = mh.GetCodeItem();
1888 while (shadow_frame != NULL) {
1889 value = Execute(self, mh, code_item, *shadow_frame, value);
1890 ShadowFrame* old_frame = shadow_frame;
1891 shadow_frame = shadow_frame->GetLink();
1892 mh.ChangeMethod(shadow_frame->GetMethod());
1893 delete old_frame;
1894 }
1895 ret_val->SetJ(value.GetJ());
1896}
1897
Ian Rogers7db619b2013-01-16 18:35:48 -08001898JValue EnterInterpreterFromStub(Thread* self, MethodHelper& mh, const DexFile::CodeItem* code_item,
1899 ShadowFrame& shadow_frame)
1900 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
1901 return Execute(self, mh, code_item, shadow_frame, JValue());
1902}
1903
Ian Rogers2fa6b2e2012-10-17 00:10:17 -07001904} // namespace interpreter
1905} // namespace art