/* * 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; } }