Files
Tksi125Monitor/sources/DataController.cpp
2024-01-22 18:30:04 +09:00

288 lines
8.4 KiB
C++

#include "DataController.h"
#include <QTimer>
#include <QtEndian>
#include <QDateTime>
#include <QSettings>
#include <QSerialPort>
#include <QCoreApplication>
#include <QDebug>
DataController::DataController(QObject* parent)
: QObject{parent}
, serialPort_(new QSerialPort)
{
connect(serialPort_.data(), &QSerialPort::readyRead, this, &DataController::serialReadyRead);
connect(serialPort_.data(), &QSerialPort::errorOccurred, this, &DataController::serialError);
// for correct qApp->exit();
QTimer::singleShot(0, this, [this]{ openSerialPort(); });
}
DataController::~DataController()
{
}
QObject* DataController::qmlInstance(QQmlEngine* /*engine*/, QJSEngine* /*scriptEngine*/)
{
return new DataController;
}
void DataController::serialReadyRead()
{
const auto data = serialPort_->readAll();
qDebug() << "Serial data received (HEX):" << QDateTime::currentDateTime() << data.toHex();
serialData_ += data;
parseSerialData();
}
void DataController::serialError()
{
const auto error = serialPort_->error();
if (error == QSerialPort::NoError)
{
return;
}
qCritical() << "A serial port error has occurred. Port:" << serialPort_->portName() << "Error:" << serialPort_->errorString();
qApp->exit(1);
}
QVariantMap DataController::readSettings()
{
QSettings settings("settings.ini", QSettings::IniFormat);
const auto keys = settings.allKeys();
if (keys.isEmpty())
{
qCritical() << "File settings.ini is empty. Please put the file next to the executable file.";
qApp->exit(1);
}
QVariantMap settingsMap;
for (const auto& key: keys)
{
settingsMap[key] = settings.value(key);
}
return settingsMap;
}
void DataController::openSerialPort()
{
const auto settings = readSettings();
if (settings.isEmpty())
{
return;
}
const auto portName = settings["name"].toString();
// for testing
if (portName == "fake")
{
mockSerialData();
parseSerialData();
return;
}
serialPort_->setPortName(portName);
if (settings.contains("baudRate"))
{
serialPort_->setBaudRate(settings["baudRate"].toInt());
}
if (settings.contains("dataBits"))
{
serialPort_->setDataBits((QSerialPort::DataBits)settings["dataBits"].toInt());
}
if (settings.contains("parity"))
{
serialPort_->setParity((QSerialPort::Parity)settings["parity"].toInt());
}
if (settings.contains("stopBits"))
{
serialPort_->setStopBits((QSerialPort::StopBits)settings["stopBits"].toInt());
}
if (settings.contains("flowControl"))
{
serialPort_->setFlowControl((QSerialPort::FlowControl)settings["flowControl"].toInt());
}
if (!serialPort_->open(QIODevice::ReadOnly))
{
qCritical() << "Can't open serial port. Port:" << serialPort_->portName() << "Error:" << serialPort_->errorString();
qApp->exit(1);
}
}
void DataController::parseSerialData()
{
const char* startBytes = "\xD0\x0D";
const auto startPosition = serialData_.indexOf(startBytes);
if (startPosition == -1)
{
serialData_.clear();
return;
}
if (startPosition > 0)
{
serialData_ = serialData_.mid(startPosition);
}
if (serialData_.size() < 35)
{
return;
}
auto extractInt = [](const QByteArray& array, int start, int length) -> uint32_t
{
uint32_t result = 0;
if (length > 4)
{
return result;
}
// reversed byte order
for (int i = 0; i < length; ++i)
{
memcpy(reinterpret_cast<char*>(&result) + i, array.data() + start + length - i - 1, 1);
}
// same byte order
// memcpy(&result, array.data() + start, length);
return result;
};
auto statusToString = [](uint32_t status) -> QString
{
auto hexStatus = "0x" + QString("%1").arg(status, 2, 16, QLatin1Char('0')).toUpper();
if (hexStatus.compare("0xFF", Qt::CaseInsensitive) == 0)
{
hexStatus = "";
}
return hexStatus;
};
auto shortIntToString = [](uint16_t parameter) -> QString
{
return parameter == std::numeric_limits<uint16_t>::max() ? "" : QString::number(parameter);
};
auto shortDoubleToString = [](uint16_t parameter) -> QString
{
return parameter == std::numeric_limits<uint16_t>::max() ? "" : QString::number(parameter / 10.0, 'f', 1);
};
const auto package = serialData_.mid(0, 35);
serialData_ = serialData_.mid(35);
quint16 calculatedChecksum = 0;
for (int i = 0; i < 33; ++i)
{
calculatedChecksum += static_cast<quint8>(package.at(i));
}
const auto checksum = extractInt(package, 33, 2);
if (calculatedChecksum != checksum)
{
qWarning() << "incorrect checksum";
return;
}
setProperty("ipbsFirstVoltage", shortDoubleToString(extractInt(package, 2, 2)));
setProperty("ipbsFirstCurrent", shortDoubleToString(extractInt(package, 4, 2)));
setProperty("ipbsSecondVoltage", shortDoubleToString(extractInt(package, 6, 2)));
setProperty("ipbsSecondCurrent", shortDoubleToString(extractInt(package, 8, 2)));
setProperty("ipbsThirdVoltage", shortDoubleToString(extractInt(package, 10, 2)));
setProperty("ipbsThirdCurrent", shortDoubleToString(extractInt(package, 12, 2)));
setProperty("mainVoltage", shortIntToString(extractInt(package, 14, 2)));
setProperty("ipbsMode", extractInt(package, 16, 1));
setProperty("firstStatus", statusToString(extractInt(package, 17, 1)));
setProperty("secondStatus", statusToString(extractInt(package, 18, 1)));
setProperty("ainFirstVoltage", shortIntToString(extractInt(package, 19, 2)));
setProperty("ainFirstCurrent", shortDoubleToString(extractInt(package, 21, 2)));
setProperty("ainSecondVoltage", shortIntToString(extractInt(package, 23, 2)));
setProperty("ainSecondCurrent", shortDoubleToString(extractInt(package, 25, 2)));
setProperty("ainThirdVoltage", shortIntToString(extractInt(package, 27, 2)));
setProperty("ainThirdCurrent", shortDoubleToString(extractInt(package, 29, 2)));
setProperty("psnInputVoltage", shortIntToString(extractInt(package, 31, 2)));
}
void DataController::mockSerialData()
{
serialData_.clear();
serialData_.append((char)0xD0);
serialData_.append((char)0x0D);
uint16_t voltage1 = qToBigEndian((uint16_t)1102);
uint16_t current1 = qToBigEndian((uint16_t)955);
serialData_.append(reinterpret_cast<char*>(&voltage1), 2);
serialData_.append(reinterpret_cast<char*>(&current1), 2);
uint16_t voltage2 = qToBigEndian((uint16_t)1097);
uint16_t current2 = qToBigEndian((uint16_t)831);
serialData_.append(reinterpret_cast<char*>(&voltage2), 2);
serialData_.append(reinterpret_cast<char*>(&current2), 2);
uint16_t voltage3 = qToBigEndian((uint16_t)1222);
uint16_t current3 = qToBigEndian((uint16_t)353);
serialData_.append(reinterpret_cast<char*>(&voltage3), 2);
serialData_.append(reinterpret_cast<char*>(&current3), 2);
uint16_t mainVoltage = qToBigEndian((uint16_t)384);
char mode = 0;
serialData_.append(reinterpret_cast<char*>(&mainVoltage), 2);
serialData_.append(mode);
char status1 = 0x0;
char status2 = 0x0;
serialData_.append(status1);
serialData_.append(status2);
uint16_t voltage4 = qToBigEndian((uint16_t)380);
uint16_t current4 = qToBigEndian((uint16_t)535);
serialData_.append(reinterpret_cast<char*>(&voltage4), 2);
serialData_.append(reinterpret_cast<char*>(&current4), 2);
uint16_t voltage5 = qToBigEndian((uint16_t)381);
uint16_t current5 = qToBigEndian((uint16_t)527);
serialData_.append(reinterpret_cast<char*>(&voltage5), 2);
serialData_.append(reinterpret_cast<char*>(&current5), 2);
uint16_t voltage6 = qToBigEndian((uint16_t)383);
uint16_t current6 = qToBigEndian((uint16_t)729);
serialData_.append(reinterpret_cast<char*>(&voltage6), 2);
serialData_.append(reinterpret_cast<char*>(&current6), 2);
uint16_t inputVoltage = qToBigEndian((uint16_t)3010);
serialData_.append(reinterpret_cast<char*>(&inputVoltage), 2);
quint16 checksum = 0;
for (int i = 0; i < serialData_.size(); ++i)
{
checksum += static_cast<quint8>(serialData_.at(i));
}
checksum = qToBigEndian(checksum);
serialData_.append(reinterpret_cast<char*>(&checksum), 2);
}