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

338 lines
12 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 20
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 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) {
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];
}
jvmtiError err = jvmti->ResumeAllVirtualThreads(EXCLUDE_CNT, exclude_list);
check_jvmti_status(jni, err, "test_vthread_resume_all: error in JVMTI ResumeAllVirtualThreads");
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
test_vthread_suspend_half(JNIEnv* jni, const jthread* thread_list) {
jvmtiError err;
LOG("\n## Agent: test_vthread_suspend_half started\n");
for (int idx = 0; idx < VTHREAD_CNT; idx++) {
if (idx % 2 == 1) {
continue; // skip odd indeces
}
jthread thread = thread_list[idx];
char* tname = get_thread_name(jvmti, jni, thread);
err = jvmti->SuspendThread(thread);
check_jvmti_status(jni, err, "test_vthread_suspend_half: error in JVMTI SuspendThread");
check_suspended_state(jni, thread, idx, tname, "SuspendThread");
deallocate(jvmti, jni, (void*)tname);
}
LOG("\n## Agent: test_vthread_suspend_half finished\n");
}
static void
test_vthread_resume_half(JNIEnv* jni, const jthread* thread_list) {
jvmtiError err;
LOG("\n## Agent: test_vthread_resume_half started\n");
for (int idx = 0; idx < VTHREAD_CNT; idx++) {
if (idx % 2 == 1) {
continue; // skip odd indeces
}
jthread thread = thread_list[idx];
char* tname = get_thread_name(jvmti, jni, thread);
err = jvmti->ResumeThread(thread);
check_jvmti_status(jni, err, "test_vthread_resume_half: error in JVMTI ResumeThread");
check_resumed_state(jni, thread, idx, tname, "ResumeThread");
deallocate(jvmti, jni, (void*)tname);
}
LOG("\n## Agent: test_vthread_resume_half: finished\n");
}
static void
check_threads_resumed_state(JNIEnv* jni, const jthread* thread_list, int thread_cnt) {
LOG("\n## Agent: check_threads_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_SuspendResume2_TestSuspendResume(JNIEnv* jni, jclass cls) {
jthread* tested_cthreads = NULL;
LOG("\n## TestSuspendResume: started\n");
test_vthread_suspend_all(jni, tested_vthreads, 0x0);
test_vthread_resume_all(jni, tested_vthreads, 0xFFFFFFFF);
test_vthread_suspend_half(jni, tested_vthreads);
test_vthread_resume_all(jni, tested_vthreads, 0x55555555);
test_vthread_suspend_all(jni, tested_vthreads, 0x0);
test_vthread_resume_half(jni, tested_vthreads);
test_vthread_resume_all(jni, tested_vthreads, 0xAAAAAAAA);
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");
jint cthread_cnt = get_cthreads(jni, &tested_cthreads);
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);
}
JNIEXPORT jint JNICALL
Java_SuspendResume2_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_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;
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);
}
}