Skip to content

Commit 4a66e63

Browse files
committed
Use ProviderChangeListener directly to report attrbiute change
1 parent 45f544b commit 4a66e63

File tree

9 files changed

+88
-42
lines changed

9 files changed

+88
-42
lines changed

src/app/codegen-data-model-provider/CodegenDataModelProvider.h

+20-1
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,18 @@ class EnumeratorCommandFinder
8080
}
8181
};
8282

83+
class DefaultProviderChangeListener : public DataModel::ProviderChangeListener
84+
{
85+
public:
86+
explicit DefaultProviderChangeListener(DataModel::Provider & provider) : mProvider(provider)
87+
{}
88+
89+
void MarkDirty(const AttributePathParams & path) override;
90+
91+
private:
92+
DataModel::Provider & mProvider;
93+
};
94+
8395
} // namespace detail
8496

8597
/// An implementation of `InteractionModel::Model` that relies on code-generation
@@ -126,6 +138,8 @@ class CodegenDataModelProvider : public DataModel::Provider
126138
};
127139

128140
public:
141+
CodegenDataModelProvider() : mChangeListener(*this) {}
142+
129143
/// clears out internal caching. Especially useful in unit tests,
130144
/// where path caching does not really apply (the same path may result in different outcomes)
131145
void Reset()
@@ -185,7 +199,10 @@ class CodegenDataModelProvider : public DataModel::Provider
185199
ConcreteCommandPath FirstGeneratedCommand(const ConcreteClusterPath & cluster) override;
186200
ConcreteCommandPath NextGeneratedCommand(const ConcreteCommandPath & before) override;
187201

188-
void Temporary_ReportAttributeChanged(const AttributePathParams & path) override;
202+
DataModel::ProviderChangeListener & GetAttributeChangeReporter() override
203+
{
204+
return mChangeListener;
205+
}
189206

190207
private:
191208
// Iteration is often done in a tight loop going through all values.
@@ -221,6 +238,8 @@ class CodegenDataModelProvider : public DataModel::Provider
221238
// Ember requires a persistence provider, so we make sure we can always have something
222239
PersistentStorageDelegate * mPersistentStorageDelegate = nullptr;
223240

241+
detail::DefaultProviderChangeListener mChangeListener;
242+
224243
/// Finds the specified ember cluster
225244
///
226245
/// Effectively the same as `emberAfFindServerCluster` except with some caching capabilities

src/app/codegen-data-model-provider/CodegenDataModelProvider_Write.cpp

+20-16
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,26 @@ std::optional<CHIP_ERROR> TryWriteViaAccessInterface(const ConcreteDataAttribute
8888

8989
} // namespace
9090

91+
namespace detail {
92+
93+
void DefaultProviderChangeListener::MarkDirty(const AttributePathParams & path)
94+
{
95+
ContextAttributesChangeListener change_listener(mProvider.CurrentContext());
96+
if (path.mClusterId != kInvalidClusterId)
97+
{
98+
emberAfAttributeChanged(path.mEndpointId, path.mClusterId, path.mAttributeId, &change_listener);
99+
}
100+
else
101+
{
102+
// When the path has wildcard cluster Id, call the emberAfEndpointChanged to mark attributes on the given endpoint
103+
// as having changing, but do NOT increase/alter any cluster data versions, as this happens when a bridged endpoint is
104+
// added or removed from a bridge and the cluster data is not changed during the process.
105+
emberAfEndpointChanged(path.mEndpointId, &change_listener);
106+
}
107+
}
108+
109+
} // namespace detail
110+
91111
DataModel::ActionReturnStatus CodegenDataModelProvider::WriteAttribute(const DataModel::WriteAttributeRequest & request,
92112
AttributeValueDecoder & decoder)
93113
{
@@ -261,21 +281,5 @@ DataModel::ActionReturnStatus CodegenDataModelProvider::WriteAttribute(const Dat
261281
return CHIP_NO_ERROR;
262282
}
263283

264-
void CodegenDataModelProvider::Temporary_ReportAttributeChanged(const AttributePathParams & path)
265-
{
266-
ContextAttributesChangeListener change_listener(CurrentContext());
267-
if (path.mClusterId != kInvalidClusterId)
268-
{
269-
emberAfAttributeChanged(path.mEndpointId, path.mClusterId, path.mAttributeId, &change_listener);
270-
}
271-
else
272-
{
273-
// When the path has wildcard cluster Id, call the emberAfEndpointChanged to mark attributes on the given endpoint
274-
// as having changing, but do NOT increase/alter any cluster data versions, as this happens when a bridged endpoint is
275-
// added or removed from a bridge and the cluster data is not changed during the process.
276-
emberAfEndpointChanged(path.mEndpointId, &change_listener);
277-
}
278-
}
279-
280284
} // namespace app
281285
} // namespace chip

src/app/data-model-provider/MetadataTypes.h

