2021-08-16 18:49:00 +02:00
|
|
|
#include <imgui.h>
|
|
|
|
#include <module.h>
|
|
|
|
#include <gui/gui.h>
|
2022-07-15 17:17:53 +02:00
|
|
|
#include <gui/style.h>
|
|
|
|
#include <signal_path/signal_path.h>
|
2021-08-16 18:49:00 +02:00
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
SDRPP_MOD_INFO{
|
2021-08-16 18:49:00 +02:00
|
|
|
/* Name: */ "scanner",
|
|
|
|
/* Description: */ "Frequency scanner for SDR++",
|
|
|
|
/* Author: */ "Ryzerth",
|
|
|
|
/* Version: */ 0, 1, 0,
|
2022-08-30 16:07:49 +02:00
|
|
|
/* Max instances */ 1
|
2021-08-16 18:49:00 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class ScannerModule : public ModuleManager::Instance {
|
|
|
|
public:
|
|
|
|
ScannerModule(std::string name) {
|
|
|
|
this->name = name;
|
|
|
|
gui::menu.registerEntry(name, menuHandler, this, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
~ScannerModule() {
|
|
|
|
gui::menu.removeEntry(name);
|
2022-07-15 17:17:53 +02:00
|
|
|
stop();
|
2021-08-16 18:49:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void postInit() {}
|
|
|
|
|
|
|
|
void enable() {
|
|
|
|
enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable() {
|
|
|
|
enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEnabled() {
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static void menuHandler(void* ctx) {
|
|
|
|
ScannerModule* _this = (ScannerModule*)ctx;
|
2022-07-15 17:17:53 +02:00
|
|
|
float menuWidth = ImGui::GetContentRegionAvail().x;
|
|
|
|
|
|
|
|
if (_this->running) { ImGui::BeginDisabled(); }
|
|
|
|
ImGui::LeftLabel("Start");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputDouble("##start_freq_scanner", &_this->startFreq, 100.0, 100000.0, "%0.0f")) {
|
|
|
|
_this->startFreq = round(_this->startFreq);
|
|
|
|
}
|
|
|
|
ImGui::LeftLabel("Stop");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputDouble("##stop_freq_scanner", &_this->stopFreq, 100.0, 100000.0, "%0.0f")) {
|
|
|
|
_this->stopFreq = round(_this->stopFreq);
|
|
|
|
}
|
|
|
|
ImGui::LeftLabel("Interval");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputDouble("##interval_scanner", &_this->interval, 100.0, 100000.0, "%0.0f")) {
|
|
|
|
_this->interval = round(_this->interval);
|
|
|
|
}
|
2022-08-30 16:07:49 +02:00
|
|
|
ImGui::LeftLabel("Passband Ratio (%)");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputDouble("##pb_ratio_scanner", &_this->passbandRatio, 1.0, 10.0, "%0.0f")) {
|
|
|
|
_this->passbandRatio = std::clamp<double>(round(_this->passbandRatio), 1.0, 100.0);
|
|
|
|
}
|
|
|
|
ImGui::LeftLabel("Tuning Time (ms)");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputInt("##tuning_time_scanner", &_this->tuningTime, 100, 1000)) {
|
|
|
|
_this->tuningTime = std::clamp<int>(_this->tuningTime, 100, 10000.0);
|
|
|
|
}
|
|
|
|
ImGui::LeftLabel("Linger Time (ms)");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
if (ImGui::InputInt("##linger_time_scanner", &_this->lingerTime, 100, 1000)) {
|
|
|
|
_this->lingerTime = std::clamp<int>(_this->lingerTime, 100, 10000.0);
|
|
|
|
}
|
2022-07-15 17:17:53 +02:00
|
|
|
if (_this->running) { ImGui::EndDisabled(); }
|
|
|
|
|
|
|
|
ImGui::LeftLabel("Level");
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
|
|
ImGui::SliderFloat("##scanner_level", &_this->level, -150.0, 0.0);
|
|
|
|
|
2022-08-30 16:07:49 +02:00
|
|
|
ImGui::BeginTable(("scanner_bottom_btn_table" + _this->name).c_str(), 2);
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
ImGui::TableSetColumnIndex(0);
|
|
|
|
if (ImGui::Button(("<<##scanner_back_" + _this->name).c_str(), ImVec2(ImGui::GetContentRegionAvail().x, 0))) {
|
|
|
|
std::lock_guard<std::mutex> lck(_this->scanMtx);
|
|
|
|
_this->reverseLock = true;
|
|
|
|
_this->receiving = false;
|
|
|
|
_this->scanUp = false;
|
|
|
|
}
|
|
|
|
ImGui::TableSetColumnIndex(1);
|
|
|
|
if (ImGui::Button((">>##scanner_forw_" + _this->name).c_str(), ImVec2(ImGui::GetContentRegionAvail().x, 0))) {
|
|
|
|
std::lock_guard<std::mutex> lck(_this->scanMtx);
|
|
|
|
_this->reverseLock = true;
|
|
|
|
_this->receiving = false;
|
|
|
|
_this->scanUp = true;
|
|
|
|
}
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
2022-07-15 17:17:53 +02:00
|
|
|
if (!_this->running) {
|
|
|
|
if (ImGui::Button("Start##scanner_start", ImVec2(menuWidth, 0))) {
|
|
|
|
_this->start();
|
|
|
|
}
|
2022-08-31 14:59:22 +02:00
|
|
|
ImGui::Text("Status: Idle");
|
2022-07-15 17:17:53 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (ImGui::Button("Stop##scanner_start", ImVec2(menuWidth, 0))) {
|
|
|
|
_this->stop();
|
|
|
|
}
|
2022-08-31 14:59:22 +02:00
|
|
|
if (_this->receiving) {
|
|
|
|
ImGui::TextColored(ImVec4(0, 1, 0, 1), "Status: Receiving");
|
|
|
|
}
|
|
|
|
else if (_this->tuning) {
|
|
|
|
ImGui::TextColored(ImVec4(0, 1, 1, 1), "Status: Tuning");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ImGui::TextColored(ImVec4(1, 1, 0, 1), "Status: Scanning");
|
|
|
|
}
|
2022-07-15 17:17:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
|
|
|
if (running) { return; }
|
|
|
|
current = startFreq;
|
|
|
|
running = true;
|
|
|
|
workerThread = std::thread(&ScannerModule::worker, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void stop() {
|
|
|
|
if (!running) { return; }
|
|
|
|
running = false;
|
|
|
|
if (workerThread.joinable()) {
|
|
|
|
workerThread.join();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void worker() {
|
|
|
|
// 10Hz scan loop
|
|
|
|
while (running) {
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(100));
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lck(scanMtx);
|
|
|
|
auto now = std::chrono::high_resolution_clock::now();
|
|
|
|
|
|
|
|
// Enforce tuning
|
2022-08-30 16:07:49 +02:00
|
|
|
if (gui::waterfall.selectedVFO.empty()) {
|
|
|
|
running = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
tuner::normalTuning(gui::waterfall.selectedVFO, current);
|
2022-07-15 17:17:53 +02:00
|
|
|
|
|
|
|
// Check if we are waiting for a tune
|
|
|
|
if (tuning) {
|
|
|
|
spdlog::warn("Tuning");
|
2022-08-30 16:07:49 +02:00
|
|
|
if ((std::chrono::duration_cast<std::chrono::milliseconds>(now - lastTuneTime)).count() > tuningTime) {
|
2022-07-15 17:17:53 +02:00
|
|
|
tuning = false;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get FFT data
|
|
|
|
int dataWidth = 0;
|
|
|
|
float* data = gui::waterfall.acquireLatestFFT(dataWidth);
|
|
|
|
if (!data) { continue; }
|
|
|
|
|
|
|
|
// Get gather waterfall data
|
|
|
|
double wfCenter = gui::waterfall.getViewOffset() + gui::waterfall.getCenterFrequency();
|
|
|
|
double wfWidth = gui::waterfall.getViewBandwidth();
|
|
|
|
double wfStart = wfCenter - (wfWidth / 2.0);
|
|
|
|
double wfEnd = wfCenter + (wfWidth / 2.0);
|
|
|
|
|
|
|
|
// Gather VFO data
|
2022-08-30 16:07:49 +02:00
|
|
|
double vfoWidth = sigpath::vfoManager.getBandwidth(gui::waterfall.selectedVFO);
|
2022-07-15 17:17:53 +02:00
|
|
|
|
|
|
|
if (receiving) {
|
|
|
|
spdlog::warn("Receiving");
|
|
|
|
|
|
|
|
float maxLevel = getMaxLevel(data, current, vfoWidth, dataWidth, wfStart, wfWidth);
|
|
|
|
if (maxLevel >= level) {
|
|
|
|
lastSignalTime = now;
|
|
|
|
}
|
2022-08-30 16:07:49 +02:00
|
|
|
else if ((std::chrono::duration_cast<std::chrono::milliseconds>(now - lastSignalTime)).count() > lingerTime) {
|
2022-07-15 17:17:53 +02:00
|
|
|
receiving = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
spdlog::warn("Seeking signal");
|
2022-08-30 16:07:49 +02:00
|
|
|
double bottomLimit = current;
|
|
|
|
double topLimit = current;
|
2022-07-15 17:17:53 +02:00
|
|
|
|
|
|
|
// Search for a signal in scan direction
|
2022-08-30 16:07:49 +02:00
|
|
|
if (findSignal(scanUp, bottomLimit, topLimit, wfStart, wfEnd, wfWidth, vfoWidth, data, dataWidth)) {
|
2022-07-15 17:17:53 +02:00
|
|
|
gui::waterfall.releaseLatestFFT();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-08-30 16:07:49 +02:00
|
|
|
// Search for signal in the inverse scan direction if direction isn't enforced
|
|
|
|
if (!reverseLock) {
|
|
|
|
if (findSignal(!scanUp, bottomLimit, topLimit, wfStart, wfEnd, wfWidth, vfoWidth, data, dataWidth)) {
|
|
|
|
gui::waterfall.releaseLatestFFT();
|
|
|
|
continue;
|
|
|
|
}
|
2022-07-15 17:17:53 +02:00
|
|
|
}
|
2022-08-30 16:07:49 +02:00
|
|
|
else { reverseLock = false; }
|
|
|
|
|
2022-07-15 17:17:53 +02:00
|
|
|
|
|
|
|
// There is no signal on the visible spectrum, tune in scan direction and retry
|
|
|
|
if (scanUp) {
|
|
|
|
current = topLimit + interval;
|
|
|
|
if (current > stopFreq) { current = startFreq; }
|
|
|
|
}
|
|
|
|
else {
|
2022-08-30 16:07:49 +02:00
|
|
|
current = bottomLimit - interval;
|
2022-07-15 17:17:53 +02:00
|
|
|
if (current < startFreq) { current = stopFreq; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the new current frequency is outside the visible bandwidth, wait for retune
|
|
|
|
if (current - (vfoWidth/2.0) < wfStart || current + (vfoWidth/2.0) > wfEnd) {
|
|
|
|
lastTuneTime = now;
|
|
|
|
tuning = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release FFT Data
|
|
|
|
gui::waterfall.releaseLatestFFT();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-30 16:07:49 +02:00
|
|
|
bool findSignal(bool scanDir, double& bottomLimit, double& topLimit, double wfStart, double wfEnd, double wfWidth, double vfoWidth, float* data, int dataWidth) {
|
2022-07-15 17:17:53 +02:00
|
|
|
bool found = false;
|
|
|
|
double freq = current;
|
|
|
|
for (freq += scanDir ? interval : -interval;
|
|
|
|
scanDir ? (freq <= stopFreq) : (freq >= startFreq);
|
|
|
|
freq += scanDir ? interval : -interval) {
|
|
|
|
|
|
|
|
// Check if signal is within bounds
|
|
|
|
if (freq - (vfoWidth/2.0) < wfStart) { break; }
|
|
|
|
if (freq + (vfoWidth/2.0) > wfEnd) { break; }
|
|
|
|
|
|
|
|
if (freq < bottomLimit) { bottomLimit = freq; }
|
|
|
|
if (freq > topLimit) { topLimit = freq; }
|
|
|
|
|
|
|
|
// Check signal level
|
2022-08-30 16:07:49 +02:00
|
|
|
float maxLevel = getMaxLevel(data, freq, vfoWidth * (passbandRatio * 0.01f), dataWidth, wfStart, wfWidth);
|
2022-07-15 17:17:53 +02:00
|
|
|
if (maxLevel >= level) {
|
|
|
|
found = true;
|
|
|
|
receiving = true;
|
|
|
|
current = freq;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
float getMaxLevel(float* data, double freq, double width, int dataWidth, double wfStart, double wfWidth) {
|
|
|
|
double low = freq - (width/2.0);
|
|
|
|
double high = freq + (width/2.0);
|
|
|
|
int lowId = std::clamp<int>((low - wfStart) * (double)dataWidth / wfWidth, 0, dataWidth - 1);
|
|
|
|
int highId = std::clamp<int>((high - wfStart) * (double)dataWidth / wfWidth, 0, dataWidth - 1);
|
|
|
|
float max = -INFINITY;
|
|
|
|
for (int i = lowId; i <= highId; i++) {
|
|
|
|
if (data[i] > max) { max = data[i]; }
|
|
|
|
}
|
|
|
|
return max;
|
2021-08-16 18:49:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string name;
|
|
|
|
bool enabled = true;
|
2022-07-15 17:17:53 +02:00
|
|
|
|
|
|
|
bool running = false;
|
2022-08-30 16:07:49 +02:00
|
|
|
//std::string selectedVFO = "Radio";
|
|
|
|
double startFreq = 88000000.0;
|
|
|
|
double stopFreq = 108000000.0;
|
2022-07-15 17:17:53 +02:00
|
|
|
double interval = 100000.0;
|
|
|
|
double current = 88000000.0;
|
2022-08-30 16:07:49 +02:00
|
|
|
double passbandRatio = 10.0;
|
|
|
|
int tuningTime = 250;
|
|
|
|
int lingerTime = 1000.0;
|
2022-07-15 17:17:53 +02:00
|
|
|
float level = -50.0;
|
|
|
|
bool receiving = true;
|
|
|
|
bool tuning = false;
|
|
|
|
bool scanUp = true;
|
2022-08-30 16:07:49 +02:00
|
|
|
bool reverseLock = false;
|
2022-07-15 17:17:53 +02:00
|
|
|
std::chrono::time_point<std::chrono::high_resolution_clock> lastSignalTime;
|
|
|
|
std::chrono::time_point<std::chrono::high_resolution_clock> lastTuneTime;
|
|
|
|
std::thread workerThread;
|
|
|
|
std::mutex scanMtx;
|
2021-08-16 18:49:00 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
MOD_EXPORT void _INIT_() {
|
|
|
|
// Nothing here
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_(std::string name) {
|
|
|
|
return new ScannerModule(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
|
|
|
|
delete (ScannerModule*)instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _END_() {
|
|
|
|
// Nothing here
|
|
|
|
}
|