/* * 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 android.os; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.UnsupportedAppUsage; import android.app.AppGlobals; import android.content.Context; import android.util.Log; import com.android.internal.util.FastPrintWriter; import com.android.internal.util.Preconditions; import com.android.internal.util.TypedProperties; import dalvik.system.VMDebug; import org.apache.harmony.dalvik.ddmc.Chunk; import org.apache.harmony.dalvik.ddmc.ChunkHandler; import org.apache.harmony.dalvik.ddmc.DdmServer; import java.io.File; import java.io.FileDescriptor; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.FileReader; import java.io.IOException; import java.io.PrintWriter; import java.io.Reader; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; import java.lang.reflect.Field; import java.lang.reflect.Modifier; import java.util.HashMap; import java.util.Map; /** * Provides various debugging methods for Android applications, including * tracing and allocation counts. *
Logging Trace Files
*Debug can create log files that give details about an application, such as * a call stack and start/stop times for any running methods. See Inspect Trace Logs with * Traceview for information about reading trace files. To start logging * trace files, call one of the startMethodTracing() methods. To stop tracing, * call {@link #stopMethodTracing()}. */ public final class Debug { private static final String TAG = "Debug"; /** * Flags for startMethodTracing(). These can be ORed together. * * TRACE_COUNT_ALLOCS adds the results from startAllocCounting to the * trace key file. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static final int TRACE_COUNT_ALLOCS = VMDebug.TRACE_COUNT_ALLOCS; /** * Flags for printLoadedClasses(). Default behavior is to only show * the class name. */ public static final int SHOW_FULL_DETAIL = 1; public static final int SHOW_CLASSLOADER = (1 << 1); public static final int SHOW_INITIALIZED = (1 << 2); // set/cleared by waitForDebugger() private static volatile boolean mWaiting = false; @UnsupportedAppUsage private Debug() {} /* * How long to wait for the debugger to finish sending requests. I've * seen this hit 800msec on the device while waiting for a response * to travel over USB and get processed, so we take that and add * half a second. */ private static final int MIN_DEBUGGER_IDLE = 1300; // msec /* how long to sleep when polling for activity */ private static final int SPIN_DELAY = 200; // msec /** * Default trace file path and file */ private static final String DEFAULT_TRACE_BODY = "dmtrace"; private static final String DEFAULT_TRACE_EXTENSION = ".trace"; /** * This class is used to retrieved various statistics about the memory mappings for this * process. The returned info is broken down by dalvik, native, and other. All results are in kB. */ public static class MemoryInfo implements Parcelable { /** The proportional set size for dalvik heap. (Doesn't include other Dalvik overhead.) */ public int dalvikPss; /** The proportional set size that is swappable for dalvik heap. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int dalvikSwappablePss; /** @hide The resident set size for dalvik heap. (Without other Dalvik overhead.) */ @UnsupportedAppUsage public int dalvikRss; /** The private dirty pages used by dalvik heap. */ public int dalvikPrivateDirty; /** The shared dirty pages used by dalvik heap. */ public int dalvikSharedDirty; /** The private clean pages used by dalvik heap. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int dalvikPrivateClean; /** The shared clean pages used by dalvik heap. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int dalvikSharedClean; /** The dirty dalvik pages that have been swapped out. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int dalvikSwappedOut; /** The dirty dalvik pages that have been swapped out, proportional. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int dalvikSwappedOutPss; /** The proportional set size for the native heap. */ public int nativePss; /** The proportional set size that is swappable for the native heap. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int nativeSwappablePss; /** @hide The resident set size for the native heap. */ @UnsupportedAppUsage public int nativeRss; /** The private dirty pages used by the native heap. */ public int nativePrivateDirty; /** The shared dirty pages used by the native heap. */ public int nativeSharedDirty; /** The private clean pages used by the native heap. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int nativePrivateClean; /** The shared clean pages used by the native heap. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int nativeSharedClean; /** The dirty native pages that have been swapped out. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int nativeSwappedOut; /** The dirty native pages that have been swapped out, proportional. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int nativeSwappedOutPss; /** The proportional set size for everything else. */ public int otherPss; /** The proportional set size that is swappable for everything else. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int otherSwappablePss; /** @hide The resident set size for everything else. */ @UnsupportedAppUsage public int otherRss; /** The private dirty pages used by everything else. */ public int otherPrivateDirty; /** The shared dirty pages used by everything else. */ public int otherSharedDirty; /** The private clean pages used by everything else. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int otherPrivateClean; /** The shared clean pages used by everything else. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int otherSharedClean; /** The dirty pages used by anyting else that have been swapped out. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int otherSwappedOut; /** The dirty pages used by anyting else that have been swapped out, proportional. */ /** @hide We may want to expose this, eventually. */ @UnsupportedAppUsage public int otherSwappedOutPss; /** Whether the kernel reports proportional swap usage */ /** @hide */ @UnsupportedAppUsage public boolean hasSwappedOutPss; /** @hide */ public static final int HEAP_UNKNOWN = 0; /** @hide */ public static final int HEAP_DALVIK = 1; /** @hide */ public static final int HEAP_NATIVE = 2; /** @hide */ public static final int OTHER_DALVIK_OTHER = 0; /** @hide */ public static final int OTHER_STACK = 1; /** @hide */ public static final int OTHER_CURSOR = 2; /** @hide */ public static final int OTHER_ASHMEM = 3; /** @hide */ public static final int OTHER_GL_DEV = 4; /** @hide */ public static final int OTHER_UNKNOWN_DEV = 5; /** @hide */ public static final int OTHER_SO = 6; /** @hide */ public static final int OTHER_JAR = 7; /** @hide */ public static final int OTHER_APK = 8; /** @hide */ public static final int OTHER_TTF = 9; /** @hide */ public static final int OTHER_DEX = 10; /** @hide */ public static final int OTHER_OAT = 11; /** @hide */ public static final int OTHER_ART = 12; /** @hide */ public static final int OTHER_UNKNOWN_MAP = 13; /** @hide */ public static final int OTHER_GRAPHICS = 14; /** @hide */ public static final int OTHER_GL = 15; /** @hide */ public static final int OTHER_OTHER_MEMTRACK = 16; // Needs to be declared here for the DVK_STAT ranges below. /** @hide */ @UnsupportedAppUsage public static final int NUM_OTHER_STATS = 17; // Dalvik subsections. /** @hide */ public static final int OTHER_DALVIK_NORMAL = 17; /** @hide */ public static final int OTHER_DALVIK_LARGE = 18; /** @hide */ public static final int OTHER_DALVIK_ZYGOTE = 19; /** @hide */ public static final int OTHER_DALVIK_NON_MOVING = 20; // Section begins and ends for dumpsys, relative to the DALVIK categories. /** @hide */ public static final int OTHER_DVK_STAT_DALVIK_START = OTHER_DALVIK_NORMAL - NUM_OTHER_STATS; /** @hide */ public static final int OTHER_DVK_STAT_DALVIK_END = OTHER_DALVIK_NON_MOVING - NUM_OTHER_STATS; // Dalvik Other subsections. /** @hide */ public static final int OTHER_DALVIK_OTHER_LINEARALLOC = 21; /** @hide */ public static final int OTHER_DALVIK_OTHER_ACCOUNTING = 22; /** @hide */ public static final int OTHER_DALVIK_OTHER_CODE_CACHE = 23; /** @hide */ public static final int OTHER_DALVIK_OTHER_COMPILER_METADATA = 24; /** @hide */ public static final int OTHER_DALVIK_OTHER_INDIRECT_REFERENCE_TABLE = 25; /** @hide */ public static final int OTHER_DVK_STAT_DALVIK_OTHER_START = OTHER_DALVIK_OTHER_LINEARALLOC - NUM_OTHER_STATS; /** @hide */ public static final int OTHER_DVK_STAT_DALVIK_OTHER_END = OTHER_DALVIK_OTHER_INDIRECT_REFERENCE_TABLE - NUM_OTHER_STATS; // Dex subsections (Boot vdex, App dex, and App vdex). /** @hide */ public static final int OTHER_DEX_BOOT_VDEX = 26; /** @hide */ public static final int OTHER_DEX_APP_DEX = 27; /** @hide */ public static final int OTHER_DEX_APP_VDEX = 28; /** @hide */ public static final int OTHER_DVK_STAT_DEX_START = OTHER_DEX_BOOT_VDEX - NUM_OTHER_STATS; /** @hide */ public static final int OTHER_DVK_STAT_DEX_END = OTHER_DEX_APP_VDEX - NUM_OTHER_STATS; // Art subsections (App image, boot image). /** @hide */ public static final int OTHER_ART_APP = 29; /** @hide */ public static final int OTHER_ART_BOOT = 30; /** @hide */ public static final int OTHER_DVK_STAT_ART_START = OTHER_ART_APP - NUM_OTHER_STATS; /** @hide */ public static final int OTHER_DVK_STAT_ART_END = OTHER_ART_BOOT - NUM_OTHER_STATS; /** @hide */ @UnsupportedAppUsage public static final int NUM_DVK_STATS = 14; /** @hide */ public static final int NUM_CATEGORIES = 9; /** @hide */ public static final int OFFSET_PSS = 0; /** @hide */ public static final int OFFSET_SWAPPABLE_PSS = 1; /** @hide */ public static final int OFFSET_RSS = 2; /** @hide */ public static final int OFFSET_PRIVATE_DIRTY = 3; /** @hide */ public static final int OFFSET_SHARED_DIRTY = 4; /** @hide */ public static final int OFFSET_PRIVATE_CLEAN = 5; /** @hide */ public static final int OFFSET_SHARED_CLEAN = 6; /** @hide */ public static final int OFFSET_SWAPPED_OUT = 7; /** @hide */ public static final int OFFSET_SWAPPED_OUT_PSS = 8; @UnsupportedAppUsage private int[] otherStats = new int[(NUM_OTHER_STATS+NUM_DVK_STATS)*NUM_CATEGORIES]; public MemoryInfo() { } /** * @hide Copy contents from another object. */ public void set(MemoryInfo other) { dalvikPss = other.dalvikPss; dalvikSwappablePss = other.dalvikSwappablePss; dalvikRss = other.dalvikRss; dalvikPrivateDirty = other.dalvikPrivateDirty; dalvikSharedDirty = other.dalvikSharedDirty; dalvikPrivateClean = other.dalvikPrivateClean; dalvikSharedClean = other.dalvikSharedClean; dalvikSwappedOut = other.dalvikSwappedOut; dalvikSwappedOutPss = other.dalvikSwappedOutPss; nativePss = other.nativePss; nativeSwappablePss = other.nativeSwappablePss; nativeRss = other.nativeRss; nativePrivateDirty = other.nativePrivateDirty; nativeSharedDirty = other.nativeSharedDirty; nativePrivateClean = other.nativePrivateClean; nativeSharedClean = other.nativeSharedClean; nativeSwappedOut = other.nativeSwappedOut; nativeSwappedOutPss = other.nativeSwappedOutPss; otherPss = other.otherPss; otherSwappablePss = other.otherSwappablePss; otherRss = other.otherRss; otherPrivateDirty = other.otherPrivateDirty; otherSharedDirty = other.otherSharedDirty; otherPrivateClean = other.otherPrivateClean; otherSharedClean = other.otherSharedClean; otherSwappedOut = other.otherSwappedOut; otherSwappedOutPss = other.otherSwappedOutPss; hasSwappedOutPss = other.hasSwappedOutPss; System.arraycopy(other.otherStats, 0, otherStats, 0, otherStats.length); } /** * Return total PSS memory usage in kB. */ public int getTotalPss() { return dalvikPss + nativePss + otherPss + getTotalSwappedOutPss(); } /** * @hide Return total PSS memory usage in kB. */ @UnsupportedAppUsage public int getTotalUss() { return dalvikPrivateClean + dalvikPrivateDirty + nativePrivateClean + nativePrivateDirty + otherPrivateClean + otherPrivateDirty; } /** * Return total PSS memory usage in kB mapping a file of one of the following extension: * .so, .jar, .apk, .ttf, .dex, .odex, .oat, .art . */ public int getTotalSwappablePss() { return dalvikSwappablePss + nativeSwappablePss + otherSwappablePss; } /** * @hide Return total RSS memory usage in kB. */ public int getTotalRss() { return dalvikRss + nativeRss + otherRss; } /** * Return total private dirty memory usage in kB. */ public int getTotalPrivateDirty() { return dalvikPrivateDirty + nativePrivateDirty + otherPrivateDirty; } /** * Return total shared dirty memory usage in kB. */ public int getTotalSharedDirty() { return dalvikSharedDirty + nativeSharedDirty + otherSharedDirty; } /** * Return total shared clean memory usage in kB. */ public int getTotalPrivateClean() { return dalvikPrivateClean + nativePrivateClean + otherPrivateClean; } /** * Return total shared clean memory usage in kB. */ public int getTotalSharedClean() { return dalvikSharedClean + nativeSharedClean + otherSharedClean; } /** * Return total swapped out memory in kB. * @hide */ public int getTotalSwappedOut() { return dalvikSwappedOut + nativeSwappedOut + otherSwappedOut; } /** * Return total swapped out memory in kB, proportional. * @hide */ public int getTotalSwappedOutPss() { return dalvikSwappedOutPss + nativeSwappedOutPss + otherSwappedOutPss; } /** @hide */ @UnsupportedAppUsage public int getOtherPss(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_PSS]; } /** @hide */ public int getOtherSwappablePss(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_SWAPPABLE_PSS]; } /** @hide */ public int getOtherRss(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_RSS]; } /** @hide */ @UnsupportedAppUsage public int getOtherPrivateDirty(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_PRIVATE_DIRTY]; } /** @hide */ @UnsupportedAppUsage public int getOtherSharedDirty(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_SHARED_DIRTY]; } /** @hide */ public int getOtherPrivateClean(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_PRIVATE_CLEAN]; } /** @hide */ @UnsupportedAppUsage public int getOtherPrivate(int which) { return getOtherPrivateClean(which) + getOtherPrivateDirty(which); } /** @hide */ public int getOtherSharedClean(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_SHARED_CLEAN]; } /** @hide */ public int getOtherSwappedOut(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_SWAPPED_OUT]; } /** @hide */ public int getOtherSwappedOutPss(int which) { return otherStats[which * NUM_CATEGORIES + OFFSET_SWAPPED_OUT_PSS]; } /** @hide */ @UnsupportedAppUsage public static String getOtherLabel(int which) { switch (which) { case OTHER_DALVIK_OTHER: return "Dalvik Other"; case OTHER_STACK: return "Stack"; case OTHER_CURSOR: return "Cursor"; case OTHER_ASHMEM: return "Ashmem"; case OTHER_GL_DEV: return "Gfx dev"; case OTHER_UNKNOWN_DEV: return "Other dev"; case OTHER_SO: return ".so mmap"; case OTHER_JAR: return ".jar mmap"; case OTHER_APK: return ".apk mmap"; case OTHER_TTF: return ".ttf mmap"; case OTHER_DEX: return ".dex mmap"; case OTHER_OAT: return ".oat mmap"; case OTHER_ART: return ".art mmap"; case OTHER_UNKNOWN_MAP: return "Other mmap"; case OTHER_GRAPHICS: return "EGL mtrack"; case OTHER_GL: return "GL mtrack"; case OTHER_OTHER_MEMTRACK: return "Other mtrack"; case OTHER_DALVIK_NORMAL: return ".Heap"; case OTHER_DALVIK_LARGE: return ".LOS"; case OTHER_DALVIK_ZYGOTE: return ".Zygote"; case OTHER_DALVIK_NON_MOVING: return ".NonMoving"; case OTHER_DALVIK_OTHER_LINEARALLOC: return ".LinearAlloc"; case OTHER_DALVIK_OTHER_ACCOUNTING: return ".GC"; case OTHER_DALVIK_OTHER_CODE_CACHE: return ".JITCache"; case OTHER_DALVIK_OTHER_COMPILER_METADATA: return ".CompilerMetadata"; case OTHER_DALVIK_OTHER_INDIRECT_REFERENCE_TABLE: return ".IndirectRef"; case OTHER_DEX_BOOT_VDEX: return ".Boot vdex"; case OTHER_DEX_APP_DEX: return ".App dex"; case OTHER_DEX_APP_VDEX: return ".App vdex"; case OTHER_ART_APP: return ".App art"; case OTHER_ART_BOOT: return ".Boot art"; default: return "????"; } } /** * Returns the value of a particular memory statistic or {@code null} if no * such memory statistic exists. * *
The following table lists the memory statistics that are supported. * Note that memory statistics may be added or removed in a future API level.
* *Memory statistic name | *Meaning | *Example | *Supported (API Levels) | *
---|---|---|---|
summary.java-heap | *The private Java Heap usage in kB. This corresponds to the Java Heap field * in the App Summary section output by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.native-heap | *The private Native Heap usage in kB. This corresponds to the Native Heap * field in the App Summary section output by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.code | *The memory usage for static code and resources in kB. This corresponds to * the Code field in the App Summary section output by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.stack | *The stack usage in kB. This corresponds to the Stack field in the * App Summary section output by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.graphics | *The graphics usage in kB. This corresponds to the Graphics field in the * App Summary section output by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.private-other | *Other private memory usage in kB. This corresponds to the Private Other * field output in the App Summary section by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.system | *Shared and system memory usage in kB. This corresponds to the System * field output in the App Summary section by dumpsys meminfo. | *{@code 1442} | *23 | *
summary.total-pss | *Total PPS memory usage in kB. | *{@code 1442} | *23 | *
summary.total-swap | *Total swap usage in kB. | *{@code 1442} | *23 | *
emulator -trace foo
* The main differences between this and {@link #startMethodTracing()} are * that tracing in the qemu emulator traces every cpu instruction of every * process, including kernel code, so we have more complete information, * including all context switches. We can also get more detailed information * such as cache misses. The sequence of calls is determined by * post-processing the instruction trace. The qemu tracing is also done * without modifying the application or perturbing the timing of calls * because no instrumentation is added to the application being traced. *
* ** One limitation of using this method compared to using * {@link #startMethodTracing()} on the real device is that the emulator * does not model all of the real hardware effects such as memory and * bus contention. The emulator also has a simple cache model and cannot * capture all the complexities of a real cache. *
*/ public static void startNativeTracing() { // Open the sysfs file for writing and write "1" to it. PrintWriter outStream = null; try { FileOutputStream fos = new FileOutputStream(SYSFS_QEMU_TRACE_STATE); outStream = new FastPrintWriter(fos); outStream.println("1"); } catch (Exception e) { } finally { if (outStream != null) outStream.close(); } VMDebug.startEmulatorTracing(); } /** * Stop qemu tracing. See {@link #startNativeTracing()} to start tracing. * *Tracing can be started and stopped as many times as desired. When * the qemu emulator itself is stopped then the buffered trace records * are flushed and written to the trace file. In fact, it is not necessary * to call this method at all; simply killing qemu is sufficient. But * starting and stopping a trace is useful for examining a specific * region of code.
*/ public static void stopNativeTracing() { VMDebug.stopEmulatorTracing(); // Open the sysfs file for writing and write "0" to it. PrintWriter outStream = null; try { FileOutputStream fos = new FileOutputStream(SYSFS_QEMU_TRACE_STATE); outStream = new FastPrintWriter(fos); outStream.println("0"); } catch (Exception e) { // We could print an error message here but we probably want // to quietly ignore errors if we are not running in the emulator. } finally { if (outStream != null) outStream.close(); } } /** * Enable "emulator traces", in which information about the current * method is made available to the "emulator -trace" feature. There * is no corresponding "disable" call -- this is intended for use by * the framework when tracing should be turned on and left that way, so * that traces captured with F9/F10 will include the necessary data. * * This puts the VM into "profile" mode, which has performance * consequences. * * To temporarily enable tracing, use {@link #startNativeTracing()}. */ public static void enableEmulatorTraceOutput() { VMDebug.startEmulatorTracing(); } /** * Start method tracing with default log name and buffer size. ** By default, the trace file is called "dmtrace.trace" and it's placed * under your package-specific directory on primary shared/external storage, * as returned by {@link Context#getExternalFilesDir(String)}. *
* See Inspect Trace Logs * with Traceview for information about reading trace files. *
* When method tracing is enabled, the VM will run more slowly than usual, * so the timings from the trace files should only be considered in relative * terms (e.g. was run #1 faster than run #2). The times for native methods * will not change, so don't try to use this to compare the performance of * interpreted and native implementations of the same method. As an * alternative, consider using sampling-based method tracing via * {@link #startMethodTracingSampling(String, int, int)} or "native" tracing * in the emulator via {@link #startNativeTracing()}. *
*/ public static void startMethodTracing() { VMDebug.startMethodTracing(fixTracePath(null), 0, 0, false, 0); } /** * Start method tracing, specifying the trace log file path. ** When a relative file path is given, the trace file will be placed under * your package-specific directory on primary shared/external storage, as * returned by {@link Context#getExternalFilesDir(String)}. *
* See Inspect Trace Logs * with Traceview for information about reading trace files. *
* When method tracing is enabled, the VM will run more slowly than usual, * so the timings from the trace files should only be considered in relative * terms (e.g. was run #1 faster than run #2). The times for native methods * will not change, so don't try to use this to compare the performance of * interpreted and native implementations of the same method. As an * alternative, consider using sampling-based method tracing via * {@link #startMethodTracingSampling(String, int, int)} or "native" tracing * in the emulator via {@link #startNativeTracing()}. *
* * @param tracePath Path to the trace log file to create. If {@code null}, * this will default to "dmtrace.trace". If the file already * exists, it will be truncated. If the path given does not end * in ".trace", it will be appended for you. */ public static void startMethodTracing(String tracePath) { startMethodTracing(tracePath, 0, 0); } /** * Start method tracing, specifying the trace log file name and the buffer * size. ** When a relative file path is given, the trace file will be placed under * your package-specific directory on primary shared/external storage, as * returned by {@link Context#getExternalFilesDir(String)}. *
* See Inspect Trace Logs * with Traceview for information about reading trace files. *
* When method tracing is enabled, the VM will run more slowly than usual, * so the timings from the trace files should only be considered in relative * terms (e.g. was run #1 faster than run #2). The times for native methods * will not change, so don't try to use this to compare the performance of * interpreted and native implementations of the same method. As an * alternative, consider using sampling-based method tracing via * {@link #startMethodTracingSampling(String, int, int)} or "native" tracing * in the emulator via {@link #startNativeTracing()}. *
* * @param tracePath Path to the trace log file to create. If {@code null}, * this will default to "dmtrace.trace". If the file already * exists, it will be truncated. If the path given does not end * in ".trace", it will be appended for you. * @param bufferSize The maximum amount of trace data we gather. If not * given, it defaults to 8MB. */ public static void startMethodTracing(String tracePath, int bufferSize) { startMethodTracing(tracePath, bufferSize, 0); } /** * Start method tracing, specifying the trace log file name, the buffer * size, and flags. ** When a relative file path is given, the trace file will be placed under * your package-specific directory on primary shared/external storage, as * returned by {@link Context#getExternalFilesDir(String)}. *
* See Inspect Trace Logs * with Traceview for information about reading trace files. *
* When method tracing is enabled, the VM will run more slowly than usual, * so the timings from the trace files should only be considered in relative * terms (e.g. was run #1 faster than run #2). The times for native methods * will not change, so don't try to use this to compare the performance of * interpreted and native implementations of the same method. As an * alternative, consider using sampling-based method tracing via * {@link #startMethodTracingSampling(String, int, int)} or "native" tracing * in the emulator via {@link #startNativeTracing()}. *
* * @param tracePath Path to the trace log file to create. If {@code null}, * this will default to "dmtrace.trace". If the file already * exists, it will be truncated. If the path given does not end * in ".trace", it will be appended for you. * @param bufferSize The maximum amount of trace data we gather. If not * given, it defaults to 8MB. * @param flags Flags to control method tracing. The only one that is * currently defined is {@link #TRACE_COUNT_ALLOCS}. */ public static void startMethodTracing(String tracePath, int bufferSize, int flags) { VMDebug.startMethodTracing(fixTracePath(tracePath), bufferSize, flags, false, 0); } /** * Start sampling-based method tracing, specifying the trace log file name, * the buffer size, and the sampling interval. ** When a relative file path is given, the trace file will be placed under * your package-specific directory on primary shared/external storage, as * returned by {@link Context#getExternalFilesDir(String)}. *
* See Inspect Trace Logs * with Traceview for information about reading trace files. * * @param tracePath Path to the trace log file to create. If {@code null}, * this will default to "dmtrace.trace". If the file already * exists, it will be truncated. If the path given does not end * in ".trace", it will be appended for you. * @param bufferSize The maximum amount of trace data we gather. If not * given, it defaults to 8MB. * @param intervalUs The amount of time between each sample in microseconds. */ public static void startMethodTracingSampling(String tracePath, int bufferSize, int intervalUs) { VMDebug.startMethodTracing(fixTracePath(tracePath), bufferSize, 0, true, intervalUs); } /** * Formats name of trace log file for method tracing. */ private static String fixTracePath(String tracePath) { if (tracePath == null || tracePath.charAt(0) != '/') { final Context context = AppGlobals.getInitialApplication(); final File dir; if (context != null) { dir = context.getExternalFilesDir(null); } else { dir = Environment.getExternalStorageDirectory(); } if (tracePath == null) { tracePath = new File(dir, DEFAULT_TRACE_BODY).getAbsolutePath(); } else { tracePath = new File(dir, tracePath).getAbsolutePath(); } } if (!tracePath.endsWith(DEFAULT_TRACE_EXTENSION)) { tracePath += DEFAULT_TRACE_EXTENSION; } return tracePath; } /** * Like startMethodTracing(String, int, int), but taking an already-opened * FileDescriptor in which the trace is written. The file name is also * supplied simply for logging. Makes a dup of the file descriptor. * * Not exposed in the SDK unless we are really comfortable with supporting * this and find it would be useful. * @hide */ public static void startMethodTracing(String traceName, FileDescriptor fd, int bufferSize, int flags, boolean streamOutput) { VMDebug.startMethodTracing(traceName, fd, bufferSize, flags, false, 0, streamOutput); } /** * Starts method tracing without a backing file. When stopMethodTracing * is called, the result is sent directly to DDMS. (If DDMS is not * attached when tracing ends, the profiling data will be discarded.) * * @hide */ public static void startMethodTracingDdms(int bufferSize, int flags, boolean samplingEnabled, int intervalUs) { VMDebug.startMethodTracingDdms(bufferSize, flags, samplingEnabled, intervalUs); } /** * Determine whether method tracing is currently active and what type is * active. * * @hide */ public static int getMethodTracingMode() { return VMDebug.getMethodTracingMode(); } /** * Stop method tracing. */ public static void stopMethodTracing() { VMDebug.stopMethodTracing(); } /** * Get an indication of thread CPU usage. The value returned * indicates the amount of time that the current thread has spent * executing code or waiting for certain types of I/O. * * The time is expressed in nanoseconds, and is only meaningful * when compared to the result from an earlier call. Note that * nanosecond resolution does not imply nanosecond accuracy. * * On system which don't support this operation, the call returns -1. */ public static long threadCpuTimeNanos() { return VMDebug.threadCpuTimeNanos(); } /** * Start counting the number and aggregate size of memory allocations. * *
The {@link #startAllocCounting() start} method resets the counts and enables counting.
* The {@link #stopAllocCounting() stop} method disables the counting so that the analysis
* code doesn't cause additional allocations. The various get
methods return
* the specified value. And the various reset
methods reset the specified
* count.
Counts are kept for the system as a whole (global) and for each thread. * The per-thread counts for threads other than the current thread * are not cleared by the "reset" or "start" calls.
* * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void startAllocCounting() { VMDebug.startAllocCounting(); } /** * Stop counting the number and aggregate size of memory allocations. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void stopAllocCounting() { VMDebug.stopAllocCounting(); } /** * Returns the global count of objects allocated by the runtime between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalAllocCount() { return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_ALLOCATED_OBJECTS); } /** * Clears the global count of objects allocated. * @see #getGlobalAllocCount() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalAllocCount() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_ALLOCATED_OBJECTS); } /** * Returns the global size, in bytes, of objects allocated by the runtime between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalAllocSize() { return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_ALLOCATED_BYTES); } /** * Clears the global size of objects allocated. * @see #getGlobalAllocSize() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalAllocSize() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_ALLOCATED_BYTES); } /** * Returns the global count of objects freed by the runtime between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalFreedCount() { return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_FREED_OBJECTS); } /** * Clears the global count of objects freed. * @see #getGlobalFreedCount() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalFreedCount() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_FREED_OBJECTS); } /** * Returns the global size, in bytes, of objects freed by the runtime between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalFreedSize() { return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_FREED_BYTES); } /** * Clears the global size of objects freed. * @see #getGlobalFreedSize() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalFreedSize() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_FREED_BYTES); } /** * Returns the number of non-concurrent GC invocations between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalGcInvocationCount() { return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_GC_INVOCATIONS); } /** * Clears the count of non-concurrent GC invocations. * @see #getGlobalGcInvocationCount() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalGcInvocationCount() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_GC_INVOCATIONS); } /** * Returns the number of classes successfully initialized (ie those that executed without * throwing an exception) between a {@link #startAllocCounting() start} and * {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalClassInitCount() { return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_CLASS_INIT_COUNT); } /** * Clears the count of classes initialized. * @see #getGlobalClassInitCount() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalClassInitCount() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_CLASS_INIT_COUNT); } /** * Returns the time spent successfully initializing classes between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getGlobalClassInitTime() { /* cumulative elapsed time for class initialization, in usec */ return VMDebug.getAllocCount(VMDebug.KIND_GLOBAL_CLASS_INIT_TIME); } /** * Clears the count of time spent initializing classes. * @see #getGlobalClassInitTime() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetGlobalClassInitTime() { VMDebug.resetAllocCount(VMDebug.KIND_GLOBAL_CLASS_INIT_TIME); } /** * This method exists for compatibility and always returns 0. * @deprecated This method is now obsolete. */ @Deprecated public static int getGlobalExternalAllocCount() { return 0; } /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static void resetGlobalExternalAllocSize() {} /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static void resetGlobalExternalAllocCount() {} /** * This method exists for compatibility and always returns 0. * @deprecated This method is now obsolete. */ @Deprecated public static int getGlobalExternalAllocSize() { return 0; } /** * This method exists for compatibility and always returns 0. * @deprecated This method is now obsolete. */ @Deprecated public static int getGlobalExternalFreedCount() { return 0; } /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static void resetGlobalExternalFreedCount() {} /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static int getGlobalExternalFreedSize() { return 0; } /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static void resetGlobalExternalFreedSize() {} /** * Returns the thread-local count of objects allocated by the runtime between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getThreadAllocCount() { return VMDebug.getAllocCount(VMDebug.KIND_THREAD_ALLOCATED_OBJECTS); } /** * Clears the thread-local count of objects allocated. * @see #getThreadAllocCount() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetThreadAllocCount() { VMDebug.resetAllocCount(VMDebug.KIND_THREAD_ALLOCATED_OBJECTS); } /** * Returns the thread-local size of objects allocated by the runtime between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * @return The allocated size in bytes. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getThreadAllocSize() { return VMDebug.getAllocCount(VMDebug.KIND_THREAD_ALLOCATED_BYTES); } /** * Clears the thread-local count of objects allocated. * @see #getThreadAllocSize() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetThreadAllocSize() { VMDebug.resetAllocCount(VMDebug.KIND_THREAD_ALLOCATED_BYTES); } /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static int getThreadExternalAllocCount() { return 0; } /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static void resetThreadExternalAllocCount() {} /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static int getThreadExternalAllocSize() { return 0; } /** * This method exists for compatibility and has no effect. * @deprecated This method is now obsolete. */ @Deprecated public static void resetThreadExternalAllocSize() {} /** * Returns the number of thread-local non-concurrent GC invocations between a * {@link #startAllocCounting() start} and {@link #stopAllocCounting() stop}. * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static int getThreadGcInvocationCount() { return VMDebug.getAllocCount(VMDebug.KIND_THREAD_GC_INVOCATIONS); } /** * Clears the thread-local count of non-concurrent GC invocations. * @see #getThreadGcInvocationCount() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetThreadGcInvocationCount() { VMDebug.resetAllocCount(VMDebug.KIND_THREAD_GC_INVOCATIONS); } /** * Clears all the global and thread-local memory allocation counters. * @see #startAllocCounting() * * @deprecated Accurate counting is a burden on the runtime and may be removed. */ @Deprecated public static void resetAllCounts() { VMDebug.resetAllocCount(VMDebug.KIND_ALL_COUNTS); } /** * Returns the value of a particular runtime statistic or {@code null} if no * such runtime statistic exists. * *The following table lists the runtime statistics that the runtime supports. * Note runtime statistics may be added or removed in a future API level.
* *Runtime statistic name | *Meaning | *Example | *Supported (API Levels) | *
---|---|---|---|
art.gc.gc-count | *The number of garbage collection runs. | *{@code 164} | *23 | *
art.gc.gc-time | *The total duration of garbage collection runs in ms. | *{@code 62364} | *23 | *
art.gc.bytes-allocated | *The total number of bytes that the application allocated. | *{@code 1463948408} | *23 | *
art.gc.bytes-freed | *The total number of bytes that garbage collection reclaimed. | *{@code 1313493084} | *23 | *
art.gc.blocking-gc-count | *The number of blocking garbage collection runs. | *{@code 2} | *23 | *
art.gc.blocking-gc-time | *The total duration of blocking garbage collection runs in ms. | *{@code 804} | *23 | *
art.gc.gc-count-rate-histogram | *Every 10 seconds, the gc-count-rate is computed as the number of garbage * collection runs that have occurred over the last 10 * seconds. art.gc.gc-count-rate-histogram is a histogram of the gc-count-rate * samples taken since the process began. The histogram can be used to identify * instances of high rates of garbage collection runs. For example, a histogram * of "0:34503,1:45350,2:11281,3:8088,4:43,5:8" shows that most of the time * there are between 0 and 2 garbage collection runs every 10 seconds, but there * were 8 distinct 10-second intervals in which 5 garbage collection runs * occurred. | *{@code 0:34503,1:45350,2:11281,3:8088,4:43,5:8} | *23 | *
art.gc.blocking-gc-count-rate-histogram | *Every 10 seconds, the blocking-gc-count-rate is computed as the number of * blocking garbage collection runs that have occurred over the last 10 * seconds. art.gc.blocking-gc-count-rate-histogram is a histogram of the * blocking-gc-count-rate samples taken since the process began. The histogram * can be used to identify instances of high rates of blocking garbage * collection runs. For example, a histogram of "0:99269,1:1,2:1" shows that * most of the time there are zero blocking garbage collection runs every 10 * seconds, but there was one 10-second interval in which one blocking garbage * collection run occurred, and there was one interval in which two blocking * garbage collection runs occurred. | *{@code 0:99269,1:1,2:1} | *23 | *
Note: this method directly retrieves memory information for the given process * from low-level data available to it. It may not be able to retrieve information about * some protected allocations, such as graphics. If you want to be sure you can see * all information about allocations by the process, use * {@link android.app.ActivityManager#getProcessMemoryInfo(int[])} instead.
*/ public static native void getMemoryInfo(MemoryInfo memoryInfo); /** * Note: currently only works when the requested pid has the same UID * as the caller. * @hide */ @UnsupportedAppUsage public static native void getMemoryInfo(int pid, MemoryInfo memoryInfo); /** * Retrieves the PSS memory used by the process as given by the * smaps. */ public static native long getPss(); /** * Retrieves the PSS memory used by the process as given by the smaps. Optionally supply a long * array of up to 3 entries to also receive (up to 3 values in order): the Uss and SwapPss and * Rss (only filled in as of {@link android.os.Build.VERSION_CODES#P}) of the process, and * another array to also retrieve the separate memtrack size. * @hide */ public static native long getPss(int pid, long[] outUssSwapPssRss, long[] outMemtrack); /** @hide */ public static final int MEMINFO_TOTAL = 0; /** @hide */ public static final int MEMINFO_FREE = 1; /** @hide */ public static final int MEMINFO_BUFFERS = 2; /** @hide */ public static final int MEMINFO_CACHED = 3; /** @hide */ public static final int MEMINFO_SHMEM = 4; /** @hide */ public static final int MEMINFO_SLAB = 5; /** @hide */ public static final int MEMINFO_SLAB_RECLAIMABLE = 6; /** @hide */ public static final int MEMINFO_SLAB_UNRECLAIMABLE = 7; /** @hide */ public static final int MEMINFO_SWAP_TOTAL = 8; /** @hide */ public static final int MEMINFO_SWAP_FREE = 9; /** @hide */ public static final int MEMINFO_ZRAM_TOTAL = 10; /** @hide */ public static final int MEMINFO_MAPPED = 11; /** @hide */ public static final int MEMINFO_VM_ALLOC_USED = 12; /** @hide */ public static final int MEMINFO_PAGE_TABLES = 13; /** @hide */ public static final int MEMINFO_KERNEL_STACK = 14; /** @hide */ public static final int MEMINFO_COUNT = 15; /** * Retrieves /proc/meminfo. outSizes is filled with fields * as defined by MEMINFO_* offsets. * @hide */ @UnsupportedAppUsage public static native void getMemInfo(long[] outSizes); /** * Establish an object allocation limit in the current thread. * This feature was never enabled in release builds. The * allocation limits feature was removed in Honeycomb. This * method exists for compatibility and always returns -1 and has * no effect. * * @deprecated This method is now obsolete. */ @Deprecated public static int setAllocationLimit(int limit) { return -1; } /** * Establish a global object allocation limit. This feature was * never enabled in release builds. The allocation limits feature * was removed in Honeycomb. This method exists for compatibility * and always returns -1 and has no effect. * * @deprecated This method is now obsolete. */ @Deprecated public static int setGlobalAllocationLimit(int limit) { return -1; } /** * Dump a list of all currently loaded class to the log file. * * @param flags See constants above. */ public static void printLoadedClasses(int flags) { VMDebug.printLoadedClasses(flags); } /** * Get the number of loaded classes. * @return the number of loaded classes. */ public static int getLoadedClassCount() { return VMDebug.getLoadedClassCount(); } /** * Dump "hprof" data to the specified file. This may cause a GC. * * @param fileName Full pathname of output file (e.g. "/sdcard/dump.hprof"). * @throws UnsupportedOperationException if the VM was built without * HPROF support. * @throws IOException if an error occurs while opening or writing files. */ public static void dumpHprofData(String fileName) throws IOException { VMDebug.dumpHprofData(fileName); } /** * Like dumpHprofData(String), but takes an already-opened * FileDescriptor to which the trace is written. The file name is also * supplied simply for logging. Makes a dup of the file descriptor. * * Primarily for use by the "am" shell command. * * @hide */ public static void dumpHprofData(String fileName, FileDescriptor fd) throws IOException { VMDebug.dumpHprofData(fileName, fd); } /** * Collect "hprof" and send it to DDMS. This may cause a GC. * * @throws UnsupportedOperationException if the VM was built without * HPROF support. * @hide */ public static void dumpHprofDataDdms() { VMDebug.dumpHprofDataDdms(); } /** * Writes native heap data to the specified file descriptor. * * @hide */ @UnsupportedAppUsage public static native void dumpNativeHeap(FileDescriptor fd); /** * Writes malloc info data to the specified file descriptor. * * @hide */ public static native void dumpNativeMallocInfo(FileDescriptor fd); /** * Returns a count of the extant instances of a class. * * @hide */ @UnsupportedAppUsage public static long countInstancesOfClass(Class cls) { return VMDebug.countInstancesOfClass(cls, true); } /** * Returns the number of sent transactions from this process. * @return The number of sent transactions or -1 if it could not read t. */ public static native int getBinderSentTransactions(); /** * Returns the number of received transactions from the binder driver. * @return The number of received transactions or -1 if it could not read the stats. */ public static native int getBinderReceivedTransactions(); /** * Returns the number of active local Binder objects that exist in the * current process. */ public static final native int getBinderLocalObjectCount(); /** * Returns the number of references to remote proxy Binder objects that * exist in the current process. */ public static final native int getBinderProxyObjectCount(); /** * Returns the number of death notification links to Binder objects that * exist in the current process. */ public static final native int getBinderDeathObjectCount(); /** * Primes the register map cache. * * Only works for classes in the bootstrap class loader. Does not * cause classes to be loaded if they're not already present. * * The classAndMethodDesc argument is a concatentation of the VM-internal * class descriptor, method name, and method descriptor. Examples: * Landroid/os/Looper;.loop:()V * Landroid/app/ActivityThread;.main:([Ljava/lang/String;)V * * @param classAndMethodDesc the method to prepare * * @hide */ public static final boolean cacheRegisterMap(String classAndMethodDesc) { return VMDebug.cacheRegisterMap(classAndMethodDesc); } /** * Dumps the contents of VM reference tables (e.g. JNI locals and * globals) to the log file. * * @hide */ @UnsupportedAppUsage public static final void dumpReferenceTables() { VMDebug.dumpReferenceTables(); } /** * API for gathering and querying instruction counts. * * Example usage: ** Debug.InstructionCount icount = new Debug.InstructionCount(); * icount.resetAndStart(); * [... do lots of stuff ...] * if (icount.collect()) { * System.out.println("Total instructions executed: " * + icount.globalTotal()); * System.out.println("Method invocations: " * + icount.globalMethodInvocations()); * } ** * @deprecated Instruction counting is no longer supported. */ @Deprecated public static class InstructionCount { public InstructionCount() { } /** * Reset counters and ensure counts are running. Counts may * have already been running. * * @return true if counting was started */ public boolean resetAndStart() { return false; } /** * Collect instruction counts. May or may not stop the * counting process. */ public boolean collect() { return false; } /** * Return the total number of instructions executed globally (i.e. in * all threads). */ public int globalTotal() { return 0; } /** * Return the total number of method-invocation instructions * executed globally. */ public int globalMethodInvocations() { return 0; } } /** * A Map of typed debug properties. */ private static final TypedProperties debugProperties; /* * Load the debug properties from the standard files into debugProperties. */ static { if (false) { final String TAG = "DebugProperties"; final String[] files = { "/system/debug.prop", "/debug.prop", "/data/debug.prop" }; final TypedProperties tp = new TypedProperties(); // Read the properties from each of the files, if present. for (String file : files) { Reader r; try { r = new FileReader(file); } catch (FileNotFoundException ex) { // It's ok if a file is missing. continue; } try { tp.load(r); } catch (Exception ex) { throw new RuntimeException("Problem loading " + file, ex); } finally { try { r.close(); } catch (IOException ex) { // Ignore this error. } } } debugProperties = tp.isEmpty() ? null : tp; } else { debugProperties = null; } } /** * Returns true if the type of the field matches the specified class. * Handles the case where the class is, e.g., java.lang.Boolean, but * the field is of the primitive "boolean" type. Also handles all of * the java.lang.Number subclasses. */ private static boolean fieldTypeMatches(Field field, Class> cl) { Class> fieldClass = field.getType(); if (fieldClass == cl) { return true; } Field primitiveTypeField; try { /* All of the classes we care about (Boolean, Integer, etc.) * have a Class field called "TYPE" that points to the corresponding * primitive class. */ primitiveTypeField = cl.getField("TYPE"); } catch (NoSuchFieldException ex) { return false; } try { return fieldClass == (Class>) primitiveTypeField.get(null); } catch (IllegalAccessException ex) { return false; } } /** * Looks up the property that corresponds to the field, and sets the field's value * if the types match. */ private static void modifyFieldIfSet(final Field field, final TypedProperties properties, final String propertyName) { if (field.getType() == java.lang.String.class) { int stringInfo = properties.getStringInfo(propertyName); switch (stringInfo) { case TypedProperties.STRING_SET: // Handle as usual below. break; case TypedProperties.STRING_NULL: try { field.set(null, null); // null object for static fields; null string } catch (IllegalAccessException ex) { throw new IllegalArgumentException( "Cannot set field for " + propertyName, ex); } return; case TypedProperties.STRING_NOT_SET: return; case TypedProperties.STRING_TYPE_MISMATCH: throw new IllegalArgumentException( "Type of " + propertyName + " " + " does not match field type (" + field.getType() + ")"); default: throw new IllegalStateException( "Unexpected getStringInfo(" + propertyName + ") return value " + stringInfo); } } Object value = properties.get(propertyName); if (value != null) { if (!fieldTypeMatches(field, value.getClass())) { throw new IllegalArgumentException( "Type of " + propertyName + " (" + value.getClass() + ") " + " does not match field type (" + field.getType() + ")"); } try { field.set(null, value); // null object for static fields } catch (IllegalAccessException ex) { throw new IllegalArgumentException( "Cannot set field for " + propertyName, ex); } } } /** * Equivalent to
setFieldsOn(cl, false)
.
*
* @see #setFieldsOn(Class, boolean)
*
* @hide
*/
public static void setFieldsOn(Class> cl) {
setFieldsOn(cl, false);
}
/**
* Reflectively sets static fields of a class based on internal debugging
* properties. This method is a no-op if false is
* false.
* * NOTE TO APPLICATION DEVELOPERS: false will * always be false in release builds. This API is typically only useful * for platform developers. *
* Class setup: define a class whose only fields are non-final, static * primitive types (except for "char") or Strings. In a static block * after the field definitions/initializations, pass the class to * this method, Debug.setFieldsOn(). Example: ** package com.example; * * import android.os.Debug; * * public class MyDebugVars { * public static String s = "a string"; * public static String s2 = "second string"; * public static String ns = null; * public static boolean b = false; * public static int i = 5; * @Debug.DebugProperty * public static float f = 0.1f; * @@Debug.DebugProperty * public static double d = 0.5d; * * // This MUST appear AFTER all fields are defined and initialized! * static { * // Sets all the fields * Debug.setFieldsOn(MyDebugVars.class); * * // Sets only the fields annotated with @Debug.DebugProperty * // Debug.setFieldsOn(MyDebugVars.class, true); * } * } ** setFieldsOn() may override the value of any field in the class based * on internal properties that are fixed at boot time. *
* These properties are only set during platform debugging, and are not * meant to be used as a general-purpose properties store. * * {@hide} * * @param cl The class to (possibly) modify * @param partial If false, sets all static fields, otherwise, only set * fields with the {@link android.os.Debug.DebugProperty} * annotation * @throws IllegalArgumentException if any fields are final or non-static, * or if the type of the field does not match the type of * the internal debugging property value. */ public static void setFieldsOn(Class> cl, boolean partial) { if (false) { if (debugProperties != null) { /* Only look for fields declared directly by the class, * so we don't mysteriously change static fields in superclasses. */ for (Field field : cl.getDeclaredFields()) { if (!partial || field.getAnnotation(DebugProperty.class) != null) { final String propertyName = cl.getName() + "." + field.getName(); boolean isStatic = Modifier.isStatic(field.getModifiers()); boolean isFinal = Modifier.isFinal(field.getModifiers()); if (!isStatic || isFinal) { throw new IllegalArgumentException(propertyName + " must be static and non-final"); } modifyFieldIfSet(field, debugProperties, propertyName); } } } } else { Log.wtf(TAG, "setFieldsOn(" + (cl == null ? "null" : cl.getName()) + ") called in non-DEBUG build"); } } /** * Annotation to put on fields you want to set with * {@link Debug#setFieldsOn(Class, boolean)}. * * @hide */ @Target({ ElementType.FIELD }) @Retention(RetentionPolicy.RUNTIME) public @interface DebugProperty { } /** * Get a debugging dump of a system service by name. * *
Most services require the caller to hold android.permission.DUMP.
*
* @param name of the service to dump
* @param fd to write dump output to (usually an output log file)
* @param args to pass to the service's dump method, may be null
* @return true if the service was dumped successfully, false if
* the service could not be found or had an error while dumping
*/
public static boolean dumpService(String name, FileDescriptor fd, String[] args) {
IBinder service = ServiceManager.getService(name);
if (service == null) {
Log.e(TAG, "Can't find service to dump: " + name);
return false;
}
try {
service.dump(fd, args);
return true;
} catch (RemoteException e) {
Log.e(TAG, "Can't dump service: " + name, e);
return false;
}
}
/**
* Append the Java stack traces of a given native process to a specified file.
*
* @param pid pid to dump.
* @param file path of file to append dump to.
* @param timeoutSecs time to wait in seconds, or 0 to wait forever.
* @hide
*/
public static native boolean dumpJavaBacktraceToFileTimeout(int pid, String file,
int timeoutSecs);
/**
* Append the native stack traces of a given process to a specified file.
*
* @param pid pid to dump.
* @param file path of file to append dump to.
* @param timeoutSecs time to wait in seconds, or 0 to wait forever.
* @hide
*/
public static native boolean dumpNativeBacktraceToFileTimeout(int pid, String file,
int timeoutSecs);
/**
* Get description of unreachable native memory.
* @param limit the number of leaks to provide info on, 0 to only get a summary.
* @param contents true to include a hex dump of the contents of unreachable memory.
* @return the String containing a description of unreachable memory.
* @hide */
public static native String getUnreachableMemory(int limit, boolean contents);
/**
* Return a String describing the calling method and location at a particular stack depth.
* @param callStack the Thread stack
* @param depth the depth of stack to return information for.
* @return the String describing the caller at that depth.
*/
private static String getCaller(StackTraceElement callStack[], int depth) {
// callStack[4] is the caller of the method that called getCallers()
if (4 + depth >= callStack.length) {
return "
* Note: agents may only be attached to debuggable apps. Otherwise, this function will
* throw a SecurityException.
*
* @param library the library containing the agent.
* @param options the options passed to the agent.
* @param classLoader the classloader determining the library search path.
*
* @throws IOException if the agent could not be attached.
* @throws SecurityException if the app is not debuggable.
*/
public static void attachJvmtiAgent(@NonNull String library, @Nullable String options,
@Nullable ClassLoader classLoader) throws IOException {
Preconditions.checkNotNull(library);
Preconditions.checkArgument(!library.contains("="));
if (options == null) {
VMDebug.attachAgent(library, classLoader);
} else {
VMDebug.attachAgent(library + "=" + options, classLoader);
}
}
/**
* Return the current free ZRAM usage in kilobytes.
*
* @hide
*/
public static native long getZramFreeKb();
}