6e228ce382
Reviewed-by: sspitsyn, kevinw
394 lines
13 KiB
Java
394 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2020, 2024, 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 Test virtual threads using core reflection
|
|
* @modules java.base/java.lang:+open
|
|
* @library /test/lib
|
|
* @run junit Reflection
|
|
*/
|
|
|
|
import java.lang.reflect.Constructor;
|
|
import java.lang.reflect.InvocationTargetException;
|
|
import java.lang.reflect.Method;
|
|
import java.util.concurrent.CountDownLatch;
|
|
import java.util.concurrent.ExecutorService;
|
|
import java.util.concurrent.Executors;
|
|
import java.util.concurrent.ThreadFactory;
|
|
import java.util.concurrent.locks.LockSupport;
|
|
|
|
import jdk.test.lib.thread.VThreadScheduler;
|
|
import jdk.test.lib.thread.VThreadRunner;
|
|
import org.junit.jupiter.api.Test;
|
|
import static org.junit.jupiter.api.Assertions.*;
|
|
import static org.junit.jupiter.api.Assumptions.*;
|
|
|
|
class Reflection {
|
|
|
|
/**
|
|
* Test invoking static method.
|
|
*/
|
|
@Test
|
|
void testInvokeStatic1() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
int result = (int) divideMethod().invoke(null, 20, 2);
|
|
assertTrue(result == 10);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that InvocationTargetException is thrown when a static method throws
|
|
* exception.
|
|
*/
|
|
@Test
|
|
void testInvokeStatic2() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
try {
|
|
divideMethod().invoke(null, 20, 0);
|
|
fail();
|
|
} catch (InvocationTargetException e) {
|
|
assertTrue(e.getCause() instanceof ArithmeticException);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that IllegalArgumentException is thrown when trying to invoke a static
|
|
* method with bad parameters.
|
|
*/
|
|
@Test
|
|
void testInvokeStatic3() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> divideMethod().invoke(null));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> divideMethod().invoke(null, 1));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> divideMethod().invoke(null, 1, 2, 3));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> divideMethod().invoke(new Object()));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> divideMethod().invoke(new Object(), 1));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> divideMethod().invoke(new Object(), 1, 2, 3));
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that ExceptionInInitializerError is thrown when invoking a static
|
|
* method triggers its class to be initialized and it fails with exception.
|
|
*/
|
|
@Test
|
|
void testInvokeStatic4() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
Method foo = BadClass1.class.getDeclaredMethod("foo");
|
|
try {
|
|
foo.invoke(null);
|
|
fail();
|
|
} catch (ExceptionInInitializerError e) {
|
|
assertTrue(e.getCause() instanceof ArithmeticException);
|
|
}
|
|
});
|
|
}
|
|
|
|
static class BadClass1 {
|
|
static {
|
|
if (1==1) throw new ArithmeticException();
|
|
}
|
|
static void foo() { }
|
|
}
|
|
|
|
/**
|
|
* Test that an error is thrown when invoking a static method triggers its
|
|
* class to be initialized and it fails with an error.
|
|
*/
|
|
@Test
|
|
void testInvokeStatic5() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
Method foo = BadClass2.class.getDeclaredMethod("foo");
|
|
assertThrows(AbstractMethodError.class, () -> foo.invoke(null));
|
|
});
|
|
}
|
|
|
|
static class BadClass2 {
|
|
static {
|
|
if (1==1) throw new AbstractMethodError();
|
|
}
|
|
static void foo() { }
|
|
}
|
|
|
|
/**
|
|
* Test that invoking a static method does not pin the carrier thread.
|
|
*/
|
|
@Test
|
|
void testInvokeStatic6() throws Exception {
|
|
assumeTrue(VThreadScheduler.supportsCustomScheduler(), "No support for custom schedulers");
|
|
Method parkMethod = Parker.class.getDeclaredMethod("park");
|
|
try (ExecutorService scheduler = Executors.newFixedThreadPool(1)) {
|
|
ThreadFactory factory = VThreadScheduler.virtualThreadFactory(scheduler);
|
|
|
|
var ready = new CountDownLatch(1);
|
|
Thread vthread = factory.newThread(() -> {
|
|
ready.countDown();
|
|
try {
|
|
parkMethod.invoke(null); // blocks
|
|
} catch (Exception e) { }
|
|
});
|
|
vthread.start();
|
|
|
|
try {
|
|
// wait for thread to run
|
|
ready.await();
|
|
|
|
// unpark with another virtual thread, runs on same carrier thread
|
|
Thread unparker = factory.newThread(() -> LockSupport.unpark(vthread));
|
|
unparker.start();
|
|
unparker.join();
|
|
} finally {
|
|
LockSupport.unpark(vthread); // in case test fails
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Test invoking instance method.
|
|
*/
|
|
@Test
|
|
void testInvokeInstance1() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
var adder = new Adder();
|
|
Adder.addMethod().invoke(adder, 5);
|
|
assertTrue(adder.sum == 5);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that InvocationTargetException is thrown when an instance method throws
|
|
* exception.
|
|
*/
|
|
@Test
|
|
void testInvokeInstance2() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
var adder = new Adder();
|
|
try {
|
|
Adder.addMethod().invoke(adder, -5);
|
|
fail();
|
|
} catch (InvocationTargetException e) {
|
|
assertTrue(e.getCause() instanceof IllegalArgumentException);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that NullPointerException and IllegalArgumentException are thrown when
|
|
* trying to invoke an instance method with null or bad parameters.
|
|
*/
|
|
@Test
|
|
void testInvokeInstance3() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
var adder = new Adder();
|
|
Method addMethod = Adder.addMethod();
|
|
assertThrows(NullPointerException.class,
|
|
() -> addMethod.invoke(null));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> addMethod.invoke(adder));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> addMethod.invoke(adder, 1, 2));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> addMethod.invoke(adder, 1, "hi"));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> addMethod.invoke(adder, "hi"));
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test invoking newInstance to create an object.
|
|
*/
|
|
@Test
|
|
void testNewInstance1() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
Constructor<?> ctor = Adder.class.getDeclaredConstructor(long.class);
|
|
Adder adder = (Adder) ctor.newInstance(10);
|
|
assertTrue(adder.sum == 10);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that InvocationTargetException is thrown when a constructor throws
|
|
* exception.
|
|
*/
|
|
@Test
|
|
void testNewInstance2() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
Constructor<?> ctor = Adder.class.getDeclaredConstructor(long.class);
|
|
try {
|
|
ctor.newInstance(-10);
|
|
fail();
|
|
} catch (InvocationTargetException e) {
|
|
assertTrue(e.getCause() instanceof IllegalArgumentException);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that IllegalArgumentException is thrown when newInstacne is called
|
|
* with bad parameters.
|
|
*/
|
|
@Test
|
|
void testNewInstance3() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
var adder = new Adder();
|
|
Constructor<?> ctor = Adder.class.getDeclaredConstructor(long.class);
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> ctor.newInstance((Object[])null));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> ctor.newInstance(adder));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> ctor.newInstance(adder, null));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> ctor.newInstance(adder, "foo"));
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> ctor.newInstance(adder, 1, 2));
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that ExceptionInInitializerError is thrown when invoking newInstance
|
|
* triggers the class to be initialized and it fails with exception.
|
|
*/
|
|
@Test
|
|
void testNewInstance4() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
Constructor<?> ctor = BadClass3.class.getDeclaredConstructor();
|
|
try {
|
|
ctor.newInstance((Object[])null);
|
|
fail();
|
|
} catch (ExceptionInInitializerError e) {
|
|
assertTrue(e.getCause() instanceof ArithmeticException);
|
|
}
|
|
});
|
|
}
|
|
|
|
static class BadClass3 {
|
|
static {
|
|
if (1==1) throw new ArithmeticException();
|
|
}
|
|
static void foo() { }
|
|
}
|
|
|
|
/**
|
|
* Test that error is thrown when invoking newInstance triggers the class
|
|
* to be initialized and it fails with an error.
|
|
*/
|
|
@Test
|
|
void testNewInstance5() throws Exception {
|
|
VThreadRunner.run(() -> {
|
|
Constructor<?> ctor = BadClass4.class.getDeclaredConstructor();
|
|
assertThrows(AbstractMethodError.class, () -> ctor.newInstance((Object[])null));
|
|
});
|
|
}
|
|
|
|
static class BadClass4 {
|
|
static {
|
|
if (1==1) throw new AbstractMethodError();
|
|
}
|
|
static void foo() { }
|
|
}
|
|
|
|
/**
|
|
* Test that newInstance does not pin the carrier thread
|
|
*/
|
|
@Test
|
|
void testNewInstance6() throws Exception {
|
|
assumeTrue(VThreadScheduler.supportsCustomScheduler(), "No support for custom schedulers");
|
|
Constructor<?> ctor = Parker.class.getDeclaredConstructor();
|
|
try (ExecutorService scheduler = Executors.newFixedThreadPool(1)) {
|
|
ThreadFactory factory = VThreadScheduler.virtualThreadFactory(scheduler);
|
|
|
|
var ready = new CountDownLatch(1);
|
|
Thread vthread = factory.newThread(() -> {
|
|
ready.countDown();
|
|
try {
|
|
ctor.newInstance();
|
|
} catch (Exception e) { }
|
|
});
|
|
vthread.start();
|
|
|
|
try {
|
|
// wait for thread to run
|
|
ready.await();
|
|
|
|
// unpark with another virtual thread, runs on same carrier thread
|
|
Thread unparker = factory.newThread(() -> LockSupport.unpark(vthread));
|
|
unparker.start();
|
|
unparker.join();
|
|
} finally {
|
|
LockSupport.unpark(vthread); // in case test fails
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// -- support classes and methods --
|
|
|
|
static int divide(int x, int y) {
|
|
return x / y;
|
|
}
|
|
|
|
static Method divideMethod() throws NoSuchMethodException {
|
|
return Reflection.class.getDeclaredMethod("divide", int.class, int.class);
|
|
}
|
|
|
|
static class Adder {
|
|
long sum;
|
|
Adder() { }
|
|
Adder(long x) {
|
|
if (x < 0)
|
|
throw new IllegalArgumentException();
|
|
sum = x;
|
|
}
|
|
Adder add(long x) {
|
|
if (x < 0)
|
|
throw new IllegalArgumentException();
|
|
sum += x;
|
|
return this;
|
|
}
|
|
static Method addMethod() throws NoSuchMethodException {
|
|
return Adder.class.getDeclaredMethod("add", long.class);
|
|
}
|
|
long sum() {
|
|
return sum;
|
|
}
|
|
}
|
|
|
|
static class Parker {
|
|
Parker() {
|
|
LockSupport.park();
|
|
}
|
|
static void park() {
|
|
LockSupport.park();
|
|
}
|
|
}
|
|
}
|