2019-08-19 23:08:54 +00:00
|
|
|
#include "lib/audiodev/AudioVoiceEngine.hpp"
|
|
|
|
|
2018-05-06 22:44:31 +00:00
|
|
|
#include "boo/boo.hpp"
|
2019-08-19 23:08:54 +00:00
|
|
|
#include "lib/audiodev/LinuxMidi.hpp"
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2019-08-19 23:08:54 +00:00
|
|
|
#include <logvisor/logvisor.hpp>
|
2018-05-06 22:44:31 +00:00
|
|
|
#include <pulse/pulseaudio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
namespace boo {
|
2018-05-06 22:44:31 +00:00
|
|
|
static logvisor::Module Log("boo::PulseAudio");
|
|
|
|
logvisor::Module ALSALog("boo::ALSA");
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
static const uint64_t StereoChans = (1 << PA_CHANNEL_POSITION_FRONT_LEFT) | (1 << PA_CHANNEL_POSITION_FRONT_RIGHT);
|
|
|
|
|
|
|
|
static const uint64_t QuadChans = (1 << PA_CHANNEL_POSITION_FRONT_LEFT) | (1 << PA_CHANNEL_POSITION_FRONT_RIGHT) |
|
|
|
|
(1 << PA_CHANNEL_POSITION_REAR_LEFT) | (1 << PA_CHANNEL_POSITION_REAR_RIGHT);
|
|
|
|
|
|
|
|
static const uint64_t S51Chans = (1 << PA_CHANNEL_POSITION_FRONT_LEFT) | (1 << PA_CHANNEL_POSITION_FRONT_RIGHT) |
|
|
|
|
(1 << PA_CHANNEL_POSITION_REAR_LEFT) | (1 << PA_CHANNEL_POSITION_REAR_RIGHT) |
|
|
|
|
(1 << PA_CHANNEL_POSITION_FRONT_CENTER) | (1 << PA_CHANNEL_POSITION_LFE);
|
|
|
|
|
|
|
|
static const uint64_t S71Chans = (1 << PA_CHANNEL_POSITION_FRONT_LEFT) | (1 << PA_CHANNEL_POSITION_FRONT_RIGHT) |
|
|
|
|
(1 << PA_CHANNEL_POSITION_REAR_LEFT) | (1 << PA_CHANNEL_POSITION_REAR_RIGHT) |
|
|
|
|
(1 << PA_CHANNEL_POSITION_FRONT_CENTER) | (1 << PA_CHANNEL_POSITION_LFE) |
|
|
|
|
(1 << PA_CHANNEL_POSITION_SIDE_LEFT) | (1 << PA_CHANNEL_POSITION_SIDE_RIGHT);
|
|
|
|
|
|
|
|
struct PulseAudioVoiceEngine : LinuxMidi {
|
|
|
|
pa_mainloop* m_mainloop = nullptr;
|
|
|
|
pa_context* m_ctx = nullptr;
|
|
|
|
pa_stream* m_stream = nullptr;
|
|
|
|
std::string m_sinkName;
|
|
|
|
bool m_handleMove = false;
|
|
|
|
pa_sample_spec m_sampleSpec = {};
|
|
|
|
pa_channel_map m_chanMap = {};
|
|
|
|
|
|
|
|
int _paWaitReady() {
|
|
|
|
int retval = 0;
|
|
|
|
while (pa_context_get_state(m_ctx) < PA_CONTEXT_READY)
|
|
|
|
pa_mainloop_iterate(m_mainloop, 1, &retval);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
int _paStreamWaitReady() {
|
|
|
|
int retval = 0;
|
|
|
|
while (pa_stream_get_state(m_stream) < PA_STREAM_READY)
|
|
|
|
pa_mainloop_iterate(m_mainloop, 1, &retval);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
int _paIterate(pa_operation* op) const {
|
|
|
|
int retval = 0;
|
|
|
|
while (pa_operation_get_state(op) == PA_OPERATION_RUNNING)
|
|
|
|
pa_mainloop_iterate(m_mainloop, 1, &retval);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool _setupSink() {
|
|
|
|
if (m_stream) {
|
|
|
|
pa_stream_disconnect(m_stream);
|
|
|
|
pa_stream_unref(m_stream);
|
|
|
|
m_stream = nullptr;
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_operation* op;
|
|
|
|
m_sampleSpec.format = PA_SAMPLE_INVALID;
|
|
|
|
op = pa_context_get_sink_info_by_name(m_ctx, m_sinkName.c_str(), pa_sink_info_cb_t(_getSinkInfoReply), this);
|
|
|
|
_paIterate(op);
|
|
|
|
pa_operation_unref(op);
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
if (m_sampleSpec.format == PA_SAMPLE_INVALID) {
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to setup audio stream"));
|
2018-12-08 05:17:51 +00:00
|
|
|
goto err;
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
m_5msFrames = m_sampleSpec.rate * 5 / 1000;
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
m_mixInfo.m_sampleRate = m_sampleSpec.rate;
|
|
|
|
m_mixInfo.m_sampleFormat = SOXR_FLOAT32;
|
|
|
|
m_mixInfo.m_bitsPerSample = 32;
|
|
|
|
m_mixInfo.m_periodFrames = m_5msFrames;
|
|
|
|
if (!(m_stream = pa_stream_new(m_ctx, "master", &m_sampleSpec, &m_chanMap))) {
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_stream_new(): {}"), pa_strerror(pa_context_errno(m_ctx)));
|
2018-12-08 05:17:51 +00:00
|
|
|
goto err;
|
|
|
|
}
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_buffer_attr bufAttr;
|
|
|
|
bufAttr.minreq = uint32_t(m_5msFrames * m_sampleSpec.channels * sizeof(float));
|
|
|
|
bufAttr.maxlength = bufAttr.minreq * 24;
|
|
|
|
bufAttr.tlength = bufAttr.maxlength;
|
|
|
|
bufAttr.prebuf = UINT32_MAX;
|
|
|
|
bufAttr.fragsize = UINT32_MAX;
|
|
|
|
|
|
|
|
if (pa_stream_connect_playback(m_stream, m_sinkName.c_str(), &bufAttr,
|
|
|
|
pa_stream_flags_t(PA_STREAM_START_UNMUTED | PA_STREAM_EARLY_REQUESTS), nullptr,
|
|
|
|
nullptr)) {
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_stream_connect_playback()"));
|
2018-12-08 05:17:51 +00:00
|
|
|
goto err;
|
|
|
|
}
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_stream_set_moved_callback(m_stream, pa_stream_notify_cb_t(_streamMoved), this);
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
_paStreamWaitReady();
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
_resetSampleRate();
|
|
|
|
return true;
|
|
|
|
err:
|
|
|
|
if (m_stream) {
|
|
|
|
pa_stream_disconnect(m_stream);
|
|
|
|
pa_stream_unref(m_stream);
|
|
|
|
m_stream = nullptr;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
PulseAudioVoiceEngine() {
|
|
|
|
if (!(m_mainloop = pa_mainloop_new())) {
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_mainloop_new()"));
|
2018-12-08 05:17:51 +00:00
|
|
|
return;
|
2018-08-18 22:08:58 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_mainloop_api* mlApi = pa_mainloop_get_api(m_mainloop);
|
|
|
|
pa_proplist* propList = pa_proplist_new();
|
|
|
|
pa_proplist_sets(propList, PA_PROP_APPLICATION_ICON_NAME, APP->getUniqueName().data());
|
2019-07-20 04:22:36 +00:00
|
|
|
pa_proplist_sets(propList, PA_PROP_APPLICATION_PROCESS_ID, fmt::format(fmt("{}"), int(getpid())).c_str());
|
2018-12-08 05:17:51 +00:00
|
|
|
if (!(m_ctx = pa_context_new_with_proplist(mlApi, APP->getFriendlyName().data(), propList))) {
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_context_new_with_proplist()"));
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_mainloop_free(m_mainloop);
|
|
|
|
m_mainloop = nullptr;
|
|
|
|
return;
|
|
|
|
}
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_operation* op;
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
if (pa_context_connect(m_ctx, nullptr, PA_CONTEXT_NOFLAGS, nullptr)) {
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_context_connect()"));
|
2018-12-08 05:17:51 +00:00
|
|
|
goto err;
|
|
|
|
}
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
_paWaitReady();
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
op = pa_context_get_server_info(m_ctx, pa_server_info_cb_t(_getServerInfoReply), this);
|
|
|
|
_paIterate(op);
|
|
|
|
pa_operation_unref(op);
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
if (!_setupSink())
|
|
|
|
goto err;
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
return;
|
|
|
|
err:
|
|
|
|
pa_context_disconnect(m_ctx);
|
|
|
|
pa_context_unref(m_ctx);
|
|
|
|
m_ctx = nullptr;
|
|
|
|
pa_mainloop_free(m_mainloop);
|
|
|
|
m_mainloop = nullptr;
|
|
|
|
}
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2019-08-13 00:52:20 +00:00
|
|
|
~PulseAudioVoiceEngine() override {
|
2018-12-08 05:17:51 +00:00
|
|
|
if (m_stream) {
|
|
|
|
pa_stream_disconnect(m_stream);
|
|
|
|
pa_stream_unref(m_stream);
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
if (m_ctx) {
|
|
|
|
pa_context_disconnect(m_ctx);
|
|
|
|
pa_context_unref(m_ctx);
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
if (m_mainloop) {
|
|
|
|
pa_mainloop_free(m_mainloop);
|
2018-08-18 22:08:58 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void _streamMoved(pa_stream* p, PulseAudioVoiceEngine* userdata) {
|
|
|
|
userdata->m_sinkName = pa_stream_get_device_name(p);
|
|
|
|
userdata->m_handleMove = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _getServerInfoReply(pa_context* c, const pa_server_info* i, PulseAudioVoiceEngine* userdata) {
|
|
|
|
userdata->m_sinkName = i->default_sink_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _parseAudioChannelSet(const pa_channel_map* chm) {
|
|
|
|
m_chanMap = *chm;
|
|
|
|
|
|
|
|
ChannelMap& chmapOut = m_mixInfo.m_channelMap;
|
|
|
|
m_mixInfo.m_channels = AudioChannelSet::Unknown;
|
|
|
|
|
|
|
|
uint64_t chBits = 0;
|
|
|
|
chmapOut.m_channelCount = chm->channels;
|
|
|
|
for (unsigned c = 0; c < chm->channels; ++c) {
|
|
|
|
chBits |= 1 << chm->map[c];
|
|
|
|
switch (chm->map[c]) {
|
|
|
|
case PA_CHANNEL_POSITION_FRONT_LEFT:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::FrontLeft;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_FRONT_RIGHT:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::FrontRight;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_REAR_LEFT:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::RearLeft;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_REAR_RIGHT:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::RearRight;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_FRONT_CENTER:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::FrontCenter;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_LFE:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::LFE;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_SIDE_LEFT:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::SideLeft;
|
|
|
|
break;
|
|
|
|
case PA_CHANNEL_POSITION_SIDE_RIGHT:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::SideRight;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
chmapOut.m_channels[c] = AudioChannel::Unknown;
|
|
|
|
break;
|
|
|
|
}
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
static const std::array<AudioChannelSet, 4> testSets = {
|
|
|
|
{AudioChannelSet::Surround71, AudioChannelSet::Surround51, AudioChannelSet::Quad, AudioChannelSet::Stereo}};
|
|
|
|
for (AudioChannelSet set : testSets) {
|
|
|
|
switch (set) {
|
|
|
|
case AudioChannelSet::Stereo: {
|
|
|
|
if ((chBits & StereoChans) == StereoChans) {
|
|
|
|
m_mixInfo.m_channels = AudioChannelSet::Stereo;
|
|
|
|
return;
|
2018-05-15 02:55:22 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case AudioChannelSet::Quad: {
|
|
|
|
if ((chBits & QuadChans) == QuadChans) {
|
|
|
|
m_mixInfo.m_channels = AudioChannelSet::Quad;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case AudioChannelSet::Surround51: {
|
|
|
|
if ((chBits & S51Chans) == S51Chans) {
|
|
|
|
m_mixInfo.m_channels = AudioChannelSet::Surround51;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case AudioChannelSet::Surround71: {
|
|
|
|
if ((chBits & S71Chans) == S71Chans) {
|
|
|
|
m_mixInfo.m_channels = AudioChannelSet::Surround71;
|
|
|
|
return;
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void _getSinkInfoReply(pa_context* c, const pa_sink_info* i, int eol, PulseAudioVoiceEngine* userdata) {
|
|
|
|
if (!i)
|
|
|
|
return;
|
|
|
|
userdata->m_sampleSpec.format = PA_SAMPLE_FLOAT32;
|
|
|
|
userdata->m_sampleSpec.rate = i->sample_spec.rate;
|
|
|
|
userdata->m_sampleSpec.channels = i->sample_spec.channels;
|
|
|
|
userdata->_parseAudioChannelSet(&i->channel_map);
|
|
|
|
}
|
|
|
|
|
|
|
|
mutable std::vector<std::pair<std::string, std::string>> m_sinks;
|
|
|
|
static void _getSinkInfoListReply(pa_context* c, const pa_sink_info* i, int eol, PulseAudioVoiceEngine* userdata) {
|
|
|
|
if (i)
|
|
|
|
userdata->m_sinks.push_back(std::make_pair(i->name, i->description));
|
|
|
|
}
|
2019-08-13 21:39:45 +00:00
|
|
|
std::vector<std::pair<std::string, std::string>> enumerateAudioOutputs() const override {
|
2018-12-08 05:17:51 +00:00
|
|
|
pa_operation* op = pa_context_get_sink_info_list(m_ctx, pa_sink_info_cb_t(_getSinkInfoListReply), (void*)this);
|
|
|
|
_paIterate(op);
|
|
|
|
pa_operation_unref(op);
|
|
|
|
std::vector<std::pair<std::string, std::string>> ret;
|
|
|
|
ret.swap(m_sinks);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-08-13 00:52:20 +00:00
|
|
|
std::string getCurrentAudioOutput() const override { return m_sinkName; }
|
2018-12-08 05:17:51 +00:00
|
|
|
|
|
|
|
bool m_sinkOk = false;
|
|
|
|
static void _checkAudioSinkReply(pa_context* c, const pa_sink_info* i, int eol, PulseAudioVoiceEngine* userdata) {
|
|
|
|
if (i)
|
|
|
|
userdata->m_sinkOk = true;
|
|
|
|
}
|
2019-08-13 00:52:20 +00:00
|
|
|
bool setCurrentAudioOutput(const char* name) override {
|
2018-12-08 05:17:51 +00:00
|
|
|
m_sinkOk = false;
|
|
|
|
pa_operation* op;
|
|
|
|
op = pa_context_get_sink_info_by_name(m_ctx, name, pa_sink_info_cb_t(_checkAudioSinkReply), this);
|
|
|
|
_paIterate(op);
|
|
|
|
pa_operation_unref(op);
|
|
|
|
if (m_sinkOk) {
|
|
|
|
m_sinkName = name;
|
|
|
|
return _setupSink();
|
2018-05-06 22:44:31 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _doIterate() {
|
|
|
|
int retval;
|
|
|
|
pa_mainloop_iterate(m_mainloop, 1, &retval);
|
|
|
|
if (m_handleMove) {
|
|
|
|
m_handleMove = false;
|
|
|
|
_setupSink();
|
2018-08-18 22:08:58 +00:00
|
|
|
}
|
2018-12-08 05:17:51 +00:00
|
|
|
}
|
|
|
|
|
2019-08-13 00:52:20 +00:00
|
|
|
void pumpAndMixVoices() override {
|
2018-12-08 05:17:51 +00:00
|
|
|
if (!m_stream) {
|
|
|
|
/* Dummy pump mode - use failsafe defaults for 1/60sec of samples */
|
|
|
|
m_mixInfo.m_sampleRate = 32000.0;
|
|
|
|
m_mixInfo.m_sampleFormat = SOXR_FLOAT32_I;
|
|
|
|
m_mixInfo.m_bitsPerSample = 32;
|
|
|
|
m_5msFrames = 32000 / 60;
|
|
|
|
m_mixInfo.m_periodFrames = m_5msFrames;
|
|
|
|
m_mixInfo.m_channels = AudioChannelSet::Stereo;
|
|
|
|
m_mixInfo.m_channelMap.m_channelCount = 2;
|
|
|
|
m_mixInfo.m_channelMap.m_channels[0] = AudioChannel::FrontLeft;
|
|
|
|
m_mixInfo.m_channelMap.m_channels[1] = AudioChannel::FrontRight;
|
|
|
|
_pumpAndMixVoices(m_5msFrames, (float*)nullptr);
|
|
|
|
return;
|
2018-08-18 22:08:58 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
size_t writableSz = pa_stream_writable_size(m_stream);
|
|
|
|
size_t frameSz = m_mixInfo.m_channelMap.m_channelCount * sizeof(float);
|
|
|
|
size_t writableFrames = writableSz / frameSz;
|
|
|
|
size_t writablePeriods = writableFrames / m_mixInfo.m_periodFrames;
|
2018-08-18 22:08:58 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
if (!writablePeriods) {
|
|
|
|
_doIterate();
|
|
|
|
return;
|
2018-08-18 22:08:58 +00:00
|
|
|
}
|
|
|
|
|
2019-04-07 04:53:41 +00:00
|
|
|
void* data = nullptr;
|
2018-12-08 05:17:51 +00:00
|
|
|
size_t periodSz = m_mixInfo.m_periodFrames * frameSz;
|
|
|
|
size_t nbytes = writablePeriods * periodSz;
|
|
|
|
if (pa_stream_begin_write(m_stream, &data, &nbytes)) {
|
|
|
|
pa_stream_state_t st = pa_stream_get_state(m_stream);
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_stream_begin_write(): {} {}"), pa_strerror(pa_context_errno(m_ctx)), st);
|
2018-12-08 05:17:51 +00:00
|
|
|
_doIterate();
|
|
|
|
return;
|
2018-08-18 22:08:58 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
writablePeriods = nbytes / periodSz;
|
|
|
|
_pumpAndMixVoices(m_mixInfo.m_periodFrames * writablePeriods, reinterpret_cast<float*>(data));
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
if (pa_stream_write(m_stream, data, nbytes, nullptr, 0, PA_SEEK_RELATIVE))
|
2019-07-20 04:22:36 +00:00
|
|
|
Log.report(logvisor::Error, fmt("Unable to pa_stream_write()"));
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
_doIterate();
|
|
|
|
}
|
2018-05-06 22:44:31 +00:00
|
|
|
};
|
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
std::unique_ptr<IAudioVoiceEngine> NewAudioVoiceEngine() { return std::make_unique<PulseAudioVoiceEngine>(); }
|
2018-05-06 22:44:31 +00:00
|
|
|
|
2018-12-08 05:17:51 +00:00
|
|
|
} // namespace boo
|