Java程序  |  86行  |  2.65 KB

/*
 * Copyright (C) 2016 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 java.lang.reflect.Method;
import java.lang.reflect.InvocationTargetException;

import dalvik.system.PathClassLoader;

// ClassLoader not delegating for non java. packages.
class DelegateLastPathClassLoader extends PathClassLoader {

  public DelegateLastPathClassLoader(String dexPath, ClassLoader parent) {
    super(dexPath, parent);
  }

  @Override
  protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
    if (!name.startsWith("java.")) {
      try {
        return findClass(name);
      } catch (ClassNotFoundException ignore) {
        // Ignore and fall through to parent class loader.
      }
    }
    return super.loadClass(name, resolve);
  }
}

public class Main {

  public static void main(String[] args) throws Exception {
    System.loadLibrary(args[0]);
    final String DEX_FILE = System.getenv("DEX_LOCATION") + "/613-inlining-dex-cache-ex.jar";
    ClassLoader loader = new DelegateLastPathClassLoader(DEX_FILE, Main.class.getClassLoader());
    Class cls = loader.loadClass("LoadedByAppClassLoader");
    Method m = cls.getDeclaredMethod("letMeInlineYou");
    // Invoke the method enough times to get JITted.
    for (int i = 0; i < 10000; ++i) {
      m.invoke(null);
    }
    ensureJitCompiled(cls, "letMeInlineYou");
    ClassLoader bLoader = areYouB();
    if (bLoader != Main.class.getClassLoader()) {
      throw new Error("Wrong class loader");
    }
  }

  public static void foo(Main o) {
    // LoadedByAppClassLoader.letMeInlineYou will try to inline this
    // method but used to pass the wrong class loader. As a result,
    // the lookup of B.foo was updating the dex cache with the other
    // class loader's B class.
    if (o != null) {
      o.myField.foo();
    }
  }

  public B myField;

  public static ClassLoader areYouB() {
    return OtherClass.getB().getClassLoader();
  }

  public static native void ensureJitCompiled(Class cls, String method_name);
}

class OtherClass {
  public static Class getB() {
    // This used to return the B class of another class loader.
    return B.class;
  }
}