1
0
mirror of https://github.com/qTox/qTox.git synced 2024-03-22 14:00:36 +08:00

refactor(core): make construction independent of CoreAV

Allows to construct a Core object without also starting CoreAV.
This commit is contained in:
sudden6 2020-08-28 00:49:54 +02:00
parent 0f5ba08fd2
commit 9971bc3a1e
No known key found for this signature in database
GPG Key ID: 279509B499E032B9
6 changed files with 74 additions and 38 deletions

View File

@ -19,8 +19,8 @@
*/ */
#include "core.h" #include "core.h"
#include "coreav.h"
#include "corefile.h" #include "corefile.h"
#include "src/core/coreav.h"
#include "src/core/dhtserver.h" #include "src/core/dhtserver.h"
#include "src/core/icoresettings.h" #include "src/core/icoresettings.h"
#include "src/core/toxlogger.h" #include "src/core/toxlogger.h"
@ -495,7 +495,6 @@ Core::~Core()
coreThread->exit(0); coreThread->exit(0);
coreThread->wait(); coreThread->wait();
av.reset();
tox.reset(); tox.reset();
} }
@ -632,17 +631,6 @@ ToxCorePtr Core::makeToxCore(const QByteArray& savedata, const ICoreSettings* co
// tox should be valid by now // tox should be valid by now
assert(core->tox != nullptr); assert(core->tox != nullptr);
// toxcore is successfully created, create toxav
// TODO(sudden6): don't create CoreAv here, Core should be usable without CoreAV
core->av = CoreAV::makeCoreAV(core->tox.get(), core->coreLoopLock);
if (!core->av) {
qCritical() << "Toxav failed to start";
if (err) {
*err = ToxCoreErrors::FAILED_TO_START;
}
return {};
}
// create CoreFile // create CoreFile
core->file = CoreFile::makeCoreFile(core.get(), core->tox.get(), core->coreLoopLock); core->file = CoreFile::makeCoreFile(core.get(), core->tox.get(), core->coreLoopLock);
if (!core->file) { if (!core->file) {
@ -692,8 +680,6 @@ void Core::onStarted()
loadGroups(); loadGroups();
process(); // starts its own timer process(); // starts its own timer
av->start();
emit avReady();
} }
/** /**
@ -706,12 +692,17 @@ void Core::start()
const CoreAV* Core::getAv() const const CoreAV* Core::getAv() const
{ {
return av.get(); return av;
} }
CoreAV* Core::getAv() CoreAV* Core::getAv()
{ {
return av.get(); return av;
}
void Core::setAv(CoreAV *coreAv)
{
av = coreAv;
} }
CoreFile* Core::getCoreFile() const CoreFile* Core::getCoreFile() const
@ -719,6 +710,16 @@ CoreFile* Core::getCoreFile() const
return file.get(); return file.get();
} }
Tox* Core::getTox() const
{
return tox.get();
}
QMutex &Core::getCoreLoopLock() const
{
return coreLoopLock;
}
/* Using the now commented out statements in checkConnection(), I watched how /* Using the now commented out statements in checkConnection(), I watched how
* many ticks disconnects-after-initial-connect lasted. Out of roughly 15 trials, * many ticks disconnects-after-initial-connect lasted. Out of roughly 15 trials,
* 5 disconnected; 4 were DCd for less than 20 ticks, while the 5th was ~50 ticks. * 5 disconnected; 4 were DCd for less than 20 ticks, while the 5th was ~50 ticks.
@ -1151,9 +1152,17 @@ void Core::removeGroup(int groupId)
tox_conference_delete(tox.get(), groupId, &error); tox_conference_delete(tox.get(), groupId, &error);
if (PARSE_ERR(error)) { if (PARSE_ERR(error)) {
emit saveRequest(); emit saveRequest();
/*
* TODO(sudden6): this is probably not (thread-)safe, but can be ignored for now since
* we don't change av at runtime.
*/
if (av) {
av->leaveGroupCall(groupId); av->leaveGroupCall(groupId);
} }
} }
}
/** /**
* @brief Returns our username, or an empty string on failure * @brief Returns our username, or an empty string on failure

View File

@ -73,7 +73,12 @@ public:
IBootstrapListGenerator& bootstrapNodes, ToxCoreErrors* err = nullptr); IBootstrapListGenerator& bootstrapNodes, ToxCoreErrors* err = nullptr);
const CoreAV* getAv() const; const CoreAV* getAv() const;
CoreAV* getAv(); CoreAV* getAv();
void setAv(CoreAV* coreAv);
CoreFile* getCoreFile() const; CoreFile* getCoreFile() const;
Tox* getTox() const;
QMutex& getCoreLoopLock() const;
~Core(); ~Core();
static const QString TOX_EXT; static const QString TOX_EXT;
@ -150,8 +155,6 @@ signals:
void failedToSetStatus(Status::Status status); void failedToSetStatus(Status::Status status);
void failedToSetTyping(bool typing); void failedToSetTyping(bool typing);
void avReady();
void saveRequest(); void saveRequest();
/** /**
@ -245,7 +248,7 @@ private:
ToxPtr tox; ToxPtr tox;
std::unique_ptr<CoreFile> file; std::unique_ptr<CoreFile> file;
std::unique_ptr<CoreAV> av; CoreAV* av = nullptr;
QTimer* toxTimer = nullptr; QTimer* toxTimer = nullptr;
// recursive, since we might call our own functions // recursive, since we might call our own functions
mutable QMutex coreLoopLock{QMutex::Recursive}; mutable QMutex coreLoopLock{QMutex::Recursive};

View File

@ -19,10 +19,11 @@
*/ */
#include "coreav.h" #include "coreav.h"
#include "audio/iaudiosettings.h"
#include "core.h" #include "core.h"
#include "src/model/friend.h" #include "src/model/friend.h"
#include "src/model/group.h" #include "src/model/group.h"
#include "src/persistence/settings.h" #include "src/persistence/igroupsettings.h"
#include "src/video/corevideosource.h" #include "src/video/corevideosource.h"
#include "src/video/videoframe.h" #include "src/video/videoframe.h"
#include <QCoreApplication> #include <QCoreApplication>
@ -68,12 +69,15 @@
* deadlock. * deadlock.
*/ */
CoreAV::CoreAV(std::unique_ptr<ToxAV, ToxAVDeleter> toxav, QMutex& toxCoreLock) CoreAV::CoreAV(std::unique_ptr<ToxAV, ToxAVDeleter> toxav, QMutex& toxCoreLock,
IAudioSettings& _audioSettings, IGroupSettings& _groupSettings)
: audio{nullptr} : audio{nullptr}
, toxav{std::move(toxav)} , toxav{std::move(toxav)}
, coreavThread{new QThread{this}} , coreavThread{new QThread{this}}
, iterateTimer{new QTimer{this}} , iterateTimer{new QTimer{this}}
, coreLock{toxCoreLock} , coreLock{toxCoreLock}
, audioSettings{_audioSettings}
, groupSettings{_groupSettings}
{ {
assert(coreavThread); assert(coreavThread);
assert(iterateTimer); assert(iterateTimer);
@ -105,7 +109,8 @@ void CoreAV::connectCallbacks(ToxAV& toxav)
* @param core pointer to the Tox instance * @param core pointer to the Tox instance
* @return CoreAV instance on success, {} on failure * @return CoreAV instance on success, {} on failure
*/ */
CoreAV::CoreAVPtr CoreAV::makeCoreAV(Tox* core, QMutex &toxCoreLock) CoreAV::CoreAVPtr CoreAV::makeCoreAV(Tox* core, QMutex& toxCoreLock,
IAudioSettings& audioSettings, IGroupSettings& groupSettings)
{ {
Toxav_Err_New err; Toxav_Err_New err;
std::unique_ptr<ToxAV, ToxAVDeleter> toxav{toxav_new(core, &err)}; std::unique_ptr<ToxAV, ToxAVDeleter> toxav{toxav_new(core, &err)};
@ -125,7 +130,7 @@ CoreAV::CoreAVPtr CoreAV::makeCoreAV(Tox* core, QMutex &toxCoreLock)
assert(toxav != nullptr); assert(toxav != nullptr);
return CoreAVPtr{new CoreAV{std::move(toxav), toxCoreLock}}; return CoreAVPtr{new CoreAV{std::move(toxav), toxCoreLock, audioSettings, groupSettings}};
} }
/** /**
@ -251,7 +256,7 @@ bool CoreAV::answerCall(uint32_t friendNum, bool video)
Toxav_Err_Answer err; Toxav_Err_Answer err;
const uint32_t videoBitrate = video ? VIDEO_DEFAULT_BITRATE : 0; const uint32_t videoBitrate = video ? VIDEO_DEFAULT_BITRATE : 0;
if (toxav_answer(toxav.get(), friendNum, Settings::getInstance().getAudioBitrate(), if (toxav_answer(toxav.get(), friendNum, audioSettings.getAudioBitrate(),
videoBitrate, &err)) { videoBitrate, &err)) {
it->second->setActive(true); it->second->setActive(true);
return true; return true;
@ -276,7 +281,7 @@ bool CoreAV::startCall(uint32_t friendNum, bool video)
} }
uint32_t videoBitrate = video ? VIDEO_DEFAULT_BITRATE : 0; uint32_t videoBitrate = video ? VIDEO_DEFAULT_BITRATE : 0;
if (!toxav_call(toxav.get(), friendNum, Settings::getInstance().getAudioBitrate(), videoBitrate, if (!toxav_call(toxav.get(), friendNum, audioSettings.getAudioBitrate(), videoBitrate,
nullptr)) nullptr))
return false; return false;
@ -476,9 +481,8 @@ void CoreAV::groupCallCallback(void* tox, uint32_t group, uint32_t peer, const i
QReadLocker locker{&cav->callsLock}; QReadLocker locker{&cav->callsLock};
const ToxPk peerPk = c->getGroupPeerPk(group, peer); const ToxPk peerPk = c->getGroupPeerPk(group, peer);
const Settings& s = Settings::getInstance();
// don't play the audio if it comes from a muted peer // don't play the audio if it comes from a muted peer
if (s.getBlackList().contains(peerPk.toString())) { if (cav->groupSettings.getBlackList().contains(peerPk.toString())) {
return; return;
} }

View File

@ -21,6 +21,7 @@
#pragma once #pragma once
#include "src/core/toxcall.h" #include "src/core/toxcall.h"
#include <QObject> #include <QObject>
#include <QMutex> #include <QMutex>
#include <QReadWriteLock> #include <QReadWriteLock>
@ -31,6 +32,8 @@
class Friend; class Friend;
class Group; class Group;
class IAudioControl; class IAudioControl;
class IAudioSettings;
class IGroupSettings;
class QThread; class QThread;
class QTimer; class QTimer;
class CoreVideoSource; class CoreVideoSource;
@ -46,7 +49,8 @@ class CoreAV : public QObject
public: public:
using CoreAVPtr = std::unique_ptr<CoreAV>; using CoreAVPtr = std::unique_ptr<CoreAV>;
static CoreAVPtr makeCoreAV(Tox* core, QMutex& coreLock); static CoreAVPtr makeCoreAV(Tox* core, QMutex& toxCoreLock,
IAudioSettings& audioSettings, IGroupSettings& groupSettings);
void setAudio(IAudioControl& newAudio); void setAudio(IAudioControl& newAudio);
IAudioControl* getAudio(); IAudioControl* getAudio();
@ -112,7 +116,8 @@ private:
} }
}; };
CoreAV(std::unique_ptr<ToxAV, ToxAVDeleter> tox, QMutex &toxCoreLock); CoreAV(std::unique_ptr<ToxAV, ToxAVDeleter> tox, QMutex &toxCoreLock,
IAudioSettings& _audioSettings, IGroupSettings& _groupSettings);
void connectCallbacks(ToxAV& toxav); void connectCallbacks(ToxAV& toxav);
void process(); void process();
@ -156,4 +161,7 @@ private:
* @note This must be a recursive mutex as we're going to lock it in callbacks * @note This must be a recursive mutex as we're going to lock it in callbacks
*/ */
QMutex& coreLock; QMutex& coreLock;
IAudioSettings& audioSettings;
IGroupSettings& groupSettings;
}; };

