/* 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/. */ //! A shareable mutable container for the DOM. use std::cell::{BorrowError, BorrowMutError}; #[cfg(not(feature = "refcell_backtrace"))] pub use std::cell::{Ref, RefCell, RefMut}; #[cfg(feature = "refcell_backtrace")] pub use accountable_refcell::{ref_filter_map, Ref, RefCell, RefMut}; #[cfg(not(feature = "refcell_backtrace"))] pub use ref_filter_map::ref_filter_map; use crate::dom::bindings::root::{assert_in_layout, assert_in_script}; /// A mutable field in the DOM. /// /// This extends the API of `std::cell::RefCell` to allow unsafe access in /// certain situations, with dynamic checking in debug builds. #[derive(Clone, Debug, Default, MallocSizeOf, PartialEq)] pub struct DomRefCell { value: RefCell, } // Functionality specific to Servo's `DomRefCell` type // =================================================== impl DomRefCell { /// Return a reference to the contents. /// /// For use in the layout thread only. #[allow(unsafe_code)] pub unsafe fn borrow_for_layout(&self) -> &T { assert_in_layout(); self.value .try_borrow_unguarded() .expect("cell is mutably borrowed") } /// Borrow the contents for the purpose of script deallocation. /// #[allow(unsafe_code)] pub unsafe fn borrow_for_script_deallocation(&self) -> &mut T { assert_in_script(); &mut *self.value.as_ptr() } /// Mutably borrow a cell for layout. Ideally this would use /// `RefCell::try_borrow_mut_unguarded` but that doesn't exist yet. #[allow(unsafe_code)] pub unsafe fn borrow_mut_for_layout(&self) -> &mut T { assert_in_layout(); &mut *self.value.as_ptr() } } // Functionality duplicated with `std::cell::RefCell` // =================================================== impl DomRefCell { /// Create a new `DomRefCell` containing `value`. pub fn new(value: T) -> DomRefCell { DomRefCell { value: RefCell::new(value), } } /// Immutably borrows the wrapped value. /// /// The borrow lasts until the returned `Ref` exits scope. Multiple /// immutable borrows can be taken out at the same time. /// /// # Panics /// /// Panics if this is called off the script thread. /// /// Panics if the value is currently mutably borrowed. pub fn borrow(&self) -> Ref { self.value.borrow() } /// Mutably borrows the wrapped value. /// /// The borrow lasts until the returned `RefMut` exits scope. The value /// cannot be borrowed while this borrow is active. /// /// # Panics /// /// Panics if this is called off the script thread. /// /// Panics if the value is currently borrowed. pub fn borrow_mut(&self) -> RefMut { self.value.borrow_mut() } /// Attempts to immutably borrow the wrapped value. /// /// The borrow lasts until the returned `Ref` exits scope. Multiple /// immutable borrows can be taken out at the same time. /// /// Returns `None` if the value is currently mutably borrowed. /// /// # Panics /// /// Panics if this is called off the script thread. pub fn try_borrow(&self) -> Result, BorrowError> { assert_in_script(); self.value.try_borrow() } /// Mutably borrows the wrapped value. /// /// The borrow lasts until the returned `RefMut` exits scope. The value /// cannot be borrowed while this borrow is active. /// /// Returns `None` if the value is currently borrowed. /// /// # Panics /// /// Panics if this is called off the script thread. pub fn try_borrow_mut(&self) -> Result, BorrowMutError> { assert_in_script(); self.value.try_borrow_mut() } }