11
11
use super :: {
12
12
central_delegate:: { CentralDelegate , CentralDelegateEvent } ,
13
13
framework:: {
14
- cb:: { self , CBManagerAuthorization , CBPeripheralState } ,
14
+ cb:: { self , CBManagerAuthorization , CBManagerState , CBPeripheralState } ,
15
15
ns,
16
16
} ,
17
17
future:: { BtlePlugFuture , BtlePlugFutureStateShared } ,
@@ -21,10 +21,7 @@ use super::{
21
21
nsuuid_to_uuid,
22
22
} ,
23
23
} ;
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 } ;
28
25
use crate :: Error ;
29
26
use cocoa:: {
30
27
base:: { id, nil} ,
@@ -148,6 +145,7 @@ pub enum CoreBluetoothReply {
148
145
ReadResult ( Vec < u8 > ) ,
149
146
Connected ( BTreeSet < Service > ) ,
150
147
State ( CBPeripheralState ) ,
148
+ ManagerState ( CBManagerState ) ,
151
149
Ok ,
152
150
Err ( String ) ,
153
151
}
@@ -408,11 +406,14 @@ pub enum CoreBluetoothMessage {
408
406
data : Vec < u8 > ,
409
407
future : CoreBluetoothReplyStateShared ,
410
408
} ,
409
+ FetchManagerState {
410
+ future : CoreBluetoothReplyStateShared ,
411
+ } ,
411
412
}
412
413
413
414
#[ derive( Debug ) ]
414
415
pub enum CoreBluetoothEvent {
415
- AdapterConnected ,
416
+ DidUpdateState ,
416
417
DeviceDiscovered {
417
418
uuid : Uuid ,
418
419
name : Option < String > ,
@@ -795,6 +796,18 @@ impl CoreBluetoothInternal {
795
796
}
796
797
}
797
798
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
+
798
811
fn write_value (
799
812
& mut self ,
800
813
peripheral_uuid : Uuid ,
@@ -837,6 +850,11 @@ impl CoreBluetoothInternal {
837
850
characteristic_uuid : Uuid ,
838
851
fut : CoreBluetoothReplyStateShared ,
839
852
) {
853
+ trace ! (
854
+ "Manager State {:?}" ,
855
+ cb:: centeralmanger_state( * self . manager)
856
+ ) ;
857
+
840
858
if let Some ( peripheral) = self . peripherals . get_mut ( & peripheral_uuid) {
841
859
if let Some ( service) = peripheral. services . get_mut ( & service_uuid) {
842
860
if let Some ( characteristic) = service. characteristics . get_mut ( & characteristic_uuid)
@@ -1010,7 +1028,7 @@ impl CoreBluetoothInternal {
1010
1028
// "ready" variable in our adapter that will cause scans/etc
1011
1029
// to fail if this hasn't updated.
1012
1030
CentralDelegateEvent :: DidUpdateState => {
1013
- self . dispatch_event( CoreBluetoothEvent :: AdapterConnected ) . await
1031
+ self . dispatch_event( CoreBluetoothEvent :: DidUpdateState ) . await
1014
1032
}
1015
1033
CentralDelegateEvent :: DiscoveredPeripheral { cbperipheral} => {
1016
1034
self . on_discovered_peripheral( cbperipheral) . await
@@ -1109,6 +1127,9 @@ impl CoreBluetoothInternal {
1109
1127
CoreBluetoothMessage :: IsConnected { peripheral_uuid, future} => {
1110
1128
self . is_connected( peripheral_uuid, future) ;
1111
1129
} ,
1130
+ CoreBluetoothMessage :: FetchManagerState { future} =>{
1131
+ self . get_manager_state( future) ;
1132
+ } ,
1112
1133
CoreBluetoothMessage :: ReadDescriptorValue { peripheral_uuid, service_uuid, characteristic_uuid, descriptor_uuid, future} => {
1113
1134
self . read_descriptor_value( peripheral_uuid, service_uuid, characteristic_uuid, descriptor_uuid, future)
1114
1135
}
@@ -1189,6 +1210,14 @@ pub fn run_corebluetooth_thread(
1189
1210
runtime. block_on ( async move {
1190
1211
let mut cbi = CoreBluetoothInternal :: new ( receiver, event_sender) ;
1191
1212
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
+ }
1192
1221
cbi. wait_for_message ( ) . await ;
1193
1222
}
1194
1223
} )
0 commit comments