@@ -154,7 +154,11 @@ @implementation MTRDeviceController {
154
154
MTRP256KeypairBridge _signingKeypairBridge;
155
155
MTRP256KeypairBridge _operationalKeypairBridge;
156
156
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;
158
162
159
163
// Counters to track assertion status and access controlled by the _assertionLock
160
164
NSUInteger _keepRunningAssertionCounter;
@@ -378,9 +382,7 @@ - (BOOL)isRunning
378
382
379
383
- (BOOL )isSuspended
380
384
{
381
- @synchronized (self) {
382
- return _suspended;
383
- }
385
+ return _suspended;
384
386
}
385
387
386
388
- (void )_notifyDelegatesOfSuspendState
@@ -397,48 +399,51 @@ - (void)suspend
397
399
{
398
400
MTR_LOG (" %@ suspending" , self);
399
401
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.
401
409
_suspended = YES ;
410
+ devicesToSuspend = [self .nodeIDToDeviceMap objectEnumerator ].allObjects ;
411
+ }
402
412
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 ];
420
416
}
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 ];
421
424
}
422
425
423
426
- (void )resume
424
427
{
425
428
MTR_LOG (" %@ resuming" , self);
426
429
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.
428
437
_suspended = NO ;
438
+ devicesToResume = [self .nodeIDToDeviceMap objectEnumerator ].allObjects ;
439
+ }
429
440
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 ];
441
444
}
445
+
446
+ [self _notifyDelegatesOfSuspendState ];
442
447
}
443
448
444
449
- (BOOL )matchesPendingShutdownControllerWithOperationalCertificate : (nullable MTRCertificateDERBytes)operationalCertificate andRootCertificate : (nullable MTRCertificateDERBytes)rootCertificate
@@ -859,7 +864,7 @@ - (BOOL)startup:(MTRDeviceControllerStartupParamsInternal *)startupParams
859
864
//
860
865
// Note that this is just an optimization to avoid throwing the information away and immediately
861
866
// 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 ) , ^{
863
868
MTR_LOG (" %@ un-retain devices loaded at startup %lu" , self, static_cast <unsigned long >(deviceList.count ));
864
869
});
865
870
}];
0 commit comments