mirror of
https://github.com/servo/servo.git
synced 2025-08-03 12:40:06 +01:00
Add support for WebGL2 read and draw buffer settings
Adds support for the `ReadBuffer` and `DrawBuffers` WebGL2 calls and the related parameter getters. See: - https://www.khronos.org/registry/webgl/specs/latest/2.0/#3.7.2 - https://www.khronos.org/registry/webgl/specs/latest/2.0/#3.7.4 - https://www.khronos.org/registry/webgl/specs/latest/2.0/#3.7.11
This commit is contained in:
parent
e8e0b70266
commit
0afe27ef18
17 changed files with 315 additions and 51 deletions
|
@ -2024,6 +2024,8 @@ impl WebGLImpl {
|
|||
attach(attachment)
|
||||
}
|
||||
},
|
||||
WebGLCommand::ReadBuffer(buffer) => gl.read_buffer(buffer),
|
||||
WebGLCommand::DrawBuffers(ref buffers) => gl.draw_buffers(buffers),
|
||||
}
|
||||
|
||||
// If debug asertions are enabled, then check the error state.
|
||||
|
|
|
@ -546,6 +546,8 @@ pub enum WebGLCommand {
|
|||
InvalidateFramebuffer(u32, Vec<u32>),
|
||||
InvalidateSubFramebuffer(u32, Vec<u32>, i32, i32, i32, i32),
|
||||
FramebufferTextureLayer(u32, u32, Option<WebGLTextureId>, i32, i32),
|
||||
ReadBuffer(u32),
|
||||
DrawBuffers(Vec<u32>),
|
||||
}
|
||||
|
||||
macro_rules! nonzero_type {
|
||||
|
|
|
@ -96,6 +96,8 @@ pub struct WebGL2RenderingContext {
|
|||
texture_pack_skip_pixels: Cell<usize>,
|
||||
texture_pack_skip_rows: Cell<usize>,
|
||||
enable_rasterizer_discard: Cell<bool>,
|
||||
default_fb_readbuffer: Cell<u32>,
|
||||
default_fb_drawbuffer: Cell<u32>,
|
||||
}
|
||||
|
||||
fn typedarray_elem_size(typeid: Type) -> usize {
|
||||
|
@ -162,6 +164,8 @@ impl WebGL2RenderingContext {
|
|||
texture_pack_skip_pixels: Cell::new(0),
|
||||
texture_pack_skip_rows: Cell::new(0),
|
||||
enable_rasterizer_discard: Cell::new(false),
|
||||
default_fb_readbuffer: Cell::new(constants::BACK),
|
||||
default_fb_drawbuffer: Cell::new(constants::BACK),
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -344,6 +348,11 @@ impl WebGL2RenderingContext {
|
|||
return self.base.webgl_error(InvalidOperation);
|
||||
}
|
||||
|
||||
let fb_slot = self.base.get_draw_framebuffer_slot();
|
||||
if fb_slot.get().is_none() && self.default_fb_readbuffer.get() == constants::NONE {
|
||||
return self.base.webgl_error(InvalidOperation);
|
||||
}
|
||||
|
||||
let dst_byte_offset = {
|
||||
let dst_elem_size = typedarray_elem_size(dst.get_array_type());
|
||||
dst_elem_offset as usize * dst_elem_size
|
||||
|
@ -800,6 +809,26 @@ impl WebGL2RenderingContextMethods for WebGL2RenderingContext {
|
|||
&self.base.get_read_framebuffer_slot().get()
|
||||
);
|
||||
},
|
||||
constants::READ_BUFFER => {
|
||||
let buffer = match self.base.get_read_framebuffer_slot().get() {
|
||||
Some(fb) => fb.read_buffer(),
|
||||
None => self.default_fb_readbuffer.get(),
|
||||
};
|
||||
return UInt32Value(buffer);
|
||||
},
|
||||
constants::DRAW_BUFFER0..=constants::DRAW_BUFFER15 => {
|
||||
let buffer = match self.base.get_read_framebuffer_slot().get() {
|
||||
Some(fb) => {
|
||||
let idx = parameter - constants::DRAW_BUFFER0;
|
||||
fb.draw_buffer_i(idx as usize)
|
||||
},
|
||||
None if parameter == constants::DRAW_BUFFER0 => {
|
||||
self.default_fb_readbuffer.get()
|
||||
},
|
||||
None => constants::NONE,
|
||||
};
|
||||
return UInt32Value(buffer);
|
||||
},
|
||||
_ => {},
|
||||
}
|
||||
|
||||
|
@ -3811,6 +3840,46 @@ impl WebGL2RenderingContextMethods for WebGL2RenderingContext {
|
|||
self.base
|
||||
.renderbuffer_storage(target, samples, internal_format, width, height)
|
||||
}
|
||||
|
||||
/// https://www.khronos.org/registry/webgl/specs/latest/2.0/#3.7.4
|
||||
fn ReadBuffer(&self, src: u32) {
|
||||
match src {
|
||||
constants::BACK | constants::NONE => {},
|
||||
_ if self.base.valid_color_attachment_enum(src) => {},
|
||||
_ => return self.base.webgl_error(InvalidEnum),
|
||||
}
|
||||
|
||||
if let Some(fb) = self.base.get_read_framebuffer_slot().get() {
|
||||
handle_potential_webgl_error!(self.base, fb.set_read_buffer(src), return)
|
||||
} else {
|
||||
match src {
|
||||
constants::NONE | constants::BACK => {},
|
||||
_ => return self.base.webgl_error(InvalidOperation),
|
||||
}
|
||||
|
||||
self.default_fb_readbuffer.set(src);
|
||||
self.base.send_command(WebGLCommand::ReadBuffer(src));
|
||||
}
|
||||
}
|
||||
|
||||
/// https://www.khronos.org/registry/webgl/specs/latest/2.0/#3.7.11
|
||||
fn DrawBuffers(&self, buffers: Vec<u32>) {
|
||||
if let Some(fb) = self.base.get_draw_framebuffer_slot().get() {
|
||||
handle_potential_webgl_error!(self.base, fb.set_draw_buffers(buffers), return)
|
||||
} else {
|
||||
if buffers.len() != 1 {
|
||||
return self.base.webgl_error(InvalidOperation);
|
||||
}
|
||||
|
||||
match buffers[0] {
|
||||
constants::NONE | constants::BACK => {},
|
||||
_ => return self.base.webgl_error(InvalidOperation),
|
||||
}
|
||||
|
||||
self.default_fb_drawbuffer.set(buffers[0]);
|
||||
self.base.send_command(WebGLCommand::DrawBuffers(buffers));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl LayoutCanvasWebGLRenderingContextHelpers for LayoutDom<WebGL2RenderingContext> {
|
||||
|
|
|
@ -101,6 +101,8 @@ pub struct WebGLFramebuffer {
|
|||
depth: DomRefCell<Option<WebGLFramebufferAttachment>>,
|
||||
stencil: DomRefCell<Option<WebGLFramebufferAttachment>>,
|
||||
depthstencil: DomRefCell<Option<WebGLFramebufferAttachment>>,
|
||||
color_read_buffer: DomRefCell<u32>,
|
||||
color_draw_buffers: DomRefCell<Vec<u32>>,
|
||||
is_initialized: Cell<bool>,
|
||||
// Framebuffers for XR keep a reference to the XR session.
|
||||
// https://github.com/immersive-web/webxr/issues/856
|
||||
|
@ -121,6 +123,8 @@ impl WebGLFramebuffer {
|
|||
depth: DomRefCell::new(None),
|
||||
stencil: DomRefCell::new(None),
|
||||
depthstencil: DomRefCell::new(None),
|
||||
color_read_buffer: DomRefCell::new(constants::COLOR_ATTACHMENT0),
|
||||
color_draw_buffers: DomRefCell::new(vec![constants::COLOR_ATTACHMENT0]),
|
||||
is_initialized: Cell::new(false),
|
||||
xr_session: Default::default(),
|
||||
}
|
||||
|
@ -914,6 +918,55 @@ impl WebGLFramebuffer {
|
|||
});
|
||||
}
|
||||
|
||||
pub fn set_read_buffer(&self, buffer: u32) -> WebGLResult<()> {
|
||||
let context = self.upcast::<WebGLObject>().context();
|
||||
|
||||
match buffer {
|
||||
constants::NONE => {},
|
||||
_ if context.valid_color_attachment_enum(buffer) => {},
|
||||
_ => return Err(WebGLError::InvalidOperation),
|
||||
};
|
||||
|
||||
*self.color_read_buffer.borrow_mut() = buffer;
|
||||
context.send_command(WebGLCommand::ReadBuffer(buffer));
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn set_draw_buffers(&self, buffers: Vec<u32>) -> WebGLResult<()> {
|
||||
let context = self.upcast::<WebGLObject>().context();
|
||||
|
||||
if buffers.len() > context.limits().max_draw_buffers as usize {
|
||||
return Err(WebGLError::InvalidValue);
|
||||
}
|
||||
|
||||
let enums_valid = buffers
|
||||
.iter()
|
||||
.all(|&val| val == constants::NONE || context.valid_color_attachment_enum(val));
|
||||
if !enums_valid {
|
||||
return Err(WebGLError::InvalidEnum);
|
||||
}
|
||||
|
||||
let values_valid = buffers.iter().enumerate().all(|(i, &val)| {
|
||||
val == constants::NONE || val == (constants::COLOR_ATTACHMENT0 + i as u32)
|
||||
});
|
||||
if !values_valid {
|
||||
return Err(WebGLError::InvalidOperation);
|
||||
}
|
||||
|
||||
*self.color_draw_buffers.borrow_mut() = buffers.clone();
|
||||
context.send_command(WebGLCommand::DrawBuffers(buffers));
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn read_buffer(&self) -> u32 {
|
||||
*self.color_read_buffer.borrow()
|
||||
}
|
||||
|
||||
pub fn draw_buffer_i(&self, index: usize) -> u32 {
|
||||
let buffers = &*self.color_draw_buffers.borrow();
|
||||
*buffers.get(index).unwrap_or(&constants::NONE)
|
||||
}
|
||||
|
||||
pub fn target(&self) -> Option<u32> {
|
||||
self.target.get()
|
||||
}
|
||||
|
|
|
@ -1510,6 +1510,11 @@ impl WebGLRenderingContext {
|
|||
|
||||
// FIXME: https://github.com/servo/servo/issues/13710
|
||||
}
|
||||
|
||||
pub fn valid_color_attachment_enum(&self, attachment: u32) -> bool {
|
||||
let last_slot = constants::COLOR_ATTACHMENT0 + self.limits().max_color_attachments - 1;
|
||||
constants::COLOR_ATTACHMENT0 <= attachment && attachment <= last_slot
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(not(feature = "webgl_backtrace"))]
|
||||
|
|
|
@ -313,7 +313,7 @@ interface mixin WebGL2RenderingContextBase
|
|||
void invalidateFramebuffer(GLenum target, sequence<GLenum> attachments);
|
||||
void invalidateSubFramebuffer(GLenum target, sequence<GLenum> attachments,
|
||||
GLint x, GLint y, GLsizei width, GLsizei height);
|
||||
// void readBuffer(GLenum src);
|
||||
void readBuffer(GLenum src);
|
||||
|
||||
/* Renderbuffer objects */
|
||||
any getInternalformatParameter(GLenum target, GLenum internalformat, GLenum pname);
|
||||
|
@ -475,7 +475,7 @@ interface mixin WebGL2RenderingContextBase
|
|||
/*[AllowShared]*/ ArrayBufferView dstData, GLuint dstOffset);
|
||||
|
||||
/* Multiple Render Targets */
|
||||
// void drawBuffers(sequence<GLenum> buffers);
|
||||
void drawBuffers(sequence<GLenum> buffers);
|
||||
|
||||
void clearBufferfv(GLenum buffer, GLint drawbuffer, Float32List values,
|
||||
optional GLuint srcOffset = 0);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue