mirror of
https://github.com/servo/servo.git
synced 2025-06-08 08:33:26 +00:00
The code was split into the following two abstractions. - OneshotTimers can be used to schedule arbitrary oneshot timers, such as XHR-Timeouts. - JsTimers (`setTimeout` and `setInterval`) which use OneshotTimers to schedule individual callbacks. With this change the implementation (of JsTimers in particular) is in much closer alignment with the specification.
487 lines
17 KiB
Rust
487 lines
17 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::callback::ExceptionHandling::Report;
|
|
use dom::bindings::cell::DOMRefCell;
|
|
use dom::bindings::codegen::Bindings::FunctionBinding::Function;
|
|
use dom::bindings::global::GlobalRef;
|
|
use dom::bindings::reflector::Reflectable;
|
|
use dom::bindings::trace::JSTraceable;
|
|
use dom::window::ScriptHelpers;
|
|
use dom::xmlhttprequest::XHRTimeoutCallback;
|
|
use euclid::length::Length;
|
|
use heapsize::HeapSizeOf;
|
|
use ipc_channel::ipc::IpcSender;
|
|
use js::jsapi::{HandleValue, Heap, RootedValue};
|
|
use js::jsval::{JSVal, UndefinedValue};
|
|
use num::traits::Saturating;
|
|
use script_traits::{MsDuration, precise_time_ms};
|
|
use script_traits::{TimerEvent, TimerEventId, TimerEventRequest, TimerSource};
|
|
use std::cell::Cell;
|
|
use std::cmp::{self, Ord, Ordering};
|
|
use std::collections::HashMap;
|
|
use std::default::Default;
|
|
use std::rc::Rc;
|
|
use util::str::DOMString;
|
|
|
|
#[derive(JSTraceable, PartialEq, Eq, Copy, Clone, HeapSizeOf, Hash, PartialOrd, Ord, Debug)]
|
|
pub struct OneshotTimerHandle(i32);
|
|
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
#[privatize]
|
|
pub struct OneshotTimers {
|
|
js_timers: JsTimers,
|
|
#[ignore_heap_size_of = "Defined in std"]
|
|
timer_event_chan: IpcSender<TimerEvent>,
|
|
#[ignore_heap_size_of = "Defined in std"]
|
|
scheduler_chan: IpcSender<TimerEventRequest>,
|
|
next_timer_handle: Cell<OneshotTimerHandle>,
|
|
timers: DOMRefCell<Vec<OneshotTimer>>,
|
|
suspended_since: Cell<Option<MsDuration>>,
|
|
/// Initially 0, increased whenever the associated document is reactivated
|
|
/// by the amount of ms the document was inactive. The current time can be
|
|
/// offset back by this amount for a coherent time across document
|
|
/// activations.
|
|
suspension_offset: Cell<MsDuration>,
|
|
/// Calls to `fire_timer` with a different argument than this get ignored.
|
|
/// They were previously scheduled and got invalidated when
|
|
/// - timers were suspended,
|
|
/// - the timer it was scheduled for got canceled or
|
|
/// - a timer was added with an earlier callback time. In this case the
|
|
/// original timer is rescheduled when it is the next one to get called.
|
|
expected_event_id: Cell<TimerEventId>,
|
|
}
|
|
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
#[privatize]
|
|
struct OneshotTimer {
|
|
handle: OneshotTimerHandle,
|
|
source: TimerSource,
|
|
callback: OneshotTimerCallback,
|
|
scheduled_for: MsDuration,
|
|
}
|
|
|
|
// This enum is required to work around the fact that trait objects do not support generic methods.
|
|
// A replacement trait would have a method such as
|
|
// `invoke<T: Reflectable>(self: Box<Self>, this: &T, js_timers: &JsTimers);`.
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
pub enum OneshotTimerCallback {
|
|
XhrTimeout(XHRTimeoutCallback),
|
|
JsTimer(JsTimerTask),
|
|
}
|
|
|
|
impl OneshotTimerCallback {
|
|
fn invoke<T: Reflectable>(self, this: &T, js_timers: &JsTimers) {
|
|
match self {
|
|
OneshotTimerCallback::XhrTimeout(callback) => callback.invoke(),
|
|
OneshotTimerCallback::JsTimer(task) => task.invoke(this, js_timers),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Ord for OneshotTimer {
|
|
fn cmp(&self, other: &OneshotTimer) -> Ordering {
|
|
match self.scheduled_for.cmp(&other.scheduled_for).reverse() {
|
|
Ordering::Equal => self.handle.cmp(&other.handle).reverse(),
|
|
res => res
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PartialOrd for OneshotTimer {
|
|
fn partial_cmp(&self, other: &OneshotTimer) -> Option<Ordering> {
|
|
Some(self.cmp(other))
|
|
}
|
|
}
|
|
|
|
impl Eq for OneshotTimer {}
|
|
impl PartialEq for OneshotTimer {
|
|
fn eq(&self, other: &OneshotTimer) -> bool {
|
|
self as *const OneshotTimer == other as *const OneshotTimer
|
|
}
|
|
}
|
|
|
|
impl OneshotTimers {
|
|
pub fn new(timer_event_chan: IpcSender<TimerEvent>,
|
|
scheduler_chan: IpcSender<TimerEventRequest>)
|
|
-> OneshotTimers {
|
|
OneshotTimers {
|
|
js_timers: JsTimers::new(),
|
|
timer_event_chan: timer_event_chan,
|
|
scheduler_chan: scheduler_chan,
|
|
next_timer_handle: Cell::new(OneshotTimerHandle(1)),
|
|
timers: DOMRefCell::new(Vec::new()),
|
|
suspended_since: Cell::new(None),
|
|
suspension_offset: Cell::new(Length::new(0)),
|
|
expected_event_id: Cell::new(TimerEventId(0)),
|
|
}
|
|
}
|
|
|
|
pub fn schedule_callback(&self,
|
|
callback: OneshotTimerCallback,
|
|
duration: MsDuration,
|
|
source: TimerSource)
|
|
-> OneshotTimerHandle {
|
|
let new_handle = self.next_timer_handle.get();
|
|
self.next_timer_handle.set(OneshotTimerHandle(new_handle.0 + 1));
|
|
|
|
let scheduled_for = self.base_time() + duration;
|
|
|
|
let timer = OneshotTimer {
|
|
handle: new_handle,
|
|
source: source,
|
|
callback: callback,
|
|
scheduled_for: scheduled_for,
|
|
};
|
|
|
|
{
|
|
let mut timers = self.timers.borrow_mut();
|
|
let insertion_index = timers.binary_search(&timer).err().unwrap();
|
|
timers.insert(insertion_index, timer);
|
|
}
|
|
|
|
if self.is_next_timer(new_handle) {
|
|
self.schedule_timer_call();
|
|
}
|
|
|
|
new_handle
|
|
}
|
|
|
|
pub fn unschedule_callback(&self, handle: OneshotTimerHandle) {
|
|
let was_next = self.is_next_timer(handle);
|
|
|
|
self.timers.borrow_mut().retain(|t| t.handle != handle);
|
|
|
|
if was_next {
|
|
self.invalidate_expected_event_id();
|
|
self.schedule_timer_call();
|
|
}
|
|
}
|
|
|
|
fn is_next_timer(&self, handle: OneshotTimerHandle) -> bool {
|
|
match self.timers.borrow().last() {
|
|
None => false,
|
|
Some(ref max_timer) => max_timer.handle == handle
|
|
}
|
|
}
|
|
|
|
pub fn fire_timer<T: Reflectable>(&self, id: TimerEventId, this: &T) {
|
|
let expected_id = self.expected_event_id.get();
|
|
if expected_id != id {
|
|
debug!("ignoring timer fire event {:?} (expected {:?})", id, expected_id);
|
|
return;
|
|
}
|
|
|
|
assert!(self.suspended_since.get().is_none());
|
|
|
|
let base_time = self.base_time();
|
|
|
|
// Since the event id was the expected one, at least one timer should be due.
|
|
assert!(base_time >= self.timers.borrow().last().unwrap().scheduled_for);
|
|
|
|
// select timers to run to prevent firing timers
|
|
// that were installed during fire of another timer
|
|
let mut timers_to_run = Vec::new();
|
|
|
|
loop {
|
|
let mut timers = self.timers.borrow_mut();
|
|
|
|
if timers.is_empty() || timers.last().unwrap().scheduled_for > base_time {
|
|
break;
|
|
}
|
|
|
|
timers_to_run.push(timers.pop().unwrap());
|
|
}
|
|
|
|
for timer in timers_to_run {
|
|
let callback = timer.callback;
|
|
callback.invoke(this, &self.js_timers);
|
|
}
|
|
|
|
self.schedule_timer_call();
|
|
}
|
|
|
|
fn base_time(&self) -> MsDuration {
|
|
let offset = self.suspension_offset.get();
|
|
|
|
match self.suspended_since.get() {
|
|
Some(time) => time - offset,
|
|
None => precise_time_ms() - offset,
|
|
}
|
|
}
|
|
|
|
pub fn suspend(&self) {
|
|
assert!(self.suspended_since.get().is_none());
|
|
|
|
self.suspended_since.set(Some(precise_time_ms()));
|
|
self.invalidate_expected_event_id();
|
|
}
|
|
|
|
pub fn resume(&self) {
|
|
assert!(self.suspended_since.get().is_some());
|
|
|
|
let additional_offset = match self.suspended_since.get() {
|
|
Some(suspended_since) => precise_time_ms() - suspended_since,
|
|
None => panic!("Timers are not suspended.")
|
|
};
|
|
|
|
self.suspension_offset.set(self.suspension_offset.get() + additional_offset);
|
|
self.suspended_since.set(None);
|
|
|
|
self.schedule_timer_call();
|
|
}
|
|
|
|
fn schedule_timer_call(&self) {
|
|
if self.suspended_since.get().is_some() {
|
|
// The timer will be scheduled when the pipeline is thawed.
|
|
return;
|
|
}
|
|
|
|
let timers = self.timers.borrow();
|
|
|
|
if let Some(timer) = timers.last() {
|
|
let expected_event_id = self.invalidate_expected_event_id();
|
|
|
|
let delay = Length::new(timer.scheduled_for.get().saturating_sub(precise_time_ms().get()));
|
|
let request = TimerEventRequest(self.timer_event_chan.clone(), timer.source,
|
|
expected_event_id, delay);
|
|
self.scheduler_chan.send(request).unwrap();
|
|
}
|
|
}
|
|
|
|
fn invalidate_expected_event_id(&self) -> TimerEventId {
|
|
let TimerEventId(currently_expected) = self.expected_event_id.get();
|
|
let next_id = TimerEventId(currently_expected + 1);
|
|
debug!("invalidating expected timer (was {:?}, now {:?}", currently_expected, next_id);
|
|
self.expected_event_id.set(next_id);
|
|
next_id
|
|
}
|
|
|
|
pub fn set_timeout_or_interval(&self,
|
|
global: GlobalRef,
|
|
callback: TimerCallback,
|
|
arguments: Vec<HandleValue>,
|
|
timeout: i32,
|
|
is_interval: IsInterval,
|
|
source: TimerSource)
|
|
-> i32 {
|
|
|
|
self.js_timers.set_timeout_or_interval(global,
|
|
callback,
|
|
arguments,
|
|
timeout,
|
|
is_interval,
|
|
source)
|
|
}
|
|
|
|
pub fn clear_timeout_or_interval(&self, global: GlobalRef, handle: i32) {
|
|
self.js_timers.clear_timeout_or_interval(global, handle)
|
|
}
|
|
}
|
|
|
|
#[derive(JSTraceable, PartialEq, Eq, Copy, Clone, HeapSizeOf, Hash, PartialOrd, Ord)]
|
|
pub struct JsTimerHandle(i32);
|
|
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
#[privatize]
|
|
pub struct JsTimers {
|
|
next_timer_handle: Cell<JsTimerHandle>,
|
|
active_timers: DOMRefCell<HashMap<JsTimerHandle, JsTimerEntry>>,
|
|
/// The nesting level of the currently executing timer task or 0.
|
|
nesting_level: Cell<u32>,
|
|
}
|
|
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
struct JsTimerEntry {
|
|
oneshot_handle: OneshotTimerHandle,
|
|
}
|
|
|
|
// Holder for the various JS values associated with setTimeout
|
|
// (ie. function value to invoke and all arguments to pass
|
|
// to the function when calling it)
|
|
// TODO: Handle rooting during invocation when movable GC is turned on
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
pub struct JsTimerTask {
|
|
#[ignore_heap_size_of = "Because it is non-owning"]
|
|
handle: JsTimerHandle,
|
|
source: TimerSource,
|
|
callback: InternalTimerCallback,
|
|
is_interval: IsInterval,
|
|
nesting_level: u32,
|
|
duration: MsDuration,
|
|
}
|
|
|
|
// Enum allowing more descriptive values for the is_interval field
|
|
#[derive(JSTraceable, PartialEq, Copy, Clone, HeapSizeOf)]
|
|
pub enum IsInterval {
|
|
Interval,
|
|
NonInterval,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub enum TimerCallback {
|
|
StringTimerCallback(DOMString),
|
|
FunctionTimerCallback(Rc<Function>),
|
|
}
|
|
|
|
#[derive(JSTraceable, Clone)]
|
|
enum InternalTimerCallback {
|
|
StringTimerCallback(DOMString),
|
|
FunctionTimerCallback(Rc<Function>, Rc<Vec<Heap<JSVal>>>),
|
|
}
|
|
|
|
impl HeapSizeOf for InternalTimerCallback {
|
|
fn heap_size_of_children(&self) -> usize {
|
|
// FIXME: Rc<T> isn't HeapSizeOf and we can't ignore it due to #6870 and #6871
|
|
0
|
|
}
|
|
}
|
|
|
|
impl JsTimers {
|
|
pub fn new() -> JsTimers {
|
|
JsTimers {
|
|
next_timer_handle: Cell::new(JsTimerHandle(1)),
|
|
active_timers: DOMRefCell::new(HashMap::new()),
|
|
nesting_level: Cell::new(0),
|
|
}
|
|
}
|
|
|
|
// see https://html.spec.whatwg.org/multipage/#timer-initialisation-steps
|
|
pub fn set_timeout_or_interval(&self,
|
|
global: GlobalRef,
|
|
callback: TimerCallback,
|
|
arguments: Vec<HandleValue>,
|
|
timeout: i32,
|
|
is_interval: IsInterval,
|
|
source: TimerSource)
|
|
-> i32 {
|
|
let callback = match callback {
|
|
TimerCallback::StringTimerCallback(code_str) =>
|
|
InternalTimerCallback::StringTimerCallback(code_str),
|
|
TimerCallback::FunctionTimerCallback(function) => {
|
|
// This is a bit complicated, but this ensures that the vector's
|
|
// buffer isn't reallocated (and moved) after setting the Heap values
|
|
let mut args = Vec::with_capacity(arguments.len());
|
|
for _ in 0..arguments.len() {
|
|
args.push(Heap::default());
|
|
}
|
|
for (i, item) in arguments.iter().enumerate() {
|
|
args.get_mut(i).unwrap().set(item.get());
|
|
}
|
|
InternalTimerCallback::FunctionTimerCallback(function, Rc::new(args))
|
|
}
|
|
};
|
|
|
|
// step 2
|
|
let JsTimerHandle(new_handle) = self.next_timer_handle.get();
|
|
self.next_timer_handle.set(JsTimerHandle(new_handle + 1));
|
|
|
|
// step 3 as part of initialize_and_schedule below
|
|
|
|
// step 4
|
|
let mut task = JsTimerTask {
|
|
handle: JsTimerHandle(new_handle),
|
|
source: source,
|
|
callback: callback,
|
|
is_interval: is_interval,
|
|
nesting_level: 0,
|
|
duration: Length::new(0),
|
|
};
|
|
|
|
// step 5
|
|
task.duration = Length::new(cmp::max(0, timeout) as u64);
|
|
|
|
// step 3, 6-9, 11-14
|
|
self.initialize_and_schedule(global, task);
|
|
|
|
// step 10
|
|
new_handle
|
|
}
|
|
|
|
pub fn clear_timeout_or_interval(&self, global: GlobalRef, handle: i32) {
|
|
let mut active_timers = self.active_timers.borrow_mut();
|
|
|
|
if let Some(entry) = active_timers.remove(&JsTimerHandle(handle)) {
|
|
global.unschedule_callback(entry.oneshot_handle);
|
|
}
|
|
}
|
|
|
|
// see https://html.spec.whatwg.org/multipage/#timer-initialisation-steps
|
|
fn initialize_and_schedule(&self, global: GlobalRef, mut task: JsTimerTask) {
|
|
let handle = task.handle;
|
|
let mut active_timers = self.active_timers.borrow_mut();
|
|
|
|
// step 6
|
|
let nesting_level = self.nesting_level.get();
|
|
|
|
// step 7
|
|
let duration = clamp_duration(nesting_level, task.duration);
|
|
|
|
// step 8, 9
|
|
task.nesting_level = nesting_level + 1;
|
|
|
|
// essentially step 11-14
|
|
let callback = OneshotTimerCallback::JsTimer(task);
|
|
let oneshot_handle = global.schedule_callback(callback, duration);
|
|
|
|
// step 3
|
|
let entry = active_timers.entry(handle).or_insert(JsTimerEntry {
|
|
oneshot_handle: oneshot_handle,
|
|
});
|
|
entry.oneshot_handle = oneshot_handle;
|
|
}
|
|
}
|
|
|
|
// see step 7 of https://html.spec.whatwg.org/multipage/#timer-initialisation-steps
|
|
fn clamp_duration(nesting_level: u32, unclamped: MsDuration) -> MsDuration {
|
|
let lower_bound = if nesting_level > 5 {
|
|
4
|
|
} else {
|
|
0
|
|
};
|
|
|
|
cmp::max(Length::new(lower_bound), unclamped)
|
|
}
|
|
|
|
impl JsTimerTask {
|
|
// see https://html.spec.whatwg.org/multipage/#timer-initialisation-steps
|
|
#[allow(unsafe_code)]
|
|
pub fn invoke<T: Reflectable>(self, this: &T, timers: &JsTimers) {
|
|
// step 4.1 can be ignored, because we proactively prevent execution
|
|
// of this task when its scheduled execution is canceled.
|
|
|
|
// prep for step 6 in nested set_timeout_or_interval calls
|
|
timers.nesting_level.set(self.nesting_level);
|
|
|
|
// step 4.2
|
|
match *&self.callback {
|
|
InternalTimerCallback::StringTimerCallback(ref code_str) => {
|
|
let cx = this.global().r().get_cx();
|
|
let mut rval = RootedValue::new(cx, UndefinedValue());
|
|
|
|
this.evaluate_js_on_global_with_result(code_str, rval.handle_mut());
|
|
},
|
|
InternalTimerCallback::FunctionTimerCallback(ref function, ref arguments) => {
|
|
let arguments: Vec<JSVal> = arguments.iter().map(|arg| arg.get()).collect();
|
|
let arguments = arguments.iter().by_ref().map(|arg| unsafe {
|
|
HandleValue::from_marked_location(arg)
|
|
}).collect();
|
|
|
|
let _ = function.Call_(this, arguments, Report);
|
|
},
|
|
};
|
|
|
|
// reset nesting level (see above)
|
|
timers.nesting_level.set(0);
|
|
|
|
// step 4.3
|
|
// Since we choose proactively prevent execution (see 4.1 above), we must only
|
|
// reschedule repeating timers when they were not canceled as part of step 4.2.
|
|
if self.is_interval == IsInterval::Interval &&
|
|
timers.active_timers.borrow().contains_key(&self.handle) {
|
|
|
|
timers.initialize_and_schedule(this.global().r(), self);
|
|
}
|
|
}
|
|
}
|