/* * Copyright (C) 2008 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.io.File; import java.io.IOException; import java.lang.reflect.Constructor; /** * DexFile tests (Dalvik-specific). */ public class Main { private static final String CLASS_PATH = System.getenv("DEX_LOCATION") + "/071-dexfile-ex.jar"; private static final String ODEX_DIR = System.getenv("DEX_LOCATION"); //private static final String ODEX_DIR = "."; private static final String ODEX_ALT = "/tmp"; private static final String LIB_DIR = "/nowhere/nothing/"; /** * Prep the environment then run the test. */ public static void main(String[] args) { Process p; try { /* * Create a sub-process to see if the ProcessManager wait * interferes with the dexopt invocation wait. * * /dev/random never hits EOF, so we're sure that we'll still * be waiting for the process to complete. On the device it * stops pretty quickly (which means the child won't be * spinning). */ ProcessBuilder pb = new ProcessBuilder("cat", "/dev/random"); p = pb.start(); } catch (IOException ioe) { System.err.println("cmd failed: " + ioe.getMessage()); p = null; } try { testDexClassLoader(); } finally { // shouldn't be necessary, but it's good to be tidy if (p != null) { p.destroy(); } // let the ProcessManager's daemon thread finish before we shut down // (avoids the occasional segmentation fault) try { Thread.sleep(500); } catch (Exception ex) {} } System.out.println("done"); } /** * Create a class loader, explicitly specifying the source DEX and * the location for the optimized DEX. */ private static void testDexClassLoader() { ClassLoader dexClassLoader = getDexClassLoader(); Class anotherClass; try { anotherClass = dexClassLoader.loadClass("Another"); } catch (ClassNotFoundException cnfe) { throw new RuntimeException("Another?", cnfe); } Object another; try { another = anotherClass.newInstance(); } catch (IllegalAccessException ie) { throw new RuntimeException("new another", ie); } catch (InstantiationException ie) { throw new RuntimeException("new another", ie); } // not expected to work; just exercises the call dexClassLoader.getResource("nonexistent"); } /* * Create an instance of DexClassLoader. The test harness doesn't * have visibility into dalvik.system.*, so we do this through * reflection. */ private static ClassLoader getDexClassLoader() { String odexDir; if (false) { String androidData = System.getenv("ANDROID_DATA"); if (androidData == null) { androidData = ""; } odexDir = androidData + "/" + ODEX_DIR; } File test = new File(ODEX_DIR); if (test.isDirectory()) { odexDir = ODEX_DIR; } else { odexDir = ODEX_ALT; } if (false) { System.out.println("Output dir is " + odexDir); } ClassLoader myLoader = Main.class.getClassLoader(); Class dclClass; try { dclClass = myLoader.loadClass("dalvik.system.DexClassLoader"); } catch (ClassNotFoundException cnfe) { throw new RuntimeException("dalvik.system.DexClassLoader not found", cnfe); } Constructor ctor; try { ctor = dclClass.getConstructor(String.class, String.class, String.class, ClassLoader.class); } catch (NoSuchMethodException nsme) { throw new RuntimeException("DCL ctor", nsme); } // create an instance, using the path we found Object dclObj; try { dclObj = ctor.newInstance(CLASS_PATH, odexDir, LIB_DIR, myLoader); } catch (Exception ex) { throw new RuntimeException("DCL newInstance", ex); } return (ClassLoader) dclObj; } }