aboutsummaryrefslogtreecommitdiffstats
path: root/components/script/dom/bluetooth.rs
diff options
context:
space:
mode:
Diffstat (limited to 'components/script/dom/bluetooth.rs')
-rw-r--r--components/script/dom/bluetooth.rs498
1 files changed, 293 insertions, 205 deletions
diff --git a/components/script/dom/bluetooth.rs b/components/script/dom/bluetooth.rs
index 3c3b88a27ac..2bdc636dc7a 100644
--- a/components/script/dom/bluetooth.rs
+++ b/components/script/dom/bluetooth.rs
@@ -1,78 +1,86 @@
/* 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 http://mozilla.org/MPL/2.0/. */
+ * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use bluetooth_traits::{BluetoothError, BluetoothRequest, GATTType};
use bluetooth_traits::{BluetoothResponse, BluetoothResponseResult};
use bluetooth_traits::blocklist::{Blocklist, uuid_is_blocklisted};
use bluetooth_traits::scanfilter::{BluetoothScanfilter, BluetoothScanfilterSequence};
use bluetooth_traits::scanfilter::{RequestDeviceoptions, ServiceUUIDSequence};
-use core::clone::Clone;
-use dom::bindings::cell::DOMRefCell;
-use dom::bindings::codegen::Bindings::BluetoothBinding::{self, BluetoothDataFilterInit, BluetoothLEScanFilterInit};
-use dom::bindings::codegen::Bindings::BluetoothBinding::{BluetoothMethods, RequestDeviceOptions};
-use dom::bindings::codegen::Bindings::BluetoothPermissionResultBinding::BluetoothPermissionDescriptor;
-use dom::bindings::codegen::Bindings::BluetoothRemoteGATTServerBinding::BluetoothRemoteGATTServerBinding::
- BluetoothRemoteGATTServerMethods;
-use dom::bindings::codegen::Bindings::EventHandlerBinding::EventHandlerNonNull;
-use dom::bindings::codegen::Bindings::PermissionStatusBinding::{PermissionName, PermissionState};
-use dom::bindings::codegen::UnionTypes::StringOrUnsignedLong;
-use dom::bindings::error::Error::{self, Network, Security, Type};
-use dom::bindings::error::Fallible;
-use dom::bindings::js::{JS, Root};
-use dom::bindings::refcounted::{Trusted, TrustedPromise};
-use dom::bindings::reflector::{DomObject, reflect_dom_object};
-use dom::bindings::str::DOMString;
-use dom::bluetoothdevice::BluetoothDevice;
-use dom::bluetoothpermissionresult::BluetoothPermissionResult;
-use dom::bluetoothuuid::{BluetoothServiceUUID, BluetoothUUID, UUID};
-use dom::eventtarget::EventTarget;
-use dom::globalscope::GlobalScope;
-use dom::permissions::{get_descriptor_permission_state, PermissionAlgorithm};
-use dom::promise::Promise;
+use crate::realms::{AlreadyInRealm, InRealm};
+use crate::dom::bindings::cell::{DomRefCell, Ref};
+use crate::dom::bindings::codegen::Bindings::BluetoothBinding::BluetoothDataFilterInit;
+use crate::dom::bindings::codegen::Bindings::BluetoothBinding::{BluetoothMethods, RequestDeviceOptions};
+use crate::dom::bindings::codegen::Bindings::BluetoothBinding::BluetoothLEScanFilterInit;
+use crate::dom::bindings::codegen::Bindings::BluetoothPermissionResultBinding::BluetoothPermissionDescriptor;
+use crate::dom::bindings::codegen::Bindings::BluetoothRemoteGATTServerBinding::BluetoothRemoteGATTServerBinding::
+BluetoothRemoteGATTServerMethods;
+use crate::dom::bindings::codegen::Bindings::PermissionStatusBinding::{PermissionName, PermissionState};
+use crate::dom::bindings::codegen::UnionTypes::{ArrayBufferViewOrArrayBuffer, StringOrUnsignedLong};
+use crate::dom::bindings::error::Error::{self, Network, Security, Type};
+use crate::dom::bindings::error::Fallible;
+use crate::dom::bindings::refcounted::{Trusted, TrustedPromise};
+use crate::dom::bindings::reflector::{DomObject, reflect_dom_object};
+use crate::dom::bindings::root::{Dom, DomRoot};
+use crate::dom::bindings::str::DOMString;
+use crate::dom::bluetoothdevice::BluetoothDevice;
+use crate::dom::bluetoothpermissionresult::BluetoothPermissionResult;
+use crate::dom::bluetoothuuid::{BluetoothServiceUUID, BluetoothUUID, UUID};
+use crate::dom::eventtarget::EventTarget;
+use crate::dom::globalscope::GlobalScope;
+use crate::dom::permissions::{get_descriptor_permission_state, PermissionAlgorithm};
+use crate::dom::promise::Promise;
+use crate::script_runtime::JSContext;
+use crate::task::TaskOnce;
use dom_struct::dom_struct;
use ipc_channel::ipc::{self, IpcSender};
use ipc_channel::router::ROUTER;
use js::conversions::ConversionResult;
-use js::jsapi::{JSAutoCompartment, JSContext, JSObject};
+use js::jsapi::JSObject;
use js::jsval::{ObjectValue, UndefinedValue};
-use script_thread::Runnable;
-use std::cell::Ref;
+use profile_traits::ipc as ProfiledIpc;
use std::collections::HashMap;
use std::rc::Rc;
use std::str::FromStr;
use std::sync::{Arc, Mutex};
-const KEY_CONVERSION_ERROR: &'static str = "This `manufacturerData` key can not be parsed as unsigned short:";
-const FILTER_EMPTY_ERROR: &'static str = "'filters' member, if present, must be nonempty to find any devices.";
+const KEY_CONVERSION_ERROR: &'static str =
+ "This `manufacturerData` key can not be parsed as unsigned short:";
+const FILTER_EMPTY_ERROR: &'static str =
+ "'filters' member, if present, must be nonempty to find any devices.";
const FILTER_ERROR: &'static str = "A filter must restrict the devices in some way.";
-const MANUFACTURER_DATA_ERROR: &'static str = "'manufacturerData', if present, must be non-empty to filter devices.";
-const MASK_LENGTH_ERROR: &'static str = "`mask`, if present, must have the same length as `dataPrefix`.";
+const MANUFACTURER_DATA_ERROR: &'static str =
+ "'manufacturerData', if present, must be non-empty to filter devices.";
+const MASK_LENGTH_ERROR: &'static str =
+ "`mask`, if present, must have the same length as `dataPrefix`.";
// 248 is the maximum number of UTF-8 code units in a Bluetooth Device Name.
const MAX_DEVICE_NAME_LENGTH: usize = 248;
const NAME_PREFIX_ERROR: &'static str = "'namePrefix', if present, must be nonempty.";
const NAME_TOO_LONG_ERROR: &'static str = "A device name can't be longer than 248 bytes.";
-const SERVICE_DATA_ERROR: &'static str = "'serviceData', if present, must be non-empty to filter devices.";
+const SERVICE_DATA_ERROR: &'static str =
+ "'serviceData', if present, must be non-empty to filter devices.";
const SERVICE_ERROR: &'static str = "'services', if present, must contain at least one service.";
const OPTIONS_ERROR: &'static str = "Fields of 'options' conflict with each other.
Either 'acceptAllDevices' member must be true, or 'filters' member must be set to a value.";
-const BT_DESC_CONVERSION_ERROR: &'static str = "Can't convert to an IDL value of type BluetoothPermissionDescriptor";
+const BT_DESC_CONVERSION_ERROR: &'static str =
+ "Can't convert to an IDL value of type BluetoothPermissionDescriptor";
-#[derive(HeapSizeOf, JSTraceable)]
+#[derive(JSTraceable, MallocSizeOf)]
+#[allow(non_snake_case)]
pub struct AllowedBluetoothDevice {
pub deviceId: DOMString,
pub mayUseGATT: bool,
}
-#[derive(HeapSizeOf, JSTraceable)]
+#[derive(JSTraceable, MallocSizeOf)]
pub struct BluetoothExtraPermissionData {
- allowed_devices: DOMRefCell<Vec<AllowedBluetoothDevice>>,
+ allowed_devices: DomRefCell<Vec<AllowedBluetoothDevice>>,
}
impl BluetoothExtraPermissionData {
pub fn new() -> BluetoothExtraPermissionData {
BluetoothExtraPermissionData {
- allowed_devices: DOMRefCell::new(Vec::new()),
+ allowed_devices: DomRefCell::new(Vec::new()),
}
}
@@ -85,7 +93,10 @@ impl BluetoothExtraPermissionData {
}
pub fn allowed_devices_contains_id(&self, id: DOMString) -> bool {
- self.allowed_devices.borrow().iter().any(|d| d.deviceId == id)
+ self.allowed_devices
+ .borrow()
+ .iter()
+ .any(|d| d.deviceId == id)
}
}
@@ -95,23 +106,24 @@ struct BluetoothContext<T: AsyncBluetoothListener + DomObject> {
}
pub trait AsyncBluetoothListener {
- fn handle_response(&self, result: BluetoothResponse, cx: *mut JSContext, promise: &Rc<Promise>);
+ fn handle_response(&self, result: BluetoothResponse, promise: &Rc<Promise>);
}
-impl<T: AsyncBluetoothListener + DomObject> BluetoothContext<T> {
+impl<T> BluetoothContext<T>
+where
+ T: AsyncBluetoothListener + DomObject,
+{
#[allow(unrooted_must_root)]
fn response(&mut self, response: BluetoothResponseResult) {
let promise = self.promise.take().expect("bt promise is missing").root();
- let promise_cx = promise.global().get_cx();
- // JSAutoCompartment needs to be manually made.
+ // JSAutoRealm needs to be manually made.
// Otherwise, Servo will crash.
- let _ac = JSAutoCompartment::new(promise_cx, promise.reflector().get_jsobject().get());
match response {
- Ok(response) => self.receiver.root().handle_response(response, promise_cx, &promise),
+ Ok(response) => self.receiver.root().handle_response(response, &promise),
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-requestdevice
// Step 3 - 4.
- Err(error) => promise.reject_error(promise_cx, Error::from(error)),
+ Err(error) => promise.reject_error(Error::from(error)),
}
}
}
@@ -120,46 +132,46 @@ impl<T: AsyncBluetoothListener + DomObject> BluetoothContext<T> {
#[dom_struct]
pub struct Bluetooth {
eventtarget: EventTarget,
- device_instance_map: DOMRefCell<HashMap<String, JS<BluetoothDevice>>>,
+ device_instance_map: DomRefCell<HashMap<String, Dom<BluetoothDevice>>>,
}
impl Bluetooth {
pub fn new_inherited() -> Bluetooth {
Bluetooth {
eventtarget: EventTarget::new_inherited(),
- device_instance_map: DOMRefCell::new(HashMap::new()),
+ device_instance_map: DomRefCell::new(HashMap::new()),
}
}
- pub fn new(global: &GlobalScope) -> Root<Bluetooth> {
- reflect_dom_object(box Bluetooth::new_inherited(),
- global,
- BluetoothBinding::Wrap)
+ pub fn new(global: &GlobalScope) -> DomRoot<Bluetooth> {
+ reflect_dom_object(Box::new(Bluetooth::new_inherited()), global)
}
fn get_bluetooth_thread(&self) -> IpcSender<BluetoothRequest> {
self.global().as_window().bluetooth_thread()
}
- pub fn get_device_map(&self) -> &DOMRefCell<HashMap<String, JS<BluetoothDevice>>> {
+ pub fn get_device_map(&self) -> &DomRefCell<HashMap<String, Dom<BluetoothDevice>>> {
&self.device_instance_map
}
// https://webbluetoothcg.github.io/web-bluetooth/#request-bluetooth-devices
- fn request_bluetooth_devices(&self,
- p: &Rc<Promise>,
- filters: &Option<Vec<BluetoothLEScanFilterInit>>,
- optional_services: &Option<Vec<BluetoothServiceUUID>>,
- sender: IpcSender<BluetoothResponseResult>) {
+ fn request_bluetooth_devices(
+ &self,
+ p: &Rc<Promise>,
+ filters: &Option<Vec<BluetoothLEScanFilterInit>>,
+ optional_services: &[BluetoothServiceUUID],
+ sender: IpcSender<BluetoothResponseResult>,
+ ) {
// TODO: Step 1: Triggered by user activation.
// Step 2.2: There are no requiredServiceUUIDS, we scan for all devices.
- let mut uuid_filters = vec!();
+ let mut uuid_filters = vec![];
if let &Some(ref filters) = filters {
// Step 2.1.
- if filters.is_empty() {
- p.reject_error(p.global().get_cx(), Type(FILTER_EMPTY_ERROR.to_owned()));
+ if filters.is_empty() {
+ p.reject_error(Type(FILTER_EMPTY_ERROR.to_owned()));
return;
}
@@ -172,7 +184,7 @@ impl Bluetooth {
// Step 2.4.2.
Ok(f) => uuid_filters.push(f),
Err(e) => {
- p.reject_error(p.global().get_cx(), e);
+ p.reject_error(e);
return;
},
}
@@ -180,105 +192,116 @@ impl Bluetooth {
}
}
- let mut optional_services_uuids = vec!();
- if let &Some(ref opt_services) = optional_services {
- for opt_service in opt_services {
- // Step 2.5 - 2.6.
- let uuid = match BluetoothUUID::service(opt_service.clone()) {
- Ok(u) => u.to_string(),
- Err(e) => {
- p.reject_error(p.global().get_cx(), e);
- return;
- },
- };
-
- // Step 2.7.
- // Note: What we are doing here, is adding the not blocklisted UUIDs to the result vector,
- // instead of removing them from an already filled vector.
- if !uuid_is_blocklisted(uuid.as_ref(), Blocklist::All) {
- optional_services_uuids.push(uuid);
- }
+ let mut optional_services_uuids = vec![];
+ for opt_service in optional_services {
+ // Step 2.5 - 2.6.
+ let uuid = match BluetoothUUID::service(opt_service.clone()) {
+ Ok(u) => u.to_string(),
+ Err(e) => {
+ p.reject_error(e);
+ return;
+ },
+ };
+
+ // Step 2.7.
+ // Note: What we are doing here, is adding the not blocklisted UUIDs to the result vector,
+ // instead of removing them from an already filled vector.
+ if !uuid_is_blocklisted(uuid.as_ref(), Blocklist::All) {
+ optional_services_uuids.push(uuid);
}
}
- let option = RequestDeviceoptions::new(BluetoothScanfilterSequence::new(uuid_filters),
- ServiceUUIDSequence::new(optional_services_uuids));
+ let option = RequestDeviceoptions::new(
+ BluetoothScanfilterSequence::new(uuid_filters),
+ ServiceUUIDSequence::new(optional_services_uuids),
+ );
// Step 4 - 5.
- if let PermissionState::Denied = get_descriptor_permission_state(PermissionName::Bluetooth, None) {
- return p.reject_error(p.global().get_cx(), Error::NotFound);
+ if let PermissionState::Denied =
+ get_descriptor_permission_state(PermissionName::Bluetooth, None)
+ {
+ return p.reject_error(Error::NotFound);
}
// Note: Step 3, 6 - 8 are implemented in
// components/net/bluetooth_thread.rs in request_device function.
- self.get_bluetooth_thread().send(BluetoothRequest::RequestDevice(option, sender)).unwrap();
+ self.get_bluetooth_thread()
+ .send(BluetoothRequest::RequestDevice(option, sender))
+ .unwrap();
}
}
pub fn response_async<T: AsyncBluetoothListener + DomObject + 'static>(
- promise: &Rc<Promise>,
- receiver: &T) -> IpcSender<BluetoothResponseResult> {
+ promise: &Rc<Promise>,
+ receiver: &T,
+) -> IpcSender<BluetoothResponseResult> {
let (action_sender, action_receiver) = ipc::channel().unwrap();
let task_source = receiver.global().networking_task_source();
let context = Arc::new(Mutex::new(BluetoothContext {
promise: Some(TrustedPromise::new(promise.clone())),
receiver: Trusted::new(receiver),
}));
- ROUTER.add_route(action_receiver.to_opaque(), box move |message| {
- struct ListenerRunnable<T: AsyncBluetoothListener + DomObject> {
- context: Arc<Mutex<BluetoothContext<T>>>,
- action: BluetoothResponseResult,
- }
+ ROUTER.add_route(
+ action_receiver.to_opaque(),
+ Box::new(move |message| {
+ struct ListenerTask<T: AsyncBluetoothListener + DomObject> {
+ context: Arc<Mutex<BluetoothContext<T>>>,
+ action: BluetoothResponseResult,
+ }
- impl<T: AsyncBluetoothListener + DomObject> Runnable for ListenerRunnable<T> {
- fn handler(self: Box<Self>) {
- let this = *self;
- let mut context = this.context.lock().unwrap();
- context.response(this.action);
+ impl<T> TaskOnce for ListenerTask<T>
+ where
+ T: AsyncBluetoothListener + DomObject,
+ {
+ fn run_once(self) {
+ let mut context = self.context.lock().unwrap();
+ context.response(self.action);
+ }
}
- }
- let runnable = box ListenerRunnable {
- context: context.clone(),
- action: message.to().unwrap(),
- };
+ let task = ListenerTask {
+ context: context.clone(),
+ action: message.to().unwrap(),
+ };
- let result = task_source.queue_wrapperless(runnable);
- if let Err(err) = result {
- warn!("failed to deliver network data: {:?}", err);
- }
- });
+ let result = task_source.queue_unconditionally(task);
+ if let Err(err) = result {
+ warn!("failed to deliver network data: {:?}", err);
+ }
+ }),
+ );
action_sender
}
-#[allow(unrooted_must_root)]
// https://webbluetoothcg.github.io/web-bluetooth/#getgattchildren
-pub fn get_gatt_children<T, F> (
- attribute: &T,
- single: bool,
- uuid_canonicalizer: F,
- uuid: Option<StringOrUnsignedLong>,
- instance_id: String,
- connected: bool,
- child_type: GATTType)
- -> Rc<Promise>
- where T: AsyncBluetoothListener + DomObject + 'static,
- F: FnOnce(StringOrUnsignedLong) -> Fallible<UUID> {
- let p = Promise::new(&attribute.global());
- let p_cx = p.global().get_cx();
+pub fn get_gatt_children<T, F>(
+ attribute: &T,
+ single: bool,
+ uuid_canonicalizer: F,
+ uuid: Option<StringOrUnsignedLong>,
+ instance_id: String,
+ connected: bool,
+ child_type: GATTType,
+) -> Rc<Promise>
+where
+ T: AsyncBluetoothListener + DomObject + 'static,
+ F: FnOnce(StringOrUnsignedLong) -> Fallible<UUID>,
+{
+ let in_realm_proof = AlreadyInRealm::assert(&attribute.global());
+ let p = Promise::new_in_current_realm(&attribute.global(), InRealm::Already(&in_realm_proof));
let result_uuid = if let Some(u) = uuid {
// Step 1.
let canonicalized = match uuid_canonicalizer(u) {
Ok(canonicalized_uuid) => canonicalized_uuid.to_string(),
Err(e) => {
- p.reject_error(p_cx, e);
+ p.reject_error(e);
return p;
- }
+ },
};
// Step 2.
if uuid_is_blocklisted(canonicalized.as_ref(), Blocklist::All) {
- p.reject_error(p_cx, Security);
+ p.reject_error(Security);
return p;
}
Some(canonicalized)
@@ -288,7 +311,7 @@ pub fn get_gatt_children<T, F> (
// Step 3 - 4.
if !connected {
- p.reject_error(p_cx, Network);
+ p.reject_error(Network);
return p;
}
@@ -297,8 +320,18 @@ pub fn get_gatt_children<T, F> (
// Note: Steps 6 - 7 are implemented in components/bluetooth/lib.rs in get_descriptor function
// and in handle_response function.
let sender = response_async(&p, attribute);
- attribute.global().as_window().bluetooth_thread().send(
- BluetoothRequest::GetGATTChildren(instance_id, result_uuid, single, child_type, sender)).unwrap();
+ attribute
+ .global()
+ .as_window()
+ .bluetooth_thread()
+ .send(BluetoothRequest::GetGATTChildren(
+ instance_id,
+ result_uuid,
+ single,
+ child_type,
+ sender,
+ ))
+ .unwrap();
return p;
}
@@ -306,11 +339,12 @@ pub fn get_gatt_children<T, F> (
fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<BluetoothScanfilter> {
// Step 1.
if filter.services.is_none() &&
- filter.name.is_none() &&
- filter.namePrefix.is_none() &&
- filter.manufacturerData.is_none() &&
- filter.serviceData.is_none() {
- return Err(Type(FILTER_ERROR.to_owned()));
+ filter.name.is_none() &&
+ filter.namePrefix.is_none() &&
+ filter.manufacturerData.is_none() &&
+ filter.serviceData.is_none()
+ {
+ return Err(Type(FILTER_ERROR.to_owned()));
}
// Step 2: There is no empty canonicalizedFilter member,
@@ -324,15 +358,15 @@ fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<Bluetooth
return Err(Type(SERVICE_ERROR.to_owned()));
}
- let mut services_vec = vec!();
+ let mut services_vec = vec![];
for service in services {
// Step 3.2 - 3.3.
- let uuid = try!(BluetoothUUID::service(service.clone())).to_string();
+ let uuid = BluetoothUUID::service(service.clone())?.to_string();
// Step 3.4.
if uuid_is_blocklisted(uuid.as_ref(), Blocklist::All) {
- return Err(Security)
+ return Err(Security);
}
services_vec.push(uuid);
@@ -340,7 +374,7 @@ fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<Bluetooth
// Step 3.5.
services_vec
},
- None => vec!(),
+ None => vec![],
};
// Step 4.
@@ -387,13 +421,18 @@ fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<Bluetooth
// Step 7.1 - 7.2.
let manufacturer_id = match u16::from_str(key.as_ref()) {
Ok(id) => id,
- Err(err) => return Err(Type(format!("{} {} {}", KEY_CONVERSION_ERROR, key, err))),
+ Err(err) => {
+ return Err(Type(format!("{} {} {}", KEY_CONVERSION_ERROR, key, err)));
+ },
};
// Step 7.3: No need to convert to IDL values since this is only used by native code.
// Step 7.4 - 7.5.
- map.insert(manufacturer_id, try!(canonicalize_bluetooth_data_filter_init(bdfi)));
+ map.insert(
+ manufacturer_id,
+ canonicalize_bluetooth_data_filter_init(bdfi)?,
+ );
}
Some(map)
},
@@ -413,11 +452,11 @@ fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<Bluetooth
// Step 9.1.
Ok(number) => StringOrUnsignedLong::UnsignedLong(number),
// Step 9.2.
- _ => StringOrUnsignedLong::String(key.clone())
+ _ => StringOrUnsignedLong::String(key.clone()),
};
// Step 9.3 - 9.4.
- let service = try!(BluetoothUUID::service(service_name)).to_string();
+ let service = BluetoothUUID::service(service_name)?.to_string();
// Step 9.5.
if uuid_is_blocklisted(service.as_ref(), Blocklist::All) {
@@ -427,7 +466,7 @@ fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<Bluetooth
// Step 9.6: No need to convert to IDL values since this is only used by native code.
// Step 9.7 - 9.8.
- map.insert(service, try!(canonicalize_bluetooth_data_filter_init(bdfi)));
+ map.insert(service, canonicalize_bluetooth_data_filter_init(bdfi)?);
}
Some(map)
},
@@ -435,18 +474,34 @@ fn canonicalize_filter(filter: &BluetoothLEScanFilterInit) -> Fallible<Bluetooth
};
// Step 10.
- Ok(BluetoothScanfilter::new(name, name_prefix, services_vec, manufacturer_data, service_data))
+ Ok(BluetoothScanfilter::new(
+ name,
+ name_prefix,
+ services_vec,
+ manufacturer_data,
+ service_data,
+ ))
}
// https://webbluetoothcg.github.io/web-bluetooth/#bluetoothdatafilterinit-canonicalizing
-fn canonicalize_bluetooth_data_filter_init(bdfi: &BluetoothDataFilterInit) -> Fallible<(Vec<u8>, Vec<u8>)> {
+fn canonicalize_bluetooth_data_filter_init(
+ bdfi: &BluetoothDataFilterInit,
+) -> Fallible<(Vec<u8>, Vec<u8>)> {
// Step 1.
- let data_prefix = bdfi.dataPrefix.clone().unwrap_or(vec![]);
+ let data_prefix = match bdfi.dataPrefix {
+ Some(ArrayBufferViewOrArrayBuffer::ArrayBufferView(ref avb)) => avb.to_vec(),
+ Some(ArrayBufferViewOrArrayBuffer::ArrayBuffer(ref ab)) => ab.to_vec(),
+ None => vec![],
+ };
// Step 2.
// If no mask present, mask will be a sequence of 0xFF bytes the same length as dataPrefix.
// Masking dataPrefix with this, leaves dataPrefix untouched.
- let mask = bdfi.mask.clone().unwrap_or(vec![0xFF; data_prefix.len()]);
+ let mask = match bdfi.mask {
+ Some(ArrayBufferViewOrArrayBuffer::ArrayBufferView(ref avb)) => avb.to_vec(),
+ Some(ArrayBufferViewOrArrayBuffer::ArrayBuffer(ref ab)) => ab.to_vec(),
+ None => vec![0xFF; data_prefix.len()],
+ };
// Step 3.
if mask.len() != data_prefix.len() {
@@ -471,14 +526,14 @@ impl From<BluetoothError> for Error {
}
impl BluetoothMethods for Bluetooth {
- #[allow(unrooted_must_root)]
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-requestdevice
- fn RequestDevice(&self, option: &RequestDeviceOptions) -> Rc<Promise> {
- let p = Promise::new(&self.global());
+ fn RequestDevice(&self, option: &RequestDeviceOptions, comp: InRealm) -> Rc<Promise> {
+ let p = Promise::new_in_current_realm(&self.global(), comp);
// Step 1.
if (option.filters.is_some() && option.acceptAllDevices) ||
- (option.filters.is_none() && !option.acceptAllDevices) {
- p.reject_error(p.global().get_cx(), Error::Type(OPTIONS_ERROR.to_owned()));
+ (option.filters.is_none() && !option.acceptAllDevices)
+ {
+ p.reject_error(Error::Type(OPTIONS_ERROR.to_owned()));
return p;
}
@@ -489,54 +544,61 @@ impl BluetoothMethods for Bluetooth {
return p;
}
- #[allow(unrooted_must_root)]
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-getavailability
- fn GetAvailability(&self) -> Rc<Promise> {
- let p = Promise::new(&self.global());
+ fn GetAvailability(&self, comp: InRealm) -> Rc<Promise> {
+ let p = Promise::new_in_current_realm(&self.global(), comp);
// Step 1. We did not override the method
// Step 2 - 3. in handle_response
let sender = response_async(&p, self);
- self.get_bluetooth_thread().send(
- BluetoothRequest::GetAvailability(sender)).unwrap();
+ self.get_bluetooth_thread()
+ .send(BluetoothRequest::GetAvailability(sender))
+ .unwrap();
p
}
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-onavailabilitychanged
- event_handler!(availabilitychanged, GetOnavailabilitychanged, SetOnavailabilitychanged);
+ event_handler!(
+ availabilitychanged,
+ GetOnavailabilitychanged,
+ SetOnavailabilitychanged
+ );
}
impl AsyncBluetoothListener for Bluetooth {
- fn handle_response(&self, response: BluetoothResponse, promise_cx: *mut JSContext, promise: &Rc<Promise>) {
+ fn handle_response(&self, response: BluetoothResponse, promise: &Rc<Promise>) {
match response {
// https://webbluetoothcg.github.io/web-bluetooth/#request-bluetooth-devices
// Step 11, 13 - 14.
BluetoothResponse::RequestDevice(device) => {
let mut device_instance_map = self.device_instance_map.borrow_mut();
if let Some(existing_device) = device_instance_map.get(&device.id.clone()) {
- return promise.resolve_native(promise_cx, &**existing_device);
+ return promise.resolve_native(&**existing_device);
}
- let bt_device = BluetoothDevice::new(&self.global(),
- DOMString::from(device.id.clone()),
- device.name.map(DOMString::from),
- &self);
- device_instance_map.insert(device.id.clone(), JS::from_ref(&bt_device));
-
- self.global().as_window().bluetooth_extra_permission_data().add_new_allowed_device(
- AllowedBluetoothDevice {
+ let bt_device = BluetoothDevice::new(
+ &self.global(),
+ DOMString::from(device.id.clone()),
+ device.name.map(DOMString::from),
+ &self,
+ );
+ device_instance_map.insert(device.id.clone(), Dom::from_ref(&bt_device));
+
+ self.global()
+ .as_window()
+ .bluetooth_extra_permission_data()
+ .add_new_allowed_device(AllowedBluetoothDevice {
deviceId: DOMString::from(device.id),
mayUseGATT: true,
- }
- );
+ });
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-requestdevice
// Step 5.
- promise.resolve_native(promise_cx, &bt_device);
+ promise.resolve_native(&bt_device);
},
// https://webbluetoothcg.github.io/web-bluetooth/#dom-bluetooth-getavailability
// Step 2 - 3.
BluetoothResponse::GetAvailability(is_available) => {
- promise.resolve_native(promise_cx, &is_available);
- }
- _ => promise.reject_error(promise_cx, Error::Type("Something went wrong...".to_owned())),
+ promise.resolve_native(&is_available);
+ },
+ _ => promise.reject_error(Error::Type("Something went wrong...".to_owned())),
}
}
}
@@ -545,25 +607,28 @@ impl PermissionAlgorithm for Bluetooth {
type Descriptor = BluetoothPermissionDescriptor;
type Status = BluetoothPermissionResult;
- #[allow(unsafe_code)]
- fn create_descriptor(cx: *mut JSContext,
- permission_descriptor_obj: *mut JSObject)
- -> Result<BluetoothPermissionDescriptor, Error> {
- rooted!(in(cx) let mut property = UndefinedValue());
- property.handle_mut().set(ObjectValue(permission_descriptor_obj));
- unsafe {
- match BluetoothPermissionDescriptor::new(cx, property.handle()) {
- Ok(ConversionResult::Success(descriptor)) => Ok(descriptor),
- Ok(ConversionResult::Failure(error)) => Err(Error::Type(error.into_owned())),
- Err(_) => Err(Error::Type(String::from(BT_DESC_CONVERSION_ERROR))),
- }
+ fn create_descriptor(
+ cx: JSContext,
+ permission_descriptor_obj: *mut JSObject,
+ ) -> Result<BluetoothPermissionDescriptor, Error> {
+ rooted!(in(*cx) let mut property = UndefinedValue());
+ property
+ .handle_mut()
+ .set(ObjectValue(permission_descriptor_obj));
+ match BluetoothPermissionDescriptor::new(cx, property.handle()) {
+ Ok(ConversionResult::Success(descriptor)) => Ok(descriptor),
+ Ok(ConversionResult::Failure(error)) => Err(Error::Type(error.into_owned())),
+ Err(_) => Err(Error::Type(String::from(BT_DESC_CONVERSION_ERROR))),
}
}
// https://webbluetoothcg.github.io/web-bluetooth/#query-the-bluetooth-permission
- fn permission_query(cx: *mut JSContext, promise: &Rc<Promise>,
- descriptor: &BluetoothPermissionDescriptor,
- status: &BluetoothPermissionResult) {
+ fn permission_query(
+ _cx: JSContext,
+ promise: &Rc<Promise>,
+ descriptor: &BluetoothPermissionDescriptor,
+ status: &BluetoothPermissionResult,
+ ) {
// Step 1: We are not using the `global` variable.
// Step 2.
@@ -572,7 +637,7 @@ impl PermissionAlgorithm for Bluetooth {
// Step 3.
if let PermissionState::Denied = status.get_state() {
status.set_devices(Vec::new());
- return promise.resolve_native(cx, status);
+ return promise.resolve_native(status);
}
// Step 4.
@@ -580,7 +645,10 @@ impl PermissionAlgorithm for Bluetooth {
// Step 5.
let global = status.global();
- let allowed_devices = global.as_window().bluetooth_extra_permission_data().get_allowed_devices();
+ let allowed_devices = global
+ .as_window()
+ .bluetooth_extra_permission_data()
+ .get_allowed_devices();
let bluetooth = status.get_bluetooth();
let device_map = bluetooth.get_device_map().borrow();
@@ -603,23 +671,28 @@ impl PermissionAlgorithm for Bluetooth {
for filter in filters {
match canonicalize_filter(&filter) {
Ok(f) => scan_filters.push(f),
- Err(error) => return promise.reject_error(cx, error),
+ Err(error) => return promise.reject_error(error),
}
}
// Step 6.2.2.
// Instead of creating an internal slot we send an ipc message to the Bluetooth thread
// to check if one of the filters matches.
- let (sender, receiver) = ipc::channel().unwrap();
- status.get_bluetooth_thread()
- .send(BluetoothRequest::MatchesFilter(device_id.clone(),
- BluetoothScanfilterSequence::new(scan_filters),
- sender)).unwrap();
+ let (sender, receiver) =
+ ProfiledIpc::channel(global.time_profiler_chan().clone()).unwrap();
+ status
+ .get_bluetooth_thread()
+ .send(BluetoothRequest::MatchesFilter(
+ device_id.clone(),
+ BluetoothScanfilterSequence::new(scan_filters),
+ sender,
+ ))
+ .unwrap();
match receiver.recv().unwrap() {
Ok(true) => (),
Ok(false) => continue,
- Err(error) => return promise.reject_error(cx, Error::from(error)),
+ Err(error) => return promise.reject_error(Error::from(error)),
};
}
@@ -627,7 +700,7 @@ impl PermissionAlgorithm for Bluetooth {
// TODO: Implement this correctly, not just using device ids here.
// https://webbluetoothcg.github.io/web-bluetooth/#get-the-bluetoothdevice-representing
if let Some(device) = device_map.get(&device_id) {
- matching_devices.push(JS::from_ref(&**device));
+ matching_devices.push(Dom::from_ref(&**device));
}
}
@@ -636,32 +709,46 @@ impl PermissionAlgorithm for Bluetooth {
// https://w3c.github.io/permissions/#dom-permissions-query
// Step 7.
- promise.resolve_native(cx, status);
+ promise.resolve_native(status);
}
// https://webbluetoothcg.github.io/web-bluetooth/#request-the-bluetooth-permission
- fn permission_request(cx: *mut JSContext, promise: &Rc<Promise>,
- descriptor: &BluetoothPermissionDescriptor,
- status: &BluetoothPermissionResult) {
+ fn permission_request(
+ _cx: JSContext,
+ promise: &Rc<Promise>,
+ descriptor: &BluetoothPermissionDescriptor,
+ status: &BluetoothPermissionResult,
+ ) {
// Step 1.
if descriptor.filters.is_some() == descriptor.acceptAllDevices {
- return promise.reject_error(cx, Error::Type(OPTIONS_ERROR.to_owned()));
+ return promise.reject_error(Error::Type(OPTIONS_ERROR.to_owned()));
}
// Step 2.
let sender = response_async(promise, status);
let bluetooth = status.get_bluetooth();
- bluetooth.request_bluetooth_devices(promise, &descriptor.filters, &descriptor.optionalServices, sender);
+ bluetooth.request_bluetooth_devices(
+ promise,
+ &descriptor.filters,
+ &descriptor.optionalServices,
+ sender,
+ );
// NOTE: Step 3. is in BluetoothPermissionResult's `handle_response` function.
}
#[allow(unrooted_must_root)]
// https://webbluetoothcg.github.io/web-bluetooth/#revoke-bluetooth-access
- fn permission_revoke(_descriptor: &BluetoothPermissionDescriptor, status: &BluetoothPermissionResult) {
+ fn permission_revoke(
+ _descriptor: &BluetoothPermissionDescriptor,
+ status: &BluetoothPermissionResult,
+ ) {
// Step 1.
let global = status.global();
- let allowed_devices = global.as_window().bluetooth_extra_permission_data().get_allowed_devices();
+ let allowed_devices = global
+ .as_window()
+ .bluetooth_extra_permission_data()
+ .get_allowed_devices();
// Step 2.
let bluetooth = status.get_bluetooth();
let device_map = bluetooth.get_device_map().borrow();
@@ -669,7 +756,8 @@ impl PermissionAlgorithm for Bluetooth {
let id = DOMString::from(id.clone());
// Step 2.1.
if allowed_devices.iter().any(|d| d.deviceId == id) &&
- !device.is_represented_device_null() {
+ !device.is_represented_device_null()
+ {
// Note: We don't need to update the allowed_services,
// because we store it in the lower level
// where it is already up-to-date