Skip to content

Commit f4b8064

Browse files
authored
[Fabric-Sync] Support icd-registration during device sync (#36569)
1 parent c197bc2 commit f4b8064

10 files changed

+507
-5
lines changed

examples/fabric-sync/admin/BUILD.gn

+6
Original file line numberDiff line numberDiff line change
@@ -44,15 +44,21 @@ source_set("fabric-admin-lib") {
4444
"FabricAdmin.h",
4545
"FabricSyncGetter.cpp",
4646
"FabricSyncGetter.h",
47+
"IcdManager.cpp",
48+
"IcdManager.h",
4749
"PairingManager.cpp",
4850
"PairingManager.h",
51+
"StayActiveSender.cpp",
52+
"StayActiveSender.h",
4953
"UniqueIdGetter.cpp",
5054
"UniqueIdGetter.h",
5155
]
5256

5357
deps = [
5458
"${chip_root}/examples/fabric-sync/bridge:fabric-bridge-lib",
5559
"${chip_root}/examples/platform/linux:app-main",
60+
"${chip_root}/src/app/icd/client:handler",
61+
"${chip_root}/src/app/icd/client:manager",
5662
"${chip_root}/src/lib",
5763
]
5864
}

examples/fabric-sync/admin/FabricAdmin.cpp

+63-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,9 @@
1616
*/
1717

1818
#include "FabricAdmin.h"
19+
#include <AppMain.h>
20+
#include <bridge/include/FabricBridge.h>
21+
#include <controller/CHIPDeviceControllerFactory.h>
1922

2023
using namespace ::chip;
2124

@@ -28,16 +31,35 @@ constexpr uint32_t kCommissionPrepareTimeMs = 500;
2831
} // namespace
2932

3033
FabricAdmin FabricAdmin::sInstance;
34+
app::DefaultICDClientStorage FabricAdmin::sICDClientStorage;
35+
app::CheckInHandler FabricAdmin::sCheckInHandler;
3136

3237
FabricAdmin & FabricAdmin::Instance()
3338
{
3439
if (!sInstance.mInitialized)
3540
{
36-
sInstance.Init();
41+
VerifyOrDie(sInstance.Init() == CHIP_NO_ERROR);
3742
}
3843
return sInstance;
3944
}
4045

