forked from mirrors/gecko-dev
<!-- Please describe your changes on the following line: --> Info about the big picture and the goals of the WebGL refactor in this thread: https://groups.google.com/forum/#!topic/mozilla.dev.servo/0WMGz60kKzQ I tried to reduce this PR as much as possible as requested in the thread. I'll do separate PRs for other features (e.g.: Batch messages or use shared memory to improve frame times) or fixes. Some tips to ease the review process: - Most changes in DOM objects follow the same pattern (remove CanvasMsg wrapper and use the new sender method). - WebGLCommands are the same ones as before (moved from webrender_api). So those lines are already reviewed. - See WebGL traits in [components/canvas_traits/webgl.rs](https://github.com/servo/servo/pull/17891/files#diff-8701045d01505418701d0631d4d45562) - See WebGLThread and WR External Image bridge in [components/canvas/webgl_thread.rs](https://github.com/servo/servo/pull/17891/files#diff-281554879f39a2a041f7a69d442a5d2e) - The implementation submitted in this PR creates a single `WebGLThread` for all ScriptThread/Pipelines. See that in [components/canvas/webgl_mode/inprocess.rs](https://github.com/servo/servo/pull/17891/files#diff-250070c6c5a38c7f9fa0f5b3c101f68b) The conformance tests will help to guarantee that we don't miss anything. --- <!-- Thank you for contributing to Servo! Please replace each `[ ]` by `[X]` when the step is complete, and replace `__` with appropriate data: --> - [x] `./mach build -d` does not report any errors - [x] `./mach test-tidy` does not report any errors - [ ] These changes fix #__ (github issue number if applicable). <!-- Either: --> - [x] There are tests for these changes OR - [ ] These changes do not require tests because _____ <!-- Also, please make sure that "Allow edits from maintainers" checkbox is checked, so that we can help you if you get stuck somewhere along the way.--> <!-- Pull requests that do not address these steps are welcome, but they will require additional verification as part of the review process. --> Source-Repo: https://github.com/servo/servo Source-Revision: 90f55ea4580e2a15f7d70d0491444f18b972d450 --HG-- rename : servo/tests/unit/style/rule_tree/mod.rs => servo/components/canvas/webgl_mode/mod.rs rename : servo/components/canvas_traits/lib.rs => servo/components/canvas_traits/canvas.rs extra : subtree_source : https%3A//hg.mozilla.org/projects/converted-servo-linear extra : subtree_revision : d476816b29986c4abfd61ff3c7b46aa189f2f50a
326 lines
11 KiB
Rust
326 lines
11 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/. */
|
|
|
|
// https://www.khronos.org/registry/webgl/specs/latest/1.0/webgl.idl
|
|
use canvas_traits::webgl::{WebGLCommand, WebGLError, WebGLMsgSender, WebGLParameter, WebGLProgramId, WebGLResult};
|
|
use canvas_traits::webgl::webgl_channel;
|
|
use dom::bindings::codegen::Bindings::WebGLProgramBinding;
|
|
use dom::bindings::codegen::Bindings::WebGLRenderingContextBinding::WebGLRenderingContextConstants as constants;
|
|
use dom::bindings::js::{MutNullableJS, Root};
|
|
use dom::bindings::reflector::{DomObject, reflect_dom_object};
|
|
use dom::bindings::str::DOMString;
|
|
use dom::webglactiveinfo::WebGLActiveInfo;
|
|
use dom::webglobject::WebGLObject;
|
|
use dom::webglrenderingcontext::MAX_UNIFORM_AND_ATTRIBUTE_LEN;
|
|
use dom::webglshader::WebGLShader;
|
|
use dom::window::Window;
|
|
use dom_struct::dom_struct;
|
|
use std::cell::Cell;
|
|
|
|
#[dom_struct]
|
|
pub struct WebGLProgram {
|
|
webgl_object: WebGLObject,
|
|
id: WebGLProgramId,
|
|
is_deleted: Cell<bool>,
|
|
link_called: Cell<bool>,
|
|
linked: Cell<bool>,
|
|
fragment_shader: MutNullableJS<WebGLShader>,
|
|
vertex_shader: MutNullableJS<WebGLShader>,
|
|
#[ignore_heap_size_of = "Defined in ipc-channel"]
|
|
renderer: WebGLMsgSender,
|
|
}
|
|
|
|
impl WebGLProgram {
|
|
fn new_inherited(renderer: WebGLMsgSender,
|
|
id: WebGLProgramId)
|
|
-> WebGLProgram {
|
|
WebGLProgram {
|
|
webgl_object: WebGLObject::new_inherited(),
|
|
id: id,
|
|
is_deleted: Cell::new(false),
|
|
link_called: Cell::new(false),
|
|
linked: Cell::new(false),
|
|
fragment_shader: Default::default(),
|
|
vertex_shader: Default::default(),
|
|
renderer: renderer,
|
|
}
|
|
}
|
|
|
|
pub fn maybe_new(window: &Window, renderer: WebGLMsgSender)
|
|
-> Option<Root<WebGLProgram>> {
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
renderer.send(WebGLCommand::CreateProgram(sender)).unwrap();
|
|
|
|
let result = receiver.recv().unwrap();
|
|
result.map(|program_id| WebGLProgram::new(window, renderer, program_id))
|
|
}
|
|
|
|
pub fn new(window: &Window,
|
|
renderer: WebGLMsgSender,
|
|
id: WebGLProgramId)
|
|
-> Root<WebGLProgram> {
|
|
reflect_dom_object(box WebGLProgram::new_inherited(renderer, id),
|
|
window,
|
|
WebGLProgramBinding::Wrap)
|
|
}
|
|
}
|
|
|
|
|
|
impl WebGLProgram {
|
|
pub fn id(&self) -> WebGLProgramId {
|
|
self.id
|
|
}
|
|
|
|
/// glDeleteProgram
|
|
pub fn delete(&self) {
|
|
if !self.is_deleted.get() {
|
|
self.is_deleted.set(true);
|
|
let _ = self.renderer.send(WebGLCommand::DeleteProgram(self.id));
|
|
|
|
if let Some(shader) = self.fragment_shader.get() {
|
|
shader.decrement_attached_counter();
|
|
}
|
|
|
|
if let Some(shader) = self.vertex_shader.get() {
|
|
shader.decrement_attached_counter();
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn is_deleted(&self) -> bool {
|
|
self.is_deleted.get()
|
|
}
|
|
|
|
pub fn is_linked(&self) -> bool {
|
|
self.linked.get()
|
|
}
|
|
|
|
/// glLinkProgram
|
|
pub fn link(&self) -> WebGLResult<()> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
self.linked.set(false);
|
|
self.link_called.set(true);
|
|
|
|
match self.fragment_shader.get() {
|
|
Some(ref shader) if shader.successfully_compiled() => {},
|
|
_ => return Ok(()), // callers use gl.LINK_STATUS to check link errors
|
|
}
|
|
|
|
match self.vertex_shader.get() {
|
|
Some(ref shader) if shader.successfully_compiled() => {},
|
|
_ => return Ok(()), // callers use gl.LINK_STATUS to check link errors
|
|
}
|
|
|
|
self.linked.set(true);
|
|
self.renderer.send(WebGLCommand::LinkProgram(self.id)).unwrap();
|
|
Ok(())
|
|
}
|
|
|
|
/// glUseProgram
|
|
pub fn use_program(&self) -> WebGLResult<()> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
if !self.linked.get() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
|
|
self.renderer.send(WebGLCommand::UseProgram(self.id)).unwrap();
|
|
Ok(())
|
|
}
|
|
|
|
/// glValidateProgram
|
|
pub fn validate(&self) -> WebGLResult<()> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
self.renderer.send(WebGLCommand::ValidateProgram(self.id)).unwrap();
|
|
Ok(())
|
|
}
|
|
|
|
/// glAttachShader
|
|
pub fn attach_shader(&self, shader: &WebGLShader) -> WebGLResult<()> {
|
|
if self.is_deleted() || shader.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
let shader_slot = match shader.gl_type() {
|
|
constants::FRAGMENT_SHADER => &self.fragment_shader,
|
|
constants::VERTEX_SHADER => &self.vertex_shader,
|
|
_ => {
|
|
error!("detachShader: Unexpected shader type");
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
};
|
|
|
|
// TODO(emilio): Differentiate between same shader already assigned and other previous
|
|
// shader.
|
|
if shader_slot.get().is_some() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
|
|
shader_slot.set(Some(shader));
|
|
shader.increment_attached_counter();
|
|
|
|
self.renderer.send(WebGLCommand::AttachShader(self.id, shader.id())).unwrap();
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// glDetachShader
|
|
pub fn detach_shader(&self, shader: &WebGLShader) -> WebGLResult<()> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
let shader_slot = match shader.gl_type() {
|
|
constants::FRAGMENT_SHADER => &self.fragment_shader,
|
|
constants::VERTEX_SHADER => &self.vertex_shader,
|
|
_ => {
|
|
error!("detachShader: Unexpected shader type");
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
};
|
|
|
|
match shader_slot.get() {
|
|
Some(ref attached_shader) if attached_shader.id() != shader.id() =>
|
|
return Err(WebGLError::InvalidOperation),
|
|
None =>
|
|
return Err(WebGLError::InvalidOperation),
|
|
_ => {}
|
|
}
|
|
|
|
shader_slot.set(None);
|
|
shader.decrement_attached_counter();
|
|
|
|
self.renderer.send(WebGLCommand::DetachShader(self.id, shader.id())).unwrap();
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// glBindAttribLocation
|
|
pub fn bind_attrib_location(&self, index: u32, name: DOMString) -> WebGLResult<()> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
if name.len() > MAX_UNIFORM_AND_ATTRIBUTE_LEN {
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
|
|
// Check if the name is reserved
|
|
if name.starts_with("gl_") || name.starts_with("webgl") || name.starts_with("_webgl_") {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
|
|
self.renderer
|
|
.send(WebGLCommand::BindAttribLocation(self.id, index, String::from(name)))
|
|
.unwrap();
|
|
Ok(())
|
|
}
|
|
|
|
pub fn get_active_uniform(&self, index: u32) -> WebGLResult<Root<WebGLActiveInfo>> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
self.renderer
|
|
.send(WebGLCommand::GetActiveUniform(self.id, index, sender))
|
|
.unwrap();
|
|
|
|
receiver.recv().unwrap().map(|(size, ty, name)|
|
|
WebGLActiveInfo::new(self.global().as_window(), size, ty, DOMString::from(name)))
|
|
}
|
|
|
|
/// glGetActiveAttrib
|
|
pub fn get_active_attrib(&self, index: u32) -> WebGLResult<Root<WebGLActiveInfo>> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
self.renderer
|
|
.send(WebGLCommand::GetActiveAttrib(self.id, index, sender))
|
|
.unwrap();
|
|
|
|
receiver.recv().unwrap().map(|(size, ty, name)|
|
|
WebGLActiveInfo::new(self.global().as_window(), size, ty, DOMString::from(name)))
|
|
}
|
|
|
|
/// glGetAttribLocation
|
|
pub fn get_attrib_location(&self, name: DOMString) -> WebGLResult<Option<i32>> {
|
|
if !self.is_linked() || self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
if name.len() > MAX_UNIFORM_AND_ATTRIBUTE_LEN {
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
|
|
// Check if the name is reserved
|
|
if name.starts_with("gl_") {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
|
|
if name.starts_with("webgl") || name.starts_with("_webgl_") {
|
|
return Ok(None);
|
|
}
|
|
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
self.renderer
|
|
.send(WebGLCommand::GetAttribLocation(self.id, String::from(name), sender))
|
|
.unwrap();
|
|
Ok(receiver.recv().unwrap())
|
|
}
|
|
|
|
/// glGetUniformLocation
|
|
pub fn get_uniform_location(&self, name: DOMString) -> WebGLResult<Option<i32>> {
|
|
if !self.is_linked() || self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
if name.len() > MAX_UNIFORM_AND_ATTRIBUTE_LEN {
|
|
return Err(WebGLError::InvalidValue);
|
|
}
|
|
|
|
// Check if the name is reserved
|
|
if name.starts_with("webgl") || name.starts_with("_webgl_") {
|
|
return Ok(None);
|
|
}
|
|
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
self.renderer
|
|
.send(WebGLCommand::GetUniformLocation(self.id, String::from(name), sender))
|
|
.unwrap();
|
|
Ok(receiver.recv().unwrap())
|
|
}
|
|
|
|
/// glGetProgramInfoLog
|
|
pub fn get_info_log(&self) -> WebGLResult<String> {
|
|
if self.is_deleted() {
|
|
return Err(WebGLError::InvalidOperation);
|
|
}
|
|
if self.link_called.get() {
|
|
let shaders_compiled = match (self.fragment_shader.get(), self.vertex_shader.get()) {
|
|
(Some(fs), Some(vs)) => fs.successfully_compiled() && vs.successfully_compiled(),
|
|
_ => false
|
|
};
|
|
if !shaders_compiled {
|
|
return Ok("One or more shaders failed to compile".to_string());
|
|
}
|
|
}
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
self.renderer.send(WebGLCommand::GetProgramInfoLog(self.id, sender)).unwrap();
|
|
Ok(receiver.recv().unwrap())
|
|
}
|
|
|
|
/// glGetProgramParameter
|
|
pub fn parameter(&self, param_id: u32) -> WebGLResult<WebGLParameter> {
|
|
let (sender, receiver) = webgl_channel().unwrap();
|
|
self.renderer.send(WebGLCommand::GetProgramParameter(self.id, param_id, sender)).unwrap();
|
|
receiver.recv().unwrap()
|
|
}
|
|
}
|
|
|
|
impl Drop for WebGLProgram {
|
|
fn drop(&mut self) {
|
|
self.delete();
|
|
}
|
|
}
|