jdk/test/java/lang/invoke/LFCaching/TestMethods.java
author kshefov
Mon, 17 Aug 2015 12:21:34 +0300
changeset 32214 7e67ff2f61d4
parent 32213 acd9aab4ae86
child 45283 fcb2af038805
permissions -rw-r--r--
8060717: [TESTBUG] Improve test coverage of MethodHandles.explicitCastArguments() Reviewed-by: vlivanov, mhaupt

/*
 * Copyright (c) 2014, 2015, 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 com.oracle.testlibrary.jsr292.Helper;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * Enumeration containing information about methods from
 * {@code j.l.i.MethodHandles} class that are used for testing lambda forms
 * caching.
 *
 * @author kshefov
 */
public enum TestMethods {

    FOLD_ARGUMENTS("foldArguments") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int modifierMHArgNum = Helper.RNG.nextInt(realArity + 1);
                    data.put("modifierMHArgNum", modifierMHArgNum);
                    Class<?> combinerReturnType;
                    if (realArity == 0) {
                        combinerReturnType = void.class;
                    } else {
                        combinerReturnType = Helper.RNG.nextBoolean() ? void.class : mtTarget.parameterType(0);
                    }
                    data.put("combinerReturnType", combinerReturnType);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    Class<?> combinerReturnType = (Class) data.get("combinerReturnType");
                    int modifierMHArgNum = (int) data.get("modifierMHArgNum");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    Class<?> rType = mtTarget.returnType();
                    int combListStart = (combinerReturnType == void.class) ? 0 : 1;
                    if (modifierMHArgNum < combListStart) {
                        modifierMHArgNum = combListStart;
                    }
                    MethodHandle combiner = TestMethods.methodHandleGenerator(combinerReturnType,
                            mtTarget.parameterList().subList(combListStart,
                                    modifierMHArgNum), kind);
                    return MethodHandles.foldArguments(target, combiner);
                }
            },
    DROP_ARGUMENTS("dropArguments") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int dropArgsPos = Helper.RNG.nextInt(realArity + 1);
                    data.put("dropArgsPos", dropArgsPos);
                    MethodType mtDropArgs = TestMethods.randomMethodTypeGenerator(
                            Helper.RNG.nextInt(super.maxArity - realArity));
                    data.put("mtDropArgs", mtDropArgs);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    MethodType mtDropArgs = (MethodType) data.get("mtDropArgs");
                    int dropArgsPos = (int) data.get("dropArgsPos");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    int mtTgtSlotsCount = TestMethods.argSlotsCount(mtTarget);
                    int mtDASlotsCount = TestMethods.argSlotsCount(mtDropArgs);
                    List<Class<?>> fakeParList;
                    if (mtTgtSlotsCount + mtDASlotsCount > super.maxArity - 1) {
                        fakeParList = TestMethods.reduceArgListToSlotsCount(mtDropArgs.parameterList(),
                                super.maxArity - mtTgtSlotsCount - 1);
                    } else {
                        fakeParList = mtDropArgs.parameterList();
                    }
                    return MethodHandles.dropArguments(target, dropArgsPos, fakeParList);
                }
            },
    EXPLICIT_CAST_ARGUMENTS("explicitCastArguments", Helper.MAX_ARITY / 2) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    MethodType mtExcplCastArgs = TestMethods.randomMethodTypeGenerator(realArity);
                    if (mtTarget.returnType() == void.class) {
                        mtExcplCastArgs = MethodType.methodType(void.class,
                                mtExcplCastArgs.parameterArray());
                    }
                    if (mtExcplCastArgs.returnType() == void.class) {
                        mtExcplCastArgs = MethodType.methodType(mtTarget.returnType(),
                                mtExcplCastArgs.parameterArray());
                    }
                    data.put("mtExcplCastArgs", mtExcplCastArgs);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    MethodType mtExcplCastArgs = (MethodType) data.get("mtExcplCastArgs");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    return MethodHandles.explicitCastArguments(target, mtExcplCastArgs);
                }
            },
    FILTER_ARGUMENTS("filterArguments", Helper.MAX_ARITY / 2) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int filterArgsPos = Helper.RNG.nextInt(realArity + 1);
                    data.put("filterArgsPos", filterArgsPos);
                    int filtersArgsArrayLength = Helper.RNG.nextInt(realArity + 1 - filterArgsPos);
                    data.put("filtersArgsArrayLength", filtersArgsArrayLength);
                    MethodType mtFilter = TestMethods.randomMethodTypeGenerator(filtersArgsArrayLength);
                    data.put("mtFilter", mtFilter);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    MethodType mtFilter = (MethodType) data.get("mtFilter");
                    int filterArgsPos = (int) data.get("filterArgsPos");
                    int filtersArgsArrayLength = (int) data.get("filtersArgsArrayLength");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    MethodHandle[] filters = new MethodHandle[filtersArgsArrayLength];
                    for (int i = 0; i < filtersArgsArrayLength; i++) {
                        filters[i] = TestMethods.filterGenerator(mtFilter.parameterType(i),
                                mtTarget.parameterType(filterArgsPos + i), kind);
                    }
                    return MethodHandles.filterArguments(target, filterArgsPos, filters);
                }
            },
    FILTER_RETURN_VALUE("filterReturnValue") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int filterArgsPos = Helper.RNG.nextInt(realArity + 1);
                    int filtersArgsArrayLength = Helper.RNG.nextInt(realArity + 1 - filterArgsPos);
                    MethodType mtFilter = TestMethods.randomMethodTypeGenerator(filtersArgsArrayLength);
                    data.put("mtFilter", mtFilter);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    MethodType mtFilter = (MethodType) data.get("mtFilter");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    MethodHandle filter = TestMethods.filterGenerator(mtTarget.returnType(),
                            mtFilter.returnType(), kind);
                    return MethodHandles.filterReturnValue(target, filter);
                }
            },
    INSERT_ARGUMENTS("insertArguments", Helper.MAX_ARITY - 3) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int insertArgsPos = Helper.RNG.nextInt(realArity + 1);
                    data.put("insertArgsPos", insertArgsPos);
                    int insertArgsArrayLength = Helper.RNG.nextInt(realArity + 1 - insertArgsPos);
                    MethodType mtInsertArgs = MethodType.methodType(void.class, mtTarget.parameterList()
                            .subList(insertArgsPos, insertArgsPos + insertArgsArrayLength));
                    data.put("mtInsertArgs", mtInsertArgs);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    MethodType mtInsertArgs = (MethodType) data.get("mtInsertArgs");
                    int insertArgsPos = (int) data.get("insertArgsPos");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    Object[] insertList = Helper.randomArgs(mtInsertArgs.parameterList());
                    return MethodHandles.insertArguments(target, insertArgsPos, insertList);
                }
            },
    PERMUTE_ARGUMENTS("permuteArguments", Helper.MAX_ARITY / 2) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int[] permuteArgsReorderArray = new int[realArity];
                    int mtPermuteArgsNum = Helper.RNG.nextInt(Helper.MAX_ARITY);
                    mtPermuteArgsNum = mtPermuteArgsNum == 0 ? 1 : mtPermuteArgsNum;
                    MethodType mtPermuteArgs = TestMethods.randomMethodTypeGenerator(mtPermuteArgsNum);
                    mtTarget = mtTarget.changeReturnType(mtPermuteArgs.returnType());
                    for (int i = 0; i < realArity; i++) {
                        int mtPermuteArgsParNum = Helper.RNG.nextInt(mtPermuteArgs.parameterCount());
                        permuteArgsReorderArray[i] = mtPermuteArgsParNum;
                        mtTarget = mtTarget.changeParameterType(
                                i, mtPermuteArgs.parameterType(mtPermuteArgsParNum));
                    }
                    data.put("mtTarget", mtTarget);
                    data.put("permuteArgsReorderArray", permuteArgsReorderArray);
                    data.put("mtPermuteArgs", mtPermuteArgs);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    MethodType mtPermuteArgs = (MethodType) data.get("mtPermuteArgs");
                    int[] permuteArgsReorderArray = (int[]) data.get("permuteArgsReorderArray");
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), kind);
                    return MethodHandles.permuteArguments(target, mtPermuteArgs, permuteArgsReorderArray);
                }
            },
    THROW_EXCEPTION("throwException") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    Class<?> rType = mtTarget.returnType();
                    return MethodHandles.throwException(rType, Exception.class
                    );
                }
            },
    GUARD_WITH_TEST("guardWithTest") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int modifierMHArgNum = Helper.RNG.nextInt(realArity + 1);
                    data.put("modifierMHArgNum", modifierMHArgNum);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    int modifierMHArgNum = (int) data.get("modifierMHArgNum");
                    TestMethods.Kind targetKind;
                    TestMethods.Kind fallbackKind;
                    if (kind.equals(TestMethods.Kind.ONE)) {
                        targetKind = TestMethods.Kind.ONE;
                        fallbackKind = TestMethods.Kind.TWO;
                    } else {
                        targetKind = TestMethods.Kind.TWO;
                        fallbackKind = TestMethods.Kind.ONE;
                    }
                    MethodHandle target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), targetKind);
                    MethodHandle fallback = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                            mtTarget.parameterList(), fallbackKind);
                    MethodHandle test = TestMethods.methodHandleGenerator(boolean.class,
                            mtTarget.parameterList().subList(0, modifierMHArgNum), kind);
                    return MethodHandles.guardWithTest(test, target, fallback);
                }
            },
    CATCH_EXCEPTION("catchException") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int modifierMHArgNum = Helper.RNG.nextInt(realArity + 1);
                    data.put("modifierMHArgNum", modifierMHArgNum);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    int modifierMHArgNum = (int) data.get("modifierMHArgNum");
                    MethodHandle target;
                    if (kind.equals(TestMethods.Kind.ONE)) {
                        target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                                mtTarget.parameterList(), TestMethods.Kind.ONE);
                    } else {
                        target = TestMethods.methodHandleGenerator(mtTarget.returnType(),
                                mtTarget.parameterList(), TestMethods.Kind.EXCEPT);
                    }
                    List<Class<?>> handlerParamList = new ArrayList<>(mtTarget.parameterCount() + 1);
                    handlerParamList.add(Exception.class);
                    handlerParamList.addAll(mtTarget.parameterList().subList(0, modifierMHArgNum));
                    MethodHandle handler = TestMethods.methodHandleGenerator(
                            mtTarget.returnType(), handlerParamList, TestMethods.Kind.TWO);
                    return MethodHandles.catchException(target, Exception.class, handler);
                }
            },
    INVOKER("invoker", Helper.MAX_ARITY - 1) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    return MethodHandles.invoker(mtTarget);
                }
            },
    EXACT_INVOKER("exactInvoker", Helper.MAX_ARITY - 1) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    return MethodHandles.exactInvoker(mtTarget);
                }
            },
    SPREAD_INVOKER("spreadInvoker", Helper.MAX_ARITY - 1) {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    // Arity after reducing because of long and double take 2 slots.
                    int realArity = mtTarget.parameterCount();
                    int modifierMHArgNum = Helper.RNG.nextInt(realArity + 1);
                    data.put("modifierMHArgNum", modifierMHArgNum);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    int modifierMHArgNum = (int) data.get("modifierMHArgNum");
                    return MethodHandles.spreadInvoker(mtTarget, modifierMHArgNum);
                }
            },
    ARRAY_ELEMENT_GETTER("arrayElementGetter") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    Class<?> rType = mtTarget.returnType();
                    if (rType == void.class) {
                        rType = Object.class;
                    }
                    return MethodHandles.arrayElementGetter(Array.newInstance(rType, 2).getClass());
                }
            },
    ARRAY_ELEMENT_SETTER("arrayElementSetter") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    Class<?> rType = mtTarget.returnType();
                    if (rType == void.class) {
                        rType = Object.class;
                    }
                    return MethodHandles.arrayElementSetter(Array.newInstance(rType, 2).getClass());
                }
            },
    CONSTANT("constant") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    Class<?> rType = mtTarget.returnType();
                    if (rType == void.class) {
                        rType = Object.class;
                    }
                    if (rType.equals(boolean.class)) {
                        // There should be the same return values because for default values there are special "zero" forms
                        return MethodHandles.constant(rType, true);
                    } else {
                        return MethodHandles.constant(rType, kind.getValue(rType));
                    }
                }
            },
    IDENTITY("identity") {
                @Override
                public Map<String, Object> getTestCaseData() {
                    Map<String, Object> data = new HashMap<>();
                    int desiredArity = Helper.RNG.nextInt(super.maxArity);
                    MethodType mtTarget = TestMethods.randomMethodTypeGenerator(desiredArity);
                    data.put("mtTarget", mtTarget);
                    return data;
                }

                @Override
                protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) {
                    MethodType mtTarget = (MethodType) data.get("mtTarget");
                    Class<?> rType = mtTarget.returnType();
                    if (rType == void.class) {
                        rType = Object.class;
                    }
                    return MethodHandles.identity(rType);
                }
            };

    /**
     * Test method's name.
     */
    public final String name;

    private final int maxArity;

    private TestMethods(String name, int maxArity) {
        this.name = name;
        this.maxArity = maxArity;
    }

    private TestMethods(String name) {
        this(name, Helper.MAX_ARITY);
    }

    protected MethodHandle getMH(Map<String, Object> data, TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
        throw new UnsupportedOperationException("TESTBUG: getMH method is not implemented for test method " + this);
    }

    /**
     * Creates an adapter method handle depending on a test method from
     * MethodHandles class. Adapter is what is returned by the test method. This
     * method is able to create two kinds of adapters, their type will be the
     * same, but return values are different.
     *
     * @param data a Map containing data to create a method handle, can be
     * obtained by {@link #getTestCaseData} method
     * @param kind defines whether adapter ONE or adapter TWO will be
     * initialized. Should be equal to TestMethods.Kind.ONE or
     * TestMethods.Kind.TWO
     * @return Method handle adapter that behaves according to
     * TestMethods.Kind.ONE or TestMethods.Kind.TWO
     * @throws java.lang.NoSuchMethodException
     * @throws java.lang.IllegalAccessException
     */
    public MethodHandle getTestCaseMH(Map<String, Object> data, TestMethods.Kind kind)
            throws NoSuchMethodException, IllegalAccessException {
        if (data == null) {
            throw new Error(String.format("TESTBUG: Data for test method %s is not prepared",
                    this.name));
        }
        if (!kind.equals(TestMethods.Kind.ONE) && !kind.equals(TestMethods.Kind.TWO)) {
            throw new IllegalArgumentException("TESTBUG: Wrong \"kind\" (" + kind
                    + ") arg to getTestCaseMH function."
                    + " Should be Kind.ONE or Kind.TWO");
        }
        return getMH(data, kind);
    }

    /**
     * Returns a data Map needed for {@link #getTestCaseMH} method.
     *
     * @return data Map needed for {@link #getTestCaseMH} method
     */
    public Map<String, Object> getTestCaseData() {
        throw new UnsupportedOperationException(
                "TESTBUG: getTestCaseData method is not implemented for test method " + this);
    }

    /**
     * Enumeration used in methodHandleGenerator to define whether a MH returned
     * by this method returns "2" in different type representations, "4", or
     * throw an Exception.
     */
    public static enum Kind {

        ONE(2),
        TWO(4),
        EXCEPT(0);

        private final int value;

        private Object getValue(Class<?> cl) {
            return Helper.castToWrapper(value, cl);
        }

        private MethodHandle getBasicMH(Class<?> rType) throws NoSuchMethodException, IllegalAccessException {
            MethodHandle result = null;
            switch (this) {
                case ONE:
                case TWO:
                    if (rType.equals(void.class)) {
                        result = MethodHandles.lookup().findVirtual(Kind.class, "returnVoid", MethodType.methodType(void.class));
                        result = MethodHandles.insertArguments(result, 0, this);
                    } else {
                        result = MethodHandles.constant(rType, getValue(rType));
                    }
                    break;
                case EXCEPT:
                    result = MethodHandles.throwException(rType, Exception.class);
                    result = MethodHandles.insertArguments(result, 0, new Exception());
                    break;
            }
            return result;
        }

        private void returnVoid() {
        }

        private Kind(int value) {
            this.value = value;
        }
    }

    /**
     * Routine used to obtain a randomly generated method type.
     *
     * @param arity Arity of returned method type.
     * @return MethodType generated randomly.
     */
    private static MethodType randomMethodTypeGenerator(int arity) {
        return Helper.randomMethodTypeGenerator(arity);
    }

    /**
     * Routine used to obtain a method handles of a given type an kind (return
     * value).
     *
     * @param returnType Type of MH return value.
     * @param argTypes Types of MH args.
     * @param kind Defines whether the obtained MH returns "1" or "2".
     * @return Method handle of the given type.
     * @throws NoSuchMethodException
     * @throws IllegalAccessException
     */
    private static MethodHandle methodHandleGenerator(Class<?> returnType,
            List<Class<?>> argTypes, TestMethods.Kind kind)
            throws NoSuchMethodException, IllegalAccessException {
        MethodHandle result;
        result = kind.getBasicMH(returnType);
        return Helper.addTrailingArgs(result, argTypes.size(), argTypes);
    }

    /**
     * Routine that generates filter method handles to test
     * MethodHandles.filterArguments method.
     *
     * @param inputType Filter's argument type.
     * @param returnType Filter's return type.
     * @param kind Filter's return value definer.
     * @return A filter method handle, that takes one argument.
     * @throws NoSuchMethodException
     * @throws IllegalAccessException
     */
    private static MethodHandle filterGenerator(Class<?> inputType, Class<?> returnType,
            TestMethods.Kind kind) throws NoSuchMethodException, IllegalAccessException {
        MethodHandle tmpMH = kind.getBasicMH(returnType);
        if (inputType.equals(void.class)) {
            return tmpMH;
        }
        ArrayList<Class<?>> inputTypeList = new ArrayList<>(1);
        inputTypeList.add(inputType);
        return Helper.addTrailingArgs(tmpMH, 1, inputTypeList);
    }

    private static int argSlotsCount(MethodType mt) {
        int result = 0;
        for (Class cl : mt.parameterArray()) {
            if (cl.equals(long.class) || cl.equals(double.class)) {
                result += 2;
            } else {
                result++;
            }
        }
        return result;
    }

    private static List<Class<?>> reduceArgListToSlotsCount(List<Class<?>> list,
            int desiredSlotCount) {
        List<Class<?>> result = new ArrayList<>(desiredSlotCount);
        int count = 0;
        for (Class<?> cl : list) {
            if (count >= desiredSlotCount) {
                break;
            }
            if (cl.equals(long.class) || cl.equals(double.class)) {
                count += 2;
            } else {
                count++;
            }
            result.add(cl);
        }
        return result;
    }
}