@@ -225,6 +225,7 @@ @implementation MTRDevice {
225
225
#ifdef DEBUG
226
226
NSUInteger _unitTestAttributesReportedSinceLastCheck;
227
227
#endif
228
+ BOOL _delegateDeviceCachePrimedCalled;
228
229
}
229
230
230
231
- (instancetype )initWithNodeID : (NSNumber *)nodeID controller : (MTRDeviceController *)controller
@@ -506,6 +507,9 @@ - (void)setDelegate:(id<MTRDeviceDelegate>)delegate queue:(dispatch_queue_t)queu
506
507
[self _setupSubscription ];
507
508
}
508
509
510
+ // Check if cache is already primed from storage
511
+ [self _checkIfCacheIsPrimed ];
512
+
509
513
os_unfair_lock_unlock (&self->_lock );
510
514
}
511
515
@@ -574,6 +578,29 @@ - (BOOL)_subscriptionAbleToReport
574
578
return (delegate != nil ) && (state == MTRDeviceStateReachable);
575
579
}
576
580
581
+ - (BOOL )_callDelegateWithBlock : (void (^)(id <MTRDeviceDelegate>))block
582
+ {
583
+ os_unfair_lock_assert_owner (&self->_lock );
584
+ id <MTRDeviceDelegate> delegate = _weakDelegate.strongObject ;
585
+ if (delegate) {
586
+ dispatch_async (_delegateQueue, ^{
587
+ block (delegate);
588
+ });
589
+ return YES ;
590
+ }
591
+ return NO ;
592
+ }
593
+
594
+ - (void )_callDelegateDeviceCachePrimed
595
+ {
596
+ os_unfair_lock_assert_owner (&self->_lock );
597
+ _delegateDeviceCachePrimedCalled = [self _callDelegateWithBlock: ^(id <MTRDeviceDelegate> delegate) {
598
+ if ([delegate respondsToSelector: @selector (deviceCachePrimed: )]) {
599
+ [delegate deviceCachePrimed: self ];
600
+ }
601
+ }];
602
+ }
603
+
577
604
// assume lock is held
578
605
- (void )_changeState : (MTRDeviceState)state
579
606
{
@@ -741,6 +768,12 @@ - (void)_handleReportEnd
741
768
_receivingReport = NO ;
742
769
_receivingPrimingReport = NO ;
743
770
_estimatedStartTimeFromGeneralDiagnosticsUpTime = nil ;
771
+
772
+ // First subscription report is priming report
773
+ if (!_delegateDeviceCachePrimedCalled) {
774
+ [self _callDelegateDeviceCachePrimed ];
775
+ }
776
+
744
777
// For unit testing only
745
778
#ifdef DEBUG
746
779
id delegate = _weakDelegate.strongObject ;
@@ -2147,6 +2180,48 @@ - (void)_removeExpectedValueForAttributePath:(MTRAttributePath *)attributePath e
2147
2180
}
2148
2181
}
2149
2182
2183
+ // This method checks if there is a need to inform delegate that the attribute cache has been "primed"
2184
+ // - The delegate callback is only called once
2185
+ - (void )_checkIfCacheIsPrimed
2186
+ {
2187
+ os_unfair_lock_assert_owner (&self->_lock );
2188
+
2189
+ // Only send the callback once per lifetime of MTRDevice
2190
+ if (_delegateDeviceCachePrimedCalled) {
2191
+ return ;
2192
+ }
2193
+
2194
+ // Check if root node descriptor exists
2195
+ NSDictionary * rootDescriptorPartsListDataValue = _readCache[[MTRAttributePath attributePathWithEndpointID: @(0 ) clusterID: @(MTRClusterIDTypeDescriptorID) attributeID: @(MTRAttributeIDTypeClusterDescriptorAttributePartsListID)]];
2196
+ if (!rootDescriptorPartsListDataValue || ![MTRArrayValueType isEqualToString: rootDescriptorPartsListDataValue[MTRTypeKey]]) {
2197
+ return ;
2198
+ }
2199
+ NSArray * partsList = rootDescriptorPartsListDataValue[MTRValueKey];
2200
+ if (![partsList isKindOfClass: [NSArray class ]] || !partsList.count ) {
2201
+ MTR_LOG_ERROR (" %@ unexpected type %@ for parts list %@" , self, [partsList class ], partsList);
2202
+ return ;
2203
+ }
2204
+
2205
+ // Check if we have cached descriptor clusters for each listed endpoint
2206
+ for (NSDictionary * endpointDataValue in partsList) {
2207
+ if (![MTRUnsignedIntegerValueType isEqual: endpointDataValue[MTRTypeKey]]) {
2208
+ MTR_LOG_ERROR (" %@ unexpected type for parts list item %@" , self, endpointDataValue);
2209
+ continue ;
2210
+ }
2211
+ NSNumber * endpoint = endpointDataValue[MTRValueKey];
2212
+ if (![endpoint isKindOfClass: [NSNumber class ]]) {
2213
+ MTR_LOG_ERROR (" %@ unexpected type for parts list item %@" , self, endpointDataValue);
2214
+ continue ;
2215
+ }
2216
+ NSDictionary * descriptorDeviceTypeListDataValue = _readCache[[MTRAttributePath attributePathWithEndpointID: endpoint clusterID: @(MTRClusterIDTypeDescriptorID) attributeID: @(MTRAttributeIDTypeClusterDescriptorAttributeDeviceTypeListID)]];
2217
+ if (![MTRArrayValueType isEqualToString: descriptorDeviceTypeListDataValue[MTRTypeKey]] || !descriptorDeviceTypeListDataValue[MTRValueKey]) {
2218
+ return ;
2219
+ }
2220
+ }
2221
+
2222
+ [self _callDelegateDeviceCachePrimed ];
2223
+ }
2224
+
2150
2225
- (MTRBaseDevice *)newBaseDevice
2151
2226
{
2152
2227
return [MTRBaseDevice deviceWithNodeID: self .nodeID controller: self .deviceController];
0 commit comments