blob: 2669538d02f1181321acc94e9b2afd798cacda61 [file] [log] [blame]
/**
*
* Copyright (c) 2020 Project CHIP Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
*
* Copyright (c) 2020 Silicon Labs
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/***************************************************************************/
/**
* @file
* @brief This file contains the code to manipulate
*the Smart Energy attribute table. This handles
*external calls to read/write the table, as well as
*internal ones.
*******************************************************************************
******************************************************************************/
// this file contains all the common includes for clusters in the zcl-util
#include "common.h"
#include "attribute-storage.h"
// for pulling in defines dealing with EITHER server or client
#include "af-main.h"
#include "gen/enums.h"
#ifdef EMBER_AF_PLUGIN_REPORTING
#include <app/reporting/reporting.h>
#endif // EMBER_AF_PLUGIN_REPORTING
using namespace chip;
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
// External Declarations
//------------------------------------------------------------------------------
// Forward Declarations
//------------------------------------------------------------------------------
// Globals
EmberAfStatus emberAfWriteAttributeExternal(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t mask,
uint16_t manufacturerCode, uint8_t * dataPtr, EmberAfAttributeType dataType)
{
EmberAfAttributeWritePermission extWritePermission =
emberAfAllowNetworkWriteAttributeCallback(endpoint, cluster, attributeID, mask, manufacturerCode, dataPtr, dataType);
switch (extWritePermission)
{
case EMBER_ZCL_ATTRIBUTE_WRITE_PERMISSION_DENY_WRITE:
return EMBER_ZCL_STATUS_FAILURE;
case EMBER_ZCL_ATTRIBUTE_WRITE_PERMISSION_ALLOW_WRITE_NORMAL:
case EMBER_ZCL_ATTRIBUTE_WRITE_PERMISSION_ALLOW_WRITE_OF_READ_ONLY:
return emAfWriteAttribute(endpoint, cluster, attributeID, mask, manufacturerCode, dataPtr, dataType,
(extWritePermission == EMBER_ZCL_ATTRIBUTE_WRITE_PERMISSION_ALLOW_WRITE_OF_READ_ONLY), false);
default:
return (EmberAfStatus) extWritePermission;
}
}
//@deprecated use emberAfWriteServerAttribute or emberAfWriteClientAttribute
EmberAfStatus emberAfWriteAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t mask,
uint8_t * dataPtr, EmberAfAttributeType dataType)
{
return emAfWriteAttribute(endpoint, cluster, attributeID, mask, EMBER_AF_NULL_MANUFACTURER_CODE, dataPtr, dataType,
true, // override read-only?
false); // just test?
}
EmberAfStatus emberAfWriteClientAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t * dataPtr,
EmberAfAttributeType dataType)
{
return emAfWriteAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_CLIENT, EMBER_AF_NULL_MANUFACTURER_CODE, dataPtr,
dataType,
true, // override read-only?
false); // just test?
}
EmberAfStatus emberAfWriteServerAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t * dataPtr,
EmberAfAttributeType dataType)
{
return emAfWriteAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_SERVER, EMBER_AF_NULL_MANUFACTURER_CODE, dataPtr,
dataType,
true, // override read-only?
false); // just test?
}
EmberAfStatus emberAfWriteManufacturerSpecificClientAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID,
uint16_t manufacturerCode, uint8_t * dataPtr,
EmberAfAttributeType dataType)
{
return emAfWriteAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_CLIENT, manufacturerCode, dataPtr, dataType,
true, // override read-only?
false); // just test?
}
EmberAfStatus emberAfWriteManufacturerSpecificServerAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID,
uint16_t manufacturerCode, uint8_t * dataPtr,
EmberAfAttributeType dataType)
{
return emAfWriteAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_SERVER, manufacturerCode, dataPtr, dataType,
true, // override read-only?
false); // just test?
}
EmberAfStatus emberAfVerifyAttributeWrite(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t mask,
uint16_t manufacturerCode, uint8_t * dataPtr, EmberAfAttributeType dataType)
{
return emAfWriteAttribute(endpoint, cluster, attributeID, mask, manufacturerCode, dataPtr, dataType,
false, // override read-only?
true); // just test?
}
EmberAfStatus emberAfReadAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t mask, uint8_t * dataPtr,
uint8_t readLength, EmberAfAttributeType * dataType)
{
return emAfReadAttribute(endpoint, cluster, attributeID, mask, EMBER_AF_NULL_MANUFACTURER_CODE, dataPtr, readLength, dataType);
}
EmberAfStatus emberAfReadServerAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t * dataPtr,
uint8_t readLength)
{
return emAfReadAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_SERVER, EMBER_AF_NULL_MANUFACTURER_CODE, dataPtr,
readLength, NULL);
}
EmberAfStatus emberAfReadClientAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t * dataPtr,
uint8_t readLength)
{
return emAfReadAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_CLIENT, EMBER_AF_NULL_MANUFACTURER_CODE, dataPtr,
readLength, NULL);
}
EmberAfStatus emberAfReadManufacturerSpecificServerAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID,
uint16_t manufacturerCode, uint8_t * dataPtr, uint8_t readLength)
{
return emAfReadAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_SERVER, manufacturerCode, dataPtr, readLength, NULL);
}
EmberAfStatus emberAfReadManufacturerSpecificClientAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID,
uint16_t manufacturerCode, uint8_t * dataPtr, uint8_t readLength)
{
return emAfReadAttribute(endpoint, cluster, attributeID, CLUSTER_MASK_CLIENT, manufacturerCode, dataPtr, readLength, NULL);
}
bool emberAfReadSequentialAttributesAddToResponse(EndpointId endpoint, ClusterId clusterId, AttributeId startAttributeId,
uint8_t mask, uint16_t manufacturerCode, uint8_t maxAttributeIds,
bool includeAccessControl)
{
uint16_t i;
uint16_t discovered = 0;
uint16_t skipped = 0;
uint16_t total = 0;
EmberAfCluster * cluster = emberAfFindClusterWithMfgCode(endpoint, clusterId, mask, manufacturerCode);
EmberAfAttributeSearchRecord record;
record.endpoint = endpoint;
record.clusterId = clusterId;
record.clusterMask = mask;
record.attributeId = startAttributeId;
record.manufacturerCode = manufacturerCode;
// If we don't have the cluster or it doesn't match the search, we're done.
if (cluster == NULL || !emAfMatchCluster(cluster, &record))
{
return true;
}
for (i = 0; i < cluster->attributeCount; i++)
{
EmberAfAttributeMetadata * metadata = &cluster->attributes[i];
// If the cluster is not manufacturer-specific, an attribute is considered
// only if its manufacturer code matches that of the command (which may be
// unset).
if (!emberAfClusterIsManufacturerSpecific(cluster))
{
record.attributeId = metadata->attributeId;
if (!emAfMatchAttribute(cluster, metadata, &record))
{
continue;
}
}
if (metadata->attributeId < startAttributeId)
{
skipped++;
}
else if (discovered < maxAttributeIds)
{
emberAfPutInt16uInResp(metadata->attributeId);
emberAfPutInt8uInResp(metadata->attributeType);
if (includeAccessControl)
{
// bit 0 : Readable <-- All our attributes are readable
// bit 1 : Writable <-- The only thing we track in the attribute metadata mask
// bit 2 : Reportable <-- All our attributes are reportable
emberAfPutInt8uInResp((metadata->mask & ATTRIBUTE_MASK_WRITABLE) ? 0x07 : 0x05);
}
discovered++;
}
else
{
// MISRA requires ..else if.. to have terminating else.
}
total++;
}
// We are finished if there are no more attributes to find, which means the
// number of attributes discovered plus the number skipped equals the total
// attributes in the cluster. For manufacturer-specific clusters, the total
// includes all attributes in the cluster. For standard ZCL clusters, if the
// the manufacturer code is set, the total is the number of attributes that
// match the manufacturer code. Otherwise, the total is the number of
// standard ZCL attributes in the cluster.
return (discovered + skipped == total);
}
static void emberAfAttributeDecodeAndPrintCluster(ClusterId cluster, uint16_t mfgCode)
{
#if defined(EMBER_AF_PRINT_ENABLE) && defined(EMBER_AF_PRINT_ATTRIBUTES)
uint16_t index = emberAfFindClusterNameIndexWithMfgCode(cluster, mfgCode);
if (index != 0xFFFF)
{
emberAfAttributesPrintln("(%p)", zclClusterNames[index].name);
}
emberAfAttributesFlush();
#endif // defined(EMBER_AF_PRINT_ENABLE) && defined(EMBER_AF_PRINT_ATTRIBUTES)
}
void emberAfPrintAttributeTable(void)
{
uint8_t data[ATTRIBUTE_LARGEST];
decltype(emberAfEndpointCount()) endpointIndex;
decltype(EmberAfEndpointType::clusterCount) clusterIndex;
uint16_t attributeIndex;
EmberAfStatus status;
uint16_t mfgCode;
for (endpointIndex = 0; endpointIndex < emberAfEndpointCount(); endpointIndex++)
{
EmberAfDefinedEndpoint * ep = &(emAfEndpoints[endpointIndex]);
emberAfAttributesPrintln("ENDPOINT %x", ep->endpoint);
emberAfAttributesPrintln("clus / side / attr / mfg /type(len)/ rw / storage / data (raw)");
emberAfAttributesFlush();
for (clusterIndex = 0; clusterIndex < ep->endpointType->clusterCount; clusterIndex++)
{
EmberAfCluster * cluster = &(ep->endpointType->cluster[clusterIndex]);
for (attributeIndex = 0; attributeIndex < cluster->attributeCount; attributeIndex++)
{
EmberAfAttributeMetadata * metaData = &(cluster->attributes[attributeIndex]);
// Depending on user config, this loop can take a very long time to
// run and watchdog reset will kick in. As a workaround, we'll
// manually reset the watchdog.
// halResetWatchdog();
emberAfAttributesPrint("%2x / %p / %2x / ", cluster->clusterId,
(emberAfAttributeIsClient(metaData) ? "clnt" : "srvr"), metaData->attributeId);
mfgCode = emAfGetManufacturerCodeForAttribute(cluster, metaData);
if (mfgCode == EMBER_AF_NULL_MANUFACTURER_CODE)
{
emberAfAttributesPrint("----");
}
else
{
emberAfAttributesPrint("%2x", mfgCode);
}
emberAfAttributesPrint(" / %x (%x) / %p / %p / ", metaData->attributeType, emberAfAttributeSize(metaData),
(emberAfAttributeIsReadOnly(metaData) ? "RO" : "RW"),
(emberAfAttributeIsTokenized(metaData)
? " token "
: (emberAfAttributeIsExternal(metaData) ? "extern " : " RAM ")));
emberAfAttributesFlush();
status = emAfReadAttribute(ep->endpoint, cluster->clusterId, metaData->attributeId,
(emberAfAttributeIsClient(metaData) ? CLUSTER_MASK_CLIENT : CLUSTER_MASK_SERVER),
mfgCode, data, ATTRIBUTE_LARGEST, NULL);
if (status == EMBER_ZCL_STATUS_UNSUPPORTED_ATTRIBUTE)
{
emberAfAttributesPrintln("Unsupported");
}
else
{
uint16_t length;
if (emberAfIsStringAttributeType(metaData->attributeType))
{
length = static_cast<uint16_t>(emberAfStringLength(data) + 1);
}
else if (emberAfIsLongStringAttributeType(metaData->attributeType))
{
length = static_cast<uint16_t>(emberAfLongStringLength(data) + 2);
}
else
{
length = emberAfAttributeSize(metaData);
}
UNUSED_VAR(length);
emberAfAttributesPrintBuffer(data, length, true);
emberAfAttributesFlush();
emberAfAttributeDecodeAndPrintCluster(cluster->clusterId, mfgCode);
}
}
}
emberAfAttributesFlush();
}
}
// given a clusterId and an attribute to read, this crafts the response
// and places it in the response buffer. Response is one of two items:
// 1) unsupported: [attrId:2] [status:1]
// 2) supported: [attrId:2] [status:1] [type:1] [data:n]
//
void emberAfRetrieveAttributeAndCraftResponse(EndpointId endpoint, ClusterId clusterId, AttributeId attrId, uint8_t mask,
uint16_t manufacturerCode, uint16_t readLength)
{
EmberAfStatus status;
uint8_t data[ATTRIBUTE_LARGEST];
uint8_t dataType;
uint16_t dataLen;
// account for at least one byte of data
if (readLength < 5)
{
return;
}
emberAfAttributesPrintln("OTA READ: ep:%x cid:%2x attid:%2x msk:%x mfcode:%2x", endpoint, clusterId, attrId, mask,
manufacturerCode);
// lookup the attribute in our table
status = emAfReadAttribute(endpoint, clusterId, attrId, mask, manufacturerCode, data, ATTRIBUTE_LARGEST, &dataType);
if (status == EMBER_ZCL_STATUS_SUCCESS)
{
dataLen = emberAfAttributeValueSize(dataType, data);
if ((readLength - 4) < dataLen)
{ // Not enough space for attribute.
return;
}
}
else
{
emberAfPutInt16uInResp(attrId);
emberAfPutInt8uInResp(status);
emberAfAttributesPrintln("READ: clus %2x, attr %2x failed %x", clusterId, attrId, status);
emberAfAttributesFlush();
return;
}
// put attribute in least sig byte first
emberAfPutInt16uInResp(attrId);
// attribute is found, so copy in the status and the data type
emberAfPutInt8uInResp(EMBER_ZCL_STATUS_SUCCESS);
emberAfPutInt8uInResp(dataType);
if (dataLen < (EMBER_AF_RESPONSE_BUFFER_LEN - appResponseLength))
{
#if (BIGENDIAN_CPU)
// strings go over the air as length byte and then in human
// readable format. These should not be flipped. Other attributes
// need to be flipped so they go little endian OTA
if (isThisDataTypeSentLittleEndianOTA(dataType))
{
uint8_t i;
for (i = 0; i < dataLen; i++)
{
appResponseData[appResponseLength + i] = data[dataLen - i - 1];
}
}
else
{
memmove(&(appResponseData[appResponseLength]), data, dataLen);
}
#else //(BIGENDIAN_CPU)
memmove(&(appResponseData[appResponseLength]), data, dataLen);
#endif //(BIGENDIAN_CPU)
// TODO: How do we know this does not overflow?
appResponseLength = static_cast<uint16_t>(appResponseLength + dataLen);
}
emberAfAttributesPrintln("READ: clus %2x, attr %2x, dataLen: %x, OK", clusterId, attrId, dataLen);
emberAfAttributesFlush();
}
// This function appends the attribute report fields for the given endpoint,
// cluster, and attribute to the buffer starting at the index. If there is
// insufficient space in the buffer or an error occurs, buffer and bufIndex will
// remain unchanged. Otherwise, bufIndex will be incremented appropriately and
// the fields will be written to the buffer.
EmberAfStatus emberAfAppendAttributeReportFields(EndpointId endpoint, ClusterId clusterId, AttributeId attributeId, uint8_t mask,
uint8_t * buffer, uint8_t bufLen, uint8_t * bufIndex)
{
EmberAfStatus status;
EmberAfAttributeType type;
uint16_t size;
uint16_t bufLen16 = (uint16_t) bufLen;
uint8_t data[ATTRIBUTE_LARGEST];
status = emberAfReadAttribute(endpoint, clusterId, attributeId, mask, data, sizeof(data), &type);
if (status != EMBER_ZCL_STATUS_SUCCESS)
{
goto kickout;
}
size = emberAfAttributeValueSize(type, data);
if (bufLen16 - *bufIndex < 3 || size > bufLen16 - (*bufIndex + 3))
{
status = EMBER_ZCL_STATUS_INSUFFICIENT_SPACE;
goto kickout;
}
buffer[(*bufIndex)++] = LOW_BYTE(attributeId);
buffer[(*bufIndex)++] = HIGH_BYTE(attributeId);
buffer[(*bufIndex)++] = type;
#if (BIGENDIAN_CPU)
if (isThisDataTypeSentLittleEndianOTA(type))
{
emberReverseMemCopy(buffer + *bufIndex, data, size);
}
else
{
memmove(buffer + *bufIndex, data, size);
}
#else
memmove(buffer + *bufIndex, data, size);
#endif
*bufIndex = static_cast<uint8_t>(*bufIndex + size);
kickout:
emberAfAttributesPrintln("REPORT: clus 0x%2x, attr 0x%2x: 0x%x", clusterId, attributeId, status);
emberAfAttributesFlush();
return status;
}
//------------------------------------------------------------------------------
// Internal Functions
// writes an attribute (identified by clusterID and attrID to the given value.
// this returns:
// - EMBER_ZCL_STATUS_UNSUPPORTED_ATTRIBUTE: if attribute isnt supported by the device (the
// device is not found in the attribute table)
// - EMBER_ZCL_STATUS_INVALID_DATA_TYPE: if the data type passed in doesnt match the type
// stored in the attribute table
// - EMBER_ZCL_STATUS_READ_ONLY: if the attribute isnt writable
// - EMBER_ZCL_STATUS_INVALID_VALUE: if the value is set out of the allowable range for
// the attribute
// - EMBER_ZCL_STATUS_SUCCESS: if the attribute was found and successfully written
//
// if true is passed in for overrideReadOnlyAndDataType then the data type is
// not checked and the read-only flag is ignored. This mode is meant for
// testing or setting the initial value of the attribute on the device.
//
// if true is passed for justTest, then the type is not written but all
// checks are done to see if the type could be written
// reads the attribute specified, returns false if the attribute is not in
// the table or the data is too large, returns true and writes to dataPtr
// if the attribute is supported and the readLength specified is less than
// the length of the data.
EmberAfStatus emAfWriteAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t mask,
uint16_t manufacturerCode, uint8_t * data, EmberAfAttributeType dataType,
bool overrideReadOnlyAndDataType, bool justTest)
{
EmberAfAttributeMetadata * metadata = NULL;
EmberAfAttributeSearchRecord record;
record.endpoint = endpoint;
record.clusterId = cluster;
record.clusterMask = mask;
record.attributeId = attributeID;
record.manufacturerCode = manufacturerCode;
emAfReadOrWriteAttribute(&record, &metadata,
NULL, // buffer
0, // buffer size
false); // write?
// if we dont support that attribute
if (metadata == NULL)
{
emberAfAttributesPrintln("%pep %x clus %2x attr %2x not supported", "WRITE ERR: ", endpoint, cluster, attributeID);
emberAfAttributesFlush();
return EMBER_ZCL_STATUS_UNSUPPORTED_ATTRIBUTE;
}
// if the data type specified by the caller is incorrect
if (!(overrideReadOnlyAndDataType))
{
if (dataType != metadata->attributeType)
{
emberAfAttributesPrintln("%pinvalid data type", "WRITE ERR: ");
emberAfAttributesFlush();
return EMBER_ZCL_STATUS_INVALID_DATA_TYPE;
}
if (emberAfAttributeIsReadOnly(metadata))
{
emberAfAttributesPrintln("%pattr not writable", "WRITE ERR: ");
emberAfAttributesFlush();
return EMBER_ZCL_STATUS_READ_ONLY;
}
}
// if the value the attribute is being set to is out of range
// return EMBER_ZCL_STATUS_INVALID_VALUE
if ((metadata->mask & ATTRIBUTE_MASK_MIN_MAX) != 0U)
{
EmberAfDefaultAttributeValue minv = metadata->defaultValue.ptrToMinMaxValue->minValue;
EmberAfDefaultAttributeValue maxv = metadata->defaultValue.ptrToMinMaxValue->maxValue;
bool isAttributeSigned = emberAfIsTypeSigned(metadata->attributeType);
uint8_t dataLen = emberAfAttributeSize(metadata);
if (dataLen <= 2)
{
int8_t minR, maxR;
uint8_t * minI = (uint8_t *) &(minv.defaultValue);
uint8_t * maxI = (uint8_t *) &(maxv.defaultValue);
// On big endian cpu with length 1 only the second byte counts
#if (BIGENDIAN_CPU)
if (dataLen == 1)
{
minI++;
maxI++;
}
#endif // BIGENDIAN_CPU
minR = emberAfCompareValues(minI, data, dataLen, isAttributeSigned);
maxR = emberAfCompareValues(maxI, data, dataLen, isAttributeSigned);
if ((minR == 1) || (maxR == -1))
{
return EMBER_ZCL_STATUS_INVALID_VALUE;
}
}
else
{
if ((emberAfCompareValues(minv.ptrToDefaultValue, data, dataLen, isAttributeSigned) == 1) ||
(emberAfCompareValues(maxv.ptrToDefaultValue, data, dataLen, isAttributeSigned) == -1))
{
return EMBER_ZCL_STATUS_INVALID_VALUE;
}
}
}
// write the data unless this is only a test
if (!justTest)
{
// Pre write attribute callback for all attribute changes,
// regardless of cluster.
EmberAfStatus status = emberAfPreAttributeChangeCallback(endpoint, cluster, attributeID, mask, manufacturerCode, dataType,
emberAfAttributeSize(metadata), data);
if (status != EMBER_ZCL_STATUS_SUCCESS)
{
return status;
}
// Pre-write attribute callback specific
// to the cluster that the attribute lives in.
status = emAfClusterPreAttributeChangedCallback(endpoint, cluster, attributeID, mask, manufacturerCode, dataType,
emberAfAttributeSize(metadata), data);
if (status != EMBER_ZCL_STATUS_SUCCESS)
{
return status;
}
// write the attribute
status = emAfReadOrWriteAttribute(&record,
NULL, // metadata
data,
0, // buffer size - unused
true); // write?
if (status != EMBER_ZCL_STATUS_SUCCESS)
{
return status;
}
// Save the attribute to token if needed
// Function itself will weed out tokens that are not tokenized.
emAfSaveAttributeToToken(data, endpoint, cluster, metadata);
#ifdef EMBER_AF_PLUGIN_REPORTING
emberAfReportingAttributeChangeCallback(endpoint, cluster, attributeID, mask, manufacturerCode, dataType, data);
#endif // EMBER_AF_PLUGIN_REPORTING
// Post write attribute callback for all attributes changes, regardless
// of cluster.
emberAfPostAttributeChangeCallback(endpoint, cluster, attributeID, mask, manufacturerCode, dataType,
emberAfAttributeSize(metadata), data);
// Post-write attribute callback specific
// to the cluster that the attribute lives in.
emAfClusterAttributeChangedCallback(endpoint, cluster, attributeID, mask, manufacturerCode);
}
else
{
// bug: 11618, we are not handling properly external attributes
// in this case... We need to do something. We don't really
// know if it will succeed.
emberAfAttributesPrintln("WRITE: no write, just a test");
emberAfAttributesFlush();
}
return EMBER_ZCL_STATUS_SUCCESS;
}
// If dataPtr is NULL, no data is copied to the caller.
// readLength should be 0 in that case.
EmberAfStatus emAfReadAttribute(EndpointId endpoint, ClusterId cluster, AttributeId attributeID, uint8_t mask,
uint16_t manufacturerCode, uint8_t * dataPtr, uint16_t readLength, EmberAfAttributeType * dataType)
{
EmberAfAttributeMetadata * metadata = NULL;
EmberAfAttributeSearchRecord record;
EmberAfStatus status;
record.endpoint = endpoint;
record.clusterId = cluster;
record.clusterMask = mask;
record.attributeId = attributeID;
record.manufacturerCode = manufacturerCode;
status = emAfReadOrWriteAttribute(&record, &metadata, dataPtr, readLength,
false); // write?
if (status == EMBER_ZCL_STATUS_SUCCESS)
{
// It worked! If the user asked for the type, set it before returning.
if (dataType != NULL)
{
(*dataType) = metadata->attributeType;
}
}
else
{ // failed, print debug info
if (status == EMBER_ZCL_STATUS_INSUFFICIENT_SPACE)
{
emberAfAttributesPrintln("READ: attribute size too large for caller");
emberAfAttributesFlush();
}
}
return status;
}