Skip to content

Commit a6f8a38

Browse files
tprinzingMandy Chung
authored and
Mandy Chung
committedFeb 17, 2022
8281000: ClassLoader::registerAsParallelCapable throws NPE if caller is null
Reviewed-by: erikj, ihse, mchung, bchristi
1 parent 4c7f8b4 commit a6f8a38

File tree

5 files changed

+216
-0
lines changed

5 files changed

+216
-0
lines changed
 

‎make/test/JtregNativeJdk.gmk

+2
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,7 @@ ifeq ($(call isTargetOs, windows), true)
6262
WIN_LIB_JLI := $(SUPPORT_OUTPUTDIR)/native/java.base/libjli/jli.lib
6363
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeJliLaunchTest := $(WIN_LIB_JLI)
6464
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeCallerAccessTest := jvm.lib
65+
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeNullCallerClassLoaderTest := jvm.lib
6566
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeNullCallerLookupTest := jvm.lib
6667
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exerevokeall := advapi32.lib
6768
BUILD_JDK_JTREG_LIBRARIES_CFLAGS_libAsyncStackWalk := /EHsc
@@ -81,6 +82,7 @@ else
8182
endif
8283
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeJliLaunchTest := -ljli
8384
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeCallerAccessTest := -ljvm
85+
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeNullCallerClassLoaderTest := -ljvm
8486
BUILD_JDK_JTREG_EXECUTABLES_LIBS_exeNullCallerLookupTest := -ljvm
8587
endif
8688

‎src/java.base/share/classes/java/lang/ClassLoader.java

