Alan Bateman 9583e3657e 8284161: Implementation of Virtual Threads (Preview)
Co-authored-by: Ron Pressler <rpressler@openjdk.org>
Co-authored-by: Alan Bateman <alanb@openjdk.org>
Co-authored-by: Erik Österlund <eosterlund@openjdk.org>
Co-authored-by: Andrew Haley <aph@openjdk.org>
Co-authored-by: Rickard Bäckman <rbackman@openjdk.org>
Co-authored-by: Markus Grönlund <mgronlun@openjdk.org>
Co-authored-by: Leonid Mesnik <lmesnik@openjdk.org>
Co-authored-by: Serguei Spitsyn <sspitsyn@openjdk.org>
Co-authored-by: Chris Plummer <cjplummer@openjdk.org>
Co-authored-by: Coleen Phillimore <coleenp@openjdk.org>
Co-authored-by: Robbin Ehn <rehn@openjdk.org>
Co-authored-by: Stefan Karlsson <stefank@openjdk.org>
Co-authored-by: Thomas Schatzl <tschatzl@openjdk.org>
Co-authored-by: Sergey Kuksenko <skuksenko@openjdk.org>
Reviewed-by: lancea, eosterlund, rehn, sspitsyn, stefank, tschatzl, dfuchs, lmesnik, dcubed, kevinw, amenkov, dlong, mchung, psandoz, bpb, coleenp, smarks, egahlin, mseledtsov, coffeys, darcy
2022-05-07 08:06:16 +00:00

395 lines
14 KiB
C++

