Skip to content

Commit 9c9ad4f

Browse files
Merge branch 'master' into bugfix/fixing_code_bugs
2 parents a16fbf1 + 9a6694f commit 9c9ad4f

23 files changed

+320
-100
lines changed

.github/workflows/darwin.yaml

+3-4
Original file line numberDiff line numberDiff line change
@@ -105,10 +105,9 @@ jobs:
105105
run: |
106106
scripts/examples/gn_build_example.sh examples/ota-requestor-app/linux out/debug/ota-requestor-app chip_config_network_layer_ble=false non_spec_compliant_ota_action_delay_floor=0
107107
- name: Run Framework Tests
108-
# For now disable unguarded-availability-new warnings because we
109-
# internally use APIs that we are annotating as only available on
110-
# new enough versions. Maybe we should change out deployment
111-
# target versions instead?
108+
# We want to ensure that our log upload runs on timeout, so use a timeout here shorter
109+
# than the 6-hour overall job timeout. 4.5 hours should be plenty.
110+
timeout-minutes: 270
112111
working-directory: src/darwin/Framework
113112
run: |
114113
mkdir -p /tmp/darwin/framework-tests

examples/chip-tool/templates/logging/DataModelLogger-src.zapt

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
{{> header}}
22

33
#include <commands/clusters/DataModelLogger.h>
4+
#include <zap-generated/cluster/logging/EntryToText.h>
45

56
using namespace chip::app::Clusters;
67

examples/chip-tool/templates/logging/EntryToText-src.zapt

