847 lines
30 KiB
Java
847 lines
30 KiB
Java
|
/*
|
||
|
* Copyright (c) 2020, 2022, Oracle and/or its affiliates. All rights reserved.
|
||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||
|
*
|
||
|
* This code is free software; you can redistribute it and/or modify it
|
||
|
* under the terms of the GNU General Public License version 2 only, as
|
||
|
* published by the Free Software Foundation.
|
||
|
*
|
||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||
|
* accompanied this code).
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License version
|
||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||
|
*
|
||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||
|
* or visit www.oracle.com if you need additional information or have any
|
||
|
* questions.
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @test
|
||
|
* @summary Unit tests for java.lang.ThreadGroup
|
||
|
* @run testng BasicTests
|
||
|
*/
|
||
|
|
||
|
import java.lang.ref.WeakReference;
|
||
|
import java.util.Arrays;
|
||
|
import java.util.Objects;
|
||
|
import java.util.Set;
|
||
|
import java.util.concurrent.locks.LockSupport;
|
||
|
import java.util.stream.Collectors;
|
||
|
|
||
|
import org.testng.annotations.Test;
|
||
|
import static org.testng.Assert.*;
|
||
|
|
||
|
public class BasicTests {
|
||
|
|
||
|
@Test
|
||
|
public void testGetName1() {
|
||
|
ThreadGroup group = new ThreadGroup(null);
|
||
|
assertTrue(group.getName() == null);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testGetName2() {
|
||
|
ThreadGroup group = new ThreadGroup("fred");
|
||
|
assertEquals(group.getName(), "fred");
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testGetParent() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
ThreadGroup group3 = new ThreadGroup(group2, "group3");
|
||
|
|
||
|
assertTrue(group1.getParent() == Thread.currentThread().getThreadGroup());
|
||
|
assertTrue(group2.getParent() == group1);
|
||
|
assertTrue(group3.getParent() == group2);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testParentOf() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
ThreadGroup group3 = new ThreadGroup(group2, "group3");
|
||
|
|
||
|
assertFalse(group1.parentOf(null));
|
||
|
assertTrue(group1.parentOf(group1));
|
||
|
assertTrue(group1.parentOf(group2));
|
||
|
assertTrue(group1.parentOf(group3));
|
||
|
|
||
|
assertFalse(group2.parentOf(null));
|
||
|
assertFalse(group2.parentOf(group1));
|
||
|
assertTrue(group2.parentOf(group2));
|
||
|
assertTrue(group2.parentOf(group3));
|
||
|
|
||
|
assertFalse(group3.parentOf(null));
|
||
|
assertFalse(group3.parentOf(group1));
|
||
|
assertFalse(group3.parentOf(group2));
|
||
|
assertTrue(group3.parentOf(group3));
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testActiveCount1() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertTrue(group.activeCount() == 0);
|
||
|
TestThread thread = TestThread.start(group, "foo");
|
||
|
try {
|
||
|
assertTrue(group.activeCount() == 1);
|
||
|
} finally {
|
||
|
thread.terminate();
|
||
|
}
|
||
|
assertTrue(group.activeCount() == 0);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testActiveCount2() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
assertTrue(group1.activeCount() == 0);
|
||
|
assertTrue(group2.activeCount() == 0);
|
||
|
TestThread thread1 = TestThread.start(group1, "foo");
|
||
|
try {
|
||
|
assertTrue(group1.activeCount() == 1);
|
||
|
assertTrue(group2.activeCount() == 0);
|
||
|
TestThread thread2 = TestThread.start(group2, "bar");
|
||
|
try {
|
||
|
assertTrue(group1.activeCount() == 2);
|
||
|
assertTrue(group2.activeCount() == 1);
|
||
|
} finally {
|
||
|
thread2.terminate();
|
||
|
}
|
||
|
assertTrue(group1.activeCount() == 1);
|
||
|
assertTrue(group2.activeCount() == 0);
|
||
|
} finally {
|
||
|
thread1.terminate();
|
||
|
}
|
||
|
assertTrue(group1.activeCount() == 0);
|
||
|
assertTrue(group2.activeCount() == 0);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void enumerateThreads1() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
Thread[] threads = new Thread[100];
|
||
|
assertTrue(group.enumerate(threads) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group.enumerate(threads, true) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
TestThread thread = TestThread.start(group, "foo");
|
||
|
try {
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread);
|
||
|
assertTrue(threads[1] == null);
|
||
|
} finally {
|
||
|
thread.terminate();
|
||
|
}
|
||
|
assertTrue(group.activeCount() == 0);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void enumerateThreads2() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
|
||
|
Thread[] threads = new Thread[100];
|
||
|
assertTrue(group1.enumerate(threads) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group1.enumerate(threads, true) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group1.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
|
||
|
TestThread thread1 = TestThread.start(group1, "foo");
|
||
|
try {
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
|
||
|
TestThread thread2 = TestThread.start(group2, "bar");
|
||
|
try {
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads) == 2);
|
||
|
assertEquals(toSet(threads, 2), Set.of(thread1, thread2));
|
||
|
assertTrue(threads[2] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, true) == 2);
|
||
|
assertEquals(toSet(threads, 2), Set.of(thread1, thread2));
|
||
|
assertTrue(threads[2] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
assertTrue(threads[1] == null);
|
||
|
} finally {
|
||
|
thread2.terminate();
|
||
|
}
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread1);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
} finally {
|
||
|
thread1.terminate();
|
||
|
}
|
||
|
assertTrue(group1.activeCount() == 0);
|
||
|
assertTrue(group2.activeCount() == 0);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void enumerateThreads3() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
ThreadGroup group3 = new ThreadGroup(group2, "group3");
|
||
|
|
||
|
Thread[] threads = new Thread[100];
|
||
|
assertTrue(group1.enumerate(threads) == 0);
|
||
|
assertTrue(group1.enumerate(threads, true) == 0);
|
||
|
assertTrue(group1.enumerate(threads, false) == 0);
|
||
|
assertTrue(group2.enumerate(threads) == 0);
|
||
|
assertTrue(group2.enumerate(threads, true) == 0);
|
||
|
assertTrue(group2.enumerate(threads, false) == 0);
|
||
|
assertTrue(group3.enumerate(threads) == 0);
|
||
|
assertTrue(group3.enumerate(threads, true) == 0);
|
||
|
assertTrue(group3.enumerate(threads, false) == 0);
|
||
|
|
||
|
TestThread thread2 = TestThread.start(group2, "foo");
|
||
|
try {
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads) == 1);
|
||
|
assertTrue(group1.enumerate(threads, true) == 1);
|
||
|
assertTrue(group1.enumerate(threads, false) == 0);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
|
||
|
assertTrue(group3.enumerate(threads) == 0);
|
||
|
assertTrue(group3.enumerate(threads, true) == 0);
|
||
|
assertTrue(group3.enumerate(threads, false) == 0);
|
||
|
|
||
|
TestThread thread3 = TestThread.start(group3, "bar");
|
||
|
try {
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads) == 2);
|
||
|
assertEquals(toSet(threads, 2), Set.of(thread2, thread3));
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, true) == 2);
|
||
|
assertEquals(toSet(threads, 2), Set.of(thread2, thread3));
|
||
|
assertTrue(group1.enumerate(threads, false) == 0);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads) == 2);
|
||
|
assertEquals(toSet(threads, 2), Set.of(thread2, thread3));
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 2);
|
||
|
assertEquals(toSet(threads, 2), Set.of(thread2, thread3));
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group3.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread3);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group3.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread3);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group3.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread3);
|
||
|
} finally {
|
||
|
thread3.terminate();
|
||
|
}
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads) == 1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group1.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads) == 1);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group2.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread2);
|
||
|
assertTrue(threads[1] == null);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group3.enumerate(threads) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group3.enumerate(threads, true) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
assertTrue(group3.enumerate(threads, false) == 0);
|
||
|
assertTrue(threads[0] == null);
|
||
|
|
||
|
} finally {
|
||
|
thread2.terminate();
|
||
|
}
|
||
|
|
||
|
assertTrue(group1.enumerate(threads) == 0);
|
||
|
assertTrue(group1.enumerate(threads, true) == 0);
|
||
|
assertTrue(group1.enumerate(threads, false) == 0);
|
||
|
assertTrue(group2.enumerate(threads) == 0);
|
||
|
assertTrue(group2.enumerate(threads, true) == 0);
|
||
|
assertTrue(group2.enumerate(threads, false) == 0);
|
||
|
assertTrue(group3.enumerate(threads) == 0);
|
||
|
assertTrue(group3.enumerate(threads, true) == 0);
|
||
|
assertTrue(group3.enumerate(threads, false) == 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Test enumerate(Thread[]) with an array of insufficient size
|
||
|
*/
|
||
|
@Test
|
||
|
public void enumerateThreads4() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
|
||
|
// array too small
|
||
|
Thread[] threads = new Thread[1];
|
||
|
|
||
|
TestThread thread1 = TestThread.start(group, "thread1");
|
||
|
TestThread thread2 = TestThread.start(group, "thread2");
|
||
|
try {
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group.enumerate(threads) == 1);
|
||
|
assertTrue(threads[0] == thread1 || threads[1] == thread2);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group.enumerate(threads, true) == 1);
|
||
|
assertTrue(threads[0] == thread1 || threads[1] == thread2);
|
||
|
|
||
|
Arrays.setAll(threads, i -> null);
|
||
|
assertTrue(group.enumerate(threads, false) == 1);
|
||
|
assertTrue(threads[0] == thread1 || threads[1] == thread2);
|
||
|
} finally {
|
||
|
thread1.terminate();
|
||
|
thread2.terminate();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testActiveGroupCount() throws Exception {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
assertTrue(group1.activeGroupCount() == 0);
|
||
|
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
assertTrue(group1.activeGroupCount() == 1);
|
||
|
assertTrue(group2.activeGroupCount() == 0);
|
||
|
|
||
|
ThreadGroup group3 = new ThreadGroup(group2, "group3");
|
||
|
assertTrue(group1.activeGroupCount() == 2);
|
||
|
assertTrue(group2.activeGroupCount() == 1);
|
||
|
assertTrue(group3.activeGroupCount() == 0);
|
||
|
|
||
|
// unreference group3 and wait for it to be GC'ed
|
||
|
var ref = new WeakReference<>(group3);
|
||
|
group3 = null;
|
||
|
System.gc();
|
||
|
while (ref.get() != null) {
|
||
|
Thread.sleep(50);
|
||
|
}
|
||
|
|
||
|
assertTrue(group1.activeGroupCount() == 1);
|
||
|
assertTrue(group2.activeGroupCount() == 0);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testEnumerateGroups1() throws Exception {
|
||
|
ThreadGroup[] groups = new ThreadGroup[100];
|
||
|
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
|
||
|
assertTrue(group1.enumerate(groups) == 0);
|
||
|
assertTrue(groups[0] == null);
|
||
|
assertTrue(group1.enumerate(groups, true) == 0);
|
||
|
assertTrue(groups[0] == null);
|
||
|
assertTrue(group1.enumerate(groups, false) == 0);
|
||
|
assertTrue(groups[0] == null);
|
||
|
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups, true) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups, false) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
ThreadGroup group3 = new ThreadGroup(group2, "group3");
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups) == 2);
|
||
|
assertEquals(toSet(groups, 2), Set.of(group2, group3));
|
||
|
assertTrue(groups[2] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups, true) == 2);
|
||
|
assertEquals(toSet(groups, 2), Set.of(group2, group3));
|
||
|
assertTrue(groups[2] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups, false) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group2.enumerate(groups) == 1);
|
||
|
assertTrue(groups[0] == group3);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group2.enumerate(groups, true) == 1);
|
||
|
assertTrue(groups[0] == group3);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group2.enumerate(groups, false) == 1);
|
||
|
assertTrue(groups[0] == group3);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group3.enumerate(groups) == 0);
|
||
|
assertTrue(groups[0] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group3.enumerate(groups, true) == 0);
|
||
|
assertTrue(groups[0] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group3.enumerate(groups, false) == 0);
|
||
|
assertTrue(groups[0] == null);
|
||
|
|
||
|
// unreference group3 and wait for it to be GC'ed
|
||
|
var ref = new WeakReference<>(group3);
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
group3 = null;
|
||
|
System.gc();
|
||
|
while (ref.get() != null) {
|
||
|
Thread.sleep(50);
|
||
|
}
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups, true) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group1.enumerate(groups, false) == 1);
|
||
|
assertTrue(groups[0] == group2);
|
||
|
assertTrue(groups[1] == null);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Test enumerate(ThreadGroup[]) with an array of insufficient size
|
||
|
*/
|
||
|
@Test
|
||
|
public void testEnumerateGroups2() throws Exception {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
ThreadGroup child1 = new ThreadGroup(group, "child1");
|
||
|
ThreadGroup child2 = new ThreadGroup(group,"child2");
|
||
|
|
||
|
// array too small
|
||
|
ThreadGroup[] groups = new ThreadGroup[1];
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group.enumerate(groups) == 1);
|
||
|
assertTrue(groups[0] == child1 || groups[1] == child2);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group.enumerate(groups, true) == 1);
|
||
|
assertTrue(groups[0] == child1 || groups[1] == child2);
|
||
|
|
||
|
Arrays.setAll(groups, i -> null);
|
||
|
assertTrue(group.enumerate(groups, false) == 1);
|
||
|
assertTrue(groups[0] == child1 || groups[1] == child2);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testMaxPriority1() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
final int maxPriority = group.getMaxPriority();
|
||
|
assertTrue(maxPriority == Thread.currentThread().getThreadGroup().getMaxPriority());
|
||
|
|
||
|
group.setMaxPriority(Thread.MIN_PRIORITY-1);
|
||
|
assertTrue(group.getMaxPriority() == maxPriority);
|
||
|
|
||
|
group.setMaxPriority(Thread.MAX_PRIORITY+1);
|
||
|
assertTrue(group.getMaxPriority() == maxPriority);
|
||
|
|
||
|
group.setMaxPriority(maxPriority+1);
|
||
|
assertTrue(group.getMaxPriority() == maxPriority);
|
||
|
|
||
|
if (maxPriority > Thread.MIN_PRIORITY) {
|
||
|
group.setMaxPriority(maxPriority-1);
|
||
|
assertTrue(group.getMaxPriority() == (maxPriority-1));
|
||
|
}
|
||
|
|
||
|
group.setMaxPriority(maxPriority);
|
||
|
assertTrue(group.getMaxPriority() == maxPriority);
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testMaxPriority2() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
int maxPriority = group1.getMaxPriority();
|
||
|
if (maxPriority > Thread.MIN_PRIORITY) {
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
assertTrue(group2.getMaxPriority() == maxPriority);
|
||
|
|
||
|
ThreadGroup group3 = new ThreadGroup(group2, "group3");
|
||
|
assertTrue(group3.getMaxPriority() == maxPriority);
|
||
|
|
||
|
group1.setMaxPriority(Thread.MIN_PRIORITY);
|
||
|
assertTrue(group1.getMaxPriority() == Thread.MIN_PRIORITY);
|
||
|
assertTrue(group2.getMaxPriority() == Thread.MIN_PRIORITY);
|
||
|
assertTrue(group3.getMaxPriority() == Thread.MIN_PRIORITY);
|
||
|
group1.setMaxPriority(maxPriority);
|
||
|
|
||
|
assertTrue(group1.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group2.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group3.getMaxPriority() == maxPriority);
|
||
|
|
||
|
group2.setMaxPriority(Thread.MIN_PRIORITY);
|
||
|
assertTrue(group1.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group2.getMaxPriority() == Thread.MIN_PRIORITY);
|
||
|
assertTrue(group3.getMaxPriority() == Thread.MIN_PRIORITY);
|
||
|
group2.setMaxPriority(maxPriority);
|
||
|
|
||
|
assertTrue(group1.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group2.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group3.getMaxPriority() == maxPriority);
|
||
|
|
||
|
group3.setMaxPriority(Thread.MIN_PRIORITY);
|
||
|
assertTrue(group1.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group2.getMaxPriority() == maxPriority);
|
||
|
assertTrue(group3.getMaxPriority() == Thread.MIN_PRIORITY);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testMaxPriority3() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
if (group.getMaxPriority() > Thread.MIN_PRIORITY) {
|
||
|
int maxPriority = Thread.MIN_PRIORITY + 1;
|
||
|
group.setMaxPriority(maxPriority);
|
||
|
assertTrue(group.getMaxPriority() == maxPriority);
|
||
|
|
||
|
Thread thread = new Thread(group, () -> { });
|
||
|
int priority = thread.getPriority();
|
||
|
|
||
|
int expectedPriority = Math.min(Thread.currentThread().getPriority(), maxPriority);
|
||
|
assertTrue(priority == expectedPriority);
|
||
|
|
||
|
thread.setPriority(Thread.MAX_PRIORITY);
|
||
|
assertTrue(thread.getPriority() == maxPriority);
|
||
|
|
||
|
thread.setPriority(maxPriority);
|
||
|
assertTrue(thread.getPriority() == maxPriority);
|
||
|
|
||
|
thread.setPriority(Thread.MIN_PRIORITY);
|
||
|
assertTrue(thread.getPriority() == Thread.MIN_PRIORITY);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testInterrupt1() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertTrue(group.activeCount() == 0);
|
||
|
TestThread thread = TestThread.start(group, "foo");
|
||
|
try {
|
||
|
group.interrupt();
|
||
|
} finally {
|
||
|
thread.terminate();
|
||
|
}
|
||
|
assertTrue(thread.wasInterrupted());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testInterrupt2() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
TestThread thread1 = TestThread.start(group1, "foo");
|
||
|
TestThread thread2 = TestThread.start(group2, "bar");
|
||
|
try {
|
||
|
group1.interrupt();
|
||
|
} finally {
|
||
|
thread1.terminate();
|
||
|
thread2.terminate();
|
||
|
}
|
||
|
assertTrue(thread2.wasInterrupted());
|
||
|
assertTrue(thread1.wasInterrupted());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testInterrupt3() {
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
TestThread thread1 = TestThread.start(group1, "foo");
|
||
|
TestThread thread2 = TestThread.start(group2, "bar");
|
||
|
try {
|
||
|
group2.interrupt();
|
||
|
} finally {
|
||
|
thread1.terminate();
|
||
|
thread2.terminate();
|
||
|
}
|
||
|
assertTrue(thread2.wasInterrupted());
|
||
|
assertFalse(thread1.wasInterrupted());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testDestroy() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertFalse(group.isDestroyed());
|
||
|
group.destroy(); // does nothing
|
||
|
assertFalse(group.isDestroyed());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testDaemon() {
|
||
|
boolean d = Thread.currentThread().getThreadGroup().isDaemon();
|
||
|
|
||
|
ThreadGroup group1 = new ThreadGroup("group1");
|
||
|
assertTrue(group1.isDaemon() == d); // inherit
|
||
|
|
||
|
group1.setDaemon(true);
|
||
|
assertTrue(group1.isDaemon());
|
||
|
ThreadGroup group2 = new ThreadGroup(group1, "group2");
|
||
|
assertTrue(group2.isDaemon()); // inherit
|
||
|
|
||
|
group1.setDaemon(false);
|
||
|
assertFalse(group1.isDaemon());
|
||
|
ThreadGroup group3 = new ThreadGroup(group1, "group2");
|
||
|
assertFalse(group3.isDaemon()); // inherit
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testList() {
|
||
|
ThreadGroup group = new ThreadGroup("foo");
|
||
|
group.list();
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testSuspend() {
|
||
|
ThreadGroup group = new ThreadGroup("foo");
|
||
|
assertThrows(UnsupportedOperationException.class, () -> group.suspend());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testResume() {
|
||
|
ThreadGroup group = new ThreadGroup("foo");
|
||
|
assertThrows(UnsupportedOperationException.class, () -> group.resume());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testStop() {
|
||
|
ThreadGroup group = new ThreadGroup("foo");
|
||
|
assertThrows(UnsupportedOperationException.class, () -> group.stop());
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testAllowThreadSuspension() {
|
||
|
ThreadGroup group = new ThreadGroup("foo");
|
||
|
assertFalse(group.allowThreadSuspension(false));
|
||
|
assertFalse(group.allowThreadSuspension(true));
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testNull1() {
|
||
|
assertThrows(NullPointerException.class,
|
||
|
() -> new ThreadGroup(null, "group"));
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testNull2() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertThrows(NullPointerException.class,
|
||
|
() -> group.enumerate((Thread[]) null));
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testNull3() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertThrows(NullPointerException.class,
|
||
|
() -> group.enumerate((Thread[]) null, false));
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testNull4() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertThrows(NullPointerException.class,
|
||
|
() -> group.enumerate((ThreadGroup[]) null));
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testNull5() {
|
||
|
ThreadGroup group = new ThreadGroup("group");
|
||
|
assertThrows(NullPointerException.class,
|
||
|
() -> group.enumerate((ThreadGroup[]) null, false));
|
||
|
}
|
||
|
|
||
|
private <T> Set<T> toSet(T[] array, int len) {
|
||
|
return Arrays.stream(array, 0, len).collect(Collectors.toSet());
|
||
|
}
|
||
|
|
||
|
static class TestThread extends Thread {
|
||
|
TestThread(ThreadGroup group, String name) {
|
||
|
super(group, name);
|
||
|
}
|
||
|
|
||
|
static TestThread start(ThreadGroup group, String name) {
|
||
|
TestThread thread = new TestThread(group, name);
|
||
|
thread.start();
|
||
|
return thread;
|
||
|
}
|
||
|
|
||
|
private volatile boolean done;
|
||
|
private volatile boolean interrupted;
|
||
|
|
||
|
public void run() {
|
||
|
if (Thread.currentThread() != this)
|
||
|
throw new IllegalCallerException();
|
||
|
while (!done) {
|
||
|
LockSupport.park();
|
||
|
if (Thread.interrupted()) {
|
||
|
interrupted = true;
|
||
|
}
|
||
|
}
|
||
|
if (Thread.interrupted()) {
|
||
|
interrupted = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
boolean wasInterrupted() {
|
||
|
return interrupted;
|
||
|
}
|
||
|
|
||
|
void terminate() {
|
||
|
done = true;
|
||
|
LockSupport.unpark(this);
|
||
|
boolean interrupted = false;
|
||
|
while (isAlive()) {
|
||
|
try {
|
||
|
Thread.sleep(10);
|
||
|
} catch (InterruptedException e) {
|
||
|
interrupted = true;
|
||
|
}
|
||
|
}
|
||
|
if (interrupted) {
|
||
|
Thread.currentThread().interrupt();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|