/*
* Copyright (c) 2021, 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 <string.h>
#include "jvmti.h"
#include "jvmti_common.h"
#include "jvmti_thread.h"
extern "C" {
/* ============================================================================= */
#define VTHREAD_CNT 10
static const char* CTHREAD_NAME_START = "ForkJoinPool";
static const int CTHREAD_NAME_START_LEN = (int)strlen("ForkJoinPool");
static jvmtiEnv *jvmti = NULL;
static jrawMonitorID agent_event_lock = NULL;
static volatile jthread agent_thread = NULL;
static jthread tested_vthreads[VTHREAD_CNT];
static int vthread_no = 0;
static jmethodID *test_methods = NULL;
jint test_method_count = 0;
jclass test_class = NULL;
static void
set_or_clear_breakpoint(JNIEnv *jni, jboolean set, const char *methodName,
jclass klass, jmethodID methods[], int method_count) {
jlocation location = (jlocation)0L;
jmethodID method = NULL;
jvmtiError err;
// Find the jmethodID of the specified method
while (--method_count >= 0) {
jmethodID meth = methods[method_count];
char* mname = get_method_name(jvmti, jni, meth);
if (strcmp(mname, methodName) == 0) {
LOG("setupBreakpoint: found method %s() to %s a breakpoint\n", mname, set ? "set" : "clear");
method = meth;
}
deallocate(jvmti, jni, (void*)mname);
}
if (method == NULL) {
LOG("setupBreakpoint: not found method %s() to %s a breakpoint\n",
methodName, set ? "set" : "clear");
jni->FatalError("Error in setupBreakpoint: not found method");
}
if (set) {
err = jvmti->SetBreakpoint(method, location);
} else {
err = jvmti->ClearBreakpoint(method, location);
}
check_jvmti_status(jni, err, "setupBreakpoint: error in JVMTI SetBreakpoint");
}
static void
set_breakpoint(JNIEnv *jni, const char *methodName,
jclass klass, jmethodID methods[], int method_count) {
set_or_clear_breakpoint(jni, JNI_TRUE, methodName, klass, methods, method_count);
}
static void
clear_breakpoint(JNIEnv *jni, const char *methodName,
jclass klass, jmethodID methods[], int method_count) {
set_or_clear_breakpoint(jni, JNI_FALSE, methodName, klass, methods, method_count);
}
JNIEXPORT void JNICALL
Java_SuspendResumeAll_setBreakpoint(JNIEnv *jni, jclass klass, jclass testKlass) {
jvmtiError err;
LOG("setBreakpoint: started\n");
test_class = (jclass)jni->NewGlobalRef(testKlass);
err = jvmti->GetClassMethods(testKlass, &test_method_count, &test_methods);
check_jvmti_status(jni, err, "setBreakpoint: error in JVMTI GetClassMethods for testKlass");
set_breakpoint(jni, "breakpointCheck", testKlass, test_methods, test_method_count);
LOG("setBreakpoint: finished\n");
}
static jint
get_cthreads(JNIEnv* jni, jthread** cthreads_p) {
jthread* tested_cthreads = NULL;
jint all_cnt = 0;
jint ct_cnt = 0;
jvmtiError err = jvmti->GetAllThreads(&all_cnt, &tested_cthreads);
check_jvmti_status(jni, err, "get_cthreads: error in JVMTI GetAllThreads");
for (int idx = 0; idx < all_cnt; idx++) {
jthread thread = tested_cthreads[idx];
char* tname = get_thread_name(jvmti, jni, thread);
if (strncmp(tname, CTHREAD_NAME_START, CTHREAD_NAME_START_LEN) != 0) {
continue;
}
tested_cthreads[ct_cnt++] = thread;
deallocate(jvmti, jni, (void*)tname);
}
*cthreads_p = tested_cthreads;
return ct_cnt;
}
static void
check_suspended_state(JNIEnv* jni, jthread thread, int thr_idx, char* tname, const char* func_name) {
void *thread_p = (void*)thread;
jint state = 0;
jvmtiError err = jvmti->GetThreadState(thread, &state);
check_jvmti_status(jni, err, "check_suspended_state: error in JVMTI GetThreadState");
LOG("## Agent: thread[%d] %p %s: state after suspend: %s (%d)\n",
thr_idx, thread_p, tname, TranslateState(state), (int)state);
if ((state & (JVMTI_THREAD_STATE_SUSPENDED | JVMTI_THREAD_STATE_TERMINATED)) == 0) {
LOG("\n## Agent: FAILED: %s did not turn on SUSPENDED flag:\n"
"# state: %s (%d)\n\n", func_name, TranslateState(state), (int)state);
set_agent_fail_status();
}
}
static void
check_resumed_state(JNIEnv* jni, jthread thread, int thr_idx, char* tname, const char* func_name) {
void *thread_p = (void*)thread;
jint state = 0;
jvmtiError err = jvmti->GetThreadState(thread, &state);
check_jvmti_status(jni, err, "check_resumed_state: error in JVMTI GetThreadState");
LOG("## Agent: thread[%d] %p %s: state after resume: %s (%d)\n",
thr_idx, thread_p, tname, TranslateState(state), (int)state);
if (!((state & (JVMTI_THREAD_STATE_SUSPENDED | JVMTI_THREAD_STATE_TERMINATED)) == 0)) {
LOG("\n## Agent: FAILED: %s did not turn off SUSPENDED flag:\n"
"# state: %s (%d)\n\n", func_name, TranslateState(state), (int)state);
set_agent_fail_status();
}
}
static void
test_vthread_suspend_all(JNIEnv* jni, const jthread* thread_list, int suspend_mask) {
LOG("\n## Agent: test_vthread_suspend_all started\n");
const jint EXCLUDE_CNT = 2;
jthread exclude_list[EXCLUDE_CNT] = { NULL, NULL };
for (int idx = 0; idx < EXCLUDE_CNT; idx++) {
exclude_list[idx] = thread_list[idx];
}
jvmtiError err = jvmti->SuspendAllVirtualThreads(EXCLUDE_CNT, exclude_list);
check_jvmti_status(jni, err, "test_vthread_suspend_all: error in JVMTI SuspendAllVirtualThreads");
for (int idx = 0; idx < VTHREAD_CNT; idx++) {
jthread thread = thread_list[idx];
char* tname = get_thread_name(jvmti, jni, thread);
if (idx < EXCLUDE_CNT && ((1 << idx) & suspend_mask) == 0) {
// thread is in exclude list and initially resumed: expected to remain resumed
check_resumed_state(jni, thread, idx, tname, "SuspendAllVirtualThreads");
err = jvmti->SuspendThread(thread);
check_jvmti_status(jni, err, "test_vthread_suspend_all: error in JVMTI SuspendThread");
} else {
// thread is not in exclude list or was initially suspended: expected to be suspended
check_suspended_state(jni, thread, idx, tname, "SuspendAllVirtualThreads");
}
deallocate(jvmti, jni, (void*)tname);
}
LOG("\n## Agent: test_vthread_suspend_all finished\n");
}
static void
test_vthread_resume_all(JNIEnv* jni, const jthread* thread_list, int suspend_mask) {
jvmtiError err;
LOG("\n## Agent: test_vthread_resume_all started\n");
const jint EXCLUDE_CNT = 2;
jthread exclude_list[EXCLUDE_CNT] = { NULL, NULL };
for (int idx = 0; idx < EXCLUDE_CNT; idx++) {
exclude_list[idx] = thread_list[idx];
// Enable Breakpoint events on excluded thread
err = jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_BREAKPOINT, thread_list[idx]);
check_jvmti_status(jni, err, "enableEvents: error in JVMTI SetEventNotificationMode: enable BREAKPOINT");
}
err = jvmti->ResumeAllVirtualThreads(EXCLUDE_CNT, exclude_list);
check_jvmti_status(jni, err, "test_vthread_resume_all: error in JVMTI ResumeAllVirtualThreads");
// wait a second to give the breakpoints a chance to be hit.
#ifdef WINDOWS
Sleep(1000);
#else
sleep(1);
#endif
for (int idx = 0; idx < EXCLUDE_CNT; idx++) {
// Disable Breakpoint events on excluded thread
err = jvmti->SetEventNotificationMode(JVMTI_DISABLE, JVMTI_EVENT_BREAKPOINT, thread_list[idx]);
check_jvmti_status(jni, err, "enableEvents: error in JVMTI SetEventNotificationMode: enable BREAKPOINT");
}
for (int idx = 0; idx < VTHREAD_CNT; idx++) {
jthread thread = thread_list[idx];
char* tname = get_thread_name(jvmti, jni, thread);
// The exclude list consists of vthreads #0 and #1, so these two vthreads were not resumed.
// If they expected to be suspended then resume them explicitly here.
if (idx < EXCLUDE_CNT && ((1 << idx) & suspend_mask) != 0) {
// thread is in exclude list and suspended: expected to remain suspended
check_suspended_state(jni, thread, idx, tname, "ResumeAllVirtualThreads");
err = jvmti->ResumeThread(thread); // now resume the thread from exclude list
check_jvmti_status(jni, err, "test_vthread_resume_all: error in JVMTI ResumeThread");
}
// thread is expected to be resumed now
check_resumed_state(jni, thread, idx, tname, "ResumeAllVirtualThreads");
deallocate(jvmti, jni, (void*)tname);
}
LOG("\n## Agent: test_vthread_resume_all: finished\n");
}
static void
check_threads_resumed_state(JNIEnv* jni, const jthread* thread_list, int thread_cnt) {
LOG("\n## Agent: check_all_vthreads_resumed_state started\n");
for (int idx = 0; idx < thread_cnt; idx++) {
jthread thread = thread_list[idx];
char* tname = get_thread_name(jvmti, jni, thread);
check_resumed_state(jni, thread, idx, tname, "<Final-Sanity-Check>");
deallocate(jvmti, jni, (void*)tname);
}
LOG("\n## Agent: check_threads_resumed_state: finished\n");
}
JNIEXPORT void JNICALL
Java_SuspendResumeAll_TestSuspendResume(JNIEnv* jni, jclass cls) {
jthread* tested_cthreads = NULL;
jint cthread_cnt = get_cthreads(jni, &tested_cthreads);
LOG("\n## TestSuspendResume: started\n");
test_vthread_suspend_all(jni, tested_vthreads, 0x0);
test_vthread_resume_all(jni, tested_vthreads, 0xFFFFFFFF);
LOG("\n\n## TestSuspendResume: Check all virtual threads are resumed\n");
check_threads_resumed_state(jni, tested_vthreads, VTHREAD_CNT);
LOG("\n\n## TestSuspendResume: Check all carrier threads are resumed\n");
check_threads_resumed_state(jni, tested_cthreads, cthread_cnt);
for (int i = 0; i < VTHREAD_CNT; i++) {
jni->DeleteGlobalRef(tested_vthreads[i]);
}
LOG("\n## TestSuspendResume: finished\n");
}
static void JNICALL
VirtualThreadStart(jvmtiEnv *jvmti, JNIEnv *jni, jthread vthread) {
RawMonitorLocker agent_start_locker(jvmti, jni, agent_event_lock);
tested_vthreads[vthread_no++] = jni->NewGlobalRef(vthread);
}
static void JNICALL
Breakpoint(jvmtiEnv *jvmti, JNIEnv* jni, jthread thread,
jmethodID method, jlocation location) {
char* mname = get_method_name(jvmti, jni, method);
jint state = 0;
jvmtiError err = jvmti->GetThreadState(thread, &state);
check_jvmti_status(jni, err, "Breakpoint: error in JVMTI GetThreadState");
LOG("## Agent: Breakpoint state(0x%x) %s\n", (int)state, TranslateState(state));
if ((state & JVMTI_THREAD_STATE_SUSPENDED) != 0) {
LOG("\n## ERROR: Breakpoint: suspended thread is running\n");
set_agent_fail_status();
}
// Turn off breakpoints notificatons for this thread.
err = jvmti->SetEventNotificationMode(JVMTI_DISABLE, JVMTI_EVENT_BREAKPOINT, thread);
if (err != JVMTI_ERROR_NONE) {
LOG("SetEventNotificationMode error: error in Breakpoint: %s (%d)\n",
TranslateError(err), err);
}
deallocate(jvmti, jni, (void*)mname);
}
JNIEXPORT jint JNICALL
Java_SuspendResumeAll_GetStatus(JNIEnv* jni, jclass cls) {
return get_agent_status();
}
jint Agent_Initialize(JavaVM *jvm, char *options, void *reserved) {
jvmtiError err;
LOG("Agent init started\n");
/* create JVMTI environment */
if (jvm->GetEnv((void **)(&jvmti), JVMTI_VERSION) != JNI_OK) {
LOG("Agent init: error in getting JvmtiEnv with GetEnv\n");
return JNI_ERR;
}
err = init_agent_data(jvmti, &agent_data);
if (err != JVMTI_ERROR_NONE) {
LOG("Agent init: error in init_agent_data: %s (%d)\n",
TranslateError(err), err);
return JNI_ERR;
}
/* add specific capabilities for suspending thread */
jvmtiCapabilities suspendCaps;
jvmtiEventCallbacks callbacks;
memset(&suspendCaps, 0, sizeof(suspendCaps));
suspendCaps.can_generate_breakpoint_events = 1;
suspendCaps.can_suspend = 1;
suspendCaps.can_support_virtual_threads = 1;
err = jvmti->AddCapabilities(&suspendCaps);
if (err != JVMTI_ERROR_NONE) {
LOG("Agent init: error in JVMTI AddCapabilities: %s (%d)\n",
TranslateError(err), err);
set_agent_fail_status();
return JNI_ERR;
}
memset(&callbacks, 0, sizeof(callbacks));
callbacks.VirtualThreadStart = &VirtualThreadStart;
callbacks.Breakpoint = &Breakpoint;
err = jvmti->SetEventCallbacks(&callbacks, sizeof(jvmtiEventCallbacks));
if (err != JVMTI_ERROR_NONE) {
LOG("Agent init: error in JVMTI SetEventCallbacks: %s (%d)\n",
TranslateError(err), err);
set_agent_fail_status();
return JNI_ERR;
}
err = jvmti->SetEventNotificationMode(JVMTI_ENABLE,
JVMTI_EVENT_VIRTUAL_THREAD_START, NULL);
if (err != JVMTI_ERROR_NONE) {
LOG("Agent init: error in JVMTI SetEventNotificationMode: %s (%d)\n",
TranslateError(err), err);
set_agent_fail_status();
return JNI_ERR;
}
agent_event_lock = create_raw_monitor(jvmti, "_agent_event_lock");
LOG("Agent init finished\n");
return JNI_OK;
}
/** Agent library initialization. */
JNIEXPORT jint JNICALL Agent_OnLoad(JavaVM *jvm, char *options, void *reserved) {
return Agent_Initialize(jvm, options, reserved);
}
JNIEXPORT jint JNICALL Agent_OnAttach(JavaVM *jvm, char *options, void *reserved) {
return Agent_Initialize(jvm, options, reserved);
}
}