8188052: JNI FindClass needs to specify the class loading context used for library lifecycle hooks

Reviewed-by: alanb, coleenp, dholmes
This commit is contained in:
Mandy Chung 2017-10-10 11:52:42 -07:00
parent 42a29333b9
commit 4c1297b778
8 changed files with 27 additions and 20 deletions

View File

@ -92,7 +92,7 @@
#include "jvmci/jvmciRuntime.hpp" #include "jvmci/jvmciRuntime.hpp"
#endif #endif
static jint CurrentVersion = JNI_VERSION_9; static jint CurrentVersion = JNI_VERSION_10;
#ifdef _WIN32 #ifdef _WIN32
extern LONG WINAPI topLevelExceptionFilter(_EXCEPTION_POINTERS* ); extern LONG WINAPI topLevelExceptionFilter(_EXCEPTION_POINTERS* );
@ -401,29 +401,31 @@ JNI_ENTRY(jclass, jni_FindClass(JNIEnv *env, const char *name))
// Find calling class // Find calling class
Klass* k = thread->security_get_caller_class(0); Klass* k = thread->security_get_caller_class(0);
if (k != NULL) { if (k != NULL) {
loader = Handle(THREAD, k->class_loader());
// Special handling to make sure JNI_OnLoad and JNI_OnUnload are executed // Special handling to make sure JNI_OnLoad and JNI_OnUnload are executed
// in the correct class context. // in the correct class context.
if (loader.is_null() && if (k->class_loader() == NULL &&
k->name() == vmSymbols::java_lang_ClassLoader_NativeLibrary()) { k->name() == vmSymbols::java_lang_ClassLoader_NativeLibrary()) {
JavaValue result(T_OBJECT); JavaValue result(T_OBJECT);
JavaCalls::call_static(&result, k, JavaCalls::call_static(&result, k,
vmSymbols::getFromClass_name(), vmSymbols::getFromClass_name(),
vmSymbols::void_class_signature(), vmSymbols::void_class_signature(),
thread); CHECK_NULL);
if (HAS_PENDING_EXCEPTION) { // When invoked from JNI_OnLoad, NativeLibrary::getFromClass returns
Handle ex(thread, thread->pending_exception()); // a non-NULL Class object. When invoked from JNI_OnUnload,
CLEAR_PENDING_EXCEPTION; // it will return NULL to indicate no context.
THROW_HANDLE_0(ex);
}
oop mirror = (oop) result.get_jobject(); oop mirror = (oop) result.get_jobject();
loader = Handle(THREAD, if (mirror != NULL) {
InstanceKlass::cast(java_lang_Class::as_Klass(mirror))->class_loader()); Klass* fromClass = java_lang_Class::as_Klass(mirror);
protection_domain = Handle(THREAD, loader = Handle(THREAD, fromClass->class_loader());
InstanceKlass::cast(java_lang_Class::as_Klass(mirror))->protection_domain()); protection_domain = Handle(THREAD, fromClass->protection_domain());
}
} else {
loader = Handle(THREAD, k->class_loader());
} }
} else { }
// We call ClassLoader.getSystemClassLoader to obtain the system class loader.
if (loader.is_null()) {
// No context and use the system class loader
loader = Handle(THREAD, SystemDictionary::java_system_loader()); loader = Handle(THREAD, SystemDictionary::java_system_loader());
} }

View File

@ -1964,6 +1964,7 @@ JNI_OnUnload(JavaVM *vm, void *reserved);
#define JNI_VERSION_1_6 0x00010006 #define JNI_VERSION_1_6 0x00010006
#define JNI_VERSION_1_8 0x00010008 #define JNI_VERSION_1_8 0x00010008
#define JNI_VERSION_9 0x00090000 #define JNI_VERSION_9 0x00090000
#define JNI_VERSION_10 0x000a0000
#ifdef __cplusplus #ifdef __cplusplus
} /* extern "C" */ } /* extern "C" */

View File

@ -4195,6 +4195,7 @@ jboolean Threads::is_supported_jni_version(jint version) {
if (version == JNI_VERSION_1_6) return JNI_TRUE; if (version == JNI_VERSION_1_6) return JNI_TRUE;
if (version == JNI_VERSION_1_8) return JNI_TRUE; if (version == JNI_VERSION_1_8) return JNI_TRUE;
if (version == JNI_VERSION_9) return JNI_TRUE; if (version == JNI_VERSION_9) return JNI_TRUE;
if (version == JNI_VERSION_10) return JNI_TRUE;
return JNI_FALSE; return JNI_FALSE;
} }

View File

@ -2381,7 +2381,7 @@ public abstract class ClassLoader {
private int jniVersion; private int jniVersion;
// the class from which the library is loaded, also indicates // the class from which the library is loaded, also indicates
// the loader this native library belongs. // the loader this native library belongs.
private final Class<?> fromClass; private Class<?> fromClass;
// the canonicalized name of the native library. // the canonicalized name of the native library.
// or static library name // or static library name
String name; String name;
@ -2404,6 +2404,8 @@ public abstract class ClassLoader {
protected void finalize() { protected void finalize() {
synchronized (loadedLibraryNames) { synchronized (loadedLibraryNames) {
if (fromClass.getClassLoader() != null && loaded) { if (fromClass.getClassLoader() != null && loaded) {
this.fromClass = null; // no context when unloaded
/* remove the native library name */ /* remove the native library name */
int size = loadedLibraryNames.size(); int size = loadedLibraryNames.size();
for (int i = 0; i < size; i++) { for (int i = 0; i < size; i++) {

View File

@ -1964,6 +1964,7 @@ JNI_OnUnload(JavaVM *vm, void *reserved);
#define JNI_VERSION_1_6 0x00010006 #define JNI_VERSION_1_6 0x00010006
#define JNI_VERSION_1_8 0x00010008 #define JNI_VERSION_1_8 0x00010008
#define JNI_VERSION_9 0x00090000 #define JNI_VERSION_9 0x00090000
#define JNI_VERSION_10 0x000a0000
#ifdef __cplusplus #ifdef __cplusplus
} /* extern "C" */ } /* extern "C" */

View File

@ -45,7 +45,7 @@ JNIEXPORT jint JNICALL DEF_JNI_OnLoad(JavaVM *vm, void *reserved)
return JNI_EVERSION; /* JNI version not supported */ return JNI_EVERSION; /* JNI version not supported */
} }
return JNI_VERSION_9; return JNI_VERSION_10;
} }
/* /*

View File

@ -39,7 +39,7 @@ JNIEXPORT jint JNICALL DEF_JNI_OnLoad(JavaVM *vm, void *reserved)
return JNI_EVERSION; /* JNI version not supported */ return JNI_EVERSION; /* JNI version not supported */
} }
return JNI_VERSION_9; return JNI_VERSION_10;
} }

View File

@ -27,12 +27,12 @@
*/ */
public class JniVersion { public class JniVersion {
public static final int JNI_VERSION_9 = 0x00090000; public static final int JNI_VERSION_10 = 0x000a0000;
public static void main(String... args) throws Exception { public static void main(String... args) throws Exception {
System.loadLibrary("JniVersion"); System.loadLibrary("JniVersion");
int res = getJniVersion(); int res = getJniVersion();
if (res != JNI_VERSION_9) { if (res != JNI_VERSION_10) {
throw new Exception("Unexpected value returned from getJniVersion(): 0x" + Integer.toHexString(res)); throw new Exception("Unexpected value returned from getJniVersion(): 0x" + Integer.toHexString(res));
} }
} }