mirror of
https://github.com/overte-org/overte.git
synced 2025-08-10 02:31:13 +02:00
implement threaded send for AudioInjector
This commit is contained in:
parent
2409b5f784
commit
bdf8c4b9b2
6 changed files with 112 additions and 4 deletions
|
@ -53,8 +53,6 @@
|
||||||
const short JITTER_BUFFER_MSECS = 12;
|
const short JITTER_BUFFER_MSECS = 12;
|
||||||
const short JITTER_BUFFER_SAMPLES = JITTER_BUFFER_MSECS * (SAMPLE_RATE / 1000.0);
|
const short JITTER_BUFFER_SAMPLES = JITTER_BUFFER_MSECS * (SAMPLE_RATE / 1000.0);
|
||||||
|
|
||||||
const unsigned int BUFFER_SEND_INTERVAL_USECS = floorf((NETWORK_BUFFER_LENGTH_SAMPLES_PER_CHANNEL / (float) SAMPLE_RATE) * 1000 * 1000);
|
|
||||||
|
|
||||||
const char AUDIO_MIXER_LOGGING_TARGET_NAME[] = "audio-mixer";
|
const char AUDIO_MIXER_LOGGING_TARGET_NAME[] = "audio-mixer";
|
||||||
|
|
||||||
void attachNewBufferToNode(Node *newNode) {
|
void attachNewBufferToNode(Node *newNode) {
|
||||||
|
|
|
@ -1321,7 +1321,9 @@ void Application::timer() {
|
||||||
static AudioInjector testInjector(QUrl("https://dl.dropboxusercontent.com/u/1864924/throw.raw"));
|
static AudioInjector testInjector(QUrl("https://dl.dropboxusercontent.com/u/1864924/throw.raw"));
|
||||||
|
|
||||||
if (testInjector.size()) {
|
if (testInjector.size()) {
|
||||||
testInjector.injectViaThread(&_audio);
|
testInjector.setPosition(_myAvatar.getHead().getPosition());
|
||||||
|
testInjector.setOrientation(_myAvatar.getOrientation());
|
||||||
|
testInjector.injectViaThread();
|
||||||
}
|
}
|
||||||
|
|
||||||
// give the MyAvatar object position to the Profile so it can propagate to the data-server
|
// give the MyAvatar object position to the Profile so it can propagate to the data-server
|
||||||
|
|
|
@ -6,18 +6,29 @@
|
||||||
// Copyright (c) 2013 HighFidelity, Inc. All rights reserved.
|
// Copyright (c) 2013 HighFidelity, Inc. All rights reserved.
|
||||||
//
|
//
|
||||||
|
|
||||||
|
#include <sys/time.h>
|
||||||
|
|
||||||
#include <QtNetwork/QNetworkAccessManager>
|
#include <QtNetwork/QNetworkAccessManager>
|
||||||
#include <QtNetwork/QNetworkReply>
|
#include <QtNetwork/QNetworkReply>
|
||||||
#include <QtNetwork/QNetworkRequest>
|
#include <QtNetwork/QNetworkRequest>
|
||||||
|
|
||||||
|
#include <NodeList.h>
|
||||||
|
#include <PacketHeaders.h>
|
||||||
|
#include <SharedUtil.h>
|
||||||
|
#include <UUID.h>
|
||||||
|
|
||||||
#include "AbstractAudioInterface.h"
|
#include "AbstractAudioInterface.h"
|
||||||
|
#include "AudioRingBuffer.h"
|
||||||
|
|
||||||
#include "AudioInjector.h"
|
#include "AudioInjector.h"
|
||||||
|
|
||||||
int abstractAudioPointerMeta = qRegisterMetaType<AbstractAudioInterface*>("AbstractAudioInterface*");
|
int abstractAudioPointerMeta = qRegisterMetaType<AbstractAudioInterface*>("AbstractAudioInterface*");
|
||||||
|
|
||||||
AudioInjector::AudioInjector(const QUrl& sampleURL) :
|
AudioInjector::AudioInjector(const QUrl& sampleURL) :
|
||||||
_sourceURL(sampleURL)
|
_currentSendPosition(0),
|
||||||
|
_sourceURL(sampleURL),
|
||||||
|
_position(0,0,0),
|
||||||
|
_orientation()
|
||||||
{
|
{
|
||||||
// we want to live on our own thread
|
// we want to live on our own thread
|
||||||
moveToThread(&_thread);
|
moveToThread(&_thread);
|
||||||
|
@ -58,5 +69,88 @@ void AudioInjector::injectAudio(AbstractAudioInterface* localAudioInterface) {
|
||||||
Q_ARG(QByteArray, _sampleByteArray));
|
Q_ARG(QByteArray, _sampleByteArray));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NodeList* nodeList = NodeList::getInstance();
|
||||||
|
|
||||||
|
// reset the current send position to the beginning
|
||||||
|
_currentSendPosition = 0;
|
||||||
|
|
||||||
|
// setup the packet for injected audio
|
||||||
|
unsigned char injectedAudioPacket[MAX_PACKET_SIZE];
|
||||||
|
unsigned char* currentPacketPosition = injectedAudioPacket;
|
||||||
|
|
||||||
|
int numBytesPacketHeader = populateTypeAndVersion(injectedAudioPacket, PACKET_TYPE_INJECT_AUDIO);
|
||||||
|
currentPacketPosition += numBytesPacketHeader;
|
||||||
|
|
||||||
|
// pack the session UUID for this Node
|
||||||
|
QByteArray rfcSessionUUID = NodeList::getInstance()->getOwnerUUID().toRfc4122();
|
||||||
|
memcpy(currentPacketPosition, rfcSessionUUID.constData(), rfcSessionUUID.size());
|
||||||
|
currentPacketPosition += rfcSessionUUID.size();
|
||||||
|
|
||||||
|
// pick a random UUID to use for this stream
|
||||||
|
QUuid randomStreamUUID;
|
||||||
|
QByteArray rfcStreamUUID = randomStreamUUID.toRfc4122();
|
||||||
|
memcpy(currentPacketPosition, rfcStreamUUID, rfcStreamUUID.size());
|
||||||
|
currentPacketPosition += rfcStreamUUID.size();
|
||||||
|
|
||||||
|
// pack the position for injected audio
|
||||||
|
memcpy(currentPacketPosition, &_position, sizeof(_position));
|
||||||
|
currentPacketPosition += sizeof(_position);
|
||||||
|
|
||||||
|
// pack a zero orientation for injected audio
|
||||||
|
memcpy(currentPacketPosition, &_orientation, sizeof(_orientation));
|
||||||
|
currentPacketPosition += sizeof(_orientation);
|
||||||
|
|
||||||
|
// pack zero for radius
|
||||||
|
float radius = 0;
|
||||||
|
memcpy(currentPacketPosition, &radius, sizeof(radius));
|
||||||
|
currentPacketPosition += sizeof(radius);
|
||||||
|
|
||||||
|
// pack 255 for attenuation byte
|
||||||
|
uchar volume = 1;
|
||||||
|
memcpy(currentPacketPosition, &volume, sizeof(volume));
|
||||||
|
currentPacketPosition += sizeof(volume);
|
||||||
|
|
||||||
|
timeval startTime = {};
|
||||||
|
gettimeofday(&startTime, NULL);
|
||||||
|
int nextFrame = 0;
|
||||||
|
|
||||||
|
// loop to send off our audio in NETWORK_BUFFER_LENGTH_SAMPLES_PER_CHANNEL byte chunks
|
||||||
|
while (_currentSendPosition < _sampleByteArray.size()) {
|
||||||
|
|
||||||
|
int bytesToCopy = std::min(NETWORK_BUFFER_LENGTH_BYTES_PER_CHANNEL,
|
||||||
|
_sampleByteArray.size() - _currentSendPosition);
|
||||||
|
|
||||||
|
// copy the next NETWORK_BUFFER_LENGTH_BYTES_PER_CHANNEL bytes to the packet
|
||||||
|
memcpy(currentPacketPosition, _sampleByteArray.data() + _currentSendPosition,
|
||||||
|
bytesToCopy);
|
||||||
|
|
||||||
|
|
||||||
|
// grab our audio mixer from the NodeList, if it exists
|
||||||
|
Node* audioMixer = nodeList->soloNodeOfType(NODE_TYPE_AUDIO_MIXER);
|
||||||
|
|
||||||
|
if (audioMixer && nodeList->getNodeActiveSocketOrPing(audioMixer)) {
|
||||||
|
// send off this audio packet
|
||||||
|
nodeList->getNodeSocket().writeDatagram((char*) injectedAudioPacket,
|
||||||
|
(currentPacketPosition - injectedAudioPacket) + bytesToCopy,
|
||||||
|
audioMixer->getActiveSocket()->getAddress(),
|
||||||
|
audioMixer->getActiveSocket()->getPort());
|
||||||
|
}
|
||||||
|
|
||||||
|
_currentSendPosition += bytesToCopy;
|
||||||
|
|
||||||
|
// send two packets before the first sleep so the mixer can start playback right away
|
||||||
|
|
||||||
|
if (_currentSendPosition != bytesToCopy && _currentSendPosition < _sampleByteArray.size()) {
|
||||||
|
// not the first packet and not done
|
||||||
|
// sleep for the appropriate time
|
||||||
|
int usecToSleep = usecTimestamp(&startTime) + (++nextFrame * BUFFER_SEND_INTERVAL_USECS) - usecTimestampNow();
|
||||||
|
|
||||||
|
if (usecToSleep > 0) {
|
||||||
|
usleep(usecToSleep);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -13,6 +13,9 @@
|
||||||
#include <QtCore/QThread>
|
#include <QtCore/QThread>
|
||||||
#include <QtCore/QUrl>
|
#include <QtCore/QUrl>
|
||||||
|
|
||||||
|
#include <glm/glm.hpp>
|
||||||
|
#include <glm/gtx/quaternion.hpp>
|
||||||
|
|
||||||
class AbstractAudioInterface;
|
class AbstractAudioInterface;
|
||||||
class QNetworkReply;
|
class QNetworkReply;
|
||||||
|
|
||||||
|
@ -22,13 +25,19 @@ public:
|
||||||
AudioInjector(const QUrl& sampleURL);
|
AudioInjector(const QUrl& sampleURL);
|
||||||
|
|
||||||
int size() const { return _sampleByteArray.size(); }
|
int size() const { return _sampleByteArray.size(); }
|
||||||
|
|
||||||
|
void setPosition(const glm::vec3& position) { _position = position; }
|
||||||
|
void setOrientation(const glm::quat& orientation) { _orientation = orientation; }
|
||||||
public slots:
|
public slots:
|
||||||
void injectViaThread(AbstractAudioInterface* localAudioInterface = NULL);
|
void injectViaThread(AbstractAudioInterface* localAudioInterface = NULL);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QByteArray _sampleByteArray;
|
QByteArray _sampleByteArray;
|
||||||
|
int _currentSendPosition;
|
||||||
QThread _thread;
|
QThread _thread;
|
||||||
QUrl _sourceURL;
|
QUrl _sourceURL;
|
||||||
|
glm::vec3 _position;
|
||||||
|
glm::quat _orientation;
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void startDownload();
|
void startDownload();
|
||||||
|
|
|
@ -25,6 +25,9 @@ const int NETWORK_BUFFER_LENGTH_SAMPLES_STEREO = NETWORK_BUFFER_LENGTH_BYTES_STE
|
||||||
const int NETWORK_BUFFER_LENGTH_BYTES_PER_CHANNEL = 512;
|
const int NETWORK_BUFFER_LENGTH_BYTES_PER_CHANNEL = 512;
|
||||||
const int NETWORK_BUFFER_LENGTH_SAMPLES_PER_CHANNEL = NETWORK_BUFFER_LENGTH_BYTES_PER_CHANNEL / sizeof(int16_t);
|
const int NETWORK_BUFFER_LENGTH_SAMPLES_PER_CHANNEL = NETWORK_BUFFER_LENGTH_BYTES_PER_CHANNEL / sizeof(int16_t);
|
||||||
|
|
||||||
|
const unsigned int BUFFER_SEND_INTERVAL_USECS = floorf((NETWORK_BUFFER_LENGTH_SAMPLES_PER_CHANNEL
|
||||||
|
/ (float) SAMPLE_RATE) * 1000 * 1000);
|
||||||
|
|
||||||
const short RING_BUFFER_LENGTH_FRAMES = 10;
|
const short RING_BUFFER_LENGTH_FRAMES = 10;
|
||||||
|
|
||||||
const int MAX_SAMPLE_VALUE = std::numeric_limits<int16_t>::max();
|
const int MAX_SAMPLE_VALUE = std::numeric_limits<int16_t>::max();
|
||||||
|
|
|
@ -42,6 +42,8 @@ int InjectedAudioRingBuffer::parseData(unsigned char* sourceBuffer, int numBytes
|
||||||
unsigned int attenuationByte = *(currentBuffer++);
|
unsigned int attenuationByte = *(currentBuffer++);
|
||||||
_attenuationRatio = attenuationByte / (float) MAX_INJECTOR_VOLUME;
|
_attenuationRatio = attenuationByte / (float) MAX_INJECTOR_VOLUME;
|
||||||
|
|
||||||
|
qDebug() << "Copying" << numBytes - (currentBuffer - sourceBuffer) << "for injected ring buffer\n";
|
||||||
|
|
||||||
currentBuffer += writeData((char*) currentBuffer, numBytes - (currentBuffer - sourceBuffer));
|
currentBuffer += writeData((char*) currentBuffer, numBytes - (currentBuffer - sourceBuffer));
|
||||||
|
|
||||||
return currentBuffer - sourceBuffer;
|
return currentBuffer - sourceBuffer;
|
||||||
|
|
Loading…
Reference in a new issue