/* * 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(); } }