Skip to content

Commit 6eab2c5

Browse files
committed
fix(macos): Break the corebluetooth loop when manager turned off
In Core Bluetooth when the device is not applicable for background bluetooth manager will create an event for state change. And then change the manager state to power off. Currently, it is not tracked at all which leads to the forever stuck unresolved issues while the connection to peripheral is still held. An additional problem I faced that there is no way to manually kill the event loop of the corebluetooth from outside so the `CoreBluetoothInternal::drop` is never called because it is always living in the stalled thread. In this change, I added an API to access the manager state and exited the event loop when if the manager turned off.
1 parent 1937ff5 commit 6eab2c5

File tree

3 files changed

+69
-11
lines changed

3 files changed

+69
-11
lines changed

src/corebluetooth/adapter.rs

+15-3
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ use super::internal::{run_corebluetooth_thread, CoreBluetoothEvent, CoreBluetoot
22
use super::peripheral::{Peripheral, PeripheralId};
33
use crate::api::{Central, CentralEvent, ScanFilter};
44
use crate::common::adapter_manager::AdapterManager;
5+
use crate::corebluetooth::internal::{CoreBluetoothReply, CoreBluetoothReplyFuture};
56
use crate::{Error, Result};
67
use async_trait::async_trait;
78
use futures::channel::mpsc::{self, Sender};
@@ -29,7 +30,7 @@ impl Adapter {
2930
debug!("Waiting on adapter connect");
3031
if !matches!(
3132
receiver.next().await,
32-
Some(CoreBluetoothEvent::AdapterConnected)
33+
Some(CoreBluetoothEvent::DidUpdateState)
3334
) {
3435
return Err(Error::Other(
3536
"Adapter failed to connect.".to_string().into(),
@@ -39,7 +40,7 @@ impl Adapter {
3940
let manager = Arc::new(AdapterManager::default());
4041

4142
let manager_clone = manager.clone();
42-
let adapter_sender_clone = adapter_sender.clone();
43+
let mut adapter_sender_clone = adapter_sender.clone();
4344
task::spawn(async move {
4445
while let Some(msg) = receiver.next().await {
4546
match msg {
@@ -67,7 +68,18 @@ impl Adapter {
6768
CoreBluetoothEvent::DeviceDisconnected { uuid } => {
6869
manager_clone.emit(CentralEvent::DeviceDisconnected(uuid.into()));
6970
}
70-
_ => {}
71+
CoreBluetoothEvent::DidUpdateState => {
72+
let fut = CoreBluetoothReplyFuture::default();
73+
let _ = adapter_sender_clone
74+
.send(CoreBluetoothMessage::FetchManagerState {
75+
future: fut.get_state_clone(),
76+
})
77+
.await;
78+
79+
if let CoreBluetoothReply::ManagerState(state) = fut.await {
80+
error!("Adapter state changed to {:?}. Aborting manager", state)
81+
}
82+
}
7183
}
7284
}
7385
});

src/corebluetooth/framework.rs

+18-1
Original file line numberDiff line numberDiff line change
@@ -163,6 +163,22 @@ pub mod cb {
163163
}
164164
}
165165

166+
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
167+
#[repr(i64)]
168+
#[allow(dead_code)]
169+
pub enum CBManagerState {
170+
Unknown = 0,
171+
Resetting = 1,
172+
Unsupported = 2,
173+
Unauthorized = 3,
174+
PoweredOff = 4,
175+
PoweredOn = 5,
176+
}
177+
178+
pub fn centeralmanger_state(cbcentralmanager: id) -> CBManagerState {
179+
unsafe { msg_send![cbcentralmanager, state] }
180+
}
181+
166182
pub fn centralmanager_stopscan(cbcentralmanager: id) {
167183
unsafe { msg_send![cbcentralmanager, stopScan] }
168184
}
@@ -207,10 +223,11 @@ pub mod cb {
207223
unsafe { msg_send![cbperipheral, name] }
208224
}
209225

226+
#[allow(dead_code)]
210227
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
211228
#[repr(i64)]
212229
pub enum CBPeripheralState {
213-
Disonnected = 0,
230+
Disconnected = 0,
214231
Connecting = 1,
215232
Connected = 2,
216233
Disconnecting = 3,

src/corebluetooth/internal.rs

+36-7
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
use super::{
1212
central_delegate::{CentralDelegate, CentralDelegateEvent},
1313
framework::{
14-
cb::{self, CBManagerAuthorization, CBPeripheralState},
14+
cb::{self, CBManagerAuthorization, CBManagerState, CBPeripheralState},
1515
ns,
1616
},
1717
future::{BtlePlugFuture, BtlePlugFutureStateShared},
@@ -21,10 +21,7 @@ use super::{
2121
nsuuid_to_uuid,
2222
},
2323
};
24-
use crate::api::{
25-
bleuuid::uuid_from_u16, CharPropFlags, Characteristic, Descriptor, ScanFilter, Service,
26-
WriteType,
27-
};
24+
use crate::api::{CharPropFlags, Characteristic, Descriptor, ScanFilter, Service, WriteType};
2825
use crate::Error;
2926
use cocoa::{
3027
base::{id, nil},
@@ -148,6 +145,7 @@ pub enum CoreBluetoothReply {
148145
ReadResult(Vec<u8>),
149146
Connected(BTreeSet<Service>),
150147
State(CBPeripheralState),
148+
ManagerState(CBManagerState),
151149
Ok,
152150
Err(String),
153151
}
@@ -408,11 +406,14 @@ pub enum CoreBluetoothMessage {
408406
data: Vec<u8>,
409407
future: CoreBluetoothReplyStateShared,
410408
},
409+
FetchManagerState {
410+
future: CoreBluetoothReplyStateShared,
411+
},
411412
}
412413

413414
#[derive(Debug)]
414415
pub enum CoreBluetoothEvent {
415-
AdapterConnected,
416+
DidUpdateState,
416417
DeviceDiscovered {
417418
uuid: Uuid,
418419
name: Option<String>,
@@ -795,6 +796,18 @@ impl CoreBluetoothInternal {
795796
}
796797
}
797798

799+
fn get_manager_state_sync(&mut self) -> CBManagerState {
800+
cb::centeralmanger_state(*self.manager)
801+
}
802+
803+
fn get_manager_state(&mut self, fut: CoreBluetoothReplyStateShared) {
804+
let state = cb::centeralmanger_state(*self.manager);
805+
trace!("Manager state {:?} ", state);
806+
fut.lock()
807+
.unwrap()
808+
.set_reply(CoreBluetoothReply::ManagerState(state));
809+
}
810+
798811
fn write_value(
799812
&mut self,
800813
peripheral_uuid: Uuid,
@@ -837,6 +850,11 @@ impl CoreBluetoothInternal {
837850
characteristic_uuid: Uuid,
838851
fut: CoreBluetoothReplyStateShared,
839852
) {
853+
trace!(
854+
"Manager State {:?}",
855+
cb::centeralmanger_state(*self.manager)
856+
);
857+
840858
if let Some(peripheral) = self.peripherals.get_mut(&peripheral_uuid) {
841859
if let Some(service) = peripheral.services.get_mut(&service_uuid) {
842860
if let Some(characteristic) = service.characteristics.get_mut(&characteristic_uuid)
@@ -1010,7 +1028,7 @@ impl CoreBluetoothInternal {
10101028
// "ready" variable in our adapter that will cause scans/etc
10111029
// to fail if this hasn't updated.
10121030
CentralDelegateEvent::DidUpdateState => {
1013-
self.dispatch_event(CoreBluetoothEvent::AdapterConnected).await
1031+
self.dispatch_event(CoreBluetoothEvent::DidUpdateState).await
10141032
}
10151033
CentralDelegateEvent::DiscoveredPeripheral{cbperipheral} => {
10161034
self.on_discovered_peripheral(cbperipheral).await
@@ -1109,6 +1127,9 @@ impl CoreBluetoothInternal {
11091127
CoreBluetoothMessage::IsConnected{peripheral_uuid, future} => {
11101128
self.is_connected(peripheral_uuid, future);
11111129
},
1130+
CoreBluetoothMessage::FetchManagerState {future} =>{
1131+
self.get_manager_state(future);
1132+
},
11121133
CoreBluetoothMessage::ReadDescriptorValue{peripheral_uuid, service_uuid, characteristic_uuid, descriptor_uuid, future} => {
11131134
self.read_descriptor_value(peripheral_uuid, service_uuid, characteristic_uuid, descriptor_uuid, future)
11141135
}
@@ -1189,6 +1210,14 @@ pub fn run_corebluetooth_thread(
11891210
runtime.block_on(async move {
11901211
let mut cbi = CoreBluetoothInternal::new(receiver, event_sender);
11911212
loop {
1213+
// When the IOS or MacOS device if powered off or locked
1214+
// the manager state will suddenly throw DidUpdateState event and turn off.
1215+
// If we are not exiting the main loop here the futures requested after
1216+
// power off will be stuck forever.
1217+
if cbi.get_manager_state_sync() == CBManagerState::PoweredOff {
1218+
trace!("Breaking out of the corebluetooth loop. Manager is off.");
1219+
break;
1220+
}
11921221
cbi.wait_for_message().await;
11931222
}
11941223
})

0 commit comments

Comments
 (0)