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