+10
Original file line numberDiff line numberDiff line change
@@ -1606,9 +1606,16 @@ protected Enumeration<URL> findResources(String name) throws IOException {
16061606
* </ol>
16071607
* <p>Note that once a class loader is registered as parallel capable, there
16081608
* is no way to change it back.</p>
1609+
* <p>
1610+
* In cases where this method is called from a context where the caller is
1611+
* not a subclass of {@code ClassLoader} or there is no caller frame on the
1612+
* stack (e.g. when called directly from a JNI attached thread),
1613+
* {@code IllegalCallerException} is thrown.
1614+
* </p>
16091615
*
16101616
* @return {@code true} if the caller is successfully registered as
16111617
* parallel capable and {@code false} if otherwise.
1618+
* @throws IllegalCallerException if the caller is not a subclass of {@code ClassLoader}
16121619
*
16131620
* @see #isRegisteredAsParallelCapable()
16141621
*
@@ -1622,6 +1629,9 @@ protected static boolean registerAsParallelCapable() {
16221629
// Caller-sensitive adapter method for reflective invocation
16231630
@CallerSensitiveAdapter
16241631
private static boolean registerAsParallelCapable(Class<?> caller) {
1632+
if ((caller == null) || !ClassLoader.class.isAssignableFrom(caller)) {
1633+
throw new IllegalCallerException(caller + " not a subclass of ClassLoader");
1634+
}
16251635
return ParallelLoaders.register(caller.asSubclass(ClassLoader.class));
16261636
}
16271637

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8281000
27+
* @summary Test ClassLoader.isRegisteredAsParallelCapable() method with an
28+
* invalid caller (non-JNI cases). This test uses reflection and opens
29+
* the java.base module so it runs separate from behavior tests of
30+
* isRegisteredParallelCapable to avoid side effects.
31+
* @modules java.base/java.lang:open
32+
* @run main/othervm BadRegisterAsParallelCapableCaller
33+
*/
34+
35+
import java.lang.reflect.InvocationTargetException;
36+
import java.lang.reflect.Method;
37+
38+
public class BadRegisterAsParallelCapableCaller {
39+
40+
public static void main(String[] args) throws Exception {
41+
42+
Throwable thrown = null;
43+
final Method m = ClassLoader.class.getDeclaredMethod("registerAsParallelCapable");
44+
m.setAccessible(true);
45+
try {
46+
m.invoke(null);
47+
} catch (InvocationTargetException ite) {
48+
thrown = ite.getCause();
49+
}
50+
if (! (thrown instanceof IllegalCallerException)) {
51+
throw new RuntimeException("Didn't get expected IllegalCallerException, got "+ thrown);
52+
} else {
53+
System.out.println("Invalid caller threw IllegalCallerException as expected");
54+
}
55+
}
56+
57+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
/**
26+
* @test
27+
* @bug 8281000
28+
* @summary Test uses custom launcher that starts VM using JNI that verifies
29+
* ClassLoader.registerAsParallelCapable with null caller class
30+
* throws an IllegalCallerException.
31+
* @library /test/lib
32+
* @requires os.family != "aix"
33+
* @run main/native NullCallerClassLoaderTest
34+
*/
35+
36+
// Test disabled on AIX since we cannot invoke the JVM on the primordial thread.
37+
38+
import java.io.File;
39+
import java.util.Map;
40+
import jdk.test.lib.Platform;
41+
import jdk.test.lib.process.OutputAnalyzer;
42+
43+
import java.io.IOException;
44+
import java.nio.file.Path;
45+
import java.nio.file.Paths;
46+
47+
public class NullCallerClassLoaderTest {
48+
public static void main(String[] args) throws IOException {
49+
Path launcher = Path.of(System.getProperty("test.nativepath"), "NullCallerClassLoaderTest");
50+
ProcessBuilder pb = new ProcessBuilder(launcher.toString());
51+
Map<String, String> env = pb.environment();
52+
53+
String libDir = Platform.libDir().toString();
54+
String vmDir = Platform.jvmLibDir().toString();
55+
56+
// set up shared library path
57+
String sharedLibraryPathEnvName = Platform.sharedLibraryPathVariableName();
58+
env.compute(sharedLibraryPathEnvName,
59+
(k, v) -> (v == null) ? libDir : v + File.pathSeparator + libDir);
60+
env.compute(sharedLibraryPathEnvName,
61+
(k, v) -> (v == null) ? vmDir : v + File.pathSeparator + vmDir);
62+
63+
System.out.println("Launching: " + launcher + " shared library path: " +
64+
env.get(sharedLibraryPathEnvName));
65+
new OutputAnalyzer(pb.start())
66+
.outputTo(System.out)
67+
.errorTo(System.err)
68+
.shouldHaveExitValue(0);
69+
}
70+
}
71+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
#include <stdio.h>
25+
#include <stdlib.h>
26+
27+
#include "jni.h"
28+
#include "assert.h"
29+
30+
static jclass class_IllegalCallerException;
31+
32+
int checkAndClearIllegalCallerExceptionThrown(JNIEnv *env) {
33+
jthrowable t = (*env)->ExceptionOccurred(env);
34+
if ((*env)->IsInstanceOf(env, t, class_IllegalCallerException) == JNI_TRUE) {
35+
(*env)->ExceptionClear(env);
36+
return JNI_TRUE;
37+
}
38+
return JNI_FALSE;
39+
}
40+
41+
int main(int argc, char** args) {
42+
JavaVM *jvm;
43+
JNIEnv *env;
44+
JavaVMInitArgs vm_args;
45+
JavaVMOption options[1];
46+
jint rc;
47+
48+
49+
vm_args.version = JNI_VERSION_1_2;
50+
vm_args.nOptions = 0;
51+
vm_args.options = options;
52+
53+
if ((rc = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args)) != JNI_OK) {
54+
printf("ERROR: cannot create VM.\n");
55+
exit(-1);
56+
}
57+
class_IllegalCallerException = (*env)->FindClass(env, "java/lang/IllegalCallerException");
58+
assert (class_IllegalCallerException != NULL);
59+
60+
// call ClassLoader.registerAsParallelCapable()
61+
jclass class_ClassLoader = (*env)->FindClass(env, "java/lang/ClassLoader");
62+
assert(class_ClassLoader != NULL);
63+
jmethodID mid_ClassLoader_registerAsParallelCapable = (*env)->GetStaticMethodID(env, class_ClassLoader, "registerAsParallelCapable", "()Z" );
64+
assert(mid_ClassLoader_registerAsParallelCapable != NULL);
65+
jboolean b = (*env)->CallStaticBooleanMethod(env, class_ClassLoader, mid_ClassLoader_registerAsParallelCapable );
66+
if ((rc = checkAndClearIllegalCallerExceptionThrown(env)) != JNI_TRUE) {
67+
printf("ERROR: Didn't get the expected IllegalCallerException.\n");
68+
exit(-1);
69+
}
70+
71+
printf("Expected IllegalCallerException was thrown\n");
72+
73+
(*jvm)->DestroyJavaVM(jvm);
74+
return 0;
75+
}
76+

0 commit comments

Comments
 (0)
Please sign in to comment.