performance: clear and disable buffer on pipeline exit

use existing buffer max size
This commit is contained in:
Gregory Terzian 2019-08-30 03:18:26 +08:00
parent 39bd45529d
commit 6d1136a37d
2 changed files with 34 additions and 27 deletions

View file

@ -57,6 +57,7 @@ const INVALID_ENTRY_NAMES: &'static [&'static str] = &[
/// Performance and PerformanceObserverEntryList interfaces implementations. /// Performance and PerformanceObserverEntryList interfaces implementations.
#[derive(JSTraceable, MallocSizeOf)] #[derive(JSTraceable, MallocSizeOf)]
pub struct PerformanceEntryList { pub struct PerformanceEntryList {
/// https://w3c.github.io/performance-timeline/#dfn-performance-entry-buffer
entries: DOMPerformanceEntryList, entries: DOMPerformanceEntryList,
} }
@ -137,10 +138,13 @@ struct PerformanceObserver {
#[dom_struct] #[dom_struct]
pub struct Performance { pub struct Performance {
eventtarget: EventTarget, eventtarget: EventTarget,
entries: DomRefCell<PerformanceEntryList>, buffer: DomRefCell<PerformanceEntryList>,
observers: DomRefCell<Vec<PerformanceObserver>>, observers: DomRefCell<Vec<PerformanceObserver>>,
pending_notification_observers_task: Cell<bool>, pending_notification_observers_task: Cell<bool>,
navigation_start_precise: u64, navigation_start_precise: u64,
/// https://w3c.github.io/performance-timeline/#dfn-maxbuffersize
/// The max-size of the buffer, set to 0 once the pipeline exits.
/// TODO: have one max-size per entry type.
resource_timing_buffer_size_limit: Cell<usize>, resource_timing_buffer_size_limit: Cell<usize>,
resource_timing_buffer_current_size: Cell<usize>, resource_timing_buffer_current_size: Cell<usize>,
resource_timing_buffer_pending_full_event: Cell<bool>, resource_timing_buffer_pending_full_event: Cell<bool>,
@ -151,7 +155,7 @@ impl Performance {
fn new_inherited(navigation_start_precise: u64) -> Performance { fn new_inherited(navigation_start_precise: u64) -> Performance {
Performance { Performance {
eventtarget: EventTarget::new_inherited(), eventtarget: EventTarget::new_inherited(),
entries: DomRefCell::new(PerformanceEntryList::new(Vec::new())), buffer: DomRefCell::new(PerformanceEntryList::new(Vec::new())),
observers: DomRefCell::new(Vec::new()), observers: DomRefCell::new(Vec::new()),
pending_notification_observers_task: Cell::new(false), pending_notification_observers_task: Cell::new(false),
navigation_start_precise, navigation_start_precise,
@ -170,6 +174,15 @@ impl Performance {
) )
} }
/// Clear all buffered performance entries, and disable the buffer.
/// Called as part of the window's "clear_js_runtime" workflow,
/// performed when exiting a pipeline.
pub fn clear_and_disable_performance_entry_buffer(&self) {
let mut buffer = self.buffer.borrow_mut();
buffer.entries.clear();
self.resource_timing_buffer_size_limit.set(0);
}
/// Add a PerformanceObserver to the list of observers with a set of /// Add a PerformanceObserver to the list of observers with a set of
/// observed entry types. /// observed entry types.
pub fn add_observer( pub fn add_observer(
@ -179,10 +192,10 @@ impl Performance {
buffered: bool, buffered: bool,
) { ) {
if buffered { if buffered {
let entries = self.entries.borrow(); let buffer = self.buffer.borrow();
let mut new_entries = entry_types let mut new_entries = entry_types
.iter() .iter()
.flat_map(|e| entries.get_entries_by_name_and_type(None, Some(e.clone()))) .flat_map(|e| buffer.get_entries_by_name_and_type(None, Some(e.clone())))
.collect::<DOMPerformanceEntryList>(); .collect::<DOMPerformanceEntryList>();
let mut obs_entries = observer.entries(); let mut obs_entries = observer.entries();
obs_entries.append(&mut new_entries); obs_entries.append(&mut new_entries);
@ -221,6 +234,7 @@ impl Performance {
/// Also this algorithm has been extented according to : /// Also this algorithm has been extented according to :
/// <https://w3c.github.io/resource-timing/#sec-extensions-performance-interface> /// <https://w3c.github.io/resource-timing/#sec-extensions-performance-interface>
pub fn queue_entry(&self, entry: &PerformanceEntry, add_to_performance_entries_buffer: bool) { pub fn queue_entry(&self, entry: &PerformanceEntry, add_to_performance_entries_buffer: bool) {
// https://w3c.github.io/performance-timeline/#dfn-determine-eligibility-for-adding-a-performance-entry
if entry.entry_type() == "resource" && !self.should_queue_resource_entry(entry) { if entry.entry_type() == "resource" && !self.should_queue_resource_entry(entry) {
return; return;
} }
@ -242,7 +256,7 @@ impl Performance {
// If the "add to performance entry buffer flag" is set, add the // If the "add to performance entry buffer flag" is set, add the
// new entry to the buffer. // new entry to the buffer.
if add_to_performance_entries_buffer { if add_to_performance_entries_buffer {
self.entries self.buffer
.borrow_mut() .borrow_mut()
.entries .entries
.push(DomRoot::from_ref(entry)); .push(DomRoot::from_ref(entry));
@ -390,14 +404,14 @@ impl PerformanceMethods for Performance {
// https://www.w3.org/TR/performance-timeline-2/#dom-performance-getentries // https://www.w3.org/TR/performance-timeline-2/#dom-performance-getentries
fn GetEntries(&self) -> Vec<DomRoot<PerformanceEntry>> { fn GetEntries(&self) -> Vec<DomRoot<PerformanceEntry>> {
self.entries self.buffer
.borrow() .borrow()
.get_entries_by_name_and_type(None, None) .get_entries_by_name_and_type(None, None)
} }
// https://www.w3.org/TR/performance-timeline-2/#dom-performance-getentriesbytype // https://www.w3.org/TR/performance-timeline-2/#dom-performance-getentriesbytype
fn GetEntriesByType(&self, entry_type: DOMString) -> Vec<DomRoot<PerformanceEntry>> { fn GetEntriesByType(&self, entry_type: DOMString) -> Vec<DomRoot<PerformanceEntry>> {
self.entries self.buffer
.borrow() .borrow()
.get_entries_by_name_and_type(None, Some(entry_type)) .get_entries_by_name_and_type(None, Some(entry_type))
} }
@ -408,7 +422,7 @@ impl PerformanceMethods for Performance {
name: DOMString, name: DOMString,
entry_type: Option<DOMString>, entry_type: Option<DOMString>,
) -> Vec<DomRoot<PerformanceEntry>> { ) -> Vec<DomRoot<PerformanceEntry>> {
self.entries self.buffer
.borrow() .borrow()
.get_entries_by_name_and_type(Some(name), entry_type) .get_entries_by_name_and_type(Some(name), entry_type)
} }
@ -435,7 +449,7 @@ impl PerformanceMethods for Performance {
// https://w3c.github.io/user-timing/#dom-performance-clearmarks // https://w3c.github.io/user-timing/#dom-performance-clearmarks
fn ClearMarks(&self, mark_name: Option<DOMString>) { fn ClearMarks(&self, mark_name: Option<DOMString>) {
self.entries self.buffer
.borrow_mut() .borrow_mut()
.clear_entries_by_name_and_type(mark_name, Some(DOMString::from("mark"))); .clear_entries_by_name_and_type(mark_name, Some(DOMString::from("mark")));
} }
@ -450,7 +464,7 @@ impl PerformanceMethods for Performance {
// Steps 1 and 2. // Steps 1 and 2.
let end_time = match end_mark { let end_time = match end_mark {
Some(name) => self Some(name) => self
.entries .buffer
.borrow() .borrow()
.get_last_entry_start_time_with_name_and_type(DOMString::from("mark"), name), .get_last_entry_start_time_with_name_and_type(DOMString::from("mark"), name),
None => self.now(), None => self.now(),
@ -459,7 +473,7 @@ impl PerformanceMethods for Performance {
// Step 3. // Step 3.
let start_time = match start_mark { let start_time = match start_mark {
Some(name) => self Some(name) => self
.entries .buffer
.borrow() .borrow()
.get_last_entry_start_time_with_name_and_type(DOMString::from("mark"), name), .get_last_entry_start_time_with_name_and_type(DOMString::from("mark"), name),
None => 0., None => 0.,
@ -485,13 +499,13 @@ impl PerformanceMethods for Performance {
// https://w3c.github.io/user-timing/#dom-performance-clearmeasures // https://w3c.github.io/user-timing/#dom-performance-clearmeasures
fn ClearMeasures(&self, measure_name: Option<DOMString>) { fn ClearMeasures(&self, measure_name: Option<DOMString>) {
self.entries self.buffer
.borrow_mut() .borrow_mut()
.clear_entries_by_name_and_type(measure_name, Some(DOMString::from("measure"))); .clear_entries_by_name_and_type(measure_name, Some(DOMString::from("measure")));
} }
// https://w3c.github.io/resource-timing/#dom-performance-clearresourcetimings // https://w3c.github.io/resource-timing/#dom-performance-clearresourcetimings
fn ClearResourceTimings(&self) { fn ClearResourceTimings(&self) {
self.entries self.buffer
.borrow_mut() .borrow_mut()
.clear_entries_by_name_and_type(None, Some(DOMString::from("resource"))); .clear_entries_by_name_and_type(None, Some(DOMString::from("resource")));
self.resource_timing_buffer_current_size.set(0); self.resource_timing_buffer_current_size.set(0);

View file

@ -872,19 +872,10 @@ impl WindowMethods for Window {
// https://dvcs.w3.org/hg/webperf/raw-file/tip/specs/ // https://dvcs.w3.org/hg/webperf/raw-file/tip/specs/
// NavigationTiming/Overview.html#sec-window.performance-attribute // NavigationTiming/Overview.html#sec-window.performance-attribute
fn Performance(&self) -> DomRoot<Performance> { fn Performance(&self) -> DomRoot<Performance> {
match self.current_state.get() { self.performance.or_init(|| {
WindowState::Alive => self.performance.or_init(|| { let global_scope = self.upcast::<GlobalScope>();
let global_scope = self.upcast::<GlobalScope>(); Performance::new(global_scope, self.navigation_start_precise.get())
Performance::new(global_scope, self.navigation_start_precise.get()) })
}),
WindowState::Zombie => {
// Don't store in Zombie state,
// as clear_js_runtime has already been called,
// and we won't have another opportunity to drop it.
let global_scope = self.upcast::<GlobalScope>();
Performance::new(global_scope, self.navigation_start_precise.get())
},
}
} }
// https://html.spec.whatwg.org/multipage/#globaleventhandlers // https://html.spec.whatwg.org/multipage/#globaleventhandlers
@ -1308,7 +1299,9 @@ impl Window {
self.current_state.set(WindowState::Zombie); self.current_state.set(WindowState::Zombie);
*self.js_runtime.borrow_mut() = None; *self.js_runtime.borrow_mut() = None;
self.window_proxy.set(None); self.window_proxy.set(None);
self.performance.set(None); if let Some(performance) = self.performance.get() {
performance.clear_and_disable_performance_entry_buffer();
}
self.ignore_all_events(); self.ignore_all_events();
} }