aboutsummaryrefslogtreecommitdiffstats
path: root/components
diff options
context:
space:
mode:
Diffstat (limited to 'components')
-rw-r--r--components/net/http_loader.rs44
-rw-r--r--components/net_traits/lib.rs13
-rw-r--r--components/script/dom/bindings/error.rs3
-rw-r--r--components/script/dom/bindings/trace.rs11
-rw-r--r--components/script/dom/domexception.rs15
-rw-r--r--components/script/dom/fakexrdevice.rs12
-rw-r--r--components/script/dom/performanceresourcetiming.rs9
-rw-r--r--components/script/dom/webglrenderingcontext.rs4
-rw-r--r--components/script/dom/webidls/DOMException.webidl1
-rw-r--r--components/script/dom/webidls/PerformanceResourceTiming.webidl2
-rw-r--r--components/script/dom/webidls/XRWebGLLayer.webidl6
-rw-r--r--components/script/dom/xrsession.rs2
-rw-r--r--components/script/dom/xrviewport.rs32
-rw-r--r--components/script/dom/xrwebgllayer.rs126
14 files changed, 216 insertions, 64 deletions
diff --git a/components/net/http_loader.rs b/components/net/http_loader.rs
index 3080221f356..dfae8624041 100644
--- a/components/net/http_loader.rs
+++ b/components/net/http_loader.rs
@@ -42,7 +42,7 @@ use net_traits::request::{RedirectMode, Referrer, Request, RequestMode};
use net_traits::request::{ResponseTainting, ServiceWorkersMode};
use net_traits::response::{HttpsState, Response, ResponseBody, ResponseType};
use net_traits::{CookieSource, FetchMetadata, NetworkError, ReferrerPolicy};
-use net_traits::{RedirectStartValue, ResourceAttribute, ResourceFetchTiming};
+use net_traits::{RedirectEndValue, RedirectStartValue, ResourceAttribute, ResourceFetchTiming};
use openssl::ssl::SslConnectorBuilder;
use servo_url::{ImmutableOrigin, ServoUrl};
use std::collections::{HashMap, HashSet};
@@ -621,8 +621,6 @@ pub fn http_fetch(
};
}
- // TODO redirect_end: last byte of response of last redirect
-
// set back to default
response.return_internal = true;
context
@@ -639,6 +637,27 @@ pub fn http_fetch(
response
}
+// Convenience struct that implements Drop, for setting redirectEnd on function return
+struct RedirectEndTimer(Option<Arc<Mutex<ResourceFetchTiming>>>);
+
+impl RedirectEndTimer {
+ fn neuter(&mut self) {
+ self.0 = None;
+ }
+}
+
+impl Drop for RedirectEndTimer {
+ fn drop(&mut self) {
+ let RedirectEndTimer(resource_fetch_timing_opt) = self;
+
+ resource_fetch_timing_opt.as_ref().map_or((), |t| {
+ t.lock()
+ .unwrap()
+ .set_attribute(ResourceAttribute::RedirectEnd(RedirectEndValue::Zero));
+ })
+ }
+}
+
/// [HTTP redirect fetch](https://fetch.spec.whatwg.org#http-redirect-fetch)
pub fn http_redirect_fetch(
request: &mut Request,
@@ -649,6 +668,8 @@ pub fn http_redirect_fetch(
done_chan: &mut DoneChannel,
context: &FetchContext,
) -> Response {
+ let mut redirect_end_timer = RedirectEndTimer(Some(context.timing.clone()));
+
// Step 1
assert!(response.return_internal);
@@ -761,7 +782,7 @@ pub fn http_redirect_fetch(
// Step 15
let recursive_flag = request.redirect_mode != RedirectMode::Manual;
- main_fetch(
+ let fetch_response = main_fetch(
request,
cache,
cors_flag,
@@ -769,7 +790,19 @@ pub fn http_redirect_fetch(
target,
done_chan,
context,
- )
+ );
+
+ // TODO: timing allow check
+ context
+ .timing
+ .lock()
+ .unwrap()
+ .set_attribute(ResourceAttribute::RedirectEnd(
+ RedirectEndValue::ResponseEnd,
+ ));
+ redirect_end_timer.neuter();
+
+ fetch_response
}
fn try_immutable_origin_to_hyper_origin(url_origin: &ImmutableOrigin) -> Option<HyperOrigin> {
@@ -1197,6 +1230,7 @@ fn http_network_fetch(
context: &FetchContext,
) -> Response {
let mut response_end_timer = ResponseEndTimer(Some(context.timing.clone()));
+
// Step 1
// nothing to do here, since credentials_flag is already a boolean
diff --git a/components/net_traits/lib.rs b/components/net_traits/lib.rs
index 566a3e1093a..b30ed9e6877 100644
--- a/components/net_traits/lib.rs
+++ b/components/net_traits/lib.rs
@@ -450,7 +450,7 @@ pub struct ResourceFetchTiming {
pub fetch_start: u64,
pub response_end: u64,
pub redirect_start: u64,
- // pub redirect_end: u64,
+ pub redirect_end: u64,
pub connect_start: u64,
pub connect_end: u64,
}
@@ -461,12 +461,18 @@ pub enum RedirectStartValue {
FetchStart,
}
+pub enum RedirectEndValue {
+ Zero,
+ ResponseEnd,
+}
+
pub enum ResourceAttribute {
RedirectCount(u16),
DomainLookupStart,
RequestStart,
ResponseStart,
RedirectStart(RedirectStartValue),
+ RedirectEnd(RedirectEndValue),
FetchStart,
ConnectStart(u64),
ConnectEnd(u64),
@@ -491,6 +497,7 @@ impl ResourceFetchTiming {
response_start: 0,
fetch_start: 0,
redirect_start: 0,
+ redirect_end: 0,
connect_start: 0,
connect_end: 0,
response_end: 0,
@@ -513,6 +520,10 @@ impl ResourceFetchTiming {
}
},
},
+ ResourceAttribute::RedirectEnd(val) => match val {
+ RedirectEndValue::Zero => self.redirect_end = 0,
+ RedirectEndValue::ResponseEnd => self.redirect_end = self.response_end,
+ },
ResourceAttribute::FetchStart => self.fetch_start = precise_time_ns(),
ResourceAttribute::ConnectStart(val) => self.connect_start = val,
ResourceAttribute::ConnectEnd(val) => self.connect_end = val,
diff --git a/components/script/dom/bindings/error.rs b/components/script/dom/bindings/error.rs
index 5953a013d6d..40d690ff8d0 100644
--- a/components/script/dom/bindings/error.rs
+++ b/components/script/dom/bindings/error.rs
@@ -84,6 +84,8 @@ pub enum Error {
InvalidModification,
/// NotReadableError DOMException
NotReadable,
+ /// OperationError DOMException
+ Operation,
/// TypeError JavaScript Error
Type(String),
@@ -136,6 +138,7 @@ pub unsafe fn throw_dom_exception(cx: *mut JSContext, global: &GlobalScope, resu
Error::TypeMismatch => DOMErrorName::TypeMismatchError,
Error::InvalidModification => DOMErrorName::InvalidModificationError,
Error::NotReadable => DOMErrorName::NotReadableError,
+ Error::Operation => DOMErrorName::OperationError,
Error::Type(message) => {
assert!(!JS_IsExceptionPending(cx));
throw_type_error(cx, &message);
diff --git a/components/script/dom/bindings/trace.rs b/components/script/dom/bindings/trace.rs
index a0307413719..7f4fad813e5 100644
--- a/components/script/dom/bindings/trace.rs
+++ b/components/script/dom/bindings/trace.rs
@@ -57,8 +57,8 @@ use devtools_traits::{CSSError, TimelineMarkerType, WorkerId};
use encoding_rs::{Decoder, Encoding};
use euclid::Length as EuclidLength;
use euclid::{
- Point2D, Rect, RigidTransform3D, Rotation3D, Transform2D, Transform3D, TypedRigidTransform3D,
- TypedScale, TypedSize2D, Vector2D,
+ Point2D, Rect, RigidTransform3D, Rotation3D, Transform2D, Transform3D, TypedRect,
+ TypedRigidTransform3D, TypedScale, TypedSize2D, Vector2D,
};
use html5ever::buffer_queue::BufferQueue;
use html5ever::{LocalName, Namespace, Prefix, QualName};
@@ -647,6 +647,13 @@ unsafe impl<U> JSTraceable for TypedSize2D<u32, U> {
}
}
+unsafe impl<U> JSTraceable for TypedRect<i32, U> {
+ #[inline]
+ unsafe fn trace(&self, _trc: *mut JSTracer) {
+ // Do nothing
+ }
+}
+
unsafe impl JSTraceable for StyleLocked<FontFaceRule> {
unsafe fn trace(&self, _trc: *mut JSTracer) {
// Do nothing.
diff --git a/components/script/dom/domexception.rs b/components/script/dom/domexception.rs
index b5b0ca5a8c4..3cfe6092dd1 100644
--- a/components/script/dom/domexception.rs
+++ b/components/script/dom/domexception.rs
@@ -13,7 +13,7 @@ use crate::dom::globalscope::GlobalScope;
use dom_struct::dom_struct;
#[repr(u16)]
-#[derive(Clone, Copy, Debug, JSTraceable, MallocSizeOf)]
+#[derive(Clone, Copy, Debug, Eq, JSTraceable, MallocSizeOf, Ord, PartialEq, PartialOrd)]
pub enum DOMErrorName {
IndexSizeError = DOMExceptionConstants::INDEX_SIZE_ERR,
HierarchyRequestError = DOMExceptionConstants::HIERARCHY_REQUEST_ERR,
@@ -36,7 +36,8 @@ pub enum DOMErrorName {
TimeoutError = DOMExceptionConstants::TIMEOUT_ERR,
InvalidNodeTypeError = DOMExceptionConstants::INVALID_NODE_TYPE_ERR,
DataCloneError = DOMExceptionConstants::DATA_CLONE_ERR,
- NotReadableError = DOMExceptionConstants::NOT_READABLE_ERR,
+ NotReadableError,
+ OperationError,
}
impl DOMErrorName {
@@ -64,6 +65,7 @@ impl DOMErrorName {
"InvalidNodeTypeError" => Some(DOMErrorName::InvalidNodeTypeError),
"DataCloneError" => Some(DOMErrorName::DataCloneError),
"NotReadableError" => Some(DOMErrorName::NotReadableError),
+ "OperationError" => Some(DOMErrorName::OperationError),
_ => None,
}
}
@@ -107,6 +109,9 @@ impl DOMException {
},
DOMErrorName::DataCloneError => "The object can not be cloned.",
DOMErrorName::NotReadableError => "The I/O read operation failed.",
+ DOMErrorName::OperationError => {
+ "The operation failed for an operation-specific reason."
+ },
};
(
@@ -147,11 +152,11 @@ impl DOMException {
}
impl DOMExceptionMethods for DOMException {
- // https://heycam.github.io/webidl/#dfn-DOMException
+ // https://heycam.github.io/webidl/#dom-domexception-code
fn Code(&self) -> u16 {
match DOMErrorName::from(&self.name) {
- Some(code) => code as u16,
- None => 0 as u16,
+ Some(code) if code <= DOMErrorName::DataCloneError => code as u16,
+ _ => 0,
}
}
diff --git a/components/script/dom/fakexrdevice.rs b/components/script/dom/fakexrdevice.rs
index e8aa3e26dff..c1268d743d9 100644
--- a/components/script/dom/fakexrdevice.rs
+++ b/components/script/dom/fakexrdevice.rs
@@ -11,6 +11,7 @@ use crate::dom::bindings::reflector::{reflect_dom_object, Reflector};
use crate::dom::bindings::root::DomRoot;
use crate::dom::globalscope::GlobalScope;
use dom_struct::dom_struct;
+use euclid::{TypedPoint2D, TypedRect, TypedSize2D};
use euclid::{TypedRigidTransform3D, TypedRotation3D, TypedTransform3D, TypedVector3D};
use ipc_channel::ipc::IpcSender;
use webxr_api::{MockDeviceMsg, View, Views};
@@ -71,13 +72,24 @@ pub fn get_views(views: &[FakeXRViewInit]) -> Fallible<Views> {
let offset_l = get_origin(&left.viewOffset)?.inverse();
let offset_r = get_origin(&right.viewOffset)?.inverse();
+ let size_l = TypedSize2D::new(views[0].resolution.width, views[0].resolution.height);
+ let size_r = TypedSize2D::new(views[1].resolution.width, views[1].resolution.height);
+
+ let origin_l = TypedPoint2D::new(0, 0);
+ let origin_r = TypedPoint2D::new(size_l.width, 0);
+
+ let viewport_l = TypedRect::new(origin_l, size_l);
+ let viewport_r = TypedRect::new(origin_r, size_r);
+
let left = View {
projection: proj_l,
transform: offset_l,
+ viewport: viewport_l,
};
let right = View {
projection: proj_r,
transform: offset_r,
+ viewport: viewport_r,
};
Ok(Views::Stereo(left, right))
}
diff --git a/components/script/dom/performanceresourcetiming.rs b/components/script/dom/performanceresourcetiming.rs
index 2457e4f23ba..5d755240458 100644
--- a/components/script/dom/performanceresourcetiming.rs
+++ b/components/script/dom/performanceresourcetiming.rs
@@ -59,8 +59,6 @@ pub struct PerformanceResourceTiming {
// TODO(#21255): duration
// TODO(#21269): next_hop
// TODO(#21264): worker_start
-// TODO(#21256): redirect_start
-// TODO(#21257): redirect_end
// TODO(#21258): fetch_start
// TODO(#21259): domain_lookup_start
// TODO(#21260): domain_lookup_end
@@ -116,7 +114,7 @@ impl PerformanceResourceTiming {
next_hop: next_hop,
worker_start: 0.,
redirect_start: resource_timing.redirect_start as f64,
- redirect_end: 0.,
+ redirect_end: resource_timing.redirect_end as f64,
fetch_start: resource_timing.fetch_start as f64,
domain_lookup_start: resource_timing.domain_lookup_start as f64,
domain_lookup_end: 0.,
@@ -187,6 +185,11 @@ impl PerformanceResourceTimingMethods for PerformanceResourceTiming {
Finite::wrap(self.redirect_start)
}
+ // https://w3c.github.io/resource-timing/#dom-performanceresourcetiming-redirectend
+ fn RedirectEnd(&self) -> DOMHighResTimeStamp {
+ Finite::wrap(self.redirect_end)
+ }
+
// https://w3c.github.io/resource-timing/#dom-performanceresourcetiming-responsestart
fn ResponseStart(&self) -> DOMHighResTimeStamp {
Finite::wrap(self.response_start)
diff --git a/components/script/dom/webglrenderingcontext.rs b/components/script/dom/webglrenderingcontext.rs
index d413bd1ac71..6e249a38ed0 100644
--- a/components/script/dom/webglrenderingcontext.rs
+++ b/components/script/dom/webglrenderingcontext.rs
@@ -349,10 +349,6 @@ impl WebGLRenderingContext {
}
}
- pub fn size(&self) -> Size2D<u32> {
- self.size.get()
- }
-
// Helper function for validating framebuffer completeness in
// calls touching the framebuffer. From the GLES 2.0.25 spec,
// page 119:
diff --git a/components/script/dom/webidls/DOMException.webidl b/components/script/dom/webidls/DOMException.webidl
index a686e5b7023..1c7bca4ce66 100644
--- a/components/script/dom/webidls/DOMException.webidl
+++ b/components/script/dom/webidls/DOMException.webidl
@@ -38,7 +38,6 @@ interface DOMException {
const unsigned short TIMEOUT_ERR = 23;
const unsigned short INVALID_NODE_TYPE_ERR = 24;
const unsigned short DATA_CLONE_ERR = 25;
- const unsigned short NOT_READABLE_ERR = 26;
// Error code as u16
readonly attribute unsigned short code;
diff --git a/components/script/dom/webidls/PerformanceResourceTiming.webidl b/components/script/dom/webidls/PerformanceResourceTiming.webidl
index 087db5f45cd..acf1682e800 100644
--- a/components/script/dom/webidls/PerformanceResourceTiming.webidl
+++ b/components/script/dom/webidls/PerformanceResourceTiming.webidl
@@ -13,7 +13,7 @@ interface PerformanceResourceTiming : PerformanceEntry {
readonly attribute DOMString nextHopProtocol;
// readonly attribute DOMHighResTimeStamp workerStart;
readonly attribute DOMHighResTimeStamp redirectStart;
- // readonly attribute DOMHighResTimeStamp redirectEnd;
+ readonly attribute DOMHighResTimeStamp redirectEnd;
readonly attribute DOMHighResTimeStamp fetchStart;
readonly attribute DOMHighResTimeStamp domainLookupStart;
// readonly attribute DOMHighResTimeStamp domainLookupEnd;
diff --git a/components/script/dom/webidls/XRWebGLLayer.webidl b/components/script/dom/webidls/XRWebGLLayer.webidl
index f551096fc05..f83e1e4c35a 100644
--- a/components/script/dom/webidls/XRWebGLLayer.webidl
+++ b/components/script/dom/webidls/XRWebGLLayer.webidl
@@ -30,9 +30,9 @@ interface XRWebGLLayer : XRLayer {
readonly attribute boolean stencil;
readonly attribute boolean alpha;
- // readonly attribute WebGLFramebuffer framebuffer;
- // readonly attribute unsigned long framebufferWidth;
- // readonly attribute unsigned long framebufferHeight;
+ readonly attribute WebGLFramebuffer framebuffer;
+ readonly attribute unsigned long framebufferWidth;
+ readonly attribute unsigned long framebufferHeight;
// // Methods
XRViewport? getViewport(XRView view);
diff --git a/components/script/dom/xrsession.rs b/components/script/dom/xrsession.rs
index 1c29978e2ae..50e89a773dc 100644
--- a/components/script/dom/xrsession.rs
+++ b/components/script/dom/xrsession.rs
@@ -103,7 +103,7 @@ impl XRSession {
ret
}
- pub fn with_session<F: FnOnce(&Session)>(&self, with: F) {
+ pub fn with_session<R, F: FnOnce(&Session) -> R>(&self, with: F) -> R {
let session = self.session.borrow();
with(&session)
}
diff --git a/components/script/dom/xrviewport.rs b/components/script/dom/xrviewport.rs
index e3065a51d43..f0b013a317e 100644
--- a/components/script/dom/xrviewport.rs
+++ b/components/script/dom/xrviewport.rs
@@ -8,36 +8,26 @@ use crate::dom::bindings::reflector::{reflect_dom_object, Reflector};
use crate::dom::bindings::root::DomRoot;
use crate::dom::globalscope::GlobalScope;
use dom_struct::dom_struct;
+use euclid::TypedRect;
+use webxr_api::Viewport;
#[dom_struct]
pub struct XRViewport {
reflector_: Reflector,
- x: u32,
- y: u32,
- width: u32,
- height: u32,
+ viewport: TypedRect<i32, Viewport>,
}
impl XRViewport {
- fn new_inherited(x: u32, y: u32, width: u32, height: u32) -> XRViewport {
+ fn new_inherited(viewport: TypedRect<i32, Viewport>) -> XRViewport {
XRViewport {
reflector_: Reflector::new(),
- x,
- y,
- width,
- height,
+ viewport,
}
}
- pub fn new(
- global: &GlobalScope,
- x: u32,
- y: u32,
- width: u32,
- height: u32,
- ) -> DomRoot<XRViewport> {
+ pub fn new(global: &GlobalScope, viewport: TypedRect<i32, Viewport>) -> DomRoot<XRViewport> {
reflect_dom_object(
- Box::new(XRViewport::new_inherited(x, y, width, height)),
+ Box::new(XRViewport::new_inherited(viewport)),
global,
XRViewportBinding::Wrap,
)
@@ -47,21 +37,21 @@ impl XRViewport {
impl XRViewportMethods for XRViewport {
/// https://immersive-web.github.io/webxr/#dom-xrviewport-x
fn X(&self) -> i32 {
- self.x as i32
+ self.viewport.origin.x
}
/// https://immersive-web.github.io/webxr/#dom-xrviewport-y
fn Y(&self) -> i32 {
- self.y as i32
+ self.viewport.origin.y
}
/// https://immersive-web.github.io/webxr/#dom-xrviewport-width
fn Width(&self) -> i32 {
- self.width as i32
+ self.viewport.size.width
}
/// https://immersive-web.github.io/webxr/#dom-xrviewport-height
fn Height(&self) -> i32 {
- self.height as i32
+ self.viewport.size.height
}
}
diff --git a/components/script/dom/xrwebgllayer.rs b/components/script/dom/xrwebgllayer.rs
index 16653383a0b..a2a7c93f37c 100644
--- a/components/script/dom/xrwebgllayer.rs
+++ b/components/script/dom/xrwebgllayer.rs
@@ -6,10 +6,15 @@ use crate::dom::bindings::codegen::Bindings::XRViewBinding::{XREye, XRViewMethod
use crate::dom::bindings::codegen::Bindings::XRWebGLLayerBinding;
use crate::dom::bindings::codegen::Bindings::XRWebGLLayerBinding::XRWebGLLayerInit;
use crate::dom::bindings::codegen::Bindings::XRWebGLLayerBinding::XRWebGLLayerMethods;
+use crate::dom::bindings::codegen::Bindings::WebGLRenderingContextBinding::WebGLRenderingContextBinding::WebGLRenderingContextMethods;
+use crate::dom::bindings::codegen::Bindings::WebGLRenderingContextBinding::WebGLRenderingContextConstants as constants;
+use crate::dom::bindings::error::Error;
use crate::dom::bindings::error::Fallible;
use crate::dom::bindings::reflector::{reflect_dom_object, DomObject};
use crate::dom::bindings::root::{Dom, DomRoot};
use crate::dom::globalscope::GlobalScope;
+use crate::dom::webgl_validations::types::TexImageTarget;
+use crate::dom::webglframebuffer::WebGLFramebuffer;
use crate::dom::webglrenderingcontext::WebGLRenderingContext;
use crate::dom::window::Window;
use crate::dom::xrlayer::XRLayer;
@@ -17,6 +22,9 @@ use crate::dom::xrsession::XRSession;
use crate::dom::xrview::XRView;
use crate::dom::xrviewport::XRViewport;
use dom_struct::dom_struct;
+use js::rust::CustomAutoRooter;
+use std::convert::TryInto;
+use webxr_api::Views;
#[dom_struct]
pub struct XRWebGLLayer {
@@ -27,6 +35,7 @@ pub struct XRWebGLLayer {
alpha: bool,
context: Dom<WebGLRenderingContext>,
session: Dom<XRSession>,
+ framebuffer: Dom<WebGLFramebuffer>,
}
impl XRWebGLLayer {
@@ -34,6 +43,7 @@ impl XRWebGLLayer {
session: &XRSession,
context: &WebGLRenderingContext,
init: &XRWebGLLayerInit,
+ framebuffer: &WebGLFramebuffer,
) -> XRWebGLLayer {
XRWebGLLayer {
xrlayer: XRLayer::new_inherited(),
@@ -43,6 +53,7 @@ impl XRWebGLLayer {
alpha: init.alpha,
context: Dom::from_ref(context),
session: Dom::from_ref(session),
+ framebuffer: Dom::from_ref(framebuffer),
}
}
@@ -51,21 +62,82 @@ impl XRWebGLLayer {
session: &XRSession,
context: &WebGLRenderingContext,
init: &XRWebGLLayerInit,
+ framebuffer: &WebGLFramebuffer,
) -> DomRoot<XRWebGLLayer> {
reflect_dom_object(
- Box::new(XRWebGLLayer::new_inherited(session, context, init)),
+ Box::new(XRWebGLLayer::new_inherited(
+ session,
+ context,
+ init,
+ framebuffer,
+ )),
global,
XRWebGLLayerBinding::Wrap,
)
}
+ /// https://immersive-web.github.io/webxr/#dom-xrwebgllayer-xrwebgllayer
pub fn Constructor(
global: &Window,
session: &XRSession,
context: &WebGLRenderingContext,
init: &XRWebGLLayerInit,
) -> Fallible<DomRoot<Self>> {
- Ok(XRWebGLLayer::new(&global.global(), session, context, init))
+ let cx = global.get_cx();
+ let old_fbo = context.bound_framebuffer();
+ let old_texture = context
+ .textures()
+ .active_texture_for_image_target(TexImageTarget::Texture2D);
+
+ // Step 8.2. "Initialize layer’s framebuffer to a new opaque framebuffer created with context."
+ let framebuffer = context.CreateFramebuffer().ok_or(Error::Operation)?;
+
+ // Step 8.3. "Allocate and initialize resources compatible with session’s XR device,
+ // including GPU accessible memory buffers, as required to support the compositing of layer."
+
+ // Create a new texture with size given by the session's recommended resolution
+ let texture = context.CreateTexture().ok_or(Error::Operation)?;
+ let resolution = session.with_session(|s| s.recommended_framebuffer_resolution());
+ let mut pixels = CustomAutoRooter::new(None);
+ context.BindTexture(constants::TEXTURE_2D, Some(&texture));
+ let sc = context.TexImage2D(
+ constants::TEXTURE_2D,
+ 0,
+ constants::RGBA,
+ resolution.width,
+ resolution.height,
+ 0,
+ constants::RGBA,
+ constants::UNSIGNED_BYTE,
+ pixels.root(cx),
+ );
+
+ // Bind the new texture to the framebuffer
+ context.BindFramebuffer(constants::FRAMEBUFFER, Some(&framebuffer));
+ context.FramebufferTexture2D(
+ constants::FRAMEBUFFER,
+ constants::COLOR_ATTACHMENT0,
+ constants::TEXTURE_2D,
+ Some(&texture),
+ 0,
+ );
+
+ // Restore the WebGL state while complaining about global mutable state
+ context.BindTexture(constants::TEXTURE_2D, old_texture.as_ref().map(|t| &**t));
+ context.BindFramebuffer(constants::FRAMEBUFFER, old_fbo.as_ref().map(|f| &**f));
+
+ // Step 8.4: "If layer’s resources were unable to be created for any reason,
+ // throw an OperationError and abort these steps."
+ sc.or(Err(Error::Operation))?;
+
+ // Step 9. "Return layer."
+ Ok(XRWebGLLayer::new(
+ &global.global(),
+ session,
+ context,
+ init,
+ &framebuffer,
+ ))
}
}
@@ -95,28 +167,48 @@ impl XRWebGLLayerMethods for XRWebGLLayer {
DomRoot::from_ref(&self.context)
}
+ /// https://immersive-web.github.io/webxr/#dom-xrwebgllayer-framebuffer
+ fn Framebuffer(&self) -> DomRoot<WebGLFramebuffer> {
+ DomRoot::from_ref(&self.framebuffer)
+ }
+
+ /// https://immersive-web.github.io/webxr/#dom-xrwebgllayer-framebufferwidth
+ fn FramebufferWidth(&self) -> u32 {
+ self.framebuffer
+ .size()
+ .unwrap_or((0, 0))
+ .0
+ .try_into()
+ .unwrap_or(0)
+ }
+
+ /// https://immersive-web.github.io/webxr/#dom-xrwebgllayer-framebufferheight
+ fn FramebufferHeight(&self) -> u32 {
+ self.framebuffer
+ .size()
+ .unwrap_or((0, 0))
+ .1
+ .try_into()
+ .unwrap_or(0)
+ }
+
/// https://immersive-web.github.io/webxr/#dom-xrwebgllayer-getviewport
fn GetViewport(&self, view: &XRView) -> Option<DomRoot<XRViewport>> {
if self.session != view.session() {
return None;
}
- let size = self.context.size();
+ let views = self.session.with_session(|s| s.views().clone());
- let x = if view.Eye() == XREye::Left {
- 0
- } else {
- size.width / 2
+ let viewport = match (view.Eye(), views) {
+ (XREye::None, Views::Mono(view)) => view.viewport,
+ (XREye::Left, Views::Stereo(view, _)) => view.viewport,
+ (XREye::Right, Views::Stereo(_, view)) => view.viewport,
+ // The spec doesn't really say what to do in this case
+ // https://github.com/immersive-web/webxr/issues/769
+ _ => return None,
};
- // XXXManishearth this assumes the WebVR default of canvases being cut in half
- // which need not be generally true for all devices, and will not work in
- // inline VR mode
- Some(XRViewport::new(
- &self.global(),
- x,
- 0,
- size.width / 2,
- size.height,
- ))
+
+ Some(XRViewport::new(&self.global(), viewport))
}
}