blob: a421c340ae6cb3fff8f924d828b8b2ec417ee885 [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"
Andreas Gampeda9b7632015-12-09 15:14:04 -080021#include "indirect_reference_table.h"
Ian Rogers68d8b422014-07-17 11:09:10 -070022#include "java_vm_ext.h"
Andreas Gampeda9b7632015-12-09 15:14:04 -080023#include "jni_env_ext.h"
Fred Shih56890e22014-06-02 11:11:52 -070024#include "mirror/string-inl.h"
Mathieu Chartier0795f232016-09-27 18:43:30 -070025#include "scoped_thread_state_change-inl.h"
Elliott Hughes726079d2011-10-07 18:43:44 -070026#include "ScopedLocalRef.h"
Elliott Hughes0c9cd562011-08-12 10:59:29 -070027
28namespace art {
29
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080030// TODO: Convert to CommonRuntimeTest. Currently MakeExecutable is used.
31class JniInternalTest : public CommonCompilerTest {
Elliott Hughesc7ac37f2011-08-12 12:21:58 -070032 protected:
33 virtual void SetUp() {
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080034 CommonCompilerTest::SetUp();
Elliott Hughes5174fe62011-08-23 15:12:35 -070035
Elliott Hughesa2501992011-08-26 19:39:54 -070036 vm_ = Runtime::Current()->GetJavaVM();
37
Elliott Hughes5174fe62011-08-23 15:12:35 -070038 // Turn on -verbose:jni for the JNI tests.
Ian Rogers79713632013-08-21 19:06:15 -070039 // gLogVerbosity.jni = true;
Elliott Hughes5174fe62011-08-23 15:12:35 -070040
Ian Rogers2d10b202014-05-12 19:15:18 -070041 vm_->AttachCurrentThread(&env_, nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -070042
Brian Carlstromea46f952013-07-30 01:26:50 -070043 ScopedLocalRef<jclass> aioobe(env_,
44 env_->FindClass("java/lang/ArrayIndexOutOfBoundsException"));
Ian Rogers2d10b202014-05-12 19:15:18 -070045 CHECK(aioobe.get() != nullptr);
Elliott Hughes726079d2011-10-07 18:43:44 -070046 aioobe_ = reinterpret_cast<jclass>(env_->NewGlobalRef(aioobe.get()));
Elliott Hughesb465ab02011-08-24 11:21:21 -070047
Elliott Hughesb264f082012-04-06 17:10:10 -070048 ScopedLocalRef<jclass> ase(env_, env_->FindClass("java/lang/ArrayStoreException"));
Ian Rogers2d10b202014-05-12 19:15:18 -070049 CHECK(ase.get() != nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -070050 ase_ = reinterpret_cast<jclass>(env_->NewGlobalRef(ase.get()));
51
Brian Carlstromea46f952013-07-30 01:26:50 -070052 ScopedLocalRef<jclass> sioobe(env_,
53 env_->FindClass("java/lang/StringIndexOutOfBoundsException"));
Ian Rogers2d10b202014-05-12 19:15:18 -070054 CHECK(sioobe.get() != nullptr);
Elliott Hughes726079d2011-10-07 18:43:44 -070055 sioobe_ = reinterpret_cast<jclass>(env_->NewGlobalRef(sioobe.get()));
56 }
57
Ian Rogers2d10b202014-05-12 19:15:18 -070058 void ExpectException(jclass exception_class) {
Ian Rogers68d8b422014-07-17 11:09:10 -070059 ScopedObjectAccess soa(env_);
60 EXPECT_TRUE(env_->ExceptionCheck())
David Sehr709b0702016-10-13 09:12:37 -070061 << mirror::Class::PrettyDescriptor(soa.Decode<mirror::Class>(exception_class));
Ian Rogers2d10b202014-05-12 19:15:18 -070062 jthrowable exception = env_->ExceptionOccurred();
63 EXPECT_NE(nullptr, exception);
64 env_->ExceptionClear();
65 EXPECT_TRUE(env_->IsInstanceOf(exception, exception_class));
66 }
67
Brian Carlstrom4d571432012-05-16 00:21:41 -070068 void CleanUpJniEnv() {
Ian Rogers2d10b202014-05-12 19:15:18 -070069 if (aioobe_ != nullptr) {
Brian Carlstrom4d571432012-05-16 00:21:41 -070070 env_->DeleteGlobalRef(aioobe_);
Ian Rogers2d10b202014-05-12 19:15:18 -070071 aioobe_ = nullptr;
Brian Carlstrom4d571432012-05-16 00:21:41 -070072 }
Ian Rogers2d10b202014-05-12 19:15:18 -070073 if (ase_ != nullptr) {
Brian Carlstrom4d571432012-05-16 00:21:41 -070074 env_->DeleteGlobalRef(ase_);
Ian Rogers2d10b202014-05-12 19:15:18 -070075 ase_ = nullptr;
Brian Carlstrom4d571432012-05-16 00:21:41 -070076 }
Ian Rogers2d10b202014-05-12 19:15:18 -070077 if (sioobe_ != nullptr) {
Brian Carlstrom4d571432012-05-16 00:21:41 -070078 env_->DeleteGlobalRef(sioobe_);
Ian Rogers2d10b202014-05-12 19:15:18 -070079 sioobe_ = nullptr;
Brian Carlstrom4d571432012-05-16 00:21:41 -070080 }
81 }
82
Ian Rogers53b8b092014-03-13 23:45:53 -070083 virtual void TearDown() OVERRIDE {
Brian Carlstrom4d571432012-05-16 00:21:41 -070084 CleanUpJniEnv();
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080085 CommonCompilerTest::TearDown();
Elliott Hughesc7ac37f2011-08-12 12:21:58 -070086 }
Elliott Hughesb465ab02011-08-24 11:21:21 -070087
Ian Rogers1d99e452014-01-02 17:36:41 -080088 jclass GetPrimitiveClass(char descriptor) {
89 ScopedObjectAccess soa(env_);
90 mirror::Class* c = class_linker_->FindPrimitiveClass(descriptor);
91 CHECK(c != nullptr);
92 return soa.AddLocalReference<jclass>(c);
93 }
94
Ian Rogers68d8b422014-07-17 11:09:10 -070095 void ExpectClassFound(const char* name) {
96 EXPECT_NE(env_->FindClass(name), nullptr) << name;
97 EXPECT_FALSE(env_->ExceptionCheck()) << name;
98 }
99
100 void ExpectClassNotFound(const char* name, bool check_jni, const char* check_jni_msg,
101 CheckJniAbortCatcher* abort_catcher) {
102 EXPECT_EQ(env_->FindClass(name), nullptr) << name;
103 if (!check_jni || check_jni_msg == nullptr) {
104 EXPECT_TRUE(env_->ExceptionCheck()) << name;
105 env_->ExceptionClear();
106 } else {
107 abort_catcher->Check(check_jni_msg);
108 }
109 }
110
111 void FindClassTest(bool check_jni) {
112 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
113 CheckJniAbortCatcher check_jni_abort_catcher;
114
115 // Null argument is always an abort.
116 env_->FindClass(nullptr);
117 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
118 : "name == null");
119
120 // Reference types...
121 ExpectClassFound("java/lang/String");
122 // ...for arrays too, where you must include "L;".
123 ExpectClassFound("[Ljava/lang/String;");
124 // Primitive arrays are okay too, if the primitive type is valid.
125 ExpectClassFound("[C");
126
127 // But primitive types aren't allowed...
128 ExpectClassNotFound("C", check_jni, nullptr, &check_jni_abort_catcher);
129 ExpectClassNotFound("V", check_jni, nullptr, &check_jni_abort_catcher);
130 ExpectClassNotFound("K", check_jni, nullptr, &check_jni_abort_catcher);
131
132 if (check_jni) {
133 // Check JNI will reject invalid class names as aborts but without pending exceptions.
134 EXPECT_EQ(env_->FindClass("java.lang.String"), nullptr);
135 EXPECT_FALSE(env_->ExceptionCheck());
136 check_jni_abort_catcher.Check("illegal class name 'java.lang.String'");
137
138 EXPECT_EQ(env_->FindClass("[Ljava.lang.String;"), nullptr);
139 EXPECT_FALSE(env_->ExceptionCheck());
140 check_jni_abort_catcher.Check("illegal class name '[Ljava.lang.String;'");
141 } else {
142 // Without check JNI we're tolerant and replace '.' with '/'.
143 ExpectClassFound("java.lang.String");
144 ExpectClassFound("[Ljava.lang.String;");
145 }
146
147 ExpectClassNotFound("Ljava.lang.String;", check_jni, "illegal class name 'Ljava.lang.String;'",
148 &check_jni_abort_catcher);
149 ExpectClassNotFound("[java.lang.String", check_jni, "illegal class name '[java.lang.String'",
150 &check_jni_abort_catcher);
151
152 // You can't include the "L;" in a JNI class descriptor.
153 ExpectClassNotFound("Ljava/lang/String;", check_jni, "illegal class name 'Ljava/lang/String;'",
154 &check_jni_abort_catcher);
155
156 // But you must include it for an array of any reference type.
157 ExpectClassNotFound("[java/lang/String", check_jni, "illegal class name '[java/lang/String'",
158 &check_jni_abort_catcher);
159
160 ExpectClassNotFound("[K", check_jni, "illegal class name '[K'", &check_jni_abort_catcher);
161
162 // Void arrays aren't allowed.
163 ExpectClassNotFound("[V", check_jni, "illegal class name '[V'", &check_jni_abort_catcher);
164
165 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
166 }
167
168 void GetFieldIdBadArgumentTest(bool check_jni) {
169 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
170 CheckJniAbortCatcher check_jni_abort_catcher;
171
172 jclass c = env_->FindClass("java/lang/String");
173 ASSERT_NE(c, nullptr);
174
175 jfieldID fid = env_->GetFieldID(nullptr, "count", "I");
176 EXPECT_EQ(nullptr, fid);
177 check_jni_abort_catcher.Check(check_jni ? "GetFieldID received NULL jclass"
178 : "java_class == null");
179 fid = env_->GetFieldID(c, nullptr, "I");
180 EXPECT_EQ(nullptr, fid);
181 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
182 : "name == null");
183 fid = env_->GetFieldID(c, "count", nullptr);
184 EXPECT_EQ(nullptr, fid);
185 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
186 : "sig == null");
187
188 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
189 }
190
191 void GetStaticFieldIdBadArgumentTest(bool check_jni) {
192 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
193 CheckJniAbortCatcher check_jni_abort_catcher;
194
195 jclass c = env_->FindClass("java/lang/String");
196 ASSERT_NE(c, nullptr);
197
198 jfieldID fid = env_->GetStaticFieldID(nullptr, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
199 EXPECT_EQ(nullptr, fid);
200 check_jni_abort_catcher.Check(check_jni ? "GetStaticFieldID received NULL jclass"
201 : "java_class == null");
202 fid = env_->GetStaticFieldID(c, nullptr, "Ljava/util/Comparator;");
203 EXPECT_EQ(nullptr, fid);
204 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
205 : "name == null");
206 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", nullptr);
207 EXPECT_EQ(nullptr, fid);
208 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
209 : "sig == null");
210
211 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
212 }
213
214 void GetMethodIdBadArgumentTest(bool check_jni) {
215 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
216 CheckJniAbortCatcher check_jni_abort_catcher;
217
218 jmethodID method = env_->GetMethodID(nullptr, "<init>", "(Ljava/lang/String;)V");
219 EXPECT_EQ(nullptr, method);
220 check_jni_abort_catcher.Check(check_jni ? "GetMethodID received NULL jclass"
221 : "java_class == null");
222 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
223 ASSERT_TRUE(jlnsme != nullptr);
224 method = env_->GetMethodID(jlnsme, nullptr, "(Ljava/lang/String;)V");
225 EXPECT_EQ(nullptr, method);
226 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
227 : "name == null");
228 method = env_->GetMethodID(jlnsme, "<init>", nullptr);
229 EXPECT_EQ(nullptr, method);
230 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
231 : "sig == null");
232
233 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
234 }
235
236 void GetStaticMethodIdBadArgumentTest(bool check_jni) {
237 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
238 CheckJniAbortCatcher check_jni_abort_catcher;
239
240 jmethodID method = env_->GetStaticMethodID(nullptr, "valueOf", "(I)Ljava/lang/String;");
241 EXPECT_EQ(nullptr, method);
242 check_jni_abort_catcher.Check(check_jni ? "GetStaticMethodID received NULL jclass"
243 : "java_class == null");
244 jclass jlstring = env_->FindClass("java/lang/String");
245 method = env_->GetStaticMethodID(jlstring, nullptr, "(I)Ljava/lang/String;");
246 EXPECT_EQ(nullptr, method);
247 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
248 : "name == null");
249 method = env_->GetStaticMethodID(jlstring, "valueOf", nullptr);
250 EXPECT_EQ(nullptr, method);
251 check_jni_abort_catcher.Check(check_jni ? "non-nullable const char* was NULL"
252 : "sig == null");
253
254 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
255 }
256
257 void GetFromReflectedField_ToReflectedFieldBadArgumentTest(bool check_jni) {
258 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
259 CheckJniAbortCatcher check_jni_abort_catcher;
260
261 jclass c = env_->FindClass("java/lang/String");
262 ASSERT_NE(c, nullptr);
263 jfieldID fid = env_->GetFieldID(c, "count", "I");
264 ASSERT_NE(fid, nullptr);
265
266 // Check class argument for null argument, not checked in non-check JNI.
267 jobject field = env_->ToReflectedField(nullptr, fid, JNI_FALSE);
268 if (check_jni) {
269 EXPECT_EQ(field, nullptr);
270 check_jni_abort_catcher.Check("ToReflectedField received NULL jclass");
271 } else {
272 EXPECT_NE(field, nullptr);
273 }
274
275 field = env_->ToReflectedField(c, nullptr, JNI_FALSE);
276 EXPECT_EQ(field, nullptr);
277 check_jni_abort_catcher.Check(check_jni ? "jfieldID was NULL"
278 : "fid == null");
279
280 fid = env_->FromReflectedField(nullptr);
281 ASSERT_EQ(fid, nullptr);
282 check_jni_abort_catcher.Check(check_jni ? "expected non-null java.lang.reflect.Field"
283 : "jlr_field == null");
284
285 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
286 }
287
288 void GetFromReflectedMethod_ToReflectedMethodBadArgumentTest(bool check_jni) {
289 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
290 CheckJniAbortCatcher check_jni_abort_catcher;
291
292 jclass c = env_->FindClass("java/lang/String");
293 ASSERT_NE(c, nullptr);
294 jmethodID mid = env_->GetMethodID(c, "<init>", "()V");
295 ASSERT_NE(mid, nullptr);
296
297 // Check class argument for null argument, not checked in non-check JNI.
298 jobject method = env_->ToReflectedMethod(nullptr, mid, JNI_FALSE);
299 if (check_jni) {
300 EXPECT_EQ(method, nullptr);
301 check_jni_abort_catcher.Check("ToReflectedMethod received NULL jclass");
302 } else {
303 EXPECT_NE(method, nullptr);
304 }
305
306 method = env_->ToReflectedMethod(c, nullptr, JNI_FALSE);
307 EXPECT_EQ(method, nullptr);
308 check_jni_abort_catcher.Check(check_jni ? "jmethodID was NULL"
309 : "mid == null");
310 mid = env_->FromReflectedMethod(method);
311 ASSERT_EQ(mid, nullptr);
312 check_jni_abort_catcher.Check(check_jni ? "expected non-null method" : "jlr_method == null");
313
314 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
315 }
316
317 void RegisterAndUnregisterNativesBadArguments(bool check_jni,
318 CheckJniAbortCatcher* check_jni_abort_catcher) {
319 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
320 // Passing a class of null is a failure.
321 {
322 JNINativeMethod methods[] = { };
323 EXPECT_EQ(env_->RegisterNatives(nullptr, methods, 0), JNI_ERR);
324 check_jni_abort_catcher->Check(check_jni ? "RegisterNatives received NULL jclass"
325 : "java_class == null");
326 }
327
328 // Passing methods as null is a failure.
329 jclass jlobject = env_->FindClass("java/lang/Object");
330 EXPECT_EQ(env_->RegisterNatives(jlobject, nullptr, 1), JNI_ERR);
331 check_jni_abort_catcher->Check("methods == null");
332
333 // Unregisters null is a failure.
334 EXPECT_EQ(env_->UnregisterNatives(nullptr), JNI_ERR);
335 check_jni_abort_catcher->Check(check_jni ? "UnregisterNatives received NULL jclass"
336 : "java_class == null");
337
338 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
339 }
340
341
342 void GetPrimitiveArrayElementsOfWrongType(bool check_jni) {
343 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
344 CheckJniAbortCatcher jni_abort_catcher;
345
346 jbooleanArray array = env_->NewBooleanArray(10);
347 jboolean is_copy;
348 EXPECT_EQ(env_->GetByteArrayElements(reinterpret_cast<jbyteArray>(array), &is_copy), nullptr);
349 jni_abort_catcher.Check(
350 check_jni ? "incompatible array type boolean[] expected byte[]"
351 : "attempt to get byte primitive array elements with an object of type boolean[]");
352 EXPECT_EQ(env_->GetShortArrayElements(reinterpret_cast<jshortArray>(array), &is_copy), nullptr);
353 jni_abort_catcher.Check(
354 check_jni ? "incompatible array type boolean[] expected short[]"
355 : "attempt to get short primitive array elements with an object of type boolean[]");
356 EXPECT_EQ(env_->GetCharArrayElements(reinterpret_cast<jcharArray>(array), &is_copy), nullptr);
357 jni_abort_catcher.Check(
358 check_jni ? "incompatible array type boolean[] expected char[]"
359 : "attempt to get char primitive array elements with an object of type boolean[]");
360 EXPECT_EQ(env_->GetIntArrayElements(reinterpret_cast<jintArray>(array), &is_copy), nullptr);
361 jni_abort_catcher.Check(
362 check_jni ? "incompatible array type boolean[] expected int[]"
363 : "attempt to get int primitive array elements with an object of type boolean[]");
364 EXPECT_EQ(env_->GetLongArrayElements(reinterpret_cast<jlongArray>(array), &is_copy), nullptr);
365 jni_abort_catcher.Check(
366 check_jni ? "incompatible array type boolean[] expected long[]"
367 : "attempt to get long primitive array elements with an object of type boolean[]");
368 EXPECT_EQ(env_->GetFloatArrayElements(reinterpret_cast<jfloatArray>(array), &is_copy), nullptr);
369 jni_abort_catcher.Check(
370 check_jni ? "incompatible array type boolean[] expected float[]"
371 : "attempt to get float primitive array elements with an object of type boolean[]");
372 EXPECT_EQ(env_->GetDoubleArrayElements(reinterpret_cast<jdoubleArray>(array), &is_copy), nullptr);
373 jni_abort_catcher.Check(
374 check_jni ? "incompatible array type boolean[] expected double[]"
375 : "attempt to get double primitive array elements with an object of type boolean[]");
376 jbyteArray array2 = env_->NewByteArray(10);
377 EXPECT_EQ(env_->GetBooleanArrayElements(reinterpret_cast<jbooleanArray>(array2), &is_copy),
378 nullptr);
379 jni_abort_catcher.Check(
380 check_jni ? "incompatible array type byte[] expected boolean[]"
381 : "attempt to get boolean primitive array elements with an object of type byte[]");
382 jobject object = env_->NewStringUTF("Test String");
383 EXPECT_EQ(env_->GetBooleanArrayElements(reinterpret_cast<jbooleanArray>(object), &is_copy),
384 nullptr);
385 jni_abort_catcher.Check(
386 check_jni ? "jarray argument has non-array type: java.lang.String"
387 : "attempt to get boolean primitive array elements with an object of type java.lang.String");
388
389 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
390 }
391
392 void ReleasePrimitiveArrayElementsOfWrongType(bool check_jni) {
393 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
394 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogersdd11d2a2014-11-19 10:06:46 -0800395 {
396 jbooleanArray array = env_->NewBooleanArray(10);
397 ASSERT_TRUE(array != nullptr);
398 jboolean is_copy;
399 jboolean* elements = env_->GetBooleanArrayElements(array, &is_copy);
400 ASSERT_TRUE(elements != nullptr);
401 env_->ReleaseByteArrayElements(reinterpret_cast<jbyteArray>(array),
402 reinterpret_cast<jbyte*>(elements), 0);
403 jni_abort_catcher.Check(
404 check_jni ? "incompatible array type boolean[] expected byte[]"
405 : "attempt to release byte primitive array elements with an object of type boolean[]");
406 env_->ReleaseShortArrayElements(reinterpret_cast<jshortArray>(array),
407 reinterpret_cast<jshort*>(elements), 0);
408 jni_abort_catcher.Check(
409 check_jni ? "incompatible array type boolean[] expected short[]"
410 : "attempt to release short primitive array elements with an object of type boolean[]");
411 env_->ReleaseCharArrayElements(reinterpret_cast<jcharArray>(array),
412 reinterpret_cast<jchar*>(elements), 0);
413 jni_abort_catcher.Check(
414 check_jni ? "incompatible array type boolean[] expected char[]"
415 : "attempt to release char primitive array elements with an object of type boolean[]");
416 env_->ReleaseIntArrayElements(reinterpret_cast<jintArray>(array),
417 reinterpret_cast<jint*>(elements), 0);
418 jni_abort_catcher.Check(
419 check_jni ? "incompatible array type boolean[] expected int[]"
420 : "attempt to release int primitive array elements with an object of type boolean[]");
421 env_->ReleaseLongArrayElements(reinterpret_cast<jlongArray>(array),
422 reinterpret_cast<jlong*>(elements), 0);
423 jni_abort_catcher.Check(
424 check_jni ? "incompatible array type boolean[] expected long[]"
425 : "attempt to release long primitive array elements with an object of type boolean[]");
426 env_->ReleaseFloatArrayElements(reinterpret_cast<jfloatArray>(array),
427 reinterpret_cast<jfloat*>(elements), 0);
428 jni_abort_catcher.Check(
429 check_jni ? "incompatible array type boolean[] expected float[]"
430 : "attempt to release float primitive array elements with an object of type boolean[]");
431 env_->ReleaseDoubleArrayElements(reinterpret_cast<jdoubleArray>(array),
432 reinterpret_cast<jdouble*>(elements), 0);
433 jni_abort_catcher.Check(
434 check_jni ? "incompatible array type boolean[] expected double[]"
435 : "attempt to release double primitive array elements with an object of type boolean[]");
Ian Rogers68d8b422014-07-17 11:09:10 -0700436
Ian Rogersdd11d2a2014-11-19 10:06:46 -0800437 // Don't leak the elements array.
438 env_->ReleaseBooleanArrayElements(array, elements, 0);
439 }
440 {
441 jbyteArray array = env_->NewByteArray(10);
442 jboolean is_copy;
443 jbyte* elements = env_->GetByteArrayElements(array, &is_copy);
444
445 env_->ReleaseBooleanArrayElements(reinterpret_cast<jbooleanArray>(array),
446 reinterpret_cast<jboolean*>(elements), 0);
447 jni_abort_catcher.Check(
448 check_jni ? "incompatible array type byte[] expected boolean[]"
449 : "attempt to release boolean primitive array elements with an object of type byte[]");
450 jobject object = env_->NewStringUTF("Test String");
451 env_->ReleaseBooleanArrayElements(reinterpret_cast<jbooleanArray>(object),
452 reinterpret_cast<jboolean*>(elements), 0);
453 jni_abort_catcher.Check(
454 check_jni ? "jarray argument has non-array type: java.lang.String"
455 : "attempt to release boolean primitive array elements with an object of type "
Ian Rogers68d8b422014-07-17 11:09:10 -0700456 "java.lang.String");
457
Ian Rogersdd11d2a2014-11-19 10:06:46 -0800458 // Don't leak the elements array.
459 env_->ReleaseByteArrayElements(array, elements, 0);
460 }
Ian Rogers68d8b422014-07-17 11:09:10 -0700461 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
462 }
463
464 void GetReleasePrimitiveArrayCriticalOfWrongType(bool check_jni) {
465 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
466 CheckJniAbortCatcher jni_abort_catcher;
467
468 jobject object = env_->NewStringUTF("Test String");
469 jboolean is_copy;
470 void* elements = env_->GetPrimitiveArrayCritical(reinterpret_cast<jarray>(object), &is_copy);
471 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
472 : "expected primitive array, given java.lang.String");
473 env_->ReleasePrimitiveArrayCritical(reinterpret_cast<jarray>(object), elements, 0);
474 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
475 : "expected primitive array, given java.lang.String");
476
477 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
478 }
479
480 void GetPrimitiveArrayRegionElementsOfWrongType(bool check_jni) {
481 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
482 CheckJniAbortCatcher jni_abort_catcher;
483 constexpr size_t kLength = 10;
484 jbooleanArray array = env_->NewBooleanArray(kLength);
485 ASSERT_TRUE(array != nullptr);
486 jboolean elements[kLength];
487 env_->GetByteArrayRegion(reinterpret_cast<jbyteArray>(array), 0, kLength,
488 reinterpret_cast<jbyte*>(elements));
489 jni_abort_catcher.Check(
490 check_jni ? "incompatible array type boolean[] expected byte[]"
491 : "attempt to get region of byte primitive array elements with an object of type boolean[]");
492 env_->GetShortArrayRegion(reinterpret_cast<jshortArray>(array), 0, kLength,
493 reinterpret_cast<jshort*>(elements));
494 jni_abort_catcher.Check(
495 check_jni ? "incompatible array type boolean[] expected short[]"
496 : "attempt to get region of short primitive array elements with an object of type boolean[]");
497 env_->GetCharArrayRegion(reinterpret_cast<jcharArray>(array), 0, kLength,
498 reinterpret_cast<jchar*>(elements));
499 jni_abort_catcher.Check(
500 check_jni ? "incompatible array type boolean[] expected char[]"
501 : "attempt to get region of char primitive array elements with an object of type boolean[]");
502 env_->GetIntArrayRegion(reinterpret_cast<jintArray>(array), 0, kLength,
503 reinterpret_cast<jint*>(elements));
504 jni_abort_catcher.Check(
505 check_jni ? "incompatible array type boolean[] expected int[]"
506 : "attempt to get region of int primitive array elements with an object of type boolean[]");
507 env_->GetLongArrayRegion(reinterpret_cast<jlongArray>(array), 0, kLength,
508 reinterpret_cast<jlong*>(elements));
509 jni_abort_catcher.Check(
510 check_jni ? "incompatible array type boolean[] expected long[]"
511 : "attempt to get region of long primitive array elements with an object of type boolean[]");
512 env_->GetFloatArrayRegion(reinterpret_cast<jfloatArray>(array), 0, kLength,
513 reinterpret_cast<jfloat*>(elements));
514 jni_abort_catcher.Check(
515 check_jni ? "incompatible array type boolean[] expected float[]"
516 : "attempt to get region of float primitive array elements with an object of type boolean[]");
517 env_->GetDoubleArrayRegion(reinterpret_cast<jdoubleArray>(array), 0, kLength,
518 reinterpret_cast<jdouble*>(elements));
519 jni_abort_catcher.Check(
520 check_jni ? "incompatible array type boolean[] expected double[]"
521 : "attempt to get region of double primitive array elements with an object of type boolean[]");
522 jbyteArray array2 = env_->NewByteArray(10);
523 env_->GetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(array2), 0, kLength,
524 reinterpret_cast<jboolean*>(elements));
525 jni_abort_catcher.Check(
526 check_jni ? "incompatible array type byte[] expected boolean[]"
527 : "attempt to get region of boolean primitive array elements with an object of type byte[]");
528 jobject object = env_->NewStringUTF("Test String");
529 env_->GetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(object), 0, kLength,
530 reinterpret_cast<jboolean*>(elements));
531 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
532 : "attempt to get region of boolean primitive array elements with an object of type "
533 "java.lang.String");
534
535 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
536 }
537
538 void SetPrimitiveArrayRegionElementsOfWrongType(bool check_jni) {
539 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
540 CheckJniAbortCatcher jni_abort_catcher;
541 constexpr size_t kLength = 10;
542 jbooleanArray array = env_->NewBooleanArray(kLength);
543 ASSERT_TRUE(array != nullptr);
544 jboolean elements[kLength];
545 env_->SetByteArrayRegion(reinterpret_cast<jbyteArray>(array), 0, kLength,
546 reinterpret_cast<jbyte*>(elements));
547 jni_abort_catcher.Check(
548 check_jni ? "incompatible array type boolean[] expected byte[]"
549 : "attempt to set region of byte primitive array elements with an object of type boolean[]");
550 env_->SetShortArrayRegion(reinterpret_cast<jshortArray>(array), 0, kLength,
551 reinterpret_cast<jshort*>(elements));
552 jni_abort_catcher.Check(
553 check_jni ? "incompatible array type boolean[] expected short[]"
554 : "attempt to set region of short primitive array elements with an object of type boolean[]");
555 env_->SetCharArrayRegion(reinterpret_cast<jcharArray>(array), 0, kLength,
556 reinterpret_cast<jchar*>(elements));
557 jni_abort_catcher.Check(
558 check_jni ? "incompatible array type boolean[] expected char[]"
559 : "attempt to set region of char primitive array elements with an object of type boolean[]");
560 env_->SetIntArrayRegion(reinterpret_cast<jintArray>(array), 0, kLength,
561 reinterpret_cast<jint*>(elements));
562 jni_abort_catcher.Check(
563 check_jni ? "incompatible array type boolean[] expected int[]"
564 : "attempt to set region of int primitive array elements with an object of type boolean[]");
565 env_->SetLongArrayRegion(reinterpret_cast<jlongArray>(array), 0, kLength,
566 reinterpret_cast<jlong*>(elements));
567 jni_abort_catcher.Check(
568 check_jni ? "incompatible array type boolean[] expected long[]"
569 : "attempt to set region of long primitive array elements with an object of type boolean[]");
570 env_->SetFloatArrayRegion(reinterpret_cast<jfloatArray>(array), 0, kLength,
571 reinterpret_cast<jfloat*>(elements));
572 jni_abort_catcher.Check(
573 check_jni ? "incompatible array type boolean[] expected float[]"
574 : "attempt to set region of float primitive array elements with an object of type boolean[]");
575 env_->SetDoubleArrayRegion(reinterpret_cast<jdoubleArray>(array), 0, kLength,
576 reinterpret_cast<jdouble*>(elements));
577 jni_abort_catcher.Check(
578 check_jni ? "incompatible array type boolean[] expected double[]"
579 : "attempt to set region of double primitive array elements with an object of type boolean[]");
580 jbyteArray array2 = env_->NewByteArray(10);
581 env_->SetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(array2), 0, kLength,
582 reinterpret_cast<jboolean*>(elements));
583 jni_abort_catcher.Check(
584 check_jni ? "incompatible array type byte[] expected boolean[]"
585 : "attempt to set region of boolean primitive array elements with an object of type byte[]");
586 jobject object = env_->NewStringUTF("Test String");
587 env_->SetBooleanArrayRegion(reinterpret_cast<jbooleanArray>(object), 0, kLength,
588 reinterpret_cast<jboolean*>(elements));
589 jni_abort_catcher.Check(check_jni ? "jarray argument has non-array type: java.lang.String"
590 : "attempt to set region of boolean primitive array elements with an object of type "
591 "java.lang.String");
592 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
593 }
594
595 void NewObjectArrayBadArguments(bool check_jni) {
596 bool old_check_jni = vm_->SetCheckJniEnabled(check_jni);
597 CheckJniAbortCatcher jni_abort_catcher;
598
599 jclass element_class = env_->FindClass("java/lang/String");
600 ASSERT_NE(element_class, nullptr);
601
602 env_->NewObjectArray(-1, element_class, nullptr);
603 jni_abort_catcher.Check(check_jni ? "negative jsize: -1" : "negative array length: -1");
604
605 env_->NewObjectArray(std::numeric_limits<jint>::min(), element_class, nullptr);
606 jni_abort_catcher.Check(check_jni ? "negative jsize: -2147483648"
607 : "negative array length: -2147483648");
608
609 EXPECT_EQ(check_jni, vm_->SetCheckJniEnabled(old_check_jni));
610 }
611
Andreas Gampe5f4a09a2015-09-28 13:16:33 -0700612 void SetUpForTest(bool direct, const char* method_name, const char* method_sig,
613 void* native_fnptr) {
614 // Initialize class loader and set generic JNI entrypoint.
615 // Note: this code is adapted from the jni_compiler_test, and taken with minimal modifications.
616 if (!runtime_->IsStarted()) {
617 {
618 ScopedObjectAccess soa(Thread::Current());
619 class_loader_ = LoadDex("MyClassNatives");
620 StackHandleScope<1> hs(soa.Self());
621 Handle<mirror::ClassLoader> loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -0700622 hs.NewHandle(soa.Decode<mirror::ClassLoader>(class_loader_)));
Andreas Gampe5f4a09a2015-09-28 13:16:33 -0700623 mirror::Class* c = class_linker_->FindClass(soa.Self(), "LMyClassNatives;", loader);
624 const auto pointer_size = class_linker_->GetImagePointerSize();
625 ArtMethod* method = direct ? c->FindDirectMethod(method_name, method_sig, pointer_size) :
626 c->FindVirtualMethod(method_name, method_sig, pointer_size);
627 ASSERT_TRUE(method != nullptr) << method_name << " " << method_sig;
628 method->SetEntryPointFromQuickCompiledCode(class_linker_->GetRuntimeQuickGenericJniStub());
629 }
630 // Start runtime.
631 Thread::Current()->TransitionFromSuspendedToRunnable();
632 bool started = runtime_->Start();
633 CHECK(started);
634 }
635 // JNI operations after runtime start.
636 env_ = Thread::Current()->GetJniEnv();
637 jklass_ = env_->FindClass("MyClassNatives");
638 ASSERT_TRUE(jklass_ != nullptr) << method_name << " " << method_sig;
639
640 if (direct) {
641 jmethod_ = env_->GetStaticMethodID(jklass_, method_name, method_sig);
642 } else {
643 jmethod_ = env_->GetMethodID(jklass_, method_name, method_sig);
644 }
645 ASSERT_TRUE(jmethod_ != nullptr) << method_name << " " << method_sig;
646
647 if (native_fnptr != nullptr) {
648 JNINativeMethod methods[] = { { method_name, method_sig, native_fnptr } };
649 ASSERT_EQ(JNI_OK, env_->RegisterNatives(jklass_, methods, 1))
650 << method_name << " " << method_sig;
651 } else {
652 env_->UnregisterNatives(jklass_);
653 }
654
655 jmethodID constructor = env_->GetMethodID(jklass_, "<init>", "()V");
656 jobj_ = env_->NewObject(jklass_, constructor);
657 ASSERT_TRUE(jobj_ != nullptr) << method_name << " " << method_sig;
658 }
659
Elliott Hughesa2501992011-08-26 19:39:54 -0700660 JavaVMExt* vm_;
Brian Carlstrom4d571432012-05-16 00:21:41 -0700661 JNIEnv* env_;
Elliott Hughes814e4032011-08-23 12:07:56 -0700662 jclass aioobe_;
Elliott Hughesb264f082012-04-06 17:10:10 -0700663 jclass ase_;
Elliott Hughesb465ab02011-08-24 11:21:21 -0700664 jclass sioobe_;
Andreas Gampe5f4a09a2015-09-28 13:16:33 -0700665
666 jclass jklass_;
667 jobject jobj_;
668 jobject class_loader_;
669 jmethodID jmethod_;
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700670};
671
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700672TEST_F(JniInternalTest, AllocObject) {
673 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700674 ASSERT_NE(c, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700675 jobject o = env_->AllocObject(c);
Ian Rogers2d10b202014-05-12 19:15:18 -0700676 ASSERT_NE(o, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700677
678 // We have an instance of the class we asked for...
679 ASSERT_TRUE(env_->IsInstanceOf(o, c));
680 // ...whose fields haven't been initialized because
681 // we didn't call a constructor.
Vladimir Markofdaf0f42016-10-13 19:29:53 +0100682 // Even with string compression empty string has `count == 0`.
683 ASSERT_EQ(0, env_->GetIntField(o, env_->GetFieldID(c, "count", "I")));
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700684}
685
Elliott Hughesc7ac37f2011-08-12 12:21:58 -0700686TEST_F(JniInternalTest, GetVersion) {
687 ASSERT_EQ(JNI_VERSION_1_6, env_->GetVersion());
688}
689
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700690TEST_F(JniInternalTest, FindClass) {
Andreas Gampe369810a2015-01-14 19:53:31 -0800691 // This tests leads to warnings in the log.
692 ScopedLogSeverity sls(LogSeverity::ERROR);
693
Ian Rogers68d8b422014-07-17 11:09:10 -0700694 FindClassTest(false);
695 FindClassTest(true);
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700696}
697
Elliott Hughescdf53122011-08-19 15:46:09 -0700698TEST_F(JniInternalTest, GetFieldID) {
699 jclass jlnsfe = env_->FindClass("java/lang/NoSuchFieldError");
Ian Rogers2d10b202014-05-12 19:15:18 -0700700 ASSERT_NE(jlnsfe, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700701 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700702 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700703
704 // Wrong type.
705 jfieldID fid = env_->GetFieldID(c, "count", "J");
Ian Rogers2d10b202014-05-12 19:15:18 -0700706 EXPECT_EQ(nullptr, fid);
707 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700708
Ian Rogersb17d08b2011-09-02 16:16:49 -0700709 // Wrong type where type doesn't exist.
710 fid = env_->GetFieldID(c, "count", "Lrod/jane/freddy;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700711 EXPECT_EQ(nullptr, fid);
712 ExpectException(jlnsfe);
Ian Rogersb17d08b2011-09-02 16:16:49 -0700713
Elliott Hughescdf53122011-08-19 15:46:09 -0700714 // Wrong name.
715 fid = env_->GetFieldID(c, "Count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700716 EXPECT_EQ(nullptr, fid);
717 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700718
719 // Good declared field lookup.
720 fid = env_->GetFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700721 EXPECT_NE(nullptr, fid);
Elliott Hughescdf53122011-08-19 15:46:09 -0700722 EXPECT_FALSE(env_->ExceptionCheck());
723
724 // Good superclass field lookup.
725 c = env_->FindClass("java/lang/StringBuilder");
726 fid = env_->GetFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700727 EXPECT_NE(nullptr, fid);
728 EXPECT_NE(fid, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700729 EXPECT_FALSE(env_->ExceptionCheck());
730
731 // Not instance.
732 fid = env_->GetFieldID(c, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700733 EXPECT_EQ(nullptr, fid);
734 ExpectException(jlnsfe);
735
736 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700737 GetFieldIdBadArgumentTest(false);
738 GetFieldIdBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700739}
740
741TEST_F(JniInternalTest, GetStaticFieldID) {
742 jclass jlnsfe = env_->FindClass("java/lang/NoSuchFieldError");
Ian Rogers2d10b202014-05-12 19:15:18 -0700743 ASSERT_NE(jlnsfe, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700744 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700745 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700746
747 // Wrong type.
748 jfieldID fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "J");
Ian Rogers2d10b202014-05-12 19:15:18 -0700749 EXPECT_EQ(nullptr, fid);
750 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700751
Ian Rogersb17d08b2011-09-02 16:16:49 -0700752 // Wrong type where type doesn't exist.
753 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "Lrod/jane/freddy;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700754 EXPECT_EQ(nullptr, fid);
755 ExpectException(jlnsfe);
Ian Rogersb17d08b2011-09-02 16:16:49 -0700756
Elliott Hughescdf53122011-08-19 15:46:09 -0700757 // Wrong name.
758 fid = env_->GetStaticFieldID(c, "cASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700759 EXPECT_EQ(nullptr, fid);
760 ExpectException(jlnsfe);
Elliott Hughescdf53122011-08-19 15:46:09 -0700761
762 // Good declared field lookup.
763 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700764 EXPECT_NE(nullptr, fid);
765 EXPECT_NE(fid, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700766 EXPECT_FALSE(env_->ExceptionCheck());
767
768 // Not static.
769 fid = env_->GetStaticFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700770 EXPECT_EQ(nullptr, fid);
771 ExpectException(jlnsfe);
772
773 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700774 GetStaticFieldIdBadArgumentTest(false);
775 GetStaticFieldIdBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700776}
777
Ian Rogers4dd71f12011-08-16 14:16:02 -0700778TEST_F(JniInternalTest, GetMethodID) {
779 jclass jlobject = env_->FindClass("java/lang/Object");
780 jclass jlstring = env_->FindClass("java/lang/String");
781 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
Brian Carlstrom004644f2014-06-18 08:34:01 -0700782 jclass jncrbc = env_->FindClass("java/nio/channels/ReadableByteChannel");
Ian Rogers4dd71f12011-08-16 14:16:02 -0700783
Brian Carlstrom004644f2014-06-18 08:34:01 -0700784 // Sanity check that no exceptions are pending.
Elliott Hughescdf53122011-08-19 15:46:09 -0700785 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700786
787 // Check that java.lang.Object.foo() doesn't exist and NoSuchMethodError is
Brian Carlstrom004644f2014-06-18 08:34:01 -0700788 // a pending exception.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700789 jmethodID method = env_->GetMethodID(jlobject, "foo", "()V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700790 EXPECT_EQ(nullptr, method);
791 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700792
Brian Carlstrom004644f2014-06-18 08:34:01 -0700793 // Check that java.lang.Object.equals() does exist.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700794 method = env_->GetMethodID(jlobject, "equals", "(Ljava/lang/Object;)Z");
Ian Rogers2d10b202014-05-12 19:15:18 -0700795 EXPECT_NE(nullptr, method);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700796 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700797
798 // Check that GetMethodID for java.lang.String.valueOf(int) fails as the
Brian Carlstrom004644f2014-06-18 08:34:01 -0700799 // method is static.
Ian Rogers4dd71f12011-08-16 14:16:02 -0700800 method = env_->GetMethodID(jlstring, "valueOf", "(I)Ljava/lang/String;");
Ian Rogers2d10b202014-05-12 19:15:18 -0700801 EXPECT_EQ(nullptr, method);
802 ExpectException(jlnsme);
Brian Carlstromea46f952013-07-30 01:26:50 -0700803
Brian Carlstrom004644f2014-06-18 08:34:01 -0700804 // Check that GetMethodID for java.lang.NoSuchMethodError.<init>(String) finds the constructor.
Brian Carlstromea46f952013-07-30 01:26:50 -0700805 method = env_->GetMethodID(jlnsme, "<init>", "(Ljava/lang/String;)V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700806 EXPECT_NE(nullptr, method);
Brian Carlstromea46f952013-07-30 01:26:50 -0700807 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers2d10b202014-05-12 19:15:18 -0700808
Brian Carlstrom004644f2014-06-18 08:34:01 -0700809 // Check that GetMethodID can find a interface method inherited from another interface.
810 method = env_->GetMethodID(jncrbc, "close", "()V");
811 EXPECT_NE(nullptr, method);
812 EXPECT_FALSE(env_->ExceptionCheck());
813
Ian Rogers2d10b202014-05-12 19:15:18 -0700814 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700815 GetMethodIdBadArgumentTest(false);
816 GetMethodIdBadArgumentTest(true);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700817}
818
Mathieu Chartiera1e78fa2014-08-20 12:09:38 -0700819TEST_F(JniInternalTest, CallVoidMethodNullReceiver) {
820 jclass jlobject = env_->FindClass("java/lang/Object");
821 jmethodID method;
822
823 // Check that GetMethodID for java.lang.NoSuchMethodError.<init>(String) finds the constructor.
824 method = env_->GetMethodID(jlobject, "<init>", "()V");
825 EXPECT_NE(nullptr, method);
826 EXPECT_FALSE(env_->ExceptionCheck());
827
828 // Null object to CallVoidMethod.
829 CheckJniAbortCatcher check_jni_abort_catcher;
830 env_->CallVoidMethod(nullptr, method);
831 check_jni_abort_catcher.Check("null");
832}
833
Ian Rogers4dd71f12011-08-16 14:16:02 -0700834TEST_F(JniInternalTest, GetStaticMethodID) {
835 jclass jlobject = env_->FindClass("java/lang/Object");
836 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
837
838 // Sanity check that no exceptions are pending
Elliott Hughescdf53122011-08-19 15:46:09 -0700839 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700840
841 // Check that java.lang.Object.foo() doesn't exist and NoSuchMethodError is
842 // a pending exception
843 jmethodID method = env_->GetStaticMethodID(jlobject, "foo", "()V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700844 EXPECT_EQ(nullptr, method);
845 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700846
847 // Check that GetStaticMethodID for java.lang.Object.equals(Object) fails as
848 // the method is not static
849 method = env_->GetStaticMethodID(jlobject, "equals", "(Ljava/lang/Object;)Z");
Ian Rogers2d10b202014-05-12 19:15:18 -0700850 EXPECT_EQ(nullptr, method);
851 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700852
853 // Check that java.lang.String.valueOf(int) does exist
Ian Rogers4dd71f12011-08-16 14:16:02 -0700854 jclass jlstring = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700855 method = env_->GetStaticMethodID(jlstring, "valueOf", "(I)Ljava/lang/String;");
856 EXPECT_NE(nullptr, method);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700857 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers2d10b202014-05-12 19:15:18 -0700858
859 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700860 GetStaticMethodIdBadArgumentTest(false);
861 GetStaticMethodIdBadArgumentTest(true);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700862}
863
Andreas Gampea8e3b862016-10-17 20:12:52 -0700864static size_t GetLocalsCapacity(JNIEnv* env) {
865 ScopedObjectAccess soa(Thread::Current());
866 return reinterpret_cast<JNIEnvExt*>(env)->locals.Capacity();
867}
868
Elliott Hughescdf53122011-08-19 15:46:09 -0700869TEST_F(JniInternalTest, FromReflectedField_ToReflectedField) {
870 jclass jlrField = env_->FindClass("java/lang/reflect/Field");
871 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700872 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700873 jfieldID fid = env_->GetFieldID(c, "count", "I");
Ian Rogers2d10b202014-05-12 19:15:18 -0700874 ASSERT_NE(fid, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700875 // Turn the fid into a java.lang.reflect.Field...
876 jobject field = env_->ToReflectedField(c, fid, JNI_FALSE);
Andreas Gampea8e3b862016-10-17 20:12:52 -0700877 size_t capacity_before = GetLocalsCapacity(env_);
878 for (size_t i = 0; i <= 10; ++i) {
Mathieu Chartier41da5962014-11-15 13:07:39 -0800879 // Regression test for b/18396311, ToReflectedField leaking local refs causing a local
880 // reference table overflows with 512 references to ArtField
881 env_->DeleteLocalRef(env_->ToReflectedField(c, fid, JNI_FALSE));
882 }
Andreas Gampea8e3b862016-10-17 20:12:52 -0700883 size_t capacity_after = GetLocalsCapacity(env_);
884 ASSERT_EQ(capacity_before, capacity_after);
885
Ian Rogers2d10b202014-05-12 19:15:18 -0700886 ASSERT_NE(c, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700887 ASSERT_TRUE(env_->IsInstanceOf(field, jlrField));
888 // ...and back again.
889 jfieldID fid2 = env_->FromReflectedField(field);
Ian Rogers2d10b202014-05-12 19:15:18 -0700890 ASSERT_NE(fid2, nullptr);
Brian Carlstromea46f952013-07-30 01:26:50 -0700891 // Make sure we can actually use it.
892 jstring s = env_->NewStringUTF("poop");
jessicahandojo3aaa37b2016-07-29 14:46:37 -0700893 if (mirror::kUseStringCompression) {
Vladimir Markofdaf0f42016-10-13 19:29:53 +0100894 ASSERT_EQ(mirror::String::GetFlaggedCount(4, /* compressible */ true),
895 env_->GetIntField(s, fid2));
jessicahandojo3aaa37b2016-07-29 14:46:37 -0700896 // Create incompressible string
897 jstring s_16 = env_->NewStringUTF("\u0444\u0444");
Vladimir Markofdaf0f42016-10-13 19:29:53 +0100898 ASSERT_EQ(mirror::String::GetFlaggedCount(2, /* compressible */ false),
899 env_->GetIntField(s_16, fid2));
jessicahandojo3aaa37b2016-07-29 14:46:37 -0700900 } else {
901 ASSERT_EQ(4, env_->GetIntField(s, fid2));
902 }
Ian Rogers2d10b202014-05-12 19:15:18 -0700903 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700904 GetFromReflectedField_ToReflectedFieldBadArgumentTest(false);
905 GetFromReflectedField_ToReflectedFieldBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700906}
907
908TEST_F(JniInternalTest, FromReflectedMethod_ToReflectedMethod) {
909 jclass jlrMethod = env_->FindClass("java/lang/reflect/Method");
Sebastien Hertzd3333762014-06-26 14:45:07 +0200910 ASSERT_NE(jlrMethod, nullptr);
911 jclass jlrConstructor = env_->FindClass("java/lang/reflect/Constructor");
912 ASSERT_NE(jlrConstructor, nullptr);
Elliott Hughescdf53122011-08-19 15:46:09 -0700913 jclass c = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -0700914 ASSERT_NE(c, nullptr);
Sebastien Hertzd3333762014-06-26 14:45:07 +0200915
916 jmethodID mid = env_->GetMethodID(c, "<init>", "()V");
Ian Rogers2d10b202014-05-12 19:15:18 -0700917 ASSERT_NE(mid, nullptr);
Sebastien Hertzd3333762014-06-26 14:45:07 +0200918 // Turn the mid into a java.lang.reflect.Constructor...
Elliott Hughescdf53122011-08-19 15:46:09 -0700919 jobject method = env_->ToReflectedMethod(c, mid, JNI_FALSE);
Andreas Gampea8e3b862016-10-17 20:12:52 -0700920 size_t capacity_before = GetLocalsCapacity(env_);
921 for (size_t i = 0; i <= 10; ++i) {
Mathieu Chartier41da5962014-11-15 13:07:39 -0800922 // Regression test for b/18396311, ToReflectedMethod leaking local refs causing a local
923 // reference table overflows with 512 references to ArtMethod
924 env_->DeleteLocalRef(env_->ToReflectedMethod(c, mid, JNI_FALSE));
925 }
Andreas Gampea8e3b862016-10-17 20:12:52 -0700926 size_t capacity_after = GetLocalsCapacity(env_);
927 ASSERT_EQ(capacity_before, capacity_after);
Sebastien Hertzd3333762014-06-26 14:45:07 +0200928 ASSERT_NE(method, nullptr);
929 ASSERT_TRUE(env_->IsInstanceOf(method, jlrConstructor));
Elliott Hughescdf53122011-08-19 15:46:09 -0700930 // ...and back again.
931 jmethodID mid2 = env_->FromReflectedMethod(method);
Ian Rogers2d10b202014-05-12 19:15:18 -0700932 ASSERT_NE(mid2, nullptr);
Brian Carlstromea46f952013-07-30 01:26:50 -0700933 // Make sure we can actually use it.
Sebastien Hertzd3333762014-06-26 14:45:07 +0200934 jstring s = reinterpret_cast<jstring>(env_->AllocObject(c));
935 ASSERT_NE(s, nullptr);
936 env_->CallVoidMethod(s, mid2);
Jeff Hao39b6c242015-05-19 20:30:23 -0700937 ASSERT_EQ(JNI_FALSE, env_->ExceptionCheck());
Jeff Hao848f70a2014-01-15 13:49:50 -0800938 env_->ExceptionClear();
Sebastien Hertzd3333762014-06-26 14:45:07 +0200939
940 mid = env_->GetMethodID(c, "length", "()I");
941 ASSERT_NE(mid, nullptr);
942 // Turn the mid into a java.lang.reflect.Method...
943 method = env_->ToReflectedMethod(c, mid, JNI_FALSE);
944 ASSERT_NE(method, nullptr);
945 ASSERT_TRUE(env_->IsInstanceOf(method, jlrMethod));
946 // ...and back again.
947 mid2 = env_->FromReflectedMethod(method);
948 ASSERT_NE(mid2, nullptr);
949 // Make sure we can actually use it.
950 s = env_->NewStringUTF("poop");
951 ASSERT_NE(s, nullptr);
Ian Rogers5d27faf2014-05-02 17:17:18 -0700952 ASSERT_EQ(4, env_->CallIntMethod(s, mid2));
Ian Rogers2d10b202014-05-12 19:15:18 -0700953
954 // Bad arguments.
Ian Rogers68d8b422014-07-17 11:09:10 -0700955 GetFromReflectedMethod_ToReflectedMethodBadArgumentTest(false);
956 GetFromReflectedMethod_ToReflectedMethodBadArgumentTest(true);
Elliott Hughescdf53122011-08-19 15:46:09 -0700957}
958
Ian Rogers2d10b202014-05-12 19:15:18 -0700959static void BogusMethod() {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700960 // You can't pass null function pointers to RegisterNatives.
Elliott Hughes5174fe62011-08-23 15:12:35 -0700961}
962
Ian Rogers2d10b202014-05-12 19:15:18 -0700963TEST_F(JniInternalTest, RegisterAndUnregisterNatives) {
Ian Rogers4dd71f12011-08-16 14:16:02 -0700964 jclass jlobject = env_->FindClass("java/lang/Object");
965 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200966 void* native_function = reinterpret_cast<void*>(BogusMethod);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700967
Ian Rogers2d10b202014-05-12 19:15:18 -0700968 // Sanity check that no exceptions are pending.
Elliott Hughescdf53122011-08-19 15:46:09 -0700969 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700970
Andreas Gampe369810a2015-01-14 19:53:31 -0800971 // The following can print errors to the log we'd like to ignore.
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200972 {
Andreas Gampe369810a2015-01-14 19:53:31 -0800973 ScopedLogSeverity sls(LogSeverity::FATAL);
974 // Check that registering method without name causes a NoSuchMethodError.
975 {
976 JNINativeMethod methods[] = { { nullptr, "()V", native_function } };
977 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
978 }
979 ExpectException(jlnsme);
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200980
Andreas Gampe369810a2015-01-14 19:53:31 -0800981 // Check that registering method without signature causes a NoSuchMethodError.
982 {
983 JNINativeMethod methods[] = { { "notify", nullptr, native_function } };
984 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
985 }
986 ExpectException(jlnsme);
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200987
Andreas Gampe369810a2015-01-14 19:53:31 -0800988 // Check that registering method without function causes a NoSuchMethodError.
989 {
990 JNINativeMethod methods[] = { { "notify", "()V", nullptr } };
991 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
992 }
993 ExpectException(jlnsme);
Sebastien Hertzfa65e842014-07-03 09:39:53 +0200994
Andreas Gampe369810a2015-01-14 19:53:31 -0800995 // Check that registering to a non-existent java.lang.Object.foo() causes a NoSuchMethodError.
996 {
997 JNINativeMethod methods[] = { { "foo", "()V", native_function } };
998 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
999 }
1000 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -07001001
Andreas Gampe369810a2015-01-14 19:53:31 -08001002 // Check that registering non-native methods causes a NoSuchMethodError.
1003 {
1004 JNINativeMethod methods[] = { { "equals", "(Ljava/lang/Object;)Z", native_function } };
1005 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_ERR);
1006 }
1007 ExpectException(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -07001008 }
Ian Rogers4dd71f12011-08-16 14:16:02 -07001009
Ian Rogers2d10b202014-05-12 19:15:18 -07001010 // Check that registering native methods is successful.
Ian Rogers4dd71f12011-08-16 14:16:02 -07001011 {
Sebastien Hertzfa65e842014-07-03 09:39:53 +02001012 JNINativeMethod methods[] = { { "notify", "()V", native_function } };
Ian Rogers2d10b202014-05-12 19:15:18 -07001013 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 1), JNI_OK);
1014 }
1015 EXPECT_FALSE(env_->ExceptionCheck());
1016 EXPECT_EQ(env_->UnregisterNatives(jlobject), JNI_OK);
1017
1018 // Check that registering no methods isn't a failure.
1019 {
1020 JNINativeMethod methods[] = { };
1021 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, 0), JNI_OK);
1022 }
1023 EXPECT_FALSE(env_->ExceptionCheck());
1024 EXPECT_EQ(env_->UnregisterNatives(jlobject), JNI_OK);
1025
1026 // Check that registering a -ve number of methods is a failure.
1027 CheckJniAbortCatcher check_jni_abort_catcher;
1028 for (int i = -10; i < 0; ++i) {
1029 JNINativeMethod methods[] = { };
1030 EXPECT_EQ(env_->RegisterNatives(jlobject, methods, i), JNI_ERR);
1031 check_jni_abort_catcher.Check("negative method count: ");
Ian Rogers4dd71f12011-08-16 14:16:02 -07001032 }
1033 EXPECT_FALSE(env_->ExceptionCheck());
Elliott Hughes5174fe62011-08-23 15:12:35 -07001034
Ian Rogers2d10b202014-05-12 19:15:18 -07001035 // Unregistering a class with no natives is a warning.
1036 EXPECT_EQ(env_->UnregisterNatives(jlnsme), JNI_OK);
Ian Rogers68d8b422014-07-17 11:09:10 -07001037
1038 RegisterAndUnregisterNativesBadArguments(false, &check_jni_abort_catcher);
1039 RegisterAndUnregisterNativesBadArguments(true, &check_jni_abort_catcher);
Ian Rogers4dd71f12011-08-16 14:16:02 -07001040}
1041
Brian Carlstromea46f952013-07-30 01:26:50 -07001042#define EXPECT_PRIMITIVE_ARRAY(new_fn, \
1043 get_region_fn, \
1044 set_region_fn, \
1045 get_elements_fn, \
1046 release_elements_fn, \
1047 scalar_type, \
1048 expected_class_descriptor) \
Ian Rogers2d10b202014-05-12 19:15:18 -07001049 jsize size = 4; \
1050 \
Ian Rogers1d99e452014-01-02 17:36:41 -08001051 { \
1052 CheckJniAbortCatcher jni_abort_catcher; \
Ian Rogers68d8b422014-07-17 11:09:10 -07001053 down_cast<JNIEnvExt*>(env_)->SetCheckJniEnabled(false); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001054 /* Allocate an negative sized array and check it has the right failure type. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001055 EXPECT_EQ(env_->new_fn(-1), nullptr); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001056 jni_abort_catcher.Check("negative array length: -1"); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001057 EXPECT_EQ(env_->new_fn(std::numeric_limits<jint>::min()), nullptr); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001058 jni_abort_catcher.Check("negative array length: -2147483648"); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001059 /* Pass the array as null. */ \
1060 EXPECT_EQ(0, env_->GetArrayLength(nullptr)); \
1061 jni_abort_catcher.Check("java_array == null"); \
1062 env_->get_region_fn(nullptr, 0, 0, nullptr); \
1063 jni_abort_catcher.Check("java_array == null"); \
1064 env_->set_region_fn(nullptr, 0, 0, nullptr); \
1065 jni_abort_catcher.Check("java_array == null"); \
1066 env_->get_elements_fn(nullptr, nullptr); \
1067 jni_abort_catcher.Check("java_array == null"); \
1068 env_->release_elements_fn(nullptr, nullptr, 0); \
1069 jni_abort_catcher.Check("java_array == null"); \
1070 /* Pass the elements for region as null. */ \
1071 scalar_type ## Array a = env_->new_fn(size); \
1072 env_->get_region_fn(a, 0, size, nullptr); \
1073 jni_abort_catcher.Check("buf == null"); \
1074 env_->set_region_fn(a, 0, size, nullptr); \
1075 jni_abort_catcher.Check("buf == null"); \
Ian Rogers68d8b422014-07-17 11:09:10 -07001076 down_cast<JNIEnvExt*>(env_)->SetCheckJniEnabled(true); \
Ian Rogers1d99e452014-01-02 17:36:41 -08001077 } \
Elliott Hughes814e4032011-08-23 12:07:56 -07001078 /* Allocate an array and check it has the right type and length. */ \
1079 scalar_type ## Array a = env_->new_fn(size); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001080 EXPECT_NE(a, nullptr); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001081 EXPECT_TRUE(env_->IsInstanceOf(a, env_->FindClass(expected_class_descriptor))); \
1082 EXPECT_EQ(size, env_->GetArrayLength(a)); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001083 \
1084 /* GetPrimitiveArrayRegion/SetPrimitiveArrayRegion */ \
Elliott Hughes814e4032011-08-23 12:07:56 -07001085 /* AIOOBE for negative start offset. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001086 env_->get_region_fn(a, -1, 1, nullptr); \
1087 ExpectException(aioobe_); \
1088 env_->set_region_fn(a, -1, 1, nullptr); \
1089 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001090 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001091 /* AIOOBE for negative length. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001092 env_->get_region_fn(a, 0, -1, nullptr); \
1093 ExpectException(aioobe_); \
1094 env_->set_region_fn(a, 0, -1, nullptr); \
1095 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001096 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001097 /* AIOOBE for buffer overrun. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001098 env_->get_region_fn(a, size - 1, size, nullptr); \
1099 ExpectException(aioobe_); \
1100 env_->set_region_fn(a, size - 1, size, nullptr); \
1101 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001102 \
Vladimir Marko795e3412015-11-06 16:57:03 +00001103 /* Regression test against integer overflow in range check. */ \
1104 env_->get_region_fn(a, 0x7fffffff, 0x7fffffff, nullptr); \
1105 ExpectException(aioobe_); \
1106 env_->set_region_fn(a, 0x7fffffff, 0x7fffffff, nullptr); \
1107 ExpectException(aioobe_); \
1108 \
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001109 /* It's okay for the buffer to be null as long as the length is 0. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001110 env_->get_region_fn(a, 2, 0, nullptr); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001111 /* Even if the offset is invalid... */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001112 env_->get_region_fn(a, 123, 0, nullptr); \
1113 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001114 \
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001115 /* It's okay for the buffer to be null as long as the length is 0. */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001116 env_->set_region_fn(a, 2, 0, nullptr); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001117 /* Even if the offset is invalid... */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001118 env_->set_region_fn(a, 123, 0, nullptr); \
1119 ExpectException(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001120 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001121 /* Prepare a couple of buffers. */ \
Chih-Hung Hsieh1a0de6a2016-08-26 15:06:11 -07001122 /* NOLINT, no parentheses around scalar_type. */ \
1123 std::unique_ptr<scalar_type[]> src_buf(new scalar_type[size]); /* NOLINT */ \
1124 std::unique_ptr<scalar_type[]> dst_buf(new scalar_type[size]); /* NOLINT */ \
Elliott Hughes814e4032011-08-23 12:07:56 -07001125 for (jsize i = 0; i < size; ++i) { src_buf[i] = scalar_type(i); } \
1126 for (jsize i = 0; i < size; ++i) { dst_buf[i] = scalar_type(-1); } \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001127 \
Elliott Hughes814e4032011-08-23 12:07:56 -07001128 /* Copy all of src_buf onto the heap. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -07001129 env_->set_region_fn(a, 0, size, &src_buf[0]); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001130 /* Copy back only part. */ \
1131 env_->get_region_fn(a, 1, size - 2, &dst_buf[1]); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001132 EXPECT_NE(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
1133 << "short copy equal"; \
Elliott Hughes814e4032011-08-23 12:07:56 -07001134 /* Copy the missing pieces. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -07001135 env_->get_region_fn(a, 0, 1, &dst_buf[0]); \
Elliott Hughes814e4032011-08-23 12:07:56 -07001136 env_->get_region_fn(a, size - 1, 1, &dst_buf[size - 1]); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001137 EXPECT_EQ(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
1138 << "fixed copy not equal"; \
Elliott Hughes814e4032011-08-23 12:07:56 -07001139 /* Copy back the whole array. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -07001140 env_->get_region_fn(a, 0, size, &dst_buf[0]); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001141 EXPECT_EQ(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
1142 << "full copy not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -07001143 /* GetPrimitiveArrayCritical */ \
Ian Rogers2d10b202014-05-12 19:15:18 -07001144 void* v = env_->GetPrimitiveArrayCritical(a, nullptr); \
Brian Carlstromea46f952013-07-30 01:26:50 -07001145 EXPECT_EQ(memcmp(&src_buf[0], v, size * sizeof(scalar_type)), 0) \
1146 << "GetPrimitiveArrayCritical not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -07001147 env_->ReleasePrimitiveArrayCritical(a, v, 0); \
1148 /* GetXArrayElements */ \
Chih-Hung Hsieh1a0de6a2016-08-26 15:06:11 -07001149 scalar_type* xs = env_->get_elements_fn(a, nullptr); /* NOLINT, scalar_type */ \
Brian Carlstromea46f952013-07-30 01:26:50 -07001150 EXPECT_EQ(memcmp(&src_buf[0], xs, size * sizeof(scalar_type)), 0) \
1151 << # get_elements_fn " not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -07001152 env_->release_elements_fn(a, xs, 0); \
Elliott Hughesbd935992011-08-22 11:59:34 -07001153
Elliott Hughes814e4032011-08-23 12:07:56 -07001154TEST_F(JniInternalTest, BooleanArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001155 EXPECT_PRIMITIVE_ARRAY(NewBooleanArray, GetBooleanArrayRegion, SetBooleanArrayRegion,
1156 GetBooleanArrayElements, ReleaseBooleanArrayElements, jboolean, "[Z");
Elliott Hughes814e4032011-08-23 12:07:56 -07001157}
1158TEST_F(JniInternalTest, ByteArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001159 EXPECT_PRIMITIVE_ARRAY(NewByteArray, GetByteArrayRegion, SetByteArrayRegion,
1160 GetByteArrayElements, ReleaseByteArrayElements, jbyte, "[B");
Elliott Hughes814e4032011-08-23 12:07:56 -07001161}
1162TEST_F(JniInternalTest, CharArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001163 EXPECT_PRIMITIVE_ARRAY(NewCharArray, GetCharArrayRegion, SetCharArrayRegion,
1164 GetCharArrayElements, ReleaseCharArrayElements, jchar, "[C");
Elliott Hughes814e4032011-08-23 12:07:56 -07001165}
1166TEST_F(JniInternalTest, DoubleArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001167 EXPECT_PRIMITIVE_ARRAY(NewDoubleArray, GetDoubleArrayRegion, SetDoubleArrayRegion,
1168 GetDoubleArrayElements, ReleaseDoubleArrayElements, jdouble, "[D");
Elliott Hughes814e4032011-08-23 12:07:56 -07001169}
1170TEST_F(JniInternalTest, FloatArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001171 EXPECT_PRIMITIVE_ARRAY(NewFloatArray, GetFloatArrayRegion, SetFloatArrayRegion,
1172 GetFloatArrayElements, ReleaseFloatArrayElements, jfloat, "[F");
Elliott Hughes814e4032011-08-23 12:07:56 -07001173}
1174TEST_F(JniInternalTest, IntArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001175 EXPECT_PRIMITIVE_ARRAY(NewIntArray, GetIntArrayRegion, SetIntArrayRegion,
1176 GetIntArrayElements, ReleaseIntArrayElements, jint, "[I");
Elliott Hughes814e4032011-08-23 12:07:56 -07001177}
1178TEST_F(JniInternalTest, LongArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001179 EXPECT_PRIMITIVE_ARRAY(NewLongArray, GetLongArrayRegion, SetLongArrayRegion,
1180 GetLongArrayElements, ReleaseLongArrayElements, jlong, "[J");
Elliott Hughes814e4032011-08-23 12:07:56 -07001181}
1182TEST_F(JniInternalTest, ShortArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001183 EXPECT_PRIMITIVE_ARRAY(NewShortArray, GetShortArrayRegion, SetShortArrayRegion,
1184 GetShortArrayElements, ReleaseShortArrayElements, jshort, "[S");
Elliott Hughesd8ddfd52011-08-15 14:32:53 -07001185}
1186
Ian Rogers2d10b202014-05-12 19:15:18 -07001187TEST_F(JniInternalTest, GetPrimitiveArrayElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001188 GetPrimitiveArrayElementsOfWrongType(false);
1189 GetPrimitiveArrayElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001190}
1191
1192TEST_F(JniInternalTest, ReleasePrimitiveArrayElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001193 ReleasePrimitiveArrayElementsOfWrongType(false);
1194 ReleasePrimitiveArrayElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001195}
Ian Rogers68d8b422014-07-17 11:09:10 -07001196
Ian Rogers2d10b202014-05-12 19:15:18 -07001197TEST_F(JniInternalTest, GetReleasePrimitiveArrayCriticalOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001198 GetReleasePrimitiveArrayCriticalOfWrongType(false);
1199 GetReleasePrimitiveArrayCriticalOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001200}
1201
1202TEST_F(JniInternalTest, GetPrimitiveArrayRegionElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001203 GetPrimitiveArrayRegionElementsOfWrongType(false);
1204 GetPrimitiveArrayRegionElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001205}
1206
1207TEST_F(JniInternalTest, SetPrimitiveArrayRegionElementsOfWrongType) {
Ian Rogers68d8b422014-07-17 11:09:10 -07001208 SetPrimitiveArrayRegionElementsOfWrongType(false);
1209 SetPrimitiveArrayRegionElementsOfWrongType(true);
Ian Rogers2d10b202014-05-12 19:15:18 -07001210}
1211
Elliott Hughesf2682d52011-08-15 16:37:04 -07001212TEST_F(JniInternalTest, NewObjectArray) {
Elliott Hughesbd935992011-08-22 11:59:34 -07001213 jclass element_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001214 ASSERT_NE(element_class, nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -07001215 jclass array_class = env_->FindClass("[Ljava/lang/String;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001216 ASSERT_NE(array_class, nullptr);
Elliott Hughesf2682d52011-08-15 16:37:04 -07001217
Ian Rogers1d99e452014-01-02 17:36:41 -08001218 jobjectArray a = env_->NewObjectArray(0, element_class, nullptr);
Ian Rogers2d10b202014-05-12 19:15:18 -07001219 EXPECT_NE(a, nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -07001220 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
1221 EXPECT_EQ(0, env_->GetArrayLength(a));
1222
Ian Rogers1d99e452014-01-02 17:36:41 -08001223 a = env_->NewObjectArray(1, element_class, nullptr);
Ian Rogers2d10b202014-05-12 19:15:18 -07001224 EXPECT_NE(a, nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -07001225 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
1226 EXPECT_EQ(1, env_->GetArrayLength(a));
Ian Rogers1d99e452014-01-02 17:36:41 -08001227 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 0), nullptr));
Ian Rogers1d99e452014-01-02 17:36:41 -08001228
Ian Rogers2d10b202014-05-12 19:15:18 -07001229 // Negative array length checks.
Ian Rogers68d8b422014-07-17 11:09:10 -07001230 NewObjectArrayBadArguments(false);
1231 NewObjectArrayBadArguments(true);
Ian Rogers1d99e452014-01-02 17:36:41 -08001232}
1233
1234TEST_F(JniInternalTest, NewObjectArrayWithPrimitiveClasses) {
1235 const char* primitive_descriptors = "VZBSCIJFD";
1236 const char* primitive_names[] = {
1237 "void", "boolean", "byte", "short", "char", "int", "long", "float", "double"
1238 };
1239 ASSERT_EQ(strlen(primitive_descriptors), arraysize(primitive_names));
1240
Ian Rogers68d8b422014-07-17 11:09:10 -07001241 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers1d99e452014-01-02 17:36:41 -08001242 CheckJniAbortCatcher jni_abort_catcher;
1243 for (size_t i = 0; i < strlen(primitive_descriptors); ++i) {
Ian Rogers2d10b202014-05-12 19:15:18 -07001244 env_->NewObjectArray(0, nullptr, nullptr);
1245 jni_abort_catcher.Check("element_jclass == null");
Ian Rogers1d99e452014-01-02 17:36:41 -08001246 jclass primitive_class = GetPrimitiveClass(primitive_descriptors[i]);
1247 env_->NewObjectArray(1, primitive_class, nullptr);
1248 std::string error_msg(StringPrintf("not an object type: %s", primitive_names[i]));
1249 jni_abort_catcher.Check(error_msg.c_str());
1250 }
Ian Rogers68d8b422014-07-17 11:09:10 -07001251 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1252 for (size_t i = 0; i < strlen(primitive_descriptors); ++i) {
1253 env_->NewObjectArray(0, nullptr, nullptr);
1254 jni_abort_catcher.Check("NewObjectArray received NULL jclass");
1255 jclass primitive_class = GetPrimitiveClass(primitive_descriptors[i]);
1256 env_->NewObjectArray(1, primitive_class, nullptr);
1257 std::string error_msg(StringPrintf("not an object type: %s", primitive_names[i]));
1258 jni_abort_catcher.Check(error_msg.c_str());
1259 }
1260 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Ian Rogers1d99e452014-01-02 17:36:41 -08001261}
1262
1263TEST_F(JniInternalTest, NewObjectArrayWithInitialValue) {
1264 jclass element_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001265 ASSERT_NE(element_class, nullptr);
Ian Rogers1d99e452014-01-02 17:36:41 -08001266 jclass array_class = env_->FindClass("[Ljava/lang/String;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001267 ASSERT_NE(array_class, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001268
1269 jstring s = env_->NewStringUTF("poop");
Ian Rogers1d99e452014-01-02 17:36:41 -08001270 jobjectArray a = env_->NewObjectArray(2, element_class, s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001271 EXPECT_NE(a, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001272 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
1273 EXPECT_EQ(2, env_->GetArrayLength(a));
1274 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 0), s));
1275 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 1), s));
Ian Rogers1d99e452014-01-02 17:36:41 -08001276
1277 // Attempt to incorrect create an array of strings with initial value of string arrays.
1278 CheckJniAbortCatcher jni_abort_catcher;
1279 env_->NewObjectArray(2, element_class, a);
1280 jni_abort_catcher.Check("cannot assign object of type 'java.lang.String[]' to array with element "
1281 "type of 'java.lang.String'");
Elliott Hughesbd935992011-08-22 11:59:34 -07001282}
1283
1284TEST_F(JniInternalTest, GetArrayLength) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001285 // Already tested in NewObjectArray/NewPrimitiveArray except for null.
Ian Rogers2f022bd2014-11-11 08:43:05 -08001286 CheckJniAbortCatcher jni_abort_catcher;
1287 bool old_check_jni = vm_->SetCheckJniEnabled(false);
1288 EXPECT_EQ(0, env_->GetArrayLength(nullptr));
1289 jni_abort_catcher.Check("java_array == null");
1290 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1291 EXPECT_EQ(JNI_ERR, env_->GetArrayLength(nullptr));
1292 jni_abort_catcher.Check("jarray was NULL");
1293 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes8a26c5c2011-08-15 18:35:43 -07001294}
1295
Elliott Hughes37f7a402011-08-22 18:56:01 -07001296TEST_F(JniInternalTest, GetObjectClass) {
1297 jclass string_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001298 ASSERT_NE(string_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001299 jclass class_class = env_->FindClass("java/lang/Class");
Ian Rogers2d10b202014-05-12 19:15:18 -07001300 ASSERT_NE(class_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001301
1302 jstring s = env_->NewStringUTF("poop");
1303 jclass c = env_->GetObjectClass(s);
1304 ASSERT_TRUE(env_->IsSameObject(string_class, c));
1305
1306 jclass c2 = env_->GetObjectClass(c);
1307 ASSERT_TRUE(env_->IsSameObject(class_class, env_->GetObjectClass(c2)));
Ian Rogers2d10b202014-05-12 19:15:18 -07001308
1309 // Null as object should fail.
1310 CheckJniAbortCatcher jni_abort_catcher;
1311 EXPECT_EQ(env_->GetObjectClass(nullptr), nullptr);
1312 jni_abort_catcher.Check("java_object == null");
Elliott Hughes37f7a402011-08-22 18:56:01 -07001313}
1314
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001315TEST_F(JniInternalTest, GetSuperclass) {
1316 jclass object_class = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07001317 ASSERT_NE(object_class, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001318 jclass string_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001319 ASSERT_NE(string_class, nullptr);
Ian Rogersdc180202012-01-29 14:47:29 -08001320 jclass runnable_interface = env_->FindClass("java/lang/Runnable");
Ian Rogers2d10b202014-05-12 19:15:18 -07001321 ASSERT_NE(runnable_interface, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001322 ASSERT_TRUE(env_->IsSameObject(object_class, env_->GetSuperclass(string_class)));
Ian Rogers2d10b202014-05-12 19:15:18 -07001323 ASSERT_EQ(env_->GetSuperclass(object_class), nullptr);
Brian Carlstrom08ac9222015-05-22 13:43:00 -07001324 ASSERT_EQ(env_->GetSuperclass(runnable_interface), nullptr);
Ian Rogers2d10b202014-05-12 19:15:18 -07001325
1326 // Null as class should fail.
1327 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001328 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers2d10b202014-05-12 19:15:18 -07001329 EXPECT_EQ(env_->GetSuperclass(nullptr), nullptr);
1330 jni_abort_catcher.Check("java_class == null");
Ian Rogers68d8b422014-07-17 11:09:10 -07001331 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1332 EXPECT_EQ(env_->GetSuperclass(nullptr), nullptr);
1333 jni_abort_catcher.Check("GetSuperclass received NULL jclass");
1334 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001335}
1336
Elliott Hughes37f7a402011-08-22 18:56:01 -07001337TEST_F(JniInternalTest, IsAssignableFrom) {
1338 jclass object_class = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07001339 ASSERT_NE(object_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001340 jclass string_class = env_->FindClass("java/lang/String");
Ian Rogers2d10b202014-05-12 19:15:18 -07001341 ASSERT_NE(string_class, nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07001342
Narayan Kamath1268b742014-07-11 19:15:11 +01001343 // A superclass is assignable from an instance of its
1344 // subclass but not vice versa.
1345 ASSERT_TRUE(env_->IsAssignableFrom(string_class, object_class));
1346 ASSERT_FALSE(env_->IsAssignableFrom(object_class, string_class));
1347
1348 jclass charsequence_interface = env_->FindClass("java/lang/CharSequence");
1349 ASSERT_NE(charsequence_interface, nullptr);
1350
1351 // An interface is assignable from an instance of an implementing
1352 // class but not vice versa.
1353 ASSERT_TRUE(env_->IsAssignableFrom(string_class, charsequence_interface));
1354 ASSERT_FALSE(env_->IsAssignableFrom(charsequence_interface, string_class));
1355
1356 // Check that arrays are covariant.
1357 jclass string_array_class = env_->FindClass("[Ljava/lang/String;");
1358 ASSERT_NE(string_array_class, nullptr);
1359 jclass object_array_class = env_->FindClass("[Ljava/lang/Object;");
1360 ASSERT_NE(object_array_class, nullptr);
1361 ASSERT_TRUE(env_->IsAssignableFrom(string_array_class, object_array_class));
1362 ASSERT_FALSE(env_->IsAssignableFrom(object_array_class, string_array_class));
1363
1364 // Primitive types are tested in 004-JniTest.
Ian Rogers2d10b202014-05-12 19:15:18 -07001365
1366 // Null as either class should fail.
1367 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001368 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers2d10b202014-05-12 19:15:18 -07001369 EXPECT_EQ(env_->IsAssignableFrom(nullptr, string_class), JNI_FALSE);
1370 jni_abort_catcher.Check("java_class1 == null");
1371 EXPECT_EQ(env_->IsAssignableFrom(object_class, nullptr), JNI_FALSE);
1372 jni_abort_catcher.Check("java_class2 == null");
Ian Rogers68d8b422014-07-17 11:09:10 -07001373 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1374 EXPECT_EQ(env_->IsAssignableFrom(nullptr, string_class), JNI_FALSE);
1375 jni_abort_catcher.Check("IsAssignableFrom received NULL jclass");
1376 EXPECT_EQ(env_->IsAssignableFrom(object_class, nullptr), JNI_FALSE);
1377 jni_abort_catcher.Check("IsAssignableFrom received NULL jclass");
1378 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes37f7a402011-08-22 18:56:01 -07001379}
1380
Elliott Hughesb465ab02011-08-24 11:21:21 -07001381TEST_F(JniInternalTest, GetObjectRefType) {
1382 jclass local = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07001383 ASSERT_TRUE(local != nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001384 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(local));
1385
1386 jobject global = env_->NewGlobalRef(local);
1387 EXPECT_EQ(JNIGlobalRefType, env_->GetObjectRefType(global));
1388
1389 jweak weak_global = env_->NewWeakGlobalRef(local);
1390 EXPECT_EQ(JNIWeakGlobalRefType, env_->GetObjectRefType(weak_global));
1391
Andreas Gampea8763072014-12-20 00:08:35 -08001392 {
1393 CheckJniAbortCatcher jni_abort_catcher;
1394 jobject invalid = reinterpret_cast<jobject>(this);
1395 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(invalid));
1396 jni_abort_catcher.Check("use of invalid jobject");
1397 }
Elliott Hughesb465ab02011-08-24 11:21:21 -07001398
1399 // TODO: invoke a native method and test that its arguments are considered local references.
Ian Rogers2d10b202014-05-12 19:15:18 -07001400
Andreas Gampea8763072014-12-20 00:08:35 -08001401 // Null as pointer should not fail and return invalid-ref. b/18820997
Ian Rogers2d10b202014-05-12 19:15:18 -07001402 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(nullptr));
Andreas Gampea8763072014-12-20 00:08:35 -08001403
1404 // TODO: Null as reference should return the original type.
1405 // This requires running a GC so a non-null object gets freed.
Elliott Hughesb465ab02011-08-24 11:21:21 -07001406}
1407
Mathieu Chartier08599992013-12-20 17:17:55 -08001408TEST_F(JniInternalTest, StaleWeakGlobal) {
1409 jclass java_lang_Class = env_->FindClass("java/lang/Class");
Ian Rogers2d10b202014-05-12 19:15:18 -07001410 ASSERT_NE(java_lang_Class, nullptr);
1411 jobjectArray local_ref = env_->NewObjectArray(1, java_lang_Class, nullptr);
1412 ASSERT_NE(local_ref, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001413 jweak weak_global = env_->NewWeakGlobalRef(local_ref);
Ian Rogers2d10b202014-05-12 19:15:18 -07001414 ASSERT_NE(weak_global, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001415 env_->DeleteLocalRef(local_ref);
1416 Runtime::Current()->GetHeap()->CollectGarbage(false); // GC should clear the weak global.
1417 jobject new_global_ref = env_->NewGlobalRef(weak_global);
Ian Rogers2d10b202014-05-12 19:15:18 -07001418 EXPECT_EQ(new_global_ref, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001419 jobject new_local_ref = env_->NewLocalRef(weak_global);
Ian Rogers2d10b202014-05-12 19:15:18 -07001420 EXPECT_EQ(new_local_ref, nullptr);
Mathieu Chartier08599992013-12-20 17:17:55 -08001421}
1422
Elliott Hughes8a26c5c2011-08-15 18:35:43 -07001423TEST_F(JniInternalTest, NewStringUTF) {
Ian Rogers2d10b202014-05-12 19:15:18 -07001424 EXPECT_EQ(env_->NewStringUTF(nullptr), nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001425 jstring s;
1426
1427 s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001428 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001429 EXPECT_EQ(0, env_->GetStringLength(s));
1430 EXPECT_EQ(0, env_->GetStringUTFLength(s));
1431 s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001432 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001433 EXPECT_EQ(5, env_->GetStringLength(s));
1434 EXPECT_EQ(5, env_->GetStringUTFLength(s));
1435
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001436 // Encoded surrogate pair.
1437 s = env_->NewStringUTF("\xed\xa0\x81\xed\xb0\x80");
1438 EXPECT_NE(s, nullptr);
1439 EXPECT_EQ(2, env_->GetStringLength(s));
Narayan Kamathe16dad12015-02-13 11:49:22 +00001440
1441 // The surrogate pair gets encoded into a 4 byte UTF sequence..
1442 EXPECT_EQ(4, env_->GetStringUTFLength(s));
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001443 const char* chars = env_->GetStringUTFChars(s, nullptr);
Narayan Kamathe16dad12015-02-13 11:49:22 +00001444 EXPECT_STREQ("\xf0\x90\x90\x80", chars);
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001445 env_->ReleaseStringUTFChars(s, chars);
1446
Narayan Kamathe16dad12015-02-13 11:49:22 +00001447 // .. but is stored as is in the utf-16 representation.
1448 const jchar* jchars = env_->GetStringChars(s, nullptr);
1449 EXPECT_EQ(0xd801, jchars[0]);
1450 EXPECT_EQ(0xdc00, jchars[1]);
1451 env_->ReleaseStringChars(s, jchars);
1452
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001453 // 4 byte UTF sequence appended to an encoded surrogate pair.
1454 s = env_->NewStringUTF("\xed\xa0\x81\xed\xb0\x80 \xf0\x9f\x8f\xa0");
1455 EXPECT_NE(s, nullptr);
Narayan Kamathe16dad12015-02-13 11:49:22 +00001456
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001457 // The 4 byte sequence {0xf0, 0x9f, 0x8f, 0xa0} is converted into a surrogate
Narayan Kamathe16dad12015-02-13 11:49:22 +00001458 // pair {0xd83c, 0xdfe0}.
1459 EXPECT_EQ(5, env_->GetStringLength(s));
1460 jchars = env_->GetStringChars(s, nullptr);
1461 // The first surrogate pair, encoded as such in the input.
1462 EXPECT_EQ(0xd801, jchars[0]);
1463 EXPECT_EQ(0xdc00, jchars[1]);
1464 // The second surrogate pair, from the 4 byte UTF sequence in the input.
1465 EXPECT_EQ(0xd83c, jchars[3]);
1466 EXPECT_EQ(0xdfe0, jchars[4]);
1467 env_->ReleaseStringChars(s, jchars);
1468
1469 EXPECT_EQ(9, env_->GetStringUTFLength(s));
1470 chars = env_->GetStringUTFChars(s, nullptr);
1471 EXPECT_STREQ("\xf0\x90\x90\x80 \xf0\x9f\x8f\xa0", chars);
Narayan Kamatha5afcfc2015-01-29 20:06:46 +00001472 env_->ReleaseStringUTFChars(s, chars);
1473
1474 // A string with 1, 2, 3 and 4 byte UTF sequences with spaces
1475 // between them
1476 s = env_->NewStringUTF("\x24 \xc2\xa2 \xe2\x82\xac \xf0\x9f\x8f\xa0");
1477 EXPECT_NE(s, nullptr);
1478 EXPECT_EQ(8, env_->GetStringLength(s));
Narayan Kamathe16dad12015-02-13 11:49:22 +00001479 EXPECT_EQ(13, env_->GetStringUTFLength(s));
Elliott Hughesf2682d52011-08-15 16:37:04 -07001480}
1481
Elliott Hughes814e4032011-08-23 12:07:56 -07001482TEST_F(JniInternalTest, NewString) {
Elliott Hughes814e4032011-08-23 12:07:56 -07001483 jchar chars[] = { 'h', 'i' };
1484 jstring s;
1485 s = env_->NewString(chars, 0);
Ian Rogers2d10b202014-05-12 19:15:18 -07001486 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001487 EXPECT_EQ(0, env_->GetStringLength(s));
1488 EXPECT_EQ(0, env_->GetStringUTFLength(s));
1489 s = env_->NewString(chars, 2);
Ian Rogers2d10b202014-05-12 19:15:18 -07001490 EXPECT_NE(s, nullptr);
Elliott Hughes814e4032011-08-23 12:07:56 -07001491 EXPECT_EQ(2, env_->GetStringLength(s));
1492 EXPECT_EQ(2, env_->GetStringUTFLength(s));
1493
1494 // TODO: check some non-ASCII strings.
1495}
1496
Jesse Wilson25e79a52011-11-18 15:31:58 -05001497TEST_F(JniInternalTest, NewStringNullCharsZeroLength) {
Ian Rogers1d99e452014-01-02 17:36:41 -08001498 jstring s = env_->NewString(nullptr, 0);
Ian Rogers2d10b202014-05-12 19:15:18 -07001499 EXPECT_NE(s, nullptr);
Jesse Wilson25e79a52011-11-18 15:31:58 -05001500 EXPECT_EQ(0, env_->GetStringLength(s));
1501}
1502
Ian Rogers1d99e452014-01-02 17:36:41 -08001503TEST_F(JniInternalTest, NewStringNullCharsNonzeroLength) {
1504 CheckJniAbortCatcher jni_abort_catcher;
1505 env_->NewString(nullptr, 1);
1506 jni_abort_catcher.Check("chars == null && char_count > 0");
1507}
1508
1509TEST_F(JniInternalTest, NewStringNegativeLength) {
1510 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001511 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers1d99e452014-01-02 17:36:41 -08001512 env_->NewString(nullptr, -1);
1513 jni_abort_catcher.Check("char_count < 0: -1");
1514 env_->NewString(nullptr, std::numeric_limits<jint>::min());
1515 jni_abort_catcher.Check("char_count < 0: -2147483648");
Ian Rogers68d8b422014-07-17 11:09:10 -07001516 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1517 env_->NewString(nullptr, -1);
1518 jni_abort_catcher.Check("negative jsize: -1");
1519 env_->NewString(nullptr, std::numeric_limits<jint>::min());
1520 jni_abort_catcher.Check("negative jsize: -2147483648");
1521 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Jesse Wilson25e79a52011-11-18 15:31:58 -05001522}
1523
Elliott Hughesb465ab02011-08-24 11:21:21 -07001524TEST_F(JniInternalTest, GetStringLength_GetStringUTFLength) {
1525 // Already tested in the NewString/NewStringUTF tests.
1526}
1527
1528TEST_F(JniInternalTest, GetStringRegion_GetStringUTFRegion) {
1529 jstring s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001530 ASSERT_TRUE(s != nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001531
Ian Rogers2d10b202014-05-12 19:15:18 -07001532 env_->GetStringRegion(s, -1, 0, nullptr);
1533 ExpectException(sioobe_);
1534 env_->GetStringRegion(s, 0, -1, nullptr);
1535 ExpectException(sioobe_);
1536 env_->GetStringRegion(s, 0, 10, nullptr);
1537 ExpectException(sioobe_);
1538 env_->GetStringRegion(s, 10, 1, nullptr);
1539 ExpectException(sioobe_);
Vladimir Marko795e3412015-11-06 16:57:03 +00001540 // Regression test against integer overflow in range check.
1541 env_->GetStringRegion(s, 0x7fffffff, 0x7fffffff, nullptr);
1542 ExpectException(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001543
1544 jchar chars[4] = { 'x', 'x', 'x', 'x' };
1545 env_->GetStringRegion(s, 1, 2, &chars[1]);
1546 EXPECT_EQ('x', chars[0]);
1547 EXPECT_EQ('e', chars[1]);
1548 EXPECT_EQ('l', chars[2]);
1549 EXPECT_EQ('x', chars[3]);
1550
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001551 // It's okay for the buffer to be null as long as the length is 0.
Ian Rogers2d10b202014-05-12 19:15:18 -07001552 env_->GetStringRegion(s, 2, 0, nullptr);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001553 // Even if the offset is invalid...
Ian Rogers2d10b202014-05-12 19:15:18 -07001554 env_->GetStringRegion(s, 123, 0, nullptr);
1555 ExpectException(sioobe_);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001556
Ian Rogers2d10b202014-05-12 19:15:18 -07001557 env_->GetStringUTFRegion(s, -1, 0, nullptr);
1558 ExpectException(sioobe_);
1559 env_->GetStringUTFRegion(s, 0, -1, nullptr);
1560 ExpectException(sioobe_);
1561 env_->GetStringUTFRegion(s, 0, 10, nullptr);
1562 ExpectException(sioobe_);
1563 env_->GetStringUTFRegion(s, 10, 1, nullptr);
1564 ExpectException(sioobe_);
Vladimir Marko795e3412015-11-06 16:57:03 +00001565 // Regression test against integer overflow in range check.
1566 env_->GetStringUTFRegion(s, 0x7fffffff, 0x7fffffff, nullptr);
1567 ExpectException(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001568
1569 char bytes[4] = { 'x', 'x', 'x', 'x' };
1570 env_->GetStringUTFRegion(s, 1, 2, &bytes[1]);
1571 EXPECT_EQ('x', bytes[0]);
1572 EXPECT_EQ('e', bytes[1]);
1573 EXPECT_EQ('l', bytes[2]);
1574 EXPECT_EQ('x', bytes[3]);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001575
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001576 // It's okay for the buffer to be null as long as the length is 0.
Ian Rogers2d10b202014-05-12 19:15:18 -07001577 env_->GetStringUTFRegion(s, 2, 0, nullptr);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -07001578 // Even if the offset is invalid...
Ian Rogers2d10b202014-05-12 19:15:18 -07001579 env_->GetStringUTFRegion(s, 123, 0, nullptr);
1580 ExpectException(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -07001581}
1582
Elliott Hughes75770752011-08-24 17:52:38 -07001583TEST_F(JniInternalTest, GetStringUTFChars_ReleaseStringUTFChars) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -07001584 // Passing in a null jstring is ignored normally, but caught by -Xcheck:jni.
Ian Rogers68d8b422014-07-17 11:09:10 -07001585 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Elliott Hughesb264f082012-04-06 17:10:10 -07001586 {
Elliott Hughesb264f082012-04-06 17:10:10 -07001587 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers2d10b202014-05-12 19:15:18 -07001588 EXPECT_EQ(env_->GetStringUTFChars(nullptr, nullptr), nullptr);
Ian Rogers68d8b422014-07-17 11:09:10 -07001589 }
1590 {
1591 CheckJniAbortCatcher check_jni_abort_catcher;
1592 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1593 EXPECT_EQ(env_->GetStringUTFChars(nullptr, nullptr), nullptr);
1594 check_jni_abort_catcher.Check("GetStringUTFChars received NULL jstring");
1595 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07001596 }
Elliott Hughes75770752011-08-24 17:52:38 -07001597
1598 jstring s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001599 ASSERT_TRUE(s != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001600
Ian Rogers2d10b202014-05-12 19:15:18 -07001601 const char* utf = env_->GetStringUTFChars(s, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001602 EXPECT_STREQ("hello", utf);
1603 env_->ReleaseStringUTFChars(s, utf);
1604
1605 jboolean is_copy = JNI_FALSE;
1606 utf = env_->GetStringUTFChars(s, &is_copy);
1607 EXPECT_EQ(JNI_TRUE, is_copy);
1608 EXPECT_STREQ("hello", utf);
1609 env_->ReleaseStringUTFChars(s, utf);
1610}
1611
1612TEST_F(JniInternalTest, GetStringChars_ReleaseStringChars) {
1613 jstring s = env_->NewStringUTF("hello");
Fred Shih56890e22014-06-02 11:11:52 -07001614 ScopedObjectAccess soa(env_);
Mathieu Chartier0795f232016-09-27 18:43:30 -07001615 ObjPtr<mirror::String> s_m = soa.Decode<mirror::String>(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001616 ASSERT_TRUE(s != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001617
1618 jchar expected[] = { 'h', 'e', 'l', 'l', 'o' };
Ian Rogers2d10b202014-05-12 19:15:18 -07001619 const jchar* chars = env_->GetStringChars(s, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001620 EXPECT_EQ(expected[0], chars[0]);
1621 EXPECT_EQ(expected[1], chars[1]);
1622 EXPECT_EQ(expected[2], chars[2]);
1623 EXPECT_EQ(expected[3], chars[3]);
1624 EXPECT_EQ(expected[4], chars[4]);
1625 env_->ReleaseStringChars(s, chars);
1626
1627 jboolean is_copy = JNI_FALSE;
1628 chars = env_->GetStringChars(s, &is_copy);
Jeff Hao848f70a2014-01-15 13:49:50 -08001629 if (Runtime::Current()->GetHeap()->IsMovableObject(s_m)) {
Fred Shih56890e22014-06-02 11:11:52 -07001630 EXPECT_EQ(JNI_TRUE, is_copy);
1631 } else {
1632 EXPECT_EQ(JNI_FALSE, is_copy);
1633 }
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_->ReleaseStringChars(s, chars);
1640}
1641
1642TEST_F(JniInternalTest, GetStringCritical_ReleaseStringCritical) {
1643 jstring s = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001644 ASSERT_TRUE(s != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001645
1646 jchar expected[] = { 'h', 'e', 'l', 'l', 'o' };
Ian Rogers2d10b202014-05-12 19:15:18 -07001647 const jchar* chars = env_->GetStringCritical(s, nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -07001648 EXPECT_EQ(expected[0], chars[0]);
1649 EXPECT_EQ(expected[1], chars[1]);
1650 EXPECT_EQ(expected[2], chars[2]);
1651 EXPECT_EQ(expected[3], chars[3]);
1652 EXPECT_EQ(expected[4], chars[4]);
1653 env_->ReleaseStringCritical(s, chars);
1654
Fred Shih56890e22014-06-02 11:11:52 -07001655 jboolean is_copy = JNI_TRUE;
Elliott Hughes75770752011-08-24 17:52:38 -07001656 chars = env_->GetStringCritical(s, &is_copy);
jessicahandojo3aaa37b2016-07-29 14:46:37 -07001657 if (mirror::kUseStringCompression) {
1658 // is_copy has to be JNI_TRUE because "hello" is all-ASCII
1659 EXPECT_EQ(JNI_TRUE, is_copy);
1660 } else {
1661 EXPECT_EQ(JNI_FALSE, is_copy);
1662 }
Elliott Hughes75770752011-08-24 17:52:38 -07001663 EXPECT_EQ(expected[0], chars[0]);
1664 EXPECT_EQ(expected[1], chars[1]);
1665 EXPECT_EQ(expected[2], chars[2]);
1666 EXPECT_EQ(expected[3], chars[3]);
1667 EXPECT_EQ(expected[4], chars[4]);
1668 env_->ReleaseStringCritical(s, chars);
jessicahandojo3aaa37b2016-07-29 14:46:37 -07001669
1670 if (mirror::kUseStringCompression) {
1671 // is_copy has to be JNI_FALSE because "\xed\xa0\x81\xed\xb0\x80" is incompressible
1672 jboolean is_copy_16 = JNI_TRUE;
1673 jstring s_16 = env_->NewStringUTF("\xed\xa0\x81\xed\xb0\x80");
1674 chars = env_->GetStringCritical(s_16, &is_copy_16);
1675 EXPECT_EQ(2, env_->GetStringLength(s_16));
1676 EXPECT_EQ(4, env_->GetStringUTFLength(s_16));
1677 env_->ReleaseStringCritical(s_16, chars);
1678 }
Elliott Hughes75770752011-08-24 17:52:38 -07001679}
1680
Elliott Hughes814e4032011-08-23 12:07:56 -07001681TEST_F(JniInternalTest, GetObjectArrayElement_SetObjectArrayElement) {
Elliott Hughesb264f082012-04-06 17:10:10 -07001682 jclass java_lang_Class = env_->FindClass("java/lang/Class");
Ian Rogers2d10b202014-05-12 19:15:18 -07001683 ASSERT_TRUE(java_lang_Class != nullptr);
Elliott Hughes289da822011-08-16 10:11:20 -07001684
Ian Rogers2d10b202014-05-12 19:15:18 -07001685 jobjectArray array = env_->NewObjectArray(1, java_lang_Class, nullptr);
1686 EXPECT_NE(array, nullptr);
1687 EXPECT_EQ(env_->GetObjectArrayElement(array, 0), nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -07001688 env_->SetObjectArrayElement(array, 0, java_lang_Class);
1689 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(array, 0), java_lang_Class));
Elliott Hughesa5b897e2011-08-16 11:33:06 -07001690
1691 // ArrayIndexOutOfBounds for negative index.
Elliott Hughesb264f082012-04-06 17:10:10 -07001692 env_->SetObjectArrayElement(array, -1, java_lang_Class);
Ian Rogers2d10b202014-05-12 19:15:18 -07001693 ExpectException(aioobe_);
Elliott Hughesa5b897e2011-08-16 11:33:06 -07001694
1695 // ArrayIndexOutOfBounds for too-large index.
Elliott Hughesb264f082012-04-06 17:10:10 -07001696 env_->SetObjectArrayElement(array, 1, java_lang_Class);
Ian Rogers2d10b202014-05-12 19:15:18 -07001697 ExpectException(aioobe_);
Elliott Hughesa5b897e2011-08-16 11:33:06 -07001698
Elliott Hughesb264f082012-04-06 17:10:10 -07001699 // ArrayStoreException thrown for bad types.
1700 env_->SetObjectArrayElement(array, 0, env_->NewStringUTF("not a jclass!"));
Ian Rogers2d10b202014-05-12 19:15:18 -07001701 ExpectException(ase_);
1702
1703 // Null as array should fail.
1704 CheckJniAbortCatcher jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001705 bool old_check_jni = vm_->SetCheckJniEnabled(false);
Ian Rogers2d10b202014-05-12 19:15:18 -07001706 EXPECT_EQ(nullptr, env_->GetObjectArrayElement(nullptr, 0));
1707 jni_abort_catcher.Check("java_array == null");
1708 env_->SetObjectArrayElement(nullptr, 0, nullptr);
1709 jni_abort_catcher.Check("java_array == null");
Ian Rogers68d8b422014-07-17 11:09:10 -07001710 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
1711 EXPECT_EQ(nullptr, env_->GetObjectArrayElement(nullptr, 0));
1712 jni_abort_catcher.Check("jarray was NULL");
1713 env_->SetObjectArrayElement(nullptr, 0, nullptr);
1714 jni_abort_catcher.Check("jarray was NULL");
1715 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes289da822011-08-16 10:11:20 -07001716}
1717
Ian Rogers647b1a82014-10-10 11:02:11 -07001718#define EXPECT_STATIC_PRIMITIVE_FIELD(expect_eq, type, field_name, sig, value1, value2) \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001719 do { \
1720 jfieldID fid = env_->GetStaticFieldID(c, field_name, sig); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001721 EXPECT_NE(fid, nullptr); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001722 env_->SetStatic ## type ## Field(c, fid, value1); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001723 expect_eq(value1, env_->GetStatic ## type ## Field(c, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001724 env_->SetStatic ## type ## Field(c, fid, value2); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001725 expect_eq(value2, env_->GetStatic ## type ## Field(c, fid)); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001726 \
Ian Rogers68d8b422014-07-17 11:09:10 -07001727 bool old_check_jni = vm_->SetCheckJniEnabled(false); \
1728 { \
1729 CheckJniAbortCatcher jni_abort_catcher; \
1730 env_->GetStatic ## type ## Field(nullptr, fid); \
1731 env_->SetStatic ## type ## Field(nullptr, fid, value1); \
1732 } \
Ian Rogers2d10b202014-05-12 19:15:18 -07001733 CheckJniAbortCatcher jni_abort_catcher; \
Ian Rogers2d10b202014-05-12 19:15:18 -07001734 env_->GetStatic ## type ## Field(c, nullptr); \
1735 jni_abort_catcher.Check("fid == null"); \
1736 env_->SetStatic ## type ## Field(c, nullptr, value1); \
1737 jni_abort_catcher.Check("fid == null"); \
Ian Rogers68d8b422014-07-17 11:09:10 -07001738 \
1739 EXPECT_FALSE(vm_->SetCheckJniEnabled(true)); \
1740 env_->GetStatic ## type ## Field(nullptr, fid); \
1741 jni_abort_catcher.Check("received NULL jclass"); \
1742 env_->SetStatic ## type ## Field(nullptr, fid, value1); \
1743 jni_abort_catcher.Check("received NULL jclass"); \
1744 env_->GetStatic ## type ## Field(c, nullptr); \
1745 jni_abort_catcher.Check("jfieldID was NULL"); \
1746 env_->SetStatic ## type ## Field(c, nullptr, value1); \
1747 jni_abort_catcher.Check("jfieldID was NULL"); \
1748 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001749 } while (false)
1750
Ian Rogers647b1a82014-10-10 11:02:11 -07001751#define EXPECT_PRIMITIVE_FIELD(expect_eq, instance, type, field_name, sig, value1, value2) \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001752 do { \
1753 jfieldID fid = env_->GetFieldID(c, field_name, sig); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001754 EXPECT_NE(fid, nullptr); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001755 env_->Set ## type ## Field(instance, fid, value1); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001756 expect_eq(value1, env_->Get ## type ## Field(instance, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001757 env_->Set ## type ## Field(instance, fid, value2); \
Ian Rogers647b1a82014-10-10 11:02:11 -07001758 expect_eq(value2, env_->Get ## type ## Field(instance, fid)); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001759 \
Ian Rogers68d8b422014-07-17 11:09:10 -07001760 bool old_check_jni = vm_->SetCheckJniEnabled(false); \
Ian Rogers2d10b202014-05-12 19:15:18 -07001761 CheckJniAbortCatcher jni_abort_catcher; \
1762 env_->Get ## type ## Field(nullptr, fid); \
1763 jni_abort_catcher.Check("obj == null"); \
1764 env_->Set ## type ## Field(nullptr, fid, value1); \
1765 jni_abort_catcher.Check("obj == null"); \
1766 env_->Get ## type ## Field(instance, nullptr); \
1767 jni_abort_catcher.Check("fid == null"); \
1768 env_->Set ## type ## Field(instance, nullptr, value1); \
1769 jni_abort_catcher.Check("fid == null"); \
Ian Rogers68d8b422014-07-17 11:09:10 -07001770 EXPECT_FALSE(vm_->SetCheckJniEnabled(true)); \
1771 env_->Get ## type ## Field(nullptr, fid); \
1772 jni_abort_catcher.Check("field operation on NULL object:"); \
1773 env_->Set ## type ## Field(nullptr, fid, value1); \
1774 jni_abort_catcher.Check("field operation on NULL object:"); \
1775 env_->Get ## type ## Field(instance, nullptr); \
1776 jni_abort_catcher.Check("jfieldID was NULL"); \
1777 env_->Set ## type ## Field(instance, nullptr, value1); \
1778 jni_abort_catcher.Check("jfieldID was NULL"); \
1779 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001780 } while (false)
1781
1782
1783TEST_F(JniInternalTest, GetPrimitiveField_SetPrimitiveField) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001784 Thread::Current()->TransitionFromSuspendedToRunnable();
1785 LoadDex("AllFields");
Brian Carlstrombd86bcc2013-03-10 20:26:16 -07001786 bool started = runtime_->Start();
Ian Rogers2d10b202014-05-12 19:15:18 -07001787 ASSERT_TRUE(started);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001788
1789 jclass c = env_->FindClass("AllFields");
Ian Rogers2d10b202014-05-12 19:15:18 -07001790 ASSERT_NE(c, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001791 jobject o = env_->AllocObject(c);
Ian Rogers2d10b202014-05-12 19:15:18 -07001792 ASSERT_NE(o, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001793
Ian Rogers647b1a82014-10-10 11:02:11 -07001794 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Boolean, "sZ", "Z", JNI_TRUE, JNI_FALSE);
1795 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Byte, "sB", "B", 1, 2);
1796 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Char, "sC", "C", 'a', 'b');
1797 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_DOUBLE_EQ, Double, "sD", "D", 1.0, 2.0);
1798 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_FLOAT_EQ, Float, "sF", "F", 1.0, 2.0);
1799 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Int, "sI", "I", 1, 2);
1800 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Long, "sJ", "J", 1, 2);
1801 EXPECT_STATIC_PRIMITIVE_FIELD(EXPECT_EQ, Short, "sS", "S", 1, 2);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001802
Ian Rogers647b1a82014-10-10 11:02:11 -07001803 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Boolean, "iZ", "Z", JNI_TRUE, JNI_FALSE);
1804 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Byte, "iB", "B", 1, 2);
1805 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Char, "iC", "C", 'a', 'b');
1806 EXPECT_PRIMITIVE_FIELD(EXPECT_DOUBLE_EQ, o, Double, "iD", "D", 1.0, 2.0);
1807 EXPECT_PRIMITIVE_FIELD(EXPECT_FLOAT_EQ, o, Float, "iF", "F", 1.0, 2.0);
1808 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Int, "iI", "I", 1, 2);
1809 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Long, "iJ", "J", 1, 2);
1810 EXPECT_PRIMITIVE_FIELD(EXPECT_EQ, o, Short, "iS", "S", 1, 2);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001811}
1812
1813TEST_F(JniInternalTest, GetObjectField_SetObjectField) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001814 Thread::Current()->TransitionFromSuspendedToRunnable();
1815 LoadDex("AllFields");
Brian Carlstrom25c33252011-09-18 15:58:35 -07001816 runtime_->Start();
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001817
1818 jclass c = env_->FindClass("AllFields");
Ian Rogers2d10b202014-05-12 19:15:18 -07001819 ASSERT_NE(c, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001820 jobject o = env_->AllocObject(c);
Ian Rogers2d10b202014-05-12 19:15:18 -07001821 ASSERT_NE(o, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001822
1823 jstring s1 = env_->NewStringUTF("hello");
Ian Rogers2d10b202014-05-12 19:15:18 -07001824 ASSERT_NE(s1, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001825 jstring s2 = env_->NewStringUTF("world");
Ian Rogers2d10b202014-05-12 19:15:18 -07001826 ASSERT_NE(s2, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001827
1828 jfieldID s_fid = env_->GetStaticFieldID(c, "sObject", "Ljava/lang/Object;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001829 ASSERT_NE(s_fid, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001830 jfieldID i_fid = env_->GetFieldID(c, "iObject", "Ljava/lang/Object;");
Ian Rogers2d10b202014-05-12 19:15:18 -07001831 ASSERT_NE(i_fid, nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -07001832
1833 env_->SetStaticObjectField(c, s_fid, s1);
1834 ASSERT_TRUE(env_->IsSameObject(s1, env_->GetStaticObjectField(c, s_fid)));
1835 env_->SetStaticObjectField(c, s_fid, s2);
1836 ASSERT_TRUE(env_->IsSameObject(s2, env_->GetStaticObjectField(c, s_fid)));
1837
1838 env_->SetObjectField(o, i_fid, s1);
1839 ASSERT_TRUE(env_->IsSameObject(s1, env_->GetObjectField(o, i_fid)));
1840 env_->SetObjectField(o, i_fid, s2);
1841 ASSERT_TRUE(env_->IsSameObject(s2, env_->GetObjectField(o, i_fid)));
1842}
1843
Ian Rogers2d10b202014-05-12 19:15:18 -07001844TEST_F(JniInternalTest, NewLocalRef_nullptr) {
1845 EXPECT_EQ(env_->NewLocalRef(nullptr), nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001846}
1847
1848TEST_F(JniInternalTest, NewLocalRef) {
1849 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001850 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001851 jobject o = env_->NewLocalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001852 EXPECT_NE(o, nullptr);
1853 EXPECT_NE(o, s);
Elliott Hughes18c07532011-08-18 15:50:51 -07001854
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001855 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(o));
Elliott Hughes18c07532011-08-18 15:50:51 -07001856}
1857
Ian Rogers2d10b202014-05-12 19:15:18 -07001858TEST_F(JniInternalTest, DeleteLocalRef_nullptr) {
1859 env_->DeleteLocalRef(nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001860}
1861
1862TEST_F(JniInternalTest, DeleteLocalRef) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001863 // This tests leads to warnings and errors in the log.
1864 ScopedLogSeverity sls(LogSeverity::FATAL);
1865
Elliott Hughes18c07532011-08-18 15:50:51 -07001866 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001867 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001868 env_->DeleteLocalRef(s);
1869
Elliott Hughes3f6635a2012-06-19 13:37:49 -07001870 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07001871 {
Ian Rogers68d8b422014-07-17 11:09:10 -07001872 bool old_check_jni = vm_->SetCheckJniEnabled(false);
1873 {
1874 CheckJniAbortCatcher check_jni_abort_catcher;
1875 env_->DeleteLocalRef(s);
1876 }
Elliott Hughesb264f082012-04-06 17:10:10 -07001877 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001878 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
Elliott Hughesb264f082012-04-06 17:10:10 -07001879 env_->DeleteLocalRef(s);
Ian Rogersc0542af2014-09-03 16:16:56 -07001880 std::string expected(StringPrintf("use of deleted local reference %p", s));
Elliott Hughesa9137c62013-01-09 10:55:21 -08001881 check_jni_abort_catcher.Check(expected.c_str());
Ian Rogers68d8b422014-07-17 11:09:10 -07001882 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07001883 }
Elliott Hughes18c07532011-08-18 15:50:51 -07001884
1885 s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001886 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001887 jobject o = env_->NewLocalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001888 ASSERT_NE(o, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001889
1890 env_->DeleteLocalRef(s);
1891 env_->DeleteLocalRef(o);
1892}
1893
Elliott Hughesaa836f72013-08-20 16:57:23 -07001894TEST_F(JniInternalTest, PushLocalFrame_10395422) {
1895 // The JNI specification is ambiguous about whether the given capacity is to be interpreted as a
1896 // maximum or as a minimum, but it seems like it's supposed to be a minimum, and that's how
1897 // Android historically treated it, and it's how the RI treats it. It's also the more useful
1898 // interpretation!
1899 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(0));
Ian Rogers2d10b202014-05-12 19:15:18 -07001900 env_->PopLocalFrame(nullptr);
Elliott Hughesaa836f72013-08-20 16:57:23 -07001901
Andreas Gampe369810a2015-01-14 19:53:31 -08001902 // The following two tests will print errors to the log.
1903 ScopedLogSeverity sls(LogSeverity::FATAL);
1904
Elliott Hughesaa836f72013-08-20 16:57:23 -07001905 // Negative capacities are not allowed.
1906 ASSERT_EQ(JNI_ERR, env_->PushLocalFrame(-1));
1907
1908 // And it's okay to have an upper limit. Ours is currently 512.
1909 ASSERT_EQ(JNI_ERR, env_->PushLocalFrame(8192));
1910}
1911
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001912TEST_F(JniInternalTest, PushLocalFrame_PopLocalFrame) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001913 // This tests leads to errors in the log.
1914 ScopedLogSeverity sls(LogSeverity::FATAL);
1915
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001916 jobject original = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001917 ASSERT_NE(original, nullptr);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001918
1919 jobject outer;
1920 jobject inner1, inner2;
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001921 ScopedObjectAccess soa(env_);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001922 {
Elliott Hughesaa836f72013-08-20 16:57:23 -07001923 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(4));
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001924 outer = env_->NewLocalRef(original);
1925
1926 {
Elliott Hughesaa836f72013-08-20 16:57:23 -07001927 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(4));
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001928 inner1 = env_->NewLocalRef(outer);
1929 inner2 = env_->NewStringUTF("survivor");
Ian Rogersc0542af2014-09-03 16:16:56 -07001930 EXPECT_NE(env_->PopLocalFrame(inner2), nullptr);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001931 }
1932
1933 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(original));
1934 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(outer));
Ian Rogersc0542af2014-09-03 16:16:56 -07001935 {
1936 CheckJniAbortCatcher check_jni_abort_catcher;
1937 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner1));
1938 check_jni_abort_catcher.Check("use of deleted local reference");
1939 }
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001940
1941 // Our local reference for the survivor is invalid because the survivor
1942 // gets a new local reference...
Ian Rogersc0542af2014-09-03 16:16:56 -07001943 {
1944 CheckJniAbortCatcher check_jni_abort_catcher;
1945 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner2));
1946 check_jni_abort_catcher.Check("use of deleted local reference");
1947 }
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001948
Ian Rogersc0542af2014-09-03 16:16:56 -07001949 EXPECT_EQ(env_->PopLocalFrame(nullptr), nullptr);
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001950 }
1951 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(original));
Ian Rogersc0542af2014-09-03 16:16:56 -07001952 CheckJniAbortCatcher check_jni_abort_catcher;
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001953 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(outer));
Ian Rogersc0542af2014-09-03 16:16:56 -07001954 check_jni_abort_catcher.Check("use of deleted local reference");
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001955 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner1));
Ian Rogersc0542af2014-09-03 16:16:56 -07001956 check_jni_abort_catcher.Check("use of deleted local reference");
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001957 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner2));
Ian Rogersc0542af2014-09-03 16:16:56 -07001958 check_jni_abort_catcher.Check("use of deleted local reference");
Elliott Hughes2ced6a52011-10-16 18:44:48 -07001959}
1960
Ian Rogers2d10b202014-05-12 19:15:18 -07001961TEST_F(JniInternalTest, NewGlobalRef_nullptr) {
1962 EXPECT_EQ(env_->NewGlobalRef(nullptr), nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001963}
1964
1965TEST_F(JniInternalTest, NewGlobalRef) {
1966 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001967 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001968 jobject o = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001969 EXPECT_NE(o, nullptr);
1970 EXPECT_NE(o, s);
Elliott Hughes18c07532011-08-18 15:50:51 -07001971
Ian Rogers2d10b202014-05-12 19:15:18 -07001972 EXPECT_EQ(env_->GetObjectRefType(o), JNIGlobalRefType);
Elliott Hughes18c07532011-08-18 15:50:51 -07001973}
1974
Ian Rogers2d10b202014-05-12 19:15:18 -07001975TEST_F(JniInternalTest, DeleteGlobalRef_nullptr) {
1976 env_->DeleteGlobalRef(nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001977}
1978
1979TEST_F(JniInternalTest, DeleteGlobalRef) {
Andreas Gampe369810a2015-01-14 19:53:31 -08001980 // This tests leads to warnings and errors in the log.
1981 ScopedLogSeverity sls(LogSeverity::FATAL);
1982
Elliott Hughes18c07532011-08-18 15:50:51 -07001983 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07001984 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001985
1986 jobject o = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07001987 ASSERT_NE(o, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07001988 env_->DeleteGlobalRef(o);
1989
Elliott Hughes3f6635a2012-06-19 13:37:49 -07001990 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07001991 {
Ian Rogers68d8b422014-07-17 11:09:10 -07001992 bool old_check_jni = vm_->SetCheckJniEnabled(false);
1993 {
1994 CheckJniAbortCatcher check_jni_abort_catcher;
1995 env_->DeleteGlobalRef(o);
1996 }
Elliott Hughesb264f082012-04-06 17:10:10 -07001997 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07001998 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
Elliott Hughesb264f082012-04-06 17:10:10 -07001999 env_->DeleteGlobalRef(o);
Ian Rogersc0542af2014-09-03 16:16:56 -07002000 std::string expected(StringPrintf("use of deleted global reference %p", o));
Elliott Hughesa9137c62013-01-09 10:55:21 -08002001 check_jni_abort_catcher.Check(expected.c_str());
Ian Rogers68d8b422014-07-17 11:09:10 -07002002 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07002003 }
Elliott Hughes18c07532011-08-18 15:50:51 -07002004
2005 jobject o1 = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002006 ASSERT_NE(o1, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002007 jobject o2 = env_->NewGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002008 ASSERT_NE(o2, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002009
2010 env_->DeleteGlobalRef(o1);
2011 env_->DeleteGlobalRef(o2);
2012}
2013
Ian Rogers2d10b202014-05-12 19:15:18 -07002014TEST_F(JniInternalTest, NewWeakGlobalRef_nullptr) {
2015 EXPECT_EQ(env_->NewWeakGlobalRef(nullptr), nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002016}
2017
2018TEST_F(JniInternalTest, NewWeakGlobalRef) {
2019 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07002020 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002021 jobject o = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002022 EXPECT_NE(o, nullptr);
2023 EXPECT_NE(o, s);
Elliott Hughes18c07532011-08-18 15:50:51 -07002024
Ian Rogers2d10b202014-05-12 19:15:18 -07002025 EXPECT_EQ(env_->GetObjectRefType(o), JNIWeakGlobalRefType);
Elliott Hughes18c07532011-08-18 15:50:51 -07002026}
2027
Ian Rogers2d10b202014-05-12 19:15:18 -07002028TEST_F(JniInternalTest, DeleteWeakGlobalRef_nullptr) {
2029 env_->DeleteWeakGlobalRef(nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002030}
2031
2032TEST_F(JniInternalTest, DeleteWeakGlobalRef) {
Andreas Gampe369810a2015-01-14 19:53:31 -08002033 // This tests leads to warnings and errors in the log.
2034 ScopedLogSeverity sls(LogSeverity::FATAL);
2035
Elliott Hughes18c07532011-08-18 15:50:51 -07002036 jstring s = env_->NewStringUTF("");
Ian Rogers2d10b202014-05-12 19:15:18 -07002037 ASSERT_NE(s, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002038
2039 jobject o = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002040 ASSERT_NE(o, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002041 env_->DeleteWeakGlobalRef(o);
2042
Elliott Hughes3f6635a2012-06-19 13:37:49 -07002043 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07002044 {
Ian Rogers68d8b422014-07-17 11:09:10 -07002045 bool old_check_jni = vm_->SetCheckJniEnabled(false);
2046 {
2047 CheckJniAbortCatcher check_jni_abort_catcher;
2048 env_->DeleteWeakGlobalRef(o);
2049 }
Elliott Hughesb264f082012-04-06 17:10:10 -07002050 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers68d8b422014-07-17 11:09:10 -07002051 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
Elliott Hughesb264f082012-04-06 17:10:10 -07002052 env_->DeleteWeakGlobalRef(o);
Ian Rogersc0542af2014-09-03 16:16:56 -07002053 std::string expected(StringPrintf("use of deleted weak global reference %p", o));
Elliott Hughesa9137c62013-01-09 10:55:21 -08002054 check_jni_abort_catcher.Check(expected.c_str());
Ian Rogers68d8b422014-07-17 11:09:10 -07002055 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughesb264f082012-04-06 17:10:10 -07002056 }
Elliott Hughes18c07532011-08-18 15:50:51 -07002057
2058 jobject o1 = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002059 ASSERT_NE(o1, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002060 jobject o2 = env_->NewWeakGlobalRef(s);
Ian Rogers2d10b202014-05-12 19:15:18 -07002061 ASSERT_NE(o2, nullptr);
Elliott Hughes18c07532011-08-18 15:50:51 -07002062
2063 env_->DeleteWeakGlobalRef(o1);
2064 env_->DeleteWeakGlobalRef(o2);
2065}
2066
Alexei Zavjalov3a1444c2014-06-25 16:04:55 +07002067TEST_F(JniInternalTest, ExceptionDescribe) {
2068 // This checks how ExceptionDescribe handles call without exception.
2069 env_->ExceptionClear();
2070 env_->ExceptionDescribe();
2071}
2072
Elliott Hughes37f7a402011-08-22 18:56:01 -07002073TEST_F(JniInternalTest, Throw) {
Elliott Hughes37f7a402011-08-22 18:56:01 -07002074 jclass exception_class = env_->FindClass("java/lang/RuntimeException");
Ian Rogers2d10b202014-05-12 19:15:18 -07002075 ASSERT_TRUE(exception_class != nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07002076 jthrowable exception = reinterpret_cast<jthrowable>(env_->AllocObject(exception_class));
Ian Rogers2d10b202014-05-12 19:15:18 -07002077 ASSERT_TRUE(exception != nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07002078
2079 EXPECT_EQ(JNI_OK, env_->Throw(exception));
2080 EXPECT_TRUE(env_->ExceptionCheck());
Elliott Hughesa2501992011-08-26 19:39:54 -07002081 jthrowable thrown_exception = env_->ExceptionOccurred();
Elliott Hughes37f7a402011-08-22 18:56:01 -07002082 env_->ExceptionClear();
Elliott Hughesa2501992011-08-26 19:39:54 -07002083 EXPECT_TRUE(env_->IsSameObject(exception, thrown_exception));
Ian Rogers68d8b422014-07-17 11:09:10 -07002084
2085 // Bad argument.
2086 bool old_check_jni = vm_->SetCheckJniEnabled(false);
2087 EXPECT_EQ(JNI_ERR, env_->Throw(nullptr));
2088 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
2089 CheckJniAbortCatcher check_jni_abort_catcher;
2090 EXPECT_EQ(JNI_ERR, env_->Throw(nullptr));
2091 check_jni_abort_catcher.Check("Throw received NULL jthrowable");
2092 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes37f7a402011-08-22 18:56:01 -07002093}
2094
2095TEST_F(JniInternalTest, ThrowNew) {
Elliott Hughes37f7a402011-08-22 18:56:01 -07002096 jclass exception_class = env_->FindClass("java/lang/RuntimeException");
Ian Rogers2d10b202014-05-12 19:15:18 -07002097 ASSERT_TRUE(exception_class != nullptr);
Elliott Hughes37f7a402011-08-22 18:56:01 -07002098
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07002099 jthrowable thrown_exception;
2100
Elliott Hughes37f7a402011-08-22 18:56:01 -07002101 EXPECT_EQ(JNI_OK, env_->ThrowNew(exception_class, "hello world"));
2102 EXPECT_TRUE(env_->ExceptionCheck());
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07002103 thrown_exception = env_->ExceptionOccurred();
2104 env_->ExceptionClear();
2105 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, exception_class));
2106
Ian Rogers2d10b202014-05-12 19:15:18 -07002107 EXPECT_EQ(JNI_OK, env_->ThrowNew(exception_class, nullptr));
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07002108 EXPECT_TRUE(env_->ExceptionCheck());
2109 thrown_exception = env_->ExceptionOccurred();
Elliott Hughes37f7a402011-08-22 18:56:01 -07002110 env_->ExceptionClear();
Elliott Hughesa2501992011-08-26 19:39:54 -07002111 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, exception_class));
Ian Rogers68d8b422014-07-17 11:09:10 -07002112
2113 // Bad argument.
2114 bool old_check_jni = vm_->SetCheckJniEnabled(false);
2115 CheckJniAbortCatcher check_jni_abort_catcher;
2116 EXPECT_EQ(JNI_ERR, env_->ThrowNew(nullptr, nullptr));
2117 check_jni_abort_catcher.Check("c == null");
2118 EXPECT_FALSE(vm_->SetCheckJniEnabled(true));
2119 EXPECT_EQ(JNI_ERR, env_->ThrowNew(nullptr, nullptr));
2120 check_jni_abort_catcher.Check("ThrowNew received NULL jclass");
2121 EXPECT_TRUE(vm_->SetCheckJniEnabled(old_check_jni));
Elliott Hughes37f7a402011-08-22 18:56:01 -07002122}
2123
Ian Rogers1d99e452014-01-02 17:36:41 -08002124TEST_F(JniInternalTest, NewDirectBuffer_GetDirectBufferAddress_GetDirectBufferCapacity) {
2125 // Start runtime.
2126 Thread* self = Thread::Current();
2127 self->TransitionFromSuspendedToRunnable();
2128 MakeExecutable(nullptr, "java.lang.Class");
2129 MakeExecutable(nullptr, "java.lang.Object");
2130 MakeExecutable(nullptr, "java.nio.DirectByteBuffer");
Yi Kong1fdb6962015-11-19 13:58:04 +00002131 MakeExecutable(nullptr, "java.nio.Bits");
Ian Rogers1d99e452014-01-02 17:36:41 -08002132 MakeExecutable(nullptr, "java.nio.MappedByteBuffer");
2133 MakeExecutable(nullptr, "java.nio.ByteBuffer");
2134 MakeExecutable(nullptr, "java.nio.Buffer");
2135 // TODO: we only load a dex file here as starting the runtime relies upon it.
2136 const char* class_name = "StaticLeafMethods";
2137 LoadDex(class_name);
2138 bool started = runtime_->Start();
2139 ASSERT_TRUE(started);
2140
Elliott Hughesb465ab02011-08-24 11:21:21 -07002141 jclass buffer_class = env_->FindClass("java/nio/Buffer");
Ian Rogers2d10b202014-05-12 19:15:18 -07002142 ASSERT_NE(buffer_class, nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07002143
2144 char bytes[1024];
2145 jobject buffer = env_->NewDirectByteBuffer(bytes, sizeof(bytes));
Ian Rogers2d10b202014-05-12 19:15:18 -07002146 ASSERT_NE(buffer, nullptr);
Elliott Hughesb465ab02011-08-24 11:21:21 -07002147 ASSERT_TRUE(env_->IsInstanceOf(buffer, buffer_class));
Ian Rogers2d10b202014-05-12 19:15:18 -07002148 ASSERT_EQ(env_->GetDirectBufferAddress(buffer), bytes);
Ian Rogers1ee99352014-05-14 14:38:16 -07002149 ASSERT_EQ(env_->GetDirectBufferCapacity(buffer), static_cast<jlong>(sizeof(bytes)));
Brian Carlstrom45d26c82014-06-24 23:36:28 -07002150
2151 {
2152 CheckJniAbortCatcher check_jni_abort_catcher;
Brian Carlstrom85a93362014-06-25 09:30:52 -07002153 env_->NewDirectByteBuffer(bytes, static_cast<jlong>(INT_MAX) + 1);
Brian Carlstrom45d26c82014-06-24 23:36:28 -07002154 check_jni_abort_catcher.Check("in call to NewDirectByteBuffer");
2155 }
Elliott Hughesb465ab02011-08-24 11:21:21 -07002156}
2157
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002158TEST_F(JniInternalTest, MonitorEnterExit) {
Andreas Gampe369810a2015-01-14 19:53:31 -08002159 // This will print some error messages. Suppress.
2160 ScopedLogSeverity sls(LogSeverity::FATAL);
2161
Ian Rogers2d10b202014-05-12 19:15:18 -07002162 // Create an object to torture.
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002163 jclass object_class = env_->FindClass("java/lang/Object");
Ian Rogers2d10b202014-05-12 19:15:18 -07002164 ASSERT_NE(object_class, nullptr);
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002165 jobject object = env_->AllocObject(object_class);
Ian Rogers2d10b202014-05-12 19:15:18 -07002166 ASSERT_NE(object, nullptr);
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002167
2168 // Expected class of exceptions
2169 jclass imse_class = env_->FindClass("java/lang/IllegalMonitorStateException");
Ian Rogers2d10b202014-05-12 19:15:18 -07002170 ASSERT_NE(imse_class, nullptr);
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002171
2172 jthrowable thrown_exception;
2173
2174 // Unlock of unowned monitor
2175 env_->MonitorExit(object);
2176 EXPECT_TRUE(env_->ExceptionCheck());
2177 thrown_exception = env_->ExceptionOccurred();
2178 env_->ExceptionClear();
2179 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, imse_class));
2180
2181 // Lock of unowned monitor
2182 env_->MonitorEnter(object);
2183 EXPECT_FALSE(env_->ExceptionCheck());
2184 // Regular unlock
2185 env_->MonitorExit(object);
2186 EXPECT_FALSE(env_->ExceptionCheck());
2187
2188 // Recursively lock a lot
2189 size_t max_recursive_lock = 1024;
2190 for (size_t i = 0; i < max_recursive_lock; i++) {
2191 env_->MonitorEnter(object);
2192 EXPECT_FALSE(env_->ExceptionCheck());
2193 }
2194 // Recursively unlock a lot
2195 for (size_t i = 0; i < max_recursive_lock; i++) {
2196 env_->MonitorExit(object);
2197 EXPECT_FALSE(env_->ExceptionCheck());
2198 }
2199
2200 // Unlock of unowned monitor
2201 env_->MonitorExit(object);
2202 EXPECT_TRUE(env_->ExceptionCheck());
2203 thrown_exception = env_->ExceptionOccurred();
2204 env_->ExceptionClear();
2205 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, imse_class));
Elliott Hughesa92853e2012-02-07 16:09:27 -08002206
Mathieu Chartier2cebb242015-04-21 16:50:40 -07002207 // It's an error to call MonitorEnter or MonitorExit on null.
Elliott Hughesb264f082012-04-06 17:10:10 -07002208 {
2209 CheckJniAbortCatcher check_jni_abort_catcher;
Ian Rogers2d10b202014-05-12 19:15:18 -07002210 env_->MonitorEnter(nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -07002211 check_jni_abort_catcher.Check("in call to MonitorEnter");
Ian Rogers2d10b202014-05-12 19:15:18 -07002212 env_->MonitorExit(nullptr);
Elliott Hughesb264f082012-04-06 17:10:10 -07002213 check_jni_abort_catcher.Check("in call to MonitorExit");
2214 }
Ian Rogers6d0b13e2012-02-07 09:25:29 -08002215}
2216
Andreas Gampe5f4a09a2015-09-28 13:16:33 -07002217void Java_MyClassNatives_foo_exit(JNIEnv* env, jobject thisObj) {
2218 // Release the monitor on self. This should trigger an abort.
2219 env->MonitorExit(thisObj);
2220}
2221
2222TEST_F(JniInternalTest, MonitorExitLockedInDifferentCall) {
2223 SetUpForTest(false, "foo", "()V", reinterpret_cast<void*>(&Java_MyClassNatives_foo_exit));
2224 ASSERT_NE(jobj_, nullptr);
2225
2226 env_->MonitorEnter(jobj_);
2227 EXPECT_FALSE(env_->ExceptionCheck());
2228
2229 CheckJniAbortCatcher check_jni_abort_catcher;
2230 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
2231 check_jni_abort_catcher.Check("Unlocking monitor that wasn't locked here");
2232}
2233
2234void Java_MyClassNatives_foo_enter_no_exit(JNIEnv* env, jobject thisObj) {
2235 // Acquire but don't release the monitor on self. This should trigger an abort on return.
2236 env->MonitorEnter(thisObj);
2237}
2238
2239TEST_F(JniInternalTest, MonitorExitNotAllUnlocked) {
2240 SetUpForTest(false,
2241 "foo",
2242 "()V",
2243 reinterpret_cast<void*>(&Java_MyClassNatives_foo_enter_no_exit));
2244 ASSERT_NE(jobj_, nullptr);
2245
2246 CheckJniAbortCatcher check_jni_abort_catcher;
2247 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
2248 check_jni_abort_catcher.Check("Still holding a locked object on JNI end");
2249}
2250
Andreas Gampeb8c4f032015-12-03 10:56:18 -08002251static bool IsLocked(JNIEnv* env, jobject jobj) {
2252 ScopedObjectAccess soa(env);
Mathieu Chartier0795f232016-09-27 18:43:30 -07002253 LockWord lock_word = soa.Decode<mirror::Object>(jobj)->GetLockWord(true);
Andreas Gampeb8c4f032015-12-03 10:56:18 -08002254 switch (lock_word.GetState()) {
2255 case LockWord::kHashCode:
2256 case LockWord::kUnlocked:
2257 return false;
2258 case LockWord::kThinLocked:
2259 return true;
2260 case LockWord::kFatLocked:
2261 return lock_word.FatLockMonitor()->IsLocked();
2262 default: {
2263 LOG(FATAL) << "Invalid monitor state " << lock_word.GetState();
2264 UNREACHABLE();
2265 }
2266 }
2267}
2268
2269TEST_F(JniInternalTest, DetachThreadUnlockJNIMonitors) {
2270 // We need to lock an object, detach, reattach, and check the locks.
2271 //
2272 // As re-attaching will create a different thread, we need to use a global
2273 // ref to keep the object around.
2274
2275 // Create an object to torture.
2276 jobject global_ref;
2277 {
2278 jclass object_class = env_->FindClass("java/lang/Object");
2279 ASSERT_NE(object_class, nullptr);
2280 jobject object = env_->AllocObject(object_class);
2281 ASSERT_NE(object, nullptr);
2282 global_ref = env_->NewGlobalRef(object);
2283 }
2284
2285 // Lock it.
2286 env_->MonitorEnter(global_ref);
2287 ASSERT_TRUE(IsLocked(env_, global_ref));
2288
2289 // Detach and re-attach.
2290 jint detach_result = vm_->DetachCurrentThread();
2291 ASSERT_EQ(detach_result, JNI_OK);
2292 jint attach_result = vm_->AttachCurrentThread(&env_, nullptr);
2293 ASSERT_EQ(attach_result, JNI_OK);
2294
2295 // Look at the global ref, check whether it's still locked.
2296 ASSERT_FALSE(IsLocked(env_, global_ref));
2297
2298 // Delete the global ref.
2299 env_->DeleteGlobalRef(global_ref);
2300}
2301
Andreas Gampeda9b7632015-12-09 15:14:04 -08002302// Test the offset computation of IndirectReferenceTable offsets. b/26071368.
2303TEST_F(JniInternalTest, IndirectReferenceTableOffsets) {
2304 // The segment_state_ field is private, and we want to avoid friend declaration. So we'll check
2305 // by modifying memory.
2306 // The parameters don't really matter here.
Richard Uhlerda0a69e2016-10-11 15:06:38 +01002307 std::string error_msg;
Andreas Gampe9d7ef622016-10-24 19:35:19 -07002308 IndirectReferenceTable irt(5,
2309 IndirectRefKind::kGlobal,
2310 IndirectReferenceTable::ResizableCapacity::kNo,
2311 &error_msg);
Richard Uhlerda0a69e2016-10-11 15:06:38 +01002312 ASSERT_TRUE(irt.IsValid()) << error_msg;
Andreas Gampee03662b2016-10-13 17:12:56 -07002313 IRTSegmentState old_state = irt.GetSegmentState();
Andreas Gampeda9b7632015-12-09 15:14:04 -08002314
2315 // Write some new state directly. We invert parts of old_state to ensure a new value.
Andreas Gampee03662b2016-10-13 17:12:56 -07002316 IRTSegmentState new_state;
2317 new_state.top_index = old_state.top_index ^ 0x07705005;
2318 ASSERT_NE(old_state.top_index, new_state.top_index);
Andreas Gampeda9b7632015-12-09 15:14:04 -08002319
2320 uint8_t* base = reinterpret_cast<uint8_t*>(&irt);
2321 int32_t segment_state_offset =
2322 IndirectReferenceTable::SegmentStateOffset(sizeof(void*)).Int32Value();
Andreas Gampee03662b2016-10-13 17:12:56 -07002323 *reinterpret_cast<IRTSegmentState*>(base + segment_state_offset) = new_state;
Andreas Gampeda9b7632015-12-09 15:14:04 -08002324
2325 // Read and compare.
Andreas Gampee03662b2016-10-13 17:12:56 -07002326 EXPECT_EQ(new_state.top_index, irt.GetSegmentState().top_index);
Andreas Gampeda9b7632015-12-09 15:14:04 -08002327}
2328
2329// Test the offset computation of JNIEnvExt offsets. b/26071368.
2330TEST_F(JniInternalTest, JNIEnvExtOffsets) {
2331 EXPECT_EQ(OFFSETOF_MEMBER(JNIEnvExt, local_ref_cookie),
Vladimir Marko46817b82016-03-29 12:21:58 +01002332 JNIEnvExt::LocalRefCookieOffset(sizeof(void*)).Uint32Value());
Andreas Gampeda9b7632015-12-09 15:14:04 -08002333
Vladimir Marko46817b82016-03-29 12:21:58 +01002334 EXPECT_EQ(OFFSETOF_MEMBER(JNIEnvExt, self), JNIEnvExt::SelfOffset(sizeof(void*)).Uint32Value());
Andreas Gampeda9b7632015-12-09 15:14:04 -08002335
2336 // segment_state_ is private in the IndirectReferenceTable. So this test isn't as good as we'd
2337 // hope it to be.
Vladimir Marko46817b82016-03-29 12:21:58 +01002338 uint32_t segment_state_now =
Andreas Gampeda9b7632015-12-09 15:14:04 -08002339 OFFSETOF_MEMBER(JNIEnvExt, locals) +
Vladimir Marko46817b82016-03-29 12:21:58 +01002340 IndirectReferenceTable::SegmentStateOffset(sizeof(void*)).Uint32Value();
2341 uint32_t segment_state_computed = JNIEnvExt::SegmentStateOffset(sizeof(void*)).Uint32Value();
Andreas Gampeda9b7632015-12-09 15:14:04 -08002342 EXPECT_EQ(segment_state_now, segment_state_computed);
2343}
2344
Ian Rogers0cfe1fb2011-08-26 03:29:44 -07002345} // namespace art