Bug 1798703 - Enforce alignment for UniformData via union rather than overalignment...
[gecko.git] / dom / canvas / HostWebGLContext.h
blobff97d97c4801b8669132df2319b88096b3060d23
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* This Source Code Form is subject to the terms of the Mozilla Public
3 * License, v. 2.0. If a copy of the MPL was not distributed with this
4 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
6 #ifndef HOSTWEBGLCONTEXT_H_
7 #define HOSTWEBGLCONTEXT_H_
9 #include "mozilla/dom/BindingUtils.h"
10 #include "mozilla/GfxMessageUtils.h"
11 #include "ClientWebGLContext.h"
12 #include "mozilla/Maybe.h"
13 #include "mozilla/UniquePtr.h"
14 #include "GLContext.h"
15 #include "WebGLContext.h"
16 #include "WebGL2Context.h"
17 #include "WebGLFramebuffer.h"
18 #include "WebGLTypes.h"
19 #include "WebGLCommandQueue.h"
21 #include <unordered_map>
22 #include <unordered_set>
23 #include <vector>
25 namespace mozilla {
27 namespace dom {
28 class WebGLParent;
30 namespace layers {
31 class CompositableHost;
34 struct LockedOutstandingContexts final {
35 private:
36 // StaticMutexAutoLock lock; // We can't use it directly (STACK_CLASS), but
37 // this is effectively what we hold via RAII.
39 public:
40 const std::unordered_set<HostWebGLContext*>& contexts;
42 LockedOutstandingContexts();
43 ~LockedOutstandingContexts();
46 /**
47 * Host endpoint of a WebGLContext. HostWebGLContext owns a WebGLContext
48 * that it uses to execute commands sent from its ClientWebGLContext.
50 * A HostWebGLContext continuously issues a Task to the Compositor thread that
51 * causes it to drain its queue of commands. It also maintains a map of WebGL
52 * objects (e.g. ObjectIdMap<WebGLShader>) that it uses associate them with
53 * their cross-process IDs.
55 * This class is not an implementation of the
56 * nsICanvasRenderingContextInternal DOM class. That is the
57 * ClientWebGLContext.
59 class HostWebGLContext final : public SupportsWeakPtr {
60 friend class WebGLContext;
61 friend class WebGLMemoryTracker;
62 friend class dom::WebGLParent;
64 using ObjectId = webgl::ObjectId;
66 static std::unique_ptr<LockedOutstandingContexts> OutstandingContexts() {
67 return std::make_unique<LockedOutstandingContexts>();
70 public:
71 struct OwnerData final {
72 ClientWebGLContext* inProcess = nullptr;
73 dom::WebGLParent* outOfProcess = nullptr;
76 static UniquePtr<HostWebGLContext> Create(const OwnerData&,
77 const webgl::InitContextDesc&,
78 webgl::InitContextResult* out);
80 private:
81 explicit HostWebGLContext(const OwnerData&);
83 public:
84 virtual ~HostWebGLContext();
86 WebGLContext* GetWebGLContext() const { return mContext; }
88 public:
89 const OwnerData mOwnerData;
91 private:
92 RefPtr<WebGLContext> mContext;
94 #define _(X) std::unordered_map<ObjectId, RefPtr<WebGL##X>> m##X##Map;
96 _(Buffer)
97 _(Framebuffer)
98 _(Program)
99 _(Query)
100 _(Renderbuffer)
101 _(Sampler)
102 _(Shader)
103 _(Sync)
104 _(Texture)
105 _(TransformFeedback)
106 _(VertexArray)
108 #undef _
110 class AutoResolveT final {
111 friend class HostWebGLContext;
113 const HostWebGLContext& mParent;
114 const ObjectId mId;
116 public:
117 AutoResolveT(const HostWebGLContext& parent, const ObjectId id)
118 : mParent(parent), mId(id) {}
120 #define _(X) \
121 WebGL##X* As(WebGL##X*) const { \
122 const auto maybe = MaybeFind(mParent.m##X##Map, mId); \
123 if (!maybe) return nullptr; \
124 return maybe->get(); \
127 _(Buffer)
128 _(Framebuffer)
129 _(Program)
130 _(Query)
131 _(Renderbuffer)
132 _(Sampler)
133 _(Shader)
134 _(Sync)
135 _(Texture)
136 _(TransformFeedback)
137 _(VertexArray)
139 #undef _
140 template <typename T>
141 MOZ_IMPLICIT operator T*() const {
142 T* coercer = nullptr;
143 return As(coercer);
146 template <typename T>
147 MOZ_IMPLICIT operator const T*() const {
148 T* coercer = nullptr;
149 return As(coercer);
153 AutoResolveT AutoResolve(const ObjectId id) const { return {*this, id}; }
154 template <typename T>
155 T* ById(const ObjectId id) const {
156 T* coercer = nullptr;
157 return AutoResolve(id).As(coercer);
160 // -------------------------------------------------------------------------
161 // Host-side methods. Calls in the client are forwarded to the host.
162 // -------------------------------------------------------------------------
164 public:
165 // ------------------------- Composition -------------------------
167 void SetCompositableHost(RefPtr<layers::CompositableHost>& compositableHost) {
168 mContext->SetCompositableHost(compositableHost);
171 void Present(const ObjectId xrFb, const layers::TextureType t,
172 const bool webvr, const webgl::SwapChainOptions& options) const {
173 return (void)mContext->Present(AutoResolve(xrFb), t, webvr, options);
175 void CopyToSwapChain(const ObjectId fb, const layers::TextureType t,
176 const webgl::SwapChainOptions& options) const {
177 return (void)mContext->CopyToSwapChain(AutoResolve(fb), t, options);
179 void EndOfFrame() const { return (void)mContext->EndOfFrame(); }
180 Maybe<layers::SurfaceDescriptor> GetFrontBuffer(ObjectId xrFb,
181 const bool webvr) const;
183 // -
185 Maybe<uvec2> FrontBufferSnapshotInto(Maybe<Range<uint8_t>> dest) const {
186 return mContext->FrontBufferSnapshotInto(dest);
189 Maybe<uvec2> FrontBufferSnapshotInto(
190 std::shared_ptr<gl::SharedSurface>& front,
191 Maybe<Range<uint8_t>> dest) const {
192 return mContext->FrontBufferSnapshotInto(front, dest);
195 void ClearVRSwapChain() const { mContext->ClearVRSwapChain(); }
197 // -
199 void Resize(const uvec2& size) { return mContext->Resize(size); }
201 uvec2 DrawingBufferSize() { return mContext->DrawingBufferSize(); }
203 void OnMemoryPressure() { return mContext->OnMemoryPressure(); }
205 void DidRefresh() { mContext->DidRefresh(); }
207 void GenerateError(const GLenum error, const std::string& text) const {
208 mContext->GenerateErrorImpl(error, text);
211 void OnContextLoss(webgl::ContextLossReason);
213 void RequestExtension(const WebGLExtensionID ext) {
214 mContext->RequestExtension(ext);
217 // -
218 // Child-ward
220 void JsWarning(const std::string&) const;
222 // -
223 // Creation and destruction
225 void CreateBuffer(ObjectId);
226 void CreateFramebuffer(ObjectId);
227 bool CreateOpaqueFramebuffer(ObjectId,
228 const webgl::OpaqueFramebufferOptions& options);
229 void CreateProgram(ObjectId);
230 void CreateQuery(ObjectId);
231 void CreateRenderbuffer(ObjectId);
232 void CreateSampler(ObjectId);
233 void CreateShader(ObjectId, GLenum type);
234 void CreateSync(ObjectId);
235 void CreateTexture(ObjectId);
236 void CreateTransformFeedback(ObjectId);
237 void CreateVertexArray(ObjectId);
239 void DeleteBuffer(ObjectId);
240 void DeleteFramebuffer(ObjectId);
241 void DeleteProgram(ObjectId);
242 void DeleteQuery(ObjectId);
243 void DeleteRenderbuffer(ObjectId);
244 void DeleteSampler(ObjectId);
245 void DeleteShader(ObjectId);
246 void DeleteSync(ObjectId);
247 void DeleteTexture(ObjectId);
248 void DeleteTransformFeedback(ObjectId);
249 void DeleteVertexArray(ObjectId);
251 // ------------------------- GL State -------------------------
252 bool IsContextLost() const { return mContext->IsContextLost(); }
254 void SetEnabled(GLenum cap, Maybe<GLuint> i, bool val) const {
255 mContext->SetEnabled(cap, i, val);
258 bool IsEnabled(GLenum cap) const { return mContext->IsEnabled(cap); }
260 Maybe<double> GetNumber(GLenum pname) const {
261 return mContext->GetParameter(pname);
264 Maybe<std::string> GetString(GLenum pname) const {
265 return mContext->GetString(pname);
268 void AttachShader(ObjectId prog, ObjectId shader) const {
269 const auto pProg = ById<WebGLProgram>(prog);
270 const auto pShader = ById<WebGLShader>(shader);
271 if (!pProg || !pShader) return;
272 mContext->AttachShader(*pProg, *pShader);
275 void BindAttribLocation(ObjectId id, GLuint location,
276 const std::string& name) const {
277 const auto obj = ById<WebGLProgram>(id);
278 if (!obj) return;
279 mContext->BindAttribLocation(*obj, location, name);
282 void BindFramebuffer(GLenum target, ObjectId id) const {
283 mContext->BindFramebuffer(target, AutoResolve(id));
286 void BlendColor(GLclampf r, GLclampf g, GLclampf b, GLclampf a) const {
287 mContext->BlendColor(r, g, b, a);
290 void BlendEquationSeparate(Maybe<GLuint> i, GLenum modeRGB,
291 GLenum modeAlpha) const {
292 mContext->BlendEquationSeparate(i, modeRGB, modeAlpha);
295 void BlendFuncSeparate(Maybe<GLuint> i, GLenum srcRGB, GLenum dstRGB,
296 GLenum srcAlpha, GLenum dstAlpha) const {
297 mContext->BlendFuncSeparate(i, srcRGB, dstRGB, srcAlpha, dstAlpha);
300 GLenum CheckFramebufferStatus(GLenum target) const {
301 return mContext->CheckFramebufferStatus(target);
304 void Clear(GLbitfield mask) const { mContext->Clear(mask); }
306 void ClearColor(GLclampf r, GLclampf g, GLclampf b, GLclampf a) const {
307 mContext->ClearColor(r, g, b, a);
310 void ClearDepth(GLclampf v) const { mContext->ClearDepth(v); }
312 void ClearStencil(GLint v) const { mContext->ClearStencil(v); }
314 void ColorMask(Maybe<GLuint> i, uint8_t mask) const {
315 mContext->ColorMask(i, mask);
318 void CompileShader(const ObjectId id) const {
319 const auto obj = ById<WebGLShader>(id);
320 if (!obj) return;
321 mContext->CompileShader(*obj);
324 void CullFace(GLenum face) const { mContext->CullFace(face); }
326 void DepthFunc(GLenum func) const { mContext->DepthFunc(func); }
328 void DepthMask(WebGLboolean b) const { mContext->DepthMask(b); }
330 void DepthRange(GLclampf zNear, GLclampf zFar) const {
331 mContext->DepthRange(zNear, zFar);
334 void DetachShader(const ObjectId prog, const ObjectId shader) const {
335 const auto pProg = ById<WebGLProgram>(prog);
336 const auto pShader = ById<WebGLShader>(shader);
337 if (!pProg || !pShader) return;
338 mContext->DetachShader(*pProg, *pShader);
341 void Flush() const { mContext->Flush(); }
343 void Finish() const { mContext->Finish(); }
345 void FramebufferAttach(const GLenum target, const GLenum attachSlot,
346 const GLenum bindImageTarget, const ObjectId id,
347 const GLint mipLevel, const GLint zLayerBase,
348 const GLsizei numViewLayers) const {
349 webgl::FbAttachInfo toAttach;
350 toAttach.rb = AutoResolve(id);
351 toAttach.tex = AutoResolve(id);
352 toAttach.mipLevel = mipLevel;
353 toAttach.zLayer = zLayerBase;
354 if (numViewLayers) {
355 toAttach.zLayerCount = numViewLayers;
356 toAttach.isMultiview = true;
359 mContext->FramebufferAttach(target, attachSlot, bindImageTarget, toAttach);
362 void FrontFace(GLenum mode) const { mContext->FrontFace(mode); }
364 Maybe<double> GetBufferParameter(GLenum target, GLenum pname) const {
365 return mContext->GetBufferParameter(target, pname);
368 webgl::CompileResult GetCompileResult(ObjectId id) const {
369 const auto obj = ById<WebGLShader>(id);
370 if (!obj) return {};
371 return mContext->GetCompileResult(*obj);
374 GLenum GetError() const { return mContext->GetError(); }
376 GLint GetFragDataLocation(ObjectId id, const std::string& name) const {
377 const auto obj = ById<WebGLProgram>(id);
378 if (!obj) return -1;
379 return mContext->GetFragDataLocation(*obj, name);
382 Maybe<double> GetFramebufferAttachmentParameter(ObjectId id,
383 GLenum attachment,
384 GLenum pname) const {
385 return mContext->GetFramebufferAttachmentParameter(AutoResolve(id),
386 attachment, pname);
389 webgl::LinkResult GetLinkResult(ObjectId id) const {
390 const auto obj = ById<WebGLProgram>(id);
391 if (!obj) return {};
392 return mContext->GetLinkResult(*obj);
395 Maybe<double> GetRenderbufferParameter(ObjectId id, GLenum pname) const {
396 const auto obj = ById<WebGLRenderbuffer>(id);
397 if (!obj) return {};
398 return mContext->GetRenderbufferParameter(*obj, pname);
401 Maybe<webgl::ShaderPrecisionFormat> GetShaderPrecisionFormat(
402 GLenum shaderType, GLenum precisionType) const {
403 return mContext->GetShaderPrecisionFormat(shaderType, precisionType);
406 webgl::GetUniformData GetUniform(ObjectId id, uint32_t loc) const {
407 const auto obj = ById<WebGLProgram>(id);
408 if (!obj) return {};
409 return mContext->GetUniform(*obj, loc);
412 void Hint(GLenum target, GLenum mode) const { mContext->Hint(target, mode); }
414 void LineWidth(GLfloat width) const { mContext->LineWidth(width); }
416 void LinkProgram(const ObjectId id) const {
417 const auto obj = ById<WebGLProgram>(id);
418 if (!obj) return;
419 mContext->LinkProgram(*obj);
422 void PolygonOffset(GLfloat factor, GLfloat units) const {
423 mContext->PolygonOffset(factor, units);
426 void SampleCoverage(GLclampf value, bool invert) const {
427 mContext->SampleCoverage(value, invert);
430 void Scissor(GLint x, GLint y, GLsizei width, GLsizei height) const {
431 mContext->Scissor(x, y, width, height);
434 // TODO: s/nsAString/std::string/
435 void ShaderSource(const ObjectId id, const std::string& source) const {
436 const auto obj = ById<WebGLShader>(id);
437 if (!obj) return;
438 mContext->ShaderSource(*obj, source);
441 void StencilFuncSeparate(GLenum face, GLenum func, GLint ref,
442 GLuint mask) const {
443 mContext->StencilFuncSeparate(face, func, ref, mask);
445 void StencilMaskSeparate(GLenum face, GLuint mask) const {
446 mContext->StencilMaskSeparate(face, mask);
448 void StencilOpSeparate(GLenum face, GLenum sfail, GLenum dpfail,
449 GLenum dppass) const {
450 mContext->StencilOpSeparate(face, sfail, dpfail, dppass);
453 void Viewport(GLint x, GLint y, GLsizei width, GLsizei height) const {
454 mContext->Viewport(x, y, width, height);
457 // ------------------------- Buffer Objects -------------------------
458 void BindBuffer(GLenum target, const ObjectId id) const {
459 mContext->BindBuffer(target, AutoResolve(id));
462 void BindBufferRange(GLenum target, GLuint index, const ObjectId id,
463 uint64_t offset, uint64_t size) const {
464 GetWebGL2Context()->BindBufferRange(target, index, AutoResolve(id), offset,
465 size);
468 void CopyBufferSubData(GLenum readTarget, GLenum writeTarget,
469 uint64_t readOffset, uint64_t writeOffset,
470 uint64_t size) const {
471 GetWebGL2Context()->CopyBufferSubData(readTarget, writeTarget, readOffset,
472 writeOffset, size);
475 bool GetBufferSubData(GLenum target, uint64_t srcByteOffset,
476 const Range<uint8_t>& dest) const {
477 return GetWebGL2Context()->GetBufferSubData(target, srcByteOffset, dest);
480 void BufferData(GLenum target, const RawBuffer<>& data, GLenum usage) const {
481 const auto& beginOrNull = data.begin();
482 mContext->BufferData(target, data.size(), beginOrNull, usage);
485 void BufferSubData(GLenum target, uint64_t dstByteOffset,
486 const RawBuffer<>& srcData) const {
487 const auto& range = srcData.Data();
488 mContext->BufferSubData(target, dstByteOffset, range.length(),
489 range.begin().get());
492 // -------------------------- Framebuffer Objects --------------------------
493 void BlitFramebuffer(GLint srcX0, GLint srcY0, GLint srcX1, GLint srcY1,
494 GLint dstX0, GLint dstY0, GLint dstX1, GLint dstY1,
495 GLbitfield mask, GLenum filter) const {
496 GetWebGL2Context()->BlitFramebuffer(srcX0, srcY0, srcX1, srcY1, dstX0,
497 dstY0, dstX1, dstY1, mask, filter);
500 void InvalidateFramebuffer(GLenum target,
501 const RawBuffer<const GLenum>& attachments) const {
502 GetWebGL2Context()->InvalidateFramebuffer(target, MakeRange(attachments));
505 void InvalidateSubFramebuffer(GLenum target,
506 const RawBuffer<const GLenum>& attachments,
507 GLint x, GLint y, GLsizei width,
508 GLsizei height) const {
509 GetWebGL2Context()->InvalidateSubFramebuffer(target, MakeRange(attachments),
510 x, y, width, height);
513 void ReadBuffer(GLenum mode) const { GetWebGL2Context()->ReadBuffer(mode); }
515 // ----------------------- Renderbuffer objects -----------------------
516 Maybe<std::vector<int32_t>> GetInternalformatParameter(GLenum target,
517 GLenum internalformat,
518 GLenum pname) const {
519 return GetWebGL2Context()->GetInternalformatParameter(
520 target, internalformat, pname);
523 void RenderbufferStorageMultisample(ObjectId id, uint32_t samples,
524 GLenum internalFormat, uint32_t width,
525 uint32_t height) const {
526 const auto obj = ById<WebGLRenderbuffer>(id);
527 if (!obj) return;
528 mContext->RenderbufferStorageMultisample(*obj, samples, internalFormat,
529 width, height);
532 // --------------------------- Texture objects ---------------------------
533 void ActiveTexture(uint32_t texUnit) const {
534 mContext->ActiveTexture(texUnit);
537 void BindTexture(GLenum texTarget, const ObjectId id) const {
538 mContext->BindTexture(texTarget, AutoResolve(id));
541 void GenerateMipmap(GLenum texTarget) const {
542 mContext->GenerateMipmap(texTarget);
545 // CompressedTexSubImage if `sub`
546 void CompressedTexImage(bool sub, GLenum imageTarget, uint32_t level,
547 GLenum format, const uvec3& offset, const uvec3& size,
548 const RawBuffer<>& src, const uint32_t pboImageSize,
549 const Maybe<uint64_t>& pboOffset) const {
550 mContext->CompressedTexImage(sub, imageTarget, level, format, offset, size,
551 MakeRange(src), pboImageSize, pboOffset);
554 // CopyTexSubImage if `!respecFormat`
555 void CopyTexImage(GLenum imageTarget, uint32_t level, GLenum respecFormat,
556 const uvec3& dstOffset, const ivec2& srcOffset,
557 const uvec2& size) const {
558 mContext->CopyTexImage(imageTarget, level, respecFormat, dstOffset,
559 srcOffset, size);
562 // TexSubImage if `!respecFormat`
563 void TexImage(uint32_t level, GLenum respecFormat, const uvec3& offset,
564 const webgl::PackingInfo& pi,
565 const webgl::TexUnpackBlobDesc& src) const {
566 mContext->TexImage(level, respecFormat, offset, pi, src);
569 void TexStorage(GLenum texTarget, uint32_t levels, GLenum internalFormat,
570 const uvec3& size) const {
571 GetWebGL2Context()->TexStorage(texTarget, levels, internalFormat, size);
574 Maybe<double> GetTexParameter(ObjectId id, GLenum pname) const {
575 const auto obj = ById<WebGLTexture>(id);
576 if (!obj) return {};
577 return mContext->GetTexParameter(*obj, pname);
580 void TexParameter_base(GLenum texTarget, GLenum pname,
581 const FloatOrInt& param) const {
582 mContext->TexParameter_base(texTarget, pname, param);
585 // ------------------- Programs and shaders --------------------------------
586 void UseProgram(ObjectId id) const { mContext->UseProgram(AutoResolve(id)); }
588 bool ValidateProgram(ObjectId id) const {
589 const auto obj = ById<WebGLProgram>(id);
590 if (!obj) return false;
591 return mContext->ValidateProgram(*obj);
594 // ------------------------ Uniforms and attributes ------------------------
596 void UniformData(uint32_t loc, bool transpose,
597 const RawBuffer<webgl::UniformDataVal>& data) const {
598 mContext->UniformData(loc, transpose, data.Data());
601 void VertexAttrib4T(GLuint index, const webgl::TypedQuad& data) const {
602 mContext->VertexAttrib4T(index, data);
605 void VertexAttribDivisor(GLuint index, GLuint divisor) const {
606 mContext->VertexAttribDivisor(index, divisor);
609 Maybe<double> GetIndexedParameter(GLenum target, GLuint index) const {
610 return GetWebGL2Context()->GetIndexedParameter(target, index);
613 void UniformBlockBinding(const ObjectId id, GLuint uniformBlockIndex,
614 GLuint uniformBlockBinding) const {
615 const auto obj = ById<WebGLProgram>(id);
616 if (!obj) return;
617 GetWebGL2Context()->UniformBlockBinding(*obj, uniformBlockIndex,
618 uniformBlockBinding);
621 void EnableVertexAttribArray(GLuint index) const {
622 mContext->EnableVertexAttribArray(index);
625 void DisableVertexAttribArray(GLuint index) const {
626 mContext->DisableVertexAttribArray(index);
629 Maybe<double> GetVertexAttrib(GLuint index, GLenum pname) const {
630 return mContext->GetVertexAttrib(index, pname);
633 void VertexAttribPointer(GLuint index,
634 const webgl::VertAttribPointerDesc& desc) const {
635 mContext->VertexAttribPointer(index, desc);
638 // --------------------------- Buffer Operations --------------------------
639 void ClearBufferTv(GLenum buffer, GLint drawBuffer,
640 const webgl::TypedQuad& data) const {
641 GetWebGL2Context()->ClearBufferTv(buffer, drawBuffer, data);
644 void ClearBufferfi(GLenum buffer, GLint drawBuffer, GLfloat depth,
645 GLint stencil) const {
646 GetWebGL2Context()->ClearBufferfi(buffer, drawBuffer, depth, stencil);
649 // ------------------------------ Readback -------------------------------
650 void ReadPixelsPbo(const webgl::ReadPixelsDesc& desc,
651 const uint64_t offset) const {
652 mContext->ReadPixelsPbo(desc, offset);
655 webgl::ReadPixelsResult ReadPixelsInto(const webgl::ReadPixelsDesc& desc,
656 const Range<uint8_t>& dest) const {
657 return mContext->ReadPixelsInto(desc, dest);
660 // ----------------------------- Sampler -----------------------------------
662 void BindSampler(GLuint unit, ObjectId id) const {
663 GetWebGL2Context()->BindSampler(unit, AutoResolve(id));
666 void SamplerParameteri(ObjectId id, GLenum pname, GLint param) const {
667 const auto obj = ById<WebGLSampler>(id);
668 if (!obj) return;
669 GetWebGL2Context()->SamplerParameteri(*obj, pname, param);
672 void SamplerParameterf(ObjectId id, GLenum pname, GLfloat param) const {
673 const auto obj = ById<WebGLSampler>(id);
674 if (!obj) return;
675 GetWebGL2Context()->SamplerParameterf(*obj, pname, param);
678 Maybe<double> GetSamplerParameter(ObjectId id, GLenum pname) const {
679 const auto obj = ById<WebGLSampler>(id);
680 if (!obj) return {};
681 return GetWebGL2Context()->GetSamplerParameter(*obj, pname);
684 // ------------------------------- GL Sync ---------------------------------
686 GLenum ClientWaitSync(ObjectId id, GLbitfield flags, GLuint64 timeout) const {
687 const auto obj = ById<WebGLSync>(id);
688 if (!obj) return LOCAL_GL_WAIT_FAILED;
689 return GetWebGL2Context()->ClientWaitSync(*obj, flags, timeout);
692 // -------------------------- Transform Feedback ---------------------------
693 void BindTransformFeedback(ObjectId id) const {
694 GetWebGL2Context()->BindTransformFeedback(AutoResolve(id));
697 void BeginTransformFeedback(GLenum primitiveMode) const {
698 GetWebGL2Context()->BeginTransformFeedback(primitiveMode);
701 void EndTransformFeedback() const {
702 GetWebGL2Context()->EndTransformFeedback();
705 void PauseTransformFeedback() const {
706 GetWebGL2Context()->PauseTransformFeedback();
709 void ResumeTransformFeedback() const {
710 GetWebGL2Context()->ResumeTransformFeedback();
713 void TransformFeedbackVaryings(ObjectId id,
714 const std::vector<std::string>& varyings,
715 GLenum bufferMode) const {
716 const auto obj = ById<WebGLProgram>(id);
717 if (!obj) return;
718 GetWebGL2Context()->TransformFeedbackVaryings(*obj, varyings, bufferMode);
721 // -------------------------- Opaque Framebuffers ---------------------------
722 void SetFramebufferIsInOpaqueRAF(ObjectId id, bool value) {
723 WebGLFramebuffer* fb = AutoResolve(id);
724 if (fb) {
725 fb->mInOpaqueRAF = value;
729 // -------------------------------------------------------------------------
730 // Host-side extension methods. Calls in the client are forwarded to the
731 // host. Some extension methods are also available in WebGL2 Contexts. For
732 // them, the final parameter is a boolean indicating if the call originated
733 // from an extension.
734 // -------------------------------------------------------------------------
736 // Misc. Extensions
737 void DrawBuffers(const std::vector<GLenum>& buffers) const {
738 mContext->DrawBuffers(buffers);
741 // VertexArrayObjectEXT
742 void BindVertexArray(ObjectId id) const {
743 mContext->BindVertexArray(AutoResolve(id));
746 // InstancedElementsEXT
747 void DrawArraysInstanced(GLenum mode, GLint first, GLsizei vertCount,
748 GLsizei primCount) const {
749 mContext->DrawArraysInstanced(mode, first, vertCount, primCount);
752 void DrawElementsInstanced(GLenum mode, GLsizei vertCount, GLenum type,
753 WebGLintptr offset, GLsizei primCount) const {
754 mContext->DrawElementsInstanced(mode, vertCount, type, offset, primCount);
757 // GLQueryEXT
758 void BeginQuery(GLenum target, ObjectId id) const {
759 const auto obj = ById<WebGLQuery>(id);
760 if (!obj) return;
761 mContext->BeginQuery(target, *obj);
764 void EndQuery(GLenum target) const { mContext->EndQuery(target); }
766 void QueryCounter(ObjectId id) const {
767 const auto obj = ById<WebGLQuery>(id);
768 if (!obj) return;
769 mContext->QueryCounter(*obj);
772 Maybe<double> GetQueryParameter(ObjectId id, GLenum pname) const {
773 const auto obj = ById<WebGLQuery>(id);
774 if (!obj) return {};
775 return mContext->GetQueryParameter(*obj, pname);
778 // -------------------------------------------------------------------------
779 // Client-side methods. Calls in the Host are forwarded to the client.
780 // -------------------------------------------------------------------------
781 public:
782 void OnLostContext();
783 void OnRestoredContext();
785 protected:
786 WebGL2Context* GetWebGL2Context() const {
787 MOZ_RELEASE_ASSERT(mContext->IsWebGL2(), "Requires WebGL2 context");
788 return static_cast<WebGL2Context*>(mContext.get());
792 } // namespace mozilla
794 #endif // HOSTWEBGLCONTEXT_H_