From f5a628fafb692e8c049c73a373b3137df620a876 Mon Sep 17 00:00:00 2001 From: "Tux3 / Mlkj / !Lev.uXFMLA" Date: Fri, 27 Jun 2014 18:51:08 +0200 Subject: [PATCH] Audio call reception Can hear, but can't talk Uses a hardwired tox_do() interval of 50ms until to_do_interval() gets fixed --- audiobuffer.cpp | 36 ++++++++++++++++++++++++++++++++++++ audiobuffer.h | 23 +++++++++++++++++++++++ core.cpp | 43 ++++++++++++++++++++++++++++--------------- core.h | 5 +++-- toxgui.pro | 6 ++++-- 5 files changed, 94 insertions(+), 19 deletions(-) create mode 100644 audiobuffer.cpp create mode 100644 audiobuffer.h diff --git a/audiobuffer.cpp b/audiobuffer.cpp new file mode 100644 index 000000000..354aabf15 --- /dev/null +++ b/audiobuffer.cpp @@ -0,0 +1,36 @@ +#include "audiobuffer.h" + +AudioBuffer::AudioBuffer() : + QIODevice(0) +{ + open(QIODevice::ReadOnly); +} + +AudioBuffer::~AudioBuffer() +{ + close(); +} + +qint64 AudioBuffer::readData(char *data, qint64 len) +{ + const qint64 total = qMin((qint64)buffer.size(), len); + memcpy(data, buffer.constData(), total); + buffer = buffer.mid(total); + return total; +} + +qint64 AudioBuffer::writeData(const char* data, qint64 len) +{ + buffer.append(data, len); + return 0; +} + +qint64 AudioBuffer::bytesAvailable() const +{ + return buffer.size() + QIODevice::bytesAvailable(); +} + +qint64 AudioBuffer::bufferSize() const +{ + return buffer.size(); +} diff --git a/audiobuffer.h b/audiobuffer.h new file mode 100644 index 000000000..5041cc3f6 --- /dev/null +++ b/audiobuffer.h @@ -0,0 +1,23 @@ +#ifndef AUDIOBUFFER_H +#define AUDIOBUFFER_H + +#include +#include + +class AudioBuffer : public QIODevice +{ + Q_OBJECT +public: + explicit AudioBuffer(); + ~AudioBuffer(); + + qint64 readData(char *data, qint64 maxlen); + qint64 writeData(const char *data, qint64 len); + qint64 bytesAvailable() const; + qint64 bufferSize() const; + +private: + QByteArray buffer; +}; + +#endif // AUDIOBUFFER_H diff --git a/core.cpp b/core.cpp index de6fe8306..4e3d60755 100644 --- a/core.cpp +++ b/core.cpp @@ -17,6 +17,7 @@ #include "core.h" #include "cdata.h" #include "cstring.h" +#include "settings.h" #include #include @@ -27,8 +28,6 @@ #include #include -#include "settings.h" - const QString Core::CONFIG_FILE_NAME = "tox_save"; QList Core::fileSendQueue; QList Core::fileRecvQueue; @@ -623,7 +622,9 @@ void Core::process() fflush(stdout); #endif checkConnection(); - toxTimer->start(tox_do_interval(tox)); + int toxInterval = tox_do_interval(tox); + //qDebug() << QString("Tox interval %1").arg(toxInterval); + toxTimer->start(50); } void Core::checkConnection() @@ -1027,7 +1028,6 @@ void Core::prepareCall(int friendId, int callId, ToxAv* toxav) qDebug() << QString("Core: preparing call %1").arg(callId); calls[callId].callId = callId; calls[callId].friendId = friendId; - calls[callId].audioBuffer.open(QBuffer::ReadWrite); calls[callId].codecSettings = av_DefaultSettings; toxav_prepare_transmission(toxav, callId, &calls[callId].codecSettings, false); QAudioFormat format; @@ -1047,11 +1047,16 @@ void Core::prepareCall(int friendId, int callId, ToxAv* toxav) calls[callId].audioOutput = new QAudioOutput(format); calls[callId].active = true; - QtConcurrent::run(playCallAudio, callId, toxav); + calls[callId].audioOutput->setBufferSize(24000); calls[callId].audioOutput->start(&calls[callId].audioBuffer); if (calls[callId].audioOutput->state() == QAudio::StoppedState && calls[callId].audioOutput->error() == QAudio::OpenError) + { qWarning() << "Core: Unable to start audio"; + } + else + qDebug() << QString("Core: Audio started, buffer size %1").arg(calls[callId].audioOutput->bufferSize()); + QtConcurrent::run(playCallAudio, callId, toxav); } void Core::cleanupCall(int callId) @@ -1062,7 +1067,6 @@ void Core::cleanupCall(int callId) { delete calls[callId].audioOutput; } - calls[callId].audioBuffer.close(); } void Core::playCallAudio(int callId, ToxAv* toxav) @@ -1070,25 +1074,34 @@ void Core::playCallAudio(int callId, ToxAv* toxav) while (calls[callId].active) { int framesize = (calls[callId].codecSettings.audio_frame_duration * calls[callId].codecSettings.audio_sample_rate) / 1000; - int16_t buf[framesize]; - int len = toxav_recv_audio(toxav, callId, framesize, buf); + uint8_t buf[framesize*2]; + int len = toxav_recv_audio(toxav, callId, framesize, (int16_t*)buf); + qDebug() << QString("Core: Received %1 audio bytes").arg(len); if (len < 0) { - qDebug() << "Core::playCallAudio: Error receiving audio"; - QThread::msleep(calls[callId].codecSettings.audio_frame_duration / 2); + qDebug() << QString("Core::playCallAudio: Error receiving audio: %1").arg(len); + QThread::msleep(5); continue; } if (len == 0) + { + qApp->processEvents(); + QThread::msleep(5); continue; - qDebug() << QString("Core::playCallAudio: Playing %1 frames/sample").arg(len); - calls[callId].audioBuffer.write((char*)buf, len*2); - calls[callId].audioBuffer.seek(0); - calls[callId].audioOutput->resume(); + } + //qDebug() << QString("Core: Received %1 bytes, %2 audio bytes free, %3 core buffer size") + //.arg(len*2).arg(calls[callId].audioOutput->bytesFree()).arg(calls[callId].audioBuffer.bufferSize()); + calls[callId].audioBuffer.writeData((char*)buf, len*2); + int state = calls[callId].audioOutput->state(); + if (state != QAudio::ActiveState) + { + qDebug() << QString("Core: Audio state is %1").arg(state); + } int error = calls[callId].audioOutput->error(); if (error != QAudio::NoError) qWarning() << QString("Core::playCallAudio: Error: %1").arg(error); - QThread::msleep(calls[callId].codecSettings.audio_frame_duration / 2); + QThread::msleep(5); } } diff --git a/core.h b/core.h index 197ccc480..ed03a9837 100644 --- a/core.h +++ b/core.h @@ -18,6 +18,7 @@ #define CORE_HPP #include "status.h" +#include "audiobuffer.h" #include #include @@ -33,7 +34,7 @@ #include #include -#define TOXAV_MAX_CALLS 32 +#define TOXAV_MAX_CALLS 16 #define GROUPCHAT_MAX_SIZE 32 #define TOX_SAVE_INTERVAL 30*1000 #define TOX_FILE_INTERVAL 20 @@ -82,7 +83,7 @@ struct ToxFile struct ToxCall { public: - QBuffer audioBuffer; + AudioBuffer audioBuffer; QAudioOutput* audioOutput; ToxAvCodecSettings codecSettings; int callId; diff --git a/toxgui.pro b/toxgui.pro index 8f5246083..1819c2dc3 100644 --- a/toxgui.pro +++ b/toxgui.pro @@ -33,7 +33,8 @@ HEADERS += widget/form/addfriendform.h \ core.h \ friendlist.h \ cdata.h \ - cstring.h + cstring.h \ + audiobuffer.h FORMS += widget.ui @@ -68,4 +69,5 @@ SOURCES += \ settings.cpp \ status.cpp \ cdata.cpp \ - cstring.cpp + cstring.cpp \ + audiobuffer.cpp