1#[cfg(feature = "servo")]
8use crate::animation::DocumentAnimationSet;
9use crate::bloom::StyleBloom;
10use crate::computed_value_flags::ComputedValueFlags;
11use crate::data::{EagerPseudoStyles, ElementData};
12use crate::dom::{SendElement, TElement};
13#[cfg(feature = "gecko")]
14use crate::gecko_bindings::structs;
15use crate::parallel::{STACK_SAFETY_MARGIN_KB, STYLE_THREAD_STACK_SIZE_KB};
16use crate::properties::ComputedValues;
17#[cfg(feature = "servo")]
18use crate::properties::PropertyId;
19use crate::rule_cache::RuleCache;
20use crate::rule_tree::StrongRuleNode;
21use crate::selector_parser::{SnapshotMap, EAGER_PSEUDO_COUNT};
22use crate::shared_lock::StylesheetGuards;
23use crate::sharing::StyleSharingCache;
24use crate::stylist::Stylist;
25use crate::thread_state::{self, ThreadState};
26use crate::traversal::DomTraversal;
27use crate::traversal_flags::TraversalFlags;
28use app_units::Au;
29use euclid::default::Size2D;
30use euclid::Scale;
31#[cfg(feature = "servo")]
32use fxhash::FxHashMap;
33use selectors::context::SelectorCaches;
34#[cfg(feature = "gecko")]
35use servo_arc::Arc;
36use std::fmt;
37use std::ops;
38use std::time::{Duration, Instant};
39use style_traits::CSSPixel;
40use style_traits::DevicePixel;
41#[cfg(feature = "servo")]
42use style_traits::SpeculativePainter;
43#[cfg(feature = "servo")]
44use stylo_atoms::Atom;
45
46pub use selectors::matching::QuirksMode;
47
48#[derive(Clone)]
51pub struct StyleSystemOptions {
52 pub disable_style_sharing_cache: bool,
54 pub dump_style_statistics: bool,
56 pub style_statistics_threshold: usize,
59}
60
61#[cfg(feature = "gecko")]
62fn get_env_bool(name: &str) -> bool {
63 use std::env;
64 match env::var(name) {
65 Ok(s) => !s.is_empty(),
66 Err(_) => false,
67 }
68}
69
70const DEFAULT_STATISTICS_THRESHOLD: usize = 50;
71
72#[cfg(feature = "gecko")]
73fn get_env_usize(name: &str) -> Option<usize> {
74 use std::env;
75 env::var(name).ok().map(|s| {
76 s.parse::<usize>()
77 .expect("Couldn't parse environmental variable as usize")
78 })
79}
80
81#[cfg(feature = "servo")]
85pub static DEFAULT_DISABLE_STYLE_SHARING_CACHE: std::sync::atomic::AtomicBool =
86 std::sync::atomic::AtomicBool::new(false);
87
88#[cfg(feature = "servo")]
92pub static DEFAULT_DUMP_STYLE_STATISTICS: std::sync::atomic::AtomicBool =
93 std::sync::atomic::AtomicBool::new(false);
94
95impl Default for StyleSystemOptions {
96 #[cfg(feature = "servo")]
97 fn default() -> Self {
98 use std::sync::atomic::Ordering;
99
100 StyleSystemOptions {
101 disable_style_sharing_cache: DEFAULT_DISABLE_STYLE_SHARING_CACHE
102 .load(Ordering::Relaxed),
103 dump_style_statistics: DEFAULT_DUMP_STYLE_STATISTICS.load(Ordering::Relaxed),
104 style_statistics_threshold: DEFAULT_STATISTICS_THRESHOLD,
105 }
106 }
107
108 #[cfg(feature = "gecko")]
109 fn default() -> Self {
110 StyleSystemOptions {
111 disable_style_sharing_cache: get_env_bool("DISABLE_STYLE_SHARING_CACHE"),
112 dump_style_statistics: get_env_bool("DUMP_STYLE_STATISTICS"),
113 style_statistics_threshold: get_env_usize("STYLE_STATISTICS_THRESHOLD")
114 .unwrap_or(DEFAULT_STATISTICS_THRESHOLD),
115 }
116 }
117}
118
119pub struct SharedStyleContext<'a> {
124 pub stylist: &'a Stylist,
126
127 pub visited_styles_enabled: bool,
132
133 pub options: StyleSystemOptions,
135
136 pub guards: StylesheetGuards<'a>,
138
139 pub current_time_for_animations: f64,
142
143 pub traversal_flags: TraversalFlags,
145
146 pub snapshot_map: &'a SnapshotMap,
148
149 #[cfg(feature = "servo")]
151 pub animations: DocumentAnimationSet,
152
153 #[cfg(feature = "servo")]
155 pub registered_speculative_painters: &'a dyn RegisteredSpeculativePainters,
156}
157
158impl<'a> SharedStyleContext<'a> {
159 pub fn viewport_size(&self) -> Size2D<Au> {
161 self.stylist.device().au_viewport_size()
162 }
163
164 pub fn device_pixel_ratio(&self) -> Scale<f32, CSSPixel, DevicePixel> {
166 self.stylist.device().device_pixel_ratio()
167 }
168
169 pub fn quirks_mode(&self) -> QuirksMode {
171 self.stylist.quirks_mode()
172 }
173}
174
175#[derive(Clone, Debug, Default)]
183pub struct CascadeInputs {
184 pub rules: Option<StrongRuleNode>,
187
188 pub visited_rules: Option<StrongRuleNode>,
193
194 pub flags: ComputedValueFlags,
196}
197
198impl CascadeInputs {
199 pub fn new_from_style(style: &ComputedValues) -> Self {
201 Self {
202 rules: style.rules.clone(),
203 visited_rules: style.visited_style().and_then(|v| v.rules.clone()),
204 flags: style.flags.for_cascade_inputs(),
205 }
206 }
207}
208
209#[derive(Debug)]
212pub struct EagerPseudoCascadeInputs(Option<[Option<CascadeInputs>; EAGER_PSEUDO_COUNT]>);
213
214impl Clone for EagerPseudoCascadeInputs {
217 fn clone(&self) -> Self {
218 if self.0.is_none() {
219 return EagerPseudoCascadeInputs(None);
220 }
221 let self_inputs = self.0.as_ref().unwrap();
222 let mut inputs: [Option<CascadeInputs>; EAGER_PSEUDO_COUNT] = Default::default();
223 for i in 0..EAGER_PSEUDO_COUNT {
224 inputs[i] = self_inputs[i].clone();
225 }
226 EagerPseudoCascadeInputs(Some(inputs))
227 }
228}
229
230impl EagerPseudoCascadeInputs {
231 fn new_from_style(styles: &EagerPseudoStyles) -> Self {
233 EagerPseudoCascadeInputs(styles.as_optional_array().map(|styles| {
234 let mut inputs: [Option<CascadeInputs>; EAGER_PSEUDO_COUNT] = Default::default();
235 for i in 0..EAGER_PSEUDO_COUNT {
236 inputs[i] = styles[i].as_ref().map(|s| CascadeInputs::new_from_style(s));
237 }
238 inputs
239 }))
240 }
241
242 pub fn into_array(self) -> Option<[Option<CascadeInputs>; EAGER_PSEUDO_COUNT]> {
244 self.0
245 }
246}
247
248#[derive(Clone, Debug)]
256pub struct ElementCascadeInputs {
257 pub primary: CascadeInputs,
259 pub pseudos: EagerPseudoCascadeInputs,
261}
262
263impl ElementCascadeInputs {
264 #[inline]
266 pub fn new_from_element_data(data: &ElementData) -> Self {
267 debug_assert!(data.has_styles());
268 ElementCascadeInputs {
269 primary: CascadeInputs::new_from_style(data.styles.primary()),
270 pseudos: EagerPseudoCascadeInputs::new_from_style(&data.styles.pseudos),
271 }
272 }
273}
274
275#[derive(AddAssign, Clone, Default)]
279pub struct PerThreadTraversalStatistics {
280 pub elements_traversed: u32,
282 pub elements_styled: u32,
284 pub elements_matched: u32,
286 pub styles_shared: u32,
288 pub styles_reused: u32,
291}
292
293#[derive(Default)]
296pub struct TraversalStatistics {
297 pub aggregated: PerThreadTraversalStatistics,
299 pub selectors: u32,
301 pub revalidation_selectors: u32,
303 pub dependency_selectors: u32,
305 pub declarations: u32,
307 pub stylist_rebuilds: u32,
309 pub traversal_time: Duration,
311 pub is_parallel: bool,
313 pub is_large: bool,
315}
316
317impl fmt::Display for TraversalStatistics {
320 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
321 writeln!(f, "[PERF] perf block start")?;
322 writeln!(
323 f,
324 "[PERF],traversal,{}",
325 if self.is_parallel {
326 "parallel"
327 } else {
328 "sequential"
329 }
330 )?;
331 writeln!(
332 f,
333 "[PERF],elements_traversed,{}",
334 self.aggregated.elements_traversed
335 )?;
336 writeln!(
337 f,
338 "[PERF],elements_styled,{}",
339 self.aggregated.elements_styled
340 )?;
341 writeln!(
342 f,
343 "[PERF],elements_matched,{}",
344 self.aggregated.elements_matched
345 )?;
346 writeln!(f, "[PERF],styles_shared,{}", self.aggregated.styles_shared)?;
347 writeln!(f, "[PERF],styles_reused,{}", self.aggregated.styles_reused)?;
348 writeln!(f, "[PERF],selectors,{}", self.selectors)?;
349 writeln!(
350 f,
351 "[PERF],revalidation_selectors,{}",
352 self.revalidation_selectors
353 )?;
354 writeln!(
355 f,
356 "[PERF],dependency_selectors,{}",
357 self.dependency_selectors
358 )?;
359 writeln!(f, "[PERF],declarations,{}", self.declarations)?;
360 writeln!(f, "[PERF],stylist_rebuilds,{}", self.stylist_rebuilds)?;
361 writeln!(
362 f,
363 "[PERF],traversal_time_ms,{}",
364 self.traversal_time.as_secs_f64() * 1000.
365 )?;
366 writeln!(f, "[PERF] perf block end")
367 }
368}
369
370impl TraversalStatistics {
371 pub fn new<E, D>(
375 aggregated: PerThreadTraversalStatistics,
376 traversal: &D,
377 parallel: bool,
378 start: Instant,
379 ) -> TraversalStatistics
380 where
381 E: TElement,
382 D: DomTraversal<E>,
383 {
384 let threshold = traversal
385 .shared_context()
386 .options
387 .style_statistics_threshold;
388 let stylist = traversal.shared_context().stylist;
389 let is_large = aggregated.elements_traversed as usize >= threshold;
390 TraversalStatistics {
391 aggregated,
392 selectors: stylist.num_selectors() as u32,
393 revalidation_selectors: stylist.num_revalidation_selectors() as u32,
394 dependency_selectors: stylist.num_invalidations() as u32,
395 declarations: stylist.num_declarations() as u32,
396 stylist_rebuilds: stylist.num_rebuilds() as u32,
397 traversal_time: Instant::now() - start,
398 is_parallel: parallel,
399 is_large,
400 }
401 }
402}
403
404#[cfg(feature = "gecko")]
405bitflags! {
406 pub struct UpdateAnimationsTasks: u8 {
409 const CSS_ANIMATIONS = structs::UpdateAnimationsTasks_CSSAnimations;
411 const CSS_TRANSITIONS = structs::UpdateAnimationsTasks_CSSTransitions;
413 const EFFECT_PROPERTIES = structs::UpdateAnimationsTasks_EffectProperties;
415 const CASCADE_RESULTS = structs::UpdateAnimationsTasks_CascadeResults;
417 const DISPLAY_CHANGED_FROM_NONE = structs::UpdateAnimationsTasks_DisplayChangedFromNone;
422 const SCROLL_TIMELINES = structs::UpdateAnimationsTasks_ScrollTimelines;
424 const VIEW_TIMELINES = structs::UpdateAnimationsTasks_ViewTimelines;
426 }
427}
428
429pub enum SequentialTask<E: TElement> {
433 Unused(SendElement<E>),
435
436 #[cfg(feature = "gecko")]
441 UpdateAnimations {
442 el: SendElement<E>,
444 before_change_style: Option<Arc<ComputedValues>>,
448 tasks: UpdateAnimationsTasks,
450 },
451}
452
453impl<E: TElement> SequentialTask<E> {
454 pub fn execute(self) {
456 use self::SequentialTask::*;
457 debug_assert!(thread_state::get().contains(ThreadState::LAYOUT));
458 match self {
459 Unused(_) => unreachable!(),
460 #[cfg(feature = "gecko")]
461 UpdateAnimations {
462 el,
463 before_change_style,
464 tasks,
465 } => {
466 el.update_animations(before_change_style, tasks);
467 },
468 }
469 }
470
471 #[cfg(feature = "gecko")]
474 pub fn update_animations(
475 el: E,
476 before_change_style: Option<Arc<ComputedValues>>,
477 tasks: UpdateAnimationsTasks,
478 ) -> Self {
479 use self::SequentialTask::*;
480 UpdateAnimations {
481 el: unsafe { SendElement::new(el) },
482 before_change_style,
483 tasks,
484 }
485 }
486}
487
488pub struct SequentialTaskList<E>(Vec<SequentialTask<E>>)
490where
491 E: TElement;
492
493impl<E> ops::Deref for SequentialTaskList<E>
494where
495 E: TElement,
496{
497 type Target = Vec<SequentialTask<E>>;
498
499 fn deref(&self) -> &Self::Target {
500 &self.0
501 }
502}
503
504impl<E> ops::DerefMut for SequentialTaskList<E>
505where
506 E: TElement,
507{
508 fn deref_mut(&mut self) -> &mut Self::Target {
509 &mut self.0
510 }
511}
512
513impl<E> Drop for SequentialTaskList<E>
514where
515 E: TElement,
516{
517 fn drop(&mut self) {
518 debug_assert!(thread_state::get().contains(ThreadState::LAYOUT));
519 for task in self.0.drain(..) {
520 task.execute()
521 }
522 }
523}
524
525pub struct StackLimitChecker {
528 lower_limit: usize,
529}
530
531impl StackLimitChecker {
532 #[inline(never)]
535 pub fn new(stack_size_limit: usize) -> Self {
536 StackLimitChecker {
537 lower_limit: StackLimitChecker::get_sp() - stack_size_limit,
538 }
539 }
540
541 #[inline(never)]
543 pub fn limit_exceeded(&self) -> bool {
544 let curr_sp = StackLimitChecker::get_sp();
545
546 if cfg!(debug_assertions) {
552 let stack_bottom = self.lower_limit - STACK_SAFETY_MARGIN_KB * 1024;
558
559 debug_assert!(stack_bottom < curr_sp);
565
566 let distance_to_stack_bottom = curr_sp - stack_bottom;
572 let max_allowable_distance = (STYLE_THREAD_STACK_SIZE_KB + 10) * 1024;
573 debug_assert!(distance_to_stack_bottom <= max_allowable_distance);
574 }
575
576 curr_sp <= self.lower_limit
578 }
579
580 #[inline(always)]
583 fn get_sp() -> usize {
584 let mut foo: usize = 42;
585 (&mut foo as *mut usize) as usize
586 }
587}
588
589pub struct ThreadLocalStyleContext<E: TElement> {
595 pub sharing_cache: StyleSharingCache<E>,
597 pub rule_cache: RuleCache,
599 pub bloom_filter: StyleBloom<E>,
601 pub tasks: SequentialTaskList<E>,
609 pub statistics: PerThreadTraversalStatistics,
611 pub stack_limit_checker: StackLimitChecker,
614 pub selector_caches: SelectorCaches,
616}
617
618impl<E: TElement> ThreadLocalStyleContext<E> {
619 pub fn new() -> Self {
621 ThreadLocalStyleContext {
622 sharing_cache: StyleSharingCache::new(),
623 rule_cache: RuleCache::new(),
624 bloom_filter: StyleBloom::new(),
625 tasks: SequentialTaskList(Vec::new()),
626 statistics: PerThreadTraversalStatistics::default(),
627 stack_limit_checker: StackLimitChecker::new(
628 (STYLE_THREAD_STACK_SIZE_KB - STACK_SAFETY_MARGIN_KB) * 1024,
629 ),
630 selector_caches: SelectorCaches::default(),
631 }
632 }
633}
634
635pub struct StyleContext<'a, E: TElement + 'a> {
638 pub shared: &'a SharedStyleContext<'a>,
640 pub thread_local: &'a mut ThreadLocalStyleContext<E>,
642}
643
644#[cfg(feature = "servo")]
646pub trait RegisteredSpeculativePainter: SpeculativePainter {
647 fn name(&self) -> Atom;
649 fn properties(&self) -> &FxHashMap<Atom, PropertyId>;
651}
652
653#[cfg(feature = "servo")]
655pub trait RegisteredSpeculativePainters: Sync {
656 fn get(&self, name: &Atom) -> Option<&dyn RegisteredSpeculativePainter>;
658}