-18
Original file line numberDiff line numberDiff line change
@@ -221,24 +221,6 @@ class ProviderMetadataTree
221221
// returned as responses.
222222
virtual ConcreteCommandPath FirstGeneratedCommand(const ConcreteClusterPath & cluster) = 0;
223223
virtual ConcreteCommandPath NextGeneratedCommand(const ConcreteCommandPath & before) = 0;
224-
225-
/// Workaround function to report attribute change.
226-
///
227-
/// When this is invoked, the caller is expected to increment the cluster data version, and the attribute path
228-
/// should be marked as `dirty` by the data model provider listener so that the reporter can notify the subscriber
229-
/// of attribute changes.
230-
/// This function should be invoked when attribute managed by attribute access interface is modified but not
231-
/// through an actual Write interaction.
232-
/// For example, if the LastNetworkingStatus attribute changes because the NetworkCommissioning driver detects a
233-
/// network connection status change and calls SetLastNetworkingStatusValue(). The data model provider can recognize
234-
/// this change by invoking this function at the point of change.
235-
///
236-
/// This is a workaround function as we cannot notify the attribute change to the data model provider. The provider
237-
/// should own its data and versions.
238-
///
239-
/// TODO: We should remove this function when the AttributeAccessInterface/CommandHandlerInterface is able to report
240-
/// the attribute changes.
241-
virtual void Temporary_ReportAttributeChanged(const AttributePathParams & path) = 0;
242224
};
243225

244226
} // namespace DataModel

src/app/data-model-provider/Provider.h

+9
Original file line numberDiff line numberDiff line change
@@ -97,6 +97,15 @@ class Provider : public ProviderMetadataTree
9797
virtual std::optional<ActionReturnStatus> Invoke(const InvokeRequest & request, chip::TLV::TLVReader & input_arguments,
9898
CommandHandler * handler) = 0;
9999

100+
/// Returns a reference to the ProviderChangeListener used for reporting attribute changes.
101+
///
102+
/// This is used to notify listeners about changes in attributes managed by the provider.
103+
/// Listeners can be used to react to updates in the underlying data model.
104+
///
105+
/// It is expected that implementations provide a valid reference, ensuring that
106+
/// attribute change reporting is functional throughout the lifecycle of the Provider.
107+
virtual ProviderChangeListener & GetAttributeChangeReporter() = 0;
108+
100109
private:
101110
InteractionModelContext mContext = { nullptr };
102111
};

src/app/reporting/reporting.cpp

+9-5
Original file line numberDiff line numberDiff line change
@@ -30,8 +30,9 @@ void MatterReportingAttributeChangeCallback(EndpointId endpoint, ClusterId clust
3030
// applications notifying about changes from their end.
3131
assertChipStackLockedByCurrentThread();
3232

33-
InteractionModelEngine::GetInstance()->GetDataModelProvider()->Temporary_ReportAttributeChanged(
34-
AttributePathParams(endpoint, clusterId, attributeId));
33+
DataModel::ProviderChangeListener & changeListener =
34+
InteractionModelEngine::GetInstance()->GetDataModelProvider()->GetAttributeChangeReporter();
35+
changeListener.MarkDirty(AttributePathParams(endpoint, clusterId, attributeId));
3536
}
3637

3738
void MatterReportingAttributeChangeCallback(const ConcreteAttributePath & aPath)
@@ -40,8 +41,9 @@ void MatterReportingAttributeChangeCallback(const ConcreteAttributePath & aPath)
4041
// applications notifying about changes from their end.
4142
assertChipStackLockedByCurrentThread();
4243

43-
InteractionModelEngine::GetInstance()->GetDataModelProvider()->Temporary_ReportAttributeChanged(
44-
AttributePathParams(aPath.mEndpointId, aPath.mClusterId, aPath.mAttributeId));
44+
DataModel::ProviderChangeListener & changeListener =
45+
InteractionModelEngine::GetInstance()->GetDataModelProvider()->GetAttributeChangeReporter();
46+
changeListener.MarkDirty(AttributePathParams(aPath.mEndpointId, aPath.mClusterId, aPath.mAttributeId));
4547
}
4648

4749
void MatterReportingAttributeChangeCallback(EndpointId endpoint)
@@ -50,5 +52,7 @@ void MatterReportingAttributeChangeCallback(EndpointId endpoint)
5052
// applications notifying about changes from their end.
5153
assertChipStackLockedByCurrentThread();
5254

53-
InteractionModelEngine::GetInstance()->GetDataModelProvider()->Temporary_ReportAttributeChanged(AttributePathParams(endpoint));
55+
DataModel::ProviderChangeListener & changeListener =
56+
InteractionModelEngine::GetInstance()->GetDataModelProvider()->GetAttributeChangeReporter();
57+
changeListener.MarkDirty(AttributePathParams(endpoint));
5458
}

src/app/tests/test-interaction-model-api.cpp

+14
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,14 @@ class TestOnlyAttributeValueDecoderAccessor
6060
AttributeValueDecoder & mDecoder;
6161
};
6262

