Skip to content

Commit 52939ca

Browse files
Finish refactor of the Sub logic when interacting with the ICDManager
1 parent feaec15 commit 52939ca

10 files changed

+312
-33
lines changed

src/app/BUILD.gn

+1
Original file line numberDiff line numberDiff line change
@@ -201,6 +201,7 @@ static_library("interaction-model") {
201201
":subscription-info-provider",
202202
"${chip_root}/src/app/MessageDef",
203203
"${chip_root}/src/app/icd/server:icd-server-config",
204+
"${chip_root}/src/app/icd/server:manager",
204205
"${chip_root}/src/app/icd/server:observer",
205206
"${chip_root}/src/lib/address_resolve",
206207
"${chip_root}/src/lib/support",

src/app/InteractionModelEngine.cpp

+28-7
Original file line numberDiff line numberDiff line change
@@ -364,8 +364,10 @@ bool InteractionModelEngine::SubjectHasPersistedSubscription(FabricIndex aFabric
364364

365365
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
366366
auto * iterator = mpSubscriptionResumptionStorage->IterateSubscriptions();
367-
SubscriptionResumptionStorage::SubscriptionInfo subscriptionInfo;
367+
// Verify that we were able t0 allocate an iterator. If not, assume we have a persisted subscription.
368+
VerifyOrReturnValue(iterator, true);
368369

370+
SubscriptionResumptionStorage::SubscriptionInfo subscriptionInfo;
369371
while (iterator->Next(subscriptionInfo))
370372
{
371373
// TODO(#31873): Persistent subscription only stores the NodeID for now. We cannot check if the CAT matches
@@ -1914,22 +1916,22 @@ CHIP_ERROR InteractionModelEngine::ResumeSubscriptions()
19141916
// future improvements: https://github.com/project-chip/connectedhomeip/issues/25439
19151917

19161918
SubscriptionResumptionStorage::SubscriptionInfo subscriptionInfo;
1917-
auto * iterator = mpSubscriptionResumptionStorage->IterateSubscriptions();
1918-
int subscriptionsToResume = 0;
1919-
uint16_t minInterval = 0;
1919+
auto * iterator = mpSubscriptionResumptionStorage->IterateSubscriptions();
1920+
mNumOfSubscriptionToResume = 0;
1921+
uint16_t minInterval = 0;
19201922
while (iterator->Next(subscriptionInfo))
19211923
{
1922-
subscriptionsToResume++;
1924+
mNumOfSubscriptionToResume++;
19231925
minInterval = std::max(minInterval, subscriptionInfo.mMinInterval);
19241926
}
19251927
iterator->Release();
19261928

1927-
if (subscriptionsToResume)
1929+
if (mNumOfSubscriptionToResume)
19281930
{
19291931
#if CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
19301932
mSubscriptionResumptionScheduled = true;
19311933
#endif
1932-
ChipLogProgress(InteractionModel, "Resuming %d subscriptions in %u seconds", subscriptionsToResume, minInterval);
1934+
ChipLogProgress(InteractionModel, "Resuming %d subscriptions in %u seconds", mNumOfSubscriptionToResume, minInterval);
19331935
ReturnErrorOnFailure(mpExchangeMgr->GetSessionManager()->SystemLayer()->StartTimer(System::Clock::Seconds16(minInterval),
19341936
ResumeSubscriptionsTimerCallback, this));
19351937
}
@@ -2051,5 +2053,24 @@ bool InteractionModelEngine::HasSubscriptionsToResume()
20512053
}
20522054
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS && CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
20532055

2056+
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
2057+
void InteractionModelEngine::DecrementNumSubscriptionToResume()
2058+
{
2059+
VerifyOrReturn(mNumOfSubscriptionToResume > 0);
2060+
#if CHIP_CONFIG_ENABLE_ICD_CIP && !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
2061+
VerifyOrDie(mICDManager);
2062+
#endif // CHIP_CONFIG_ENABLE_ICD_CIP && CHIP_CONFIG_PERSIST_SUBSCRIPTIONS && !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
2063+
2064+
mNumOfSubscriptionToResume--;
2065+
2066+
#if CHIP_CONFIG_ENABLE_ICD_CIP && !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
2067+
if (!mNumOfSubscriptionToResume)
2068+
{
2069+
mICDManager->SetBootUpResumeSubscriptionExecuted();
2070+
}
2071+
#endif // CHIP_CONFIG_ENABLE_ICD_CIP && !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
2072+
}
2073+
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
2074+
20542075
} // namespace app
20552076
} // namespace chip

