3f1dd83c32
Adds a new test for the deadlock found in JDK-8027670 and fixed in JDK-8029281. Reviewed-by: mchung
341 lines
13 KiB
Java
341 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2014, 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.
|
|
*/
|
|
import java.io.File;
|
|
import java.io.PrintStream;
|
|
import java.lang.management.ManagementFactory;
|
|
import java.lang.management.ThreadInfo;
|
|
import java.security.Permission;
|
|
import java.security.Policy;
|
|
import java.security.ProtectionDomain;
|
|
import java.util.ArrayList;
|
|
import java.util.Collections;
|
|
import java.util.HashSet;
|
|
import java.util.List;
|
|
import java.util.Random;
|
|
import java.util.Set;
|
|
import java.util.concurrent.atomic.AtomicLong;
|
|
import java.util.logging.Level;
|
|
import java.util.logging.LogManager;
|
|
import java.util.logging.Logger;
|
|
|
|
|
|
/**
|
|
* @test
|
|
* @bug 8027670 8029281
|
|
* @summary Deadlock in drainLoggerRefQueueBounded / readConfiguration
|
|
* caused by synchronization issues in Logger and LogManager.
|
|
* @run main/othervm TestLogConfigurationDeadLockWithConf
|
|
* @author danielfuchs
|
|
*/
|
|
// This test is a best effort to try & detect issues. The test itself will run
|
|
// for 8secs. This is usually sufficient to detect issues.
|
|
// However to get a greater confidence it is recommended to run this test in a loop:
|
|
// e.g. use something like:
|
|
// $ while jtreg -jdk:$JDK -verbose:all \
|
|
// test/java/util/logging/TestLogConfigurationDeadLockWithConf.java ; \
|
|
// do echo Running test again ; done
|
|
// and let it run for a few hours...
|
|
//
|
|
public class TestLogConfigurationDeadLockWithConf {
|
|
|
|
static volatile Exception thrown = null;
|
|
static volatile boolean goOn = true;
|
|
|
|
static final int READERS = 2;
|
|
static final int LOGGERS = 2;
|
|
static final long TIME = 4 * 1000; // 4 sec.
|
|
static final long STEP = 1 * 1000; // message every 1 sec.
|
|
static final int LCOUNT = 50; // 50 loggers created in a row...
|
|
static final AtomicLong nextLogger = new AtomicLong(0);
|
|
static final AtomicLong readCount = new AtomicLong(0);
|
|
static final AtomicLong checkCount = new AtomicLong(0);
|
|
|
|
/**
|
|
* This test will run both with and without a security manager.
|
|
*
|
|
* The test starts a number of threads that will call
|
|
* LogManager.readConfiguration() concurrently (ReadConf), then starts
|
|
* a number of threads that will create new loggers concurrently
|
|
* (AddLogger), and then two additional threads: one (Stopper) that
|
|
* will stop the test after 4secs (TIME ms), and one DeadlockDetector
|
|
* that will attempt to detect deadlocks.
|
|
* If after 4secs no deadlock was detected and no exception was thrown
|
|
* then the test is considered a success and passes.
|
|
*
|
|
* This procedure is done twice: once without a security manager and once
|
|
* again with a security manager - which means the test takes ~8secs to
|
|
* run.
|
|
*
|
|
* Note that 8sec may not be enough to detect issues if there are some.
|
|
* This is a best effort test.
|
|
*
|
|
* @param args the command line arguments
|
|
* @throws java.lang.Exception if the test fails.
|
|
*/
|
|
public static void main(String[] args) throws Exception {
|
|
File config = new File(System.getProperty("test.src", "."),
|
|
"deadlockconf.properties");
|
|
if (!config.canRead()) {
|
|
System.err.println("Can't read config file: test cannot execute.");
|
|
System.err.println("Please check your test environment: ");
|
|
System.err.println("\t -Dtest.src=" + System.getProperty("test.src", "."));
|
|
System.err.println("\t config file is: " + config.getAbsolutePath());
|
|
throw new RuntimeException("Can't read config file: "
|
|
+ config.getAbsolutePath());
|
|
}
|
|
|
|
System.setProperty("java.util.logging.config.file",
|
|
config.getAbsolutePath());
|
|
|
|
// test without security
|
|
System.out.println("No security");
|
|
test();
|
|
|
|
// test with security
|
|
System.out.println("\nWith security");
|
|
Policy.setPolicy(new Policy() {
|
|
@Override
|
|
public boolean implies(ProtectionDomain domain, Permission permission) {
|
|
if (super.implies(domain, permission)) return true;
|
|
// System.out.println("Granting " + permission);
|
|
return true; // all permissions
|
|
}
|
|
});
|
|
System.setSecurityManager(new SecurityManager());
|
|
test();
|
|
}
|
|
|
|
static Random rand = new Random(System.currentTimeMillis());
|
|
private static int getBarCount() {
|
|
return rand.nextInt(10);
|
|
}
|
|
|
|
/**
|
|
* Starts all threads, wait 4secs, then stops all threads.
|
|
* @throws Exception if a deadlock was detected or an error occurred.
|
|
*/
|
|
public static void test() throws Exception {
|
|
goOn = true;
|
|
thrown = null;
|
|
long sNextLogger = nextLogger.get();
|
|
long sReadCount = readCount.get();
|
|
long sCheckCount = checkCount.get();
|
|
List<Thread> threads = new ArrayList<>();
|
|
for (int i = 0; i<READERS; i++) {
|
|
threads.add(new ReadConf());
|
|
}
|
|
for (int i = 0; i<LOGGERS; i++) {
|
|
threads.add(new AddLogger());
|
|
}
|
|
DeadlockDetector detector = new DeadlockDetector();
|
|
threads.add(detector);
|
|
threads.add(0, new Stopper(TIME));
|
|
for (Thread t : threads) {
|
|
t.start();
|
|
}
|
|
|
|
// wait for the detector to finish.
|
|
detector.join();
|
|
|
|
final PrintStream out = thrown == null ? System.out : System.err;
|
|
|
|
// Try to wait for all threads to finish.
|
|
// This is a best effort: if some threads are in deadlock we can't
|
|
// obviously wait for them, and other threads may have joined in
|
|
// the deadlock since we last checked.
|
|
// However, all threads which are succeptible of deadlocking
|
|
// extend DeamonThread.
|
|
for (Thread t : threads) {
|
|
if (t == detector) {
|
|
continue;
|
|
}
|
|
if (detector.deadlocked.contains(t.getId())) {
|
|
out.println("Skipping deadlocked thread "
|
|
+ t.getClass().getSimpleName() + ": " + t);
|
|
continue; // don't wait for deadlocked thread: they won't terminate
|
|
}
|
|
try {
|
|
if (detector.deadlocked.isEmpty()) {
|
|
t.join();
|
|
} else {
|
|
if (t instanceof DaemonThread) {
|
|
// Some other threads may have join the deadlock.
|
|
// don't wait forever.
|
|
t.join(100);
|
|
} else {
|
|
// Those threads that don't extend DaemonThread
|
|
// should be safe from deadlock.
|
|
out.println("Waiting for "
|
|
+ t.getClass().getSimpleName() + ": " + t);
|
|
t.join();
|
|
}
|
|
}
|
|
} catch (Exception x) {
|
|
fail(x);
|
|
}
|
|
}
|
|
out.println("All threads joined.");
|
|
|
|
final String status = thrown == null ? "Passed" : "FAILED";
|
|
|
|
out.println(status + ": " + (nextLogger.get() - sNextLogger)
|
|
+ " loggers created by " + LOGGERS + " Thread(s),");
|
|
out.println("\t LogManager.readConfiguration() called "
|
|
+ (readCount.get() - sReadCount) + " times by " + READERS
|
|
+ " Thread(s).");
|
|
out.println("\t ThreadMXBean.findDeadlockedThreads called "
|
|
+ (checkCount.get() -sCheckCount) + " times by 1 Thread.");
|
|
|
|
if (thrown != null) {
|
|
out.println("\t Error is: "+thrown.getMessage());
|
|
throw thrown;
|
|
}
|
|
}
|
|
|
|
static class DaemonThread extends Thread {
|
|
public DaemonThread() {
|
|
this.setDaemon(true);
|
|
}
|
|
}
|
|
|
|
final static class ReadConf extends DaemonThread {
|
|
@Override
|
|
public void run() {
|
|
while (goOn) {
|
|
try {
|
|
LogManager.getLogManager().readConfiguration();
|
|
readCount.incrementAndGet();
|
|
Thread.sleep(1);
|
|
} catch (Exception x) {
|
|
fail(x);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
final static class AddLogger extends DaemonThread {
|
|
@Override
|
|
public void run() {
|
|
try {
|
|
while (goOn) {
|
|
Logger l;
|
|
int barcount = getBarCount();
|
|
for (int i=0; i < LCOUNT ; i++) {
|
|
l = Logger.getLogger("foo.bar"+barcount+".l"+nextLogger.incrementAndGet());
|
|
l.fine("I'm fine");
|
|
if (!goOn) break;
|
|
Thread.sleep(1);
|
|
}
|
|
}
|
|
} catch (InterruptedException | RuntimeException x ) {
|
|
fail(x);
|
|
}
|
|
}
|
|
}
|
|
|
|
final static class DeadlockDetector extends Thread {
|
|
|
|
final Set<Long> deadlocked = Collections.synchronizedSet(new HashSet<Long>());
|
|
|
|
static List<Long> asList(long... ids) {
|
|
final List<Long> list = new ArrayList<>(ids.length);
|
|
for (long id : ids) {
|
|
list.add(id);
|
|
}
|
|
return list;
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
while(goOn) {
|
|
try {
|
|
long[] ids = ManagementFactory.getThreadMXBean().findDeadlockedThreads();
|
|
checkCount.incrementAndGet();
|
|
ids = ids == null ? new long[0] : ids;
|
|
if (ids.length > 0) {
|
|
deadlocked.addAll(asList(ids));
|
|
}
|
|
if (ids.length == 1) {
|
|
throw new RuntimeException("Found 1 deadlocked thread: "+ids[0]);
|
|
} else if (ids.length > 0) {
|
|
ThreadInfo[] infos = ManagementFactory.getThreadMXBean().getThreadInfo(ids, Integer.MAX_VALUE);
|
|
System.err.println("Found "+ids.length+" deadlocked threads: ");
|
|
for (ThreadInfo inf : infos) {
|
|
System.err.println(inf.toString());
|
|
}
|
|
throw new RuntimeException("Found "+ids.length+" deadlocked threads");
|
|
}
|
|
Thread.sleep(100);
|
|
} catch(InterruptedException | RuntimeException x) {
|
|
fail(x);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
static final class Stopper extends Thread {
|
|
long start;
|
|
long time;
|
|
|
|
static final Logger logger = Logger.getLogger("remaining");
|
|
|
|
Stopper(long time) {
|
|
start = System.currentTimeMillis();
|
|
this.time = time;
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
try {
|
|
long rest, previous;
|
|
previous = time;
|
|
while (goOn && (rest = start - System.currentTimeMillis() + time) > 0) {
|
|
if (previous == time || previous - rest >= STEP) {
|
|
logger.log(Level.INFO,
|
|
"{0}ms remaining...", String.valueOf(rest));
|
|
previous = rest == time ? rest -1 : rest;
|
|
System.gc();
|
|
}
|
|
if (goOn == false) break;
|
|
Thread.sleep(Math.min(rest, 100));
|
|
}
|
|
System.out.println(System.currentTimeMillis() - start
|
|
+ " ms elapsed ("+time+ " requested)");
|
|
goOn = false;
|
|
} catch(InterruptedException | RuntimeException x) {
|
|
fail(x);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
static void fail(Exception x) {
|
|
x.printStackTrace();
|
|
if (thrown == null) {
|
|
thrown = x;
|
|
}
|
|
goOn = false;
|
|
}
|
|
}
|