Skip to content

Commit 4799d0e

Browse files
add VOR receiver module
1 parent ea3675d commit 4799d0e

File tree

8 files changed

+2366
-0
lines changed

8 files changed

+2366
-0
lines changed

CMakeLists.txt

+5
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,7 @@ option(OPT_BUILD_METEOR_DEMODULATOR "Build the meteor demodulator module (no dep
5353
option(OPT_BUILD_PAGER_DECODER "Build the pager decoder module (no dependencies required)" ON)
5454
option(OPT_BUILD_RADIO "Main audio modulation decoder (AM, FM, SSB, etc...)" ON)
5555
option(OPT_BUILD_RYFI_DECODER "RyFi data link decoder" OFF)
56+
option(OPT_BUILD_VOR_RECEIVER "VOR beacon receiver" ON)
5657
option(OPT_BUILD_WEATHER_SAT_DECODER "Build the HRPT decoder module (no dependencies required)" OFF)
5758

5859
# Misc
@@ -289,6 +290,10 @@ if (OPT_BUILD_RYFI_DECODER)
289290
add_subdirectory("decoder_modules/ryfi_decoder")
290291
endif (OPT_BUILD_RYFI_DECODER)
291292

293+
if (OPT_BUILD_VOR_RECEIVER)
294+
add_subdirectory("decoder_modules/vor_receiver")
295+
endif (OPT_BUILD_VOR_RECEIVER)
296+
292297
if (OPT_BUILD_WEATHER_SAT_DECODER)
293298
add_subdirectory("decoder_modules/weather_sat_decoder")
294299
endif (OPT_BUILD_WEATHER_SAT_DECODER)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
cmake_minimum_required(VERSION 3.13)
2+
project(vor_receiver)
3+
4+
file(GLOB_RECURSE SRC "src/*.cpp")
5+
6+
include(${SDRPP_MODULE_CMAKE})
7+
8+
target_include_directories(vor_receiver PRIVATE "src/")
+128
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
#include <imgui.h>
2+
#include <config.h>
3+
#include <core.h>
4+
#include <gui/style.h>
5+
#include <gui/gui.h>
6+
#include <signal_path/signal_path.h>
7+
#include <module.h>
8+
#include <filesystem>
9+
#include <dsp/buffer/reshaper.h>
10+
#include <dsp/sink/handler_sink.h>
11+
#include <gui/widgets/constellation_diagram.h>
12+
#include "vor_decoder.h"
13+
#include <fstream>
14+
15+
#define CONCAT(a, b) ((std::string(a) + b).c_str())
16+
17+
SDRPP_MOD_INFO{
18+
/* Name: */ "vor_receiver",
19+
/* Description: */ "VOR Receiver for SDR++",
20+
/* Author: */ "Ryzerth",
21+
/* Version: */ 0, 1, 0,
22+
/* Max instances */ -1
23+
};
24+
25+
ConfigManager config;
26+
27+
#define INPUT_SAMPLE_RATE VOR_IN_SR
28+
29+
class VORReceiverModule : public ModuleManager::Instance {
30+
public:
31+
VORReceiverModule(std::string name) {
32+
this->name = name;
33+
34+
// Load config
35+
config.acquire();
36+
// TODO: Load config
37+
config.release();
38+
39+
vfo = sigpath::vfoManager.createVFO(name, ImGui::WaterfallVFO::REF_CENTER, 0, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, true);
40+
decoder = new vor::Decoder(vfo->output, 1);
41+
decoder->onBearing.bind(&VORReceiverModule::onBearing, this);
42+
43+
decoder->start();
44+
45+
gui::menu.registerEntry(name, menuHandler, this, this);
46+
}
47+
48+
~VORReceiverModule() {
49+
decoder->stop();
50+
sigpath::vfoManager.deleteVFO(vfo);
51+
gui::menu.removeEntry(name);
52+
delete decoder;
53+
}
54+
55+
void postInit() {}
56+
57+
void enable() {
58+
double bw = gui::waterfall.getBandwidth();
59+
vfo = sigpath::vfoManager.createVFO(name, ImGui::WaterfallVFO::REF_CENTER, 0, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, true);
60+
61+
decoder->setInput(vfo->output);
62+
63+
decoder->start();
64+
65+
enabled = true;
66+
}
67+
68+
void disable() {
69+
decoder->stop();
70+
71+
sigpath::vfoManager.deleteVFO(vfo);
72+
enabled = false;
73+
}
74+
75+
bool isEnabled() {
76+
return enabled;
77+
}
78+
79+
private:
80+
static void menuHandler(void* ctx) {
81+
VORReceiverModule* _this = (VORReceiverModule*)ctx;
82+
83+
float menuWidth = ImGui::GetContentRegionAvail().x;
84+
85+
if (!_this->enabled) { style::beginDisabled(); }
86+
87+
ImGui::Text("Bearing: %f°", _this->bearing);
88+
ImGui::Text("Quality: %0.1f%%", _this->quality);
89+
90+
if (!_this->enabled) { style::endDisabled(); }
91+
}
92+
93+
void onBearing(float nbearing, float nquality) {
94+
bearing = (180.0f * nbearing / FL_M_PI);
95+
quality = nquality * 100.0f;
96+
}
97+
98+
std::string name;
99+
bool enabled = true;
100+
101+
// DSP Chain
102+
VFOManager::VFO* vfo;
103+
vor::Decoder* decoder;
104+
105+
float bearing = 0.0f, quality = 0.0f;
106+
};
107+
108+
MOD_EXPORT void _INIT_() {
109+
// Create default recording directory
110+
std::string root = (std::string)core::args["root"];
111+
json def = json({});
112+
config.setPath(root + "/vor_receiver_config.json");
113+
config.load(def);
114+
config.enableAutoSave();
115+
}
116+
117+
MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_(std::string name) {
118+
return new VORReceiverModule(name);
119+
}
120+
121+
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
122+
delete (VORReceiverModule*)instance;
123+
}
124+
125+
MOD_EXPORT void _END_() {
126+
config.disableAutoSave();
127+
config.save();
128+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
#include "vor_decoder.h"
2+
3+
#define STDDEV_NORM_FACTOR 1.813799364234218f // 2.0f * FL_M_PI / sqrt(12)
4+
5+
namespace vor {
6+
Decoder::Decoder(dsp::stream<dsp::complex_t>* in, double integrationTime) {
7+
rx.init(in);
8+
reshape.init(&rx.out, round(1000.0 * integrationTime), 0);
9+
symSink.init(&reshape.out, dataHandler, this);
10+
}
11+
12+
Decoder::~Decoder() {
13+
// TODO
14+
}
15+
16+
void Decoder::setInput(dsp::stream<dsp::complex_t>* in) {
17+
rx.setInput(in);
18+
}
19+
20+
void Decoder::start() {
21+
rx.start();
22+
reshape.start();
23+
symSink.start();
24+
}
25+
26+
void Decoder::stop() {
27+
rx.stop();
28+
reshape.stop();
29+
symSink.stop();
30+
}
31+
32+
void Decoder::dataHandler(float* data, int count, void* ctx) {
33+
// Get the instance from context
34+
Decoder* _this = (Decoder*)ctx;
35+
36+
// Compute the mean and standard deviation of the
37+
float mean, stddev;
38+
volk_32f_stddev_and_mean_32f_x2(&stddev, &mean, data, count);
39+
40+
// Compute the signal quality
41+
float quality = std::max<float>(1.0f - (stddev / STDDEV_NORM_FACTOR), 0.0f);
42+
43+
// Convert the phase difference to a compass heading
44+
mean = -mean;
45+
if (mean < 0) { mean = 2.0f*FL_M_PI + mean; }
46+
47+
// Call the handler
48+
_this->onBearing(mean, quality);
49+
}
50+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
#include "vor_receiver.h"
2+
#include <dsp/buffer/reshaper.h>
3+
#include <dsp/sink/handler_sink.h>
4+
#include <utils/new_event.h>
5+
6+
namespace vor {
7+
// Note: hard coded to 22KHz samplerate
8+
class Decoder {
9+
public:
10+
/**
11+
* Create an instance of a VOR decoder.
12+
* @param in Input IQ stream at 22 KHz sampling rate.
13+
* @param integrationTime Integration time of the bearing data in seconds.
14+
*/
15+
Decoder(dsp::stream<dsp::complex_t>* in, double integrationTime);
16+
17+
// Destructor
18+
~Decoder();
19+
20+
/**
21+
* Set the input stream.
22+
* @param in Input IQ stream at 22 KHz sampling rate.
23+
*/
24+
void setInput(dsp::stream<dsp::complex_t>* in);
25+
26+
/**
27+
* Start the decoder.
28+
*/
29+
void start();
30+
31+
/**
32+
* Stop the decoder.
33+
*/
34+
void stop();
35+
36+
/**
37+
* handler(bearing, signalQuality);
38+
*/
39+
NewEvent<float, float> onBearing;
40+
41+
private:
42+
static void dataHandler(float* data, int count, void* ctx);
43+
44+
// DSP
45+
Receiver rx;
46+
dsp::buffer::Reshaper<float> reshape;
47+
dsp::sink::Handler<float> symSink;
48+
};
49+
}

0 commit comments

Comments
 (0)