2025-04-28 12:25:20 +08:00
|
|
|
#include "XNUDPService.h"
|
|
|
|
#include "XNUDPService_p.h"
|
|
|
|
#include <XNCore/XNServiceManager.h>
|
|
|
|
#include <XNCore/XNDDSManager.h>
|
|
|
|
#include <QFile>
|
|
|
|
#include <QDomDocument>
|
|
|
|
|
|
|
|
XN_DLL_INITIALIZE(XNUDPService)
|
|
|
|
|
|
|
|
XN_REGISTER_SERVICE_BEGIN_SERVICE(XNUDPService)
|
|
|
|
XN_REGISTER_SERVICE_END_SERVICE(XNUDPService)
|
|
|
|
|
|
|
|
XNUDPService::XNUDPService(QObject *parent)
|
|
|
|
: XNServiceObject(*new XNUDPServicePrivate(this), parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
XNUDPService::~XNUDPService()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
XNUDPService::XNUDPService(XNUDPServicePrivate &dd, QObject *parent) : XNServiceObject(dd, parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void XNUDPService::OnInitialize()
|
|
|
|
{
|
|
|
|
Q_D(XNUDPService);
|
|
|
|
XNServiceObject::OnInitialize();
|
|
|
|
|
|
|
|
// 读取配置文件
|
|
|
|
QFile file(GetXmlPath());
|
|
|
|
if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
|
|
|
LOG_WARNING("Failed to open config file:%1", GetXmlPath());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDomDocument doc;
|
|
|
|
if (!doc.setContent(&file)) {
|
|
|
|
file.close();
|
|
|
|
LOG_WARNING("Failed to parse config file:%1", GetXmlPath());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
// 读取UDP配置
|
|
|
|
QDomElement udpElement = doc.documentElement().firstChildElement("UDP");
|
|
|
|
if (!udpElement.isNull()) {
|
|
|
|
d->localPort = udpElement.firstChildElement("LocalPort").text().toUInt();
|
|
|
|
d->targetHost = udpElement.firstChildElement("TargetHost").text();
|
|
|
|
d->targetPort = udpElement.firstChildElement("TargetPort").text().toUInt();
|
|
|
|
} else {
|
|
|
|
LOG_WARNING("UDP configuration not found, using default values");
|
|
|
|
d->localPort = 12345;
|
|
|
|
d->targetHost = "127.0.0.1";
|
|
|
|
d->targetPort = 54321;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void XNUDPService::OnPrepareForExecute()
|
|
|
|
{
|
|
|
|
Q_D(XNUDPService);
|
|
|
|
XNServiceObject::OnPrepareForExecute();
|
|
|
|
|
|
|
|
// 初始化UDP socket
|
|
|
|
d->udpSocket = new QUdpSocket(this);
|
|
|
|
|
|
|
|
// 绑定本地端口
|
|
|
|
if (!d->udpSocket->bind(QHostAddress::Any, d->localPort)) {
|
|
|
|
LOG_WARNING("UDP socket bind failed on port:%1", d->localPort);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 连接UDP socket的信号
|
|
|
|
connect(d->udpSocket, &QUdpSocket::readyRead, this, &XNUDPService::HandleIncomingData);
|
|
|
|
|
|
|
|
RegisterRTEventHandler("SendUDPData",
|
|
|
|
std::bind(&XNUDPService::SendData, this, std::placeholders::_1));
|
|
|
|
}
|
|
|
|
|
|
|
|
void XNUDPService::HandleIncomingData()
|
|
|
|
{
|
|
|
|
Q_D(XNUDPService);
|
|
|
|
while (d->udpSocket->hasPendingDatagrams()) {
|
|
|
|
QByteArray datagram;
|
|
|
|
datagram.resize(d->udpSocket->pendingDatagramSize());
|
|
|
|
QHostAddress sender;
|
|
|
|
quint16 senderPort;
|
|
|
|
|
|
|
|
d->udpSocket->readDatagram(datagram.data(), datagram.size(), &sender, &senderPort);
|
|
|
|
|
2025-04-29 11:55:56 +08:00
|
|
|
if (datagram.size() <= 6) {
|
2025-04-28 16:41:21 +08:00
|
|
|
LOG_WARNING("Invalid size of UDP datagram received, size:%1", datagram.size());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 使用处理后的数据
|
2025-04-29 11:55:56 +08:00
|
|
|
quint8 header = (quint8)datagram[0];
|
|
|
|
quint8 type = (quint8)datagram[1];
|
|
|
|
quint8 subType1 = (quint8)datagram[2];
|
|
|
|
quint8 subType2 = (quint8)datagram[3];
|
|
|
|
quint16 size = (quint8)datagram[4] << 8 | (quint8)datagram[5];
|
2025-04-28 16:41:21 +08:00
|
|
|
if (header != 0x0b) {
|
|
|
|
LOG_WARNING("Invalid header of UDP datagram received, header:%1", header);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2025-04-29 11:55:56 +08:00
|
|
|
if (datagram.size() < size) {
|
2025-04-28 16:41:21 +08:00
|
|
|
LOG_WARNING("Invalid size of UDP datagram received, size:%1, expected:%2",
|
2025-04-29 11:55:56 +08:00
|
|
|
datagram.size(), size);
|
2025-04-28 12:25:20 +08:00
|
|
|
continue;
|
|
|
|
}
|
2025-04-28 16:41:21 +08:00
|
|
|
|
|
|
|
if (type == 0x04) {
|
|
|
|
if (subType1 == 0x00 && subType2 == 0x00) {
|
2025-04-29 11:55:56 +08:00
|
|
|
TriggerRTEvent("ATA04AeroInput", datagram);
|
2025-04-28 12:25:20 +08:00
|
|
|
continue;
|
2025-04-28 16:41:21 +08:00
|
|
|
} else if (subType1 == 0x01 && subType2 == 0x00) {
|
2025-04-29 11:55:56 +08:00
|
|
|
TriggerRTEvent("ATA04GhInput", datagram);
|
2025-04-28 12:25:20 +08:00
|
|
|
continue;
|
2025-04-28 16:41:21 +08:00
|
|
|
} else if (subType1 == 0x02 && subType2 == 0x00) {
|
2025-04-29 11:55:56 +08:00
|
|
|
TriggerRTEvent("ATA04WbInput", datagram);
|
2025-04-28 12:25:20 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// TODO: 根据具体需求处理其它数据
|
2025-04-29 11:55:56 +08:00
|
|
|
TriggerRTEvent("ReceiveUDPData", datagram);
|
2025-04-28 12:25:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void XNUDPService::SendData(const QVariant &data)
|
|
|
|
{
|
|
|
|
Q_D(XNUDPService);
|
|
|
|
if (!d->udpSocket) {
|
|
|
|
LOG_WARNING("UDP socket not initialized");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 将QVariant转换为字节数组
|
2025-04-29 11:55:56 +08:00
|
|
|
QByteArray datagram = data.toByteArray();
|
2025-04-28 12:25:20 +08:00
|
|
|
|
|
|
|
// 发送数据
|
|
|
|
qint64 bytesSent =
|
|
|
|
d->udpSocket->writeDatagram(datagram, QHostAddress(d->targetHost), d->targetPort);
|
|
|
|
if (bytesSent == -1) {
|
|
|
|
LOG_WARNING("Failed to send UDP datagram:%1", d->udpSocket->errorString());
|
|
|
|
}
|
|
|
|
}
|