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