mirror of
https://github.com/servo/servo.git
synced 2025-06-04 07:35:36 +00:00
* Use 2024 style edition Signed-off-by: Simon Wülker <simon.wuelker@arcor.de> * Reformat all code Signed-off-by: Simon Wülker <simon.wuelker@arcor.de> --------- Signed-off-by: Simon Wülker <simon.wuelker@arcor.de>
991 lines
36 KiB
Rust
991 lines
36 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
|
|
pub mod adapter;
|
|
pub mod bluetooth;
|
|
#[cfg(not(any(
|
|
all(target_os = "linux", feature = "native-bluetooth"),
|
|
all(target_os = "android", feature = "native-bluetooth"),
|
|
all(target_os = "macos", feature = "native-bluetooth")
|
|
)))]
|
|
mod empty;
|
|
mod macros;
|
|
pub mod test;
|
|
|
|
use std::borrow::ToOwned;
|
|
use std::collections::{HashMap, HashSet};
|
|
use std::string::String;
|
|
use std::thread;
|
|
use std::time::Duration;
|
|
|
|
use base::id::WebViewId;
|
|
use bitflags::bitflags;
|
|
use bluetooth_traits::blocklist::{Blocklist, uuid_is_blocklisted};
|
|
use bluetooth_traits::scanfilter::{
|
|
BluetoothScanfilter, BluetoothScanfilterSequence, RequestDeviceoptions,
|
|
};
|
|
use bluetooth_traits::{
|
|
BluetoothCharacteristicMsg, BluetoothDescriptorMsg, BluetoothDeviceMsg, BluetoothError,
|
|
BluetoothRequest, BluetoothResponse, BluetoothResponseResult, BluetoothResult,
|
|
BluetoothServiceMsg, GATTType,
|
|
};
|
|
use embedder_traits::{EmbedderMsg, EmbedderProxy};
|
|
use ipc_channel::ipc::{self, IpcReceiver, IpcSender};
|
|
use log::warn;
|
|
use servo_config::pref;
|
|
use servo_rand::{self, Rng};
|
|
|
|
use crate::bluetooth::{
|
|
BluetoothAdapter, BluetoothDevice, BluetoothGATTCharacteristic, BluetoothGATTDescriptor,
|
|
BluetoothGATTService,
|
|
};
|
|
|
|
// A transaction not completed within 30 seconds shall time out. Such a transaction shall be considered to have failed.
|
|
// https://www.bluetooth.org/DocMan/handlers/DownloadDoc.ashx?doc_id=286439 (Vol. 3, page 480)
|
|
const MAXIMUM_TRANSACTION_TIME: u8 = 30;
|
|
const CONNECTION_TIMEOUT_MS: u64 = 1000;
|
|
// The discovery session needs some time to find any nearby devices
|
|
const DISCOVERY_TIMEOUT_MS: u64 = 1500;
|
|
|
|
bitflags! {
|
|
struct Flags: u32 {
|
|
const BROADCAST = 0b000000001;
|
|
const READ = 0b000000010;
|
|
const WRITE_WITHOUT_RESPONSE = 0b000000100;
|
|
const WRITE = 0b000001000;
|
|
const NOTIFY = 0b000010000;
|
|
const INDICATE = 0b000100000;
|
|
const AUTHENTICATED_SIGNED_WRITES = 0b001000000;
|
|
const RELIABLE_WRITE = 0b010000000;
|
|
const WRITABLE_AUXILIARIES = 0b100000000;
|
|
}
|
|
}
|
|
|
|
macro_rules! return_if_cached(
|
|
($cache:expr, $key:expr) => (
|
|
if $cache.contains_key($key) {
|
|
return $cache.get($key);
|
|
}
|
|
);
|
|
);
|
|
|
|
pub trait BluetoothThreadFactory {
|
|
fn new(embedder_proxy: EmbedderProxy) -> Self;
|
|
}
|
|
|
|
impl BluetoothThreadFactory for IpcSender<BluetoothRequest> {
|
|
fn new(embedder_proxy: EmbedderProxy) -> IpcSender<BluetoothRequest> {
|
|
let (sender, receiver) = ipc::channel().unwrap();
|
|
let adapter = if pref!(dom_bluetooth_enabled) {
|
|
BluetoothAdapter::new()
|
|
} else {
|
|
BluetoothAdapter::new_mock()
|
|
}
|
|
.ok();
|
|
thread::Builder::new()
|
|
.name("Bluetooth".to_owned())
|
|
.spawn(move || {
|
|
BluetoothManager::new(receiver, adapter, embedder_proxy).start();
|
|
})
|
|
.expect("Thread spawning failed");
|
|
sender
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#matches-a-filter
|
|
fn matches_filter(device: &BluetoothDevice, filter: &BluetoothScanfilter) -> bool {
|
|
if filter.is_empty_or_invalid() {
|
|
return false;
|
|
}
|
|
|
|
// Step 1.
|
|
if let Some(name) = filter.get_name() {
|
|
if device.get_name().ok() != Some(name.to_string()) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Step 2.
|
|
if !filter.get_name_prefix().is_empty() {
|
|
if let Ok(device_name) = device.get_name() {
|
|
if !device_name.starts_with(filter.get_name_prefix()) {
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Step 3.
|
|
if !filter.get_services().is_empty() {
|
|
if let Ok(device_uuids) = device.get_uuids() {
|
|
for service in filter.get_services() {
|
|
if !device_uuids.iter().any(|x| x == service) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Step 4.
|
|
if let Some(manufacturer_data) = filter.get_manufacturer_data() {
|
|
let advertised_manufacturer_data = match device.get_manufacturer_data() {
|
|
Ok(data) => data,
|
|
Err(_) => return false,
|
|
};
|
|
for (id, (prefix, mask)) in manufacturer_data.iter() {
|
|
if let Some(advertised_data) = advertised_manufacturer_data.get(id) {
|
|
if !data_filter_matches(advertised_data, prefix, mask) {
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Step 5.
|
|
if let Some(service_data) = filter.get_service_data() {
|
|
let advertised_service_data = match device.get_service_data() {
|
|
Ok(data) => data,
|
|
Err(_) => return false,
|
|
};
|
|
for (uuid, (prefix, mask)) in service_data.iter() {
|
|
if let Some(advertised_data) = advertised_service_data.get(uuid.as_str()) {
|
|
if !data_filter_matches(advertised_data, prefix, mask) {
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Step 6.
|
|
true
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#bluetoothdatafilterinit-matches
|
|
fn data_filter_matches(data: &[u8], prefix: &[u8], mask: &[u8]) -> bool {
|
|
// Step 1-2: No need to copy the bytes here.
|
|
// Step 3.
|
|
if data.len() < prefix.len() {
|
|
return false;
|
|
}
|
|
|
|
// Step 4.
|
|
for ((data, mask), prefix) in data.iter().zip(mask.iter()).zip(prefix.iter()) {
|
|
if data & mask != prefix & mask {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Step 5.
|
|
true
|
|
}
|
|
|
|
fn matches_filters(device: &BluetoothDevice, filters: &BluetoothScanfilterSequence) -> bool {
|
|
if filters.has_empty_or_invalid_filter() {
|
|
return false;
|
|
}
|
|
|
|
filters.iter().any(|f| matches_filter(device, f))
|
|
}
|
|
|
|
fn is_mock_adapter(adapter: &BluetoothAdapter) -> bool {
|
|
matches!(adapter, &BluetoothAdapter::Mock(_))
|
|
}
|
|
|
|
pub struct BluetoothManager {
|
|
receiver: IpcReceiver<BluetoothRequest>,
|
|
adapter: Option<BluetoothAdapter>,
|
|
address_to_id: HashMap<String, String>,
|
|
service_to_device: HashMap<String, String>,
|
|
characteristic_to_service: HashMap<String, String>,
|
|
descriptor_to_characteristic: HashMap<String, String>,
|
|
cached_devices: HashMap<String, BluetoothDevice>,
|
|
cached_services: HashMap<String, BluetoothGATTService>,
|
|
cached_characteristics: HashMap<String, BluetoothGATTCharacteristic>,
|
|
cached_descriptors: HashMap<String, BluetoothGATTDescriptor>,
|
|
allowed_services: HashMap<String, HashSet<String>>,
|
|
embedder_proxy: EmbedderProxy,
|
|
}
|
|
|
|
impl BluetoothManager {
|
|
pub fn new(
|
|
receiver: IpcReceiver<BluetoothRequest>,
|
|
adapter: Option<BluetoothAdapter>,
|
|
embedder_proxy: EmbedderProxy,
|
|
) -> BluetoothManager {
|
|
BluetoothManager {
|
|
receiver,
|
|
adapter,
|
|
address_to_id: HashMap::new(),
|
|
service_to_device: HashMap::new(),
|
|
characteristic_to_service: HashMap::new(),
|
|
descriptor_to_characteristic: HashMap::new(),
|
|
cached_devices: HashMap::new(),
|
|
cached_services: HashMap::new(),
|
|
cached_characteristics: HashMap::new(),
|
|
cached_descriptors: HashMap::new(),
|
|
allowed_services: HashMap::new(),
|
|
embedder_proxy,
|
|
}
|
|
}
|
|
|
|
fn start(&mut self) {
|
|
while let Ok(msg) = self.receiver.recv() {
|
|
match msg {
|
|
BluetoothRequest::RequestDevice(options, sender) => {
|
|
let _ = sender.send(self.request_device(options));
|
|
},
|
|
BluetoothRequest::GATTServerConnect(device_id, sender) => {
|
|
let _ = sender.send(self.gatt_server_connect(device_id));
|
|
},
|
|
BluetoothRequest::GATTServerDisconnect(device_id, sender) => {
|
|
let _ = sender.send(self.gatt_server_disconnect(device_id));
|
|
},
|
|
BluetoothRequest::GetGATTChildren(id, uuid, single, child_type, sender) => {
|
|
let _ = sender.send(self.get_gatt_children(id, uuid, single, child_type));
|
|
},
|
|
BluetoothRequest::ReadValue(id, sender) => {
|
|
let _ = sender.send(self.read_value(id));
|
|
},
|
|
BluetoothRequest::WriteValue(id, value, sender) => {
|
|
let _ = sender.send(self.write_value(id, value));
|
|
},
|
|
BluetoothRequest::EnableNotification(id, enable, sender) => {
|
|
let _ = sender.send(self.enable_notification(id, enable));
|
|
},
|
|
BluetoothRequest::WatchAdvertisements(id, sender) => {
|
|
let _ = sender.send(self.watch_advertisements(id));
|
|
},
|
|
BluetoothRequest::Test(data_set_name, sender) => {
|
|
let _ = sender.send(self.test(data_set_name));
|
|
},
|
|
BluetoothRequest::SetRepresentedToNull(
|
|
service_ids,
|
|
characteristic_ids,
|
|
descriptor_ids,
|
|
) => self.remove_ids_from_caches(service_ids, characteristic_ids, descriptor_ids),
|
|
BluetoothRequest::IsRepresentedDeviceNull(id, sender) => {
|
|
let _ = sender.send(!self.device_is_cached(&id));
|
|
},
|
|
BluetoothRequest::GetAvailability(sender) => {
|
|
let _ = sender.send(self.get_availability());
|
|
},
|
|
BluetoothRequest::MatchesFilter(id, filters, sender) => {
|
|
let _ = sender.send(self.device_matches_filter(&id, &filters));
|
|
},
|
|
BluetoothRequest::Exit => break,
|
|
}
|
|
}
|
|
}
|
|
|
|
// Test
|
|
|
|
fn test(&mut self, data_set_name: String) -> BluetoothResult<()> {
|
|
self.address_to_id.clear();
|
|
self.service_to_device.clear();
|
|
self.characteristic_to_service.clear();
|
|
self.descriptor_to_characteristic.clear();
|
|
self.cached_devices.clear();
|
|
self.cached_services.clear();
|
|
self.cached_characteristics.clear();
|
|
self.cached_descriptors.clear();
|
|
self.allowed_services.clear();
|
|
self.adapter = BluetoothAdapter::new_mock().ok();
|
|
match test::test(self, data_set_name) {
|
|
Ok(_) => Ok(()),
|
|
Err(error) => Err(BluetoothError::Type(error.to_string())),
|
|
}
|
|
}
|
|
|
|
fn remove_ids_from_caches(
|
|
&mut self,
|
|
service_ids: Vec<String>,
|
|
characteristic_ids: Vec<String>,
|
|
descriptor_ids: Vec<String>,
|
|
) {
|
|
for id in service_ids {
|
|
self.cached_services.remove(&id);
|
|
self.service_to_device.remove(&id);
|
|
}
|
|
|
|
for id in characteristic_ids {
|
|
self.cached_characteristics.remove(&id);
|
|
self.characteristic_to_service.remove(&id);
|
|
}
|
|
|
|
for id in descriptor_ids {
|
|
self.cached_descriptors.remove(&id);
|
|
self.descriptor_to_characteristic.remove(&id);
|
|
}
|
|
}
|
|
|
|
// Adapter
|
|
|
|
pub fn get_or_create_adapter(&mut self) -> Option<BluetoothAdapter> {
|
|
let adapter_valid = self
|
|
.adapter
|
|
.as_ref()
|
|
.is_some_and(|a| a.get_address().is_ok());
|
|
if !adapter_valid {
|
|
self.adapter = BluetoothAdapter::new().ok();
|
|
}
|
|
|
|
let adapter = self.adapter.as_ref()?;
|
|
|
|
if is_mock_adapter(adapter) && !adapter.is_present().unwrap_or(false) {
|
|
return None;
|
|
}
|
|
|
|
self.adapter.clone()
|
|
}
|
|
|
|
fn get_adapter(&mut self) -> BluetoothResult<BluetoothAdapter> {
|
|
match self.get_or_create_adapter() {
|
|
Some(adapter) => {
|
|
if !adapter.is_powered().unwrap_or(false) {
|
|
return Err(BluetoothError::NotFound);
|
|
}
|
|
Ok(adapter)
|
|
},
|
|
None => Err(BluetoothError::NotFound),
|
|
}
|
|
}
|
|
|
|
// Device
|
|
|
|
fn get_and_cache_devices(&mut self, adapter: &mut BluetoothAdapter) -> Vec<BluetoothDevice> {
|
|
let devices = adapter.get_devices().unwrap_or_default();
|
|
for device in &devices {
|
|
if let Ok(address) = device.get_address() {
|
|
#[allow(clippy::map_entry)] // False positive, the fix creates a borrowing error
|
|
if !self.address_to_id.contains_key(&address) {
|
|
let generated_id = self.generate_device_id();
|
|
self.address_to_id.insert(address, generated_id.clone());
|
|
self.cached_devices
|
|
.insert(generated_id.clone(), device.clone());
|
|
self.allowed_services.insert(generated_id, HashSet::new());
|
|
}
|
|
}
|
|
}
|
|
self.cached_devices.values().cloned().collect()
|
|
}
|
|
|
|
fn get_device(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
device_id: &str,
|
|
) -> Option<&BluetoothDevice> {
|
|
return_if_cached!(self.cached_devices, device_id);
|
|
self.get_and_cache_devices(adapter);
|
|
return_if_cached!(self.cached_devices, device_id);
|
|
None
|
|
}
|
|
|
|
fn select_device(
|
|
&mut self,
|
|
webview_id: WebViewId,
|
|
devices: Vec<BluetoothDevice>,
|
|
adapter: &BluetoothAdapter,
|
|
) -> Option<String> {
|
|
if is_mock_adapter(adapter) {
|
|
for device in &devices {
|
|
if let Ok(address) = device.get_address() {
|
|
return Some(address);
|
|
}
|
|
}
|
|
return None;
|
|
}
|
|
|
|
let mut dialog_rows: Vec<String> = vec![];
|
|
for device in devices {
|
|
dialog_rows.extend_from_slice(&[
|
|
device.get_address().unwrap_or("".to_string()),
|
|
device.get_name().unwrap_or("".to_string()),
|
|
]);
|
|
}
|
|
|
|
let (ipc_sender, ipc_receiver) = ipc::channel().expect("Failed to create IPC channel!");
|
|
self.embedder_proxy
|
|
.send(EmbedderMsg::GetSelectedBluetoothDevice(
|
|
webview_id,
|
|
dialog_rows,
|
|
ipc_sender,
|
|
));
|
|
|
|
match ipc_receiver.recv() {
|
|
Ok(result) => result,
|
|
Err(e) => {
|
|
warn!("Failed to receive files from embedder ({:?}).", e);
|
|
None
|
|
},
|
|
}
|
|
}
|
|
|
|
fn generate_device_id(&mut self) -> String {
|
|
let mut device_id;
|
|
let mut rng = servo_rand::thread_rng();
|
|
loop {
|
|
device_id = rng.gen::<u32>().to_string();
|
|
if !self.cached_devices.contains_key(&device_id) {
|
|
break;
|
|
}
|
|
}
|
|
device_id
|
|
}
|
|
|
|
fn device_from_service_id(&self, service_id: &str) -> Option<BluetoothDevice> {
|
|
let device_id = self.service_to_device.get(service_id)?;
|
|
self.cached_devices.get(device_id).cloned()
|
|
}
|
|
|
|
fn device_is_cached(&self, device_id: &str) -> bool {
|
|
self.cached_devices.contains_key(device_id) &&
|
|
self.address_to_id.values().any(|v| v == device_id)
|
|
}
|
|
|
|
fn device_matches_filter(
|
|
&mut self,
|
|
device_id: &str,
|
|
filters: &BluetoothScanfilterSequence,
|
|
) -> BluetoothResult<bool> {
|
|
let mut adapter = self.get_adapter()?;
|
|
match self.get_device(&mut adapter, device_id) {
|
|
Some(device) => Ok(matches_filters(device, filters)),
|
|
None => Ok(false),
|
|
}
|
|
}
|
|
|
|
// Service
|
|
|
|
fn get_and_cache_gatt_services(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
device_id: &str,
|
|
) -> Vec<BluetoothGATTService> {
|
|
let mut services = match self.get_device(adapter, device_id) {
|
|
Some(d) => d.get_gatt_services().unwrap_or_default(),
|
|
None => vec![],
|
|
};
|
|
|
|
services.retain(|s| {
|
|
!uuid_is_blocklisted(&s.get_uuid().unwrap_or_default(), Blocklist::All) &&
|
|
self.allowed_services
|
|
.get(device_id)
|
|
.is_some_and(|uuids| uuids.contains(&s.get_uuid().unwrap_or_default()))
|
|
});
|
|
for service in &services {
|
|
self.cached_services
|
|
.insert(service.get_id(), service.clone());
|
|
self.service_to_device
|
|
.insert(service.get_id(), device_id.to_owned());
|
|
}
|
|
services
|
|
}
|
|
|
|
fn get_gatt_service(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
service_id: &str,
|
|
) -> Option<&BluetoothGATTService> {
|
|
return_if_cached!(self.cached_services, service_id);
|
|
let device_id = self.service_to_device.get(service_id)?.clone();
|
|
self.get_and_cache_gatt_services(adapter, &device_id);
|
|
return_if_cached!(self.cached_services, service_id);
|
|
None
|
|
}
|
|
|
|
fn service_is_cached(&self, service_id: &str) -> bool {
|
|
self.cached_services.contains_key(service_id) &&
|
|
self.service_to_device.contains_key(service_id)
|
|
}
|
|
|
|
// Characteristic
|
|
|
|
fn get_and_cache_gatt_characteristics(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
service_id: &str,
|
|
) -> Vec<BluetoothGATTCharacteristic> {
|
|
let mut characteristics = match self.get_gatt_service(adapter, service_id) {
|
|
Some(s) => s.get_gatt_characteristics().unwrap_or_default(),
|
|
None => vec![],
|
|
};
|
|
|
|
characteristics
|
|
.retain(|c| !uuid_is_blocklisted(&c.get_uuid().unwrap_or_default(), Blocklist::All));
|
|
for characteristic in &characteristics {
|
|
self.cached_characteristics
|
|
.insert(characteristic.get_id(), characteristic.clone());
|
|
self.characteristic_to_service
|
|
.insert(characteristic.get_id(), service_id.to_owned());
|
|
}
|
|
characteristics
|
|
}
|
|
|
|
fn get_gatt_characteristic(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
characteristic_id: &str,
|
|
) -> Option<&BluetoothGATTCharacteristic> {
|
|
return_if_cached!(self.cached_characteristics, characteristic_id);
|
|
let service_id = self
|
|
.characteristic_to_service
|
|
.get(characteristic_id)?
|
|
.clone();
|
|
self.get_and_cache_gatt_characteristics(adapter, &service_id);
|
|
return_if_cached!(self.cached_characteristics, characteristic_id);
|
|
None
|
|
}
|
|
|
|
fn get_characteristic_properties(&self, characteristic: &BluetoothGATTCharacteristic) -> Flags {
|
|
let mut props: Flags = Flags::empty();
|
|
let flags = characteristic.get_flags().unwrap_or_default();
|
|
for flag in flags {
|
|
match flag.as_ref() {
|
|
"broadcast" => props.insert(Flags::BROADCAST),
|
|
"read" => props.insert(Flags::READ),
|
|
"write-without-response" => props.insert(Flags::WRITE_WITHOUT_RESPONSE),
|
|
"write" => props.insert(Flags::WRITE),
|
|
"notify" => props.insert(Flags::NOTIFY),
|
|
"indicate" => props.insert(Flags::INDICATE),
|
|
"authenticated-signed-writes" => props.insert(Flags::AUTHENTICATED_SIGNED_WRITES),
|
|
"reliable-write" => props.insert(Flags::RELIABLE_WRITE),
|
|
"writable-auxiliaries" => props.insert(Flags::WRITABLE_AUXILIARIES),
|
|
_ => (),
|
|
}
|
|
}
|
|
props
|
|
}
|
|
|
|
fn characteristic_is_cached(&self, characteristic_id: &str) -> bool {
|
|
self.cached_characteristics.contains_key(characteristic_id) &&
|
|
self.characteristic_to_service
|
|
.contains_key(characteristic_id)
|
|
}
|
|
|
|
// Descriptor
|
|
|
|
fn get_and_cache_gatt_descriptors(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
characteristic_id: &str,
|
|
) -> Vec<BluetoothGATTDescriptor> {
|
|
let mut descriptors = match self.get_gatt_characteristic(adapter, characteristic_id) {
|
|
Some(c) => c.get_gatt_descriptors().unwrap_or_default(),
|
|
None => vec![],
|
|
};
|
|
|
|
descriptors
|
|
.retain(|d| !uuid_is_blocklisted(&d.get_uuid().unwrap_or_default(), Blocklist::All));
|
|
for descriptor in &descriptors {
|
|
self.cached_descriptors
|
|
.insert(descriptor.get_id(), descriptor.clone());
|
|
self.descriptor_to_characteristic
|
|
.insert(descriptor.get_id(), characteristic_id.to_owned());
|
|
}
|
|
descriptors
|
|
}
|
|
|
|
fn get_gatt_descriptor(
|
|
&mut self,
|
|
adapter: &mut BluetoothAdapter,
|
|
descriptor_id: &str,
|
|
) -> Option<&BluetoothGATTDescriptor> {
|
|
return_if_cached!(self.cached_descriptors, descriptor_id);
|
|
let characteristic_id = self
|
|
.descriptor_to_characteristic
|
|
.get(descriptor_id)?
|
|
.clone();
|
|
self.get_and_cache_gatt_descriptors(adapter, &characteristic_id);
|
|
return_if_cached!(self.cached_descriptors, descriptor_id);
|
|
None
|
|
}
|
|
|
|
// Methods
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#request-bluetooth-devices
|
|
fn request_device(&mut self, options: RequestDeviceoptions) -> BluetoothResponseResult {
|
|
// Step 6.
|
|
let mut adapter = self.get_adapter()?;
|
|
|
|
// Step 7.
|
|
// Note: There are no requiredServiceUUIDS, we scan for all devices.
|
|
if let Ok(ref session) = adapter.create_discovery_session() {
|
|
if session.start_discovery().is_ok() && !is_mock_adapter(&adapter) {
|
|
thread::sleep(Duration::from_millis(DISCOVERY_TIMEOUT_MS));
|
|
}
|
|
|
|
let _ = session.stop_discovery();
|
|
}
|
|
|
|
let mut matched_devices = self.get_and_cache_devices(&mut adapter);
|
|
|
|
// Step 8.
|
|
if !options.is_accepting_all_devices() {
|
|
matched_devices.retain(|d| matches_filters(d, options.get_filters()));
|
|
}
|
|
|
|
// Step 9.
|
|
if let Some(address) = self.select_device(options.webview_id(), matched_devices, &adapter) {
|
|
let device_id = match self.address_to_id.get(&address) {
|
|
Some(id) => id.clone(),
|
|
None => return Err(BluetoothError::NotFound),
|
|
};
|
|
let mut services = options.get_services_set();
|
|
if let Some(services_set) = self.allowed_services.get(&device_id) {
|
|
services = services_set | &services;
|
|
}
|
|
self.allowed_services.insert(device_id.clone(), services);
|
|
if let Some(device) = self.get_device(&mut adapter, &device_id) {
|
|
let message = BluetoothDeviceMsg {
|
|
id: device_id,
|
|
name: device.get_name().ok(),
|
|
};
|
|
return Ok(BluetoothResponse::RequestDevice(message));
|
|
}
|
|
}
|
|
// Step 10.
|
|
Err(BluetoothError::NotFound)
|
|
// Step 12: Missing, because it is optional.
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattserver-connect
|
|
fn gatt_server_connect(&mut self, device_id: String) -> BluetoothResponseResult {
|
|
// Step 2.
|
|
if !self.device_is_cached(&device_id) {
|
|
return Err(BluetoothError::Network);
|
|
}
|
|
let mut adapter = self.get_adapter()?;
|
|
|
|
// Step 5.1.1.
|
|
match self.get_device(&mut adapter, &device_id) {
|
|
Some(d) => {
|
|
if d.is_connected().unwrap_or(false) {
|
|
return Ok(BluetoothResponse::GATTServerConnect(true));
|
|
}
|
|
let _ = d.connect();
|
|
for _ in 0..MAXIMUM_TRANSACTION_TIME {
|
|
if d.is_connected().unwrap_or(false) {
|
|
return Ok(BluetoothResponse::GATTServerConnect(true));
|
|
} else {
|
|
if is_mock_adapter(&adapter) {
|
|
break;
|
|
}
|
|
thread::sleep(Duration::from_millis(CONNECTION_TIMEOUT_MS));
|
|
}
|
|
// TODO: Step 5.1.4: Use the exchange MTU procedure.
|
|
}
|
|
// Step 5.1.3.
|
|
Err(BluetoothError::Network)
|
|
},
|
|
None => Err(BluetoothError::NotFound),
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattserver-disconnect
|
|
fn gatt_server_disconnect(&mut self, device_id: String) -> BluetoothResult<()> {
|
|
let mut adapter = self.get_adapter()?;
|
|
match self.get_device(&mut adapter, &device_id) {
|
|
Some(d) => {
|
|
// Step 2.
|
|
if !d.is_connected().unwrap_or(true) {
|
|
return Ok(());
|
|
}
|
|
let _ = d.disconnect();
|
|
for _ in 0..MAXIMUM_TRANSACTION_TIME {
|
|
if d.is_connected().unwrap_or(true) {
|
|
thread::sleep(Duration::from_millis(CONNECTION_TIMEOUT_MS))
|
|
} else {
|
|
return Ok(());
|
|
}
|
|
}
|
|
Err(BluetoothError::Network)
|
|
},
|
|
None => Err(BluetoothError::NotFound),
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#getgattchildren
|
|
fn get_gatt_children(
|
|
&mut self,
|
|
id: String,
|
|
uuid: Option<String>,
|
|
single: bool,
|
|
child_type: GATTType,
|
|
) -> BluetoothResponseResult {
|
|
let mut adapter = self.get_adapter()?;
|
|
match child_type {
|
|
GATTType::PrimaryService => {
|
|
// Step 5.
|
|
if !self.device_is_cached(&id) {
|
|
return Err(BluetoothError::InvalidState);
|
|
}
|
|
// Step 6.
|
|
if let Some(ref uuid) = uuid {
|
|
if !self
|
|
.allowed_services
|
|
.get(&id)
|
|
.is_some_and(|s| s.contains(uuid))
|
|
{
|
|
return Err(BluetoothError::Security);
|
|
}
|
|
}
|
|
let mut services = self.get_and_cache_gatt_services(&mut adapter, &id);
|
|
if let Some(uuid) = uuid {
|
|
services.retain(|e| e.get_uuid().unwrap_or_default() == uuid);
|
|
}
|
|
let mut services_vec = vec![];
|
|
for service in services {
|
|
if service.is_primary().unwrap_or(false) {
|
|
if let Ok(uuid) = service.get_uuid() {
|
|
services_vec.push(BluetoothServiceMsg {
|
|
uuid,
|
|
is_primary: true,
|
|
instance_id: service.get_id(),
|
|
});
|
|
}
|
|
}
|
|
}
|
|
// Step 7.
|
|
if services_vec.is_empty() {
|
|
return Err(BluetoothError::NotFound);
|
|
}
|
|
|
|
Ok(BluetoothResponse::GetPrimaryServices(services_vec, single))
|
|
},
|
|
GATTType::Characteristic => {
|
|
// Step 5.
|
|
if !self.service_is_cached(&id) {
|
|
return Err(BluetoothError::InvalidState);
|
|
}
|
|
// Step 6.
|
|
let mut characteristics =
|
|
self.get_and_cache_gatt_characteristics(&mut adapter, &id);
|
|
if let Some(uuid) = uuid {
|
|
characteristics.retain(|e| e.get_uuid().unwrap_or_default() == uuid);
|
|
}
|
|
let mut characteristics_vec = vec![];
|
|
for characteristic in characteristics {
|
|
if let Ok(uuid) = characteristic.get_uuid() {
|
|
let properties = self.get_characteristic_properties(&characteristic);
|
|
characteristics_vec.push(BluetoothCharacteristicMsg {
|
|
uuid,
|
|
instance_id: characteristic.get_id(),
|
|
broadcast: properties.contains(Flags::BROADCAST),
|
|
read: properties.contains(Flags::READ),
|
|
write_without_response: properties
|
|
.contains(Flags::WRITE_WITHOUT_RESPONSE),
|
|
write: properties.contains(Flags::WRITE),
|
|
notify: properties.contains(Flags::NOTIFY),
|
|
indicate: properties.contains(Flags::INDICATE),
|
|
authenticated_signed_writes: properties
|
|
.contains(Flags::AUTHENTICATED_SIGNED_WRITES),
|
|
reliable_write: properties.contains(Flags::RELIABLE_WRITE),
|
|
writable_auxiliaries: properties.contains(Flags::WRITABLE_AUXILIARIES),
|
|
});
|
|
}
|
|
}
|
|
|
|
// Step 7.
|
|
if characteristics_vec.is_empty() {
|
|
return Err(BluetoothError::NotFound);
|
|
}
|
|
|
|
Ok(BluetoothResponse::GetCharacteristics(
|
|
characteristics_vec,
|
|
single,
|
|
))
|
|
},
|
|
GATTType::IncludedService => {
|
|
// Step 5.
|
|
if !self.service_is_cached(&id) {
|
|
return Err(BluetoothError::InvalidState);
|
|
}
|
|
// Step 6.
|
|
let device = match self.device_from_service_id(&id) {
|
|
Some(device) => device,
|
|
None => return Err(BluetoothError::NotFound),
|
|
};
|
|
let primary_service = match self.get_gatt_service(&mut adapter, &id) {
|
|
Some(s) => s,
|
|
None => return Err(BluetoothError::NotFound),
|
|
};
|
|
let services = primary_service.get_includes(device).unwrap_or(vec![]);
|
|
let mut services_vec = vec![];
|
|
for service in services {
|
|
if let Ok(service_uuid) = service.get_uuid() {
|
|
services_vec.push(BluetoothServiceMsg {
|
|
uuid: service_uuid,
|
|
is_primary: service.is_primary().unwrap_or(false),
|
|
instance_id: service.get_id(),
|
|
});
|
|
}
|
|
}
|
|
if let Some(uuid) = uuid {
|
|
services_vec.retain(|s| s.uuid == uuid);
|
|
}
|
|
services_vec.retain(|s| !uuid_is_blocklisted(&s.uuid, Blocklist::All));
|
|
|
|
// Step 7.
|
|
if services_vec.is_empty() {
|
|
return Err(BluetoothError::NotFound);
|
|
}
|
|
|
|
Ok(BluetoothResponse::GetIncludedServices(services_vec, single))
|
|
},
|
|
GATTType::Descriptor => {
|
|
// Step 5.
|
|
if !self.characteristic_is_cached(&id) {
|
|
return Err(BluetoothError::InvalidState);
|
|
}
|
|
// Step 6.
|
|
let mut descriptors = self.get_and_cache_gatt_descriptors(&mut adapter, &id);
|
|
if let Some(uuid) = uuid {
|
|
descriptors.retain(|e| e.get_uuid().unwrap_or_default() == uuid);
|
|
}
|
|
let mut descriptors_vec = vec![];
|
|
for descriptor in descriptors {
|
|
if let Ok(uuid) = descriptor.get_uuid() {
|
|
descriptors_vec.push(BluetoothDescriptorMsg {
|
|
uuid,
|
|
instance_id: descriptor.get_id(),
|
|
});
|
|
}
|
|
}
|
|
|
|
// Step 7.
|
|
if descriptors_vec.is_empty() {
|
|
return Err(BluetoothError::NotFound);
|
|
}
|
|
Ok(BluetoothResponse::GetDescriptors(descriptors_vec, single))
|
|
},
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-readvalue
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattdescriptor-readvalue
|
|
fn read_value(&mut self, id: String) -> BluetoothResponseResult {
|
|
// (Characteristic) Step 5.2: Missing because it is optional.
|
|
// (Descriptor) Step 5.1: Missing because it is optional.
|
|
let mut adapter = self.get_adapter()?;
|
|
|
|
// (Characteristic) Step 5.3.
|
|
let mut value = self
|
|
.get_gatt_characteristic(&mut adapter, &id)
|
|
.map(|c| c.read_value().unwrap_or_default());
|
|
|
|
// (Characteristic) TODO: Step 5.4: Handle all the errors returned from the read_value call.
|
|
|
|
// (Descriptor) Step 5.2.
|
|
if value.is_none() {
|
|
value = self
|
|
.get_gatt_descriptor(&mut adapter, &id)
|
|
.map(|d| d.read_value().unwrap_or_default());
|
|
}
|
|
|
|
// (Descriptor) TODO: Step 5.3: Handle all the errors returned from the read_value call.
|
|
|
|
match value {
|
|
// (Characteristic) Step 5.5.4.
|
|
// (Descriptor) Step 5.4.3.
|
|
Some(v) => Ok(BluetoothResponse::ReadValue(v)),
|
|
|
|
// (Characteristic) Step 4.
|
|
// (Descriptor) Step 4.
|
|
None => Err(BluetoothError::InvalidState),
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-writevalue
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattdescriptor-writevalue
|
|
fn write_value(&mut self, id: String, value: Vec<u8>) -> BluetoothResponseResult {
|
|
// (Characteristic) Step 7.2: Missing because it is optional.
|
|
// (Descriptor) Step 7.1: Missing because it is optional.
|
|
let mut adapter = self.get_adapter()?;
|
|
|
|
// (Characteristic) Step 7.3.
|
|
let mut result = self
|
|
.get_gatt_characteristic(&mut adapter, &id)
|
|
.map(|c| c.write_value(value.clone()));
|
|
|
|
// (Characteristic) TODO: Step 7.4: Handle all the errors returned from the write_value call.
|
|
|
|
// (Descriptor) Step 7.2.
|
|
if result.is_none() {
|
|
result = self
|
|
.get_gatt_descriptor(&mut adapter, &id)
|
|
.map(|d| d.write_value(value.clone()));
|
|
}
|
|
|
|
// (Descriptor) TODO: Step 7.3: Handle all the errors returned from the write_value call.
|
|
|
|
match result {
|
|
Some(v) => match v {
|
|
// (Characteristic) Step 7.5.3.
|
|
// (Descriptor) Step 7.4.3.
|
|
Ok(_) => Ok(BluetoothResponse::WriteValue(value)),
|
|
|
|
// (Characteristic) Step 7.1.
|
|
Err(_) => Err(BluetoothError::NotSupported),
|
|
},
|
|
|
|
// (Characteristic) Step 6.
|
|
// (Descriptor) Step 6.
|
|
None => Err(BluetoothError::InvalidState),
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-startnotifications
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothremotegattcharacteristic-stopnotifications
|
|
fn enable_notification(&mut self, id: String, enable: bool) -> BluetoothResponseResult {
|
|
// (StartNotifications) Step 3 - 4.
|
|
// (StopNotifications) Step 1 - 2.
|
|
if !self.characteristic_is_cached(&id) {
|
|
return Err(BluetoothError::InvalidState);
|
|
}
|
|
|
|
// (StartNotification) TODO: Step 7: Missing because it is optional.
|
|
let mut adapter = self.get_adapter()?;
|
|
match self.get_gatt_characteristic(&mut adapter, &id) {
|
|
Some(c) => {
|
|
let result = if enable {
|
|
// (StartNotification) Step 8.
|
|
// TODO: Handle all the errors returned from the start_notify call.
|
|
c.start_notify()
|
|
} else {
|
|
// (StopNotification) Step 4.
|
|
c.stop_notify()
|
|
};
|
|
match result {
|
|
// (StartNotification) Step 11.
|
|
// (StopNotification) Step 5.
|
|
Ok(_) => Ok(BluetoothResponse::EnableNotification(())),
|
|
|
|
// (StartNotification) Step 5.
|
|
Err(_) => Err(BluetoothError::NotSupported),
|
|
}
|
|
},
|
|
// (StartNotification) Step 4.
|
|
None => Err(BluetoothError::InvalidState),
|
|
}
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetoothdevice-watchadvertisements
|
|
fn watch_advertisements(&mut self, _device_id: String) -> BluetoothResponseResult {
|
|
// Step 2.
|
|
// TODO: Implement this when supported in lower level
|
|
Err(BluetoothError::NotSupported)
|
|
}
|
|
|
|
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-getavailability
|
|
fn get_availability(&mut self) -> BluetoothResponseResult {
|
|
Ok(BluetoothResponse::GetAvailability(
|
|
self.get_adapter().is_ok(),
|
|
))
|
|
}
|
|
}
|