mirror of
https://github.com/cjcliffe/CubicSDR.git
synced 2024-11-10 14:23:27 -05:00
567 lines
16 KiB
C++
567 lines
16 KiB
C++
#include "AppConfig.h"
|
|
#include "CubicSDR.h"
|
|
|
|
DeviceConfig::DeviceConfig() : deviceId("") {
|
|
ppm.store(0);
|
|
offset.store(0);
|
|
}
|
|
|
|
DeviceConfig::DeviceConfig(std::string deviceId) : DeviceConfig() {
|
|
this->deviceId = deviceId;
|
|
}
|
|
|
|
void DeviceConfig::setPPM(int ppm) {
|
|
this->ppm.store(ppm);
|
|
}
|
|
|
|
int DeviceConfig::getPPM() {
|
|
return ppm.load();
|
|
}
|
|
|
|
void DeviceConfig::setOffset(long long offset) {
|
|
this->offset.store(offset);
|
|
}
|
|
|
|
long long DeviceConfig::getOffset() {
|
|
return offset.load();
|
|
}
|
|
|
|
void DeviceConfig::setDeviceId(std::string deviceId) {
|
|
busy_lock.lock();
|
|
this->deviceId = deviceId;
|
|
busy_lock.unlock();
|
|
}
|
|
|
|
std::string DeviceConfig::getDeviceId() {
|
|
std::string tmp;
|
|
|
|
busy_lock.lock();
|
|
tmp = deviceId;
|
|
busy_lock.unlock();
|
|
|
|
return tmp;
|
|
}
|
|
|
|
void DeviceConfig::setDeviceName(std::string deviceName) {
|
|
busy_lock.lock();
|
|
this->deviceName = deviceName;
|
|
busy_lock.unlock();
|
|
}
|
|
|
|
std::string DeviceConfig::getDeviceName() {
|
|
std::string tmp;
|
|
|
|
busy_lock.lock();
|
|
tmp = (deviceName=="")?deviceId:deviceName;
|
|
busy_lock.unlock();
|
|
|
|
return tmp;
|
|
}
|
|
|
|
void DeviceConfig::save(DataNode *node) {
|
|
busy_lock.lock();
|
|
*node->newChild("id") = deviceId;
|
|
*node->newChild("name") = deviceName;
|
|
*node->newChild("ppm") = (int)ppm;
|
|
*node->newChild("offset") = offset;
|
|
DataNode *streamOptsNode = node->newChild("streamOpts");
|
|
for (ConfigSettings::const_iterator opt_i = streamOpts.begin(); opt_i != streamOpts.end(); opt_i++) {
|
|
*streamOptsNode->newChild(opt_i->first.c_str()) = opt_i->second;
|
|
}
|
|
DataNode *settingsNode = node->newChild("settings");
|
|
for (ConfigSettings::const_iterator set_i = settings.begin(); set_i != settings.end(); set_i++) {
|
|
*settingsNode->newChild(set_i->first.c_str()) = set_i->second;
|
|
}
|
|
DataNode *rigIFs = node->newChild("rig_ifs");
|
|
for (std::map<int, long long>::const_iterator rigIF_i = rigIF.begin(); rigIF_i != rigIF.end(); rigIF_i++) {
|
|
DataNode *ifNode = rigIFs->newChild("rig_if");
|
|
*ifNode->newChild("model") = rigIF_i->first;
|
|
*ifNode->newChild("sdr_if") = rigIF_i->second;
|
|
}
|
|
busy_lock.unlock();
|
|
}
|
|
|
|
void DeviceConfig::load(DataNode *node) {
|
|
busy_lock.lock();
|
|
if (node->hasAnother("name")) {
|
|
deviceName = node->getNext("name")->element()->toString();
|
|
}
|
|
if (node->hasAnother("ppm")) {
|
|
DataNode *ppm_node = node->getNext("ppm");
|
|
int ppmValue = 0;
|
|
ppm_node->element()->get(ppmValue);
|
|
setPPM(ppmValue);
|
|
std::cout << "Loaded PPM for device '" << deviceId << "' at " << ppmValue << "ppm" << std::endl;
|
|
}
|
|
if (node->hasAnother("offset")) {
|
|
DataNode *offset_node = node->getNext("offset");
|
|
long long offsetValue = 0;
|
|
offset_node->element()->get(offsetValue);
|
|
setOffset(offsetValue);
|
|
std::cout << "Loaded offset for device '" << deviceId << "' at " << offsetValue << "Hz" << std::endl;
|
|
}
|
|
if (node->hasAnother("streamOpts")) {
|
|
DataNode *streamOptsNode = node->getNext("streamOpts");
|
|
for (int i = 0, iMax = streamOptsNode->numChildren(); i<iMax; i++) {
|
|
DataNode *streamOptNode = streamOptsNode->child(i);
|
|
std::string keyName = streamOptNode->getName();
|
|
std::string strSettingValue = streamOptNode->element()->toString();
|
|
|
|
if (keyName != "") {
|
|
setStreamOpt(keyName, strSettingValue);
|
|
}
|
|
}
|
|
}
|
|
if (node->hasAnother("settings")) {
|
|
DataNode *settingsNode = node->getNext("settings");
|
|
for (int i = 0, iMax = settingsNode->numChildren(); i<iMax; i++) {
|
|
DataNode *settingNode = settingsNode->child(i);
|
|
std::string keyName = settingNode->getName();
|
|
std::string strSettingValue = settingNode->element()->toString();
|
|
|
|
if (keyName != "") {
|
|
setSetting(keyName, strSettingValue);
|
|
}
|
|
}
|
|
}
|
|
if (node->hasAnother("rig_ifs")) {
|
|
DataNode *rigIFNodes = node->getNext("rig_ifs");
|
|
while (rigIFNodes->hasAnother("rig_if")) {
|
|
DataNode *rigIFNode = rigIFNodes->getNext("rig_if");
|
|
if (rigIFNode->hasAnother("model") && rigIFNode->hasAnother("sdr_if")) {
|
|
int load_model;
|
|
long long load_freq;
|
|
|
|
rigIFNode->getNext("model")->element()->get(load_model);
|
|
rigIFNode->getNext("sdr_if")->element()->get(load_freq);
|
|
|
|
rigIF[load_model] = load_freq;
|
|
}
|
|
}
|
|
}
|
|
busy_lock.unlock();
|
|
}
|
|
|
|
void DeviceConfig::setStreamOpts(ConfigSettings opts) {
|
|
streamOpts = opts;
|
|
}
|
|
|
|
ConfigSettings DeviceConfig::getStreamOpts() {
|
|
return streamOpts;
|
|
}
|
|
|
|
void DeviceConfig::setStreamOpt(std::string key, std::string value) {
|
|
streamOpts[key] = value;
|
|
}
|
|
|
|
std::string DeviceConfig::getStreamOpt(std::string key, std::string defaultValue) {
|
|
if (streamOpts.find(key) == streamOpts.end()) {
|
|
return defaultValue;
|
|
}
|
|
|
|
return streamOpts[key];
|
|
}
|
|
|
|
void DeviceConfig::setSettings(ConfigSettings settings) {
|
|
this->settings = settings;
|
|
}
|
|
|
|
void DeviceConfig::setSetting(std::string key, std::string value) {
|
|
this->settings[key] = value;
|
|
}
|
|
|
|
std::string DeviceConfig::getSetting(std::string key, std::string defaultValue) {
|
|
if (settings.find(key) == settings.end()) {
|
|
return defaultValue;
|
|
}
|
|
return settings[key];
|
|
}
|
|
|
|
ConfigSettings DeviceConfig::getSettings() {
|
|
return settings;
|
|
}
|
|
|
|
void DeviceConfig::setRigIF(int rigType, long long freq) {
|
|
rigIF[rigType] = freq;
|
|
}
|
|
|
|
long long DeviceConfig::getRigIF(int rigType) {
|
|
if (rigIF.find(rigType) != rigIF.end()) {
|
|
return rigIF[rigType];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
AppConfig::AppConfig() : configName("") {
|
|
winX.store(0);
|
|
winY.store(0);
|
|
winW.store(0);
|
|
winH.store(0);
|
|
winMax.store(false);
|
|
themeId.store(0);
|
|
snap.store(1);
|
|
centerFreq.store(100000000);
|
|
waterfallLinesPerSec.store(DEFAULT_WATERFALL_LPS);
|
|
spectrumAvgSpeed.store(0.65f);
|
|
#ifdef USE_HAMLIB
|
|
rigModel.store(1);
|
|
rigRate.store(57600);
|
|
rigPort = "/dev/ttyUSB0";
|
|
#endif
|
|
}
|
|
|
|
DeviceConfig *AppConfig::getDevice(std::string deviceId) {
|
|
if (deviceConfig.find(deviceId) == deviceConfig.end()) {
|
|
deviceConfig[deviceId] = new DeviceConfig();
|
|
}
|
|
DeviceConfig *conf = deviceConfig[deviceId];
|
|
conf->setDeviceId(deviceId);
|
|
return conf;
|
|
}
|
|
|
|
std::string AppConfig::getConfigDir() {
|
|
std::string dataDir = wxStandardPaths::Get().GetUserDataDir().ToStdString();
|
|
|
|
bool mkStatus = false;
|
|
|
|
if (!wxDir::Exists(dataDir)) {
|
|
mkStatus = wxDir::Make(dataDir);
|
|
} else {
|
|
mkStatus = true;
|
|
}
|
|
|
|
if (!mkStatus) {
|
|
std::cout << "Warning, unable to initialize user data directory." << std::endl;
|
|
}
|
|
|
|
return dataDir;
|
|
}
|
|
|
|
|
|
void AppConfig::setWindow(wxPoint winXY, wxSize winWH) {
|
|
winX.store(winXY.x);
|
|
winY.store(winXY.y);
|
|
winW.store(winWH.x);
|
|
winH.store(winWH.y);
|
|
}
|
|
|
|
void AppConfig::setWindowMaximized(bool max) {
|
|
winMax.store(max);
|
|
}
|
|
|
|
bool AppConfig::getWindowMaximized() {
|
|
return winMax.load();
|
|
}
|
|
|
|
wxRect *AppConfig::getWindow() {
|
|
wxRect *r = NULL;
|
|
if (winH.load() && winW.load()) {
|
|
r = new wxRect(winX.load(),winY.load(),winW.load(),winH.load());
|
|
}
|
|
return r;
|
|
}
|
|
|
|
void AppConfig::setTheme(int themeId) {
|
|
this->themeId.store(themeId);
|
|
}
|
|
|
|
int AppConfig::getTheme() {
|
|
return themeId.load();
|
|
}
|
|
|
|
|
|
void AppConfig::setSnap(long long snapVal) {
|
|
this->snap.store(snapVal);
|
|
}
|
|
|
|
long long AppConfig::getSnap() {
|
|
return snap.load();
|
|
}
|
|
|
|
void AppConfig::setCenterFreq(long long freqVal) {
|
|
centerFreq.store(freqVal);
|
|
}
|
|
|
|
long long AppConfig::getCenterFreq() {
|
|
return centerFreq.load();
|
|
}
|
|
|
|
|
|
void AppConfig::setWaterfallLinesPerSec(int lps) {
|
|
waterfallLinesPerSec.store(lps);
|
|
}
|
|
|
|
int AppConfig::getWaterfallLinesPerSec() {
|
|
return waterfallLinesPerSec.load();
|
|
}
|
|
|
|
void AppConfig::setSpectrumAvgSpeed(float avgSpeed) {
|
|
spectrumAvgSpeed.store(avgSpeed);
|
|
}
|
|
|
|
float AppConfig::getSpectrumAvgSpeed() {
|
|
return spectrumAvgSpeed.load();
|
|
}
|
|
|
|
void AppConfig::setManualDevices(std::vector<SDRManualDef> manuals) {
|
|
manualDevices = manuals;
|
|
}
|
|
|
|
std::vector<SDRManualDef> AppConfig::getManualDevices() {
|
|
return manualDevices;
|
|
}
|
|
|
|
void AppConfig::setConfigName(std::string configName) {
|
|
this->configName = configName;
|
|
}
|
|
|
|
std::string AppConfig::getConfigFileName(bool ignoreName) {
|
|
std::string cfgFileDir = getConfigDir();
|
|
|
|
wxFileName cfgFile;
|
|
if (configName.length() && !ignoreName) {
|
|
std::string tempFn("config-");
|
|
tempFn.append(configName);
|
|
tempFn.append(".xml");
|
|
cfgFile = wxFileName(cfgFileDir, tempFn);
|
|
} else {
|
|
cfgFile = wxFileName(cfgFileDir, "config.xml");
|
|
}
|
|
|
|
std::string cfgFileName = cfgFile.GetFullPath(wxPATH_NATIVE).ToStdString();
|
|
|
|
return cfgFileName;
|
|
}
|
|
|
|
bool AppConfig::save() {
|
|
DataTree cfg;
|
|
|
|
cfg.rootNode()->setName("cubicsdr_config");
|
|
|
|
if (winW.load() && winH.load()) {
|
|
DataNode *window_node = cfg.rootNode()->newChild("window");
|
|
|
|
*window_node->newChild("x") = winX.load();
|
|
*window_node->newChild("y") = winY.load();
|
|
*window_node->newChild("w") = winW.load();
|
|
*window_node->newChild("h") = winH.load();
|
|
|
|
*window_node->newChild("max") = winMax.load();
|
|
*window_node->newChild("theme") = themeId.load();
|
|
*window_node->newChild("snap") = snap.load();
|
|
*window_node->newChild("center_freq") = centerFreq.load();
|
|
*window_node->newChild("waterfall_lps") = waterfallLinesPerSec.load();
|
|
*window_node->newChild("spectrum_avg") = spectrumAvgSpeed.load();
|
|
}
|
|
|
|
DataNode *devices_node = cfg.rootNode()->newChild("devices");
|
|
|
|
std::map<std::string, DeviceConfig *>::iterator device_config_i;
|
|
for (device_config_i = deviceConfig.begin(); device_config_i != deviceConfig.end(); device_config_i++) {
|
|
DataNode *device_node = devices_node->newChild("device");
|
|
device_config_i->second->save(device_node);
|
|
}
|
|
|
|
if (manualDevices.size()) {
|
|
DataNode *manual_node = cfg.rootNode()->newChild("manual_devices");
|
|
for (std::vector<SDRManualDef>::const_iterator i = manualDevices.begin(); i != manualDevices.end(); i++) {
|
|
DataNode *rig_node = manual_node->newChild("device");
|
|
*rig_node->newChild("factory") = i->factory;
|
|
*rig_node->newChild("params") = i->params;
|
|
}
|
|
}
|
|
|
|
#ifdef USE_HAMLIB
|
|
DataNode *rig_node = cfg.rootNode()->newChild("rig");
|
|
*rig_node->newChild("model") = rigModel.load();
|
|
*rig_node->newChild("rate") = rigRate.load();
|
|
*rig_node->newChild("port") = rigPort;
|
|
#endif
|
|
|
|
std::string cfgFileName = getConfigFileName();
|
|
|
|
if (!cfg.SaveToFileXML(cfgFileName)) {
|
|
std::cout << "Error saving :: configuration file '" << cfgFileName << "' is not writable!" << std::endl;
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AppConfig::load() {
|
|
DataTree cfg;
|
|
std::string cfgFileDir = getConfigDir();
|
|
|
|
std::string cfgFileName = getConfigFileName();
|
|
wxFileName cfgFile = wxFileName(cfgFileName);
|
|
|
|
if (!cfgFile.Exists()) {
|
|
if (configName.length()) {
|
|
wxFileName baseConfig = wxFileName(getConfigFileName(true));
|
|
if (baseConfig.Exists()) {
|
|
std::string baseConfigFileName = baseConfig.GetFullPath(wxPATH_NATIVE).ToStdString();
|
|
std::cout << "Creating new configuration file '" << cfgFileName << "' by copying '" << baseConfigFileName << "'..";
|
|
wxCopyFile(baseConfigFileName, cfgFileName);
|
|
if (!cfgFile.Exists()) {
|
|
std::cout << "failed." << std::endl;
|
|
return true;
|
|
}
|
|
std::cout << "ok." << std::endl;
|
|
} else {
|
|
return true;
|
|
}
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (cfgFile.IsFileReadable()) {
|
|
std::cout << "Loading:: configuration file '" << cfgFileName << "'" << std::endl;
|
|
|
|
cfg.LoadFromFileXML(cfgFileName);
|
|
} else {
|
|
std::cout << "Error loading:: configuration file '" << cfgFileName << "' is not readable!" << std::endl;
|
|
return false;
|
|
}
|
|
|
|
if (cfg.rootNode()->hasAnother("window")) {
|
|
int x,y,w,h;
|
|
int max;
|
|
|
|
DataNode *win_node = cfg.rootNode()->getNext("window");
|
|
|
|
if (win_node->hasAnother("w") && win_node->hasAnother("h") && win_node->hasAnother("x") && win_node->hasAnother("y")) {
|
|
win_node->getNext("x")->element()->get(x);
|
|
win_node->getNext("y")->element()->get(y);
|
|
win_node->getNext("w")->element()->get(w);
|
|
win_node->getNext("h")->element()->get(h);
|
|
|
|
winX.store(x);
|
|
winY.store(y);
|
|
winW.store(w);
|
|
winH.store(h);
|
|
}
|
|
|
|
if (win_node->hasAnother("max")) {
|
|
win_node->getNext("max")->element()->get(max);
|
|
winMax.store(max?true:false);
|
|
}
|
|
|
|
if (win_node->hasAnother("theme")) {
|
|
int theme;
|
|
win_node->getNext("theme")->element()->get(theme);
|
|
themeId.store(theme);
|
|
}
|
|
|
|
if (win_node->hasAnother("snap")) {
|
|
long long snapVal;
|
|
win_node->getNext("snap")->element()->get(snapVal);
|
|
snap.store(snapVal);
|
|
}
|
|
|
|
if (win_node->hasAnother("center_freq")) {
|
|
long long freqVal;
|
|
win_node->getNext("center_freq")->element()->get(freqVal);
|
|
centerFreq.store(freqVal);
|
|
}
|
|
|
|
if (win_node->hasAnother("waterfall_lps")) {
|
|
int lpsVal;
|
|
win_node->getNext("waterfall_lps")->element()->get(lpsVal);
|
|
waterfallLinesPerSec.store(lpsVal);
|
|
}
|
|
|
|
if (win_node->hasAnother("spectrum_avg")) {
|
|
float avgVal;
|
|
win_node->getNext("spectrum_avg")->element()->get(avgVal);
|
|
spectrumAvgSpeed.store(avgVal);
|
|
}
|
|
}
|
|
|
|
if (cfg.rootNode()->hasAnother("devices")) {
|
|
DataNode *devices_node = cfg.rootNode()->getNext("devices");
|
|
|
|
while (devices_node->hasAnother("device")) {
|
|
DataNode *device_node = devices_node->getNext("device");
|
|
if (device_node->hasAnother("id")) {
|
|
std::string deviceId = device_node->getNext("id")->element()->toString();
|
|
|
|
getDevice(deviceId)->load(device_node);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (cfg.rootNode()->hasAnother("manual_devices")) {
|
|
DataNode *manuals_node = cfg.rootNode()->getNext("manual_devices");
|
|
|
|
while (manuals_node->hasAnother("device")) {
|
|
DataNode *manual_node = manuals_node->getNext("device");
|
|
if (manual_node->hasAnother("factory") && manual_node->hasAnother("params")) {
|
|
SDRManualDef mdef;
|
|
|
|
mdef.factory = manual_node->getNext("factory")->element()->toString();
|
|
mdef.params = manual_node->getNext("params")->element()->toString();
|
|
|
|
manualDevices.push_back(mdef);
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef USE_HAMLIB
|
|
if (cfg.rootNode()->hasAnother("rig")) {
|
|
DataNode *rig_node = cfg.rootNode()->getNext("rig");
|
|
|
|
if (rig_node->hasAnother("model")) {
|
|
int loadModel;
|
|
rig_node->getNext("model")->element()->get(loadModel);
|
|
rigModel.store(loadModel?loadModel:1);
|
|
}
|
|
if (rig_node->hasAnother("rate")) {
|
|
int loadRate;
|
|
rig_node->getNext("rate")->element()->get(loadRate);
|
|
rigRate.store(loadRate?loadRate:57600);
|
|
}
|
|
if (rig_node->hasAnother("port")) {
|
|
rigPort = rig_node->getNext("port")->element()->toString();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AppConfig::reset() {
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
#if USE_HAMLIB
|
|
|
|
int AppConfig::getRigModel() {
|
|
return rigModel.load();
|
|
}
|
|
|
|
void AppConfig::setRigModel(int rigModel) {
|
|
this->rigModel.store(rigModel);
|
|
}
|
|
|
|
int AppConfig::getRigRate() {
|
|
return rigRate.load();
|
|
}
|
|
|
|
void AppConfig::setRigRate(int rigRate) {
|
|
this->rigRate.store(rigRate);
|
|
}
|
|
|
|
std::string AppConfig::getRigPort() {
|
|
return rigPort;
|
|
}
|
|
|
|
void AppConfig::setRigPort(std::string rigPort) {
|
|
this->rigPort = rigPort;
|
|
}
|
|
|
|
#endif
|