mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
183 lines
6.2 KiB
Rust
183 lines
6.2 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 https://mozilla.org/MPL/2.0/. */
|
|
|
|
//! Data needed by the layout thread.
|
|
|
|
use crate::display_list::items::{OpaqueNode, WebRenderImageInfo};
|
|
use fnv::FnvHasher;
|
|
use gfx::font_cache_thread::FontCacheThread;
|
|
use gfx::font_context::FontContext;
|
|
use malloc_size_of::{MallocSizeOf, MallocSizeOfOps};
|
|
use msg::constellation_msg::PipelineId;
|
|
use net_traits::image_cache::{
|
|
ImageCache, ImageCacheResult, ImageOrMetadataAvailable, UsePlaceholder,
|
|
};
|
|
use parking_lot::RwLock;
|
|
use script_layout_interface::{PendingImage, PendingImageState};
|
|
use script_traits::Painter;
|
|
use servo_atoms::Atom;
|
|
use servo_url::{ImmutableOrigin, ServoUrl};
|
|
use std::cell::{RefCell, RefMut};
|
|
use std::collections::HashMap;
|
|
use std::hash::BuildHasherDefault;
|
|
use std::sync::{Arc, Mutex};
|
|
use std::thread;
|
|
use style::context::RegisteredSpeculativePainter;
|
|
use style::context::SharedStyleContext;
|
|
|
|
pub type LayoutFontContext = FontContext<FontCacheThread>;
|
|
|
|
thread_local!(static FONT_CONTEXT_KEY: RefCell<Option<LayoutFontContext>> = RefCell::new(None));
|
|
|
|
pub fn with_thread_local_font_context<F, R>(layout_context: &LayoutContext, f: F) -> R
|
|
where
|
|
F: FnOnce(&mut LayoutFontContext) -> R,
|
|
{
|
|
FONT_CONTEXT_KEY.with(|k| {
|
|
let mut font_context = k.borrow_mut();
|
|
if font_context.is_none() {
|
|
let font_cache_thread = layout_context.font_cache_thread.lock().unwrap().clone();
|
|
*font_context = Some(FontContext::new(font_cache_thread));
|
|
}
|
|
f(&mut RefMut::map(font_context, |x| x.as_mut().unwrap()))
|
|
})
|
|
}
|
|
|
|
pub fn malloc_size_of_persistent_local_context(ops: &mut MallocSizeOfOps) -> usize {
|
|
FONT_CONTEXT_KEY.with(|r| {
|
|
if let Some(ref context) = *r.borrow() {
|
|
context.size_of(ops)
|
|
} else {
|
|
0
|
|
}
|
|
})
|
|
}
|
|
|
|
/// Layout information shared among all workers. This must be thread-safe.
|
|
pub struct LayoutContext<'a> {
|
|
/// The pipeline id of this LayoutContext.
|
|
pub id: PipelineId,
|
|
|
|
/// The origin of this layout context.
|
|
pub origin: ImmutableOrigin,
|
|
|
|
/// Bits shared by the layout and style system.
|
|
pub style_context: SharedStyleContext<'a>,
|
|
|
|
/// Reference to the script thread image cache.
|
|
pub image_cache: Arc<dyn ImageCache>,
|
|
|
|
/// Interface to the font cache thread.
|
|
pub font_cache_thread: Mutex<FontCacheThread>,
|
|
|
|
/// A cache of WebRender image info.
|
|
pub webrender_image_cache: Arc<
|
|
RwLock<
|
|
HashMap<(ServoUrl, UsePlaceholder), WebRenderImageInfo, BuildHasherDefault<FnvHasher>>,
|
|
>,
|
|
>,
|
|
|
|
/// Paint worklets
|
|
pub registered_painters: &'a dyn RegisteredPainters,
|
|
|
|
/// A list of in-progress image loads to be shared with the script thread.
|
|
pub pending_images: Mutex<Vec<PendingImage>>,
|
|
}
|
|
|
|
impl<'a> Drop for LayoutContext<'a> {
|
|
fn drop(&mut self) {
|
|
if !thread::panicking() {
|
|
assert!(self.pending_images.lock().unwrap().is_empty());
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> LayoutContext<'a> {
|
|
#[inline(always)]
|
|
pub fn shared_context(&self) -> &SharedStyleContext {
|
|
&self.style_context
|
|
}
|
|
|
|
pub fn get_or_request_image_or_meta(
|
|
&self,
|
|
node: OpaqueNode,
|
|
url: ServoUrl,
|
|
use_placeholder: UsePlaceholder,
|
|
) -> Option<ImageOrMetadataAvailable> {
|
|
// Check for available image or start tracking.
|
|
let cache_result = self.image_cache.get_cached_image_status(
|
|
url.clone(),
|
|
self.origin.clone(),
|
|
None,
|
|
use_placeholder,
|
|
);
|
|
|
|
match cache_result {
|
|
ImageCacheResult::Available(img_or_meta) => Some(img_or_meta),
|
|
// Image has been requested, is still pending. Return no image for this paint loop.
|
|
// When the image loads it will trigger a reflow and/or repaint.
|
|
ImageCacheResult::Pending(id) => {
|
|
let image = PendingImage {
|
|
state: PendingImageState::PendingResponse,
|
|
node: node.into(),
|
|
id,
|
|
origin: self.origin.clone(),
|
|
};
|
|
self.pending_images.lock().unwrap().push(image);
|
|
None
|
|
},
|
|
// Not yet requested - request image or metadata from the cache
|
|
ImageCacheResult::ReadyForRequest(id) => {
|
|
let image = PendingImage {
|
|
state: PendingImageState::Unrequested(url),
|
|
node: node.into(),
|
|
id,
|
|
origin: self.origin.clone(),
|
|
};
|
|
self.pending_images.lock().unwrap().push(image);
|
|
None
|
|
},
|
|
// Image failed to load, so just return nothing
|
|
ImageCacheResult::LoadError => None,
|
|
}
|
|
}
|
|
|
|
pub fn get_webrender_image_for_url(
|
|
&self,
|
|
node: OpaqueNode,
|
|
url: ServoUrl,
|
|
use_placeholder: UsePlaceholder,
|
|
) -> Option<WebRenderImageInfo> {
|
|
if let Some(existing_webrender_image) = self
|
|
.webrender_image_cache
|
|
.read()
|
|
.get(&(url.clone(), use_placeholder))
|
|
{
|
|
return Some((*existing_webrender_image).clone());
|
|
}
|
|
|
|
match self.get_or_request_image_or_meta(node, url.clone(), use_placeholder) {
|
|
Some(ImageOrMetadataAvailable::ImageAvailable { image, .. }) => {
|
|
let image_info = WebRenderImageInfo::from_image(&*image);
|
|
if image_info.key.is_none() {
|
|
Some(image_info)
|
|
} else {
|
|
let mut webrender_image_cache = self.webrender_image_cache.write();
|
|
webrender_image_cache.insert((url, use_placeholder), image_info);
|
|
Some(image_info)
|
|
}
|
|
},
|
|
None | Some(ImageOrMetadataAvailable::MetadataAvailable(_)) => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A registered painter
|
|
pub trait RegisteredPainter: RegisteredSpeculativePainter + Painter {}
|
|
|
|
/// A set of registered painters
|
|
pub trait RegisteredPainters: Sync {
|
|
/// Look up a painter
|
|
fn get(&self, name: &Atom) -> Option<&dyn RegisteredPainter>;
|
|
}
|