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

org.robolectric.internal.bytecode.InvokeDynamicSupport Maven / Gradle / Ivy

There is a newer version: 4.14.1
Show newest version
package org.robolectric.internal.bytecode;

import static java.lang.invoke.MethodHandles.catchException;
import static java.lang.invoke.MethodHandles.constant;
import static java.lang.invoke.MethodHandles.dropArguments;
import static java.lang.invoke.MethodHandles.exactInvoker;
import static java.lang.invoke.MethodHandles.filterArguments;
import static java.lang.invoke.MethodHandles.foldArguments;
import static java.lang.invoke.MethodHandles.throwException;
import static java.lang.invoke.MethodType.methodType;
import static org.robolectric.internal.bytecode.MethodCallSite.Kind.REGULAR;
import static org.robolectric.internal.bytecode.MethodCallSite.Kind.STATIC;

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;
import java.lang.invoke.SwitchPoint;
import java.lang.invoke.WrongMethodTypeException;
import org.robolectric.util.PerfStatsCollector;
import org.robolectric.util.ReflectionHelpers;

@SuppressWarnings("RethrowReflectiveOperationExceptionAsLinkageError")
public class InvokeDynamicSupport {
  @SuppressWarnings("unused")
  private static Interceptors INTERCEPTORS;

  private static final MethodHandle BIND_CALL_SITE;
  private static final MethodHandle BIND_INIT_CALL_SITE;
  private static final MethodHandle EXCEPTION_HANDLER;
  private static final MethodHandle GET_SHADOW;

