@@ -61,8 +61,7 @@ static bool is_suffix(const char* suffix, const char* str) {
61
61
return strncmp (str + (str_len - suffix_len), suffix, suffix_len) == 0 ;
62
62
}
63
63
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) {
66
65
// don't care about the program name
67
66
argc--;
68
67
argv++;
@@ -90,10 +89,9 @@ static int init_jvm(int argc, char **argv, bool disable_error_handling) {
90
89
args.options = options;
91
90
args.ignoreUnrecognized = JNI_FALSE;
92
91
93
- JavaVM* jvm;
94
92
JNIEnv* env;
95
93
96
- int ret = JNI_CreateJavaVM (&jvm , (void **)&env, &args);
94
+ int ret = JNI_CreateJavaVM (jvm_ptr , (void **)&env, &args);
97
95
if (ret == JNI_OK) {
98
96
// CreateJavaVM leaves WXExec context, while gtests
99
97
// calls internal functions assuming running in WXWwrite.
@@ -111,26 +109,31 @@ class JVMInitializerListener : public ::testing::EmptyTestEventListener {
111
109
private:
112
110
int _argc;
113
111
char ** _argv;
114
- bool _is_initialized;
115
-
116
- void initialize_jvm () {
117
- }
112
+ JavaVM* _jvm;
118
113
119
114
public:
120
115
JVMInitializerListener (int argc, char ** argv) :
121
- _argc (argc), _argv(argv), _is_initialized( false ) {
116
+ _argc (argc), _argv(argv), _jvm( nullptr ) {
122
117
}
123
118
124
119
virtual void OnTestStart (const ::testing::TestInfo& test_info) {
125
120
const char * name = test_info.name ();
126
- if (!_is_initialized && is_same_vm_test (name)) {
121
+ if (_jvm == nullptr && is_same_vm_test (name)) {
127
122
// 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 );
129
124
if (ret_val != 0 ) {
130
- ADD_FAILURE () << " Could not initialize the JVM" ;
125
+ ADD_FAILURE () << " Could not initialize the JVM: " << ret_val ;
131
126
exit (1 );
132
127
}
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
+ }
134
137
}
135
138
}
136
139
};
@@ -208,6 +211,18 @@ static char** remove_test_runner_arguments(int* argcp, char **argv) {
208
211
return new_argv;
209
212
}
210
213
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.
211
226
static void runUnitTestsInner (int argc, char ** argv) {
212
227
::testing::InitGoogleMock (&argc, argv);
213
228
::testing::GTEST_FLAG (death_test_style) = "threadsafe";
@@ -253,22 +268,38 @@ static void runUnitTestsInner(int argc, char** argv) {
253
268
#endif // _WIN32
254
269
argv = remove_test_runner_arguments (&argc, argv);
255
270
271
+
272
+ JVMInitializerListener* jvm_listener = NULL ;
273
+
256
274
if (is_vmassert_test || is_othervm_test) {
275
+ JavaVM* jvm = NULL ;
257
276
// both vmassert and other vm tests require inited jvm
258
277
// 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);
260
281
abort ();
261
282
}
262
283
} else {
263
284
::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);
265
287
}
266
288
267
289
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
+
268
295
if (result != 0 ) {
269
296
fprintf (stderr, " ERROR: RUN_ALL_TESTS() failed. Error %d\n " , result);
270
297
exit (2 );
271
298
}
299
+
300
+ if (jvm_listener != NULL ) {
301
+ jvm_listener->destroy_jvm ();
302
+ }
272
303
}
273
304
274
305
// Thread support for -new-thread option
0 commit comments