mirror of
https://github.com/AlexandreRouma/SDRPlusPlus.git
synced 2025-01-11 18:57:11 +01:00
Push before potential f*ck up
This commit is contained in:
parent
afadb71d64
commit
19e516f206
@ -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")
|
||||||
|
@ -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) {
|
||||||
|
@ -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();
|
||||||
|
@ -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();
|
||||||
|
@ -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);
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
"Source",
|
"Source",
|
||||||
"Radio",
|
"Radio",
|
||||||
"Recorder",
|
"Recorder",
|
||||||
|
"Sinks",
|
||||||
"Audio",
|
"Audio",
|
||||||
"Scripting",
|
"Scripting",
|
||||||
"Band Plan",
|
"Band Plan",
|
||||||
|
Loading…
Reference in New Issue
Block a user