mirror of
https://github.com/servo/servo.git
synced 2025-06-08 16:43:28 +00:00
Servo currently uses `heapsize`, but Stylo/Gecko use `malloc_size_of`. `malloc_size_of` is better -- it handles various cases that `heapsize` does not -- so this patch changes Servo to use `malloc_size_of`. This patch makes the following changes to the `malloc_size_of` crate. - Adds `MallocSizeOf` trait implementations for numerous types, some built-in (e.g. `VecDeque`), some external and Servo-only (e.g. `string_cache`). - Makes `enclosing_size_of_op` optional, because vanilla jemalloc doesn't support that operation. - For `HashSet`/`HashMap`, falls back to a computed estimate when `enclosing_size_of_op` isn't available. - Adds an extern "C" `malloc_size_of` function that does the actual heap measurement; this is based on the same functions from the `heapsize` crate. This patch makes the following changes elsewhere. - Converts all the uses of `heapsize` to instead use `malloc_size_of`. - Disables the "heapsize"/"heap_size" feature for the external crates that provide it. - Removes the `HeapSizeOf` implementation from `hashglobe`. - Adds `ignore` annotations to a few `Rc`/`Arc`, because `malloc_size_of` doesn't derive those types, unlike `heapsize`.
164 lines
5.7 KiB
Rust
164 lines
5.7 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 http://mozilla.org/MPL/2.0/. */
|
|
|
|
use dom::bindings::cell::DomRefCell;
|
|
use dom::bindings::codegen::Bindings::EventListenerBinding::EventListener;
|
|
use dom::bindings::codegen::Bindings::EventTargetBinding::AddEventListenerOptions;
|
|
use dom::bindings::codegen::Bindings::EventTargetBinding::EventListenerOptions;
|
|
use dom::bindings::codegen::Bindings::MediaQueryListBinding::{self, MediaQueryListMethods};
|
|
use dom::bindings::inheritance::Castable;
|
|
use dom::bindings::reflector::DomObject;
|
|
use dom::bindings::reflector::reflect_dom_object;
|
|
use dom::bindings::root::{Dom, DomRoot};
|
|
use dom::bindings::str::DOMString;
|
|
use dom::bindings::trace::JSTraceable;
|
|
use dom::bindings::weakref::{WeakRef, WeakRefVec};
|
|
use dom::document::Document;
|
|
use dom::event::Event;
|
|
use dom::eventtarget::EventTarget;
|
|
use dom::mediaquerylistevent::MediaQueryListEvent;
|
|
use dom_struct::dom_struct;
|
|
use js::jsapi::JSTracer;
|
|
use std::cell::Cell;
|
|
use std::rc::Rc;
|
|
use style::media_queries::MediaList;
|
|
use style_traits::ToCss;
|
|
|
|
pub enum MediaQueryListMatchState {
|
|
Same(bool),
|
|
Changed(bool),
|
|
}
|
|
|
|
#[dom_struct]
|
|
pub struct MediaQueryList {
|
|
eventtarget: EventTarget,
|
|
document: Dom<Document>,
|
|
media_query_list: MediaList,
|
|
last_match_state: Cell<Option<bool>>
|
|
}
|
|
|
|
impl MediaQueryList {
|
|
fn new_inherited(document: &Document, media_query_list: MediaList) -> MediaQueryList {
|
|
MediaQueryList {
|
|
eventtarget: EventTarget::new_inherited(),
|
|
document: Dom::from_ref(document),
|
|
media_query_list: media_query_list,
|
|
last_match_state: Cell::new(None),
|
|
}
|
|
}
|
|
|
|
pub fn new(document: &Document, media_query_list: MediaList) -> DomRoot<MediaQueryList> {
|
|
reflect_dom_object(Box::new(MediaQueryList::new_inherited(document, media_query_list)),
|
|
document.window(),
|
|
MediaQueryListBinding::Wrap)
|
|
}
|
|
}
|
|
|
|
impl MediaQueryList {
|
|
fn evaluate_changes(&self) -> MediaQueryListMatchState {
|
|
let matches = self.evaluate();
|
|
|
|
let result = if let Some(old_matches) = self.last_match_state.get() {
|
|
if old_matches == matches {
|
|
MediaQueryListMatchState::Same(matches)
|
|
} else {
|
|
MediaQueryListMatchState::Changed(matches)
|
|
}
|
|
} else {
|
|
MediaQueryListMatchState::Changed(matches)
|
|
};
|
|
|
|
self.last_match_state.set(Some(matches));
|
|
result
|
|
}
|
|
|
|
pub fn evaluate(&self) -> bool {
|
|
self.document.device().map_or(false, |device| {
|
|
self.media_query_list.evaluate(&device, self.document.quirks_mode())
|
|
})
|
|
}
|
|
}
|
|
|
|
impl MediaQueryListMethods for MediaQueryList {
|
|
// https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-media
|
|
fn Media(&self) -> DOMString {
|
|
let mut s = String::new();
|
|
self.media_query_list.to_css(&mut s).unwrap();
|
|
DOMString::from_string(s)
|
|
}
|
|
|
|
// https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-matches
|
|
fn Matches(&self) -> bool {
|
|
match self.last_match_state.get() {
|
|
None => self.evaluate(),
|
|
Some(state) => state,
|
|
}
|
|
}
|
|
|
|
// https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-addlistener
|
|
fn AddListener(&self, listener: Option<Rc<EventListener>>) {
|
|
self.upcast::<EventTarget>().add_event_listener(
|
|
DOMString::from_string("change".to_owned()),
|
|
listener,
|
|
AddEventListenerOptions { parent: EventListenerOptions { capture: false } },
|
|
);
|
|
}
|
|
|
|
// https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-removelistener
|
|
fn RemoveListener(&self, listener: Option<Rc<EventListener>>) {
|
|
self.upcast::<EventTarget>().remove_event_listener(
|
|
DOMString::from_string("change".to_owned()),
|
|
listener,
|
|
EventListenerOptions { capture: false },
|
|
);
|
|
}
|
|
|
|
// https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-onchange
|
|
event_handler!(change, GetOnchange, SetOnchange);
|
|
}
|
|
|
|
#[derive(MallocSizeOf)]
|
|
pub struct WeakMediaQueryListVec {
|
|
cell: DomRefCell<WeakRefVec<MediaQueryList>>,
|
|
}
|
|
|
|
impl WeakMediaQueryListVec {
|
|
/// Create a new vector of weak references to MediaQueryList
|
|
pub fn new() -> Self {
|
|
WeakMediaQueryListVec { cell: DomRefCell::new(WeakRefVec::new()) }
|
|
}
|
|
|
|
pub fn push(&self, mql: &MediaQueryList) {
|
|
self.cell.borrow_mut().push(WeakRef::new(mql));
|
|
}
|
|
|
|
/// Evaluate media query lists and report changes
|
|
/// <https://drafts.csswg.org/cssom-view/#evaluate-media-queries-and-report-changes>
|
|
pub fn evaluate_and_report_changes(&self) {
|
|
rooted_vec!(let mut mql_list);
|
|
self.cell.borrow_mut().update(|mql| {
|
|
let mql = mql.root().unwrap();
|
|
if let MediaQueryListMatchState::Changed(_) = mql.evaluate_changes() {
|
|
// Recording list of changed Media Queries
|
|
mql_list.push(Dom::from_ref(&*mql));
|
|
}
|
|
});
|
|
// Sending change events for all changed Media Queries
|
|
for mql in mql_list.iter() {
|
|
let event = MediaQueryListEvent::new(&mql.global(),
|
|
atom!("change"),
|
|
false, false,
|
|
mql.Media(),
|
|
mql.Matches());
|
|
event.upcast::<Event>().fire(mql.upcast::<EventTarget>());
|
|
}
|
|
}
|
|
}
|
|
|
|
#[allow(unsafe_code)]
|
|
unsafe impl JSTraceable for WeakMediaQueryListVec {
|
|
unsafe fn trace(&self, _: *mut JSTracer) {
|
|
self.cell.borrow_mut().retain_alive()
|
|
}
|
|
}
|