/* * Copyright (c) 2020, 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 * @bug 8284161 * @summary Test java.lang.management.ThreadMXBean with virtual threads * @enablePreview * @modules java.base/java.lang:+open java.management * @run testng/othervm VirtualThreads */ /** * @test * @requires vm.continuations * @enablePreview * @modules java.base/java.lang:+open java.management * @run testng/othervm -XX:+UnlockExperimentalVMOptions -XX:-VMContinuations VirtualThreads */ import java.lang.management.ManagementFactory; import java.lang.management.ThreadInfo; import java.lang.management.ThreadMXBean; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.nio.channels.Selector; import java.util.Arrays; import java.util.concurrent.Executor; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.locks.LockSupport; import org.testng.SkipException; import org.testng.annotations.Test; import static org.testng.Assert.*; public class VirtualThreads { /** * Test that ThreadMXBean::getAllThreadsIds does not include thread ids for * virtual threads. */ @Test public void testGetAllThreadIds() throws Exception { runInVirtualThread(() -> { long currentTid = Thread.currentThread().getId(); long[] tids = ManagementFactory.getThreadMXBean().getAllThreadIds(); boolean found = Arrays.stream(tids).anyMatch(tid -> tid == currentTid); assertFalse(found); }); } /** * Test that ThreadMXBean::getThreadInfo returns null for a virual thread. */ @Test public void testGetThreadInfo1() throws Exception { runInVirtualThread(() -> { long tid = Thread.currentThread().getId(); ThreadInfo info = ManagementFactory.getThreadMXBean().getThreadInfo(tid); assertTrue(info == null); }); } /** * Test that ThreadMXBean::getThreadInfo on a carrier thread. The stack * frames of the virtual thread should not be returned. */ @Test public void testGetThreadInfo2() throws Exception { if (!supportsCustomScheduler()) throw new SkipException("No support for custom schedulers"); try (ExecutorService pool = Executors.newFixedThreadPool(1)) { var carrierRef = new AtomicReference(); Executor scheduler = (task) -> { pool.execute(() -> { carrierRef.set(Thread.currentThread()); task.run(); }); }; // start virtual thread so carrier Thread can be captured virtualThreadBuilder(scheduler).start(() -> { }).join(); Thread carrier = carrierRef.get(); assertTrue(carrier != null && !carrier.isVirtual()); try (Selector sel = Selector.open()) { // start virtual thread that blocks in a native method virtualThreadBuilder(scheduler).start(() -> { try { sel.select(); } catch (Exception e) { } }); // invoke getThreadInfo get and check the stack trace long tid = carrier.getId(); ThreadInfo info = ManagementFactory.getThreadMXBean().getThreadInfo(tid, 100); // should only see carrier frames StackTraceElement[] stack = info.getStackTrace(); assertTrue(contains(stack, "java.util.concurrent.ThreadPoolExecutor")); assertFalse(contains(stack, "java.nio.channels.Selector")); // carrier should not be holding any monitors assertTrue(info.getLockedMonitors().length == 0); } } } /** * Test that getThreadCpuTime returns -1 for a virual thread.. */ @Test public void testGetThreadCpuTime() throws Exception { runInVirtualThread(() -> { long tid = Thread.currentThread().getId(); long cpuTime = ManagementFactory.getThreadMXBean().getThreadCpuTime(tid); assertTrue(cpuTime == -1L); }); } /** * Test that getThreadUserTime returns -1 for a virual thread. */ @Test public void testGetThreadUserTime() throws Exception { runInVirtualThread(() -> { long tid = Thread.currentThread().getId(); long cpuTime = ManagementFactory.getThreadMXBean().getThreadUserTime(tid); assertTrue(cpuTime == -1L); }); } /** * Test that ThreadMXBean::getCurrentThreadCpuTime throws UOE when invoked * on a virtual thread. */ @Test(expectedExceptions = { UnsupportedOperationException.class }) public void testGetCurrentThreadCpuTime() throws Exception { runInVirtualThread(() -> { ManagementFactory.getThreadMXBean().getCurrentThreadCpuTime(); }); } /** * Test that ThreadMXBean::getCurrentThreadUserTime throws UOE when * invoked on a virtual thread. */ @Test(expectedExceptions = { UnsupportedOperationException.class }) public void testGetCurrentThreadUserTime() throws Exception { runInVirtualThread(() -> { ManagementFactory.getThreadMXBean().getCurrentThreadUserTime(); }); } /** * Test that ThreadMXBean::getCurrentThreadAllocatedBytes returns -1 when * invoked on a virtual thread. */ @Test public void testGetCurrentThreadAllocatedBytes() throws Exception { runInVirtualThread(() -> { long allocated = ManagementFactory.getPlatformMXBean(com.sun.management.ThreadMXBean.class) .getCurrentThreadAllocatedBytes(); assertTrue(allocated == -1L); }); } interface ThrowingRunnable { void run() throws Exception; } private static void runInVirtualThread(ThrowingRunnable task) throws Exception { AtomicReference exc = new AtomicReference<>(); Runnable target = () -> { try { task.run(); } catch (Error e) { exc.set(new RuntimeException(e)); } catch (Exception e) { exc.set(e); } }; Thread thread = Thread.ofVirtual().start(target); thread.join(); Exception e = exc.get(); if (e != null) { throw e; } } private static boolean contains(StackTraceElement[] stack, String className) { return Arrays.stream(stack) .map(StackTraceElement::getClassName) .anyMatch(className::equals); } /** * Returns a builder to create virtual threads that use the given scheduler. * @throws UnsupportedOperationException if there is no support for custom schedulers */ private static Thread.Builder.OfVirtual virtualThreadBuilder(Executor scheduler) { Thread.Builder.OfVirtual builder = Thread.ofVirtual(); try { Class clazz = Class.forName("java.lang.ThreadBuilders$VirtualThreadBuilder"); Constructor ctor = clazz.getDeclaredConstructor(Executor.class); ctor.setAccessible(true); return (Thread.Builder.OfVirtual) ctor.newInstance(scheduler); } catch (InvocationTargetException e) { Throwable cause = e.getCause(); if (cause instanceof RuntimeException re) { throw re; } throw new RuntimeException(e); } catch (Exception e) { throw new RuntimeException(e); } } /** * Return true if custom schedulers are supported. */ private static boolean supportsCustomScheduler() { try (var pool = Executors.newCachedThreadPool()) { try { virtualThreadBuilder(pool); return true; } catch (UnsupportedOperationException e) { return false; } } } }