blob: 071b658fb92964e670baaa5b8345e6617e11649f [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
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080019#include "common_compiler_test.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070020#include "mirror/art_method-inl.h"
Elliott Hughes726079d2011-10-07 18:43:44 -070021#include "ScopedLocalRef.h"
Elliott Hughes0c9cd562011-08-12 10:59:29 -070022
23namespace art {
24
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080025// TODO: Convert to CommonRuntimeTest. Currently MakeExecutable is used.
26class JniInternalTest : public CommonCompilerTest {
Elliott Hughesc7ac37f2011-08-12 12:21:58 -070027 protected:
28 virtual void SetUp() {
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080029 CommonCompilerTest::SetUp();
Elliott Hughes5174fe62011-08-23 15:12:35 -070030
Elliott Hughesa2501992011-08-26 19:39:54 -070031 vm_ = Runtime::Current()->GetJavaVM();
32
Elliott Hughes5174fe62011-08-23 15:12:35 -070033 // Turn on -verbose:jni for the JNI tests.
Ian Rogers79713632013-08-21 19:06:15 -070034 // gLogVerbosity.jni = true;
Elliott Hughes5174fe62011-08-23 15:12:35 -070035
Brian Carlstrom4d571432012-05-16 00:21:41 -070036 vm_->AttachCurrentThread(&env_, NULL);
Elliott Hughesb465ab02011-08-24 11:21:21 -070037
Brian Carlstromea46f952013-07-30 01:26:50 -070038 ScopedLocalRef<jclass> aioobe(env_,
39 env_->FindClass("java/lang/ArrayIndexOutOfBoundsException"));
Elliott Hughes726079d2011-10-07 18:43:44 -070040 CHECK(aioobe.get() != NULL);
41 aioobe_ = reinterpret_cast<jclass>(env_->NewGlobalRef(aioobe.get()));
Elliott Hughesb465ab02011-08-24 11:21:21 -070042
Elliott Hughesb264f082012-04-06 17:10:10 -070043 ScopedLocalRef<jclass> ase(env_, env_->FindClass("java/lang/ArrayStoreException"));
44 CHECK(ase.get() != NULL);
45 ase_ = reinterpret_cast<jclass>(env_->NewGlobalRef(ase.get()));
46
Brian Carlstromea46f952013-07-30 01:26:50 -070047 ScopedLocalRef<jclass> sioobe(env_,
48 env_->FindClass("java/lang/StringIndexOutOfBoundsException"));
Elliott Hughes726079d2011-10-07 18:43:44 -070049 CHECK(sioobe.get() != NULL);
50 sioobe_ = reinterpret_cast<jclass>(env_->NewGlobalRef(sioobe.get()));
51 }
52
Brian Carlstrom4d571432012-05-16 00:21:41 -070053 void CleanUpJniEnv() {
54 if (aioobe_ != NULL) {
55 env_->DeleteGlobalRef(aioobe_);
56 aioobe_ = NULL;
57 }
58 if (ase_ != NULL) {
59 env_->DeleteGlobalRef(ase_);
60 ase_ = NULL;
61 }
62 if (sioobe_ != NULL) {
63 env_->DeleteGlobalRef(sioobe_);
64 sioobe_ = NULL;
65 }
66 }
67
Ian Rogers53b8b092014-03-13 23:45:53 -070068 virtual void TearDown() OVERRIDE {
Brian Carlstrom4d571432012-05-16 00:21:41 -070069 CleanUpJniEnv();
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080070 CommonCompilerTest::TearDown();
Elliott Hughesc7ac37f2011-08-12 12:21:58 -070071 }
Elliott Hughesb465ab02011-08-24 11:21:21 -070072
Ian Rogers1d99e452014-01-02 17:36:41 -080073 jclass GetPrimitiveClass(char descriptor) {
74 ScopedObjectAccess soa(env_);
75 mirror::Class* c = class_linker_->FindPrimitiveClass(descriptor);
76 CHECK(c != nullptr);
77 return soa.AddLocalReference<jclass>(c);
78 }
79
Elliott Hughesa2501992011-08-26 19:39:54 -070080 JavaVMExt* vm_;
Brian Carlstrom4d571432012-05-16 00:21:41 -070081 JNIEnv* env_;
Elliott Hughes814e4032011-08-23 12:07:56 -070082 jclass aioobe_;
Elliott Hughesb264f082012-04-06 17:10:10 -070083 jclass ase_;
Elliott Hughesb465ab02011-08-24 11:21:21 -070084 jclass sioobe_;
Elliott Hughes0c9cd562011-08-12 10:59:29 -070085};
86
Elliott Hughes885c3bd2011-08-22 16:59:20 -070087TEST_F(JniInternalTest, AllocObject) {
88 jclass c = env_->FindClass("java/lang/String");
89 ASSERT_TRUE(c != NULL);
90 jobject o = env_->AllocObject(c);
91 ASSERT_TRUE(o != NULL);
92
93 // We have an instance of the class we asked for...
94 ASSERT_TRUE(env_->IsInstanceOf(o, c));
95 // ...whose fields haven't been initialized because
96 // we didn't call a constructor.
97 ASSERT_EQ(0, env_->GetIntField(o, env_->GetFieldID(c, "count", "I")));
98 ASSERT_EQ(0, env_->GetIntField(o, env_->GetFieldID(c, "offset", "I")));
99 ASSERT_TRUE(env_->GetObjectField(o, env_->GetFieldID(c, "value", "[C")) == NULL);
100}
101
Elliott Hughesc7ac37f2011-08-12 12:21:58 -0700102TEST_F(JniInternalTest, GetVersion) {
103 ASSERT_EQ(JNI_VERSION_1_6, env_->GetVersion());
104}
105
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700106#define EXPECT_CLASS_FOUND(NAME) \
Elliott Hughesbd935992011-08-22 11:59:34 -0700107 EXPECT_TRUE(env_->FindClass(NAME) != NULL); \
108 EXPECT_FALSE(env_->ExceptionCheck())
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700109
110#define EXPECT_CLASS_NOT_FOUND(NAME) \
Elliott Hughesbd935992011-08-22 11:59:34 -0700111 EXPECT_TRUE(env_->FindClass(NAME) == NULL); \
112 EXPECT_TRUE(env_->ExceptionCheck()); \
113 env_->ExceptionClear()
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700114
115TEST_F(JniInternalTest, FindClass) {
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700116 // Reference types...
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700117 EXPECT_CLASS_FOUND("java/lang/String");
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700118 // ...for arrays too, where you must include "L;".
119 EXPECT_CLASS_FOUND("[Ljava/lang/String;");
Elliott Hughes3f6635a2012-06-19 13:37:49 -0700120 // Primitive arrays are okay too, if the primitive type is valid.
121 EXPECT_CLASS_FOUND("[C");
Elliott Hughesa2501992011-08-26 19:39:54 -0700122
Elliott Hughesb264f082012-04-06 17:10:10 -0700123 {
Elliott Hughesb264f082012-04-06 17:10:10 -0700124 // We support . as well as / for compatibility, if -Xcheck:jni is off.
Elliott Hughes3f6635a2012-06-19 13:37:49 -0700125 CheckJniAbortCatcher check_jni_abort_catcher;
Elliott Hughesb264f082012-04-06 17:10:10 -0700126 EXPECT_CLASS_FOUND("java.lang.String");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700127 check_jni_abort_catcher.Check("illegal class name 'java.lang.String'");
Elliott Hughesb264f082012-04-06 17:10:10 -0700128 EXPECT_CLASS_NOT_FOUND("Ljava.lang.String;");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700129 check_jni_abort_catcher.Check("illegal class name 'Ljava.lang.String;'");
Elliott Hughesb264f082012-04-06 17:10:10 -0700130 EXPECT_CLASS_FOUND("[Ljava.lang.String;");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700131 check_jni_abort_catcher.Check("illegal class name '[Ljava.lang.String;'");
Elliott Hughesb264f082012-04-06 17:10:10 -0700132 EXPECT_CLASS_NOT_FOUND("[java.lang.String");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700133 check_jni_abort_catcher.Check("illegal class name '[java.lang.String'");
Elliott Hughesb264f082012-04-06 17:10:10 -0700134
135 // You can't include the "L;" in a JNI class descriptor.
136 EXPECT_CLASS_NOT_FOUND("Ljava/lang/String;");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700137 check_jni_abort_catcher.Check("illegal class name 'Ljava/lang/String;'");
Elliott Hughes3f6635a2012-06-19 13:37:49 -0700138
Elliott Hughesb264f082012-04-06 17:10:10 -0700139 // But you must include it for an array of any reference type.
140 EXPECT_CLASS_NOT_FOUND("[java/lang/String");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700141 check_jni_abort_catcher.Check("illegal class name '[java/lang/String'");
Elliott Hughes3f6635a2012-06-19 13:37:49 -0700142
143 EXPECT_CLASS_NOT_FOUND("[K");
Elliott Hughes56ef0422012-06-19 14:35:04 -0700144 check_jni_abort_catcher.Check("illegal class name '[K'");
Elliott Hughesb264f082012-04-06 17:10:10 -0700145 }
Elliott Hughesa2501992011-08-26 19:39:54 -0700146
Elliott Hughes0c9cd562011-08-12 10:59:29 -0700147 // But primitive types aren't allowed...
148 EXPECT_CLASS_NOT_FOUND("C");
149 EXPECT_CLASS_NOT_FOUND("K");
150}
151
Elliott Hughescdf53122011-08-19 15:46:09 -0700152#define EXPECT_EXCEPTION(exception_class) \
153 do { \
154 EXPECT_TRUE(env_->ExceptionCheck()); \
155 jthrowable exception = env_->ExceptionOccurred(); \
156 EXPECT_NE(static_cast<jthrowable>(NULL), exception); \
Elliott Hughescdf53122011-08-19 15:46:09 -0700157 env_->ExceptionClear(); \
Elliott Hughesa2501992011-08-26 19:39:54 -0700158 EXPECT_TRUE(env_->IsInstanceOf(exception, exception_class)); \
Elliott Hughescdf53122011-08-19 15:46:09 -0700159 } while (false)
160
161TEST_F(JniInternalTest, GetFieldID) {
162 jclass jlnsfe = env_->FindClass("java/lang/NoSuchFieldError");
163 ASSERT_TRUE(jlnsfe != NULL);
164 jclass c = env_->FindClass("java/lang/String");
165 ASSERT_TRUE(c != NULL);
166
167 // Wrong type.
168 jfieldID fid = env_->GetFieldID(c, "count", "J");
169 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
170 EXPECT_EXCEPTION(jlnsfe);
171
Ian Rogersb17d08b2011-09-02 16:16:49 -0700172 // Wrong type where type doesn't exist.
173 fid = env_->GetFieldID(c, "count", "Lrod/jane/freddy;");
174 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
175 EXPECT_EXCEPTION(jlnsfe);
176
Elliott Hughescdf53122011-08-19 15:46:09 -0700177 // Wrong name.
178 fid = env_->GetFieldID(c, "Count", "I");
179 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
180 EXPECT_EXCEPTION(jlnsfe);
181
182 // Good declared field lookup.
183 fid = env_->GetFieldID(c, "count", "I");
184 EXPECT_NE(static_cast<jfieldID>(NULL), fid);
185 EXPECT_TRUE(fid != NULL);
186 EXPECT_FALSE(env_->ExceptionCheck());
187
188 // Good superclass field lookup.
189 c = env_->FindClass("java/lang/StringBuilder");
190 fid = env_->GetFieldID(c, "count", "I");
191 EXPECT_NE(static_cast<jfieldID>(NULL), fid);
192 EXPECT_TRUE(fid != NULL);
193 EXPECT_FALSE(env_->ExceptionCheck());
194
195 // Not instance.
196 fid = env_->GetFieldID(c, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
197 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
198 EXPECT_EXCEPTION(jlnsfe);
199}
200
201TEST_F(JniInternalTest, GetStaticFieldID) {
202 jclass jlnsfe = env_->FindClass("java/lang/NoSuchFieldError");
203 ASSERT_TRUE(jlnsfe != NULL);
204 jclass c = env_->FindClass("java/lang/String");
205 ASSERT_TRUE(c != NULL);
206
207 // Wrong type.
208 jfieldID fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "J");
209 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
210 EXPECT_EXCEPTION(jlnsfe);
211
Ian Rogersb17d08b2011-09-02 16:16:49 -0700212 // Wrong type where type doesn't exist.
213 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "Lrod/jane/freddy;");
214 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
215 EXPECT_EXCEPTION(jlnsfe);
216
Elliott Hughescdf53122011-08-19 15:46:09 -0700217 // Wrong name.
218 fid = env_->GetStaticFieldID(c, "cASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
219 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
220 EXPECT_EXCEPTION(jlnsfe);
221
222 // Good declared field lookup.
223 fid = env_->GetStaticFieldID(c, "CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
224 EXPECT_NE(static_cast<jfieldID>(NULL), fid);
225 EXPECT_TRUE(fid != NULL);
226 EXPECT_FALSE(env_->ExceptionCheck());
227
228 // Not static.
229 fid = env_->GetStaticFieldID(c, "count", "I");
230 EXPECT_EQ(static_cast<jfieldID>(NULL), fid);
231 EXPECT_EXCEPTION(jlnsfe);
232}
233
Ian Rogers4dd71f12011-08-16 14:16:02 -0700234TEST_F(JniInternalTest, GetMethodID) {
235 jclass jlobject = env_->FindClass("java/lang/Object");
236 jclass jlstring = env_->FindClass("java/lang/String");
237 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
238
239 // Sanity check that no exceptions are pending
Elliott Hughescdf53122011-08-19 15:46:09 -0700240 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700241
242 // Check that java.lang.Object.foo() doesn't exist and NoSuchMethodError is
243 // a pending exception
244 jmethodID method = env_->GetMethodID(jlobject, "foo", "()V");
245 EXPECT_EQ(static_cast<jmethodID>(NULL), method);
Elliott Hughescdf53122011-08-19 15:46:09 -0700246 EXPECT_EXCEPTION(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700247
248 // Check that java.lang.Object.equals() does exist
Ian Rogers4dd71f12011-08-16 14:16:02 -0700249 method = env_->GetMethodID(jlobject, "equals", "(Ljava/lang/Object;)Z");
250 EXPECT_NE(static_cast<jmethodID>(NULL), method);
251 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700252
253 // Check that GetMethodID for java.lang.String.valueOf(int) fails as the
254 // method is static
255 method = env_->GetMethodID(jlstring, "valueOf", "(I)Ljava/lang/String;");
256 EXPECT_EQ(static_cast<jmethodID>(NULL), method);
Elliott Hughescdf53122011-08-19 15:46:09 -0700257 EXPECT_EXCEPTION(jlnsme);
Brian Carlstromea46f952013-07-30 01:26:50 -0700258
259 // Check that GetMethodID for java.lang.NoSuchMethodError.<init>(String) finds the constructor
260 method = env_->GetMethodID(jlnsme, "<init>", "(Ljava/lang/String;)V");
261 EXPECT_NE(static_cast<jmethodID>(NULL), method);
262 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700263}
264
265TEST_F(JniInternalTest, GetStaticMethodID) {
266 jclass jlobject = env_->FindClass("java/lang/Object");
267 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
268
269 // Sanity check that no exceptions are pending
Elliott Hughescdf53122011-08-19 15:46:09 -0700270 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700271
272 // Check that java.lang.Object.foo() doesn't exist and NoSuchMethodError is
273 // a pending exception
274 jmethodID method = env_->GetStaticMethodID(jlobject, "foo", "()V");
275 EXPECT_EQ(static_cast<jmethodID>(NULL), method);
Elliott Hughescdf53122011-08-19 15:46:09 -0700276 EXPECT_EXCEPTION(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700277
278 // Check that GetStaticMethodID for java.lang.Object.equals(Object) fails as
279 // the method is not static
280 method = env_->GetStaticMethodID(jlobject, "equals", "(Ljava/lang/Object;)Z");
281 EXPECT_EQ(static_cast<jmethodID>(NULL), method);
Elliott Hughescdf53122011-08-19 15:46:09 -0700282 EXPECT_EXCEPTION(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700283
284 // Check that java.lang.String.valueOf(int) does exist
Ian Rogers4dd71f12011-08-16 14:16:02 -0700285 jclass jlstring = env_->FindClass("java/lang/String");
286 method = env_->GetStaticMethodID(jlstring, "valueOf",
287 "(I)Ljava/lang/String;");
288 EXPECT_NE(static_cast<jmethodID>(NULL), method);
289 EXPECT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700290}
291
Elliott Hughescdf53122011-08-19 15:46:09 -0700292TEST_F(JniInternalTest, FromReflectedField_ToReflectedField) {
293 jclass jlrField = env_->FindClass("java/lang/reflect/Field");
294 jclass c = env_->FindClass("java/lang/String");
295 ASSERT_TRUE(c != NULL);
296 jfieldID fid = env_->GetFieldID(c, "count", "I");
297 ASSERT_TRUE(fid != NULL);
298 // Turn the fid into a java.lang.reflect.Field...
299 jobject field = env_->ToReflectedField(c, fid, JNI_FALSE);
300 ASSERT_TRUE(c != NULL);
301 ASSERT_TRUE(env_->IsInstanceOf(field, jlrField));
302 // ...and back again.
303 jfieldID fid2 = env_->FromReflectedField(field);
304 ASSERT_TRUE(fid2 != NULL);
Brian Carlstromea46f952013-07-30 01:26:50 -0700305 // Make sure we can actually use it.
306 jstring s = env_->NewStringUTF("poop");
307 ASSERT_EQ(4, env_->GetIntField(s, fid2));
Elliott Hughescdf53122011-08-19 15:46:09 -0700308}
309
310TEST_F(JniInternalTest, FromReflectedMethod_ToReflectedMethod) {
311 jclass jlrMethod = env_->FindClass("java/lang/reflect/Method");
312 jclass c = env_->FindClass("java/lang/String");
313 ASSERT_TRUE(c != NULL);
314 jmethodID mid = env_->GetMethodID(c, "length", "()I");
315 ASSERT_TRUE(mid != NULL);
316 // Turn the mid into a java.lang.reflect.Method...
317 jobject method = env_->ToReflectedMethod(c, mid, JNI_FALSE);
318 ASSERT_TRUE(c != NULL);
319 ASSERT_TRUE(env_->IsInstanceOf(method, jlrMethod));
320 // ...and back again.
321 jmethodID mid2 = env_->FromReflectedMethod(method);
322 ASSERT_TRUE(mid2 != NULL);
Brian Carlstromea46f952013-07-30 01:26:50 -0700323 // Make sure we can actually use it.
324 jstring s = env_->NewStringUTF("poop");
325 // TODO: this should return 4, but the runtime skips the method
326 // invoke because the runtime isn't started. In the future it would
327 // be nice to use interpretter for things like this. This still does
328 // validate that we have a sane jmethodID value.
329 ASSERT_EQ(0, env_->CallIntMethod(s, mid2));
Elliott Hughescdf53122011-08-19 15:46:09 -0700330}
331
Elliott Hughes5174fe62011-08-23 15:12:35 -0700332void BogusMethod() {
333 // You can't pass NULL function pointers to RegisterNatives.
334}
335
Ian Rogers4dd71f12011-08-16 14:16:02 -0700336TEST_F(JniInternalTest, RegisterNatives) {
337 jclass jlobject = env_->FindClass("java/lang/Object");
338 jclass jlnsme = env_->FindClass("java/lang/NoSuchMethodError");
339
340 // Sanity check that no exceptions are pending
Elliott Hughescdf53122011-08-19 15:46:09 -0700341 ASSERT_FALSE(env_->ExceptionCheck());
Ian Rogers4dd71f12011-08-16 14:16:02 -0700342
343 // Check that registering to a non-existent java.lang.Object.foo() causes a
344 // NoSuchMethodError
345 {
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700346 JNINativeMethod methods[] = { { "foo", "()V", NULL } };
Ian Rogers4dd71f12011-08-16 14:16:02 -0700347 env_->RegisterNatives(jlobject, methods, 1);
348 }
Elliott Hughescdf53122011-08-19 15:46:09 -0700349 EXPECT_EXCEPTION(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700350
351 // Check that registering non-native methods causes a NoSuchMethodError
352 {
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700353 JNINativeMethod methods[] = { { "equals", "(Ljava/lang/Object;)Z", NULL } };
Ian Rogers4dd71f12011-08-16 14:16:02 -0700354 env_->RegisterNatives(jlobject, methods, 1);
355 }
Elliott Hughescdf53122011-08-19 15:46:09 -0700356 EXPECT_EXCEPTION(jlnsme);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700357
358 // Check that registering native methods is successful
359 {
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700360 JNINativeMethod methods[] = { { "notify", "()V", reinterpret_cast<void*>(BogusMethod) } };
Ian Rogers4dd71f12011-08-16 14:16:02 -0700361 env_->RegisterNatives(jlobject, methods, 1);
362 }
363 EXPECT_FALSE(env_->ExceptionCheck());
Elliott Hughes5174fe62011-08-23 15:12:35 -0700364
365 env_->UnregisterNatives(jlobject);
Ian Rogers4dd71f12011-08-16 14:16:02 -0700366}
367
Brian Carlstromea46f952013-07-30 01:26:50 -0700368#define EXPECT_PRIMITIVE_ARRAY(new_fn, \
369 get_region_fn, \
370 set_region_fn, \
371 get_elements_fn, \
372 release_elements_fn, \
373 scalar_type, \
374 expected_class_descriptor) \
Ian Rogers1d99e452014-01-02 17:36:41 -0800375 { \
376 CheckJniAbortCatcher jni_abort_catcher; \
377 /* Allocate an negative sized array and check it has the right failure type. */ \
378 env_->new_fn(-1); \
379 jni_abort_catcher.Check("negative array length: -1"); \
380 env_->new_fn(std::numeric_limits<jint>::min()); \
381 jni_abort_catcher.Check("negative array length: -2147483648"); \
382 } \
Elliott Hughes814e4032011-08-23 12:07:56 -0700383 jsize size = 4; \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700384 \
Elliott Hughes814e4032011-08-23 12:07:56 -0700385 /* Allocate an array and check it has the right type and length. */ \
386 scalar_type ## Array a = env_->new_fn(size); \
387 EXPECT_TRUE(a != NULL); \
388 EXPECT_TRUE(env_->IsInstanceOf(a, env_->FindClass(expected_class_descriptor))); \
389 EXPECT_EQ(size, env_->GetArrayLength(a)); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700390 \
391 /* GetPrimitiveArrayRegion/SetPrimitiveArrayRegion */ \
Elliott Hughes814e4032011-08-23 12:07:56 -0700392 /* AIOOBE for negative start offset. */ \
393 env_->get_region_fn(a, -1, 1, NULL); \
394 EXPECT_EXCEPTION(aioobe_); \
395 env_->set_region_fn(a, -1, 1, NULL); \
396 EXPECT_EXCEPTION(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700397 \
Elliott Hughes814e4032011-08-23 12:07:56 -0700398 /* AIOOBE for negative length. */ \
399 env_->get_region_fn(a, 0, -1, NULL); \
400 EXPECT_EXCEPTION(aioobe_); \
401 env_->set_region_fn(a, 0, -1, NULL); \
402 EXPECT_EXCEPTION(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700403 \
Elliott Hughes814e4032011-08-23 12:07:56 -0700404 /* AIOOBE for buffer overrun. */ \
405 env_->get_region_fn(a, size - 1, size, NULL); \
406 EXPECT_EXCEPTION(aioobe_); \
407 env_->set_region_fn(a, size - 1, size, NULL); \
408 EXPECT_EXCEPTION(aioobe_); \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700409 \
410 /* It's okay for the buffer to be NULL as long as the length is 0. */ \
411 env_->get_region_fn(a, 2, 0, NULL); \
412 /* Even if the offset is invalid... */ \
413 env_->get_region_fn(a, 123, 0, NULL); \
414 EXPECT_EXCEPTION(aioobe_); \
415 \
416 /* It's okay for the buffer to be NULL as long as the length is 0. */ \
417 env_->set_region_fn(a, 2, 0, NULL); \
418 /* Even if the offset is invalid... */ \
419 env_->set_region_fn(a, 123, 0, NULL); \
420 EXPECT_EXCEPTION(aioobe_); \
421 \
Elliott Hughes814e4032011-08-23 12:07:56 -0700422 /* Prepare a couple of buffers. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -0700423 UniquePtr<scalar_type[]> src_buf(new scalar_type[size]); \
424 UniquePtr<scalar_type[]> dst_buf(new scalar_type[size]); \
Elliott Hughes814e4032011-08-23 12:07:56 -0700425 for (jsize i = 0; i < size; ++i) { src_buf[i] = scalar_type(i); } \
426 for (jsize i = 0; i < size; ++i) { dst_buf[i] = scalar_type(-1); } \
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700427 \
Elliott Hughes814e4032011-08-23 12:07:56 -0700428 /* Copy all of src_buf onto the heap. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -0700429 env_->set_region_fn(a, 0, size, &src_buf[0]); \
Elliott Hughes814e4032011-08-23 12:07:56 -0700430 /* Copy back only part. */ \
431 env_->get_region_fn(a, 1, size - 2, &dst_buf[1]); \
Brian Carlstromea46f952013-07-30 01:26:50 -0700432 EXPECT_NE(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
433 << "short copy equal"; \
Elliott Hughes814e4032011-08-23 12:07:56 -0700434 /* Copy the missing pieces. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -0700435 env_->get_region_fn(a, 0, 1, &dst_buf[0]); \
Elliott Hughes814e4032011-08-23 12:07:56 -0700436 env_->get_region_fn(a, size - 1, 1, &dst_buf[size - 1]); \
Brian Carlstromea46f952013-07-30 01:26:50 -0700437 EXPECT_EQ(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
438 << "fixed copy not equal"; \
Elliott Hughes814e4032011-08-23 12:07:56 -0700439 /* Copy back the whole array. */ \
Elliott Hughesee0fa762012-03-26 17:12:41 -0700440 env_->get_region_fn(a, 0, size, &dst_buf[0]); \
Brian Carlstromea46f952013-07-30 01:26:50 -0700441 EXPECT_EQ(memcmp(&src_buf[0], &dst_buf[0], size * sizeof(scalar_type)), 0) \
442 << "full copy not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -0700443 /* GetPrimitiveArrayCritical */ \
444 void* v = env_->GetPrimitiveArrayCritical(a, NULL); \
Brian Carlstromea46f952013-07-30 01:26:50 -0700445 EXPECT_EQ(memcmp(&src_buf[0], v, size * sizeof(scalar_type)), 0) \
446 << "GetPrimitiveArrayCritical not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -0700447 env_->ReleasePrimitiveArrayCritical(a, v, 0); \
448 /* GetXArrayElements */ \
449 scalar_type* xs = env_->get_elements_fn(a, NULL); \
Brian Carlstromea46f952013-07-30 01:26:50 -0700450 EXPECT_EQ(memcmp(&src_buf[0], xs, size * sizeof(scalar_type)), 0) \
451 << # get_elements_fn " not equal"; \
Elliott Hughes75770752011-08-24 17:52:38 -0700452 env_->release_elements_fn(a, xs, 0); \
Elliott Hughesbd935992011-08-22 11:59:34 -0700453
Elliott Hughes814e4032011-08-23 12:07:56 -0700454TEST_F(JniInternalTest, BooleanArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700455 EXPECT_PRIMITIVE_ARRAY(NewBooleanArray, GetBooleanArrayRegion, SetBooleanArrayRegion,
456 GetBooleanArrayElements, ReleaseBooleanArrayElements, jboolean, "[Z");
Elliott Hughes814e4032011-08-23 12:07:56 -0700457}
458TEST_F(JniInternalTest, ByteArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700459 EXPECT_PRIMITIVE_ARRAY(NewByteArray, GetByteArrayRegion, SetByteArrayRegion,
460 GetByteArrayElements, ReleaseByteArrayElements, jbyte, "[B");
Elliott Hughes814e4032011-08-23 12:07:56 -0700461}
462TEST_F(JniInternalTest, CharArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700463 EXPECT_PRIMITIVE_ARRAY(NewCharArray, GetCharArrayRegion, SetCharArrayRegion,
464 GetCharArrayElements, ReleaseCharArrayElements, jchar, "[C");
Elliott Hughes814e4032011-08-23 12:07:56 -0700465}
466TEST_F(JniInternalTest, DoubleArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700467 EXPECT_PRIMITIVE_ARRAY(NewDoubleArray, GetDoubleArrayRegion, SetDoubleArrayRegion,
468 GetDoubleArrayElements, ReleaseDoubleArrayElements, jdouble, "[D");
Elliott Hughes814e4032011-08-23 12:07:56 -0700469}
470TEST_F(JniInternalTest, FloatArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700471 EXPECT_PRIMITIVE_ARRAY(NewFloatArray, GetFloatArrayRegion, SetFloatArrayRegion,
472 GetFloatArrayElements, ReleaseFloatArrayElements, jfloat, "[F");
Elliott Hughes814e4032011-08-23 12:07:56 -0700473}
474TEST_F(JniInternalTest, IntArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700475 EXPECT_PRIMITIVE_ARRAY(NewIntArray, GetIntArrayRegion, SetIntArrayRegion,
476 GetIntArrayElements, ReleaseIntArrayElements, jint, "[I");
Elliott Hughes814e4032011-08-23 12:07:56 -0700477}
478TEST_F(JniInternalTest, LongArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700479 EXPECT_PRIMITIVE_ARRAY(NewLongArray, GetLongArrayRegion, SetLongArrayRegion,
480 GetLongArrayElements, ReleaseLongArrayElements, jlong, "[J");
Elliott Hughes814e4032011-08-23 12:07:56 -0700481}
482TEST_F(JniInternalTest, ShortArrays) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700483 EXPECT_PRIMITIVE_ARRAY(NewShortArray, GetShortArrayRegion, SetShortArrayRegion,
484 GetShortArrayElements, ReleaseShortArrayElements, jshort, "[S");
Elliott Hughesd8ddfd52011-08-15 14:32:53 -0700485}
486
Elliott Hughesf2682d52011-08-15 16:37:04 -0700487TEST_F(JniInternalTest, NewObjectArray) {
Elliott Hughesbd935992011-08-22 11:59:34 -0700488 jclass element_class = env_->FindClass("java/lang/String");
Ian Rogers1d99e452014-01-02 17:36:41 -0800489 ASSERT_TRUE(element_class != nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -0700490 jclass array_class = env_->FindClass("[Ljava/lang/String;");
Ian Rogers1d99e452014-01-02 17:36:41 -0800491 ASSERT_TRUE(array_class != nullptr);
Elliott Hughesf2682d52011-08-15 16:37:04 -0700492
Ian Rogers1d99e452014-01-02 17:36:41 -0800493 jobjectArray a = env_->NewObjectArray(0, element_class, nullptr);
494 EXPECT_TRUE(a != nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -0700495 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
496 EXPECT_EQ(0, env_->GetArrayLength(a));
497
Ian Rogers1d99e452014-01-02 17:36:41 -0800498 a = env_->NewObjectArray(1, element_class, nullptr);
499 EXPECT_TRUE(a != nullptr);
Elliott Hughesbd935992011-08-22 11:59:34 -0700500 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
501 EXPECT_EQ(1, env_->GetArrayLength(a));
Ian Rogers1d99e452014-01-02 17:36:41 -0800502 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 0), nullptr));
503}
504
505TEST_F(JniInternalTest, NewObjectArrayWithNegativeLength) {
506 jclass element_class = env_->FindClass("java/lang/String");
507 ASSERT_TRUE(element_class != nullptr);
508 jclass array_class = env_->FindClass("[Ljava/lang/String;");
509 ASSERT_TRUE(array_class != nullptr);
510 CheckJniAbortCatcher jni_abort_catcher;
511
512 env_->NewObjectArray(-1, element_class, nullptr);
513 jni_abort_catcher.Check("negative array length: -1");
514
515 env_->NewObjectArray(std::numeric_limits<jint>::min(), element_class, nullptr);
516 jni_abort_catcher.Check("negative array length: -2147483648");
517}
518
519TEST_F(JniInternalTest, NewObjectArrayWithPrimitiveClasses) {
520 const char* primitive_descriptors = "VZBSCIJFD";
521 const char* primitive_names[] = {
522 "void", "boolean", "byte", "short", "char", "int", "long", "float", "double"
523 };
524 ASSERT_EQ(strlen(primitive_descriptors), arraysize(primitive_names));
525
526 CheckJniAbortCatcher jni_abort_catcher;
527 for (size_t i = 0; i < strlen(primitive_descriptors); ++i) {
528 jclass primitive_class = GetPrimitiveClass(primitive_descriptors[i]);
529 env_->NewObjectArray(1, primitive_class, nullptr);
530 std::string error_msg(StringPrintf("not an object type: %s", primitive_names[i]));
531 jni_abort_catcher.Check(error_msg.c_str());
532 }
533}
534
535TEST_F(JniInternalTest, NewObjectArrayWithInitialValue) {
536 jclass element_class = env_->FindClass("java/lang/String");
537 ASSERT_TRUE(element_class != nullptr);
538 jclass array_class = env_->FindClass("[Ljava/lang/String;");
539 ASSERT_TRUE(array_class != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -0700540
541 jstring s = env_->NewStringUTF("poop");
Ian Rogers1d99e452014-01-02 17:36:41 -0800542 jobjectArray a = env_->NewObjectArray(2, element_class, s);
543 EXPECT_TRUE(a != nullptr);
Elliott Hughes75770752011-08-24 17:52:38 -0700544 EXPECT_TRUE(env_->IsInstanceOf(a, array_class));
545 EXPECT_EQ(2, env_->GetArrayLength(a));
546 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 0), s));
547 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(a, 1), s));
Ian Rogers1d99e452014-01-02 17:36:41 -0800548
549 // Attempt to incorrect create an array of strings with initial value of string arrays.
550 CheckJniAbortCatcher jni_abort_catcher;
551 env_->NewObjectArray(2, element_class, a);
552 jni_abort_catcher.Check("cannot assign object of type 'java.lang.String[]' to array with element "
553 "type of 'java.lang.String'");
Elliott Hughesbd935992011-08-22 11:59:34 -0700554}
555
556TEST_F(JniInternalTest, GetArrayLength) {
557 // Already tested in NewObjectArray/NewPrimitiveArray.
Elliott Hughes8a26c5c2011-08-15 18:35:43 -0700558}
559
Elliott Hughes37f7a402011-08-22 18:56:01 -0700560TEST_F(JniInternalTest, GetObjectClass) {
561 jclass string_class = env_->FindClass("java/lang/String");
562 ASSERT_TRUE(string_class != NULL);
563 jclass class_class = env_->FindClass("java/lang/Class");
564 ASSERT_TRUE(class_class != NULL);
565
566 jstring s = env_->NewStringUTF("poop");
567 jclass c = env_->GetObjectClass(s);
568 ASSERT_TRUE(env_->IsSameObject(string_class, c));
569
570 jclass c2 = env_->GetObjectClass(c);
571 ASSERT_TRUE(env_->IsSameObject(class_class, env_->GetObjectClass(c2)));
572}
573
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700574TEST_F(JniInternalTest, GetSuperclass) {
575 jclass object_class = env_->FindClass("java/lang/Object");
576 ASSERT_TRUE(object_class != NULL);
577 jclass string_class = env_->FindClass("java/lang/String");
578 ASSERT_TRUE(string_class != NULL);
Ian Rogersdc180202012-01-29 14:47:29 -0800579 jclass runnable_interface = env_->FindClass("java/lang/Runnable");
580 ASSERT_TRUE(runnable_interface != NULL);
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700581 ASSERT_TRUE(env_->IsSameObject(object_class, env_->GetSuperclass(string_class)));
582 ASSERT_TRUE(env_->GetSuperclass(object_class) == NULL);
Ian Rogersdc180202012-01-29 14:47:29 -0800583 ASSERT_TRUE(env_->IsSameObject(object_class, env_->GetSuperclass(runnable_interface)));
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700584}
585
Elliott Hughes37f7a402011-08-22 18:56:01 -0700586TEST_F(JniInternalTest, IsAssignableFrom) {
587 jclass object_class = env_->FindClass("java/lang/Object");
588 ASSERT_TRUE(object_class != NULL);
589 jclass string_class = env_->FindClass("java/lang/String");
590 ASSERT_TRUE(string_class != NULL);
591
592 ASSERT_TRUE(env_->IsAssignableFrom(object_class, string_class));
593 ASSERT_FALSE(env_->IsAssignableFrom(string_class, object_class));
594}
595
Elliott Hughesb465ab02011-08-24 11:21:21 -0700596TEST_F(JniInternalTest, GetObjectRefType) {
597 jclass local = env_->FindClass("java/lang/Object");
598 ASSERT_TRUE(local != NULL);
599 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(local));
600
601 jobject global = env_->NewGlobalRef(local);
602 EXPECT_EQ(JNIGlobalRefType, env_->GetObjectRefType(global));
603
604 jweak weak_global = env_->NewWeakGlobalRef(local);
605 EXPECT_EQ(JNIWeakGlobalRefType, env_->GetObjectRefType(weak_global));
606
607 jobject invalid = reinterpret_cast<jobject>(this);
608 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(invalid));
609
610 // TODO: invoke a native method and test that its arguments are considered local references.
611}
612
Mathieu Chartier08599992013-12-20 17:17:55 -0800613TEST_F(JniInternalTest, StaleWeakGlobal) {
614 jclass java_lang_Class = env_->FindClass("java/lang/Class");
615 ASSERT_TRUE(java_lang_Class != NULL);
616 jobjectArray local_ref = env_->NewObjectArray(1, java_lang_Class, NULL);
617 ASSERT_TRUE(local_ref != NULL);
618 jweak weak_global = env_->NewWeakGlobalRef(local_ref);
619 ASSERT_TRUE(weak_global != NULL);
620 env_->DeleteLocalRef(local_ref);
621 Runtime::Current()->GetHeap()->CollectGarbage(false); // GC should clear the weak global.
622 jobject new_global_ref = env_->NewGlobalRef(weak_global);
623 EXPECT_TRUE(new_global_ref == NULL);
624 jobject new_local_ref = env_->NewLocalRef(weak_global);
625 EXPECT_TRUE(new_local_ref == NULL);
626}
627
Elliott Hughes8a26c5c2011-08-15 18:35:43 -0700628TEST_F(JniInternalTest, NewStringUTF) {
629 EXPECT_TRUE(env_->NewStringUTF(NULL) == NULL);
Elliott Hughes814e4032011-08-23 12:07:56 -0700630 jstring s;
631
632 s = env_->NewStringUTF("");
633 EXPECT_TRUE(s != NULL);
634 EXPECT_EQ(0, env_->GetStringLength(s));
635 EXPECT_EQ(0, env_->GetStringUTFLength(s));
636 s = env_->NewStringUTF("hello");
637 EXPECT_TRUE(s != NULL);
638 EXPECT_EQ(5, env_->GetStringLength(s));
639 EXPECT_EQ(5, env_->GetStringUTFLength(s));
640
Elliott Hughes8a26c5c2011-08-15 18:35:43 -0700641 // TODO: check some non-ASCII strings.
Elliott Hughesf2682d52011-08-15 16:37:04 -0700642}
643
Elliott Hughes814e4032011-08-23 12:07:56 -0700644TEST_F(JniInternalTest, NewString) {
Elliott Hughes814e4032011-08-23 12:07:56 -0700645 jchar chars[] = { 'h', 'i' };
646 jstring s;
647 s = env_->NewString(chars, 0);
648 EXPECT_TRUE(s != NULL);
649 EXPECT_EQ(0, env_->GetStringLength(s));
650 EXPECT_EQ(0, env_->GetStringUTFLength(s));
651 s = env_->NewString(chars, 2);
652 EXPECT_TRUE(s != NULL);
653 EXPECT_EQ(2, env_->GetStringLength(s));
654 EXPECT_EQ(2, env_->GetStringUTFLength(s));
655
656 // TODO: check some non-ASCII strings.
657}
658
Jesse Wilson25e79a52011-11-18 15:31:58 -0500659TEST_F(JniInternalTest, NewStringNullCharsZeroLength) {
Ian Rogers1d99e452014-01-02 17:36:41 -0800660 jstring s = env_->NewString(nullptr, 0);
661 EXPECT_TRUE(s != nullptr);
Jesse Wilson25e79a52011-11-18 15:31:58 -0500662 EXPECT_EQ(0, env_->GetStringLength(s));
663}
664
Ian Rogers1d99e452014-01-02 17:36:41 -0800665TEST_F(JniInternalTest, NewStringNullCharsNonzeroLength) {
666 CheckJniAbortCatcher jni_abort_catcher;
667 env_->NewString(nullptr, 1);
668 jni_abort_catcher.Check("chars == null && char_count > 0");
669}
670
671TEST_F(JniInternalTest, NewStringNegativeLength) {
672 CheckJniAbortCatcher jni_abort_catcher;
673 env_->NewString(nullptr, -1);
674 jni_abort_catcher.Check("char_count < 0: -1");
675 env_->NewString(nullptr, std::numeric_limits<jint>::min());
676 jni_abort_catcher.Check("char_count < 0: -2147483648");
Jesse Wilson25e79a52011-11-18 15:31:58 -0500677}
678
Elliott Hughesb465ab02011-08-24 11:21:21 -0700679TEST_F(JniInternalTest, GetStringLength_GetStringUTFLength) {
680 // Already tested in the NewString/NewStringUTF tests.
681}
682
683TEST_F(JniInternalTest, GetStringRegion_GetStringUTFRegion) {
684 jstring s = env_->NewStringUTF("hello");
685 ASSERT_TRUE(s != NULL);
686
687 env_->GetStringRegion(s, -1, 0, NULL);
688 EXPECT_EXCEPTION(sioobe_);
689 env_->GetStringRegion(s, 0, -1, NULL);
690 EXPECT_EXCEPTION(sioobe_);
691 env_->GetStringRegion(s, 0, 10, NULL);
692 EXPECT_EXCEPTION(sioobe_);
693 env_->GetStringRegion(s, 10, 1, NULL);
694 EXPECT_EXCEPTION(sioobe_);
695
696 jchar chars[4] = { 'x', 'x', 'x', 'x' };
697 env_->GetStringRegion(s, 1, 2, &chars[1]);
698 EXPECT_EQ('x', chars[0]);
699 EXPECT_EQ('e', chars[1]);
700 EXPECT_EQ('l', chars[2]);
701 EXPECT_EQ('x', chars[3]);
702
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700703 // It's okay for the buffer to be NULL as long as the length is 0.
704 env_->GetStringRegion(s, 2, 0, NULL);
705 // Even if the offset is invalid...
706 env_->GetStringRegion(s, 123, 0, NULL);
707 EXPECT_EXCEPTION(sioobe_);
708
Elliott Hughesb465ab02011-08-24 11:21:21 -0700709 env_->GetStringUTFRegion(s, -1, 0, NULL);
710 EXPECT_EXCEPTION(sioobe_);
711 env_->GetStringUTFRegion(s, 0, -1, NULL);
712 EXPECT_EXCEPTION(sioobe_);
713 env_->GetStringUTFRegion(s, 0, 10, NULL);
714 EXPECT_EXCEPTION(sioobe_);
715 env_->GetStringUTFRegion(s, 10, 1, NULL);
716 EXPECT_EXCEPTION(sioobe_);
717
718 char bytes[4] = { 'x', 'x', 'x', 'x' };
719 env_->GetStringUTFRegion(s, 1, 2, &bytes[1]);
720 EXPECT_EQ('x', bytes[0]);
721 EXPECT_EQ('e', bytes[1]);
722 EXPECT_EQ('l', bytes[2]);
723 EXPECT_EQ('x', bytes[3]);
Elliott Hughesd7d7f6e2013-09-18 12:00:45 -0700724
725 // It's okay for the buffer to be NULL as long as the length is 0.
726 env_->GetStringUTFRegion(s, 2, 0, NULL);
727 // Even if the offset is invalid...
728 env_->GetStringUTFRegion(s, 123, 0, NULL);
729 EXPECT_EXCEPTION(sioobe_);
Elliott Hughesb465ab02011-08-24 11:21:21 -0700730}
731
Elliott Hughes75770752011-08-24 17:52:38 -0700732TEST_F(JniInternalTest, GetStringUTFChars_ReleaseStringUTFChars) {
Elliott Hughes3f6635a2012-06-19 13:37:49 -0700733 // Passing in a NULL jstring is ignored normally, but caught by -Xcheck:jni.
Elliott Hughesb264f082012-04-06 17:10:10 -0700734 {
Elliott Hughesb264f082012-04-06 17:10:10 -0700735 CheckJniAbortCatcher check_jni_abort_catcher;
736 EXPECT_TRUE(env_->GetStringUTFChars(NULL, NULL) == NULL);
Elliott Hughes56ef0422012-06-19 14:35:04 -0700737 check_jni_abort_catcher.Check("GetStringUTFChars received null jstring");
Elliott Hughesb264f082012-04-06 17:10:10 -0700738 }
Elliott Hughes75770752011-08-24 17:52:38 -0700739
740 jstring s = env_->NewStringUTF("hello");
741 ASSERT_TRUE(s != NULL);
742
743 const char* utf = env_->GetStringUTFChars(s, NULL);
744 EXPECT_STREQ("hello", utf);
745 env_->ReleaseStringUTFChars(s, utf);
746
747 jboolean is_copy = JNI_FALSE;
748 utf = env_->GetStringUTFChars(s, &is_copy);
749 EXPECT_EQ(JNI_TRUE, is_copy);
750 EXPECT_STREQ("hello", utf);
751 env_->ReleaseStringUTFChars(s, utf);
752}
753
754TEST_F(JniInternalTest, GetStringChars_ReleaseStringChars) {
755 jstring s = env_->NewStringUTF("hello");
756 ASSERT_TRUE(s != NULL);
757
758 jchar expected[] = { 'h', 'e', 'l', 'l', 'o' };
759 const jchar* chars = env_->GetStringChars(s, NULL);
760 EXPECT_EQ(expected[0], chars[0]);
761 EXPECT_EQ(expected[1], chars[1]);
762 EXPECT_EQ(expected[2], chars[2]);
763 EXPECT_EQ(expected[3], chars[3]);
764 EXPECT_EQ(expected[4], chars[4]);
765 env_->ReleaseStringChars(s, chars);
766
767 jboolean is_copy = JNI_FALSE;
768 chars = env_->GetStringChars(s, &is_copy);
Mathieu Chartier590fee92013-09-13 13:46:47 -0700769 EXPECT_EQ(JNI_TRUE, is_copy);
Elliott Hughes75770752011-08-24 17:52:38 -0700770 EXPECT_EQ(expected[0], chars[0]);
771 EXPECT_EQ(expected[1], chars[1]);
772 EXPECT_EQ(expected[2], chars[2]);
773 EXPECT_EQ(expected[3], chars[3]);
774 EXPECT_EQ(expected[4], chars[4]);
775 env_->ReleaseStringChars(s, chars);
776}
777
778TEST_F(JniInternalTest, GetStringCritical_ReleaseStringCritical) {
779 jstring s = env_->NewStringUTF("hello");
780 ASSERT_TRUE(s != NULL);
781
782 jchar expected[] = { 'h', 'e', 'l', 'l', 'o' };
783 const jchar* chars = env_->GetStringCritical(s, NULL);
784 EXPECT_EQ(expected[0], chars[0]);
785 EXPECT_EQ(expected[1], chars[1]);
786 EXPECT_EQ(expected[2], chars[2]);
787 EXPECT_EQ(expected[3], chars[3]);
788 EXPECT_EQ(expected[4], chars[4]);
789 env_->ReleaseStringCritical(s, chars);
790
791 jboolean is_copy = JNI_FALSE;
792 chars = env_->GetStringCritical(s, &is_copy);
Mathieu Chartier590fee92013-09-13 13:46:47 -0700793 // TODO: Fix GetStringCritical to use the same mechanism as GetPrimitiveArrayElementsCritical.
794 EXPECT_EQ(JNI_TRUE, is_copy);
Elliott Hughes75770752011-08-24 17:52:38 -0700795 EXPECT_EQ(expected[0], chars[0]);
796 EXPECT_EQ(expected[1], chars[1]);
797 EXPECT_EQ(expected[2], chars[2]);
798 EXPECT_EQ(expected[3], chars[3]);
799 EXPECT_EQ(expected[4], chars[4]);
800 env_->ReleaseStringCritical(s, chars);
801}
802
Elliott Hughes814e4032011-08-23 12:07:56 -0700803TEST_F(JniInternalTest, GetObjectArrayElement_SetObjectArrayElement) {
Elliott Hughesb264f082012-04-06 17:10:10 -0700804 jclass java_lang_Class = env_->FindClass("java/lang/Class");
805 ASSERT_TRUE(java_lang_Class != NULL);
Elliott Hughes289da822011-08-16 10:11:20 -0700806
Elliott Hughesb264f082012-04-06 17:10:10 -0700807 jobjectArray array = env_->NewObjectArray(1, java_lang_Class, NULL);
Elliott Hughes289da822011-08-16 10:11:20 -0700808 EXPECT_TRUE(array != NULL);
Elliott Hughes814e4032011-08-23 12:07:56 -0700809 EXPECT_TRUE(env_->GetObjectArrayElement(array, 0) == NULL);
Elliott Hughesb264f082012-04-06 17:10:10 -0700810 env_->SetObjectArrayElement(array, 0, java_lang_Class);
811 EXPECT_TRUE(env_->IsSameObject(env_->GetObjectArrayElement(array, 0), java_lang_Class));
Elliott Hughesa5b897e2011-08-16 11:33:06 -0700812
813 // ArrayIndexOutOfBounds for negative index.
Elliott Hughesb264f082012-04-06 17:10:10 -0700814 env_->SetObjectArrayElement(array, -1, java_lang_Class);
Elliott Hughes814e4032011-08-23 12:07:56 -0700815 EXPECT_EXCEPTION(aioobe_);
Elliott Hughesa5b897e2011-08-16 11:33:06 -0700816
817 // ArrayIndexOutOfBounds for too-large index.
Elliott Hughesb264f082012-04-06 17:10:10 -0700818 env_->SetObjectArrayElement(array, 1, java_lang_Class);
Elliott Hughes814e4032011-08-23 12:07:56 -0700819 EXPECT_EXCEPTION(aioobe_);
Elliott Hughesa5b897e2011-08-16 11:33:06 -0700820
Elliott Hughesb264f082012-04-06 17:10:10 -0700821 // ArrayStoreException thrown for bad types.
822 env_->SetObjectArrayElement(array, 0, env_->NewStringUTF("not a jclass!"));
823 EXPECT_EXCEPTION(ase_);
Elliott Hughes289da822011-08-16 10:11:20 -0700824}
825
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700826#define EXPECT_STATIC_PRIMITIVE_FIELD(type, field_name, sig, value1, value2) \
827 do { \
828 jfieldID fid = env_->GetStaticFieldID(c, field_name, sig); \
829 EXPECT_TRUE(fid != NULL); \
830 env_->SetStatic ## type ## Field(c, fid, value1); \
Brian Carlstrom2e3d1b22012-01-09 18:01:56 -0800831 EXPECT_TRUE(value1 == env_->GetStatic ## type ## Field(c, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700832 env_->SetStatic ## type ## Field(c, fid, value2); \
Brian Carlstrom2e3d1b22012-01-09 18:01:56 -0800833 EXPECT_TRUE(value2 == env_->GetStatic ## type ## Field(c, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700834 } while (false)
835
836#define EXPECT_PRIMITIVE_FIELD(instance, type, field_name, sig, value1, value2) \
837 do { \
838 jfieldID fid = env_->GetFieldID(c, field_name, sig); \
839 EXPECT_TRUE(fid != NULL); \
840 env_->Set ## type ## Field(instance, fid, value1); \
Brian Carlstrom2e3d1b22012-01-09 18:01:56 -0800841 EXPECT_TRUE(value1 == env_->Get ## type ## Field(instance, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700842 env_->Set ## type ## Field(instance, fid, value2); \
Brian Carlstrom2e3d1b22012-01-09 18:01:56 -0800843 EXPECT_TRUE(value2 == env_->Get ## type ## Field(instance, fid)); \
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700844 } while (false)
845
846
847TEST_F(JniInternalTest, GetPrimitiveField_SetPrimitiveField) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800848 TEST_DISABLED_FOR_PORTABLE();
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700849 Thread::Current()->TransitionFromSuspendedToRunnable();
850 LoadDex("AllFields");
Brian Carlstrombd86bcc2013-03-10 20:26:16 -0700851 bool started = runtime_->Start();
852 CHECK(started);
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700853
854 jclass c = env_->FindClass("AllFields");
855 ASSERT_TRUE(c != NULL);
856 jobject o = env_->AllocObject(c);
857 ASSERT_TRUE(o != NULL);
858
859 EXPECT_STATIC_PRIMITIVE_FIELD(Boolean, "sZ", "Z", true, false);
860 EXPECT_STATIC_PRIMITIVE_FIELD(Byte, "sB", "B", 1, 2);
861 EXPECT_STATIC_PRIMITIVE_FIELD(Char, "sC", "C", 'a', 'b');
862 EXPECT_STATIC_PRIMITIVE_FIELD(Double, "sD", "D", 1.0, 2.0);
863 EXPECT_STATIC_PRIMITIVE_FIELD(Float, "sF", "F", 1.0, 2.0);
864 EXPECT_STATIC_PRIMITIVE_FIELD(Int, "sI", "I", 1, 2);
865 EXPECT_STATIC_PRIMITIVE_FIELD(Long, "sJ", "J", 1, 2);
866 EXPECT_STATIC_PRIMITIVE_FIELD(Short, "sS", "S", 1, 2);
867
868 EXPECT_PRIMITIVE_FIELD(o, Boolean, "iZ", "Z", true, false);
869 EXPECT_PRIMITIVE_FIELD(o, Byte, "iB", "B", 1, 2);
870 EXPECT_PRIMITIVE_FIELD(o, Char, "iC", "C", 'a', 'b');
871 EXPECT_PRIMITIVE_FIELD(o, Double, "iD", "D", 1.0, 2.0);
872 EXPECT_PRIMITIVE_FIELD(o, Float, "iF", "F", 1.0, 2.0);
873 EXPECT_PRIMITIVE_FIELD(o, Int, "iI", "I", 1, 2);
874 EXPECT_PRIMITIVE_FIELD(o, Long, "iJ", "J", 1, 2);
875 EXPECT_PRIMITIVE_FIELD(o, Short, "iS", "S", 1, 2);
876}
877
878TEST_F(JniInternalTest, GetObjectField_SetObjectField) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800879 TEST_DISABLED_FOR_PORTABLE();
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700880 Thread::Current()->TransitionFromSuspendedToRunnable();
881 LoadDex("AllFields");
Brian Carlstrom25c33252011-09-18 15:58:35 -0700882 runtime_->Start();
Elliott Hughes885c3bd2011-08-22 16:59:20 -0700883
884 jclass c = env_->FindClass("AllFields");
885 ASSERT_TRUE(c != NULL);
886 jobject o = env_->AllocObject(c);
887 ASSERT_TRUE(o != NULL);
888
889 jstring s1 = env_->NewStringUTF("hello");
890 ASSERT_TRUE(s1 != NULL);
891 jstring s2 = env_->NewStringUTF("world");
892 ASSERT_TRUE(s2 != NULL);
893
894 jfieldID s_fid = env_->GetStaticFieldID(c, "sObject", "Ljava/lang/Object;");
895 ASSERT_TRUE(s_fid != NULL);
896 jfieldID i_fid = env_->GetFieldID(c, "iObject", "Ljava/lang/Object;");
897 ASSERT_TRUE(i_fid != NULL);
898
899 env_->SetStaticObjectField(c, s_fid, s1);
900 ASSERT_TRUE(env_->IsSameObject(s1, env_->GetStaticObjectField(c, s_fid)));
901 env_->SetStaticObjectField(c, s_fid, s2);
902 ASSERT_TRUE(env_->IsSameObject(s2, env_->GetStaticObjectField(c, s_fid)));
903
904 env_->SetObjectField(o, i_fid, s1);
905 ASSERT_TRUE(env_->IsSameObject(s1, env_->GetObjectField(o, i_fid)));
906 env_->SetObjectField(o, i_fid, s2);
907 ASSERT_TRUE(env_->IsSameObject(s2, env_->GetObjectField(o, i_fid)));
908}
909
Elliott Hughes18c07532011-08-18 15:50:51 -0700910TEST_F(JniInternalTest, NewLocalRef_NULL) {
911 EXPECT_TRUE(env_->NewLocalRef(NULL) == NULL);
912}
913
914TEST_F(JniInternalTest, NewLocalRef) {
915 jstring s = env_->NewStringUTF("");
916 ASSERT_TRUE(s != NULL);
917 jobject o = env_->NewLocalRef(s);
918 EXPECT_TRUE(o != NULL);
919 EXPECT_TRUE(o != s);
920
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700921 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(o));
Elliott Hughes18c07532011-08-18 15:50:51 -0700922}
923
924TEST_F(JniInternalTest, DeleteLocalRef_NULL) {
925 env_->DeleteLocalRef(NULL);
926}
927
928TEST_F(JniInternalTest, DeleteLocalRef) {
929 jstring s = env_->NewStringUTF("");
930 ASSERT_TRUE(s != NULL);
931 env_->DeleteLocalRef(s);
932
Elliott Hughes3f6635a2012-06-19 13:37:49 -0700933 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -0700934 {
Elliott Hughesb264f082012-04-06 17:10:10 -0700935 CheckJniAbortCatcher check_jni_abort_catcher;
936 env_->DeleteLocalRef(s);
Elliott Hughesa9137c62013-01-09 10:55:21 -0800937
Brian Carlstromea46f952013-07-30 01:26:50 -0700938 std::string expected(StringPrintf("native code passing in reference to "
939 "invalid local reference: %p", s));
Elliott Hughesa9137c62013-01-09 10:55:21 -0800940 check_jni_abort_catcher.Check(expected.c_str());
Elliott Hughesb264f082012-04-06 17:10:10 -0700941 }
Elliott Hughes18c07532011-08-18 15:50:51 -0700942
943 s = env_->NewStringUTF("");
944 ASSERT_TRUE(s != NULL);
945 jobject o = env_->NewLocalRef(s);
946 ASSERT_TRUE(o != NULL);
947
948 env_->DeleteLocalRef(s);
949 env_->DeleteLocalRef(o);
950}
951
Elliott Hughesaa836f72013-08-20 16:57:23 -0700952TEST_F(JniInternalTest, PushLocalFrame_10395422) {
953 // The JNI specification is ambiguous about whether the given capacity is to be interpreted as a
954 // maximum or as a minimum, but it seems like it's supposed to be a minimum, and that's how
955 // Android historically treated it, and it's how the RI treats it. It's also the more useful
956 // interpretation!
957 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(0));
958 env_->PopLocalFrame(NULL);
959
960 // Negative capacities are not allowed.
961 ASSERT_EQ(JNI_ERR, env_->PushLocalFrame(-1));
962
963 // And it's okay to have an upper limit. Ours is currently 512.
964 ASSERT_EQ(JNI_ERR, env_->PushLocalFrame(8192));
965}
966
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700967TEST_F(JniInternalTest, PushLocalFrame_PopLocalFrame) {
968 jobject original = env_->NewStringUTF("");
969 ASSERT_TRUE(original != NULL);
970
971 jobject outer;
972 jobject inner1, inner2;
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700973 ScopedObjectAccess soa(env_);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800974 mirror::Object* inner2_direct_pointer;
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700975 {
Elliott Hughesaa836f72013-08-20 16:57:23 -0700976 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(4));
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700977 outer = env_->NewLocalRef(original);
978
979 {
Elliott Hughesaa836f72013-08-20 16:57:23 -0700980 ASSERT_EQ(JNI_OK, env_->PushLocalFrame(4));
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700981 inner1 = env_->NewLocalRef(outer);
982 inner2 = env_->NewStringUTF("survivor");
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800983 inner2_direct_pointer = soa.Decode<mirror::Object*>(inner2);
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700984 env_->PopLocalFrame(inner2);
985 }
986
987 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(original));
988 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(outer));
989 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner1));
990
991 // Our local reference for the survivor is invalid because the survivor
992 // gets a new local reference...
993 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner2));
994 // ...but the survivor should be in the local reference table.
Brian Carlstrom4d571432012-05-16 00:21:41 -0700995 JNIEnvExt* env = reinterpret_cast<JNIEnvExt*>(env_);
996 EXPECT_TRUE(env->locals.ContainsDirectPointer(inner2_direct_pointer));
Elliott Hughes2ced6a52011-10-16 18:44:48 -0700997
998 env_->PopLocalFrame(NULL);
999 }
1000 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(original));
1001 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(outer));
1002 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner1));
1003 EXPECT_EQ(JNIInvalidRefType, env_->GetObjectRefType(inner2));
1004}
1005
Elliott Hughes18c07532011-08-18 15:50:51 -07001006TEST_F(JniInternalTest, NewGlobalRef_NULL) {
1007 EXPECT_TRUE(env_->NewGlobalRef(NULL) == NULL);
1008}
1009
1010TEST_F(JniInternalTest, NewGlobalRef) {
1011 jstring s = env_->NewStringUTF("");
1012 ASSERT_TRUE(s != NULL);
1013 jobject o = env_->NewGlobalRef(s);
1014 EXPECT_TRUE(o != NULL);
1015 EXPECT_TRUE(o != s);
1016
1017 // TODO: check that o is a global reference.
1018}
1019
1020TEST_F(JniInternalTest, DeleteGlobalRef_NULL) {
1021 env_->DeleteGlobalRef(NULL);
1022}
1023
1024TEST_F(JniInternalTest, DeleteGlobalRef) {
1025 jstring s = env_->NewStringUTF("");
1026 ASSERT_TRUE(s != NULL);
1027
1028 jobject o = env_->NewGlobalRef(s);
1029 ASSERT_TRUE(o != NULL);
1030 env_->DeleteGlobalRef(o);
1031
Elliott Hughes3f6635a2012-06-19 13:37:49 -07001032 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07001033 {
Elliott Hughesb264f082012-04-06 17:10:10 -07001034 CheckJniAbortCatcher check_jni_abort_catcher;
1035 env_->DeleteGlobalRef(o);
Elliott Hughesa9137c62013-01-09 10:55:21 -08001036
Brian Carlstromea46f952013-07-30 01:26:50 -07001037 std::string expected(StringPrintf("native code passing in reference to "
1038 "invalid global reference: %p", o));
Elliott Hughesa9137c62013-01-09 10:55:21 -08001039 check_jni_abort_catcher.Check(expected.c_str());
Elliott Hughesb264f082012-04-06 17:10:10 -07001040 }
Elliott Hughes18c07532011-08-18 15:50:51 -07001041
1042 jobject o1 = env_->NewGlobalRef(s);
1043 ASSERT_TRUE(o1 != NULL);
1044 jobject o2 = env_->NewGlobalRef(s);
1045 ASSERT_TRUE(o2 != NULL);
1046
1047 env_->DeleteGlobalRef(o1);
1048 env_->DeleteGlobalRef(o2);
1049}
1050
1051TEST_F(JniInternalTest, NewWeakGlobalRef_NULL) {
1052 EXPECT_TRUE(env_->NewWeakGlobalRef(NULL) == NULL);
1053}
1054
1055TEST_F(JniInternalTest, NewWeakGlobalRef) {
1056 jstring s = env_->NewStringUTF("");
1057 ASSERT_TRUE(s != NULL);
1058 jobject o = env_->NewWeakGlobalRef(s);
1059 EXPECT_TRUE(o != NULL);
1060 EXPECT_TRUE(o != s);
1061
1062 // TODO: check that o is a weak global reference.
1063}
1064
1065TEST_F(JniInternalTest, DeleteWeakGlobalRef_NULL) {
1066 env_->DeleteWeakGlobalRef(NULL);
1067}
1068
1069TEST_F(JniInternalTest, DeleteWeakGlobalRef) {
1070 jstring s = env_->NewStringUTF("");
1071 ASSERT_TRUE(s != NULL);
1072
1073 jobject o = env_->NewWeakGlobalRef(s);
1074 ASSERT_TRUE(o != NULL);
1075 env_->DeleteWeakGlobalRef(o);
1076
Elliott Hughes3f6635a2012-06-19 13:37:49 -07001077 // Currently, deleting an already-deleted reference is just a CheckJNI warning.
Elliott Hughesb264f082012-04-06 17:10:10 -07001078 {
Elliott Hughesb264f082012-04-06 17:10:10 -07001079 CheckJniAbortCatcher check_jni_abort_catcher;
1080 env_->DeleteWeakGlobalRef(o);
Elliott Hughesa9137c62013-01-09 10:55:21 -08001081
Brian Carlstromea46f952013-07-30 01:26:50 -07001082 std::string expected(StringPrintf("native code passing in reference to "
1083 "invalid weak global reference: %p", o));
Elliott Hughesa9137c62013-01-09 10:55:21 -08001084 check_jni_abort_catcher.Check(expected.c_str());
Elliott Hughesb264f082012-04-06 17:10:10 -07001085 }
Elliott Hughes18c07532011-08-18 15:50:51 -07001086
1087 jobject o1 = env_->NewWeakGlobalRef(s);
1088 ASSERT_TRUE(o1 != NULL);
1089 jobject o2 = env_->NewWeakGlobalRef(s);
1090 ASSERT_TRUE(o2 != NULL);
1091
1092 env_->DeleteWeakGlobalRef(o1);
1093 env_->DeleteWeakGlobalRef(o2);
1094}
1095
Elliott Hughes37f7a402011-08-22 18:56:01 -07001096TEST_F(JniInternalTest, Throw) {
1097 EXPECT_EQ(JNI_ERR, env_->Throw(NULL));
1098
1099 jclass exception_class = env_->FindClass("java/lang/RuntimeException");
1100 ASSERT_TRUE(exception_class != NULL);
1101 jthrowable exception = reinterpret_cast<jthrowable>(env_->AllocObject(exception_class));
1102 ASSERT_TRUE(exception != NULL);
1103
1104 EXPECT_EQ(JNI_OK, env_->Throw(exception));
1105 EXPECT_TRUE(env_->ExceptionCheck());
Elliott Hughesa2501992011-08-26 19:39:54 -07001106 jthrowable thrown_exception = env_->ExceptionOccurred();
Elliott Hughes37f7a402011-08-22 18:56:01 -07001107 env_->ExceptionClear();
Elliott Hughesa2501992011-08-26 19:39:54 -07001108 EXPECT_TRUE(env_->IsSameObject(exception, thrown_exception));
Elliott Hughes37f7a402011-08-22 18:56:01 -07001109}
1110
1111TEST_F(JniInternalTest, ThrowNew) {
1112 EXPECT_EQ(JNI_ERR, env_->Throw(NULL));
1113
1114 jclass exception_class = env_->FindClass("java/lang/RuntimeException");
1115 ASSERT_TRUE(exception_class != NULL);
1116
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07001117 jthrowable thrown_exception;
1118
Elliott Hughes37f7a402011-08-22 18:56:01 -07001119 EXPECT_EQ(JNI_OK, env_->ThrowNew(exception_class, "hello world"));
1120 EXPECT_TRUE(env_->ExceptionCheck());
Elliott Hughes5cb5ad22011-10-02 12:13:39 -07001121 thrown_exception = env_->ExceptionOccurred();
1122 env_->ExceptionClear();
1123 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, exception_class));
1124
1125 EXPECT_EQ(JNI_OK, env_->ThrowNew(exception_class, NULL));
1126 EXPECT_TRUE(env_->ExceptionCheck());
1127 thrown_exception = env_->ExceptionOccurred();
Elliott Hughes37f7a402011-08-22 18:56:01 -07001128 env_->ExceptionClear();
Elliott Hughesa2501992011-08-26 19:39:54 -07001129 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, exception_class));
Elliott Hughes37f7a402011-08-22 18:56:01 -07001130}
1131
Ian Rogers1d99e452014-01-02 17:36:41 -08001132TEST_F(JniInternalTest, NewDirectBuffer_GetDirectBufferAddress_GetDirectBufferCapacity) {
1133 // Start runtime.
1134 Thread* self = Thread::Current();
1135 self->TransitionFromSuspendedToRunnable();
1136 MakeExecutable(nullptr, "java.lang.Class");
1137 MakeExecutable(nullptr, "java.lang.Object");
1138 MakeExecutable(nullptr, "java.nio.DirectByteBuffer");
1139 MakeExecutable(nullptr, "java.nio.MemoryBlock");
1140 MakeExecutable(nullptr, "java.nio.MemoryBlock$UnmanagedBlock");
1141 MakeExecutable(nullptr, "java.nio.MappedByteBuffer");
1142 MakeExecutable(nullptr, "java.nio.ByteBuffer");
1143 MakeExecutable(nullptr, "java.nio.Buffer");
1144 // TODO: we only load a dex file here as starting the runtime relies upon it.
1145 const char* class_name = "StaticLeafMethods";
1146 LoadDex(class_name);
1147 bool started = runtime_->Start();
1148 ASSERT_TRUE(started);
1149
Elliott Hughesb465ab02011-08-24 11:21:21 -07001150 jclass buffer_class = env_->FindClass("java/nio/Buffer");
1151 ASSERT_TRUE(buffer_class != NULL);
1152
1153 char bytes[1024];
1154 jobject buffer = env_->NewDirectByteBuffer(bytes, sizeof(bytes));
1155 ASSERT_TRUE(buffer != NULL);
1156 ASSERT_TRUE(env_->IsInstanceOf(buffer, buffer_class));
1157 ASSERT_TRUE(env_->GetDirectBufferAddress(buffer) == bytes);
1158 ASSERT_TRUE(env_->GetDirectBufferCapacity(buffer) == sizeof(bytes));
1159}
1160
Ian Rogers6d0b13e2012-02-07 09:25:29 -08001161TEST_F(JniInternalTest, MonitorEnterExit) {
1162 // Create an object to torture
1163 jclass object_class = env_->FindClass("java/lang/Object");
1164 ASSERT_TRUE(object_class != NULL);
1165 jobject object = env_->AllocObject(object_class);
1166 ASSERT_TRUE(object != NULL);
1167
1168 // Expected class of exceptions
1169 jclass imse_class = env_->FindClass("java/lang/IllegalMonitorStateException");
1170 ASSERT_TRUE(imse_class != NULL);
1171
1172 jthrowable thrown_exception;
1173
1174 // Unlock of unowned monitor
1175 env_->MonitorExit(object);
1176 EXPECT_TRUE(env_->ExceptionCheck());
1177 thrown_exception = env_->ExceptionOccurred();
1178 env_->ExceptionClear();
1179 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, imse_class));
1180
1181 // Lock of unowned monitor
1182 env_->MonitorEnter(object);
1183 EXPECT_FALSE(env_->ExceptionCheck());
1184 // Regular unlock
1185 env_->MonitorExit(object);
1186 EXPECT_FALSE(env_->ExceptionCheck());
1187
1188 // Recursively lock a lot
1189 size_t max_recursive_lock = 1024;
1190 for (size_t i = 0; i < max_recursive_lock; i++) {
1191 env_->MonitorEnter(object);
1192 EXPECT_FALSE(env_->ExceptionCheck());
1193 }
1194 // Recursively unlock a lot
1195 for (size_t i = 0; i < max_recursive_lock; i++) {
1196 env_->MonitorExit(object);
1197 EXPECT_FALSE(env_->ExceptionCheck());
1198 }
1199
1200 // Unlock of unowned monitor
1201 env_->MonitorExit(object);
1202 EXPECT_TRUE(env_->ExceptionCheck());
1203 thrown_exception = env_->ExceptionOccurred();
1204 env_->ExceptionClear();
1205 EXPECT_TRUE(env_->IsInstanceOf(thrown_exception, imse_class));
Elliott Hughesa92853e2012-02-07 16:09:27 -08001206
1207 // It's an error to call MonitorEnter or MonitorExit on NULL.
Elliott Hughesb264f082012-04-06 17:10:10 -07001208 {
1209 CheckJniAbortCatcher check_jni_abort_catcher;
1210 env_->MonitorEnter(NULL);
1211 check_jni_abort_catcher.Check("in call to MonitorEnter");
Elliott Hughesb264f082012-04-06 17:10:10 -07001212
Elliott Hughesb264f082012-04-06 17:10:10 -07001213 env_->MonitorExit(NULL);
1214 check_jni_abort_catcher.Check("in call to MonitorExit");
1215 }
Ian Rogers6d0b13e2012-02-07 09:25:29 -08001216}
1217
Brian Carlstrom4d571432012-05-16 00:21:41 -07001218TEST_F(JniInternalTest, DetachCurrentThread) {
1219 CleanUpJniEnv(); // cleanup now so TearDown won't have junk from wrong JNIEnv
1220 jint ok = vm_->DetachCurrentThread();
1221 EXPECT_EQ(JNI_OK, ok);
1222
1223 jint err = vm_->DetachCurrentThread();
1224 EXPECT_EQ(JNI_ERR, err);
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -08001225 vm_->AttachCurrentThread(&env_, NULL); // need attached thread for CommonRuntimeTest::TearDown
Brian Carlstrom4d571432012-05-16 00:21:41 -07001226}
1227
Ian Rogers0cfe1fb2011-08-26 03:29:44 -07001228} // namespace art