blob: f8b7013957f0bc5807c8a3b1a90dc17d79f81ccb [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 */
Ian Rogersb033c752011-07-20 12:22:35 -070016
17#include <sys/mman.h>
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070018
Elliott Hughes90a33692011-08-30 13:27:07 -070019#include "UniquePtr.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070020#include "class_linker.h"
21#include "common_test.h"
22#include "dex_file.h"
Elliott Hughes90a33692011-08-30 13:27:07 -070023#include "gtest/gtest.h"
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070024#include "indirect_reference_table.h"
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070025#include "jni_internal.h"
Brian Carlstromdb4d5402011-08-09 12:18:28 -070026#include "mem_map.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070027#include "runtime.h"
Ian Rogersaaa20802011-09-11 21:47:37 -070028#include "scoped_jni_thread_state.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070029#include "thread.h"
Ian Rogersb033c752011-07-20 12:22:35 -070030
Elliott Hughesb264f082012-04-06 17:10:10 -070031extern "C" JNIEXPORT jint JNICALL Java_MyClassNatives_bar(JNIEnv*, jobject, jint count) {
Brian Carlstromb9cc1ca2012-01-27 00:57:42 -080032 return count + 1;
33}
34
Elliott Hughesb264f082012-04-06 17:10:10 -070035extern "C" JNIEXPORT jint JNICALL Java_MyClassNatives_sbar(JNIEnv*, jclass, jint count) {
Ian Rogers1cefdbd2012-02-29 09:34:50 -080036 return count + 1;
37}
38
Ian Rogersb033c752011-07-20 12:22:35 -070039namespace art {
40
Brian Carlstromf734cf52011-08-17 16:28:14 -070041class JniCompilerTest : public CommonTest {
Ian Rogersb033c752011-07-20 12:22:35 -070042 protected:
Ian Rogersb033c752011-07-20 12:22:35 -070043
Brian Carlstrom40381fb2011-10-19 14:13:40 -070044 void CompileForTest(ClassLoader* class_loader, bool direct,
45 const char* method_name, const char* method_sig) {
Brian Carlstrom25c33252011-09-18 15:58:35 -070046 // Compile the native method before starting the runtime
Elliott Hughesb264f082012-04-06 17:10:10 -070047 Class* c = class_linker_->FindClass("LMyClassNatives;", class_loader);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070048 Method* method;
49 if (direct) {
buzbeec143c552011-08-20 17:38:58 -070050 method = c->FindDirectMethod(method_name, method_sig);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070051 } else {
buzbeec143c552011-08-20 17:38:58 -070052 method = c->FindVirtualMethod(method_name, method_sig);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070053 }
54 ASSERT_TRUE(method != NULL);
Brian Carlstrom25c33252011-09-18 15:58:35 -070055 if (method->GetCode() != NULL) {
56 return;
57 }
Brian Carlstrom9baa4ae2011-09-01 21:14:14 -070058 CompileMethod(method);
Elliott Hughes1240dad2011-09-09 16:24:50 -070059 ASSERT_TRUE(method->GetCode() != NULL);
Brian Carlstrom25c33252011-09-18 15:58:35 -070060 }
61
Elliott Hughesb264f082012-04-06 17:10:10 -070062 void SetUpForTest(ClassLoader* class_loader, bool direct,
Brian Carlstrom40381fb2011-10-19 14:13:40 -070063 const char* method_name, const char* method_sig,
Brian Carlstrom25c33252011-09-18 15:58:35 -070064 void* native_fnptr) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -070065 CompileForTest(class_loader, direct, method_name, method_sig);
Brian Carlstrom25c33252011-09-18 15:58:35 -070066 if (!runtime_->IsStarted()) {
67 runtime_->Start();
68 }
69
70 // JNI operations after runtime start
71 env_ = Thread::Current()->GetJniEnv();
Elliott Hughesb264f082012-04-06 17:10:10 -070072 jklass_ = env_->FindClass("MyClassNatives");
Brian Carlstrom25c33252011-09-18 15:58:35 -070073 ASSERT_TRUE(jklass_ != NULL);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070074
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070075 if (direct) {
76 jmethod_ = env_->GetStaticMethodID(jklass_, method_name, method_sig);
77 } else {
78 jmethod_ = env_->GetMethodID(jklass_, method_name, method_sig);
79 }
80 ASSERT_TRUE(jmethod_ != NULL);
81
Ian Rogersbdb03912011-09-14 00:55:44 -070082 if (native_fnptr != NULL) {
Elliott Hughesb25c3f62012-03-26 16:35:06 -070083 JNINativeMethod methods[] = { { method_name, method_sig, native_fnptr } };
Shih-wei Liao31384c52011-09-06 15:27:45 -070084 ASSERT_EQ(JNI_OK, env_->RegisterNatives(jklass_, methods, 1));
Ian Rogersbdb03912011-09-14 00:55:44 -070085 } else {
86 env_->UnregisterNatives(jklass_);
Shih-wei Liao31384c52011-09-06 15:27:45 -070087 }
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070088
89 jmethodID constructor = env_->GetMethodID(jklass_, "<init>", "()V");
90 jobj_ = env_->NewObject(jklass_, constructor);
91 ASSERT_TRUE(jobj_ != NULL);
Ian Rogersb033c752011-07-20 12:22:35 -070092 }
93
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070094 public:
95 static jclass jklass_;
96 static jobject jobj_;
97 protected:
Ian Rogerscdd1d2d2011-08-18 09:58:17 -070098 JNIEnv* env_;
99 jmethodID jmethod_;
Ian Rogersb033c752011-07-20 12:22:35 -0700100};
101
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700102jclass JniCompilerTest::jklass_;
103jobject JniCompilerTest::jobj_;
104
Elliott Hughesb264f082012-04-06 17:10:10 -0700105int gJava_MyClassNatives_foo_calls = 0;
106void Java_MyClassNatives_foo(JNIEnv* env, jobject thisObj) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700107 // 2 = SirtRef<ClassLoader> + thisObj
TDYa12728f1a142012-03-15 21:51:52 -0700108 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700109 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700110 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
111 EXPECT_TRUE(thisObj != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700112 EXPECT_TRUE(env->IsInstanceOf(thisObj, JniCompilerTest::jklass_));
Elliott Hughesb264f082012-04-06 17:10:10 -0700113 gJava_MyClassNatives_foo_calls++;
Ian Rogersb033c752011-07-20 12:22:35 -0700114}
115
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700116TEST_F(JniCompilerTest, CompileAndRunNoArgMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700117 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700118 SetUpForTest(class_loader.get(), false, "foo", "()V",
119 reinterpret_cast<void*>(&Java_MyClassNatives_foo));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700120
Elliott Hughesb264f082012-04-06 17:10:10 -0700121 EXPECT_EQ(0, gJava_MyClassNatives_foo_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700122 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
Elliott Hughesb264f082012-04-06 17:10:10 -0700123 EXPECT_EQ(1, gJava_MyClassNatives_foo_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700124 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
Elliott Hughesb264f082012-04-06 17:10:10 -0700125 EXPECT_EQ(2, gJava_MyClassNatives_foo_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700126}
127
Shih-wei Liao31384c52011-09-06 15:27:45 -0700128TEST_F(JniCompilerTest, CompileAndRunIntMethodThroughStub) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700129 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700130 SetUpForTest(class_loader.get(), false, "bar", "(I)I",
131 NULL /* calling through stub will link with &Java_MyClassNatives_bar */);
Shih-wei Liao31384c52011-09-06 15:27:45 -0700132
Shih-wei Liao31384c52011-09-06 15:27:45 -0700133 std::string reason;
Brian Carlstromb9cc1ca2012-01-27 00:57:42 -0800134 ASSERT_TRUE(Runtime::Current()->GetJavaVM()->LoadNativeLibrary("", class_loader.get(), reason))
135 << reason;
Shih-wei Liao31384c52011-09-06 15:27:45 -0700136
137 jint result = env_->CallNonvirtualIntMethod(jobj_, jklass_, jmethod_, 24);
138 EXPECT_EQ(25, result);
139}
140
Ian Rogers1cefdbd2012-02-29 09:34:50 -0800141TEST_F(JniCompilerTest, CompileAndRunStaticIntMethodThroughStub) {
142 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700143 SetUpForTest(class_loader.get(), true, "sbar", "(I)I",
144 NULL /* calling through stub will link with &Java_MyClassNatives_sbar */);
Ian Rogers1cefdbd2012-02-29 09:34:50 -0800145
146 std::string reason;
147 ASSERT_TRUE(Runtime::Current()->GetJavaVM()->LoadNativeLibrary("", class_loader.get(), reason))
148 << reason;
149
150 jint result = env_->CallStaticIntMethod(jklass_, jmethod_, 42);
151 EXPECT_EQ(43, result);
152}
153
Elliott Hughesb264f082012-04-06 17:10:10 -0700154int gJava_MyClassNatives_fooI_calls = 0;
155jint Java_MyClassNatives_fooI(JNIEnv* env, jobject thisObj, jint x) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700156 // 2 = SirtRef<ClassLoader> + thisObj
TDYa12728f1a142012-03-15 21:51:52 -0700157 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700158 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700159 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
160 EXPECT_TRUE(thisObj != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700161 EXPECT_TRUE(env->IsInstanceOf(thisObj, JniCompilerTest::jklass_));
Elliott Hughesb264f082012-04-06 17:10:10 -0700162 gJava_MyClassNatives_fooI_calls++;
Ian Rogersb033c752011-07-20 12:22:35 -0700163 return x;
164}
165
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700166TEST_F(JniCompilerTest, CompileAndRunIntMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700167 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700168 SetUpForTest(class_loader.get(), false, "fooI", "(I)I",
169 reinterpret_cast<void*>(&Java_MyClassNatives_fooI));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700170
Elliott Hughesb264f082012-04-06 17:10:10 -0700171 EXPECT_EQ(0, gJava_MyClassNatives_fooI_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700172 jint result = env_->CallNonvirtualIntMethod(jobj_, jklass_, jmethod_, 42);
173 EXPECT_EQ(42, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700174 EXPECT_EQ(1, gJava_MyClassNatives_fooI_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700175 result = env_->CallNonvirtualIntMethod(jobj_, jklass_, jmethod_, 0xCAFED00D);
176 EXPECT_EQ(static_cast<jint>(0xCAFED00D), result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700177 EXPECT_EQ(2, gJava_MyClassNatives_fooI_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700178}
179
Elliott Hughesb264f082012-04-06 17:10:10 -0700180int gJava_MyClassNatives_fooII_calls = 0;
181jint Java_MyClassNatives_fooII(JNIEnv* env, jobject thisObj, jint x, jint y) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700182 // 2 = SirtRef<ClassLoader> + thisObj
TDYa12728f1a142012-03-15 21:51:52 -0700183 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700184 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700185 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
186 EXPECT_TRUE(thisObj != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700187 EXPECT_TRUE(env->IsInstanceOf(thisObj, JniCompilerTest::jklass_));
Elliott Hughesb264f082012-04-06 17:10:10 -0700188 gJava_MyClassNatives_fooII_calls++;
Ian Rogersb033c752011-07-20 12:22:35 -0700189 return x - y; // non-commutative operator
190}
191
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700192TEST_F(JniCompilerTest, CompileAndRunIntIntMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700193 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700194 SetUpForTest(class_loader.get(), false, "fooII", "(II)I",
195 reinterpret_cast<void*>(&Java_MyClassNatives_fooII));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700196
Elliott Hughesb264f082012-04-06 17:10:10 -0700197 EXPECT_EQ(0, gJava_MyClassNatives_fooII_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700198 jint result = env_->CallNonvirtualIntMethod(jobj_, jklass_, jmethod_, 99, 10);
199 EXPECT_EQ(99 - 10, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700200 EXPECT_EQ(1, gJava_MyClassNatives_fooII_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700201 result = env_->CallNonvirtualIntMethod(jobj_, jklass_, jmethod_, 0xCAFEBABE,
202 0xCAFED00D);
203 EXPECT_EQ(static_cast<jint>(0xCAFEBABE - 0xCAFED00D), result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700204 EXPECT_EQ(2, gJava_MyClassNatives_fooII_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700205}
206
Elliott Hughesb264f082012-04-06 17:10:10 -0700207int gJava_MyClassNatives_fooJJ_calls = 0;
208jlong Java_MyClassNatives_fooJJ(JNIEnv* env, jobject thisObj, jlong x, jlong y) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700209 // 2 = SirtRef<ClassLoader> + thisObj
TDYa12728f1a142012-03-15 21:51:52 -0700210 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700211 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogers9b269d22011-09-04 14:06:05 -0700212 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
213 EXPECT_TRUE(thisObj != NULL);
214 EXPECT_TRUE(env->IsInstanceOf(thisObj, JniCompilerTest::jklass_));
Elliott Hughesb264f082012-04-06 17:10:10 -0700215 gJava_MyClassNatives_fooJJ_calls++;
Ian Rogers9b269d22011-09-04 14:06:05 -0700216 return x - y; // non-commutative operator
217}
218
219TEST_F(JniCompilerTest, CompileAndRunLongLongMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700220 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700221 SetUpForTest(class_loader.get(), false, "fooJJ", "(JJ)J",
222 reinterpret_cast<void*>(&Java_MyClassNatives_fooJJ));
Ian Rogers9b269d22011-09-04 14:06:05 -0700223
Elliott Hughesb264f082012-04-06 17:10:10 -0700224 EXPECT_EQ(0, gJava_MyClassNatives_fooJJ_calls);
Ian Rogers9b269d22011-09-04 14:06:05 -0700225 jlong a = 0x1234567890ABCDEFll;
226 jlong b = 0xFEDCBA0987654321ll;
227 jlong result = env_->CallNonvirtualLongMethod(jobj_, jklass_, jmethod_, a, b);
228 EXPECT_EQ(a - b, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700229 EXPECT_EQ(1, gJava_MyClassNatives_fooJJ_calls);
Ian Rogers9b269d22011-09-04 14:06:05 -0700230 result = env_->CallNonvirtualLongMethod(jobj_, jklass_, jmethod_, b, a);
231 EXPECT_EQ(b - a, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700232 EXPECT_EQ(2, gJava_MyClassNatives_fooJJ_calls);
Ian Rogers9b269d22011-09-04 14:06:05 -0700233}
234
Elliott Hughesb264f082012-04-06 17:10:10 -0700235int gJava_MyClassNatives_fooDD_calls = 0;
236jdouble Java_MyClassNatives_fooDD(JNIEnv* env, jobject thisObj, jdouble x, jdouble y) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700237 // 2 = SirtRef<ClassLoader> + thisObj
TDYa12728f1a142012-03-15 21:51:52 -0700238 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700239 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700240 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
241 EXPECT_TRUE(thisObj != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700242 EXPECT_TRUE(env->IsInstanceOf(thisObj, JniCompilerTest::jklass_));
Elliott Hughesb264f082012-04-06 17:10:10 -0700243 gJava_MyClassNatives_fooDD_calls++;
Ian Rogersb033c752011-07-20 12:22:35 -0700244 return x - y; // non-commutative operator
245}
246
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700247TEST_F(JniCompilerTest, CompileAndRunDoubleDoubleMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700248 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700249 SetUpForTest(class_loader.get(), false, "fooDD", "(DD)D",
250 reinterpret_cast<void*>(&Java_MyClassNatives_fooDD));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700251
Elliott Hughesb264f082012-04-06 17:10:10 -0700252 EXPECT_EQ(0, gJava_MyClassNatives_fooDD_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700253 jdouble result = env_->CallNonvirtualDoubleMethod(jobj_, jklass_, jmethod_,
254 99.0, 10.0);
255 EXPECT_EQ(99.0 - 10.0, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700256 EXPECT_EQ(1, gJava_MyClassNatives_fooDD_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700257 jdouble a = 3.14159265358979323846;
258 jdouble b = 0.69314718055994530942;
259 result = env_->CallNonvirtualDoubleMethod(jobj_, jklass_, jmethod_, a, b);
260 EXPECT_EQ(a - b, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700261 EXPECT_EQ(2, gJava_MyClassNatives_fooDD_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700262}
263
Elliott Hughesb264f082012-04-06 17:10:10 -0700264int gJava_MyClassNatives_fooIOO_calls = 0;
265jobject Java_MyClassNatives_fooIOO(JNIEnv* env, jobject thisObj, jint x, jobject y,
Ian Rogersb033c752011-07-20 12:22:35 -0700266 jobject z) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700267 // 4 = SirtRef<ClassLoader> + this + y + z
TDYa12728f1a142012-03-15 21:51:52 -0700268 EXPECT_EQ(4U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700269 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700270 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
271 EXPECT_TRUE(thisObj != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700272 EXPECT_TRUE(env->IsInstanceOf(thisObj, JniCompilerTest::jklass_));
Elliott Hughesb264f082012-04-06 17:10:10 -0700273 gJava_MyClassNatives_fooIOO_calls++;
Ian Rogersb033c752011-07-20 12:22:35 -0700274 switch (x) {
275 case 1:
276 return y;
277 case 2:
278 return z;
279 default:
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700280 return thisObj;
Ian Rogersb033c752011-07-20 12:22:35 -0700281 }
282}
283
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700284TEST_F(JniCompilerTest, CompileAndRunIntObjectObjectMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700285 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700286 SetUpForTest(class_loader.get(), false, "fooIOO",
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700287 "(ILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
Elliott Hughesb264f082012-04-06 17:10:10 -0700288 reinterpret_cast<void*>(&Java_MyClassNatives_fooIOO));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700289
Elliott Hughesb264f082012-04-06 17:10:10 -0700290 EXPECT_EQ(0, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700291 jobject result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 0, NULL, NULL);
292 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700293 EXPECT_EQ(1, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700294
295 result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 0, NULL, jklass_);
296 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700297 EXPECT_EQ(2, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700298 result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 1, NULL, jklass_);
299 EXPECT_TRUE(env_->IsSameObject(NULL, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700300 EXPECT_EQ(3, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700301 result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 2, NULL, jklass_);
302 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700303 EXPECT_EQ(4, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700304
305 result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 0, jklass_, NULL);
306 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700307 EXPECT_EQ(5, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700308 result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 1, jklass_, NULL);
309 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700310 EXPECT_EQ(6, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700311 result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, 2, jklass_, NULL);
312 EXPECT_TRUE(env_->IsSameObject(NULL, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700313 EXPECT_EQ(7, gJava_MyClassNatives_fooIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700314}
315
Elliott Hughesb264f082012-04-06 17:10:10 -0700316int gJava_MyClassNatives_fooSII_calls = 0;
317jint Java_MyClassNatives_fooSII(JNIEnv* env, jclass klass, jint x, jint y) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700318 // 2 = SirtRef<ClassLoader> + klass
TDYa12728f1a142012-03-15 21:51:52 -0700319 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700320 EXPECT_EQ(kNative, Thread::Current()->GetState());
Shih-wei Liao82da44b2011-09-01 00:38:04 -0700321 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
322 EXPECT_TRUE(klass != NULL);
323 EXPECT_TRUE(env->IsInstanceOf(JniCompilerTest::jobj_, klass));
Elliott Hughesb264f082012-04-06 17:10:10 -0700324 gJava_MyClassNatives_fooSII_calls++;
Shih-wei Liao82da44b2011-09-01 00:38:04 -0700325 return x + y;
326}
327
Shih-wei Liao82da44b2011-09-01 00:38:04 -0700328TEST_F(JniCompilerTest, CompileAndRunStaticIntIntMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700329 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700330 SetUpForTest(class_loader.get(), true, "fooSII", "(II)I",
331 reinterpret_cast<void*>(&Java_MyClassNatives_fooSII));
Shih-wei Liao82da44b2011-09-01 00:38:04 -0700332
Elliott Hughesb264f082012-04-06 17:10:10 -0700333 EXPECT_EQ(0, gJava_MyClassNatives_fooSII_calls);
Shih-wei Liao82da44b2011-09-01 00:38:04 -0700334 jint result = env_->CallStaticIntMethod(jklass_, jmethod_, 20, 30);
335 EXPECT_EQ(50, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700336 EXPECT_EQ(1, gJava_MyClassNatives_fooSII_calls);
Shih-wei Liao82da44b2011-09-01 00:38:04 -0700337}
338
Elliott Hughesb264f082012-04-06 17:10:10 -0700339int gJava_MyClassNatives_fooSDD_calls = 0;
340jdouble Java_MyClassNatives_fooSDD(JNIEnv* env, jclass klass, jdouble x, jdouble y) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700341 // 2 = SirtRef<ClassLoader> + klass
TDYa12728f1a142012-03-15 21:51:52 -0700342 EXPECT_EQ(2U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700343 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogers7a99c112011-09-07 12:48:27 -0700344 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
345 EXPECT_TRUE(klass != NULL);
346 EXPECT_TRUE(env->IsInstanceOf(JniCompilerTest::jobj_, klass));
Elliott Hughesb264f082012-04-06 17:10:10 -0700347 gJava_MyClassNatives_fooSDD_calls++;
Ian Rogers7a99c112011-09-07 12:48:27 -0700348 return x - y; // non-commutative operator
349}
350
351TEST_F(JniCompilerTest, CompileAndRunStaticDoubleDoubleMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700352 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700353 SetUpForTest(class_loader.get(), true, "fooSDD", "(DD)D",
354 reinterpret_cast<void*>(&Java_MyClassNatives_fooSDD));
Ian Rogers7a99c112011-09-07 12:48:27 -0700355
Elliott Hughesb264f082012-04-06 17:10:10 -0700356 EXPECT_EQ(0, gJava_MyClassNatives_fooSDD_calls);
Ian Rogers7a99c112011-09-07 12:48:27 -0700357 jdouble result = env_->CallStaticDoubleMethod(jklass_, jmethod_, 99.0, 10.0);
358 EXPECT_EQ(99.0 - 10.0, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700359 EXPECT_EQ(1, gJava_MyClassNatives_fooSDD_calls);
Ian Rogers7a99c112011-09-07 12:48:27 -0700360 jdouble a = 3.14159265358979323846;
361 jdouble b = 0.69314718055994530942;
362 result = env_->CallStaticDoubleMethod(jklass_, jmethod_, a, b);
363 EXPECT_EQ(a - b, result);
Elliott Hughesb264f082012-04-06 17:10:10 -0700364 EXPECT_EQ(2, gJava_MyClassNatives_fooSDD_calls);
Ian Rogers7a99c112011-09-07 12:48:27 -0700365}
366
Elliott Hughesb264f082012-04-06 17:10:10 -0700367int gJava_MyClassNatives_fooSIOO_calls = 0;
368jobject Java_MyClassNatives_fooSIOO(JNIEnv* env, jclass klass, jint x, jobject y,
Ian Rogersb033c752011-07-20 12:22:35 -0700369 jobject z) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700370 // 4 = SirtRef<ClassLoader> + klass + y + z
TDYa12728f1a142012-03-15 21:51:52 -0700371 EXPECT_EQ(4U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700372 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700373 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
374 EXPECT_TRUE(klass != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700375 EXPECT_TRUE(env->IsInstanceOf(JniCompilerTest::jobj_, klass));
Elliott Hughesb264f082012-04-06 17:10:10 -0700376 gJava_MyClassNatives_fooSIOO_calls++;
Ian Rogersb033c752011-07-20 12:22:35 -0700377 switch (x) {
378 case 1:
379 return y;
380 case 2:
381 return z;
382 default:
383 return klass;
384 }
385}
386
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700387
388TEST_F(JniCompilerTest, CompileAndRunStaticIntObjectObjectMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700389 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700390 SetUpForTest(class_loader.get(), true, "fooSIOO",
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700391 "(ILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
Elliott Hughesb264f082012-04-06 17:10:10 -0700392 reinterpret_cast<void*>(&Java_MyClassNatives_fooSIOO));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700393
Elliott Hughesb264f082012-04-06 17:10:10 -0700394 EXPECT_EQ(0, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700395 jobject result = env_->CallStaticObjectMethod(jklass_, jmethod_, 0, NULL, NULL);
396 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700397 EXPECT_EQ(1, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700398
399 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 0, NULL, jobj_);
400 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700401 EXPECT_EQ(2, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700402 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 1, NULL, jobj_);
403 EXPECT_TRUE(env_->IsSameObject(NULL, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700404 EXPECT_EQ(3, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700405 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 2, NULL, jobj_);
406 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700407 EXPECT_EQ(4, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700408
409 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 0, jobj_, NULL);
410 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700411 EXPECT_EQ(5, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700412 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 1, jobj_, NULL);
413 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700414 EXPECT_EQ(6, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700415 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 2, jobj_, NULL);
416 EXPECT_TRUE(env_->IsSameObject(NULL, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700417 EXPECT_EQ(7, gJava_MyClassNatives_fooSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700418}
419
Elliott Hughesb264f082012-04-06 17:10:10 -0700420int gJava_MyClassNatives_fooSSIOO_calls = 0;
421jobject Java_MyClassNatives_fooSSIOO(JNIEnv* env, jclass klass, jint x, jobject y,
Ian Rogersdf20fe02011-07-20 20:34:16 -0700422 jobject z) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700423 // 4 = SirtRef<ClassLoader> + klass + y + z
TDYa12728f1a142012-03-15 21:51:52 -0700424 EXPECT_EQ(4U, Thread::Current()->NumStackReferences());
Elliott Hughes34e06962012-04-09 13:55:55 -0700425 EXPECT_EQ(kNative, Thread::Current()->GetState());
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700426 EXPECT_EQ(Thread::Current()->GetJniEnv(), env);
427 EXPECT_TRUE(klass != NULL);
Ian Rogersa8cd9f42011-08-19 16:43:41 -0700428 EXPECT_TRUE(env->IsInstanceOf(JniCompilerTest::jobj_, klass));
Elliott Hughesb264f082012-04-06 17:10:10 -0700429 gJava_MyClassNatives_fooSSIOO_calls++;
Ian Rogersdf20fe02011-07-20 20:34:16 -0700430 switch (x) {
431 case 1:
432 return y;
433 case 2:
434 return z;
435 default:
436 return klass;
437 }
438}
439
Ian Rogersdf20fe02011-07-20 20:34:16 -0700440TEST_F(JniCompilerTest, CompileAndRunStaticSynchronizedIntObjectObjectMethod) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700441 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700442 SetUpForTest(class_loader.get(), true, "fooSSIOO",
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700443 "(ILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
Elliott Hughesb264f082012-04-06 17:10:10 -0700444 reinterpret_cast<void*>(&Java_MyClassNatives_fooSSIOO));
Carl Shapiro419ec7b2011-08-03 14:48:33 -0700445
Elliott Hughesb264f082012-04-06 17:10:10 -0700446 EXPECT_EQ(0, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700447 jobject result = env_->CallStaticObjectMethod(jklass_, jmethod_, 0, NULL, NULL);
448 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700449 EXPECT_EQ(1, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700450
451 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 0, NULL, jobj_);
452 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700453 EXPECT_EQ(2, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700454 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 1, NULL, jobj_);
455 EXPECT_TRUE(env_->IsSameObject(NULL, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700456 EXPECT_EQ(3, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700457 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 2, NULL, jobj_);
458 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700459 EXPECT_EQ(4, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700460
461 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 0, jobj_, NULL);
462 EXPECT_TRUE(env_->IsSameObject(jklass_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700463 EXPECT_EQ(5, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700464 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 1, jobj_, NULL);
465 EXPECT_TRUE(env_->IsSameObject(jobj_, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700466 EXPECT_EQ(6, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700467 result = env_->CallStaticObjectMethod(jklass_, jmethod_, 2, jobj_, NULL);
468 EXPECT_TRUE(env_->IsSameObject(NULL, result));
Elliott Hughesb264f082012-04-06 17:10:10 -0700469 EXPECT_EQ(7, gJava_MyClassNatives_fooSSIOO_calls);
Ian Rogersdf20fe02011-07-20 20:34:16 -0700470}
471
Elliott Hughesb264f082012-04-06 17:10:10 -0700472void Java_MyClassNatives_throwException(JNIEnv* env, jobject) {
Elliott Hughesa2501992011-08-26 19:39:54 -0700473 jclass c = env->FindClass("java/lang/RuntimeException");
474 env->ThrowNew(c, "hello");
475}
Ian Rogers45a76cb2011-07-21 22:00:15 -0700476
Elliott Hughesa2501992011-08-26 19:39:54 -0700477TEST_F(JniCompilerTest, ExceptionHandling) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700478 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
479
Elliott Hughesb264f082012-04-06 17:10:10 -0700480 // all compilation needs to happen before SetUpForTest calls Runtime::Start
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700481 CompileForTest(class_loader.get(), false, "foo", "()V");
482 CompileForTest(class_loader.get(), false, "throwException", "()V");
483 CompileForTest(class_loader.get(), false, "foo", "()V");
Brian Carlstrom25c33252011-09-18 15:58:35 -0700484
Elliott Hughesb264f082012-04-06 17:10:10 -0700485 gJava_MyClassNatives_foo_calls = 0;
Elliott Hughesa2501992011-08-26 19:39:54 -0700486
Ian Rogers67375ac2011-09-14 00:55:44 -0700487 // Check a single call of a JNI method is ok
Elliott Hughesb264f082012-04-06 17:10:10 -0700488 SetUpForTest(class_loader.get(), false, "foo", "()V", reinterpret_cast<void*>(&Java_MyClassNatives_foo));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700489 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
Elliott Hughesb264f082012-04-06 17:10:10 -0700490 EXPECT_EQ(1, gJava_MyClassNatives_foo_calls);
Ian Rogers67375ac2011-09-14 00:55:44 -0700491 EXPECT_FALSE(Thread::Current()->IsExceptionPending());
Elliott Hughesa2501992011-08-26 19:39:54 -0700492
Ian Rogers67375ac2011-09-14 00:55:44 -0700493 // Get class for exception we expect to be thrown
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700494 Class* jlre = class_linker_->FindClass("Ljava/lang/RuntimeException;", class_loader.get());
Elliott Hughesb264f082012-04-06 17:10:10 -0700495 SetUpForTest(class_loader.get(), false, "throwException", "()V",
496 reinterpret_cast<void*>(&Java_MyClassNatives_throwException));
497 // Call Java_MyClassNatives_throwException (JNI method that throws exception)
Elliott Hughesa2501992011-08-26 19:39:54 -0700498 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
Elliott Hughesb264f082012-04-06 17:10:10 -0700499 EXPECT_EQ(1, gJava_MyClassNatives_foo_calls);
Ian Rogers67375ac2011-09-14 00:55:44 -0700500 EXPECT_TRUE(Thread::Current()->IsExceptionPending());
501 EXPECT_TRUE(Thread::Current()->GetException()->InstanceOf(jlre));
502 Thread::Current()->ClearException();
Elliott Hughesa2501992011-08-26 19:39:54 -0700503
Ian Rogers67375ac2011-09-14 00:55:44 -0700504 // Check a single call of a JNI method is ok
Elliott Hughesb264f082012-04-06 17:10:10 -0700505 SetUpForTest(class_loader.get(), false, "foo", "()V", reinterpret_cast<void*>(&Java_MyClassNatives_foo));
Ian Rogerscdd1d2d2011-08-18 09:58:17 -0700506 env_->CallNonvirtualVoidMethod(jobj_, jklass_, jmethod_);
Elliott Hughesb264f082012-04-06 17:10:10 -0700507 EXPECT_EQ(2, gJava_MyClassNatives_foo_calls);
Ian Rogers45a76cb2011-07-21 22:00:15 -0700508}
509
Elliott Hughesb264f082012-04-06 17:10:10 -0700510jint Java_MyClassNatives_nativeUpCall(JNIEnv* env, jobject thisObj, jint i) {
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700511 if (i <= 0) {
Ian Rogersaaa20802011-09-11 21:47:37 -0700512 // We want to check raw Object*/Array* below
513 ScopedJniThreadState ts(env);
514
515 // Build stack trace
Elliott Hughes01158d72011-09-19 19:47:10 -0700516 jobject internal = Thread::Current()->CreateInternalStackTrace(env);
517 jobjectArray ste_array = Thread::InternalStackTraceToStackTraceElementArray(env, internal);
Ian Rogersaaa20802011-09-11 21:47:37 -0700518 ObjectArray<StackTraceElement>* trace_array =
519 Decode<ObjectArray<StackTraceElement>*>(env, ste_array);
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700520 EXPECT_TRUE(trace_array != NULL);
521 EXPECT_EQ(11, trace_array->GetLength());
522
Ian Rogersaaa20802011-09-11 21:47:37 -0700523 // Check stack trace entries have expected values
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700524 for (int32_t i = 0; i < trace_array->GetLength(); ++i) {
525 EXPECT_EQ(-2, trace_array->Get(i)->GetLineNumber());
Ian Rogersaaa20802011-09-11 21:47:37 -0700526 StackTraceElement* ste = trace_array->Get(i);
527 EXPECT_STREQ("MyClassNatives.java", ste->GetFileName()->ToModifiedUtf8().c_str());
Elliott Hughesb264f082012-04-06 17:10:10 -0700528 EXPECT_STREQ("MyClassNatives", ste->GetDeclaringClass()->ToModifiedUtf8().c_str());
Ian Rogersaaa20802011-09-11 21:47:37 -0700529 EXPECT_STREQ("fooI", ste->GetMethodName()->ToModifiedUtf8().c_str());
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700530 }
Ian Rogersaaa20802011-09-11 21:47:37 -0700531
532 // end recursion
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700533 return 0;
534 } else {
Elliott Hughesb264f082012-04-06 17:10:10 -0700535 jclass jklass = env->FindClass("MyClassNatives");
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700536 EXPECT_TRUE(jklass != NULL);
537 jmethodID jmethod = env->GetMethodID(jklass, "fooI", "(I)I");
538 EXPECT_TRUE(jmethod != NULL);
539
Ian Rogersaaa20802011-09-11 21:47:37 -0700540 // Recurse with i - 1
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700541 jint result = env->CallNonvirtualIntMethod(thisObj, jklass, jmethod, i - 1);
Ian Rogersaaa20802011-09-11 21:47:37 -0700542
543 // Return sum of all depths
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700544 return i + result;
545 }
546}
547
548TEST_F(JniCompilerTest, NativeStackTraceElement) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700549 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700550 SetUpForTest(class_loader.get(), false, "fooI", "(I)I",
551 reinterpret_cast<void*>(&Java_MyClassNatives_nativeUpCall));
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700552 jint result = env_->CallNonvirtualIntMethod(jobj_, jklass_, jmethod_, 10);
Ian Rogersaaa20802011-09-11 21:47:37 -0700553 EXPECT_EQ(10+9+8+7+6+5+4+3+2+1, result);
Shih-wei Liaoff0f9be2011-08-29 15:43:53 -0700554}
555
Elliott Hughesb264f082012-04-06 17:10:10 -0700556jobject Java_MyClassNatives_fooO(JNIEnv* env, jobject, jobject x) {
Shih-wei Liao558788e2011-09-01 02:39:11 -0700557 return env->NewGlobalRef(x);
558}
559
Ian Rogersb9231c82011-09-05 22:13:19 -0700560TEST_F(JniCompilerTest, ReturnGlobalRef) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700561 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700562 SetUpForTest(class_loader.get(), false, "fooO", "(Ljava/lang/Object;)Ljava/lang/Object;",
563 reinterpret_cast<void*>(&Java_MyClassNatives_fooO));
Shih-wei Liao558788e2011-09-01 02:39:11 -0700564 jobject result = env_->CallNonvirtualObjectMethod(jobj_, jklass_, jmethod_, jobj_);
565 EXPECT_EQ(JNILocalRefType, env_->GetObjectRefType(result));
566 EXPECT_TRUE(env_->IsSameObject(result, jobj_));
567}
568
Ian Rogersdc51b792011-09-22 20:41:37 -0700569jint local_ref_test(JNIEnv* env, jobject thisObj, jint x) {
570 // Add 10 local references
Ian Rogers5a7a74a2011-09-26 16:32:29 -0700571 for (int i = 0; i < 10; i++) {
Ian Rogersdc51b792011-09-22 20:41:37 -0700572 AddLocalReference<jobject>(env, Decode<Object*>(env, thisObj));
573 }
574 return x+1;
575}
576
577TEST_F(JniCompilerTest, LocalReferenceTableClearingTest) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700578 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700579 SetUpForTest(class_loader.get(), false, "fooI", "(I)I", reinterpret_cast<void*>(&local_ref_test));
Ian Rogersdc51b792011-09-22 20:41:37 -0700580 // 1000 invocations of a method that adds 10 local references
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700581 for (int i = 0; i < 1000; i++) {
Ian Rogersdc51b792011-09-22 20:41:37 -0700582 jint result = env_->CallIntMethod(jobj_, jmethod_, i);
583 EXPECT_TRUE(result == i + 1);
584 }
585}
586
Ian Rogersb9231c82011-09-05 22:13:19 -0700587void my_arraycopy(JNIEnv* env, jclass klass, jobject src, jint src_pos, jobject dst, jint dst_pos, jint length) {
588 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jklass_, klass));
589 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jklass_, dst));
Ian Rogers82f3e092011-09-05 22:54:45 -0700590 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jobj_, src));
Ian Rogersb9231c82011-09-05 22:13:19 -0700591 EXPECT_EQ(1234, src_pos);
592 EXPECT_EQ(5678, dst_pos);
593 EXPECT_EQ(9876, length);
594}
595
596TEST_F(JniCompilerTest, JavaLangSystemArrayCopy) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700597 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700598 SetUpForTest(class_loader.get(), true, "arraycopy", "(Ljava/lang/Object;ILjava/lang/Object;II)V",
Ian Rogersb9231c82011-09-05 22:13:19 -0700599 reinterpret_cast<void*>(&my_arraycopy));
Ian Rogers82f3e092011-09-05 22:54:45 -0700600 env_->CallStaticVoidMethod(jklass_, jmethod_, jobj_, 1234, jklass_, 5678, 9876);
Ian Rogersb9231c82011-09-05 22:13:19 -0700601}
602
Ian Rogers67375ac2011-09-14 00:55:44 -0700603jboolean my_casi(JNIEnv* env, jobject unsafe, jobject obj, jlong offset, jint expected, jint newval) {
604 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jobj_, unsafe));
605 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jobj_, obj));
606 EXPECT_EQ(0x12345678ABCDEF88ll, offset);
607 EXPECT_EQ(static_cast<jint>(0xCAFEF00D), expected);
608 EXPECT_EQ(static_cast<jint>(0xEBADF00D), newval);
609 return JNI_TRUE;
610}
611
612TEST_F(JniCompilerTest, CompareAndSwapInt) {
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700613 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700614 SetUpForTest(class_loader.get(), false, "compareAndSwapInt", "(Ljava/lang/Object;JII)Z",
Ian Rogers67375ac2011-09-14 00:55:44 -0700615 reinterpret_cast<void*>(&my_casi));
616 jboolean result = env_->CallBooleanMethod(jobj_, jmethod_, jobj_, 0x12345678ABCDEF88ll, 0xCAFEF00D, 0xEBADF00D);
617 EXPECT_EQ(result, JNI_TRUE);
618}
619
Ian Rogersc7792842012-03-03 15:36:20 -0800620jint my_gettext(JNIEnv* env, jclass klass, jlong val1, jobject obj1, jlong val2, jobject obj2) {
621 EXPECT_TRUE(env->IsInstanceOf(JniCompilerTest::jobj_, klass));
622 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jobj_, obj1));
623 EXPECT_TRUE(env->IsSameObject(JniCompilerTest::jobj_, obj2));
624 EXPECT_EQ(0x12345678ABCDEF88ll, val1);
625 EXPECT_EQ(0x7FEDCBA987654321ll, val2);
626 return 42;
627}
628
629TEST_F(JniCompilerTest, GetText) {
630 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
Elliott Hughesb264f082012-04-06 17:10:10 -0700631 SetUpForTest(class_loader.get(), true, "getText", "(JLjava/lang/Object;JLjava/lang/Object;)I",
Ian Rogersc7792842012-03-03 15:36:20 -0800632 reinterpret_cast<void*>(&my_gettext));
633 jint result = env_->CallStaticIntMethod(jklass_, jmethod_, 0x12345678ABCDEF88ll, jobj_,
634 0x7FEDCBA987654321ll, jobj_);
635 EXPECT_EQ(result, 42);
636}
637
Elliott Hughesb264f082012-04-06 17:10:10 -0700638// This should return jclass, but we're imitating a bug pattern.
639jobject Java_MyClassNatives_instanceMethodThatShouldReturnClass(JNIEnv* env, jobject) {
640 return env->NewStringUTF("not a class!");
641}
642
643// This should return jclass, but we're imitating a bug pattern.
644jobject Java_MyClassNatives_staticMethodThatShouldReturnClass(JNIEnv* env, jclass) {
645 return env->NewStringUTF("not a class!");
646}
647
648TEST_F(JniCompilerTest, UpcallReturnTypeChecking_Instance) {
649 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
650 SetUpForTest(class_loader.get(), false, "instanceMethodThatShouldReturnClass", "()Ljava/lang/Class;",
651 reinterpret_cast<void*>(&Java_MyClassNatives_instanceMethodThatShouldReturnClass));
652
653 CheckJniAbortCatcher check_jni_abort_catcher;
654
655 // This native method is bad, and tries to return a jstring as a jclass.
656 env_->CallObjectMethod(jobj_, jmethod_);
657 check_jni_abort_catcher.Check("java.lang.Class MyClassNatives.instanceMethodThatShouldReturnClass");
658
659 // Here, we just call the method wrong; we should catch that too.
660 env_->CallVoidMethod(jobj_, jmethod_);
661 check_jni_abort_catcher.Check("java.lang.Class MyClassNatives.instanceMethodThatShouldReturnClass");
662}
663
664TEST_F(JniCompilerTest, UpcallReturnTypeChecking_Static) {
665 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
666 SetUpForTest(class_loader.get(), true, "staticMethodThatShouldReturnClass", "()Ljava/lang/Class;",
667 reinterpret_cast<void*>(&Java_MyClassNatives_staticMethodThatShouldReturnClass));
668
669 CheckJniAbortCatcher check_jni_abort_catcher;
670
671 // This native method is bad, and tries to return a jstring as a jclass.
672 env_->CallStaticObjectMethod(jklass_, jmethod_);
673 check_jni_abort_catcher.Check("java.lang.Class MyClassNatives.staticMethodThatShouldReturnClass");
674
675 // Here, we just call the method wrong; we should catch that too.
676 env_->CallVoidMethod(jobj_, jmethod_);
677 check_jni_abort_catcher.Check("java.lang.Class MyClassNatives.staticMethodThatShouldReturnClass");
678}
679
680// This should take jclass, but we're imitating a bug pattern.
681void Java_MyClassNatives_instanceMethodThatShouldTakeClass(JNIEnv*, jobject, jclass) {
682}
683
684// This should take jclass, but we're imitating a bug pattern.
685void Java_MyClassNatives_staticMethodThatShouldTakeClass(JNIEnv*, jclass, jclass) {
686}
687
688TEST_F(JniCompilerTest, UpcallArgumentTypeChecking_Instance) {
689 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
690 SetUpForTest(class_loader.get(), false, "instanceMethodThatShouldTakeClass", "(ILjava/lang/Class;)V",
691 reinterpret_cast<void*>(&Java_MyClassNatives_instanceMethodThatShouldTakeClass));
692
693 CheckJniAbortCatcher check_jni_abort_catcher;
694 // We deliberately pass a bad second argument here.
695 env_->CallVoidMethod(jobj_, jmethod_, 123, env_->NewStringUTF("not a class!"));
696 check_jni_abort_catcher.Check("Aborting because JNI app bug detected");
697}
698
699TEST_F(JniCompilerTest, UpcallArgumentTypeChecking_Static) {
700 SirtRef<ClassLoader> class_loader(LoadDex("MyClassNatives"));
701 SetUpForTest(class_loader.get(), true, "staticMethodThatShouldTakeClass", "(ILjava/lang/Class;)V",
702 reinterpret_cast<void*>(&Java_MyClassNatives_staticMethodThatShouldTakeClass));
703
704 CheckJniAbortCatcher check_jni_abort_catcher;
705 // We deliberately pass a bad second argument here.
706 env_->CallStaticVoidMethod(jklass_, jmethod_, 123, env_->NewStringUTF("not a class!"));
707 check_jni_abort_catcher.Check("Aborting because JNI app bug detected");
708}
709
Ian Rogersb033c752011-07-20 12:22:35 -0700710} // namespace art