Skip to content

Commit 6eb7c3a

Browse files
author
David Holmes
committedAug 12, 2022
8290732: JNI DestroyJavaVM can start shutdown when one non-daemon thread remains
Reviewed-by: stuefe, rehn
1 parent 083e014 commit 6eb7c3a

File tree

5 files changed

+906
-656
lines changed

5 files changed

+906
-656
lines changed
 

‎make/test/JtregNativeHotspot.gmk

+653-651
Large diffs are not rendered by default.

‎src/hotspot/share/runtime/threads.cpp

+11-5
Original file line numberDiff line numberDiff line change
@@ -1043,10 +1043,14 @@ void Threads::destroy_vm() {
10431043
#ifdef ASSERT
10441044
_vm_complete = false;
10451045
#endif
1046-
// Wait until we are the last non-daemon thread to execute
1046+
// Wait until we are the last non-daemon thread to execute, or
1047+
// if we are a daemon then wait until the last non-daemon thread has
1048+
// executed.
1049+
bool daemon = java_lang_Thread::is_daemon(thread->threadObj());
1050+
int expected = daemon ? 0 : 1;
10471051
{
10481052
MonitorLocker nu(Threads_lock);
1049-
while (Threads::number_of_non_daemon_threads() > 1)
1053+
while (Threads::number_of_non_daemon_threads() > expected)
10501054
// This wait should make safepoint checks, wait without a timeout.
10511055
nu.wait(0);
10521056
}
@@ -1223,9 +1227,11 @@ void Threads::remove(JavaThread* p, bool is_daemon) {
12231227
if (!is_daemon) {
12241228
_number_of_non_daemon_threads--;
12251229

1226-
// Only one thread left, do a notify on the Threads_lock so a thread waiting
1227-
// on destroy_vm will wake up.
1228-
if (number_of_non_daemon_threads() == 1) {
1230+
// If this is the last non-daemon thread then we need to do
1231+
// a notify on the Threads_lock so a thread waiting
1232+
// on destroy_vm will wake up. But that thread could be a daemon
1233+
// or non-daemon, so we notify for both the 0 and 1 case.
1234+
if (number_of_non_daemon_threads() <= 1) {
12291235
ml.notify_all();
12301236
}
12311237
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
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+
* Simple application that checks that shutdown does not commence until
27+
* after the last non-daemon thread has terminated. Reporting failure is
28+
* tricky because we can't uses exceptions (as they are ignored from Shutdown
29+
* hooks) and we can't call System.exit. So we rely on System.out being checked
30+
* in the TestDaemonDestroy application.
31+
*/
32+
public class Main {
33+
34+
static volatile Thread t1;
35+
36+
public static void main() {
37+
t1 = new Thread(() -> {
38+
System.out.println("T1 started");
39+
try {
40+
Thread.sleep(5000);
41+
} catch (InterruptedException ignore) { }
42+
System.out.println("T1 finished");
43+
}, "T1");
44+
45+
t1.setDaemon(false);
46+
47+
Thread hook = new Thread(() -> {
48+
System.out.println("HOOK started");
49+
if (t1.isAlive()) {
50+
System.out.println("Error: T1 isAlive");
51+
}
52+
}, "HOOK");
53+
Runtime.getRuntime().addShutdownHook(hook);
54+
t1.start();
55+
}
56+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
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 8290732
28+
* @comment Test uses custom launcher that attempts to destroy the VM on both
29+
* a daemon and non-daemon thread. The result should be the same in
30+
* both cases.
31+
* @requires vm.flagless
32+
* @library /test/lib
33+
* @build Main
34+
* @run main/native TestDaemonDestroy
35+
* @run main/native TestDaemonDestroy daemon
36+
*/
37+
38+
// Logic copied from SigTestDriver
39+
40+
import jdk.test.lib.Platform;
41+
import jdk.test.lib.Utils;
42+
import jdk.test.lib.process.OutputAnalyzer;
43+
44+
import java.io.File;
45+
import java.io.IOException;
46+
import java.nio.file.Files;
47+
import java.nio.file.Paths;
48+
import java.nio.file.Path;
49+
import java.util.ArrayList;
50+
import java.util.List;
51+
52+
public class TestDaemonDestroy {
53+
54+
public static void main(String[] args) throws IOException {
55+
Path launcher = Paths.get(Utils.TEST_NATIVE_PATH)
56+
.resolve("daemonDestroy" + (Platform.isWindows() ? ".exe" : ""))
57+
.toAbsolutePath();
58+
59+
System.out.println("Launcher = " + launcher +
60+
(Files.exists(launcher) ? " (exists)" : " (missing)"));
61+
62+
List<String> cmd = new ArrayList<>();
63+
cmd.add(launcher.toString());
64+
cmd.add("-Djava.class.path=" + Utils.TEST_CLASS_PATH);
65+
if (args.length > 0) {
66+
cmd.add("daemon");
67+
}
68+
ProcessBuilder pb = new ProcessBuilder(cmd);
69+
70+
// Need to add libjvm location to LD_LIBRARY_PATH
71+
String envVar = Platform.sharedLibraryPathVariableName();
72+
pb.environment().merge(envVar, Platform.jvmLibDir().toString(),
73+
(x, y) -> y + File.pathSeparator + x);
74+
75+
OutputAnalyzer oa = new OutputAnalyzer(pb.start());
76+
oa.shouldHaveExitValue(0);
77+
oa.shouldNotContain("Error: T1 isAlive");
78+
oa.shouldContain("T1 finished");
79+
oa.reportDiagnosticSummary();
80+
}
81+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
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 <jni.h>
25+
#include <stdlib.h>
26+
27+
static JavaVMOption options[] = {
28+
{ "-Djava.class.path=.", NULL }, // gets overwritten with real value
29+
};
30+
31+
static JavaVMInitArgs vm_args = {
32+
JNI_VERSION_19,
33+
sizeof(options) / sizeof(JavaVMOption),
34+
options,
35+
JNI_FALSE
36+
};
37+
38+
int main(int argc, char *argv[]) {
39+
JavaVM *jvm;
40+
JNIEnv *env;
41+
42+
if (argc < 2) {
43+
fprintf(stderr, "Usage: main <classpath property> [daemon]\n");
44+
exit(1);
45+
}
46+
47+
char* cp = argv[1];
48+
49+
printf("Test using classpath: %s\n", cp);
50+
51+
options[0].optionString = cp;
52+
53+
jint res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
54+
if (res != JNI_OK) {
55+
fprintf(stderr, "Test Error: JNI_CreateJavaVM failed: %d\n", res);
56+
exit(1);
57+
}
58+
59+
jclass cls = (*env)->FindClass(env, "Main");
60+
if (cls == NULL) {
61+
fprintf(stderr, "Test Error. Can't load class Main\n");
62+
(*env)->ExceptionDescribe(env);
63+
exit(1);
64+
}
65+
66+
jmethodID mid = (*env)->GetStaticMethodID(env, cls, "main", "()V");
67+
if (mid == NULL) {
68+
fprintf(stderr, "Test Error. Can't find method main\n");
69+
(*env)->ExceptionDescribe(env);
70+
exit(1);
71+
}
72+
73+
(*env)->CallStaticVoidMethod(env, cls, mid);
74+
75+
res = (*jvm)->DetachCurrentThread(jvm);
76+
if (res != JNI_OK) {
77+
fprintf(stderr, "Test Error: DetachCurrentThread failed: %d\n", res);
78+
exit(1);
79+
}
80+
81+
// Any additional arg implies to use a daemon thread.
82+
if (argc > 2) {
83+
res = (*jvm)->AttachCurrentThreadAsDaemon(jvm, (void **)&env, NULL);
84+
if (res != JNI_OK) {
85+
fprintf(stderr, "Test Error: AttachCurrentThreadAsDaemon failed: %d\n", res);
86+
exit(1);
87+
}
88+
puts("Test: attached as daemon");
89+
} else {
90+
res = (*jvm)->AttachCurrentThread(jvm, (void **)&env, NULL);
91+
if (res != JNI_OK) {
92+
fprintf(stderr, "Test Error: AttachCurrentThread failed: %d\n", res);
93+
exit(1);
94+
}
95+
puts("Test: attached as non-daemon");
96+
}
97+
98+
puts("Test: calling DestroyJavaVM");
99+
res = (*jvm)->DestroyJavaVM(jvm);
100+
if (res != JNI_OK) {
101+
fprintf(stderr, "Test Error: DestroyJavaVM failed: %d\n", res);
102+
exit(1);
103+
}
104+
puts("Test: DestroyJavaVM returned");
105+
}

0 commit comments

Comments
 (0)
Please sign in to comment.