1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * Copyright (c) 1995, 2006, Oracle and/or its affiliates. All rights reserved. 4 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 5 * 6 * This code is free software; you can redistribute it and/or modify it 7 * under the terms of the GNU General Public License version 2 only, as 8 * published by the Free Software Foundation. Oracle designates this 9 * particular file as subject to the "Classpath" exception as provided 10 * by Oracle in the LICENSE file that accompanied this code. 11 * 12 * This code is distributed in the hope that it will be useful, but WITHOUT 13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 15 * version 2 for more details (a copy is included in the LICENSE file that 16 * accompanied this code). 17 * 18 * You should have received a copy of the GNU General Public License version 19 * 2 along with this work; if not, write to the Free Software Foundation, 20 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 21 * 22 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 23 * or visit www.oracle.com if you need additional information or have any 24 * questions. 25 */ 26 27 package java.lang; 28 29 import java.io.*; 30 import java.util.StringTokenizer; 31 import sun.reflect.CallerSensitive; 32 import java.lang.ref.FinalizerReference; 33 import java.util.ArrayList; 34 import java.util.List; 35 import dalvik.system.BaseDexClassLoader; 36 import dalvik.system.VMDebug; 37 import dalvik.system.VMStack; 38 import dalvik.system.VMRuntime; 39 import libcore.io.IoUtils; 40 import libcore.io.Libcore; 41 import libcore.util.EmptyArray; 42 import static android.system.OsConstants._SC_NPROCESSORS_CONF; 43 44 /** 45 * Every Java application has a single instance of class 46 * <code>Runtime</code> that allows the application to interface with 47 * the environment in which the application is running. The current 48 * runtime can be obtained from the <code>getRuntime</code> method. 49 * <p> 50 * An application cannot create its own instance of this class. 51 * 52 * @author unascribed 53 * @see java.lang.Runtime#getRuntime() 54 * @since JDK1.0 55 */ 56 57 public class Runtime { 58 private static Runtime currentRuntime = new Runtime(); 59 60 /** 61 * Holds the list of threads to run when the VM terminates 62 */ 63 private List<Thread> shutdownHooks = new ArrayList<Thread>(); 64 65 /** 66 * Reflects whether finalization should be run for all objects 67 * when the VM terminates. 68 */ 69 private static boolean finalizeOnExit; 70 71 /** 72 * Reflects whether we are already shutting down the VM. 73 */ 74 private boolean shuttingDown; 75 76 /** 77 * Reflects whether we are tracing method calls. 78 */ 79 private boolean tracingMethods; 80 nativeExit(int code)81 private static native void nativeExit(int code); 82 83 /** 84 * Returns the runtime object associated with the current Java application. 85 * Most of the methods of class <code>Runtime</code> are instance 86 * methods and must be invoked with respect to the current runtime object. 87 * 88 * @return the <code>Runtime</code> object associated with the current 89 * Java application. 90 */ getRuntime()91 public static Runtime getRuntime() { 92 return currentRuntime; 93 } 94 95 /** Don't let anyone else instantiate this class */ Runtime()96 private Runtime() {} 97 98 /** 99 * Terminates the currently running Java virtual machine by initiating its 100 * shutdown sequence. This method never returns normally. The argument 101 * serves as a status code; by convention, a nonzero status code indicates 102 * abnormal termination. 103 * 104 * <p> The virtual machine's shutdown sequence consists of two phases. In 105 * the first phase all registered {@link #addShutdownHook shutdown hooks}, 106 * if any, are started in some unspecified order and allowed to run 107 * concurrently until they finish. In the second phase all uninvoked 108 * finalizers are run if {@link #runFinalizersOnExit finalization-on-exit} 109 * has been enabled. Once this is done the virtual machine {@link #halt 110 * halts}. 111 * 112 * <p> If this method is invoked after the virtual machine has begun its 113 * shutdown sequence then if shutdown hooks are being run this method will 114 * block indefinitely. If shutdown hooks have already been run and on-exit 115 * finalization has been enabled then this method halts the virtual machine 116 * with the given status code if the status is nonzero; otherwise, it 117 * blocks indefinitely. 118 * 119 * <p> The <tt>{@link System#exit(int) System.exit}</tt> method is the 120 * conventional and convenient means of invoking this method. <p> 121 * 122 * @param status 123 * Termination status. By convention, a nonzero status code 124 * indicates abnormal termination. 125 * 126 * @throws SecurityException 127 * If a security manager is present and its <tt>{@link 128 * SecurityManager#checkExit checkExit}</tt> method does not permit 129 * exiting with the specified status 130 * 131 * @see java.lang.SecurityException 132 * @see java.lang.SecurityManager#checkExit(int) 133 * @see #addShutdownHook 134 * @see #removeShutdownHook 135 * @see #runFinalizersOnExit 136 * @see #halt(int) 137 */ exit(int status)138 public void exit(int status) { 139 // Make sure we don't try this several times 140 synchronized(this) { 141 if (!shuttingDown) { 142 shuttingDown = true; 143 144 Thread[] hooks; 145 synchronized (shutdownHooks) { 146 // create a copy of the hooks 147 hooks = new Thread[shutdownHooks.size()]; 148 shutdownHooks.toArray(hooks); 149 } 150 151 // Start all shutdown hooks concurrently 152 for (Thread hook : hooks) { 153 hook.start(); 154 } 155 156 // Wait for all shutdown hooks to finish 157 for (Thread hook : hooks) { 158 try { 159 hook.join(); 160 } catch (InterruptedException ex) { 161 // Ignore, since we are at VM shutdown. 162 } 163 } 164 165 // Ensure finalization on exit, if requested 166 if (finalizeOnExit) { 167 runFinalization(); 168 } 169 170 // Get out of here finally... 171 nativeExit(status); 172 } 173 } 174 } 175 176 /** 177 * Registers a new virtual-machine shutdown hook. 178 * 179 * <p> The Java virtual machine <i>shuts down</i> in response to two kinds 180 * of events: 181 * 182 * <ul> 183 * 184 * <p> <li> The program <i>exits</i> normally, when the last non-daemon 185 * thread exits or when the <tt>{@link #exit exit}</tt> (equivalently, 186 * <tt>{@link System#exit(int) System.exit}</tt>) method is invoked, or 187 * 188 * <p> <li> The virtual machine is <i>terminated</i> in response to a 189 * user interrupt, such as typing <tt>^C</tt>, or a system-wide event, 190 * such as user logoff or system shutdown. 191 * 192 * </ul> 193 * 194 * <p> A <i>shutdown hook</i> is simply an initialized but unstarted 195 * thread. When the virtual machine begins its shutdown sequence it will 196 * start all registered shutdown hooks in some unspecified order and let 197 * them run concurrently. When all the hooks have finished it will then 198 * run all uninvoked finalizers if finalization-on-exit has been enabled. 199 * Finally, the virtual machine will halt. Note that daemon threads will 200 * continue to run during the shutdown sequence, as will non-daemon threads 201 * if shutdown was initiated by invoking the <tt>{@link #exit exit}</tt> 202 * method. 203 * 204 * <p> Once the shutdown sequence has begun it can be stopped only by 205 * invoking the <tt>{@link #halt halt}</tt> method, which forcibly 206 * terminates the virtual machine. 207 * 208 * <p> Once the shutdown sequence has begun it is impossible to register a 209 * new shutdown hook or de-register a previously-registered hook. 210 * Attempting either of these operations will cause an 211 * <tt>{@link IllegalStateException}</tt> to be thrown. 212 * 213 * <p> Shutdown hooks run at a delicate time in the life cycle of a virtual 214 * machine and should therefore be coded defensively. They should, in 215 * particular, be written to be thread-safe and to avoid deadlocks insofar 216 * as possible. They should also not rely blindly upon services that may 217 * have registered their own shutdown hooks and therefore may themselves in 218 * the process of shutting down. Attempts to use other thread-based 219 * services such as the AWT event-dispatch thread, for example, may lead to 220 * deadlocks. 221 * 222 * <p> Shutdown hooks should also finish their work quickly. When a 223 * program invokes <tt>{@link #exit exit}</tt> the expectation is 224 * that the virtual machine will promptly shut down and exit. When the 225 * virtual machine is terminated due to user logoff or system shutdown the 226 * underlying operating system may only allow a fixed amount of time in 227 * which to shut down and exit. It is therefore inadvisable to attempt any 228 * user interaction or to perform a long-running computation in a shutdown 229 * hook. 230 * 231 * <p> Uncaught exceptions are handled in shutdown hooks just as in any 232 * other thread, by invoking the <tt>{@link ThreadGroup#uncaughtException 233 * uncaughtException}</tt> method of the thread's <tt>{@link 234 * ThreadGroup}</tt> object. The default implementation of this method 235 * prints the exception's stack trace to <tt>{@link System#err}</tt> and 236 * terminates the thread; it does not cause the virtual machine to exit or 237 * halt. 238 * 239 * <p> In rare circumstances the virtual machine may <i>abort</i>, that is, 240 * stop running without shutting down cleanly. This occurs when the 241 * virtual machine is terminated externally, for example with the 242 * <tt>SIGKILL</tt> signal on Unix or the <tt>TerminateProcess</tt> call on 243 * Microsoft Windows. The virtual machine may also abort if a native 244 * method goes awry by, for example, corrupting internal data structures or 245 * attempting to access nonexistent memory. If the virtual machine aborts 246 * then no guarantee can be made about whether or not any shutdown hooks 247 * will be run. <p> 248 * 249 * @param hook 250 * An initialized but unstarted <tt>{@link Thread}</tt> object 251 * 252 * @throws IllegalArgumentException 253 * If the specified hook has already been registered, 254 * or if it can be determined that the hook is already running or 255 * has already been run 256 * 257 * @throws IllegalStateException 258 * If the virtual machine is already in the process 259 * of shutting down 260 * 261 * @throws SecurityException 262 * If a security manager is present and it denies 263 * <tt>{@link RuntimePermission}("shutdownHooks")</tt> 264 * 265 * @see #removeShutdownHook 266 * @see #halt(int) 267 * @see #exit(int) 268 * @since 1.3 269 */ addShutdownHook(Thread hook)270 public void addShutdownHook(Thread hook) { 271 // Sanity checks 272 if (hook == null) { 273 throw new NullPointerException("hook == null"); 274 } 275 276 if (shuttingDown) { 277 throw new IllegalStateException("VM already shutting down"); 278 } 279 280 if (hook.started) { 281 throw new IllegalArgumentException("Hook has already been started"); 282 } 283 284 synchronized (shutdownHooks) { 285 if (shutdownHooks.contains(hook)) { 286 throw new IllegalArgumentException("Hook already registered."); 287 } 288 289 shutdownHooks.add(hook); 290 } 291 } 292 293 /** 294 * De-registers a previously-registered virtual-machine shutdown hook. <p> 295 * 296 * @param hook the hook to remove 297 * @return <tt>true</tt> if the specified hook had previously been 298 * registered and was successfully de-registered, <tt>false</tt> 299 * otherwise. 300 * 301 * @throws IllegalStateException 302 * If the virtual machine is already in the process of shutting 303 * down 304 * 305 * @throws SecurityException 306 * If a security manager is present and it denies 307 * <tt>{@link RuntimePermission}("shutdownHooks")</tt> 308 * 309 * @see #addShutdownHook 310 * @see #exit(int) 311 * @since 1.3 312 */ removeShutdownHook(Thread hook)313 public boolean removeShutdownHook(Thread hook) { 314 // Sanity checks 315 if (hook == null) { 316 throw new NullPointerException("hook == null"); 317 } 318 319 if (shuttingDown) { 320 throw new IllegalStateException("VM already shutting down"); 321 } 322 323 synchronized (shutdownHooks) { 324 return shutdownHooks.remove(hook); 325 } 326 } 327 328 /** 329 * Forcibly terminates the currently running Java virtual machine. This 330 * method never returns normally. 331 * 332 * <p> This method should be used with extreme caution. Unlike the 333 * <tt>{@link #exit exit}</tt> method, this method does not cause shutdown 334 * hooks to be started and does not run uninvoked finalizers if 335 * finalization-on-exit has been enabled. If the shutdown sequence has 336 * already been initiated then this method does not wait for any running 337 * shutdown hooks or finalizers to finish their work. <p> 338 * 339 * @param status 340 * Termination status. By convention, a nonzero status code 341 * indicates abnormal termination. If the <tt>{@link Runtime#exit 342 * exit}</tt> (equivalently, <tt>{@link System#exit(int) 343 * System.exit}</tt>) method has already been invoked then this 344 * status code will override the status code passed to that method. 345 * 346 * @throws SecurityException 347 * If a security manager is present and its <tt>{@link 348 * SecurityManager#checkExit checkExit}</tt> method does not permit 349 * an exit with the specified status 350 * 351 * @see #exit 352 * @see #addShutdownHook 353 * @see #removeShutdownHook 354 * @since 1.3 355 */ halt(int status)356 public void halt(int status) { 357 nativeExit(status); 358 } 359 360 /** 361 * Enable or disable finalization on exit; doing so specifies that the 362 * finalizers of all objects that have finalizers that have not yet been 363 * automatically invoked are to be run before the Java runtime exits. 364 * By default, finalization on exit is disabled. 365 * 366 * <p>If there is a security manager, 367 * its <code>checkExit</code> method is first called 368 * with 0 as its argument to ensure the exit is allowed. 369 * This could result in a SecurityException. 370 * 371 * @param value true to enable finalization on exit, false to disable 372 * @deprecated This method is inherently unsafe. It may result in 373 * finalizers being called on live objects while other threads are 374 * concurrently manipulating those objects, resulting in erratic 375 * behavior or deadlock. 376 * 377 * @throws SecurityException 378 * if a security manager exists and its <code>checkExit</code> 379 * method doesn't allow the exit. 380 * 381 * @see java.lang.Runtime#exit(int) 382 * @see java.lang.Runtime#gc() 383 * @see java.lang.SecurityManager#checkExit(int) 384 * @since JDK1.1 385 */ 386 @Deprecated runFinalizersOnExit(boolean value)387 public static void runFinalizersOnExit(boolean value) { 388 finalizeOnExit = value; 389 } 390 391 /** 392 * Executes the specified string command in a separate process. 393 * 394 * <p>This is a convenience method. An invocation of the form 395 * <tt>exec(command)</tt> 396 * behaves in exactly the same way as the invocation 397 * <tt>{@link #exec(String, String[], File) exec}(command, null, null)</tt>. 398 * 399 * @param command a specified system command. 400 * 401 * @return A new {@link Process} object for managing the subprocess 402 * 403 * @throws SecurityException 404 * If a security manager exists and its 405 * {@link SecurityManager#checkExec checkExec} 406 * method doesn't allow creation of the subprocess 407 * 408 * @throws IOException 409 * If an I/O error occurs 410 * 411 * @throws NullPointerException 412 * If <code>command</code> is <code>null</code> 413 * 414 * @throws IllegalArgumentException 415 * If <code>command</code> is empty 416 * 417 * @see #exec(String[], String[], File) 418 * @see ProcessBuilder 419 */ exec(String command)420 public Process exec(String command) throws IOException { 421 return exec(command, null, null); 422 } 423 424 /** 425 * Executes the specified string command in a separate process with the 426 * specified environment. 427 * 428 * <p>This is a convenience method. An invocation of the form 429 * <tt>exec(command, envp)</tt> 430 * behaves in exactly the same way as the invocation 431 * <tt>{@link #exec(String, String[], File) exec}(command, envp, null)</tt>. 432 * 433 * @param command a specified system command. 434 * 435 * @param envp array of strings, each element of which 436 * has environment variable settings in the format 437 * <i>name</i>=<i>value</i>, or 438 * <tt>null</tt> if the subprocess should inherit 439 * the environment of the current process. 440 * 441 * @return A new {@link Process} object for managing the subprocess 442 * 443 * @throws SecurityException 444 * If a security manager exists and its 445 * {@link SecurityManager#checkExec checkExec} 446 * method doesn't allow creation of the subprocess 447 * 448 * @throws IOException 449 * If an I/O error occurs 450 * 451 * @throws NullPointerException 452 * If <code>command</code> is <code>null</code>, 453 * or one of the elements of <code>envp</code> is <code>null</code> 454 * 455 * @throws IllegalArgumentException 456 * If <code>command</code> is empty 457 * 458 * @see #exec(String[], String[], File) 459 * @see ProcessBuilder 460 */ exec(String command, String[] envp)461 public Process exec(String command, String[] envp) throws IOException { 462 return exec(command, envp, null); 463 } 464 465 /** 466 * Executes the specified string command in a separate process with the 467 * specified environment and working directory. 468 * 469 * <p>This is a convenience method. An invocation of the form 470 * <tt>exec(command, envp, dir)</tt> 471 * behaves in exactly the same way as the invocation 472 * <tt>{@link #exec(String[], String[], File) exec}(cmdarray, envp, dir)</tt>, 473 * where <code>cmdarray</code> is an array of all the tokens in 474 * <code>command</code>. 475 * 476 * <p>More precisely, the <code>command</code> string is broken 477 * into tokens using a {@link StringTokenizer} created by the call 478 * <code>new {@link StringTokenizer}(command)</code> with no 479 * further modification of the character categories. The tokens 480 * produced by the tokenizer are then placed in the new string 481 * array <code>cmdarray</code>, in the same order. 482 * 483 * @param command a specified system command. 484 * 485 * @param envp array of strings, each element of which 486 * has environment variable settings in the format 487 * <i>name</i>=<i>value</i>, or 488 * <tt>null</tt> if the subprocess should inherit 489 * the environment of the current process. 490 * 491 * @param dir the working directory of the subprocess, or 492 * <tt>null</tt> if the subprocess should inherit 493 * the working directory of the current process. 494 * 495 * @return A new {@link Process} object for managing the subprocess 496 * 497 * @throws SecurityException 498 * If a security manager exists and its 499 * {@link SecurityManager#checkExec checkExec} 500 * method doesn't allow creation of the subprocess 501 * 502 * @throws IOException 503 * If an I/O error occurs 504 * 505 * @throws NullPointerException 506 * If <code>command</code> is <code>null</code>, 507 * or one of the elements of <code>envp</code> is <code>null</code> 508 * 509 * @throws IllegalArgumentException 510 * If <code>command</code> is empty 511 * 512 * @see ProcessBuilder 513 * @since 1.3 514 */ exec(String command, String[] envp, File dir)515 public Process exec(String command, String[] envp, File dir) 516 throws IOException { 517 if (command.length() == 0) 518 throw new IllegalArgumentException("Empty command"); 519 520 StringTokenizer st = new StringTokenizer(command); 521 String[] cmdarray = new String[st.countTokens()]; 522 for (int i = 0; st.hasMoreTokens(); i++) 523 cmdarray[i] = st.nextToken(); 524 return exec(cmdarray, envp, dir); 525 } 526 527 /** 528 * Executes the specified command and arguments in a separate process. 529 * 530 * <p>This is a convenience method. An invocation of the form 531 * <tt>exec(cmdarray)</tt> 532 * behaves in exactly the same way as the invocation 533 * <tt>{@link #exec(String[], String[], File) exec}(cmdarray, null, null)</tt>. 534 * 535 * @param cmdarray array containing the command to call and 536 * its arguments. 537 * 538 * @return A new {@link Process} object for managing the subprocess 539 * 540 * @throws SecurityException 541 * If a security manager exists and its 542 * {@link SecurityManager#checkExec checkExec} 543 * method doesn't allow creation of the subprocess 544 * 545 * @throws IOException 546 * If an I/O error occurs 547 * 548 * @throws NullPointerException 549 * If <code>cmdarray</code> is <code>null</code>, 550 * or one of the elements of <code>cmdarray</code> is <code>null</code> 551 * 552 * @throws IndexOutOfBoundsException 553 * If <code>cmdarray</code> is an empty array 554 * (has length <code>0</code>) 555 * 556 * @see ProcessBuilder 557 */ exec(String cmdarray[])558 public Process exec(String cmdarray[]) throws IOException { 559 return exec(cmdarray, null, null); 560 } 561 562 /** 563 * Executes the specified command and arguments in a separate process 564 * with the specified environment. 565 * 566 * <p>This is a convenience method. An invocation of the form 567 * <tt>exec(cmdarray, envp)</tt> 568 * behaves in exactly the same way as the invocation 569 * <tt>{@link #exec(String[], String[], File) exec}(cmdarray, envp, null)</tt>. 570 * 571 * @param cmdarray array containing the command to call and 572 * its arguments. 573 * 574 * @param envp array of strings, each element of which 575 * has environment variable settings in the format 576 * <i>name</i>=<i>value</i>, or 577 * <tt>null</tt> if the subprocess should inherit 578 * the environment of the current process. 579 * 580 * @return A new {@link Process} object for managing the subprocess 581 * 582 * @throws SecurityException 583 * If a security manager exists and its 584 * {@link SecurityManager#checkExec checkExec} 585 * method doesn't allow creation of the subprocess 586 * 587 * @throws IOException 588 * If an I/O error occurs 589 * 590 * @throws NullPointerException 591 * If <code>cmdarray</code> is <code>null</code>, 592 * or one of the elements of <code>cmdarray</code> is <code>null</code>, 593 * or one of the elements of <code>envp</code> is <code>null</code> 594 * 595 * @throws IndexOutOfBoundsException 596 * If <code>cmdarray</code> is an empty array 597 * (has length <code>0</code>) 598 * 599 * @see ProcessBuilder 600 */ exec(String[] cmdarray, String[] envp)601 public Process exec(String[] cmdarray, String[] envp) throws IOException { 602 return exec(cmdarray, envp, null); 603 } 604 605 606 /** 607 * Executes the specified command and arguments in a separate process with 608 * the specified environment and working directory. 609 * 610 * <p>Given an array of strings <code>cmdarray</code>, representing the 611 * tokens of a command line, and an array of strings <code>envp</code>, 612 * representing "environment" variable settings, this method creates 613 * a new process in which to execute the specified command. 614 * 615 * <p>This method checks that <code>cmdarray</code> is a valid operating 616 * system command. Which commands are valid is system-dependent, 617 * but at the very least the command must be a non-empty list of 618 * non-null strings. 619 * 620 * <p>If <tt>envp</tt> is <tt>null</tt>, the subprocess inherits the 621 * environment settings of the current process. 622 * 623 * <p>A minimal set of system dependent environment variables may 624 * be required to start a process on some operating systems. 625 * As a result, the subprocess may inherit additional environment variable 626 * settings beyond those in the specified environment. 627 * 628 * <p>{@link ProcessBuilder#start()} is now the preferred way to 629 * start a process with a modified environment. 630 * 631 * <p>The working directory of the new subprocess is specified by <tt>dir</tt>. 632 * If <tt>dir</tt> is <tt>null</tt>, the subprocess inherits the 633 * current working directory of the current process. 634 * 635 * <p>If a security manager exists, its 636 * {@link SecurityManager#checkExec checkExec} 637 * method is invoked with the first component of the array 638 * <code>cmdarray</code> as its argument. This may result in a 639 * {@link SecurityException} being thrown. 640 * 641 * <p>Starting an operating system process is highly system-dependent. 642 * Among the many things that can go wrong are: 643 * <ul> 644 * <li>The operating system program file was not found. 645 * <li>Access to the program file was denied. 646 * <li>The working directory does not exist. 647 * </ul> 648 * 649 * <p>In such cases an exception will be thrown. The exact nature 650 * of the exception is system-dependent, but it will always be a 651 * subclass of {@link IOException}. 652 * 653 * 654 * @param cmdarray array containing the command to call and 655 * its arguments. 656 * 657 * @param envp array of strings, each element of which 658 * has environment variable settings in the format 659 * <i>name</i>=<i>value</i>, or 660 * <tt>null</tt> if the subprocess should inherit 661 * the environment of the current process. 662 * 663 * @param dir the working directory of the subprocess, or 664 * <tt>null</tt> if the subprocess should inherit 665 * the working directory of the current process. 666 * 667 * @return A new {@link Process} object for managing the subprocess 668 * 669 * @throws SecurityException 670 * If a security manager exists and its 671 * {@link SecurityManager#checkExec checkExec} 672 * method doesn't allow creation of the subprocess 673 * 674 * @throws IOException 675 * If an I/O error occurs 676 * 677 * @throws NullPointerException 678 * If <code>cmdarray</code> is <code>null</code>, 679 * or one of the elements of <code>cmdarray</code> is <code>null</code>, 680 * or one of the elements of <code>envp</code> is <code>null</code> 681 * 682 * @throws IndexOutOfBoundsException 683 * If <code>cmdarray</code> is an empty array 684 * (has length <code>0</code>) 685 * 686 * @see ProcessBuilder 687 * @since 1.3 688 */ exec(String[] cmdarray, String[] envp, File dir)689 public Process exec(String[] cmdarray, String[] envp, File dir) 690 throws IOException { 691 return new ProcessBuilder(cmdarray) 692 .environment(envp) 693 .directory(dir) 694 .start(); 695 } 696 697 /** 698 * Returns the number of processors available to the Java virtual machine. 699 * 700 * <p> This value may change during a particular invocation of the virtual 701 * machine. Applications that are sensitive to the number of available 702 * processors should therefore occasionally poll this property and adjust 703 * their resource usage appropriately. </p> 704 * 705 * @return the maximum number of processors available to the virtual 706 * machine; never smaller than one 707 * @since 1.4 708 */ availableProcessors()709 public int availableProcessors() { 710 return (int) Libcore.os.sysconf(_SC_NPROCESSORS_CONF); 711 } 712 713 /** 714 * Returns the amount of free memory in the Java Virtual Machine. 715 * Calling the 716 * <code>gc</code> method may result in increasing the value returned 717 * by <code>freeMemory.</code> 718 * 719 * @return an approximation to the total amount of memory currently 720 * available for future allocated objects, measured in bytes. 721 */ freeMemory()722 public native long freeMemory(); 723 724 /** 725 * Returns the total amount of memory in the Java virtual machine. 726 * The value returned by this method may vary over time, depending on 727 * the host environment. 728 * <p> 729 * Note that the amount of memory required to hold an object of any 730 * given type may be implementation-dependent. 731 * 732 * @return the total amount of memory currently available for current 733 * and future objects, measured in bytes. 734 */ totalMemory()735 public native long totalMemory(); 736 737 /** 738 * Returns the maximum amount of memory that the Java virtual machine will 739 * attempt to use. If there is no inherent limit then the value {@link 740 * java.lang.Long#MAX_VALUE} will be returned. </p> 741 * 742 * @return the maximum amount of memory that the virtual machine will 743 * attempt to use, measured in bytes 744 * @since 1.4 745 */ maxMemory()746 public native long maxMemory(); 747 748 /** 749 * Runs the garbage collector. 750 * Calling this method suggests that the Java virtual machine expend 751 * effort toward recycling unused objects in order to make the memory 752 * they currently occupy available for quick reuse. When control 753 * returns from the method call, the virtual machine has made 754 * its best effort to recycle all discarded objects. 755 * <p> 756 * The name <code>gc</code> stands for "garbage 757 * collector". The virtual machine performs this recycling 758 * process automatically as needed, in a separate thread, even if the 759 * <code>gc</code> method is not invoked explicitly. 760 * <p> 761 * The method {@link System#gc()} is the conventional and convenient 762 * means of invoking this method. 763 */ gc()764 public native void gc(); 765 766 /* Wormhole for calling java.lang.ref.Finalizer.runFinalization */ runFinalization0()767 private static native void runFinalization0(); 768 769 /** 770 * Runs the finalization methods of any objects pending finalization. 771 * Calling this method suggests that the Java virtual machine expend 772 * effort toward running the <code>finalize</code> methods of objects 773 * that have been found to be discarded but whose <code>finalize</code> 774 * methods have not yet been run. When control returns from the 775 * method call, the virtual machine has made a best effort to 776 * complete all outstanding finalizations. 777 * <p> 778 * The virtual machine performs the finalization process 779 * automatically as needed, in a separate thread, if the 780 * <code>runFinalization</code> method is not invoked explicitly. 781 * <p> 782 * The method {@link System#runFinalization()} is the conventional 783 * and convenient means of invoking this method. 784 * 785 * @see java.lang.Object#finalize() 786 */ runFinalization()787 public void runFinalization() { 788 VMRuntime.runFinalization(0); 789 } 790 791 /** 792 * Enables/Disables tracing of instructions. 793 * If the <code>boolean</code> argument is <code>true</code>, this 794 * method suggests that the Java virtual machine emit debugging 795 * information for each instruction in the virtual machine as it 796 * is executed. The format of this information, and the file or other 797 * output stream to which it is emitted, depends on the host environment. 798 * The virtual machine may ignore this request if it does not support 799 * this feature. The destination of the trace output is system 800 * dependent. 801 * <p> 802 * If the <code>boolean</code> argument is <code>false</code>, this 803 * method causes the virtual machine to stop performing the 804 * detailed instruction trace it is performing. 805 * 806 * @param enable <code>true</code> to enable instruction tracing; 807 * <code>false</code> to disable this feature. 808 */ 809 // Android changed - param name s/on/enable traceInstructions(boolean enable)810 public void traceInstructions(boolean enable) { 811 } 812 813 /** 814 * Enables/Disables tracing of method calls. 815 * If the <code>boolean</code> argument is <code>true</code>, this 816 * method suggests that the Java virtual machine emit debugging 817 * information for each method in the virtual machine as it is 818 * called. The format of this information, and the file or other output 819 * stream to which it is emitted, depends on the host environment. The 820 * virtual machine may ignore this request if it does not support 821 * this feature. 822 * <p> 823 * Calling this method with argument false suggests that the 824 * virtual machine cease emitting per-call debugging information. 825 * 826 * @param enable <code>true</code> to enable instruction tracing; 827 * <code>false</code> to disable this feature. 828 */ 829 // Android changed - param name s/on/enable traceMethodCalls(boolean enable)830 public void traceMethodCalls(boolean enable) { 831 if (enable != tracingMethods) { 832 if (enable) { 833 VMDebug.startMethodTracing(); 834 } else { 835 VMDebug.stopMethodTracing(); 836 } 837 tracingMethods = enable; 838 } 839 } 840 841 /** 842 * Loads the specified filename as a dynamic library. The filename 843 * argument must be a complete path name, 844 * (for example 845 * <code>Runtime.getRuntime().load("/home/avh/lib/libX11.so");</code>). 846 * <p> 847 * First, if there is a security manager, its <code>checkLink</code> 848 * method is called with the <code>filename</code> as its argument. 849 * This may result in a security exception. 850 * <p> 851 * This is similar to the method {@link #loadLibrary(String)}, but it 852 * accepts a general file name as an argument rather than just a library 853 * name, allowing any file of native code to be loaded. 854 * <p> 855 * The method {@link System#load(String)} is the conventional and 856 * convenient means of invoking this method. 857 * 858 * @param filename the file to load. 859 * @exception SecurityException if a security manager exists and its 860 * <code>checkLink</code> method doesn't allow 861 * loading of the specified dynamic library 862 * @exception UnsatisfiedLinkError if the file does not exist. 863 * @exception NullPointerException if <code>filename</code> is 864 * <code>null</code> 865 * @see java.lang.Runtime#getRuntime() 866 * @see java.lang.SecurityException 867 * @see java.lang.SecurityManager#checkLink(java.lang.String) 868 */ 869 @CallerSensitive load(String filename)870 public void load(String filename) { 871 load0(VMStack.getStackClass1(), filename); 872 } 873 874 /** Check target sdk, if it's higher than N, we throw an UnsupportedOperationException */ checkTargetSdkVersionForLoad(String methodName)875 private void checkTargetSdkVersionForLoad(String methodName) { 876 final int targetSdkVersion = VMRuntime.getRuntime().getTargetSdkVersion(); 877 if (targetSdkVersion > 24) { 878 throw new UnsupportedOperationException(methodName + " is not supported on SDK " + 879 targetSdkVersion); 880 } 881 } 882 883 // Fixes b/25859957 regression. Depending on private methods is bad, mkay. load(String absolutePath, ClassLoader loader)884 void load(String absolutePath, ClassLoader loader) { 885 checkTargetSdkVersionForLoad("java.lang.Runtime#load(String, ClassLoader)"); 886 887 java.lang.System.logE("java.lang.Runtime#load(String, ClassLoader)" + 888 " is private and will be removed in a future Android release"); 889 if (absolutePath == null) { 890 throw new NullPointerException("absolutePath == null"); 891 } 892 String error = doLoad(absolutePath, loader); 893 if (error != null) { 894 throw new UnsatisfiedLinkError(error); 895 } 896 } 897 load0(Class fromClass, String filename)898 synchronized void load0(Class fromClass, String filename) { 899 if (!(new File(filename).isAbsolute())) { 900 throw new UnsatisfiedLinkError( 901 "Expecting an absolute path of the library: " + filename); 902 } 903 if (filename == null) { 904 throw new NullPointerException("filename == null"); 905 } 906 String error = doLoad(filename, fromClass.getClassLoader()); 907 if (error != null) { 908 throw new UnsatisfiedLinkError(error); 909 } 910 } 911 912 /** 913 * Loads the dynamic library with the specified library name. 914 * A file containing native code is loaded from the local file system 915 * from a place where library files are conventionally obtained. The 916 * details of this process are implementation-dependent. The 917 * mapping from a library name to a specific filename is done in a 918 * system-specific manner. 919 * <p> 920 * First, if there is a security manager, its <code>checkLink</code> 921 * method is called with the <code>libname</code> as its argument. 922 * This may result in a security exception. 923 * <p> 924 * The method {@link System#loadLibrary(String)} is the conventional 925 * and convenient means of invoking this method. If native 926 * methods are to be used in the implementation of a class, a standard 927 * strategy is to put the native code in a library file (call it 928 * <code>LibFile</code>) and then to put a static initializer: 929 * <blockquote><pre> 930 * static { System.loadLibrary("LibFile"); } 931 * </pre></blockquote> 932 * within the class declaration. When the class is loaded and 933 * initialized, the necessary native code implementation for the native 934 * methods will then be loaded as well. 935 * <p> 936 * If this method is called more than once with the same library 937 * name, the second and subsequent calls are ignored. 938 * 939 * @param libname the name of the library. 940 * @exception SecurityException if a security manager exists and its 941 * <code>checkLink</code> method doesn't allow 942 * loading of the specified dynamic library 943 * @exception UnsatisfiedLinkError if the library does not exist. 944 * @exception NullPointerException if <code>libname</code> is 945 * <code>null</code> 946 * @see java.lang.SecurityException 947 * @see java.lang.SecurityManager#checkLink(java.lang.String) 948 */ 949 @CallerSensitive loadLibrary(String libname)950 public void loadLibrary(String libname) { 951 loadLibrary0(VMStack.getCallingClassLoader(), libname); 952 } 953 954 /** 955 * Temporarily preserved for backward compatibility. Applications call this 956 * method using reflection. 957 * 958 * **** THIS METHOD WILL BE REMOVED IN A FUTURE ANDROID VERSION **** 959 * 960 * http://b/26217329 961 * 962 * @hide 963 */ loadLibrary(String libname, ClassLoader classLoader)964 public void loadLibrary(String libname, ClassLoader classLoader) { 965 checkTargetSdkVersionForLoad("java.lang.Runtime#loadLibrary(String, ClassLoader)"); 966 java.lang.System.logE("java.lang.Runtime#loadLibrary(String, ClassLoader)" + 967 " is private and will be removed in a future Android release"); 968 loadLibrary0(classLoader, libname); 969 } 970 loadLibrary0(ClassLoader loader, String libname)971 synchronized void loadLibrary0(ClassLoader loader, String libname) { 972 if (libname.indexOf((int)File.separatorChar) != -1) { 973 throw new UnsatisfiedLinkError( 974 "Directory separator should not appear in library name: " + libname); 975 } 976 String libraryName = libname; 977 if (loader != null) { 978 String filename = loader.findLibrary(libraryName); 979 if (filename == null) { 980 // It's not necessarily true that the ClassLoader used 981 // System.mapLibraryName, but the default setup does, and it's 982 // misleading to say we didn't find "libMyLibrary.so" when we 983 // actually searched for "liblibMyLibrary.so.so". 984 throw new UnsatisfiedLinkError(loader + " couldn't find \"" + 985 System.mapLibraryName(libraryName) + "\""); 986 } 987 String error = doLoad(filename, loader); 988 if (error != null) { 989 throw new UnsatisfiedLinkError(error); 990 } 991 return; 992 } 993 994 String filename = System.mapLibraryName(libraryName); 995 List<String> candidates = new ArrayList<String>(); 996 String lastError = null; 997 for (String directory : getLibPaths()) { 998 String candidate = directory + filename; 999 candidates.add(candidate); 1000 1001 if (IoUtils.canOpenReadOnly(candidate)) { 1002 String error = doLoad(candidate, loader); 1003 if (error == null) { 1004 return; // We successfully loaded the library. Job done. 1005 } 1006 lastError = error; 1007 } 1008 } 1009 1010 if (lastError != null) { 1011 throw new UnsatisfiedLinkError(lastError); 1012 } 1013 throw new UnsatisfiedLinkError("Library " + libraryName + " not found; tried " + candidates); 1014 } 1015 1016 private volatile String[] mLibPaths = null; 1017 getLibPaths()1018 private String[] getLibPaths() { 1019 if (mLibPaths == null) { 1020 synchronized(this) { 1021 if (mLibPaths == null) { 1022 mLibPaths = initLibPaths(); 1023 } 1024 } 1025 } 1026 return mLibPaths; 1027 } 1028 initLibPaths()1029 private static String[] initLibPaths() { 1030 String javaLibraryPath = System.getProperty("java.library.path"); 1031 if (javaLibraryPath == null) { 1032 return EmptyArray.STRING; 1033 } 1034 String[] paths = javaLibraryPath.split(":"); 1035 // Add a '/' to the end of each directory so we don't have to do it every time. 1036 for (int i = 0; i < paths.length; ++i) { 1037 if (!paths[i].endsWith("/")) { 1038 paths[i] += "/"; 1039 } 1040 } 1041 return paths; 1042 } doLoad(String name, ClassLoader loader)1043 private String doLoad(String name, ClassLoader loader) { 1044 // Android apps are forked from the zygote, so they can't have a custom LD_LIBRARY_PATH, 1045 // which means that by default an app's shared library directory isn't on LD_LIBRARY_PATH. 1046 1047 // The PathClassLoader set up by frameworks/base knows the appropriate path, so we can load 1048 // libraries with no dependencies just fine, but an app that has multiple libraries that 1049 // depend on each other needed to load them in most-dependent-first order. 1050 1051 // We added API to Android's dynamic linker so we can update the library path used for 1052 // the currently-running process. We pull the desired path out of the ClassLoader here 1053 // and pass it to nativeLoad so that it can call the private dynamic linker API. 1054 1055 // We didn't just change frameworks/base to update the LD_LIBRARY_PATH once at the 1056 // beginning because multiple apks can run in the same process and third party code can 1057 // use its own BaseDexClassLoader. 1058 1059 // We didn't just add a dlopen_with_custom_LD_LIBRARY_PATH call because we wanted any 1060 // dlopen(3) calls made from a .so's JNI_OnLoad to work too. 1061 1062 // So, find out what the native library search path is for the ClassLoader in question... 1063 String librarySearchPath = null; 1064 if (loader != null && loader instanceof BaseDexClassLoader) { 1065 BaseDexClassLoader dexClassLoader = (BaseDexClassLoader) loader; 1066 librarySearchPath = dexClassLoader.getLdLibraryPath(); 1067 } 1068 // nativeLoad should be synchronized so there's only one LD_LIBRARY_PATH in use regardless 1069 // of how many ClassLoaders are in the system, but dalvik doesn't support synchronized 1070 // internal natives. 1071 synchronized (this) { 1072 return nativeLoad(name, loader, librarySearchPath); 1073 } 1074 } 1075 1076 // TODO: should be synchronized, but dalvik doesn't support synchronized internal natives. nativeLoad(String filename, ClassLoader loader, String librarySearchPath)1077 private static native String nativeLoad(String filename, ClassLoader loader, 1078 String librarySearchPath); 1079 1080 /** 1081 * Creates a localized version of an input stream. This method takes 1082 * an <code>InputStream</code> and returns an <code>InputStream</code> 1083 * equivalent to the argument in all respects except that it is 1084 * localized: as characters in the local character set are read from 1085 * the stream, they are automatically converted from the local 1086 * character set to Unicode. 1087 * <p> 1088 * If the argument is already a localized stream, it may be returned 1089 * as the result. 1090 * 1091 * @param in InputStream to localize 1092 * @return a localized input stream 1093 * @see java.io.InputStream 1094 * @see java.io.BufferedReader#BufferedReader(java.io.Reader) 1095 * @see java.io.InputStreamReader#InputStreamReader(java.io.InputStream) 1096 * @deprecated As of JDK 1.1, the preferred way to translate a byte 1097 * stream in the local encoding into a character stream in Unicode is via 1098 * the <code>InputStreamReader</code> and <code>BufferedReader</code> 1099 * classes. 1100 */ 1101 @Deprecated getLocalizedInputStream(InputStream in)1102 public InputStream getLocalizedInputStream(InputStream in) { 1103 return in; 1104 } 1105 1106 /** 1107 * Creates a localized version of an output stream. This method 1108 * takes an <code>OutputStream</code> and returns an 1109 * <code>OutputStream</code> equivalent to the argument in all respects 1110 * except that it is localized: as Unicode characters are written to 1111 * the stream, they are automatically converted to the local 1112 * character set. 1113 * <p> 1114 * If the argument is already a localized stream, it may be returned 1115 * as the result. 1116 * 1117 * @deprecated As of JDK 1.1, the preferred way to translate a 1118 * Unicode character stream into a byte stream in the local encoding is via 1119 * the <code>OutputStreamWriter</code>, <code>BufferedWriter</code>, and 1120 * <code>PrintWriter</code> classes. 1121 * 1122 * @param out OutputStream to localize 1123 * @return a localized output stream 1124 * @see java.io.OutputStream 1125 * @see java.io.BufferedWriter#BufferedWriter(java.io.Writer) 1126 * @see java.io.OutputStreamWriter#OutputStreamWriter(java.io.OutputStream) 1127 * @see java.io.PrintWriter#PrintWriter(java.io.OutputStream) 1128 */ 1129 @Deprecated getLocalizedOutputStream(OutputStream out)1130 public OutputStream getLocalizedOutputStream(OutputStream out) { 1131 return out; 1132 } 1133 1134 } 1135