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
177 lines
8.7 KiB
Java
177 lines
8.7 KiB
Java
/*
|
|
* Copyright (c) 2019, 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
|
|
* @bug 8246774
|
|
* @summary Basic tests for ObjectMethods
|
|
* @run testng ObjectMethodsTest
|
|
*/
|
|
|
|
import java.util.List;
|
|
import java.lang.invoke.CallSite;
|
|
import java.lang.invoke.MethodHandle;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.invoke.MethodType;
|
|
import java.lang.runtime.ObjectMethods;
|
|
import org.testng.annotations.Test;
|
|
import static java.lang.invoke.MethodType.methodType;
|
|
import static org.testng.Assert.assertEquals;
|
|
import static org.testng.Assert.assertThrows;
|
|
import static org.testng.Assert.assertFalse;
|
|
import static org.testng.Assert.assertTrue;
|
|
|
|
@Test
|
|
public class ObjectMethodsTest {
|
|
|
|
public static class C {
|
|
static final MethodType EQUALS_DESC = methodType(boolean.class, C.class, Object.class);
|
|
static final MethodType HASHCODE_DESC = methodType(int.class, C.class);
|
|
static final MethodType TO_STRING_DESC = methodType(String.class, C.class);
|
|
|
|
static final MethodHandle[] ACCESSORS = accessors();
|
|
static final String NAME_LIST = "x;y";
|
|
private static MethodHandle[] accessors() {
|
|
try {
|
|
return new MethodHandle[]{
|
|
MethodHandles.lookup().findGetter(C.class, "x", int.class),
|
|
MethodHandles.lookup().findGetter(C.class, "y", int.class),
|
|
};
|
|
} catch (Exception e) {
|
|
throw new AssertionError(e);
|
|
}
|
|
}
|
|
|
|
private final int x;
|
|
private final int y;
|
|
C (int x, int y) { this.x = x; this.y = y; }
|
|
public int x() { return x; }
|
|
public int y() { return y; }
|
|
}
|
|
|
|
static class Empty {
|
|
static final MethodType EQUALS_DESC = methodType(boolean.class, Empty.class, Object.class);
|
|
static final MethodType HASHCODE_DESC = methodType(int.class, Empty.class);
|
|
static final MethodType TO_STRING_DESC = methodType(String.class, Empty.class);
|
|
static final MethodHandle[] ACCESSORS = new MethodHandle[] { };
|
|
static final String NAME_LIST = "";
|
|
Empty () { }
|
|
}
|
|
|
|
static final MethodHandles.Lookup LOOKUP = MethodHandles.lookup();
|
|
|
|
public void testEqualsC() throws Throwable {
|
|
CallSite cs = (CallSite)ObjectMethods.bootstrap(LOOKUP, "equals", C.EQUALS_DESC, C.class, C.NAME_LIST, C.ACCESSORS);
|
|
MethodHandle handle = cs.dynamicInvoker();
|
|
C c = new C(5, 5);
|
|
assertTrue((boolean)handle.invokeExact(c, (Object)c));
|
|
assertTrue((boolean)handle.invokeExact(c, (Object)new C(5, 5)));
|
|
assertFalse((boolean)handle.invokeExact(c, (Object)new C(5, 4)));
|
|
assertFalse((boolean)handle.invokeExact(c, (Object)new C(4, 5)));
|
|
assertFalse((boolean)handle.invokeExact(c, (Object)null));
|
|
assertFalse((boolean)handle.invokeExact(c, new Object()));
|
|
}
|
|
|
|
public void testEqualsEmpty() throws Throwable {
|
|
CallSite cs = (CallSite)ObjectMethods.bootstrap(LOOKUP, "equals", Empty.EQUALS_DESC, Empty.class, Empty.NAME_LIST, Empty.ACCESSORS);
|
|
MethodHandle handle = cs.dynamicInvoker();
|
|
Empty e = new Empty();
|
|
assertTrue((boolean)handle.invokeExact(e, (Object)e));
|
|
assertTrue((boolean)handle.invokeExact(e, (Object)new Empty()));
|
|
assertFalse((boolean)handle.invokeExact(e, (Object)null));
|
|
assertFalse((boolean)handle.invokeExact(e, new Object()));
|
|
}
|
|
|
|
public void testHashCodeC() throws Throwable {
|
|
CallSite cs = (CallSite)ObjectMethods.bootstrap(LOOKUP, "hashCode", C.HASHCODE_DESC, C.class, "x;y", C.ACCESSORS);
|
|
MethodHandle handle = cs.dynamicInvoker();
|
|
C c = new C(6, 7);
|
|
int hc = (int)handle.invokeExact(c);
|
|
assertEquals(hc, hashCombiner(c.x(), c.y()));
|
|
|
|
assertEquals((int)handle.invokeExact(new C(100, 1)), hashCombiner(100, 1));
|
|
assertEquals((int)handle.invokeExact(new C(0, 0)), hashCombiner(0, 0));
|
|
assertEquals((int)handle.invokeExact(new C(-1, 100)), hashCombiner(-1, 100));
|
|
assertEquals((int)handle.invokeExact(new C(100, 1)), hashCombiner(100, 1));
|
|
assertEquals((int)handle.invokeExact(new C(100, -1)), hashCombiner(100, -1));
|
|
}
|
|
|
|
public void testHashCodeEmpty() throws Throwable {
|
|
CallSite cs = (CallSite)ObjectMethods.bootstrap(LOOKUP, "hashCode", Empty.HASHCODE_DESC, Empty.class, "", Empty.ACCESSORS);
|
|
MethodHandle handle = cs.dynamicInvoker();
|
|
Empty e = new Empty();
|
|
assertEquals((int)handle.invokeExact(e), 0);
|
|
}
|
|
|
|
public void testToStringC() throws Throwable {
|
|
CallSite cs = (CallSite)ObjectMethods.bootstrap(LOOKUP, "toString", C.TO_STRING_DESC, C.class, C.NAME_LIST, C.ACCESSORS);
|
|
MethodHandle handle = cs.dynamicInvoker();
|
|
assertEquals((String)handle.invokeExact(new C(8, 9)), "C[x=8, y=9]" );
|
|
assertEquals((String)handle.invokeExact(new C(10, 11)), "C[x=10, y=11]" );
|
|
assertEquals((String)handle.invokeExact(new C(100, -9)), "C[x=100, y=-9]");
|
|
assertEquals((String)handle.invokeExact(new C(0, 0)), "C[x=0, y=0]" );
|
|
}
|
|
|
|
public void testToStringEmpty() throws Throwable {
|
|
CallSite cs = (CallSite)ObjectMethods.bootstrap(LOOKUP, "toString", Empty.TO_STRING_DESC, Empty.class, Empty.NAME_LIST, Empty.ACCESSORS);
|
|
MethodHandle handle = cs.dynamicInvoker();
|
|
assertEquals((String)handle.invokeExact(new Empty()), "Empty[]");
|
|
}
|
|
|
|
Class<NullPointerException> NPE = NullPointerException.class;
|
|
Class<IllegalArgumentException> IAE = IllegalArgumentException.class;
|
|
|
|
public void exceptions() {
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "badName", C.EQUALS_DESC, C.class, C.NAME_LIST, C.ACCESSORS));
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "toString", C.TO_STRING_DESC, C.class, "x;y;z", C.ACCESSORS));
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "toString", C.TO_STRING_DESC, C.class, "x;y", new MethodHandle[]{}));
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "toString", C.TO_STRING_DESC, this.getClass(), "x;y", C.ACCESSORS));
|
|
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "toString", C.EQUALS_DESC, C.class, "x;y", C.ACCESSORS));
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "hashCode", C.TO_STRING_DESC, C.class, "x;y", C.ACCESSORS));
|
|
assertThrows(IAE, () -> ObjectMethods.bootstrap(LOOKUP, "equals", C.HASHCODE_DESC, C.class, "x;y", C.ACCESSORS));
|
|
|
|
record NamePlusType(String mn, MethodType mt) {}
|
|
List<NamePlusType> namePlusTypeList = List.of(
|
|
new NamePlusType("toString", C.TO_STRING_DESC),
|
|
new NamePlusType("equals", C.EQUALS_DESC),
|
|
new NamePlusType("hashCode", C.HASHCODE_DESC)
|
|
);
|
|
|
|
for (NamePlusType npt : namePlusTypeList) {
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(LOOKUP, npt.mn(), npt.mt(), C.class, "x;y", null));
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(LOOKUP, npt.mn(), npt.mt(), C.class, "x;y", new MethodHandle[]{null}));
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(LOOKUP, npt.mn(), npt.mt(), C.class, null, C.ACCESSORS));
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(LOOKUP, npt.mn(), npt.mt(), null, "x;y", C.ACCESSORS));
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(LOOKUP, npt.mn(), null, C.class, "x;y", C.ACCESSORS));
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(LOOKUP, null, npt.mt(), C.class, "x;y", C.ACCESSORS));
|
|
assertThrows(NPE, () -> ObjectMethods.bootstrap(null, npt.mn(), npt.mt(), C.class, "x;y", C.ACCESSORS));
|
|
}
|
|
}
|
|
|
|
// Based on the ObjectMethods internal implementation
|
|
private static int hashCombiner(int x, int y) {
|
|
return x*31 + y;
|
|
}
|
|
}
|