mirror of
https://github.com/moonlight-stream/moonlight-qt
synced 2025-01-09 01:38:44 +00:00
Batch delayed flushes to improve performance on macOS
This commit is contained in:
parent
18130fd8de
commit
aa74e6930b
2 changed files with 42 additions and 20 deletions
|
@ -15,6 +15,9 @@
|
|||
#define SER_HOSTS "hosts"
|
||||
#define SER_HOSTS_BACKUP "hostsbackup"
|
||||
|
||||
// We will wait up to this length of time to batch delayed flushes
|
||||
#define DELAYED_FLUSH_TIMEOUT_MS 10000
|
||||
|
||||
class PcMonitorThread : public QThread
|
||||
{
|
||||
Q_OBJECT
|
||||
|
@ -149,7 +152,7 @@ ComputerManager::ComputerManager(QObject *parent)
|
|||
m_PollingRef(0),
|
||||
m_MdnsBrowser(nullptr),
|
||||
m_CompatFetcher(nullptr),
|
||||
m_NeedsDelayedFlush(false)
|
||||
m_NeedsFlush(FlushType::None)
|
||||
{
|
||||
QSettings settings;
|
||||
|
||||
|
@ -190,16 +193,17 @@ ComputerManager::~ComputerManager()
|
|||
// Stop the delayed flush thread before acquiring the lock in write mode
|
||||
// to avoid deadlocking with a flush that needs the lock in read mode.
|
||||
{
|
||||
// Wake the delayed flush thread
|
||||
// Wake the delayed flush thread for an immediate flush
|
||||
m_DelayedFlushThread->requestInterruption();
|
||||
m_DelayedFlushCondition.wakeOne();
|
||||
m_ImmediateFlushCondition.wakeOne();
|
||||
|
||||
// Wait for it to terminate (and finish any pending flush)
|
||||
m_DelayedFlushThread->wait();
|
||||
delete m_DelayedFlushThread;
|
||||
|
||||
// Delayed flushes should have completed by now
|
||||
Q_ASSERT(!m_NeedsDelayedFlush);
|
||||
Q_ASSERT(m_NeedsFlush == FlushType::None);
|
||||
}
|
||||
|
||||
QWriteLocker lock(&m_Lock);
|
||||
|
@ -237,19 +241,25 @@ void DelayedFlushThread::run() {
|
|||
{
|
||||
QMutexLocker locker(&m_ComputerManager->m_DelayedFlushMutex);
|
||||
|
||||
while (!QThread::currentThread()->isInterruptionRequested() && !m_ComputerManager->m_NeedsDelayedFlush) {
|
||||
// Wait for any flush to be pending
|
||||
while (!QThread::currentThread()->isInterruptionRequested() && m_ComputerManager->m_NeedsFlush == ComputerManager::FlushType::None) {
|
||||
m_ComputerManager->m_DelayedFlushCondition.wait(&m_ComputerManager->m_DelayedFlushMutex);
|
||||
}
|
||||
|
||||
// Bail without flushing if we woke up for an interruption alone.
|
||||
// If we have both an interruption and a flush request, do the flush.
|
||||
if (!m_ComputerManager->m_NeedsDelayedFlush) {
|
||||
if (m_ComputerManager->m_NeedsFlush == ComputerManager::FlushType::None) {
|
||||
Q_ASSERT(QThread::currentThread()->isInterruptionRequested());
|
||||
break;
|
||||
}
|
||||
|
||||
// Reset the delayed flush flag to ensure any racing saveHosts() call will set it again
|
||||
m_ComputerManager->m_NeedsDelayedFlush = false;
|
||||
// If this is a delayed flush and we're not being interrupted, wait on the immediate flush condition (up to the delayed flush timeout).
|
||||
while (!QThread::currentThread()->isInterruptionRequested() &&
|
||||
m_ComputerManager->m_NeedsFlush != ComputerManager::FlushType::Immediate &&
|
||||
m_ComputerManager->m_ImmediateFlushCondition.wait(&m_ComputerManager->m_DelayedFlushMutex, DELAYED_FLUSH_TIMEOUT_MS));
|
||||
|
||||
// Reset the flush condition
|
||||
m_ComputerManager->m_NeedsFlush = ComputerManager::FlushType::None;
|
||||
}
|
||||
|
||||
// Perform the flush
|
||||
|
@ -287,15 +297,24 @@ void DelayedFlushThread::run() {
|
|||
}
|
||||
}
|
||||
|
||||
void ComputerManager::saveHosts()
|
||||
void ComputerManager::saveHosts(bool immediate)
|
||||
{
|
||||
Q_ASSERT(m_DelayedFlushThread != nullptr && m_DelayedFlushThread->isRunning());
|
||||
|
||||
// Punt to a worker thread because QSettings on macOS can take ages (> 500 ms)
|
||||
// to persist our host list to disk (especially when a host has a bunch of apps).
|
||||
QMutexLocker locker(&m_DelayedFlushMutex);
|
||||
m_NeedsDelayedFlush = true;
|
||||
if (m_NeedsFlush == FlushType::None) {
|
||||
m_NeedsFlush = immediate ? FlushType::Immediate : FlushType::Delayed;
|
||||
}
|
||||
else if (m_NeedsFlush == FlushType::Delayed && immediate) {
|
||||
// Upgrade a delayed flush to an immediate flush if an immediate request comes in
|
||||
m_NeedsFlush = FlushType::Immediate;
|
||||
}
|
||||
m_DelayedFlushCondition.wakeOne();
|
||||
if (immediate) {
|
||||
m_ImmediateFlushCondition.wakeOne();
|
||||
}
|
||||
}
|
||||
|
||||
QHostAddress ComputerManager::getBestGlobalAddressV6(QVector<QHostAddress> &addresses)
|
||||
|
@ -442,8 +461,8 @@ void ComputerManager::handleComputerStateChanged(NvComputer* computer)
|
|||
emit quitAppCompleted(QVariant());
|
||||
}
|
||||
|
||||
// Save updated hosts to QSettings
|
||||
saveHosts();
|
||||
// Save updated hosts to QSettings (delayed and batchable)
|
||||
saveHosts(false);
|
||||
}
|
||||
|
||||
QVector<NvComputer*> ComputerManager::getComputers()
|
||||
|
@ -479,8 +498,8 @@ public:
|
|||
m_ComputerManager->m_KnownHosts.remove(m_Computer->uuid);
|
||||
}
|
||||
|
||||
// Persist the new host list
|
||||
m_ComputerManager->saveHosts();
|
||||
// Persist the new host list immediately
|
||||
m_ComputerManager->saveHosts(true);
|
||||
|
||||
// Delete the polling entry first. This will stop all polling threads too.
|
||||
delete pollingEntry;
|
||||
|
@ -520,9 +539,6 @@ void ComputerManager::renameHost(NvComputer* computer, QString name)
|
|||
|
||||
void ComputerManager::clientSideAttributeUpdated(NvComputer* computer)
|
||||
{
|
||||
// Persist the change
|
||||
saveHosts();
|
||||
|
||||
// Notify the UI of the state change
|
||||
handleComputerStateChanged(computer);
|
||||
}
|
||||
|
@ -579,8 +595,9 @@ private:
|
|||
emit pairingCompleted(m_Computer, tr("Another pairing attempt is already in progress."));
|
||||
break;
|
||||
case NvPairingManager::PairState::PAIRED:
|
||||
// Persist the newly pinned server certificate for this host
|
||||
m_ComputerManager->saveHosts();
|
||||
// Persist the newly pinned server certificate for this host. We do this immediately because
|
||||
// loss of the pinned cert invalidates the pairing. This is also a very infrequent operation.
|
||||
m_ComputerManager->saveHosts(true);
|
||||
|
||||
emit pairingCompleted(m_Computer, nullptr);
|
||||
break;
|
||||
|
|
|
@ -249,7 +249,7 @@ private slots:
|
|||
void handleMdnsServiceResolved(MdnsPendingComputer* computer, QVector<QHostAddress>& addresses);
|
||||
|
||||
private:
|
||||
void saveHosts();
|
||||
void saveHosts(bool immediate);
|
||||
|
||||
QHostAddress getBestGlobalAddressV6(QVector<QHostAddress>& addresses);
|
||||
|
||||
|
@ -266,5 +266,10 @@ private:
|
|||
DelayedFlushThread* m_DelayedFlushThread;
|
||||
QMutex m_DelayedFlushMutex;
|
||||
QWaitCondition m_DelayedFlushCondition;
|
||||
bool m_NeedsDelayedFlush;
|
||||
QWaitCondition m_ImmediateFlushCondition;
|
||||
enum class FlushType {
|
||||
None,
|
||||
Delayed,
|
||||
Immediate
|
||||
} m_NeedsFlush;
|
||||
};
|
||||
|
|
Loading…
Reference in a new issue