blob: 93ff7aa9fa98144f42d753fed1cc5ececd36911b [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,
Ian Rogersef7d42f2014-01-06 12:55:46 -080029 mirror::ArtMethod* referrer,
Brian Carlstromea46f952013-07-30 01:26:50 -070030 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 Hertzd4beb6b2013-10-02 17:07:20 +020038 field = FindFieldFromCode<StaticPrimitiveRead, true>(field_idx, referrer, self, sizeof(int32_t));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -080046 mirror::ArtMethod* referrer,
Brian Carlstromea46f952013-07-30 01:26:50 -070047 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 Hertzd4beb6b2013-10-02 17:07:20 +020055 field = FindFieldFromCode<StaticPrimitiveRead, true>(field_idx, referrer, self, sizeof(int64_t));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -080063 mirror::ArtMethod* referrer,
Brian Carlstromea46f952013-07-30 01:26:50 -070064 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 Rogersef7d42f2014-01-06 12:55:46 -080067 sizeof(mirror::HeapReference<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);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +020072 field = FindFieldFromCode<StaticObjectRead, true>(field_idx, referrer, self,
Ian Rogersef7d42f2014-01-06 12:55:46 -080073 sizeof(mirror::HeapReference<mirror::Object>));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -080081 mirror::ArtMethod* referrer, Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -070082 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);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +020090 field = FindFieldFromCode<InstancePrimitiveRead, true>(field_idx, referrer, self,
91 sizeof(int32_t));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800104 mirror::ArtMethod* referrer, Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -0700105 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);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +0200113 field = FindFieldFromCode<InstancePrimitiveRead, true>(field_idx, referrer, self,
114 sizeof(int64_t));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800127 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,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800132 sizeof(mirror::HeapReference<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);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +0200137 field = FindFieldFromCode<InstanceObjectRead, true>(field_idx, referrer, self,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800138 sizeof(mirror::HeapReference<mirror::Object>));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800151 mirror::ArtMethod* referrer, Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -0700152 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 Hertzd4beb6b2013-10-02 17:07:20 +0200161 field = FindFieldFromCode<StaticPrimitiveWrite, true>(field_idx, referrer, self, sizeof(int32_t));
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
Ian Rogersef7d42f2014-01-06 12:55:46 -0800169extern "C" int artSet64StaticFromCode(uint32_t field_idx, mirror::ArtMethod* referrer,
Brian Carlstromea46f952013-07-30 01:26:50 -0700170 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 Hertzd4beb6b2013-10-02 17:07:20 +0200179 field = FindFieldFromCode<StaticPrimitiveWrite, true>(field_idx, referrer, self, sizeof(int64_t));
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,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800188 mirror::ArtMethod* referrer, Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -0700189 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,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800192 sizeof(mirror::HeapReference<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 Hertzd4beb6b2013-10-02 17:07:20 +0200200 field = FindFieldFromCode<StaticObjectWrite, true>(field_idx, referrer, self,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800201 sizeof(mirror::HeapReference<mirror::Object>));
Ian Rogers57b86d42012-03-27 16:05:41 -0700202 if (LIKELY(field != NULL)) {
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700203 field->SetObj(field->GetDeclaringClass(), new_value);
Ian Rogers57b86d42012-03-27 16:05:41 -0700204 return 0; // success
205 }
206 return -1; // failure
207}
208
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800209extern "C" int artSet32InstanceFromCode(uint32_t field_idx, mirror::Object* obj, uint32_t new_value,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800210 mirror::ArtMethod* referrer, Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -0700211 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700212 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700213 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstancePrimitiveWrite,
214 sizeof(int32_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700215 if (LIKELY(field != NULL && obj != NULL)) {
216 field->Set32(obj, new_value);
217 return 0; // success
218 }
219 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +0200220 field = FindFieldFromCode<InstancePrimitiveWrite, true>(field_idx, referrer, self,
221 sizeof(int32_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700222 if (LIKELY(field != NULL)) {
223 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800224 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
225 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false);
Ian Rogers57b86d42012-03-27 16:05:41 -0700226 } else {
227 field->Set32(obj, new_value);
228 return 0; // success
229 }
230 }
231 return -1; // failure
232}
233
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800234extern "C" int artSet64InstanceFromCode(uint32_t field_idx, mirror::Object* obj, uint64_t new_value,
Brian Carlstromea46f952013-07-30 01:26:50 -0700235 Thread* self, mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700236 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700237 mirror::ArtMethod* callee_save = Runtime::Current()->GetCalleeSaveMethod(Runtime::kRefsOnly);
238 mirror::ArtMethod* referrer =
239 sp[callee_save->GetFrameSizeInBytes() / sizeof(mirror::ArtMethod*)];
240 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstancePrimitiveWrite,
241 sizeof(int64_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700242 if (LIKELY(field != NULL && obj != NULL)) {
243 field->Set64(obj, new_value);
244 return 0; // success
245 }
246 *sp = callee_save;
247 self->SetTopOfStack(sp, 0);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +0200248 field = FindFieldFromCode<InstancePrimitiveWrite, true>(field_idx, referrer, self,
249 sizeof(int64_t));
Ian Rogers57b86d42012-03-27 16:05:41 -0700250 if (LIKELY(field != NULL)) {
251 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800252 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
253 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false);
Ian Rogers57b86d42012-03-27 16:05:41 -0700254 } else {
255 field->Set64(obj, new_value);
256 return 0; // success
257 }
258 }
259 return -1; // failure
260}
261
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800262extern "C" int artSetObjInstanceFromCode(uint32_t field_idx, mirror::Object* obj,
263 mirror::Object* new_value,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800264 mirror::ArtMethod* referrer, Thread* self,
Brian Carlstromea46f952013-07-30 01:26:50 -0700265 mirror::ArtMethod** sp)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700266 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700267 mirror::ArtField* field = FindFieldFast(field_idx, referrer, InstanceObjectWrite,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800268 sizeof(mirror::HeapReference<mirror::Object>));
Ian Rogers57b86d42012-03-27 16:05:41 -0700269 if (LIKELY(field != NULL && obj != NULL)) {
270 field->SetObj(obj, new_value);
271 return 0; // success
272 }
273 FinishCalleeSaveFrameSetup(self, sp, Runtime::kRefsOnly);
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +0200274 field = FindFieldFromCode<InstanceObjectWrite, true>(field_idx, referrer, self,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800275 sizeof(mirror::HeapReference<mirror::Object>));
Ian Rogers57b86d42012-03-27 16:05:41 -0700276 if (LIKELY(field != NULL)) {
277 if (UNLIKELY(obj == NULL)) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800278 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
279 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false);
Ian Rogers57b86d42012-03-27 16:05:41 -0700280 } else {
281 field->SetObj(obj, new_value);
282 return 0; // success
283 }
284 }
285 return -1; // failure
286}
287
288} // namespace art