8339725: Concurrent GC crashed due to GetMethodDeclaringClass
Reviewed-by: lmesnik, coleenp, eosterlund, stefank
This commit is contained in:
parent
fa502ecd2d
commit
c91fa278fe
@ -885,7 +885,7 @@ BUILD_HOTSPOT_JTREG_EXECUTABLES_JDK_LIBS_exedaemonDestroy := java.base:libjvm
|
|||||||
|
|
||||||
ifeq ($(call isTargetOs, windows), true)
|
ifeq ($(call isTargetOs, windows), true)
|
||||||
BUILD_HOTSPOT_JTREG_EXECUTABLES_CFLAGS_exeFPRegs := -MT
|
BUILD_HOTSPOT_JTREG_EXECUTABLES_CFLAGS_exeFPRegs := -MT
|
||||||
BUILD_HOTSPOT_JTREG_EXCLUDE += exesigtest.c libterminatedThread.c libTestJNI.c libCompleteExit.c libMonitorWithDeadObjectTest.c libTestPsig.c exeGetCreatedJavaVMs.c
|
BUILD_HOTSPOT_JTREG_EXCLUDE += exesigtest.c libterminatedThread.c libTestJNI.c libCompleteExit.c libMonitorWithDeadObjectTest.c libTestPsig.c exeGetCreatedJavaVMs.c libTestUnloadedClass.cpp
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_JDK_LIBS_libnativeStack := java.base:libjvm
|
BUILD_HOTSPOT_JTREG_LIBRARIES_JDK_LIBS_libnativeStack := java.base:libjvm
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_JDK_LIBS_libVThreadEventTest := java.base:libjvm
|
BUILD_HOTSPOT_JTREG_LIBRARIES_JDK_LIBS_libVThreadEventTest := java.base:libjvm
|
||||||
else
|
else
|
||||||
@ -1526,6 +1526,7 @@ else
|
|||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libCompleteExit += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libCompleteExit += -lpthread
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libMonitorWithDeadObjectTest += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libMonitorWithDeadObjectTest += -lpthread
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libnativeStack += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libnativeStack += -lpthread
|
||||||
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libTestUnloadedClass += -lpthread
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_JDK_LIBS_libVThreadEventTest := java.base:libjvm
|
BUILD_HOTSPOT_JTREG_LIBRARIES_JDK_LIBS_libVThreadEventTest := java.base:libjvm
|
||||||
BUILD_HOTSPOT_JTREG_EXECUTABLES_LIBS_exeGetCreatedJavaVMs := -lpthread
|
BUILD_HOTSPOT_JTREG_EXECUTABLES_LIBS_exeGetCreatedJavaVMs := -lpthread
|
||||||
BUILD_HOTSPOT_JTREG_EXECUTABLES_JDK_LIBS_exeGetCreatedJavaVMs := java.base:libjvm
|
BUILD_HOTSPOT_JTREG_EXECUTABLES_JDK_LIBS_exeGetCreatedJavaVMs := java.base:libjvm
|
||||||
|
@ -3132,7 +3132,9 @@ JvmtiEnv::GetFieldName(fieldDescriptor* fdesc_ptr, char** name_ptr, char** signa
|
|||||||
// declaring_class_ptr - pre-checked for null
|
// declaring_class_ptr - pre-checked for null
|
||||||
jvmtiError
|
jvmtiError
|
||||||
JvmtiEnv::GetFieldDeclaringClass(fieldDescriptor* fdesc_ptr, jclass* declaring_class_ptr) {
|
JvmtiEnv::GetFieldDeclaringClass(fieldDescriptor* fdesc_ptr, jclass* declaring_class_ptr) {
|
||||||
|
// As for the GetFieldDeclaringClass method, the XSL generated C++ code that calls it has
|
||||||
|
// a jclass of the relevant class or a subclass of it, which is fine in terms of ensuring
|
||||||
|
// the holder is kept alive.
|
||||||
*declaring_class_ptr = get_jni_class_non_null(fdesc_ptr->field_holder());
|
*declaring_class_ptr = get_jni_class_non_null(fdesc_ptr->field_holder());
|
||||||
return JVMTI_ERROR_NONE;
|
return JVMTI_ERROR_NONE;
|
||||||
} /* end GetFieldDeclaringClass */
|
} /* end GetFieldDeclaringClass */
|
||||||
@ -3210,7 +3212,9 @@ JvmtiEnv::GetMethodName(Method* method, char** name_ptr, char** signature_ptr, c
|
|||||||
jvmtiError
|
jvmtiError
|
||||||
JvmtiEnv::GetMethodDeclaringClass(Method* method, jclass* declaring_class_ptr) {
|
JvmtiEnv::GetMethodDeclaringClass(Method* method, jclass* declaring_class_ptr) {
|
||||||
NULL_CHECK(method, JVMTI_ERROR_INVALID_METHODID);
|
NULL_CHECK(method, JVMTI_ERROR_INVALID_METHODID);
|
||||||
(*declaring_class_ptr) = get_jni_class_non_null(method->method_holder());
|
Klass* k = method->method_holder();
|
||||||
|
Handle holder(Thread::current(), k->klass_holder()); // keep the klass alive
|
||||||
|
(*declaring_class_ptr) = get_jni_class_non_null(k);
|
||||||
return JVMTI_ERROR_NONE;
|
return JVMTI_ERROR_NONE;
|
||||||
} /* end GetMethodDeclaringClass */
|
} /* end GetMethodDeclaringClass */
|
||||||
|
|
||||||
|
@ -599,6 +599,7 @@ JvmtiEnvBase::jvf_for_thread_and_depth(JavaThread* java_thread, jint depth) {
|
|||||||
jclass
|
jclass
|
||||||
JvmtiEnvBase::get_jni_class_non_null(Klass* k) {
|
JvmtiEnvBase::get_jni_class_non_null(Klass* k) {
|
||||||
assert(k != nullptr, "k != null");
|
assert(k != nullptr, "k != null");
|
||||||
|
assert(k->is_loader_alive(), "Must be alive");
|
||||||
Thread *thread = Thread::current();
|
Thread *thread = Thread::current();
|
||||||
return (jclass)jni_reference(Handle(thread, k->java_mirror()));
|
return (jclass)jni_reference(Handle(thread, k->java_mirror()));
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,106 @@
|
|||||||
|
/*
|
||||||
|
* Copyright Amazon.com Inc. or its affiliates. All Rights Reserved.
|
||||||
|
* Copyright (c) 2024, Alibaba Group Holding Limited. 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 8339725
|
||||||
|
* @summary Stress test GetMethodDeclaringClass
|
||||||
|
* @requires vm.jvmti
|
||||||
|
* @requires (os.family == "linux")
|
||||||
|
* @library /test/lib
|
||||||
|
* @run driver/timeout=300 TestUnloadedClass
|
||||||
|
*/
|
||||||
|
|
||||||
|
import jdk.test.lib.process.OutputAnalyzer;
|
||||||
|
import jdk.test.lib.Utils;
|
||||||
|
import jdk.test.lib.process.ProcessTools;
|
||||||
|
import jdk.test.lib.Platform;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.lang.reflect.Constructor;
|
||||||
|
|
||||||
|
public class TestUnloadedClass {
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
ProcessBuilder pb = ProcessTools.createTestJavaProcessBuilder(
|
||||||
|
"-agentpath:" + Utils.TEST_NATIVE_PATH + File.separator + System.mapLibraryName("TestUnloadedClass"),
|
||||||
|
"-Xmx50m",
|
||||||
|
"Test");
|
||||||
|
OutputAnalyzer output = new OutputAnalyzer(pb.start());
|
||||||
|
if (!Platform.isDebugBuild()) {
|
||||||
|
output.shouldContain("OutOfMemoryError");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Test {
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
long last = System.nanoTime();
|
||||||
|
for (int i = 0;;i++) {
|
||||||
|
if (Platform.isDebugBuild() && i >= 1000) {
|
||||||
|
// Debug build costs too much time to OOM so limit the loop iteration
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
CustomClassLoader loader = new CustomClassLoader();
|
||||||
|
Class<?> k = loader.findClass("MyClass");
|
||||||
|
Constructor<?> c = k.getDeclaredConstructor();
|
||||||
|
c.setAccessible(true);
|
||||||
|
c.newInstance();
|
||||||
|
|
||||||
|
// call gc every ~1 second.
|
||||||
|
if ((System.nanoTime() - last) >= 1e9) {
|
||||||
|
System.gc();
|
||||||
|
last = System.nanoTime();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class CustomClassLoader extends ClassLoader {
|
||||||
|
static byte[] BYTES;
|
||||||
|
|
||||||
|
static {
|
||||||
|
try (InputStream in = CustomClassLoader.class.getResourceAsStream("MyClass.class")) {
|
||||||
|
try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
|
||||||
|
byte[] buf = new byte[4096];
|
||||||
|
int len;
|
||||||
|
while ((len = in.read(buf)) != -1) {
|
||||||
|
baos.write(buf, 0, len);
|
||||||
|
}
|
||||||
|
BYTES = baos.toByteArray();
|
||||||
|
}
|
||||||
|
} catch (Throwable t) {
|
||||||
|
throw new RuntimeException(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Class findClass(String name) throws ClassNotFoundException {
|
||||||
|
return defineClass(name, BYTES, 0, BYTES.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MyClass {
|
||||||
|
}
|
@ -0,0 +1,129 @@
|
|||||||
|
/*
|
||||||
|
* Copyright Amazon.com Inc. or its affiliates. All Rights Reserved.
|
||||||
|
* Copyright (c) 2024, Alibaba Group Holding Limited. 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
|
|
||||||
|
#include <jvmti.h>
|
||||||
|
#include <jni.h>
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
static jvmtiEnv *_jvmti;
|
||||||
|
static JavaVM *_jvm;
|
||||||
|
|
||||||
|
#define BUFFER_SIZE 100000
|
||||||
|
static std::atomic<jmethodID> ring_buffer[BUFFER_SIZE];
|
||||||
|
|
||||||
|
void get_method_details(jmethodID method) {
|
||||||
|
jclass method_class;
|
||||||
|
char *class_name = NULL;
|
||||||
|
if (_jvmti->GetMethodDeclaringClass(method, &method_class) == JVMTI_ERROR_NONE) {
|
||||||
|
if (_jvmti->GetClassSignature(method_class, &class_name, NULL) == JVMTI_ERROR_NONE) {
|
||||||
|
_jvmti->Deallocate((unsigned char *)class_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void* read_ringbuffer(void* arg) {
|
||||||
|
JNIEnv *env;
|
||||||
|
_jvm->AttachCurrentThreadAsDaemon((void **)&env, NULL);
|
||||||
|
for (;;) {
|
||||||
|
jmethodID id = ring_buffer[rand() % BUFFER_SIZE].load(std::memory_order_relaxed);
|
||||||
|
if (id != (jmethodID)0) {
|
||||||
|
get_method_details(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void JNICALL ClassPrepareCallback(jvmtiEnv *jvmti_env,
|
||||||
|
JNIEnv *jni_env,
|
||||||
|
jthread thread,
|
||||||
|
jclass klass) {
|
||||||
|
static bool reader_created = false;
|
||||||
|
static int ring_buffer_idx = 0;
|
||||||
|
|
||||||
|
char *class_name = NULL;
|
||||||
|
if (jvmti_env->GetClassSignature(klass, &class_name, NULL) != JVMTI_ERROR_NONE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// We only care MyClass and only one thread loads it
|
||||||
|
bool is_my_class = strcmp(class_name, "LMyClass;") == 0;
|
||||||
|
jvmti_env->Deallocate((unsigned char *)class_name);
|
||||||
|
if (!is_my_class) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!reader_created) {
|
||||||
|
pthread_t tid;
|
||||||
|
pthread_create(&tid, NULL, read_ringbuffer, NULL);
|
||||||
|
reader_created = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
jint method_count;
|
||||||
|
jmethodID *methods;
|
||||||
|
if (jvmti_env->GetClassMethods(klass, &method_count, &methods) == JVMTI_ERROR_NONE) {
|
||||||
|
ring_buffer[ring_buffer_idx++].store(methods[0], std::memory_order_relaxed);
|
||||||
|
ring_buffer_idx = ring_buffer_idx % BUFFER_SIZE;
|
||||||
|
jvmti_env->Deallocate((unsigned char *)methods);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT jint JNICALL Agent_OnLoad(JavaVM *jvm, char *options, void *reserved) {
|
||||||
|
for (int i = 0; i < BUFFER_SIZE; i++) {
|
||||||
|
ring_buffer[i].store(0, std::memory_order_relaxed);
|
||||||
|
}
|
||||||
|
|
||||||
|
jvmtiEventCallbacks callbacks;
|
||||||
|
jvmtiError error;
|
||||||
|
|
||||||
|
_jvm = jvm;
|
||||||
|
|
||||||
|
if (jvm->GetEnv((void **)&_jvmti, JVMTI_VERSION_1_0) != JNI_OK) {
|
||||||
|
fprintf(stderr, "Unable to access JVMTI!\n");
|
||||||
|
return JNI_ERR;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up the event callbacks
|
||||||
|
memset(&callbacks, 0, sizeof(callbacks));
|
||||||
|
callbacks.ClassPrepare = &ClassPrepareCallback;
|
||||||
|
|
||||||
|
// Register the callbacks
|
||||||
|
error = _jvmti->SetEventCallbacks(&callbacks, sizeof(callbacks));
|
||||||
|
if (error != JVMTI_ERROR_NONE) {
|
||||||
|
fprintf(stderr, "Error setting event callbacks: %d\n", error);
|
||||||
|
return JNI_ERR;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enable the ClassPrepare event
|
||||||
|
error = _jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_CLASS_PREPARE, NULL);
|
||||||
|
if (error != JVMTI_ERROR_NONE) {
|
||||||
|
fprintf(stderr, "Error enabling ClassPrepare event: %d\n", error);
|
||||||
|
return JNI_ERR;
|
||||||
|
}
|
||||||
|
|
||||||
|
return JNI_OK;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user