2020-11-12 00:53:38 +01:00
|
|
|
#include <imgui.h>
|
2020-12-22 20:00:51 +01:00
|
|
|
#include <module.h>
|
2020-11-12 00:53:38 +01:00
|
|
|
#include <gui/gui.h>
|
|
|
|
#include <signal_path/signal_path.h>
|
|
|
|
#include <signal_path/sink.h>
|
2020-11-22 18:26:48 +01:00
|
|
|
#include <dsp/audio.h>
|
2021-02-20 15:27:43 +01:00
|
|
|
#include <dsp/processing.h>
|
2020-12-06 17:57:44 +01:00
|
|
|
#include <spdlog/spdlog.h>
|
2021-02-23 00:26:35 +01:00
|
|
|
#include <RtAudio.h>
|
|
|
|
#include <config.h>
|
|
|
|
#include <options.h>
|
2020-11-12 00:53:38 +01:00
|
|
|
|
|
|
|
#define CONCAT(a, b) ((std::string(a) + b).c_str())
|
|
|
|
|
2020-12-08 04:36:37 +01:00
|
|
|
SDRPP_MOD_INFO {
|
|
|
|
/* Name: */ "audio_sink",
|
|
|
|
/* Description: */ "Audio sink module for SDR++",
|
|
|
|
/* Author: */ "Ryzerth",
|
|
|
|
/* Version: */ 0, 1, 0,
|
|
|
|
/* Max instances */ 1
|
|
|
|
};
|
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
ConfigManager config;
|
|
|
|
|
2020-11-12 00:53:38 +01:00
|
|
|
class AudioSink : SinkManager::Sink {
|
|
|
|
public:
|
2020-11-29 20:55:00 +01:00
|
|
|
AudioSink(SinkManager::Stream* stream, std::string streamName) {
|
2020-11-12 00:53:38 +01:00
|
|
|
_stream = stream;
|
2020-11-29 20:55:00 +01:00
|
|
|
_streamName = streamName;
|
2020-11-30 05:51:33 +01:00
|
|
|
s2m.init(_stream->sinkOut);
|
2021-02-23 00:26:35 +01:00
|
|
|
monoPacker.init(&s2m.out, 512);
|
|
|
|
stereoPacker.init(_stream->sinkOut, 512);
|
|
|
|
|
|
|
|
bool created = false;
|
|
|
|
std::string device = "";
|
2021-07-09 14:24:07 -04:00
|
|
|
config.acquire();
|
2021-02-23 00:26:35 +01:00
|
|
|
if (!config.conf.contains(_streamName)) {
|
|
|
|
created = true;
|
|
|
|
config.conf[_streamName]["device"] = "";
|
|
|
|
config.conf[_streamName]["devices"] = json({});
|
|
|
|
}
|
|
|
|
device = config.conf[_streamName]["device"];
|
|
|
|
config.release(created);
|
|
|
|
|
|
|
|
int count = audio.getDeviceCount();
|
|
|
|
RtAudio::DeviceInfo info;
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
info = audio.getDeviceInfo(i);
|
|
|
|
if (!info.probed) { continue; }
|
|
|
|
if (info.outputChannels == 0) { continue; }
|
|
|
|
if (info.isDefaultOutput) { defaultDevId = devList.size(); }
|
|
|
|
devList.push_back(info);
|
|
|
|
deviceIds.push_back(i);
|
|
|
|
txtDevList += info.name;
|
2020-11-22 18:26:48 +01:00
|
|
|
txtDevList += '\0';
|
|
|
|
}
|
2020-11-29 20:55:00 +01:00
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
selectByName(device);
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
~AudioSink() {
|
2021-02-23 00:26:35 +01:00
|
|
|
|
2020-11-22 18:26:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
2020-11-29 20:55:00 +01:00
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doStart();
|
|
|
|
running = true;
|
2020-11-22 18:26:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void stop() {
|
2020-11-29 20:55:00 +01:00
|
|
|
if (!running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doStop();
|
|
|
|
running = false;
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
2021-02-23 00:26:35 +01:00
|
|
|
|
|
|
|
void selectFirst() {
|
|
|
|
selectById(defaultDevId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void selectByName(std::string name) {
|
|
|
|
for (int i = 0; i < devList.size(); i++) {
|
|
|
|
if (devList[i].name == name) {
|
|
|
|
selectById(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
selectFirst();
|
|
|
|
}
|
|
|
|
|
|
|
|
void selectById(int id) {
|
|
|
|
devId = id;
|
|
|
|
bool created = false;
|
2021-07-09 14:24:07 -04:00
|
|
|
config.acquire();
|
2021-02-23 00:26:35 +01:00
|
|
|
if (!config.conf[_streamName]["devices"].contains(devList[id].name)) {
|
|
|
|
created = true;
|
|
|
|
config.conf[_streamName]["devices"][devList[id].name] = devList[id].preferredSampleRate;
|
|
|
|
}
|
|
|
|
sampleRate = config.conf[_streamName]["devices"][devList[id].name];
|
|
|
|
config.release(created);
|
|
|
|
|
|
|
|
sampleRates = devList[id].sampleRates;
|
|
|
|
sampleRatesTxt = "";
|
|
|
|
char buf[256];
|
|
|
|
bool found = false;
|
|
|
|
unsigned int defaultId = 0;
|
|
|
|
unsigned int defaultSr = devList[id].preferredSampleRate;
|
|
|
|
for (int i = 0; i < sampleRates.size(); i++) {
|
|
|
|
if (sampleRates[i] == sampleRate) {
|
|
|
|
found = true;
|
|
|
|
srId = i;
|
|
|
|
}
|
|
|
|
if (sampleRates[i] == defaultSr) {
|
|
|
|
defaultId = i;
|
|
|
|
}
|
|
|
|
sprintf(buf, "%d", sampleRates[i]);
|
|
|
|
sampleRatesTxt += buf;
|
|
|
|
sampleRatesTxt += '\0';
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
sampleRate = defaultSr;
|
|
|
|
srId = defaultId;
|
|
|
|
}
|
|
|
|
|
|
|
|
_stream->setSampleRate(sampleRate);
|
|
|
|
|
|
|
|
if (running) { doStop(); }
|
|
|
|
if (running) { doStart(); }
|
|
|
|
}
|
2020-11-12 00:53:38 +01:00
|
|
|
|
|
|
|
void menuHandler() {
|
2020-11-30 05:51:33 +01:00
|
|
|
float menuWidth = ImGui::GetContentRegionAvailWidth();
|
|
|
|
|
|
|
|
ImGui::SetNextItemWidth(menuWidth);
|
2021-02-23 00:26:35 +01:00
|
|
|
if (ImGui::Combo(("##_audio_sink_dev_"+_streamName).c_str(), &devId, txtDevList.c_str())) {
|
|
|
|
selectById(devId);
|
2021-07-09 14:24:07 -04:00
|
|
|
config.acquire();
|
2021-02-23 00:26:35 +01:00
|
|
|
config.conf[_streamName]["device"] = devList[devId].name;
|
|
|
|
config.release(true);
|
2020-11-29 20:55:00 +01:00
|
|
|
}
|
|
|
|
|
2020-11-30 05:51:33 +01:00
|
|
|
ImGui::SetNextItemWidth(menuWidth);
|
2021-02-23 00:26:35 +01:00
|
|
|
if (ImGui::Combo(("##_audio_sink_sr_"+_streamName).c_str(), &srId, sampleRatesTxt.c_str())) {
|
|
|
|
sampleRate = sampleRates[srId];
|
|
|
|
_stream->setSampleRate(sampleRate);
|
2020-11-29 20:55:00 +01:00
|
|
|
if (running) {
|
|
|
|
doStop();
|
|
|
|
doStart();
|
|
|
|
}
|
2021-07-09 14:24:07 -04:00
|
|
|
config.acquire();
|
2021-02-23 00:26:35 +01:00
|
|
|
config.conf[_streamName]["devices"][devList[devId].name] = sampleRate;
|
|
|
|
config.release(true);
|
2020-11-29 20:55:00 +01:00
|
|
|
}
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2020-11-29 20:55:00 +01:00
|
|
|
void doStart() {
|
2021-02-23 00:26:35 +01:00
|
|
|
RtAudio::StreamParameters parameters;
|
|
|
|
parameters.deviceId = deviceIds[devId];
|
|
|
|
parameters.nChannels = 2;
|
|
|
|
unsigned int bufferFrames = sampleRate / 60;
|
|
|
|
RtAudio::StreamOptions opts;
|
|
|
|
opts.flags = RTAUDIO_MINIMIZE_LATENCY;
|
2021-02-28 16:32:57 +01:00
|
|
|
opts.streamName = _streamName;
|
2021-02-23 00:26:35 +01:00
|
|
|
|
|
|
|
try {
|
|
|
|
audio.openStream(¶meters, NULL, RTAUDIO_FLOAT32, sampleRate, &bufferFrames, &callback, this, &opts);
|
2021-07-24 19:53:57 +02:00
|
|
|
stereoPacker.setSampleCount(bufferFrames);
|
2021-02-23 00:26:35 +01:00
|
|
|
audio.startStream();
|
|
|
|
stereoPacker.start();
|
2020-11-29 20:55:00 +01:00
|
|
|
}
|
2021-02-23 00:26:35 +01:00
|
|
|
catch ( RtAudioError& e ) {
|
|
|
|
spdlog::error("Could not open audio device");
|
2020-11-29 20:55:00 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
spdlog::info("RtAudio stream open");
|
2020-11-29 20:55:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void doStop() {
|
2020-12-25 19:58:52 +01:00
|
|
|
s2m.stop();
|
2021-02-23 00:26:35 +01:00
|
|
|
monoPacker.stop();
|
|
|
|
stereoPacker.stop();
|
|
|
|
monoPacker.out.stopReader();
|
|
|
|
stereoPacker.out.stopReader();
|
|
|
|
audio.stopStream();
|
|
|
|
audio.closeStream();
|
|
|
|
monoPacker.out.clearReadStop();
|
|
|
|
stereoPacker.out.clearReadStop();
|
2020-11-29 20:55:00 +01:00
|
|
|
}
|
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
static int callback( void *outputBuffer, void *inputBuffer, unsigned int nBufferFrames, double streamTime, RtAudioStreamStatus status, void *userData) {
|
2020-11-29 20:55:00 +01:00
|
|
|
AudioSink* _this = (AudioSink*)userData;
|
2021-02-23 00:26:35 +01:00
|
|
|
int count = _this->stereoPacker.out.read();
|
|
|
|
if (count < 0) { return 0; }
|
2021-05-09 03:06:57 +02:00
|
|
|
|
2021-06-29 02:43:04 +02:00
|
|
|
// For debug purposes only...
|
|
|
|
// if (nBufferFrames != count) { spdlog::warn("Buffer size missmatch, wanted {0}, was asked for {1}", count, nBufferFrames); }
|
|
|
|
// for (int i = 0; i < count; i++) {
|
|
|
|
// if (_this->stereoPacker.out.readBuf[i].l == NAN || _this->stereoPacker.out.readBuf[i].r == NAN) { spdlog::error("NAN in audio data"); }
|
|
|
|
// if (_this->stereoPacker.out.readBuf[i].l == INFINITY || _this->stereoPacker.out.readBuf[i].r == INFINITY) { spdlog::error("INFINITY in audio data"); }
|
|
|
|
// if (_this->stereoPacker.out.readBuf[i].l == -INFINITY || _this->stereoPacker.out.readBuf[i].r == -INFINITY) { spdlog::error("-INFINITY in audio data"); }
|
|
|
|
// }
|
2021-05-09 03:06:57 +02:00
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
memcpy(outputBuffer, _this->stereoPacker.out.readBuf, nBufferFrames * sizeof(dsp::stereo_t));
|
|
|
|
_this->stereoPacker.out.flush();
|
2020-11-29 20:55:00 +01:00
|
|
|
return 0;
|
|
|
|
}
|
2020-11-22 18:26:48 +01:00
|
|
|
|
2020-11-12 00:53:38 +01:00
|
|
|
SinkManager::Stream* _stream;
|
2020-11-22 18:26:48 +01:00
|
|
|
dsp::StereoToMono s2m;
|
2021-02-23 00:26:35 +01:00
|
|
|
dsp::Packer<float> monoPacker;
|
|
|
|
dsp::Packer<dsp::stereo_t> stereoPacker;
|
2021-02-20 15:27:43 +01:00
|
|
|
|
2020-11-29 20:55:00 +01:00
|
|
|
std::string _streamName;
|
2020-11-22 18:26:48 +01:00
|
|
|
|
|
|
|
int srId = 0;
|
|
|
|
int devCount;
|
|
|
|
int devId = 0;
|
2020-11-29 20:55:00 +01:00
|
|
|
bool running = false;
|
2020-11-22 18:26:48 +01:00
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
unsigned int defaultDevId = 0;
|
|
|
|
|
|
|
|
std::vector<RtAudio::DeviceInfo> devList;
|
|
|
|
std::vector<unsigned int> deviceIds;
|
2020-11-22 18:26:48 +01:00
|
|
|
std::string txtDevList;
|
2020-11-12 00:53:38 +01:00
|
|
|
|
2021-02-23 00:26:35 +01:00
|
|
|
std::vector<unsigned int> sampleRates;
|
|
|
|
std::string sampleRatesTxt;
|
|
|
|
unsigned int sampleRate = 48000;
|
|
|
|
|
|
|
|
RtAudio audio;
|
|
|
|
|
2020-11-12 00:53:38 +01:00
|
|
|
};
|
|
|
|
|
2020-12-08 04:36:37 +01:00
|
|
|
class AudioSinkModule : public ModuleManager::Instance {
|
2020-11-12 00:53:38 +01:00
|
|
|
public:
|
|
|
|
AudioSinkModule(std::string name) {
|
|
|
|
this->name = name;
|
|
|
|
provider.create = create_sink;
|
|
|
|
provider.ctx = this;
|
2020-12-26 00:42:15 +01:00
|
|
|
|
2020-11-12 00:53:38 +01:00
|
|
|
sigpath::sinkManager.registerSinkProvider("Audio", provider);
|
|
|
|
}
|
|
|
|
|
|
|
|
~AudioSinkModule() {
|
2021-07-26 15:54:33 +02:00
|
|
|
// Unregister sink, this will automatically stop and delete all instances of the audio sink
|
|
|
|
sigpath::sinkManager.unregisterSinkProvider("Audio");
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
|
|
|
|
2021-07-26 03:11:51 +02:00
|
|
|
void postInit() {}
|
|
|
|
|
2020-12-08 04:36:37 +01:00
|
|
|
void enable() {
|
|
|
|
enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable() {
|
|
|
|
enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEnabled() {
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
2020-11-12 00:53:38 +01:00
|
|
|
private:
|
2020-11-29 20:55:00 +01:00
|
|
|
static SinkManager::Sink* create_sink(SinkManager::Stream* stream, std::string streamName, void* ctx) {
|
|
|
|
return (SinkManager::Sink*)(new AudioSink(stream, streamName));
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string name;
|
2020-12-08 04:36:37 +01:00
|
|
|
bool enabled = true;
|
2020-11-12 00:53:38 +01:00
|
|
|
SinkManager::SinkProvider provider;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
MOD_EXPORT void _INIT_() {
|
2021-02-23 00:26:35 +01:00
|
|
|
json def = json({});
|
|
|
|
config.setPath(options::opts.root + "/audio_sink_config.json");
|
|
|
|
config.load(def);
|
|
|
|
config.enableAutoSave();
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void* _CREATE_INSTANCE_(std::string name) {
|
|
|
|
AudioSinkModule* instance = new AudioSinkModule(name);
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
2021-04-27 16:48:31 +02:00
|
|
|
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
|
|
|
|
delete (AudioSinkModule*)instance;
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 04:36:37 +01:00
|
|
|
MOD_EXPORT void _END_() {
|
2021-04-27 16:48:31 +02:00
|
|
|
config.disableAutoSave();
|
|
|
|
config.save();
|
2020-11-12 00:53:38 +01:00
|
|
|
}
|