mirror of
https://github.com/monero-project/monero-gui.git
synced 2025-01-10 12:54:30 +00:00
qt: implement FutureScheduler, always await async code to complete
This commit is contained in:
parent
c7956f76ea
commit
be7810c5a8
9 changed files with 277 additions and 187 deletions
|
@ -63,6 +63,7 @@ HEADERS += \
|
||||||
src/libwalletqt/UnsignedTransaction.h \
|
src/libwalletqt/UnsignedTransaction.h \
|
||||||
Logger.h \
|
Logger.h \
|
||||||
MainApp.h \
|
MainApp.h \
|
||||||
|
src/qt/FutureScheduler.h \
|
||||||
src/qt/ipc.h \
|
src/qt/ipc.h \
|
||||||
src/qt/mime.h \
|
src/qt/mime.h \
|
||||||
src/qt/KeysFiles.h \
|
src/qt/KeysFiles.h \
|
||||||
|
@ -96,6 +97,7 @@ SOURCES += main.cpp \
|
||||||
src/libwalletqt/UnsignedTransaction.cpp \
|
src/libwalletqt/UnsignedTransaction.cpp \
|
||||||
Logger.cpp \
|
Logger.cpp \
|
||||||
MainApp.cpp \
|
MainApp.cpp \
|
||||||
|
src/qt/FutureScheduler.cpp \
|
||||||
src/qt/ipc.cpp \
|
src/qt/ipc.cpp \
|
||||||
src/qt/mime.cpp \
|
src/qt/mime.cpp \
|
||||||
src/qt/KeysFiles.cpp \
|
src/qt/KeysFiles.cpp \
|
||||||
|
|
|
@ -131,19 +131,12 @@ bool DaemonManager::start(const QString &flags, NetworkType::Type nettype, const
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start start watcher
|
// Start start watcher
|
||||||
QFuture<bool> future = QtConcurrent::run(this, &DaemonManager::startWatcher, nettype);
|
m_scheduler.run([this, nettype] {
|
||||||
QFutureWatcher<bool> * watcher = new QFutureWatcher<bool>();
|
if (startWatcher(nettype))
|
||||||
connect(watcher, &QFutureWatcher<bool>::finished,
|
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<bool> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
if(future.result())
|
|
||||||
emit daemonStarted();
|
emit daemonStarted();
|
||||||
else
|
else
|
||||||
emit daemonStartFailure();
|
emit daemonStartFailure();
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -155,17 +148,12 @@ bool DaemonManager::stop(NetworkType::Type nettype)
|
||||||
qDebug() << message;
|
qDebug() << message;
|
||||||
|
|
||||||
// Start stop watcher - Will kill if not shutting down
|
// Start stop watcher - Will kill if not shutting down
|
||||||
QFuture<bool> future = QtConcurrent::run(this, &DaemonManager::stopWatcher, nettype);
|
m_scheduler.run([this, nettype] {
|
||||||
QFutureWatcher<bool> * watcher = new QFutureWatcher<bool>();
|
if (stopWatcher(nettype))
|
||||||
connect(watcher, &QFutureWatcher<bool>::finished,
|
{
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<bool> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
if(future.result()) {
|
|
||||||
emit daemonStopped();
|
emit daemonStopped();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -330,6 +318,7 @@ QVariantMap DaemonManager::validateDataDir(const QString &dataDir) const
|
||||||
|
|
||||||
DaemonManager::DaemonManager(QObject *parent)
|
DaemonManager::DaemonManager(QObject *parent)
|
||||||
: QObject(parent)
|
: QObject(parent)
|
||||||
|
, m_scheduler(this)
|
||||||
{
|
{
|
||||||
|
|
||||||
// Platform depetent path to monerod
|
// Platform depetent path to monerod
|
||||||
|
@ -344,3 +333,8 @@ DaemonManager::DaemonManager(QObject *parent)
|
||||||
m_has_daemon = false;
|
m_has_daemon = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DaemonManager::~DaemonManager()
|
||||||
|
{
|
||||||
|
m_scheduler.shutdownWaitForFinished();
|
||||||
|
}
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <QProcess>
|
#include <QProcess>
|
||||||
#include <QVariantMap>
|
#include <QVariantMap>
|
||||||
|
#include "qt/FutureScheduler.h"
|
||||||
#include "NetworkType.h"
|
#include "NetworkType.h"
|
||||||
|
|
||||||
class DaemonManager : public QObject
|
class DaemonManager : public QObject
|
||||||
|
@ -71,6 +72,8 @@ public slots:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
explicit DaemonManager(QObject *parent = 0);
|
explicit DaemonManager(QObject *parent = 0);
|
||||||
|
~DaemonManager();
|
||||||
|
|
||||||
static DaemonManager * m_instance;
|
static DaemonManager * m_instance;
|
||||||
static QStringList m_clArgs;
|
static QStringList m_clArgs;
|
||||||
QProcess *m_daemon;
|
QProcess *m_daemon;
|
||||||
|
@ -79,6 +82,7 @@ private:
|
||||||
bool m_has_daemon = true;
|
bool m_has_daemon = true;
|
||||||
bool m_app_exit = false;
|
bool m_app_exit = false;
|
||||||
|
|
||||||
|
FutureScheduler m_scheduler;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // DAEMONMANAGER_H
|
#endif // DAEMONMANAGER_H
|
||||||
|
|
|
@ -150,13 +150,8 @@ NetworkType::Type Wallet::nettype() const
|
||||||
|
|
||||||
void Wallet::updateConnectionStatusAsync()
|
void Wallet::updateConnectionStatusAsync()
|
||||||
{
|
{
|
||||||
QFuture<Monero::Wallet::ConnectionStatus> future = QtConcurrent::run(m_walletImpl, &Monero::Wallet::connected);
|
m_scheduler.run([this] {
|
||||||
QFutureWatcher<Monero::Wallet::ConnectionStatus> *connectionWatcher = new QFutureWatcher<Monero::Wallet::ConnectionStatus>();
|
ConnectionStatus newStatus = static_cast<ConnectionStatus>(m_walletImpl->connected());
|
||||||
|
|
||||||
connect(connectionWatcher, &QFutureWatcher<Monero::Wallet::ConnectionStatus>::finished, [=]() {
|
|
||||||
QFuture<Monero::Wallet::ConnectionStatus> future = connectionWatcher->future();
|
|
||||||
connectionWatcher->deleteLater();
|
|
||||||
ConnectionStatus newStatus = static_cast<ConnectionStatus>(future.result());
|
|
||||||
if (newStatus != m_connectionStatus || !m_initialized) {
|
if (newStatus != m_connectionStatus || !m_initialized) {
|
||||||
m_initialized = true;
|
m_initialized = true;
|
||||||
m_connectionStatus = newStatus;
|
m_connectionStatus = newStatus;
|
||||||
|
@ -166,7 +161,6 @@ void Wallet::updateConnectionStatusAsync()
|
||||||
// Release lock
|
// Release lock
|
||||||
m_connectionStatusRunning = false;
|
m_connectionStatusRunning = false;
|
||||||
});
|
});
|
||||||
connectionWatcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Wallet::ConnectionStatus Wallet::connected(bool forceCheck)
|
Wallet::ConnectionStatus Wallet::connected(bool forceCheck)
|
||||||
|
@ -246,15 +240,10 @@ void Wallet::initAsync(const QString &daemonAddress, quint64 upperTransactionLim
|
||||||
emit connectionStatusChanged(m_connectionStatus);
|
emit connectionStatusChanged(m_connectionStatus);
|
||||||
}
|
}
|
||||||
|
|
||||||
QFuture<bool> future = QtConcurrent::run(this, &Wallet::init,
|
m_scheduler.run([this, daemonAddress, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight] {
|
||||||
daemonAddress, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight);
|
bool success = init(daemonAddress, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight);
|
||||||
QFutureWatcher<bool> * watcher = new QFutureWatcher<bool>();
|
if (success)
|
||||||
|
{
|
||||||
connect(watcher, &QFutureWatcher<bool>::finished,
|
|
||||||
this, [this, watcher, daemonAddress, upperTransactionLimit, isRecovering, restoreHeight]() {
|
|
||||||
QFuture<bool> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
if(future.result()){
|
|
||||||
emit walletCreationHeightChanged();
|
emit walletCreationHeightChanged();
|
||||||
qDebug() << "init async finished - starting refresh";
|
qDebug() << "init async finished - starting refresh";
|
||||||
connected(true);
|
connected(true);
|
||||||
|
@ -262,7 +251,6 @@ void Wallet::initAsync(const QString &daemonAddress, quint64 upperTransactionLim
|
||||||
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//! create a view only wallet
|
//! create a view only wallet
|
||||||
|
@ -348,16 +336,32 @@ void Wallet::setSubaddressLabel(quint32 accountIndex, quint32 addressIndex, cons
|
||||||
m_walletImpl->setSubaddressLabel(accountIndex, addressIndex, label.toStdString());
|
m_walletImpl->setSubaddressLabel(accountIndex, addressIndex, label.toStdString());
|
||||||
}
|
}
|
||||||
|
|
||||||
void Wallet::refreshHeightAsync() const
|
void Wallet::refreshHeightAsync()
|
||||||
{
|
{
|
||||||
QtConcurrent::run([this] {
|
m_scheduler.run([this] {
|
||||||
QFuture<quint64> daemonHeight = QtConcurrent::run([this] {
|
quint64 daemonHeight;
|
||||||
return daemonBlockChainHeight();
|
QPair<bool, QFuture<void>> daemonHeightFuture = m_scheduler.run([this, &daemonHeight] {
|
||||||
|
daemonHeight = daemonBlockChainHeight();
|
||||||
});
|
});
|
||||||
QFuture<quint64> targetHeight = QtConcurrent::run([this] {
|
if (!daemonHeightFuture.first)
|
||||||
return daemonBlockChainTargetHeight();
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
quint64 targetHeight;
|
||||||
|
QPair<bool, QFuture<void>> targetHeightFuture = m_scheduler.run([this, &targetHeight] {
|
||||||
|
targetHeight = daemonBlockChainTargetHeight();
|
||||||
});
|
});
|
||||||
emit heightRefreshed(blockChainHeight(), daemonHeight.result(), targetHeight.result());
|
if (!targetHeightFuture.first)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
quint64 walletHeight = blockChainHeight();
|
||||||
|
daemonHeightFuture.second.waitForFinished();
|
||||||
|
targetHeightFuture.second.waitForFinished();
|
||||||
|
|
||||||
|
emit heightRefreshed(walletHeight, daemonHeight, targetHeight);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -458,17 +462,10 @@ void Wallet::createTransactionAsync(const QString &dst_addr, const QString &paym
|
||||||
quint64 amount, quint32 mixin_count,
|
quint64 amount, quint32 mixin_count,
|
||||||
PendingTransaction::Priority priority)
|
PendingTransaction::Priority priority)
|
||||||
{
|
{
|
||||||
QFuture<PendingTransaction*> future = QtConcurrent::run(this, &Wallet::createTransaction,
|
m_scheduler.run([this, dst_addr, payment_id, amount, mixin_count, priority] {
|
||||||
dst_addr, payment_id,amount, mixin_count, priority);
|
PendingTransaction *tx = createTransaction(dst_addr, payment_id, amount, mixin_count, priority);
|
||||||
QFutureWatcher<PendingTransaction*> * watcher = new QFutureWatcher<PendingTransaction*>();
|
emit transactionCreated(tx, dst_addr, payment_id, mixin_count);
|
||||||
|
|
||||||
connect(watcher, &QFutureWatcher<PendingTransaction*>::finished,
|
|
||||||
this, [this, watcher,dst_addr,payment_id,mixin_count]() {
|
|
||||||
QFuture<PendingTransaction*> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit transactionCreated(future.result(),dst_addr,payment_id,mixin_count);
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PendingTransaction *Wallet::createTransactionAll(const QString &dst_addr, const QString &payment_id,
|
PendingTransaction *Wallet::createTransactionAll(const QString &dst_addr, const QString &payment_id,
|
||||||
|
@ -486,17 +483,10 @@ void Wallet::createTransactionAllAsync(const QString &dst_addr, const QString &p
|
||||||
quint32 mixin_count,
|
quint32 mixin_count,
|
||||||
PendingTransaction::Priority priority)
|
PendingTransaction::Priority priority)
|
||||||
{
|
{
|
||||||
QFuture<PendingTransaction*> future = QtConcurrent::run(this, &Wallet::createTransactionAll,
|
m_scheduler.run([this, dst_addr, payment_id, mixin_count, priority] {
|
||||||
dst_addr, payment_id, mixin_count, priority);
|
PendingTransaction *tx = createTransactionAll(dst_addr, payment_id, mixin_count, priority);
|
||||||
QFutureWatcher<PendingTransaction*> * watcher = new QFutureWatcher<PendingTransaction*>();
|
emit transactionCreated(tx, dst_addr, payment_id, mixin_count);
|
||||||
|
|
||||||
connect(watcher, &QFutureWatcher<PendingTransaction*>::finished,
|
|
||||||
this, [this, watcher,dst_addr,payment_id,mixin_count]() {
|
|
||||||
QFuture<PendingTransaction*> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit transactionCreated(future.result(),dst_addr,payment_id,mixin_count);
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PendingTransaction *Wallet::createSweepUnmixableTransaction()
|
PendingTransaction *Wallet::createSweepUnmixableTransaction()
|
||||||
|
@ -508,16 +498,10 @@ PendingTransaction *Wallet::createSweepUnmixableTransaction()
|
||||||
|
|
||||||
void Wallet::createSweepUnmixableTransactionAsync()
|
void Wallet::createSweepUnmixableTransactionAsync()
|
||||||
{
|
{
|
||||||
QFuture<PendingTransaction*> future = QtConcurrent::run(this, &Wallet::createSweepUnmixableTransaction);
|
m_scheduler.run([this] {
|
||||||
QFutureWatcher<PendingTransaction*> * watcher = new QFutureWatcher<PendingTransaction*>();
|
PendingTransaction *tx = createSweepUnmixableTransaction();
|
||||||
|
emit transactionCreated(tx, "", "", 0);
|
||||||
connect(watcher, &QFutureWatcher<PendingTransaction*>::finished,
|
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<PendingTransaction*> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit transactionCreated(future.result(),"","",0);
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
UnsignedTransaction * Wallet::loadTxFile(const QString &fileName)
|
UnsignedTransaction * Wallet::loadTxFile(const QString &fileName)
|
||||||
|
@ -539,18 +523,9 @@ bool Wallet::submitTxFile(const QString &fileName) const
|
||||||
|
|
||||||
void Wallet::commitTransactionAsync(PendingTransaction *t)
|
void Wallet::commitTransactionAsync(PendingTransaction *t)
|
||||||
{
|
{
|
||||||
QStringList txid(t->txid());
|
m_scheduler.run([this, t] {
|
||||||
QFuture<bool> future = QtConcurrent::run(t, &PendingTransaction::commit);
|
emit transactionCommitted(t->commit(), t, t->txid());
|
||||||
|
|
||||||
QFutureWatcher<bool> * watcher = new QFutureWatcher<bool>();
|
|
||||||
|
|
||||||
connect(watcher, &QFutureWatcher<bool>::finished,
|
|
||||||
this, [this, watcher, t, txid]() {
|
|
||||||
QFuture<bool> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit transactionCommitted(future.result(), t, txid);
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Wallet::disposeTransaction(PendingTransaction *t)
|
void Wallet::disposeTransaction(PendingTransaction *t)
|
||||||
|
@ -662,23 +637,11 @@ QString Wallet::getTxKey(const QString &txid) const
|
||||||
return QString::fromStdString(m_walletImpl->getTxKey(txid.toStdString()));
|
return QString::fromStdString(m_walletImpl->getTxKey(txid.toStdString()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void Wallet::getTxKeyAsync(const QString &txid, const QJSValue &ref)
|
void Wallet::getTxKeyAsync(const QString &txid, const QJSValue &callback)
|
||||||
{
|
{
|
||||||
QFuture<QString> future = QtConcurrent::run(this, &Wallet::getTxKey, txid);
|
m_scheduler.run([this, txid] {
|
||||||
auto watcher = new QFutureWatcher<QString>(this);
|
return QJSValueList({txid, getTxKey(txid)});
|
||||||
|
}, callback);
|
||||||
connect(watcher, &QFutureWatcher<QString>::finished,
|
|
||||||
this, [watcher, txid, ref]() {
|
|
||||||
QFuture<QString> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
|
|
||||||
auto txKey = future.result();
|
|
||||||
if (ref.isCallable()){
|
|
||||||
QJSValue cb(ref);
|
|
||||||
cb.call(QJSValueList {txid, txKey});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QString Wallet::checkTxKey(const QString &txid, const QString &tx_key, const QString &address)
|
QString Wallet::checkTxKey(const QString &txid, const QString &tx_key, const QString &address)
|
||||||
|
@ -699,23 +662,11 @@ QString Wallet::getTxProof(const QString &txid, const QString &address, const QS
|
||||||
return QString::fromStdString(result);
|
return QString::fromStdString(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Wallet::getTxProofAsync(const QString &txid, const QString &address, const QString &message, const QJSValue &ref)
|
void Wallet::getTxProofAsync(const QString &txid, const QString &address, const QString &message, const QJSValue &callback)
|
||||||
{
|
{
|
||||||
QFuture<QString> future = QtConcurrent::run(this, &Wallet::getTxProof, txid, address, message);
|
m_scheduler.run([this, txid, address, message] {
|
||||||
auto watcher = new QFutureWatcher<QString>(this);
|
return QJSValueList({txid, getTxProof(txid, address, message)});
|
||||||
|
}, callback);
|
||||||
connect(watcher, &QFutureWatcher<QString>::finished,
|
|
||||||
this, [watcher, txid, ref]() {
|
|
||||||
QFuture<QString> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
|
|
||||||
auto proof = future.result();
|
|
||||||
if (ref.isCallable()){
|
|
||||||
QJSValue cb(ref);
|
|
||||||
cb.call(QJSValueList {txid, proof});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QString Wallet::checkTxProof(const QString &txid, const QString &address, const QString &message, const QString &signature)
|
QString Wallet::checkTxProof(const QString &txid, const QString &address, const QString &message, const QString &signature)
|
||||||
|
@ -737,23 +688,11 @@ Q_INVOKABLE QString Wallet::getSpendProof(const QString &txid, const QString &me
|
||||||
return QString::fromStdString(result);
|
return QString::fromStdString(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Wallet::getSpendProofAsync(const QString &txid, const QString &message, const QJSValue &ref)
|
void Wallet::getSpendProofAsync(const QString &txid, const QString &message, const QJSValue &callback)
|
||||||
{
|
{
|
||||||
QFuture<QString> future = QtConcurrent::run(this, &Wallet::getSpendProof, txid, message);
|
m_scheduler.run([this, txid, message] {
|
||||||
auto watcher = new QFutureWatcher<QString>(this);
|
return QJSValueList({txid, getSpendProof(txid, message)});
|
||||||
|
}, callback);
|
||||||
connect(watcher, &QFutureWatcher<QString>::finished,
|
|
||||||
this, [watcher, txid, ref]() {
|
|
||||||
QFuture<QString> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
|
|
||||||
auto proof = future.result();
|
|
||||||
if (ref.isCallable()){
|
|
||||||
QJSValue cb(ref);
|
|
||||||
cb.call(QJSValueList {txid, proof});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Q_INVOKABLE QString Wallet::checkSpendProof(const QString &txid, const QString &message, const QString &signature) const
|
Q_INVOKABLE QString Wallet::checkSpendProof(const QString &txid, const QString &message, const QString &signature) const
|
||||||
|
@ -1007,6 +946,7 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
||||||
, m_daemonBlockChainTargetHeightTtl(DAEMON_BLOCKCHAIN_TARGET_HEIGHT_CACHE_TTL_SECONDS)
|
, m_daemonBlockChainTargetHeightTtl(DAEMON_BLOCKCHAIN_TARGET_HEIGHT_CACHE_TTL_SECONDS)
|
||||||
, m_connectionStatusTtl(WALLET_CONNECTION_STATUS_CACHE_TTL_SECONDS)
|
, m_connectionStatusTtl(WALLET_CONNECTION_STATUS_CACHE_TTL_SECONDS)
|
||||||
, m_currentSubaddressAccount(0)
|
, m_currentSubaddressAccount(0)
|
||||||
|
, m_scheduler(this)
|
||||||
{
|
{
|
||||||
m_history = new TransactionHistory(m_walletImpl->history(), this);
|
m_history = new TransactionHistory(m_walletImpl->history(), this);
|
||||||
m_addressBook = new AddressBook(m_walletImpl->addressBook(), this);
|
m_addressBook = new AddressBook(m_walletImpl->addressBook(), this);
|
||||||
|
@ -1028,6 +968,9 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
||||||
Wallet::~Wallet()
|
Wallet::~Wallet()
|
||||||
{
|
{
|
||||||
qDebug("~Wallet: Closing wallet");
|
qDebug("~Wallet: Closing wallet");
|
||||||
|
|
||||||
|
m_scheduler.shutdownWaitForFinished();
|
||||||
|
|
||||||
delete m_addressBook;
|
delete m_addressBook;
|
||||||
m_addressBook = NULL;
|
m_addressBook = NULL;
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,7 @@
|
||||||
#include <QtConcurrent/QtConcurrent>
|
#include <QtConcurrent/QtConcurrent>
|
||||||
|
|
||||||
#include "wallet/api/wallet2_api.h" // we need to have an access to the Monero::Wallet::Status enum here;
|
#include "wallet/api/wallet2_api.h" // we need to have an access to the Monero::Wallet::Status enum here;
|
||||||
|
#include "qt/FutureScheduler.h"
|
||||||
#include "PendingTransaction.h" // we need to have an access to the PendingTransaction::Priority enum here;
|
#include "PendingTransaction.h" // we need to have an access to the PendingTransaction::Priority enum here;
|
||||||
#include "UnsignedTransaction.h"
|
#include "UnsignedTransaction.h"
|
||||||
#include "NetworkType.h"
|
#include "NetworkType.h"
|
||||||
|
@ -182,7 +183,7 @@ public:
|
||||||
//! returns if view only wallet
|
//! returns if view only wallet
|
||||||
Q_INVOKABLE bool viewOnly() const;
|
Q_INVOKABLE bool viewOnly() const;
|
||||||
|
|
||||||
Q_INVOKABLE void refreshHeightAsync() const;
|
Q_INVOKABLE void refreshHeightAsync();
|
||||||
|
|
||||||
//! export/import key images
|
//! export/import key images
|
||||||
Q_INVOKABLE bool exportKeyImages(const QString& path);
|
Q_INVOKABLE bool exportKeyImages(const QString& path);
|
||||||
|
@ -290,13 +291,13 @@ public:
|
||||||
Q_INVOKABLE bool setUserNote(const QString &txid, const QString ¬e);
|
Q_INVOKABLE bool setUserNote(const QString &txid, const QString ¬e);
|
||||||
Q_INVOKABLE QString getUserNote(const QString &txid) const;
|
Q_INVOKABLE QString getUserNote(const QString &txid) const;
|
||||||
Q_INVOKABLE QString getTxKey(const QString &txid) const;
|
Q_INVOKABLE QString getTxKey(const QString &txid) const;
|
||||||
Q_INVOKABLE void getTxKeyAsync(const QString &txid, const QJSValue &ref);
|
Q_INVOKABLE void getTxKeyAsync(const QString &txid, const QJSValue &callback);
|
||||||
Q_INVOKABLE QString checkTxKey(const QString &txid, const QString &tx_key, const QString &address);
|
Q_INVOKABLE QString checkTxKey(const QString &txid, const QString &tx_key, const QString &address);
|
||||||
Q_INVOKABLE QString getTxProof(const QString &txid, const QString &address, const QString &message) const;
|
Q_INVOKABLE QString getTxProof(const QString &txid, const QString &address, const QString &message) const;
|
||||||
Q_INVOKABLE void getTxProofAsync(const QString &txid, const QString &address, const QString &message, const QJSValue &ref);
|
Q_INVOKABLE void getTxProofAsync(const QString &txid, const QString &address, const QString &message, const QJSValue &callback);
|
||||||
Q_INVOKABLE QString checkTxProof(const QString &txid, const QString &address, const QString &message, const QString &signature);
|
Q_INVOKABLE QString checkTxProof(const QString &txid, const QString &address, const QString &message, const QString &signature);
|
||||||
Q_INVOKABLE QString getSpendProof(const QString &txid, const QString &message) const;
|
Q_INVOKABLE QString getSpendProof(const QString &txid, const QString &message) const;
|
||||||
Q_INVOKABLE void getSpendProofAsync(const QString &txid, const QString &message, const QJSValue &ref);
|
Q_INVOKABLE void getSpendProofAsync(const QString &txid, const QString &message, const QJSValue &callback);
|
||||||
Q_INVOKABLE QString checkSpendProof(const QString &txid, const QString &message, const QString &signature) const;
|
Q_INVOKABLE QString checkSpendProof(const QString &txid, const QString &message, const QString &signature) const;
|
||||||
// Rescan spent outputs
|
// Rescan spent outputs
|
||||||
Q_INVOKABLE bool rescanSpent();
|
Q_INVOKABLE bool rescanSpent();
|
||||||
|
@ -356,7 +357,7 @@ signals:
|
||||||
// emitted when transaction is created async
|
// emitted when transaction is created async
|
||||||
void transactionCreated(PendingTransaction * transaction, QString address, QString paymentId, quint32 mixinCount);
|
void transactionCreated(PendingTransaction * transaction, QString address, QString paymentId, quint32 mixinCount);
|
||||||
|
|
||||||
void connectionStatusChanged(ConnectionStatus status) const;
|
void connectionStatusChanged(int status) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Wallet(QObject * parent = nullptr);
|
Wallet(QObject * parent = nullptr);
|
||||||
|
@ -406,6 +407,7 @@ private:
|
||||||
QString m_daemonUsername;
|
QString m_daemonUsername;
|
||||||
QString m_daemonPassword;
|
QString m_daemonPassword;
|
||||||
Monero::WalletListener *m_walletListener;
|
Monero::WalletListener *m_walletListener;
|
||||||
|
FutureScheduler m_scheduler;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -145,17 +145,9 @@ Wallet *WalletManager::openWallet(const QString &path, const QString &password,
|
||||||
|
|
||||||
void WalletManager::openWalletAsync(const QString &path, const QString &password, NetworkType::Type nettype, quint64 kdfRounds)
|
void WalletManager::openWalletAsync(const QString &path, const QString &password, NetworkType::Type nettype, quint64 kdfRounds)
|
||||||
{
|
{
|
||||||
QFuture<Wallet*> future = QtConcurrent::run(this, &WalletManager::openWallet,
|
m_scheduler.run([this, path, password, nettype, kdfRounds] {
|
||||||
path, password, nettype, kdfRounds);
|
emit walletOpened(openWallet(path, password, nettype, kdfRounds));
|
||||||
QFutureWatcher<Wallet*> * watcher = new QFutureWatcher<Wallet*>();
|
|
||||||
|
|
||||||
connect(watcher, &QFutureWatcher<Wallet*>::finished,
|
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<Wallet*> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit walletOpened(future.result());
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -216,21 +208,10 @@ Wallet *WalletManager::createWalletFromDevice(const QString &path, const QString
|
||||||
void WalletManager::createWalletFromDeviceAsync(const QString &path, const QString &password, NetworkType::Type nettype,
|
void WalletManager::createWalletFromDeviceAsync(const QString &path, const QString &password, NetworkType::Type nettype,
|
||||||
const QString &deviceName, quint64 restoreHeight, const QString &subaddressLookahead)
|
const QString &deviceName, quint64 restoreHeight, const QString &subaddressLookahead)
|
||||||
{
|
{
|
||||||
auto lmbd = [=](){
|
m_scheduler.run([this, path, password, nettype, deviceName, restoreHeight, subaddressLookahead] {
|
||||||
return this->createWalletFromDevice(path, password, nettype, deviceName, restoreHeight, subaddressLookahead);
|
Wallet *wallet = createWalletFromDevice(path, password, nettype, deviceName, restoreHeight, subaddressLookahead);
|
||||||
};
|
emit walletCreated(wallet);
|
||||||
|
|
||||||
QFuture<Wallet *> future = QtConcurrent::run(lmbd);
|
|
||||||
|
|
||||||
QFutureWatcher<Wallet *> * watcher = new QFutureWatcher<Wallet *>();
|
|
||||||
|
|
||||||
connect(watcher, &QFutureWatcher<Wallet *>::finished,
|
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<Wallet *> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit walletCreated(future.result());
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QString WalletManager::closeWallet()
|
QString WalletManager::closeWallet()
|
||||||
|
@ -249,16 +230,9 @@ QString WalletManager::closeWallet()
|
||||||
|
|
||||||
void WalletManager::closeWalletAsync()
|
void WalletManager::closeWalletAsync()
|
||||||
{
|
{
|
||||||
QFuture<QString> future = QtConcurrent::run(this, &WalletManager::closeWallet);
|
m_scheduler.run([this] {
|
||||||
QFutureWatcher<QString> * watcher = new QFutureWatcher<QString>();
|
emit walletClosed(closeWallet());
|
||||||
|
|
||||||
connect(watcher, &QFutureWatcher<QString>::finished,
|
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<QString> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
emit walletClosed(future.result());
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool WalletManager::walletExists(const QString &path) const
|
bool WalletManager::walletExists(const QString &path) const
|
||||||
|
@ -333,7 +307,7 @@ QString WalletManager::paymentIdFromAddress(const QString &address, NetworkType:
|
||||||
|
|
||||||
void WalletManager::setDaemonAddressAsync(const QString &address)
|
void WalletManager::setDaemonAddressAsync(const QString &address)
|
||||||
{
|
{
|
||||||
QtConcurrent::run([this, address] {
|
m_scheduler.run([this, address] {
|
||||||
m_pimpl->setDaemonAddress(address.toStdString());
|
m_pimpl->setDaemonAddress(address.toStdString());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -376,9 +350,9 @@ bool WalletManager::isMining() const
|
||||||
return m_pimpl->isMining();
|
return m_pimpl->isMining();
|
||||||
}
|
}
|
||||||
|
|
||||||
void WalletManager::miningStatusAsync() const
|
void WalletManager::miningStatusAsync()
|
||||||
{
|
{
|
||||||
QtConcurrent::run([this] {
|
m_scheduler.run([this] {
|
||||||
emit miningStatus(isMining());
|
emit miningStatus(isMining());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -488,19 +462,11 @@ bool WalletManager::saveQrCode(const QString &code, const QString &path) const
|
||||||
return QRCodeImageProvider::genQrImage(code, &size).scaled(size.expandedTo(QSize(240, 240)), Qt::KeepAspectRatio).save(path, "PNG", 100);
|
return QRCodeImageProvider::genQrImage(code, &size).scaled(size.expandedTo(QSize(240, 240)), Qt::KeepAspectRatio).save(path, "PNG", 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
void WalletManager::checkUpdatesAsync(const QString &software, const QString &subdir) const
|
void WalletManager::checkUpdatesAsync(const QString &software, const QString &subdir)
|
||||||
{
|
{
|
||||||
QFuture<QString> future = QtConcurrent::run(this, &WalletManager::checkUpdates,
|
m_scheduler.run([this, software, subdir] {
|
||||||
software, subdir);
|
emit checkUpdatesComplete(checkUpdates(software, subdir));
|
||||||
QFutureWatcher<QString> * watcher = new QFutureWatcher<QString>();
|
|
||||||
connect(watcher, &QFutureWatcher<Wallet*>::finished,
|
|
||||||
this, [this, watcher]() {
|
|
||||||
QFuture<QString> future = watcher->future();
|
|
||||||
watcher->deleteLater();
|
|
||||||
qDebug() << "Checking for updates - done";
|
|
||||||
emit checkUpdatesComplete(future.result());
|
|
||||||
});
|
});
|
||||||
watcher->setFuture(future);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -532,11 +498,18 @@ bool WalletManager::clearWalletCache(const QString &wallet_path) const
|
||||||
return walletCache.rename(newFileName);
|
return walletCache.rename(newFileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
WalletManager::WalletManager(QObject *parent) : QObject(parent)
|
WalletManager::WalletManager(QObject *parent)
|
||||||
|
: QObject(parent)
|
||||||
|
, m_scheduler(this)
|
||||||
{
|
{
|
||||||
m_pimpl = Monero::WalletManagerFactory::getWalletManager();
|
m_pimpl = Monero::WalletManagerFactory::getWalletManager();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
WalletManager::~WalletManager()
|
||||||
|
{
|
||||||
|
m_scheduler.shutdownWaitForFinished();
|
||||||
|
}
|
||||||
|
|
||||||
void WalletManager::onWalletPassphraseNeeded(Monero::Wallet *)
|
void WalletManager::onWalletPassphraseNeeded(Monero::Wallet *)
|
||||||
{
|
{
|
||||||
m_mutex_pass.lock();
|
m_mutex_pass.lock();
|
||||||
|
|
|
@ -37,6 +37,7 @@
|
||||||
#include <QPointer>
|
#include <QPointer>
|
||||||
#include <QWaitCondition>
|
#include <QWaitCondition>
|
||||||
#include <QMutex>
|
#include <QMutex>
|
||||||
|
#include "qt/FutureScheduler.h"
|
||||||
#include "NetworkType.h"
|
#include "NetworkType.h"
|
||||||
|
|
||||||
class Wallet;
|
class Wallet;
|
||||||
|
@ -151,7 +152,7 @@ public:
|
||||||
Q_INVOKABLE bool localDaemonSynced() const;
|
Q_INVOKABLE bool localDaemonSynced() const;
|
||||||
Q_INVOKABLE bool isDaemonLocal(const QString &daemon_address) const;
|
Q_INVOKABLE bool isDaemonLocal(const QString &daemon_address) const;
|
||||||
|
|
||||||
Q_INVOKABLE void miningStatusAsync() const;
|
Q_INVOKABLE void miningStatusAsync();
|
||||||
Q_INVOKABLE bool startMining(const QString &address, quint32 threads, bool backgroundMining, bool ignoreBattery);
|
Q_INVOKABLE bool startMining(const QString &address, quint32 threads, bool backgroundMining, bool ignoreBattery);
|
||||||
Q_INVOKABLE bool stopMining();
|
Q_INVOKABLE bool stopMining();
|
||||||
|
|
||||||
|
@ -175,7 +176,7 @@ public:
|
||||||
Q_INVOKABLE bool parse_uri(const QString &uri, QString &address, QString &payment_id, uint64_t &amount, QString &tx_description, QString &recipient_name, QVector<QString> &unknown_parameters, QString &error) const;
|
Q_INVOKABLE bool parse_uri(const QString &uri, QString &address, QString &payment_id, uint64_t &amount, QString &tx_description, QString &recipient_name, QVector<QString> &unknown_parameters, QString &error) const;
|
||||||
Q_INVOKABLE QVariantMap parse_uri_to_object(const QString &uri) const;
|
Q_INVOKABLE QVariantMap parse_uri_to_object(const QString &uri) const;
|
||||||
Q_INVOKABLE bool saveQrCode(const QString &, const QString &) const;
|
Q_INVOKABLE bool saveQrCode(const QString &, const QString &) const;
|
||||||
Q_INVOKABLE void checkUpdatesAsync(const QString &software, const QString &subdir) const;
|
Q_INVOKABLE void checkUpdatesAsync(const QString &software, const QString &subdir);
|
||||||
Q_INVOKABLE QString checkUpdates(const QString &software, const QString &subdir) const;
|
Q_INVOKABLE QString checkUpdates(const QString &software, const QString &subdir) const;
|
||||||
|
|
||||||
// clear/rename wallet cache
|
// clear/rename wallet cache
|
||||||
|
@ -200,6 +201,7 @@ private:
|
||||||
friend class WalletPassphraseListenerImpl;
|
friend class WalletPassphraseListenerImpl;
|
||||||
|
|
||||||
explicit WalletManager(QObject *parent = 0);
|
explicit WalletManager(QObject *parent = 0);
|
||||||
|
~WalletManager();
|
||||||
|
|
||||||
bool isMining() const;
|
bool isMining() const;
|
||||||
|
|
||||||
|
@ -212,6 +214,8 @@ private:
|
||||||
QMutex m_mutex_pass;
|
QMutex m_mutex_pass;
|
||||||
QString m_passphrase;
|
QString m_passphrase;
|
||||||
bool m_passphrase_abort;
|
bool m_passphrase_abort;
|
||||||
|
|
||||||
|
FutureScheduler m_scheduler;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // WALLETMANAGER_H
|
#endif // WALLETMANAGER_H
|
||||||
|
|
89
src/qt/FutureScheduler.cpp
Normal file
89
src/qt/FutureScheduler.cpp
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
#include "FutureScheduler.h"
|
||||||
|
|
||||||
|
FutureScheduler::FutureScheduler(QObject *parent)
|
||||||
|
: QObject(parent), Alive(0), Stopping(false)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
FutureScheduler::~FutureScheduler()
|
||||||
|
{
|
||||||
|
shutdownWaitForFinished();
|
||||||
|
}
|
||||||
|
|
||||||
|
void FutureScheduler::shutdownWaitForFinished() noexcept
|
||||||
|
{
|
||||||
|
QMutexLocker locker(&Mutex);
|
||||||
|
|
||||||
|
Stopping = true;
|
||||||
|
while (Alive > 0)
|
||||||
|
{
|
||||||
|
Condition.wait(&Mutex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QPair<bool, QFuture<void>> FutureScheduler::run(std::function<void()> function) noexcept
|
||||||
|
{
|
||||||
|
return execute<void>([this, function](QFutureWatcher<void> *) {
|
||||||
|
return QtConcurrent::run([this, function] {
|
||||||
|
try
|
||||||
|
{
|
||||||
|
function();
|
||||||
|
}
|
||||||
|
catch (const std::exception &exception)
|
||||||
|
{
|
||||||
|
qWarning() << "Exception thrown from async function: " << exception.what();
|
||||||
|
}
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
QPair<bool, QFuture<QJSValueList>> FutureScheduler::run(std::function<QJSValueList() noexcept> function, const QJSValue &callback) noexcept
|
||||||
|
{
|
||||||
|
if (!callback.isCallable())
|
||||||
|
{
|
||||||
|
throw std::runtime_error("js callback must be callable");
|
||||||
|
}
|
||||||
|
|
||||||
|
return execute<QJSValueList>([this, function, callback](QFutureWatcher<QJSValueList> *watcher) {
|
||||||
|
connect(watcher, &QFutureWatcher<QJSValueList>::finished, [watcher, callback] {
|
||||||
|
QJSValue(callback).call(watcher->future().result());
|
||||||
|
});
|
||||||
|
return QtConcurrent::run([this, function] {
|
||||||
|
QJSValueList result;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
result = function();
|
||||||
|
}
|
||||||
|
catch (const std::exception &exception)
|
||||||
|
{
|
||||||
|
qWarning() << "Exception thrown from async function: " << exception.what();
|
||||||
|
}
|
||||||
|
done();
|
||||||
|
return result;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
bool FutureScheduler::add() noexcept
|
||||||
|
{
|
||||||
|
QMutexLocker locker(&Mutex);
|
||||||
|
|
||||||
|
if (Stopping)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
++Alive;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void FutureScheduler::done() noexcept
|
||||||
|
{
|
||||||
|
{
|
||||||
|
QMutexLocker locker(&Mutex);
|
||||||
|
--Alive;
|
||||||
|
}
|
||||||
|
|
||||||
|
Condition.wakeAll();
|
||||||
|
}
|
79
src/qt/FutureScheduler.h
Normal file
79
src/qt/FutureScheduler.h
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
#ifndef FUTURE_SCHEDULER_H
|
||||||
|
#define FUTURE_SCHEDULER_H
|
||||||
|
|
||||||
|
#include <functional>
|
||||||
|
|
||||||
|
#include <QtConcurrent/QtConcurrent>
|
||||||
|
#include <QFuture>
|
||||||
|
#include <QJSValue>
|
||||||
|
#include <QMutex>
|
||||||
|
#include <QMutexLocker>
|
||||||
|
#include <QPair>
|
||||||
|
#include <QWaitCondition>
|
||||||
|
|
||||||
|
class FutureScheduler : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
FutureScheduler(QObject *parent);
|
||||||
|
~FutureScheduler();
|
||||||
|
|
||||||
|
void shutdownWaitForFinished() noexcept;
|
||||||
|
|
||||||
|
QPair<bool, QFuture<void>> run(std::function<void()> function) noexcept;
|
||||||
|
QPair<bool, QFuture<QJSValueList>> run(std::function<QJSValueList() noexcept> function, const QJSValue &callback) noexcept;
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool add() noexcept;
|
||||||
|
void done() noexcept;
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
QFutureWatcher<T> *newWatcher()
|
||||||
|
{
|
||||||
|
QFutureWatcher<T> *watcher = new QFutureWatcher<T>();
|
||||||
|
QThread *schedulerThread = this->thread();
|
||||||
|
if (watcher->thread() != schedulerThread)
|
||||||
|
{
|
||||||
|
watcher->moveToThread(schedulerThread);
|
||||||
|
}
|
||||||
|
watcher->setParent(this);
|
||||||
|
|
||||||
|
return watcher;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
QPair<bool, QFuture<T>> execute(std::function<QFuture<T>(QFutureWatcher<T> *)> makeFuture) noexcept
|
||||||
|
{
|
||||||
|
if (add())
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
auto *watcher = newWatcher<T>();
|
||||||
|
watcher->setFuture(makeFuture(watcher));
|
||||||
|
connect(watcher, &QFutureWatcher<T>::finished, [this, watcher] {
|
||||||
|
watcher->deleteLater();
|
||||||
|
});
|
||||||
|
return qMakePair(true, watcher->future());
|
||||||
|
}
|
||||||
|
catch (const std::exception &exception)
|
||||||
|
{
|
||||||
|
qCritical() << "Failed to schedule async function: " << exception.what();
|
||||||
|
done();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return qMakePair(false, QFuture<T>());
|
||||||
|
}
|
||||||
|
|
||||||
|
QFutureWatcher<void> schedule(std::function<void()> function);
|
||||||
|
QFutureWatcher<QJSValueList> schedule(std::function<QJSValueList() noexcept> function, const QJSValue &callback);
|
||||||
|
|
||||||
|
private:
|
||||||
|
size_t Alive;
|
||||||
|
QWaitCondition Condition;
|
||||||
|
QMutex Mutex;
|
||||||
|
bool Stopping;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // FUTURE_SCHEDULER_H
|
Loading…
Reference in a new issue