blob: b657aec30dc52db2db865006cca37bc48c59ba67 [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"
Hiroshi Yamauchi02d2f292015-04-03 13:35:16 -070027#include "mirror/field-inl.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
Vladimir Markoc1363122015-04-09 14:13:13 +010094static jobject Class_findOverriddenMethodIfProxy(JNIEnv* env, jclass, jobject art_method) {
95 ScopedFastNativeObjectAccess soa(env);
96 mirror::ArtMethod* method = soa.Decode<mirror::ArtMethod*>(art_method);
97 mirror::Class* declaring_klass = method->GetDeclaringClass();
98 if (!declaring_klass->IsProxyClass()) {
99 return art_method;
100 }
101 uint32_t dex_method_index = method->GetDexMethodIndex();
102 mirror::ArtMethod* overriden_method = method->GetDexCacheResolvedMethods()->Get(dex_method_index);
103 return soa.AddLocalReference<jobject>(overriden_method);
104}
105
Elliott Hughes0512f022012-03-15 22:10:52 -0700106static jstring Class_getNameNative(JNIEnv* env, jobject javaThis) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700107 ScopedFastNativeObjectAccess soa(env);
Mathieu Chartierf8322842014-05-16 10:59:25 -0700108 StackHandleScope<1> hs(soa.Self());
109 mirror::Class* const c = DecodeClass(soa, javaThis);
110 return soa.AddLocalReference<jstring>(mirror::Class::ComputeName(hs.NewHandle(c)));
Elliott Hughes6bdc3b22011-09-16 19:24:10 -0700111}
112
Elliott Hughes2ed52c42012-03-21 16:56:56 -0700113static jobjectArray Class_getProxyInterfaces(JNIEnv* env, jobject javaThis) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700114 ScopedFastNativeObjectAccess soa(env);
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700115 mirror::Class* c = DecodeClass(soa, javaThis);
Ian Rogers50b35e22012-10-04 10:09:15 -0700116 return soa.AddLocalReference<jobjectArray>(c->GetInterfaces()->Clone(soa.Self()));
Elliott Hughes2ed52c42012-03-21 16:56:56 -0700117}
118
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700119static mirror::ObjectArray<mirror::Field>* GetDeclaredFields(
120 Thread* self, mirror::Class* klass, bool public_only, bool force_resolve)
121 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
122 StackHandleScope<3> hs(self);
123 auto h_ifields = hs.NewHandle(klass->GetIFields());
124 auto h_sfields = hs.NewHandle(klass->GetSFields());
125 const int32_t num_ifields = h_ifields.Get() != nullptr ? h_ifields->GetLength() : 0;
126 const int32_t num_sfields = h_sfields.Get() != nullptr ? h_sfields->GetLength() : 0;
127 int32_t array_size = num_ifields + num_sfields;
128 if (public_only) {
129 // Lets go subtract all the non public fields.
130 for (int32_t i = 0; i < num_ifields; ++i) {
131 if (!h_ifields->GetWithoutChecks(i)->IsPublic()) {
132 --array_size;
133 }
134 }
135 for (int32_t i = 0; i < num_sfields; ++i) {
136 if (!h_sfields->GetWithoutChecks(i)->IsPublic()) {
137 --array_size;
138 }
139 }
140 }
141 int32_t array_idx = 0;
142 auto object_array = hs.NewHandle(mirror::ObjectArray<mirror::Field>::Alloc(
143 self, mirror::Field::ArrayClass(), array_size));
144 if (object_array.Get() == nullptr) {
145 return nullptr;
146 }
147 for (int32_t i = 0; i < num_ifields; ++i) {
148 auto* art_field = h_ifields->GetWithoutChecks(i);
149 if (!public_only || art_field->IsPublic()) {
150 auto* field = mirror::Field::CreateFromArtField(self, art_field, force_resolve);
151 if (field == nullptr) {
152 if (kIsDebugBuild) {
153 self->AssertPendingException();
154 }
155 // Maybe null due to OOME or type resolving exception.
156 return nullptr;
157 }
158 object_array->SetWithoutChecks<false>(array_idx++, field);
159 }
160 }
161 for (int32_t i = 0; i < num_sfields; ++i) {
162 auto* art_field = h_sfields->GetWithoutChecks(i);
163 if (!public_only || art_field->IsPublic()) {
164 auto* field = mirror::Field::CreateFromArtField(self, art_field, force_resolve);
165 if (field == nullptr) {
166 if (kIsDebugBuild) {
167 self->AssertPendingException();
168 }
169 return nullptr;
170 }
171 object_array->SetWithoutChecks<false>(array_idx++, field);
172 }
173 }
174 CHECK_EQ(array_idx, array_size);
175 return object_array.Get();
176}
177
178static jobjectArray Class_getDeclaredFieldsUnchecked(JNIEnv* env, jobject javaThis,
179 jboolean publicOnly) {
180 ScopedFastNativeObjectAccess soa(env);
181 return soa.AddLocalReference<jobjectArray>(
182 GetDeclaredFields(soa.Self(), DecodeClass(soa, javaThis), publicOnly != JNI_FALSE, false));
183}
184
185static jobjectArray Class_getDeclaredFields(JNIEnv* env, jobject javaThis) {
186 ScopedFastNativeObjectAccess soa(env);
187 return soa.AddLocalReference<jobjectArray>(
188 GetDeclaredFields(soa.Self(), DecodeClass(soa, javaThis), false, true));
189}
190
191static jobjectArray Class_getPublicDeclaredFields(JNIEnv* env, jobject javaThis) {
192 ScopedFastNativeObjectAccess soa(env);
193 return soa.AddLocalReference<jobjectArray>(
194 GetDeclaredFields(soa.Self(), DecodeClass(soa, javaThis), true, true));
195}
196
197// Performs a binary search through an array of fields, TODO: Is this fast enough if we don't use
198// the dex cache for lookups? I think CompareModifiedUtf8ToUtf16AsCodePointValues should be fairly
199// fast.
200ALWAYS_INLINE static inline mirror::ArtField* FindFieldByName(
201 Thread* self ATTRIBUTE_UNUSED, mirror::String* name,
202 mirror::ObjectArray<mirror::ArtField>* fields)
203 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
204 uint32_t low = 0;
205 uint32_t high = fields->GetLength();
206 const uint16_t* const data = name->GetCharArray()->GetData() + name->GetOffset();
207 const size_t length = name->GetLength();
208 while (low < high) {
209 auto mid = (low + high) / 2;
210 mirror::ArtField* const field = fields->GetWithoutChecks(mid);
211 int result = CompareModifiedUtf8ToUtf16AsCodePointValues(field->GetName(), data, length);
212 // Alternate approach, only a few % faster at the cost of more allocations.
213 // int result = field->GetStringName(self, true)->CompareTo(name);
214 if (result < 0) {
215 low = mid + 1;
216 } else if (result > 0) {
217 high = mid;
218 } else {
219 return field;
220 }
221 }
222 if (kIsDebugBuild) {
223 for (int32_t i = 0; i < fields->GetLength(); ++i) {
224 CHECK_NE(fields->GetWithoutChecks(i)->GetName(), name->ToModifiedUtf8());
225 }
226 }
227 return nullptr;
228}
229
230ALWAYS_INLINE static inline mirror::Field* GetDeclaredField(
231 Thread* self, mirror::Class* c, mirror::String* name)
232 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
233 auto* instance_fields = c->GetIFields();
234 if (instance_fields != nullptr) {
235 auto* art_field = FindFieldByName(self, name, instance_fields);
236 if (art_field != nullptr) {
237 return mirror::Field::CreateFromArtField(self, art_field, true);
238 }
239 }
240 auto* static_fields = c->GetSFields();
241 if (static_fields != nullptr) {
242 auto* art_field = FindFieldByName(self, name, static_fields);
243 if (art_field != nullptr) {
244 return mirror::Field::CreateFromArtField(self, art_field, true);
245 }
246 }
247 return nullptr;
248}
249
250static jobject Class_getDeclaredFieldInternal(JNIEnv* env, jobject javaThis, jstring name) {
251 ScopedFastNativeObjectAccess soa(env);
252 auto* name_string = soa.Decode<mirror::String*>(name);
253 return soa.AddLocalReference<jobject>(
254 GetDeclaredField(soa.Self(), DecodeClass(soa, javaThis), name_string));
255}
256
257static jobject Class_getDeclaredField(JNIEnv* env, jobject javaThis, jstring name) {
258 ScopedFastNativeObjectAccess soa(env);
259 auto* name_string = soa.Decode<mirror::String*>(name);
260 if (name == nullptr) {
261 ThrowNullPointerException("name == null");
262 return nullptr;
263 }
264 auto* klass = DecodeClass(soa, javaThis);
265 mirror::Field* result = GetDeclaredField(soa.Self(), klass, name_string);
266 if (result == nullptr) {
267 std::string name_str = name_string->ToModifiedUtf8();
Mathieu Chartierca239af2015-03-29 18:27:50 -0700268 // We may have a pending exception if we failed to resolve.
269 if (!soa.Self()->IsExceptionPending()) {
270 soa.Self()->ThrowNewException("Ljava/lang/NoSuchFieldException;", name_str.c_str());
271 }
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700272 return nullptr;
273 }
274 return soa.AddLocalReference<jobject>(result);
275}
276
Elliott Hughesd369bb72011-09-12 14:41:14 -0700277static JNINativeMethod gMethods[] = {
Ian Rogers53b8b092014-03-13 23:45:53 -0700278 NATIVE_METHOD(Class, classForName, "!(Ljava/lang/String;ZLjava/lang/ClassLoader;)Ljava/lang/Class;"),
Vladimir Markoc1363122015-04-09 14:13:13 +0100279 NATIVE_METHOD(Class, findOverriddenMethodIfProxy,
280 "!(Ljava/lang/reflect/ArtMethod;)Ljava/lang/reflect/ArtMethod;"),
Ian Rogers1eb512d2013-10-18 15:42:20 -0700281 NATIVE_METHOD(Class, getNameNative, "!()Ljava/lang/String;"),
282 NATIVE_METHOD(Class, getProxyInterfaces, "!()[Ljava/lang/Class;"),
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700283 NATIVE_METHOD(Class, getDeclaredFields, "!()[Ljava/lang/reflect/Field;"),
284 NATIVE_METHOD(Class, getPublicDeclaredFields, "!()[Ljava/lang/reflect/Field;"),
285 NATIVE_METHOD(Class, getDeclaredFieldsUnchecked, "!(Z)[Ljava/lang/reflect/Field;"),
286 NATIVE_METHOD(Class, getDeclaredFieldInternal, "!(Ljava/lang/String;)Ljava/lang/reflect/Field;"),
287 NATIVE_METHOD(Class, getDeclaredField, "!(Ljava/lang/String;)Ljava/lang/reflect/Field;"),
Elliott Hughesd369bb72011-09-12 14:41:14 -0700288};
289
Elliott Hughesd369bb72011-09-12 14:41:14 -0700290void register_java_lang_Class(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700291 REGISTER_NATIVE_METHODS("java/lang/Class");
Elliott Hughesd369bb72011-09-12 14:41:14 -0700292}
293
294} // namespace art