blob: d1687d7874a3c7b061024803193d40d469b4c3ac [file] [log] [blame]
Elliott Hughes2faa5f12012-01-30 14:42:07 -08001/*
2 * Copyright (C) 2011 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 */
Elliott Hughes0c9cd562011-08-12 10:59:29 -070016
Carl Shapiro9b9ba282011-08-14 15:30:39 -070017#include "jni_internal.h"
Elliott Hughes0c9cd562011-08-12 10:59:29 -070018
Mathieu Chartiere401d142015-04-22 13:56:20 -070019#include "art_method-inl.h"
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080020#include "common_compiler_test.h"
Ian Rogers68d8b422014-07-17 11:09:10 -070021#include "java_vm_ext.h"
Fred Shih56890e22014-06-02 11:11:52 -070022#include "mirror/string-inl.h"
Ian Rogerse63db272014-07-15 15:36:11 -070023#include "scoped_thread_state_change.h"
Elliott Hughes726079d2011-10-07 18:43:44 -070024#include "ScopedLocalRef.h"
Elliott Hughes0c9cd562011-08-12 10:59:29 -070025
26namespace art {
27
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080028// TODO: Convert to CommonRuntimeTest. Currently MakeExecutable is used.
29class JniInternalTest : public CommonCompilerTest {
Elliott Hughesc7ac37f2011-08-12 12:21:58 -070030 protected:
31 virtual void SetUp() {
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080032 CommonCompilerTest::SetUp();
Elliott Hughes5174fe62011-08-23 15:12:35 -070033
Elliott Hughesa2501992011-08-26 19:39:54 -070034 vm_ = Runtime::Current()->GetJavaVM();
35
Elliott Hughes5174fe62011-08-23 15:12:35 -070036 // Turn on -verbose:jni for the JNI tests.
Ian Rogers79713632013-08-21 19:06:15 -070037 // gLogVerbosity.jni = true;
Elliott Hughes5174fe62011-08-23 15:12:35 -070038
Ian Rogers2d10b202014-05-12 19:15:18 -070039 vm_->AttachCurrentThread(&env_, nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -070040
Brian Carlstromea46f952013-07-30 01:26:50 -070041 ScopedLocalRef<jclass> aioobe(env_,
42 env_->FindClass("java/lang/ArrayIndexOutOfBoundsException"));
Ian Rogers2d10b202014-05-12 19:15:18 -070043 CHECK(aioobe.get() != nullptr);
Elliott Hughes726079d2011-10-07 18:43:44 -070044 aioobe_ = reinterpret_cast<jclass>(env_->NewGlobalRef(aioobe.get()));
Elliott Hughesb465ab02011-08-24 11:21:21 -070045
Elliott Hughesb264f082012-04-06 17:10:10 -070046 ScopedLocalRef<jclass> ase(env_, env_->FindClass("java/lang/ArrayStoreException"));
Ian Rogers2d10b202014-05-12 19:15:18 -070047 CHECK(ase.get() != nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -070048 ase_ = reinterpret_cast<jclass>(env_->NewGlobalRef(ase.get()));
49
Brian Carlstromea46f952013-07-30 01:26:50 -070050 ScopedLocalRef<jclass> sioobe(env_,
51 env_->FindClass("java/lang/StringIndexOutOfBoundsException"));
Ian Rogers2d10b202014-05-12 19:15:18 -070052 CHECK(sioobe.get() != nullptr);
Elliott Hughes726079d2011-10-07 18:43:44 -070053 sioobe_ = reinterpret_cast<jclass>(env_->NewGlobalRef(sioobe.get()));
54 }
55
Ian Rogers2d10b202014-05-12 19:15:18 -070056 void ExpectException(jclass exception_class) {
Ian Rogers68d8b422014-07-17 11:09:10 -070057 ScopedObjectAccess soa(env_);
58 EXPECT_TRUE(env_->ExceptionCheck())
59 << PrettyDescriptor(soa.Decode<mirror::Class*>(exception_class));
Ian Rogers2d10b202014-05-12 19:15:18 -070060 jthrowable exception = env_->ExceptionOccurred();
61 EXPECT_NE(nullptr, exception);
62 env_->ExceptionClear();
63 EXPECT_TRUE(env_->IsInstanceOf(exception, exception_class));
64 }
65
Brian Carlstrom4d571432012-05-16 00:21:41 -070066 void CleanUpJniEnv() {
Ian Rogers2d10b202014-05-12 19:15:18 -070067 if (aioobe_ != nullptr) {
Brian Carlstrom4d571432012-05-16 00:21:41 -070068 env_->DeleteGlobalRef(aioobe_);
Ian Rogers2d10b202014-05-12 19:15:18 -070069 aioobe_ = nullptr;
Brian Carlstrom4d571432012-05-16 00:21:41 -070070 }
Ian Rogers2d10b202014-05-12 19:15:18 -070071 if (ase_ != nullptr) {
Brian Carlstrom4d571432012-05-16 00:21:41 -070072 env_->DeleteGlobalRef(ase_);
Ian Rogers2d10b202014-05-12 19:15:18 -070073 ase_ = nullptr;
Brian Carlstrom4d571432012-05-16 00:21:41 -070074 }
Ian Rogers2d10b202014-05-12 19:15:18 -070075 if (sioobe_ != nullptr) {
Brian Carlstrom4d571432012-05-16 00:21:41 -070076 env_->DeleteGlobalRef(sioobe_);
Ian Rogers2d10b202014-05-12 19:15:18 -070077 sioobe_ = nullptr;
Brian Carlstrom4d571432012-05-16 00:21:41 -070078 }
79 }
80
Ian Rogers53b8b092014-03-13 23:45:53 -070081 virtual void TearDown() OVERRIDE {
Brian Carlstrom4d571432012-05-16 00:21:41 -070082 CleanUpJniEnv();
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080083 CommonCompilerTest::TearDown();
Elliott Hughesc7ac37f2011-08-12 12:21:58 -070084 }
Elliott Hughesb465ab02011-08-24 11:21:21 -070085
Ian Rogers1d99e452014-01-02 17:36:41 -080086 jclass GetPrimitiveClass(char descriptor) {
87 ScopedObjectAccess soa(env_);
88 mirror::Class* c = class_linker_->FindPrimitiveClass(descriptor);
89 CHECK(c != nullptr);
90 return soa.AddLocalReference<jclass>(c);
91 }
92
Ian Rogers68d8b422014-07-17 11:09:10 -070093 void ExpectClassFound(const char* name) {
94 EXPECT_NE(env_->FindClass(name), nullptr) << name;
95 EXPECT_FALSE(env_->ExceptionCheck()) << name;
96 }
97
98 void ExpectClassNotFound(const char* name, bool check_jni, const char* check_jni_msg,
99 CheckJniAbortCatcher* abort_catcher) {
100 EXPECT_EQ(env_->FindClass(name), nullptr) << name;
101 if (!check_jni || check_jni_msg == nullptr) {
102 EXPECT_TRUE(env_->ExceptionCheck()) << name;
103 env_->ExceptionClear();
104 } else {
105 abort_catcher->Check(check_jni_msg);
106 }
107 }
108
109 void FindClassTest(bool check_jni) {
110 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
111 CheckJniAbortCatcher check_jni_abort_catcher;
112
113 // Null argument is always an abort.
114 env_->FindClass(nullptr);
115 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
116 : "name == null");
117
118 // Reference types...
119 ExpectClassFound("java/lang/String");
120 // ...for arrays too, where you must include "L;".
121 ExpectClassFound("[Ljava/lang/String;");
122 // Primitive arrays are okay too, if the primitive type is valid.
123 ExpectClassFound("[C");
124
125 // But primitive types aren't allowed...
126 ExpectClassNotFound("C", check_jni, nullptr, &check_jni_abort_catcher);
127 ExpectClassNotFound("V", check_jni, nullptr, &check_jni_abort_catcher);
128 ExpectClassNotFound("K", check_jni, nullptr, &check_jni_abort_catcher);
129
130 if (check_jni) {
131 // Check JNI will reject invalid class names as aborts but without pending exceptions.
132 EXPECT_EQ(env_->FindClass("java.lang.String"), nullptr);
133 EXPECT_FALSE(env_->ExceptionCheck());
134 check_jni_abort_catcher.Check("illegal class name 'java.lang.String'");
135
136 EXPECT_EQ(env_->FindClass("[Ljava.lang.String;"), nullptr);
137 EXPECT_FALSE(env_->ExceptionCheck());
138 check_jni_abort_catcher.Check("illegal class name '[Ljava.lang.String;'");
139 } else {
140 // Without check JNI we're tolerant and replace '.' with '/'.
141 ExpectClassFound("java.lang.String");
142 ExpectClassFound("[Ljava.lang.String;");
143 }
144
145 ExpectClassNotFound("Ljava.lang.String;", check_jni, "illegal class name 'Ljava.lang.String;'",
146 &check_jni_abort_catcher);
147 ExpectClassNotFound("[java.lang.String", check_jni, "illegal class name '[java.lang.String'",
148 &check_jni_abort_catcher);
149
150 // You can't include the "L;" in a JNI class descriptor.
151 ExpectClassNotFound("Ljava/lang/String;", check_jni, "illegal class name 'Ljava/lang/String;'",
152 &check_jni_abort_catcher);
153
154 // But you must include it for an array of any reference type.
155 ExpectClassNotFound("[java/lang/String", check_jni, "illegal class name '[java/lang/String'",
156 &check_jni_abort_catcher);
157
158 ExpectClassNotFound("[K", check_jni, "illegal class name '[K'", &check_jni_abort_catcher);
159
160 // Void arrays aren't allowed.
161 ExpectClassNotFound("[V", check_jni, "illegal class name '[V'", &check_jni_abort_catcher);
162
163 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
164 }
165
166 void GetFieldIdBadArgumentTest(bool check_jni) {
167 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
168 CheckJniAbortCatcher check_jni_abort_catcher;
169
170 jclass c = env_->FindClass("java/lang/String");
171 ASSERT_NE(c, nullptr);
172
173 jfieldID fid = env_->GetFieldID(nullptr, "count", "I");
174 EXPECT_EQ(nullptr, fid);
175 check_jni_abort_catcher.Check(check_jni ? "GetFieldID received NULL jclass"
176 : "java_class == null");
177 fid = env_->GetFieldID(c, nullptr, "I");
178 EXPECT_EQ(nullptr, fid);
179 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
180 : "name == null");
181 fid = env_->GetFieldID(c, "count", nullptr);
182 EXPECT_EQ(nullptr, fid);
183 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
184 : "sig == null");
185
186 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
187 }
188
189 void GetStaticFieldIdBadArgumentTest(bool check_jni) {
190 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
191 CheckJniAbortCatcher check_jni_abort_catcher;
192
193 jclass c = env_->FindClass("java/lang/String");
194 ASSERT_NE(c, nullptr);
195
196 jfieldID fid = env_->GetStaticFieldID(nullptr, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
197 EXPECT_EQ(nullptr, fid);
198 check_jni_abort_catcher.Check(check_jni ? "GetStaticFieldID received NULL jclass"
199 : "java_class == null");
200 fid = env_->GetStaticFieldID(c, nullptr, "Ljava/util/Comparator;");
201 EXPECT_EQ(nullptr, fid);
202 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
203 : "name == null");
204 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", nullptr);
205 EXPECT_EQ(nullptr, fid);
206 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
207 : "sig == null");
208
209 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
210 }
211
212 void GetMethodIdBadArgumentTest(bool check_jni) {
213 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
214 CheckJniAbortCatcher check_jni_abort_catcher;
215
216 jmethodID method = env_->GetMethodID(nullptr, "<init>", "(Ljava/lang/String;)V");
217 EXPECT_EQ(nullptr, method);
218 check_jni_abort_catcher.Check(check_jni ? "GetMethodID received NULL jclass"
219 : "java_class == null");
220 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
221 ASSERT_TRUE(jlnsme != nullptr);
222 method = env_->GetMethodID(jlnsme, nullptr, "(Ljava/lang/String;)V");
223 EXPECT_EQ(nullptr, method);
224 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
225 : "name == null");
226 method = env_->GetMethodID(jlnsme, "<init>", nullptr);
227 EXPECT_EQ(nullptr, method);
228 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
229 : "sig == null");
230
231 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
232 }
233
234 void GetStaticMethodIdBadArgumentTest(bool check_jni) {
235 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
236 CheckJniAbortCatcher check_jni_abort_catcher;
237
238 jmethodID method = env_->GetStaticMethodID(nullptr, "valueOf", "(I)Ljava/lang/String;");
239 EXPECT_EQ(nullptr, method);
240 check_jni_abort_catcher.Check(check_jni ? "GetStaticMethodID received NULL jclass"
241 : "java_class == null");
242 jclass jlstring = env_->FindClass("java/lang/String");
243 method = env_->GetStaticMethodID(jlstring, nullptr, "(I)Ljava/lang/String;");
244 EXPECT_EQ(nullptr, method);
245 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
246 : "name == null");
247 method = env_->GetStaticMethodID(jlstring, "valueOf", nullptr);
248 EXPECT_EQ(nullptr, method);
249 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
250 : "sig == null");
251
252 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
253 }
254
255 void GetFromReflectedField_ToReflectedFieldBadArgumentTest(bool check_jni) {
256 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
257 CheckJniAbortCatcher check_jni_abort_catcher;
258
259 jclass c = env_->FindClass("java/lang/String");
260 ASSERT_NE(c, nullptr);
261 jfieldID fid = env_->GetFieldID(c, "count", "I");
262 ASSERT_NE(fid, nullptr);
263
264 // Check class argument for null argument, not checked in non-check JNI.
265 jobject field = env_->ToReflectedField(nullptr, fid, JNI_FALSE);
266 if (check_jni) {
267 EXPECT_EQ(field, nullptr);
268 check_jni_abort_catcher.Check("ToReflectedField received NULL jclass");
269 } else {
270 EXPECT_NE(field, nullptr);
271 }
272
273 field = env_->ToReflectedField(c, nullptr, JNI_FALSE);
274 EXPECT_EQ(field, nullptr);
275 check_jni_abort_catcher.Check(check_jni ? "jfieldID was NULL"
276 : "fid == null");
277
278 fid = env_->FromReflectedField(nullptr);
279 ASSERT_EQ(fid, nullptr);
280 check_jni_abort_catcher.Check(check_jni ? "expected non-null java.lang.reflect.Field"
281 : "jlr_field == null");
282
283 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
284 }
285
286 void GetFromReflectedMethod_ToReflectedMethodBadArgumentTest(bool check_jni) {
287 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
288 CheckJniAbortCatcher check_jni_abort_catcher;
289
290 jclass c = env_->FindClass("java/lang/String");
291 ASSERT_NE(c, nullptr);
292 jmethodID mid = env_->GetMethodID(c, "<init>", "()V");
293 ASSERT_NE(mid, nullptr);
294
295 // Check class argument for null argument, not checked in non-check JNI.
296 jobject method = env_->ToReflectedMethod(nullptr, mid, JNI_FALSE);
297 if (check_jni) {
298 EXPECT_EQ(method, nullptr);
299 check_jni_abort_catcher.Check("ToReflectedMethod received NULL jclass");
300 } else {
301 EXPECT_NE(method, nullptr);
302 }
303
304 method = env_->ToReflectedMethod(c, nullptr, JNI_FALSE);
305 EXPECT_EQ(method, nullptr);
306 check_jni_abort_catcher.Check(check_jni ? "jmethodID was NULL"
307 : "mid == null");
308 mid = env_->FromReflectedMethod(method);
309 ASSERT_EQ(mid, nullptr);
310 check_jni_abort_catcher.Check(check_jni ? "expected non-null method" : "jlr_method == null");
311
312 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
313 }
314
315 void RegisterAndUnregisterNativesBadArguments(bool check_jni,
316 CheckJniAbortCatcher* check_jni_abort_catcher) {
317 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
318 // Passing a class of null is a failure.
319 {
320 JNINativeMethod methods[] = { };
321 EXPECT_EQ(env_->RegisterNatives(nullptr, methods, 0), JNI_ERR);
322 check_jni_abort_catcher->Check(check_jni ? "RegisterNatives received NULL jclass"
323 : "java_class == null");
324 }
325
326 // Passing methods as null is a failure.
327 jclass jlobject = env_->FindClass("java/lang/Object");
328 EXPECT_EQ(env_->RegisterNatives(jlobject, nullptr, 1), JNI_ERR);
329 check_jni_abort_catcher->Check("methods == null");
330
331 // Unregisters null is a failure.
332 EXPECT_EQ(env_->UnregisterNatives(nullptr), JNI_ERR);
333 check_jni_abort_catcher->Check(check_jni ? "UnregisterNatives received NULL jclass"
334 : "java_class == null");
335
336 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
337 }
338
339
340 void GetPrimitiveArrayElementsOfWrongType(bool check_jni) {
341 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
342 CheckJniAbortCatcher jni_abort_catcher;
343
344 jbooleanArray array = env_->NewBooleanArray(10);
345 jboolean is_copy;
346 EXPECT_EQ(env_->GetByteArrayElements(reinterpret_cast<jbyteArray>(array), &is_copy), nullptr);
347 jni_abort_catcher.Check(
348 check_jni ? "incompatible array type boolean[] expected byte[]"
349 : "attempt to get byte primitive array elements with an object of type boolean[]");
350 EXPECT_EQ(env_->GetShortArrayElements(reinterpret_cast<jshortArray>(array), &is_copy), nullptr);
351 jni_abort_catcher.Check(
352 check_jni ? "incompatible array type boolean[] expected short[]"
353 : "attempt to get short primitive array elements with an object of type boolean[]");
354 EXPECT_EQ(env_->GetCharArrayElements(reinterpret_cast<jcharArray>(array), &is_copy), nullptr);
355 jni_abort_catcher.Check(
356 check_jni ? "incompatible array type boolean[] expected char[]"
357 : "attempt to get char primitive array elements with an object of type boolean[]");
358 EXPECT_EQ(env_->GetIntArrayElements(reinterpret_cast<jintArray>(array), &is_copy), nullptr);
359 jni_abort_catcher.Check(
360 check_jni ? "incompatible array type boolean[] expected int[]"
361 : "attempt to get int primitive array elements with an object of type boolean[]");
362 EXPECT_EQ(env_->GetLongArrayElements(reinterpret_cast<jlongArray>(array), &is_copy), nullptr);
363 jni_abort_catcher.Check(
364 check_jni ? "incompatible array type boolean[] expected long[]"
365 : "attempt to get long primitive array elements with an object of type boolean[]");
366 EXPECT_EQ(env_->GetFloatArrayElements(reinterpret_cast<jfloatArray>(array), &is_copy), nullptr);
367 jni_abort_catcher.Check(
368 check_jni ? "incompatible array type boolean[] expected float[]"
369 : "attempt to get float primitive array elements with an object of type boolean[]");
370 EXPECT_EQ(env_->GetDoubleArrayElements(reinterpret_cast<jdoubleArray>(array), &is_copy), nullptr);
371 jni_abort_catcher.Check(
372 check_jni ? "incompatible array type boolean[] expected double[]"
373 : "attempt to get double primitive array elements with an object of type boolean[]");
374 jbyteArray array2 = env_->NewByteArray(10);
375 EXPECT_EQ(env_->GetBooleanArrayElements(reinterpret_cast<jbooleanArray>(array2), &is_copy),
376 nullptr);
377 jni_abort_catcher.Check(
378 check_jni ? "incompatible array type byte[] expected boolean[]"
379 : "attempt to get boolean primitive array elements with an object of type byte[]");
380 jobject object = env_->NewStringUTF("Test String");
381 EXPECT_EQ(env_->GetBooleanArrayElements(reinterpret_cast<jbooleanArray>(object), &is_copy),
382 nullptr);
383 jni_abort_catcher.Check(
384 check_jni ? "jarray argument has non-array type: java.lang.String"
385 : "attempt to get boolean primitive array elements with an object of type java.lang.String");
386
387 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
388 }
389
390 void ReleasePrimitiveArrayElementsOfWrongType(bool check_jni) {
391 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
392 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogersdd11d2a2014-11-19 10:06:46 -0800393 {
394 jbooleanArray array = env_->NewBooleanArray(10);
395 ASSERT_TRUE(array != nullptr);
396 jboolean is_copy;
397 jboolean* elements = env_->GetBooleanArrayElements(array, &is_copy);
398 ASSERT_TRUE(elements != nullptr);
399 env_->ReleaseByteArrayElements(reinterpret_cast<jbyteArray>(array),
400 reinterpret_cast<jbyte*>(elements), 0);
401 jni_abort_catcher.Check(
402 check_jni ? "incompatible array type boolean[] expected byte[]"
403 : "attempt to release byte primitive array elements with an object of type boolean[]");
404 env_->ReleaseShortArrayElements(reinterpret_cast<jshortArray>(array),
405 reinterpret_cast<jshort*>(elements), 0);
406 jni_abort_catcher.Check(
407 check_jni ? "incompatible array type boolean[] expected short[]"
408 : "attempt to release short primitive array elements with an object of type boolean[]");
409 env_->ReleaseCharArrayElements(reinterpret_cast<jcharArray>(array),
410 reinterpret_cast<jchar*>(elements), 0);
411 jni_abort_catcher.Check(
412 check_jni ? "incompatible array type boolean[] expected char[]"
413 : "attempt to release char primitive array elements with an object of type boolean[]");
414 env_->ReleaseIntArrayElements(reinterpret_cast<jintArray>(array),
415 reinterpret_cast<jint*>(elements), 0);
416 jni_abort_catcher.Check(
417 check_jni ? "incompatible array type boolean[] expected int[]"
418 : "attempt to release int primitive array elements with an object of type boolean[]");
419 env_->ReleaseLongArrayElements(reinterpret_cast<jlongArray>(array),
420 reinterpret_cast<jlong*>(elements), 0);
421 jni_abort_catcher.Check(
422 check_jni ? "incompatible array type boolean[] expected long[]"
423 : "attempt to release long primitive array elements with an object of type boolean[]");
424 env_->ReleaseFloatArrayElements(reinterpret_cast<jfloatArray>(array),
425 reinterpret_cast<jfloat*>(elements), 0);
426 jni_abort_catcher.Check(
427 check_jni ? "incompatible array type boolean[] expected float[]"
428 : "attempt to release float primitive array elements with an object of type boolean[]");
429 env_->ReleaseDoubleArrayElements(reinterpret_cast<jdoubleArray>(array),
430 reinterpret_cast<jdouble*>(elements), 0);
431 jni_abort_catcher.Check(
432 check_jni ? "incompatible array type boolean[] expected double[]"
433 : "attempt to release double primitive array elements with an object of type boolean[]");
Ian Rogers68d8b422014-07-17 11:09:10 -0700434
Ian Rogersdd11d2a2014-11-19 10:06:46 -0800435 // Don't leak the elements array.
436 env_->ReleaseBooleanArrayElements(array, elements, 0);
437 }
438 {
439 jbyteArray array = env_->NewByteArray(10);
440 jboolean is_copy;
441 jbyte* elements = env_->GetByteArrayElements(array, &is_copy);
442
443 env_->ReleaseBooleanArrayElements(reinterpret_cast<jbooleanArray>(array),
444 reinterpret_cast<jboolean*>(elements), 0);
445 jni_abort_catcher.Check(
446 check_jni ? "incompatible array type byte[] expected boolean[]"
447 : "attempt to release boolean primitive array elements with an object of type byte[]");
448 jobject object = env_->NewStringUTF("Test String");
449 env_->ReleaseBooleanArrayElements(reinterpret_cast<jbooleanArray>(object),
450 reinterpret_cast<jboolean*>(elements), 0);
451 jni_abort_catcher.Check(
452 check_jni ? "jarray argument has non-array type: java.lang.String"
453 : "attempt to release boolean primitive array elements with an object of type "
Ian Rogers68d8b422014-07-17 11:09:10 -0700454 "java.lang.String");
455
Ian Rogersdd11d2a2014-11-19 10:06:46 -0800456 // Don't leak the elements array.
457 env_->ReleaseByteArrayElements(array, elements, 0);
458 }
Ian Rogers68d8b422014-07-17 11:09:10 -0700459 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
460 }
461
462 void GetReleasePrimitiveArrayCriticalOfWrongType(bool check_jni) {
463 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
464 CheckJniAbortCatcher jni_abort_catcher;
465
466 jobject object = env_->NewStringUTF("Test String");
467 jboolean is_copy;
468 void* elements = env_->GetPrimitiveArrayCritical(reinterpret_cast<jarray>(object), &is_copy);
469 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
470 : "expected primitive array, given java.lang.String");
471 env_->ReleasePrimitiveArrayCritical(reinterpret_cast<jarray>(object), elements, 0);
472 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
473 : "expected primitive array, given java.lang.String");
474
475 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
476 }
477
478 void GetPrimitiveArrayRegionElementsOfWrongType(bool check_jni) {
479 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
480 CheckJniAbortCatcher jni_abort_catcher;
481 constexpr size_t kLength = 10;
482 jbooleanArray array = env_->NewBooleanArray(kLength);
483 ASSERT_TRUE(array != nullptr);
484 jboolean elements[kLength];
485 env_->GetByteArrayRegion(reinterpret_cast<jbyteArray>(array), 0, kLength,
486 reinterpret_cast<jbyte*>(elements));
487 jni_abort_catcher.Check(
488 check_jni ? "incompatible array type boolean[] expected byte[]"
489 : "attempt to get region of byte primitive array elements with an object of type boolean[]");
490 env_->GetShortArrayRegion(reinterpret_cast<jshortArray>(array), 0, kLength,
491 reinterpret_cast<jshort*>(elements));
492 jni_abort_catcher.Check(
493 check_jni ? "incompatible array type boolean[] expected short[]"
494 : "attempt to get region of short primitive array elements with an object of type boolean[]");
495 env_->GetCharArrayRegion(reinterpret_cast<jcharArray>(array), 0, kLength,
496 reinterpret_cast<jchar*>(elements));
497 jni_abort_catcher.Check(
498 check_jni ? "incompatible array type boolean[] expected char[]"
499 : "attempt to get region of char primitive array elements with an object of type boolean[]");
500 env_->GetIntArrayRegion(reinterpret_cast<jintArray>(array), 0, kLength,
501 reinterpret_cast<jint*>(elements));
502 jni_abort_catcher.Check(
503 check_jni ? "incompatible array type boolean[] expected int[]"
504 : "attempt to get region of int primitive array elements with an object of type boolean[]");
505 env_->GetLongArrayRegion(reinterpret_cast<jlongArray>(array), 0, kLength,
506 reinterpret_cast<jlong*>(elements));
507 jni_abort_catcher.Check(
508 check_jni ? "incompatible array type boolean[] expected long[]"
509 : "attempt to get region of long primitive array elements with an object of type boolean[]");
510 env_->GetFloatArrayRegion(reinterpret_cast<jfloatArray>(array), 0, kLength,
511 reinterpret_cast<jfloat*>(elements));
512 jni_abort_catcher.Check(
513 check_jni ? "incompatible array type boolean[] expected float[]"
514 : "attempt to get region of float primitive array elements with an object of type boolean[]");
515 env_->GetDoubleArrayRegion(reinterpret_cast<jdoubleArray>(array), 0, kLength,
516 reinterpret_cast<jdouble*>(elements));
517 jni_abort_catcher.Check(
518 check_jni ? "incompatible array type boolean[] expected double[]"
519 : "attempt to get region of double primitive array elements with an object of type boolean[]");
520 jbyteArray array2 = env_->NewByteArray(10);
521 env_->GetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(array2), 0, kLength,
522 reinterpret_cast<jboolean*>(elements));
523 jni_abort_catcher.Check(
524 check_jni ? "incompatible array type byte[] expected boolean[]"
525 : "attempt to get region of boolean primitive array elements with an object of type byte[]");
526 jobject object = env_->NewStringUTF("Test String");
527 env_->GetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(object), 0, kLength,
528 reinterpret_cast<jboolean*>(elements));
529 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
530 : "attempt to get region of boolean primitive array elements with an object of type "
531 "java.lang.String");
532
533 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
534 }
535
536 void SetPrimitiveArrayRegionElementsOfWrongType(bool check_jni) {
537 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
538 CheckJniAbortCatcher jni_abort_catcher;
539 constexpr size_t kLength = 10;
540 jbooleanArray array = env_->NewBooleanArray(kLength);
541 ASSERT_TRUE(array != nullptr);
542 jboolean elements[kLength];
543 env_->SetByteArrayRegion(reinterpret_cast<jbyteArray>(array), 0, kLength,
544 reinterpret_cast<jbyte*>(elements));
545 jni_abort_catcher.Check(
546 check_jni ? "incompatible array type boolean[] expected byte[]"
547 : "attempt to set region of byte primitive array elements with an object of type boolean[]");
548 env_->SetShortArrayRegion(reinterpret_cast<jshortArray>(array), 0, kLength,
549 reinterpret_cast<jshort*>(elements));
550 jni_abort_catcher.Check(
551 check_jni ? "incompatible array type boolean[] expected short[]"
552 : "attempt to set region of short primitive array elements with an object of type boolean[]");
553 env_->SetCharArrayRegion(reinterpret_cast<jcharArray>(array), 0, kLength,
554 reinterpret_cast<jchar*>(elements));
555 jni_abort_catcher.Check(
556 check_jni ? "incompatible array type boolean[] expected char[]"
557 : "attempt to set region of char primitive array elements with an object of type boolean[]");
558 env_->SetIntArrayRegion(reinterpret_cast<jintArray>(array), 0, kLength,
559 reinterpret_cast<jint*>(elements));
560 jni_abort_catcher.Check(
561 check_jni ? "incompatible array type boolean[] expected int[]"
562 : "attempt to set region of int primitive array elements with an object of type boolean[]");
563 env_->SetLongArrayRegion(reinterpret_cast<jlongArray>(array), 0, kLength,
564 reinterpret_cast<jlong*>(elements));
565 jni_abort_catcher.Check(
566 check_jni ? "incompatible array type boolean[] expected long[]"
567 : "attempt to set region of long primitive array elements with an object of type boolean[]");
568 env_->SetFloatArrayRegion(reinterpret_cast<jfloatArray>(array), 0, kLength,
569 reinterpret_cast<jfloat*>(elements));
570 jni_abort_catcher.Check(
571 check_jni ? "incompatible array type boolean[] expected float[]"
572 : "attempt to set region of float primitive array elements with an object of type boolean[]");
573 env_->SetDoubleArrayRegion(reinterpret_cast<jdoubleArray>(array), 0, kLength,
574 reinterpret_cast<jdouble*>(elements));
575 jni_abort_catcher.Check(
576 check_jni ? "incompatible array type boolean[] expected double[]"
577 : "attempt to set region of double primitive array elements with an object of type boolean[]");
578 jbyteArray array2 = env_->NewByteArray(10);
579 env_->SetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(array2), 0, kLength,
580 reinterpret_cast<jboolean*>(elements));
581 jni_abort_catcher.Check(
582 check_jni ? "incompatible array type byte[] expected boolean[]"
583 : "attempt to set region of boolean primitive array elements with an object of type byte[]");
584 jobject object = env_->NewStringUTF("Test String");
585 env_->SetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(object), 0, kLength,
586 reinterpret_cast<jboolean*>(elements));
587 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
588 : "attempt to set region of boolean primitive array elements with an object of type "
589 "java.lang.String");
590 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
591 }
592
593 void NewObjectArrayBadArguments(bool check_jni) {
594 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
595 CheckJniAbortCatcher jni_abort_catcher;
596
597 jclass element_class = env_->FindClass("java/lang/String");
598 ASSERT_NE(element_class, nullptr);
599
600 env_->NewObjectArray(-1, element_class, nullptr);
601 jni_abort_catcher.Check(check_jni ? "negative jsize: -1" : "negative array length: -1");
602
603 env_->NewObjectArray(std::numeric_limits<jint>::min(), element_class, nullptr);
604 jni_abort_catcher.Check(check_jni ? "negative jsize: -2147483648"
605 : "negative array length: -2147483648");
606
607 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
608 }
609
Andreas Gampe5f4a09a2015-09-28 13:16:33 -0700610 void SetUpForTest(bool direct, const char* method_name, const char* method_sig,
611 void* native_fnptr) {
612 // Initialize class loader and set generic JNI entrypoint.
613 // Note: this code is adapted from the jni_compiler_test, and taken with minimal modifications.
614 if (!runtime_->IsStarted()) {
615 {
616 ScopedObjectAccess soa(Thread::Current());
617 class_loader_ = LoadDex("MyClassNatives");
618 StackHandleScope<1> hs(soa.Self());
619 Handle<mirror::ClassLoader> loader(
620 hs.NewHandle(soa.Decode<mirror::ClassLoader*>(class_loader_)));
621 mirror::Class* c = class_linker_->FindClass(soa.Self(), "LMyClassNatives;", loader);
622 const auto pointer_size = class_linker_->GetImagePointerSize();
623 ArtMethod* method = direct ? c->FindDirectMethod(method_name, method_sig, pointer_size) :
624 c->FindVirtualMethod(method_name, method_sig, pointer_size);
625 ASSERT_TRUE(method != nullptr) << method_name << " " << method_sig;
626 method->SetEntryPointFromQuickCompiledCode(class_linker_->GetRuntimeQuickGenericJniStub());
627 }
628 // Start runtime.
629 Thread::Current()->TransitionFromSuspendedToRunnable();
630 bool started = runtime_->Start();
631 CHECK(started);
632 }
633 // JNI operations after runtime start.
634 env_ = Thread::Current()->GetJniEnv();
635 jklass_ = env_->FindClass("MyClassNatives");
636 ASSERT_TRUE(jklass_ != nullptr) << method_name << " " << method_sig;
637
638 if (direct) {
639 jmethod_ = env_->GetStaticMethodID(jklass_, method_name, method_sig);
640 } else {
641 jmethod_ = env_->GetMethodID(jklass_, method_name, method_sig);
642 }
643 ASSERT_TRUE(jmethod_ != nullptr) << method_name << " " << method_sig;
644
645 if (native_fnptr != nullptr) {
646 JNINativeMethod methods[] = { { method_name, method_sig, native_fnptr } };
647 ASSERT_EQ(JNI_OK, env_->RegisterNatives(jklass_, methods, 1))
648 << method_name << " " << method_sig;
649 } else {
650 env_->UnregisterNatives(jklass_);
651 }
652
653 jmethodID constructor = env_->GetMethodID(jklass_, "<init>", "()V");
654 jobj_ = env_->NewObject(jklass_, constructor);
655 ASSERT_TRUE(jobj_ != nullptr) << method_name << " " << method_sig;
656 }
657
Elliott Hughesa2501992011-08-26 19:39:54 -0700658 JavaVMExt* vm_;
Brian Carlstrom4d571432012-05-16 00:21:41 -0700659 JNIEnv* env_;
Elliott Hughes814e4032011-08-23 12:07:56 -0700660 jclass aioobe_;
Elliott Hughesb264f082012-04-06 17:10:10 -0700661 jclass ase_;
Elliott Hughesb465ab02011-08-24 11:21:21 -0700662 jclass sioobe_;
Andreas Gampe5f4a09a2015-09-28 13:16:33 -0700663
664 jclass jklass_;
665 jobject jobj_;
666 jobject class_loader_;
667 jmethodID jmethod_;
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700668};
669
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700670TEST_F(JniInternalTest, AllocObject) {
671 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700672 ASSERT_NE(c, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700673 jobject o = env_->AllocObject(c);
Ian Rogers2d10b202014-05-12 19:15:18 -0700674 ASSERT_NE(o, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700675
676 // We have an instance of the class we asked for...
677 ASSERT_TRUE(env_->IsInstanceOf(o, c));
678 // ...whose fields haven't been initialized because
679 // we didn't call a constructor.
680 ASSERT_EQ(0, env_->GetIntField(o, env_->GetFieldID(c, "count", "I")));
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700681}
682
Elliott Hughesc7ac37f2011-08-12 12:21:58 -0700683TEST_F(JniInternalTest, GetVersion) {
684 ASSERT_EQ(JNI_VERSION_1_6, env_->GetVersion());
685}
686
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700687TEST_F(JniInternalTest, FindClass) {
Andreas Gampe369810a2015-01-14 19:53:31 -0800688 // This tests leads to warnings in the log.
689 ScopedLogSeverity sls(LogSeverity::ERROR);
690
Ian Rogers68d8b422014-07-17 11:09:10 -0700691 FindClassTest(false);
692 FindClassTest(true);
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700693}
694
Elliott Hughescdf53122011-08-19 15:46:09 -0700695TEST_F(JniInternalTest, GetFieldID) {
696 jclass jlnsfe = env_->FindClass("java/lang/NoSuchFieldError");
Ian Rogers2d10b202014-05-12 19:15:18 -0700697 ASSERT_NE(jlnsfe, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700698 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700699 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700700
701 // Wrong type.
702 jfieldID fid = env_->GetFieldID(c, "count", "J");
Ian Rogers2d10b202014-05-12 19:15:18 -0700703 EXPECT_EQ(nullptr, fid);
704 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700705
Ian Rogersb17d08b2011-09-02 16:16:49 -0700706 // Wrong type where type doesn't exist.
707 fid = env_->GetFieldID(c, "count", "Lrod/jane/freddy;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700708 EXPECT_EQ(nullptr, fid);
709 ExpectException(jlnsfe);
Ian Rogersb17d08b2011-09-02 16:16:49 -0700710
Elliott Hughescdf53122011-08-19 15:46:09 -0700711 // Wrong name.
712 fid = env_->GetFieldID(c, "Count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700713 EXPECT_EQ(nullptr, fid);
714 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700715
716 // Good declared field lookup.
717 fid = env_->GetFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700718 EXPECT_NE(nullptr, fid);
Elliott Hughescdf53122011-08-19 15:46:09 -0700719 EXPECT_FALSE(env_->ExceptionCheck());
720
721 // Good superclass field lookup.
722 c = env_->FindClass("java/lang/StringBuilder");
723 fid = env_->GetFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700724 EXPECT_NE(nullptr, fid);
725 EXPECT_NE(fid, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700726 EXPECT_FALSE(env_->ExceptionCheck());
727
728 // Not instance.
729 fid = env_->GetFieldID(c, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700730 EXPECT_EQ(nullptr, fid);
731 ExpectException(jlnsfe);
732
733 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700734 GetFieldIdBadArgumentTest(false);
735 GetFieldIdBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700736}
737
738TEST_F(JniInternalTest, GetStaticFieldID) {
739 jclass jlnsfe = env_->FindClass("java/lang/NoSuchFieldError");
Ian Rogers2d10b202014-05-12 19:15:18 -0700740 ASSERT_NE(jlnsfe, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700741 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700742 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700743
744 // Wrong type.
745 jfieldID fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "J");
Ian Rogers2d10b202014-05-12 19:15:18 -0700746 EXPECT_EQ(nullptr, fid);
747 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700748
Ian Rogersb17d08b2011-09-02 16:16:49 -0700749 // Wrong type where type doesn't exist.
750 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "Lrod/jane/freddy;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700751 EXPECT_EQ(nullptr, fid);
752 ExpectException(jlnsfe);
Ian Rogersb17d08b2011-09-02 16:16:49 -0700753
Elliott Hughescdf53122011-08-19 15:46:09 -0700754 // Wrong name.
755 fid = env_->GetStaticFieldID(c, "cASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700756 EXPECT_EQ(nullptr, fid);
757 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700758
759 // Good declared field lookup.
760 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700761 EXPECT_NE(nullptr, fid);
762 EXPECT_NE(fid, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700763 EXPECT_FALSE(env_->ExceptionCheck());
764
765 // Not static.
766 fid = env_->GetStaticFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700767 EXPECT_EQ(nullptr, fid);
768 ExpectException(jlnsfe);
769
770 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700771 GetStaticFieldIdBadArgumentTest(false);
772 GetStaticFieldIdBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700773}
774
Ian Rogers4dd71f12011-08-16 14:16:02 -0700775TEST_F(JniInternalTest, GetMethodID) {
776 jclass jlobject = env_->FindClass("java/lang/Object");
777 jclass jlstring = env_->FindClass("java/lang/String");
778 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
Brian Carlstrom004644f2014-06-18 08:34:01 -0700779 jclass jncrbc = env_->FindClass("java/nio/channels/ReadableByteChannel");
Ian Rogers4dd71f12011-08-16 14:16:02 -0700780
Brian Carlstrom004644f2014-06-18 08:34:01 -0700781 // Sanity check that no exceptions are pending.
Elliott Hughescdf53122011-08-19 15:46:09 -0700782 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700783
784 // Check that java.lang.Object.foo() doesn't exist and NoSuchMethodError is
Brian Carlstrom004644f2014-06-18 08:34:01 -0700785 // a pending exception.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700786 jmethodID method = env_->GetMethodID(jlobject, "foo", "()V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700787 EXPECT_EQ(nullptr, method);
788 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700789
Brian Carlstrom004644f2014-06-18 08:34:01 -0700790 // Check that java.lang.Object.equals() does exist.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700791 method = env_->GetMethodID(jlobject, "equals", "(Ljava/lang/Object;)Z");
Ian Rogers2d10b202014-05-12 19:15:18 -0700792 EXPECT_NE(nullptr, method);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700793 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700794
795 // Check that GetMethodID for java.lang.String.valueOf(int) fails as the
Brian Carlstrom004644f2014-06-18 08:34:01 -0700796 // method is static.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700797 method = env_->GetMethodID(jlstring, "valueOf", "(I)Ljava/lang/String;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700798 EXPECT_EQ(nullptr, method);
799 ExpectException(jlnsme);
Brian Carlstromea46f952013-07-30 01:26:50 -0700800
Brian Carlstrom004644f2014-06-18 08:34:01 -0700801 // Check that GetMethodID for java.lang.NoSuchMethodError.<init>(String) finds the constructor.
Brian Carlstromea46f952013-07-30 01:26:50 -0700802 method = env_->GetMethodID(jlnsme, "<init>", "(Ljava/lang/String;)V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700803 EXPECT_NE(nullptr, method);
Brian Carlstromea46f952013-07-30 01:26:50 -0700804 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers2d10b202014-05-12 19:15:18 -0700805
Brian Carlstrom004644f2014-06-18 08:34:01 -0700806 // Check that GetMethodID can find a interface method inherited from another interface.
807 method = env_->GetMethodID(jncrbc, "close", "()V");
808 EXPECT_NE(nullptr, method);
809 EXPECT_FALSE(env_->ExceptionCheck());
810
Ian Rogers2d10b202014-05-12 19:15:18 -0700811 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700812 GetMethodIdBadArgumentTest(false);
813 GetMethodIdBadArgumentTest(true);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700814}
815
Mathieu Chartiera1e78fa2014-08-20 12:09:38 -0700816TEST_F(JniInternalTest, CallVoidMethodNullReceiver) {
817 jclass jlobject = env_->FindClass("java/lang/Object");
818 jmethodID method;
819
820 // Check that GetMethodID for java.lang.NoSuchMethodError.<init>(String) finds the constructor.
821 method = env_->GetMethodID(jlobject, "<init>", "()V");
822 EXPECT_NE(nullptr, method);
823 EXPECT_FALSE(env_->ExceptionCheck());
824
825 // Null object to CallVoidMethod.
826 CheckJniAbortCatcher check_jni_abort_catcher;
827 env_->CallVoidMethod(nullptr, method);
828 check_jni_abort_catcher.Check("null");
829}
830
Ian Rogers4dd71f12011-08-16 14:16:02 -0700831TEST_F(JniInternalTest, GetStaticMethodID) {
832 jclass jlobject = env_->FindClass("java/lang/Object");
833 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
834
835 // Sanity check that no exceptions are pending
Elliott Hughescdf53122011-08-19 15:46:09 -0700836 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700837
838 // Check that java.lang.Object.foo() doesn't exist and NoSuchMethodError is
839 // a pending exception
840 jmethodID method = env_->GetStaticMethodID(jlobject, "foo", "()V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700841 EXPECT_EQ(nullptr, method);
842 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700843
844 // Check that GetStaticMethodID for java.lang.Object.equals(Object) fails as
845 // the method is not static
846 method = env_->GetStaticMethodID(jlobject, "equals", "(Ljava/lang/Object;)Z");
Ian Rogers2d10b202014-05-12 19:15:18 -0700847 EXPECT_EQ(nullptr, method);
848 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700849
850 // Check that java.lang.String.valueOf(int) does exist
Ian Rogers4dd71f12011-08-16 14:16:02 -0700851 jclass jlstring = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700852 method = env_->GetStaticMethodID(jlstring, "valueOf", "(I)Ljava/lang/String;");
853 EXPECT_NE(nullptr, method);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700854 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers2d10b202014-05-12 19:15:18 -0700855
856 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700857 GetStaticMethodIdBadArgumentTest(false);
858 GetStaticMethodIdBadArgumentTest(true);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700859}
860
Elliott Hughescdf53122011-08-19 15:46:09 -0700861TEST_F(JniInternalTest, FromReflectedField_ToReflectedField) {
862 jclass jlrField = env_->FindClass("java/lang/reflect/Field");
863 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700864 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700865 jfieldID fid = env_->GetFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700866 ASSERT_NE(fid, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700867 // Turn the fid into a java.lang.reflect.Field...
868 jobject field = env_->ToReflectedField(c, fid, JNI_FALSE);
Mathieu Chartier78959462014-11-17 10:24:04 -0800869 for (size_t i = 0; i <= kLocalsMax; ++i) {
Mathieu Chartier41da5962014-11-15 13:07:39 -0800870 // Regression test for b/18396311, ToReflectedField leaking local refs causing a local
871 // reference table overflows with 512 references to ArtField
872 env_->DeleteLocalRef(env_->ToReflectedField(c, fid, JNI_FALSE));
873 }
Ian Rogers2d10b202014-05-12 19:15:18 -0700874 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700875 ASSERT_TRUE(env_->IsInstanceOf(field, jlrField));
876 // ...and back again.
877 jfieldID fid2 = env_->FromReflectedField(field);
Ian Rogers2d10b202014-05-12 19:15:18 -0700878 ASSERT_NE(fid2, nullptr);
Brian Carlstromea46f952013-07-30 01:26:50 -0700879 // Make sure we can actually use it.
880 jstring s = env_->NewStringUTF("poop");
881 ASSERT_EQ(4, env_->GetIntField(s, fid2));
Ian Rogers2d10b202014-05-12 19:15:18 -0700882
883 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700884 GetFromReflectedField_ToReflectedFieldBadArgumentTest(false);
885 GetFromReflectedField_ToReflectedFieldBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700886}
887
888TEST_F(JniInternalTest, FromReflectedMethod_ToReflectedMethod) {
889 jclass jlrMethod = env_->FindClass("java/lang/reflect/Method");
Sebastien Hertzd3333762014-06-26 14:45:07 +0200890 ASSERT_NE(jlrMethod, nullptr);
891 jclass jlrConstructor = env_->FindClass("java/lang/reflect/Constructor");
892 ASSERT_NE(jlrConstructor, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700893 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700894 ASSERT_NE(c, nullptr);
Sebastien Hertzd3333762014-06-26 14:45:07 +0200895
896 jmethodID mid = env_->GetMethodID(c, "<init>", "()V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700897 ASSERT_NE(mid, nullptr);
Sebastien Hertzd3333762014-06-26 14:45:07 +0200898 // Turn the mid into a java.lang.reflect.Constructor...
Elliott Hughescdf53122011-08-19 15:46:09 -0700899 jobject method = env_->ToReflectedMethod(c, mid, JNI_FALSE);
Mathieu Chartier78959462014-11-17 10:24:04 -0800900 for (size_t i = 0; i <= kLocalsMax; ++i) {
Mathieu Chartier41da5962014-11-15 13:07:39 -0800901 // Regression test for b/18396311, ToReflectedMethod leaking local refs causing a local
902 // reference table overflows with 512 references to ArtMethod
903 env_->DeleteLocalRef(env_->ToReflectedMethod(c, mid, JNI_FALSE));
904 }
Sebastien Hertzd3333762014-06-26 14:45:07 +0200905 ASSERT_NE(method, nullptr);
906 ASSERT_TRUE(env_->IsInstanceOf(method, jlrConstructor));
Elliott Hughescdf53122011-08-19 15:46:09 -0700907 // ...and back again.
908 jmethodID mid2 = env_->FromReflectedMethod(method);
Ian Rogers2d10b202014-05-12 19:15:18 -0700909 ASSERT_NE(mid2, nullptr);
Brian Carlstromea46f952013-07-30 01:26:50 -0700910 // Make sure we can actually use it.
Sebastien Hertzd3333762014-06-26 14:45:07 +0200911 jstring s = reinterpret_cast<jstring>(env_->AllocObject(c));
912 ASSERT_NE(s, nullptr);
913 env_->CallVoidMethod(s, mid2);
Jeff Hao39b6c242015-05-19 20:30:23 -0700914 ASSERT_EQ(JNI_FALSE, env_->ExceptionCheck());
Jeff Hao848f70a2014-01-15 13:49:50 -0800915 env_->ExceptionClear();
Sebastien Hertzd3333762014-06-26 14:45:07 +0200916
917 mid = env_->GetMethodID(c, "length", "()I");
918 ASSERT_NE(mid, nullptr);
919 // Turn the mid into a java.lang.reflect.Method...
920 method = env_->ToReflectedMethod(c, mid, JNI_FALSE);
921 ASSERT_NE(method, nullptr);
922 ASSERT_TRUE(env_->IsInstanceOf(method, jlrMethod));
923 // ...and back again.
924 mid2 = env_->FromReflectedMethod(method);
925 ASSERT_NE(mid2, nullptr);
926 // Make sure we can actually use it.
927 s = env_->NewStringUTF("poop");
928 ASSERT_NE(s, nullptr);
Ian Rogers5d27faf2014-05-02 17:17:18 -0700929 ASSERT_EQ(4, env_->CallIntMethod(s, mid2));
Ian Rogers2d10b202014-05-12 19:15:18 -0700930
931 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700932 GetFromReflectedMethod_ToReflectedMethodBadArgumentTest(false);
933 GetFromReflectedMethod_ToReflectedMethodBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700934}
935
Ian Rogers2d10b202014-05-12 19:15:18 -0700936static void BogusMethod() {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700937 // You can't pass null function pointers to RegisterNatives.
Elliott Hughes5174fe62011-08-23 15:12:35 -0700938}
939
Ian Rogers2d10b202014-05-12 19:15:18 -0700940TEST_F(JniInternalTest, RegisterAndUnregisterNatives) {
Ian Rogers4dd71f12011-08-16 14:16:02 -0700941 jclass jlobject = env_->FindClass("java/lang/Object");
942 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200943 void* native_function = reinterpret_cast<void*>(BogusMethod);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700944
Ian Rogers2d10b202014-05-12 19:15:18 -0700945 // Sanity check that no exceptions are pending.
Elliott Hughescdf53122011-08-19 15:46:09 -0700946 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700947
Andreas Gampe369810a2015-01-14 19:53:31 -0800948 // The following can print errors to the log we'd like to ignore.
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200949 {
Andreas Gampe369810a2015-01-14 19:53:31 -0800950 ScopedLogSeverity sls(LogSeverity::FATAL);
951 // Check that registering method without name causes a NoSuchMethodError.
952 {
953 JNINativeMethod methods[] = { { nullptr, "()V", native_function } };
954 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
955 }
956 ExpectException(jlnsme);
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200957
Andreas Gampe369810a2015-01-14 19:53:31 -0800958 // Check that registering method without signature causes a NoSuchMethodError.
959 {
960 JNINativeMethod methods[] = { { "notify", nullptr, native_function } };
961 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
962 }
963 ExpectException(jlnsme);
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200964
Andreas Gampe369810a2015-01-14 19:53:31 -0800965 // Check that registering method without function causes a NoSuchMethodError.
966 {
967 JNINativeMethod methods[] = { { "notify", "()V", nullptr } };
968 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
969 }
970 ExpectException(jlnsme);
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200971
Andreas Gampe369810a2015-01-14 19:53:31 -0800972 // Check that registering to a non-existent java.lang.Object.foo() causes a NoSuchMethodError.
973 {
974 JNINativeMethod methods[] = { { "foo", "()V", native_function } };
975 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
976 }
977 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700978
Andreas Gampe369810a2015-01-14 19:53:31 -0800979 // Check that registering non-native methods causes a NoSuchMethodError.
980 {
981 JNINativeMethod methods[] = { { "equals", "(Ljava/lang/Object;)Z", native_function } };
982 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
983 }
984 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700985 }
Ian Rogers4dd71f12011-08-16 14:16:02 -0700986
Ian Rogers2d10b202014-05-12 19:15:18 -0700987 // Check that registering native methods is successful.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700988 {
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200989 JNINativeMethod methods[] = { { "notify", "()V", native_function } };
Ian Rogers2d10b202014-05-12 19:15:18 -0700990 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_OK);
991 }
992 EXPECT_FALSE(env_->ExceptionCheck());
993 EXPECT_EQ(env_->UnregisterNatives(jlobject), JNI_OK);
994
995 // Check that registering no methods isn't a failure.
996 {
997 JNINativeMethod methods[] = { };
998 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 0), JNI_OK);
999 }
1000 EXPECT_FALSE(env_->ExceptionCheck());
1001 EXPECT_EQ(env_->UnregisterNatives(jlobject), JNI_OK);
1002
1003 // Check that registering a -ve number of methods is a failure.
1004 CheckJniAbortCatcher check_jni_abort_catcher;
1005 for (int i = -10; i < 0; ++i) {
1006 JNINativeMethod methods[] = { };
1007 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, i), JNI_ERR);
1008 check_jni_abort_catcher.Check("negative method count: ");
Ian Rogers4dd71f12011-08-16 14:16:02 -07001009 }
1010 EXPECT_FALSE(env_->ExceptionCheck());
Elliott Hughes5174fe62011-08-23 15:12:35 -07001011
Ian Rogers2d10b202014-05-12 19:15:18 -07001012 // Unregistering a class with no natives is a warning.
1013 EXPECT_EQ(env_->UnregisterNatives(jlnsme), JNI_OK);
Ian Rogers68d8b422014-07-17 11:09:10 -07001014
1015 RegisterAndUnregisterNativesBadArguments(false, &check_jni_abort_catcher);
1016 RegisterAndUnregisterNativesBadArguments(true, &check_jni_abort_catcher);
Ian Rogers4dd71f12011-08-16 14:16:02 -07001017}
1018
Brian Carlstromea46f952013-07-30 01:26:50 -07001019#define EXPECT_PRIMITIVE_ARRAY(new_fn, \
1020 get_region_fn, \
1021 set_region_fn, \
1022 get_elements_fn, \
1023 release_elements_fn, \
1024 scalar_type, \
1025 expected_class_descriptor) \
Ian Rogers2d10b202014-05-12 19:15:18 -07001026 jsize size = 4; \
1027 \
Ian Rogers1d99e452014-01-02 17:36:41 -08001028 { \
1029 CheckJniAbortCatcher jni_abort_catcher; \
Ian Rogers68d8b422014-07-17 11:09:10 -07001030 down_cast<JNIEnvExt*>(env_)->SetCheckJniEnabled(false); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001031 /* Allocate an negative sized array and check it has the right failure type. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001032 EXPECT_EQ(env_->new_fn(-1), nullptr); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001033 jni_abort_catcher.Check("negative array length: -1"); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001034 EXPECT_EQ(env_->new_fn(std::numeric_limits<jint>::min()), nullptr); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001035 jni_abort_catcher.Check("negative array length: -2147483648"); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001036 /* Pass the array as null. */ \
1037 EXPECT_EQ(0, env_->GetArrayLength(nullptr)); \
1038 jni_abort_catcher.Check("java_array == null"); \
1039 env_->get_region_fn(nullptr, 0, 0, nullptr); \
1040 jni_abort_catcher.Check("java_array == null"); \
1041 env_->set_region_fn(nullptr, 0, 0, nullptr); \
1042 jni_abort_catcher.Check("java_array == null"); \
1043 env_->get_elements_fn(nullptr, nullptr); \
1044 jni_abort_catcher.Check("java_array == null"); \
1045 env_->release_elements_fn(nullptr, nullptr, 0); \
1046 jni_abort_catcher.Check("java_array == null"); \
1047 /* Pass the elements for region as null. */ \
1048 scalar_type ## Array a = env_->new_fn(size); \
1049 env_->get_region_fn(a, 0, size, nullptr); \
1050 jni_abort_catcher.Check("buf == null"); \
1051 env_->set_region_fn(a, 0, size, nullptr); \
1052 jni_abort_catcher.Check("buf == null"); \
Ian Rogers68d8b422014-07-17 11:09:10 -07001053 down_cast<JNIEnvExt*>(env_)->SetCheckJniEnabled(true); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001054 } \
Elliott Hughes814e4032011-08-23 12:07:56 -07001055 /* Allocate an array and check it has the right type and length. */ \
1056 scalar_type ## Array a = env_->new_fn(size); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001057 EXPECT_NE(a, nullptr); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001058 EXPECT_TRUE(env_->IsInstanceOf(a, env_->FindClass(expected_class_descriptor))); \
1059 EXPECT_EQ(size, env_->GetArrayLength(a)); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001060 \
1061 /* GetPrimitiveArrayRegion/SetPrimitiveArrayRegion */ \
Elliott Hughes814e4032011-08-23 12:07:56 -07001062 /* AIOOBE for negative start offset. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001063 env_->get_region_fn(a, -1, 1, nullptr); \
1064 ExpectException(aioobe_); \
1065 env_->set_region_fn(a, -1, 1, nullptr); \
1066 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001067 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001068 /* AIOOBE for negative length. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001069 env_->get_region_fn(a, 0, -1, nullptr); \
1070 ExpectException(aioobe_); \
1071 env_->set_region_fn(a, 0, -1, nullptr); \
1072 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001073 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001074 /* AIOOBE for buffer overrun. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001075 env_->get_region_fn(a, size - 1, size, nullptr); \
1076 ExpectException(aioobe_); \
1077 env_->set_region_fn(a, size - 1, size, nullptr); \
1078 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001079 \
Vladimir Marko795e3412015-11-06 16:57:03 +00001080 /* Regression test against integer overflow in range check. */ \
1081 env_->get_region_fn(a, 0x7fffffff, 0x7fffffff, nullptr); \
1082 ExpectException(aioobe_); \
1083 env_->set_region_fn(a, 0x7fffffff, 0x7fffffff, nullptr); \
1084 ExpectException(aioobe_); \
1085 \
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001086 /* It's okay for the buffer to be null as long as the length is 0. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001087 env_->get_region_fn(a, 2, 0, nullptr); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001088 /* Even if the offset is invalid... */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001089 env_->get_region_fn(a, 123, 0, nullptr); \
1090 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001091 \
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001092 /* It's okay for the buffer to be null as long as the length is 0. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001093 env_->set_region_fn(a, 2, 0, nullptr); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001094 /* Even if the offset is invalid... */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001095 env_->set_region_fn(a, 123, 0, nullptr); \
1096 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001097 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001098 /* Prepare a couple of buffers. */ \
Ian Rogers700a4022014-05-19 16:49:03 -07001099 std::unique_ptr<scalar_type[]> src_buf(new scalar_type[size]); \
1100 std::unique_ptr<scalar_type[]> dst_buf(new scalar_type[size]); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001101 for (jsize i = 0; i < size; ++i) { src_buf[i] = scalar_type(i); } \
1102 for (jsize i = 0; i < size; ++i) { dst_buf[i] = scalar_type(-1); } \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001103 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001104 /* Copy all of src_buf onto the heap. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -07001105 env_->set_region_fn(a, 0, size, &src_buf[0]); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001106 /* Copy back only part. */ \
1107 env_->get_region_fn(a, 1, size - 2, &dst_buf[1]); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001108 EXPECT_NE(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
1109 << "short copy equal"; \
Elliott Hughes814e4032011-08-23 12:07:56 -07001110 /* Copy the missing pieces. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -07001111 env_->get_region_fn(a, 0, 1, &dst_buf[0]); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001112 env_->get_region_fn(a, size - 1, 1, &dst_buf[size - 1]); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001113 EXPECT_EQ(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
1114 << "fixed copy not equal"; \
Elliott Hughes814e4032011-08-23 12:07:56 -07001115 /* Copy back the whole array. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -07001116 env_->get_region_fn(a, 0, size, &dst_buf[0]); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001117 EXPECT_EQ(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
1118 << "full copy not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -07001119 /* GetPrimitiveArrayCritical */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001120 void* v = env_->GetPrimitiveArrayCritical(a, nullptr); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001121 EXPECT_EQ(memcmp(&src_buf[0], v, size * sizeof(scalar_type)), 0) \
1122 << "GetPrimitiveArrayCritical not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -07001123 env_->ReleasePrimitiveArrayCritical(a, v, 0); \
1124 /* GetXArrayElements */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001125 scalar_type* xs = env_->get_elements_fn(a, nullptr); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001126 EXPECT_EQ(memcmp(&src_buf[0], xs, size * sizeof(scalar_type)), 0) \
1127 << # get_elements_fn " not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -07001128 env_->release_elements_fn(a, xs, 0); \
Elliott Hughesbd935992011-08-22 11:59:34 -07001129
Elliott Hughes814e4032011-08-23 12:07:56 -07001130TEST_F(JniInternalTest, BooleanArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001131 EXPECT_PRIMITIVE_ARRAY(NewBooleanArray, GetBooleanArrayRegion, SetBooleanArrayRegion,
1132 GetBooleanArrayElements, ReleaseBooleanArrayElements, jboolean, "[Z");
Elliott Hughes814e4032011-08-23 12:07:56 -07001133}
1134TEST_F(JniInternalTest, ByteArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001135 EXPECT_PRIMITIVE_ARRAY(NewByteArray, GetByteArrayRegion, SetByteArrayRegion,
1136 GetByteArrayElements, ReleaseByteArrayElements, jbyte, "[B");
Elliott Hughes814e4032011-08-23 12:07:56 -07001137}
1138TEST_F(JniInternalTest, CharArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001139 EXPECT_PRIMITIVE_ARRAY(NewCharArray, GetCharArrayRegion, SetCharArrayRegion,
1140 GetCharArrayElements, ReleaseCharArrayElements, jchar, "[C");
Elliott Hughes814e4032011-08-23 12:07:56 -07001141}
1142TEST_F(JniInternalTest, DoubleArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001143 EXPECT_PRIMITIVE_ARRAY(NewDoubleArray, GetDoubleArrayRegion, SetDoubleArrayRegion,
1144 GetDoubleArrayElements, ReleaseDoubleArrayElements, jdouble, "[D");
Elliott Hughes814e4032011-08-23 12:07:56 -07001145}
1146TEST_F(JniInternalTest, FloatArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001147 EXPECT_PRIMITIVE_ARRAY(NewFloatArray, GetFloatArrayRegion, SetFloatArrayRegion,
1148 GetFloatArrayElements, ReleaseFloatArrayElements, jfloat, "[F");
Elliott Hughes814e4032011-08-23 12:07:56 -07001149}
1150TEST_F(JniInternalTest, IntArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001151 EXPECT_PRIMITIVE_ARRAY(NewIntArray, GetIntArrayRegion, SetIntArrayRegion,
1152 GetIntArrayElements, ReleaseIntArrayElements, jint, "[I");
Elliott Hughes814e4032011-08-23 12:07:56 -07001153}
1154TEST_F(JniInternalTest, LongArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001155 EXPECT_PRIMITIVE_ARRAY(NewLongArray, GetLongArrayRegion, SetLongArrayRegion,
1156 GetLongArrayElements, ReleaseLongArrayElements, jlong, "[J");
Elliott Hughes814e4032011-08-23 12:07:56 -07001157}
1158TEST_F(JniInternalTest, ShortArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001159 EXPECT_PRIMITIVE_ARRAY(NewShortArray, GetShortArrayRegion, SetShortArrayRegion,
1160 GetShortArrayElements, ReleaseShortArrayElements, jshort, "[S");
Elliott Hughesd8ddfd52011-08-15 14:32:53 -07001161}
1162
Ian Rogers2d10b202014-05-12 19:15:18 -07001163TEST_F(JniInternalTest, GetPrimitiveArrayElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001164 GetPrimitiveArrayElementsOfWrongType(false);
1165 GetPrimitiveArrayElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001166}
1167
1168TEST_F(JniInternalTest, ReleasePrimitiveArrayElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001169 ReleasePrimitiveArrayElementsOfWrongType(false);
1170 ReleasePrimitiveArrayElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001171}
Ian Rogers68d8b422014-07-17 11:09:10 -07001172
Ian Rogers2d10b202014-05-12 19:15:18 -07001173TEST_F(JniInternalTest, GetReleasePrimitiveArrayCriticalOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001174 GetReleasePrimitiveArrayCriticalOfWrongType(false);
1175 GetReleasePrimitiveArrayCriticalOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001176}
1177
1178TEST_F(JniInternalTest, GetPrimitiveArrayRegionElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001179 GetPrimitiveArrayRegionElementsOfWrongType(false);
1180 GetPrimitiveArrayRegionElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001181}
1182
1183TEST_F(JniInternalTest, SetPrimitiveArrayRegionElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001184 SetPrimitiveArrayRegionElementsOfWrongType(false);
1185 SetPrimitiveArrayRegionElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001186}
1187
Elliott Hughesf2682d52011-08-15 16:37:04 -07001188TEST_F(JniInternalTest, NewObjectArray) {
Elliott Hughesbd935992011-08-22 11:59:34 -07001189 jclass element_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001190 ASSERT_NE(element_class, nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -07001191 jclass array_class = env_->FindClass("[Ljava/lang/String;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001192 ASSERT_NE(array_class, nullptr);
Elliott Hughesf2682d52011-08-15 16:37:04 -07001193
Ian Rogers1d99e452014-01-02 17:36:41 -08001194 jobjectArray a = env_->NewObjectArray(0, element_class, nullptr);
Ian Rogers2d10b202014-05-12 19:15:18 -07001195 EXPECT_NE(a, nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -07001196 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
1197 EXPECT_EQ(0, env_->GetArrayLength(a));
1198
Ian Rogers1d99e452014-01-02 17:36:41 -08001199 a = env_->NewObjectArray(1, element_class, nullptr);
Ian Rogers2d10b202014-05-12 19:15:18 -07001200 EXPECT_NE(a, nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -07001201 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
1202 EXPECT_EQ(1, env_->GetArrayLength(a));
Ian Rogers1d99e452014-01-02 17:36:41 -08001203 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 0), nullptr));
Ian Rogers1d99e452014-01-02 17:36:41 -08001204
Ian Rogers2d10b202014-05-12 19:15:18 -07001205 // Negative array length checks.
Ian Rogers68d8b422014-07-17 11:09:10 -07001206 NewObjectArrayBadArguments(false);
1207 NewObjectArrayBadArguments(true);
Ian Rogers1d99e452014-01-02 17:36:41 -08001208}
1209
1210TEST_F(JniInternalTest, NewObjectArrayWithPrimitiveClasses) {
1211 const char* primitive_descriptors = "VZBSCIJFD";
1212 const char* primitive_names[] = {
1213 "void", "boolean", "byte", "short", "char", "int", "long", "float", "double"
1214 };
1215 ASSERT_EQ(strlen(primitive_descriptors), arraysize(primitive_names));
1216
Ian Rogers68d8b422014-07-17 11:09:10 -07001217 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers1d99e452014-01-02 17:36:41 -08001218 CheckJniAbortCatcher jni_abort_catcher;
1219 for (size_t i = 0; i < strlen(primitive_descriptors); ++i) {
Ian Rogers2d10b202014-05-12 19:15:18 -07001220 env_->NewObjectArray(0, nullptr, nullptr);
1221 jni_abort_catcher.Check("element_jclass == null");
Ian Rogers1d99e452014-01-02 17:36:41 -08001222 jclass primitive_class = GetPrimitiveClass(primitive_descriptors[i]);
1223 env_->NewObjectArray(1, primitive_class, nullptr);
1224 std::string error_msg(StringPrintf("not an object type: %s", primitive_names[i]));
1225 jni_abort_catcher.Check(error_msg.c_str());
1226 }
Ian Rogers68d8b422014-07-17 11:09:10 -07001227 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1228 for (size_t i = 0; i < strlen(primitive_descriptors); ++i) {
1229 env_->NewObjectArray(0, nullptr, nullptr);
1230 jni_abort_catcher.Check("NewObjectArray received NULL jclass");
1231 jclass primitive_class = GetPrimitiveClass(primitive_descriptors[i]);
1232 env_->NewObjectArray(1, primitive_class, nullptr);
1233 std::string error_msg(StringPrintf("not an object type: %s", primitive_names[i]));
1234 jni_abort_catcher.Check(error_msg.c_str());
1235 }
1236 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Ian Rogers1d99e452014-01-02 17:36:41 -08001237}
1238
1239TEST_F(JniInternalTest, NewObjectArrayWithInitialValue) {
1240 jclass element_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001241 ASSERT_NE(element_class, nullptr);
Ian Rogers1d99e452014-01-02 17:36:41 -08001242 jclass array_class = env_->FindClass("[Ljava/lang/String;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001243 ASSERT_NE(array_class, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001244
1245 jstring s = env_->NewStringUTF("poop");
Ian Rogers1d99e452014-01-02 17:36:41 -08001246 jobjectArray a = env_->NewObjectArray(2, element_class, s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001247 EXPECT_NE(a, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001248 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
1249 EXPECT_EQ(2, env_->GetArrayLength(a));
1250 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 0), s));
1251 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 1), s));
Ian Rogers1d99e452014-01-02 17:36:41 -08001252
1253 // Attempt to incorrect create an array of strings with initial value of string arrays.
1254 CheckJniAbortCatcher jni_abort_catcher;
1255 env_->NewObjectArray(2, element_class, a);
1256 jni_abort_catcher.Check("cannot assign object of type 'java.lang.String[]' to array with element "
1257 "type of 'java.lang.String'");
Elliott Hughesbd935992011-08-22 11:59:34 -07001258}
1259
1260TEST_F(JniInternalTest, GetArrayLength) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001261 // Already tested in NewObjectArray/NewPrimitiveArray except for null.
Ian Rogers2f022bd2014-11-11 08:43:05 -08001262 CheckJniAbortCatcher jni_abort_catcher;
1263 bool old_check_jni = vm_->SetCheckJniEnabled(false);
1264 EXPECT_EQ(0, env_->GetArrayLength(nullptr));
1265 jni_abort_catcher.Check("java_array == null");
1266 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1267 EXPECT_EQ(JNI_ERR, env_->GetArrayLength(nullptr));
1268 jni_abort_catcher.Check("jarray was NULL");
1269 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes8a26c5c2011-08-15 18:35:43 -07001270}
1271
Elliott Hughes37f7a402011-08-22 18:56:01 -07001272TEST_F(JniInternalTest, GetObjectClass) {
1273 jclass string_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001274 ASSERT_NE(string_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001275 jclass class_class = env_->FindClass("java/lang/Class");
Ian Rogers2d10b202014-05-12 19:15:18 -07001276 ASSERT_NE(class_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001277
1278 jstring s = env_->NewStringUTF("poop");
1279 jclass c = env_->GetObjectClass(s);
1280 ASSERT_TRUE(env_->IsSameObject(string_class, c));
1281
1282 jclass c2 = env_->GetObjectClass(c);
1283 ASSERT_TRUE(env_->IsSameObject(class_class, env_->GetObjectClass(c2)));
Ian Rogers2d10b202014-05-12 19:15:18 -07001284
1285 // Null as object should fail.
1286 CheckJniAbortCatcher jni_abort_catcher;
1287 EXPECT_EQ(env_->GetObjectClass(nullptr), nullptr);
1288 jni_abort_catcher.Check("java_object == null");
Elliott Hughes37f7a402011-08-22 18:56:01 -07001289}
1290
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001291TEST_F(JniInternalTest, GetSuperclass) {
1292 jclass object_class = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07001293 ASSERT_NE(object_class, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001294 jclass string_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001295 ASSERT_NE(string_class, nullptr);
Ian Rogersdc180202012-01-29 14:47:29 -08001296 jclass runnable_interface = env_->FindClass("java/lang/Runnable");
Ian Rogers2d10b202014-05-12 19:15:18 -07001297 ASSERT_NE(runnable_interface, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001298 ASSERT_TRUE(env_->IsSameObject(object_class, env_->GetSuperclass(string_class)));
Ian Rogers2d10b202014-05-12 19:15:18 -07001299 ASSERT_EQ(env_->GetSuperclass(object_class), nullptr);
Brian Carlstrom08ac9222015-05-22 13:43:00 -07001300 ASSERT_EQ(env_->GetSuperclass(runnable_interface), nullptr);
Ian Rogers2d10b202014-05-12 19:15:18 -07001301
1302 // Null as class should fail.
1303 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001304 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers2d10b202014-05-12 19:15:18 -07001305 EXPECT_EQ(env_->GetSuperclass(nullptr), nullptr);
1306 jni_abort_catcher.Check("java_class == null");
Ian Rogers68d8b422014-07-17 11:09:10 -07001307 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1308 EXPECT_EQ(env_->GetSuperclass(nullptr), nullptr);
1309 jni_abort_catcher.Check("GetSuperclass received NULL jclass");
1310 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001311}
1312
Elliott Hughes37f7a402011-08-22 18:56:01 -07001313TEST_F(JniInternalTest, IsAssignableFrom) {
1314 jclass object_class = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07001315 ASSERT_NE(object_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001316 jclass string_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001317 ASSERT_NE(string_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001318
Narayan Kamath1268b742014-07-11 19:15:11 +01001319 // A superclass is assignable from an instance of its
1320 // subclass but not vice versa.
1321 ASSERT_TRUE(env_->IsAssignableFrom(string_class, object_class));
1322 ASSERT_FALSE(env_->IsAssignableFrom(object_class, string_class));
1323
1324 jclass charsequence_interface = env_->FindClass("java/lang/CharSequence");
1325 ASSERT_NE(charsequence_interface, nullptr);
1326
1327 // An interface is assignable from an instance of an implementing
1328 // class but not vice versa.
1329 ASSERT_TRUE(env_->IsAssignableFrom(string_class, charsequence_interface));
1330 ASSERT_FALSE(env_->IsAssignableFrom(charsequence_interface, string_class));
1331
1332 // Check that arrays are covariant.
1333 jclass string_array_class = env_->FindClass("[Ljava/lang/String;");
1334 ASSERT_NE(string_array_class, nullptr);
1335 jclass object_array_class = env_->FindClass("[Ljava/lang/Object;");
1336 ASSERT_NE(object_array_class, nullptr);
1337 ASSERT_TRUE(env_->IsAssignableFrom(string_array_class, object_array_class));
1338 ASSERT_FALSE(env_->IsAssignableFrom(object_array_class, string_array_class));
1339
1340 // Primitive types are tested in 004-JniTest.
Ian Rogers2d10b202014-05-12 19:15:18 -07001341
1342 // Null as either class should fail.
1343 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001344 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers2d10b202014-05-12 19:15:18 -07001345 EXPECT_EQ(env_->IsAssignableFrom(nullptr, string_class), JNI_FALSE);
1346 jni_abort_catcher.Check("java_class1 == null");
1347 EXPECT_EQ(env_->IsAssignableFrom(object_class, nullptr), JNI_FALSE);
1348 jni_abort_catcher.Check("java_class2 == null");
Ian Rogers68d8b422014-07-17 11:09:10 -07001349 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1350 EXPECT_EQ(env_->IsAssignableFrom(nullptr, string_class), JNI_FALSE);
1351 jni_abort_catcher.Check("IsAssignableFrom received NULL jclass");
1352 EXPECT_EQ(env_->IsAssignableFrom(object_class, nullptr), JNI_FALSE);
1353 jni_abort_catcher.Check("IsAssignableFrom received NULL jclass");
1354 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes37f7a402011-08-22 18:56:01 -07001355}
1356
Elliott Hughesb465ab02011-08-24 11:21:21 -07001357TEST_F(JniInternalTest, GetObjectRefType) {
1358 jclass local = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07001359 ASSERT_TRUE(local != nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001360 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(local));
1361
1362 jobject global = env_->NewGlobalRef(local);
1363 EXPECT_EQ(JNIGlobalRefType, env_->GetObjectRefType(global));
1364
1365 jweak weak_global = env_->NewWeakGlobalRef(local);
1366 EXPECT_EQ(JNIWeakGlobalRefType, env_->GetObjectRefType(weak_global));
1367
Andreas Gampea8763072014-12-20 00:08:35 -08001368 {
1369 CheckJniAbortCatcher jni_abort_catcher;
1370 jobject invalid = reinterpret_cast<jobject>(this);
1371 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(invalid));
1372 jni_abort_catcher.Check("use of invalid jobject");
1373 }
Elliott Hughesb465ab02011-08-24 11:21:21 -07001374
1375 // TODO: invoke a native method and test that its arguments are considered local references.
Ian Rogers2d10b202014-05-12 19:15:18 -07001376
Andreas Gampea8763072014-12-20 00:08:35 -08001377 // Null as pointer should not fail and return invalid-ref. b/18820997
Ian Rogers2d10b202014-05-12 19:15:18 -07001378 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(nullptr));
Andreas Gampea8763072014-12-20 00:08:35 -08001379
1380 // TODO: Null as reference should return the original type.
1381 // This requires running a GC so a non-null object gets freed.
Elliott Hughesb465ab02011-08-24 11:21:21 -07001382}
1383
Mathieu Chartier08599992013-12-20 17:17:55 -08001384TEST_F(JniInternalTest, StaleWeakGlobal) {
1385 jclass java_lang_Class = env_->FindClass("java/lang/Class");
Ian Rogers2d10b202014-05-12 19:15:18 -07001386 ASSERT_NE(java_lang_Class, nullptr);
1387 jobjectArray local_ref = env_->NewObjectArray(1, java_lang_Class, nullptr);
1388 ASSERT_NE(local_ref, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001389 jweak weak_global = env_->NewWeakGlobalRef(local_ref);
Ian Rogers2d10b202014-05-12 19:15:18 -07001390 ASSERT_NE(weak_global, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001391 env_->DeleteLocalRef(local_ref);
1392 Runtime::Current()->GetHeap()->CollectGarbage(false); // GC should clear the weak global.
1393 jobject new_global_ref = env_->NewGlobalRef(weak_global);
Ian Rogers2d10b202014-05-12 19:15:18 -07001394 EXPECT_EQ(new_global_ref, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001395 jobject new_local_ref = env_->NewLocalRef(weak_global);
Ian Rogers2d10b202014-05-12 19:15:18 -07001396 EXPECT_EQ(new_local_ref, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001397}
1398
Elliott Hughes8a26c5c2011-08-15 18:35:43 -07001399TEST_F(JniInternalTest, NewStringUTF) {
Ian Rogers2d10b202014-05-12 19:15:18 -07001400 EXPECT_EQ(env_->NewStringUTF(nullptr), nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001401 jstring s;
1402
1403 s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001404 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001405 EXPECT_EQ(0, env_->GetStringLength(s));
1406 EXPECT_EQ(0, env_->GetStringUTFLength(s));
1407 s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001408 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001409 EXPECT_EQ(5, env_->GetStringLength(s));
1410 EXPECT_EQ(5, env_->GetStringUTFLength(s));
1411
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001412 // Encoded surrogate pair.
1413 s = env_->NewStringUTF("\xed\xa0\x81\xed\xb0\x80");
1414 EXPECT_NE(s, nullptr);
1415 EXPECT_EQ(2, env_->GetStringLength(s));
Narayan Kamathe16dad12015-02-13 11:49:22 +00001416
1417 // The surrogate pair gets encoded into a 4 byte UTF sequence..
1418 EXPECT_EQ(4, env_->GetStringUTFLength(s));
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001419 const char* chars = env_->GetStringUTFChars(s, nullptr);
Narayan Kamathe16dad12015-02-13 11:49:22 +00001420 EXPECT_STREQ("\xf0\x90\x90\x80", chars);
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001421 env_->ReleaseStringUTFChars(s, chars);
1422
Narayan Kamathe16dad12015-02-13 11:49:22 +00001423 // .. but is stored as is in the utf-16 representation.
1424 const jchar* jchars = env_->GetStringChars(s, nullptr);
1425 EXPECT_EQ(0xd801, jchars[0]);
1426 EXPECT_EQ(0xdc00, jchars[1]);
1427 env_->ReleaseStringChars(s, jchars);
1428
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001429 // 4 byte UTF sequence appended to an encoded surrogate pair.
1430 s = env_->NewStringUTF("\xed\xa0\x81\xed\xb0\x80 \xf0\x9f\x8f\xa0");
1431 EXPECT_NE(s, nullptr);
Narayan Kamathe16dad12015-02-13 11:49:22 +00001432
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001433 // The 4 byte sequence {0xf0, 0x9f, 0x8f, 0xa0} is converted into a surrogate
Narayan Kamathe16dad12015-02-13 11:49:22 +00001434 // pair {0xd83c, 0xdfe0}.
1435 EXPECT_EQ(5, env_->GetStringLength(s));
1436 jchars = env_->GetStringChars(s, nullptr);
1437 // The first surrogate pair, encoded as such in the input.
1438 EXPECT_EQ(0xd801, jchars[0]);
1439 EXPECT_EQ(0xdc00, jchars[1]);
1440 // The second surrogate pair, from the 4 byte UTF sequence in the input.
1441 EXPECT_EQ(0xd83c, jchars[3]);
1442 EXPECT_EQ(0xdfe0, jchars[4]);
1443 env_->ReleaseStringChars(s, jchars);
1444
1445 EXPECT_EQ(9, env_->GetStringUTFLength(s));
1446 chars = env_->GetStringUTFChars(s, nullptr);
1447 EXPECT_STREQ("\xf0\x90\x90\x80 \xf0\x9f\x8f\xa0", chars);
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001448 env_->ReleaseStringUTFChars(s, chars);
1449
1450 // A string with 1, 2, 3 and 4 byte UTF sequences with spaces
1451 // between them
1452 s = env_->NewStringUTF("\x24 \xc2\xa2 \xe2\x82\xac \xf0\x9f\x8f\xa0");
1453 EXPECT_NE(s, nullptr);
1454 EXPECT_EQ(8, env_->GetStringLength(s));
Narayan Kamathe16dad12015-02-13 11:49:22 +00001455 EXPECT_EQ(13, env_->GetStringUTFLength(s));
Elliott Hughesf2682d52011-08-15 16:37:04 -07001456}
1457
Elliott Hughes814e4032011-08-23 12:07:56 -07001458TEST_F(JniInternalTest, NewString) {
Elliott Hughes814e4032011-08-23 12:07:56 -07001459 jchar chars[] = { 'h', 'i' };
1460 jstring s;
1461 s = env_->NewString(chars, 0);
Ian Rogers2d10b202014-05-12 19:15:18 -07001462 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001463 EXPECT_EQ(0, env_->GetStringLength(s));
1464 EXPECT_EQ(0, env_->GetStringUTFLength(s));
1465 s = env_->NewString(chars, 2);
Ian Rogers2d10b202014-05-12 19:15:18 -07001466 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001467 EXPECT_EQ(2, env_->GetStringLength(s));
1468 EXPECT_EQ(2, env_->GetStringUTFLength(s));
1469
1470 // TODO: check some non-ASCII strings.
1471}
1472
Jesse Wilson25e79a52011-11-18 15:31:58 -05001473TEST_F(JniInternalTest, NewStringNullCharsZeroLength) {
Ian Rogers1d99e452014-01-02 17:36:41 -08001474 jstring s = env_->NewString(nullptr, 0);
Ian Rogers2d10b202014-05-12 19:15:18 -07001475 EXPECT_NE(s, nullptr);
Jesse Wilson25e79a52011-11-18 15:31:58 -05001476 EXPECT_EQ(0, env_->GetStringLength(s));
1477}
1478
Ian Rogers1d99e452014-01-02 17:36:41 -08001479TEST_F(JniInternalTest, NewStringNullCharsNonzeroLength) {
1480 CheckJniAbortCatcher jni_abort_catcher;
1481 env_->NewString(nullptr, 1);
1482 jni_abort_catcher.Check("chars == null && char_count > 0");
1483}
1484
1485TEST_F(JniInternalTest, NewStringNegativeLength) {
1486 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001487 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers1d99e452014-01-02 17:36:41 -08001488 env_->NewString(nullptr, -1);
1489 jni_abort_catcher.Check("char_count < 0: -1");
1490 env_->NewString(nullptr, std::numeric_limits<jint>::min());
1491 jni_abort_catcher.Check("char_count < 0: -2147483648");
Ian Rogers68d8b422014-07-17 11:09:10 -07001492 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1493 env_->NewString(nullptr, -1);
1494 jni_abort_catcher.Check("negative jsize: -1");
1495 env_->NewString(nullptr, std::numeric_limits<jint>::min());
1496 jni_abort_catcher.Check("negative jsize: -2147483648");
1497 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Jesse Wilson25e79a52011-11-18 15:31:58 -05001498}
1499
Elliott Hughesb465ab02011-08-24 11:21:21 -07001500TEST_F(JniInternalTest, GetStringLength_GetStringUTFLength) {
1501 // Already tested in the NewString/NewStringUTF tests.
1502}
1503
1504TEST_F(JniInternalTest, GetStringRegion_GetStringUTFRegion) {
1505 jstring s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001506 ASSERT_TRUE(s != nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001507
Ian Rogers2d10b202014-05-12 19:15:18 -07001508 env_->GetStringRegion(s, -1, 0, nullptr);
1509 ExpectException(sioobe_);
1510 env_->GetStringRegion(s, 0, -1, nullptr);
1511 ExpectException(sioobe_);
1512 env_->GetStringRegion(s, 0, 10, nullptr);
1513 ExpectException(sioobe_);
1514 env_->GetStringRegion(s, 10, 1, nullptr);
1515 ExpectException(sioobe_);
Vladimir Marko795e3412015-11-06 16:57:03 +00001516 // Regression test against integer overflow in range check.
1517 env_->GetStringRegion(s, 0x7fffffff, 0x7fffffff, nullptr);
1518 ExpectException(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001519
1520 jchar chars[4] = { 'x', 'x', 'x', 'x' };
1521 env_->GetStringRegion(s, 1, 2, &chars[1]);
1522 EXPECT_EQ('x', chars[0]);
1523 EXPECT_EQ('e', chars[1]);
1524 EXPECT_EQ('l', chars[2]);
1525 EXPECT_EQ('x', chars[3]);
1526
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001527 // It's okay for the buffer to be null as long as the length is 0.
Ian Rogers2d10b202014-05-12 19:15:18 -07001528 env_->GetStringRegion(s, 2, 0, nullptr);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001529 // Even if the offset is invalid...
Ian Rogers2d10b202014-05-12 19:15:18 -07001530 env_->GetStringRegion(s, 123, 0, nullptr);
1531 ExpectException(sioobe_);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001532
Ian Rogers2d10b202014-05-12 19:15:18 -07001533 env_->GetStringUTFRegion(s, -1, 0, nullptr);
1534 ExpectException(sioobe_);
1535 env_->GetStringUTFRegion(s, 0, -1, nullptr);
1536 ExpectException(sioobe_);
1537 env_->GetStringUTFRegion(s, 0, 10, nullptr);
1538 ExpectException(sioobe_);
1539 env_->GetStringUTFRegion(s, 10, 1, nullptr);
1540 ExpectException(sioobe_);
Vladimir Marko795e3412015-11-06 16:57:03 +00001541 // Regression test against integer overflow in range check.
1542 env_->GetStringUTFRegion(s, 0x7fffffff, 0x7fffffff, nullptr);
1543 ExpectException(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001544
1545 char bytes[4] = { 'x', 'x', 'x', 'x' };
1546 env_->GetStringUTFRegion(s, 1, 2, &bytes[1]);
1547 EXPECT_EQ('x', bytes[0]);
1548 EXPECT_EQ('e', bytes[1]);
1549 EXPECT_EQ('l', bytes[2]);
1550 EXPECT_EQ('x', bytes[3]);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001551
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001552 // It's okay for the buffer to be null as long as the length is 0.
Ian Rogers2d10b202014-05-12 19:15:18 -07001553 env_->GetStringUTFRegion(s, 2, 0, nullptr);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001554 // Even if the offset is invalid...
Ian Rogers2d10b202014-05-12 19:15:18 -07001555 env_->GetStringUTFRegion(s, 123, 0, nullptr);
1556 ExpectException(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001557}
1558
Elliott Hughes75770752011-08-24 17:52:38 -07001559TEST_F(JniInternalTest, GetStringUTFChars_ReleaseStringUTFChars) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001560 // Passing in a null jstring is ignored normally, but caught by -Xcheck:jni.
Ian Rogers68d8b422014-07-17 11:09:10 -07001561 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Elliott Hughesb264f082012-04-06 17:10:10 -07001562 {
Elliott Hughesb264f082012-04-06 17:10:10 -07001563 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers2d10b202014-05-12 19:15:18 -07001564 EXPECT_EQ(env_->GetStringUTFChars(nullptr, nullptr), nullptr);
Ian Rogers68d8b422014-07-17 11:09:10 -07001565 }
1566 {
1567 CheckJniAbortCatcher check_jni_abort_catcher;
1568 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1569 EXPECT_EQ(env_->GetStringUTFChars(nullptr, nullptr), nullptr);
1570 check_jni_abort_catcher.Check("GetStringUTFChars received NULL jstring");
1571 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07001572 }
Elliott Hughes75770752011-08-24 17:52:38 -07001573
1574 jstring s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001575 ASSERT_TRUE(s != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001576
Ian Rogers2d10b202014-05-12 19:15:18 -07001577 const char* utf = env_->GetStringUTFChars(s, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001578 EXPECT_STREQ("hello", utf);
1579 env_->ReleaseStringUTFChars(s, utf);
1580
1581 jboolean is_copy = JNI_FALSE;
1582 utf = env_->GetStringUTFChars(s, &is_copy);
1583 EXPECT_EQ(JNI_TRUE, is_copy);
1584 EXPECT_STREQ("hello", utf);
1585 env_->ReleaseStringUTFChars(s, utf);
1586}
1587
1588TEST_F(JniInternalTest, GetStringChars_ReleaseStringChars) {
1589 jstring s = env_->NewStringUTF("hello");
Fred Shih56890e22014-06-02 11:11:52 -07001590 ScopedObjectAccess soa(env_);
1591 mirror::String* s_m = soa.Decode<mirror::String*>(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001592 ASSERT_TRUE(s != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001593
1594 jchar expected[] = { 'h', 'e', 'l', 'l', 'o' };
Ian Rogers2d10b202014-05-12 19:15:18 -07001595 const jchar* chars = env_->GetStringChars(s, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001596 EXPECT_EQ(expected[0], chars[0]);
1597 EXPECT_EQ(expected[1], chars[1]);
1598 EXPECT_EQ(expected[2], chars[2]);
1599 EXPECT_EQ(expected[3], chars[3]);
1600 EXPECT_EQ(expected[4], chars[4]);
1601 env_->ReleaseStringChars(s, chars);
1602
1603 jboolean is_copy = JNI_FALSE;
1604 chars = env_->GetStringChars(s, &is_copy);
Jeff Hao848f70a2014-01-15 13:49:50 -08001605 if (Runtime::Current()->GetHeap()->IsMovableObject(s_m)) {
Fred Shih56890e22014-06-02 11:11:52 -07001606 EXPECT_EQ(JNI_TRUE, is_copy);
1607 } else {
1608 EXPECT_EQ(JNI_FALSE, is_copy);
1609 }
Elliott Hughes75770752011-08-24 17:52:38 -07001610 EXPECT_EQ(expected[0], chars[0]);
1611 EXPECT_EQ(expected[1], chars[1]);
1612 EXPECT_EQ(expected[2], chars[2]);
1613 EXPECT_EQ(expected[3], chars[3]);
1614 EXPECT_EQ(expected[4], chars[4]);
1615 env_->ReleaseStringChars(s, chars);
1616}
1617
1618TEST_F(JniInternalTest, GetStringCritical_ReleaseStringCritical) {
1619 jstring s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001620 ASSERT_TRUE(s != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001621
1622 jchar expected[] = { 'h', 'e', 'l', 'l', 'o' };
Ian Rogers2d10b202014-05-12 19:15:18 -07001623 const jchar* chars = env_->GetStringCritical(s, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001624 EXPECT_EQ(expected[0], chars[0]);
1625 EXPECT_EQ(expected[1], chars[1]);
1626 EXPECT_EQ(expected[2], chars[2]);
1627 EXPECT_EQ(expected[3], chars[3]);
1628 EXPECT_EQ(expected[4], chars[4]);
1629 env_->ReleaseStringCritical(s, chars);
1630
Fred Shih56890e22014-06-02 11:11:52 -07001631 jboolean is_copy = JNI_TRUE;
Elliott Hughes75770752011-08-24 17:52:38 -07001632 chars = env_->GetStringCritical(s, &is_copy);
Fred Shih56890e22014-06-02 11:11:52 -07001633 EXPECT_EQ(JNI_FALSE, is_copy);
Elliott Hughes75770752011-08-24 17:52:38 -07001634 EXPECT_EQ(expected[0], chars[0]);
1635 EXPECT_EQ(expected[1], chars[1]);
1636 EXPECT_EQ(expected[2], chars[2]);
1637 EXPECT_EQ(expected[3], chars[3]);
1638 EXPECT_EQ(expected[4], chars[4]);
1639 env_->ReleaseStringCritical(s, chars);
1640}
1641
Elliott Hughes814e4032011-08-23 12:07:56 -07001642TEST_F(JniInternalTest, GetObjectArrayElement_SetObjectArrayElement) {
Elliott Hughesb264f082012-04-06 17:10:10 -07001643 jclass java_lang_Class = env_->FindClass("java/lang/Class");
Ian Rogers2d10b202014-05-12 19:15:18 -07001644 ASSERT_TRUE(java_lang_Class != nullptr);
Elliott Hughes289da822011-08-16 10:11:20 -07001645
Ian Rogers2d10b202014-05-12 19:15:18 -07001646 jobjectArray array = env_->NewObjectArray(1, java_lang_Class, nullptr);
1647 EXPECT_NE(array, nullptr);
1648 EXPECT_EQ(env_->GetObjectArrayElement(array, 0), nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -07001649 env_->SetObjectArrayElement(array, 0, java_lang_Class);
1650 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(array, 0), java_lang_Class));
Elliott Hughesa5b897e2011-08-16 11:33:06 -07001651
1652 // ArrayIndexOutOfBounds for negative index.
Elliott Hughesb264f082012-04-06 17:10:10 -07001653 env_->SetObjectArrayElement(array, -1, java_lang_Class);
Ian Rogers2d10b202014-05-12 19:15:18 -07001654 ExpectException(aioobe_);
Elliott Hughesa5b897e2011-08-16 11:33:06 -07001655
1656 // ArrayIndexOutOfBounds for too-large index.
Elliott Hughesb264f082012-04-06 17:10:10 -07001657 env_->SetObjectArrayElement(array, 1, java_lang_Class);
Ian Rogers2d10b202014-05-12 19:15:18 -07001658 ExpectException(aioobe_);
Elliott Hughesa5b897e2011-08-16 11:33:06 -07001659
Elliott Hughesb264f082012-04-06 17:10:10 -07001660 // ArrayStoreException thrown for bad types.
1661 env_->SetObjectArrayElement(array, 0, env_->NewStringUTF("not a jclass!"));
Ian Rogers2d10b202014-05-12 19:15:18 -07001662 ExpectException(ase_);
1663
1664 // Null as array should fail.
1665 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001666 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers2d10b202014-05-12 19:15:18 -07001667 EXPECT_EQ(nullptr, env_->GetObjectArrayElement(nullptr, 0));
1668 jni_abort_catcher.Check("java_array == null");
1669 env_->SetObjectArrayElement(nullptr, 0, nullptr);
1670 jni_abort_catcher.Check("java_array == null");
Ian Rogers68d8b422014-07-17 11:09:10 -07001671 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1672 EXPECT_EQ(nullptr, env_->GetObjectArrayElement(nullptr, 0));
1673 jni_abort_catcher.Check("jarray was NULL");
1674 env_->SetObjectArrayElement(nullptr, 0, nullptr);
1675 jni_abort_catcher.Check("jarray was NULL");
1676 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes289da822011-08-16 10:11:20 -07001677}
1678
Ian Rogers647b1a82014-10-10 11:02:11 -07001679#define EXPECT_STATIC_PRIMITIVE_FIELD(expect_eq, type, field_name, sig, value1, value2) \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001680 do { \
1681 jfieldID fid = env_->GetStaticFieldID(c, field_name, sig); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001682 EXPECT_NE(fid, nullptr); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001683 env_->SetStatic ## type ## Field(c, fid, value1); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001684 expect_eq(value1, env_->GetStatic ## type ## Field(c, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001685 env_->SetStatic ## type ## Field(c, fid, value2); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001686 expect_eq(value2, env_->GetStatic ## type ## Field(c, fid)); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001687 \
Ian Rogers68d8b422014-07-17 11:09:10 -07001688 bool old_check_jni = vm_->SetCheckJniEnabled(false); \
1689 { \
1690 CheckJniAbortCatcher jni_abort_catcher; \
1691 env_->GetStatic ## type ## Field(nullptr, fid); \
1692 env_->SetStatic ## type ## Field(nullptr, fid, value1); \
1693 } \
Ian Rogers2d10b202014-05-12 19:15:18 -07001694 CheckJniAbortCatcher jni_abort_catcher; \
Ian Rogers2d10b202014-05-12 19:15:18 -07001695 env_->GetStatic ## type ## Field(c, nullptr); \
1696 jni_abort_catcher.Check("fid == null"); \
1697 env_->SetStatic ## type ## Field(c, nullptr, value1); \
1698 jni_abort_catcher.Check("fid == null"); \
Ian Rogers68d8b422014-07-17 11:09:10 -07001699 \
1700 EXPECT_FALSE(vm_->SetCheckJniEnabled(true)); \
1701 env_->GetStatic ## type ## Field(nullptr, fid); \
1702 jni_abort_catcher.Check("received NULL jclass"); \
1703 env_->SetStatic ## type ## Field(nullptr, fid, value1); \
1704 jni_abort_catcher.Check("received NULL jclass"); \
1705 env_->GetStatic ## type ## Field(c, nullptr); \
1706 jni_abort_catcher.Check("jfieldID was NULL"); \
1707 env_->SetStatic ## type ## Field(c, nullptr, value1); \
1708 jni_abort_catcher.Check("jfieldID was NULL"); \
1709 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001710 } while (false)
1711
Ian Rogers647b1a82014-10-10 11:02:11 -07001712#define EXPECT_PRIMITIVE_FIELD(expect_eq, instance, type, field_name, sig, value1, value2) \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001713 do { \
1714 jfieldID fid = env_->GetFieldID(c, field_name, sig); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001715 EXPECT_NE(fid, nullptr); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001716 env_->Set ## type ## Field(instance, fid, value1); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001717 expect_eq(value1, env_->Get ## type ## Field(instance, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001718 env_->Set ## type ## Field(instance, fid, value2); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001719 expect_eq(value2, env_->Get ## type ## Field(instance, fid)); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001720 \
Ian Rogers68d8b422014-07-17 11:09:10 -07001721 bool old_check_jni = vm_->SetCheckJniEnabled(false); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001722 CheckJniAbortCatcher jni_abort_catcher; \
1723 env_->Get ## type ## Field(nullptr, fid); \
1724 jni_abort_catcher.Check("obj == null"); \
1725 env_->Set ## type ## Field(nullptr, fid, value1); \
1726 jni_abort_catcher.Check("obj == null"); \
1727 env_->Get ## type ## Field(instance, nullptr); \
1728 jni_abort_catcher.Check("fid == null"); \
1729 env_->Set ## type ## Field(instance, nullptr, value1); \
1730 jni_abort_catcher.Check("fid == null"); \
Ian Rogers68d8b422014-07-17 11:09:10 -07001731 EXPECT_FALSE(vm_->SetCheckJniEnabled(true)); \
1732 env_->Get ## type ## Field(nullptr, fid); \
1733 jni_abort_catcher.Check("field operation on NULL object:"); \
1734 env_->Set ## type ## Field(nullptr, fid, value1); \
1735 jni_abort_catcher.Check("field operation on NULL object:"); \
1736 env_->Get ## type ## Field(instance, nullptr); \
1737 jni_abort_catcher.Check("jfieldID was NULL"); \
1738 env_->Set ## type ## Field(instance, nullptr, value1); \
1739 jni_abort_catcher.Check("jfieldID was NULL"); \
1740 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001741 } while (false)
1742
1743
1744TEST_F(JniInternalTest, GetPrimitiveField_SetPrimitiveField) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001745 Thread::Current()->TransitionFromSuspendedToRunnable();
1746 LoadDex("AllFields");
Brian Carlstrombd86bcc2013-03-10 20:26:16 -07001747 bool started = runtime_->Start();
Ian Rogers2d10b202014-05-12 19:15:18 -07001748 ASSERT_TRUE(started);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001749
1750 jclass c = env_->FindClass("AllFields");
Ian Rogers2d10b202014-05-12 19:15:18 -07001751 ASSERT_NE(c, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001752 jobject o = env_->AllocObject(c);
Ian Rogers2d10b202014-05-12 19:15:18 -07001753 ASSERT_NE(o, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001754
Ian Rogers647b1a82014-10-10 11:02:11 -07001755 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Boolean, "sZ", "Z", JNI_TRUE, JNI_FALSE);
1756 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Byte, "sB", "B", 1, 2);
1757 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Char, "sC", "C", 'a', 'b');
1758 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_DOUBLE_EQ, Double, "sD", "D", 1.0, 2.0);
1759 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_FLOAT_EQ, Float, "sF", "F", 1.0, 2.0);
1760 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Int, "sI", "I", 1, 2);
1761 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Long, "sJ", "J", 1, 2);
1762 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Short, "sS", "S", 1, 2);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001763
Ian Rogers647b1a82014-10-10 11:02:11 -07001764 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Boolean, "iZ", "Z", JNI_TRUE, JNI_FALSE);
1765 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Byte, "iB", "B", 1, 2);
1766 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Char, "iC", "C", 'a', 'b');
1767 EXPECT_PRIMITIVE_FIELD(EXPECT_DOUBLE_EQ, o, Double, "iD", "D", 1.0, 2.0);
1768 EXPECT_PRIMITIVE_FIELD(EXPECT_FLOAT_EQ, o, Float, "iF", "F", 1.0, 2.0);
1769 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Int, "iI", "I", 1, 2);
1770 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Long, "iJ", "J", 1, 2);
1771 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Short, "iS", "S", 1, 2);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001772}
1773
1774TEST_F(JniInternalTest, GetObjectField_SetObjectField) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001775 Thread::Current()->TransitionFromSuspendedToRunnable();
1776 LoadDex("AllFields");
Brian Carlstrom25c33252011-09-18 15:58:35 -07001777 runtime_->Start();
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001778
1779 jclass c = env_->FindClass("AllFields");
Ian Rogers2d10b202014-05-12 19:15:18 -07001780 ASSERT_NE(c, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001781 jobject o = env_->AllocObject(c);
Ian Rogers2d10b202014-05-12 19:15:18 -07001782 ASSERT_NE(o, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001783
1784 jstring s1 = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001785 ASSERT_NE(s1, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001786 jstring s2 = env_->NewStringUTF("world");
Ian Rogers2d10b202014-05-12 19:15:18 -07001787 ASSERT_NE(s2, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001788
1789 jfieldID s_fid = env_->GetStaticFieldID(c, "sObject", "Ljava/lang/Object;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001790 ASSERT_NE(s_fid, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001791 jfieldID i_fid = env_->GetFieldID(c, "iObject", "Ljava/lang/Object;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001792 ASSERT_NE(i_fid, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001793
1794 env_->SetStaticObjectField(c, s_fid, s1);
1795 ASSERT_TRUE(env_->IsSameObject(s1, env_->GetStaticObjectField(c, s_fid)));
1796 env_->SetStaticObjectField(c, s_fid, s2);
1797 ASSERT_TRUE(env_->IsSameObject(s2, env_->GetStaticObjectField(c, s_fid)));
1798
1799 env_->SetObjectField(o, i_fid, s1);
1800 ASSERT_TRUE(env_->IsSameObject(s1, env_->GetObjectField(o, i_fid)));
1801 env_->SetObjectField(o, i_fid, s2);
1802 ASSERT_TRUE(env_->IsSameObject(s2, env_->GetObjectField(o, i_fid)));
1803}
1804
Ian Rogers2d10b202014-05-12 19:15:18 -07001805TEST_F(JniInternalTest, NewLocalRef_nullptr) {
1806 EXPECT_EQ(env_->NewLocalRef(nullptr), nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001807}
1808
1809TEST_F(JniInternalTest, NewLocalRef) {
1810 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001811 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001812 jobject o = env_->NewLocalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001813 EXPECT_NE(o, nullptr);
1814 EXPECT_NE(o, s);
Elliott Hughes18c07532011-08-18 15:50:51 -07001815
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001816 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(o));
Elliott Hughes18c07532011-08-18 15:50:51 -07001817}
1818
Ian Rogers2d10b202014-05-12 19:15:18 -07001819TEST_F(JniInternalTest, DeleteLocalRef_nullptr) {
1820 env_->DeleteLocalRef(nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001821}
1822
1823TEST_F(JniInternalTest, DeleteLocalRef) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001824 // This tests leads to warnings and errors in the log.
1825 ScopedLogSeverity sls(LogSeverity::FATAL);
1826
Elliott Hughes18c07532011-08-18 15:50:51 -07001827 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001828 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001829 env_->DeleteLocalRef(s);
1830
Elliott Hughes3f6635a2012-06-19 13:37:49 -07001831 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07001832 {
Ian Rogers68d8b422014-07-17 11:09:10 -07001833 bool old_check_jni = vm_->SetCheckJniEnabled(false);
1834 {
1835 CheckJniAbortCatcher check_jni_abort_catcher;
1836 env_->DeleteLocalRef(s);
1837 }
Elliott Hughesb264f082012-04-06 17:10:10 -07001838 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001839 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
Elliott Hughesb264f082012-04-06 17:10:10 -07001840 env_->DeleteLocalRef(s);
Ian Rogersc0542af2014-09-03 16:16:56 -07001841 std::string expected(StringPrintf("use of deleted local reference %p", s));
Elliott Hughesa9137c62013-01-09 10:55:21 -08001842 check_jni_abort_catcher.Check(expected.c_str());
Ian Rogers68d8b422014-07-17 11:09:10 -07001843 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07001844 }
Elliott Hughes18c07532011-08-18 15:50:51 -07001845
1846 s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001847 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001848 jobject o = env_->NewLocalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001849 ASSERT_NE(o, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001850
1851 env_->DeleteLocalRef(s);
1852 env_->DeleteLocalRef(o);
1853}
1854
Elliott Hughesaa836f72013-08-20 16:57:23 -07001855TEST_F(JniInternalTest, PushLocalFrame_10395422) {
1856 // The JNI specification is ambiguous about whether the given capacity is to be interpreted as a
1857 // maximum or as a minimum, but it seems like it's supposed to be a minimum, and that's how
1858 // Android historically treated it, and it's how the RI treats it. It's also the more useful
1859 // interpretation!
1860 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(0));
Ian Rogers2d10b202014-05-12 19:15:18 -07001861 env_->PopLocalFrame(nullptr);
Elliott Hughesaa836f72013-08-20 16:57:23 -07001862
Andreas Gampe369810a2015-01-14 19:53:31 -08001863 // The following two tests will print errors to the log.
1864 ScopedLogSeverity sls(LogSeverity::FATAL);
1865
Elliott Hughesaa836f72013-08-20 16:57:23 -07001866 // Negative capacities are not allowed.
1867 ASSERT_EQ(JNI_ERR, env_->PushLocalFrame(-1));
1868
1869 // And it's okay to have an upper limit. Ours is currently 512.
1870 ASSERT_EQ(JNI_ERR, env_->PushLocalFrame(8192));
1871}
1872
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001873TEST_F(JniInternalTest, PushLocalFrame_PopLocalFrame) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001874 // This tests leads to errors in the log.
1875 ScopedLogSeverity sls(LogSeverity::FATAL);
1876
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001877 jobject original = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001878 ASSERT_NE(original, nullptr);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001879
1880 jobject outer;
1881 jobject inner1, inner2;
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001882 ScopedObjectAccess soa(env_);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001883 {
Elliott Hughesaa836f72013-08-20 16:57:23 -07001884 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(4));
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001885 outer = env_->NewLocalRef(original);
1886
1887 {
Elliott Hughesaa836f72013-08-20 16:57:23 -07001888 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(4));
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001889 inner1 = env_->NewLocalRef(outer);
1890 inner2 = env_->NewStringUTF("survivor");
Ian Rogersc0542af2014-09-03 16:16:56 -07001891 EXPECT_NE(env_->PopLocalFrame(inner2), nullptr);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001892 }
1893
1894 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(original));
1895 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(outer));
Ian Rogersc0542af2014-09-03 16:16:56 -07001896 {
1897 CheckJniAbortCatcher check_jni_abort_catcher;
1898 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner1));
1899 check_jni_abort_catcher.Check("use of deleted local reference");
1900 }
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001901
1902 // Our local reference for the survivor is invalid because the survivor
1903 // gets a new local reference...
Ian Rogersc0542af2014-09-03 16:16:56 -07001904 {
1905 CheckJniAbortCatcher check_jni_abort_catcher;
1906 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner2));
1907 check_jni_abort_catcher.Check("use of deleted local reference");
1908 }
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001909
Ian Rogersc0542af2014-09-03 16:16:56 -07001910 EXPECT_EQ(env_->PopLocalFrame(nullptr), nullptr);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001911 }
1912 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(original));
Ian Rogersc0542af2014-09-03 16:16:56 -07001913 CheckJniAbortCatcher check_jni_abort_catcher;
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001914 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(outer));
Ian Rogersc0542af2014-09-03 16:16:56 -07001915 check_jni_abort_catcher.Check("use of deleted local reference");
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001916 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner1));
Ian Rogersc0542af2014-09-03 16:16:56 -07001917 check_jni_abort_catcher.Check("use of deleted local reference");
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001918 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner2));
Ian Rogersc0542af2014-09-03 16:16:56 -07001919 check_jni_abort_catcher.Check("use of deleted local reference");
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001920}
1921
Ian Rogers2d10b202014-05-12 19:15:18 -07001922TEST_F(JniInternalTest, NewGlobalRef_nullptr) {
1923 EXPECT_EQ(env_->NewGlobalRef(nullptr), nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001924}
1925
1926TEST_F(JniInternalTest, NewGlobalRef) {
1927 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001928 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001929 jobject o = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001930 EXPECT_NE(o, nullptr);
1931 EXPECT_NE(o, s);
Elliott Hughes18c07532011-08-18 15:50:51 -07001932
Ian Rogers2d10b202014-05-12 19:15:18 -07001933 EXPECT_EQ(env_->GetObjectRefType(o), JNIGlobalRefType);
Elliott Hughes18c07532011-08-18 15:50:51 -07001934}
1935
Ian Rogers2d10b202014-05-12 19:15:18 -07001936TEST_F(JniInternalTest, DeleteGlobalRef_nullptr) {
1937 env_->DeleteGlobalRef(nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001938}
1939
1940TEST_F(JniInternalTest, DeleteGlobalRef) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001941 // This tests leads to warnings and errors in the log.
1942 ScopedLogSeverity sls(LogSeverity::FATAL);
1943
Elliott Hughes18c07532011-08-18 15:50:51 -07001944 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001945 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001946
1947 jobject o = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001948 ASSERT_NE(o, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001949 env_->DeleteGlobalRef(o);
1950
Elliott Hughes3f6635a2012-06-19 13:37:49 -07001951 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07001952 {
Ian Rogers68d8b422014-07-17 11:09:10 -07001953 bool old_check_jni = vm_->SetCheckJniEnabled(false);
1954 {
1955 CheckJniAbortCatcher check_jni_abort_catcher;
1956 env_->DeleteGlobalRef(o);
1957 }
Elliott Hughesb264f082012-04-06 17:10:10 -07001958 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001959 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
Elliott Hughesb264f082012-04-06 17:10:10 -07001960 env_->DeleteGlobalRef(o);
Ian Rogersc0542af2014-09-03 16:16:56 -07001961 std::string expected(StringPrintf("use of deleted global reference %p", o));
Elliott Hughesa9137c62013-01-09 10:55:21 -08001962 check_jni_abort_catcher.Check(expected.c_str());
Ian Rogers68d8b422014-07-17 11:09:10 -07001963 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07001964 }
Elliott Hughes18c07532011-08-18 15:50:51 -07001965
1966 jobject o1 = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001967 ASSERT_NE(o1, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001968 jobject o2 = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001969 ASSERT_NE(o2, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001970
1971 env_->DeleteGlobalRef(o1);
1972 env_->DeleteGlobalRef(o2);
1973}
1974
Ian Rogers2d10b202014-05-12 19:15:18 -07001975TEST_F(JniInternalTest, NewWeakGlobalRef_nullptr) {
1976 EXPECT_EQ(env_->NewWeakGlobalRef(nullptr), nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001977}
1978
1979TEST_F(JniInternalTest, NewWeakGlobalRef) {
1980 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001981 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001982 jobject o = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001983 EXPECT_NE(o, nullptr);
1984 EXPECT_NE(o, s);
Elliott Hughes18c07532011-08-18 15:50:51 -07001985
Ian Rogers2d10b202014-05-12 19:15:18 -07001986 EXPECT_EQ(env_->GetObjectRefType(o), JNIWeakGlobalRefType);
Elliott Hughes18c07532011-08-18 15:50:51 -07001987}
1988
Ian Rogers2d10b202014-05-12 19:15:18 -07001989TEST_F(JniInternalTest, DeleteWeakGlobalRef_nullptr) {
1990 env_->DeleteWeakGlobalRef(nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001991}
1992
1993TEST_F(JniInternalTest, DeleteWeakGlobalRef) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001994 // This tests leads to warnings and errors in the log.
1995 ScopedLogSeverity sls(LogSeverity::FATAL);
1996
Elliott Hughes18c07532011-08-18 15:50:51 -07001997 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001998 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001999
2000 jobject o = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002001 ASSERT_NE(o, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002002 env_->DeleteWeakGlobalRef(o);
2003
Elliott Hughes3f6635a2012-06-19 13:37:49 -07002004 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07002005 {
Ian Rogers68d8b422014-07-17 11:09:10 -07002006 bool old_check_jni = vm_->SetCheckJniEnabled(false);
2007 {
2008 CheckJniAbortCatcher check_jni_abort_catcher;
2009 env_->DeleteWeakGlobalRef(o);
2010 }
Elliott Hughesb264f082012-04-06 17:10:10 -07002011 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07002012 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
Elliott Hughesb264f082012-04-06 17:10:10 -07002013 env_->DeleteWeakGlobalRef(o);
Ian Rogersc0542af2014-09-03 16:16:56 -07002014 std::string expected(StringPrintf("use of deleted weak global reference %p", o));
Elliott Hughesa9137c62013-01-09 10:55:21 -08002015 check_jni_abort_catcher.Check(expected.c_str());
Ian Rogers68d8b422014-07-17 11:09:10 -07002016 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07002017 }
Elliott Hughes18c07532011-08-18 15:50:51 -07002018
2019 jobject o1 = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002020 ASSERT_NE(o1, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002021 jobject o2 = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002022 ASSERT_NE(o2, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002023
2024 env_->DeleteWeakGlobalRef(o1);
2025 env_->DeleteWeakGlobalRef(o2);
2026}
2027
Alexei Zavjalov3a1444c2014-06-25 16:04:55 +07002028TEST_F(JniInternalTest, ExceptionDescribe) {
2029 // This checks how ExceptionDescribe handles call without exception.
2030 env_->ExceptionClear();
2031 env_->ExceptionDescribe();
2032}
2033
Elliott Hughes37f7a402011-08-22 18:56:01 -07002034TEST_F(JniInternalTest, Throw) {
Elliott Hughes37f7a402011-08-22 18:56:01 -07002035 jclass exception_class = env_->FindClass("java/lang/RuntimeException");
Ian Rogers2d10b202014-05-12 19:15:18 -07002036 ASSERT_TRUE(exception_class != nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07002037 jthrowable exception = reinterpret_cast<jthrowable>(env_->AllocObject(exception_class));
Ian Rogers2d10b202014-05-12 19:15:18 -07002038 ASSERT_TRUE(exception != nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07002039
2040 EXPECT_EQ(JNI_OK, env_->Throw(exception));
2041 EXPECT_TRUE(env_->ExceptionCheck());
Elliott Hughesa2501992011-08-26 19:39:54 -07002042 jthrowable thrown_exception = env_->ExceptionOccurred();
Elliott Hughes37f7a402011-08-22 18:56:01 -07002043 env_->ExceptionClear();
Elliott Hughesa2501992011-08-26 19:39:54 -07002044 EXPECT_TRUE(env_->IsSameObject(exception, thrown_exception));
Ian Rogers68d8b422014-07-17 11:09:10 -07002045
2046 // Bad argument.
2047 bool old_check_jni = vm_->SetCheckJniEnabled(false);
2048 EXPECT_EQ(JNI_ERR, env_->Throw(nullptr));
2049 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
2050 CheckJniAbortCatcher check_jni_abort_catcher;
2051 EXPECT_EQ(JNI_ERR, env_->Throw(nullptr));
2052 check_jni_abort_catcher.Check("Throw received NULL jthrowable");
2053 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes37f7a402011-08-22 18:56:01 -07002054}
2055
2056TEST_F(JniInternalTest, ThrowNew) {
Elliott Hughes37f7a402011-08-22 18:56:01 -07002057 jclass exception_class = env_->FindClass("java/lang/RuntimeException");
Ian Rogers2d10b202014-05-12 19:15:18 -07002058 ASSERT_TRUE(exception_class != nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07002059
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07002060 jthrowable thrown_exception;
2061
Elliott Hughes37f7a402011-08-22 18:56:01 -07002062 EXPECT_EQ(JNI_OK, env_->ThrowNew(exception_class, "hello world"));
2063 EXPECT_TRUE(env_->ExceptionCheck());
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07002064 thrown_exception = env_->ExceptionOccurred();
2065 env_->ExceptionClear();
2066 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, exception_class));
2067
Ian Rogers2d10b202014-05-12 19:15:18 -07002068 EXPECT_EQ(JNI_OK, env_->ThrowNew(exception_class, nullptr));
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07002069 EXPECT_TRUE(env_->ExceptionCheck());
2070 thrown_exception = env_->ExceptionOccurred();
Elliott Hughes37f7a402011-08-22 18:56:01 -07002071 env_->ExceptionClear();
Elliott Hughesa2501992011-08-26 19:39:54 -07002072 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, exception_class));
Ian Rogers68d8b422014-07-17 11:09:10 -07002073
2074 // Bad argument.
2075 bool old_check_jni = vm_->SetCheckJniEnabled(false);
2076 CheckJniAbortCatcher check_jni_abort_catcher;
2077 EXPECT_EQ(JNI_ERR, env_->ThrowNew(nullptr, nullptr));
2078 check_jni_abort_catcher.Check("c == null");
2079 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
2080 EXPECT_EQ(JNI_ERR, env_->ThrowNew(nullptr, nullptr));
2081 check_jni_abort_catcher.Check("ThrowNew received NULL jclass");
2082 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes37f7a402011-08-22 18:56:01 -07002083}
2084
Ian Rogers1d99e452014-01-02 17:36:41 -08002085TEST_F(JniInternalTest, NewDirectBuffer_GetDirectBufferAddress_GetDirectBufferCapacity) {
2086 // Start runtime.
2087 Thread* self = Thread::Current();
2088 self->TransitionFromSuspendedToRunnable();
2089 MakeExecutable(nullptr, "java.lang.Class");
2090 MakeExecutable(nullptr, "java.lang.Object");
2091 MakeExecutable(nullptr, "java.nio.DirectByteBuffer");
2092 MakeExecutable(nullptr, "java.nio.MemoryBlock");
2093 MakeExecutable(nullptr, "java.nio.MemoryBlock$UnmanagedBlock");
2094 MakeExecutable(nullptr, "java.nio.MappedByteBuffer");
2095 MakeExecutable(nullptr, "java.nio.ByteBuffer");
2096 MakeExecutable(nullptr, "java.nio.Buffer");
2097 // TODO: we only load a dex file here as starting the runtime relies upon it.
2098 const char* class_name = "StaticLeafMethods";
2099 LoadDex(class_name);
2100 bool started = runtime_->Start();
2101 ASSERT_TRUE(started);
2102
Elliott Hughesb465ab02011-08-24 11:21:21 -07002103 jclass buffer_class = env_->FindClass("java/nio/Buffer");
Ian Rogers2d10b202014-05-12 19:15:18 -07002104 ASSERT_NE(buffer_class, nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07002105
2106 char bytes[1024];
2107 jobject buffer = env_->NewDirectByteBuffer(bytes, sizeof(bytes));
Ian Rogers2d10b202014-05-12 19:15:18 -07002108 ASSERT_NE(buffer, nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07002109 ASSERT_TRUE(env_->IsInstanceOf(buffer, buffer_class));
Ian Rogers2d10b202014-05-12 19:15:18 -07002110 ASSERT_EQ(env_->GetDirectBufferAddress(buffer), bytes);
Ian Rogers1ee99352014-05-14 14:38:16 -07002111 ASSERT_EQ(env_->GetDirectBufferCapacity(buffer), static_cast<jlong>(sizeof(bytes)));
Brian Carlstrom45d26c82014-06-24 23:36:28 -07002112
2113 {
2114 CheckJniAbortCatcher check_jni_abort_catcher;
Brian Carlstrom85a93362014-06-25 09:30:52 -07002115 env_->NewDirectByteBuffer(bytes, static_cast<jlong>(INT_MAX) + 1);
Brian Carlstrom45d26c82014-06-24 23:36:28 -07002116 check_jni_abort_catcher.Check("in call to NewDirectByteBuffer");
2117 }
Elliott Hughesb465ab02011-08-24 11:21:21 -07002118}
2119
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002120TEST_F(JniInternalTest, MonitorEnterExit) {
Andreas Gampe369810a2015-01-14 19:53:31 -08002121 // This will print some error messages. Suppress.
2122 ScopedLogSeverity sls(LogSeverity::FATAL);
2123
Ian Rogers2d10b202014-05-12 19:15:18 -07002124 // Create an object to torture.
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002125 jclass object_class = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07002126 ASSERT_NE(object_class, nullptr);
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002127 jobject object = env_->AllocObject(object_class);
Ian Rogers2d10b202014-05-12 19:15:18 -07002128 ASSERT_NE(object, nullptr);
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002129
2130 // Expected class of exceptions
2131 jclass imse_class = env_->FindClass("java/lang/IllegalMonitorStateException");
Ian Rogers2d10b202014-05-12 19:15:18 -07002132 ASSERT_NE(imse_class, nullptr);
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002133
2134 jthrowable thrown_exception;
2135
2136 // Unlock of unowned monitor
2137 env_->MonitorExit(object);
2138 EXPECT_TRUE(env_->ExceptionCheck());
2139 thrown_exception = env_->ExceptionOccurred();
2140 env_->ExceptionClear();
2141 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, imse_class));
2142
2143 // Lock of unowned monitor
2144 env_->MonitorEnter(object);
2145 EXPECT_FALSE(env_->ExceptionCheck());
2146 // Regular unlock
2147 env_->MonitorExit(object);
2148 EXPECT_FALSE(env_->ExceptionCheck());
2149
2150 // Recursively lock a lot
2151 size_t max_recursive_lock = 1024;
2152 for (size_t i = 0; i < max_recursive_lock; i++) {
2153 env_->MonitorEnter(object);
2154 EXPECT_FALSE(env_->ExceptionCheck());
2155 }
2156 // Recursively unlock a lot
2157 for (size_t i = 0; i < max_recursive_lock; i++) {
2158 env_->MonitorExit(object);
2159 EXPECT_FALSE(env_->ExceptionCheck());
2160 }
2161
2162 // Unlock of unowned monitor
2163 env_->MonitorExit(object);
2164 EXPECT_TRUE(env_->ExceptionCheck());
2165 thrown_exception = env_->ExceptionOccurred();
2166 env_->ExceptionClear();
2167 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, imse_class));
Elliott Hughesa92853e2012-02-07 16:09:27 -08002168
Mathieu Chartier2cebb242015-04-21 16:50:40 -07002169 // It's an error to call MonitorEnter or MonitorExit on null.
Elliott Hughesb264f082012-04-06 17:10:10 -07002170 {
2171 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers2d10b202014-05-12 19:15:18 -07002172 env_->MonitorEnter(nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -07002173 check_jni_abort_catcher.Check("in call to MonitorEnter");
Ian Rogers2d10b202014-05-12 19:15:18 -07002174 env_->MonitorExit(nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -07002175 check_jni_abort_catcher.Check("in call to MonitorExit");
2176 }
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002177}
2178
Andreas Gampe5f4a09a2015-09-28 13:16:33 -07002179void Java_MyClassNatives_foo_exit(JNIEnv* env, jobject thisObj) {
2180 // Release the monitor on self. This should trigger an abort.
2181 env->MonitorExit(thisObj);
2182}
2183
2184TEST_F(JniInternalTest, MonitorExitLockedInDifferentCall) {
2185 SetUpForTest(false, "foo", "()V", reinterpret_cast<void*>(&Java_MyClassNatives_foo_exit));
2186 ASSERT_NE(jobj_, nullptr);
2187
2188 env_->MonitorEnter(jobj_);
2189 EXPECT_FALSE(env_->ExceptionCheck());
2190
2191 CheckJniAbortCatcher check_jni_abort_catcher;
2192 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
2193 check_jni_abort_catcher.Check("Unlocking monitor that wasn't locked here");
2194}
2195
2196void Java_MyClassNatives_foo_enter_no_exit(JNIEnv* env, jobject thisObj) {
2197 // Acquire but don't release the monitor on self. This should trigger an abort on return.
2198 env->MonitorEnter(thisObj);
2199}
2200
2201TEST_F(JniInternalTest, MonitorExitNotAllUnlocked) {
2202 SetUpForTest(false,
2203 "foo",
2204 "()V",
2205 reinterpret_cast<void*>(&Java_MyClassNatives_foo_enter_no_exit));
2206 ASSERT_NE(jobj_, nullptr);
2207
2208 CheckJniAbortCatcher check_jni_abort_catcher;
2209 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
2210 check_jni_abort_catcher.Check("Still holding a locked object on JNI end");
2211}
2212
Andreas Gampeb8c4f032015-12-03 10:56:18 -08002213static bool IsLocked(JNIEnv* env, jobject jobj) {
2214 ScopedObjectAccess soa(env);
2215 LockWord lock_word = soa.Decode<mirror::Object*>(jobj)->GetLockWord(true);
2216 switch (lock_word.GetState()) {
2217 case LockWord::kHashCode:
2218 case LockWord::kUnlocked:
2219 return false;
2220 case LockWord::kThinLocked:
2221 return true;
2222 case LockWord::kFatLocked:
2223 return lock_word.FatLockMonitor()->IsLocked();
2224 default: {
2225 LOG(FATAL) << "Invalid monitor state " << lock_word.GetState();
2226 UNREACHABLE();
2227 }
2228 }
2229}
2230
2231TEST_F(JniInternalTest, DetachThreadUnlockJNIMonitors) {
2232 // We need to lock an object, detach, reattach, and check the locks.
2233 //
2234 // As re-attaching will create a different thread, we need to use a global
2235 // ref to keep the object around.
2236
2237 // Create an object to torture.
2238 jobject global_ref;
2239 {
2240 jclass object_class = env_->FindClass("java/lang/Object");
2241 ASSERT_NE(object_class, nullptr);
2242 jobject object = env_->AllocObject(object_class);
2243 ASSERT_NE(object, nullptr);
2244 global_ref = env_->NewGlobalRef(object);
2245 }
2246
2247 // Lock it.
2248 env_->MonitorEnter(global_ref);
2249 ASSERT_TRUE(IsLocked(env_, global_ref));
2250
2251 // Detach and re-attach.
2252 jint detach_result = vm_->DetachCurrentThread();
2253 ASSERT_EQ(detach_result, JNI_OK);
2254 jint attach_result = vm_->AttachCurrentThread(&env_, nullptr);
2255 ASSERT_EQ(attach_result, JNI_OK);
2256
2257 // Look at the global ref, check whether it's still locked.
2258 ASSERT_FALSE(IsLocked(env_, global_ref));
2259
2260 // Delete the global ref.
2261 env_->DeleteGlobalRef(global_ref);
2262}
2263
Ian Rogers0cfe1fb2011-08-26 03:29:44 -07002264} // namespace art