View File

@ -232,7 +232,7 @@ bool logCreateToxDataError(const CreateToxDataError& error, const QString& userN
QStringList Profile::profiles; QStringList Profile::profiles;
void Profile::initCore(const QByteArray& toxsave, const ICoreSettings& s, bool isNewProfile) void Profile::initCore(const QByteArray& toxsave, Settings& s, bool isNewProfile)
{ {
if (toxsave.isEmpty() && !isNewProfile) { if (toxsave.isEmpty() && !isNewProfile) {
qCritical() << "Existing toxsave is empty"; qCritical() << "Existing toxsave is empty";
@ -245,7 +245,7 @@ void Profile::initCore(const QByteArray& toxsave, const ICoreSettings& s, bool i
} }
bootstrapNodes = std::unique_ptr<BootstrapNodeUpdater>( bootstrapNodes = std::unique_ptr<BootstrapNodeUpdater>(
new BootstrapNodeUpdater(Settings::getInstance().getProxy(), paths)); new BootstrapNodeUpdater(s.getProxy(), paths));
Core::ToxCoreErrors err; Core::ToxCoreErrors err;
core = Core::makeToxCore(toxsave, &s, *bootstrapNodes, &err); core = Core::makeToxCore(toxsave, &s, *bootstrapNodes, &err);
@ -265,6 +265,17 @@ void Profile::initCore(const QByteArray& toxsave, const ICoreSettings& s, bool i
return; return;
} }
coreAv = CoreAV::makeCoreAV(core->getTox(), core->getCoreLoopLock(), s, s);
if (!coreAv) {
qDebug() << "Failed to start ToxAV";
emit failedToStart();
return;
}
// Tell Core that we run with AV before doing anything else
core->setAv(coreAv.get());
coreAv->start();
if (isNewProfile) { if (isNewProfile) {
core->setStatusMessage(tr("Toxing on qTox")); core->setStatusMessage(tr("Toxing on qTox"));
core->setUsername(name); core->setUsername(name);
@ -282,7 +293,7 @@ void Profile::initCore(const QByteArray& toxsave, const ICoreSettings& s, bool i
avatarBroadcaster = std::unique_ptr<AvatarBroadcaster>(new AvatarBroadcaster(*core)); avatarBroadcaster = std::unique_ptr<AvatarBroadcaster>(new AvatarBroadcaster(*core));
} }
Profile::Profile(const QString& name, const QString& password, std::unique_ptr<ToxEncrypt> passkey, Paths& paths_) Profile::Profile(const QString& name, std::unique_ptr<ToxEncrypt> passkey, Paths& paths_)
: name{name} : name{name}
, passkey{std::move(passkey)} , passkey{std::move(passkey)}
, isRemoved{false} , isRemoved{false}
@ -320,7 +331,7 @@ Profile* Profile::loadProfile(const QString& name, const QString& password, Sett
return nullptr; return nullptr;
} }
Profile* p = new Profile(name, password, std::move(tmpKey), settings.getPaths()); Profile* p = new Profile(name, std::move(tmpKey), settings.getPaths());
// Core settings are saved per profile, need to load them before starting Core // Core settings are saved per profile, need to load them before starting Core
settings.updateProfileData(p, parser); settings.updateProfileData(p, parser);
@ -351,7 +362,7 @@ Profile* Profile::createProfile(const QString& name, const QString& password, Se
} }
settings.createPersonal(name); settings.createPersonal(name);
Profile* p = new Profile(name, password, std::move(tmpKey), settings.getPaths()); Profile* p = new Profile(name, std::move(tmpKey), settings.getPaths());
settings.updateProfileData(p, parser); settings.updateProfileData(p, parser);
p->initCore(QByteArray(), settings, /*isNewProfile*/ true); p->initCore(QByteArray(), settings, /*isNewProfile*/ true);

View File

@ -104,15 +104,16 @@ private slots:
void onAvatarOfferReceived(uint32_t friendId, uint32_t fileId, const QByteArray& avatarHash); void onAvatarOfferReceived(uint32_t friendId, uint32_t fileId, const QByteArray& avatarHash);
private: private:
Profile(const QString& name, const QString& password, std::unique_ptr<ToxEncrypt> passkey, Paths& paths); Profile(const QString& name, std::unique_ptr<ToxEncrypt> passkey, Paths& paths);
static QStringList getFilesByExt(QString extension); static QStringList getFilesByExt(QString extension);
QString avatarPath(const ToxPk& owner, bool forceUnencrypted = false); QString avatarPath(const ToxPk& owner, bool forceUnencrypted = false);
bool saveToxSave(QByteArray data); bool saveToxSave(QByteArray data);
void initCore(const QByteArray& toxsave, const ICoreSettings& s, bool isNewProfile); void initCore(const QByteArray& toxsave, Settings &s, bool isNewProfile);
private: private:
std::unique_ptr<AvatarBroadcaster> avatarBroadcaster; std::unique_ptr<AvatarBroadcaster> avatarBroadcaster;
std::unique_ptr<Core> core; std::unique_ptr<Core> core;
std::unique_ptr<CoreAV> coreAv;
QString name; QString name;
std::unique_ptr<ToxEncrypt> passkey; std::unique_ptr<ToxEncrypt> passkey;
std::shared_ptr<RawDatabase> database; std::shared_ptr<RawDatabase> database;