63+
class TestProviderChangeListener : public DataModel::ProviderChangeListener
64+
{
65+
public:
66+
explicit TestProviderChangeListener() {}
67+
68+
void MarkDirty(const AttributePathParams & path) override {}
69+
};
70+
6371
// strong defintion in TestCommandInteraction.cpp
6472
__attribute__((weak)) void DispatchSingleClusterCommand(const ConcreteCommandPath & aRequestCommandPath,
6573
chip::TLV::TLVReader & aReader, CommandHandler * apCommandObj)
@@ -149,6 +157,12 @@ std::optional<ActionReturnStatus> TestImCustomDataModel::Invoke(const InvokeRequ
149157
return std::make_optional<ActionReturnStatus>(CHIP_ERROR_NOT_IMPLEMENTED);
150158
}
151159

160+
ProviderChangeListener & TestImCustomDataModel::GetAttributeChangeReporter()
161+
{
162+
static TestProviderChangeListener changeListener;
163+
return changeListener;
164+
}
165+
152166
DataModel::EndpointEntry TestImCustomDataModel::FirstEndpoint()
153167
{
154168
return CodegenDataModelProviderInstance(nullptr /* delegate */)->FirstEndpoint();

src/app/tests/test-interaction-model-api.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -109,6 +109,7 @@ class TestImCustomDataModel : public DataModel::Provider
109109
AttributeValueDecoder & decoder) override;
110110
std::optional<DataModel::ActionReturnStatus> Invoke(const DataModel::InvokeRequest & request,
111111
chip::TLV::TLVReader & input_arguments, CommandHandler * handler) override;
112+
DataModel::ProviderChangeListener & GetAttributeChangeReporter() override;
112113

113114
DataModel::EndpointEntry FirstEndpoint() override;
114115
DataModel::EndpointEntry NextEndpoint(EndpointId before) override;
@@ -131,7 +132,6 @@ class TestImCustomDataModel : public DataModel::Provider
131132
std::optional<DataModel::CommandInfo> GetAcceptedCommandInfo(const ConcreteCommandPath & path) override;
132133
ConcreteCommandPath FirstGeneratedCommand(const ConcreteClusterPath & cluster) override;
133134
ConcreteCommandPath NextGeneratedCommand(const ConcreteCommandPath & before) override;
134-
void Temporary_ReportAttributeChanged(const AttributePathParams & path) override {}
135135
};
136136

137137
} // namespace app

src/controller/tests/data_model/DataModelFixtures.cpp

+14
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,14 @@ class TestOnlyAttributeValueDecoderAccessor
6767
AttributeValueDecoder & mDecoder;
6868
};
6969

70+
class TestProviderChangeListener : public DataModel::ProviderChangeListener
71+
{
72+
public:
73+
explicit TestProviderChangeListener() {}
74+
75+
void MarkDirty(const AttributePathParams & path) override {}
76+
};
77+
7078
namespace DataModelTests {
7179

7280
ScopedChangeOnly<ReadResponseDirective> gReadResponseDirective(ReadResponseDirective::kSendDataResponse);
@@ -474,6 +482,12 @@ std::optional<ActionReturnStatus> CustomDataModel::Invoke(const InvokeRequest &
474482
return std::nullopt; // handler status is set by the dispatch
475483
}
476484

485+
ProviderChangeListener & CustomDataModel::GetAttributeChangeReporter()
486+
{
487+
static TestProviderChangeListener changeListener;
488+
return changeListener;
489+
}
490+
477491
DataModel::EndpointEntry CustomDataModel::FirstEndpoint()
478492
{
479493
return CodegenDataModelProviderInstance(nullptr /* delegate */)->FirstEndpoint();

src/controller/tests/data_model/DataModelFixtures.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,7 @@ class CustomDataModel : public DataModel::Provider
121121
AttributeValueDecoder & decoder) override;
122122
std::optional<DataModel::ActionReturnStatus> Invoke(const DataModel::InvokeRequest & request,
123123
chip::TLV::TLVReader & input_arguments, CommandHandler * handler) override;
124+
DataModel::ProviderChangeListener & GetAttributeChangeReporter() override;
124125

125126
DataModel::EndpointEntry FirstEndpoint() override;
126127
DataModel::EndpointEntry NextEndpoint(EndpointId before) override;
@@ -143,7 +144,6 @@ class CustomDataModel : public DataModel::Provider
143144
std::optional<DataModel::CommandInfo> GetAcceptedCommandInfo(const ConcreteCommandPath & path) override;
144145
ConcreteCommandPath FirstGeneratedCommand(const ConcreteClusterPath & cluster) override;
145146
ConcreteCommandPath NextGeneratedCommand(const ConcreteCommandPath & before) override;
146-
void Temporary_ReportAttributeChanged(const AttributePathParams & path) override {}
147147
};
148148

149149
} // namespace DataModelTests

0 commit comments

Comments
 (0)