Skip to content

Commit 843943c

Browse files
author
David Holmes
committedAug 9, 2021
8263567: gtests don't terminate the VM safely
Reviewed-by: stuefe, dcubed
1 parent 7fc99cf commit 843943c

File tree

2 files changed

+56
-16
lines changed

2 files changed

+56
-16
lines changed
 

‎test/hotspot/gtest/gtestMain.cpp

+46-15
Original file line numberDiff line numberDiff line change
@@ -61,8 +61,7 @@ static bool is_suffix(const char* suffix, const char* str) {
6161
return strncmp(str + (str_len - suffix_len), suffix, suffix_len) == 0;
6262
}
6363

64-
65-
static int init_jvm(int argc, char **argv, bool disable_error_handling) {
64+
static int init_jvm(int argc, char **argv, bool disable_error_handling, JavaVM** jvm_ptr) {
6665
// don't care about the program name
6766
argc--;
6867
argv++;
@@ -90,10 +89,9 @@ static int init_jvm(int argc, char **argv, bool disable_error_handling) {
9089
args.options = options;
9190
args.ignoreUnrecognized = JNI_FALSE;
9291

93-
JavaVM* jvm;
9492
JNIEnv* env;
9593

96-
int ret = JNI_CreateJavaVM(&jvm, (void**)&env, &args);
94+
int ret = JNI_CreateJavaVM(jvm_ptr, (void**)&env, &args);
9795
if (ret == JNI_OK) {
9896
// CreateJavaVM leaves WXExec context, while gtests
9997
// calls internal functions assuming running in WXWwrite.
@@ -111,26 +109,31 @@ class JVMInitializerListener : public ::testing::EmptyTestEventListener {
111109
private:
112110
int _argc;
113111
char** _argv;
114-
bool _is_initialized;
115-
116-
void initialize_jvm() {
117-
}
112+
JavaVM* _jvm;
118113

119114
public:
120115
JVMInitializerListener(int argc, char** argv) :
121-
_argc(argc), _argv(argv), _is_initialized(false) {
116+
_argc(argc), _argv(argv), _jvm(nullptr) {
122117
}
123118

124119
virtual void OnTestStart(const ::testing::TestInfo& test_info) {
125120
const char* name = test_info.name();
126-
if (!_is_initialized && is_same_vm_test(name)) {
121+
if (_jvm == nullptr && is_same_vm_test(name)) {
127122
// we want to have hs_err and core files when we execute regular tests
128-
int ret_val = init_jvm(_argc, _argv, false);
123+
int ret_val = init_jvm(_argc, _argv, false, &_jvm);
129124
if (ret_val != 0) {
130-
ADD_FAILURE() << "Could not initialize the JVM";
125+
ADD_FAILURE() << "Could not initialize the JVM: " << ret_val;
131126
exit(1);
132127
}
133-
_is_initialized = true;
128+
}
129+
}
130+
131+
void destroy_jvm() {
132+
if (_jvm != NULL) {
133+
int ret = _jvm->DestroyJavaVM();
134+
if (ret != 0) {
135+
fprintf(stderr, "Warning: DestroyJavaVM error %d\n", ret);
136+
}
134137
}
135138
}
136139
};
@@ -208,6 +211,18 @@ static char** remove_test_runner_arguments(int* argcp, char **argv) {
208211
return new_argv;
209212
}
210213

214+
// This is generally run once for a set of tests. But if that set includes a vm_assert or
215+
// other_vm test, then a new process is forked, and runUnitTestsInner is called, passing
216+
// just that test as the one to be executed.
217+
//
218+
// When we execute a vm_assert or other_vm test we create and initialize the JVM below.
219+
//
220+
// A vm_assert test crashes the VM so no cleanup is needed, but for other_vm we call
221+
// DestroyJavaVM via the TEST_OTHER_VM macro prior to the call to exit().
222+
//
223+
// For same_vm tests we use an event listener to create the JVM when the first same_vm
224+
// test is executed. Once all tests are completed we can then call DestroyJavaVM on that
225+
// JVM directly.
211226
static void runUnitTestsInner(int argc, char** argv) {
212227
::testing::InitGoogleMock(&argc, argv);
213228
::testing::GTEST_FLAG(death_test_style) = "threadsafe";
@@ -253,22 +268,38 @@ static void runUnitTestsInner(int argc, char** argv) {
253268
#endif // _WIN32
254269
argv = remove_test_runner_arguments(&argc, argv);
255270

271+
272+
JVMInitializerListener* jvm_listener = NULL;
273+
256274
if (is_vmassert_test || is_othervm_test) {
275+
JavaVM* jvm = NULL;
257276
// both vmassert and other vm tests require inited jvm
258277
// but only vmassert tests disable hs_err and core file generation
259-
if (init_jvm(argc, argv, is_vmassert_test) != 0) {
278+
int ret;
279+
if ((ret = init_jvm(argc, argv, is_vmassert_test, &jvm)) != 0) {
280+
fprintf(stderr, "ERROR: JNI_CreateJavaVM failed: %d\n", ret);
260281
abort();
261282
}
262283
} else {
263284
::testing::TestEventListeners& listeners = ::testing::UnitTest::GetInstance()->listeners();
264-
listeners.Append(new JVMInitializerListener(argc, argv));
285+
jvm_listener = new JVMInitializerListener(argc, argv);
286+
listeners.Append(jvm_listener);
265287
}
266288

267289
int result = RUN_ALL_TESTS();
290+
291+
// vm_assert and other_vm tests never reach this point as they either abort, or call
292+
// exit() - see TEST_OTHER_VM macro. We will reach here when all same_vm tests have
293+
// completed for this run, so we can terminate the VM used for that case.
294+
268295
if (result != 0) {
269296
fprintf(stderr, "ERROR: RUN_ALL_TESTS() failed. Error %d\n", result);
270297
exit(2);
271298
}
299+
300+
if (jvm_listener != NULL) {
301+
jvm_listener->destroy_jvm();
302+
}
272303
}
273304

274305
// Thread support for -new-thread option

‎test/hotspot/gtest/unittest.hpp

+10-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2016, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2016, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -88,6 +88,15 @@
8888
static void child_ ## category ## _ ## name ## _() { \
8989
::testing::GTEST_FLAG(throw_on_failure) = true; \
9090
test_ ## category ## _ ## name ## _(); \
91+
JavaVM* jvm[1]; \
92+
jsize nVMs = 0; \
93+
JNI_GetCreatedJavaVMs(&jvm[0], 1, &nVMs); \
94+
if (nVMs == 1) { \
95+
int ret = jvm[0]->DestroyJavaVM(); \
96+
if (ret != 0) { \
97+
fprintf(stderr, "Warning: DestroyJavaVM error %d\n", ret); \
98+
} \
99+
} \
91100
fprintf(stderr, "OKIDOKI"); \
92101
exit(0); \
93102
} \

0 commit comments

Comments
 (0)
Please sign in to comment.