Skip to content

Commit e8568b8

Browse files
LudwikJaniukegahlin
authored andcommittedJul 12, 2022
8290020: Deadlock in leakprofiler::emit_events during shutdown
Reviewed-by: mgronlun, dholmes, egahlin
1 parent 7f0e9bd commit e8568b8

File tree

6 files changed

+30
-34
lines changed

6 files changed

+30
-34
lines changed
 

‎src/hotspot/share/jfr/jfr.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -116,8 +116,8 @@ void Jfr::on_resolution(const Parse* parse, const ciKlass* holder, const ciMetho
116116
}
117117
#endif
118118

119-
void Jfr::on_vm_shutdown(bool exception_handler) {
120-
if (JfrRecorder::is_recording()) {
119+
void Jfr::on_vm_shutdown(bool exception_handler, bool halt) {
120+
if (!halt && JfrRecorder::is_recording()) {
121121
JfrEmergencyDump::on_vm_shutdown(exception_handler);
122122
}
123123
}

‎src/hotspot/share/jfr/jfr.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ class Jfr : AllStatic {
6565
static void on_resolution(const GraphBuilder* builder, const ciKlass* holder, const ciMethod* target);
6666
static void on_java_thread_start(JavaThread* starter, JavaThread* startee);
6767
static void on_set_current_thread(JavaThread* jt, oop thread);
68-
static void on_vm_shutdown(bool exception_handler = false);
68+
static void on_vm_shutdown(bool exception_handler = false, bool halt = false);
6969
static void on_vm_error_report(outputStream* st);
7070
static bool on_flight_recorder_option(const JavaVMOption** option, char* delimiter);
7171
static bool on_start_flight_recording_option(const JavaVMOption** option, char* delimiter);

‎src/hotspot/share/prims/jvm.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -441,7 +441,7 @@ JVM_END
441441

442442

443443
JVM_ENTRY_NO_ENV(void, JVM_Halt(jint code))
444-
before_exit(thread);
444+
before_exit(thread, true);
445445
vm_exit(code);
446446
JVM_END
447447

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

+2-2
Original file line numberDiff line numberDiff line change
@@ -401,7 +401,7 @@ void print_statistics() {
401401
// Note: before_exit() can be executed only once, if more than one threads
402402
// are trying to shutdown the VM at the same time, only one thread
403403
// can run before_exit() and all other threads must wait.
404-
void before_exit(JavaThread* thread) {
404+
void before_exit(JavaThread* thread, bool halt) {
405405
#define BEFORE_EXIT_NOT_RUN 0
406406
#define BEFORE_EXIT_RUNNING 1
407407
#define BEFORE_EXIT_DONE 2
@@ -448,7 +448,7 @@ void before_exit(JavaThread* thread) {
448448
event.commit();
449449
}
450450

451-
JFR_ONLY(Jfr::on_vm_shutdown();)
451+
JFR_ONLY(Jfr::on_vm_shutdown(false, halt);)
452452

453453
// Stop the WatcherThread. We do this before disenrolling various
454454
// PeriodicTasks to reduce the likelihood of races.

‎src/hotspot/share/runtime/java.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ class JavaThread;
3333
class Symbol;
3434

3535
// Execute code before all handles are released and thread is killed; prologue to vm_exit
36-
extern void before_exit(JavaThread * thread);
36+
extern void before_exit(JavaThread * thread, bool halt = false);
3737

3838
// Forced VM exit (i.e, internal error or JVM_Exit)
3939
extern void vm_exit(int code);

‎test/jdk/jdk/jfr/jvm/TestDumpOnCrash.java

+23-27
Original file line numberDiff line numberDiff line change
@@ -76,41 +76,33 @@ public static void main(String[] args) throws Exception {
7676

7777
public static void main(String[] args) throws Exception {
7878
// Test without dumppath
79-
test(CrasherIllegalAccess.class, "", true);
80-
test(CrasherIllegalAccess.class, "", false);
81-
test(CrasherHalt.class, "", true);
82-
test(CrasherHalt.class, "", false);
79+
test(CrasherIllegalAccess.class, "", true, null, true);
80+
test(CrasherIllegalAccess.class, "", false, null, true);
81+
82+
// JDK-8290020 disables dumps when calling halt, so expect no dump.
83+
test(CrasherHalt.class, "", true, null, false);
84+
test(CrasherHalt.class, "", false, null, false);
8385

8486
// Test with dumppath
8587
Path dumppath = Files.createTempDirectory(null);
8688
try {
87-
test(CrasherIllegalAccess.class, "", true, dumppath.toString());
88-
test(CrasherIllegalAccess.class, "", false, dumppath.toString());
89-
test(CrasherHalt.class, "", true, dumppath.toString());
90-
test(CrasherHalt.class, "", false, dumppath.toString());
89+
test(CrasherIllegalAccess.class, "", true, dumppath.toString(), true);
90+
test(CrasherIllegalAccess.class, "", false, dumppath.toString(), true);
9191
} finally {
9292
dumppath.toFile().delete();
9393
}
9494

9595
// Test is excluded until 8219680 is fixed
9696
// @ignore 8219680
97-
// test(CrasherSig.class, "FPE", true);
98-
}
99-
100-
private static void test(Class<?> crasher, String signal, boolean disk) throws Exception {
101-
test(crasher, signal, disk, null);
97+
// test(CrasherSig.class, "FPE", true, true);
10298
}
10399

104-
private static void test(Class<?> crasher, String signal, boolean disk, String dumppath) throws Exception {
105-
test(crasher, signal, disk, dumppath, dumppath);
106-
}
107-
108-
private static void test(Class<?> crasher, String signal, boolean disk, String dumppath, String expectedPath) throws Exception {
100+
private static void test(Class<?> crasher, String signal, boolean disk, String dumppath, boolean expectDump) throws Exception {
109101
// The JVM may be in a state it can't recover from, so try three times
110102
// before concluding functionality is not working.
111103
for (int attempt = 0; attempt < ATTEMPTS; attempt++) {
112104
try {
113-
verify(runProcess(crasher, signal, disk, dumppath), expectedPath);
105+
verify(runProcess(crasher, signal, disk, dumppath), dumppath, expectDump);
114106
return;
115107
} catch (Exception e) {
116108
System.out.println("Attempt " + attempt + ". Verification failed:");
@@ -148,19 +140,23 @@ private static long runProcess(Class<?> crasher, String signal, boolean disk, St
148140
return p.pid();
149141
}
150142

151-
private static void verify(long pid, String dumppath) throws IOException {
143+
private static void verify(long pid, String dumppath, boolean expectDump) throws IOException {
152144
String fileName = "hs_err_pid" + pid + ".jfr";
153145
Path file = (dumppath == null) ? Paths.get(fileName) : Paths.get(dumppath, fileName);
154146
file = file.toAbsolutePath().normalize();
155147

156-
Asserts.assertTrue(Files.exists(file), "No emergency jfr recording file " + file + " exists");
157-
Asserts.assertNotEquals(Files.size(file), 0L, "File length 0. Should at least be some bytes");
158-
System.out.printf("File size=%d%n", Files.size(file));
148+
if (expectDump) {
149+
Asserts.assertTrue(Files.exists(file), "No emergency jfr recording file " + file + " exists");
150+
Asserts.assertNotEquals(Files.size(file), 0L, "File length 0. Should at least be some bytes");
151+
System.out.printf("File size=%d%n", Files.size(file));
159152

160-
List<RecordedEvent> events = RecordingFile.readAllEvents(file);
161-
Asserts.assertFalse(events.isEmpty(), "No event found");
162-
System.out.printf("Found event %s%n", events.get(0).getEventType().getName());
153+
List<RecordedEvent> events = RecordingFile.readAllEvents(file);
154+
Asserts.assertFalse(events.isEmpty(), "No event found");
155+
System.out.printf("Found event %s%n", events.get(0).getEventType().getName());
163156

164-
Files.delete(file);
157+
Files.delete(file);
158+
} else {
159+
Asserts.assertFalse(Files.exists(file), "Emergency jfr recording file " + file + " exists but wasn't expected");
160+
}
165161
}
166162
}

0 commit comments

Comments
 (0)
Please sign in to comment.