All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.headius.invokebinder.transform.TryFinally Maven / Gradle / Ivy

/*
 * Copyright 2012-2014 headius.
 *
 * 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.
 */
package com.headius.invokebinder.transform;

import com.headius.invokebinder.Binder;
import com.headius.invokebinder.Util;

import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.lang.reflect.InvocationTargetException;

/**
 * An try-finally transform.
 *
 * Equivalent call: A combination of folds and catches.
 *
 * MethodHandle exceptionHandler = [drop exception and invoke post logic]
 * target = MethodHandles.catchException(target, Throwable.class, exceptionHandler)
 * target = MethodHandles.foldArguments(post, target)
 */
public class TryFinally extends Transform {

    private final MethodHandle post;

    private static final MethodHandle tryFinallyJava9;

    public TryFinally(MethodHandle post) {
        this.post = post;
    }

    public MethodHandle up(MethodHandle target) {
        if (Util.isJava9()) return nativeTryFinally(target, post);

        MethodHandle exceptionHandler = Binder
                .from(target.type().insertParameterTypes(0, Throwable.class).changeReturnType(void.class))
                .drop(0)
                .invoke(post);

        MethodHandle rethrow = Binder
                .from(target.type().insertParameterTypes(0, Throwable.class))
                .fold(exceptionHandler)
                .drop(1, target.type().parameterCount())
                .throwException();

        target = MethodHandles.catchException(target, Throwable.class, rethrow);

        // if target returns a value, we must return it regardless of post
        MethodHandle realPost = post;
        if (target.type().returnType() != void.class) {
            // modify post to ignore return value
            MethodHandle newPost = Binder
                    .from(target.type().insertParameterTypes(0, target.type().returnType()).changeReturnType(void.class))
                    .drop(0)
                    .invoke(post);

            // fold post into an identity chain that only returns the value
            realPost = Binder
                    .from(target.type().insertParameterTypes(0, target.type().returnType()))
                    .fold(newPost)
                    .drop(1, target.type().parameterCount())
                    .identity();
        }

        return MethodHandles.foldArguments(realPost, target);
    }

    private MethodHandle nativeTryFinally(MethodHandle target, MethodHandle post) {
        MethodType targetType = target.type();
        boolean voidReturn = targetType.returnType() == Void.TYPE;
        MethodType finallyType = targetType.insertParameterTypes(0, Throwable.class);
        int dropCount = 1;

        if (!voidReturn) {
            finallyType = finallyType.insertParameterTypes(1, targetType.returnType());
            dropCount = 2;
        }

        MethodHandle wrapPost = Binder
                .from(finallyType)
                .drop(0, dropCount)
                .invoke(post);

        if (!voidReturn) {
            wrapPost = Binder.from(finallyType)
                    .foldVoid(wrapPost)
                    .permute(1)
                    .identity();
        }

        try {
            return (MethodHandle) tryFinallyJava9.invokeExact(target, wrapPost);
        } catch (Throwable t) {
            throw new RuntimeException("Java 9 detected but MethodHandles.tryFinally missing", t);
        }
    }

    public MethodType down(MethodType type) {
        return type;
    }

    public String toString() {
        return "try/finally with " + post;
    }

    static {
        if (Util.isJava9()) {
            tryFinallyJava9 = Binder.from(MethodHandle.class, MethodHandle.class, MethodHandle.class).invokeStaticQuiet(MethodHandles.lookup(), MethodHandles.class, "tryFinally");
        } else {
            tryFinallyJava9 = null;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy