<html> <head> <title>Controlling the Embedded VM</title> <link rel=stylesheet href="android.css"> </head> <body> <h1>Controlling the Embedded VM</h1> <ul> <li><a href="#overview">Overview</a> <li><a href="#checkjni">Extended JNI Checks</a> <li><a href="#assertions">Assertions</a> <li><a href="#verifier">Bytecode Verification and Optimization</a> <li><a href="#execmode">Execution Mode</a> <li><a href="#dp">Deadlock Prediction</a> <li><a href="#stackdump">Stack Dumps</a> </ul> <h2><a name="overview">Overview</a></h2> <p>The Dalvik VM supports a variety of command-line arguments (use <code>adb shell dalvikvm -help</code> to get a summary), but it's not possible to pass arbitrary arguments through the Android application runtime. It is, however, possible to affect the VM behavior through certain system properties. <p>For all of the features described below, you would set the system property with <code>setprop</code>, issuing a shell command on the device like this: <pre>adb shell setprop <name> <value></pre> <p>The Android runtime must be restarted before the changes will take effect (<code>adb shell stop; adb shell start</code>). This is because the settings are processed in the "zygote" process, which starts early and stays around "forever". <p>You could also add a line to <code>/data/local.prop</code> that looks like: <pre><name> = <value></pre> <p>Such changes will survive reboots, but will be removed by anything that wipes the data partition. (Hint: create a <code>local.prop</code> on your workstation, then <code>adb push local.prop /data</code> .) <h2><a name="checkjni">Extended JNI Checks</a></h2> <p>JNI, the Java Native Interface, provides a way for code written in the Java programming language interact with native (C/C++) code. The extended JNI checks will cause the system to run more slowly, but they can spot a variety of nasty bugs before they have a chance to cause problems. <p>There are two system properties that affect this feature, which is enabled with the <code>-Xcheck:jni</code> command-line argument. The first is <code>ro.kernel.android.checkjni</code>. This is set by the Android build system for development builds. (It may also be set by the Android emulator unless the <code>-nojni</code> flag is provided on the emulator command line.) Because this is an "ro." property, the value cannot be changed once the device has started. <p>To allow toggling of the CheckJNI flag, a second property, <code>dalvik.vm.checkjni</code>, is also checked. The value of this overrides the value from <code>ro.kernel.android.checkjni</code>. <p>If neither property is defined, or <code>dalvik.vm.checkjni</code> is set to <code>false</code>, the <code>-Xcheck:jni</code> flag is not passed in, and JNI checks will be disabled. <p>To enable JNI checking: <pre>adb shell setprop dalvik.vm.checkjni true</pre> <p>You can also pass JNI-checking options into the VM through a system property. The value set for <code>dalvik.vm.jniopts</code> will be passed in as the <code>-Xjniopts</code> argument. <p>For more information about JNI checks, see <a href="jni-tips.html">JNI Tips</a>. <h2><a name="assertions">Assertions</a></h2> <p>Dalvik VM supports the Java programming language "assert" statement. By default they are off, but the <code>dalvik.vm.enableassertions</code> property provides a way to set the value for a <code>-ea</code> argument. <p>The argument behaves the same as it does in other desktop VMs. You can provide a class name, a package name (followed by "..."), or the special value "all". <p>For example, this: <pre>adb shell setprop dalvik.vm.enableassertions all</pre> enables assertions in all non-system classes. <p>The system property is much more limited than the full command line. It is not possible to specify more than one <code>-ea</code> entry, and there is no way to specify a <code>-da</code> entry. There is presently no equivalent for <code>-esa</code>/<code>-dsa</code>. <h2><a name="verifier">Bytecode Verification and Optimization</a></h2> <p>The system tries to pre-verify all classes in a DEX file to reduce class load overhead, and performs a series of optimizations to improve runtime performance. Both of these are done by the <code>dexopt</code> command, either in the build system or by the installer. On a development device, <code>dexopt</code> may be run the first time a DEX file is used and whenever it or one of its dependencies is updated ("just-in-time" optimization and verification). <p>There are two command-line flags that control the just-in-time verification and optimization, <code>-Xverify</code> and <code>-Xdexopt</code>. The Android framework configures these based on the <code>dalvik.vm.dexopt-flags</code> property. <p>If you set: <pre>adb shell setprop dalvik.vm.dexopt-flags v=a,o=v</pre> then the framework will pass <code>-Xverify:all -Xdexopt:verified</code> to the VM. This enables verification, and only optimizes classes that successfully verified. This is the safest setting, and is the default. <p>You could also set <code>dalvik.vm.dexopt-flags</code> to <code>v=n</code> to have the framework pass <code>-Xverify:none -Xdexopt:verified</code> to disable verification. (We could pass in <code>-Xdexopt:all</code> to allow optimization, but that wouldn't necessarily optimize more of the code, since classes that fail verification may well be skipped by the optimizer for the same reasons.) Classes will not be verified by <code>dexopt</code>, and unverified code will be loaded and executed. <p>Enabling verification will make the <code>dexopt</code> command take significantly longer, because the verification process is fairly slow. Once the verified and optimized DEX files have been prepared, verification incurs no additional overhead except when loading classes that failed to pre-verify. <p>If your DEX files are processed with verification disabled, and you later turn the verifier on, application loading will be noticeably slower (perhaps 40% or more) as classes are verified on first use. <p>For best results you should force a re-dexopt of all DEX files when this property changes. You can do this with: <pre>adb shell "rm /data/dalvik-cache/*"</pre> This removes the cached versions of the DEX files. Remember to stop and restart the runtime (<code>adb shell stop; adb shell start</code>). <p>(Previous version of the runtime supported the boolean <code>dalvik.vm.verify-bytecode</code> property, but that has been superceded by <code>dalvik.vm.dexopt-flags</code>.)</p> <h2><a name="execmode">Execution Mode</a></h2> <p>The current implementation of the Dalvik VM includes three distinct interpreter cores. These are referred to as "fast", "portable", and "debug". The "fast" interpreter is optimized for the current platform, and might consist of hand-optimized assembly routines. In constrast, the "portable" interpreter is written in C and expected to run on a broad range of platforms. The "debug" interpreter is a variant of "portable" that includes support for profiling and single-stepping. <p>The VM allows you to choose between "fast" and "portable" with an extended form of the <code>-Xint</code> argument. The value of this argument can be set through the <code>dalvik.vm.execution-mode</code> system property. <p>To select the "portable" interpreter, you would use: <pre>adb shell setprop dalvik.vm.execution-mode int:portable</pre> If the property is not specified, the most appropriate interpreter will be selected automatically. At some point this mechanism may allow selection of other modes, such as JIT compilation. <p>Not all platforms have an optimized implementation. In such cases, the "fast" interpreter is generated as a series of C stubs, and the result will be slower than the "portable" version. (When we have optimized versions for all popular architectures the naming convention will be more accurate.) <p>If profiling is enabled or a debugger is attached, the VM switches to the "debug" interpreter. When profiling ends or the debugger disconnects, the original interpreter is resumed. (The "debug" interpreter is substantially slower, something to keep in mind when evaluating profiling data.) <h2><a name="dp">Deadlock Prediction</a></h2> <p>If the VM is built with <code>WITH_DEADLOCK_PREDICTION</code>, the deadlock predictor can be enabled with the <code>-Xdeadlockpredict</code> argument. (The output from <code>dalvikvm -help</code> will tell you if the VM was built appropriately -- look for <code>deadlock_prediction</code> on the <code>Configured with:</code> line.) This feature tells the VM to keep track of the order in which object monitor locks are acquired. If the program attempts to acquire a set of locks in a different order from what was seen earlier, the VM logs a warning and optionally throws an exception. <p>The command-line argument is set based on the <code>dalvik.vm.deadlock-predict</code> property. Valid values are <code>off</code> to disable it (default), <code>warn</code> to log the problem but continue executing, <code>err</code> to cause a <code>dalvik.system.PotentialDeadlockError</code> to be thrown from the <code>monitor-enter</code> instruction, and <code>abort</code> to have the entire VM abort. <p>You will usually want to use: <pre>adb shell setprop dalvik.vm.deadlock-predict err</pre> unless you are keeping an eye on the logs as they scroll by. <p>Please note that this feature is deadlock prediction, not deadlock detection -- in the current implementation, the computations are performed after the lock is acquired (this simplifies the code, reducing the overhead added to every mutex operation). You can spot a deadlock in a hung process by sending a <code>kill -3</code> and examining the stack trace written to the log. <p>This only takes monitors into account. Native mutexes and other resources can also be the cause of deadlocks, but will not be detected by this. <h2><a name="stackdump">Stack Dumps</a></h2> <p>Like other desktop VMs, when the Dalvik VM receives a SIGQUIT (Ctrl-\ or <code>kill -3</code>), it dumps stack traces for all threads. By default this goes to the Android log, but it can also be written to a file. <p>The <code>dalvik.vm.stack-trace-file</code> property allows you to specify the name of the file where the thread stack traces will be written. The file will be created (world writable) if it doesn't exist, and the new information will be appended to the end of the file. The filename is passed into the VM via the <code>-Xstacktracefile</code> argument. <p>For example: <pre>adb shell setprop dalvik.vm.stack-trace-file /tmp/stack-traces.txt</pre> <p>If the property is not defined, the VM will write the stack traces to the Android log when the signal arrives. <address>Copyright © 2008 The Android Open Source Project</address> </body></html>