2021-04-01 16:54:16 +02:00
|
|
|
#include <imgui.h>
|
|
|
|
#include <config.h>
|
|
|
|
#include <core.h>
|
|
|
|
#include <gui/style.h>
|
2021-06-20 21:17:11 +02:00
|
|
|
#include <gui/gui.h>
|
2021-04-01 16:54:16 +02:00
|
|
|
#include <signal_path/signal_path.h>
|
|
|
|
#include <module.h>
|
2021-04-22 19:18:19 +02:00
|
|
|
#include <filesystem>
|
2022-07-16 00:08:25 +02:00
|
|
|
#include "meteor_demod.h"
|
2022-07-02 16:53:09 +02:00
|
|
|
#include <dsp/routing/splitter.h>
|
|
|
|
#include <dsp/buffer/reshaper.h>
|
|
|
|
#include <dsp/sink/handler_sink.h>
|
2021-07-18 04:30:55 +02:00
|
|
|
#include <meteor_demodulator_interface.h>
|
2021-04-01 16:54:16 +02:00
|
|
|
#include <gui/widgets/folder_select.h>
|
|
|
|
#include <gui/widgets/constellation_diagram.h>
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
#define CONCAT(a, b) ((std::string(a) + b).c_str())
|
2021-04-01 16:54:16 +02:00
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
SDRPP_MOD_INFO{
|
2021-04-01 16:54:16 +02:00
|
|
|
/* Name: */ "meteor_demodulator",
|
|
|
|
/* Description: */ "Meteor demodulator for SDR++",
|
|
|
|
/* Author: */ "Ryzerth",
|
|
|
|
/* Version: */ 0, 1, 0,
|
|
|
|
/* Max instances */ -1
|
|
|
|
};
|
|
|
|
|
2021-04-22 19:18:19 +02:00
|
|
|
ConfigManager config;
|
|
|
|
|
2021-04-01 16:54:16 +02:00
|
|
|
std::string genFileName(std::string prefix, std::string suffix) {
|
|
|
|
time_t now = time(0);
|
2021-12-19 22:11:44 +01:00
|
|
|
tm* ltm = localtime(&now);
|
2021-04-01 16:54:16 +02:00
|
|
|
char buf[1024];
|
|
|
|
sprintf(buf, "%s_%02d-%02d-%02d_%02d-%02d-%02d%s", prefix.c_str(), ltm->tm_hour, ltm->tm_min, ltm->tm_sec, ltm->tm_mday, ltm->tm_mon + 1, ltm->tm_year + 1900, suffix.c_str());
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define INPUT_SAMPLE_RATE 150000
|
|
|
|
|
|
|
|
class MeteorDemodulatorModule : public ModuleManager::Instance {
|
|
|
|
public:
|
|
|
|
MeteorDemodulatorModule(std::string name) : folderSelect("%ROOT%/recordings") {
|
|
|
|
this->name = name;
|
|
|
|
|
|
|
|
writeBuffer = new int8_t[STREAM_BUFFER_SIZE];
|
|
|
|
|
2021-04-22 19:18:19 +02:00
|
|
|
// Load config
|
2021-07-09 14:24:07 -04:00
|
|
|
config.acquire();
|
2022-07-16 00:08:25 +02:00
|
|
|
// Note: this first one may not be needed but I'm paranoid
|
2021-04-22 19:18:19 +02:00
|
|
|
if (!config.conf.contains(name)) {
|
2022-07-16 00:08:25 +02:00
|
|
|
config.conf[name] = json({});
|
2021-04-22 19:18:19 +02:00
|
|
|
}
|
2022-07-16 00:08:25 +02:00
|
|
|
if (config.conf[name].contains("recPath")) {
|
|
|
|
folderSelect.setPath(config.conf[name]["recPath"]);
|
|
|
|
}
|
|
|
|
if (config.conf[name].contains("brokenModulation")) {
|
|
|
|
brokenModulation = config.conf[name]["brokenModulation"];
|
|
|
|
}
|
|
|
|
config.release();
|
2021-04-22 19:18:19 +02:00
|
|
|
|
2021-04-17 22:37:50 +02:00
|
|
|
vfo = sigpath::vfoManager.createVFO(name, ImGui::WaterfallVFO::REF_CENTER, 0, 150000, INPUT_SAMPLE_RATE, 150000, 150000, true);
|
2022-07-16 00:08:25 +02:00
|
|
|
demod.init(vfo->output, 72000.0f, INPUT_SAMPLE_RATE, 33, 0.6f, 0.1f, 0.005f, brokenModulation, 1e-6, 0.01);
|
2022-07-02 16:53:09 +02:00
|
|
|
split.init(&demod.out);
|
2021-04-01 16:54:16 +02:00
|
|
|
split.bindStream(&symSinkStream);
|
|
|
|
split.bindStream(&sinkStream);
|
|
|
|
reshape.init(&symSinkStream, 1024, (72000 / 30) - 1024);
|
|
|
|
symSink.init(&reshape.out, symSinkHandler, this);
|
|
|
|
sink.init(&sinkStream, sinkHandler, this);
|
|
|
|
|
|
|
|
demod.start();
|
|
|
|
split.start();
|
|
|
|
reshape.start();
|
|
|
|
symSink.start();
|
|
|
|
sink.start();
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-04-01 16:54:16 +02:00
|
|
|
gui::menu.registerEntry(name, menuHandler, this, this);
|
2021-07-18 04:30:55 +02:00
|
|
|
core::modComManager.registerInterface("meteor_demodulator", name, moduleInterfaceHandler, this);
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
~MeteorDemodulatorModule() {
|
2021-05-05 04:31:37 +02:00
|
|
|
if (recording) {
|
|
|
|
std::lock_guard<std::mutex> lck(recMtx);
|
|
|
|
recording = false;
|
|
|
|
recFile.close();
|
|
|
|
}
|
|
|
|
demod.stop();
|
|
|
|
split.stop();
|
|
|
|
reshape.stop();
|
|
|
|
symSink.stop();
|
|
|
|
sink.stop();
|
|
|
|
sigpath::vfoManager.deleteVFO(vfo);
|
|
|
|
gui::menu.removeEntry(name);
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
|
|
|
|
2021-07-26 03:11:51 +02:00
|
|
|
void postInit() {}
|
|
|
|
|
2021-04-01 16:54:16 +02:00
|
|
|
void enable() {
|
2021-04-22 05:58:20 +02:00
|
|
|
double bw = gui::waterfall.getBandwidth();
|
2021-12-19 22:11:44 +01:00
|
|
|
vfo = sigpath::vfoManager.createVFO(name, ImGui::WaterfallVFO::REF_CENTER, std::clamp<double>(0, -bw / 2.0, bw / 2.0), 150000, INPUT_SAMPLE_RATE, 150000, 150000, true);
|
2021-04-01 16:54:16 +02:00
|
|
|
|
|
|
|
demod.setInput(vfo->output);
|
|
|
|
|
|
|
|
demod.start();
|
|
|
|
split.start();
|
|
|
|
reshape.start();
|
|
|
|
symSink.start();
|
|
|
|
sink.start();
|
|
|
|
|
|
|
|
enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable() {
|
|
|
|
demod.stop();
|
|
|
|
split.stop();
|
|
|
|
reshape.stop();
|
|
|
|
symSink.stop();
|
|
|
|
sink.stop();
|
|
|
|
|
|
|
|
sigpath::vfoManager.deleteVFO(vfo);
|
|
|
|
enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEnabled() {
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static void menuHandler(void* ctx) {
|
|
|
|
MeteorDemodulatorModule* _this = (MeteorDemodulatorModule*)ctx;
|
|
|
|
|
2022-02-14 23:33:52 +01:00
|
|
|
float menuWidth = ImGui::GetContentRegionAvail().x;
|
2021-04-01 16:54:16 +02:00
|
|
|
|
|
|
|
if (!_this->enabled) { style::beginDisabled(); }
|
|
|
|
|
|
|
|
ImGui::SetNextItemWidth(menuWidth);
|
|
|
|
_this->constDiagram.draw();
|
|
|
|
|
2022-07-16 00:08:25 +02:00
|
|
|
if (_this->folderSelect.render("##meteor_rec" + _this->name)) {
|
2021-04-22 19:18:19 +02:00
|
|
|
if (_this->folderSelect.pathIsValid()) {
|
2021-07-09 14:24:07 -04:00
|
|
|
config.acquire();
|
2021-04-22 19:18:19 +02:00
|
|
|
config.conf[_this->name]["recPath"] = _this->folderSelect.path;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
}
|
2021-04-01 16:54:16 +02:00
|
|
|
|
2022-07-16 00:08:25 +02:00
|
|
|
if (ImGui::Checkbox(CONCAT("Broken modulation##meteor_rec", _this->name), &_this->brokenModulation)) {
|
|
|
|
_this->demod.setBrokenModulation(_this->brokenModulation);
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["brokenModulation"] = _this->brokenModulation;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
|
2021-04-01 16:54:16 +02:00
|
|
|
if (!_this->folderSelect.pathIsValid() && _this->enabled) { style::beginDisabled(); }
|
|
|
|
|
|
|
|
if (_this->recording) {
|
2022-07-16 00:08:25 +02:00
|
|
|
if (ImGui::Button(CONCAT("Stop##meteor_rec_", _this->name), ImVec2(menuWidth, 0))) {
|
2021-07-18 04:30:55 +02:00
|
|
|
_this->stopRecording();
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
|
|
|
ImGui::TextColored(ImVec4(1.0f, 0.0f, 0.0f, 1.0f), "Recording %.2fMB", (float)_this->dataWritten / 1000000.0f);
|
|
|
|
}
|
|
|
|
else {
|
2022-07-16 00:08:25 +02:00
|
|
|
if (ImGui::Button(CONCAT("Record##meteor_rec_", _this->name), ImVec2(menuWidth, 0))) {
|
2021-12-19 22:11:44 +01:00
|
|
|
_this->startRecording();
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
2022-01-26 14:50:16 +01:00
|
|
|
ImGui::TextUnformatted("Idle --.--MB");
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
if (!_this->folderSelect.pathIsValid() && _this->enabled) { style::endDisabled(); }
|
2021-04-01 16:54:16 +02:00
|
|
|
|
|
|
|
if (!_this->enabled) { style::endDisabled(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
static void symSinkHandler(dsp::complex_t* data, int count, void* ctx) {
|
|
|
|
MeteorDemodulatorModule* _this = (MeteorDemodulatorModule*)ctx;
|
|
|
|
|
2021-07-09 14:24:07 -04:00
|
|
|
dsp::complex_t* buf = _this->constDiagram.acquireBuffer();
|
2021-04-01 16:54:16 +02:00
|
|
|
memcpy(buf, data, 1024 * sizeof(dsp::complex_t));
|
|
|
|
_this->constDiagram.releaseBuffer();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sinkHandler(dsp::complex_t* data, int count, void* ctx) {
|
|
|
|
MeteorDemodulatorModule* _this = (MeteorDemodulatorModule*)ctx;
|
|
|
|
std::lock_guard<std::mutex> lck(_this->recMtx);
|
|
|
|
if (!_this->recording) { return; }
|
|
|
|
for (int i = 0; i < count; i++) {
|
2021-04-01 20:57:03 +02:00
|
|
|
_this->writeBuffer[(2 * i)] = std::clamp<int>(data[i].re * 84.0f, -127, 127);
|
|
|
|
_this->writeBuffer[(2 * i) + 1] = std::clamp<int>(data[i].im * 84.0f, -127, 127);
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
|
|
|
_this->recFile.write((char*)_this->writeBuffer, count * 2);
|
|
|
|
_this->dataWritten += count * 2;
|
|
|
|
}
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
void startRecording() {
|
|
|
|
std::lock_guard<std::mutex> lck(recMtx);
|
|
|
|
dataWritten = 0;
|
|
|
|
std::string filename = genFileName(folderSelect.expandString(folderSelect.path) + "/meteor", ".s");
|
|
|
|
recFile = std::ofstream(filename, std::ios::binary);
|
|
|
|
if (recFile.is_open()) {
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::info("Recording to '{0}'", filename);
|
2021-07-18 04:30:55 +02:00
|
|
|
recording = true;
|
|
|
|
}
|
|
|
|
else {
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::error("Could not open file for recording!");
|
2021-12-19 22:11:44 +01:00
|
|
|
}
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void stopRecording() {
|
|
|
|
std::lock_guard<std::mutex> lck(recMtx);
|
|
|
|
recording = false;
|
|
|
|
recFile.close();
|
|
|
|
dataWritten = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void moduleInterfaceHandler(int code, void* in, void* out, void* ctx) {
|
|
|
|
MeteorDemodulatorModule* _this = (MeteorDemodulatorModule*)ctx;
|
|
|
|
if (code == METEOR_DEMODULATOR_IFACE_CMD_START) {
|
|
|
|
if (!_this->recording) { _this->startRecording(); }
|
|
|
|
}
|
|
|
|
else if (code == METEOR_DEMODULATOR_IFACE_CMD_STOP) {
|
|
|
|
if (_this->recording) { _this->stopRecording(); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-01 16:54:16 +02:00
|
|
|
std::string name;
|
|
|
|
bool enabled = true;
|
|
|
|
|
|
|
|
// DSP Chain
|
|
|
|
VFOManager::VFO* vfo;
|
2022-07-16 00:08:25 +02:00
|
|
|
dsp::demod::Meteor demod;
|
2022-07-02 16:53:09 +02:00
|
|
|
dsp::routing::Splitter<dsp::complex_t> split;
|
2021-04-01 16:54:16 +02:00
|
|
|
|
|
|
|
dsp::stream<dsp::complex_t> symSinkStream;
|
|
|
|
dsp::stream<dsp::complex_t> sinkStream;
|
2022-07-02 16:53:09 +02:00
|
|
|
dsp::buffer::Reshaper<dsp::complex_t> reshape;
|
|
|
|
dsp::sink::Handler<dsp::complex_t> symSink;
|
|
|
|
dsp::sink::Handler<dsp::complex_t> sink;
|
2021-04-01 16:54:16 +02:00
|
|
|
|
|
|
|
ImGui::ConstellationDiagram constDiagram;
|
|
|
|
|
|
|
|
FolderSelect folderSelect;
|
|
|
|
|
|
|
|
std::mutex recMtx;
|
|
|
|
bool recording = false;
|
|
|
|
uint64_t dataWritten = 0;
|
|
|
|
std::ofstream recFile;
|
2022-07-16 00:08:25 +02:00
|
|
|
bool brokenModulation = false;
|
2021-04-01 16:54:16 +02:00
|
|
|
|
|
|
|
int8_t* writeBuffer;
|
|
|
|
};
|
|
|
|
|
|
|
|
MOD_EXPORT void _INIT_() {
|
2021-04-22 19:18:19 +02:00
|
|
|
// Create default recording directory
|
2022-02-24 21:01:51 +01:00
|
|
|
std::string root = (std::string)core::args["root"];
|
2022-02-24 20:49:53 +01:00
|
|
|
if (!std::filesystem::exists(root + "/recordings")) {
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::warn("Recordings directory does not exist, creating it");
|
2022-02-24 20:49:53 +01:00
|
|
|
if (!std::filesystem::create_directory(root + "/recordings")) {
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::error("Could not create recordings directory");
|
2021-04-22 19:18:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
json def = json({});
|
2022-02-24 20:49:53 +01:00
|
|
|
config.setPath(root + "/meteor_demodulator_config.json");
|
2021-04-22 19:18:19 +02:00
|
|
|
config.load(def);
|
|
|
|
config.enableAutoSave();
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_(std::string name) {
|
|
|
|
return new MeteorDemodulatorModule(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
|
|
|
|
delete (MeteorDemodulatorModule*)instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _END_() {
|
2021-04-22 19:18:19 +02:00
|
|
|
config.disableAutoSave();
|
|
|
|
config.save();
|
2021-04-01 16:54:16 +02:00
|
|
|
}
|