moonlight-qt/app/backend/computermanager.h

276 lines
6.5 KiB
C
Raw Normal View History

2018-06-24 22:13:37 +00:00
#pragma once
#include "nvcomputer.h"
2018-07-06 06:12:55 +00:00
#include "nvpairingmanager.h"
#include "settings/compatfetcher.h"
2018-06-24 22:13:37 +00:00
2018-07-01 06:07:31 +00:00
#include <qmdnsengine/server.h>
#include <qmdnsengine/cache.h>
#include <qmdnsengine/browser.h>
#include <qmdnsengine/service.h>
#include <qmdnsengine/resolver.h>
2018-06-27 04:47:01 +00:00
#include <QThread>
#include <QReadWriteLock>
#include <QSettings>
2018-07-06 06:12:55 +00:00
#include <QRunnable>
#include <QTimer>
#include <QMutex>
#include <QWaitCondition>
2022-10-25 07:08:46 +00:00
class ComputerManager;
class DelayedFlushThread : public QThread
{
Q_OBJECT
public:
DelayedFlushThread(ComputerManager* cm)
: m_ComputerManager(cm)
{
setObjectName("CM Delayed Flush Thread");
}
void run();
private:
ComputerManager* m_ComputerManager;
};
2018-06-24 22:13:37 +00:00
2018-07-01 06:07:31 +00:00
class MdnsPendingComputer : public QObject
{
Q_OBJECT
public:
explicit MdnsPendingComputer(const QSharedPointer<QMdnsEngine::Server> server,
2018-07-01 06:07:31 +00:00
const QMdnsEngine::Service& service)
: m_Hostname(service.hostname()),
m_Port(service.port()),
m_ServerWeak(server),
2019-08-05 01:04:07 +00:00
m_Resolver(nullptr)
2018-07-01 06:07:31 +00:00
{
2019-08-05 01:04:07 +00:00
// Start resolving
resolve();
}
virtual ~MdnsPendingComputer()
{
delete m_Resolver;
2018-07-01 06:07:31 +00:00
}
QString hostname()
{
return m_Hostname;
}
uint16_t port()
{
return m_Port;
}
2018-07-01 06:07:31 +00:00
private slots:
void handleResolvedTimeout()
2018-07-01 06:07:31 +00:00
{
2019-08-05 01:04:07 +00:00
if (m_Addresses.isEmpty()) {
// Try again
qInfo() << "Resolving" << hostname() << "timed out. Retrying...";
resolve();
}
else {
Q_ASSERT(!m_Addresses.isEmpty());
emit resolvedHost(this, m_Addresses);
}
}
void handleResolvedAddress(const QHostAddress& address)
{
qInfo() << "Resolved" << hostname() << "to" << address;
m_Addresses.push_back(address);
2018-07-01 06:07:31 +00:00
}
signals:
void resolvedHost(MdnsPendingComputer*,QVector<QHostAddress>&);
2018-07-01 06:07:31 +00:00
private:
2019-08-05 01:04:07 +00:00
void resolve()
{
// Delete our resolver, so we're guaranteed that nothing is referencing m_Server.
2019-08-05 01:04:07 +00:00
delete m_Resolver;
m_Resolver = nullptr;
// Now delete our strong reference that we held on behalf of m_Resolver.
// The server may be destroyed after we make this call.
m_Server.reset();
// Re-acquire a strong reference if the server still exists.
m_Server = m_ServerWeak.toStrongRef();
if (!m_Server) {
return;
}
2023-07-16 05:13:09 +00:00
m_Resolver = new QMdnsEngine::Resolver(m_Server.data(), m_Hostname);
2019-08-05 01:04:07 +00:00
connect(m_Resolver, &QMdnsEngine::Resolver::resolved,
this, &MdnsPendingComputer::handleResolvedAddress);
2021-03-03 00:14:15 +00:00
QTimer::singleShot(2000, this, &MdnsPendingComputer::handleResolvedTimeout);
2019-08-05 01:04:07 +00:00
}
2018-07-01 06:07:31 +00:00
QByteArray m_Hostname;
uint16_t m_Port;
QWeakPointer<QMdnsEngine::Server> m_ServerWeak;
QSharedPointer<QMdnsEngine::Server> m_Server;
2019-08-05 01:04:07 +00:00
QMdnsEngine::Resolver* m_Resolver;
QVector<QHostAddress> m_Addresses;
2018-07-01 06:07:31 +00:00
};
class ComputerPollingEntry
{
public:
ComputerPollingEntry()
: m_ActiveThread(nullptr)
{
}
virtual ~ComputerPollingEntry()
{
interrupt();
// interrupt() should have taken care of this
Q_ASSERT(m_ActiveThread == nullptr);
for (QThread* thread : m_InactiveList) {
thread->wait();
delete thread;
}
}
bool isActive()
{
cleanInactiveList();
return m_ActiveThread != nullptr;
}
void setActiveThread(QThread* thread)
{
cleanInactiveList();
Q_ASSERT(!isActive());
m_ActiveThread = thread;
}
void interrupt()
{
cleanInactiveList();
if (m_ActiveThread != nullptr) {
// Interrupt the active thread
m_ActiveThread->requestInterruption();
// Place it on the inactive list awaiting death
m_InactiveList.append(m_ActiveThread);
m_ActiveThread = nullptr;
}
}
private:
void cleanInactiveList()
{
QMutableListIterator<QThread*> i(m_InactiveList);
// Reap any threads that have finished
while (i.hasNext()) {
i.next();
QThread* thread = i.value();
if (thread->isFinished()) {
delete thread;
i.remove();
}
}
}
QThread* m_ActiveThread;
QList<QThread*> m_InactiveList;
};
2018-06-27 04:47:01 +00:00
class ComputerManager : public QObject
{
Q_OBJECT
2018-07-06 06:12:55 +00:00
friend class DeferredHostDeletionTask;
2018-07-06 07:34:16 +00:00
friend class PendingAddTask;
friend class PendingPairingTask;
friend class DelayedFlushThread;
2018-07-06 06:12:55 +00:00
2018-06-27 04:47:01 +00:00
public:
2018-06-28 02:55:44 +00:00
explicit ComputerManager(QObject *parent = nullptr);
2018-06-27 04:47:01 +00:00
virtual ~ComputerManager();
Q_INVOKABLE void startPolling();
2018-06-27 04:47:01 +00:00
Q_INVOKABLE void stopPollingAsync();
2018-06-27 04:47:01 +00:00
Q_INVOKABLE void addNewHostManually(QString address);
void addNewHost(NvAddress address, bool mdns, NvAddress mdnsIpv6Address = NvAddress());
2018-06-27 04:47:01 +00:00
QString generatePinString();
2018-07-06 06:12:55 +00:00
void pairHost(NvComputer* computer, QString pin);
void quitRunningApp(NvComputer* computer);
QVector<NvComputer*> getComputers();
// computer is deleted inside this call
void deleteHost(NvComputer* computer);
2020-05-02 01:34:15 +00:00
void renameHost(NvComputer* computer, QString name);
void clientSideAttributeUpdated(NvComputer* computer);
2018-06-27 04:47:01 +00:00
signals:
void computerStateChanged(NvComputer* computer);
2018-07-06 06:12:55 +00:00
void pairingCompleted(NvComputer* computer, QString error);
void computerAddCompleted(QVariant success, QVariant detectedPortBlocking);
2018-07-06 07:34:16 +00:00
2018-08-02 05:32:21 +00:00
void quitAppCompleted(QVariant error);
2018-06-27 04:47:01 +00:00
private slots:
void handleAboutToQuit();
2018-06-27 04:47:01 +00:00
void handleComputerStateChanged(NvComputer* computer);
void handleMdnsServiceResolved(MdnsPendingComputer* computer, QVector<QHostAddress>& addresses);
2018-07-01 06:07:31 +00:00
2018-06-27 04:47:01 +00:00
private:
void saveHosts(bool immediate);
QHostAddress getBestGlobalAddressV6(QVector<QHostAddress>& addresses);
2018-06-27 04:47:01 +00:00
void startPollingComputer(NvComputer* computer);
int m_PollingRef;
2018-06-27 04:47:01 +00:00
QReadWriteLock m_Lock;
QMap<QString, NvComputer*> m_KnownHosts;
QMap<QString, ComputerPollingEntry*> m_PollEntries;
QSharedPointer<QMdnsEngine::Server> m_MdnsServer;
2018-07-01 06:07:31 +00:00
QMdnsEngine::Browser* m_MdnsBrowser;
QVector<MdnsPendingComputer*> m_PendingResolution;
CompatFetcher m_CompatFetcher;
DelayedFlushThread* m_DelayedFlushThread;
QMutex m_DelayedFlushMutex;
QWaitCondition m_DelayedFlushCondition;
QWaitCondition m_ImmediateFlushCondition;
enum class FlushType {
None,
Delayed,
Immediate
} m_NeedsFlush;
2018-06-24 22:13:37 +00:00
};