8284274: Error reporting crashes because missing ResourceMarks

Reviewed-by: dholmes, stuefe
This commit is contained in:
Coleen Phillimore 2022-04-13 14:46:27 +00:00
parent 8ee2944cc4
commit e245f9d200
3 changed files with 132 additions and 3 deletions

View File

@ -28,6 +28,7 @@
#include "runtime/atomic.hpp"
#include "runtime/thread.inline.hpp"
#include "services/memTracker.hpp"
#include "utilities/vmError.hpp"
void ResourceArea::bias_to(MEMFLAGS new_flags) {
if (new_flags != _flags) {
@ -43,7 +44,7 @@ void ResourceArea::bias_to(MEMFLAGS new_flags) {
#ifdef ASSERT
void ResourceArea::verify_has_resource_mark() {
if (_nesting <= 0) {
if (_nesting <= 0 && !VMError::is_error_reported()) {
// Only report the first occurrence of an allocating thread that
// is missing a ResourceMark, to avoid possible recursive errors
// in error handling.

View File

@ -544,23 +544,33 @@ void VMError::report(outputStream* st, bool _verbose) {
#ifdef ASSERT
// Error handler self tests
// Meaning of codes passed through in the tests.
#define TEST_SECONDARY_CRASH 14
#define TEST_RESOURCE_MARK_CRASH 2
// test secondary error handling. Test it twice, to test that resetting
// error handler after a secondary crash works.
STEP("test secondary crash 1")
if (_verbose && TestCrashInErrorHandler != 0) {
if (_verbose && TestCrashInErrorHandler == TEST_SECONDARY_CRASH) {
st->print_cr("Will crash now (TestCrashInErrorHandler=" UINTX_FORMAT ")...",
TestCrashInErrorHandler);
controlled_crash(TestCrashInErrorHandler);
}
STEP("test secondary crash 2")
if (_verbose && TestCrashInErrorHandler != 0) {
if (_verbose && TestCrashInErrorHandler == TEST_SECONDARY_CRASH) {
st->print_cr("Will crash now (TestCrashInErrorHandler=" UINTX_FORMAT ")...",
TestCrashInErrorHandler);
controlled_crash(TestCrashInErrorHandler);
}
STEP("test missing ResourceMark does not crash")
if (_verbose && TestCrashInErrorHandler == TEST_RESOURCE_MARK_CRASH) {
stringStream message;
message.print("This is a message with no ResourceMark");
tty->print_cr("%s", message.as_string());
}
// TestUnresponsiveErrorHandler: We want to test both step timeouts and global timeout.
// Step to global timeout ratio is 4:1, so in order to be absolutely sure we hit the
// global timeout, let's execute the timeout step five times.

View File

@ -0,0 +1,118 @@
/*
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 8284274
* @summary Test that reporting doesn't crash because missing ResourceMarks
* @library /test/lib
* @requires vm.debug
* @requires os.family != "windows"
* @modules java.base/jdk.internal.misc
* java.management
* @run driver ResourceMarkTest
*/
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStreamReader;
import java.util.regex.Pattern;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
public class ResourceMarkTest {
public static void main(String[] args) throws Exception {
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(
"-XX:+UnlockDiagnosticVMOptions",
"-Xmx100M",
"-XX:-CreateCoredumpOnCrash",
"-XX:ErrorHandlerTest=15",
"-XX:TestCrashInErrorHandler=2",
"-version");
OutputAnalyzer output_detail = new OutputAnalyzer(pb.start());
// we should have crashed with a SIGFPE
output_detail.shouldMatch("# A fatal error has been detected by the Java Runtime Environment:.*");
output_detail.shouldMatch("#.+SIGFPE.*");
output_detail.shouldMatch("This is a message with no ResourceMark");
// extract hs-err file
String hs_err_file = output_detail.firstMatch("# *(\\S*hs_err_pid\\d+\\.log)", 1);
if (hs_err_file == null) {
throw new RuntimeException("Did not find hs-err file in output.\n");
}
// scan hs-err file: File should NOT contain the "[error occurred during error reporting..]"
// markers which show that the secondary error handling kicked in and handled the
// error successfully. As an added test, we check that the last line contains "END.",
// which is an end marker written in the last step and proves that hs-err file was
// completely written.
File f = new File(hs_err_file);
if (!f.exists()) {
throw new RuntimeException("hs-err file missing at "
+ f.getAbsolutePath() + ".\n");
}
System.out.println("Found hs_err file. Scanning...");
FileInputStream fis = new FileInputStream(f);
BufferedReader br = new BufferedReader(new InputStreamReader(fis));
String line = null;
Pattern [] pattern = new Pattern[] {
Pattern.compile("\\[error occurred during error reporting \\(test missing ResourceMark does not crash\\), id 0xe0000000, Internal Error \\(.*resourceArea.cpp:.*\\)\\]"),
};
int currentPattern = 0;
String lastLine = null;
while ((line = br.readLine()) != null) {
if (currentPattern < pattern.length) {
if (pattern[currentPattern].matcher(line).matches()) {
System.out.println("Found: " + line + ".");
currentPattern ++;
}
}
lastLine = line;
}
br.close();
if (currentPattern == pattern.length) {
throw new RuntimeException("hs-err file found secondary crash for ResourceMark");
}
if (!lastLine.equals("END.")) {
throw new RuntimeException("hs-err file incomplete (missing END marker.)");
} else {
System.out.println("End marker found.");
}
System.out.println("OK.");
}
}