LibreVNA/Software/PC_Application/LibreVNA-GUI/Device/deviceusblogview.cpp
2022-11-01 01:12:04 +01:00

320 lines
15 KiB
C++

#include "deviceusblogview.h"
#include "ui_deviceusblogview.h"
#include "CustomWidgets/informationbox.h"
#include "unit.h"
#include <fstream>
#include <iomanip>
#include <QPushButton>
#include <QFileDialog>
#include <QMessageBox>
using namespace std;
DeviceUSBLogView::DeviceUSBLogView(QWidget *parent) :
QDialog(parent),
ui(new Ui::DeviceUSBLogView)
{
setAttribute(Qt::WA_DeleteOnClose);
ui->setupUi(this);
// connect(&log, &DeviceUSBLog::entryAdded, this, &DeviceUSBLogView::addEntry);
connect(ui->buttonBox->button(QDialogButtonBox::Reset), &QPushButton::clicked, [=](){
DeviceUSBLog::getInstance().reset();
updateTree();
});
connect(ui->buttonBox->button(QDialogButtonBox::Save), &QPushButton::clicked, [=](){
QString filename = QFileDialog::getSaveFileName(nullptr, "Load USB log data", "", "USB log files (*.usblog)", nullptr, QFileDialog::DontUseNativeDialog);
if(filename.isEmpty()) {
// aborted selection
return;
}
if(!filename.endsWith(".usblog")) {
filename.append(".usblog");
}
ofstream file;
file.open(filename.toStdString());
file << setw(1) << DeviceUSBLog::getInstance().toJSON() << endl;
file.close();
});
connect(ui->buttonBox->button(QDialogButtonBox::Open), &QPushButton::clicked, [=](){
QString filename = QFileDialog::getOpenFileName(nullptr, "Load USB log data", "", "USB log files (*.usblog)", nullptr, QFileDialog::DontUseNativeDialog);
if(filename.isEmpty()) {
// aborted selection
return;
}
ifstream file;
file.open(filename.toStdString());
if(!file.is_open()) {
qWarning() << "Unable to open file:" << filename;
return;
}
nlohmann::json j;
try {
// TODO this can take a long time, move to thread
file >> j;
} catch (exception &e) {
InformationBox::ShowError("Error", "Failed to parse the USB log file (" + QString(e.what()) + ")");
qWarning() << "Parsing of USB log file failed: " << e.what();
file.close();
return;
}
file.close();
DeviceUSBLog::getInstance().fromJSON(j);
updateTree();
});
updateTree();
}
DeviceUSBLogView::~DeviceUSBLogView()
{
delete ui;
}
void DeviceUSBLogView::updateTree()
{
auto &log = DeviceUSBLog::getInstance();
ui->tree->clear();
ui->tree->setColumnCount(4);
ui->tree->setHeaderLabels({"Timestamp","Source","Type","Content"});
unsigned int i=0;
try {
// will throw once all entries have been added (number of entries may change while the loop is running)
for(i=0;;i++) {
addEntry(log.getEntry(i));
}
} catch (...) {
}
QString status = "Log contains "+QString::number(i) + " entries, using ";
status += Unit::ToString(log.getUsedStorageSize(), "B", " kMG") + " (maximum: "+Unit::ToString(log.getMaxStorageSize(), "B", " kMG")+")";
ui->status->setText(status);
}
void DeviceUSBLogView::addEntry(const DeviceUSBLog::LogEntry &e)
{
auto item = new QTreeWidgetItem;
item->setData(0, Qt::DisplayRole, e.timestamp.toString(Qt::DateFormat::ISODateWithMs));
item->setData(1, Qt::DisplayRole, e.serial.size() > 0 ? e.serial : "LibreVNA-GUI");
item->setData(2, Qt::DisplayRole, e.type == DeviceUSBLog::LogEntry::Type::Packet ? "Packet" : "Invalid bytes");
if(e.type == DeviceUSBLog::LogEntry::Type::Packet) {
item->setData(2, Qt::DisplayRole, "Packet");
static const QStringList packetNames = {"None", "Datapoint", "SweepSettings", "ManualStatusV1", "ManualControlV1", "DeviceInfo", "FirmwarePacket", "Ack",
"ClearFlash", "PerformFirmwareUpdate", "Nack", "Reference", "Generator", "SpectrumAnalyzerSettings",
"SpectrumAnalyzerResult", "RequestDeviceInfo", "RequestSourceCal", "RequestReceiverCal", "SourceCalPoint",
"ReceiverCalPoint", "SetIdle", "RequestFrequencyCorrection", "FrequencyCorrection", "RequestAcquisitionFrequencySettings",
"AcquisitionFrequencySettings", "DeviceStatusV1", "RequestDeviceStatus", "VNADatapoint", "SetTrigger", "ClearTrigger"};
item->setData(3, Qt::DisplayRole, "Type "+QString::number((int)e.p->type)+"("+packetNames[(int)e.p->type]+")");
auto addDouble = [=](QTreeWidgetItem *parent, QString name, double value, QString unit = "", int precision = 8) {
auto subitem = new QTreeWidgetItem;
subitem->setData(2, Qt::DisplayRole, name);
subitem->setData(3, Qt::DisplayRole, Unit::ToString(value, unit, "pnum kMG", precision));
parent->addChild(subitem);
};
auto addInteger = [=](QTreeWidgetItem *parent, QString name, int value) {
auto subitem = new QTreeWidgetItem;
subitem->setData(2, Qt::DisplayRole, name);
subitem->setData(3, Qt::DisplayRole, value);
parent->addChild(subitem);
};
auto addBool = [=](QTreeWidgetItem *parent, QString name, bool value) {
auto subitem = new QTreeWidgetItem;
subitem->setData(2, Qt::DisplayRole, name);
subitem->setData(3, Qt::DisplayRole, value ? "true" : "false");
parent->addChild(subitem);
};
auto addEnum = [=](QTreeWidgetItem *parent, QString name, int value, QStringList names) {
auto subitem = new QTreeWidgetItem;
subitem->setData(2, Qt::DisplayRole, name);
subitem->setData(3, Qt::DisplayRole, names[value]);
parent->addChild(subitem);
};
auto addString = [=](QTreeWidgetItem *parent, QString name, QString value) {
auto subitem = new QTreeWidgetItem;
subitem->setData(2, Qt::DisplayRole, name);
subitem->setData(3, Qt::DisplayRole, value);
parent->addChild(subitem);
};
switch(e.p->type) {
case Protocol::PacketType::SweepSettings: {
Protocol::SweepSettings s = e.p->settings;
addDouble(item, "Start frequency", s.f_start, "Hz");
addDouble(item, "Stop frequency", s.f_stop, "Hz");
addInteger(item, "Points", s.points);
addDouble(item, "IF bandwidth", s.if_bandwidth, "Hz");
addDouble(item, "Start power", (double) s.cdbm_excitation_start / 100.0, "dBm");
addDouble(item, "Stop power", (double) s.cdbm_excitation_stop / 100.0, "dBm");
addEnum(item, "Sync mode", s.syncMode, {"no synchronization", "USB", "Ext. reference", "Ext. trigger"});
addBool(item, "Sync Master", s.syncMaster);
addBool(item, "Suppress peaks", s.suppressPeaks);
addBool(item, "Fixed power setting", s.fixedPowerSetting);
addBool(item, "Logarithmic sweep", s.logSweep);
addInteger(item, "Stages", s.stages);
addInteger(item, "Port 1 stage", s.port1Stage);
addInteger(item, "Port 2 stage", s.port2Stage);
}
break;
case Protocol::PacketType::Reference: {
Protocol::ReferenceSettings s = e.p->reference;
addDouble(item, "Output frequency", s.ExtRefOuputFreq, "Hz");
addBool(item, "Switch automatically", s.AutomaticSwitch);
addBool(item, "Use external reference", s.UseExternalRef);
}
break;
case Protocol::PacketType::Generator: {
Protocol::GeneratorSettings s = e.p->generator;
addDouble(item, "Frequency", s.frequency, "Hz");
addDouble(item, "Power", (double) s.cdbm_level / 100.0, "dBm");
addBool(item, "Use source calibration", s.applyAmplitudeCorrection);
addInteger(item, "Active port", s.activePort);
}
break;
case Protocol::PacketType::DeviceStatusV1: {
Protocol::DeviceStatusV1 s = e.p->statusV1;
addBool(item, "External reference available", s.extRefAvailable);
addBool(item, "External reference in use", s.extRefInUse);
addBool(item, "FPGA configured", s.FPGA_configured);
addBool(item, "Source locked", s.source_locked);
addBool(item, "1.LO locked", s.LO1_locked);
addBool(item, "ADC overload", s.ADC_overload);
addBool(item, "Unlevel", s.unlevel);
addInteger(item, "Source temperature", s.temp_source);
addInteger(item, "1.LO temperature", s.temp_LO1);
addInteger(item, "MCU temperature", s.temp_MCU);
}
break;
case Protocol::PacketType::DeviceInfo: {
Protocol::DeviceInfo s = e.p->info;
addInteger(item, "Protocol version", s.ProtocolVersion);
addInteger(item, "Firmware major", s.FW_major);
addInteger(item, "Firmware minor", s.FW_minor);
addInteger(item, "Firmware patch", s.FW_patch);
addInteger(item, "Hardware version", s.hardware_version);
addString(item, "Hardware version", QString(s.HW_Revision));
addDouble(item, "Minimum frequency", s.limits_minFreq, "Hz");
addDouble(item, "Maximum frequency", s.limits_maxFreq, "Hz");
addDouble(item, "Minimum IF bandwidth", s.limits_minIFBW, "Hz");
addDouble(item, "Maximum IF bandwidht", s.limits_maxIFBW, "Hz");
addInteger(item, "Maximum points", s.limits_maxPoints);
addDouble(item, "Minimum power", (double) s.limits_cdbm_min / 100.0, "dBm");
addDouble(item, "Maximum power", (double) s.limits_cdbm_max / 100.0, "dBm");
addDouble(item, "Minimum RBW", s.limits_minRBW, "Hz");
addDouble(item, "Maximum RBW", s.limits_maxRBW, "Hz");
addInteger(item, "Maximum amplitude points", s.limits_maxAmplitudePoints);
addDouble(item, "Maximum harmonic frequency", s.limits_maxFreqHarmonic, "Hz");
}
break;
case Protocol::PacketType::ManualControlV1:
case Protocol::PacketType::ManualStatusV1:
// TODO
break;
case Protocol::PacketType::SpectrumAnalyzerSettings: {
Protocol::SpectrumAnalyzerSettings s = e.p->spectrumSettings;
addDouble(item, "Start frequency", s.f_start, "Hz");
addDouble(item, "Stop frequency", s.f_stop, "Hz");
addInteger(item, "Points", s.pointNum);
addDouble(item, "Resolution bandwidth", s.RBW, "Hz");
addEnum(item, "Window", s.syncMode, {"None", "Kaiser", "Hann", "Flattop"});
addBool(item, "Signal ID", s.SignalID);
addBool(item, "Use DFT", s.UseDFT);
addBool(item, "Use receiver calibration", s.applyReceiverCorrection);
addBool(item, "Use source calibration", s.applySourceCorrection);
addEnum(item, "Detector", s.Detector, {"Positive peak", "Negative peak", "Sample", "Normal", "Average"});
addEnum(item, "Sync mode", s.syncMode, {"no synchronization", "USB", "Ext. reference", "Ext. trigger"});
addBool(item, "Sync Master", s.syncMaster);
addBool(item, "Enable tracking generator", s.trackingGenerator);
addInteger(item, "Tracking generator port", s.trackingGeneratorPort+1);
addDouble(item, "Tracking generator offset", s.trackingGeneratorOffset, "Hz");
addDouble(item, "Tracking generator power", (double) s.trackingPower / 100.0, "dBm");
}
break;
case Protocol::PacketType::VNADatapoint: {
// TODO VNAdatapoint will already be deleted. Create deep copy when creating the log entry?
// Protocol::VNADatapoint<32>* s = e.p->VNAdatapoint;
// addInteger(item, "Point number", s->pointNum);
// addDouble(item, "Frequency/time", s->frequency, "Hz");
// addDouble(item, "Power", (double) s->cdBm / 100.0, "dBm");
// for(unsigned int i=0;i<s->getNumValues();i++) {
// auto v = s->getValue(i);
// vector<int> ports;
// if(v.flags & 0x01) {
// ports.push_back(1);
// }
// if(v.flags & 0x02) {
// ports.push_back(2);
// }
// if(v.flags & 0x04) {
// ports.push_back(3);
// }
// if(v.flags & 0x08) {
// ports.push_back(4);
// }
// bool reference = v.flags & 0x10;
// int stage = v.flags >> 5;
// auto vitem = new QTreeWidgetItem;
// vitem->setData(2, Qt::DisplayRole, "Measurement "+QString::number(i+1));
// vitem->setData(3, Qt::DisplayRole, "Real: "+QString::number(v.value.real())+" Imag: "+QString::number(v.value.imag()));
// addInteger(vitem, "Stage", stage);
// addBool(vitem, "Reference", reference);
// QString sports = QString::number(ports.front());
// for(unsigned int j=1;j<ports.size();j++) {
// sports += ","+QString::number(ports[j]);
// }
// addString(vitem, "Ports", sports);
// item->addChild(vitem);
// }
}
break;
case Protocol::PacketType::SpectrumAnalyzerResult: {
Protocol::SpectrumAnalyzerResult s = e.p->spectrumResult;
addDouble(item, "Port 1 level", s.port1);
addDouble(item, "Port 2 level", s.port2);
addDouble(item, "Frequency/time", s.frequency, "Hz");
addInteger(item, "Point number", s.pointNum);
}
break;
case Protocol::PacketType::SourceCalPoint:
case Protocol::PacketType::ReceiverCalPoint: {
Protocol::AmplitudeCorrectionPoint s = e.p->amplitudePoint;
addInteger(item, "Total points", s.totalPoints);
addInteger(item, "Point number", s.pointNum);
addDouble(item, "Frequency", s.freq, "Hz");
addInteger(item, "Port 1 correction", s.port1);
addInteger(item, "Port 2 correction", s.port2);
}
break;
case Protocol::PacketType::FrequencyCorrection: {
Protocol::FrequencyCorrection s = e.p->frequencyCorrection;
addDouble(item, "ppm", s.ppm, "");
}
break;
case Protocol::PacketType::AcquisitionFrequencySettings: {
Protocol::AcquisitionFrequencySettings s = e.p->acquisitionFrequencySettings;
addDouble(item, "1.IF", s.IF1, "Hz");
addInteger(item, "ADC prescaler", s.ADCprescaler);
addInteger(item, "DFT phase increment", s.DFTphaseInc);
}
break;
default:
break;
}
} else {
item->setData(2, Qt::DisplayRole, "Invalid bytes");
item->setData(3, Qt::DisplayRole, QString::number(e.bytes.size())+ " bytes");
for(auto b : e.bytes) {
auto subitem = new QTreeWidgetItem;
subitem->setData(3, Qt::DisplayRole, "0x"+QString::number(b, 16));
item->addChild(subitem);
}
}
ui->tree->addTopLevelItem(item);
}