/*
* Copyright (C) 2018 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import annotations.BootstrapMethod;
import annotations.CalledByIndy;
import java.lang.invoke.CallSite;
import java.lang.invoke.ConstantCallSite;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
class TestInvocationKinds extends TestBase {
private static int static_field;
private double instance_field;
static CallSite lookupStaticFieldGetter(
MethodHandles.Lookup lookup, String name, MethodType methodType) throws Throwable {
// methodType = "()LfieldType;"
MethodHandle mh =
lookup.findStaticGetter(TestInvocationKinds.class, name, methodType.returnType());
return new ConstantCallSite(mh);
}
@CalledByIndy(
bootstrapMethod =
@BootstrapMethod(
enclosingType = TestInvocationKinds.class,
name = "lookupStaticFieldSetter"
),
fieldOrMethodName = "static_field",
returnType = void.class,
parameterTypes = {int.class}
)
private static void setStaticField(int value) {
assertNotReached();
}
static CallSite lookupStaticFieldSetter(
MethodHandles.Lookup lookup, String name, MethodType methodType) throws Throwable {
// methodType = "(LfieldType;)V"
MethodHandle mh =
lookup.findStaticSetter(
TestInvocationKinds.class, name, methodType.parameterType(0));
return new ConstantCallSite(mh);
}
@CalledByIndy(
bootstrapMethod =
@BootstrapMethod(
enclosingType = TestInvocationKinds.class,
name = "lookupStaticFieldGetter"
),
fieldOrMethodName = "static_field",
returnType = int.class,
parameterTypes = {}
)
private static int getStaticField() {
assertNotReached();
return 0;
}
static CallSite lookupInstanceFieldSetter(
MethodHandles.Lookup lookup, String name, MethodType methodType) throws Throwable {
// methodType = "(Lreceiver;LfieldType;)V"
MethodHandle mh =
lookup.findSetter(methodType.parameterType(0), name, methodType.parameterType(1));
return new ConstantCallSite(mh);
}
@CalledByIndy(
bootstrapMethod =
@BootstrapMethod(
enclosingType = TestInvocationKinds.class,
name = "lookupInstanceFieldSetter"
),
fieldOrMethodName = "instance_field",
returnType = void.class,
parameterTypes = {TestInvocationKinds.class, double.class}
)
private static void setInstanceField(TestInvocationKinds instance, double value) {
assertNotReached();
instance.instance_field = Double.NaN;
}
static CallSite lookupInstanceFieldGetter(
MethodHandles.Lookup lookup, String name, MethodType methodType) throws Throwable {
// methodType = "(Lreceiver;)LfieldType;"
MethodHandle mh =
lookup.findGetter(methodType.parameterType(0), name, methodType.returnType());
return new ConstantCallSite(mh);
}
@CalledByIndy(
bootstrapMethod =
@BootstrapMethod(
enclosingType = TestInvocationKinds.class,
name = "lookupInstanceFieldGetter"
),
fieldOrMethodName = "instance_field",
returnType = double.class,
parameterTypes = {TestInvocationKinds.class}
)
private static double getInstanceField(TestInvocationKinds instance) {
assertNotReached();
return Double.NaN;
}
private static void testStaticFieldAccessors() {
System.out.println("testStaticFieldAccessors");
setStaticField(3);
assertEquals(static_field, 3);
setStaticField(4);
assertEquals(static_field, 4);
assertEquals(static_field, getStaticField());
static_field = Integer.MAX_VALUE;
assertEquals(Integer.MAX_VALUE, getStaticField());
}
private static void testInstanceFieldAccessors() {
System.out.println("testInstanceFieldAccessors");
TestInvocationKinds instance = new TestInvocationKinds();
instance.instance_field = Double.MIN_VALUE;
setInstanceField(instance, Math.PI);
assertEquals(Math.PI, instance.instance_field);
instance.instance_field = Math.E;
assertEquals(Math.E, getInstanceField(instance));
}
private static CallSite lookupVirtual(
MethodHandles.Lookup lookup, String name, MethodType methodType) throws Throwable {
// To get the point-of-use and invokedynamic to work the methodType here has the
// receiver type as the leading paramter which needs to be dropped for findVirtual().
MethodType mt = methodType.dropParameterTypes(0, 1);
MethodHandle mh = lookup.findVirtual(TestInvocationKinds.class, name, mt);
return new ConstantCallSite(mh);
}
@CalledByIndy(
bootstrapMethod =
@BootstrapMethod(enclosingType = TestInvocationKinds.class, name = "lookupVirtual"),
fieldOrMethodName = "getMaxIntegerValue",
returnType = int.class,
parameterTypes = {TestInvocationKinds.class, int.class, int.class}
)
private static int maxIntegerValue(TestInvocationKinds receiver, int x, int y) {
assertNotReached();
return 0;
}
public int getMaxIntegerValue(int x, int y) {
return x > y ? x : y;
}
static void testInvokeVirtual() {
System.out.print("testInvokeVirtual => max(77, -3) = ");
TestInvocationKinds receiver = new TestInvocationKinds();
int result = maxIntegerValue(receiver, 77, -3);
System.out.println(result);
}
static class Widget {
int value;
public Widget(int value) {}
}
private static CallSite lookupConstructor(
MethodHandles.Lookup lookup, String name, MethodType methodType) throws Throwable {
// methodType = (constructorParams);classToBeConstructed
Class<?> cls = methodType.returnType();
MethodType constructorMethodType = methodType.changeReturnType(void.class);
MethodHandle mh = lookup.findConstructor(cls, constructorMethodType);
return new ConstantCallSite(mh);
}
@CalledByIndy(
bootstrapMethod =
@BootstrapMethod(
enclosingType = TestInvocationKinds.class,
name = "lookupConstructor"
),
fieldOrMethodName = "unused",
returnType = Widget.class,
parameterTypes = {int.class}
)
private static Widget makeWidget(int v) {
assertNotReached();
return null;
}
static void testConstructor() {
System.out.print("testConstructor => ");
Widget receiver = makeWidget(3);
assertEquals(Widget.class, receiver.getClass());
System.out.println(receiver.getClass());
}
public static void test() {
System.out.println(TestInvocationKinds.class.getName());
testStaticFieldAccessors();
testInstanceFieldAccessors();
testInvokeVirtual();
testConstructor();
}
}