@@ -350,6 +350,14 @@ @implementation MTRDevice_Concrete {
350
350
NSDate * _Nullable _mostRecentReportTimeForDescription;
351
351
// Copy of _lastSubscriptionFailureTime that is safe to use in description.
352
352
NSDate * _Nullable _lastSubscriptionFailureTimeForDescription;
353
+ // Copy of _state that is safe to use in description.
354
+ MTRDeviceState _deviceStateForDescription;
355
+ // Copy of _deviceCachePrimed that is safe to use in description.
356
+ BOOL _deviceCachePrimedForDescription;
357
+ // Copy of _estimatedStartTime that is safe to use in description.
358
+ NSDate * _Nullable _estimatedStartTimeForDescription;
359
+ // Copy of _estimatedSubscriptionLatency that is safe to use in description.
360
+ NSNumber * _Nullable _estimatedSubscriptionLatencyForDescription;
353
361
}
354
362
355
363
// synthesize superclass property readwrite accessors
@@ -473,11 +481,16 @@ - (NSDictionary *)_internalProperties
473
481
MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyKeyVendorID, _vid, properties);
474
482
MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyKeyProductID, _pid, properties);
475
483
MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyNetworkFeatures, _allNetworkFeatures, properties);
476
- MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyDeviceState , [NSNumber numberWithUnsignedInteger:_internalDeviceStateForDescription], properties);
484
+ MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyDeviceInternalState , [NSNumber numberWithUnsignedInteger:_internalDeviceStateForDescription], properties);
477
485
MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyLastSubscriptionAttemptWait, [NSNumber numberWithUnsignedInt:_lastSubscriptionAttemptWaitForDescription], properties);
478
486
MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyMostRecentReportTime, _mostRecentReportTimeForDescription, properties);
479
487
MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyLastSubscriptionFailureTime, _lastSubscriptionFailureTimeForDescription, properties);
480
488
489
+ MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyDeviceState, [NSNumber numberWithUnsignedInteger:_deviceStateForDescription], properties);
490
+ MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyDeviceCachePrimed, @(_deviceCachePrimedForDescription), properties);
491
+ MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyEstimatedStartTime, _estimatedStartTimeForDescription, properties);
492
+ MTR_OPTIONAL_ATTRIBUTE(kMTRDeviceInternalPropertyEstimatedSubscriptionLatency, _estimatedSubscriptionLatencyForDescription, properties);
493
+
481
494
return properties;
482
495
}
483
496
@@ -964,11 +977,15 @@ - (void)_changeState:(MTRDeviceState)state
964
977
os_unfair_lock_assert_owner(&self->_lock);
965
978
MTRDeviceState lastState = _state;
966
979
_state = state;
980
+ {
981
+ std::lock_guard lock(_descriptionLock);
982
+ _deviceStateForDescription = _state;
983
+ }
967
984
if (lastState != state) {
968
985
if (state != MTRDeviceStateReachable) {
969
986
MTR_LOG("%@ reachability state change %lu => %lu, set estimated start time to nil", self, static_cast<unsigned long>(lastState),
970
987
static_cast<unsigned long>(state));
971
- _estimatedStartTime = nil;
988
+ [self _updateEstimatedStartTime: nil] ;
972
989
_estimatedStartTimeFromGeneralDiagnosticsUpTime = nil;
973
990
} else {
974
991
MTR_LOG(
@@ -1025,6 +1042,15 @@ - (MTRInternalDeviceState)_getInternalState
1025
1042
}
1026
1043
#endif
1027
1044
1045
+ - (void)_updateEstimatedSubscriptionLatency:(NSNumber *)estimatedSubscriptionLatency
1046
+ {
1047
+ os_unfair_lock_assert_owner(&_lock);
1048
+ _estimatedSubscriptionLatency = estimatedSubscriptionLatency;
1049
+
1050
+ std::lock_guard lock(_descriptionLock);
1051
+ _estimatedSubscriptionLatencyForDescription = estimatedSubscriptionLatency;
1052
+ }
1053
+
1028
1054
// First Time Sync happens 2 minutes after reachability (this can be changed in the future)
1029
1055
#define MTR_DEVICE_TIME_UPDATE_INITIAL_WAIT_TIME_SEC (60 * 2)
1030
1056
- (void)_handleSubscriptionEstablished
@@ -1056,10 +1082,10 @@ - (void)_handleSubscriptionEstablished
1056
1082
// way around.
1057
1083
NSTimeInterval subscriptionLatency = -[initialSubscribeStart timeIntervalSinceNow];
1058
1084
if (_estimatedSubscriptionLatency == nil) {
1059
- _estimatedSubscriptionLatency = @(subscriptionLatency);
1085
+ [self _updateEstimatedSubscriptionLatency: @(subscriptionLatency)] ;
1060
1086
} else {
1061
1087
NSTimeInterval newSubscriptionLatencyEstimate = MTRDEVICE_SUBSCRIPTION_LATENCY_NEW_VALUE_WEIGHT * subscriptionLatency + (1 - MTRDEVICE_SUBSCRIPTION_LATENCY_NEW_VALUE_WEIGHT) * _estimatedSubscriptionLatency.doubleValue;
1062
- _estimatedSubscriptionLatency = @(newSubscriptionLatencyEstimate);
1088
+ [self _updateEstimatedSubscriptionLatency: @(newSubscriptionLatencyEstimate)] ;
1063
1089
}
1064
1090
[self _storePersistedDeviceData];
1065
1091
}
@@ -1733,6 +1759,15 @@ - (void)setStorageBehaviorConfiguration:(MTRDeviceStorageBehaviorConfiguration *
1733
1759
[self _resetStorageBehaviorState];
1734
1760
}
1735
1761
1762
+ - (void)_updateDeviceCachePrimed:(BOOL)deviceCachePrimed
1763
+ {
1764
+ os_unfair_lock_assert_owner(&_lock);
1765
+ _deviceCachePrimed = deviceCachePrimed;
1766
+
1767
+ std::lock_guard lock(_descriptionLock);
1768
+ _deviceCachePrimedForDescription = deviceCachePrimed;
1769
+ }
1770
+
1736
1771
- (void)_handleReportEnd
1737
1772
{
1738
1773
MTR_LOG("%@ handling report end", self);
@@ -1764,7 +1799,7 @@ - (void)_handleReportEnd
1764
1799
if (!_deviceCachePrimed) {
1765
1800
// This is the end of the priming sequence of data reports, so we have
1766
1801
// all the data for the device now.
1767
- _deviceCachePrimed = YES;
1802
+ [self _updateDeviceCachePrimed: YES] ;
1768
1803
[self _callDelegateDeviceCachePrimed];
1769
1804
}
1770
1805
@@ -1954,6 +1989,15 @@ - (BOOL)_interestedPaths:(NSArray * _Nullable)interestedPaths includesEventPath:
1954
1989
return filteredEvents;
1955
1990
}
1956
1991
1992
+ - (void)_updateEstimatedStartTime:(NSDate *)estimatedStartTime
1993
+ {
1994
+ os_unfair_lock_assert_owner(&_lock);
1995
+ _estimatedStartTime = estimatedStartTime;
1996
+
1997
+ std::lock_guard lock(_descriptionLock);
1998
+ _estimatedStartTimeForDescription = _estimatedStartTime;
1999
+ }
2000
+
1957
2001
- (void)_handleEventReport:(NSArray<NSDictionary<NSString *, id> *> *)eventReport
1958
2002
{
1959
2003
std::lock_guard lock(_lock);
@@ -1999,11 +2043,11 @@ - (void)_handleEventReport:(NSArray<NSDictionary<NSString *, id> *> *)eventRepor
1999
2043
// If UpTime was received, make use of it as mark of system start time
2000
2044
MTR_LOG("%@ StartUp event: set estimated start time forward to %@", self,
2001
2045
_estimatedStartTimeFromGeneralDiagnosticsUpTime);
2002
- _estimatedStartTime = _estimatedStartTimeFromGeneralDiagnosticsUpTime;
2046
+ [self _updateEstimatedStartTime: _estimatedStartTimeFromGeneralDiagnosticsUpTime] ;
2003
2047
} else {
2004
2048
// If UpTime was not received, reset estimated start time in case of reboot
2005
2049
MTR_LOG("%@ StartUp event: set estimated start time to nil", self);
2006
- _estimatedStartTime = nil;
2050
+ [self _updateEstimatedStartTime: nil] ;
2007
2051
}
2008
2052
}
2009
2053
@@ -2023,7 +2067,7 @@ - (void)_handleEventReport:(NSArray<NSDictionary<NSString *, id> *> *)eventRepor
2023
2067
NSTimeInterval eventTimeValue = eventTimeValueNumber.doubleValue;
2024
2068
NSDate * potentialSystemStartTime = [NSDate dateWithTimeIntervalSinceNow:-eventTimeValue];
2025
2069
if (!_estimatedStartTime || ([potentialSystemStartTime compare:_estimatedStartTime] == NSOrderedAscending)) {
2026
- _estimatedStartTime = potentialSystemStartTime;
2070
+ [self _updateEstimatedStartTime: potentialSystemStartTime] ;
2027
2071
}
2028
2072
}
2029
2073
@@ -3601,7 +3645,7 @@ - (NSArray *)_getAttributesToReportWithReportedValues:(NSArray<NSDictionary<NSSt
3601
3645
if (!_estimatedStartTime || ([potentialSystemStartTime compare:_estimatedStartTime] == NSOrderedAscending)) {
3602
3646
MTR_LOG("%@ General Diagnostics UpTime %.3lf: estimated start time %@ => %@", self, upTime,
3603
3647
oldSystemStartTime, potentialSystemStartTime);
3604
- _estimatedStartTime = potentialSystemStartTime;
3648
+ [self _updateEstimatedStartTime: potentialSystemStartTime] ;
3605
3649
}
3606
3650
3607
3651
// Save estimate in the subscription resumption case, for when StartUp event uses it
@@ -3693,7 +3737,7 @@ - (void)setPersistedClusterData:(NSDictionary<MTRClusterPath *, MTRDeviceCluster
3693
3737
3694
3738
// We have some stored data. Since we don't store data until the end of the
3695
3739
// initial priming report, our device cache must be primed.
3696
- _deviceCachePrimed = YES;
3740
+ [self _updateDeviceCachePrimed: YES] ;
3697
3741
}
3698
3742
3699
3743
- (void)_setLastInitialSubscribeLatency:(id)latency
@@ -3705,7 +3749,7 @@ - (void)_setLastInitialSubscribeLatency:(id)latency
3705
3749
return;
3706
3750
}
3707
3751
3708
- _estimatedSubscriptionLatency = latency;
3752
+ [self _updateEstimatedSubscriptionLatency: latency] ;
3709
3753
}
3710
3754
3711
3755
- (void)setPersistedDeviceData:(NSDictionary<NSString *, id> *)data
0 commit comments