2021-07-30 21:56:34 +02:00
|
|
|
#include <utils/networking.h>
|
|
|
|
#include <imgui.h>
|
|
|
|
#include <module.h>
|
|
|
|
#include <gui/gui.h>
|
|
|
|
#include <signal_path/signal_path.h>
|
|
|
|
#include <signal_path/sink.h>
|
2022-06-17 17:34:23 +02:00
|
|
|
#include <dsp/buffer/packer.h>
|
|
|
|
#include <dsp/convert/stereo_to_mono.h>
|
|
|
|
#include <dsp/sink/handler_sink.h>
|
2023-02-25 18:12:34 +01:00
|
|
|
#include <utils/flog.h>
|
2021-07-30 21:56:34 +02:00
|
|
|
#include <config.h>
|
|
|
|
#include <gui/style.h>
|
2022-02-24 21:24:46 +01:00
|
|
|
#include <core.h>
|
2021-07-30 21:56:34 +02:00
|
|
|
|
|
|
|
#define CONCAT(a, b) ((std::string(a) + b).c_str())
|
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
SDRPP_MOD_INFO{
|
2021-07-30 21:56:34 +02:00
|
|
|
/* Name: */ "network_sink",
|
|
|
|
/* Description: */ "Network sink module for SDR++",
|
|
|
|
/* Author: */ "Ryzerth",
|
|
|
|
/* Version: */ 0, 1, 0,
|
|
|
|
/* Max instances */ 1
|
|
|
|
};
|
|
|
|
|
|
|
|
ConfigManager config;
|
|
|
|
|
|
|
|
enum {
|
|
|
|
SINK_MODE_TCP,
|
|
|
|
SINK_MODE_UDP
|
|
|
|
};
|
|
|
|
|
|
|
|
const char* sinkModesTxt = "TCP\0UDP\0";
|
|
|
|
|
|
|
|
class NetworkSink : SinkManager::Sink {
|
|
|
|
public:
|
|
|
|
NetworkSink(SinkManager::Stream* stream, std::string streamName) {
|
|
|
|
_stream = stream;
|
|
|
|
_streamName = streamName;
|
|
|
|
|
|
|
|
// Load config
|
|
|
|
config.acquire();
|
|
|
|
if (!config.conf.contains(_streamName)) {
|
|
|
|
config.conf[_streamName]["hostname"] = "localhost";
|
|
|
|
config.conf[_streamName]["port"] = 7355;
|
|
|
|
config.conf[_streamName]["protocol"] = SINK_MODE_UDP; // UDP
|
|
|
|
config.conf[_streamName]["sampleRate"] = 48000.0;
|
|
|
|
config.conf[_streamName]["stereo"] = false;
|
|
|
|
config.conf[_streamName]["listening"] = false;
|
|
|
|
}
|
|
|
|
std::string host = config.conf[_streamName]["hostname"];
|
|
|
|
strcpy(hostname, host.c_str());
|
|
|
|
port = config.conf[_streamName]["port"];
|
|
|
|
modeId = config.conf[_streamName]["protocol"];
|
|
|
|
sampleRate = config.conf[_streamName]["sampleRate"];
|
|
|
|
stereo = config.conf[_streamName]["stereo"];
|
|
|
|
bool startNow = config.conf[_streamName]["listening"];
|
|
|
|
config.release(true);
|
|
|
|
|
|
|
|
netBuf = new int16_t[STREAM_BUFFER_SIZE];
|
|
|
|
|
|
|
|
packer.init(_stream->sinkOut, 512);
|
|
|
|
s2m.init(&packer.out);
|
|
|
|
monoSink.init(&s2m.out, monoHandler, this);
|
|
|
|
stereoSink.init(&packer.out, stereoHandler, this);
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-30 21:56:34 +02:00
|
|
|
|
|
|
|
// Create a list of sample rates
|
|
|
|
for (int sr = 12000; sr < 200000; sr += 12000) {
|
|
|
|
sampleRates.push_back(sr);
|
|
|
|
}
|
|
|
|
for (int sr = 11025; sr < 192000; sr += 11025) {
|
|
|
|
sampleRates.push_back(sr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort sample rate list
|
|
|
|
std::sort(sampleRates.begin(), sampleRates.end(), [](double a, double b) { return (a < b); });
|
|
|
|
|
|
|
|
// Generate text list for UI
|
|
|
|
char buffer[128];
|
|
|
|
int id = 0;
|
|
|
|
int _48kId;
|
|
|
|
bool found = false;
|
|
|
|
for (auto sr : sampleRates) {
|
|
|
|
sprintf(buffer, "%d", (int)sr);
|
|
|
|
sampleRatesTxt += buffer;
|
|
|
|
sampleRatesTxt += '\0';
|
2021-12-19 22:11:44 +01:00
|
|
|
if (sr == sampleRate) {
|
|
|
|
srId = id;
|
|
|
|
found = true;
|
|
|
|
}
|
2021-07-30 21:56:34 +02:00
|
|
|
if (sr == 48000.0) { _48kId = id; }
|
|
|
|
id++;
|
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
if (!found) {
|
|
|
|
srId = _48kId;
|
|
|
|
sampleRate = 48000.0;
|
|
|
|
}
|
2021-07-30 21:56:34 +02:00
|
|
|
_stream->setSampleRate(sampleRate);
|
|
|
|
|
|
|
|
// Start if needed
|
|
|
|
if (startNow) { startServer(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
~NetworkSink() {
|
|
|
|
stopServer();
|
|
|
|
delete[] netBuf;
|
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doStart();
|
|
|
|
running = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void stop() {
|
|
|
|
if (!running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doStop();
|
|
|
|
running = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void menuHandler() {
|
2022-02-14 23:33:52 +01:00
|
|
|
float menuWidth = ImGui::GetContentRegionAvail().x;
|
2021-07-30 21:56:34 +02:00
|
|
|
|
|
|
|
bool listening = (listener && listener->isListening()) || (conn && conn->isOpen());
|
|
|
|
|
|
|
|
if (listening) { style::beginDisabled(); }
|
|
|
|
if (ImGui::InputText(CONCAT("##_network_sink_host_", _streamName), hostname, 1023)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["hostname"] = hostname;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
ImGui::SameLine();
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputInt(CONCAT("##_network_sink_port_", _streamName), &port, 0, 0)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["port"] = port;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
|
2021-08-31 18:39:48 +02:00
|
|
|
ImGui::LeftLabel("Protocol");
|
2021-07-30 21:56:34 +02:00
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::Combo(CONCAT("##_network_sink_mode_", _streamName), &modeId, sinkModesTxt)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["protocol"] = modeId;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listening) { style::endDisabled(); }
|
|
|
|
|
2021-08-31 18:39:48 +02:00
|
|
|
ImGui::LeftLabel("Samplerate");
|
2021-07-30 21:56:34 +02:00
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::Combo(CONCAT("##_network_sink_sr_", _streamName), &srId, sampleRatesTxt.c_str())) {
|
|
|
|
sampleRate = sampleRates[srId];
|
|
|
|
_stream->setSampleRate(sampleRate);
|
|
|
|
packer.setSampleCount(sampleRate / 60);
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["sampleRate"] = sampleRate;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::Checkbox(CONCAT("Stereo##_network_sink_stereo_", _streamName), &stereo)) {
|
|
|
|
stop();
|
|
|
|
start();
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["stereo"] = stereo;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listening && ImGui::Button(CONCAT("Stop##_network_sink_stop_", _streamName), ImVec2(menuWidth, 0))) {
|
|
|
|
stopServer();
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["listening"] = false;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
else if (!listening && ImGui::Button(CONCAT("Start##_network_sink_stop_", _streamName), ImVec2(menuWidth, 0))) {
|
|
|
|
startServer();
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_streamName]["listening"] = true;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
|
2022-01-26 14:50:16 +01:00
|
|
|
ImGui::TextUnformatted("Status:");
|
2021-07-30 21:56:34 +02:00
|
|
|
ImGui::SameLine();
|
|
|
|
if (conn && conn->isOpen()) {
|
|
|
|
ImGui::TextColored(ImVec4(0.0, 1.0, 0.0, 1.0), (modeId == SINK_MODE_TCP) ? "Connected" : "Sending");
|
|
|
|
}
|
|
|
|
else if (listening) {
|
|
|
|
ImGui::TextColored(ImVec4(1.0, 1.0, 0.0, 1.0), "Listening");
|
|
|
|
}
|
|
|
|
else {
|
2022-01-26 14:50:16 +01:00
|
|
|
ImGui::TextUnformatted("Idle");
|
2021-07-30 21:56:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void doStart() {
|
|
|
|
packer.start();
|
|
|
|
if (stereo) {
|
|
|
|
stereoSink.start();
|
|
|
|
}
|
|
|
|
else {
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::warn("Starting");
|
2021-07-30 21:56:34 +02:00
|
|
|
s2m.start();
|
|
|
|
monoSink.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void doStop() {
|
|
|
|
packer.stop();
|
|
|
|
s2m.stop();
|
|
|
|
monoSink.stop();
|
|
|
|
stereoSink.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
void startServer() {
|
|
|
|
if (modeId == SINK_MODE_TCP) {
|
|
|
|
listener = net::listen(hostname, port);
|
|
|
|
if (listener) {
|
|
|
|
listener->acceptAsync(clientHandler, this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
conn = net::openUDP("0.0.0.0", port, hostname, port, false);
|
|
|
|
}
|
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-30 21:56:34 +02:00
|
|
|
void stopServer() {
|
|
|
|
if (conn) { conn->close(); }
|
|
|
|
if (listener) { listener->close(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
static void monoHandler(float* samples, int count, void* ctx) {
|
|
|
|
NetworkSink* _this = (NetworkSink*)ctx;
|
|
|
|
std::lock_guard lck(_this->connMtx);
|
|
|
|
if (!_this->conn || !_this->conn->isOpen()) { return; }
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-30 21:56:34 +02:00
|
|
|
volk_32f_s32f_convert_16i(_this->netBuf, (float*)samples, 32768.0f, count);
|
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
_this->conn->write(count * sizeof(int16_t), (uint8_t*)_this->netBuf);
|
2021-07-30 21:56:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void stereoHandler(dsp::stereo_t* samples, int count, void* ctx) {
|
|
|
|
NetworkSink* _this = (NetworkSink*)ctx;
|
|
|
|
std::lock_guard lck(_this->connMtx);
|
|
|
|
if (!_this->conn || !_this->conn->isOpen()) { return; }
|
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
volk_32f_s32f_convert_16i(_this->netBuf, (float*)samples, 32768.0f, count * 2);
|
2021-07-30 21:56:34 +02:00
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
_this->conn->write(count * 2 * sizeof(int16_t), (uint8_t*)_this->netBuf);
|
2021-07-30 21:56:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void clientHandler(net::Conn client, void* ctx) {
|
|
|
|
NetworkSink* _this = (NetworkSink*)ctx;
|
|
|
|
|
|
|
|
{
|
|
|
|
std::lock_guard lck(_this->connMtx);
|
|
|
|
_this->conn = std::move(client);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_this->conn) {
|
|
|
|
_this->conn->waitForEnd();
|
|
|
|
_this->conn->close();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
}
|
|
|
|
|
|
|
|
_this->listener->acceptAsync(clientHandler, _this);
|
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-30 21:56:34 +02:00
|
|
|
SinkManager::Stream* _stream;
|
2022-06-17 17:34:23 +02:00
|
|
|
dsp::buffer::Packer<dsp::stereo_t> packer;
|
|
|
|
dsp::convert::StereoToMono s2m;
|
|
|
|
dsp::sink::Handler<float> monoSink;
|
|
|
|
dsp::sink::Handler<dsp::stereo_t> stereoSink;
|
2021-07-30 21:56:34 +02:00
|
|
|
|
|
|
|
std::string _streamName;
|
|
|
|
|
|
|
|
int srId = 0;
|
|
|
|
bool running = false;
|
|
|
|
|
|
|
|
char hostname[1024];
|
|
|
|
int port = 4242;
|
|
|
|
|
|
|
|
int modeId = 1;
|
|
|
|
|
|
|
|
std::vector<unsigned int> sampleRates;
|
|
|
|
std::string sampleRatesTxt;
|
|
|
|
unsigned int sampleRate = 48000;
|
|
|
|
bool stereo = false;
|
|
|
|
|
|
|
|
int16_t* netBuf;
|
|
|
|
|
|
|
|
net::Listener listener;
|
|
|
|
net::Conn conn;
|
|
|
|
std::mutex connMtx;
|
|
|
|
};
|
|
|
|
|
|
|
|
class NetworkSinkModule : public ModuleManager::Instance {
|
|
|
|
public:
|
|
|
|
NetworkSinkModule(std::string name) {
|
|
|
|
this->name = name;
|
|
|
|
provider.create = create_sink;
|
|
|
|
provider.ctx = this;
|
|
|
|
|
|
|
|
sigpath::sinkManager.registerSinkProvider("Network", provider);
|
|
|
|
}
|
|
|
|
|
|
|
|
~NetworkSinkModule() {
|
|
|
|
// Unregister sink, this will automatically stop and delete all instances of the audio sink
|
|
|
|
sigpath::sinkManager.unregisterSinkProvider("Network");
|
|
|
|
}
|
|
|
|
|
|
|
|
void postInit() {}
|
|
|
|
|
|
|
|
void enable() {
|
|
|
|
enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable() {
|
|
|
|
enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEnabled() {
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static SinkManager::Sink* create_sink(SinkManager::Stream* stream, std::string streamName, void* ctx) {
|
|
|
|
return (SinkManager::Sink*)(new NetworkSink(stream, streamName));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string name;
|
|
|
|
bool enabled = true;
|
|
|
|
SinkManager::SinkProvider provider;
|
|
|
|
};
|
|
|
|
|
|
|
|
MOD_EXPORT void _INIT_() {
|
|
|
|
json def = json({});
|
2022-02-24 20:49:53 +01:00
|
|
|
config.setPath(core::args["root"].s() + "/network_sink_config.json");
|
2021-07-30 21:56:34 +02:00
|
|
|
config.load(def);
|
|
|
|
config.enableAutoSave();
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void* _CREATE_INSTANCE_(std::string name) {
|
|
|
|
NetworkSinkModule* instance = new NetworkSinkModule(name);
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
|
|
|
|
delete (NetworkSinkModule*)instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _END_() {
|
|
|
|
config.disableAutoSave();
|
|
|
|
config.save();
|
|
|
|
}
|