2021-07-16 01:49:41 +02:00
|
|
|
#include <utils/networking.h>
|
|
|
|
#include <imgui.h>
|
|
|
|
#include <module.h>
|
|
|
|
#include <gui/gui.h>
|
|
|
|
#include <gui/style.h>
|
|
|
|
#include <signal_path/signal_path.h>
|
|
|
|
#include <core.h>
|
|
|
|
#include <recorder_interface.h>
|
2021-07-18 04:30:55 +02:00
|
|
|
#include <meteor_demodulator_interface.h>
|
|
|
|
#include <config.h>
|
2021-07-28 04:21:51 +02:00
|
|
|
#include <cctype>
|
|
|
|
#include <radio_interface.h>
|
2021-07-18 04:30:55 +02:00
|
|
|
#define CONCAT(a, b) ((std::string(a) + b).c_str())
|
2021-07-16 01:49:41 +02:00
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
#define MAX_COMMAND_LENGTH 8192
|
2021-07-16 01:49:41 +02:00
|
|
|
|
2021-12-19 22:11:44 +01:00
|
|
|
SDRPP_MOD_INFO{
|
2021-07-16 01:49:41 +02:00
|
|
|
/* Name: */ "rigctl_server",
|
|
|
|
/* Description: */ "My fancy new module",
|
|
|
|
/* Author: */ "Ryzerth",
|
|
|
|
/* Version: */ 0, 1, 0,
|
|
|
|
/* Max instances */ -1
|
|
|
|
};
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
enum {
|
|
|
|
RECORDER_TYPE_RECORDER,
|
|
|
|
RECORDER_TYPE_METEOR_DEMODULATOR
|
|
|
|
};
|
|
|
|
|
|
|
|
ConfigManager config;
|
|
|
|
|
2021-07-16 01:49:41 +02:00
|
|
|
class SigctlServerModule : public ModuleManager::Instance {
|
|
|
|
public:
|
|
|
|
SigctlServerModule(std::string name) {
|
|
|
|
this->name = name;
|
2021-07-18 04:30:55 +02:00
|
|
|
|
|
|
|
config.acquire();
|
|
|
|
if (!config.conf.contains(name)) {
|
|
|
|
config.conf[name]["host"] = "localhost";
|
|
|
|
config.conf[name]["port"] = 4532;
|
|
|
|
config.conf[name]["tuning"] = true;
|
|
|
|
config.conf[name]["recording"] = false;
|
|
|
|
config.conf[name]["autoStart"] = false;
|
|
|
|
config.conf[name]["vfo"] = "";
|
|
|
|
config.conf[name]["recorder"] = "";
|
|
|
|
}
|
|
|
|
std::string host = config.conf[name]["host"];
|
|
|
|
strcpy(hostname, host.c_str());
|
|
|
|
port = config.conf[name]["port"];
|
|
|
|
tuningEnabled = config.conf[name]["tuning"];
|
|
|
|
recordingEnabled = config.conf[name]["recording"];
|
|
|
|
autoStart = config.conf[name]["autoStart"];
|
|
|
|
selectedVfo = config.conf[name]["vfo"];
|
|
|
|
selectedRecorder = config.conf[name]["recorder"];
|
|
|
|
config.release(true);
|
|
|
|
|
2021-07-16 01:49:41 +02:00
|
|
|
gui::menu.registerEntry(name, menuHandler, this, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
~SigctlServerModule() {
|
|
|
|
gui::menu.removeEntry(name);
|
2021-07-26 03:11:51 +02:00
|
|
|
sigpath::vfoManager.onVfoCreated.unbindHandler(&vfoCreatedHandler);
|
|
|
|
sigpath::vfoManager.onVfoDeleted.unbindHandler(&vfoDeletedHandler);
|
|
|
|
core::moduleManager.onInstanceCreated.unbindHandler(&modChangedHandler);
|
|
|
|
core::moduleManager.onInstanceDeleted.unbindHandler(&modChangedHandler);
|
|
|
|
if (client) { client->close(); }
|
|
|
|
if (listener) { listener->close(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
void postInit() {
|
|
|
|
// Refresh modules
|
|
|
|
refreshModules();
|
|
|
|
|
|
|
|
// Select VFO and recorder from config
|
|
|
|
selectVfoByName(selectedVfo);
|
|
|
|
selectRecorderByName(selectedRecorder);
|
|
|
|
|
|
|
|
// Bind handlers
|
|
|
|
vfoCreatedHandler.handler = _vfoCreatedHandler;
|
|
|
|
vfoCreatedHandler.ctx = this;
|
|
|
|
vfoDeletedHandler.handler = _vfoDeletedHandler;
|
|
|
|
vfoDeletedHandler.ctx = this;
|
|
|
|
modChangedHandler.handler = _modChangeHandler;
|
|
|
|
modChangedHandler.ctx = this;
|
|
|
|
sigpath::vfoManager.onVfoCreated.bindHandler(&vfoCreatedHandler);
|
|
|
|
sigpath::vfoManager.onVfoDeleted.bindHandler(&vfoDeletedHandler);
|
|
|
|
core::moduleManager.onInstanceCreated.bindHandler(&modChangedHandler);
|
|
|
|
core::moduleManager.onInstanceDeleted.bindHandler(&modChangedHandler);
|
|
|
|
|
|
|
|
// If autostart is enabled, start the server
|
|
|
|
if (autoStart) { startServer(); }
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void enable() {
|
|
|
|
enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable() {
|
|
|
|
enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEnabled() {
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static void menuHandler(void* ctx) {
|
|
|
|
SigctlServerModule* _this = (SigctlServerModule*)ctx;
|
2022-02-14 23:33:52 +01:00
|
|
|
float menuWidth = ImGui::GetContentRegionAvail().x;
|
2021-07-16 01:49:41 +02:00
|
|
|
|
|
|
|
bool listening = (_this->listener && _this->listener->isListening());
|
|
|
|
|
|
|
|
if (listening) { style::beginDisabled(); }
|
2021-07-18 04:30:55 +02:00
|
|
|
if (ImGui::InputText(CONCAT("##_rigctl_srv_host_", _this->name), _this->hostname, 1023)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["host"] = std::string(_this->hostname);
|
|
|
|
config.release(true);
|
|
|
|
}
|
2021-07-16 01:49:41 +02:00
|
|
|
ImGui::SameLine();
|
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
2021-07-18 04:30:55 +02:00
|
|
|
if (ImGui::InputInt(CONCAT("##_rigctl_srv_port_", _this->name), &_this->port, 0, 0)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["port"] = _this->port;
|
|
|
|
config.release(true);
|
|
|
|
}
|
2021-07-16 01:49:41 +02:00
|
|
|
if (listening) { style::endDisabled(); }
|
|
|
|
|
2021-08-31 18:39:48 +02:00
|
|
|
ImGui::LeftLabel("Controlled VFO");
|
2021-07-16 01:49:41 +02:00
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
2021-07-18 04:30:55 +02:00
|
|
|
{
|
|
|
|
std::lock_guard lck(_this->vfoMtx);
|
|
|
|
if (ImGui::Combo(CONCAT("##_rigctl_srv_vfo_", _this->name), &_this->vfoId, _this->vfoNamesTxt.c_str())) {
|
|
|
|
_this->selectVfoByName(_this->vfoNames[_this->vfoId], false);
|
2023-01-13 17:11:16 +01:00
|
|
|
if (!_this->selectedVfo.empty()) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["vfo"] = _this->selectedVfo;
|
|
|
|
config.release(true);
|
|
|
|
}
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-08-31 18:39:48 +02:00
|
|
|
ImGui::LeftLabel("Controlled Recorder");
|
2021-07-16 01:49:41 +02:00
|
|
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
2021-07-18 04:30:55 +02:00
|
|
|
{
|
|
|
|
std::lock_guard lck(_this->vfoMtx);
|
|
|
|
if (ImGui::Combo(CONCAT("##_rigctl_srv_rec_", _this->name), &_this->recorderId, _this->recorderNamesTxt.c_str())) {
|
|
|
|
_this->selectRecorderByName(_this->recorderNames[_this->recorderId], false);
|
2023-01-13 22:43:42 +01:00
|
|
|
if (!_this->selectedRecorder.empty()) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["recorder"] = _this->selectedRecorder;
|
|
|
|
config.release(true);
|
|
|
|
}
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
ImGui::BeginTable(CONCAT("Stop##_rigctl_srv_tbl_", _this->name), 2);
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
ImGui::TableSetColumnIndex(0);
|
|
|
|
if (ImGui::Checkbox(CONCAT("Tuning##_rigctl_srv_tune_ena_", _this->name), &_this->tuningEnabled)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["tuning"] = _this->tuningEnabled;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
ImGui::TableSetColumnIndex(1);
|
|
|
|
if (ImGui::Checkbox(CONCAT("Recording##_rigctl_srv_tune_ena_", _this->name), &_this->recordingEnabled)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["recording"] = _this->recordingEnabled;
|
|
|
|
config.release(true);
|
|
|
|
}
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
|
|
|
if (ImGui::Checkbox(CONCAT("Listen on startup##_rigctl_srv_auto_lst_", _this->name), &_this->autoStart)) {
|
|
|
|
config.acquire();
|
|
|
|
config.conf[_this->name]["autoStart"] = _this->autoStart;
|
|
|
|
config.release(true);
|
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
if (listening && ImGui::Button(CONCAT("Stop##_rigctl_srv_stop_", _this->name), ImVec2(menuWidth, 0))) {
|
|
|
|
_this->stopServer();
|
|
|
|
}
|
|
|
|
else if (!listening && ImGui::Button(CONCAT("Start##_rigctl_srv_stop_", _this->name), ImVec2(menuWidth, 0))) {
|
|
|
|
_this->startServer();
|
|
|
|
}
|
|
|
|
|
2022-01-26 14:50:16 +01:00
|
|
|
ImGui::TextUnformatted("Status:");
|
2021-07-16 01:49:41 +02:00
|
|
|
ImGui::SameLine();
|
|
|
|
if (_this->client && _this->client->isOpen()) {
|
|
|
|
ImGui::TextColored(ImVec4(0.0, 1.0, 0.0, 1.0), "Connected");
|
|
|
|
}
|
|
|
|
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-16 01:49:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
void startServer() {
|
|
|
|
try {
|
2021-07-30 21:56:34 +02:00
|
|
|
listener = net::listen(hostname, port);
|
2021-07-18 04:30:55 +02:00
|
|
|
listener->acceptAsync(clientHandler, this);
|
|
|
|
}
|
2024-01-22 19:46:01 +01:00
|
|
|
catch (const std::exception& e) {
|
|
|
|
flog::error("Could not start rigctl server: {}", e.what());
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void stopServer() {
|
|
|
|
if (client) { client->close(); }
|
|
|
|
listener->close();
|
|
|
|
}
|
|
|
|
|
|
|
|
void refreshModules() {
|
|
|
|
vfoNames.clear();
|
|
|
|
vfoNamesTxt.clear();
|
|
|
|
recorderNames.clear();
|
|
|
|
recorderNamesTxt.clear();
|
|
|
|
|
|
|
|
// List recording capable modules
|
|
|
|
for (auto const& [_name, inst] : core::moduleManager.instances) {
|
|
|
|
std::string mod = core::moduleManager.getInstanceModuleName(_name);
|
|
|
|
if (mod != "recorder" && mod != "meteor_demodulator") { continue; }
|
|
|
|
recorderNames.push_back(_name);
|
|
|
|
recorderNamesTxt += _name;
|
|
|
|
recorderNamesTxt += '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
// List VFOs
|
|
|
|
for (auto const& [_name, vfo] : gui::waterfall.vfos) {
|
|
|
|
vfoNames.push_back(_name);
|
|
|
|
vfoNamesTxt += _name;
|
|
|
|
vfoNamesTxt += '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void selectVfoByName(std::string _name, bool lock = true) {
|
|
|
|
if (vfoNames.empty()) {
|
|
|
|
if (lock) { std::lock_guard lck(vfoMtx); }
|
|
|
|
selectedVfo.clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the ID of the VFO, if not found, select first VFO in the list
|
|
|
|
auto vfoIt = std::find(vfoNames.begin(), vfoNames.end(), _name);
|
|
|
|
if (vfoIt == vfoNames.end()) {
|
|
|
|
selectVfoByName(vfoNames[0]);
|
|
|
|
return;
|
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// Select the VFO
|
|
|
|
{
|
|
|
|
if (lock) { std::lock_guard lck(vfoMtx); }
|
|
|
|
vfoId = std::distance(vfoNames.begin(), vfoIt);
|
|
|
|
selectedVfo = _name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void selectRecorderByName(std::string _name, bool lock = true) {
|
|
|
|
if (recorderNames.empty()) {
|
|
|
|
if (lock) { std::lock_guard lck(recorderMtx); }
|
|
|
|
selectedRecorder.clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the ID of the VFO, if not found, select first VFO in the list
|
|
|
|
auto recIt = std::find(recorderNames.begin(), recorderNames.end(), _name);
|
|
|
|
if (recIt == recorderNames.end()) {
|
|
|
|
selectRecorderByName(recorderNames[0]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string type = core::modComManager.getModuleName(_name);
|
2021-12-19 22:11:44 +01:00
|
|
|
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// Select the VFO
|
|
|
|
{
|
|
|
|
if (lock) { std::lock_guard lck(recorderMtx); }
|
|
|
|
recorderId = std::distance(recorderNames.begin(), recIt);
|
|
|
|
selectedRecorder = _name;
|
|
|
|
if (type == "meteor_demodulator") {
|
|
|
|
recorderType = RECORDER_TYPE_METEOR_DEMODULATOR;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
recorderType = RECORDER_TYPE_RECORDER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _vfoCreatedHandler(VFOManager::VFO* vfo, void* ctx) {
|
|
|
|
SigctlServerModule* _this = (SigctlServerModule*)ctx;
|
|
|
|
_this->refreshModules();
|
|
|
|
_this->selectVfoByName(_this->selectedVfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _vfoDeletedHandler(std::string _name, void* ctx) {
|
|
|
|
SigctlServerModule* _this = (SigctlServerModule*)ctx;
|
|
|
|
_this->refreshModules();
|
|
|
|
_this->selectVfoByName(_this->selectedVfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _modChangeHandler(std::string _name, void* ctx) {
|
|
|
|
SigctlServerModule* _this = (SigctlServerModule*)ctx;
|
|
|
|
_this->refreshModules();
|
|
|
|
_this->selectRecorderByName(_this->selectedRecorder);
|
|
|
|
}
|
|
|
|
|
2021-07-16 01:49:41 +02:00
|
|
|
static void clientHandler(net::Conn _client, void* ctx) {
|
|
|
|
SigctlServerModule* _this = (SigctlServerModule*)ctx;
|
2023-02-25 18:12:34 +01:00
|
|
|
//flog::info("New client!");
|
2021-07-16 01:49:41 +02:00
|
|
|
|
|
|
|
_this->client = std::move(_client);
|
2022-01-22 16:36:48 +01:00
|
|
|
_this->client->readAsync(1024, _this->dataBuf, dataHandler, _this, false);
|
2021-07-16 01:49:41 +02:00
|
|
|
_this->client->waitForEnd();
|
|
|
|
_this->client->close();
|
|
|
|
|
2023-02-25 18:12:34 +01:00
|
|
|
//flog::info("Client disconnected!");
|
2021-07-16 01:49:41 +02:00
|
|
|
|
|
|
|
_this->listener->acceptAsync(clientHandler, _this);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dataHandler(int count, uint8_t* data, void* ctx) {
|
|
|
|
SigctlServerModule* _this = (SigctlServerModule*)ctx;
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
if (data[i] == '\n') {
|
|
|
|
_this->commandHandler(_this->command);
|
|
|
|
_this->command.clear();
|
|
|
|
continue;
|
|
|
|
}
|
2021-07-18 04:30:55 +02:00
|
|
|
if (_this->command.size() < MAX_COMMAND_LENGTH) { _this->command += (char)data[i]; }
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
|
|
|
|
2022-01-22 16:36:48 +01:00
|
|
|
_this->client->readAsync(1024, _this->dataBuf, dataHandler, _this, false);
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
|
|
|
|
2023-09-06 05:38:00 +02:00
|
|
|
std::map<int, const char*> radioModeToString = {
|
|
|
|
{ RADIO_IFACE_MODE_NFM, "NFM" },
|
|
|
|
{ RADIO_IFACE_MODE_WFM, "WFM" },
|
|
|
|
{ RADIO_IFACE_MODE_AM, "AM" },
|
|
|
|
{ RADIO_IFACE_MODE_DSB, "DSB" },
|
|
|
|
{ RADIO_IFACE_MODE_USB, "USB" },
|
|
|
|
{ RADIO_IFACE_MODE_CW, "CW" },
|
|
|
|
{ RADIO_IFACE_MODE_LSB, "LSB" },
|
|
|
|
{ RADIO_IFACE_MODE_RAW, "RAW" }
|
|
|
|
};
|
|
|
|
|
2021-07-16 01:49:41 +02:00
|
|
|
void commandHandler(std::string cmd) {
|
|
|
|
std::string corr = "";
|
|
|
|
std::vector<std::string> parts;
|
|
|
|
bool lastWasSpace = false;
|
|
|
|
std::string resp = "";
|
|
|
|
|
|
|
|
// Split command into parts and remove excess spaces
|
|
|
|
for (char c : cmd) {
|
|
|
|
if (lastWasSpace && c == ' ') { continue; }
|
|
|
|
else if (c == ' ') {
|
|
|
|
parts.push_back(corr);
|
|
|
|
corr.clear();
|
|
|
|
lastWasSpace = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
lastWasSpace = false;
|
|
|
|
corr += c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!corr.empty()) {
|
|
|
|
parts.push_back(corr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: THIS STUFF ISN'T THREADSAFE AND WILL LIKELY BREAK.
|
|
|
|
|
2021-07-28 04:21:51 +02:00
|
|
|
// If the command is empty, do nothing
|
2021-07-16 01:49:41 +02:00
|
|
|
if (parts.size() == 0) { return; }
|
2021-07-26 20:30:03 -04:00
|
|
|
|
2021-07-28 04:21:51 +02:00
|
|
|
// If the command is a compound command, execute each one separately
|
2021-08-08 21:38:01 +02:00
|
|
|
if (parts[0].size() > 1 && parts[0][0] != '\\' && parts[0] != "AOS" && parts[0] != "LOS") {
|
2021-07-28 04:21:51 +02:00
|
|
|
std::string arguments;
|
|
|
|
if (parts.size() > 1) { arguments = cmd.substr(parts[0].size()); }
|
|
|
|
for (char c : parts[0]) {
|
|
|
|
commandHandler(c + arguments);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::info("Rigctl command: '{0}'", cmd);
|
2021-07-28 04:21:51 +02:00
|
|
|
|
|
|
|
// Otherwise, execute the command
|
|
|
|
if (parts[0] == "F" || parts[0] == "\\set_freq") {
|
2021-07-18 04:30:55 +02:00
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
|
2021-07-16 01:49:41 +02:00
|
|
|
// if number of arguments isn't correct, return error
|
|
|
|
if (parts.size() != 2) {
|
|
|
|
resp = "RPRT 1\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
return;
|
|
|
|
}
|
2021-07-26 20:30:03 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// If not controlling the VFO, return
|
|
|
|
if (!tuningEnabled) {
|
2021-07-26 20:30:03 -04:00
|
|
|
resp = "RPRT 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-29 21:20:40 +02:00
|
|
|
return;
|
2021-07-26 20:30:03 -04:00
|
|
|
}
|
|
|
|
|
2021-07-16 01:49:41 +02:00
|
|
|
// Parse frequency and assign it to the VFO
|
|
|
|
long long freq = std::stoll(parts[1]);
|
2021-07-18 04:30:55 +02:00
|
|
|
tuner::tune(tuner::TUNER_MODE_NORMAL, selectedVfo, freq);
|
2021-07-16 01:49:41 +02:00
|
|
|
resp = "RPRT 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
else if (parts[0] == "f" || parts[0] == "\\get_freq") {
|
2021-07-18 04:30:55 +02:00
|
|
|
std::lock_guard lck(vfoMtx);
|
2021-07-26 20:30:03 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// Get center frequency of the SDR
|
2021-07-16 01:49:41 +02:00
|
|
|
double freq = gui::waterfall.getCenterFrequency();
|
2021-07-26 20:30:03 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// Add the offset of the VFO if it exists
|
|
|
|
if (sigpath::vfoManager.vfoExists(selectedVfo)) {
|
|
|
|
freq += sigpath::vfoManager.getOffset(selectedVfo);
|
2021-07-26 20:30:03 -04:00
|
|
|
}
|
2021-07-27 12:18:50 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// Respond with the frequency
|
2021-07-16 01:49:41 +02:00
|
|
|
char buf[128];
|
2021-07-16 03:11:49 +02:00
|
|
|
sprintf(buf, "%" PRIu64 "\n", (uint64_t)freq);
|
2021-07-16 01:49:41 +02:00
|
|
|
client->write(strlen(buf), (uint8_t*)buf);
|
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
else if (parts[0] == "M" || parts[0] == "\\set_mode") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp = "RPRT 0\n";
|
2021-07-27 12:18:50 -04:00
|
|
|
|
2021-07-29 12:38:23 -04:00
|
|
|
// If client is querying, respond accordingly
|
2021-07-28 04:21:51 +02:00
|
|
|
if (parts.size() >= 2 && parts[1] == "?") {
|
|
|
|
resp = "FM WFM AM DSB USB CW LSB RAW\n";
|
2021-07-27 11:57:51 -04:00
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-28 04:21:51 +02:00
|
|
|
return;
|
2021-07-27 11:57:51 -04:00
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
|
2021-08-20 20:16:37 +03:00
|
|
|
// if number of arguments isn't correct, return error
|
2021-07-28 04:21:51 +02:00
|
|
|
if (parts.size() != 3) {
|
|
|
|
resp = "RPRT 1\n";
|
2021-07-27 11:57:51 -04:00
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-28 04:21:51 +02:00
|
|
|
return;
|
2021-07-27 11:57:51 -04:00
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
|
2021-08-20 20:16:37 +03:00
|
|
|
// Check that the bandwidth is an integer (0 or -1 for default bandwidth)
|
|
|
|
int pos = 0;
|
2021-07-28 04:21:51 +02:00
|
|
|
for (char c : parts[2]) {
|
2021-08-20 20:16:37 +03:00
|
|
|
if (!std::isdigit(c) && !(c == '-' && !pos)) {
|
2021-07-27 12:18:50 -04:00
|
|
|
resp = "RPRT 1\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
return;
|
|
|
|
}
|
2021-08-20 20:16:37 +03:00
|
|
|
pos++;
|
2021-07-27 11:57:51 -04:00
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
|
2023-09-06 05:38:00 +02:00
|
|
|
const std::string& newModeStr = parts[1];
|
2021-07-28 04:21:51 +02:00
|
|
|
float newBandwidth = std::atoi(parts[2].c_str());
|
2023-09-06 05:38:00 +02:00
|
|
|
|
|
|
|
auto it = std::find_if(radioModeToString.begin(), radioModeToString.end(), [&newModeStr](const auto& e) {
|
|
|
|
return e.second == newModeStr;
|
|
|
|
});
|
|
|
|
if (it == radioModeToString.end()) {
|
2021-07-28 04:21:51 +02:00
|
|
|
resp = "RPRT 1\n";
|
2021-07-27 11:57:51 -04:00
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-28 04:21:51 +02:00
|
|
|
return;
|
2021-07-27 11:57:51 -04:00
|
|
|
}
|
2023-09-06 05:38:00 +02:00
|
|
|
int newMode = it->first;
|
2021-07-26 20:00:32 -04:00
|
|
|
|
2021-07-28 04:21:51 +02:00
|
|
|
// If tuning is enabled, set the mode and optionally the bandwidth
|
|
|
|
if (!selectedVfo.empty() && core::modComManager.getModuleName(selectedVfo) == "radio" && tuningEnabled) {
|
|
|
|
core::modComManager.callInterface(selectedVfo, RADIO_IFACE_CMD_SET_MODE, &newMode, NULL);
|
2021-08-20 20:16:37 +03:00
|
|
|
if (newBandwidth > 0) {
|
2021-07-28 04:21:51 +02:00
|
|
|
core::modComManager.callInterface(selectedVfo, RADIO_IFACE_CMD_SET_BANDWIDTH, &newBandwidth, NULL);
|
2021-07-26 20:00:32 -04:00
|
|
|
}
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
2021-12-19 22:11:44 +01:00
|
|
|
|
2021-07-28 04:21:51 +02:00
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
|
|
|
else if (parts[0] == "m" || parts[0] == "\\get_mode") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
2021-08-20 21:45:07 +02:00
|
|
|
resp = "RAW\n";
|
2021-07-28 04:21:51 +02:00
|
|
|
|
|
|
|
if (!selectedVfo.empty() && core::modComManager.getModuleName(selectedVfo) == "radio") {
|
|
|
|
int mode;
|
|
|
|
core::modComManager.callInterface(selectedVfo, RADIO_IFACE_CMD_GET_MODE, NULL, &mode);
|
2023-09-06 05:38:00 +02:00
|
|
|
resp = std::string(radioModeToString[mode]) + "\n";
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
2023-09-06 05:38:00 +02:00
|
|
|
else if (!selectedVfo.empty()) {
|
2021-07-28 04:34:44 +02:00
|
|
|
resp += std::to_string((int)sigpath::vfoManager.getBandwidth(selectedVfo)) + "\n";
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
|
|
|
else {
|
2021-07-28 04:34:44 +02:00
|
|
|
resp += "0\n";
|
|
|
|
}
|
|
|
|
|
2021-07-28 04:21:51 +02:00
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
|
|
|
else if (parts[0] == "V" || parts[0] == "\\set_vfo") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp = "RPRT 0\n";
|
|
|
|
|
|
|
|
// if number of arguments isn't correct or the VFO is not "VFO", return error
|
|
|
|
if (parts.size() != 2) {
|
2021-07-26 20:00:32 -04:00
|
|
|
resp = "RPRT 1\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
return;
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
|
|
|
|
if (parts[1] == "?") {
|
|
|
|
resp = "VFO\n";
|
|
|
|
}
|
|
|
|
else if (parts[1] != "VFO") {
|
|
|
|
resp = "RPRT 1\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
2021-07-28 04:21:51 +02:00
|
|
|
else if (parts[0] == "v" || parts[0] == "\\get_vfo") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp = "VFO\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
2021-08-20 20:16:37 +03:00
|
|
|
else if (parts[0] == "\\chk_vfo") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp = "CHKVFO 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
|
|
|
else if (parts[0] == "s") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp = "0\nVFOA\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
|
|
|
else if (parts[0] == "S") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp = "RPRT 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
2021-07-29 21:20:40 +02:00
|
|
|
else if (parts[0] == "AOS" || parts[0] == "\\recorder_start") {
|
2021-07-18 04:30:55 +02:00
|
|
|
std::lock_guard lck(recorderMtx);
|
2021-07-29 12:38:23 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// If not controlling the recorder, return
|
|
|
|
if (!recordingEnabled) {
|
|
|
|
resp = "RPRT 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
return;
|
|
|
|
}
|
2021-07-26 20:00:32 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// Send the command to the selected recorder
|
|
|
|
if (recorderType == RECORDER_TYPE_METEOR_DEMODULATOR) {
|
|
|
|
core::modComManager.callInterface(selectedRecorder, METEOR_DEMODULATOR_IFACE_CMD_START, NULL, NULL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
core::modComManager.callInterface(selectedRecorder, RECORDER_IFACE_CMD_START, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
2021-09-20 19:59:35 +02:00
|
|
|
// Respond with a success
|
2021-07-16 03:11:49 +02:00
|
|
|
resp = "RPRT 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
2021-07-29 21:20:40 +02:00
|
|
|
else if (parts[0] == "LOS" || parts[0] == "\\recorder_stop") {
|
2021-07-18 04:30:55 +02:00
|
|
|
std::lock_guard lck(recorderMtx);
|
2021-07-26 20:00:32 -04:00
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
// If not controlling the recorder, return
|
|
|
|
if (!recordingEnabled) {
|
|
|
|
resp = "RPRT 0\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send the command to the selected recorder
|
|
|
|
if (recorderType == RECORDER_TYPE_METEOR_DEMODULATOR) {
|
|
|
|
core::modComManager.callInterface(selectedRecorder, METEOR_DEMODULATOR_IFACE_CMD_STOP, NULL, NULL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
core::modComManager.callInterface(selectedRecorder, RECORDER_IFACE_CMD_STOP, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
2021-08-20 20:16:37 +03:00
|
|
|
// Respond with a success
|
2021-07-17 19:43:44 +02:00
|
|
|
resp = "RPRT 0\n";
|
2021-07-16 03:11:49 +02:00
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
2021-07-29 21:20:40 +02:00
|
|
|
else if (parts[0] == "q" || parts[0] == "\\quit") {
|
2021-07-16 01:49:41 +02:00
|
|
|
// Will close automatically
|
|
|
|
}
|
2021-11-05 19:42:49 +01:00
|
|
|
else if (parts[0] == "\\start") {
|
|
|
|
gui::mainWindow.setPlayState(true);
|
|
|
|
}
|
|
|
|
else if (parts[0] == "\\stop") {
|
|
|
|
gui::mainWindow.setPlayState(false);
|
|
|
|
}
|
2021-08-20 20:16:37 +03:00
|
|
|
else if (parts[0] == "\\dump_state") {
|
|
|
|
std::lock_guard lck(vfoMtx);
|
|
|
|
resp =
|
|
|
|
/* rigctl protocol version */
|
|
|
|
"0\n"
|
|
|
|
/* rigctl model */
|
|
|
|
"2\n"
|
|
|
|
/* ITU region */
|
|
|
|
"1\n"
|
|
|
|
/* RX/TX frequency ranges
|
|
|
|
* start, end, modes, low_power, high_power, vfo, ant
|
|
|
|
* start/end - Start/End frequency [Hz]
|
|
|
|
* modes - Bit field of RIG_MODE's (AM|AMS|CW|CWR|USB|LSB|FM|WFM)
|
|
|
|
* low_power/high_power - Lower/Higher RF power in mW,
|
|
|
|
* -1 for no power (ie. rx list)
|
|
|
|
* vfo - VFO list equipped with this range (RIG_VFO_A)
|
|
|
|
* ant - Antenna list equipped with this range, 0 means all
|
|
|
|
* FIXME: get limits from receiver
|
|
|
|
*/
|
|
|
|
"0.000000 10000000000.000000 0x2ef -1 -1 0x1 0x0\n"
|
|
|
|
/* End of RX frequency ranges. */
|
|
|
|
"0 0 0 0 0 0 0\n"
|
|
|
|
/* End of TX frequency ranges. The SDR++ is reciver only. */
|
|
|
|
"0 0 0 0 0 0 0\n"
|
|
|
|
/* Tuning steps: modes, tuning_step */
|
|
|
|
"0xef 1\n"
|
|
|
|
"0xef 0\n"
|
|
|
|
/* End of tuning steps */
|
|
|
|
"0 0\n"
|
|
|
|
/* Filter sizes: modes, width
|
|
|
|
* FIXME: get filter sizes from presets
|
|
|
|
*/
|
|
|
|
"0x82 500\n" /* CW | CWR normal */
|
|
|
|
"0x82 200\n" /* CW | CWR narrow */
|
|
|
|
"0x82 2000\n" /* CW | CWR wide */
|
|
|
|
"0x221 10000\n" /* AM | AMS | FM normal */
|
|
|
|
"0x221 5000\n" /* AM | AMS | FM narrow */
|
|
|
|
"0x221 20000\n" /* AM | AMS | FM wide */
|
|
|
|
"0x0c 2700\n" /* SSB normal */
|
|
|
|
"0x0c 1400\n" /* SSB narrow */
|
|
|
|
"0x0c 3900\n" /* SSB wide */
|
|
|
|
"0x40 160000\n" /* WFM normal */
|
|
|
|
"0x40 120000\n" /* WFM narrow */
|
|
|
|
"0x40 200000\n" /* WFM wide */
|
|
|
|
/* End of filter sizes */
|
|
|
|
"0 0\n"
|
|
|
|
/* max_rit */
|
|
|
|
"0\n"
|
|
|
|
/* max_xit */
|
|
|
|
"0\n"
|
|
|
|
/* max_ifshift */
|
|
|
|
"0\n"
|
|
|
|
/* Announces (bit field list) */
|
|
|
|
"0\n" /* RIG_ANN_NONE */
|
|
|
|
/* Preamp list in dB, 0 terminated */
|
|
|
|
"0\n"
|
|
|
|
/* Attenuator list in dB, 0 terminated */
|
|
|
|
"0\n"
|
|
|
|
/* Bit field list of get functions */
|
|
|
|
"0\n" /* RIG_FUNC_NONE */
|
|
|
|
/* Bit field list of set functions */
|
|
|
|
"0\n" /* RIG_FUNC_NONE */
|
|
|
|
/* Bit field list of get level */
|
|
|
|
"0x40000020\n" /* RIG_LEVEL_SQL | RIG_LEVEL_STRENGTH */
|
|
|
|
/* Bit field list of set level */
|
2021-12-19 22:11:44 +01:00
|
|
|
"0x20\n" /* RIG_LEVEL_SQL */
|
2021-08-20 20:16:37 +03:00
|
|
|
/* Bit field list of get parm */
|
|
|
|
"0\n" /* RIG_PARM_NONE */
|
|
|
|
/* Bit field list of set parm */
|
|
|
|
"0\n" /* RIG_PARM_NONE */;
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
2023-06-25 13:51:51 +09:00
|
|
|
// This get_powerstat stuff is a wordaround for WSJT-X 2.7.0
|
|
|
|
else if (parts[0] == "\\get_powerstat") {
|
|
|
|
resp = "1\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
}
|
2021-07-16 01:49:41 +02:00
|
|
|
else {
|
2021-07-29 12:38:23 -04:00
|
|
|
// If command is not recognized, return error
|
2023-02-25 18:12:34 +01:00
|
|
|
flog::error("Rigctl client sent invalid command: '{0}'", cmd);
|
2021-07-16 01:49:41 +02:00
|
|
|
resp = "RPRT 1\n";
|
|
|
|
client->write(resp.size(), (uint8_t*)resp.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string name;
|
|
|
|
bool enabled = true;
|
|
|
|
|
|
|
|
char hostname[1024];
|
|
|
|
int port = 4532;
|
|
|
|
uint8_t dataBuf[1024];
|
|
|
|
net::Listener listener;
|
|
|
|
net::Conn client;
|
|
|
|
|
|
|
|
std::string command = "";
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
EventHandler<std::string> modChangedHandler;
|
|
|
|
EventHandler<VFOManager::VFO*> vfoCreatedHandler;
|
|
|
|
EventHandler<std::string> vfoDeletedHandler;
|
|
|
|
|
|
|
|
std::vector<std::string> vfoNames;
|
|
|
|
std::string vfoNamesTxt;
|
|
|
|
std::vector<std::string> recorderNames;
|
|
|
|
std::string recorderNamesTxt;
|
|
|
|
std::mutex vfoMtx;
|
|
|
|
std::mutex recorderMtx;
|
|
|
|
|
|
|
|
std::string selectedVfo = "";
|
|
|
|
std::string selectedRecorder = "";
|
|
|
|
int vfoId = 0;
|
|
|
|
int recorderId = 0;
|
|
|
|
int recorderType = RECORDER_TYPE_RECORDER;
|
|
|
|
|
|
|
|
bool tuningEnabled = true;
|
|
|
|
bool recordingEnabled = false;
|
|
|
|
bool autoStart = false;
|
2021-07-16 01:49:41 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
MOD_EXPORT void _INIT_() {
|
2022-02-24 20:49:53 +01:00
|
|
|
config.setPath(core::args["root"].s() + "/rigctl_server_config.json");
|
2021-07-18 04:30:55 +02:00
|
|
|
config.load(json::object());
|
|
|
|
config.enableAutoSave();
|
2021-07-16 01:49:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_(std::string name) {
|
|
|
|
return new SigctlServerModule(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
|
|
|
|
delete (SigctlServerModule*)instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOD_EXPORT void _END_() {
|
2021-07-18 04:30:55 +02:00
|
|
|
config.disableAutoSave();
|
|
|
|
config.save();
|
2021-07-26 20:00:32 -04:00
|
|
|
}
|