blob: c7af269fe0b6271aab9a5e7260602f5398b4b92e [file] [log] [blame]
// Copyright 2011 Google Inc. All Rights Reserved.
// Author: irogers@google.com (Ian Rogers)
#include <sys/mman.h>
#include "assembler.h"
#include "class_linker.h"
#include "common_test.h"
#include "dex_file.h"
#include "jni_compiler.h"
#include "runtime.h"
#include "thread.h"
#include "gtest/gtest.h"
namespace art {
class JniCompilerTest : public RuntimeTest {
protected:
virtual void SetUp() {
RuntimeTest::SetUp();
// Create runtime and attach thread
std::vector<RawDexFile*> boot_class_path;
boot_class_path.push_back(java_lang_raw_dex_file_.get());
runtime_ = Runtime::Create(boot_class_path);
CHECK(runtime_->AttachCurrentThread());
// Create thunk code that performs the native to managed transition
thunk_code_size_ = 4096;
thunk_ = mmap(NULL, thunk_code_size_, PROT_READ | PROT_WRITE | PROT_EXEC,
MAP_ANONYMOUS | MAP_PRIVATE, -1, 0);
CHECK_NE(MAP_FAILED, thunk_);
Assembler thk_asm;
// TODO: shouldn't have machine specific code in a general purpose file
#if defined(__i386__)
thk_asm.pushl(EDI); // preserve EDI
thk_asm.movl(EAX, Address(ESP, 8)); // EAX = method->GetCode()
thk_asm.movl(EDI, Address(ESP, 12)); // EDI = method
thk_asm.pushl(Immediate(0)); // push pad
thk_asm.pushl(Immediate(0)); // push pad
thk_asm.pushl(Address(ESP, 40)); // push pad or jlong high
thk_asm.pushl(Address(ESP, 40)); // push jint or jlong low
thk_asm.pushl(Address(ESP, 40)); // push jint or jlong high
thk_asm.pushl(Address(ESP, 40)); // push jint or jlong low
thk_asm.pushl(Address(ESP, 40)); // push jobject
thk_asm.call(EAX); // Continue in method->GetCode()
thk_asm.addl(ESP, Immediate(28)); // pop arguments
thk_asm.popl(EDI); // restore EDI
thk_asm.ret();
#else
LOG(FATAL) << "Unimplemented";
#endif
size_t cs = thk_asm.CodeSize();
MemoryRegion code(thunk_, cs);
thk_asm.FinalizeInstructions(code);
thunk_entry1_ = reinterpret_cast<jint (*)(const void*, art::Method*,
jobject, jint, jint, jint)
>(code.pointer());
thunk_entry2_ = reinterpret_cast<jdouble (*)(const void*, art::Method*,
jobject, jdouble, jdouble)
>(code.pointer());
}
virtual void TearDown() {
// Release thunk code
CHECK(runtime_->DetachCurrentThread());
CHECK_EQ(0, munmap(thunk_, thunk_code_size_));
}
// Run generated code associated with method passing and returning int size
// arguments
jvalue RunMethod(Method* method, jvalue a, jvalue b, jvalue c, jvalue d) {
jvalue result;
// sanity checks
EXPECT_NE(static_cast<void*>(NULL), method->GetCode());
EXPECT_EQ(0u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kRunnable, Thread::Current()->GetState());
// perform call
result.i = (*thunk_entry1_)(method->GetCode(), method, a.l, b.i, c.i, d.i);
// sanity check post-call
EXPECT_EQ(0u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kRunnable, Thread::Current()->GetState());
return result;
}
// Run generated code associated with method passing and returning double size
// arguments
jvalue RunMethodD(Method* method, jvalue a, jvalue b, jvalue c) {
jvalue result;
// sanity checks
EXPECT_NE(static_cast<void*>(NULL), method->GetCode());
EXPECT_EQ(0u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kRunnable, Thread::Current()->GetState());
// perform call
result.d = (*thunk_entry2_)(method->GetCode(), method, a.l, b.d, c.d);
// sanity check post-call
EXPECT_EQ(0u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kRunnable, Thread::Current()->GetState());
return result;
}
Runtime* runtime_;
void* thunk_;
size_t thunk_code_size_;
jint (*thunk_entry1_)(const void*, Method*, jobject, jint, jint, jint);
jdouble (*thunk_entry2_)(const void*, Method*, jobject, jdouble, jdouble);
};
int gJava_MyClass_foo_calls = 0;
void Java_MyClass_foo(JNIEnv*, jobject) {
EXPECT_EQ(1u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_foo_calls++;
}
int gJava_MyClass_fooI_calls = 0;
jint Java_MyClass_fooI(JNIEnv*, jobject, jint x) {
EXPECT_EQ(1u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_fooI_calls++;
return x;
}
int gJava_MyClass_fooII_calls = 0;
jint Java_MyClass_fooII(JNIEnv*, jobject, jint x, jint y) {
EXPECT_EQ(1u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_fooII_calls++;
return x - y; // non-commutative operator
}
int gJava_MyClass_fooDD_calls = 0;
jdouble Java_MyClass_fooDD(JNIEnv*, jobject, jdouble x, jdouble y) {
EXPECT_EQ(1u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_fooDD_calls++;
return x - y; // non-commutative operator
}
int gJava_MyClass_fooIOO_calls = 0;
jobject Java_MyClass_fooIOO(JNIEnv*, jobject thisObject, jint x, jobject y,
jobject z) {
EXPECT_EQ(3u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_fooIOO_calls++;
switch (x) {
case 1:
return y;
case 2:
return z;
default:
return thisObject;
}
}
int gJava_MyClass_fooSIOO_calls = 0;
jobject Java_MyClass_fooSIOO(JNIEnv*, jclass klass, jint x, jobject y,
jobject z) {
EXPECT_EQ(3u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_fooSIOO_calls++;
switch (x) {
case 1:
return y;
case 2:
return z;
default:
return klass;
}
}
int gJava_MyClass_fooSSIOO_calls = 0;
jobject Java_MyClass_fooSSIOO(JNIEnv*, jclass klass, jint x, jobject y,
jobject z) {
EXPECT_EQ(3u, Thread::Current()->NumShbHandles());
EXPECT_EQ(Thread::kNative, Thread::Current()->GetState());
gJava_MyClass_fooSSIOO_calls++;
switch (x) {
case 1:
return y;
case 2:
return z;
default:
return klass;
}
}
TEST_F(JniCompilerTest, CompileAndRunNoArgMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("foo");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
// JNIEnv* env = Thread::Current()->GetJniEnv();
// JNINativeMethod methods[] = {{"foo", "()V", (void*)&Java_MyClass_foo}};
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_foo));
jvalue a;
a.l = (jobject)NULL;
gJava_MyClass_foo_calls = 0;
RunMethod(method, a, a, a, a);
EXPECT_EQ(1, gJava_MyClass_foo_calls);
RunMethod(method, a, a, a, a);
EXPECT_EQ(2, gJava_MyClass_foo_calls);
}
TEST_F(JniCompilerTest, CompileAndRunIntMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("fooI");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooI));
jvalue a, b, c;
a.l = (jobject)NULL;
b.i = 42;
EXPECT_EQ(0, gJava_MyClass_fooI_calls);
c = RunMethod(method, a, b, a, a);
ASSERT_EQ(42, c.i);
EXPECT_EQ(1, gJava_MyClass_fooI_calls);
b.i = 0xCAFED00D;
c = RunMethod(method, a, b, a, a);
ASSERT_EQ((jint)0xCAFED00D, c.i);
EXPECT_EQ(2, gJava_MyClass_fooI_calls);
}
TEST_F(JniCompilerTest, CompileAndRunIntIntMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("fooII");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooII));
jvalue a, b, c, d;
a.l = (jobject)NULL;
b.i = 99;
c.i = 10;
EXPECT_EQ(0, gJava_MyClass_fooII_calls);
d = RunMethod(method, a, b, c, a);
ASSERT_EQ(99 - 10, d.i);
EXPECT_EQ(1, gJava_MyClass_fooII_calls);
b.i = 0xCAFEBABE;
c.i = 0xCAFED00D;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jint)(0xCAFEBABE - 0xCAFED00D), d.i);
EXPECT_EQ(2, gJava_MyClass_fooII_calls);
}
TEST_F(JniCompilerTest, CompileAndRunDoubleDoubleMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("fooDD");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooDD));
jvalue a, b, c, d;
a.l = (jobject)NULL;
b.d = 99;
c.d = 10;
EXPECT_EQ(0, gJava_MyClass_fooDD_calls);
d = RunMethodD(method, a, b, c);
ASSERT_EQ(b.d - c.d, d.d);
EXPECT_EQ(1, gJava_MyClass_fooDD_calls);
b.d = 3.14159265358979323846;
c.d = 0.69314718055994530942;
d = RunMethodD(method, a, b, c);
ASSERT_EQ(b.d - c.d, d.d);
EXPECT_EQ(2, gJava_MyClass_fooDD_calls);
}
TEST_F(JniCompilerTest, CompileAndRunIntObjectObjectMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("fooIOO");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooIOO));
jvalue a, b, c, d, e;
a.l = (jobject)NULL;
b.i = 0;
c.l = (jobject)NULL;
d.l = (jobject)NULL;
EXPECT_EQ(0, gJava_MyClass_fooIOO_calls);
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)NULL, e.l);
EXPECT_EQ(1, gJava_MyClass_fooIOO_calls);
a.l = (jobject)8;
b.i = 0;
c.l = (jobject)NULL;
d.l = (jobject)16;
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)8, e.l);
EXPECT_EQ(2, gJava_MyClass_fooIOO_calls);
b.i = 1;
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)NULL, e.l);
EXPECT_EQ(3, gJava_MyClass_fooIOO_calls);
b.i = 2;
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)16, e.l);
EXPECT_EQ(4, gJava_MyClass_fooIOO_calls);
a.l = (jobject)8;
b.i = 0;
c.l = (jobject)16;
d.l = (jobject)NULL;
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)8, e.l);
EXPECT_EQ(5, gJava_MyClass_fooIOO_calls);
b.i = 1;
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)16, e.l);
EXPECT_EQ(6, gJava_MyClass_fooIOO_calls);
b.i = 2;
e = RunMethod(method, a, b, c, d);
ASSERT_EQ((jobject)NULL, e.l);
EXPECT_EQ(7, gJava_MyClass_fooIOO_calls);
}
TEST_F(JniCompilerTest, CompileAndRunStaticIntObjectObjectMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindDirectMethod("fooSIOO");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooSIOO));
jvalue a, b, c, d;
a.i = 0;
b.l = (jobject)NULL;
c.l = (jobject)NULL;
EXPECT_EQ(0, gJava_MyClass_fooSIOO_calls);
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)method->GetClass(), d.l);
EXPECT_EQ(1, gJava_MyClass_fooSIOO_calls);
a.i = 0;
b.l = (jobject)NULL;
c.l = (jobject)16;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)method->GetClass(), d.l);
EXPECT_EQ(2, gJava_MyClass_fooSIOO_calls);
a.i = 1;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)NULL, d.l);
EXPECT_EQ(3, gJava_MyClass_fooSIOO_calls);
a.i = 2;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)16, d.l);
EXPECT_EQ(4, gJava_MyClass_fooSIOO_calls);
a.i = 0;
b.l = (jobject)16;
c.l = (jobject)NULL;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)method->GetClass(), d.l);
EXPECT_EQ(5, gJava_MyClass_fooSIOO_calls);
a.i = 1;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)16, d.l);
EXPECT_EQ(6, gJava_MyClass_fooSIOO_calls);
a.i = 2;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)NULL, d.l);
EXPECT_EQ(7, gJava_MyClass_fooSIOO_calls);
}
TEST_F(JniCompilerTest, CompileAndRunStaticSynchronizedIntObjectObjectMethod) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindDirectMethod("fooSSIOO");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooSSIOO));
jvalue a, b, c, d;
a.i = 0;
b.l = (jobject)NULL;
c.l = (jobject)NULL;
EXPECT_EQ(0, gJava_MyClass_fooSSIOO_calls);
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)method->GetClass(), d.l);
EXPECT_EQ(1, gJava_MyClass_fooSSIOO_calls);
a.i = 0;
b.l = (jobject)NULL;
c.l = (jobject)16;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)method->GetClass(), d.l);
EXPECT_EQ(2, gJava_MyClass_fooSSIOO_calls);
a.i = 1;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)NULL, d.l);
EXPECT_EQ(3, gJava_MyClass_fooSSIOO_calls);
a.i = 2;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)16, d.l);
EXPECT_EQ(4, gJava_MyClass_fooSSIOO_calls);
a.i = 0;
b.l = (jobject)16;
c.l = (jobject)NULL;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)method->GetClass(), d.l);
EXPECT_EQ(5, gJava_MyClass_fooSSIOO_calls);
a.i = 1;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)16, d.l);
EXPECT_EQ(6, gJava_MyClass_fooSSIOO_calls);
a.i = 2;
d = RunMethod(method, a, b, c, a);
ASSERT_EQ((jobject)NULL, d.l);
EXPECT_EQ(7, gJava_MyClass_fooSSIOO_calls);
}
int gSuspendCounterHandler_calls;
void SuspendCountHandler(Method** frame) {
EXPECT_EQ(0, (*frame)->GetName().compare("fooI"));
gSuspendCounterHandler_calls++;
Thread::Current()->DecrementSuspendCount();
}
TEST_F(JniCompilerTest, SuspendCountAcknolewdgement) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("fooI");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_fooI));
Thread::Current()->RegisterSuspendCountEntryPoint(&SuspendCountHandler);
gSuspendCounterHandler_calls = 0;
gJava_MyClass_fooI_calls = 0;
jvalue a, b, c;
a.l = (jobject)NULL;
b.i = 42;
c = RunMethod(method, a, b, a, a);
ASSERT_EQ(42, c.i);
EXPECT_EQ(1, gJava_MyClass_fooI_calls);
EXPECT_EQ(0, gSuspendCounterHandler_calls);
Thread::Current()->IncrementSuspendCount();
c = RunMethod(method, a, b, a, a);
ASSERT_EQ(42, c.i);
EXPECT_EQ(2, gJava_MyClass_fooI_calls);
EXPECT_EQ(1, gSuspendCounterHandler_calls);
c = RunMethod(method, a, b, a, a);
ASSERT_EQ(42, c.i);
EXPECT_EQ(3, gJava_MyClass_fooI_calls);
EXPECT_EQ(1, gSuspendCounterHandler_calls);
}
int gExceptionHandler_calls;
void ExceptionHandler(Method** frame) {
EXPECT_EQ(0, (*frame)->GetName().compare("foo"));
gExceptionHandler_calls++;
Thread::Current()->ClearException();
}
TEST_F(JniCompilerTest, ExceptionHandling) {
scoped_ptr<RawDexFile> dex(OpenRawDexFileBase64(kMyClassNativesDex));
class_linker_.get()->RegisterDexFile(dex.get());
Class* klass = class_linker_.get()->FindClass("LMyClass;", NULL, dex.get());
Method* method = klass->FindVirtualMethod("foo");
Assembler jni_asm;
JniCompiler jni_compiler;
jni_compiler.Compile(&jni_asm, method);
// TODO: should really use JNIEnv to RegisterNative, but missing a
// complete story on this, so hack the RegisterNative below
method->RegisterNative(reinterpret_cast<void*>(&Java_MyClass_foo));
Thread::Current()->RegisterExceptionEntryPoint(&ExceptionHandler);
gExceptionHandler_calls = 0;
gJava_MyClass_foo_calls = 0;
jvalue a;
a.l = (jobject)NULL;
RunMethod(method, a, a, a, a);
EXPECT_EQ(1, gJava_MyClass_foo_calls);
EXPECT_EQ(0, gExceptionHandler_calls);
// TODO: create a real exception here
Thread::Current()->SetException(reinterpret_cast<Object*>(8));
RunMethod(method, a, a, a, a);
EXPECT_EQ(2, gJava_MyClass_foo_calls);
EXPECT_EQ(1, gExceptionHandler_calls);
RunMethod(method, a, a, a, a);
EXPECT_EQ(3, gJava_MyClass_foo_calls);
EXPECT_EQ(1, gExceptionHandler_calls);
}
} // namespace art