46+
CHIP_ERROR FabricAdmin::Init()
47+
{
48+
IcdManager::Instance().SetDelegate(&sInstance);
49+
50+
ReturnLogErrorOnFailure(sICDClientStorage.Init(GetPersistentStorageDelegate(), GetSessionKeystore()));
51+
52+
auto engine = chip::app::InteractionModelEngine::GetInstance();
53+
VerifyOrReturnError(engine != nullptr, CHIP_ERROR_INCORRECT_STATE);
54+
ReturnLogErrorOnFailure(IcdManager::Instance().Init(&sICDClientStorage, engine));
55+
ReturnLogErrorOnFailure(sCheckInHandler.Init(Controller::DeviceControllerFactory::GetInstance().GetSystemState()->ExchangeMgr(),
56+
&sICDClientStorage, &IcdManager::Instance(), engine));
57+
58+
mInitialized = true;
59+
60+
return CHIP_NO_ERROR;
61+
}
62+
4163
CHIP_ERROR FabricAdmin::OpenCommissioningWindow(Controller::CommissioningWindowVerifierParams params, FabricIndex fabricIndex)
4264
{
4365
ScopedNodeId scopedNodeId(params.GetNodeId(), fabricIndex);
@@ -116,6 +138,46 @@ CHIP_ERROR FabricAdmin::KeepActive(ScopedNodeId scopedNodeId, uint32_t stayActiv
116138
return CHIP_NO_ERROR;
117139
}
118140

141+
void FabricAdmin::OnCheckInCompleted(const app::ICDClientInfo & clientInfo)
142+
{
143+
// Accessing mPendingCheckIn should only be done while holding ChipStackLock
144+
assertChipStackLockedByCurrentThread();
145+
ScopedNodeId scopedNodeId = clientInfo.peer_node;
146+
auto it = mPendingCheckIn.find(scopedNodeId);
147+
VerifyOrReturn(it != mPendingCheckIn.end());
148+
149+
KeepActiveDataForCheckIn checkInData = it->second;
150+
// Removed from pending map as check-in from this node has occured and we will handle the pending KeepActive
151+
// request.
152+
mPendingCheckIn.erase(scopedNodeId);
153+
154+
auto timeNow = System::SystemClock().GetMonotonicTimestamp();
155+
if (timeNow > checkInData.mRequestExpiryTimestamp)
156+
{
157+
ChipLogError(NotSpecified,
158+
"ICD check-in for device we have been waiting, came after KeepActive expiry. Request dropped for ID: "
159+
"[%d:0x " ChipLogFormatX64 "]",
160+
scopedNodeId.GetFabricIndex(), ChipLogValueX64(scopedNodeId.GetNodeId()));
161+
return;
162+
}
163+
164+
// TODO https://github.com/CHIP-Specifications/connectedhomeip-spec/issues/10448. Spec does
165+
// not define what to do if we fail to send the StayActiveRequest. We are assuming that any
166+
// further attempts to send a StayActiveRequest will result in a similar failure. Because
167+
// there is no mechanism for us to communicate with the client that sent out the KeepActive
168+
// command that there was a failure, we simply fail silently. After spec issue is
169+
// addressed, we can implement what spec defines here.
170+
auto onDone = [=](uint32_t promisedActiveDuration) {
171+
bridge::FabricBridge::Instance().ActiveChanged(scopedNodeId, promisedActiveDuration);
172+
};
173+
CHIP_ERROR err = StayActiveSender::SendStayActiveCommand(checkInData.mStayActiveDurationMs, clientInfo.peer_node,
174+
app::InteractionModelEngine::GetInstance(), onDone);
175+
if (err != CHIP_NO_ERROR)
176+
{
177+
ChipLogError(NotSpecified, "Failed to send StayActive command %s", err.AsString());
178+
}
179+
}
180+
119181
void FabricAdmin::OnCommissioningComplete(NodeId deviceId, CHIP_ERROR err)
120182
{
121183
if (mNodeId != deviceId)

examples/fabric-sync/admin/FabricAdmin.h

+12-2
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,12 @@
1818
#pragma once
1919

2020
#include "DeviceManager.h"
21+
#include "IcdManager.h"
22+
#include "StayActiveSender.h"
2123

24+
#include <app/icd/client/CheckInHandler.h>
25+
#include <app/icd/client/DefaultCheckInDelegate.h>
26+
#include <app/icd/client/DefaultICDClientStorage.h>
2227
#include <bridge/include/FabricAdminDelegate.h>
2328
#include <map>
2429
#include <setup_payload/QRCodeSetupPayloadGenerator.h>
@@ -37,10 +42,11 @@ struct ScopedNodeIdHasher
3742
}
3843
};
3944

