@@ -106,7 +106,8 @@ static bool IsValidCATNumber(id _Nullable value)
106
106
@implementation MTRDeviceControllerDataStore {
107
107
id <MTRDeviceControllerStorageDelegate> _storageDelegate;
108
108
dispatch_queue_t _storageDelegateQueue;
109
- MTRDeviceController * _controller;
109
+ // Controller owns us, so we have to make sure to not keep it alive.
110
+ __weak MTRDeviceController * _controller;
110
111
// Array of nodes with resumption info, oldest-stored first.
111
112
NSMutableArray <NSNumber *> * _nodesWithResumptionInfo;
112
113
}
@@ -126,7 +127,9 @@ - (nullable instancetype)initWithController:(MTRDeviceController *)controller
126
127
__block id resumptionNodeList;
127
128
dispatch_sync (_storageDelegateQueue, ^{
128
129
@autoreleasepool {
129
- resumptionNodeList = [_storageDelegate controller: _controller
130
+ // NOTE: controller, not our weak ref, since we know it's still
131
+ // valid under this sync dispatch.
132
+ resumptionNodeList = [_storageDelegate controller: controller
130
133
valueForKey: sResumptionNodeListKey
131
134
securityLevel: MTRStorageSecurityLevelSecure
132
135
sharingType: MTRStorageSharingTypeNotShared];
@@ -154,9 +157,15 @@ - (nullable instancetype)initWithController:(MTRDeviceController *)controller
154
157
- (void )fetchAttributeDataForAllDevices : (MTRDeviceControllerDataStoreClusterDataHandler)clusterDataHandler
155
158
{
156
159
__block NSDictionary <NSString *, id > * dataStoreSecureLocalValues = nil ;
160
+ MTRDeviceController * controller = _controller;
161
+ if (controller == nil ) {
162
+ // Not expected; no way to call delegate without controller.
163
+ return ;
164
+ }
165
+
157
166
dispatch_sync (_storageDelegateQueue, ^{
158
167
if ([self ->_storageDelegate respondsToSelector: @selector (valuesForController:securityLevel:sharingType: )]) {
159
- dataStoreSecureLocalValues = [self ->_storageDelegate valuesForController: self ->_controller securityLevel: MTRStorageSecurityLevelSecure sharingType: MTRStorageSharingTypeNotShared];
168
+ dataStoreSecureLocalValues = [self ->_storageDelegate valuesForController: controller securityLevel: MTRStorageSecurityLevelSecure sharingType: MTRStorageSharingTypeNotShared];
160
169
}
161
170
});
162
171
@@ -177,32 +186,38 @@ - (nullable MTRCASESessionResumptionInfo *)findResumptionInfoByResumptionID:(NSD
177
186
178
187
- (void )storeResumptionInfo : (MTRCASESessionResumptionInfo *)resumptionInfo
179
188
{
189
+ MTRDeviceController * controller = _controller;
190
+ if (controller == nil ) {
191
+ // Not expected; no way to call delegate without controller.
192
+ return ;
193
+ }
194
+
180
195
auto * oldInfo = [self findResumptionInfoByNodeID: resumptionInfo.nodeID];
181
196
dispatch_sync (_storageDelegateQueue, ^{
182
197
if (oldInfo != nil ) {
183
198
// Remove old resumption id key. No need to do that for the
184
199
// node id, because we are about to overwrite it.
185
- [_storageDelegate controller: _controller
200
+ [_storageDelegate controller: controller
186
201
removeValueForKey: ResumptionByResumptionIDKey (oldInfo.resumptionID)
187
202
securityLevel: MTRStorageSecurityLevelSecure
188
203
sharingType: MTRStorageSharingTypeNotShared];
189
204
[_nodesWithResumptionInfo removeObject: resumptionInfo.nodeID];
190
205
}
191
206
192
- [_storageDelegate controller: _controller
207
+ [_storageDelegate controller: controller
193
208
storeValue: resumptionInfo
194
209
forKey: ResumptionByNodeIDKey (resumptionInfo.nodeID)
195
210
securityLevel: MTRStorageSecurityLevelSecure
196
211
sharingType: MTRStorageSharingTypeNotShared];
197
- [_storageDelegate controller: _controller
212
+ [_storageDelegate controller: controller
198
213
storeValue: resumptionInfo
199
214
forKey: ResumptionByResumptionIDKey (resumptionInfo.resumptionID)
200
215
securityLevel: MTRStorageSecurityLevelSecure
201
216
sharingType: MTRStorageSharingTypeNotShared];
202
217
203
218
// Update our resumption info node list.
204
219
[_nodesWithResumptionInfo addObject: resumptionInfo.nodeID];
205
- [_storageDelegate controller: _controller
220
+ [_storageDelegate controller: controller
206
221
storeValue: [_nodesWithResumptionInfo copy ]
207
222
forKey: sResumptionNodeListKey
208
223
securityLevel: MTRStorageSecurityLevelSecure
@@ -212,17 +227,23 @@ - (void)storeResumptionInfo:(MTRCASESessionResumptionInfo *)resumptionInfo
212
227
213
228
- (void )clearAllResumptionInfo
214
229
{
230
+ MTRDeviceController * controller = _controller;
231
+ if (controller == nil ) {
232
+ // Not expected; no way to call delegate without controller.
233
+ return ;
234
+ }
235
+
215
236
// Can we do less dispatch? We would need to have a version of
216
237
// _findResumptionInfoWithKey that assumes we are already on the right queue.
217
238
for (NSNumber * nodeID in _nodesWithResumptionInfo) {
218
239
auto * oldInfo = [self findResumptionInfoByNodeID: nodeID];
219
240
if (oldInfo != nil ) {
220
241
dispatch_sync (_storageDelegateQueue, ^{
221
- [_storageDelegate controller: _controller
242
+ [_storageDelegate controller: controller
222
243
removeValueForKey: ResumptionByResumptionIDKey (oldInfo.resumptionID)
223
244
securityLevel: MTRStorageSecurityLevelSecure
224
245
sharingType: MTRStorageSharingTypeNotShared];
225
- [_storageDelegate controller: _controller
246
+ [_storageDelegate controller: controller
226
247
removeValueForKey: ResumptionByNodeIDKey (oldInfo.nodeID)
227
248
securityLevel: MTRStorageSecurityLevelSecure
228
249
sharingType: MTRStorageSharingTypeNotShared];
@@ -235,9 +256,15 @@ - (void)clearAllResumptionInfo
235
256
236
257
- (CHIP_ERROR)storeLastLocallyUsedNOC : (MTRCertificateTLVBytes)noc
237
258
{
259
+ MTRDeviceController * controller = _controller;
260
+ if (controller == nil ) {
261
+ // Not expected; no way to call delegate without controller.
262
+ return CHIP_ERROR_PERSISTED_STORAGE_FAILED;
263
+ }
264
+
238
265
__block BOOL ok;
239
266
dispatch_sync (_storageDelegateQueue, ^{
240
- ok = [_storageDelegate controller: _controller
267
+ ok = [_storageDelegate controller: controller
241
268
storeValue: noc
242
269
forKey: sLastLocallyUsedNOCKey
243
270
securityLevel: MTRStorageSecurityLevelSecure
@@ -248,10 +275,16 @@ - (CHIP_ERROR)storeLastLocallyUsedNOC:(MTRCertificateTLVBytes)noc
248
275
249
276
- (MTRCertificateTLVBytes _Nullable)fetchLastLocallyUsedNOC
250
277
{
278
+ MTRDeviceController * controller = _controller;
279
+ if (controller == nil ) {
280
+ // Not expected; no way to call delegate without controller.
281
+ return nil ;
282
+ }
283
+
251
284
__block id data;
252
285
dispatch_sync (_storageDelegateQueue, ^{
253
286
@autoreleasepool {
254
- data = [_storageDelegate controller: _controller
287
+ data = [_storageDelegate controller: controller
255
288
valueForKey: sLastLocallyUsedNOCKey
256
289
securityLevel: MTRStorageSecurityLevelSecure
257
290
sharingType: MTRStorageSharingTypeNotShared];
@@ -271,6 +304,12 @@ - (MTRCertificateTLVBytes _Nullable)fetchLastLocallyUsedNOC
271
304
272
305
- (nullable MTRCASESessionResumptionInfo *)_findResumptionInfoWithKey : (nullable NSString *)key
273
306
{
307
+ MTRDeviceController * controller = _controller;
308
+ if (controller == nil ) {
309
+ // Not expected; no way to call delegate without controller.
310
+ return nil ;
311
+ }
312
+
274
313
// key could be nil if [NSString stringWithFormat] returns nil for some reason.
275
314
if (key == nil ) {
276
315
return nil ;
@@ -279,7 +318,7 @@ - (nullable MTRCASESessionResumptionInfo *)_findResumptionInfoWithKey:(nullable
279
318
__block id resumptionInfo;
280
319
dispatch_sync (_storageDelegateQueue, ^{
281
320
@autoreleasepool {
282
- resumptionInfo = [_storageDelegate controller: _controller
321
+ resumptionInfo = [_storageDelegate controller: controller
283
322
valueForKey: key
284
323
securityLevel: MTRStorageSecurityLevelSecure
285
324
sharingType: MTRStorageSharingTypeNotShared];
@@ -318,9 +357,15 @@ - (nullable MTRCASESessionResumptionInfo *)_findResumptionInfoWithKey:(nullable
318
357
319
358
- (id )_fetchAttributeCacheValueForKey : (NSString *)key expectedClass : (Class )expectedClass ;
320
359
{
360
+ MTRDeviceController * controller = _controller;
361
+ if (controller == nil ) {
362
+ // Not expected; no way to call delegate without controller.
363
+ return nil ;
364
+ }
365
+
321
366
id data;
322
367
@autoreleasepool {
323
- data = [_storageDelegate controller: _controller
368
+ data = [_storageDelegate controller: controller
324
369
valueForKey: key
325
370
securityLevel: MTRStorageSecurityLevelSecure
326
371
sharingType: MTRStorageSharingTypeNotShared];
@@ -338,7 +383,13 @@ - (id)_fetchAttributeCacheValueForKey:(NSString *)key expectedClass:(Class)expec
338
383
339
384
- (BOOL )_storeAttributeCacheValue : (id )value forKey : (NSString *)key
340
385
{
341
- return [_storageDelegate controller: _controller
386
+ MTRDeviceController * controller = _controller;
387
+ if (controller == nil ) {
388
+ // Not expected; no way to call delegate without controller.
389
+ return NO ;
390
+ }
391
+
392
+ return [_storageDelegate controller: controller
342
393
storeValue: value
343
394
forKey: key
344
395
securityLevel: MTRStorageSecurityLevelSecure
@@ -347,15 +398,27 @@ - (BOOL)_storeAttributeCacheValue:(id)value forKey:(NSString *)key
347
398
348
399
- (BOOL )_bulkStoreAttributeCacheValues : (NSDictionary <NSString *, id<NSSecureCoding >> *)values
349
400
{
350
- return [_storageDelegate controller: _controller
401
+ MTRDeviceController * controller = _controller;
402
+ if (controller == nil ) {
403
+ // Not expected; no way to call delegate without controller.
404
+ return NO ;
405
+ }
406
+
407
+ return [_storageDelegate controller: controller
351
408
storeValues: values
352
409
securityLevel: MTRStorageSecurityLevelSecure
353
410
sharingType: MTRStorageSharingTypeNotShared];
354
411
}
355
412
356
413
- (BOOL )_removeAttributeCacheValueForKey : (NSString *)key
357
414
{
358
- return [_storageDelegate controller: _controller
415
+ MTRDeviceController * controller = _controller;
416
+ if (controller == nil ) {
417
+ // Not expected; no way to call delegate without controller.
418
+ return NO ;
419
+ }
420
+
421
+ return [_storageDelegate controller: controller
359
422
removeValueForKey: key
360
423
securityLevel: MTRStorageSecurityLevelSecure
361
424
sharingType: MTRStorageSharingTypeNotShared];
0 commit comments