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:
parent
0f5ba08fd2
commit
9971bc3a1e
@ -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,8 +1152,16 @@ 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);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -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};
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
};
|
};
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user