This patch separates the transactionSequenceNumber used for sending from the one received. According to the specification, the transactionSequenceNumber is not meant to equally increase on both ends, but instead really just be a "random" number which allows to match a reply to a request. Syncing them on both ends has the outcome to increase the likelyhood of collisions if a device sends a notification at the same time we send a request and thus even may wrongly interpret that incoming command as a reply to the request. In fact, ideally TSNs for outgoing messages would stay away as far as possible from incoming ones. The old code additionally had the problem that it would re-use the last received TSN for outgoing requests, given it used a post-increment when reading m_transactionSequenceNumber after setting it to the last received TSN. The new code will use a single static upcounting TSN for all outgoing requests but will still allow overriding it with a custom TSN of for some reason a certain device requires a specific TSN (apparently those exist). It will not do anything with incoming TSNs but forward them now to the application layer which may decide to use it match its own transactions or to deduplicate packets. This allows fixing the issue in nymea that remote controls sometimes produce duplicate pressed events (seen most often with the Tradfri Symfonisk) by discarding commands that didn't increase the TSN.
514 lines
21 KiB
C++
514 lines
21 KiB
C++
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
*
|
|
* Copyright 2013 - 2020, nymea GmbH
|
|
* Contact: contact@nymea.io
|
|
*
|
|
* This file is part of nymea-zigbee.
|
|
* This project including source code and documentation is protected by copyright law, and
|
|
* remains the property of nymea GmbH. All rights, including reproduction, publication,
|
|
* editing and translation, are reserved. The use of this project is subject to the terms of a
|
|
* license agreement to be concluded with nymea GmbH in accordance with the terms
|
|
* of use of nymea GmbH, available under https://nymea.io/license
|
|
*
|
|
* GNU Lesser General Public License Usage
|
|
* Alternatively, this project may be redistributed and/or modified under the terms of the GNU
|
|
* Lesser General Public License as published by the Free Software Foundation; version 3.
|
|
* this project is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
* See the GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License along with this project.
|
|
* If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* For any further details and any questions please contact us under contact@nymea.io
|
|
* or see our FAQ/Licensing Information on https://nymea.io/license/faq
|
|
*
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
#include "zigbeeutils.h"
|
|
#include "zigbeenetwork.h"
|
|
#include "zigbeecluster.h"
|
|
#include "loggingcategory.h"
|
|
#include "zigbeenetworkreply.h"
|
|
#include "zigbeeclusterlibrary.h"
|
|
#include "zigbeenetworkrequest.h"
|
|
|
|
#include <QDataStream>
|
|
#include <QMetaEnum>
|
|
|
|
ZigbeeCluster::ZigbeeCluster(ZigbeeNetwork *network, ZigbeeNode *node, ZigbeeNodeEndpoint *endpoint, ZigbeeClusterLibrary::ClusterId clusterId, Direction direction, QObject *parent) :
|
|
QObject(parent),
|
|
m_network(network),
|
|
m_node(node),
|
|
m_endpoint(endpoint),
|
|
m_clusterId(clusterId),
|
|
m_direction(direction)
|
|
{
|
|
//qCDebug(dcZigbeeCluster()) << "Create cluster" << ZigbeeUtils::convertUint16ToHexString(clusterId) << direction;
|
|
}
|
|
|
|
ZigbeeNode *ZigbeeCluster::node() const
|
|
{
|
|
return m_node;
|
|
}
|
|
|
|
ZigbeeNodeEndpoint *ZigbeeCluster::endpoint() const
|
|
{
|
|
return m_endpoint;
|
|
}
|
|
|
|
ZigbeeCluster::Direction ZigbeeCluster::direction() const
|
|
{
|
|
return m_direction;
|
|
}
|
|
|
|
ZigbeeClusterLibrary::ClusterId ZigbeeCluster::clusterId() const
|
|
{
|
|
return m_clusterId;
|
|
}
|
|
|
|
QString ZigbeeCluster::clusterName() const
|
|
{
|
|
return ZigbeeUtils::clusterIdToString(static_cast<ZigbeeClusterLibrary::ClusterId>(m_clusterId));
|
|
}
|
|
|
|
QList<ZigbeeClusterAttribute> ZigbeeCluster::attributes() const
|
|
{
|
|
return m_attributes.values();
|
|
}
|
|
|
|
bool ZigbeeCluster::hasAttribute(quint16 attributeId) const
|
|
{
|
|
if (m_attributes.keys().isEmpty())
|
|
return false;
|
|
|
|
return m_attributes.keys().contains(attributeId);
|
|
}
|
|
|
|
ZigbeeClusterAttribute ZigbeeCluster::attribute(quint16 attributeId)
|
|
{
|
|
return m_attributes.value(attributeId);
|
|
}
|
|
|
|
void ZigbeeCluster::setAttribute(const ZigbeeClusterAttribute &attribute)
|
|
{
|
|
qCDebug(dcZigbeeCluster()) << "Update attribute" << m_node << m_endpoint << this << attribute;
|
|
updateOrAddAttribute(attribute);
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::readAttributes(QList<quint16> attributes, quint16 manufacturerCode)
|
|
{
|
|
qCDebug(dcZigbeeCluster()) << "Read attributes from" << m_node << m_endpoint << this << attributes;
|
|
|
|
// ZCL payload
|
|
QByteArray payload;
|
|
QDataStream stream(&payload, QIODevice::WriteOnly);
|
|
stream.setByteOrder(QDataStream::LittleEndian);
|
|
foreach (quint16 attribute, attributes) {
|
|
stream << attribute;
|
|
}
|
|
|
|
return executeGlobalCommand(ZigbeeClusterLibrary::CommandReadAttributes, payload, manufacturerCode);
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::writeAttributes(QList<ZigbeeClusterLibrary::WriteAttributeRecord> writeAttributeRecords, quint16 manufacturerCode)
|
|
{
|
|
qCDebug(dcZigbeeCluster()) << "Write attributes on" << m_node << m_endpoint << this;
|
|
QByteArray payload;
|
|
foreach (const ZigbeeClusterLibrary::WriteAttributeRecord &writeAttributeRecord, writeAttributeRecords) {
|
|
payload += ZigbeeClusterLibrary::buildWriteAttributeRecord(writeAttributeRecord);
|
|
}
|
|
|
|
return executeGlobalCommand(ZigbeeClusterLibrary::CommandWriteAttributes, payload, manufacturerCode);
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::configureReporting(QList<ZigbeeClusterLibrary::AttributeReportingConfiguration> reportingConfigurations, quint16 manufacturerCode)
|
|
{
|
|
qCDebug(dcZigbeeCluster()) << "Configure reporting on" << m_node << m_endpoint << this << reportingConfigurations;
|
|
|
|
QByteArray payload;
|
|
foreach (const ZigbeeClusterLibrary::AttributeReportingConfiguration reportingConfiguration, reportingConfigurations) {
|
|
payload += ZigbeeClusterLibrary::buildAttributeReportingConfiguration(reportingConfiguration);
|
|
}
|
|
|
|
return executeGlobalCommand(ZigbeeClusterLibrary::CommandConfigureReporting, payload, manufacturerCode);
|
|
}
|
|
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::executeGlobalCommand(quint8 command, const QByteArray &payload, quint16 manufacturerCode, quint8 transactionSequenceNumber)
|
|
{
|
|
// Build the request
|
|
ZigbeeNetworkRequest request = createGeneralRequest();
|
|
|
|
// Build ZCL frame
|
|
|
|
// Note: for basic commands the frame control files has to be zero accoring to spec ZCL 2.4.1.1
|
|
ZigbeeClusterLibrary::FrameControl frameControl;
|
|
frameControl.frameType = ZigbeeClusterLibrary::FrameTypeGlobal;
|
|
if (manufacturerCode != 0) {
|
|
frameControl.manufacturerSpecific = true;
|
|
} else {
|
|
frameControl.manufacturerSpecific = false;
|
|
}
|
|
|
|
if (m_direction == Direction::Server) {
|
|
frameControl.direction = ZigbeeClusterLibrary::DirectionClientToServer;
|
|
} else {
|
|
frameControl.direction = ZigbeeClusterLibrary::DirectionServerToClient;
|
|
}
|
|
frameControl.disableDefaultResponse = true;
|
|
|
|
// ZCL header
|
|
ZigbeeClusterLibrary::Header header;
|
|
header.frameControl = frameControl;
|
|
header.command = command;
|
|
if (manufacturerCode != 0) {
|
|
header.manufacturerCode = manufacturerCode;
|
|
}
|
|
header.transactionSequenceNumber = transactionSequenceNumber;
|
|
|
|
// Put them together
|
|
ZigbeeClusterLibrary::Frame frame;
|
|
frame.header = header;
|
|
frame.payload = payload;
|
|
|
|
request.setTxOptions(Zigbee::ZigbeeTxOptions(Zigbee::ZigbeeTxOptionAckTransmission));
|
|
request.setAsdu(ZigbeeClusterLibrary::buildFrame(frame));
|
|
|
|
ZigbeeClusterReply *zclReply = createClusterReply(request, frame);
|
|
ZigbeeNetworkReply *networkReply = m_network->sendRequest(request);
|
|
connect(networkReply, &ZigbeeNetworkReply::finished, this, [this, networkReply, zclReply](){
|
|
if (!verifyNetworkError(zclReply, networkReply)) {
|
|
finishZclReply(zclReply);
|
|
return;
|
|
}
|
|
|
|
// The request was successfully sent to the device
|
|
// Now check if the expected indication response received already
|
|
if (zclReply->isComplete()) {
|
|
finishZclReply(zclReply);
|
|
return;
|
|
}
|
|
});
|
|
|
|
return zclReply;
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::createClusterReply(const ZigbeeNetworkRequest &request, ZigbeeClusterLibrary::Frame frame)
|
|
{
|
|
ZigbeeClusterReply *zclReply = new ZigbeeClusterReply(request, frame, this);
|
|
connect(zclReply, &ZigbeeClusterReply::finished, zclReply, &ZigbeeClusterReply::deleteLater, Qt::QueuedConnection);
|
|
zclReply->m_transactionSequenceNumber = frame.header.transactionSequenceNumber;
|
|
m_pendingReplies.insert(zclReply->transactionSequenceNumber(), zclReply);
|
|
return zclReply;
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::executeClusterCommand(quint8 command, const QByteArray &payload)
|
|
{
|
|
ZigbeeNetworkRequest request = createGeneralRequest();
|
|
|
|
// Build ZCL frame control
|
|
ZigbeeClusterLibrary::FrameControl frameControl;
|
|
frameControl.frameType = ZigbeeClusterLibrary::FrameTypeClusterSpecific;
|
|
frameControl.manufacturerSpecific = false;
|
|
frameControl.direction = ZigbeeClusterLibrary::DirectionClientToServer;
|
|
frameControl.disableDefaultResponse = false;
|
|
|
|
// Build ZCL header
|
|
ZigbeeClusterLibrary::Header header;
|
|
header.frameControl = frameControl;
|
|
header.command = command;
|
|
header.transactionSequenceNumber = newTransactionSequenceNumber();
|
|
|
|
// Build ZCL frame
|
|
ZigbeeClusterLibrary::Frame frame;
|
|
frame.header = header;
|
|
frame.payload = payload;
|
|
|
|
request.setTxOptions(Zigbee::ZigbeeTxOptions(Zigbee::ZigbeeTxOptionAckTransmission));
|
|
request.setAsdu(ZigbeeClusterLibrary::buildFrame(frame));
|
|
|
|
ZigbeeClusterReply *zclReply = createClusterReply(request, frame);
|
|
qCDebug(dcZigbeeCluster()) << "Executing command" << ZigbeeUtils::convertByteToHexString(command) << ZigbeeUtils::convertByteArrayToHexString(payload);
|
|
ZigbeeNetworkReply *networkReply = m_network->sendRequest(request);
|
|
connect(networkReply, &ZigbeeNetworkReply::finished, this, [this, networkReply, zclReply](){
|
|
if (!verifyNetworkError(zclReply, networkReply)) {
|
|
finishZclReply(zclReply);
|
|
return;
|
|
}
|
|
|
|
// The request was successfully sent to the device
|
|
// Now check if the expected indication response received already
|
|
if (zclReply->isComplete()) {
|
|
finishZclReply(zclReply);
|
|
return;
|
|
}
|
|
});
|
|
|
|
return zclReply;
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::sendClusterServerResponse(quint8 command, quint8 transactionSequenceNumber, const QByteArray &payload)
|
|
{
|
|
ZigbeeNetworkRequest request = createGeneralRequest();
|
|
|
|
// Build ZCL frame control
|
|
ZigbeeClusterLibrary::FrameControl frameControl;
|
|
frameControl.frameType = ZigbeeClusterLibrary::FrameTypeClusterSpecific;
|
|
frameControl.manufacturerSpecific = false;
|
|
frameControl.direction = ZigbeeClusterLibrary::DirectionServerToClient;
|
|
frameControl.disableDefaultResponse = true;
|
|
|
|
// Build ZCL header
|
|
ZigbeeClusterLibrary::Header header;
|
|
header.frameControl = frameControl;
|
|
header.command = command;
|
|
header.transactionSequenceNumber = transactionSequenceNumber;
|
|
|
|
// Build ZCL frame
|
|
ZigbeeClusterLibrary::Frame frame;
|
|
frame.header = header;
|
|
frame.payload = payload;
|
|
|
|
request.setTxOptions(Zigbee::ZigbeeTxOptions(Zigbee::ZigbeeTxOptionAckTransmission));
|
|
request.setAsdu(ZigbeeClusterLibrary::buildFrame(frame));
|
|
|
|
ZigbeeClusterReply *zclReply = createClusterReply(request, frame);
|
|
qCDebug(dcZigbeeCluster()) << "Send command response" << ZigbeeUtils::convertByteToHexString(command) << "TSN:" << ZigbeeUtils::convertByteToHexString(transactionSequenceNumber) << ZigbeeUtils::convertByteArrayToHexString(payload);
|
|
ZigbeeNetworkReply *networkReply = m_network->sendRequest(request);
|
|
connect(networkReply, &ZigbeeNetworkReply::finished, this, [this, networkReply, zclReply](){
|
|
if (!verifyNetworkError(zclReply, networkReply)) {
|
|
finishZclReply(zclReply);
|
|
return;
|
|
}
|
|
|
|
// Note: since this is a response to a request, we don't expect any additional indications and the reply is finished
|
|
finishZclReply(zclReply);
|
|
});
|
|
|
|
return zclReply;
|
|
}
|
|
|
|
ZigbeeClusterReply *ZigbeeCluster::sendDefaultResponse(quint8 transactionSequenceNumber, quint8 command, quint8 status)
|
|
{
|
|
ZigbeeNetworkRequest request = createGeneralRequest();
|
|
|
|
// Build ZCL frame control
|
|
ZigbeeClusterLibrary::FrameControl frameControl;
|
|
frameControl.frameType = ZigbeeClusterLibrary::FrameTypeGlobal;
|
|
frameControl.manufacturerSpecific = false;
|
|
frameControl.direction = ZigbeeClusterLibrary::DirectionServerToClient;
|
|
frameControl.disableDefaultResponse = true;
|
|
|
|
// Build ZCL header
|
|
ZigbeeClusterLibrary::Header header;
|
|
header.frameControl = frameControl;
|
|
header.command = ZigbeeClusterLibrary::CommandDefaultResponse;
|
|
header.transactionSequenceNumber = transactionSequenceNumber;
|
|
|
|
QByteArray payload;
|
|
QDataStream stream(&payload, QIODevice::WriteOnly);
|
|
stream.setByteOrder(QDataStream::LittleEndian);
|
|
stream << command << status;
|
|
|
|
// Build ZCL frame
|
|
ZigbeeClusterLibrary::Frame frame;
|
|
frame.header = header;
|
|
frame.payload = payload;
|
|
|
|
request.setTxOptions(Zigbee::ZigbeeTxOptions(Zigbee::ZigbeeTxOptionAckTransmission));
|
|
request.setAsdu(ZigbeeClusterLibrary::buildFrame(frame));
|
|
|
|
ZigbeeClusterReply *zclReply = createClusterReply(request, frame);
|
|
qCDebug(dcZigbeeCluster()) << "Send default response" << "TSN:" << ZigbeeUtils::convertByteToHexString(transactionSequenceNumber) << ZigbeeUtils::convertByteArrayToHexString(payload);
|
|
ZigbeeNetworkReply *networkReply = m_network->sendRequest(request);
|
|
connect(networkReply, &ZigbeeNetworkReply::finished, this, [this, networkReply, zclReply](){
|
|
if (!verifyNetworkError(zclReply, networkReply)) {
|
|
finishZclReply(zclReply);
|
|
return;
|
|
}
|
|
|
|
// Note: since this is a response to a request, we don't expect any additional indications and the reply is finished
|
|
finishZclReply(zclReply);
|
|
});
|
|
|
|
return zclReply;
|
|
}
|
|
|
|
ZigbeeNetworkRequest ZigbeeCluster::createGeneralRequest()
|
|
{
|
|
// Build the request
|
|
ZigbeeNetworkRequest request;
|
|
request.setRequestId(m_network->generateSequenceNumber());
|
|
request.setDestinationAddressMode(Zigbee::DestinationAddressModeShortAddress);
|
|
request.setDestinationShortAddress(m_node->shortAddress());
|
|
request.setProfileId(Zigbee::ZigbeeProfileHomeAutomation); // Note: in Zigbee 3.0 this is the Application Profile (0x0104)
|
|
request.setClusterId(m_clusterId);
|
|
request.setSourceEndpoint(0x01);
|
|
request.setDestinationEndpoint(m_endpoint->endpointId());
|
|
request.setRadius(0);
|
|
request.setTxOptions(Zigbee::ZigbeeTxOptions(Zigbee::ZigbeeTxOptionAckTransmission));
|
|
return request;
|
|
}
|
|
|
|
bool ZigbeeCluster::verifyNetworkError(ZigbeeClusterReply *zclReply, ZigbeeNetworkReply *networkReply)
|
|
{
|
|
bool success = false;
|
|
switch (networkReply->error()) {
|
|
case ZigbeeNetworkReply::ErrorNoError:
|
|
// The request has been transported successfully to he destination, now
|
|
// wait for the expected indication or check if we already recieved it
|
|
zclReply->m_apsConfirmReceived = true;
|
|
success = true;
|
|
break;
|
|
case ZigbeeNetworkReply::ErrorTimeout:
|
|
zclReply->m_error = ZigbeeClusterReply::ErrorTimeout;
|
|
qCWarning(dcZigbeeClusterLibrary()) << "Failed to send request to" << m_node << zclReply->error();
|
|
break;
|
|
case ZigbeeNetworkReply::ErrorInterfaceError:
|
|
zclReply->m_error = ZigbeeClusterReply::ErrorInterfaceError;
|
|
qCWarning(dcZigbeeClusterLibrary()) << "Failed to send request to" << m_node << zclReply->error();
|
|
break;
|
|
case ZigbeeNetworkReply::ErrorNetworkOffline:
|
|
zclReply->m_error = ZigbeeClusterReply::ErrorNetworkOffline;
|
|
qCWarning(dcZigbeeClusterLibrary()) << "Failed to send request to" << m_node << zclReply->error();
|
|
break;
|
|
case ZigbeeNetworkReply::ErrorZigbeeApsStatusError:
|
|
zclReply->m_apsConfirmReceived = true;
|
|
zclReply->m_error = ZigbeeClusterReply::ErrorZigbeeApsStatusError;
|
|
zclReply->m_zigbeeApsStatus = networkReply->zigbeeApsStatus();
|
|
qCWarning(dcZigbeeClusterLibrary()) << "Failed to send request to" << m_node << zclReply->zigbeeApsStatus();
|
|
break;
|
|
case ZigbeeNetworkReply::ErrorZigbeeNwkStatusError:
|
|
zclReply->m_apsConfirmReceived = true;
|
|
zclReply->m_error = ZigbeeClusterReply::ErrorZigbeeNwkStatusError;
|
|
zclReply->m_zigbeeNwkStatus = networkReply->zigbeeNwkStatus();
|
|
qCWarning(dcZigbeeClusterLibrary()) << "Failed to send request to" << m_node << zclReply->zigbeeNwkStatus();
|
|
break;
|
|
case ZigbeeNetworkReply::ErrorZigbeeMacStatusError:
|
|
zclReply->m_apsConfirmReceived = true;
|
|
zclReply->m_error = ZigbeeClusterReply::ErrorZigbeeMacStatusError;
|
|
zclReply->m_zigbeeMacStatus = networkReply->zigbeeMacStatus();
|
|
qCWarning(dcZigbeeClusterLibrary()) << "Failed to send request to" << m_node << zclReply->zigbeeMacStatus();
|
|
break;
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
void ZigbeeCluster::finishZclReply(ZigbeeClusterReply *zclReply)
|
|
{
|
|
m_pendingReplies.remove(zclReply->transactionSequenceNumber());
|
|
qCDebug(dcZigbeeCluster()) << "ZigbeeClusterReply finished" << zclReply->request() << zclReply->requestFrame() << zclReply->responseFrame();
|
|
// FIXME: Set the status
|
|
emit zclReply->finished();
|
|
}
|
|
|
|
void ZigbeeCluster::processDataIndication(ZigbeeClusterLibrary::Frame frame)
|
|
{
|
|
// Warn about the unhandled cluster indication, you can override this method in cluster implementations
|
|
qCWarning(dcZigbeeCluster()) << "Unhandled ZCL indication in" << m_node << m_endpoint << this << frame;
|
|
}
|
|
|
|
void ZigbeeCluster::updateOrAddAttribute(const ZigbeeClusterAttribute &attribute)
|
|
{
|
|
if (hasAttribute(attribute.id())) {
|
|
m_attributes[attribute.id()] = attribute;
|
|
emit attributeChanged(attribute);
|
|
} else {
|
|
m_attributes.insert(attribute.id(), attribute);
|
|
emit attributeChanged(attribute);
|
|
}
|
|
}
|
|
|
|
quint8 ZigbeeCluster::newTransactionSequenceNumber()
|
|
{
|
|
static quint8 tsn = 1;
|
|
return tsn++;
|
|
}
|
|
|
|
void ZigbeeCluster::processApsDataIndication(const QByteArray &asdu, const ZigbeeClusterLibrary::Frame &frame)
|
|
{
|
|
// Check if this indication is for a pending reply
|
|
if (m_pendingReplies.contains(frame.header.transactionSequenceNumber)) {
|
|
ZigbeeClusterReply *reply = m_pendingReplies.value(frame.header.transactionSequenceNumber);
|
|
reply->m_responseData = asdu;
|
|
reply->m_responseFrame = frame;
|
|
reply->m_zclIndicationReceived = true;
|
|
if (reply->isComplete())
|
|
finishZclReply(reply);
|
|
|
|
// Update the attributes internally if this was a read command
|
|
if (frame.header.frameControl.frameType == ZigbeeClusterLibrary::FrameTypeGlobal) {
|
|
ZigbeeClusterLibrary::Command globalCommand = static_cast<ZigbeeClusterLibrary::Command>(frame.header.command);
|
|
if (globalCommand == ZigbeeClusterLibrary::CommandReadAttributesResponse) {
|
|
// Update the attributes from the attribut status reports internally
|
|
QList<ZigbeeClusterLibrary::ReadAttributeStatusRecord> attributeStatusRecords = ZigbeeClusterLibrary::parseAttributeStatusRecords(frame.payload);
|
|
foreach (const ZigbeeClusterLibrary::ReadAttributeStatusRecord &attributeStatusRecord, attributeStatusRecords) {
|
|
qCDebug(dcZigbeeCluster()) << "Received read attribute status record" << this << attributeStatusRecord;
|
|
if (attributeStatusRecord.attributeStatus == ZigbeeClusterLibrary::StatusSuccess) {
|
|
setAttribute(ZigbeeClusterAttribute(attributeStatusRecord.attributeId, attributeStatusRecord.dataType));
|
|
} else {
|
|
qCWarning(dcZigbeeCluster()) << "Reading attribute status record returned an error" << attributeStatusRecord;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// Check for server clusters and if this is an attribute report
|
|
if (m_direction == Server && frame.header.frameControl.frameType == ZigbeeClusterLibrary::FrameTypeGlobal) {
|
|
ZigbeeClusterLibrary::Command globalCommand = static_cast<ZigbeeClusterLibrary::Command>(frame.header.command);
|
|
if (globalCommand == ZigbeeClusterLibrary::CommandReportAttributes) {
|
|
// Read the attribute reports and update/set the attributes
|
|
QDataStream stream(frame.payload);
|
|
stream.setByteOrder(QDataStream::LittleEndian);
|
|
while (!stream.atEnd()) {
|
|
quint16 attributeId = 0; quint8 type = 0;
|
|
stream >> attributeId >> type;
|
|
ZigbeeDataType dataType = ZigbeeClusterLibrary::readDataType(&stream, static_cast<Zigbee::DataType>(type));
|
|
qCDebug(dcZigbeeCluster()) << "Received attributes report" << this << frame;
|
|
setAttribute(ZigbeeClusterAttribute(attributeId, dataType));
|
|
}
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Not for a reply or not an attribute report, let the cluster process this message internally
|
|
processDataIndication(frame);
|
|
}
|
|
|
|
QDebug operator<<(QDebug debug, ZigbeeCluster *cluster)
|
|
{
|
|
debug.nospace().noquote() << "ZigbeeCluster("
|
|
<< ZigbeeUtils::convertUint16ToHexString(static_cast<quint16>(cluster->clusterId())) << ", "
|
|
<< cluster->clusterName() << ", ";
|
|
switch (cluster->direction()) {
|
|
case ZigbeeCluster::Server:
|
|
debug.nospace().noquote() << "Server)";
|
|
break;
|
|
case ZigbeeCluster::Client:
|
|
debug.nospace().noquote() << "Client)";
|
|
break;
|
|
}
|
|
|
|
return debug.space();
|
|
}
|
|
|
|
QDebug operator<<(QDebug debug, const ZigbeeClusterAttributeReport &attributeReport)
|
|
{
|
|
debug.nospace().noquote() << "AttributeReport("
|
|
<< attributeReport.clusterId << ", "
|
|
<< attributeReport.attributeId << ", "
|
|
<< attributeReport.attributeStatus << ", "
|
|
<< attributeReport.dataType << ", "
|
|
<< ZigbeeUtils::convertByteArrayToHexString(attributeReport.data)
|
|
<< ")";
|
|
|
|
return debug.space();
|
|
}
|
|
|