  /**
   * Represents the boolean 'true' as an integer. Due to a JVM bug, invokedynamic bootstrap methods
   * currently do not support extra primitive boolean parameters. Integers are required to convey
   * booleans.
   *
   * 

See https://bugs.java.com/bugdatabase/view_bug?bug_id=JDK-8322510 */ private static final int BOOLEAN_TRUE = 1; static { try { MethodHandles.Lookup lookup = MethodHandles.lookup(); BIND_CALL_SITE = lookup.findStatic( InvokeDynamicSupport.class, "bindCallSite", methodType(MethodHandle.class, MethodCallSite.class)); BIND_INIT_CALL_SITE = lookup.findStatic( InvokeDynamicSupport.class, "bindInitCallSite", methodType(MethodHandle.class, RoboCallSite.class)); MethodHandle cleanStackTrace = lookup.findStatic( RobolectricInternals.class, "cleanStackTrace", methodType(Throwable.class, Throwable.class)); EXCEPTION_HANDLER = filterArguments(throwException(void.class, Throwable.class), 0, cleanStackTrace); GET_SHADOW = lookup.findVirtual(ShadowedObject.class, "$$robo$getData", methodType(Object.class)); } catch (NoSuchMethodException | IllegalAccessException e) { throw new AssertionError(e); } } @SuppressWarnings("UnusedDeclaration") public static CallSite bootstrapInit(MethodHandles.Lookup caller, String name, MethodType type) { return PerfStatsCollector.getInstance() .measure( "invokedynamic bootstrap init", () -> { RoboCallSite site = new RoboCallSite(type, caller.lookupClass()); bindInitCallSite(site); return site; }); } @SuppressWarnings("UnusedDeclaration") public static CallSite bootstrap( MethodHandles.Lookup caller, String name, MethodType type, MethodHandle original, int isNative /* 1 == originally native, 0 == not originally native */) throws IllegalAccessException { return PerfStatsCollector.getInstance() .measure( "invokedynamic bootstrap", () -> { MethodCallSite site = new MethodCallSite( caller.lookupClass(), type, name, original, REGULAR, isNative == BOOLEAN_TRUE); bindCallSite(site); return site; }); } @SuppressWarnings("UnusedDeclaration") public static CallSite bootstrapStatic( MethodHandles.Lookup caller, String name, MethodType type, MethodHandle original, int isNative /* 1 == originally native, 0 == not originally native */) throws IllegalAccessException { return PerfStatsCollector.getInstance() .measure( "invokedynamic bootstrap static", () -> { MethodCallSite site = new MethodCallSite( caller.lookupClass(), type, name, original, STATIC, isNative == BOOLEAN_TRUE); bindCallSite(site); return site; }); } @SuppressWarnings("UnusedDeclaration") public static CallSite bootstrapIntrinsic( MethodHandles.Lookup caller, String name, MethodType type, String callee) throws IllegalAccessException { return PerfStatsCollector.getInstance() .measure( "invokedynamic bootstrap intrinsic", () -> { MethodHandle mh = getMethodHandle(callee, name, type); if (mh == null) { throw new IllegalArgumentException( "Could not find intrinsic for " + callee + ":" + name); } return new ConstantCallSite(mh.asType(type)); }); } private static final MethodHandle NOTHING = constant(Void.class, null).asType(methodType(void.class)); private static MethodHandle getMethodHandle( String className, String methodName, MethodType type) { Interceptor interceptor = INTERCEPTORS.findInterceptor(className, methodName); if (interceptor != null) { try { // reload interceptor in sandbox... Class theClass = (Class) ReflectionHelpers.loadClass( RobolectricInternals.getClassLoader(), interceptor.getClass().getName()) .asSubclass(Interceptor.class); return ReflectionHelpers.newInstance(theClass).getMethodHandle(methodName, type); } catch (NoSuchMethodException | IllegalAccessException e) { throw new RuntimeException(e); } } if (type.parameterCount() != 0) { return dropArguments(NOTHING, 0, type.parameterArray()); } else { return NOTHING; } } private static MethodHandle bindInitCallSite(RoboCallSite site) { MethodHandle mh = RobolectricInternals.getShadowCreator(site.getTheClass()); return bindWithFallback(site, mh, BIND_INIT_CALL_SITE); } private static MethodHandle bindCallSite(MethodCallSite site) throws IllegalAccessException { MethodHandle mh = RobolectricInternals.findShadowMethodHandle( site.getTheClass(), site.getName(), site.type(), site.isStatic(), site.isNative()); if (mh == null) { // call original code mh = site.getOriginal(); } else if (mh == ShadowWrangler.DO_NOTHING) { // no-op mh = dropArguments(mh, 0, site.type().parameterList()); } else if (!site.isStatic()) { // drop arg 0 (this) for static methods Class mhType = mh.type().parameterType(0); // At this point, thisType is either a shadow type, or in the case of native method // invocations, it can be equivalent to the original type. if (!mhType.equals(site.getTheClass())) { // Only invoke getShadow if the method is on class that is decoupled from the original. mh = filterArguments(mh, 0, GET_SHADOW.asType(methodType(mhType, site.thisType()))); } } try { return bindWithFallback(site, cleanStackTraces(mh), BIND_CALL_SITE); } catch (Throwable t) { // The error that bubbles up is currently not very helpful so we print any error messages // here t.printStackTrace(); System.err.println(site.getTheClass()); throw t; } } private static MethodHandle bindWithFallback( RoboCallSite site, MethodHandle mh, MethodHandle fallback) { SwitchPoint switchPoint = getInvalidator(site.getTheClass()); MethodType type = site.type(); MethodHandle boundFallback = foldArguments(exactInvoker(type), fallback.bindTo(site)); try { mh = switchPoint.guardWithTest(mh.asType(type), boundFallback); } catch (WrongMethodTypeException e) { if (site instanceof MethodCallSite) { MethodCallSite methodCallSite = (MethodCallSite) site; throw new RuntimeException( "failed to bind " + methodCallSite.thisType() + "." + methodCallSite.getName(), e); } else { throw e; } } site.setTarget(mh); return mh; } private static SwitchPoint getInvalidator(Class cl) { return RobolectricInternals.getShadowInvalidator().getSwitchPoint(cl); } private static MethodHandle cleanStackTraces(MethodHandle mh) { MethodType type = EXCEPTION_HANDLER.type().changeReturnType(mh.type().returnType()); return catchException(mh, Throwable.class, EXCEPTION_HANDLER.asType(type)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy