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

org.robolectric.interceptors.AndroidInterceptors Maven / Gradle / Ivy

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

import static java.lang.Math.min;
import static java.lang.invoke.MethodHandles.constant;
import static java.lang.invoke.MethodHandles.dropArguments;
import static java.lang.invoke.MethodType.methodType;
import static java.util.Arrays.asList;
import static org.robolectric.util.ReflectionHelpers.callStaticMethod;

import com.google.common.base.Throwables;
import java.io.FileDescriptor;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.SoftReference;
import java.lang.ref.WeakReference;
import java.lang.reflect.Field;
import java.net.Socket;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
import javax.annotation.Nullable;
import org.robolectric.fakes.CleanerCompat;
import org.robolectric.internal.bytecode.Interceptor;
import org.robolectric.internal.bytecode.MethodRef;
import org.robolectric.internal.bytecode.MethodSignature;
import org.robolectric.util.Function;
import org.robolectric.util.Util;

public class AndroidInterceptors {
  private static final MethodHandles.Lookup lookup = MethodHandles.lookup();

  public static Collection all() {
    List interceptors =
        new ArrayList<>(
            asList(
                new LinkedHashMapEldestInterceptor(),
                new SystemTimeInterceptor(),
                new SystemArrayCopyInterceptor(),
                new LocaleAdjustLanguageCodeInterceptor(),
                new SystemLogInterceptor(),
                new FileDescriptorInterceptor(),
                new NoOpInterceptor(),
                new SocketInterceptor(),
                new ReferenceRefersToInterceptor(),
                new NioUtilsFreeDirectBufferInterceptor()));

    if (Util.getJavaVersion() >= 9) {
      interceptors.add(new CleanerInterceptor());
    }

    return interceptors;
  }