+10
Original file line numberDiff line numberDiff line change
@@ -81,4 +81,14 @@ char const * GeneratedCommandIdToText(chip::ClusterId cluster, chip::CommandId i
8181
{{/zcl_clusters}}
8282
default: return "Unknown";
8383
}
84+
}
85+
86+
char const * DeviceTypeIdToText(chip::DeviceTypeId id) {
87+
switch(id)
88+
{
89+
{{#zcl_device_types}}
90+
case {{asHex code 8}}: return "{{caption}}";
91+
{{/zcl_device_types}}
92+
default: return "Unknown";
93+
}
8494
}

examples/chip-tool/templates/logging/EntryToText.zapt

+3-1
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,6 @@ char const * AttributeIdToText(chip::ClusterId cluster, chip::AttributeId id);
1010

1111
char const * AcceptedCommandIdToText(chip::ClusterId cluster, chip::CommandId id);
1212

13-
char const * GeneratedCommandIdToText(chip::ClusterId cluster, chip::CommandId id);
13+
char const * GeneratedCommandIdToText(chip::ClusterId cluster, chip::CommandId id);
14+
15+
char const * DeviceTypeIdToText(chip::DeviceTypeId id);

examples/chip-tool/templates/partials/StructLoggerImpl.zapt

+22
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,34 @@ CHIP_ERROR DataModelLogger::LogValue(const char * label, size_t indent, const ch
33
DataModelLogger::LogString(label, indent, "{");
44
{{#zcl_struct_items}}
55
{
6+
{{#if (isEqual type "devtype_id") }}
7+
{{#if isNullable }}
8+
if (value.{{asLowerCamelCase label}}.IsNull())
9+
{
10+
CHIP_ERROR err = LogValue("{{asUpperCamelCase label}}", indent + 1, value.{{asLowerCamelCase label}});
11+
if (err != CHIP_NO_ERROR)
12+
{
13+
DataModelLogger::LogString(indent + 1, "Struct truncated due to invalid value for '{{asUpperCamelCase label}}'");
14+
return err;
15+
}
16+
}
17+
else
18+
{
19+
std::string item = std::to_string(value.{{asLowerCamelCase label}}.Value()) + " (" + DeviceTypeIdToText(value.{{asLowerCamelCase label}}.Value()) + ")";
20+
DataModelLogger::LogString("{{asUpperCamelCase label}}", indent + 1, item);
21+
}
22+
{{else}}
23+
std::string item = std::to_string(value.{{asLowerCamelCase label}}) + " (" + DeviceTypeIdToText(value.{{asLowerCamelCase label}}) + ")";
24+
DataModelLogger::LogString("{{asUpperCamelCase label}}", indent + 1, item);
25+
{{/if}}
26+
{{else}}
627
CHIP_ERROR err = LogValue("{{asUpperCamelCase label}}", indent + 1, value.{{asLowerCamelCase label}});
728
if (err != CHIP_NO_ERROR)
829
{
930
DataModelLogger::LogString(indent + 1, "Struct truncated due to invalid value for '{{asUpperCamelCase label}}'");
1031
return err;
1132
}
33+
{{/if}}
1234
}
1335
{{/zcl_struct_items}}
1436
DataModelLogger::LogString(indent, "}");

src/app/util/attribute-storage.h

+6
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,12 @@ static constexpr uint16_t kEmberInvalidEndpointIndex = 0xFFFF;
6060
} /* cluster revision */ \
6161
}
6262

63+
// The attrMask must contain the relevant ATTRIBUTE_MASK_* bits from
64+
// attribute-metadata.h. Specifically:
65+
//
66+
// * Writable attributes must have ATTRIBUTE_MASK_WRITABLE
67+
// * Nullable attributes (have X in the quality column in the spec) must have ATTRIBUTE_MASK_NULLABLE
68+
// * Attributes that have T in the Access column in the spec must have ATTRIBUTE_MASK_MUST_USE_TIMED_WRITE
6369
#define DECLARE_DYNAMIC_ATTRIBUTE(attId, attType, attSizeBytes, attrMask) \
6470
{ \
6571
ZAP_EMPTY_DEFAULT(), attId, attSizeBytes, ZAP_TYPE(attType), attrMask | ZAP_ATTRIBUTE_MASK(EXTERNAL_STORAGE) \

src/darwin/Framework/CHIP/MTRDeviceController.mm

+40-35
Original file line numberDiff line numberDiff line change
@@ -154,7 +154,11 @@ @implementation MTRDeviceController {
154154
MTRP256KeypairBridge _signingKeypairBridge;
155155
MTRP256KeypairBridge _operationalKeypairBridge;
156156

157-
BOOL _suspended;
157+
// For now, we just ensure that access to _suspended is atomic, but don't
158+
// guarantee atomicity of the the entire suspend/resume operation. The
159+
// expectation is that suspend/resume on a given controller happen on some
160+
// specific queue, so can't race against each other.
161+
std::atomic<bool> _suspended;
158162

159163
// Counters to track assertion status and access controlled by the _assertionLock
160164
NSUInteger _keepRunningAssertionCounter;
@@ -378,9 +382,7 @@ - (BOOL)isRunning
378382

379383
- (BOOL)isSuspended
380384
{
381-
@synchronized(self) {
382-
return _suspended;
383-
}
385+
return _suspended;
384386
}
385387

386388
- (void)_notifyDelegatesOfSuspendState
@@ -397,48 +399,51 @@ - (void)suspend
397399
{
398400
MTR_LOG("%@ suspending", self);
399401

400-
@synchronized(self) {
402+
NSArray * devicesToSuspend;
403+
{
404+
std::lock_guard lock(*self.deviceMapLock);
405+
// Set _suspended under the device map lock. This guarantees that
406+
// for any given device exactly one of two things is true:
407+
// * It is in the snapshot we are creating
408+
// * It is created after we have changed our _suspended state.
401409
_suspended = YES;
410+
devicesToSuspend = [self.nodeIDToDeviceMap objectEnumerator].allObjects;
411+
}
402412

403-
NSArray * devicesToSuspend;
404-
{
405-
std::lock_guard lock(*self.deviceMapLock);
406-
devicesToSuspend = [self.nodeIDToDeviceMap objectEnumerator].allObjects;
407-
}
408-
409-
for (MTRDevice * device in devicesToSuspend) {
410-
[device controllerSuspended];
411-
}
412-
413-
// TODO: In the concrete class, consider what should happen with:
414-
//
415-
// * Active commissioning sessions (presumably close them?)
416-
// * CASE sessions in general.
417-
// * Possibly try to see whether we can change our fabric entry to not advertise and restart advertising.
418-
419-
[self _notifyDelegatesOfSuspendState];
413+
MTR_LOG("%@ found %lu devices to suspend", self, static_cast<unsigned long>(devicesToSuspend.count));
414+
for (MTRDevice * device in devicesToSuspend) {
415+
[device controllerSuspended];
420416
}
417+
418+
// TODO: In the concrete class, consider what should happen with:
419+
//
420+
// * Active commissioning sessions (presumably close them?)
421+
// * CASE sessions in general.
422+
// * Possibly try to see whether we can change our fabric entry to not advertise and restart advertising.
423+
[self _notifyDelegatesOfSuspendState];
421424
}
422425

423426
- (void)resume
424427
{
425428
MTR_LOG("%@ resuming", self);
426429

427-
@synchronized(self) {
430+
NSArray * devicesToResume;
431+
{
432+
std::lock_guard lock(*self.deviceMapLock);
433+
// Set _suspended under the device map lock. This guarantees that
434+
// for any given device exactly one of two things is true:
435+
// * It is in the snapshot we are creating
436+
// * It is created after we have changed our _suspended state.
428437
_suspended = NO;
438+
devicesToResume = [self.nodeIDToDeviceMap objectEnumerator].allObjects;
439+
}
429440

430-
NSArray * devicesToResume;
431-
{
432-
std::lock_guard lock(*self.deviceMapLock);
433-
devicesToResume = [self.nodeIDToDeviceMap objectEnumerator].allObjects;
434-
}
435-
436-
for (MTRDevice * device in devicesToResume) {
437-
[device controllerResumed];
438-
}
439-
440-
[self _notifyDelegatesOfSuspendState];
441+
MTR_LOG("%@ found %lu devices to resume", self, static_cast<unsigned long>(devicesToResume.count));
442+
for (MTRDevice * device in devicesToResume) {
443+
[device controllerResumed];
441444
}
445+
446+
[self _notifyDelegatesOfSuspendState];
442447
}
443448

444449
- (BOOL)matchesPendingShutdownControllerWithOperationalCertificate:(nullable MTRCertificateDERBytes)operationalCertificate andRootCertificate:(nullable MTRCertificateDERBytes)rootCertificate
@@ -859,7 +864,7 @@ - (BOOL)startup:(MTRDeviceControllerStartupParamsInternal *)startupParams
859864
//
860865
// Note that this is just an optimization to avoid throwing the information away and immediately
861866
// re-reading it from storage.
862-
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t) (kSecondsToWaitBeforeAPIClientRetainsMTRDevice * NSEC_PER_SEC)), self.chipWorkQueue, ^{
867+
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t) (kSecondsToWaitBeforeAPIClientRetainsMTRDevice * NSEC_PER_SEC)), dispatch_get_global_queue(QOS_CLASS_DEFAULT, 0), ^{
863868
MTR_LOG("%@ un-retain devices loaded at startup %lu", self, static_cast<unsigned long>(deviceList.count));
864869
});
865870
}];

src/darwin/Framework/CHIP/MTRDeviceController_Concrete.mm

+1-1
Original file line numberDiff line numberDiff line change
@@ -761,7 +761,7 @@ - (BOOL)startup:(MTRDeviceControllerStartupParamsInternal *)startupParams
761761
//
762762
// Note that this is just an optimization to avoid throwing the information away and immediately
763763
// re-reading it from storage.
764-
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t) (kSecondsToWaitBeforeAPIClientRetainsMTRDevice * NSEC_PER_SEC)), self.chipWorkQueue, ^{
764+
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t) (kSecondsToWaitBeforeAPIClientRetainsMTRDevice * NSEC_PER_SEC)), dispatch_get_global_queue(QOS_CLASS_DEFAULT, 0), ^{
765765
MTR_LOG("%@ un-retain devices loaded at startup %lu", self, static_cast<unsigned long>(deviceList.count));
766766
});
767767
}];

