mirror of
https://github.com/qTox/qTox.git
synced 2024-03-22 14:00:36 +08:00
refactor(audio): move audio output to new IAudioSink interface
This commit is contained in:
parent
c61fcd1f2b
commit
5b908184fc
|
@ -236,6 +236,8 @@ execute_process(COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||||
set(${PROJECT_NAME}_SOURCES
|
set(${PROJECT_NAME}_SOURCES
|
||||||
src/audio/audio.cpp
|
src/audio/audio.cpp
|
||||||
src/audio/audio.h
|
src/audio/audio.h
|
||||||
|
src/audio/backend/alsink.cpp
|
||||||
|
src/audio/backend/alsink.h
|
||||||
src/audio/backend/openal.cpp
|
src/audio/backend/openal.cpp
|
||||||
src/audio/backend/openal.h
|
src/audio/backend/openal.h
|
||||||
src/audio/iaudiosettings.h
|
src/audio/iaudiosettings.h
|
||||||
|
|
|
@ -23,39 +23,12 @@
|
||||||
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
|
||||||
#include <cassert>
|
class IAudioSink;
|
||||||
|
|
||||||
class Audio : public QObject
|
class Audio : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
enum class Sound
|
|
||||||
{
|
|
||||||
NewMessage,
|
|
||||||
Test,
|
|
||||||
IncomingCall,
|
|
||||||
OutgoingCall,
|
|
||||||
CallEnd
|
|
||||||
};
|
|
||||||
|
|
||||||
inline static QString getSound(Sound s)
|
|
||||||
{
|
|
||||||
switch (s) {
|
|
||||||
case Sound::Test:
|
|
||||||
return QStringLiteral(":/audio/notification.s16le.pcm");
|
|
||||||
case Sound::NewMessage:
|
|
||||||
return QStringLiteral(":/audio/notification.s16le.pcm");
|
|
||||||
case Sound::IncomingCall:
|
|
||||||
return QStringLiteral(":/audio/ToxIncomingCall.s16le.pcm");
|
|
||||||
case Sound::OutgoingCall:
|
|
||||||
return QStringLiteral(":/audio/ToxOutgoingCall.s16le.pcm");
|
|
||||||
case Sound::CallEnd:
|
|
||||||
return QStringLiteral(":/audio/ToxEndCall.s16le.pcm");
|
|
||||||
}
|
|
||||||
assert(false);
|
|
||||||
return QString();
|
|
||||||
}
|
|
||||||
static Audio& getInstance();
|
static Audio& getInstance();
|
||||||
|
|
||||||
virtual qreal outputVolume() const = 0;
|
virtual qreal outputVolume() const = 0;
|
||||||
|
@ -86,21 +59,12 @@ public:
|
||||||
virtual QStringList outDeviceNames() = 0;
|
virtual QStringList outDeviceNames() = 0;
|
||||||
virtual QStringList inDeviceNames() = 0;
|
virtual QStringList inDeviceNames() = 0;
|
||||||
|
|
||||||
virtual void subscribeOutput(uint& sourceId) = 0;
|
|
||||||
virtual void unsubscribeOutput(uint& sourceId) = 0;
|
|
||||||
|
|
||||||
virtual void subscribeInput() = 0;
|
virtual void subscribeInput() = 0;
|
||||||
virtual void unsubscribeInput() = 0;
|
virtual void unsubscribeInput() = 0;
|
||||||
|
|
||||||
virtual void startLoop() = 0;
|
|
||||||
virtual void stopLoop() = 0;
|
|
||||||
virtual void playMono16Sound(const QByteArray& data) = 0;
|
|
||||||
virtual void playMono16Sound(const QString& path) = 0;
|
|
||||||
|
|
||||||
virtual void stopActive() = 0;
|
virtual void stopActive() = 0;
|
||||||
|
|
||||||
virtual void playAudioBuffer(uint sourceId, const int16_t* data, int samples, unsigned channels,
|
virtual IAudioSink* makeSink() = 0;
|
||||||
int sampleRate) = 0;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// Public default audio settings
|
// Public default audio settings
|
||||||
|
|
94
src/audio/backend/alsink.cpp
Normal file
94
src/audio/backend/alsink.cpp
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
#include "src/audio/backend/alsink.h"
|
||||||
|
#include "src/audio/backend/openal.h"
|
||||||
|
|
||||||
|
#include <QDebug>
|
||||||
|
#include <QMutexLocker>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Can play audio via the speakers or some other audio device. Allocates
|
||||||
|
* and frees the audio ressources internally.
|
||||||
|
*/
|
||||||
|
|
||||||
|
AlSink::~AlSink()
|
||||||
|
{
|
||||||
|
QMutexLocker{&killLock};
|
||||||
|
|
||||||
|
// unsubscribe only if not already killed
|
||||||
|
if(!killed) {
|
||||||
|
audio.destroySink(*this);
|
||||||
|
killed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AlSink::playAudioBuffer(const int16_t *data, int samples, unsigned channels, int sampleRate) const
|
||||||
|
{
|
||||||
|
QMutexLocker{&killLock};
|
||||||
|
|
||||||
|
if(killed) {
|
||||||
|
qCritical() << "Trying to play audio on an invalid sink";
|
||||||
|
} else {
|
||||||
|
audio.playAudioBuffer(sourceId, data, samples, channels, sampleRate);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AlSink::playMono16Sound(const IAudioSink::Sound &sound)
|
||||||
|
{
|
||||||
|
QMutexLocker{&killLock};
|
||||||
|
|
||||||
|
if(killed) {
|
||||||
|
qCritical() << "Trying to play sound on an invalid sink";
|
||||||
|
} else {
|
||||||
|
audio.playMono16Sound(*this, sound);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AlSink::startLoop()
|
||||||
|
{
|
||||||
|
QMutexLocker{&killLock};
|
||||||
|
|
||||||
|
if(killed) {
|
||||||
|
qCritical() << "Trying to start loop on an invalid sink";
|
||||||
|
} else {
|
||||||
|
audio.startLoop(sourceId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AlSink::stopLoop()
|
||||||
|
{
|
||||||
|
QMutexLocker{&killLock};
|
||||||
|
|
||||||
|
if(killed) {
|
||||||
|
qCritical() << "Trying to stop loop on an invalid sink";
|
||||||
|
} else {
|
||||||
|
audio.stopLoop(sourceId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint AlSink::getSourceId() const
|
||||||
|
{
|
||||||
|
uint tmp = sourceId;
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AlSink::kill()
|
||||||
|
{
|
||||||
|
killLock.lock();
|
||||||
|
// this flag is only set once here, afterwards the object is considered dead
|
||||||
|
killed = true;
|
||||||
|
killLock.unlock();
|
||||||
|
emit invalidated();
|
||||||
|
}
|
||||||
|
|
||||||
|
AlSink::AlSink(OpenAL& al, uint sourceId)
|
||||||
|
: audio{al}
|
||||||
|
, sourceId{sourceId}
|
||||||
|
, killLock{QMutex::Recursive}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
AlSink::operator bool() const
|
||||||
|
{
|
||||||
|
QMutexLocker{&killLock};
|
||||||
|
|
||||||
|
return !killed;
|
||||||
|
}
|
40
src/audio/backend/alsink.h
Normal file
40
src/audio/backend/alsink.h
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
#ifndef ALSINK_H
|
||||||
|
#define ALSINK_H
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QMutex>
|
||||||
|
|
||||||
|
#include "src/audio/iaudiosink.h"
|
||||||
|
|
||||||
|
class OpenAL;
|
||||||
|
class QMutex;
|
||||||
|
class AlSink : public IAudioSink
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
AlSink(OpenAL &al, uint sourceId);
|
||||||
|
AlSink(const AlSink &src) = delete;
|
||||||
|
AlSink & operator=(const AlSink&) = delete;
|
||||||
|
AlSink(AlSink&& other) = delete;
|
||||||
|
AlSink & operator=(AlSink&& other) = delete;
|
||||||
|
~AlSink();
|
||||||
|
|
||||||
|
void playAudioBuffer(const int16_t* data, int samples, unsigned channels,
|
||||||
|
int sampleRate) const;
|
||||||
|
void playMono16Sound(const IAudioSink::Sound& sound);
|
||||||
|
void startLoop();
|
||||||
|
void stopLoop();
|
||||||
|
|
||||||
|
operator bool() const;
|
||||||
|
|
||||||
|
uint getSourceId() const;
|
||||||
|
void kill();
|
||||||
|
|
||||||
|
private:
|
||||||
|
OpenAL& audio;
|
||||||
|
uint sourceId;
|
||||||
|
bool killed = false;
|
||||||
|
mutable QMutex killLock;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // ALSINK_H
|
|
@ -68,7 +68,6 @@ OpenAL::OpenAL()
|
||||||
audioThread->setObjectName("qTox Audio");
|
audioThread->setObjectName("qTox Audio");
|
||||||
QObject::connect(audioThread, &QThread::finished, &voiceTimer, &QTimer::stop);
|
QObject::connect(audioThread, &QThread::finished, &voiceTimer, &QTimer::stop);
|
||||||
QObject::connect(audioThread, &QThread::finished, &captureTimer, &QTimer::stop);
|
QObject::connect(audioThread, &QThread::finished, &captureTimer, &QTimer::stop);
|
||||||
QObject::connect(audioThread, &QThread::finished, &playMono16Timer, &QTimer::stop);
|
|
||||||
QObject::connect(audioThread, &QThread::finished, audioThread, &QThread::deleteLater);
|
QObject::connect(audioThread, &QThread::finished, audioThread, &QThread::deleteLater);
|
||||||
|
|
||||||
moveToThread(audioThread);
|
moveToThread(audioThread);
|
||||||
|
@ -85,9 +84,11 @@ OpenAL::OpenAL()
|
||||||
// TODO for Qt 5.6+: use qOverload
|
// TODO for Qt 5.6+: use qOverload
|
||||||
connect(audioThread, &QThread::started, &captureTimer, static_cast<void (QTimer::*)(void)>(&QTimer::start));
|
connect(audioThread, &QThread::started, &captureTimer, static_cast<void (QTimer::*)(void)>(&QTimer::start));
|
||||||
|
|
||||||
connect(&playMono16Timer, &QTimer::timeout, this, &OpenAL::playMono16SoundCleanup);
|
cleanupTimer.setInterval(1000);
|
||||||
playMono16Timer.setSingleShot(true);
|
cleanupTimer.setSingleShot(false);
|
||||||
playMono16Timer.moveToThread(audioThread);
|
connect(&cleanupTimer, &QTimer::timeout, this, &OpenAL::cleanupSound);
|
||||||
|
// TODO for Qt 5.6+: use qOverload
|
||||||
|
connect(audioThread, &QThread::started, &cleanupTimer, static_cast<void (QTimer::*)(void)>(&QTimer::start));
|
||||||
|
|
||||||
audioThread->start();
|
audioThread->start();
|
||||||
}
|
}
|
||||||
|
@ -222,8 +223,82 @@ void OpenAL::reinitInput(const QString& inDevDesc)
|
||||||
bool OpenAL::reinitOutput(const QString& outDevDesc)
|
bool OpenAL::reinitOutput(const QString& outDevDesc)
|
||||||
{
|
{
|
||||||
QMutexLocker locker(&audioLock);
|
QMutexLocker locker(&audioLock);
|
||||||
|
|
||||||
|
const auto bakSinks = sinks;
|
||||||
|
|
||||||
|
sinks.clear();
|
||||||
|
|
||||||
cleanupOutput();
|
cleanupOutput();
|
||||||
return initOutput(outDevDesc);
|
const bool result = initOutput(outDevDesc);
|
||||||
|
|
||||||
|
locker.unlock();
|
||||||
|
// this must happen outside `audioLock`, to avoid a deadlock when
|
||||||
|
// a slot on AlSink::invalidate tries to create a new source immedeately.
|
||||||
|
for (auto& sink : bakSinks) {
|
||||||
|
sink->kill();
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Allocates ressources for a new audio output
|
||||||
|
* @return AudioSink on success, nullptr on failure
|
||||||
|
*/
|
||||||
|
IAudioSink *OpenAL::makeSink()
|
||||||
|
{
|
||||||
|
QMutexLocker locker(&audioLock);
|
||||||
|
|
||||||
|
if (!autoInitOutput()) {
|
||||||
|
qWarning("Failed to subscribe to audio output device.");
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
ALuint sid;
|
||||||
|
alGenSources(1, &sid);
|
||||||
|
|
||||||
|
auto const sink = new AlSink(*this, sid);
|
||||||
|
if (sink == nullptr) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
sinks.insert(sink);
|
||||||
|
qDebug() << "Audio source" << sid << "created. Sources active:" << sinks.size();
|
||||||
|
|
||||||
|
return sink;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Must be called by the destructor of AlSink to remove the internal ressources.
|
||||||
|
* If no sinks are opened, the output is closed afterwards.
|
||||||
|
* @param sink Audio sink to remove.
|
||||||
|
*/
|
||||||
|
void OpenAL::destroySink(AlSink& sink)
|
||||||
|
{
|
||||||
|
QMutexLocker locker(&audioLock);
|
||||||
|
|
||||||
|
const auto sinksErased = sinks.erase(&sink);
|
||||||
|
const auto soundSinksErased = soundSinks.erase(&sink);
|
||||||
|
|
||||||
|
if (sinksErased == 0 && soundSinksErased == 0) {
|
||||||
|
qWarning() << "Destroying non-existant source";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const uint sid = sink.getSourceId();
|
||||||
|
|
||||||
|
if (alIsSource(sid)) {
|
||||||
|
// stop playing, marks all buffers as processed
|
||||||
|
alSourceStop(sid);
|
||||||
|
cleanupBuffers(sid);
|
||||||
|
qDebug() << "Audio source" << sid << "deleted. Sources active:" << sinks.size();
|
||||||
|
} else {
|
||||||
|
qWarning() << "Trying to delete invalid audio source" << sid;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sinks.empty() && soundSinks.empty()) {
|
||||||
|
cleanupOutput();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -332,7 +407,8 @@ bool OpenAL::initInput(const QString& deviceName, uint32_t channels)
|
||||||
*/
|
*/
|
||||||
bool OpenAL::initOutput(const QString& deviceName)
|
bool OpenAL::initOutput(const QString& deviceName)
|
||||||
{
|
{
|
||||||
peerSources.clear();
|
// there should be no sinks when initializing the output
|
||||||
|
assert(sinks.size() == 0);
|
||||||
|
|
||||||
outputInitialized = false;
|
outputInitialized = false;
|
||||||
if (!Settings::getInstance().getAudioOutDevEnabled())
|
if (!Settings::getInstance().getAudioOutDevEnabled())
|
||||||
|
@ -359,61 +435,77 @@ bool OpenAL::initOutput(const QString& deviceName)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
alGenSources(1, &alMainSource);
|
|
||||||
checkAlError();
|
|
||||||
|
|
||||||
// init master volume
|
// init master volume
|
||||||
alListenerf(AL_GAIN, Settings::getInstance().getOutVolume() * 0.01f);
|
alListenerf(AL_GAIN, Settings::getInstance().getOutVolume() * 0.01f);
|
||||||
checkAlError();
|
checkAlError();
|
||||||
|
|
||||||
Core* core = Core::getInstance();
|
|
||||||
if (core) {
|
|
||||||
// reset each call's audio source
|
|
||||||
core->getAv()->invalidateCallSources();
|
|
||||||
}
|
|
||||||
|
|
||||||
outputInitialized = true;
|
outputInitialized = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Play a 48kHz mono 16bit PCM sound from a file
|
|
||||||
*
|
|
||||||
* @param[in] path the path to the sound file
|
|
||||||
*/
|
|
||||||
void OpenAL::playMono16Sound(const QString& path)
|
|
||||||
{
|
|
||||||
QFile sndFile(path);
|
|
||||||
sndFile.open(QIODevice::ReadOnly);
|
|
||||||
playMono16Sound(QByteArray(sndFile.readAll()));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Play a 48kHz mono 16bit PCM sound
|
* @brief Play a 48kHz mono 16bit PCM sound
|
||||||
*/
|
*/
|
||||||
void OpenAL::playMono16Sound(const QByteArray& data)
|
void OpenAL::playMono16Sound(AlSink& sink, const IAudioSink::Sound &sound)
|
||||||
|
{
|
||||||
|
const uint sourceId = sink.getSourceId();
|
||||||
|
QFile sndFile(IAudioSink::getSound(sound));
|
||||||
|
if(!sndFile.exists()) {
|
||||||
|
qDebug() << "Trying to open non existent sound file";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
sndFile.open(QIODevice::ReadOnly);
|
||||||
|
const QByteArray data{sndFile.readAll()};
|
||||||
|
if(data.isEmpty()) {
|
||||||
|
qDebug() << "Sound file contained no data";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QMutexLocker locker(&audioLock);
|
||||||
|
|
||||||
|
// interrupt possibly playing sound, we don't buffer here
|
||||||
|
alSourceStop(sourceId);
|
||||||
|
|
||||||
|
ALint processed = 0;
|
||||||
|
alGetSourcei(sourceId, AL_BUFFERS_PROCESSED, &processed);
|
||||||
|
alSourcei(sourceId, AL_LOOPING, AL_FALSE);
|
||||||
|
|
||||||
|
ALuint bufid;
|
||||||
|
if (processed == 0) {
|
||||||
|
// create new buffer
|
||||||
|
alGenBuffers(1, &bufid);
|
||||||
|
} else {
|
||||||
|
// we only reserve space for one buffer
|
||||||
|
assert(processed == 1);
|
||||||
|
// unqueue all processed buffers
|
||||||
|
alSourceUnqueueBuffers(sourceId, processed, &bufid);
|
||||||
|
}
|
||||||
|
|
||||||
|
alBufferData(bufid, AL_FORMAT_MONO16, data.constData(), data.size(), AUDIO_SAMPLE_RATE);
|
||||||
|
alSourcei(sourceId, AL_BUFFER, bufid);
|
||||||
|
alSourcePlay(sourceId);
|
||||||
|
soundSinks.insert(&sink);
|
||||||
|
}
|
||||||
|
|
||||||
|
void OpenAL::cleanupSound()
|
||||||
{
|
{
|
||||||
QMutexLocker locker(&audioLock);
|
QMutexLocker locker(&audioLock);
|
||||||
|
|
||||||
if (!autoInitOutput())
|
auto sinkIt = soundSinks.begin();
|
||||||
return;
|
while (sinkIt != soundSinks.end()) {
|
||||||
|
auto sink = *sinkIt;
|
||||||
|
ALuint sourceId = sink->getSourceId();
|
||||||
|
ALint state = 0;
|
||||||
|
|
||||||
if (!alMainBuffer)
|
alGetSourcei(sourceId, AL_SOURCE_STATE, &state);
|
||||||
alGenBuffers(1, &alMainBuffer);
|
if (state != AL_PLAYING) {
|
||||||
|
sinkIt = soundSinks.erase(sinkIt);
|
||||||
ALint state;
|
emit sink->finishedPlaying();
|
||||||
alGetSourcei(alMainSource, AL_SOURCE_STATE, &state);
|
} else {
|
||||||
if (state == AL_PLAYING) {
|
++sinkIt;
|
||||||
alSourceStop(alMainSource);
|
}
|
||||||
alSourcei(alMainSource, AL_BUFFER, AL_NONE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
alBufferData(alMainBuffer, AL_FORMAT_MONO16, data.constData(), data.size(), AUDIO_SAMPLE_RATE);
|
|
||||||
alSourcei(alMainSource, AL_BUFFER, static_cast<ALint>(alMainBuffer));
|
|
||||||
alSourcePlay(alMainSource);
|
|
||||||
|
|
||||||
int durationMs = data.size() * 1000 / 2 / AUDIO_SAMPLE_RATE;
|
|
||||||
QMetaObject::invokeMethod(&playMono16Timer, "start", Q_ARG(int, durationMs + 50));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenAL::playAudioBuffer(uint sourceId, const int16_t* data, int samples, unsigned channels,
|
void OpenAL::playAudioBuffer(uint sourceId, const int16_t* data, int samples, unsigned channels,
|
||||||
|
@ -483,15 +575,6 @@ void OpenAL::cleanupOutput()
|
||||||
outputInitialized = false;
|
outputInitialized = false;
|
||||||
|
|
||||||
if (alOutDev) {
|
if (alOutDev) {
|
||||||
alSourcei(alMainSource, AL_LOOPING, AL_FALSE);
|
|
||||||
alSourceStop(alMainSource);
|
|
||||||
alDeleteSources(1, &alMainSource);
|
|
||||||
|
|
||||||
if (alMainBuffer) {
|
|
||||||
alDeleteBuffers(1, &alMainBuffer);
|
|
||||||
alMainBuffer = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!alcMakeContextCurrent(nullptr)) {
|
if (!alcMakeContextCurrent(nullptr)) {
|
||||||
qWarning("Failed to clear audio context.");
|
qWarning("Failed to clear audio context.");
|
||||||
}
|
}
|
||||||
|
@ -508,29 +591,6 @@ void OpenAL::cleanupOutput()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Called after a mono16 sound stopped playing
|
|
||||||
*/
|
|
||||||
void OpenAL::playMono16SoundCleanup()
|
|
||||||
{
|
|
||||||
QMutexLocker locker(&audioLock);
|
|
||||||
|
|
||||||
ALint state;
|
|
||||||
alGetSourcei(alMainSource, AL_SOURCE_STATE, &state);
|
|
||||||
if (state == AL_STOPPED) {
|
|
||||||
alSourcei(alMainSource, AL_BUFFER, AL_NONE);
|
|
||||||
alDeleteBuffers(1, &alMainBuffer);
|
|
||||||
alMainBuffer = 0;
|
|
||||||
// close the audio device if no other sources active
|
|
||||||
if (peerSources.isEmpty()) {
|
|
||||||
cleanupOutput();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// the audio didn't finish, try again later
|
|
||||||
playMono16Timer.start(10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Called by doCapture to calculate volume of the audio buffer
|
* @brief Called by doCapture to calculate volume of the audio buffer
|
||||||
*
|
*
|
||||||
|
@ -605,10 +665,7 @@ void OpenAL::doAudio()
|
||||||
{
|
{
|
||||||
QMutexLocker lock(&audioLock);
|
QMutexLocker lock(&audioLock);
|
||||||
|
|
||||||
// Output section
|
// Output section does nothing
|
||||||
if (outputInitialized && !peerSources.isEmpty()) {
|
|
||||||
doOutput();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Input section
|
// Input section
|
||||||
if (alInDev && inSubscriptions) {
|
if (alInDev && inSubscriptions) {
|
||||||
|
@ -664,72 +721,36 @@ QStringList OpenAL::inDeviceNames()
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenAL::subscribeOutput(uint& sid)
|
/**
|
||||||
{
|
* @brief Free all buffers that finished playing on a source
|
||||||
QMutexLocker locker(&audioLock);
|
* @param sourceId where to remove the buffers from
|
||||||
|
*/
|
||||||
if (!autoInitOutput()) {
|
void OpenAL::cleanupBuffers(uint sourceId) {
|
||||||
qWarning("Failed to subscribe to audio output device.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
alGenSources(1, &sid);
|
|
||||||
assert(sid);
|
|
||||||
peerSources << sid;
|
|
||||||
|
|
||||||
qDebug() << "Audio source" << sid << "created. Sources active:" << peerSources.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
void OpenAL::unsubscribeOutput(uint& sid)
|
|
||||||
{
|
|
||||||
QMutexLocker locker(&audioLock);
|
|
||||||
|
|
||||||
peerSources.removeAll(sid);
|
|
||||||
|
|
||||||
if (sid) {
|
|
||||||
if (alIsSource(sid)) {
|
|
||||||
// stop playing, marks all buffers as processed
|
|
||||||
alSourceStop(sid);
|
|
||||||
// unqueue all buffers from the source
|
// unqueue all buffers from the source
|
||||||
ALint processed = 0;
|
ALint processed = 0;
|
||||||
alGetSourcei(sid, AL_BUFFERS_PROCESSED, &processed);
|
alGetSourcei(sourceId, AL_BUFFERS_PROCESSED, &processed);
|
||||||
ALuint* bufids = new ALuint[processed];
|
std::vector<ALuint> bufids;
|
||||||
alSourceUnqueueBuffers(sid, processed, bufids);
|
// should never be out of range, just to be sure
|
||||||
|
assert(processed >= 0);
|
||||||
|
assert(processed <= SIZE_MAX);
|
||||||
|
bufids.resize(processed);
|
||||||
|
alSourceUnqueueBuffers(sourceId, processed, bufids.data());
|
||||||
// delete all buffers
|
// delete all buffers
|
||||||
alDeleteBuffers(processed, bufids);
|
alDeleteBuffers(processed, bufids.data());
|
||||||
delete[] bufids;
|
|
||||||
alDeleteSources(1, &sid);
|
|
||||||
qDebug() << "Audio source" << sid << "deleted. Sources active:" << peerSources.size();
|
|
||||||
} else {
|
|
||||||
qWarning() << "Trying to delete invalid audio source" << sid;
|
|
||||||
}
|
|
||||||
|
|
||||||
sid = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (peerSources.isEmpty())
|
|
||||||
cleanupOutput();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenAL::startLoop()
|
void OpenAL::startLoop(uint sourceId)
|
||||||
{
|
{
|
||||||
QMutexLocker locker(&audioLock);
|
QMutexLocker locker(&audioLock);
|
||||||
alSourcei(alMainSource, AL_LOOPING, AL_TRUE);
|
alSourcei(sourceId, AL_LOOPING, AL_TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenAL::stopLoop()
|
void OpenAL::stopLoop(uint sourceId)
|
||||||
{
|
{
|
||||||
QMutexLocker locker(&audioLock);
|
QMutexLocker locker(&audioLock);
|
||||||
alSourcei(alMainSource, AL_LOOPING, AL_FALSE);
|
alSourcei(sourceId, AL_LOOPING, AL_FALSE);
|
||||||
alSourceStop(alMainSource);
|
alSourceStop(sourceId);
|
||||||
|
cleanupBuffers(sourceId);
|
||||||
ALint state;
|
|
||||||
alGetSourcei(alMainSource, AL_SOURCE_STATE, &state);
|
|
||||||
if (state == AL_STOPPED) {
|
|
||||||
alSourcei(alMainSource, AL_BUFFER, AL_NONE);
|
|
||||||
alDeleteBuffers(1, &alMainBuffer);
|
|
||||||
alMainBuffer = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
qreal OpenAL::inputGain() const
|
qreal OpenAL::inputGain() const
|
||||||
|
|
|
@ -22,6 +22,10 @@
|
||||||
#define OPENAL_H
|
#define OPENAL_H
|
||||||
|
|
||||||
#include "src/audio/audio.h"
|
#include "src/audio/audio.h"
|
||||||
|
#include "src/audio/backend/alsink.h"
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
#include <unordered_set>
|
||||||
|
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
@ -76,16 +80,15 @@ public:
|
||||||
QStringList outDeviceNames();
|
QStringList outDeviceNames();
|
||||||
QStringList inDeviceNames();
|
QStringList inDeviceNames();
|
||||||
|
|
||||||
void subscribeOutput(uint& sourceId);
|
IAudioSink* makeSink();
|
||||||
void unsubscribeOutput(uint& sourceId);
|
void destroySink(AlSink& sink);
|
||||||
|
|
||||||
void subscribeInput();
|
void subscribeInput();
|
||||||
void unsubscribeInput();
|
void unsubscribeInput();
|
||||||
|
|
||||||
void startLoop();
|
void startLoop(uint sourceId);
|
||||||
void stopLoop();
|
void stopLoop(uint sourceId);
|
||||||
void playMono16Sound(const QByteArray& data);
|
void playMono16Sound(AlSink& sink, const IAudioSink::Sound& sound);
|
||||||
void playMono16Sound(const QString& path);
|
|
||||||
void stopActive();
|
void stopActive();
|
||||||
|
|
||||||
void playAudioBuffer(uint sourceId, const int16_t* data, int samples, unsigned channels,
|
void playAudioBuffer(uint sourceId, const int16_t* data, int samples, unsigned channels,
|
||||||
|
@ -113,24 +116,35 @@ protected:
|
||||||
private:
|
private:
|
||||||
virtual bool initInput(const QString& deviceName);
|
virtual bool initInput(const QString& deviceName);
|
||||||
virtual bool initOutput(const QString& outDevDescr);
|
virtual bool initOutput(const QString& outDevDescr);
|
||||||
void playMono16SoundCleanup();
|
|
||||||
|
void cleanupBuffers(uint sourceId);
|
||||||
|
void cleanupSound();
|
||||||
|
|
||||||
float getVolume();
|
float getVolume();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
QThread* audioThread;
|
QThread* audioThread;
|
||||||
mutable QMutex audioLock{QMutex::Recursive};
|
mutable QMutex audioLock{QMutex::Recursive};
|
||||||
|
QString inDev{};
|
||||||
|
QString outDev{};
|
||||||
|
|
||||||
ALCdevice* alInDev = nullptr;
|
ALCdevice* alInDev = nullptr;
|
||||||
quint32 inSubscriptions = 0;
|
quint32 inSubscriptions = 0;
|
||||||
QTimer captureTimer, playMono16Timer;
|
QTimer captureTimer;
|
||||||
|
QTimer cleanupTimer;
|
||||||
|
|
||||||
ALCdevice* alOutDev = nullptr;
|
ALCdevice* alOutDev = nullptr;
|
||||||
ALCcontext* alOutContext = nullptr;
|
ALCcontext* alOutContext = nullptr;
|
||||||
ALuint alMainSource = 0;
|
|
||||||
ALuint alMainBuffer = 0;
|
|
||||||
bool outputInitialized = false;
|
bool outputInitialized = false;
|
||||||
|
|
||||||
QList<ALuint> peerSources;
|
// Qt containers need copy operators, so use stdlib containers
|
||||||
|
std::unordered_set<AlSink*> sinks;
|
||||||
|
std::unordered_set<AlSink*> soundSinks;
|
||||||
|
|
||||||
|
// number of output sources
|
||||||
|
int outCount = 0;
|
||||||
|
|
||||||
int channels = 0;
|
int channels = 0;
|
||||||
qreal gain = 0;
|
qreal gain = 0;
|
||||||
qreal gainFactor = 1;
|
qreal gainFactor = 1;
|
||||||
|
|
|
@ -205,7 +205,7 @@ bool OpenAL2::initOutputEchoCancel()
|
||||||
*/
|
*/
|
||||||
bool OpenAL2::initOutput(const QString& deviceName)
|
bool OpenAL2::initOutput(const QString& deviceName)
|
||||||
{
|
{
|
||||||
peerSources.clear();
|
assert(sinks.empty());
|
||||||
|
|
||||||
outputInitialized = false;
|
outputInitialized = false;
|
||||||
if (!Settings::getInstance().getAudioOutDevEnabled()) {
|
if (!Settings::getInstance().getAudioOutDevEnabled()) {
|
||||||
|
@ -243,19 +243,10 @@ bool OpenAL2::initOutput(const QString& deviceName)
|
||||||
alProxyContext = alOutContext;
|
alProxyContext = alOutContext;
|
||||||
}
|
}
|
||||||
|
|
||||||
alGenSources(1, &alMainSource);
|
|
||||||
checkAlError();
|
|
||||||
|
|
||||||
// init master volume
|
// init master volume
|
||||||
alListenerf(AL_GAIN, Settings::getInstance().getOutVolume() * 0.01f);
|
alListenerf(AL_GAIN, Settings::getInstance().getOutVolume() * 0.01f);
|
||||||
checkAlError();
|
checkAlError();
|
||||||
|
|
||||||
Core* core = Core::getInstance();
|
|
||||||
if (core) {
|
|
||||||
// reset each call's audio source
|
|
||||||
core->getAv()->invalidateCallSources();
|
|
||||||
}
|
|
||||||
|
|
||||||
// ensure alProxyContext is active
|
// ensure alProxyContext is active
|
||||||
alcMakeContextCurrent(alProxyContext);
|
alcMakeContextCurrent(alProxyContext);
|
||||||
outputInitialized = true;
|
outputInitialized = true;
|
||||||
|
|
|
@ -21,8 +21,8 @@
|
||||||
#ifndef OPENAL2_H
|
#ifndef OPENAL2_H
|
||||||
#define OPENAL2_H
|
#define OPENAL2_H
|
||||||
|
|
||||||
#include "openal.h"
|
|
||||||
#include "src/audio/audio.h"
|
#include "src/audio/audio.h"
|
||||||
|
#include "src/audio/backend/openal.h"
|
||||||
|
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
|
|
@ -63,15 +63,15 @@ public:
|
||||||
{
|
{
|
||||||
switch (s) {
|
switch (s) {
|
||||||
case Sound::Test:
|
case Sound::Test:
|
||||||
return QStringLiteral(":/audio/notification.pcm");
|
return QStringLiteral(":/audio/notification.s16le.pcm");
|
||||||
case Sound::NewMessage:
|
case Sound::NewMessage:
|
||||||
return QStringLiteral(":/audio/notification.pcm");
|
return QStringLiteral(":/audio/notification.s16le.pcm");
|
||||||
case Sound::IncomingCall:
|
case Sound::IncomingCall:
|
||||||
return QStringLiteral(":/audio/ToxIncomingCall.pcm");
|
return QStringLiteral(":/audio/ToxIncomingCall.s16le.pcm");
|
||||||
case Sound::OutgoingCall:
|
case Sound::OutgoingCall:
|
||||||
return QStringLiteral(":/audio/ToxOutgoingCall.pcm");
|
return QStringLiteral(":/audio/ToxOutgoingCall.s16le.pcm");
|
||||||
case Sound::CallEnd:
|
case Sound::CallEnd:
|
||||||
return QStringLiteral(":/audio/ToxEndCall.pcm");
|
return QStringLiteral(":/audio/ToxEndCall.s16le.pcm");
|
||||||
}
|
}
|
||||||
assert(false);
|
assert(false);
|
||||||
return {};
|
return {};
|
||||||
|
|
|
@ -502,12 +502,7 @@ void CoreAV::groupCallCallback(void* tox, uint32_t group, uint32_t peer, const i
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Audio& audio = Audio::getInstance();
|
call.getAudioSink(peerPk)->playAudioBuffer(data, samples, channels, sample_rate);
|
||||||
if(!call.havePeer(peerPk)) {
|
|
||||||
call.addPeer(peerPk);
|
|
||||||
}
|
|
||||||
|
|
||||||
audio.playAudioBuffer(call.getAlSource(peerPk), data, samples, channels, sample_rate);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -678,21 +673,6 @@ bool CoreAV::isCallOutputMuted(const Friend* f) const
|
||||||
return (it != calls.end()) && it->second->getMuteVol();
|
return (it != calls.end()) && it->second->getMuteVol();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Forces to regenerate each call's audio sources.
|
|
||||||
*/
|
|
||||||
void CoreAV::invalidateCallSources()
|
|
||||||
{
|
|
||||||
for (auto& kv : groupCalls) {
|
|
||||||
kv.second->clearPeers();
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto& kv : calls) {
|
|
||||||
// TODO: this is wrong, "0" is a valid source id
|
|
||||||
kv.second->setAlSource(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Signal to all peers that we're not sending video anymore.
|
* @brief Signal to all peers that we're not sending video anymore.
|
||||||
* @note The next frame sent cancels this.
|
* @note The next frame sent cancels this.
|
||||||
|
@ -886,14 +866,7 @@ void CoreAV::audioFrameCallback(ToxAV*, uint32_t friendNum, const int16_t* pcm,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Audio& audio = Audio::getInstance();
|
call.getAudioSink()->playAudioBuffer(pcm, sampleCount, channels, samplingRate);
|
||||||
if (!call.getAlSource()) {
|
|
||||||
quint32 sourceId;
|
|
||||||
audio.subscribeOutput(sourceId);
|
|
||||||
call.setAlSource(sourceId);
|
|
||||||
}
|
|
||||||
|
|
||||||
audio.playAudioBuffer(call.getAlSource(), pcm, sampleCount, channels, samplingRate);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CoreAV::videoFrameCallback(ToxAV*, uint32_t friendNum, uint16_t w, uint16_t h,
|
void CoreAV::videoFrameCallback(ToxAV*, uint32_t friendNum, uint16_t w, uint16_t h,
|
||||||
|
|
|
@ -62,7 +62,6 @@ public:
|
||||||
uint32_t rate) const;
|
uint32_t rate) const;
|
||||||
|
|
||||||
VideoSource* getVideoSourceFromCall(int callNumber) const;
|
VideoSource* getVideoSourceFromCall(int callNumber) const;
|
||||||
void invalidateCallSources();
|
|
||||||
void sendNoVideo();
|
void sendNoVideo();
|
||||||
|
|
||||||
void joinGroupCall(int groupNum);
|
void joinGroupCall(int groupNum);
|
||||||
|
|
|
@ -101,18 +101,10 @@ CoreVideoSource* ToxCall::getVideoSource() const
|
||||||
return videoSource;
|
return videoSource;
|
||||||
}
|
}
|
||||||
|
|
||||||
quint32 ToxFriendCall::getAlSource() const
|
|
||||||
{
|
|
||||||
return alSource;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ToxFriendCall::setAlSource(const quint32& value)
|
|
||||||
{
|
|
||||||
alSource = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
ToxFriendCall::ToxFriendCall(uint32_t FriendNum, bool VideoEnabled, CoreAV& av)
|
ToxFriendCall::ToxFriendCall(uint32_t FriendNum, bool VideoEnabled, CoreAV& av)
|
||||||
: ToxCall(VideoEnabled, av)
|
: ToxCall(VideoEnabled, av)
|
||||||
|
, sink(Audio::getInstance().makeSink())
|
||||||
|
, friendId{FriendNum}
|
||||||
{
|
{
|
||||||
// register audio
|
// register audio
|
||||||
Audio& audio = Audio::getInstance();
|
Audio& audio = Audio::getInstance();
|
||||||
|
@ -127,7 +119,11 @@ ToxFriendCall::ToxFriendCall(uint32_t FriendNum, bool VideoEnabled, CoreAV& av)
|
||||||
qDebug() << "Audio input connection not working";
|
qDebug() << "Audio input connection not working";
|
||||||
}
|
}
|
||||||
|
|
||||||
audio.subscribeOutput(alSource);
|
audioSinkInvalid = QObject::connect(sink.get(), &IAudioSink::invalidated,
|
||||||
|
[this]() {
|
||||||
|
this->onAudioSinkInvalidated();
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
// register video
|
// register video
|
||||||
if (videoEnabled) {
|
if (videoEnabled) {
|
||||||
|
@ -150,8 +146,18 @@ ToxFriendCall::ToxFriendCall(uint32_t FriendNum, bool VideoEnabled, CoreAV& av)
|
||||||
|
|
||||||
ToxFriendCall::~ToxFriendCall()
|
ToxFriendCall::~ToxFriendCall()
|
||||||
{
|
{
|
||||||
auto& audio = Audio::getInstance();
|
QObject::disconnect(audioSinkInvalid);
|
||||||
audio.unsubscribeOutput(alSource);
|
}
|
||||||
|
|
||||||
|
void ToxFriendCall::onAudioSinkInvalidated()
|
||||||
|
{
|
||||||
|
const auto newSink = Audio::getInstance().makeSink();
|
||||||
|
|
||||||
|
audioSinkInvalid = QObject::connect(newSink, &IAudioSink::invalidated,
|
||||||
|
[this]() {
|
||||||
|
this->onAudioSinkInvalidated();
|
||||||
|
});
|
||||||
|
sink.reset(newSink);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToxFriendCall::startTimeout(uint32_t callId)
|
void ToxFriendCall::startTimeout(uint32_t callId)
|
||||||
|
@ -187,8 +193,14 @@ void ToxFriendCall::setState(const TOXAV_FRIEND_CALL_STATE& value)
|
||||||
state = value;
|
state = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const std::unique_ptr<IAudioSink> &ToxFriendCall::getAudioSink() const
|
||||||
|
{
|
||||||
|
return sink;
|
||||||
|
}
|
||||||
|
|
||||||
ToxGroupCall::ToxGroupCall(int GroupNum, CoreAV& av)
|
ToxGroupCall::ToxGroupCall(int GroupNum, CoreAV& av)
|
||||||
: ToxCall(false, av)
|
: ToxCall(false, av)
|
||||||
|
, groupId{GroupNum}
|
||||||
{
|
{
|
||||||
// register audio
|
// register audio
|
||||||
Audio& audio = Audio::getInstance();
|
Audio& audio = Audio::getInstance();
|
||||||
|
@ -206,57 +218,64 @@ ToxGroupCall::ToxGroupCall(int GroupNum, CoreAV& av)
|
||||||
|
|
||||||
ToxGroupCall::~ToxGroupCall()
|
ToxGroupCall::~ToxGroupCall()
|
||||||
{
|
{
|
||||||
Audio& audio = Audio::getInstance();
|
// disconnect all Qt connections
|
||||||
|
clearPeers();
|
||||||
|
}
|
||||||
|
|
||||||
for (quint32 v : peers) {
|
void ToxGroupCall::onAudioSinkInvalidated(ToxPk peerId)
|
||||||
audio.unsubscribeOutput(v);
|
{
|
||||||
}
|
removePeer(peerId);
|
||||||
|
addPeer(peerId);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToxGroupCall::removePeer(ToxPk peerId)
|
void ToxGroupCall::removePeer(ToxPk peerId)
|
||||||
{
|
{
|
||||||
auto& audio = Audio::getInstance();
|
const auto& source = peers.find(peerId);
|
||||||
const auto& sourceId = peers.find(peerId);
|
if(source == peers.cend()) {
|
||||||
if(sourceId == peers.constEnd()) {
|
qDebug() << "Peer:" << peerId.toString() << "does not have a source, can't remove";
|
||||||
qDebug() << "Peer does not have a source, can't remove";
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
audio.unsubscribeOutput(sourceId.value());
|
peers.erase(source);
|
||||||
peers.remove(peerId);
|
QObject::disconnect(sinkInvalid[peerId]);
|
||||||
|
sinkInvalid.erase(peerId);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToxGroupCall::addPeer(ToxPk peerId)
|
void ToxGroupCall::addPeer(ToxPk peerId)
|
||||||
{
|
{
|
||||||
auto& audio = Audio::getInstance();
|
auto& audio = Audio::getInstance();
|
||||||
uint sourceId = 0;
|
IAudioSink* newSink = audio.makeSink();
|
||||||
audio.subscribeOutput(sourceId);
|
peers.emplace(peerId, std::unique_ptr<IAudioSink>(newSink));
|
||||||
peers.insert(peerId, sourceId);
|
|
||||||
|
QMetaObject::Connection con = QObject::connect(newSink, &IAudioSink::invalidated,
|
||||||
|
[this, peerId]() {
|
||||||
|
this->onAudioSinkInvalidated(peerId);
|
||||||
|
});
|
||||||
|
|
||||||
|
sinkInvalid.insert({peerId, con});
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ToxGroupCall::havePeer(ToxPk peerId)
|
bool ToxGroupCall::havePeer(ToxPk peerId)
|
||||||
{
|
{
|
||||||
const auto& sourceId = peers.find(peerId);
|
const auto& source = peers.find(peerId);
|
||||||
return sourceId != peers.constEnd();
|
return source != peers.cend();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToxGroupCall::clearPeers()
|
void ToxGroupCall::clearPeers()
|
||||||
{
|
{
|
||||||
Audio& audio = Audio::getInstance();
|
peers.clear();
|
||||||
|
for(auto con : sinkInvalid) {
|
||||||
for (quint32 v : peers) {
|
QObject::disconnect(con.second);
|
||||||
audio.unsubscribeOutput(v);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
peers.clear();
|
sinkInvalid.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
quint32 ToxGroupCall::getAlSource(ToxPk peer)
|
const std::unique_ptr<IAudioSink> &ToxGroupCall::getAudioSink(ToxPk peer)
|
||||||
{
|
{
|
||||||
if(!havePeer(peer)) {
|
if(!havePeer(peer)) {
|
||||||
qWarning() << "Getting alSource for non-existant peer";
|
addPeer(peer);
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
const auto& source = peers.find(peer);
|
||||||
return peers[peer];
|
return source->second;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
#ifndef TOXCALL_H
|
#ifndef TOXCALL_H
|
||||||
#define TOXCALL_H
|
#define TOXCALL_H
|
||||||
|
|
||||||
|
#include "src/audio/iaudiosink.h"
|
||||||
#include <src/core/toxpk.h>
|
#include <src/core/toxpk.h>
|
||||||
#include <tox/toxav.h>
|
#include <tox/toxav.h>
|
||||||
|
|
||||||
|
@ -76,16 +77,21 @@ public:
|
||||||
TOXAV_FRIEND_CALL_STATE getState() const;
|
TOXAV_FRIEND_CALL_STATE getState() const;
|
||||||
void setState(const TOXAV_FRIEND_CALL_STATE& value);
|
void setState(const TOXAV_FRIEND_CALL_STATE& value);
|
||||||
|
|
||||||
quint32 getAlSource() const;
|
const std::unique_ptr<IAudioSink> &getAudioSink() const;
|
||||||
void setAlSource(const quint32& value);
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::unique_ptr<QTimer> timeoutTimer;
|
std::unique_ptr<QTimer> timeoutTimer;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QMetaObject::Connection audioSinkInvalid;
|
||||||
|
void onAudioSourceInvalidated();
|
||||||
|
void onAudioSinkInvalidated();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TOXAV_FRIEND_CALL_STATE state{TOXAV_FRIEND_CALL_STATE_NONE};
|
TOXAV_FRIEND_CALL_STATE state{TOXAV_FRIEND_CALL_STATE_NONE};
|
||||||
static constexpr int CALL_TIMEOUT = 45000;
|
static constexpr int CALL_TIMEOUT = 45000;
|
||||||
quint32 alSource{0};
|
std::unique_ptr<IAudioSink> sink = nullptr;
|
||||||
|
uint32_t friendId;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ToxGroupCall : public ToxCall
|
class ToxGroupCall : public ToxCall
|
||||||
|
@ -103,12 +109,14 @@ public:
|
||||||
bool havePeer(ToxPk peerId);
|
bool havePeer(ToxPk peerId);
|
||||||
void clearPeers();
|
void clearPeers();
|
||||||
|
|
||||||
quint32 getAlSource(ToxPk peer);
|
const std::unique_ptr<IAudioSink> &getAudioSink(ToxPk peer);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QMap<ToxPk, quint32> peers;
|
std::map<ToxPk, std::unique_ptr<IAudioSink>> peers;
|
||||||
|
std::map<ToxPk, QMetaObject::Connection> sinkInvalid;
|
||||||
|
int groupId;
|
||||||
|
|
||||||
// If you add something here, don't forget to override the ctors and move operators!
|
void onAudioSourceInvalidated();
|
||||||
|
void onAudioSinkInvalidated(ToxPk peerId);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // TOXCALL_H
|
#endif // TOXCALL_H
|
||||||
|
|
|
@ -122,11 +122,12 @@ void AVForm::hideEvent(QHideEvent* event)
|
||||||
{
|
{
|
||||||
if (subscribedToAudioIn) {
|
if (subscribedToAudioIn) {
|
||||||
// TODO: This should not be done in show/hide events
|
// TODO: This should not be done in show/hide events
|
||||||
audio->unsubscribeOutput(alSource);
|
|
||||||
audio->unsubscribeInput();
|
audio->unsubscribeInput();
|
||||||
subscribedToAudioIn = false;
|
subscribedToAudioIn = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
audioSink.reset();
|
||||||
|
|
||||||
if (camVideoSurface) {
|
if (camVideoSurface) {
|
||||||
camVideoSurface->setSource(nullptr);
|
camVideoSurface->setSource(nullptr);
|
||||||
killVideoSurface();
|
killVideoSurface();
|
||||||
|
@ -145,11 +146,14 @@ void AVForm::showEvent(QShowEvent* event)
|
||||||
|
|
||||||
if (!subscribedToAudioIn) {
|
if (!subscribedToAudioIn) {
|
||||||
// TODO: This should not be done in show/hide events
|
// TODO: This should not be done in show/hide events
|
||||||
audio->subscribeOutput(alSource);
|
|
||||||
audio->subscribeInput();
|
audio->subscribeInput();
|
||||||
subscribedToAudioIn = true;
|
subscribedToAudioIn = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(audioSink == nullptr) {
|
||||||
|
audioSink.reset(audio->makeSink());
|
||||||
|
}
|
||||||
|
|
||||||
GenericForm::showEvent(event);
|
GenericForm::showEvent(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -560,14 +564,19 @@ void AVForm::on_outDevCombobox_currentIndexChanged(int deviceIndex)
|
||||||
const bool outputEnabled = deviceIndex > 0;
|
const bool outputEnabled = deviceIndex > 0;
|
||||||
audioSettings->setAudioOutDevEnabled(outputEnabled);
|
audioSettings->setAudioOutDevEnabled(outputEnabled);
|
||||||
|
|
||||||
QString deviceName;
|
QString deviceName{};
|
||||||
if (outputEnabled) {
|
if (outputEnabled) {
|
||||||
deviceName = outDevCombobox->itemText(deviceIndex);
|
deviceName = outDevCombobox->itemText(deviceIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
audioSettings->setOutDev(deviceName);
|
const QString oldName = audioSettings->getOutDev();
|
||||||
|
|
||||||
|
if(oldName != deviceName) {
|
||||||
|
audioSettings->setOutDev(deviceName);
|
||||||
audio->reinitOutput(deviceName);
|
audio->reinitOutput(deviceName);
|
||||||
|
audioSink.reset(Audio::getInstance().makeSink());
|
||||||
|
}
|
||||||
|
|
||||||
playbackSlider->setEnabled(outputEnabled);
|
playbackSlider->setEnabled(outputEnabled);
|
||||||
playbackSlider->setSliderPosition(
|
playbackSlider->setSliderPosition(
|
||||||
getStepsFromValue(audio->outputVolume(), audio->minOutputVolume(), audio->maxOutputVolume()));
|
getStepsFromValue(audio->outputVolume(), audio->minOutputVolume(), audio->maxOutputVolume()));
|
||||||
|
@ -584,8 +593,9 @@ void AVForm::on_playbackSlider_valueChanged(int sliderSteps)
|
||||||
getValueFromSteps(sliderSteps, audio->minOutputVolume(), audio->maxOutputVolume());
|
getValueFromSteps(sliderSteps, audio->minOutputVolume(), audio->maxOutputVolume());
|
||||||
audio->setOutputVolume(volume);
|
audio->setOutputVolume(volume);
|
||||||
|
|
||||||
if (cbEnableTestSound->isChecked())
|
if (cbEnableTestSound->isChecked() && audioSink) {
|
||||||
audio->playMono16Sound(Audio::getSound(Audio::Sound::Test));
|
audioSink->playMono16Sound(IAudioSink::Sound::Test);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -593,8 +603,9 @@ void AVForm::on_cbEnableTestSound_stateChanged()
|
||||||
{
|
{
|
||||||
audioSettings->setEnableTestSound(cbEnableTestSound->isChecked());
|
audioSettings->setEnableTestSound(cbEnableTestSound->isChecked());
|
||||||
|
|
||||||
if (cbEnableTestSound->isChecked() && audio->isOutputReady())
|
if (cbEnableTestSound->isChecked() && audio->isOutputReady() && audioSink) {
|
||||||
audio->playMono16Sound(Audio::getSound(Audio::Sound::Test));
|
audioSink->playMono16Sound(IAudioSink::Sound::Test);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void AVForm::on_microphoneSlider_valueChanged(int sliderSteps)
|
void AVForm::on_microphoneSlider_valueChanged(int sliderSteps)
|
||||||
|
|
|
@ -28,13 +28,15 @@
|
||||||
#include "ui_avform.h"
|
#include "ui_avform.h"
|
||||||
#include "src/video/videomode.h"
|
#include "src/video/videomode.h"
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
class Audio;
|
class Audio;
|
||||||
class CameraSource;
|
class CameraSource;
|
||||||
class CoreAV;
|
class CoreAV;
|
||||||
class IAudioSettings;
|
class IAudioSettings;
|
||||||
class IVideoSettings;
|
class IVideoSettings;
|
||||||
class VideoSurface;
|
class VideoSurface;
|
||||||
|
class IAudioSink;
|
||||||
class AVForm : public GenericForm, private Ui::AVForm
|
class AVForm : public GenericForm, private Ui::AVForm
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
@ -100,6 +102,7 @@ private:
|
||||||
IVideoSettings* videoSettings;
|
IVideoSettings* videoSettings;
|
||||||
|
|
||||||
bool subscribedToAudioIn;
|
bool subscribedToAudioIn;
|
||||||
|
std::unique_ptr<IAudioSink> audioSink = nullptr;
|
||||||
VideoSurface* camVideoSurface;
|
VideoSurface* camVideoSurface;
|
||||||
CameraSource& camera;
|
CameraSource& camera;
|
||||||
QVector<QPair<QString, QString>> videoDeviceList;
|
QVector<QPair<QString, QString>> videoDeviceList;
|
||||||
|
|
|
@ -928,27 +928,53 @@ void Widget::reloadHistory()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Plays a sound via the audioNotification AudioSink
|
||||||
|
* @param sound Sound to play
|
||||||
|
* @param loop if true, loop the sound until onStopNotification() is called
|
||||||
|
*/
|
||||||
|
void Widget::playNotificationSound(IAudioSink::Sound sound, bool loop) {
|
||||||
|
if(audioNotification == nullptr) {
|
||||||
|
audioNotification = std::unique_ptr<IAudioSink>(Audio::getInstance().makeSink());
|
||||||
|
if(audioNotification == nullptr) {
|
||||||
|
qDebug() << "Failed to allocate AudioSink";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
connect(audioNotification.get(), &IAudioSink::finishedPlaying,
|
||||||
|
this, &Widget::cleanupNotificationSound);
|
||||||
|
|
||||||
|
if(loop) {
|
||||||
|
audioNotification->startLoop();
|
||||||
|
}
|
||||||
|
|
||||||
|
audioNotification->playMono16Sound(sound);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Widget::cleanupNotificationSound()
|
||||||
|
{
|
||||||
|
audioNotification.reset();
|
||||||
|
}
|
||||||
|
|
||||||
void Widget::incomingNotification(uint32_t friendnumber)
|
void Widget::incomingNotification(uint32_t friendnumber)
|
||||||
{
|
{
|
||||||
const auto& friendId = FriendList::id2Key(friendnumber);
|
const auto& friendId = FriendList::id2Key(friendnumber);
|
||||||
newFriendMessageAlert(friendId, false);
|
newFriendMessageAlert(friendId, false);
|
||||||
|
|
||||||
Audio& audio = Audio::getInstance();
|
// loop until call answered or rejected
|
||||||
audio.startLoop();
|
playNotificationSound(IAudioSink::Sound::IncomingCall, true);
|
||||||
audio.playMono16Sound(Audio::getSound(Audio::Sound::IncomingCall));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Widget::outgoingNotification()
|
void Widget::outgoingNotification()
|
||||||
{
|
{
|
||||||
Audio& audio = Audio::getInstance();
|
// loop until call answered or rejected
|
||||||
audio.startLoop();
|
playNotificationSound(IAudioSink::Sound::OutgoingCall, true);
|
||||||
audio.playMono16Sound(Audio::getSound(Audio::Sound::OutgoingCall));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Widget::onCallEnd()
|
void Widget::onCallEnd()
|
||||||
{
|
{
|
||||||
Audio& audio = Audio::getInstance();
|
playNotificationSound(IAudioSink::Sound::CallEnd);
|
||||||
audio.playMono16Sound(Audio::getSound(Audio::Sound::CallEnd));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -956,7 +982,7 @@ void Widget::onCallEnd()
|
||||||
*/
|
*/
|
||||||
void Widget::onStopNotification()
|
void Widget::onStopNotification()
|
||||||
{
|
{
|
||||||
Audio::getInstance().stopLoop();
|
audioNotification.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Widget::onRejectCall(uint32_t friendId)
|
void Widget::onRejectCall(uint32_t friendId)
|
||||||
|
@ -1446,8 +1472,7 @@ bool Widget::newMessageAlert(QWidget* currentWindow, bool isActive, bool sound,
|
||||||
bool notifySound = settings.getNotifySound();
|
bool notifySound = settings.getNotifySound();
|
||||||
|
|
||||||
if (notifySound && sound && (!isBusy || busySound)) {
|
if (notifySound && sound && (!isBusy || busySound)) {
|
||||||
QString soundPath = Audio::getSound(Audio::Sound::NewMessage);
|
playNotificationSound(IAudioSink::Sound::NewMessage);
|
||||||
Audio::getInstance().playMono16Sound(soundPath);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
|
|
||||||
#include "genericchatitemwidget.h"
|
#include "genericchatitemwidget.h"
|
||||||
|
|
||||||
|
#include "src/audio/iaudiosink.h"
|
||||||
#include "src/core/core.h"
|
#include "src/core/core.h"
|
||||||
#include "src/core/toxfile.h"
|
#include "src/core/toxfile.h"
|
||||||
#include "src/core/groupid.h"
|
#include "src/core/groupid.h"
|
||||||
|
@ -45,6 +46,7 @@ class MainWindow;
|
||||||
}
|
}
|
||||||
|
|
||||||
class AddFriendForm;
|
class AddFriendForm;
|
||||||
|
class AlSink;
|
||||||
class Camera;
|
class Camera;
|
||||||
class ChatForm;
|
class ChatForm;
|
||||||
class CircleWidget;
|
class CircleWidget;
|
||||||
|
@ -256,6 +258,8 @@ private:
|
||||||
void retranslateUi();
|
void retranslateUi();
|
||||||
void focusChatInput();
|
void focusChatInput();
|
||||||
void openDialog(GenericChatroomWidget* widget, bool newWindow);
|
void openDialog(GenericChatroomWidget* widget, bool newWindow);
|
||||||
|
void playNotificationSound(IAudioSink::Sound sound, bool loop = false);
|
||||||
|
void cleanupNotificationSound();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
SystemTrayIcon* icon = nullptr;
|
SystemTrayIcon* icon = nullptr;
|
||||||
|
@ -308,6 +312,7 @@ private:
|
||||||
QPushButton* groupInvitesButton;
|
QPushButton* groupInvitesButton;
|
||||||
unsigned int unreadGroupInvites;
|
unsigned int unreadGroupInvites;
|
||||||
int icon_size;
|
int icon_size;
|
||||||
|
std::unique_ptr<IAudioSink> audioNotification = nullptr;
|
||||||
Settings& settings;
|
Settings& settings;
|
||||||
|
|
||||||
QMap<ToxPk, FriendWidget*> friendWidgets;
|
QMap<ToxPk, FriendWidget*> friendWidgets;
|
||||||
|
|
Loading…
Reference in New Issue
Block a user