Push before potential f*ck up

This commit is contained in:
Ryzerth 2020-11-29 20:55:00 +01:00
parent afadb71d64
commit 19e516f206
6 changed files with 145 additions and 16 deletions

View File

@ -8,7 +8,7 @@ add_subdirectory("recorder")
add_subdirectory("soapy") add_subdirectory("soapy")
add_subdirectory("file_source") add_subdirectory("file_source")
add_subdirectory("rtl_tcp_source") add_subdirectory("rtl_tcp_source")
# add_subdirectory("audio_sink") add_subdirectory("audio_sink")
add_subdirectory("rx888_source") add_subdirectory("rx888_source")
add_subdirectory("plutosdr_source") add_subdirectory("plutosdr_source")
add_subdirectory("demo") add_subdirectory("demo")

View File

@ -14,12 +14,14 @@ public:
std::string name; std::string name;
int index; int index;
int channels; int channels;
int srId;
std::vector<double> sampleRates; std::vector<double> sampleRates;
std::string txtSampleRates; std::string txtSampleRates;
}; };
AudioSink(SinkManager::Stream* stream) { AudioSink(SinkManager::Stream* stream, std::string streamName) {
_stream = stream; _stream = stream;
_streamName = streamName;
audioStream = _stream->bindStream(); audioStream = _stream->bindStream();
s2m.init(audioStream); s2m.init(audioStream);
monoRB.init(&s2m.out); monoRB.init(&s2m.out);
@ -65,11 +67,14 @@ public:
devListId = devices.size(); devListId = devices.size();
defaultDev = devListId; defaultDev = devListId;
} }
dev.srId = 0;
devices.push_back(dev); devices.push_back(dev);
deviceNames.push_back(deviceInfo->name); deviceNames.push_back(deviceInfo->name);
txtDevList += deviceInfo->name; txtDevList += deviceInfo->name;
txtDevList += '\0'; txtDevList += '\0';
} }
// Load config from file
} }
~AudioSink() { ~AudioSink() {
@ -77,31 +82,118 @@ public:
} }
void start() { void start() {
if (running) {
return;
}
doStart();
running = true;
} }
void stop() { void stop() {
if (!running) {
return;
}
doStop();
running = false;
} }
void menuHandler() { void menuHandler() {
if (ImGui::Combo(("##_audio_sink_dev_"+_streamName).c_str(), &devListId, txtDevList.c_str())) {
// TODO: Load SR from config
if (running) {
doStop();
doStart();
}
// TODO: Save to config
}
AudioDevice_t dev = devices[devListId];
if (ImGui::Combo(("##_audio_sink_sr_"+_streamName).c_str(), &dev.srId, txtDevList.c_str())) {
if (running) {
doStop();
doStart();
}
// TODO: Save to config
}
} }
private: private:
void doStart() {
const PaDeviceInfo *deviceInfo;
AudioDevice_t dev = devices[devListId];
PaStreamParameters outputParams;
deviceInfo = Pa_GetDeviceInfo(dev.index);
outputParams.channelCount = 2;
outputParams.sampleFormat = paFloat32;
outputParams.hostApiSpecificStreamInfo = NULL;
outputParams.device = dev.index;
outputParams.suggestedLatency = Pa_GetDeviceInfo(outputParams.device)->defaultLowOutputLatency;
PaError err;
float sampleRate = dev.sampleRates[dev.srId];
int bufferSize = sampleRate / 60.0f;
if (dev.channels == 2) {
stereoRB.data.setMaxLatency(bufferSize * 2);
err = Pa_OpenStream(&stream, NULL, &outputParams, sampleRate, bufferSize, 0, _stereo_cb, this);
}
else {
monoRB.data.setMaxLatency(bufferSize * 2);
err = Pa_OpenStream(&stream, NULL, &outputParams, sampleRate, bufferSize, 0, _mono_cb, this);
}
if (err != 0) {
spdlog::error("Error while opening audio stream: ({0}) => {1}", err, Pa_GetErrorText(err));
return;
}
err = Pa_StartStream(stream);
if (err != 0) {
spdlog::error("Error while starting audio stream: ({0}) => {1}", err, Pa_GetErrorText(err));
return;
}
spdlog::info("Audio device open.");
running = true;
}
void doStop() {
monoRB.data.stopReader();
stereoRB.data.stopReader();
Pa_StopStream(stream);
Pa_CloseStream(stream);
monoRB.data.clearReadStop();
stereoRB.data.clearReadStop();
}
static int _mono_cb(const void *input, void *output, unsigned long frameCount,
const PaStreamCallbackTimeInfo* timeInfo, PaStreamCallbackFlags statusFlags, void *userData) {
AudioSink* _this = (AudioSink*)userData;
_this->monoRB.data.read((float*)output, frameCount);
return 0;
}
static int _stereo_cb(const void *input, void *output, unsigned long frameCount,
const PaStreamCallbackTimeInfo* timeInfo, PaStreamCallbackFlags statusFlags, void *userData) {
AudioSink* _this = (AudioSink*)userData;
_this->stereoRB.data.read((dsp::stereo_t*)output, frameCount);
return 0;
}
SinkManager::Stream* _stream; SinkManager::Stream* _stream;
dsp::stream<dsp::stereo_t>* audioStream; dsp::stream<dsp::stereo_t>* audioStream;
dsp::StereoToMono s2m; dsp::StereoToMono s2m;
dsp::RingBufferSink<float> monoRB; dsp::RingBufferSink<float> monoRB;
dsp::RingBufferSink<dsp::stereo_t> stereoRB; dsp::RingBufferSink<dsp::stereo_t> stereoRB;
std::string _streamName;
PaStream *stream;
int srId = 0; int srId = 0;
float sampleRate;
int devCount; int devCount;
int devId = 0; int devId = 0;
int devListId = 0; int devListId = 0;
int defaultDev = 0; int defaultDev = 0;
bool running = false;
const double POSSIBLE_SAMP_RATE[6] = { const double POSSIBLE_SAMP_RATE[6] = {
48000.0f, 48000.0f,
@ -132,8 +224,8 @@ public:
} }
private: private:
static SinkManager::Sink* create_sink(SinkManager::Stream* stream, void* ctx) { static SinkManager::Sink* create_sink(SinkManager::Stream* stream, std::string streamName, void* ctx) {
return (SinkManager::Sink*)(new AudioSink(stream)); return (SinkManager::Sink*)(new AudioSink(stream, streamName));
} }
std::string name; std::string name;
@ -143,6 +235,7 @@ private:
MOD_EXPORT void _INIT_() { MOD_EXPORT void _INIT_() {
// Nothing here // Nothing here
// TODO: Do instancing here (in source modules as well) to prevent multiple loads
} }
MOD_EXPORT void* _CREATE_INSTANCE_(std::string name) { MOD_EXPORT void* _CREATE_INSTANCE_(std::string name) {

View File

@ -26,6 +26,7 @@
#include <gui/menus/source.h> #include <gui/menus/source.h>
#include <gui/menus/display.h> #include <gui/menus/display.h>
#include <gui/menus/bandplan.h> #include <gui/menus/bandplan.h>
#include <gui/menus/sink.h>
#include <gui/menus/audio.h> #include <gui/menus/audio.h>
#include <gui/menus/scripting.h> #include <gui/menus/scripting.h>
#include <gui/dialogs/credits.h> #include <gui/dialogs/credits.h>
@ -95,6 +96,7 @@ void windowInit() {
core::configManager.release(); core::configManager.release();
gui::menu.registerEntry("Source", sourecmenu::draw, NULL); gui::menu.registerEntry("Source", sourecmenu::draw, NULL);
gui::menu.registerEntry("Sinks", sinkmenu::draw, NULL);
gui::menu.registerEntry("Audio", audiomenu::draw, NULL); gui::menu.registerEntry("Audio", audiomenu::draw, NULL);
gui::menu.registerEntry("Scripting", scriptingmenu::draw, NULL); gui::menu.registerEntry("Scripting", scriptingmenu::draw, NULL);
gui::menu.registerEntry("Band Plan", bandplanmenu::draw, NULL); gui::menu.registerEntry("Band Plan", bandplanmenu::draw, NULL);
@ -117,6 +119,7 @@ void windowInit() {
mod::loadFromList(ROOT_DIR "/module_list.json"); mod::loadFromList(ROOT_DIR "/module_list.json");
sourecmenu::init(); sourecmenu::init();
sinkmenu::init();
audiomenu::init(); audiomenu::init();
scriptingmenu::init(); scriptingmenu::init();
bandplanmenu::init(); bandplanmenu::init();

View File

@ -3,8 +3,12 @@
#include <imgui/imgui.h> #include <imgui/imgui.h>
#include <core.h> #include <core.h>
SinkManager::SinkManager() { #define CONCAT(a, b) ((std::string(a) + b).c_str())
SinkManager::SinkManager() {
SinkManager::SinkProvider prov;
prov.create = SinkManager::NullSink::create;
registerSinkProvider("None", prov);
} }
SinkManager::Stream::Stream(dsp::stream<dsp::stereo_t>* in, const Event<float>::EventHandler& srChangeHandler, float sampleRate) { SinkManager::Stream::Stream(dsp::stream<dsp::stereo_t>* in, const Event<float>::EventHandler& srChangeHandler, float sampleRate) {
@ -15,11 +19,19 @@ SinkManager::Stream::Stream(dsp::stream<dsp::stereo_t>* in, const Event<float>::
} }
void SinkManager::Stream::start() { void SinkManager::Stream::start() {
if (running) {
return;
}
sink->start(); sink->start();
running = true;
} }
void SinkManager::Stream::stop() { void SinkManager::Stream::stop() {
if (!running) {
return;
}
sink->stop(); sink->stop();
running = false;
} }
void SinkManager::Stream::setInput(dsp::stream<dsp::stereo_t>* in) { void SinkManager::Stream::setInput(dsp::stream<dsp::stereo_t>* in) {
@ -72,7 +84,7 @@ void SinkManager::registerStream(std::string name, SinkManager::Stream* stream)
provider = providers[providerName]; provider = providers[providerName];
} }
stream->sink = provider.create(stream, provider.ctx); stream->sink = provider.create(stream, name, provider.ctx);
} }
void SinkManager::unregisterStream(std::string name) { void SinkManager::unregisterStream(std::string name) {
@ -136,8 +148,16 @@ void SinkManager::showMenu() {
ImGui::SetCursorPosX((menuWidth / 2.0f) - (ImGui::CalcTextSize(name.c_str()).x / 2.0f)); ImGui::SetCursorPosX((menuWidth / 2.0f) - (ImGui::CalcTextSize(name.c_str()).x / 2.0f));
ImGui::Text("%s", name.c_str()); ImGui::Text("%s", name.c_str());
if (ImGui::Combo("", &stream->providerId, provStr.c_str())) { if (ImGui::Combo(CONCAT("##_sdrpp_sink_select_", name), &stream->providerId, provStr.c_str())) {
if (stream->running) {
stream->sink->stop();
}
delete stream->sink;
SinkManager::SinkProvider prov = providers[providerNames[stream->providerId]];
stream->sink = prov.create(stream, name, prov.ctx);
if (stream->running) {
stream->sink->start();
}
} }
stream->sink->menuHandler(); stream->sink->menuHandler();

View File

@ -4,6 +4,7 @@
#include <dsp/stream.h> #include <dsp/stream.h>
#include <dsp/types.h> #include <dsp/types.h>
#include <dsp/routing.h> #include <dsp/routing.h>
#include <dsp/sink.h>
#include <mutex> #include <mutex>
#include <event.h> #include <event.h>
#include <vector> #include <vector>
@ -35,24 +36,35 @@ public:
friend SinkManager::Sink; friend SinkManager::Sink;
Event<float> srChange; Event<float> srChange;
SinkManager::Sink* sink;
int providerId = 0;
private: private:
void setSampleRate(float sampleRate); void setSampleRate(float sampleRate);
dsp::stream<dsp::stereo_t>* _in; dsp::stream<dsp::stereo_t>* _in;
dsp::Splitter<dsp::stereo_t> splitter; dsp::Splitter<dsp::stereo_t> splitter;
SinkManager::Sink* sink;
std::mutex ctrlMtx; std::mutex ctrlMtx;
float _sampleRate; float _sampleRate;
int providerId = 0;
bool running = false;
}; };
struct SinkProvider { struct SinkProvider {
SinkManager::Sink* (*create)(SinkManager::Stream* stream, void* ctx); SinkManager::Sink* (*create)(SinkManager::Stream* stream, std::string streamName, void* ctx);
void* ctx; void* ctx;
}; };
class NullSink : SinkManager::Sink {
public:
void start() {}
void stop() {}
void menuHandler() {}
static SinkManager::Sink* create(SinkManager::Stream* stream, std::string streamName, void* ctx) {
return new SinkManager::NullSink;
}
};
void registerSinkProvider(std::string name, SinkProvider provider); void registerSinkProvider(std::string name, SinkProvider provider);
void registerStream(std::string name, Stream* stream); void registerStream(std::string name, Stream* stream);

View File

@ -27,6 +27,7 @@
"Source", "Source",
"Radio", "Radio",
"Recorder", "Recorder",
"Sinks",
"Audio", "Audio",
"Scripting", "Scripting",
"Band Plan", "Band Plan",