Skip to content

Commit

Permalink
Fix the error code returned when writing global list attributes.
Browse files Browse the repository at this point in the history
It should be UNSUPPORTED_WRITE, not UNSUPPORTED_ATTRIBUTE.

Fixes #31448
  • Loading branch information
bzbarsky-apple committed Jan 16, 2024
1 parent 72fd9e9 commit 2f96124
Show file tree
Hide file tree
Showing 4 changed files with 76 additions and 829 deletions.
2 changes: 2 additions & 0 deletions examples/darwin-framework-tool/templates/tests/ciTests.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@
"disable": [
"Test_TC_SC_4_1",
"Test_TC_SC_5_2",
"TestBasicInformation disabled because codegen test don't support writing readonly attributes",
"TestBasicInformation",
"TestClusterComplexTypes",
"TestEvents",
"TestDiscovery",
Expand Down
40 changes: 40 additions & 0 deletions src/app/tests/suites/TestBasicInformation.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -215,3 +215,43 @@ tests:
response:
# For now all-clusters-app only supports 1 max paths per invoke.
value: 1

- label: "Write global ClusterRevision attribute"
command: "writeAttribute"
attribute: "ClusterRevision"
arguments:
value: 1
response:
error: UNSUPPORTED_WRITE

- label: "Write global FeatureMap attribute"
command: "writeAttribute"
attribute: "FeatureMap"
arguments:
value: 1
response:
error: UNSUPPORTED_WRITE

- label: "Write global AttributeList attribute"
command: "writeAttribute"
attribute: "AttributeList"
arguments:
value: []
response:
error: UNSUPPORTED_WRITE

- label: "Write global GeneratedCommandList attribute"
command: "writeAttribute"
attribute: "GeneratedCommandList"
arguments:
value: []
response:
error: UNSUPPORTED_WRITE

- label: "Write global AcceptedCommandList attribute"
command: "writeAttribute"
attribute: "AcceptedCommandList"
arguments:
value: []
response:
error: UNSUPPORTED_WRITE
53 changes: 34 additions & 19 deletions src/app/util/ember-compatibility-functions.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -503,6 +503,31 @@ Protocols::InteractionModel::Status UnsupportedAttributeStatus(const ConcreteAtt
return Status::UnsupportedAttribute;
}

// Will set exactly one of the out-params (aAttributeCluster or
// aAttributeMetadata) to non-null.
void FindAttributeMetadata(const ConcreteAttributePath & aPath, const EmberAfCluster ** aAttributeCluster,
const EmberAfAttributeMetadata ** aAttributeMetadata)
{
*aAttributeCluster = nullptr;
*aAttributeMetadata = nullptr;

bool isGlobalAttributeNotInMetadata = false;
for (auto & attr : GlobalAttributesNotInMetadata)
{
if (attr == aPath.mAttributeId)
{
isGlobalAttributeNotInMetadata = true;
*aAttributeCluster = emberAfFindServerCluster(aPath.mEndpointId, aPath.mClusterId);
break;
}
}

if (!isGlobalAttributeNotInMetadata)
{
*aAttributeMetadata = emberAfLocateAttributeMetadata(aPath.mEndpointId, aPath.mClusterId, aPath.mAttributeId);
}
}

} // anonymous namespace

bool ConcreteAttributePathExists(const ConcreteAttributePath & aPath)
Expand Down Expand Up @@ -530,22 +555,7 @@ CHIP_ERROR ReadSingleClusterData(const SubjectDescriptor & aSubjectDescriptor, b

const EmberAfCluster * attributeCluster = nullptr;
const EmberAfAttributeMetadata * attributeMetadata = nullptr;

bool isGlobalAttributeNotInMetadata = false;
for (auto & attr : GlobalAttributesNotInMetadata)
{
if (attr == aPath.mAttributeId)
{
isGlobalAttributeNotInMetadata = true;
attributeCluster = emberAfFindServerCluster(aPath.mEndpointId, aPath.mClusterId);
break;
}
}

if (!isGlobalAttributeNotInMetadata)
{
attributeMetadata = emberAfLocateAttributeMetadata(aPath.mEndpointId, aPath.mClusterId, aPath.mAttributeId);
}
FindAttributeMetadata(aPath, &attributeCluster, &attributeMetadata);

if (attributeCluster == nullptr && attributeMetadata == nullptr)
{
Expand Down Expand Up @@ -972,14 +982,19 @@ const EmberAfAttributeMetadata * GetAttributeMetadata(const ConcreteAttributePat
CHIP_ERROR WriteSingleClusterData(const SubjectDescriptor & aSubjectDescriptor, const ConcreteDataAttributePath & aPath,
TLV::TLVReader & aReader, WriteHandler * apWriteHandler)
{
const EmberAfAttributeMetadata * attributeMetadata = GetAttributeMetadata(aPath);
// Check attribute existence. This includes attributes with registered metadata, but also specially handled
// mandatory global attributes (which just check for cluster on endpoint).
const EmberAfCluster * attributeCluster = nullptr;
const EmberAfAttributeMetadata * attributeMetadata = nullptr;
FindAttributeMetadata(aPath, &attributeCluster, &attributeMetadata);

if (attributeMetadata == nullptr)
if (attributeCluster == nullptr && attributeMetadata == nullptr)
{
return apWriteHandler->AddStatus(aPath, UnsupportedAttributeStatus(aPath));
}

if (attributeMetadata->IsReadOnly())
// All the global attributes we don't have metadata for are readonly.
if (attributeMetadata == nullptr || attributeMetadata->IsReadOnly())
{
return apWriteHandler->AddStatus(aPath, Protocols::InteractionModel::Status::UnsupportedWrite);
}
Expand Down
Loading

0 comments on commit 2f96124

Please sign in to comment.