  /**
   * Intercepts calls to libcore-extensions to {@link FileDescriptor}.
   *
   * 

libcore implements extensions to {@link FileDescriptor} that support ownership tracking of * unix FDs, which are not part of the Java API. This intercepts calls to these and maps them to * the OpenJDK API. */ public static class FileDescriptorInterceptor extends Interceptor { public FileDescriptorInterceptor() { super( new MethodRef(FileDescriptor.class, "release$"), new MethodRef(FileDescriptor.class, "getInt$"), new MethodRef(FileDescriptor.class, "setInt$")); } private static void moveField( FileDescriptor out, FileDescriptor in, String fieldName, Object movedOutValue) throws ReflectiveOperationException { Field fieldAccessor = FileDescriptor.class.getDeclaredField(fieldName); fieldAccessor.setAccessible(true); fieldAccessor.set(out, fieldAccessor.get(in)); fieldAccessor.set(in, movedOutValue); } static Object setInt(FileDescriptor input, int value) { try { input.getClass().getDeclaredField("fd").setInt(input, value); } catch (Exception e) { // Ignore } return null; } static int getInt(FileDescriptor input) { try { Field fieldAccessor = input.getClass().getDeclaredField("fd"); fieldAccessor.setAccessible(true); return fieldAccessor.getInt(input); } catch (IllegalAccessException e) { // Should not happen since we set this to be accessible return -1; } catch (NoSuchFieldException e) { return -2; } } static FileDescriptor release(FileDescriptor input) { synchronized (input) { try { FileDescriptor ret = new FileDescriptor(); moveField(ret, input, "fd", /*movedOutValue=*/ -1); // "closed" is irrelevant if the fd is already -1. moveField(ret, input, "closed", /*movedOutValue=*/ false); // N.B.: FileDescriptor.attach() is not implemented in libcore (yet), so these won't be // used. moveField(ret, input, "parent", /*movedOutValue=*/ null); moveField(ret, input, "otherParents", /*movedOutValue=*/ null); // These only exist on Windows. try { moveField(ret, input, "handle", /*movedOutValue=*/ -1); moveField(ret, input, "append", /*movedOutValue=*/ false); } catch (ReflectiveOperationException ex) { // Ignore. } return ret; } catch (ReflectiveOperationException ex) { throw new RuntimeException(ex); } } } @Override public Function handle(MethodSignature methodSignature) { return new Function() { @Override public Object call(Class theClass, Object value, Object[] params) { if ("release$".equals(methodSignature.methodName)) { return release((FileDescriptor) value); } else if ("getInt$".equals(methodSignature.methodName)) { return getInt((FileDescriptor) value); } else { return setInt((FileDescriptor) value, (int) params[0]); } } }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { if ("release$".equals(methodName)) { return lookup.findStatic( getClass(), "release", methodType(FileDescriptor.class, FileDescriptor.class)); } else if ("getInt$".equals(methodName)) { return lookup.findStatic(getClass(), "getInt", methodType(int.class, FileDescriptor.class)); } else { return lookup.findStatic( getClass(), "setInt", methodType(Object.class, FileDescriptor.class, int.class)); } } } // @Intercept(value = LinkedHashMap.class, method = "eldest") public static class LinkedHashMapEldestInterceptor extends Interceptor { public LinkedHashMapEldestInterceptor() { super(new MethodRef(LinkedHashMap.class, "eldest")); } @Nullable static Object eldest(LinkedHashMap map) { return map.isEmpty() ? null : map.entrySet().iterator().next(); } @Override public Function handle(MethodSignature methodSignature) { return new Function() { @Override public Object call(Class theClass, Object value, Object[] params) { return eldest((LinkedHashMap) value); } }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic(getClass(), "eldest", methodType(Object.class, LinkedHashMap.class)); } } public static class SystemTimeInterceptor extends Interceptor { public SystemTimeInterceptor() { super( new MethodRef(System.class, "nanoTime"), new MethodRef(System.class, "currentTimeMillis")); } @Override public Function handle(final MethodSignature methodSignature) { return new Function() { @Override public Object call(Class theClass, Object value, Object[] params) { ClassLoader cl = theClass.getClassLoader(); try { Class shadowSystemClass = cl.loadClass("org.robolectric.shadows.ShadowSystem"); return callStaticMethod(shadowSystemClass, methodSignature.methodName); } catch (ClassNotFoundException e) { throw new RuntimeException(e); } } }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { Class shadowSystemClass; try { shadowSystemClass = getClass().getClassLoader().loadClass("org.robolectric.shadows.ShadowSystem"); } catch (ClassNotFoundException e) { throw new RuntimeException(e); } switch (methodName) { case "nanoTime": return lookup.findStatic(shadowSystemClass, "nanoTime", methodType(long.class)); case "currentTimeMillis": return lookup.findStatic(shadowSystemClass, "currentTimeMillis", methodType(long.class)); } throw new UnsupportedOperationException(); } } public static class SystemArrayCopyInterceptor extends Interceptor { public SystemArrayCopyInterceptor() { super(new MethodRef(System.class, "arraycopy")); } @Override public Function handle(MethodSignature methodSignature) { return new Function() { @Override public Object call(Class theClass, Object value, Object[] params) { //noinspection SuspiciousSystemArraycopy System.arraycopy( params[0], (Integer) params[1], params[2], (Integer) params[3], (Integer) params[4]); return null; } }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic( System.class, "arraycopy", methodType(void.class, Object.class, int.class, Object.class, int.class, int.class)); } } public static class LocaleAdjustLanguageCodeInterceptor extends Interceptor { public LocaleAdjustLanguageCodeInterceptor() { super(new MethodRef(Locale.class, "adjustLanguageCode")); } static String adjustLanguageCode(String languageCode) { String adjusted = languageCode.toLowerCase(Locale.US); // Map new language codes to the obsolete language // codes so the correct resource bundles will be used. if (languageCode.equals("he")) { adjusted = "iw"; } else if (languageCode.equals("id")) { adjusted = "in"; } else if (languageCode.equals("yi")) { adjusted = "ji"; } return adjusted; } @Override public Function handle(MethodSignature methodSignature) { return new Function() { @Override public Object call(Class theClass, Object value, Object[] params) { return adjustLanguageCode((String) params[0]); } }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic( getClass(), "adjustLanguageCode", methodType(String.class, String.class)); } } /** AndroidInterceptor for System.logE and System.logW. */ public static class SystemLogInterceptor extends Interceptor { public SystemLogInterceptor() { super( new MethodRef(System.class.getName(), "logE"), new MethodRef(System.class.getName(), "logW")); } static void logE(Object... params) { log("System.logE: ", params); } static void logW(Object... params) { log("System.logW: ", params); } /** * Write logs to {@code System#err}. * * @param prefix A non-null prefix to write. * @param params Optional parameters. Expected parameters are [String message] or [String * message, Throwable tr] in which case the throwable's stack trace is written to the * output. */ static void log(String prefix, Object... params) { StringBuilder message = new StringBuilder(prefix); for (int i = 0; i < min(2, params.length); i++) { Object param = params[i]; if (i > 0 && param instanceof Throwable) { message.append('\n').append(Throwables.getStackTraceAsString((Throwable) param)); } else { message.append(param); } } System.err.println(message); } @Override public Function handle(MethodSignature methodSignature) { return (theClass, value, params) -> { if ("logE".equals(methodSignature.methodName)) { logE(params); } else if ("logW".equals(methodSignature.methodName)) { logW(params); } return null; }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic(getClass(), methodName, methodType(void.class, Object[].class)); } } /** * Maps calls to Cleaner, which moved between Java 8 and 9: * *

    *
  • {@code sun.misc.Cleaner.create()} -> {@code new java.lang.ref.Cleaner().register()} *
  • {@code sun.misc.Cleaner.clean()} -> {@code java.lang.ref.Cleaner.Cleanable().clean()} *
*/ public static class CleanerInterceptor extends Interceptor { public CleanerInterceptor() { super( new MethodRef("sun.misc.Cleaner", "create"), new MethodRef("sun.misc.Cleaner", "clean")); } static Object create(Object obj, Runnable action) { return CleanerCompat.register(obj, action); } static void clean(Object cleanable) { CleanerCompat.clean(cleanable); } @Override public Function handle(MethodSignature methodSignature) { switch (methodSignature.methodName) { case "create": return (theClass, value, params) -> create(params[0], (Runnable) params[1]); case "clean": return (theClass, value, params) -> { clean(value); return null; }; default: throw new IllegalStateException(); } } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { switch (methodName) { case "create": return lookup.findStatic( getClass(), "create", methodType(Object.class, Object.class, Runnable.class)); case "clean": return lookup.findStatic(getClass(), "clean", methodType(void.class, Object.class)); default: throw new IllegalStateException(); } } } public static class NoOpInterceptor extends Interceptor { public NoOpInterceptor() { super( new MethodRef("java.lang.System", "loadLibrary"), new MethodRef("android.os.StrictMode", "trackActivity"), new MethodRef("android.os.StrictMode", "incrementExpectedActivityCount"), new MethodRef("android.util.LocaleUtil", "getLayoutDirectionFromLocale"), new MethodRef("android.view.FallbackEventHandler", "*")); } @Override public Function handle(MethodSignature methodSignature) { return returnDefaultValue(methodSignature); } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { MethodHandle nothing = constant(Void.class, null).asType(methodType(void.class)); if (type.parameterCount() != 0) { return dropArguments(nothing, 0, type.parameterArray()); } else { return nothing; } } } /** Intercepts calls to methods in {@link Socket} not present in the OpenJDK. */ public static class SocketInterceptor extends Interceptor { public SocketInterceptor() { super(new MethodRef(Socket.class, "getFileDescriptor$")); } @Nullable static FileDescriptor getFileDescriptor(Socket socket) { return null; } @Override public Function handle(MethodSignature methodSignature) { return new Function() { @Override public Object call(Class theClass, Object value, Object[] params) { return getFileDescriptor((Socket) value); } }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic( getClass(), "getFileDescriptor", methodType(FileDescriptor.class, Socket.class)); } } /** AndroidInterceptor for Reference.refersTo which is not available until JDK 16. */ public static class ReferenceRefersToInterceptor extends Interceptor { private static final String METHOD = "refersTo"; public ReferenceRefersToInterceptor() { super( new MethodRef(WeakReference.class.getName(), METHOD), new MethodRef(SoftReference.class.getName(), METHOD), new MethodRef(PhantomReference.class.getName(), METHOD)); } static boolean refersTo(Reference ref, Object obj) { return ref.get() == obj; } @Override public Function handle(MethodSignature methodSignature) { return (theClass, value, params) -> refersTo((Reference) value, params[0]); } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic( getClass(), METHOD, methodType(boolean.class, Reference.class, Object.class)); } } /** * AndroidInterceptor for NioUtils.freeDirectBuffer. * *

This method is invoked by ImageReader.java. The class is not part of the JDK. */ public static class NioUtilsFreeDirectBufferInterceptor extends Interceptor { private static final String METHOD = "freeDirectBuffer"; public NioUtilsFreeDirectBufferInterceptor() { super(new MethodRef("java.nio.NioUtils", METHOD)); } static void freeDirectBuffer(ByteBuffer buffer) { // Following the layoutlib/java/NioUtils_Delegate.java implementation, this is a no-op: "it // does not seem we have to do anything in here as we are only referencing the existing native // buffer and do not perform any allocation on creation." // // Note: for the interceptor to work, this method _must_ be present even though it doesn't // do anything. Otherwise the bytecode can't reference it at runtime. } @Override public Function handle(MethodSignature methodSignature) { return (theClass, value, params) -> { freeDirectBuffer((ByteBuffer) value); return null; }; } @Override public MethodHandle getMethodHandle(String methodName, MethodType type) throws NoSuchMethodException, IllegalAccessException { return lookup.findStatic(getClass(), METHOD, methodType(void.class, ByteBuffer.class)); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy