1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * Copyright (c) 1995, 2011, 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.PrintStream; 30 import java.util.Arrays; 31 import sun.misc.VM; 32 33 /** 34 * A thread group represents a set of threads. In addition, a thread 35 * group can also include other thread groups. The thread groups form 36 * a tree in which every thread group except the initial thread group 37 * has a parent. 38 * <p> 39 * A thread is allowed to access information about its own thread 40 * group, but not to access information about its thread group's 41 * parent thread group or any other thread groups. 42 * 43 * @author unascribed 44 * @since JDK1.0 45 */ 46 /* The locking strategy for this code is to try to lock only one level of the 47 * tree wherever possible, but otherwise to lock from the bottom up. 48 * That is, from child thread groups to parents. 49 * This has the advantage of limiting the number of locks that need to be held 50 * and in particular avoids having to grab the lock for the root thread group, 51 * (or a global lock) which would be a source of contention on a 52 * multi-processor system with many thread groups. 53 * This policy often leads to taking a snapshot of the state of a thread group 54 * and working off of that snapshot, rather than holding the thread group locked 55 * while we work on the children. 56 */ 57 public 58 class ThreadGroup implements Thread.UncaughtExceptionHandler { 59 /* the runtime uses these directly; do not rename */ 60 static final ThreadGroup systemThreadGroup = new ThreadGroup(); 61 62 static final ThreadGroup mainThreadGroup = new ThreadGroup(systemThreadGroup, "main"); 63 64 private final ThreadGroup parent; 65 String name; 66 int maxPriority; 67 boolean destroyed; 68 boolean daemon; 69 boolean vmAllowSuspension; 70 71 int nUnstartedThreads = 0; 72 int nthreads; 73 Thread threads[]; 74 75 int ngroups; 76 ThreadGroup groups[]; 77 78 /** 79 * Creates an empty Thread group that is not in any Thread group. 80 * This method is used to create the system Thread group. 81 */ 82 private ThreadGroup() { // called from C code 83 this.name = "system"; 84 this.maxPriority = Thread.MAX_PRIORITY; 85 this.parent = null; 86 } 87 88 /** 89 * Constructs a new thread group. The parent of this new group is 90 * the thread group of the currently running thread. 91 * <p> 92 * The <code>checkAccess</code> method of the parent thread group is 93 * called with no arguments; this may result in a security exception. 94 * 95 * @param name the name of the new thread group. 96 * @exception SecurityException if the current thread cannot create a 97 * thread in the specified thread group. 98 * @see java.lang.ThreadGroup#checkAccess() 99 * @since JDK1.0 100 */ 101 public ThreadGroup(String name) { 102 this(Thread.currentThread().getThreadGroup(), name); 103 } 104 105 /** 106 * Creates a new thread group. The parent of this new group is the 107 * specified thread group. 108 * <p> 109 * The <code>checkAccess</code> method of the parent thread group is 110 * called with no arguments; this may result in a security exception. 111 * 112 * @param parent the parent thread group. 113 * @param name the name of the new thread group. 114 * @exception NullPointerException if the thread group argument is 115 * <code>null</code>. 116 * @exception SecurityException if the current thread cannot create a 117 * thread in the specified thread group. 118 * @see java.lang.SecurityException 119 * @see java.lang.ThreadGroup#checkAccess() 120 * @since JDK1.0 121 */ 122 public ThreadGroup(ThreadGroup parent, String name) { 123 this(checkParentAccess(parent), parent, name); 124 } 125 126 private ThreadGroup(Void unused, ThreadGroup parent, String name) { 127 this.name = name; 128 this.maxPriority = parent.maxPriority; 129 this.daemon = parent.daemon; 130 this.vmAllowSuspension = parent.vmAllowSuspension; 131 this.parent = parent; 132 parent.add(this); 133 } 134 135 /* 136 * @throws NullPointerException if the parent argument is {@code null} 137 * @throws SecurityException if the current thread cannot create a 138 * thread in the specified thread group. 139 */ 140 private static Void checkParentAccess(ThreadGroup parent) { 141 parent.checkAccess(); 142 return null; 143 } 144 145 /** 146 * Returns the name of this thread group. 147 * 148 * @return the name of this thread group. 149 * @since JDK1.0 150 */ 151 public final String getName() { 152 return name; 153 } 154 155 /** 156 * Returns the parent of this thread group. 157 * <p> 158 * First, if the parent is not <code>null</code>, the 159 * <code>checkAccess</code> method of the parent thread group is 160 * called with no arguments; this may result in a security exception. 161 * 162 * @return the parent of this thread group. The top-level thread group 163 * is the only thread group whose parent is <code>null</code>. 164 * @exception SecurityException if the current thread cannot modify 165 * this thread group. 166 * @see java.lang.ThreadGroup#checkAccess() 167 * @see java.lang.SecurityException 168 * @see java.lang.RuntimePermission 169 * @since JDK1.0 170 */ 171 public final ThreadGroup getParent() { 172 if (parent != null) 173 parent.checkAccess(); 174 return parent; 175 } 176 177 /** 178 * Returns the maximum priority of this thread group. Threads that are 179 * part of this group cannot have a higher priority than the maximum 180 * priority. 181 * 182 * @return the maximum priority that a thread in this thread group 183 * can have. 184 * @see #setMaxPriority 185 * @since JDK1.0 186 */ 187 public final int getMaxPriority() { 188 return maxPriority; 189 } 190 191 /** 192 * Tests if this thread group is a daemon thread group. A 193 * daemon thread group is automatically destroyed when its last 194 * thread is stopped or its last thread group is destroyed. 195 * 196 * @return <code>true</code> if this thread group is a daemon thread group; 197 * <code>false</code> otherwise. 198 * @since JDK1.0 199 */ 200 public final boolean isDaemon() { 201 return daemon; 202 } 203 204 /** 205 * Tests if this thread group has been destroyed. 206 * 207 * @return true if this object is destroyed 208 * @since JDK1.1 209 */ 210 public synchronized boolean isDestroyed() { 211 return destroyed; 212 } 213 214 /** 215 * Changes the daemon status of this thread group. 216 * <p> 217 * First, the <code>checkAccess</code> method of this thread group is 218 * called with no arguments; this may result in a security exception. 219 * <p> 220 * A daemon thread group is automatically destroyed when its last 221 * thread is stopped or its last thread group is destroyed. 222 * 223 * @param daemon if <code>true</code>, marks this thread group as 224 * a daemon thread group; otherwise, marks this 225 * thread group as normal. 226 * @exception SecurityException if the current thread cannot modify 227 * this thread group. 228 * @see java.lang.SecurityException 229 * @see java.lang.ThreadGroup#checkAccess() 230 * @since JDK1.0 231 */ 232 public final void setDaemon(boolean daemon) { 233 checkAccess(); 234 this.daemon = daemon; 235 } 236 237 /** 238 * Sets the maximum priority of the group. Threads in the thread 239 * group that already have a higher priority are not affected. 240 * <p> 241 * First, the <code>checkAccess</code> method of this thread group is 242 * called with no arguments; this may result in a security exception. 243 * <p> 244 * If the <code>pri</code> argument is less than 245 * {@link Thread#MIN_PRIORITY} or greater than 246 * {@link Thread#MAX_PRIORITY}, it is clamped to those values. 247 * <p> 248 * Otherwise, the priority of this ThreadGroup object is set to the 249 * smaller of the specified <code>pri</code> and the maximum permitted 250 * priority of the parent of this thread group. (If this thread group 251 * is the system thread group, which has no parent, then its maximum 252 * priority is simply set to <code>pri</code>.) Then this method is 253 * called recursively, with <code>pri</code> as its argument, for 254 * every thread group that belongs to this thread group. 255 * 256 * @param pri the new priority of the thread group. 257 * @exception SecurityException if the current thread cannot modify 258 * this thread group. 259 * @see #getMaxPriority 260 * @see java.lang.SecurityException 261 * @see java.lang.ThreadGroup#checkAccess() 262 * @since JDK1.0 263 */ 264 // Android changed: We clamp the priority to the range [MIN_PRIORITY, MAX_PRIORITY] 265 // before using it. 266 public final void setMaxPriority(int pri) { 267 int ngroupsSnapshot; 268 ThreadGroup[] groupsSnapshot; 269 synchronized (this) { 270 checkAccess(); 271 // Android changed: Clamp to MIN_PRIORITY, MAX_PRIORITY. 272 // if (pri < Thread.MIN_PRIORITY || pri > Thread.MAX_PRIORITY) { 273 // return; 274 // } 275 if (pri < Thread.MIN_PRIORITY) { 276 pri = Thread.MIN_PRIORITY; 277 } 278 if (pri > Thread.MAX_PRIORITY) { 279 pri = Thread.MAX_PRIORITY; 280 } 281 282 maxPriority = (parent != null) ? Math.min(pri, parent.maxPriority) : pri; 283 ngroupsSnapshot = ngroups; 284 if (groups != null) { 285 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 286 } else { 287 groupsSnapshot = null; 288 } 289 } 290 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 291 groupsSnapshot[i].setMaxPriority(pri); 292 } 293 } 294 295 /** 296 * Tests if this thread group is either the thread group 297 * argument or one of its ancestor thread groups. 298 * 299 * @param g a thread group. 300 * @return <code>true</code> if this thread group is the thread group 301 * argument or one of its ancestor thread groups; 302 * <code>false</code> otherwise. 303 * @since JDK1.0 304 */ 305 public final boolean parentOf(ThreadGroup g) { 306 for (; g != null ; g = g.parent) { 307 if (g == this) { 308 return true; 309 } 310 } 311 return false; 312 } 313 314 /** 315 * Determines if the currently running thread has permission to 316 * modify this thread group. 317 * <p> 318 * If there is a security manager, its <code>checkAccess</code> method 319 * is called with this thread group as its argument. This may result 320 * in throwing a <code>SecurityException</code>. 321 * 322 * @exception SecurityException if the current thread is not allowed to 323 * access this thread group. 324 * @see java.lang.SecurityManager#checkAccess(java.lang.ThreadGroup) 325 * @since JDK1.0 326 */ 327 public final void checkAccess() { 328 } 329 330 /** 331 * Returns an estimate of the number of active threads in this thread 332 * group and its subgroups. Recursively iterates over all subgroups in 333 * this thread group. 334 * 335 * <p> The value returned is only an estimate because the number of 336 * threads may change dynamically while this method traverses internal 337 * data structures, and might be affected by the presence of certain 338 * system threads. This method is intended primarily for debugging 339 * and monitoring purposes. 340 * 341 * @return an estimate of the number of active threads in this thread 342 * group and in any other thread group that has this thread 343 * group as an ancestor 344 * 345 * @since JDK1.0 346 */ 347 public int activeCount() { 348 int result; 349 // Snapshot sub-group data so we don't hold this lock 350 // while our children are computing. 351 int ngroupsSnapshot; 352 ThreadGroup[] groupsSnapshot; 353 synchronized (this) { 354 if (destroyed) { 355 return 0; 356 } 357 result = nthreads; 358 ngroupsSnapshot = ngroups; 359 if (groups != null) { 360 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 361 } else { 362 groupsSnapshot = null; 363 } 364 } 365 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 366 result += groupsSnapshot[i].activeCount(); 367 } 368 return result; 369 } 370 371 /** 372 * Copies into the specified array every active thread in this 373 * thread group and its subgroups. 374 * 375 * <p> An invocation of this method behaves in exactly the same 376 * way as the invocation 377 * 378 * <blockquote> 379 * {@linkplain #enumerate(Thread[], boolean) enumerate}{@code (list, true)} 380 * </blockquote> 381 * 382 * @param list 383 * an array into which to put the list of threads 384 * 385 * @return the number of threads put into the array 386 * 387 * @throws SecurityException 388 * if {@linkplain #checkAccess checkAccess} determines that 389 * the current thread cannot access this thread group 390 * 391 * @since JDK1.0 392 */ 393 public int enumerate(Thread list[]) { 394 checkAccess(); 395 return enumerate(list, 0, true); 396 } 397 398 /** 399 * Copies into the specified array every active thread in this 400 * thread group. If {@code recurse} is {@code true}, 401 * this method recursively enumerates all subgroups of this 402 * thread group and references to every active thread in these 403 * subgroups are also included. If the array is too short to 404 * hold all the threads, the extra threads are silently ignored. 405 * 406 * <p> An application might use the {@linkplain #activeCount activeCount} 407 * method to get an estimate of how big the array should be, however 408 * <i>if the array is too short to hold all the threads, the extra threads 409 * are silently ignored.</i> If it is critical to obtain every active 410 * thread in this thread group, the caller should verify that the returned 411 * int value is strictly less than the length of {@code list}. 412 * 413 * <p> Due to the inherent race condition in this method, it is recommended 414 * that the method only be used for debugging and monitoring purposes. 415 * 416 * @param list 417 * an array into which to put the list of threads 418 * 419 * @param recurse 420 * if {@code true}, recursively enumerate all subgroups of this 421 * thread group 422 * 423 * @return the number of threads put into the array 424 * 425 * @throws SecurityException 426 * if {@linkplain #checkAccess checkAccess} determines that 427 * the current thread cannot access this thread group 428 * 429 * @since JDK1.0 430 */ 431 public int enumerate(Thread list[], boolean recurse) { 432 checkAccess(); 433 return enumerate(list, 0, recurse); 434 } 435 436 private int enumerate(Thread list[], int n, boolean recurse) { 437 int ngroupsSnapshot = 0; 438 ThreadGroup[] groupsSnapshot = null; 439 synchronized (this) { 440 if (destroyed) { 441 return 0; 442 } 443 int nt = nthreads; 444 if (nt > list.length - n) { 445 nt = list.length - n; 446 } 447 for (int i = 0; i < nt; i++) { 448 if (threads[i].isAlive()) { 449 list[n++] = threads[i]; 450 } 451 } 452 if (recurse) { 453 ngroupsSnapshot = ngroups; 454 if (groups != null) { 455 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 456 } else { 457 groupsSnapshot = null; 458 } 459 } 460 } 461 if (recurse) { 462 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 463 n = groupsSnapshot[i].enumerate(list, n, true); 464 } 465 } 466 return n; 467 } 468 469 /** 470 * Returns an estimate of the number of active groups in this 471 * thread group and its subgroups. Recursively iterates over 472 * all subgroups in this thread group. 473 * 474 * <p> The value returned is only an estimate because the number of 475 * thread groups may change dynamically while this method traverses 476 * internal data structures. This method is intended primarily for 477 * debugging and monitoring purposes. 478 * 479 * @return the number of active thread groups with this thread group as 480 * an ancestor 481 * 482 * @since JDK1.0 483 */ 484 public int activeGroupCount() { 485 int ngroupsSnapshot; 486 ThreadGroup[] groupsSnapshot; 487 synchronized (this) { 488 if (destroyed) { 489 return 0; 490 } 491 ngroupsSnapshot = ngroups; 492 if (groups != null) { 493 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 494 } else { 495 groupsSnapshot = null; 496 } 497 } 498 int n = ngroupsSnapshot; 499 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 500 n += groupsSnapshot[i].activeGroupCount(); 501 } 502 return n; 503 } 504 505 /** 506 * Copies into the specified array references to every active 507 * subgroup in this thread group and its subgroups. 508 * 509 * <p> An invocation of this method behaves in exactly the same 510 * way as the invocation 511 * 512 * <blockquote> 513 * {@linkplain #enumerate(ThreadGroup[], boolean) enumerate}{@code (list, true)} 514 * </blockquote> 515 * 516 * @param list 517 * an array into which to put the list of thread groups 518 * 519 * @return the number of thread groups put into the array 520 * 521 * @throws SecurityException 522 * if {@linkplain #checkAccess checkAccess} determines that 523 * the current thread cannot access this thread group 524 * 525 * @since JDK1.0 526 */ 527 public int enumerate(ThreadGroup list[]) { 528 checkAccess(); 529 return enumerate(list, 0, true); 530 } 531 532 /** 533 * Copies into the specified array references to every active 534 * subgroup in this thread group. If {@code recurse} is 535 * {@code true}, this method recursively enumerates all subgroups of this 536 * thread group and references to every active thread group in these 537 * subgroups are also included. 538 * 539 * <p> An application might use the 540 * {@linkplain #activeGroupCount activeGroupCount} method to 541 * get an estimate of how big the array should be, however <i>if the 542 * array is too short to hold all the thread groups, the extra thread 543 * groups are silently ignored.</i> If it is critical to obtain every 544 * active subgroup in this thread group, the caller should verify that 545 * the returned int value is strictly less than the length of 546 * {@code list}. 547 * 548 * <p> Due to the inherent race condition in this method, it is recommended 549 * that the method only be used for debugging and monitoring purposes. 550 * 551 * @param list 552 * an array into which to put the list of thread groups 553 * 554 * @param recurse 555 * if {@code true}, recursively enumerate all subgroups 556 * 557 * @return the number of thread groups put into the array 558 * 559 * @throws SecurityException 560 * if {@linkplain #checkAccess checkAccess} determines that 561 * the current thread cannot access this thread group 562 * 563 * @since JDK1.0 564 */ 565 public int enumerate(ThreadGroup list[], boolean recurse) { 566 checkAccess(); 567 return enumerate(list, 0, recurse); 568 } 569 570 private int enumerate(ThreadGroup list[], int n, boolean recurse) { 571 int ngroupsSnapshot = 0; 572 ThreadGroup[] groupsSnapshot = null; 573 synchronized (this) { 574 if (destroyed) { 575 return 0; 576 } 577 int ng = ngroups; 578 if (ng > list.length - n) { 579 ng = list.length - n; 580 } 581 if (ng > 0) { 582 System.arraycopy(groups, 0, list, n, ng); 583 n += ng; 584 } 585 if (recurse) { 586 ngroupsSnapshot = ngroups; 587 if (groups != null) { 588 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 589 } else { 590 groupsSnapshot = null; 591 } 592 } 593 } 594 if (recurse) { 595 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 596 n = groupsSnapshot[i].enumerate(list, n, true); 597 } 598 } 599 return n; 600 } 601 602 /** 603 * Stops all threads in this thread group. 604 * <p> 605 * First, the <code>checkAccess</code> method of this thread group is 606 * called with no arguments; this may result in a security exception. 607 * <p> 608 * This method then calls the <code>stop</code> method on all the 609 * threads in this thread group and in all of its subgroups. 610 * 611 * @exception SecurityException if the current thread is not allowed 612 * to access this thread group or any of the threads in 613 * the thread group. 614 * @see java.lang.SecurityException 615 * @see java.lang.Thread#stop() 616 * @see java.lang.ThreadGroup#checkAccess() 617 * @since JDK1.0 618 * @deprecated This method is inherently unsafe. See 619 * {@link Thread#stop} for details. 620 */ 621 @Deprecated 622 public final void stop() { 623 if (stopOrSuspend(false)) 624 Thread.currentThread().stop(); 625 } 626 627 /** 628 * Interrupts all threads in this thread group. 629 * <p> 630 * First, the <code>checkAccess</code> method of this thread group is 631 * called with no arguments; this may result in a security exception. 632 * <p> 633 * This method then calls the <code>interrupt</code> method on all the 634 * threads in this thread group and in all of its subgroups. 635 * 636 * @exception SecurityException if the current thread is not allowed 637 * to access this thread group or any of the threads in 638 * the thread group. 639 * @see java.lang.Thread#interrupt() 640 * @see java.lang.SecurityException 641 * @see java.lang.ThreadGroup#checkAccess() 642 * @since 1.2 643 */ 644 public final void interrupt() { 645 int ngroupsSnapshot; 646 ThreadGroup[] groupsSnapshot; 647 synchronized (this) { 648 checkAccess(); 649 for (int i = 0 ; i < nthreads ; i++) { 650 threads[i].interrupt(); 651 } 652 ngroupsSnapshot = ngroups; 653 if (groups != null) { 654 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 655 } else { 656 groupsSnapshot = null; 657 } 658 } 659 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 660 groupsSnapshot[i].interrupt(); 661 } 662 } 663 664 /** 665 * Suspends all threads in this thread group. 666 * <p> 667 * First, the <code>checkAccess</code> method of this thread group is 668 * called with no arguments; this may result in a security exception. 669 * <p> 670 * This method then calls the <code>suspend</code> method on all the 671 * threads in this thread group and in all of its subgroups. 672 * 673 * @exception SecurityException if the current thread is not allowed 674 * to access this thread group or any of the threads in 675 * the thread group. 676 * @see java.lang.Thread#suspend() 677 * @see java.lang.SecurityException 678 * @see java.lang.ThreadGroup#checkAccess() 679 * @since JDK1.0 680 * @deprecated This method is inherently deadlock-prone. See 681 * {@link Thread#suspend} for details. 682 */ 683 @Deprecated 684 public final void suspend() { 685 if (stopOrSuspend(true)) 686 Thread.currentThread().suspend(); 687 } 688 689 /** 690 * Helper method: recursively stops or suspends (as directed by the 691 * boolean argument) all of the threads in this thread group and its 692 * subgroups, except the current thread. This method returns true 693 * if (and only if) the current thread is found to be in this thread 694 * group or one of its subgroups. 695 */ 696 private boolean stopOrSuspend(boolean suspend) { 697 boolean suicide = false; 698 Thread us = Thread.currentThread(); 699 int ngroupsSnapshot; 700 ThreadGroup[] groupsSnapshot = null; 701 synchronized (this) { 702 checkAccess(); 703 for (int i = 0 ; i < nthreads ; i++) { 704 if (threads[i]==us) 705 suicide = true; 706 else if (suspend) 707 threads[i].suspend(); 708 else 709 threads[i].stop(); 710 } 711 712 ngroupsSnapshot = ngroups; 713 if (groups != null) { 714 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 715 } 716 } 717 for (int i = 0 ; i < ngroupsSnapshot ; i++) 718 suicide = groupsSnapshot[i].stopOrSuspend(suspend) || suicide; 719 720 return suicide; 721 } 722 723 /** 724 * Resumes all threads in this thread group. 725 * <p> 726 * First, the <code>checkAccess</code> method of this thread group is 727 * called with no arguments; this may result in a security exception. 728 * <p> 729 * This method then calls the <code>resume</code> method on all the 730 * threads in this thread group and in all of its sub groups. 731 * 732 * @exception SecurityException if the current thread is not allowed to 733 * access this thread group or any of the threads in the 734 * thread group. 735 * @see java.lang.SecurityException 736 * @see java.lang.Thread#resume() 737 * @see java.lang.ThreadGroup#checkAccess() 738 * @since JDK1.0 739 * @deprecated This method is used solely in conjunction with 740 * <tt>Thread.suspend</tt> and <tt>ThreadGroup.suspend</tt>, 741 * both of which have been deprecated, as they are inherently 742 * deadlock-prone. See {@link Thread#suspend} for details. 743 */ 744 @Deprecated 745 public final void resume() { 746 int ngroupsSnapshot; 747 ThreadGroup[] groupsSnapshot; 748 synchronized (this) { 749 checkAccess(); 750 for (int i = 0 ; i < nthreads ; i++) { 751 threads[i].resume(); 752 } 753 ngroupsSnapshot = ngroups; 754 if (groups != null) { 755 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 756 } else { 757 groupsSnapshot = null; 758 } 759 } 760 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 761 groupsSnapshot[i].resume(); 762 } 763 } 764 765 /** 766 * Destroys this thread group and all of its subgroups. This thread 767 * group must be empty, indicating that all threads that had been in 768 * this thread group have since stopped. 769 * <p> 770 * First, the <code>checkAccess</code> method of this thread group is 771 * called with no arguments; this may result in a security exception. 772 * 773 * @exception IllegalThreadStateException if the thread group is not 774 * empty or if the thread group has already been destroyed. 775 * @exception SecurityException if the current thread cannot modify this 776 * thread group. 777 * @see java.lang.ThreadGroup#checkAccess() 778 * @since JDK1.0 779 */ 780 public final void destroy() { 781 int ngroupsSnapshot; 782 ThreadGroup[] groupsSnapshot; 783 synchronized (this) { 784 checkAccess(); 785 if (destroyed || (nthreads > 0)) { 786 throw new IllegalThreadStateException(); 787 } 788 ngroupsSnapshot = ngroups; 789 if (groups != null) { 790 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 791 } else { 792 groupsSnapshot = null; 793 } 794 if (parent != null) { 795 destroyed = true; 796 ngroups = 0; 797 groups = null; 798 nthreads = 0; 799 threads = null; 800 } 801 } 802 for (int i = 0 ; i < ngroupsSnapshot ; i += 1) { 803 groupsSnapshot[i].destroy(); 804 } 805 if (parent != null) { 806 parent.remove(this); 807 } 808 } 809 810 /** 811 * Adds the specified Thread group to this group. 812 * @param g the specified Thread group to be added 813 * @exception IllegalThreadStateException If the Thread group has been destroyed. 814 */ 815 private final void add(ThreadGroup g){ 816 synchronized (this) { 817 if (destroyed) { 818 throw new IllegalThreadStateException(); 819 } 820 if (groups == null) { 821 groups = new ThreadGroup[4]; 822 } else if (ngroups == groups.length) { 823 groups = Arrays.copyOf(groups, ngroups * 2); 824 } 825 groups[ngroups] = g; 826 827 // This is done last so it doesn't matter in case the 828 // thread is killed 829 ngroups++; 830 } 831 } 832 833 /** 834 * Removes the specified Thread group from this group. 835 * @param g the Thread group to be removed 836 * @return if this Thread has already been destroyed. 837 */ 838 private void remove(ThreadGroup g) { 839 synchronized (this) { 840 if (destroyed) { 841 return; 842 } 843 for (int i = 0 ; i < ngroups ; i++) { 844 if (groups[i] == g) { 845 ngroups -= 1; 846 System.arraycopy(groups, i + 1, groups, i, ngroups - i); 847 // Zap dangling reference to the dead group so that 848 // the garbage collector will collect it. 849 groups[ngroups] = null; 850 break; 851 } 852 } 853 if (nthreads == 0) { 854 notifyAll(); 855 } 856 if (daemon && (nthreads == 0) && 857 (nUnstartedThreads == 0) && (ngroups == 0)) 858 { 859 destroy(); 860 } 861 } 862 } 863 864 865 /** 866 * Increments the count of unstarted threads in the thread group. 867 * Unstarted threads are not added to the thread group so that they 868 * can be collected if they are never started, but they must be 869 * counted so that daemon thread groups with unstarted threads in 870 * them are not destroyed. 871 */ 872 void addUnstarted() { 873 synchronized(this) { 874 if (destroyed) { 875 throw new IllegalThreadStateException(); 876 } 877 nUnstartedThreads++; 878 } 879 } 880 881 /** 882 * Adds the specified thread to this thread group. 883 * 884 * <p> Note: This method is called from both library code 885 * and the Virtual Machine. It is called from VM to add 886 * certain system threads to the system thread group. 887 * 888 * @param t 889 * the Thread to be added 890 * 891 * @throws IllegalThreadStateException 892 * if the Thread group has been destroyed 893 */ 894 void add(Thread t) { 895 synchronized (this) { 896 if (destroyed) { 897 throw new IllegalThreadStateException(); 898 } 899 if (threads == null) { 900 threads = new Thread[4]; 901 } else if (nthreads == threads.length) { 902 threads = Arrays.copyOf(threads, nthreads * 2); 903 } 904 threads[nthreads] = t; 905 906 // This is done last so it doesn't matter in case the 907 // thread is killed 908 nthreads++; 909 910 // The thread is now a fully fledged member of the group, even 911 // though it may, or may not, have been started yet. It will prevent 912 // the group from being destroyed so the unstarted Threads count is 913 // decremented. 914 nUnstartedThreads--; 915 } 916 } 917 918 /** 919 * Notifies the group that the thread {@code t} has failed 920 * an attempt to start. 921 * 922 * <p> The state of this thread group is rolled back as if the 923 * attempt to start the thread has never occurred. The thread is again 924 * considered an unstarted member of the thread group, and a subsequent 925 * attempt to start the thread is permitted. 926 * 927 * @param t 928 * the Thread whose start method was invoked 929 * 930 * @param failed 931 * true if the thread could not be started successfully 932 */ 933 void threadStartFailed(Thread t) { 934 synchronized(this) { 935 remove(t); 936 nUnstartedThreads++; 937 } 938 } 939 940 /** 941 * Notifies the group that the thread {@code t} has terminated. 942 * 943 * <p> Destroy the group if all of the following conditions are 944 * true: this is a daemon thread group; there are no more alive 945 * or unstarted threads in the group; there are no subgroups in 946 * this thread group. 947 * 948 * @param t 949 * the Thread that has terminated 950 */ 951 void threadTerminated(Thread t) { 952 synchronized (this) { 953 remove(t); 954 955 if (nthreads == 0) { 956 notifyAll(); 957 } 958 if (daemon && (nthreads == 0) && 959 (nUnstartedThreads == 0) && (ngroups == 0)) 960 { 961 destroy(); 962 } 963 } 964 } 965 966 /** 967 * Removes the specified Thread from this group. Invoking this method 968 * on a thread group that has been destroyed has no effect. 969 * 970 * @param t 971 * the Thread to be removed 972 */ 973 private void remove(Thread t) { 974 synchronized (this) { 975 if (destroyed) { 976 return; 977 } 978 for (int i = 0 ; i < nthreads ; i++) { 979 if (threads[i] == t) { 980 System.arraycopy(threads, i + 1, threads, i, --nthreads - i); 981 // Zap dangling reference to the dead thread so that 982 // the garbage collector will collect it. 983 threads[nthreads] = null; 984 break; 985 } 986 } 987 } 988 } 989 990 /** 991 * Prints information about this thread group to the standard 992 * output. This method is useful only for debugging. 993 * 994 * @since JDK1.0 995 */ 996 public void list() { 997 list(System.out, 0); 998 } 999 void list(PrintStream out, int indent) { 1000 int ngroupsSnapshot; 1001 ThreadGroup[] groupsSnapshot; 1002 synchronized (this) { 1003 for (int j = 0 ; j < indent ; j++) { 1004 out.print(" "); 1005 } 1006 out.println(this); 1007 indent += 4; 1008 for (int i = 0 ; i < nthreads ; i++) { 1009 for (int j = 0 ; j < indent ; j++) { 1010 out.print(" "); 1011 } 1012 out.println(threads[i]); 1013 } 1014 ngroupsSnapshot = ngroups; 1015 if (groups != null) { 1016 groupsSnapshot = Arrays.copyOf(groups, ngroupsSnapshot); 1017 } else { 1018 groupsSnapshot = null; 1019 } 1020 } 1021 for (int i = 0 ; i < ngroupsSnapshot ; i++) { 1022 groupsSnapshot[i].list(out, indent); 1023 } 1024 } 1025 1026 /** 1027 * Called by the Java Virtual Machine when a thread in this 1028 * thread group stops because of an uncaught exception, and the thread 1029 * does not have a specific {@link Thread.UncaughtExceptionHandler} 1030 * installed. 1031 * <p> 1032 * The <code>uncaughtException</code> method of 1033 * <code>ThreadGroup</code> does the following: 1034 * <ul> 1035 * <li>If this thread group has a parent thread group, the 1036 * <code>uncaughtException</code> method of that parent is called 1037 * with the same two arguments. 1038 * <li>Otherwise, this method checks to see if there is a 1039 * {@linkplain Thread#getDefaultUncaughtExceptionHandler default 1040 * uncaught exception handler} installed, and if so, its 1041 * <code>uncaughtException</code> method is called with the same 1042 * two arguments. 1043 * <li>Otherwise, this method determines if the <code>Throwable</code> 1044 * argument is an instance of {@link ThreadDeath}. If so, nothing 1045 * special is done. Otherwise, a message containing the 1046 * thread's name, as returned from the thread's {@link 1047 * Thread#getName getName} method, and a stack backtrace, 1048 * using the <code>Throwable</code>'s {@link 1049 * Throwable#printStackTrace printStackTrace} method, is 1050 * printed to the {@linkplain System#err standard error stream}. 1051 * </ul> 1052 * <p> 1053 * Applications can override this method in subclasses of 1054 * <code>ThreadGroup</code> to provide alternative handling of 1055 * uncaught exceptions. 1056 * 1057 * @param t the thread that is about to exit. 1058 * @param e the uncaught exception. 1059 * @since JDK1.0 1060 */ 1061 public void uncaughtException(Thread t, Throwable e) { 1062 if (parent != null) { 1063 parent.uncaughtException(t, e); 1064 } else { 1065 Thread.UncaughtExceptionHandler ueh = 1066 Thread.getDefaultUncaughtExceptionHandler(); 1067 if (ueh != null) { 1068 ueh.uncaughtException(t, e); 1069 } else if (!(e instanceof ThreadDeath)) { 1070 System.err.print("Exception in thread \"" 1071 + t.getName() + "\" "); 1072 e.printStackTrace(System.err); 1073 } 1074 } 1075 } 1076 1077 /** 1078 * Used by VM to control lowmem implicit suspension. 1079 * 1080 * @param b boolean to allow or disallow suspension 1081 * @return true on success 1082 * @since JDK1.1 1083 * @deprecated The definition of this call depends on {@link #suspend}, 1084 * which is deprecated. Further, the behavior of this call 1085 * was never specified. 1086 */ 1087 @Deprecated 1088 public boolean allowThreadSuspension(boolean b) { 1089 this.vmAllowSuspension = b; 1090 if (!b) { 1091 VM.unsuspendSomeThreads(); 1092 } 1093 return true; 1094 } 1095 1096 /** 1097 * Returns a string representation of this Thread group. 1098 * 1099 * @return a string representation of this thread group. 1100 * @since JDK1.0 1101 */ 1102 public String toString() { 1103 return getClass().getName() + "[name=" + getName() + ",maxpri=" + maxPriority + "]"; 1104 } 1105 } 1106