blob: ad88109a86e41cbf62fe3ada24e3d86572a803dd [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"
Ian Rogers22d5e732014-07-15 22:23:51 -070021#include "field_helper.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070022#include "jni_internal.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070023#include "mirror/art_field-inl.h"
Jeff Hao11d5d8f2014-03-26 15:08:20 -070024#include "mirror/art_method-inl.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070025#include "mirror/class-inl.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,
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -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();
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -070096 if ((*f)->IsStatic()) {
97 StackHandleScope<2> hs(soa.Self());
98 HandleWrapper<mirror::ArtField> h_f(hs.NewHandleWrapper(f));
99 Handle<mirror::Class> h_klass(hs.NewHandle((*f)->GetDeclaringClass()));
Ian Rogers7b078e82014-09-10 14:44:24 -0700100 if (UNLIKELY(!Runtime::Current()->GetClassLinker()->EnsureInitialized(soa.Self(), h_klass, true,
101 true))) {
Ian Rogers62f05122014-03-21 11:21:29 -0700102 DCHECK(soa.Self()->IsExceptionPending());
103 *class_or_rcvr = nullptr;
104 return false;
105 }
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700106 *class_or_rcvr = h_klass.Get();
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700107 return true;
108 }
109
Ian Rogers62f05122014-03-21 11:21:29 -0700110 *class_or_rcvr = soa.Decode<mirror::Object*>(j_rcvr);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700111 mirror::Class* declaringClass = (*f)->GetDeclaringClass();
Ian Rogers62f05122014-03-21 11:21:29 -0700112 if (!VerifyObjectIsClass(*class_or_rcvr, declaringClass)) {
113 DCHECK(soa.Self()->IsExceptionPending());
114 *class_or_rcvr = nullptr;
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700115 return false;
116 }
117 return true;
118}
119
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700120static jobject Field_get(JNIEnv* env, jobject javaField, jobject javaObj, jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700121 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700122 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
123 mirror::Object* o = nullptr;
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700124 if (!CheckReceiver(soa, javaObj, &f, &o)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700125 DCHECK(soa.Self()->IsExceptionPending());
126 return nullptr;
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800127 }
Jeff Haocb4581a2014-03-28 15:43:37 -0700128 // If field is not set to be accessible, verify it can be accessed by the caller.
129 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, false)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700130 DCHECK(soa.Self()->IsExceptionPending());
131 return nullptr;
132 }
Ian Rogers62f05122014-03-21 11:21:29 -0700133 // We now don't expect suspension unless an exception is thrown.
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800134 // Get the field's value, boxing if necessary.
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700135 Primitive::Type field_type = f->GetTypeAsPrimitiveType();
Elliott Hughes1d878f32012-04-11 15:17:54 -0700136 JValue value;
Ian Rogers62f05122014-03-21 11:21:29 -0700137 if (!GetFieldValue(soa, o, f, field_type, true, &value)) {
138 DCHECK(soa.Self()->IsExceptionPending());
139 return nullptr;
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800140 }
Ian Rogers62f05122014-03-21 11:21:29 -0700141 return soa.AddLocalReference<jobject>(BoxPrimitive(field_type, value));
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800142}
143
Brian Carlstromea46f952013-07-30 01:26:50 -0700144static JValue GetPrimitiveField(JNIEnv* env, jobject javaField, jobject javaObj,
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700145 char dst_descriptor, jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700146 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700147 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
148 mirror::Object* o = nullptr;
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700149 if (!CheckReceiver(soa, javaObj, &f, &o)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700150 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.
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700162 Primitive::Type field_type = 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 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
261 // Check that the receiver is non-null and an instance of the field's declaring class.
262 mirror::Object* o = nullptr;
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700263 if (!CheckReceiver(soa, javaObj, &f, &o)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700264 DCHECK(soa.Self()->IsExceptionPending());
265 return;
266 }
Ian Rogers62f05122014-03-21 11:21:29 -0700267 mirror::Class* field_type;
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700268 const char* field_type_desciptor = f->GetTypeDescriptor();
269 Primitive::Type field_prim_type = Primitive::GetType(field_type_desciptor[0]);
270 if (field_prim_type == Primitive::kPrimNot) {
271 StackHandleScope<2> hs(soa.Self());
272 HandleWrapper<mirror::Object> h_o(hs.NewHandleWrapper(&o));
273 HandleWrapper<mirror::ArtField> h_f(hs.NewHandleWrapper(&f));
274 FieldHelper fh(h_f);
275 // May cause resolution.
276 field_type = fh.GetType(true);
277 if (field_type == nullptr) {
278 DCHECK(soa.Self()->IsExceptionPending());
279 return;
Ian Rogers62f05122014-03-21 11:21:29 -0700280 }
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700281 } else {
282 field_type = Runtime::Current()->GetClassLinker()->FindPrimitiveClass(field_type_desciptor[0]);
Ian Rogers62f05122014-03-21 11:21:29 -0700283 }
284 // We now don't expect suspension unless an exception is thrown.
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800285 // Unbox the value, if necessary.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800286 mirror::Object* boxed_value = soa.Decode<mirror::Object*>(javaValue);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800287 JValue unboxed_value;
Ian Rogers84956ff2014-03-26 23:52:41 -0700288 if (!UnboxPrimitiveForField(boxed_value, field_type, f, &unboxed_value)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700289 DCHECK(soa.Self()->IsExceptionPending());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800290 return;
291 }
Jeff Haocb4581a2014-03-28 15:43:37 -0700292 // If field is not set to be accessible, verify it can be accessed by the caller.
293 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, true)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700294 DCHECK(soa.Self()->IsExceptionPending());
295 return;
296 }
Ian Rogers62f05122014-03-21 11:21:29 -0700297 SetFieldValue(soa, o, f, field_prim_type, true, unboxed_value);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800298}
299
Elliott Hughes0512f022012-03-15 22:10:52 -0700300static void SetPrimitiveField(JNIEnv* env, jobject javaField, jobject javaObj, char src_descriptor,
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700301 const JValue& new_value, jboolean accessible) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700302 ScopedFastNativeObjectAccess soa(env);
Ian Rogers62f05122014-03-21 11:21:29 -0700303 mirror::ArtField* f = mirror::ArtField::FromReflectedField(soa, javaField);
304 mirror::Object* o = nullptr;
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700305 if (!CheckReceiver(soa, javaObj, &f, &o)) {
Elliott Hughesed1c1e32011-10-02 14:31:05 -0700306 return;
Elliott Hughes33203b52011-09-20 19:42:01 -0700307 }
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700308 Primitive::Type field_type = f->GetTypeAsPrimitiveType();
Ian Rogers62f05122014-03-21 11:21:29 -0700309 if (UNLIKELY(field_type == Primitive::kPrimNot)) {
310 ThrowIllegalArgumentException(nullptr, StringPrintf("Not a primitive field: %s",
311 PrettyField(f).c_str()).c_str());
Jesse Wilsonc129a6b2011-11-24 14:47:46 -0500312 return;
313 }
Elliott Hughes33203b52011-09-20 19:42:01 -0700314
315 // Widen the value if necessary (and possible).
316 JValue wide_value;
Ian Rogers62f05122014-03-21 11:21:29 -0700317 if (!ConvertPrimitiveValue(nullptr, false, Primitive::GetType(src_descriptor),
Ian Rogers84956ff2014-03-26 23:52:41 -0700318 field_type, new_value, &wide_value)) {
Ian Rogers62f05122014-03-21 11:21:29 -0700319 DCHECK(soa.Self()->IsExceptionPending());
Elliott Hughes33203b52011-09-20 19:42:01 -0700320 return;
321 }
322
Jeff Haocb4581a2014-03-28 15:43:37 -0700323 // If field is not set to be accessible, verify it can be accessed by the caller.
324 if ((accessible == JNI_FALSE) && !VerifyFieldAccess(f, o, true)) {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700325 DCHECK(soa.Self()->IsExceptionPending());
326 return;
327 }
328
Elliott Hughes33203b52011-09-20 19:42:01 -0700329 // Write the value.
Ian Rogers62f05122014-03-21 11:21:29 -0700330 SetFieldValue(soa, o, f, field_type, false, wide_value);
Elliott Hughes33203b52011-09-20 19:42:01 -0700331}
332
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700333static void Field_setBoolean(JNIEnv* env, jobject javaField, jobject javaObj, jboolean z,
334 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700335 JValue value;
336 value.SetZ(z);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700337 SetPrimitiveField(env, javaField, javaObj, 'Z', value, accessible);
Elliott Hughes33203b52011-09-20 19:42:01 -0700338}
339
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700340static void Field_setByte(JNIEnv* env, jobject javaField, jobject javaObj, jbyte b,
341 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700342 JValue value;
343 value.SetB(b);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700344 SetPrimitiveField(env, javaField, javaObj, 'B', value, accessible);
Elliott Hughes33203b52011-09-20 19:42:01 -0700345}
346
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700347static void Field_setChar(JNIEnv* env, jobject javaField, jobject javaObj, jchar c,
348 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700349 JValue value;
350 value.SetC(c);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700351 SetPrimitiveField(env, javaField, javaObj, 'C', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800352}
Elliott Hughes33203b52011-09-20 19:42:01 -0700353
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700354static void Field_setDouble(JNIEnv* env, jobject javaField, jobject javaObj, jdouble d,
355 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700356 JValue value;
357 value.SetD(d);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700358 SetPrimitiveField(env, javaField, javaObj, 'D', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800359}
Elliott Hughes33203b52011-09-20 19:42:01 -0700360
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700361static void Field_setFloat(JNIEnv* env, jobject javaField, jobject javaObj, jfloat f,
362 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700363 JValue value;
364 value.SetF(f);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700365 SetPrimitiveField(env, javaField, javaObj, 'F', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800366}
367
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700368static void Field_setInt(JNIEnv* env, jobject javaField, jobject javaObj, jint i,
369 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700370 JValue value;
371 value.SetI(i);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700372 SetPrimitiveField(env, javaField, javaObj, 'I', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800373}
374
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700375static void Field_setLong(JNIEnv* env, jobject javaField, jobject javaObj, jlong j,
376 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700377 JValue value;
378 value.SetJ(j);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700379 SetPrimitiveField(env, javaField, javaObj, 'J', value, accessible);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800380}
381
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700382static void Field_setShort(JNIEnv* env, jobject javaField, jobject javaObj, jshort s,
383 jboolean accessible) {
Elliott Hughesf24d3ce2012-04-11 17:43:37 -0700384 JValue value;
385 value.SetS(s);
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700386 SetPrimitiveField(env, javaField, javaObj, 'S', value, accessible);
Elliott Hughes33203b52011-09-20 19:42:01 -0700387}
388
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700389static JNINativeMethod gMethods[] = {
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700390 NATIVE_METHOD(Field, get, "!(Ljava/lang/Object;Z)Ljava/lang/Object;"),
391 NATIVE_METHOD(Field, getBoolean, "!(Ljava/lang/Object;Z)Z"),
392 NATIVE_METHOD(Field, getByte, "!(Ljava/lang/Object;Z)B"),
393 NATIVE_METHOD(Field, getChar, "!(Ljava/lang/Object;Z)C"),
394 NATIVE_METHOD(Field, getDouble, "!(Ljava/lang/Object;Z)D"),
395 NATIVE_METHOD(Field, getFloat, "!(Ljava/lang/Object;Z)F"),
396 NATIVE_METHOD(Field, getInt, "!(Ljava/lang/Object;Z)I"),
397 NATIVE_METHOD(Field, getLong, "!(Ljava/lang/Object;Z)J"),
398 NATIVE_METHOD(Field, getShort, "!(Ljava/lang/Object;Z)S"),
399 NATIVE_METHOD(Field, set, "!(Ljava/lang/Object;Ljava/lang/Object;Z)V"),
400 NATIVE_METHOD(Field, setBoolean, "!(Ljava/lang/Object;ZZ)V"),
401 NATIVE_METHOD(Field, setByte, "!(Ljava/lang/Object;BZ)V"),
402 NATIVE_METHOD(Field, setChar, "!(Ljava/lang/Object;CZ)V"),
403 NATIVE_METHOD(Field, setDouble, "!(Ljava/lang/Object;DZ)V"),
404 NATIVE_METHOD(Field, setFloat, "!(Ljava/lang/Object;FZ)V"),
405 NATIVE_METHOD(Field, setInt, "!(Ljava/lang/Object;IZ)V"),
406 NATIVE_METHOD(Field, setLong, "!(Ljava/lang/Object;JZ)V"),
407 NATIVE_METHOD(Field, setShort, "!(Ljava/lang/Object;SZ)V"),
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700408};
409
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700410void register_java_lang_reflect_Field(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700411 REGISTER_NATIVE_METHODS("java/lang/reflect/Field");
Brian Carlstromf867b6f2011-09-16 12:17:25 -0700412}
413
414} // namespace art