db85090553
Co-authored-by: Sean Mullan <mullan@openjdk.org> Co-authored-by: Alan Bateman <alanb@openjdk.org> Co-authored-by: Weijun Wang <weijun@openjdk.org> Co-authored-by: Aleksei Efimov <aefimov@openjdk.org> Co-authored-by: Brian Burkhalter <bpb@openjdk.org> Co-authored-by: Daniel Fuchs <dfuchs@openjdk.org> Co-authored-by: Harshitha Onkar <honkar@openjdk.org> Co-authored-by: Joe Wang <joehw@openjdk.org> Co-authored-by: Jorn Vernee <jvernee@openjdk.org> Co-authored-by: Justin Lu <jlu@openjdk.org> Co-authored-by: Kevin Walls <kevinw@openjdk.org> Co-authored-by: Lance Andersen <lancea@openjdk.org> Co-authored-by: Naoto Sato <naoto@openjdk.org> Co-authored-by: Roger Riggs <rriggs@openjdk.org> Co-authored-by: Brent Christian <bchristi@openjdk.org> Co-authored-by: Stuart Marks <smarks@openjdk.org> Co-authored-by: Ian Graves <igraves@openjdk.org> Co-authored-by: Phil Race <prr@openjdk.org> Co-authored-by: Erik Gahlin <egahlin@openjdk.org> Co-authored-by: Jaikiran Pai <jpai@openjdk.org> Reviewed-by: kevinw, aivanov, rriggs, lancea, coffeys, dfuchs, ihse, erikj, cjplummer, coleenp, naoto, mchung, prr, weijun, joehw, azvegint, psadhukhan, bchristi, sundar, attila
249 lines
11 KiB
Java
249 lines
11 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.
|
|
*/
|
|
import static jdk.dynalink.StandardNamespace.PROPERTY;
|
|
import static jdk.dynalink.StandardOperation.GET;
|
|
|
|
import java.lang.invoke.CallSite;
|
|
import java.lang.invoke.MethodHandle;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.invoke.MethodType;
|
|
import jdk.dynalink.CallSiteDescriptor;
|
|
import jdk.dynalink.DynamicLinker;
|
|
import jdk.dynalink.DynamicLinkerFactory;
|
|
import jdk.dynalink.NoSuchDynamicMethodException;
|
|
import jdk.dynalink.Operation;
|
|
import jdk.dynalink.StandardNamespace;
|
|
import jdk.dynalink.StandardOperation;
|
|
import jdk.dynalink.beans.StaticClass;
|
|
import jdk.dynalink.linker.GuardedInvocation;
|
|
import jdk.dynalink.linker.LinkRequest;
|
|
import jdk.dynalink.linker.LinkerServices;
|
|
import jdk.dynalink.support.SimpleRelinkableCallSite;
|
|
import org.testng.Assert;
|
|
import org.testng.annotations.Test;
|
|
|
|
/**
|
|
* @test
|
|
* @build TestGuardingDynamicLinkerExporter
|
|
* @run testng/othervm TrustedDynamicLinkerFactoryTest
|
|
*/
|
|
public class TrustedDynamicLinkerFactoryTest {
|
|
|
|
private static final Operation GET_PROPERTY = GET.withNamespace(PROPERTY);
|
|
|
|
private static DynamicLinkerFactory newDynamicLinkerFactory(final boolean resetClassLoader) {
|
|
final DynamicLinkerFactory factory = new DynamicLinkerFactory();
|
|
if (resetClassLoader) {
|
|
factory.setClassLoader(null);
|
|
}
|
|
return factory;
|
|
}
|
|
|
|
@Test
|
|
public void callSiteCreationTest() {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(true);
|
|
final DynamicLinker linker = factory.createLinker();
|
|
final StandardOperation[] operations = StandardOperation.values();
|
|
final MethodType mt = MethodType.methodType(Object.class, Object.class);
|
|
for (final Operation op : operations) {
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(new CallSiteDescriptor(
|
|
MethodHandles.publicLookup(), op, mt)));
|
|
Assert.assertNotNull(cs);
|
|
Assert.assertEquals(cs.type(), mt);
|
|
Assert.assertNotNull(cs.getTarget());
|
|
}
|
|
}
|
|
|
|
@Test
|
|
public void fallbackLinkerTest() {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(true);
|
|
final Operation myOperation = new Operation() {
|
|
};
|
|
final boolean[] reachedFallback = { false };
|
|
factory.setFallbackLinkers((final LinkRequest linkRequest, final LinkerServices linkerServices) -> {
|
|
Assert.assertEquals(linkRequest.getCallSiteDescriptor().getOperation(), myOperation);
|
|
reachedFallback[0] = true;
|
|
return null;
|
|
});
|
|
|
|
final DynamicLinker linker = factory.createLinker();
|
|
final MethodType mt = MethodType.methodType(Object.class);
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(new CallSiteDescriptor(
|
|
MethodHandles.publicLookup(), myOperation, mt)));
|
|
|
|
// linking the call site initially does not invoke the linkers!
|
|
Assert.assertFalse(reachedFallback[0]);
|
|
try {
|
|
cs.getTarget().invoke();
|
|
} catch (final NoSuchDynamicMethodException nsdm) {
|
|
// we do expect NoSuchDynamicMethod!
|
|
// because our dummy fallback linker returns null!
|
|
} catch (final Throwable th) {
|
|
throw new RuntimeException("should not reach here with: " + th);
|
|
}
|
|
|
|
// check that the control reached fallback linker!
|
|
Assert.assertTrue(reachedFallback[0]);
|
|
}
|
|
|
|
@Test
|
|
public void priorityLinkerTest() {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(true);
|
|
final Operation myOperation = new Operation() {
|
|
};
|
|
final boolean[] reachedProrityLinker = { false };
|
|
factory.setPrioritizedLinker((final LinkRequest linkRequest, final LinkerServices linkerServices) -> {
|
|
Assert.assertEquals(linkRequest.getCallSiteDescriptor().getOperation(), myOperation);
|
|
reachedProrityLinker[0] = true;
|
|
return null;
|
|
});
|
|
|
|
final DynamicLinker linker = factory.createLinker();
|
|
final MethodType mt = MethodType.methodType(Object.class);
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(new CallSiteDescriptor(
|
|
MethodHandles.publicLookup(), myOperation, mt)));
|
|
|
|
// linking the call site initially does not invoke the linkers!
|
|
Assert.assertFalse(reachedProrityLinker[0]);
|
|
try {
|
|
cs.getTarget().invoke();
|
|
} catch (final NoSuchDynamicMethodException nsdm) {
|
|
// we do expect NoSuchDynamicMethod!
|
|
// because our dummy priority linker returns null!
|
|
} catch (final Throwable th) {
|
|
throw new RuntimeException("should not reach here with: " + th);
|
|
}
|
|
|
|
// check that the control reached fallback linker!
|
|
Assert.assertTrue(reachedProrityLinker[0]);
|
|
}
|
|
|
|
@Test
|
|
public void priorityAndFallbackLinkerTest() {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(true);
|
|
final Operation myOperation = new Operation() {
|
|
};
|
|
final int[] linkerReachCounter = { 0 };
|
|
factory.setPrioritizedLinker((final LinkRequest linkRequest, final LinkerServices linkerServices) -> {
|
|
Assert.assertEquals(linkRequest.getCallSiteDescriptor().getOperation(), myOperation);
|
|
linkerReachCounter[0]++;
|
|
return null;
|
|
});
|
|
factory.setFallbackLinkers((final LinkRequest linkRequest, final LinkerServices linkerServices) -> {
|
|
Assert.assertEquals(linkRequest.getCallSiteDescriptor().getOperation(), myOperation);
|
|
Assert.assertEquals(linkerReachCounter[0], 1);
|
|
linkerReachCounter[0]++;
|
|
return null;
|
|
});
|
|
|
|
final DynamicLinker linker = factory.createLinker();
|
|
final MethodType mt = MethodType.methodType(Object.class);
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(new CallSiteDescriptor(
|
|
MethodHandles.publicLookup(), myOperation, mt)));
|
|
|
|
// linking the call site initially does not invoke the linkers!
|
|
Assert.assertEquals(linkerReachCounter[0], 0);
|
|
|
|
try {
|
|
cs.getTarget().invoke();
|
|
} catch (final NoSuchDynamicMethodException nsdm) {
|
|
// we do expect NoSuchDynamicMethod!
|
|
} catch (final Throwable th) {
|
|
throw new RuntimeException("should not reach here with: " + th);
|
|
}
|
|
|
|
Assert.assertEquals(linkerReachCounter[0], 2);
|
|
}
|
|
|
|
@Test
|
|
public void prelinkTransformerTest() throws Throwable {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(true);
|
|
final boolean[] reachedPrelinkTransformer = { false };
|
|
|
|
factory.setPrelinkTransformer((final GuardedInvocation inv, final LinkRequest linkRequest, final LinkerServices linkerServices) -> {
|
|
reachedPrelinkTransformer[0] = true;
|
|
// just identity transformer!
|
|
return inv;
|
|
});
|
|
|
|
final MethodType mt = MethodType.methodType(Object.class, Object.class, String.class);
|
|
final DynamicLinker linker = factory.createLinker();
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(new CallSiteDescriptor(
|
|
MethodHandles.publicLookup(), GET_PROPERTY, mt)));
|
|
Assert.assertFalse(reachedPrelinkTransformer[0]);
|
|
Assert.assertEquals(cs.getTarget().invoke(new Object(), "class"), Object.class);
|
|
Assert.assertTrue(reachedPrelinkTransformer[0]);
|
|
}
|
|
|
|
@Test
|
|
public void internalObjectsFilterTest() throws Throwable {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(true);
|
|
final boolean[] reachedInternalObjectsFilter = { false };
|
|
|
|
factory.setInternalObjectsFilter((final MethodHandle mh) -> {
|
|
reachedInternalObjectsFilter[0] = true;
|
|
return mh;
|
|
});
|
|
|
|
final MethodType mt = MethodType.methodType(Object.class, Object.class, String.class);
|
|
final DynamicLinker linker = factory.createLinker();
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(new CallSiteDescriptor(
|
|
MethodHandles.publicLookup(), GET_PROPERTY, mt)));
|
|
Assert.assertFalse(reachedInternalObjectsFilter[0]);
|
|
Assert.assertEquals(cs.getTarget().invoke(new Object(), "class"), Object.class);
|
|
Assert.assertTrue(reachedInternalObjectsFilter[0]);
|
|
}
|
|
|
|
@Test
|
|
public void autoLoadedLinkerTest() {
|
|
testAutoLoadedLinkerInvoked(new Object(), "toString");
|
|
}
|
|
|
|
@Test
|
|
public void autoLoadedLinkerSeesStaticMethod() {
|
|
testAutoLoadedLinkerInvoked(StaticClass.forClass(System.class), "currentTimeMillis");
|
|
}
|
|
|
|
private static void testAutoLoadedLinkerInvoked(final Object target, final String methodName) {
|
|
final DynamicLinkerFactory factory = newDynamicLinkerFactory(false);
|
|
final DynamicLinker linker = factory.createLinker();
|
|
|
|
final MethodType mt = MethodType.methodType(Object.class, Object.class);
|
|
final CallSiteDescriptor testDescriptor = new CallSiteDescriptor(MethodHandles.publicLookup(),
|
|
GET.withNamespace(StandardNamespace.METHOD).named(methodName), mt);
|
|
final CallSite cs = linker.link(new SimpleRelinkableCallSite(testDescriptor));
|
|
|
|
TestGuardingDynamicLinkerExporter.enable();
|
|
try {
|
|
cs.getTarget().invoke(target);
|
|
// The linker was loaded and it observed our invocation
|
|
Assert.assertTrue(TestGuardingDynamicLinkerExporter.isLastCallSiteDescriptor(testDescriptor));
|
|
} catch (final Throwable th) {
|
|
throw new RuntimeException(th);
|
|
} finally {
|
|
TestGuardingDynamicLinkerExporter.disable();
|
|
}
|
|
|
|
}
|
|
}
|