Skip to content

Commit 819d2df

Browse files
committedOct 21, 2021
8274794: Print all owned locks in hs_err file
Reviewed-by: stuefe, dholmes
1 parent c41ce6d commit 819d2df

File tree

8 files changed

+242
-110
lines changed

8 files changed

+242
-110
lines changed
 

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

+68-15
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
#include "runtime/osThread.hpp"
3232
#include "runtime/safepointMechanism.inline.hpp"
3333
#include "runtime/thread.inline.hpp"
34+
#include "runtime/threadCritical.hpp"
3435
#include "utilities/events.hpp"
3536
#include "utilities/macros.hpp"
3637

@@ -267,7 +268,36 @@ bool Monitor::wait(int64_t timeout) {
267268
return wait_status != 0; // return true IFF timeout
268269
}
269270

271+
// Array used to print owned locks on error.
272+
static Mutex* _mutex_array = NULL;
273+
274+
void Mutex::add_to_global_list() {
275+
// Add mutex to print_owned_locks_on_error array
276+
ThreadCritical tc;
277+
Mutex* next = _mutex_array;
278+
_next_mutex = next;
279+
_prev_mutex = nullptr;
280+
_mutex_array = this;
281+
if (next != nullptr) {
282+
next->_prev_mutex = this;
283+
}
284+
}
285+
286+
void Mutex::remove_from_global_list() {
287+
// Remove mutex from print_owned_locks_on_error array
288+
ThreadCritical tc;
289+
Mutex* old_next = _next_mutex;
290+
assert(old_next != nullptr, "this list can never be empty");
291+
old_next->_prev_mutex = _prev_mutex;
292+
if (_prev_mutex == nullptr) {
293+
_mutex_array = old_next;
294+
} else {
295+
_prev_mutex->_next_mutex = old_next;
296+
}
297+
}
298+
270299
Mutex::~Mutex() {
300+
remove_from_global_list();
271301
assert_owner(NULL);
272302
os::free(const_cast<char*>(_name));
273303
}
@@ -280,6 +310,8 @@ Mutex::Mutex(Rank rank, const char * name, bool allow_vm_block) : _owner(NULL) {
280310
_allow_vm_block = allow_vm_block;
281311
_rank = rank;
282312
_skip_rank_check = false;
313+
_next = nullptr;
314+
_last_owner = nullptr;
283315

284316
assert(_rank >= static_cast<Rank>(0) && _rank <= safepoint, "Bad lock rank %s: %s", rank_name(), name);
285317

@@ -288,16 +320,51 @@ Mutex::Mutex(Rank rank, const char * name, bool allow_vm_block) : _owner(NULL) {
288320
assert(_rank > nosafepoint || _allow_vm_block,
289321
"Locks that don't check for safepoint should always allow the vm to block: %s", name);
290322
#endif
323+
add_to_global_list();
291324
}
292325

293326
bool Mutex::owned_by_self() const {
294327
return owner() == Thread::current();
295328
}
296329

297-
void Mutex::print_on_error(outputStream* st) const {
330+
void Mutex::print_on(outputStream* st) const {
298331
st->print("[" PTR_FORMAT, p2i(this));
299332
st->print("] %s", _name);
300333
st->print(" - owner thread: " PTR_FORMAT, p2i(owner()));
334+
#ifdef ASSERT
335+
if (_allow_vm_block) {
336+
st->print_raw(" allow_vm_block");
337+
}
338+
st->print(" %s", rank_name());
339+
#endif
340+
st->cr();
341+
}
342+
343+
// Print all mutexes/monitors that are currently owned by a thread; called
344+
// by fatal error handler.
345+
// This function doesn't take the ThreadCritical lock to avoid potential
346+
// deadlock during error reporting.
347+
void Mutex::print_owned_locks_on_error(outputStream* st) {
348+
assert(VMError::is_error_reported(), "should only be called during error reporting");
349+
ResourceMark rm;
350+
st->print("VM Mutexes/Monitors currently owned by a thread: ");
351+
bool none = true;
352+
Mutex *m = _mutex_array;
353+
int array_count = 0;
354+
while (m != nullptr) {
355+
array_count++;
356+
// see if it has an owner
357+
if (m->owner() != NULL) {
358+
if (none) {
359+
st->cr();
360+
none = false;
361+
}
362+
m->print_on(st);
363+
}
364+
m = m->_next_mutex;
365+
}
366+
if (none) st->print_cr("None");
367+
st->print_cr("Total Mutex count %d", array_count);
301368
}
302369

303370
// ----------------------------------------------------------------------------------
@@ -343,21 +410,7 @@ void Mutex::assert_no_overlap(Rank orig, Rank adjusted, int adjust) {
343410
rank_name_internal(orig), adjust, rank_name_internal(adjusted));
344411
}
345412
}
346-
#endif // ASSERT
347-
348-
#ifndef PRODUCT
349-
void Mutex::print_on(outputStream* st) const {
350-
st->print("Mutex: [" PTR_FORMAT "] %s - owner: " PTR_FORMAT,
351-
p2i(this), _name, p2i(owner()));
352-
if (_allow_vm_block) {
353-
st->print("%s", " allow_vm_block");
354-
}
355-
DEBUG_ONLY(st->print(" %s", rank_name()));
356-
st->cr();
357-
}
358-
#endif // PRODUCT
359413

