blob: 0d5477275a85cabe5ecd0b8aa588e573b72b82f3 [file] [log] [blame]
Brian Carlstromf867b6f2011-09-16 12:17:25 -07001/*
2 * Copyright (C) 2008 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
Brian Carlstromf867b6f2011-09-16 12:17:25 -070017#include "class_linker.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080018#include "class_linker-inl.h"
Ian Rogers62d6c772013-02-27 08:32:07 -080019#include "common_throws.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070020#include "dex_file-inl.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070021#include "jni_internal.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070022#include "mirror/art_field-inl.h"
Jeff Hao11d5d8f2014-03-26 15:08:20 -070023#include "mirror/art_method-inl.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070024#include "mirror/class-inl.h"
Ian Rogers6d4d9fc2011-11-30 16:24:48 -080025#include "object_utils.h"
Elliott Hughes418d20f2011-09-22 14:00:39 -070026#include "reflection.h"
Ian Rogers1eb512d2013-10-18 15:42:20 -070027#include "scoped_fast_native_object_access.h"
Brian Carlstromf867b6f2011-09-16 12:17:25 -070028
Brian Carlstromf867b6f2011-09-16 12:17:25 -070029namespace art {
30
Jeff Haocb4581a2014-03-28 15:43:37 -070031static bool VerifyFieldAccess(mirror::ArtField* field, mirror::Object* obj, bool is_set)
Jeff Hao11d5d8f2014-03-26 15:08:20 -070032 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
33 if (field->IsFinal() && is_set) {
34 ThrowIllegalAccessException(nullptr, StringPrintf("Cannot set final field: %s",
35 PrettyField(field).c_str()).c_str());
36 return false;
37 }
Jeff Haocb4581a2014-03-28 15:43:37 -070038 if (!VerifyAccess(obj, field->GetDeclaringClass(), field->GetAccessFlags())) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -070039 ThrowIllegalAccessException(nullptr, StringPrintf("Cannot access field: %s",
40 PrettyField(field).c_str()).c_str());
41 return false;
42 }
43 return true;
44}
45
Ian Rogers1eb512d2013-10-18 15:42:20 -070046static bool GetFieldValue(const ScopedFastNativeObjectAccess& soa, mirror::Object* o,
Ian Rogers62f05122014-03-21 11:21:29 -070047 mirror::ArtField* f, Primitive::Type field_type, bool allow_references,
48 JValue* value)
Ian Rogersb726dcb2012-09-05 08:57:23 -070049 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers62f05122014-03-21 11:21:29 -070050 DCHECK_EQ(value->GetJ(), INT64_C(0));
Ian Rogers62f05122014-03-21 11:21:29 -070051 switch (field_type) {
52 case Primitive::kPrimBoolean:
53 value->SetZ(f->GetBoolean(o));
Elliott Hughes33203b52011-09-20 19:42:01 -070054 return true;
Ian Rogers62f05122014-03-21 11:21:29 -070055 case Primitive::kPrimByte:
56 value->SetB(f->GetByte(o));
57 return true;
58 case Primitive::kPrimChar:
59 value->SetC(f->GetChar(o));
60 return true;
61 case Primitive::kPrimDouble:
62 value->SetD(f->GetDouble(o));
63 return true;
64 case Primitive::kPrimFloat:
65 value->SetF(f->GetFloat(o));
66 return true;
67 case Primitive::kPrimInt:
68 value->SetI(f->GetInt(o));
69 return true;
70 case Primitive::kPrimLong:
71 value->SetJ(f->GetLong(o));
72 return true;
73 case Primitive::kPrimShort:
74 value->SetS(f->GetShort(o));
75 return true;
76 case Primitive::kPrimNot:
77 if (allow_references) {
78 value->SetL(f->GetObject(o));
79 return true;
80 }
81 // Else break to report an error.
82 break;
83 case Primitive::kPrimVoid:
84 // Never okay.
85 break;
Elliott Hughes33203b52011-09-20 19:42:01 -070086 }
Ian Rogers62f05122014-03-21 11:21:29 -070087 ThrowIllegalArgumentException(nullptr, StringPrintf("Not a primitive field: %s",
88 PrettyField(f).c_str()).c_str());
Elliott Hughes33203b52011-09-20 19:42:01 -070089 return false;
90}
91
Ian Rogers1eb512d2013-10-18 15:42:20 -070092static bool CheckReceiver(const ScopedFastNativeObjectAccess& soa, jobject j_rcvr,
Ian Rogers62f05122014-03-21 11:21:29 -070093 mirror::ArtField* f, mirror::Object** class_or_rcvr)
Ian Rogersb726dcb2012-09-05 08:57:23 -070094 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers62f05122014-03-21 11:21:29 -070095 soa.Self()->AssertThreadSuspensionIsAllowable();
Elliott Hughesed1c1e32011-10-02 14:31:05 -070096 if (f->IsStatic()) {
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070097 StackHandleScope<1> hs(soa.Self());
98 Handle<mirror::Class> h_klass(hs.NewHandle(f->GetDeclaringClass()));
99 if (UNLIKELY(!Runtime::Current()->GetClassLinker()->EnsureInitialized(h_klass, true, true))) {
Ian Rogers62f05122014-03-21 11:21:29 -0700100 DCHECK(soa.Self()->IsExceptionPending());
101 *class_or_rcvr = nullptr;
102 return false;
103 }
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700104 *class_or_rcvr = h_klass.Get();
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700105 return true;
106 }
107
Ian Rogers62f05122014-03-21 11:21:29 -0700108 *class_or_rcvr = soa.Decode<mirror::Object*>(j_rcvr);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800109 mirror::Class* declaringClass = f->GetDeclaringClass();
Ian Rogers62f05122014-03-21 11:21:29 -0700110 if (!VerifyObjectIsClass(*class_or_rcvr, declaringClass)) {
111 DCHECK(soa.Self()->IsExceptionPending());
112 *class_or_rcvr = nullptr;
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700113 return false;
114 }
115 return true;
116}
117
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700118static jobject Field_get(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700119 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700120 CHECK(!kMovingFields) << "CheckReceiver may trigger thread suspension for initialization";
121 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
122 mirror::Object* o = nullptr;
123 if (!CheckReceiver(soa, javaObj, f, &o)) {
124 DCHECK(soa.Self()->IsExceptionPending());
125 return nullptr;
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800126 }
Jeff Haocb4581a2014-03-28 15:43:37 -0700127 // If field is not set to be accessible, verify it can be accessed by the caller.
128 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, false)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700129 DCHECK(soa.Self()->IsExceptionPending());
130 return nullptr;
131 }
Ian Rogers62f05122014-03-21 11:21:29 -0700132 // We now don't expect suspension unless an exception is thrown.
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800133 // Get the field's value, boxing if necessary.
Ian Rogers62f05122014-03-21 11:21:29 -0700134 Primitive::Type field_type = FieldHelper(f).GetTypeAsPrimitiveType();
Elliott Hughes1d878f32012-04-11 15:17:54 -0700135 JValue value;
Ian Rogers62f05122014-03-21 11:21:29 -0700136 if (!GetFieldValue(soa, o, f, field_type, true, &value)) {
137 DCHECK(soa.Self()->IsExceptionPending());
138 return nullptr;
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800139 }
Ian Rogers62f05122014-03-21 11:21:29 -0700140 return soa.AddLocalReference<jobject>(BoxPrimitive(field_type, value));
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800141}
142
Brian Carlstromea46f952013-07-30 01:26:50 -0700143static JValue GetPrimitiveField(JNIEnv* env, jobject javaField, jobject javaObj,
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700144 char dst_descriptor, jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700145 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700146 CHECK(!kMovingFields) << "CheckReceiver may trigger thread suspension for initialization";
147 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
148 mirror::Object* o = nullptr;
149 if (!CheckReceiver(soa, javaObj, f, &o)) {
150 DCHECK(soa.Self()->IsExceptionPending());
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700151 return JValue();
Elliott Hughes33203b52011-09-20 19:42:01 -0700152 }
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700153
Jeff Haocb4581a2014-03-28 15:43:37 -0700154 // If field is not set to be accessible, verify it can be accessed by the caller.
155 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, false)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700156 DCHECK(soa.Self()->IsExceptionPending());
157 return JValue();
158 }
159
Ian Rogers62f05122014-03-21 11:21:29 -0700160 // We now don't expect suspension unless an exception is thrown.
Elliott Hughes33203b52011-09-20 19:42:01 -0700161 // Read the value.
Ian Rogers62f05122014-03-21 11:21:29 -0700162 Primitive::Type field_type = FieldHelper(f).GetTypeAsPrimitiveType();
Elliott Hughes1d878f32012-04-11 15:17:54 -0700163 JValue field_value;
Ian Rogers62f05122014-03-21 11:21:29 -0700164 if (!GetFieldValue(soa, o, f, field_type, false, &field_value)) {
165 DCHECK(soa.Self()->IsExceptionPending());
Elliott Hughes33203b52011-09-20 19:42:01 -0700166 return JValue();
167 }
168
169 // Widen it if necessary (and possible).
170 JValue wide_value;
Ian Rogers62f05122014-03-21 11:21:29 -0700171 if (!ConvertPrimitiveValue(NULL, false, field_type, Primitive::GetType(dst_descriptor),
Ian Rogers84956ff2014-03-26 23:52:41 -0700172 field_value, &wide_value)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700173 DCHECK(soa.Self()->IsExceptionPending());
Elliott Hughes33203b52011-09-20 19:42:01 -0700174 return JValue();
175 }
176 return wide_value;
177}
178
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700179static jboolean Field_getBoolean(JNIEnv* env, jobject javaField, jobject javaObj,
180 jboolean accessible) {
181 return GetPrimitiveField(env, javaField, javaObj, 'Z', accessible).GetZ();
Elliott Hughes33203b52011-09-20 19:42:01 -0700182}
183
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700184static jbyte Field_getByte(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
185 return GetPrimitiveField(env, javaField, javaObj, 'B', accessible).GetB();
Elliott Hughes33203b52011-09-20 19:42:01 -0700186}
187
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700188static jchar Field_getChar(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
189 return GetPrimitiveField(env, javaField, javaObj, 'C', accessible).GetC();
Elliott Hughes33203b52011-09-20 19:42:01 -0700190}
191
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700192static jdouble Field_getDouble(JNIEnv* env, jobject javaField, jobject javaObj,
193 jboolean accessible) {
194 return GetPrimitiveField(env, javaField, javaObj, 'D', accessible).GetD();
Elliott Hughes33203b52011-09-20 19:42:01 -0700195}
196
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700197static jfloat Field_getFloat(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
198 return GetPrimitiveField(env, javaField, javaObj, 'F', accessible).GetF();
Elliott Hughes33203b52011-09-20 19:42:01 -0700199}
200
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700201static jint Field_getInt(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
202 return GetPrimitiveField(env, javaField, javaObj, 'I', accessible).GetI();
Elliott Hughes33203b52011-09-20 19:42:01 -0700203}
204
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700205static jlong Field_getLong(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
206 return GetPrimitiveField(env, javaField, javaObj, 'J', accessible).GetJ();
Elliott Hughes33203b52011-09-20 19:42:01 -0700207}
208
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700209static jshort Field_getShort(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
210 return GetPrimitiveField(env, javaField, javaObj, 'S', accessible).GetS();
Elliott Hughes33203b52011-09-20 19:42:01 -0700211}
212
Mathieu Chartierc528dba2013-11-26 12:00:11 -0800213static void SetFieldValue(ScopedFastNativeObjectAccess& soa, mirror::Object* o,
Ian Rogers62f05122014-03-21 11:21:29 -0700214 mirror::ArtField* f, Primitive::Type field_type, bool allow_references,
215 const JValue& new_value)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700216 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers62f05122014-03-21 11:21:29 -0700217 DCHECK(f->GetDeclaringClass()->IsInitialized());
218 switch (field_type) {
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700219 case Primitive::kPrimBoolean:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100220 f->SetBoolean<false>(o, new_value.GetZ());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700221 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700222 case Primitive::kPrimByte:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100223 f->SetByte<false>(o, new_value.GetB());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700224 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700225 case Primitive::kPrimChar:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100226 f->SetChar<false>(o, new_value.GetC());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700227 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700228 case Primitive::kPrimDouble:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100229 f->SetDouble<false>(o, new_value.GetD());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700230 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700231 case Primitive::kPrimFloat:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100232 f->SetFloat<false>(o, new_value.GetF());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700233 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700234 case Primitive::kPrimInt:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100235 f->SetInt<false>(o, new_value.GetI());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700236 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700237 case Primitive::kPrimLong:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100238 f->SetLong<false>(o, new_value.GetJ());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700239 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700240 case Primitive::kPrimShort:
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100241 f->SetShort<false>(o, new_value.GetS());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700242 break;
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700243 case Primitive::kPrimNot:
Elliott Hughes33203b52011-09-20 19:42:01 -0700244 if (allow_references) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100245 f->SetObject<false>(o, new_value.GetL());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700246 break;
Elliott Hughes33203b52011-09-20 19:42:01 -0700247 }
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700248 // Else fall through to report an error.
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700249 case Primitive::kPrimVoid:
Elliott Hughes33203b52011-09-20 19:42:01 -0700250 // Never okay.
Ian Rogers62f05122014-03-21 11:21:29 -0700251 ThrowIllegalArgumentException(nullptr, StringPrintf("Not a primitive field: %s",
252 PrettyField(f).c_str()).c_str());
Elliott Hughesfe6207f2011-09-26 17:24:06 -0700253 return;
Elliott Hughes33203b52011-09-20 19:42:01 -0700254 }
Elliott Hughes33203b52011-09-20 19:42:01 -0700255}
256
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700257static void Field_set(JNIEnv* env, jobject javaField, jobject javaObj, jobject javaValue,
258 jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700259 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700260 CHECK(!kMovingFields) << "CheckReceiver may trigger thread suspension for initialization";
261 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
262 // Check that the receiver is non-null and an instance of the field's declaring class.
263 mirror::Object* o = nullptr;
264 if (!CheckReceiver(soa, javaObj, f, &o)) {
265 DCHECK(soa.Self()->IsExceptionPending());
266 return;
267 }
268 Primitive::Type field_prim_type;
269 mirror::Class* field_type;
270 {
271 FieldHelper fh(f);
272 const char* field_type_desciptor = fh.GetTypeDescriptor();
273 field_prim_type = Primitive::GetType(field_type_desciptor[0]);
274 if (field_prim_type == Primitive::kPrimNot) {
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700275 StackHandleScope<1> hs(soa.Self());
276 HandleWrapper<mirror::Object> h(hs.NewHandleWrapper(&o));
Ian Rogers62f05122014-03-21 11:21:29 -0700277 // May cause resolution.
278 CHECK(!kMovingFields) << "Resolution may trigger thread suspension";
279 field_type = fh.GetType(true);
280 if (field_type == nullptr) {
281 DCHECK(soa.Self()->IsExceptionPending());
282 return;
283 }
284 } else {
285 field_type = Runtime::Current()->GetClassLinker()->FindPrimitiveClass(field_type_desciptor[0]);
286 }
287 }
288 // We now don't expect suspension unless an exception is thrown.
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800289 // Unbox the value, if necessary.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800290 mirror::Object* boxed_value = soa.Decode<mirror::Object*>(javaValue);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800291 JValue unboxed_value;
Ian Rogers84956ff2014-03-26 23:52:41 -0700292 if (!UnboxPrimitiveForField(boxed_value, field_type, f, &unboxed_value)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700293 DCHECK(soa.Self()->IsExceptionPending());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800294 return;
295 }
Jeff Haocb4581a2014-03-28 15:43:37 -0700296 // If field is not set to be accessible, verify it can be accessed by the caller.
297 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, true)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700298 DCHECK(soa.Self()->IsExceptionPending());
299 return;
300 }
Ian Rogers62f05122014-03-21 11:21:29 -0700301 SetFieldValue(soa, o, f, field_prim_type, true, unboxed_value);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800302}
303
Elliott Hughes0512f022012-03-15 22:10:52 -0700304static void SetPrimitiveField(JNIEnv* env, jobject javaField, jobject javaObj, char src_descriptor,
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700305 const JValue& new_value, jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700306 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700307 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
308 mirror::Object* o = nullptr;
309 if (!CheckReceiver(soa, javaObj, f, &o)) {
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700310 return;
Elliott Hughes33203b52011-09-20 19:42:01 -0700311 }
Ian Rogers62f05122014-03-21 11:21:29 -0700312 Primitive::Type field_type = FieldHelper(f).GetTypeAsPrimitiveType();
313 if (UNLIKELY(field_type == Primitive::kPrimNot)) {
314 ThrowIllegalArgumentException(nullptr, StringPrintf("Not a primitive field: %s",
315 PrettyField(f).c_str()).c_str());
Jesse Wilsonc129a6b2011-11-24 14:47:46 -0500316 return;
317 }
Elliott Hughes33203b52011-09-20 19:42:01 -0700318
319 // Widen the value if necessary (and possible).
320 JValue wide_value;
Ian Rogers62f05122014-03-21 11:21:29 -0700321 if (!ConvertPrimitiveValue(nullptr, false, Primitive::GetType(src_descriptor),
Ian Rogers84956ff2014-03-26 23:52:41 -0700322 field_type, new_value, &wide_value)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700323 DCHECK(soa.Self()->IsExceptionPending());
Elliott Hughes33203b52011-09-20 19:42:01 -0700324 return;
325 }
326
Jeff Haocb4581a2014-03-28 15:43:37 -0700327 // If field is not set to be accessible, verify it can be accessed by the caller.
328 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, true)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700329 DCHECK(soa.Self()->IsExceptionPending());
330 return;
331 }
332
Elliott Hughes33203b52011-09-20 19:42:01 -0700333 // Write the value.
Ian Rogers62f05122014-03-21 11:21:29 -0700334 SetFieldValue(soa, o, f, field_type, false, wide_value);
Elliott Hughes33203b52011-09-20 19:42:01 -0700335}
336
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700337static void Field_setBoolean(JNIEnv* env, jobject javaField, jobject javaObj, jboolean z,
338 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700339 JValue value;
340 value.SetZ(z);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700341 SetPrimitiveField(env, javaField, javaObj, 'Z', value, accessible);
Elliott Hughes33203b52011-09-20 19:42:01 -0700342}
343
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700344static void Field_setByte(JNIEnv* env, jobject javaField, jobject javaObj, jbyte b,
345 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700346 JValue value;
347 value.SetB(b);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700348 SetPrimitiveField(env, javaField, javaObj, 'B', value, accessible);
Elliott Hughes33203b52011-09-20 19:42:01 -0700349}
350
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700351static void Field_setChar(JNIEnv* env, jobject javaField, jobject javaObj, jchar c,
352 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700353 JValue value;
354 value.SetC(c);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700355 SetPrimitiveField(env, javaField, javaObj, 'C', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800356}
Elliott Hughes33203b52011-09-20 19:42:01 -0700357
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700358static void Field_setDouble(JNIEnv* env, jobject javaField, jobject javaObj, jdouble d,
359 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700360 JValue value;
361 value.SetD(d);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700362 SetPrimitiveField(env, javaField, javaObj, 'D', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800363}
Elliott Hughes33203b52011-09-20 19:42:01 -0700364
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700365static void Field_setFloat(JNIEnv* env, jobject javaField, jobject javaObj, jfloat f,
366 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700367 JValue value;
368 value.SetF(f);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700369 SetPrimitiveField(env, javaField, javaObj, 'F', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800370}
371
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700372static void Field_setInt(JNIEnv* env, jobject javaField, jobject javaObj, jint i,
373 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700374 JValue value;
375 value.SetI(i);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700376 SetPrimitiveField(env, javaField, javaObj, 'I', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800377}
378
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700379static void Field_setLong(JNIEnv* env, jobject javaField, jobject javaObj, jlong j,
380 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700381 JValue value;
382 value.SetJ(j);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700383 SetPrimitiveField(env, javaField, javaObj, 'J', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800384}
385
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700386static void Field_setShort(JNIEnv* env, jobject javaField, jobject javaObj, jshort s,
387 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700388 JValue value;
389 value.SetS(s);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700390 SetPrimitiveField(env, javaField, javaObj, 'S', value, accessible);
Elliott Hughes33203b52011-09-20 19:42:01 -0700391}
392
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700393static JNINativeMethod gMethods[] = {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700394 NATIVE_METHOD(Field, get, "!(Ljava/lang/Object;Z)Ljava/lang/Object;"),
395 NATIVE_METHOD(Field, getBoolean, "!(Ljava/lang/Object;Z)Z"),
396 NATIVE_METHOD(Field, getByte, "!(Ljava/lang/Object;Z)B"),
397 NATIVE_METHOD(Field, getChar, "!(Ljava/lang/Object;Z)C"),
398 NATIVE_METHOD(Field, getDouble, "!(Ljava/lang/Object;Z)D"),
399 NATIVE_METHOD(Field, getFloat, "!(Ljava/lang/Object;Z)F"),
400 NATIVE_METHOD(Field, getInt, "!(Ljava/lang/Object;Z)I"),
401 NATIVE_METHOD(Field, getLong, "!(Ljava/lang/Object;Z)J"),
402 NATIVE_METHOD(Field, getShort, "!(Ljava/lang/Object;Z)S"),
403 NATIVE_METHOD(Field, set, "!(Ljava/lang/Object;Ljava/lang/Object;Z)V"),
404 NATIVE_METHOD(Field, setBoolean, "!(Ljava/lang/Object;ZZ)V"),
405 NATIVE_METHOD(Field, setByte, "!(Ljava/lang/Object;BZ)V"),
406 NATIVE_METHOD(Field, setChar, "!(Ljava/lang/Object;CZ)V"),
407 NATIVE_METHOD(Field, setDouble, "!(Ljava/lang/Object;DZ)V"),
408 NATIVE_METHOD(Field, setFloat, "!(Ljava/lang/Object;FZ)V"),
409 NATIVE_METHOD(Field, setInt, "!(Ljava/lang/Object;IZ)V"),
410 NATIVE_METHOD(Field, setLong, "!(Ljava/lang/Object;JZ)V"),
411 NATIVE_METHOD(Field, setShort, "!(Ljava/lang/Object;SZ)V"),
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700412};
413
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700414void register_java_lang_reflect_Field(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700415 REGISTER_NATIVE_METHODS("java/lang/reflect/Field");
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700416}
417
418} // namespace art