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:
sawinglogz 2020-07-20 19:12:46 -04:00
parent 162e5695b1
commit 5e9d391ccc
2 changed files with 28 additions and 20 deletions

View File

@ -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
} }

View File

@ -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);