blob: 0ec1eb7920bef8f3fae17e222a65d286d4567eb1 [file] [log] [blame]
Ian Rogers57b86d42012-03-27 16:05:41 -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 "callee_save_frame.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070018#include "dex_file-inl.h"
Ian Rogers7655f292013-07-29 11:07:13 -070019#include "entrypoints/entrypoint_utils.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070020#include "mirror/art_field-inl.h"
21#include "mirror/art_method-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080022#include "mirror/class-inl.h"
Ian Rogers57b86d42012-03-27 16:05:41 -070023
24#include <stdint.h>
25
26namespace art {
27
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080028extern "C" uint32_t artGet32StaticFromCode(uint32_t field_idx,
Brian Carlstromea46f952013-07-30 01:26:50 -070029 const mirror::ArtMethod* referrer,
30 Thread* self, mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -070031 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -070032 mirror::ArtField* field = FindFieldFast(field_idx, referrer, StaticPrimitiveRead,
33 sizeof(int32_t));
Ian Rogers57b86d42012-03-27 16:05:41 -070034 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070035 return field->Get32(field->GetDeclaringClass());
Ian Rogers57b86d42012-03-27 16:05:41 -070036 }
37 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertz233ea8e2013-06-06 11:57:09 +020038 field = FindFieldFromCode(field_idx, referrer, self, StaticPrimitiveRead, sizeof(int32_t), true);
Ian Rogers57b86d42012-03-27 16:05:41 -070039 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070040 return field->Get32(field->GetDeclaringClass());
Ian Rogers57b86d42012-03-27 16:05:41 -070041 }
42 return 0; // Will throw exception by checking with Thread::Current
43}
44
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080045extern "C" uint64_t artGet64StaticFromCode(uint32_t field_idx,
Brian Carlstromea46f952013-07-30 01:26:50 -070046 const mirror::ArtMethod* referrer,
47 Thread* self, mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -070048 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -070049 mirror::ArtField* field = FindFieldFast(field_idx, referrer, StaticPrimitiveRead,
50 sizeof(int64_t));
Ian Rogers57b86d42012-03-27 16:05:41 -070051 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070052 return field->Get64(field->GetDeclaringClass());
Ian Rogers57b86d42012-03-27 16:05:41 -070053 }
54 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertz233ea8e2013-06-06 11:57:09 +020055 field = FindFieldFromCode(field_idx, referrer, self, StaticPrimitiveRead, sizeof(int64_t), true);
Ian Rogers57b86d42012-03-27 16:05:41 -070056 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070057 return field->Get64(field->GetDeclaringClass());
Ian Rogers57b86d42012-03-27 16:05:41 -070058 }
59 return 0; // Will throw exception by checking with Thread::Current
60}
61
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080062extern "C" mirror::Object* artGetObjStaticFromCode(uint32_t field_idx,
Brian Carlstromea46f952013-07-30 01:26:50 -070063 const mirror::ArtMethod* referrer,
64 Thread* self, mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -070065 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -070066 mirror::ArtField* field = FindFieldFast(field_idx, referrer, StaticObjectRead,
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080067 sizeof(mirror::Object*));
Ian Rogers57b86d42012-03-27 16:05:41 -070068 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070069 return field->GetObj(field->GetDeclaringClass());
Ian Rogers57b86d42012-03-27 16:05:41 -070070 }
71 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Brian Carlstromea46f952013-07-30 01:26:50 -070072 field = FindFieldFromCode(field_idx, referrer, self, StaticObjectRead, sizeof(mirror::Object*),
73 true);
Ian Rogers57b86d42012-03-27 16:05:41 -070074 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -070075 return field->GetObj(field->GetDeclaringClass());
Ian Rogers57b86d42012-03-27 16:05:41 -070076 }
77 return NULL; // Will throw exception by checking with Thread::Current
78}
79
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080080extern "C" uint32_t artGet32InstanceFromCode(uint32_t field_idx, mirror::Object* obj,
Brian Carlstromea46f952013-07-30 01:26:50 -070081 const mirror::ArtMethod* referrer, Thread* self,
82 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -070083 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -070084 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstancePrimitiveRead,
85 sizeof(int32_t));
Ian Rogers57b86d42012-03-27 16:05:41 -070086 if (LIKELY(field != NULL && obj != NULL)) {
87 return field->Get32(obj);
88 }
89 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Brian Carlstromea46f952013-07-30 01:26:50 -070090 field = FindFieldFromCode(field_idx, referrer, self, InstancePrimitiveRead, sizeof(int32_t),
91 true);
Ian Rogers57b86d42012-03-27 16:05:41 -070092 if (LIKELY(field != NULL)) {
93 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -080094 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
95 ThrowNullPointerExceptionForFieldAccess(throw_location, field, true);
Ian Rogers57b86d42012-03-27 16:05:41 -070096 } else {
97 return field->Get32(obj);
98 }
99 }
100 return 0; // Will throw exception by checking with Thread::Current
101}
102
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800103extern "C" uint64_t artGet64InstanceFromCode(uint32_t field_idx, mirror::Object* obj,
Brian Carlstromea46f952013-07-30 01:26:50 -0700104 const mirror::ArtMethod* referrer, Thread* self,
105 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700106 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700107 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstancePrimitiveRead,
108 sizeof(int64_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700109 if (LIKELY(field != NULL && obj != NULL)) {
110 return field->Get64(obj);
111 }
112 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Brian Carlstromea46f952013-07-30 01:26:50 -0700113 field = FindFieldFromCode(field_idx, referrer, self, InstancePrimitiveRead, sizeof(int64_t),
114 true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700115 if (LIKELY(field != NULL)) {
116 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800117 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
118 ThrowNullPointerExceptionForFieldAccess(throw_location, field, true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700119 } else {
120 return field->Get64(obj);
121 }
122 }
123 return 0; // Will throw exception by checking with Thread::Current
124}
125
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800126extern "C" mirror::Object* artGetObjInstanceFromCode(uint32_t field_idx, mirror::Object* obj,
Brian Carlstromea46f952013-07-30 01:26:50 -0700127 const mirror::ArtMethod* referrer,
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800128 Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -0700129 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700130 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700131 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstanceObjectRead,
132 sizeof(mirror::Object*));
Ian Rogers57b86d42012-03-27 16:05:41 -0700133 if (LIKELY(field != NULL && obj != NULL)) {
134 return field->GetObj(obj);
135 }
136 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Brian Carlstromea46f952013-07-30 01:26:50 -0700137 field = FindFieldFromCode(field_idx, referrer, self, InstanceObjectRead, sizeof(mirror::Object*),
138 true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700139 if (LIKELY(field != NULL)) {
140 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800141 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
142 ThrowNullPointerExceptionForFieldAccess(throw_location, field, true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700143 } else {
144 return field->GetObj(obj);
145 }
146 }
147 return NULL; // Will throw exception by checking with Thread::Current
148}
149
150extern "C" int artSet32StaticFromCode(uint32_t field_idx, uint32_t new_value,
Brian Carlstromea46f952013-07-30 01:26:50 -0700151 const mirror::ArtMethod* referrer, Thread* self,
152 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700153 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700154 mirror::ArtField* field = FindFieldFast(field_idx, referrer, StaticPrimitiveWrite,
155 sizeof(int32_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700156 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700157 field->Set32(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700158 return 0; // success
159 }
160 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertz233ea8e2013-06-06 11:57:09 +0200161 field = FindFieldFromCode(field_idx, referrer, self, StaticPrimitiveWrite, sizeof(int32_t), true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700162 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700163 field->Set32(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700164 return 0; // success
165 }
166 return -1; // failure
167}
168
Brian Carlstromea46f952013-07-30 01:26:50 -0700169extern "C" int artSet64StaticFromCode(uint32_t field_idx, const mirror::ArtMethod* referrer,
170 uint64_t new_value, Thread* self, mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700171 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700172 mirror::ArtField* field = FindFieldFast(field_idx, referrer, StaticPrimitiveWrite,
173 sizeof(int64_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700174 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700175 field->Set64(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700176 return 0; // success
177 }
178 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertz233ea8e2013-06-06 11:57:09 +0200179 field = FindFieldFromCode(field_idx, referrer, self, StaticPrimitiveWrite, sizeof(int64_t), true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700180 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700181 field->Set64(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700182 return 0; // success
183 }
184 return -1; // failure
185}
186
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800187extern "C" int artSetObjStaticFromCode(uint32_t field_idx, mirror::Object* new_value,
Brian Carlstromea46f952013-07-30 01:26:50 -0700188 const mirror::ArtMethod* referrer, Thread* self,
189 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700190 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700191 mirror::ArtField* field = FindFieldFast(field_idx, referrer, StaticObjectWrite,
192 sizeof(mirror::Object*));
Ian Rogers57b86d42012-03-27 16:05:41 -0700193 if (LIKELY(field != NULL)) {
194 if (LIKELY(!FieldHelper(field).IsPrimitiveType())) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700195 field->SetObj(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700196 return 0; // success
197 }
198 }
199 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertz233ea8e2013-06-06 11:57:09 +0200200 field = FindFieldFromCode(field_idx, referrer, self, StaticObjectWrite, sizeof(mirror::Object*), true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700201 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700202 field->SetObj(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700203 return 0; // success
204 }
205 return -1; // failure
206}
207
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800208extern "C" int artSet32InstanceFromCode(uint32_t field_idx, mirror::Object* obj, uint32_t new_value,
Brian Carlstromea46f952013-07-30 01:26:50 -0700209 const mirror::ArtMethod* referrer, Thread* self,
210 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700211 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700212 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstancePrimitiveWrite,
213 sizeof(int32_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700214 if (LIKELY(field != NULL && obj != NULL)) {
215 field->Set32(obj, new_value);
216 return 0; // success
217 }
218 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Brian Carlstromea46f952013-07-30 01:26:50 -0700219 field = FindFieldFromCode(field_idx, referrer, self, InstancePrimitiveWrite, sizeof(int32_t),
220 true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700221 if (LIKELY(field != NULL)) {
222 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800223 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
224 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false);
Ian Rogers57b86d42012-03-27 16:05:41 -0700225 } else {
226 field->Set32(obj, new_value);
227 return 0; // success
228 }
229 }
230 return -1; // failure
231}
232
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800233extern "C" int artSet64InstanceFromCode(uint32_t field_idx, mirror::Object* obj, uint64_t new_value,
Brian Carlstromea46f952013-07-30 01:26:50 -0700234 Thread* self, mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700235 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700236 mirror::ArtMethod* callee_save = Runtime::Current()->GetCalleeSaveMethod(Runtime::kRefsOnly);
237 mirror::ArtMethod* referrer =
238 sp[callee_save->GetFrameSizeInBytes() / sizeof(mirror::ArtMethod*)];
239 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstancePrimitiveWrite,
240 sizeof(int64_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700241 if (LIKELY(field != NULL && obj != NULL)) {
242 field->Set64(obj, new_value);
243 return 0; // success
244 }
245 *sp = callee_save;
246 self->SetTopOfStack(sp, 0);
Brian Carlstromea46f952013-07-30 01:26:50 -0700247 field = FindFieldFromCode(field_idx, referrer, self, InstancePrimitiveWrite, sizeof(int64_t),
248 true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700249 if (LIKELY(field != NULL)) {
250 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800251 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
252 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false);
Ian Rogers57b86d42012-03-27 16:05:41 -0700253 } else {
254 field->Set64(obj, new_value);
255 return 0; // success
256 }
257 }
258 return -1; // failure
259}
260
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800261extern "C" int artSetObjInstanceFromCode(uint32_t field_idx, mirror::Object* obj,
262 mirror::Object* new_value,
Brian Carlstromea46f952013-07-30 01:26:50 -0700263 const mirror::ArtMethod* referrer, Thread* self,
264 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700265 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700266 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstanceObjectWrite,
267 sizeof(mirror::Object*));
Ian Rogers57b86d42012-03-27 16:05:41 -0700268 if (LIKELY(field != NULL && obj != NULL)) {
269 field->SetObj(obj, new_value);
270 return 0; // success
271 }
272 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800273 field = FindFieldFromCode(field_idx, referrer, self, InstanceObjectWrite,
Sebastien Hertz233ea8e2013-06-06 11:57:09 +0200274 sizeof(mirror::Object*), true);
Ian Rogers57b86d42012-03-27 16:05:41 -0700275 if (LIKELY(field != NULL)) {
276 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800277 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
278 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false);
Ian Rogers57b86d42012-03-27 16:05:41 -0700279 } else {
280 field->SetObj(obj, new_value);
281 return 0; // success
282 }
283 }
284 return -1; // failure
285}
286
287} // namespace art