src/darwin/Framework/CHIP/MTRDevice_Concrete.mm

+12-4
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,7 @@ @interface MTRDevice_Concrete ()
6262
@property (nonatomic, readwrite) MTRDeviceState state;
6363
@property (nonatomic, readwrite, nullable) NSDate * estimatedStartTime;
6464
@property (nonatomic, readwrite, nullable, copy) NSNumber * estimatedSubscriptionLatency;
65+
@property (nonatomic, readwrite, assign) BOOL suspended;
6566

6667
@end
6768

@@ -395,6 +396,8 @@ - (instancetype)initWithNodeID:(NSNumber *)nodeID controller:(MTRDeviceControlle
395396
}];
396397
}
397398

399+
self.suspended = controller.suspended;
400+
398401
MTR_LOG_DEBUG("%@ init with hex nodeID 0x%016llX", self, _nodeID.unsignedLongLongValue);
399402
}
400403
return self;
@@ -719,8 +722,8 @@ - (BOOL)_subscriptionsAllowed
719722
{
720723
os_unfair_lock_assert_owner(&self->_lock);
721724

722-
// We should not allow a subscription for suspended controllers or device controllers over XPC.
723-
return _deviceController.isSuspended == NO && ![_deviceController isKindOfClass:MTRDeviceControllerOverXPC.class];
725+
// We should not allow a subscription when we are suspended or for device controllers over XPC.
726+
return self.suspended == NO && ![_deviceController isKindOfClass:MTRDeviceControllerOverXPC.class];
724727
}
725728

