/* * Copyright (C) 2007 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. */ package dalvik.system; import android.system.ErrnoException; import dalvik.annotation.compat.UnsupportedAppUsage; import dalvik.annotation.optimization.ReachabilitySensitive; import java.io.File; import java.io.FileNotFoundException; import java.io.IOException; import java.nio.ByteBuffer; import java.util.Arrays; import java.util.Enumeration; import java.util.List; import libcore.io.Libcore; /** * Loads DEX files. This class is meant for internal use and should not be used * by applications. * * @deprecated This class should not be used directly by applications. It will hurt * performance in most cases and will lead to incorrect execution of bytecode in * the worst case. Applications should use one of the standard classloaders such * as {@link dalvik.system.PathClassLoader} instead. This API will be removed * in a future Android release. */ @libcore.api.CorePlatformApi @Deprecated public final class DexFile { /** * If close is called, mCookie becomes null but the internal cookie is preserved if the close * failed so that we can free resources in the finalizer. */ @UnsupportedAppUsage @ReachabilitySensitive private Object mCookie; @UnsupportedAppUsage private Object mInternalCookie; @UnsupportedAppUsage private final String mFileName; /** * Opens a DEX file from a given File object. * * @deprecated Applications should use one of the standard classloaders such * as {@link dalvik.system.PathClassLoader} instead. This API will be removed * in a future Android release. */ @Deprecated public DexFile(File file) throws IOException { this(file.getPath()); } /* * Private version with class loader argument. * * @param file * the File object referencing the actual DEX file * @param loader * the class loader object creating the DEX file object * @param elements * the temporary dex path list elements from DexPathList.makeElements */ DexFile(File file, ClassLoader loader, DexPathList.Element[] elements) throws IOException { this(file.getPath(), loader, elements); } /** * Opens a DEX file from a given filename. * * @deprecated Applications should use one of the standard classloaders such * as {@link dalvik.system.PathClassLoader} instead. This API will be removed * in a future Android release. */ @Deprecated public DexFile(String fileName) throws IOException { this(fileName, null, null); } /* * Private version with class loader argument. * * @param fileName * the filename of the DEX file * @param loader * the class loader creating the DEX file object * @param elements * the temporary dex path list elements from DexPathList.makeElements */ DexFile(String fileName, ClassLoader loader, DexPathList.Element[] elements) throws IOException { mCookie = openDexFile(fileName, null, 0, loader, elements); mInternalCookie = mCookie; mFileName = fileName; //System.out.println("DEX FILE cookie is " + mCookie + " fileName=" + fileName); } DexFile(ByteBuffer[] bufs, ClassLoader loader, DexPathList.Element[] elements) throws IOException { mCookie = openInMemoryDexFiles(bufs, loader, elements); mInternalCookie = mCookie; mFileName = null; } /** * Opens a DEX file from a given filename, using a specified file * to hold the optimized data. * * @param sourceName * Jar or APK file with "classes.dex". * @param outputName * File that will hold the optimized form of the DEX data. * @param flags * Enable optional features. * @param loader * The class loader creating the DEX file object. * @param elements * The temporary dex path list elements from DexPathList.makeElements */ private DexFile(String sourceName, String outputName, int flags, ClassLoader loader, DexPathList.Element[] elements) throws IOException { if (outputName != null) { try { String parent = new File(outputName).getParent(); if (Libcore.os.getuid() != Libcore.os.stat(parent).st_uid) { throw new IllegalArgumentException("Optimized data directory " + parent + " is not owned by the current user. Shared storage cannot protect" + " your application from code injection attacks."); } } catch (ErrnoException ignored) { // assume we'll fail with a more contextual error later } } mCookie = openDexFile(sourceName, outputName, flags, loader, elements); mInternalCookie = mCookie; mFileName = sourceName; //System.out.println("DEX FILE cookie is " + mCookie + " sourceName=" + sourceName + " outputName=" + outputName); } /** * Open a DEX file, specifying the file in which the optimized DEX * data should be written. If the optimized form exists and appears * to be current, it will be used; if not, the VM will attempt to * regenerate it. * * @deprecated Applications should use one of the standard classloaders such * as {@link dalvik.system.PathClassLoader} instead. This API will be removed * in a future Android release. */ @Deprecated static public DexFile loadDex(String sourcePathName, String outputPathName, int flags) throws IOException { /* * TODO: we may want to cache previously-opened DexFile objects. * The cache would be synchronized with close(). This would help * us avoid mapping the same DEX more than once when an app * decided to open it multiple times. In practice this may not * be a real issue. */ return loadDex(sourcePathName, outputPathName, flags, null, null); } /* * Private version of loadDex that also takes a class loader. * * @param sourcePathName * Jar or APK file with "classes.dex". (May expand this to include * "raw DEX" in the future.) * @param outputPathName * File that will hold the optimized form of the DEX data. * @param flags * Enable optional features. (Currently none defined.) * @param loader * Class loader that is aloading the DEX file. * @param elements * The temporary dex path list elements from DexPathList.makeElements * @return * A new or previously-opened DexFile. * @throws IOException * If unable to open the source or output file. */ @UnsupportedAppUsage static DexFile loadDex(String sourcePathName, String outputPathName, int flags, ClassLoader loader, DexPathList.Element[] elements) throws IOException { /* * TODO: we may want to cache previously-opened DexFile objects. * The cache would be synchronized with close(). This would help * us avoid mapping the same DEX more than once when an app * decided to open it multiple times. In practice this may not * be a real issue. */ return new DexFile(sourcePathName, outputPathName, flags, loader, elements); } /** * Gets the name of the (already opened) DEX file. * * @return the file name */ public String getName() { return mFileName; } @Override public String toString() { if (mFileName != null) { return getName(); } else { return "InMemoryDexFile[cookie=" + Arrays.toString((long[]) mCookie) + "]"; } } /** * Closes the DEX file. *

* This may not be able to release all of the resources. If classes from this DEX file are * still resident, the DEX file can't be unmapped. In the case where we do not release all * the resources, close is called again in the finalizer. * * @throws IOException * if an I/O error occurs during closing the file, which * normally should not happen */ public void close() throws IOException { if (mInternalCookie != null) { if (closeDexFile(mInternalCookie)) { mInternalCookie = null; } mCookie = null; } } /** * Loads a class. Returns the class on success, or a {@code null} reference * on failure. *

* If you are not calling this from a class loader, this is most likely not * going to do what you want. Use {@link Class#forName(String)} instead. *

* The method does not throw {@link ClassNotFoundException} if the class * isn't found because it isn't reasonable to throw exceptions wildly every * time a class is not found in the first DEX file we look at. * * @param name * the class name, which should look like "java/lang/String" * * @param loader * the class loader that tries to load the class (in most cases * the caller of the method * * @return the {@link Class} object representing the class, or {@code null} * if the class cannot be loaded */ public Class loadClass(String name, ClassLoader loader) { String slashName = name.replace('.', '/'); return loadClassBinaryName(slashName, loader, null); } /** * See {@link #loadClass(String, ClassLoader)}. * * This takes a "binary" class name to better match ClassLoader semantics. * * @hide */ @UnsupportedAppUsage public Class loadClassBinaryName(String name, ClassLoader loader, List suppressed) { return defineClass(name, loader, mCookie, this, suppressed); } private static Class defineClass(String name, ClassLoader loader, Object cookie, DexFile dexFile, List suppressed) { Class result = null; try { result = defineClassNative(name, loader, cookie, dexFile); } catch (NoClassDefFoundError e) { if (suppressed != null) { suppressed.add(e); } } catch (ClassNotFoundException e) { if (suppressed != null) { suppressed.add(e); } } return result; } /** * Enumerate the names of the classes in this DEX file. * * @return an enumeration of names of classes contained in the DEX file, in * the usual internal form (like "java/lang/String"). */ public Enumeration entries() { return new DFEnum(this); } /* * Helper class. */ private static class DFEnum implements Enumeration { private int mIndex; @UnsupportedAppUsage private String[] mNameList; DFEnum(DexFile df) { mIndex = 0; mNameList = getClassNameList(df.mCookie); } public boolean hasMoreElements() { return (mIndex < mNameList.length); } public String nextElement() { return mNameList[mIndex++]; } } /** * Called when the class is finalized. Makes sure the DEX file is closed. * * @throws IOException * if an I/O error occurs during closing the file, which * normally should not happen */ @Override protected void finalize() throws Throwable { try { if (mInternalCookie != null && !closeDexFile(mInternalCookie)) { throw new AssertionError("Failed to close dex file in finalizer."); } mInternalCookie = null; mCookie = null; } finally { super.finalize(); } } /* * Open a DEX file. The value returned is a magic VM cookie. On * failure, an IOException is thrown. */ @UnsupportedAppUsage private static Object openDexFile(String sourceName, String outputName, int flags, ClassLoader loader, DexPathList.Element[] elements) throws IOException { // Use absolute paths to enable the use of relative paths when testing on host. return openDexFileNative(new File(sourceName).getAbsolutePath(), (outputName == null) ? null : new File(outputName).getAbsolutePath(), flags, loader, elements); } private static Object openInMemoryDexFiles(ByteBuffer[] bufs, ClassLoader loader, DexPathList.Element[] elements) throws IOException { // Preprocess the ByteBuffers for openInMemoryDexFilesNative. We extract // the backing array (non-direct buffers only) and start/end positions // so that the native method does not have to call Java methods anymore. byte[][] arrays = new byte[bufs.length][]; int[] starts = new int[bufs.length]; int[] ends = new int[bufs.length]; for (int i = 0; i < bufs.length; ++i) { arrays[i] = bufs[i].isDirect() ? null : bufs[i].array(); starts[i] = bufs[i].position(); ends[i] = bufs[i].limit(); } return openInMemoryDexFilesNative(bufs, arrays, starts, ends, loader, elements); } private static native Object openInMemoryDexFilesNative(ByteBuffer[] bufs, byte[][] arrays, int[] starts, int[] ends, ClassLoader loader, DexPathList.Element[] elements); /* * Initiates background verification of this DexFile. This is a sepearate down-call * from openDexFile and openInMemoryDexFiles because it requires the class loader's * DexPathList to have been initialized for its classes to be resolvable by ART. * DexPathList will open the dex files first, finalize `dexElements` and then call this. */ /*package*/ void verifyInBackground(ClassLoader classLoader, String classLoaderContext) { verifyInBackgroundNative(mCookie, classLoader, classLoaderContext); } private static native void verifyInBackgroundNative(Object mCookie, ClassLoader classLoader, String classLoaderContext); /*package*/ static native String getClassLoaderContext(ClassLoader classLoader, DexPathList.Element[] elements); /* * Returns true if the dex file is backed by a valid oat file. */ @UnsupportedAppUsage /*package*/ boolean isBackedByOatFile() { return isBackedByOatFile(mCookie); } /* * Set the dex file as trusted: it can access hidden APIs of the platform. */ /*package*/ void setTrusted() { setTrusted(mCookie); } /* * Returns true if we managed to close the dex file. */ private static native boolean closeDexFile(Object cookie); private static native Class defineClassNative(String name, ClassLoader loader, Object cookie, DexFile dexFile) throws ClassNotFoundException, NoClassDefFoundError; @UnsupportedAppUsage private static native String[] getClassNameList(Object cookie); private static native boolean isBackedByOatFile(Object cookie); private static native void setTrusted(Object cookie); /* * Open a DEX file. The value returned is a magic VM cookie. On * failure, an IOException is thrown. */ @UnsupportedAppUsage private static native Object openDexFileNative(String sourceName, String outputName, int flags, ClassLoader loader, DexPathList.Element[] elements); /** * Returns true if the VM believes that the apk/jar file is out of date * and should be passed through "dexopt" again. * * @param fileName the absolute path to the apk/jar file to examine. * @return true if dexopt should be called on the file, false otherwise. * @throws java.io.FileNotFoundException if fileName is not readable, * not a file, or not present. * @throws java.io.IOException if fileName is not a valid apk/jar file or * if problems occur while parsing it. * @throws java.lang.NullPointerException if fileName is null. */ public static native boolean isDexOptNeeded(String fileName) throws FileNotFoundException, IOException; /** * No dexopt should (or can) be done to update the apk/jar. * * See {@link #getDexOptNeeded(String, String, String, boolean, boolean)}. * * @hide */ @libcore.api.CorePlatformApi public static final int NO_DEXOPT_NEEDED = 0; /** * dex2oat should be run to update the apk/jar from scratch. * * See {@link #getDexOptNeeded(String, String, String, boolean, boolean)}. * * @hide */ public static final int DEX2OAT_FROM_SCRATCH = 1; /** * dex2oat should be run to update the apk/jar because the existing code * is out of date with respect to the boot image. * * See {@link #getDexOptNeeded(String, String, String, boolean, boolean)}. * * @hide */ public static final int DEX2OAT_FOR_BOOT_IMAGE = 2; /** * dex2oat should be run to update the apk/jar because the existing code * is out of date with respect to the target compiler filter. * * See {@link #getDexOptNeeded(String, String, String, boolean, boolean)}. * * @hide */ @libcore.api.CorePlatformApi public static final int DEX2OAT_FOR_FILTER = 3; /** * Calls {@link #getDexOptNeeded(String, String, String, String, String, boolean, boolean)} * with a null class loader context. * * TODO(ngeoffray, calin): deprecate / remove. * @hide */ public static int getDexOptNeeded(String fileName, String instructionSet, String compilerFilter, boolean newProfile, boolean downgrade) throws FileNotFoundException, IOException { return getDexOptNeeded( fileName, instructionSet, compilerFilter, null, newProfile, downgrade); } /** * Returns the VM's opinion of what kind of dexopt is needed to make the * apk/jar file up to date, where {@code targetMode} is used to indicate what * type of compilation the caller considers up-to-date, and {@code newProfile} * is used to indicate whether profile information has changed recently. * * @param fileName the absolute path to the apk/jar file to examine. * @param compilerFilter a compiler filter to use for what a caller considers up-to-date. * @param classLoaderContext a string encoding the class loader context the dex file * is intended to have at runtime. * @param newProfile flag that describes whether a profile corresponding * to the dex file has been recently updated and should be considered * in the state of the file. * @param downgrade flag that describes if the purpose of dexopt is to downgrade the * compiler filter. If set to false, will be evaluated as an upgrade request. * @return NO_DEXOPT_NEEDED, or DEX2OAT_*. See documentation * of the particular status code for more information on its * meaning. Returns a positive status code if the status refers to * the oat file in the oat location. Returns a negative status * code if the status refers to the oat file in the odex location. * @throws java.io.FileNotFoundException if fileName is not readable, * not a file, or not present. * @throws java.io.IOException if fileName is not a valid apk/jar file or * if problems occur while parsing it. * @throws java.lang.NullPointerException if fileName is null. * * @hide */ @libcore.api.CorePlatformApi public static native int getDexOptNeeded(String fileName, String instructionSet, String compilerFilter, String classLoaderContext, boolean newProfile, boolean downgrade) throws FileNotFoundException, IOException; /** * Returns the status of the dex file {@code fileName}. The returned string is * an opaque, human readable representation of the current status. The output * is only meant for debugging and is not guaranteed to be stable across * releases and/or devices. * * @hide */ public static native String getDexFileStatus(String fileName, String instructionSet) throws FileNotFoundException; /** * Encapsulates information about the optimizations performed on a dex file. * * Note that the info is only meant for debugging and is not guaranteed to be * stable across releases and/or devices. * * @hide */ @libcore.api.CorePlatformApi public static final class OptimizationInfo { // The optimization status. private final String status; // The optimization reason. The reason might be "unknown" if the // the compiler artifacts were not annotated during optimizations. private final String reason; private OptimizationInfo(String status, String reason) { this.status = status; this.reason = reason; } @libcore.api.CorePlatformApi public String getStatus() { return status; } @libcore.api.CorePlatformApi public String getReason() { return reason; } } /** * Retrieves the optimization info for a dex file. * * @hide */ @libcore.api.CorePlatformApi public static OptimizationInfo getDexFileOptimizationInfo( String fileName, String instructionSet) throws FileNotFoundException { String[] status = getDexFileOptimizationStatus(fileName, instructionSet); return new OptimizationInfo(status[0], status[1]); } /** * Returns the optimization status of the dex file {@code fileName}. The returned * array will have 2 elements which specify: * - index 0: the level of optimizations * - index 1: the optimization reason. The reason might be "unknown" if the * the compiler artifacts were not annotated during optimizations. * * The output is only meant for debugging and is not guaranteed to be stable across * releases and/or devices. * * @hide */ private static native String[] getDexFileOptimizationStatus( String fileName, String instructionSet) throws FileNotFoundException; /** * Returns the paths of the optimized files generated for {@code fileName}. * If no optimized code exists the method returns null. * @hide */ @libcore.api.CorePlatformApi public static native String[] getDexFileOutputPaths(String fileName, String instructionSet) throws FileNotFoundException; /** * Returns whether the given filter is a valid filter. * * @hide */ @libcore.api.CorePlatformApi public native static boolean isValidCompilerFilter(String filter); /** * Returns whether the given filter is based on profiles. * * @hide */ @libcore.api.CorePlatformApi public native static boolean isProfileGuidedCompilerFilter(String filter); /** * Returns the version of the compiler filter that is not based on profiles. * If the input is not a valid filter, or the filter is already not based on * profiles, this returns the input. * * @hide */ public native static String getNonProfileGuidedCompilerFilter(String filter); /** * Returns the version of the compiler filter that is suitable for safe mode. * If the input is not a valid filter, or the filter is already suitable for * safe mode, this returns the input. * * @hide */ @libcore.api.CorePlatformApi public native static String getSafeModeCompilerFilter(String filter); /** * Returns the static file size of the original dex file. * The original size of the uncompressed dex file is returned. * On device the dex file may be compressed or embedded in some other * file (e.g. oat) in a platform implementation dependent manner. This * method abstracts away from those details and provides an efficient * implementation given that the dex file in question has already been * uncompressed, extracted, and/or loaded by the runtime as appropriate. *

* In the case of multidex, returns the sum of the original uncompressed * multidex entry file sizes. * * @hide */ public long getStaticSizeOfDexFile() { return getStaticSizeOfDexFile(mCookie); } private static native long getStaticSizeOfDexFile(Object cookie); }