blob: 0ca9d24824f8423e6fdf92efc561df068febbb3d [file] [log] [blame]
Elliott Hughesd369bb72011-09-12 14:41:14 -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
Andreas Gampe277ccbd2014-11-03 21:36:10 -080017#include "java_lang_Class.h"
18
Elliott Hughesd369bb72011-09-12 14:41:14 -070019#include "class_linker.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070020#include "common_throws.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070021#include "dex_file-inl.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070022#include "jni_internal.h"
Elliott Hughes6a144332012-04-03 13:07:11 -070023#include "nth_caller_visitor.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070024#include "mirror/art_field-inl.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070025#include "mirror/class-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080026#include "mirror/class_loader.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070027#include "mirror/field.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080028#include "mirror/object-inl.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070029#include "mirror/object_array-inl.h"
30#include "mirror/string-inl.h"
Ian Rogers00f7d0e2012-07-19 15:28:27 -070031#include "scoped_thread_state_change.h"
Ian Rogers1eb512d2013-10-18 15:42:20 -070032#include "scoped_fast_native_object_access.h"
Elliott Hughes80609252011-09-23 17:24:51 -070033#include "ScopedLocalRef.h"
Brian Carlstromf91c8c32011-09-21 17:30:34 -070034#include "ScopedUtfChars.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070035#include "utf.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070036#include "well_known_classes.h"
Elliott Hughesd369bb72011-09-12 14:41:14 -070037
38namespace art {
39
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070040ALWAYS_INLINE static inline mirror::Class* DecodeClass(
41 const ScopedFastNativeObjectAccess& soa, jobject java_class)
Ian Rogersb726dcb2012-09-05 08:57:23 -070042 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080043 mirror::Class* c = soa.Decode<mirror::Class*>(java_class);
Elliott Hughes15216932012-03-21 21:53:06 -070044 DCHECK(c != NULL);
45 DCHECK(c->IsClass());
Elliott Hughes923e8b82012-03-23 11:44:07 -070046 // TODO: we could EnsureInitialized here, rather than on every reflective get/set or invoke .
47 // For now, we conservatively preserve the old dalvik behavior. A quick "IsInitialized" check
48 // every time probably doesn't make much difference to reflection performance anyway.
49 return c;
Elliott Hughes15216932012-03-21 21:53:06 -070050}
51
Brian Carlstromf91c8c32011-09-21 17:30:34 -070052// "name" is in "binary name" format, e.g. "dalvik.system.Debug$1".
Ian Rogers98379392014-02-24 16:53:16 -080053static jclass Class_classForName(JNIEnv* env, jclass, jstring javaName, jboolean initialize,
54 jobject javaLoader) {
Ian Rogers53b8b092014-03-13 23:45:53 -070055 ScopedFastNativeObjectAccess soa(env);
Brian Carlstromf91c8c32011-09-21 17:30:34 -070056 ScopedUtfChars name(env, javaName);
Mathieu Chartierc528dba2013-11-26 12:00:11 -080057 if (name.c_str() == nullptr) {
58 return nullptr;
Brian Carlstromf91c8c32011-09-21 17:30:34 -070059 }
60
61 // We need to validate and convert the name (from x.y.z to x/y/z). This
62 // is especially handy for array types, since we want to avoid
63 // auto-generating bogus array classes.
Elliott Hughes906e6852011-10-28 14:52:10 -070064 if (!IsValidBinaryClassName(name.c_str())) {
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +000065 soa.Self()->ThrowNewExceptionF("Ljava/lang/ClassNotFoundException;",
Ian Rogers62d6c772013-02-27 08:32:07 -080066 "Invalid name: %s", name.c_str());
Mathieu Chartierc528dba2013-11-26 12:00:11 -080067 return nullptr;
Brian Carlstromf91c8c32011-09-21 17:30:34 -070068 }
69
70 std::string descriptor(DotToDescriptor(name.c_str()));
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070071 StackHandleScope<2> hs(soa.Self());
72 Handle<mirror::ClassLoader> class_loader(hs.NewHandle(soa.Decode<mirror::ClassLoader*>(javaLoader)));
Brian Carlstromf91c8c32011-09-21 17:30:34 -070073 ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070074 Handle<mirror::Class> c(
75 hs.NewHandle(class_linker->FindClass(soa.Self(), descriptor.c_str(), class_loader)));
76 if (c.Get() == nullptr) {
Elliott Hughes844f9a02012-01-24 20:19:58 -080077 ScopedLocalRef<jthrowable> cause(env, env->ExceptionOccurred());
Brian Carlstrom395520e2011-09-25 19:35:00 -070078 env->ExceptionClear();
Elliott Hugheseac76672012-05-24 21:56:51 -070079 jthrowable cnfe = reinterpret_cast<jthrowable>(env->NewObject(WellKnownClasses::java_lang_ClassNotFoundException,
80 WellKnownClasses::java_lang_ClassNotFoundException_init,
81 javaName, cause.get()));
Ian Rogersc114b5f2014-07-21 08:55:01 -070082 if (cnfe != nullptr) {
83 // Make sure allocation didn't fail with an OOME.
84 env->Throw(cnfe);
85 }
Mathieu Chartierc528dba2013-11-26 12:00:11 -080086 return nullptr;
Brian Carlstrom395520e2011-09-25 19:35:00 -070087 }
Brian Carlstromf91c8c32011-09-21 17:30:34 -070088 if (initialize) {
Ian Rogers7b078e82014-09-10 14:44:24 -070089 class_linker->EnsureInitialized(soa.Self(), c, true, true);
Brian Carlstromf91c8c32011-09-21 17:30:34 -070090 }
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070091 return soa.AddLocalReference<jclass>(c.Get());
Brian Carlstromf91c8c32011-09-21 17:30:34 -070092}
93
Elliott Hughes0512f022012-03-15 22:10:52 -070094static jstring Class_getNameNative(JNIEnv* env, jobject javaThis) {
Ian Rogers1eb512d2013-10-18 15:42:20 -070095 ScopedFastNativeObjectAccess soa(env);
Mathieu Chartierf8322842014-05-16 10:59:25 -070096 StackHandleScope<1> hs(soa.Self());
97 mirror::Class* const c = DecodeClass(soa, javaThis);
98 return soa.AddLocalReference<jstring>(mirror::Class::ComputeName(hs.NewHandle(c)));
Elliott Hughes6bdc3b22011-09-16 19:24:10 -070099}
100
Elliott Hughes2ed52c42012-03-21 16:56:56 -0700101static jobjectArray Class_getProxyInterfaces(JNIEnv* env, jobject javaThis) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700102 ScopedFastNativeObjectAccess soa(env);
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700103 mirror::Class* c = DecodeClass(soa, javaThis);
Ian Rogers50b35e22012-10-04 10:09:15 -0700104 return soa.AddLocalReference<jobjectArray>(c->GetInterfaces()->Clone(soa.Self()));
Elliott Hughes2ed52c42012-03-21 16:56:56 -0700105}
106
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700107static mirror::ObjectArray<mirror::Field>* GetDeclaredFields(
108 Thread* self, mirror::Class* klass, bool public_only, bool force_resolve)
109 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
110 StackHandleScope<3> hs(self);
111 auto h_ifields = hs.NewHandle(klass->GetIFields());
112 auto h_sfields = hs.NewHandle(klass->GetSFields());
113 const int32_t num_ifields = h_ifields.Get() != nullptr ? h_ifields->GetLength() : 0;
114 const int32_t num_sfields = h_sfields.Get() != nullptr ? h_sfields->GetLength() : 0;
115 int32_t array_size = num_ifields + num_sfields;
116 if (public_only) {
117 // Lets go subtract all the non public fields.
118 for (int32_t i = 0; i < num_ifields; ++i) {
119 if (!h_ifields->GetWithoutChecks(i)->IsPublic()) {
120 --array_size;
121 }
122 }
123 for (int32_t i = 0; i < num_sfields; ++i) {
124 if (!h_sfields->GetWithoutChecks(i)->IsPublic()) {
125 --array_size;
126 }
127 }
128 }
129 int32_t array_idx = 0;
130 auto object_array = hs.NewHandle(mirror::ObjectArray<mirror::Field>::Alloc(
131 self, mirror::Field::ArrayClass(), array_size));
132 if (object_array.Get() == nullptr) {
133 return nullptr;
134 }
135 for (int32_t i = 0; i < num_ifields; ++i) {
136 auto* art_field = h_ifields->GetWithoutChecks(i);
137 if (!public_only || art_field->IsPublic()) {
138 auto* field = mirror::Field::CreateFromArtField(self, art_field, force_resolve);
139 if (field == nullptr) {
140 if (kIsDebugBuild) {
141 self->AssertPendingException();
142 }
143 // Maybe null due to OOME or type resolving exception.
144 return nullptr;
145 }
146 object_array->SetWithoutChecks<false>(array_idx++, field);
147 }
148 }
149 for (int32_t i = 0; i < num_sfields; ++i) {
150 auto* art_field = h_sfields->GetWithoutChecks(i);
151 if (!public_only || art_field->IsPublic()) {
152 auto* field = mirror::Field::CreateFromArtField(self, art_field, force_resolve);
153 if (field == nullptr) {
154 if (kIsDebugBuild) {
155 self->AssertPendingException();
156 }
157 return nullptr;
158 }
159 object_array->SetWithoutChecks<false>(array_idx++, field);
160 }
161 }
162 CHECK_EQ(array_idx, array_size);
163 return object_array.Get();
164}
165
166static jobjectArray Class_getDeclaredFieldsUnchecked(JNIEnv* env, jobject javaThis,
167 jboolean publicOnly) {
168 ScopedFastNativeObjectAccess soa(env);
169 return soa.AddLocalReference<jobjectArray>(
170 GetDeclaredFields(soa.Self(), DecodeClass(soa, javaThis), publicOnly != JNI_FALSE, false));
171}
172
173static jobjectArray Class_getDeclaredFields(JNIEnv* env, jobject javaThis) {
174 ScopedFastNativeObjectAccess soa(env);
175 return soa.AddLocalReference<jobjectArray>(
176 GetDeclaredFields(soa.Self(), DecodeClass(soa, javaThis), false, true));
177}
178
179static jobjectArray Class_getPublicDeclaredFields(JNIEnv* env, jobject javaThis) {
180 ScopedFastNativeObjectAccess soa(env);
181 return soa.AddLocalReference<jobjectArray>(
182 GetDeclaredFields(soa.Self(), DecodeClass(soa, javaThis), true, true));
183}
184
185// Performs a binary search through an array of fields, TODO: Is this fast enough if we don't use
186// the dex cache for lookups? I think CompareModifiedUtf8ToUtf16AsCodePointValues should be fairly
187// fast.
188ALWAYS_INLINE static inline mirror::ArtField* FindFieldByName(
189 Thread* self ATTRIBUTE_UNUSED, mirror::String* name,
190 mirror::ObjectArray<mirror::ArtField>* fields)
191 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
192 uint32_t low = 0;
193 uint32_t high = fields->GetLength();
194 const uint16_t* const data = name->GetCharArray()->GetData() + name->GetOffset();
195 const size_t length = name->GetLength();
196 while (low < high) {
197 auto mid = (low + high) / 2;
198 mirror::ArtField* const field = fields->GetWithoutChecks(mid);
199 int result = CompareModifiedUtf8ToUtf16AsCodePointValues(field->GetName(), data, length);
200 // Alternate approach, only a few % faster at the cost of more allocations.
201 // int result = field->GetStringName(self, true)->CompareTo(name);
202 if (result < 0) {
203 low = mid + 1;
204 } else if (result > 0) {
205 high = mid;
206 } else {
207 return field;
208 }
209 }
210 if (kIsDebugBuild) {
211 for (int32_t i = 0; i < fields->GetLength(); ++i) {
212 CHECK_NE(fields->GetWithoutChecks(i)->GetName(), name->ToModifiedUtf8());
213 }
214 }
215 return nullptr;
216}
217
218ALWAYS_INLINE static inline mirror::Field* GetDeclaredField(
219 Thread* self, mirror::Class* c, mirror::String* name)
220 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
221 auto* instance_fields = c->GetIFields();
222 if (instance_fields != nullptr) {
223 auto* art_field = FindFieldByName(self, name, instance_fields);
224 if (art_field != nullptr) {
225 return mirror::Field::CreateFromArtField(self, art_field, true);
226 }
227 }
228 auto* static_fields = c->GetSFields();
229 if (static_fields != nullptr) {
230 auto* art_field = FindFieldByName(self, name, static_fields);
231 if (art_field != nullptr) {
232 return mirror::Field::CreateFromArtField(self, art_field, true);
233 }
234 }
235 return nullptr;
236}
237
238static jobject Class_getDeclaredFieldInternal(JNIEnv* env, jobject javaThis, jstring name) {
239 ScopedFastNativeObjectAccess soa(env);
240 auto* name_string = soa.Decode<mirror::String*>(name);
241 return soa.AddLocalReference<jobject>(
242 GetDeclaredField(soa.Self(), DecodeClass(soa, javaThis), name_string));
243}
244
245static jobject Class_getDeclaredField(JNIEnv* env, jobject javaThis, jstring name) {
246 ScopedFastNativeObjectAccess soa(env);
247 auto* name_string = soa.Decode<mirror::String*>(name);
248 if (name == nullptr) {
249 ThrowNullPointerException("name == null");
250 return nullptr;
251 }
252 auto* klass = DecodeClass(soa, javaThis);
253 mirror::Field* result = GetDeclaredField(soa.Self(), klass, name_string);
254 if (result == nullptr) {
255 std::string name_str = name_string->ToModifiedUtf8();
Mathieu Chartierca239af2015-03-29 18:27:50 -0700256 // We may have a pending exception if we failed to resolve.
257 if (!soa.Self()->IsExceptionPending()) {
258 soa.Self()->ThrowNewException("Ljava/lang/NoSuchFieldException;", name_str.c_str());
259 }
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700260 return nullptr;
261 }
262 return soa.AddLocalReference<jobject>(result);
263}
264
Elliott Hughesd369bb72011-09-12 14:41:14 -0700265static JNINativeMethod gMethods[] = {
Ian Rogers53b8b092014-03-13 23:45:53 -0700266 NATIVE_METHOD(Class, classForName, "!(Ljava/lang/String;ZLjava/lang/ClassLoader;)Ljava/lang/Class;"),
Ian Rogers1eb512d2013-10-18 15:42:20 -0700267 NATIVE_METHOD(Class, getNameNative, "!()Ljava/lang/String;"),
268 NATIVE_METHOD(Class, getProxyInterfaces, "!()[Ljava/lang/Class;"),
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700269 NATIVE_METHOD(Class, getDeclaredFields, "!()[Ljava/lang/reflect/Field;"),
270 NATIVE_METHOD(Class, getPublicDeclaredFields, "!()[Ljava/lang/reflect/Field;"),
271 NATIVE_METHOD(Class, getDeclaredFieldsUnchecked, "!(Z)[Ljava/lang/reflect/Field;"),
272 NATIVE_METHOD(Class, getDeclaredFieldInternal, "!(Ljava/lang/String;)Ljava/lang/reflect/Field;"),
273 NATIVE_METHOD(Class, getDeclaredField, "!(Ljava/lang/String;)Ljava/lang/reflect/Field;"),
Elliott Hughesd369bb72011-09-12 14:41:14 -0700274};
275
Elliott Hughesd369bb72011-09-12 14:41:14 -0700276void register_java_lang_Class(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700277 REGISTER_NATIVE_METHODS("java/lang/Class");
Elliott Hughesd369bb72011-09-12 14:41:14 -0700278}
279
280} // namespace art