From 035eaeecabd484d6db629c8b4056fa4b3a73f960 Mon Sep 17 00:00:00 2001 From: Serguei Spitsyn Date: Fri, 18 Nov 2022 20:52:56 +0000 Subject: [PATCH] 8296324: JVMTI GetStackTrace truncates vthread stack trace for agents loaded into running VM Reviewed-by: cjplummer, lmesnik --- src/hotspot/share/prims/jvmtiExport.cpp | 4 + .../VirtualStackTraceTest.java | 80 ++++++++++++++++ .../libVirtualStackTraceTest.cpp | 96 +++++++++++++++++++ 3 files changed, 180 insertions(+) create mode 100644 test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/VirtualStackTraceTest.java create mode 100644 test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/libVirtualStackTraceTest.cpp diff --git a/src/hotspot/share/prims/jvmtiExport.cpp b/src/hotspot/share/prims/jvmtiExport.cpp index 1daca3e4ab4..949141104b0 100644 --- a/src/hotspot/share/prims/jvmtiExport.cpp +++ b/src/hotspot/share/prims/jvmtiExport.cpp @@ -383,6 +383,10 @@ JvmtiExport::get_jvmti_interface(JavaVM *jvm, void **penv, jint version) { if (Continuations::enabled()) { // Virtual threads support. There is a performance impact when VTMS transitions are enabled. java_lang_VirtualThread::set_notify_jvmti_events(true); + if (JvmtiEnv::get_phase() == JVMTI_PHASE_LIVE) { + ThreadInVMfromNative __tiv(JavaThread::current()); + java_lang_VirtualThread::init_static_notify_jvmti_events(); + } } if (JvmtiEnv::get_phase() == JVMTI_PHASE_LIVE) { diff --git a/test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/VirtualStackTraceTest.java b/test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/VirtualStackTraceTest.java new file mode 100644 index 00000000000..8336d5d235a --- /dev/null +++ b/test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/VirtualStackTraceTest.java @@ -0,0 +1,80 @@ +/* + * 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 + * @summary Verifies JVMTI GetStackTrace does not truncate virtual thread stack trace with agent attach + * @requires vm.jvmti + * @requires vm.continuations + * @enablePreview + * @run main/othervm/native -Djdk.attach.allowAttachSelf=true VirtualStackTraceTest + */ + +import com.sun.tools.attach.VirtualMachine; + +import java.util.Arrays; +import java.util.List; +import java.util.Objects; + +public class VirtualStackTraceTest { + private static final String AGENT_LIB = "VirtualStackTraceTest"; + + public static native String[] getStackTrace(); + + public static void main(String[] args) throws Exception { + VirtualMachine vm = VirtualMachine.attach(String.valueOf(ProcessHandle.current().pid())); + vm.loadAgentLibrary(AGENT_LIB); + VirtualStackTraceTest t = new VirtualStackTraceTest(); + t.runTest(); + } + + void runTest() throws Exception { + Thread thr = Thread.ofVirtual().name("VT").start(VirtualStackTraceTest::test); + thr.join(); + } + + private static void test() { + work(); + } + + private static void work() { + inner(); + } + + private static void inner() { + checkCurrentThread(); + } + + private static void checkCurrentThread() { + System.out.println("Stack trace for " + Thread.currentThread() + ": "); + var javaStackTrace = Arrays.stream(Thread.currentThread().getStackTrace()).map(StackTraceElement::getMethodName).toList(); + var jvmtiStackTrace = List.of(getStackTrace()); + + System.out.println("JVMTI: " + jvmtiStackTrace); + System.out.println("Java : " + javaStackTrace); + + if (!Objects.equals(jvmtiStackTrace, javaStackTrace)) { + throw new RuntimeException("VirtualStackTraceTest failed: stack traces do not match"); + } + } +} diff --git a/test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/libVirtualStackTraceTest.cpp b/test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/libVirtualStackTraceTest.cpp new file mode 100644 index 00000000000..eb387a1f8dc --- /dev/null +++ b/test/hotspot/jtreg/serviceability/jvmti/vthread/VirtualStackTraceTest/libVirtualStackTraceTest.cpp @@ -0,0 +1,96 @@ +/* + * 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. + */ + +#include +#include +#include +#include "jvmti_common.h" + +extern "C" { + +const int MAX_COUNT = 50; +static jvmtiEnv *jvmti; + +JNIEXPORT jobjectArray JNICALL +Java_VirtualStackTraceTest_getStackTrace(JNIEnv* jni, jclass clazz) { + jvmtiError err; + jint count = 0; + jint skipped = 0; + + jobject visibleFrames[MAX_COUNT]; + jvmtiFrameInfo frameInfo[MAX_COUNT]; + + err = jvmti->GetStackTrace(NULL, 0, MAX_COUNT, frameInfo, &count); + check_jvmti_status(jni, err, "event handler: error in JVMTI GetStackTrace call"); + + for (int idx = 0; idx < count; idx++) { + jclass declaringClass = NULL; + char *clasSignature = NULL; + char *methodName = NULL; + + err = jvmti->GetMethodDeclaringClass(frameInfo[idx].method, &declaringClass); + check_jvmti_status(jni, err, "event handler: error in JVMTI GetMethodDeclaringClass call"); + + err = jvmti->GetClassSignature(declaringClass, &clasSignature, NULL); + check_jvmti_status(jni, err, "event handler: error in JVMTI GetClassSignature call"); + + err = jvmti->GetMethodName(frameInfo[idx].method, &methodName, NULL, NULL); + check_jvmti_status(jni, err, "event handler: error in JVMTI GetMethodName call"); + + if (strchr(clasSignature, '.')) { + skipped++; + continue; + } + visibleFrames[idx - skipped] = jni->NewStringUTF(methodName); + + jvmti->Deallocate(reinterpret_cast(methodName)); + jvmti->Deallocate(reinterpret_cast(clasSignature)); + } + jobjectArray methodNames = jni->NewObjectArray(count - skipped, jni->FindClass("java/lang/String"), NULL); + for (int idx = 0; idx < count - skipped; idx++) { + jni->SetObjectArrayElement(methodNames, idx, visibleFrames[idx]); + } + print_stack_trace(jvmti, jni, NULL); + + return methodNames; +} + +JNIEXPORT jint JNICALL +Agent_OnLoad(JavaVM *jvm, char *options, void *reserved) { + LOG("Agent_OnLoad started\n"); + if (jvm->GetEnv((void **) (&jvmti), JVMTI_VERSION) != JNI_OK) { + return JNI_ERR; + } + return JNI_OK; +} + +JNIEXPORT jint JNICALL +Agent_OnAttach(JavaVM *jvm, char *options, void *reserved) { + LOG("Agent_OnAttach started\n"); + if (jvm->GetEnv((void **) (&jvmti), JVMTI_VERSION) != JNI_OK) { + return JNI_ERR; + } + return JNI_OK; +} + +} // extern "C"