src/app/InteractionModelEngine.h

+22-2
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@
4949
#include <app/TimedHandler.h>
5050
#include <app/WriteClient.h>
5151
#include <app/WriteHandler.h>
52+
#include <app/icd/server/ICDServerConfig.h>
5253
#include <app/reporting/Engine.h>
5354
#include <app/reporting/ReportScheduler.h>
5455
#include <app/util/attribute-metadata.h>
@@ -68,6 +69,10 @@
6869

6970
#include <app/CASESessionManager.h>
7071

72+
#if CHIP_CONFIG_ENABLE_ICD_SERVER
73+
#include <app/icd/server/ICDManager.h> // nogncheck
74+
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
75+
7176
namespace chip {
7277
namespace app {
7378

@@ -128,6 +133,10 @@ class InteractionModelEngine : public Messaging::UnsolicitedMessageHandler,
128133

129134
void Shutdown();
130135

136+
#if CHIP_CONFIG_ENABLE_ICD_SERVER
137+
void SetICDManager(ICDManager * manager) { mICDManager = manager; };
138+
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
139+
131140
Messaging::ExchangeManager * GetExchangeManager(void) const { return mpExchangeMgr; }
132141

133142
/**
@@ -318,6 +327,10 @@ class InteractionModelEngine : public Messaging::UnsolicitedMessageHandler,
318327

319328
bool SubjectHasPersistedSubscription(FabricIndex aFabricIndex, NodeId subjectID) override;
320329

330+
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
331+
void DecrementNumSubscriptionToResume();
332+
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
333+
321334
#if CONFIG_BUILD_FOR_HOST_UNIT_TEST
322335
//
323336
// Get direct access to the underlying read handler pool
@@ -602,6 +615,10 @@ class InteractionModelEngine : public Messaging::UnsolicitedMessageHandler,
602615

603616
CommandHandlerInterface * mCommandHandlerList = nullptr;
604617

618+
#if CHIP_CONFIG_ENABLE_ICD_SERVER
619+
ICDManager * mICDManager = nullptr;
620+
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
621+
605622
ObjectPool<CommandHandler, CHIP_IM_MAX_NUM_COMMAND_HANDLER> mCommandHandlerObjs;
606623
ObjectPool<TimedHandler, CHIP_IM_MAX_NUM_TIMED_HANDLER> mTimedHandlers;
607624
WriteHandler mWriteHandlers[CHIP_IM_MAX_NUM_WRITE_HANDLER];
@@ -660,12 +677,15 @@ class InteractionModelEngine : public Messaging::UnsolicitedMessageHandler,
660677
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS && CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
661678
#endif // CONFIG_BUILD_FOR_HOST_UNIT_TEST
662679

663-
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS && CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
680+
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
681+
int8_t mNumOfSubscriptionToResume = 0;
682+
#if CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
664683
bool HasSubscriptionsToResume();
665684
uint32_t ComputeTimeSecondsTillNextSubscriptionResumption();
666685
uint32_t mNumSubscriptionResumptionRetries = 0;
667686
bool mSubscriptionResumptionScheduled = false;
668-
#endif
687+
#endif // CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
688+
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
669689

670690
FabricTable * mpFabricTable;
671691

src/app/SubscriptionResumptionSessionEstablisher.cpp

+12-1
Original file line numberDiff line numberDiff line change
@@ -90,15 +90,21 @@ void SubscriptionResumptionSessionEstablisher::HandleDeviceConnected(void * cont
9090
AutoDeleteEstablisher establisher(static_cast<SubscriptionResumptionSessionEstablisher *>(context));
9191
SubscriptionResumptionStorage::SubscriptionInfo & subscriptionInfo = establisher->mSubscriptionInfo;
9292
InteractionModelEngine * imEngine = InteractionModelEngine::GetInstance();
93+
94+
// Decrement the number of subscriptions to resume
95+
imEngine->DecrementNumSubscriptionToResume();
96+
9397
if (!imEngine->EnsureResourceForSubscription(subscriptionInfo.mFabricIndex, subscriptionInfo.mAttributePaths.AllocatedSize(),
9498
subscriptionInfo.mEventPaths.AllocatedSize()))
9599
{
100+
// TODO - Should we keep the subscription here?
96101
ChipLogProgress(InteractionModel, "no resource for subscription resumption");
97102
return;
98103
}
99104
ReadHandler * readHandler = imEngine->mReadHandlers.CreateObject(*imEngine, imEngine->GetReportScheduler());
100105
if (readHandler == nullptr)
101106
{
107+
// TODO - Should we keep the subscription here?
102108
ChipLogProgress(InteractionModel, "no resource for ReadHandler creation");
103109
return;
104110
}
@@ -118,10 +124,15 @@ void SubscriptionResumptionSessionEstablisher::HandleDeviceConnectionFailure(voi
118124
CHIP_ERROR error)
119125
{
120126
AutoDeleteEstablisher establisher(static_cast<SubscriptionResumptionSessionEstablisher *>(context));
127+
InteractionModelEngine * imEngine = InteractionModelEngine::GetInstance();
121128
SubscriptionResumptionStorage::SubscriptionInfo & subscriptionInfo = establisher->mSubscriptionInfo;
122129
ChipLogError(DataManagement, "Failed to establish CASE for subscription-resumption with error '%" CHIP_ERROR_FORMAT "'",
123130
error.Format());
124-
auto * subscriptionResumptionStorage = InteractionModelEngine::GetInstance()->GetSubscriptionResumptionStorage();
131+
132+
// Decrement the number of subscriptions to resume
133+
imEngine->DecrementNumSubscriptionToResume();
134+
135+
auto * subscriptionResumptionStorage = imEngine->GetSubscriptionResumptionStorage();
125136
if (!subscriptionResumptionStorage)
126137
{
127138
ChipLogError(DataManagement, "Failed to get subscription resumption storage");

src/app/icd/server/BUILD.gn

+5
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,11 @@ buildconfig_header("icd-server-buildconfig") {
2222
header = "ICDServerBuildConfig.h"
2323
header_dir = "app/icd/server"
2424

25+
if (chip_enable_icd_lit || chip_enable_icd_checkin ||
26+
chip_enable_icd_user_active_mode_trigger) {
27+
assert(chip_enable_icd_lit)
28+
}
29+
2530
if (chip_enable_icd_lit) {
2631
assert(chip_enable_icd_checkin && chip_enable_icd_user_active_mode_trigger)
2732
}

src/app/icd/server/ICDManager.cpp

+73-9
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,10 @@ void ICDManager::Shutdown()
105105
mFabricTable = nullptr;
106106
mSubInfoProvider = nullptr;
107107
mICDSenderPool.ReleaseAll();
108+
109+
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS && !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
110+
mIsBootUpResumeSubscriptionExecuted = false;
111+
#endif // !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
108112
#endif // CHIP_CONFIG_ENABLE_ICD_CIP
109113
}
110114

@@ -152,6 +156,7 @@ void ICDManager::SendCheckInMsgs()
152156

153157
ICDMonitoringTable table(*mStorage, fabricInfo.GetFabricIndex(), supported_clients /*Table entry limit*/,
154158
mSymmetricKeystore);
159+
155160
if (table.IsEmpty())
156161
{
157162
continue;
@@ -173,8 +178,7 @@ void ICDManager::SendCheckInMsgs()
173178
continue;
174179
}
175180

176-
bool active = mSubInfoProvider->SubjectHasActiveSubscription(entry.fabricIndex, entry.monitoredSubject);
177-
if (active)
181+
if (!CheckInWouldBeSentAtActiveModeVerifier(entry.fabricIndex, entry.monitoredSubject))
178182
{
179183
continue;
180184
}
@@ -244,17 +248,77 @@ bool ICDManager::CheckInMessagesWouldBeSent(std::function<RegistrationVerificati
244248
return false;
245249
}
246250

251+
/**
252+
* CheckInWouldBeSentAtActiveModeVerifier is used to determine if a Check-In message is required for a given registration.
253+
* Due to how the ICD Check-In use-case interacts with the persistent subscription and subscription timeout resumption,
254+
* having a single implementation of the function renders the implematention very difficult to understand and maintain.
255+
* Because of this, each valid feature combination has its own implementation of the verifier.
256+
*/
257+
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
258+
#if CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
259+
/**
260+
* @brief Implementation for when the persistant subscription and subscription timeout resumption feature are present.
261+
* Verifier checks that there no active or persisted subscriptions for a given fabricIndex or subjectID.
262+
*
263+
* @param aFabricIndex
264+
* @param subjectID subjectID to check. Can be an opperationnal node id or a CAT
265+
*
266+
* @return true Returns true if the fabricIndex and subjectId combination does not have an active or a persisted subscription.
267+
* @return false Returns false if the fabricIndex and subjectId combination has an active or persisted subscription.
268+
*/
269+
bool ICDManager::CheckInWouldBeSentAtActiveModeVerifier(FabricIndex aFabricIndex, NodeId subjectID)
270+
{
271+
return !(mSubInfoProvider->SubjectHasActiveSubscription(aFabricIndex, subjectID) ||
272+
mSubInfoProvider->SubjectHasPersistedSubscription(aFabricIndex, subjectID));
273+
}
274+
#else
275+
/**
276+
* @brief Implementation for when the persistant subscription is present without the subscription timeout resumption feature.
277+
* Verifier checks that there no active subscriptions. If the boot up subscription resumption has been completed,
278+
* verifier also checks if there are persisted subscriptions.
279+
*
280+
* @note The persistent subscriptions feature tries to resume subscriptions at the highest min interval
281+
* of all the persisted subscriptions. As such, it is possible for the ICD to return to Idle Mode
282+
* until the timer elaspses. We do not when to send Check-In message to clients with persisted subscriptions
283+
* until we have tried to resubscribe.
284+
*
285+
* @param aFabricIndex
286+
* @param subjectID subjectID to check. Can be an opperationnal node id or a CAT
287+
*
288+
* @return true Returns true if the fabricIndex and subjectId combination does not have an active subscription.
289+
* If the boot up susbscription has not been completed, there must not be a persisted subscription either.
290+
* @return false Returns false if the fabricIndex and subjectId combination has an active subscription.
291+
* If the boot up susbscription has not been completed,
292+
* if the fabricIndex and subjectId combination has a persisted subscription.
293+
*/
247294
bool ICDManager::CheckInWouldBeSentAtActiveModeVerifier(FabricIndex aFabricIndex, NodeId subjectID)
248295
{
249-
VerifyOrReturnValue(mSubInfoProvider->SubjectHasActiveSubscription(aFabricIndex, subjectID), true);
296+
bool wouldSendCheckIn = !(mSubInfoProvider->SubjectHasActiveSubscription(aFabricIndex, subjectID));
250297

251-
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
252-
// If at least one registration has a persisted entry, do not send Check-In message.
253-
// The resumption of the persisted subscription will serve the same function a check-in would have served.
254-
VerifyOrReturnValue(mSubInfoProvider->SubjectHasPersistedSubscription(aFabricIndex, subjectID), true);
255-
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
256-
return false;
298+
if (!mIsBootUpResumeSubscriptionExecuted)
299+
{
300+
wouldSendCheckIn = wouldSendCheckIn && !mSubInfoProvider->SubjectHasPersistedSubscription(aFabricIndex, subjectID);
301+
}
302+
303+
return wouldSendCheckIn;
304+
}
305+
#endif // CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
306+
#else
307+
/**
308+
* @brief Implementation for when neither the persistant subscription and subscription timeout resumption feature are present.
309+
* Verifier checks that there no active sbuscriptions for a given fabricIndex and subjectId combination.
310+
*
311+
* @param aFabricIndex
312+
* @param subjectID subjectID to check. Can be an opperationnal node id or a CAT
313+
*
314+
* @return true Returns true if the fabricIndex and subjectId combination does not have an active subscription.
315+
* @return false Returns false if the fabricIndex and subjectId combination has an active subscription.
316+
*/
317+
bool ICDManager::CheckInWouldBeSentAtActiveModeVerifier(FabricIndex aFabricIndex, NodeId subjectID)
318+
{
319+
return !(mSubInfoProvider->SubjectHasActiveSubscription(aFabricIndex, subjectID));
257320
}
321+
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
258322

259323
void ICDManager::TriggerCheckInMessages(const std::function<RegistrationVerificationFunction> & verifier)
260324
{

src/app/icd/server/ICDManager.h

+13
Original file line numberDiff line numberDiff line change
@@ -138,10 +138,20 @@ class ICDManager : public ICDListener
138138
* @param[in] algo Verifier function to use to determine if we need to send check-in messages
139139
*/
140140
void TriggerCheckInMessages(const std::function<RegistrationVerificationFunction> & verifier);
141+
142+
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS && !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
143+
/**
144+
* @brief Set mSubCheckInBootCheckExecuted to true
145+
* Function allows the InteractionModelEngine to notify the ICDManager that the boot up subscription resumption has been
146+
* completed.
147+
*/
148+
void SetBootUpResumeSubscriptionExecuted() { mIsBootUpResumeSubscriptionExecuted = true; };
149+
#endif // !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION && CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
141150
#endif // CHIP_CONFIG_ENABLE_ICD_CIP
142151

143152
#if CONFIG_BUILD_FOR_HOST_UNIT_TEST
144153
void SetTestFeatureMapValue(uint32_t featureMap) { mFeatureMap = featureMap; };
154+
bool GetIsBootUpResumeSubscriptionExecuted() { return mIsBootUpResumeSubscriptionExecuted; };
145155
#endif
146156

147157
// Implementation of ICDListener functions.
@@ -212,6 +222,9 @@ class ICDManager : public ICDListener
212222
ObjectPool<ObserverPointer, CHIP_CONFIG_ICD_OBSERVERS_POOL_SIZE> mStateObserverPool;
213223

214224
#if CHIP_CONFIG_ENABLE_ICD_CIP
225+
#if !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION && CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
226+
bool mIsBootUpResumeSubscriptionExecuted = false;
227+
#endif // !CHIP_CONFIG_SUBSCRIPTION_TIMEOUT_RESUMPTION
215228
PersistentStorageDelegate * mStorage = nullptr;
216229
FabricTable * mFabricTable = nullptr;
217230
Messaging::ExchangeManager * mExchangeManager = nullptr;

src/app/server/Server.cpp

+11-4
Original file line numberDiff line numberDiff line change
@@ -332,6 +332,10 @@ CHIP_ERROR Server::Init(const ServerInitParams & initParams)
332332
&mCASESessionManager, mSubscriptionResumptionStorage);
333333
SuccessOrExit(err);
334334

335+
#if CHIP_CONFIG_ENABLE_ICD_SERVER
336+
chip::app::InteractionModelEngine::GetInstance()->SetICDManager(&mICDManager);
337+
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
338+
335339
// ICD Init needs to be after data model init and InteractionModel Init
336340
#if CHIP_CONFIG_ENABLE_ICD_SERVER
337341

@@ -452,7 +456,7 @@ void Server::OnPlatformEvent(const DeviceLayer::ChipDeviceEvent & event)
452456
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER && CHIP_CONFIG_ENABLE_ICD_CIP
453457
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
454458
ResumeSubscriptions();
455-
#endif
459+
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
456460
break;
457461
#if CHIP_SYSTEM_CONFIG_USE_OPEN_THREAD_ENDPOINT
458462
case DeviceEventType::kThreadConnectivityChange:
@@ -524,9 +528,9 @@ void Server::RejoinExistingMulticastGroups()
524528
bool Server::CheckInWouldBeSentAtBootVerifier(FabricIndex aFabricIndex, NodeId subjectID)
525529
{
526530
#if CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
527-
// At least one registration has a persisted entry. Do not send Check-In message.
528-
// This is to cover the use-case where the subscription resumption feature is used with the Check-In message.
529-
VerifyOrReturnValue(!chip::app::InteractionModelEngine::GetInstance()->SubjectHasPersistedSubscription(aFabricIndex, subjectID),
531+
// If at least one registration has a persisted entry, do not send Check-In message.
532+
// The resumption of the persisted subscription will serve the same function a check-in would have served.
533+
VerifyOrReturnValue(!app::InteractionModelEngine::GetInstance()->SubjectHasPersistedSubscription(aFabricIndex, subjectID),
530534
false);
531535
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
532536

@@ -576,6 +580,9 @@ void Server::Shutdown()
576580

577581
chip::Dnssd::Resolver::Instance().Shutdown();
578582
chip::app::InteractionModelEngine::GetInstance()->Shutdown();
583+
#if CHIP_CONFIG_ENABLE_ICD_SERVER
584+
chip::app::InteractionModelEngine::GetInstance()->SetICDManager(nullptr);
585+
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
579586
mCommissioningWindowManager.Shutdown();
580587
mMessageCounterManager.Shutdown();
581588
mExchangeMgr.Shutdown();

0 commit comments

Comments
 (0)