Skip to content

Commit 6315ea5

Browse files
Darwin: Bug fixes in BleConnectionDelegateImpl (#29059)
* Avoid clearing the `ble` global if it points to a different instance. * Capture the CBPeripheral strong ref instead of the void * in dispatches. * Also read characteristic value before dispatching and improve logging.
1 parent c0bf847 commit 6315ea5

File tree

2 files changed

+23
-11
lines changed

2 files changed

+23
-11
lines changed

src/platform/Darwin/BleConnectionDelegate.h

+3
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,9 @@ class BleConnectionDelegateImpl : public Ble::BleConnectionDelegate
3131
virtual void NewConnection(Ble::BleLayer * bleLayer, void * appState, const SetupDiscriminator & connDiscriminator);
3232
virtual void NewConnection(Ble::BleLayer * bleLayer, void * appState, BLE_CONNECTION_OBJECT connObj);
3333
virtual CHIP_ERROR CancelConnection();
34+
35+
private:
36+
CHIP_ERROR DoCancel();
3437
};
3538

3639
} // namespace Internal

src/platform/Darwin/BleConnectionDelegateImpl.mm

+20-11
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,7 @@ - (void)removePeripheralsFromCache;
103103
// Make a copy of the device discriminator for the block to capture.
104104
SetupDiscriminator deviceDiscriminator = inDeviceDiscriminator;
105105

106-
ChipLogProgress(Ble, "%s", __FUNCTION__);
106+
ChipLogProgress(Ble, "ConnectionDelegate NewConnection with discriminator");
107107
if (!bleWorkQueue) {
108108
bleWorkQueue = dispatch_queue_create(kBleWorkQueueName, DISPATCH_QUEUE_SERIAL_WITH_AUTORELEASE_POOL);
109109
}
@@ -134,12 +134,13 @@ - (void)removePeripheralsFromCache;
134134
{
135135
assertChipStackLockedByCurrentThread();
136136

137-
ChipLogProgress(Ble, "%s", __FUNCTION__);
137+
ChipLogProgress(Ble, "ConnectionDelegate NewConnection with conn obj: %p", connObj);
138138

139139
if (!bleWorkQueue) {
140140
bleWorkQueue = dispatch_queue_create(kBleWorkQueueName, DISPATCH_QUEUE_SERIAL_WITH_AUTORELEASE_POOL);
141141
}
142142

143+
CBPeripheral * peripheral = (__bridge CBPeripheral *) connObj; // bridge (and retain) before dispatching
143144
dispatch_async(bleWorkQueue, ^{
144145
// The BLE_CONNECTION_OBJECT represent a CBPeripheral object. In order for it to be valid the central
145146
// manager needs to still be running.
@@ -157,15 +158,15 @@ - (void)removePeripheralsFromCache;
157158
ble.appState = appState;
158159
ble.onConnectionComplete = OnConnectionComplete;
159160
ble.onConnectionError = OnConnectionError;
160-
[ble updateWithPeripheral:(__bridge CBPeripheral *) connObj];
161+
[ble updateWithPeripheral:peripheral];
161162
});
162163
}
163164

164165
void BleConnectionDelegateImpl::StartScan(BleScannerDelegate * delegate)
165166
{
166167
assertChipStackLockedByCurrentThread();
167168

168-
ChipLogProgress(Ble, "%s", __FUNCTION__);
169+
ChipLogProgress(Ble, "ConnectionDelegate StartScan%s", (delegate ? " with delegate" : ""));
169170

170171
if (!bleWorkQueue) {
171172
bleWorkQueue = dispatch_queue_create(kBleWorkQueueName, DISPATCH_QUEUE_SERIAL_WITH_AUTORELEASE_POOL);
@@ -191,15 +192,19 @@ - (void)removePeripheralsFromCache;
191192

192193
void BleConnectionDelegateImpl::StopScan()
193194
{
194-
assertChipStackLockedByCurrentThread();
195-
CancelConnection();
195+
ChipLogProgress(Ble, "ConnectionDelegate StopScan");
196+
DoCancel();
196197
}
197198

198199
CHIP_ERROR BleConnectionDelegateImpl::CancelConnection()
199200
{
200-
assertChipStackLockedByCurrentThread();
201+
ChipLogProgress(Ble, "ConnectionDelegate CancelConnection");
202+
return DoCancel();
203+
}
201204

202-
ChipLogProgress(Ble, "%s", __FUNCTION__);
205+
CHIP_ERROR BleConnectionDelegateImpl::DoCancel()
206+
{
207+
assertChipStackLockedByCurrentThread();
203208
if (bleWorkQueue == nil) {
204209
return CHIP_NO_ERROR;
205210
}
@@ -284,6 +289,7 @@ - (void)setupTimer:(uint64_t)timeout
284289

285290
_timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, _workQueue);
286291
dispatch_source_set_event_handler(_timer, ^{
292+
ChipLogProgress(Ble, "ConnectionDelegate timeout");
287293
[self stop];
288294
[self dispatchConnectionError:BLE_ERROR_APP_CLOSED_CONNECTION];
289295
});
@@ -374,7 +380,7 @@ - (void)centralManager:(CBCentralManager *)central
374380
}
375381

376382
const uint8_t * bytes = (const uint8_t *) [serviceData bytes];
377-
if ([serviceData length] != 8) {
383+
if ([serviceData length] != sizeof(ChipBLEDeviceIdentificationInfo)) {
378384
NSMutableString * hexString = [NSMutableString stringWithCapacity:([serviceData length] * 2)];
379385
for (NSUInteger i = 0; i < [serviceData length]; i++) {
380386
[hexString appendString:[NSString stringWithFormat:@"%02lx", (unsigned long) bytes[i]]];
@@ -521,10 +527,11 @@ - (void)peripheral:(CBPeripheral *)peripheral
521527
chip::Ble::ChipBleUUID svcId;
522528
chip::Ble::ChipBleUUID charId;
523529
[BleConnection fillServiceWithCharacteristicUuids:characteristic svcId:&svcId charId:&charId];
530+
auto * value = characteristic.value; // read immediately before dispatching
524531

525532
dispatch_async(_chipWorkQueue, ^{
526533
// build a inet buffer from the rxEv and send to blelayer.
527-
auto msgBuf = chip::System::PacketBufferHandle::NewWithData(characteristic.value.bytes, characteristic.value.length);
534+
auto msgBuf = chip::System::PacketBufferHandle::NewWithData(value.bytes, value.length);
528535

529536
if (msgBuf.IsNull()) {
530537
ChipLogError(Ble, "Failed at allocating buffer for incoming BLE data");
@@ -583,7 +590,9 @@ - (void)stop
583590
_centralManager.delegate = nil;
584591
_centralManager = nil;
585592
_peripheral = nil;
586-
chip::DeviceLayer::Internal::ble = nil;
593+
if (chip::DeviceLayer::Internal::ble == self) {
594+
chip::DeviceLayer::Internal::ble = nil;
595+
}
587596
});
588597
});
589598
}

0 commit comments

Comments
 (0)