1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2015 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker 17*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Constructor; 18*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Field; 19*795d594fSAndroid Build Coastguard Worker 20*795d594fSAndroid Build Coastguard Worker /** 21*795d594fSAndroid Build Coastguard Worker * PathClassLoader test. 22*795d594fSAndroid Build Coastguard Worker */ 23*795d594fSAndroid Build Coastguard Worker public class Main { 24*795d594fSAndroid Build Coastguard Worker createClassLoader(String dexPath, ClassLoader parent)25*795d594fSAndroid Build Coastguard Worker private static ClassLoader createClassLoader(String dexPath, ClassLoader parent) { 26*795d594fSAndroid Build Coastguard Worker try { 27*795d594fSAndroid Build Coastguard Worker Class<?> myClassLoaderClass = Class.forName("MyPathClassLoader"); 28*795d594fSAndroid Build Coastguard Worker Constructor<?> constructor = myClassLoaderClass.getConstructor(String.class, 29*795d594fSAndroid Build Coastguard Worker ClassLoader.class); 30*795d594fSAndroid Build Coastguard Worker return (ClassLoader)constructor.newInstance(dexPath, parent); 31*795d594fSAndroid Build Coastguard Worker } catch (Exception e) { 32*795d594fSAndroid Build Coastguard Worker // Ups, not available?!?! 33*795d594fSAndroid Build Coastguard Worker throw new RuntimeException(e); 34*795d594fSAndroid Build Coastguard Worker } 35*795d594fSAndroid Build Coastguard Worker } 36*795d594fSAndroid Build Coastguard Worker 37*795d594fSAndroid Build Coastguard Worker /** 38*795d594fSAndroid Build Coastguard Worker * Main entry point. 39*795d594fSAndroid Build Coastguard Worker */ main(String[] args)40*795d594fSAndroid Build Coastguard Worker public static void main(String[] args) throws Exception { 41*795d594fSAndroid Build Coastguard Worker // Check the class-path for the second file. We'll use that one as the source of the 42*795d594fSAndroid Build Coastguard Worker // new classloader. 43*795d594fSAndroid Build Coastguard Worker String cp = System.getProperty("java.class.path"); 44*795d594fSAndroid Build Coastguard Worker if (cp.split(System.getProperty("path.separator")).length != 1) { 45*795d594fSAndroid Build Coastguard Worker throw new IllegalStateException("Didn't find exactly one classpath element in " + cp); 46*795d594fSAndroid Build Coastguard Worker } 47*795d594fSAndroid Build Coastguard Worker if (!cp.endsWith("classloader2.jar")) { 48*795d594fSAndroid Build Coastguard Worker throw new IllegalStateException("Don't understand classpath " + cp); 49*795d594fSAndroid Build Coastguard Worker } 50*795d594fSAndroid Build Coastguard Worker cp = cp.replace("classloader2.jar", "classloader2-ex.jar"); 51*795d594fSAndroid Build Coastguard Worker 52*795d594fSAndroid Build Coastguard Worker ClassLoader myClassLoader = createClassLoader( 53*795d594fSAndroid Build Coastguard Worker cp, ClassLoader.getSystemClassLoader().getParent()); 54*795d594fSAndroid Build Coastguard Worker 55*795d594fSAndroid Build Coastguard Worker // Now load our test class. 56*795d594fSAndroid Build Coastguard Worker Class<?> srcClass = A.class; 57*795d594fSAndroid Build Coastguard Worker Class<?> exClass = myClassLoader.loadClass("A"); 58*795d594fSAndroid Build Coastguard Worker 59*795d594fSAndroid Build Coastguard Worker // First check: classes should be different. 60*795d594fSAndroid Build Coastguard Worker if (srcClass == exClass) { 61*795d594fSAndroid Build Coastguard Worker throw new IllegalStateException("Loaded class instances are the same"); 62*795d594fSAndroid Build Coastguard Worker } 63*795d594fSAndroid Build Coastguard Worker 64*795d594fSAndroid Build Coastguard Worker // Secondary checks: get the static field values and make sure they aren't the same. 65*795d594fSAndroid Build Coastguard Worker String srcValue = (String)srcClass.getDeclaredField("value").get(null); 66*795d594fSAndroid Build Coastguard Worker if (!"Src-A".equals(srcValue)) { 67*795d594fSAndroid Build Coastguard Worker throw new IllegalStateException("Expected Src-A, found " + srcValue); 68*795d594fSAndroid Build Coastguard Worker } 69*795d594fSAndroid Build Coastguard Worker String exValue = (String)exClass.getDeclaredField("value").get(null); 70*795d594fSAndroid Build Coastguard Worker if (!"Ex-A".equals(exValue)) { 71*795d594fSAndroid Build Coastguard Worker throw new IllegalStateException("Expected Ex-A, found " + exValue); 72*795d594fSAndroid Build Coastguard Worker } 73*795d594fSAndroid Build Coastguard Worker 74*795d594fSAndroid Build Coastguard Worker // Try to load a dex file with bad dex code. Use new instance to force verification. 75*795d594fSAndroid Build Coastguard Worker VerifyError existing = null; 76*795d594fSAndroid Build Coastguard Worker try { 77*795d594fSAndroid Build Coastguard Worker Class<?> badClass = Main.class.getClassLoader().loadClass("B"); 78*795d594fSAndroid Build Coastguard Worker System.out.println("Loaded class B."); 79*795d594fSAndroid Build Coastguard Worker badClass.newInstance(); 80*795d594fSAndroid Build Coastguard Worker System.out.println("Should not be able to instantiate B with bad dex bytecode."); 81*795d594fSAndroid Build Coastguard Worker } catch (VerifyError e) { 82*795d594fSAndroid Build Coastguard Worker System.out.println("Caught VerifyError."); 83*795d594fSAndroid Build Coastguard Worker existing = e; 84*795d594fSAndroid Build Coastguard Worker } 85*795d594fSAndroid Build Coastguard Worker 86*795d594fSAndroid Build Coastguard Worker // Make sure the same error is rethrown when reloading the bad class. 87*795d594fSAndroid Build Coastguard Worker try { 88*795d594fSAndroid Build Coastguard Worker Class<?> badClass = Main.class.getClassLoader().loadClass("B"); 89*795d594fSAndroid Build Coastguard Worker System.out.println("Loaded class B."); 90*795d594fSAndroid Build Coastguard Worker badClass.newInstance(); 91*795d594fSAndroid Build Coastguard Worker System.out.println("Should not be able to instantiate B with bad dex bytecode."); 92*795d594fSAndroid Build Coastguard Worker } catch (VerifyError e) { 93*795d594fSAndroid Build Coastguard Worker if (e == existing) { 94*795d594fSAndroid Build Coastguard Worker System.out.println("Caught existing VerifyError."); 95*795d594fSAndroid Build Coastguard Worker } else { 96*795d594fSAndroid Build Coastguard Worker e.printStackTrace(System.out); 97*795d594fSAndroid Build Coastguard Worker } 98*795d594fSAndroid Build Coastguard Worker } 99*795d594fSAndroid Build Coastguard Worker 100*795d594fSAndroid Build Coastguard Worker System.out.println("Everything OK."); 101*795d594fSAndroid Build Coastguard Worker } 102*795d594fSAndroid Build Coastguard Worker } 103