mirror of
https://github.com/servo/servo.git
synced 2025-06-10 01:23:13 +00:00
Each canvas context returns snapshot instead of just raw bytes. This allows as to hold off conversions (BGRA <-> RGBA, (un)premultiply) to when/if they are actually needed. For example when loading snapshot into webgl we can load both RGBA and BGRA so no conversion is really needed. Currently whole thing is designed to be able to be extend on https://github.com/servo/ipc-channel/pull/356, to make less copies. Hence some commented out code. Fixes #35759 There are tests for these changes in WPT --------- Signed-off-by: sagudev <16504129+sagudev@users.noreply.github.com>
410 lines
15 KiB
Rust
410 lines
15 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/. */
|
|
|
|
use std::borrow::Cow;
|
|
use std::cell::RefCell;
|
|
|
|
use arrayvec::ArrayVec;
|
|
use dom_struct::dom_struct;
|
|
use ipc_channel::ipc::{self};
|
|
use script_layout_interface::HTMLCanvasDataSource;
|
|
use snapshot::Snapshot;
|
|
use webgpu_traits::{
|
|
ContextConfiguration, PRESENTATION_BUFFER_COUNT, WebGPU, WebGPUContextId, WebGPURequest,
|
|
WebGPUTexture,
|
|
};
|
|
use webrender_api::ImageKey;
|
|
use webrender_api::units::DeviceIntSize;
|
|
use wgpu_core::id;
|
|
|
|
use super::gpuconvert::convert_texture_descriptor;
|
|
use super::gputexture::GPUTexture;
|
|
use crate::canvas_context::{CanvasContext, CanvasHelpers};
|
|
use crate::conversions::Convert;
|
|
use crate::dom::bindings::codegen::Bindings::GPUCanvasContextBinding::GPUCanvasContextMethods;
|
|
use crate::dom::bindings::codegen::Bindings::WebGPUBinding::GPUTexture_Binding::GPUTextureMethods;
|
|
use crate::dom::bindings::codegen::Bindings::WebGPUBinding::{
|
|
GPUCanvasAlphaMode, GPUCanvasConfiguration, GPUDeviceMethods, GPUExtent3D, GPUExtent3DDict,
|
|
GPUObjectDescriptorBase, GPUTextureDescriptor, GPUTextureDimension, GPUTextureFormat,
|
|
GPUTextureUsageConstants,
|
|
};
|
|
use crate::dom::bindings::codegen::UnionTypes::HTMLCanvasElementOrOffscreenCanvas;
|
|
use crate::dom::bindings::error::{Error, Fallible};
|
|
use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object};
|
|
use crate::dom::bindings::root::{DomRoot, LayoutDom, MutNullableDom};
|
|
use crate::dom::bindings::str::USVString;
|
|
use crate::dom::bindings::weakref::WeakRef;
|
|
use crate::dom::document::WebGPUContextsMap;
|
|
use crate::dom::globalscope::GlobalScope;
|
|
use crate::dom::htmlcanvaselement::{HTMLCanvasElement, LayoutCanvasRenderingContextHelpers};
|
|
use crate::dom::node::NodeTraits;
|
|
use crate::script_runtime::CanGc;
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#supported-context-formats>
|
|
fn supported_context_format(format: GPUTextureFormat) -> bool {
|
|
// TODO: GPUTextureFormat::Rgba16float
|
|
matches!(
|
|
format,
|
|
GPUTextureFormat::Bgra8unorm | GPUTextureFormat::Rgba8unorm
|
|
)
|
|
}
|
|
|
|
#[derive(Clone, Debug, Default, JSTraceable, MallocSizeOf)]
|
|
/// Helps observe changes on swapchain
|
|
struct DrawingBuffer {
|
|
#[no_trace]
|
|
size: DeviceIntSize,
|
|
/// image is transparent black
|
|
cleared: bool,
|
|
#[ignore_malloc_size_of = "Defined in wgpu"]
|
|
#[no_trace]
|
|
config: Option<ContextConfiguration>,
|
|
}
|
|
|
|
#[dom_struct]
|
|
pub(crate) struct GPUCanvasContext {
|
|
reflector_: Reflector,
|
|
#[ignore_malloc_size_of = "channels are hard"]
|
|
#[no_trace]
|
|
channel: WebGPU,
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-canvas>
|
|
canvas: HTMLCanvasElementOrOffscreenCanvas,
|
|
// TODO: can we have wgpu surface that is hw accelerated inside wr ...
|
|
#[ignore_malloc_size_of = "Defined in webrender"]
|
|
#[no_trace]
|
|
webrender_image: ImageKey,
|
|
#[no_trace]
|
|
context_id: WebGPUContextId,
|
|
#[ignore_malloc_size_of = "manual writing is hard"]
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-configuration-slot>
|
|
configuration: RefCell<Option<GPUCanvasConfiguration>>,
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-texturedescriptor-slot>
|
|
texture_descriptor: RefCell<Option<GPUTextureDescriptor>>,
|
|
/// Conceptually <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-drawingbuffer-slot>
|
|
drawing_buffer: RefCell<DrawingBuffer>,
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-currenttexture-slot>
|
|
current_texture: MutNullableDom<GPUTexture>,
|
|
/// This is used for clearing
|
|
#[ignore_malloc_size_of = "Rc are hard"]
|
|
webgpu_contexts: WebGPUContextsMap,
|
|
}
|
|
|
|
impl GPUCanvasContext {
|
|
fn new_inherited(
|
|
global: &GlobalScope,
|
|
canvas: HTMLCanvasElementOrOffscreenCanvas,
|
|
channel: WebGPU,
|
|
webgpu_contexts: WebGPUContextsMap,
|
|
) -> Self {
|
|
let (sender, receiver) = ipc::channel().unwrap();
|
|
let size = canvas.size().cast().cast_unit();
|
|
let mut buffer_ids = ArrayVec::<id::BufferId, PRESENTATION_BUFFER_COUNT>::new();
|
|
for _ in 0..PRESENTATION_BUFFER_COUNT {
|
|
buffer_ids.push(global.wgpu_id_hub().create_buffer_id());
|
|
}
|
|
if let Err(e) = channel.0.send(WebGPURequest::CreateContext {
|
|
buffer_ids,
|
|
size,
|
|
sender,
|
|
}) {
|
|
warn!("Failed to send CreateContext ({:?})", e);
|
|
}
|
|
let (external_id, webrender_image) = receiver.recv().unwrap();
|
|
Self {
|
|
reflector_: Reflector::new(),
|
|
channel,
|
|
canvas,
|
|
webrender_image,
|
|
context_id: WebGPUContextId(external_id.0),
|
|
drawing_buffer: RefCell::new(DrawingBuffer {
|
|
size,
|
|
cleared: true,
|
|
..Default::default()
|
|
}),
|
|
configuration: RefCell::new(None),
|
|
texture_descriptor: RefCell::new(None),
|
|
current_texture: MutNullableDom::default(),
|
|
webgpu_contexts,
|
|
}
|
|
}
|
|
|
|
pub(crate) fn new(
|
|
global: &GlobalScope,
|
|
canvas: &HTMLCanvasElement,
|
|
channel: WebGPU,
|
|
can_gc: CanGc,
|
|
) -> DomRoot<Self> {
|
|
let document = canvas.owner_document();
|
|
let this = reflect_dom_object(
|
|
Box::new(GPUCanvasContext::new_inherited(
|
|
global,
|
|
HTMLCanvasElementOrOffscreenCanvas::HTMLCanvasElement(DomRoot::from_ref(canvas)),
|
|
channel,
|
|
document.webgpu_contexts(),
|
|
)),
|
|
global,
|
|
can_gc,
|
|
);
|
|
this.webgpu_contexts
|
|
.borrow_mut()
|
|
.entry(this.context_id())
|
|
.or_insert_with(|| WeakRef::new(&this));
|
|
this
|
|
}
|
|
}
|
|
|
|
// Abstract ops from spec
|
|
impl GPUCanvasContext {
|
|
/// <https://gpuweb.github.io/gpuweb/#abstract-opdef-gputexturedescriptor-for-the-canvas-and-configuration>
|
|
fn texture_descriptor_for_canvas(
|
|
&self,
|
|
configuration: &GPUCanvasConfiguration,
|
|
) -> GPUTextureDescriptor {
|
|
let size = self.size();
|
|
GPUTextureDescriptor {
|
|
format: configuration.format,
|
|
// We need to add `COPY_SRC` so we can copy texture to presentation buffer
|
|
// causes FAIL on webgpu:web_platform,canvas,configure:usage:*
|
|
usage: configuration.usage | GPUTextureUsageConstants::COPY_SRC,
|
|
size: GPUExtent3D::GPUExtent3DDict(GPUExtent3DDict {
|
|
width: size.width as u32,
|
|
height: size.height as u32,
|
|
depthOrArrayLayers: 1,
|
|
}),
|
|
viewFormats: configuration.viewFormats.clone(),
|
|
// other members to default
|
|
mipLevelCount: 1,
|
|
sampleCount: 1,
|
|
parent: GPUObjectDescriptorBase {
|
|
label: USVString::default(),
|
|
},
|
|
dimension: GPUTextureDimension::_2d,
|
|
}
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#abstract-opdef-expire-the-current-texture>
|
|
fn expire_current_texture(&self) {
|
|
if let Some(current_texture) = self.current_texture.take() {
|
|
// Make copy of texture content
|
|
self.send_swap_chain_present(current_texture.id());
|
|
// Step 1
|
|
current_texture.Destroy()
|
|
}
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#abstract-opdef-replace-the-drawing-buffer>
|
|
fn replace_drawing_buffer(&self) {
|
|
// Step 1
|
|
self.expire_current_texture();
|
|
// Step 2
|
|
let configuration = self.configuration.borrow();
|
|
// Step 3
|
|
let mut drawing_buffer = self.drawing_buffer.borrow_mut();
|
|
drawing_buffer.size = self.size().cast().cast_unit();
|
|
drawing_buffer.cleared = true;
|
|
if let Some(configuration) = configuration.as_ref() {
|
|
drawing_buffer.config = Some(ContextConfiguration {
|
|
device_id: configuration.device.id().0,
|
|
queue_id: configuration.device.queue_id().0,
|
|
format: configuration.format.convert(),
|
|
is_opaque: matches!(configuration.alphaMode, GPUCanvasAlphaMode::Opaque),
|
|
});
|
|
} else {
|
|
drawing_buffer.config.take();
|
|
};
|
|
// TODO: send less
|
|
self.channel
|
|
.0
|
|
.send(WebGPURequest::UpdateContext {
|
|
context_id: self.context_id,
|
|
size: drawing_buffer.size,
|
|
configuration: drawing_buffer.config,
|
|
})
|
|
.expect("Failed to update webgpu context");
|
|
}
|
|
}
|
|
|
|
// Internal helper methods
|
|
impl GPUCanvasContext {
|
|
fn layout_handle(&self) -> HTMLCanvasDataSource {
|
|
if self.drawing_buffer.borrow().cleared {
|
|
HTMLCanvasDataSource::Empty
|
|
} else {
|
|
HTMLCanvasDataSource::WebGPU(self.webrender_image)
|
|
}
|
|
}
|
|
|
|
fn send_swap_chain_present(&self, texture_id: WebGPUTexture) {
|
|
self.drawing_buffer.borrow_mut().cleared = false;
|
|
let encoder_id = self.global().wgpu_id_hub().create_command_encoder_id();
|
|
if let Err(e) = self.channel.0.send(WebGPURequest::SwapChainPresent {
|
|
context_id: self.context_id,
|
|
texture_id: texture_id.0,
|
|
encoder_id,
|
|
}) {
|
|
warn!(
|
|
"Failed to send UpdateWebrenderData({:?}) ({})",
|
|
self.context_id, e
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
impl CanvasContext for GPUCanvasContext {
|
|
type ID = WebGPUContextId;
|
|
|
|
fn context_id(&self) -> WebGPUContextId {
|
|
self.context_id
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#abstract-opdef-updating-the-rendering-of-a-webgpu-canvas>
|
|
fn update_rendering(&self) {
|
|
// Step 1
|
|
self.expire_current_texture();
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#abstract-opdef-update-the-canvas-size>
|
|
fn resize(&self) {
|
|
// Step 1
|
|
self.replace_drawing_buffer();
|
|
// Step 2
|
|
let configuration = self.configuration.borrow();
|
|
// Step 3
|
|
if let Some(configuration) = configuration.as_ref() {
|
|
self.texture_descriptor
|
|
.replace(Some(self.texture_descriptor_for_canvas(configuration)));
|
|
}
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#ref-for-abstract-opdef-get-a-copy-of-the-image-contents-of-a-context%E2%91%A5>
|
|
fn get_image_data(&self) -> Option<Snapshot> {
|
|
// 1. Return a copy of the image contents of context.
|
|
Some(if self.drawing_buffer.borrow().cleared {
|
|
Snapshot::cleared(self.size())
|
|
} else {
|
|
let (sender, receiver) = ipc::channel().unwrap();
|
|
self.channel
|
|
.0
|
|
.send(WebGPURequest::GetImage {
|
|
context_id: self.context_id,
|
|
sender,
|
|
})
|
|
.unwrap();
|
|
receiver.recv().unwrap().to_owned()
|
|
})
|
|
}
|
|
|
|
fn canvas(&self) -> HTMLCanvasElementOrOffscreenCanvas {
|
|
self.canvas.clone()
|
|
}
|
|
}
|
|
|
|
impl LayoutCanvasRenderingContextHelpers for LayoutDom<'_, GPUCanvasContext> {
|
|
fn canvas_data_source(self) -> HTMLCanvasDataSource {
|
|
(*self.unsafe_get()).layout_handle()
|
|
}
|
|
}
|
|
|
|
impl GPUCanvasContextMethods<crate::DomTypeHolder> for GPUCanvasContext {
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-canvas>
|
|
fn Canvas(&self) -> HTMLCanvasElementOrOffscreenCanvas {
|
|
self.canvas.clone()
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-configure>
|
|
fn Configure(&self, configuration: &GPUCanvasConfiguration) -> Fallible<()> {
|
|
// Step 1: Let device be configuration.device
|
|
let device = &configuration.device;
|
|
|
|
// Step 5: Let descriptor be the GPUTextureDescriptor for the canvas and configuration.
|
|
let descriptor = self.texture_descriptor_for_canvas(configuration);
|
|
|
|
// Step 2&3: Validate texture format required features
|
|
let (mut desc, _) = convert_texture_descriptor(&descriptor, device)?;
|
|
desc.label = Some(Cow::Borrowed(
|
|
"dummy texture for texture descriptor validation",
|
|
));
|
|
|
|
// Step 4: If Supported context formats does not contain configuration.format, throw a TypeError
|
|
if !supported_context_format(configuration.format) {
|
|
return Err(Error::Type(format!(
|
|
"Unsupported context format: {:?}",
|
|
configuration.format
|
|
)));
|
|
}
|
|
|
|
// Step 5
|
|
self.configuration.replace(Some(configuration.clone()));
|
|
|
|
// Step 6
|
|
self.texture_descriptor.replace(Some(descriptor));
|
|
|
|
// Step 7
|
|
self.replace_drawing_buffer();
|
|
|
|
// Step 8: Validate texture descriptor
|
|
let texture_id = self.global().wgpu_id_hub().create_texture_id();
|
|
self.channel
|
|
.0
|
|
.send(WebGPURequest::ValidateTextureDescriptor {
|
|
device_id: device.id().0,
|
|
texture_id,
|
|
descriptor: desc,
|
|
})
|
|
.expect("Failed to create WebGPU SwapChain");
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-unconfigure>
|
|
fn Unconfigure(&self) {
|
|
// Step 1
|
|
self.configuration.take();
|
|
// Step 2
|
|
self.current_texture.take();
|
|
// Step 3
|
|
self.replace_drawing_buffer();
|
|
}
|
|
|
|
/// <https://gpuweb.github.io/gpuweb/#dom-gpucanvascontext-getcurrenttexture>
|
|
fn GetCurrentTexture(&self) -> Fallible<DomRoot<GPUTexture>> {
|
|
// Step 1: If this.[[configuration]] is null, throw an InvalidStateError and return.
|
|
let configuration = self.configuration.borrow();
|
|
let Some(configuration) = configuration.as_ref() else {
|
|
return Err(Error::InvalidState);
|
|
};
|
|
// Step 2: Assert this.[[textureDescriptor]] is not null.
|
|
let texture_descriptor = self.texture_descriptor.borrow();
|
|
let texture_descriptor = texture_descriptor.as_ref().unwrap();
|
|
// Step 6
|
|
let current_texture = if let Some(current_texture) = self.current_texture.get() {
|
|
current_texture
|
|
} else {
|
|
// Step 4.1
|
|
self.replace_drawing_buffer();
|
|
// Step 4.2
|
|
let current_texture = configuration.device.CreateTexture(texture_descriptor)?;
|
|
self.current_texture.set(Some(¤t_texture));
|
|
// We only need to mark new texture
|
|
self.mark_as_dirty();
|
|
current_texture
|
|
};
|
|
// Step 6
|
|
Ok(current_texture)
|
|
}
|
|
}
|
|
|
|
impl Drop for GPUCanvasContext {
|
|
fn drop(&mut self) {
|
|
self.webgpu_contexts.borrow_mut().remove(&self.context_id());
|
|
if let Err(e) = self.channel.0.send(WebGPURequest::DestroyContext {
|
|
context_id: self.context_id,
|
|
}) {
|
|
warn!(
|
|
"Failed to send DestroySwapChain-ImageKey({:?}) ({})",
|
|
self.webrender_image, e
|
|
);
|
|
}
|
|
}
|
|
}
|