2020-12-06 16:13:47 +01:00
|
|
|
#pragma once
|
|
|
|
#include <radio_demod.h>
|
|
|
|
#include <dsp/demodulator.h>
|
|
|
|
#include <dsp/resampling.h>
|
|
|
|
#include <dsp/filter.h>
|
|
|
|
#include <dsp/audio.h>
|
|
|
|
#include <string>
|
2020-12-09 15:16:38 +01:00
|
|
|
#include <config.h>
|
2020-12-06 16:13:47 +01:00
|
|
|
#include <imgui.h>
|
2021-02-07 23:47:17 +01:00
|
|
|
|
2020-12-06 16:13:47 +01:00
|
|
|
|
|
|
|
class LSBDemodulator : public Demodulator {
|
|
|
|
public:
|
|
|
|
LSBDemodulator() {}
|
2020-12-09 15:16:38 +01:00
|
|
|
LSBDemodulator(std::string prefix, VFOManager::VFO* vfo, float audioSampleRate, float bandWidth, ConfigManager* config) {
|
|
|
|
init(prefix, vfo, audioSampleRate, bandWidth, config);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
|
|
|
|
2020-12-09 15:16:38 +01:00
|
|
|
void init(std::string prefix, VFOManager::VFO* vfo, float audioSampleRate, float bandWidth, ConfigManager* config) {
|
2020-12-06 16:13:47 +01:00
|
|
|
uiPrefix = prefix;
|
|
|
|
_vfo = vfo;
|
|
|
|
audioSampRate = audioSampleRate;
|
|
|
|
bw = bandWidth;
|
2020-12-09 15:16:38 +01:00
|
|
|
_config = config;
|
|
|
|
|
2021-07-09 20:24:07 +02:00
|
|
|
_config->acquire();
|
2020-12-09 15:16:38 +01:00
|
|
|
if(_config->conf.contains(prefix)) {
|
|
|
|
if(!_config->conf[prefix].contains("LSB")) {
|
2021-04-22 05:38:25 +02:00
|
|
|
_config->conf[prefix]["LSB"]["bandwidth"] = bw;
|
|
|
|
_config->conf[prefix]["LSB"]["snapInterval"] = snapInterval;
|
|
|
|
_config->conf[prefix]["LSB"]["squelchLevel"] = squelchLevel;
|
2020-12-09 15:16:38 +01:00
|
|
|
}
|
|
|
|
json conf = _config->conf[prefix]["LSB"];
|
2021-04-22 05:38:25 +02:00
|
|
|
if (conf.contains("bandwidth")) { bw = conf["bandwidth"]; }
|
|
|
|
if (conf.contains("snapInterval")) { snapInterval = conf["snapInterval"]; }
|
|
|
|
if (conf.contains("squelchLevel")) { squelchLevel = conf["squelchLevel"]; }
|
2020-12-09 15:16:38 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
_config->conf[prefix]["LSB"]["bandwidth"] = bw;
|
|
|
|
_config->conf[prefix]["LSB"]["snapInterval"] = snapInterval;
|
2021-04-22 05:38:25 +02:00
|
|
|
_config->conf[prefix]["LSB"]["squelchLevel"] = squelchLevel;
|
2020-12-09 15:16:38 +01:00
|
|
|
}
|
|
|
|
_config->release(true);
|
2020-12-10 05:18:40 +01:00
|
|
|
|
|
|
|
squelch.init(_vfo->output, squelchLevel);
|
2020-12-06 16:13:47 +01:00
|
|
|
|
2021-02-15 20:16:40 +01:00
|
|
|
demod.init(&squelch.out, bbSampRate, bw, dsp::SSBDemod::MODE_LSB);
|
2020-12-06 16:13:47 +01:00
|
|
|
|
2020-12-31 14:26:12 +01:00
|
|
|
agc.init(&demod.out, 20.0f, bbSampRate);
|
2020-12-06 16:13:47 +01:00
|
|
|
|
2020-12-06 19:51:56 +01:00
|
|
|
float audioBW = std::min<float>(audioSampRate / 2.0f, bw);
|
2020-12-06 16:13:47 +01:00
|
|
|
win.init(audioBW, audioBW, bbSampRate);
|
|
|
|
resamp.init(&agc.out, &win, bbSampRate, audioSampRate);
|
|
|
|
win.setSampleRate(bbSampRate * resamp.getInterpolation());
|
|
|
|
resamp.updateWindow(&win);
|
|
|
|
|
|
|
|
m2s.init(&resamp.out);
|
2021-08-20 20:40:14 +02:00
|
|
|
|
|
|
|
onUserChangedBandwidthHandler.handler = vfoUserChangedBandwidthHandler;
|
|
|
|
onUserChangedBandwidthHandler.ctx = this;
|
|
|
|
|
|
|
|
_vfo->wtfVFO->onUserChangedBandwidth.bindHandler(&onUserChangedBandwidthHandler);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
2020-12-10 05:18:40 +01:00
|
|
|
squelch.start();
|
2020-12-06 16:13:47 +01:00
|
|
|
demod.start();
|
|
|
|
agc.start();
|
|
|
|
resamp.start();
|
|
|
|
m2s.start();
|
|
|
|
running = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void stop() {
|
2020-12-10 05:18:40 +01:00
|
|
|
squelch.stop();
|
2020-12-06 16:13:47 +01:00
|
|
|
demod.stop();
|
|
|
|
agc.stop();
|
|
|
|
resamp.stop();
|
|
|
|
m2s.stop();
|
|
|
|
running = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isRunning() {
|
|
|
|
return running;
|
|
|
|
}
|
|
|
|
|
|
|
|
void select() {
|
|
|
|
_vfo->setSampleRate(bbSampRate, bw);
|
|
|
|
_vfo->setSnapInterval(snapInterval);
|
|
|
|
_vfo->setReference(ImGui::WaterfallVFO::REF_UPPER);
|
2021-04-17 22:37:50 +02:00
|
|
|
_vfo->setBandwidthLimits(bwMin, bwMax, false);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void setVFO(VFOManager::VFO* vfo) {
|
|
|
|
_vfo = vfo;
|
2020-12-10 05:18:40 +01:00
|
|
|
squelch.setInput(_vfo->output);
|
2021-08-20 20:40:14 +02:00
|
|
|
_vfo->wtfVFO->onUserChangedBandwidth.bindHandler(&onUserChangedBandwidthHandler);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
VFOManager::VFO* getVFO() {
|
|
|
|
return _vfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setAudioSampleRate(float sampleRate) {
|
|
|
|
if (running) {
|
|
|
|
resamp.stop();
|
|
|
|
}
|
|
|
|
audioSampRate = sampleRate;
|
2020-12-06 19:51:56 +01:00
|
|
|
float audioBW = std::min<float>(audioSampRate / 2.0f, bw);
|
2020-12-06 16:13:47 +01:00
|
|
|
resamp.setOutSampleRate(audioSampRate);
|
|
|
|
win.setSampleRate(bbSampRate * resamp.getInterpolation());
|
|
|
|
win.setCutoff(audioBW);
|
|
|
|
win.setTransWidth(audioBW);
|
|
|
|
resamp.updateWindow(&win);
|
|
|
|
if (running) {
|
|
|
|
resamp.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
float getAudioSampleRate() {
|
|
|
|
return audioSampRate;
|
|
|
|
}
|
|
|
|
|
|
|
|
dsp::stream<dsp::stereo_t>* getOutput() {
|
|
|
|
return &m2s.out;
|
|
|
|
}
|
|
|
|
|
|
|
|
void showMenu() {
|
|
|
|
float menuWidth = ImGui::GetContentRegionAvailWidth();
|
|
|
|
|
|
|
|
ImGui::SetNextItemWidth(menuWidth);
|
2021-02-07 23:47:17 +01:00
|
|
|
if (ImGui::InputFloat(("##_radio_lsb_bw_" + uiPrefix).c_str(), &bw, 1, 100, "%.0f", 0)) {
|
2020-12-06 16:13:47 +01:00
|
|
|
bw = std::clamp<float>(bw, bwMin, bwMax);
|
|
|
|
setBandwidth(bw);
|
2021-07-09 20:24:07 +02:00
|
|
|
_config->acquire();
|
2020-12-09 15:16:38 +01:00
|
|
|
_config->conf[uiPrefix]["LSB"]["bandwidth"] = bw;
|
|
|
|
_config->release(true);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
|
|
|
|
2021-08-31 18:39:48 +02:00
|
|
|
ImGui::LeftLabel("Snap Interval");
|
2020-12-06 20:02:22 +01:00
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
2021-02-07 23:47:17 +01:00
|
|
|
if (ImGui::InputFloat(("##_radio_lsb_snap_" + uiPrefix).c_str(), &snapInterval, 1, 100, "%.0f", 0)) {
|
2021-02-15 20:16:40 +01:00
|
|
|
if (snapInterval < 1) { snapInterval = 1; }
|
2020-12-06 16:13:47 +01:00
|
|
|
setSnapInterval(snapInterval);
|
2021-07-09 20:24:07 +02:00
|
|
|
_config->acquire();
|
2020-12-09 15:16:38 +01:00
|
|
|
_config->conf[uiPrefix]["LSB"]["snapInterval"] = snapInterval;
|
|
|
|
_config->release(true);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
2020-12-10 05:18:40 +01:00
|
|
|
|
2021-08-31 18:39:48 +02:00
|
|
|
ImGui::LeftLabel("Squelch");
|
2020-12-10 05:18:40 +01:00
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
2021-02-07 23:47:17 +01:00
|
|
|
if (ImGui::SliderFloat(("##_radio_lsb_squelch_" + uiPrefix).c_str(), &squelchLevel, -100.0f, 0.0f, "%.3fdB")) {
|
2020-12-10 05:18:40 +01:00
|
|
|
squelch.setLevel(squelchLevel);
|
2021-07-09 20:24:07 +02:00
|
|
|
_config->acquire();
|
2020-12-10 05:18:40 +01:00
|
|
|
_config->conf[uiPrefix]["LSB"]["squelchLevel"] = squelchLevel;
|
|
|
|
_config->release(true);
|
|
|
|
}
|
2021-08-20 20:40:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void vfoUserChangedBandwidthHandler(double newBw, void* ctx) {
|
|
|
|
LSBDemodulator* _this = (LSBDemodulator*)ctx;
|
|
|
|
if (_this->running) {
|
|
|
|
_this->bw = newBw;
|
|
|
|
_this->setBandwidth(_this->bw, false);
|
|
|
|
_this->_config->acquire();
|
|
|
|
_this->_config->conf[_this->uiPrefix]["LSB"]["bandwidth"] = _this->bw;
|
|
|
|
_this->_config->release(true);
|
|
|
|
}
|
|
|
|
}
|
2020-12-06 16:13:47 +01:00
|
|
|
|
2021-04-17 22:37:50 +02:00
|
|
|
void setBandwidth(float bandWidth, bool updateWaterfall = true) {
|
2021-07-05 01:09:48 +02:00
|
|
|
bandWidth = std::clamp<float>(bandWidth, bwMin, bwMax);
|
2020-12-06 16:13:47 +01:00
|
|
|
bw = bandWidth;
|
2021-04-17 22:37:50 +02:00
|
|
|
_vfo->setBandwidth(bw, updateWaterfall);
|
2020-12-31 14:26:12 +01:00
|
|
|
demod.setBandWidth(bw);
|
|
|
|
float audioBW = std::min<float>(audioSampRate / 2.0f, bw);
|
|
|
|
win.setSampleRate(bbSampRate * resamp.getInterpolation());
|
|
|
|
win.setCutoff(audioBW);
|
|
|
|
win.setTransWidth(audioBW);
|
|
|
|
resamp.updateWindow(&win);
|
2020-12-06 16:13:47 +01:00
|
|
|
}
|
|
|
|
|
2021-07-31 21:00:47 +02:00
|
|
|
void saveParameters(bool lock = true) {
|
|
|
|
if (lock) { _config->acquire(); }
|
2021-08-20 20:40:14 +02:00
|
|
|
_config->conf[uiPrefix]["LSB"]["bandwidth"] = bw;
|
|
|
|
_config->conf[uiPrefix]["LSB"]["snapInterval"] = snapInterval;
|
|
|
|
_config->conf[uiPrefix]["LSB"]["squelchLevel"] = squelchLevel;
|
2021-07-31 21:00:47 +02:00
|
|
|
if (lock) { _config->release(true); }
|
|
|
|
}
|
|
|
|
|
2021-07-05 01:09:48 +02:00
|
|
|
private:
|
2020-12-06 16:13:47 +01:00
|
|
|
void setSnapInterval(float snapInt) {
|
|
|
|
snapInterval = snapInt;
|
|
|
|
_vfo->setSnapInterval(snapInterval);
|
|
|
|
}
|
|
|
|
|
2021-02-10 23:15:19 +01:00
|
|
|
const float bwMax = 12000;
|
2020-12-06 16:13:47 +01:00
|
|
|
const float bwMin = 500;
|
2021-02-10 23:15:19 +01:00
|
|
|
const float bbSampRate = 24000;
|
2020-12-06 16:13:47 +01:00
|
|
|
|
|
|
|
std::string uiPrefix;
|
|
|
|
float snapInterval = 100;
|
|
|
|
float audioSampRate = 48000;
|
|
|
|
float bw = 3000;
|
|
|
|
bool running = false;
|
2020-12-10 05:18:40 +01:00
|
|
|
float squelchLevel = -100.0f;
|
2020-12-06 16:13:47 +01:00
|
|
|
|
|
|
|
VFOManager::VFO* _vfo;
|
2020-12-10 05:18:40 +01:00
|
|
|
dsp::Squelch squelch;
|
2020-12-06 16:13:47 +01:00
|
|
|
dsp::SSBDemod demod;
|
|
|
|
dsp::AGC agc;
|
|
|
|
dsp::filter_window::BlackmanWindow win;
|
|
|
|
dsp::PolyphaseResampler<float> resamp;
|
|
|
|
dsp::MonoToStereo m2s;
|
|
|
|
|
2020-12-09 15:16:38 +01:00
|
|
|
ConfigManager* _config;
|
|
|
|
|
2021-08-20 20:40:14 +02:00
|
|
|
EventHandler<double> onUserChangedBandwidthHandler;
|
|
|
|
|
2020-12-06 16:13:47 +01:00
|
|
|
};
|