360-
#ifdef ASSERT
361414
void Mutex::assert_owner(Thread * expected) {
362415
const char* msg = "invalid owner";
363416
if (expected == NULL) {

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

+15-10
Original file line numberDiff line numberDiff line change
@@ -88,19 +88,24 @@ class Mutex : public CHeapObj<mtSynchronizer> {
8888
Thread* volatile _owner;
8989
void raw_set_owner(Thread* new_owner) { Atomic::store(&_owner, new_owner); }
9090

91+
// Embed pointers for mutex array for error reporting.
92+
Mutex* _next_mutex;
93+
Mutex* _prev_mutex;
94+
95+
void add_to_global_list();
96+
void remove_from_global_list();
97+
9198
protected: // Monitor-Mutex metadata
9299
os::PlatformMonitor _lock; // Native monitor implementation
93100
const char* _name; // Name of mutex/monitor
94101

95-
// Debugging fields for naming, deadlock detection, etc. (some only used in debug mode)
96-
#ifndef PRODUCT
97-
bool _allow_vm_block;
98-
#endif
102+
// Debugging fields for naming, deadlock detection, etc.
99103
#ifdef ASSERT
100104
Rank _rank; // rank (to avoid/detect potential deadlocks)
101105
Mutex* _next; // Used by a Thread to link up owned locks
102106
Thread* _last_owner; // the last thread to own the lock
103-
bool _skip_rank_check; // read only by owner when doing rank checks
107+
bool _skip_rank_check; // read only by owner when doing rank checks
108+
bool _allow_vm_block;
104109

105110
static bool contains(Mutex* locks, Mutex* lock);
106111
static Mutex* get_least_ranked_lock(Mutex* locks);
@@ -189,11 +194,11 @@ class Mutex : public CHeapObj<mtSynchronizer> {
189194

190195
const char *name() const { return _name; }
191196

192-
void print_on_error(outputStream* st) const;
193-
#ifndef PRODUCT
194-
void print_on(outputStream* st) const;
195-
void print() const { print_on(::tty); }
196-
#endif
197+
// Print all mutexes/monitors that are currently owned by a thread; called
198+
// by fatal error handler.
199+
static void print_owned_locks_on_error(outputStream* st);
200+
void print_on(outputStream* st) const;
201+
void print() const { print_on(::tty); }
197202
};
198203

199204
class Monitor : public Mutex {

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

-33
Original file line numberDiff line numberDiff line change
@@ -157,11 +157,6 @@ Mutex* Bootclasspath_lock = NULL;
157157
Monitor* JVMCI_lock = NULL;
158158
#endif
159159

160-
161-
#define MAX_NUM_MUTEX 128
162-
static Mutex* _mutex_array[MAX_NUM_MUTEX];
163-
static int _num_mutex;
164-
165160
#ifdef ASSERT
166161
void assert_locked_or_safepoint(const Mutex* lock) {
167162
// check if this thread owns the lock (common case)
@@ -194,26 +189,18 @@ void assert_locked_or_safepoint_or_handshake(const Mutex* lock, const JavaThread
194189
}
195190
#endif
196191

197-
static void add_mutex(Mutex* var) {
198-
assert(_num_mutex < MAX_NUM_MUTEX, "increase MAX_NUM_MUTEX");
199-
_mutex_array[_num_mutex++] = var;
200-
}
201-
202192
#define def(var, type, pri, vm_block) { \
203193
var = new type(Mutex::pri, #var, vm_block); \
204-
add_mutex(var); \
205194
}
206195

207196
// Specify relative ranked lock
208197
#ifdef ASSERT
209198
#define defl(var, type, held_lock, vm_block) { \
210199
var = new type(held_lock->rank()-1, #var, vm_block); \
211-
add_mutex(var); \
212200
}
213201
#else
214202
#define defl(var, type, held_lock, vm_block) { \
215203
var = new type(Mutex::safepoint, #var, vm_block); \
216-
add_mutex(var); \
217204
}
218205
#endif
219206

@@ -377,23 +364,3 @@ GCMutexLocker::GCMutexLocker(Mutex* mutex) {
377364
_mutex->lock();
378365
}
379366
}
380-
381-
// Print all mutexes/monitors that are currently owned by a thread; called
382-
// by fatal error handler.
383-
void print_owned_locks_on_error(outputStream* st) {
384-
st->print("VM Mutex/Monitor currently owned by a thread: ");
385-
bool none = true;
386-
for (int i = 0; i < _num_mutex; i++) {
387-
// see if it has an owner
388-
if (_mutex_array[i]->owner() != NULL) {
389-
if (none) {
390-
// print format used by Mutex::print_on_error()
391-
st->print_cr(" ([mutex/lock_event])");
392-
none = false;
393-
}
394-
_mutex_array[i]->print_on_error(st);
395-
st->cr();
396-
}
397-
}
398-
if (none) st->print_cr("None");
399-
}

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

-6
Original file line numberDiff line numberDiff line change
@@ -169,12 +169,6 @@ extern Mutex* tty_lock; // lock to synchronize output.
169169
// order. If their implementations change such that these assumptions
170170
// are violated, a whole lot of code will break.
171171

172-
// Print all mutexes/monitors that are currently owned by a thread; called
173-
// by fatal error handler.
174-
void print_owned_locks_on_error(outputStream* st);
175-
176-
char *lock_name(Mutex *mutex);
177-
178172
// for debugging: check that we're already owning this lock (or are at a safepoint / handshake)
179173
#ifdef ASSERT
180174
void assert_locked_or_safepoint(const Mutex* lock);

‎src/hotspot/share/utilities/vmError.cpp

+7-1
Original file line numberDiff line numberDiff line change
@@ -1015,7 +1015,7 @@ void VMError::report(outputStream* st, bool _verbose) {
10151015

10161016
// mutexes/monitors that currently have an owner
10171017
if (_verbose) {
1018-
print_owned_locks_on_error(st);
1018+
Mutex::print_owned_locks_on_error(st);
10191019
st->cr();
10201020
}
10211021

@@ -1913,6 +1913,12 @@ void VMError::controlled_crash(int how) {
19131913
switch (how) {
19141914
case 1: assert(how == 0, "test assert"); break;
19151915
case 2: guarantee(how == 0, "test guarantee"); break;
1916+
case 3: {
1917+
Mutex* ErrorTest_lock = new Mutex(Mutex::nosafepoint, "ErrorTest_lock");
1918+
MutexLocker ml(ErrorTest_lock, Mutex::_no_safepoint_check_flag);
1919+
assert(how == 0, "test assert with lock");
1920+
break;
1921+
}
19161922

19171923
// The other cases are unused.
19181924
case 14: crash_with_segfault(); break;

‎test/hotspot/jtreg/runtime/ErrorHandling/ErrorFileOverwriteTest.java

+5-45
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2019, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2019, 2021, Oracle and/or its affiliates. All rights reserved.
33
* Copyright (c) 2019, SAP. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
@@ -42,46 +42,6 @@
4242

4343
public class ErrorFileOverwriteTest {
4444

45-
private static File findHsErrorFileInOutput(OutputAnalyzer output) {
46-
47-
String hs_err_file = output.firstMatch("# *(\\S*hs_err_pid.*\\.log)", 1);
48-
if(hs_err_file ==null) {
49-
throw new RuntimeException("Did not find hs-err file in output.\n");
50-
}
51-
52-
File f = new File(hs_err_file);
53-
if (!f.exists()) {
54-
throw new RuntimeException("hs-err file missing at "
55-
+ f.getAbsolutePath() + ".\n");
56-
}
57-
58-
return f;
59-
60-
}
61-
62-
private static void scanHsErrorFileForContent(File f, Pattern[] pattern) throws IOException {
63-
FileInputStream fis = new FileInputStream(f);
64-
BufferedReader br = new BufferedReader(new InputStreamReader(fis));
65-
String line = null;
66-
67-
int currentPattern = 0;
68-
69-
String lastLine = null;
70-
while ((line = br.readLine()) != null && currentPattern < pattern.length) {
71-
if (pattern[currentPattern].matcher(line).matches()) {
72-
System.out.println("Found: " + line + ".");
73-
currentPattern++;
74-
}
75-
lastLine = line;
76-
}
77-
br.close();
78-
79-
if (currentPattern < pattern.length) {
80-
throw new RuntimeException("hs-err file incomplete (first missing pattern: " + pattern[currentPattern] + ")");
81-
}
82-
83-
}
84-
8545
public static void do_test(boolean with_percent_p) throws Exception {
8646

8747
// Crash twice.
@@ -110,10 +70,10 @@ public static void do_test(boolean with_percent_p) throws Exception {
11070
output_detail.shouldMatch("# " + errorFileStem + ".*");
11171
System.out.println("First crash: Found expected output on tty. Ok.");
11272

113-
File f = findHsErrorFileInOutput(output_detail);
73+
File f = ErrorFileScanner.findHsErrorFileInOutput(output_detail);
11474
System.out.println("First crash: Found hs error file at " + f.getAbsolutePath());
11575

116-
scanHsErrorFileForContent(f, new Pattern[] {
76+
ErrorFileScanner.scanHsErrorFileForContent(f, new Pattern[] {
11777
Pattern.compile("# *Internal Error.*"),
11878
Pattern.compile("Command Line:.*-XX:ErrorHandlerTest=1.*-XX:ErrorFile=" + errorFileStem + ".*")
11979
});
@@ -136,7 +96,7 @@ public static void do_test(boolean with_percent_p) throws Exception {
13696
output_detail.shouldMatch("# " + errorFileStem + ".*");
13797
System.out.println("Second crash: Found expected output on tty. Ok.");
13898

139-
File f2 = findHsErrorFileInOutput(output_detail);
99+
File f2 = ErrorFileScanner.findHsErrorFileInOutput(output_detail);
140100
System.out.println("Second crash: Found hs error file at " + f2.getAbsolutePath());
141101

142102
if (with_percent_p) {
@@ -145,7 +105,7 @@ public static void do_test(boolean with_percent_p) throws Exception {
145105
}
146106
}
147107

148-
scanHsErrorFileForContent(f2, new Pattern[] {
108+
ErrorFileScanner.scanHsErrorFileForContent(f2, new Pattern[] {
149109
Pattern.compile("# *Internal Error.*"),
150110
Pattern.compile("Command Line:.*-XX:ErrorHandlerTest=2.*-XX:ErrorFile=" + errorFileStem + ".*")
151111
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
* Copyright (c) 2021, 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+
import java.io.BufferedReader;
25+
import java.io.File;
26+
import java.io.FileInputStream;
27+
import java.io.InputStreamReader;
28+
import java.io.IOException;
29+
30+
import java.util.regex.Pattern;
31+
32+
import jdk.test.lib.process.OutputAnalyzer;
33+
34+
public class ErrorFileScanner {
35+
36+
public static File findHsErrorFileInOutput(OutputAnalyzer output) {
37+
38+
String hs_err_file = output.firstMatch("# *(\\S*hs_err_pid.*\\.log)", 1);
39+
if(hs_err_file ==null) {
40+
throw new RuntimeException("Did not find hs-err file in output.\n");
41+
}
42+
43+
File f = new File(hs_err_file);
44+
if (!f.exists()) {
45+
throw new RuntimeException("hs-err file missing at "
46+
+ f.getAbsolutePath() + ".\n");
47+
}
48+
49+
return f;
50+
51+
}
52+
53+
public static void scanHsErrorFileForContent(File f, Pattern[] pattern) throws IOException {
54+
FileInputStream fis = new FileInputStream(f);
55+
BufferedReader br = new BufferedReader(new InputStreamReader(fis));
56+
String line = null;
57+
58+
int currentPattern = 0;
59+
60+
String lastLine = null;
61+
while ((line = br.readLine()) != null && currentPattern < pattern.length) {
62+
if (pattern[currentPattern].matcher(line).matches()) {
63+
System.out.println("Found: " + line + ".");
64+
currentPattern++;
65+
}
66+
lastLine = line;
67+
}
68+
br.close();
69+
70+
if (currentPattern < pattern.length) {
71+
throw new RuntimeException("hs-err file incomplete (first missing pattern: " + pattern[currentPattern] + ")");
72+
}
73+
74+
}
75+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
/*
2+
* Copyright (c) 2021, 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 8274794
28+
* @summary Test that locks are printed in the Error file.
29+
* @library /test/lib
30+
* @modules java.base/jdk.internal.misc
31+
* @requires (vm.debug == true)
32+
* @run driver TestErrorFileMutex
33+
*/
34+
35+
import jdk.test.lib.process.OutputAnalyzer;
36+
import jdk.test.lib.process.ProcessTools;
37+
38+
import java.io.*;
39+
import java.util.regex.Pattern;
40+
41+
public class TestErrorFileMutex {
42+
43+
public static void do_test() throws Exception {
44+
45+
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(
46+
"-Xmx64M",
47+
"-XX:-CreateCoredumpOnCrash",
48+
"-XX:ErrorHandlerTest=3",
49+
"-version");
50+
51+
OutputAnalyzer output_detail = new OutputAnalyzer(pb.start());
52+
output_detail.shouldMatch("# A fatal error has been detected by the Java Runtime Environment:.*");
53+
54+
File f = ErrorFileScanner.findHsErrorFileInOutput(output_detail);
55+
System.out.println("Found hs error file at " + f.getAbsolutePath());
56+
57+
ErrorFileScanner.scanHsErrorFileForContent(f, new Pattern[] {
58+
Pattern.compile("# *Internal Error.*"),
59+
Pattern.compile(".*VM Mutexes/Monitors currently owned by a thread:.*"),
60+
Pattern.compile(".*ErrorTest_lock - owner thread:.*"),
61+
Pattern.compile(".*Threads_lock - owner thread:.*")
62+
});
63+
}
64+
65+
public static void main(String[] args) throws Exception {
66+
do_test();
67+
}
68+
69+
}
70+
71+
72+

0 commit comments

Comments
 (0)
Please sign in to comment.