/* * This file is part of the L2J Mobius project. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program 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 for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package com.l2jmobius.commons.util; import java.lang.management.LockInfo; import java.lang.management.ManagementFactory; import java.lang.management.MonitorInfo; import java.lang.management.ThreadInfo; import java.lang.management.ThreadMXBean; import java.time.Duration; import java.util.logging.Level; import java.util.logging.Logger; import com.l2jmobius.Config; /** * Thread to check for deadlocked threads. * @author -Nemesiss- L2M */ public class DeadLockDetector extends Thread { private static Logger LOGGER = Logger.getLogger(DeadLockDetector.class.getName()); private final Duration _checkInterval; private final Runnable _deadLockCallback; private final ThreadMXBean tmx; public DeadLockDetector(Duration checkInterval, Runnable deadLockCallback) { super("DeadLockDetector"); _checkInterval = checkInterval; _deadLockCallback = deadLockCallback; tmx = ManagementFactory.getThreadMXBean(); } @Override public final void run() { boolean deadlock = false; while (!deadlock) { try { final long[] ids = tmx.findDeadlockedThreads(); if (ids != null) { deadlock = true; final ThreadInfo[] tis = tmx.getThreadInfo(ids, true, true); final StringBuilder info = new StringBuilder(); info.append("DeadLock Found!"); info.append(Config.EOL); for (ThreadInfo ti : tis) { info.append(ti.toString()); } for (ThreadInfo ti : tis) { final LockInfo[] locks = ti.getLockedSynchronizers(); final MonitorInfo[] monitors = ti.getLockedMonitors(); if ((locks.length == 0) && (monitors.length == 0)) { continue; } ThreadInfo dl = ti; info.append("Java-level deadlock:"); info.append(Config.EOL); info.append('\t'); info.append(dl.getThreadName()); info.append(" is waiting to lock "); info.append(dl.getLockInfo().toString()); info.append(" which is held by "); info.append(dl.getLockOwnerName()); info.append(Config.EOL); while ((dl = tmx.getThreadInfo(new long[] { dl.getLockOwnerId() }, true, true)[0]).getThreadId() != ti.getThreadId()) { info.append('\t'); info.append(dl.getThreadName()); info.append(" is waiting to lock "); info.append(dl.getLockInfo().toString()); info.append(" which is held by "); info.append(dl.getLockOwnerName()); info.append(Config.EOL); } } LOGGER.warning(info.toString()); if (_deadLockCallback != null) { _deadLockCallback.run(); } } Thread.sleep(_checkInterval.toMillis()); } catch (Exception e) { LOGGER.log(Level.WARNING, "", e); } } } }