726729
- (void)_delegateAdded
@@ -1290,7 +1293,7 @@ - (void)_doHandleSubscriptionReset:(NSNumber * _Nullable)retryDelay
12901293

12911294
os_unfair_lock_assert_owner(&_lock);
12921295

1293-
if (_deviceController.isSuspended) {
1296+
if (self.suspended) {
12941297
MTR_LOG("%@ ignoring expected subscription reset on controller suspend", self);
12951298
return;
12961299
}
@@ -1378,7 +1381,6 @@ - (void)_reattemptSubscriptionNowIfNeededWithReason:(NSString *)reason
13781381
}
13791382

13801383
MTR_LOG("%@ reattempting subscription with reason %@", self, reason);
1381-
self.reattemptingSubscription = NO;
13821384
[self _setupSubscriptionWithReason:reason];
13831385
}
13841386

@@ -2302,6 +2304,10 @@ - (void)_setupSubscriptionWithReason:(NSString *)reason
23022304

23032305
os_unfair_lock_assert_owner(&self->_lock);
23042306

2307+
// If we have a pending subscription reattempt, make sure it does not
2308+
// actually happen, since we are trying to do a subscription now.
2309+
self.reattemptingSubscription = NO;
2310+
23052311
if (![self _subscriptionsAllowed]) {
23062312
MTR_LOG("%@ _setupSubscription: Subscriptions not allowed. Do not set up subscription (reason: %@)", self, reason);
23072313
return;
@@ -3991,6 +3997,7 @@ - (void)controllerSuspended
39913997
[super controllerSuspended];
39923998

39933999
std::lock_guard lock(self->_lock);
4000+
self.suspended = YES;
39944001
[self _resetSubscriptionWithReasonString:@"Controller suspended"];
39954002

39964003
// Ensure that any pre-existing resubscribe attempts we control don't try to
@@ -4003,6 +4010,7 @@ - (void)controllerResumed
40034010
[super controllerResumed];
40044011

40054012
std::lock_guard lock(self->_lock);
4013+
self.suspended = NO;
40064014

40074015
if (![self _delegateExists]) {
40084016
MTR_LOG("%@ ignoring controller resume: no delegates", self);

src/darwin/Framework/CHIPTests/MTRCommissionableBrowserTests.m

+11-1
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@ @interface DeviceScannerDelegate : NSObject <MTRCommissionableBrowserDelegate>
5555
@property (nonatomic, nullable) XCTestExpectation * expectation;
5656
@property (nonatomic) NSMutableArray<NSDictionary<NSString *, id> *> * results;
5757
@property (nonatomic) NSMutableArray<NSDictionary<NSString *, id> *> * removedResults;
58+
@property (nonatomic) BOOL expectedResultsCountReached;
5859

5960
- (instancetype)initWithExpectation:(XCTestExpectation *)expectation;
6061
- (void)controller:(MTRDeviceController *)controller didFindCommissionableDevice:(MTRCommissionableBrowserResult *)device;
@@ -71,6 +72,7 @@ - (instancetype)initWithExpectation:(XCTestExpectation *)expectation
7172
_expectation = expectation;
7273
_results = [[NSMutableArray alloc] init];
7374
_removedResults = [[NSMutableArray alloc] init];
75+
_expectedResultsCountReached = NO;
7476
return self;
7577
}
7678

@@ -83,6 +85,7 @@ - (instancetype)init
8385
_expectation = nil;
8486
_results = [[NSMutableArray alloc] init];
8587
_removedResults = [[NSMutableArray alloc] init];
88+
_expectedResultsCountReached = NO;
8689
return self;
8790
}
8891