40-
class FabricAdmin final : public bridge::FabricAdminDelegate, public PairingDelegate
45+
class FabricAdmin final : public bridge::FabricAdminDelegate, public PairingDelegate, public IcdManager::Delegate
4146
{
4247
public:
4348
static FabricAdmin & Instance();
49+
static chip::app::DefaultICDClientStorage & GetDefaultICDClientStorage() { return sICDClientStorage; }
4450

4551
CHIP_ERROR OpenCommissioningWindow(chip::Controller::CommissioningWindowVerifierParams params,
4652
chip::FabricIndex fabricIndex) override;
@@ -51,6 +57,8 @@ class FabricAdmin final : public bridge::FabricAdminDelegate, public PairingDele
5157

5258
CHIP_ERROR KeepActive(chip::ScopedNodeId scopedNodeId, uint32_t stayActiveDurationMs, uint32_t timeoutMs) override;
5359

60+
void OnCheckInCompleted(const chip::app::ICDClientInfo & clientInfo) override;
61+
5462
void OnCommissioningComplete(chip::NodeId deviceId, CHIP_ERROR err) override;
5563

5664
void ScheduleSendingKeepActiveOnCheckIn(chip::ScopedNodeId scopedNodeId, uint32_t stayActiveDurationMs, uint32_t timeoutMs);
@@ -89,11 +97,13 @@ class FabricAdmin final : public bridge::FabricAdminDelegate, public PairingDele
8997
std::unordered_map<chip::ScopedNodeId, KeepActiveDataForCheckIn, ScopedNodeIdHasher> mPendingCheckIn;
9098

9199
static FabricAdmin sInstance;
100+
static chip::app::DefaultICDClientStorage sICDClientStorage;
101+
static chip::app::CheckInHandler sCheckInHandler;
92102

93103
bool mInitialized = false;
94104
chip::NodeId mNodeId = chip::kUndefinedNodeId;
95105

96-
void Init() { mInitialized = true; }
106+
CHIP_ERROR Init();
97107
};
98108

99109
} // namespace admin
+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
/*
2+
* Copyright (c) 2024 Project CHIP Authors
3+
* All rights reserved.
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*
17+
*/
18+
19+
#include "IcdManager.h"
20+
21+
namespace admin {
22+
23+
IcdManager IcdManager::sInstance;
24+
25+
IcdManager & IcdManager::Instance()
26+
{
27+
return sInstance;
28+
}
29+
30+
void IcdManager::OnCheckInComplete(const chip::app::ICDClientInfo & clientInfo)
31+
{
32+
DefaultCheckInDelegate::OnCheckInComplete(clientInfo);
33+
if (mDelegate)
34+
{
35+
mDelegate->OnCheckInCompleted(clientInfo);
36+
}
37+
}
38+
39+
void IcdManager::SetDelegate(Delegate * delegate)
40+
{
41+
// To keep IcdManager simple, there is an assumption that there is only ever
42+
// one delegate set and it's lifetime is identical to IcdManager. In the
43+
// future this assumption can change should there be a need, but that will
44+
// require code changes to IcdManager. For now we will crash if someone tries
45+
// to call SetDelegate for a second time or if delegate is non-null.
46+
VerifyOrDie(delegate);
47+
VerifyOrDie(!mDelegate);
48+
mDelegate = delegate;
49+
}
50+
51+
} // namespace admin
+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
/*
2+
* Copyright (c) 2024 Project CHIP Authors
3+
* All rights reserved.
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*
17+
*/
18+
19+
#pragma once
20+
21+
#include <app/icd/client/DefaultCheckInDelegate.h>
22+
23+
namespace admin {
24+
25+
/**
26+
* @brief Manages check-ins from ICD devices.
27+
*
28+
* Intended to be used as a thin CheckInDelegate. This allows a delegate register
29+
* themselves so they can be aware when ICD device checks-in allowing the
30+
* delegate to interact with the ICD device during the short window that it is
31+
* awake.
32+
*/
33+
class IcdManager : public chip::app::DefaultCheckInDelegate
34+
{
35+
public:
36+
class Delegate
37+
{
38+
public:
39+
virtual ~Delegate() = default;
40+
virtual void OnCheckInCompleted(const chip::app::ICDClientInfo & clientInfo) = 0;
41+
};
42+
43+
static IcdManager & Instance();
44+
void OnCheckInComplete(const chip::app::ICDClientInfo & clientInfo) override;
45+
46+
// There is an assumption delegate assigned only happens once and that it lives
47+
// for the entirety of the lifetime of fabric admin.
48+
void SetDelegate(Delegate * delegate);
49+
50+
private:
51+
static IcdManager sInstance;
52+
Delegate * mDelegate = nullptr;
53+
};
54+
55+
} // namespace admin

0 commit comments

Comments
 (0)