mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-06 11:10:44 +00:00
Fix a rare race condition when shutting down the logger.
This resulted in an occasional use-after-free crash every once in a while.
This commit is contained in:
parent
162e5695b1
commit
5e9d391ccc
@ -105,6 +105,7 @@ void LogThread::logToFile()
|
|||||||
}
|
}
|
||||||
logTrigger.wakeAll();
|
logTrigger.wakeAll();
|
||||||
strlock.unlock();
|
strlock.unlock();
|
||||||
|
|
||||||
if (m_logStream) {
|
if (m_logStream) {
|
||||||
qDebug().noquote() << "Logging to" << debugLog;
|
qDebug().noquote() << "Logging to" << debugLog;
|
||||||
} else {
|
} else {
|
||||||
@ -112,6 +113,20 @@ void LogThread::logToFile()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LogThread::~LogThread()
|
||||||
|
{
|
||||||
|
QMutexLocker lock(&strlock);
|
||||||
|
|
||||||
|
Q_ASSERT(running == false);
|
||||||
|
if (m_logStream) {
|
||||||
|
delete m_logStream;
|
||||||
|
m_logStream = nullptr;
|
||||||
|
Q_ASSERT(m_logFile);
|
||||||
|
delete m_logFile;
|
||||||
|
m_logFile = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
QString LogThread::logFileName()
|
QString LogThread::logFileName()
|
||||||
{
|
{
|
||||||
if (!m_logFile) {
|
if (!m_logFile) {
|
||||||
@ -124,7 +139,8 @@ void shutdownLogger()
|
|||||||
{
|
{
|
||||||
if (logger) {
|
if (logger) {
|
||||||
logger->quit();
|
logger->quit();
|
||||||
otherThreadPool->waitForDone(-1);
|
// The thread is automatically destroyed when its run() method exits.
|
||||||
|
otherThreadPool->waitForDone(-1); // wait until that happens
|
||||||
logger = NULL;
|
logger = NULL;
|
||||||
}
|
}
|
||||||
delete otherThreadPool;
|
delete otherThreadPool;
|
||||||
@ -150,32 +166,23 @@ void LogThread::appendClean(QString msg)
|
|||||||
|
|
||||||
void LogThread::quit() {
|
void LogThread::quit() {
|
||||||
qDebug() << "Shutting down logging thread";
|
qDebug() << "Shutting down logging thread";
|
||||||
running = false;
|
qInstallMessageHandler(0); // Remove our logger.
|
||||||
|
|
||||||
strlock.lock();
|
strlock.lock();
|
||||||
while (!buffer.isEmpty()) {
|
running = false; // Force the thread to exit after its next iteration.
|
||||||
QString msg = buffer.takeFirst();
|
logTrigger.wakeAll(); // Trigger the final flush.
|
||||||
if (m_logStream) {
|
strlock.unlock(); // Release the lock so that the thread can complete.
|
||||||
*m_logStream << msg << endl;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (m_logStream) {
|
|
||||||
delete m_logStream;
|
|
||||||
m_logStream = nullptr;
|
|
||||||
Q_ASSERT(m_logFile);
|
|
||||||
delete m_logFile;
|
|
||||||
m_logFile = nullptr;
|
|
||||||
}
|
|
||||||
strlock.unlock();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void LogThread::run()
|
void LogThread::run()
|
||||||
{
|
{
|
||||||
|
QMutexLocker lock(&strlock);
|
||||||
|
|
||||||
running = true;
|
running = true;
|
||||||
s_LoggerRunning.unlock(); // unlock as soon as the thread begins to run
|
s_LoggerRunning.unlock(); // unlock as soon as the thread begins to run
|
||||||
do {
|
do {
|
||||||
strlock.lock();
|
logTrigger.wait(&strlock); // releases strlock while it waits
|
||||||
logTrigger.wait(&strlock);
|
|
||||||
while (connected && m_logFile && !buffer.isEmpty()) {
|
while (connected && m_logFile && !buffer.isEmpty()) {
|
||||||
QString msg = buffer.takeFirst();
|
QString msg = buffer.takeFirst();
|
||||||
if (m_logStream) {
|
if (m_logStream) {
|
||||||
@ -183,8 +190,9 @@ void LogThread::run()
|
|||||||
}
|
}
|
||||||
emit outputLog(msg);
|
emit outputLog(msg);
|
||||||
}
|
}
|
||||||
strlock.unlock();
|
|
||||||
} while (running);
|
} while (running);
|
||||||
|
|
||||||
|
// strlock will be released when lock goes out of scope
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@ class LogThread:public QObject, public QRunnable
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit LogThread() : QRunnable() { running = false; logtime.start(); connected = false; m_logFile = nullptr; m_logStream = nullptr; }
|
explicit LogThread() : QRunnable() { running = false; logtime.start(); connected = false; m_logFile = nullptr; m_logStream = nullptr; }
|
||||||
virtual ~LogThread() {}
|
virtual ~LogThread();
|
||||||
|
|
||||||
void run();
|
void run();
|
||||||
void append(QString msg);
|
void append(QString msg);
|
||||||
|
Loading…
Reference in New Issue
Block a user