@@ -111,7 +114,14 @@ - (void)controller:(MTRDeviceController *)controller didFindCommissionableDevice
111114
// Ensure that we just saw the same results as our final set popping in and out if things
112115
// ever got removed here.
113116
XCTAssertEqualObjects(finalResultsSet, allResultsSet);
114-
[self.expectation fulfill];
117+
118+
// If we have a remove and re-add after the result count reached the
119+
// expected one, we can end up in this branch again. Doing the above
120+
// checks is fine, but we shouldn't double-fulfill the expectation.
121+
if (self.expectedResultsCountReached == NO) {
122+
self.expectedResultsCountReached = YES;
123+
[self.expectation fulfill];
124+
}
115125
}
116126

117127
XCTAssertLessThanOrEqual(_results.count, kExpectedDiscoveredDevicesCount);

src/platform/silabs/KeyValueStoreManagerImpl.cpp

+5-5
Original file line numberDiff line numberDiff line change
@@ -88,9 +88,9 @@ uint16_t KeyValueStoreManagerImpl::hashKvsKeyString(const char * key) const
8888
CHIP_ERROR KeyValueStoreManagerImpl::MapKvsKeyToNvm3(const char * key, uint16_t hash, uint32_t & nvm3Key, bool isSlotNeeded) const
8989
{
9090
CHIP_ERROR err;
91-
char * strPrefix = nullptr;
92-
uint8_t firstEmptyKeySlot = kMaxEntries;
93-
for (uint8_t keyIndex = 0; keyIndex < kMaxEntries; keyIndex++)
91+
char * strPrefix = nullptr;
92+
uint16_t firstEmptyKeySlot = kMaxEntries;
93+
for (uint16_t keyIndex = 0; keyIndex < kMaxEntries; keyIndex++)
9494
{
9595
if (mKvsKeyMap[keyIndex] == hash)
9696
{
@@ -165,7 +165,7 @@ void KeyValueStoreManagerImpl::ScheduleKeyMapSave(void)
165165
Commit the key map in nvm once it as stabilized.
166166
*/
167167
SystemLayer().StartTimer(
168-
std::chrono::duration_cast<System::Clock::Timeout>(System::Clock::Seconds32(SILABS_KVS_SAVE_DELAY_SECONDS)),
168+
std::chrono::duration_cast<System::Clock::Timeout>(System::Clock::Seconds32(SL_KVS_SAVE_DELAY_SECONDS)),
169169
KeyValueStoreManagerImpl::OnScheduledKeyMapSave, NULL);
170170
}
171171

@@ -247,7 +247,7 @@ CHIP_ERROR KeyValueStoreManagerImpl::_Delete(const char * key)
247247
void KeyValueStoreManagerImpl::ErasePartition(void)
248248
{
249249
// Iterate over all the Matter Kvs nvm3 records and delete each one...
250-
for (uint32_t nvm3Key = SilabsConfig::kMinConfigKey_MatterKvs; nvm3Key <= SilabsConfig::kMaxConfigKey_MatterKvs; nvm3Key++)
250+
for (uint32_t nvm3Key = SilabsConfig::kMinConfigKey_MatterKvs; nvm3Key <= SilabsConfig::kConfigKey_KvsLastKeySlot; nvm3Key++)
251251
{
252252
SilabsConfig::ClearConfigValue(nvm3Key);
253253
}

0 commit comments

Comments
 (0)