diff options
Diffstat (limited to '')
-rw-r--r-- | dom/canvas/WebGLContext.h | 1430 |
1 files changed, 1430 insertions, 0 deletions
diff --git a/dom/canvas/WebGLContext.h b/dom/canvas/WebGLContext.h new file mode 100644 index 0000000000..5d4ca6a68b --- /dev/null +++ b/dom/canvas/WebGLContext.h @@ -0,0 +1,1430 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* 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/. */ + +#ifndef WEBGLCONTEXT_H_ +#define WEBGLCONTEXT_H_ + +#include <bitset> +#include <memory> +#include <stdarg.h> + +#include "GLContextTypes.h" +#include "GLDefs.h" +#include "GLScreenBuffer.h" +#include "js/ScalarType.h" // js::Scalar::Type +#include "mozilla/Attributes.h" +#include "mozilla/Atomics.h" +#include "mozilla/CheckedInt.h" +#include "mozilla/dom/BindingDeclarations.h" +#include "mozilla/dom/HTMLCanvasElement.h" +#include "mozilla/dom/Nullable.h" +#include "mozilla/dom/TypedArray.h" +#include "mozilla/EnumeratedArray.h" +#include "mozilla/gfx/2D.h" +#include "mozilla/Mutex.h" +#include "mozilla/StaticMutex.h" +#include "mozilla/StaticPrefs_webgl.h" +#include "mozilla/UniquePtr.h" +#include "mozilla/WeakPtr.h" +#include "nsICanvasRenderingContextInternal.h" +#include "nsTArray.h" +#include "SurfaceTypes.h" +#include "ScopedGLHelpers.h" +#include "TexUnpackBlob.h" + +// Local +#include "CacheInvalidator.h" +#include "WebGLContextLossHandler.h" +#include "WebGLExtensions.h" +#include "WebGLObjectModel.h" +#include "WebGLStrongTypes.h" +#include "WebGLTypes.h" + +// Generated +#include "mozilla/dom/WebGLRenderingContextBinding.h" +#include "mozilla/dom/WebGL2RenderingContextBinding.h" + +#include <list> + +class nsIDocShell; + +// WebGL-only GLenums +// clang-format off +#define LOCAL_GL_BROWSER_DEFAULT_WEBGL 0x9244 +#define LOCAL_GL_CONTEXT_LOST_WEBGL 0x9242 +#define LOCAL_GL_MAX_CLIENT_WAIT_TIMEOUT_WEBGL 0x9247 +#define LOCAL_GL_UNPACK_COLORSPACE_CONVERSION_WEBGL 0x9243 +#define LOCAL_GL_UNPACK_FLIP_Y_WEBGL 0x9240 +#define LOCAL_GL_UNPACK_PREMULTIPLY_ALPHA_WEBGL 0x9241 +// clang-format on + +namespace mozilla { +class HostWebGLContext; +class ScopedCopyTexImageSource; +class ScopedDrawCallWrapper; +class ScopedResolveTexturesForDraw; +class WebGLBuffer; +class WebGLExtensionBase; +class WebGLFramebuffer; +class WebGLProgram; +class WebGLQuery; +class WebGLRenderbuffer; +class WebGLSampler; +class WebGLShader; +class WebGLSync; +class WebGLTexture; +class WebGLTransformFeedback; +class WebGLVertexArray; + +namespace dom { +class Document; +class Element; +class ImageData; +class OwningHTMLCanvasElementOrOffscreenCanvas; +struct WebGLContextAttributes; +} // namespace dom + +namespace gfx { +class SourceSurface; +class VRLayerChild; +} // namespace gfx + +namespace gl { +class GLScreenBuffer; +class MozFramebuffer; +class SharedSurface; +class Texture; +} // namespace gl + +namespace layers { +class CompositableHost; +class RemoteTextureOwnerClient; +class SurfaceDescriptor; +} // namespace layers + +namespace webgl { +class AvailabilityRunnable; +struct CachedDrawFetchLimits; +struct FbAttachInfo; +struct FormatInfo; +class FormatUsageAuthority; +struct FormatUsageInfo; +struct ImageInfo; +struct LinkedProgramInfo; +struct SamplerUniformInfo; +struct SamplingState; +class ScopedPrepForResourceClear; +class ShaderValidator; +class TexUnpackBlob; +struct UniformInfo; +struct UniformBlockInfo; +struct VertAttribPointerDesc; +} // namespace webgl + +struct WebGLTexImageData { + TexImageTarget mTarget; + int32_t mRowLength; + uint32_t mWidth; + uint32_t mHeight; + uint32_t mDepth; + gfxAlphaType mSrcAlphaType; +}; + +struct WebGLTexPboOffset { + TexImageTarget mTarget; + uint32_t mWidth; + uint32_t mHeight; + uint32_t mDepth; + WebGLsizeiptr mPboOffset; + bool mHasExpectedImageSize; + GLsizei mExpectedImageSize; +}; + +WebGLTexelFormat GetWebGLTexelFormat(TexInternalFormat format); + +void AssertUintParamCorrect(gl::GLContext* gl, GLenum pname, GLuint shadow); + +// From WebGLContextUtils +TexTarget TexImageTargetToTexTarget(TexImageTarget texImageTarget); + +struct WebGLIntOrFloat { + const enum { Int, Float, Uint } mType; + + union { + GLint i; + GLfloat f; + GLuint u; + } mValue; + + explicit WebGLIntOrFloat(GLint i) : mType(Int) { mValue.i = i; } + explicit WebGLIntOrFloat(GLfloat f) : mType(Float) { mValue.f = f; } + + GLint AsInt() const { + return (mType == Int) ? mValue.i : NS_lroundf(mValue.f); + } + GLfloat AsFloat() const { + return (mType == Float) ? mValue.f : GLfloat(mValue.i); + } +}; + +struct IndexedBufferBinding { + RefPtr<WebGLBuffer> mBufferBinding; + uint64_t mRangeStart; + uint64_t mRangeSize; + + IndexedBufferBinding(); + + uint64_t ByteCount() const; +}; + +//////////////////////////////////// + +namespace webgl { + +class AvailabilityRunnable final : public DiscardableRunnable { + public: + const WeakPtr<const ClientWebGLContext> mWebGL; + std::vector<WeakPtr<WebGLQueryJS>> mQueries; + std::vector<WeakPtr<WebGLSyncJS>> mSyncs; + + explicit AvailabilityRunnable(const ClientWebGLContext* webgl); + ~AvailabilityRunnable(); + + NS_IMETHOD Run() override; +}; + +struct BufferAndIndex final { + const WebGLBuffer* buffer = nullptr; + uint32_t id = -1; +}; + +} // namespace webgl + +//////////////////////////////////////////////////////////////////////////////// + +class WebGLContext : public VRefCounted, public SupportsWeakPtr { + friend class ScopedDrawCallWrapper; + friend class ScopedDrawWithTransformFeedback; + friend class ScopedFakeVertexAttrib0; + friend class ScopedFBRebinder; + friend class WebGL2Context; + friend class WebGLContextUserData; + friend class WebGLExtensionCompressedTextureASTC; + friend class WebGLExtensionCompressedTextureBPTC; + friend class WebGLExtensionCompressedTextureES3; + friend class WebGLExtensionCompressedTextureETC1; + friend class WebGLExtensionCompressedTexturePVRTC; + friend class WebGLExtensionCompressedTextureRGTC; + friend class WebGLExtensionCompressedTextureS3TC; + friend class WebGLExtensionCompressedTextureS3TC_SRGB; + friend class WebGLExtensionDepthTexture; + friend class WebGLExtensionDisjointTimerQuery; + friend class WebGLExtensionDrawBuffers; + friend class WebGLExtensionLoseContext; + friend class WebGLExtensionMOZDebug; + friend class WebGLExtensionVertexArray; + friend class WebGLMemoryTracker; + friend class webgl::AvailabilityRunnable; + friend struct webgl::LinkedProgramInfo; + friend struct webgl::SamplerUniformInfo; + friend class webgl::ScopedPrepForResourceClear; + friend struct webgl::UniformBlockInfo; + + friend const webgl::CachedDrawFetchLimits* ValidateDraw(WebGLContext*, GLenum, + uint32_t); + friend RefPtr<const webgl::LinkedProgramInfo> QueryProgramInfo( + WebGLProgram* prog, gl::GLContext* gl); + + MOZ_DECLARE_REFCOUNTED_VIRTUAL_TYPENAME(WebGLContext, override) + + enum { + UNPACK_FLIP_Y_WEBGL = 0x9240, + UNPACK_PREMULTIPLY_ALPHA_WEBGL = 0x9241, + // We throw InvalidOperation in TexImage if we fail to use GPU fast-path + // for texture copy when it is set to true, only for debug purpose. + UNPACK_REQUIRE_FASTPATH = 0x10001, + CONTEXT_LOST_WEBGL = 0x9242, + UNPACK_COLORSPACE_CONVERSION_WEBGL = 0x9243, + BROWSER_DEFAULT_WEBGL = 0x9244, + UNMASKED_VENDOR_WEBGL = 0x9245, + UNMASKED_RENDERER_WEBGL = 0x9246 + }; + + private: + class LruPosition final { + std::list<WebGLContext*>::iterator mItr; + + LruPosition(const LruPosition&) = delete; + LruPosition(LruPosition&&) = delete; + LruPosition& operator=(const LruPosition&) = delete; + LruPosition& operator=(LruPosition&&) = delete; + + public: + void AssignLocked(WebGLContext& aContext) MOZ_REQUIRES(sLruMutex); + void Reset(); + void ResetLocked() MOZ_REQUIRES(sLruMutex); + bool IsInsertedLocked() const MOZ_REQUIRES(sLruMutex); + + LruPosition(); + explicit LruPosition(WebGLContext&); + + ~LruPosition() { Reset(); } + }; + + mutable LruPosition mLruPosition MOZ_GUARDED_BY(sLruMutex); + + void BumpLruLocked() MOZ_REQUIRES(sLruMutex); + + public: + void BumpLru(); + void LoseLruContextIfLimitExceeded(); + + // - + + // We've had issues in the past with nulling `gl` without actually releasing + // all of our resources. This construction ensures that we are aware that we + // should only null `gl` in DestroyResourcesAndContext. + RefPtr<gl::GLContext> mGL_OnlyClearInDestroyResourcesAndContext; + + public: + // Grab a const reference so we can see changes, but can't make changes. + const decltype(mGL_OnlyClearInDestroyResourcesAndContext)& gl; + + public: + void CheckForInactivity(); + + protected: + const WeakPtr<HostWebGLContext> mHost; + const bool mResistFingerprinting; + WebGLContextOptions mOptions; + const uint32_t mPrincipalKey; + Maybe<webgl::Limits> mLimits; + const uint32_t mMaxVertIdsPerDraw = + StaticPrefs::webgl_max_vert_ids_per_draw(); + + bool mIsContextLost = false; + Atomic<bool> mPendingContextLoss; + webgl::ContextLossReason mPendingContextLossReason = + webgl::ContextLossReason::None; + const uint32_t mMaxPerfWarnings; + mutable uint64_t mNumPerfWarnings = 0; + const uint32_t mMaxAcceptableFBStatusInvals; + bool mWarnOnce_DepthTexCompareFilterable = true; + + uint64_t mNextFenceId = 1; + uint64_t mCompletedFenceId = 0; + + std::unique_ptr<gl::Texture> mIncompleteTexOverride; + + public: + class FuncScope; + + private: + mutable FuncScope* mFuncScope = nullptr; + + public: + static RefPtr<WebGLContext> Create(HostWebGLContext&, + const webgl::InitContextDesc&, + webgl::InitContextResult* out); + + private: + void FinishInit(); + + protected: + WebGLContext(HostWebGLContext&, const webgl::InitContextDesc&); + virtual ~WebGLContext(); + + RefPtr<layers::CompositableHost> mCompositableHost; + + layers::LayersBackend mBackend = layers::LayersBackend::LAYERS_NONE; + + public: + void Resize(uvec2 size); + + void SetCompositableHost(RefPtr<layers::CompositableHost>& aCompositableHost); + + /** + * An abstract base class to be implemented by callers wanting to be notified + * that a refresh has occurred. Callers must ensure an observer is removed + * before it is destroyed. + */ + virtual void DidRefresh(); + + void OnMemoryPressure(); + + // - + + /* + + Here are the bind calls that are supposed to be fully-validated client side, + so that client's binding state doesn't diverge: + * AttachShader + * DetachShader + * BindFramebuffer + * FramebufferAttach + * BindBuffer + * BindBufferRange + * BindTexture + * UseProgram + * BindSampler + * BindTransformFeedback + * BindVertexArray + * BeginQuery + * EndQuery + * ActiveTexture + + */ + + const auto& CurFuncScope() const { return *mFuncScope; } + const char* FuncName() const; + + class FuncScope final { + public: + const WebGLContext& mWebGL; + const char* const mFuncName; + bool mBindFailureGuard = false; + + public: + FuncScope(const WebGLContext& webgl, const char* funcName); + ~FuncScope(); + }; + + void GenerateErrorImpl(const GLenum err, const nsACString& text) const { + GenerateErrorImpl(err, std::string(text.BeginReading())); + } + void GenerateErrorImpl(const GLenum err, const std::string& text) const; + + void GenerateError(const webgl::ErrorInfo& err) { + GenerateError(err.type, "%s", err.info.c_str()); + } + + template <typename... Args> + void GenerateError(const GLenum err, const char* const fmt, + const Args&... args) const { + MOZ_ASSERT(FuncName()); + + nsCString text; + text.AppendPrintf("WebGL warning: %s: ", FuncName()); + +#ifdef __clang__ +# pragma clang diagnostic push +# pragma clang diagnostic ignored "-Wformat-security" +#elif defined(__GNUC__) +# pragma GCC diagnostic push +# pragma GCC diagnostic ignored "-Wformat-security" +#endif + text.AppendPrintf(fmt, args...); +#ifdef __clang__ +# pragma clang diagnostic pop +#elif defined(__GNUC__) +# pragma GCC diagnostic pop +#endif + + GenerateErrorImpl(err, text); + } + + template <typename... Args> + void ErrorInvalidEnum(const char* const fmt, const Args&... args) const { + GenerateError(LOCAL_GL_INVALID_ENUM, fmt, args...); + } + template <typename... Args> + void ErrorInvalidOperation(const char* const fmt, const Args&... args) const { + GenerateError(LOCAL_GL_INVALID_OPERATION, fmt, args...); + } + template <typename... Args> + void ErrorInvalidValue(const char* const fmt, const Args&... args) const { + GenerateError(LOCAL_GL_INVALID_VALUE, fmt, args...); + } + template <typename... Args> + void ErrorInvalidFramebufferOperation(const char* const fmt, + const Args&... args) const { + GenerateError(LOCAL_GL_INVALID_FRAMEBUFFER_OPERATION, fmt, args...); + } + template <typename... Args> + void ErrorOutOfMemory(const char* const fmt, const Args&... args) const { + GenerateError(LOCAL_GL_OUT_OF_MEMORY, fmt, args...); + } + + template <typename... Args> + void ErrorImplementationBug(const char* const fmt, + const Args&... args) const { + const nsPrintfCString newFmt( + "Implementation bug, please file at %s! %s", + "https://bugzilla.mozilla.org/" + "enter_bug.cgi?product=Core&component=Canvas%3A+WebGL", + fmt); + GenerateError(LOCAL_GL_OUT_OF_MEMORY, newFmt.BeginReading(), args...); + MOZ_ASSERT(false, "WebGLContext::ErrorImplementationBug"); + NS_ERROR("WebGLContext::ErrorImplementationBug"); + } + + void ErrorInvalidEnumInfo(const char* info, GLenum enumValue) const; + void ErrorInvalidEnumArg(const char* argName, GLenum val) const; + + static const char* ErrorName(GLenum error); + + /** + * Return displayable name for GLenum. + * This version is like gl::GLenumToStr but with out the GL_ prefix to + * keep consistency with how errors are reported from WebGL. + * Returns hex formatted version of glenum if glenum is unknown. + */ + static void EnumName(GLenum val, nsCString* out_name); + + void DummyReadFramebufferOperation(); + + WebGLTexture* GetActiveTex(const GLenum texTarget) const; + + gl::GLContext* GL() const { return gl; } + + bool IsPremultAlpha() const { return mOptions.premultipliedAlpha; } + + bool IsPreservingDrawingBuffer() const { + return mOptions.preserveDrawingBuffer; + } + + // Present to compositor + private: + bool PresentInto(gl::SwapChain& swapChain); + bool PresentIntoXR(gl::SwapChain& swapChain, const gl::MozFramebuffer& xrFb); + + public: + // Present swaps the front and back buffers of the swap chain for compositing. + // This assumes the framebuffer may directly alias with the back buffer, + // dependent on remoting state or other concerns. Framebuffer and swap chain + // surface formats are assumed to be similar to enable this aliasing. As such, + // the back buffer may be invalidated by this swap with the front buffer, + // unless overriden by explicitly setting the preserveDrawingBuffer option, + // which may incur a further copy to preserve the back buffer. + void Present( + WebGLFramebuffer*, layers::TextureType, const bool webvr, + const webgl::SwapChainOptions& options = webgl::SwapChainOptions()); + // CopyToSwapChain forces a copy from the supplied framebuffer into the back + // buffer before swapping the front and back buffers of the swap chain for + // compositing. The formats of the framebuffer and the swap chain buffers + // may differ subject to available format conversion options. Since this + // operation uses an explicit copy, it inherently preserves the framebuffer + // without need to set the preserveDrawingBuffer option. + void CopyToSwapChain( + WebGLFramebuffer*, layers::TextureType, + const webgl::SwapChainOptions& options = webgl::SwapChainOptions()); + // In use cases where a framebuffer is used as an offscreen framebuffer and + // does not need to be committed to the swap chain, it may still be useful + // for the implementation to delineate distinct frames, such as when sharing + // a single WebGLContext amongst many distinct users. EndOfFrame signals that + // frame rendering is complete so that any implementation side-effects such + // as resetting internal profile counters or resource queues may be handled + // appropriately. + void EndOfFrame(); + RefPtr<gfx::DataSourceSurface> GetFrontBufferSnapshot(); + Maybe<uvec2> FrontBufferSnapshotInto( + const Maybe<Range<uint8_t>> dest, + const Maybe<size_t> destStride = Nothing()); + Maybe<uvec2> FrontBufferSnapshotInto( + const std::shared_ptr<gl::SharedSurface>& front, + const Maybe<Range<uint8_t>> dest, + const Maybe<size_t> destStride = Nothing()); + Maybe<uvec2> SnapshotInto(GLuint srcFb, const gfx::IntSize& size, + const Range<uint8_t>& dest, + const Maybe<size_t> destStride = Nothing()); + gl::SwapChain* GetSwapChain(WebGLFramebuffer*, const bool webvr); + Maybe<layers::SurfaceDescriptor> GetFrontBuffer(WebGLFramebuffer*, + const bool webvr); + + void ClearVRSwapChain(); + + void RunContextLossTimer(); + void CheckForContextLoss(); + void HandlePendingContextLoss(); + + bool TryToRestoreContext(); + + void AssertCachedBindings() const; + void AssertCachedGlobalState() const; + + // WebIDL WebGLRenderingContext API + void Commit(); + + uvec2 DrawingBufferSize(); + + public: + void GetContextAttributes(dom::Nullable<dom::WebGLContextAttributes>& retval); + + // This is the entrypoint. Don't test against it directly. + bool IsContextLost() const { + auto* self = const_cast<WebGLContext*>(this); + if (self->mPendingContextLoss.exchange(false)) { + self->HandlePendingContextLoss(); + } + return mIsContextLost; + } + + // - + + RefPtr<WebGLBuffer> CreateBuffer(); + RefPtr<WebGLFramebuffer> CreateFramebuffer(); + RefPtr<WebGLFramebuffer> CreateOpaqueFramebuffer( + const webgl::OpaqueFramebufferOptions& options); + RefPtr<WebGLProgram> CreateProgram(); + RefPtr<WebGLQuery> CreateQuery(); + RefPtr<WebGLRenderbuffer> CreateRenderbuffer(); + RefPtr<WebGLShader> CreateShader(GLenum type); + RefPtr<WebGLTexture> CreateTexture(); + RefPtr<WebGLVertexArray> CreateVertexArray(); + + // - + + void AttachShader(WebGLProgram& prog, WebGLShader& shader); + void BindAttribLocation(WebGLProgram& prog, GLuint location, + const std::string& name) const; + void BindFramebuffer(GLenum target, WebGLFramebuffer* fb); + void BindRenderbuffer(GLenum target, WebGLRenderbuffer* fb); + void BindVertexArray(WebGLVertexArray* vao); + void BlendColor(GLclampf r, GLclampf g, GLclampf b, GLclampf a); + void BlendEquationSeparate(Maybe<GLuint> i, GLenum modeRGB, GLenum modeAlpha); + void BlendFuncSeparate(Maybe<GLuint> i, GLenum srcRGB, GLenum dstRGB, + GLenum srcAlpha, GLenum dstAlpha); + GLenum CheckFramebufferStatus(GLenum target); + void Clear(GLbitfield mask); + void ClearColor(GLclampf r, GLclampf g, GLclampf b, GLclampf a); + void ClearDepth(GLclampf v); + void ClearStencil(GLint v); + void ColorMask(Maybe<GLuint> i, uint8_t mask); + void CompileShader(WebGLShader& shader); + + private: + void CompileShaderANGLE(WebGLShader* shader); + void CompileShaderBypass(WebGLShader* shader, const nsCString& shaderSource); + + public: + void CullFace(GLenum face); + void DepthFunc(GLenum func); + void DepthMask(WebGLboolean b); + void DepthRange(GLclampf zNear, GLclampf zFar); + void DetachShader(WebGLProgram& prog, const WebGLShader& shader); + void DrawBuffers(const std::vector<GLenum>& buffers); + void Flush(); + void Finish(); + + void FramebufferAttach(GLenum target, GLenum attachSlot, + GLenum bindImageTarget, + const webgl::FbAttachInfo& toAttach); + + void FrontFace(GLenum mode); + + Maybe<double> GetBufferParameter(GLenum target, GLenum pname); + webgl::CompileResult GetCompileResult(const WebGLShader&) const; + GLenum GetError(); + GLint GetFragDataLocation(const WebGLProgram&, const std::string& name) const; + + Maybe<double> GetFramebufferAttachmentParameter(WebGLFramebuffer*, + GLenum attachment, + GLenum pname) const; + + Maybe<double> GetRenderbufferParameter(const WebGLRenderbuffer&, + GLenum pname) const; + webgl::LinkResult GetLinkResult(const WebGLProgram&) const; + + Maybe<webgl::ShaderPrecisionFormat> GetShaderPrecisionFormat( + GLenum shadertype, GLenum precisiontype) const; + + webgl::GetUniformData GetUniform(const WebGLProgram&, uint32_t loc) const; + + void Hint(GLenum target, GLenum mode); + + void LineWidth(GLfloat width); + void LinkProgram(WebGLProgram& prog); + void PolygonOffset(GLfloat factor, GLfloat units); + void ProvokingVertex(webgl::ProvokingVertex) const; + + //// + + webgl::PackingInfo ValidImplementationColorReadPI( + const webgl::FormatUsageInfo* usage) const; + + protected: + webgl::ReadPixelsResult ReadPixelsImpl(const webgl::ReadPixelsDesc&, + uintptr_t dest, uint64_t availBytes); + bool DoReadPixelsAndConvert(const webgl::FormatInfo* srcFormat, + const webgl::ReadPixelsDesc&, uintptr_t dest, + uint64_t dataLen, uint32_t rowStride); + + public: + void ReadPixelsPbo(const webgl::ReadPixelsDesc&, uint64_t offset); + webgl::ReadPixelsResult ReadPixelsInto(const webgl::ReadPixelsDesc&, + const Range<uint8_t>& dest); + + //// + + void RenderbufferStorageMultisample(WebGLRenderbuffer&, uint32_t samples, + GLenum internalformat, uint32_t width, + uint32_t height) const; + + public: + void SampleCoverage(GLclampf value, WebGLboolean invert); + void Scissor(GLint x, GLint y, GLsizei width, GLsizei height); + void ShaderSource(WebGLShader& shader, const std::string& source) const; + void StencilFuncSeparate(GLenum face, GLenum func, GLint ref, GLuint mask); + void StencilMaskSeparate(GLenum face, GLuint mask); + void StencilOpSeparate(GLenum face, GLenum sfail, GLenum dpfail, + GLenum dppass); + + ////////////////////////// + + void UniformData(uint32_t loc, bool transpose, + const Range<const webgl::UniformDataVal>& data) const; + + //////////////////////////////////// + + void UseProgram(WebGLProgram* prog); + + bool ValidateAttribArraySetter(uint32_t count, uint32_t arrayLength); + bool ValidateProgram(const WebGLProgram& prog) const; + void Viewport(GLint x, GLint y, GLsizei width, GLsizei height); + + // ----------------------------------------------------------------------------- + // Buffer Objects (WebGLContextBuffers.cpp) + void BindBuffer(GLenum target, WebGLBuffer* buffer); + void BindBufferRange(GLenum target, GLuint index, WebGLBuffer* buf, + uint64_t offset, uint64_t size); + + void BufferData(GLenum target, uint64_t dataLen, const uint8_t* data, + GLenum usage) const; + // The unsynchronized flag may allow for better performance when + // interleaving buffer updates with draw calls. However, care must + // be taken. This has similar semantics to glMapBufferRange's + // GL_MAP_UNSYNCHRONIZED_BIT: the results of any pending operations + // that reference the region of the buffer being updated are + // undefined. + void BufferSubData(GLenum target, uint64_t dstByteOffset, uint64_t srcDataLen, + const uint8_t* srcData, bool unsynchronized = false) const; + + protected: + // bound buffer state + RefPtr<WebGLBuffer> mBoundArrayBuffer; + RefPtr<WebGLBuffer> mBoundCopyReadBuffer; + RefPtr<WebGLBuffer> mBoundCopyWriteBuffer; + RefPtr<WebGLBuffer> mBoundPixelPackBuffer; + RefPtr<WebGLBuffer> mBoundPixelUnpackBuffer; + RefPtr<WebGLBuffer> mBoundTransformFeedbackBuffer; + RefPtr<WebGLBuffer> mBoundUniformBuffer; + + std::vector<IndexedBufferBinding> mIndexedUniformBufferBindings; + + RefPtr<WebGLBuffer>& GetBufferSlotByTarget(GLenum target); + RefPtr<WebGLBuffer>& GetBufferSlotByTargetIndexed(GLenum target, + GLuint index); + + // - + + void GenErrorIllegalUse(GLenum useTarget, uint32_t useId, GLenum boundTarget, + uint32_t boundId) const; + + bool ValidateBufferForNonTf(const WebGLBuffer&, GLenum nonTfTarget, + uint32_t nonTfId) const; + + bool ValidateBufferForNonTf(const WebGLBuffer* const nonTfBuffer, + const GLenum nonTfTarget, + const uint32_t nonTfId = -1) const { + if (!nonTfBuffer) return true; + return ValidateBufferForNonTf(*nonTfBuffer, nonTfTarget, nonTfId); + } + + bool ValidateBuffersForTf(const WebGLTransformFeedback&, + const webgl::LinkedProgramInfo&) const; + bool ValidateBuffersForTf( + const std::vector<webgl::BufferAndIndex>& tfBuffers) const; + + // ----------------------------------------------------------------------------- + // Queries (WebGL2ContextQueries.cpp) + protected: + RefPtr<WebGLQuery> mQuerySlot_SamplesPassed; + RefPtr<WebGLQuery> mQuerySlot_TFPrimsWritten; + RefPtr<WebGLQuery> mQuerySlot_TimeElapsed; + + RefPtr<WebGLQuery>* ValidateQuerySlotByTarget(GLenum target); + + public: + void BeginQuery(GLenum target, WebGLQuery& query); + void EndQuery(GLenum target); + Maybe<double> GetQueryParameter(const WebGLQuery& query, GLenum pname) const; + void QueryCounter(WebGLQuery&) const; + + // ----------------------------------------------------------------------------- + // State and State Requests (WebGLContextState.cpp) + void SetEnabled(GLenum cap, Maybe<GLuint> i, bool enabled); + bool GetStencilBits(GLint* const out_stencilBits) const; + + virtual Maybe<double> GetParameter(GLenum pname); + Maybe<std::string> GetString(GLenum pname) const; + + bool IsEnabled(GLenum cap); + + private: + static StaticMutex sLruMutex; + static std::list<WebGLContext*> sLru MOZ_GUARDED_BY(sLruMutex); + + // State tracking slots + bool mDitherEnabled = true; + bool mRasterizerDiscardEnabled = false; + bool mScissorTestEnabled = false; + bool mDepthTestEnabled = false; + bool mStencilTestEnabled = false; + GLenum mGenerateMipmapHint = LOCAL_GL_DONT_CARE; + + struct ScissorRect final { + GLint x; + GLint y; + GLsizei w; + GLsizei h; + + void Apply(gl::GLContext&) const; + }; + ScissorRect mScissorRect = {}; + + bool ValidateCapabilityEnum(GLenum cap); + bool* GetStateTrackingSlot(GLenum cap, GLuint i); + + // Allocation debugging variables + mutable uint64_t mDataAllocGLCallCount = 0; + + void OnDataAllocCall() const { mDataAllocGLCallCount++; } + + uint64_t GetNumGLDataAllocCalls() const { return mDataAllocGLCallCount; } + + void OnEndOfFrame(); + + // ----------------------------------------------------------------------------- + // Texture funcions (WebGLContextTextures.cpp) + public: + void ActiveTexture(uint32_t texUnit); + void BindTexture(GLenum texTarget, WebGLTexture* tex); + void GenerateMipmap(GLenum texTarget); + + Maybe<double> GetTexParameter(const WebGLTexture&, GLenum pname) const; + void TexParameter_base(GLenum texTarget, GLenum pname, + const FloatOrInt& param); + + virtual bool IsTexParamValid(GLenum pname) const; + + //////////////////////////////////// + // Uploads + + // CompressedTexSubImage if `sub` + void CompressedTexImage(bool sub, GLenum imageTarget, uint32_t level, + GLenum format, uvec3 offset, uvec3 size, + const Range<const uint8_t>& src, + const uint32_t pboImageSize, + const Maybe<uint64_t>& pboOffset) const; + + // CopyTexSubImage if `!respectFormat` + void CopyTexImage(GLenum imageTarget, uint32_t level, GLenum respecFormat, + uvec3 dstOffset, const ivec2& srcOffset, + const uvec2& size) const; + + // TexSubImage if `!respectFormat` + void TexImage(uint32_t level, GLenum respecFormat, uvec3 offset, + const webgl::PackingInfo& pi, + const webgl::TexUnpackBlobDesc&) const; + + void TexStorage(GLenum texTarget, uint32_t levels, GLenum sizedFormat, + uvec3 size) const; + + UniquePtr<webgl::TexUnpackBlob> ToTexUnpackBytes( + const WebGLTexImageData& imageData); + + UniquePtr<webgl::TexUnpackBytes> ToTexUnpackBytes(WebGLTexPboOffset& aPbo); + + //////////////////////////////////// + // WebGLTextureUpload.cpp + protected: + bool ValidateTexImageSpecification(uint8_t funcDims, GLenum texImageTarget, + GLint level, GLsizei width, GLsizei height, + GLsizei depth, GLint border, + TexImageTarget* const out_target, + WebGLTexture** const out_texture, + webgl::ImageInfo** const out_imageInfo); + bool ValidateTexImageSelection(uint8_t funcDims, GLenum texImageTarget, + GLint level, GLint xOffset, GLint yOffset, + GLint zOffset, GLsizei width, GLsizei height, + GLsizei depth, + TexImageTarget* const out_target, + WebGLTexture** const out_texture, + webgl::ImageInfo** const out_imageInfo); + bool ValidateUnpackInfo(bool usePBOs, GLenum format, GLenum type, + webgl::PackingInfo* const out); + + // ----------------------------------------------------------------------------- + // Vertices Feature (WebGLContextVertices.cpp) + GLenum mPrimRestartTypeBytes = 0; + + public: + void DrawArraysInstanced(GLenum mode, GLint first, GLsizei vertexCount, + GLsizei instanceCount); + void DrawElementsInstanced(GLenum mode, GLsizei vertexCount, GLenum type, + WebGLintptr byteOffset, GLsizei instanceCount); + + void EnableVertexAttribArray(GLuint index); + void DisableVertexAttribArray(GLuint index); + + Maybe<double> GetVertexAttrib(GLuint index, GLenum pname); + + //// + + void VertexAttrib4T(GLuint index, const webgl::TypedQuad&); + + //// + + void VertexAttribPointer(uint32_t index, const webgl::VertAttribPointerDesc&); + + void VertexAttribDivisor(GLuint index, GLuint divisor); + + private: + WebGLBuffer* DrawElements_check(GLsizei indexCount, GLenum type, + WebGLintptr byteOffset, + GLsizei instanceCount); + void Draw_cleanup(); + + void VertexAttrib1fv_base(GLuint index, uint32_t arrayLength, + const GLfloat* ptr); + void VertexAttrib2fv_base(GLuint index, uint32_t arrayLength, + const GLfloat* ptr); + void VertexAttrib3fv_base(GLuint index, uint32_t arrayLength, + const GLfloat* ptr); + void VertexAttrib4fv_base(GLuint index, uint32_t arrayLength, + const GLfloat* ptr); + + bool BindArrayAttribToLocation0(WebGLProgram* prog); + + // ----------------------------------------------------------------------------- + // PROTECTED + protected: + WebGLVertexAttrib0Status WhatDoesVertexAttrib0Need() const; + bool DoFakeVertexAttrib0(uint64_t fakeVertexCount, + WebGLVertexAttrib0Status whatDoesAttrib0Need); + void UndoFakeVertexAttrib0(); + + bool mResetLayer = true; + bool mOptionsFrozen = false; + bool mIsMesa = false; + bool mLoseContextOnMemoryPressure = false; + bool mCanLoseContextInForeground = true; + bool mShouldPresent = false; + bool mDisableFragHighP = false; + bool mForceResizeOnPresent = false; + bool mVRReady = false; + + template <typename WebGLObjectType> + void DeleteWebGLObjectsArray(nsTArray<WebGLObjectType>& array); + + GLuint mActiveTexture = 0; + GLenum mDefaultFB_DrawBuffer0 = LOCAL_GL_BACK; + GLenum mDefaultFB_ReadBuffer = LOCAL_GL_BACK; + + mutable GLenum mWebGLError = 0; + + std::unique_ptr<webgl::ShaderValidator> CreateShaderValidator( + GLenum shaderType) const; + + // some GL constants + uint32_t mGLMaxFragmentUniformVectors = 0; + uint32_t mGLMaxVertexUniformVectors = 0; + uint32_t mGLMaxVertexOutputVectors = 0; + uint32_t mGLMaxFragmentInputVectors = 0; + + uint32_t mGLMaxVertexTextureImageUnits = 0; + uint32_t mGLMaxFragmentTextureImageUnits = 0; + uint32_t mGLMaxCombinedTextureImageUnits = 0; + + // ES3: + uint32_t mGLMinProgramTexelOffset = 0; + uint32_t mGLMaxProgramTexelOffset = 0; + + public: + auto GLMaxDrawBuffers() const { return mLimits->maxColorDrawBuffers; } + + uint32_t MaxValidDrawBuffers() const { + if (IsWebGL2() || + IsExtensionEnabled(WebGLExtensionID::WEBGL_draw_buffers)) { + return GLMaxDrawBuffers(); + } + return 1; + } + + GLenum LastColorAttachmentEnum() const { + return LOCAL_GL_COLOR_ATTACHMENT0 + MaxValidDrawBuffers() - 1; + } + + const auto& Options() const { return mOptions; } + + protected: + uint32_t mGLMaxRenderbufferSize = 0; + + public: + const auto& Limits() const { return *mLimits; } + auto MaxVertexAttribs() const { return mLimits->maxVertexAttribs; } + auto GLMaxTextureUnits() const { return mLimits->maxTexUnits; } + + bool IsFormatValidForFB(TexInternalFormat format) const; + + protected: + // ------------------------------------------------------------------------- + // WebGL extensions (implemented in WebGLContextExtensions.cpp) + + EnumeratedArray<WebGLExtensionID, WebGLExtensionID::Max, + std::unique_ptr<WebGLExtensionBase>> + mExtensions; + + public: + void RequestExtension(WebGLExtensionID, bool explicitly = true); + + // returns true if the extension has been enabled by calling getExtension. + bool IsExtensionEnabled(const WebGLExtensionID id) const { + return bool(mExtensions[id]); + } + + bool IsExtensionExplicit(WebGLExtensionID) const; + void WarnIfImplicit(WebGLExtensionID) const; + + bool IsExtensionSupported(WebGLExtensionID) const; + + // ------------------------------------------------------------------------- + // WebGL 2 specifics (implemented in WebGL2Context.cpp) + public: + virtual bool IsWebGL2() const { return false; } + + struct FailureReason { + nsCString key; // For reporting. + nsCString info; + + FailureReason() = default; + + template <typename A, typename B> + FailureReason(const A& _key, const B& _info) + : key(nsCString(_key)), info(nsCString(_info)) {} + }; + + protected: + bool InitWebGL2(FailureReason* const out_failReason); + + bool CreateAndInitGL(bool forceEnabled, + std::vector<FailureReason>* const out_failReasons); + + // ------------------------------------------------------------------------- + // Validation functions (implemented in WebGLContextValidate.cpp) + bool InitAndValidateGL(FailureReason* const out_failReason); + + bool ValidateBlendEquationEnum(GLenum cap, const char* info); + bool ValidateBlendFuncEnumsCompatibility(GLenum sfactor, GLenum dfactor, + const char* info); + bool ValidateStencilOpEnum(GLenum action, const char* info); + bool ValidateFaceEnum(GLenum face); + bool ValidateTexInputData(GLenum type, js::Scalar::Type jsArrayType, + WebGLTexImageFunc func, WebGLTexDimensions dims); + bool ValidateAttribPointer(bool integerMode, GLuint index, GLint size, + GLenum type, WebGLboolean normalized, + GLsizei stride, WebGLintptr byteOffset, + const char* info); + bool ValidateStencilParamsForDrawCall() const; + + bool ValidateCopyTexImage(TexInternalFormat srcFormat, + TexInternalFormat dstformat, WebGLTexImageFunc func, + WebGLTexDimensions dims); + + bool ValidateTexImage(TexImageTarget texImageTarget, GLint level, + GLenum internalFormat, GLint xoffset, GLint yoffset, + GLint zoffset, GLint width, GLint height, GLint depth, + GLint border, GLenum format, GLenum type, + WebGLTexImageFunc func, WebGLTexDimensions dims); + bool ValidateTexImageFormat(GLenum internalFormat, WebGLTexImageFunc func, + WebGLTexDimensions dims); + bool ValidateTexImageType(GLenum type, WebGLTexImageFunc func, + WebGLTexDimensions dims); + bool ValidateTexImageFormatAndType(GLenum format, GLenum type, + WebGLTexImageFunc func, + WebGLTexDimensions dims); + bool ValidateCompTexImageInternalFormat(GLenum format, WebGLTexImageFunc func, + WebGLTexDimensions dims); + bool ValidateCopyTexImageInternalFormat(GLenum format, WebGLTexImageFunc func, + WebGLTexDimensions dims); + bool ValidateTexImageSize(TexImageTarget texImageTarget, GLint level, + GLint width, GLint height, GLint depth, + WebGLTexImageFunc func, WebGLTexDimensions dims); + bool ValidateTexSubImageSize(GLint x, GLint y, GLint z, GLsizei width, + GLsizei height, GLsizei depth, GLsizei baseWidth, + GLsizei baseHeight, GLsizei baseDepth, + WebGLTexImageFunc func, WebGLTexDimensions dims); + bool ValidateCompTexImageSize(GLint level, GLenum internalFormat, + GLint xoffset, GLint yoffset, GLsizei width, + GLsizei height, GLsizei levelWidth, + GLsizei levelHeight, WebGLTexImageFunc func, + WebGLTexDimensions dims); + bool ValidateCompTexImageDataSize(GLint level, GLenum internalFormat, + GLsizei width, GLsizei height, + uint32_t byteLength, WebGLTexImageFunc func, + WebGLTexDimensions dims); + + bool HasDrawBuffers() const { + return IsWebGL2() || + IsExtensionEnabled(WebGLExtensionID::WEBGL_draw_buffers); + } + + RefPtr<WebGLBuffer>* ValidateBufferSlot(GLenum target); + + public: + WebGLBuffer* ValidateBufferSelection(GLenum target) const; + + protected: + IndexedBufferBinding* ValidateIndexedBufferSlot(GLenum target, GLuint index); + + bool ValidateIndexedBufferBinding( + GLenum target, GLuint index, + RefPtr<WebGLBuffer>** const out_genericBinding, + IndexedBufferBinding** const out_indexedBinding); + + public: + bool ValidateNonNegative(const char* argName, int64_t val) const { + if (MOZ_UNLIKELY(val < 0)) { + ErrorInvalidValue("`%s` must be non-negative.", argName); + return false; + } + return true; + } + + template <typename T> + bool ValidateNonNull(const char* const argName, + const dom::Nullable<T>& maybe) const { + if (maybe.IsNull()) { + ErrorInvalidValue("%s: Cannot be null.", argName); + return false; + } + return true; + } + + //// + + protected: + void DestroyResourcesAndContext(); + + // helpers + + bool ConvertImage(size_t width, size_t height, size_t srcStride, + size_t dstStride, const uint8_t* src, uint8_t* dst, + WebGLTexelFormat srcFormat, bool srcPremultiplied, + WebGLTexelFormat dstFormat, bool dstPremultiplied, + size_t dstTexelSize); + + ////// + public: + template <typename T> + bool ValidateObject(const char* const argName, const T& object) const { + // Todo: Remove all callers. + return true; + } + + template <typename T> + bool ValidateObject(const char* const argName, const T* const object) const { + // Todo: Remove most (all?) callers. + if (!object) { + ErrorInvalidOperation( + "%s: Object argument cannot have been marked for" + " deletion.", + argName); + return false; + } + return true; + } + + //// + + private: + void LoseContextLruLocked(webgl::ContextLossReason reason) + MOZ_REQUIRES(sLruMutex); + + public: + void LoseContext( + webgl::ContextLossReason reason = webgl::ContextLossReason::None); + + protected: + nsTArray<RefPtr<WebGLTexture>> mBound2DTextures; + nsTArray<RefPtr<WebGLTexture>> mBoundCubeMapTextures; + nsTArray<RefPtr<WebGLTexture>> mBound3DTextures; + nsTArray<RefPtr<WebGLTexture>> mBound2DArrayTextures; + nsTArray<RefPtr<WebGLSampler>> mBoundSamplers; + + void ResolveTexturesForDraw() const; + + RefPtr<WebGLProgram> mCurrentProgram; + RefPtr<const webgl::LinkedProgramInfo> mActiveProgramLinkInfo; + + bool ValidateFramebufferTarget(GLenum target) const; + bool ValidateInvalidateFramebuffer(GLenum target, + const Range<const GLenum>& attachments, + std::vector<GLenum>* const scopedVector, + GLsizei* const out_glNumAttachments, + const GLenum** const out_glAttachments); + + RefPtr<WebGLFramebuffer> mBoundDrawFramebuffer; + RefPtr<WebGLFramebuffer> mBoundReadFramebuffer; + RefPtr<WebGLTransformFeedback> mBoundTransformFeedback; + RefPtr<WebGLVertexArray> mBoundVertexArray; + + public: + const auto& BoundReadFb() const { return mBoundReadFramebuffer; } + + protected: + RefPtr<WebGLTransformFeedback> mDefaultTransformFeedback; + RefPtr<WebGLVertexArray> mDefaultVertexArray; + + //////////////////////////////////// + + protected: + GLuint mEmptyTFO = 0; + + // Generic Vertex Attributes + // Though CURRENT_VERTEX_ATTRIB is listed under "Vertex Shader State" in the + // spec state tables, this isn't vertex shader /object/ state. This array is + // merely state useful to vertex shaders, but is global state. + std::vector<webgl::AttribBaseType> mGenericVertexAttribTypes; + CacheInvalidator mGenericVertexAttribTypeInvalidator; + + GLuint mFakeVertexAttrib0BufferObject = 0; + intptr_t mFakeVertexAttrib0BufferObjectSize = 0; + bool mFakeVertexAttrib0DataDefined = false; + alignas(alignof(float)) uint8_t + mGenericVertexAttrib0Data[sizeof(float) * 4] = {}; + alignas(alignof(float)) uint8_t + mFakeVertexAttrib0Data[sizeof(float) * 4] = {}; + + GLint mStencilRefFront = 0; + GLint mStencilRefBack = 0; + GLuint mStencilValueMaskFront = 0; + GLuint mStencilValueMaskBack = 0; + GLuint mStencilWriteMaskFront = 0; + GLuint mStencilWriteMaskBack = 0; + uint8_t mColorWriteMask0 = 0xf; // bitmask + mutable uint8_t mDriverColorMask0 = 0xf; + bool mDepthWriteMask = true; + GLfloat mColorClearValue[4] = {0, 0, 0, 0}; + GLint mStencilClearValue = 0; + GLfloat mDepthClearValue = 1.0f; + + std::bitset<webgl::kMaxDrawBuffers> mColorWriteMaskNonzero = -1; + std::bitset<webgl::kMaxDrawBuffers> mBlendEnabled = 0; + + GLint mViewportX = 0; + GLint mViewportY = 0; + GLsizei mViewportWidth = 0; + GLsizei mViewportHeight = 0; + bool mAlreadyWarnedAboutViewportLargerThanDest = false; + + GLfloat mLineWidth = 1.0; + + WebGLContextLossHandler mContextLossHandler; + + // Used for some hardware (particularly Tegra 2 and 4) that likes to + // be Flushed while doing hundreds of draw calls. + mutable uint64_t mDrawCallsSinceLastFlush = 0; + + mutable uint64_t mWarningCount = 0; + const uint64_t mMaxWarnings; + bool mAlreadyWarnedAboutFakeVertexAttrib0 = false; + + bool ShouldGenerateWarnings() const { return mWarningCount < mMaxWarnings; } + + bool ShouldGeneratePerfWarnings() const { + return mNumPerfWarnings < mMaxPerfWarnings; + } + + bool mNeedsFakeNoAlpha = false; + bool mNeedsFakeNoDepth = false; + bool mNeedsFakeNoStencil = false; + bool mNeedsFakeNoStencil_UserFBs = false; + + bool mDriverDepthTest = false; + bool mDriverStencilTest = false; + + bool mNeedsLegacyVertexAttrib0Handling = false; + bool mMaybeNeedsLegacyVertexAttrib0Handling = false; + bool mNeedsIndexValidation = false; + bool mBug_DrawArraysInstancedUserAttribFetchAffectedByFirst = false; + + const bool mAllowFBInvalidation; + + bool Has64BitTimestamps() const; + + // -- + + const uint8_t mMsaaSamples; + mutable uvec2 mRequestedSize; + mutable UniquePtr<gl::MozFramebuffer> mDefaultFB; + mutable bool mDefaultFB_IsInvalid = false; + mutable UniquePtr<gl::MozFramebuffer> mResolvedDefaultFB; + + gl::SwapChain mSwapChain; + gl::SwapChain mWebVRSwapChain; + + RefPtr<layers::RemoteTextureOwnerClient> mRemoteTextureOwner; + + bool PushRemoteTexture(WebGLFramebuffer*, gl::SwapChain&, + std::shared_ptr<gl::SharedSurface>, + const webgl::SwapChainOptions& options); + + // -- + + bool EnsureDefaultFB(); + bool ValidateAndInitFB( + const WebGLFramebuffer* fb, + GLenum incompleteFbError = LOCAL_GL_INVALID_FRAMEBUFFER_OPERATION); + void DoBindFB(const WebGLFramebuffer* fb, + GLenum target = LOCAL_GL_FRAMEBUFFER) const; + + bool BindCurFBForDraw(); + bool BindCurFBForColorRead( + const webgl::FormatUsageInfo** out_format, uint32_t* out_width, + uint32_t* out_height, + GLenum incompleteFbError = LOCAL_GL_INVALID_FRAMEBUFFER_OPERATION); + void DoColorMask(Maybe<GLuint> i, uint8_t bitmask) const; + void BlitBackbufferToCurDriverFB( + WebGLFramebuffer* const srcAsWebglFb = nullptr, + const gl::MozFramebuffer* const srcAsMozFb = nullptr, + bool srcIsBGRA = false) const; + bool BindDefaultFBForRead(); + + // -- + + public: + // console logging helpers + template <typename... Args> + void GenerateWarning(const char* const fmt, const Args&... args) const { + GenerateError(0, fmt, args...); + } + + template <typename... Args> + void GeneratePerfWarning(const char* const fmt, const Args&... args) const { + GenerateError(webgl::kErrorPerfWarning, fmt, args...); + } + + public: + UniquePtr<webgl::FormatUsageAuthority> mFormatUsage; + + virtual UniquePtr<webgl::FormatUsageAuthority> CreateFormatUsage( + gl::GLContext* gl) const; + + const decltype(mBound2DTextures)* TexListForElemType(GLenum elemType) const; + + // Friend list + friend class ScopedCopyTexImageSource; + friend class ScopedResolveTexturesForDraw; + friend class webgl::TexUnpackBlob; + friend class webgl::TexUnpackBytes; + friend class webgl::TexUnpackImage; + friend class webgl::TexUnpackSurface; + friend struct webgl::UniformInfo; + friend class WebGLTexture; + friend class WebGLFBAttachPoint; + friend class WebGLFramebuffer; + friend class WebGLRenderbuffer; + friend class WebGLProgram; + friend class WebGLQuery; + friend class WebGLBuffer; + friend class WebGLSampler; + friend class WebGLShader; + friend class WebGLSync; + friend class WebGLTransformFeedback; + friend class WebGLVertexArray; + friend class WebGLVertexArrayFake; + friend class WebGLVertexArrayGL; +}; + +// Returns `value` rounded to the next highest multiple of `multiple`. +// AKA PadToAlignment, StrideForAlignment. +template <typename V, typename M> +V RoundUpToMultipleOf(const V& value, const M& multiple) { + return ((value + multiple - 1) / multiple) * multiple; +} + +class ScopedFBRebinder final { + private: + const WebGLContext* const mWebGL; + + public: + explicit ScopedFBRebinder(const WebGLContext* const webgl) : mWebGL(webgl) {} + ~ScopedFBRebinder(); +}; + +// - + +constexpr inline bool IsBufferTargetLazilyBound(const GLenum target) { + return target != LOCAL_GL_ELEMENT_ARRAY_BUFFER; +} + +void DoBindBuffer(gl::GLContext&, GLenum target, const WebGLBuffer*); + +class ScopedLazyBind final { + private: + gl::GLContext& mGL; + const GLenum mTarget; + + public: + ScopedLazyBind(gl::GLContext* const gl, const GLenum target, + const WebGLBuffer* const buf) + : mGL(*gl), mTarget(IsBufferTargetLazilyBound(target) ? target : 0) { + if (mTarget) { + DoBindBuffer(mGL, mTarget, buf); + } + } + + ~ScopedLazyBind() { + if (mTarget) { + DoBindBuffer(mGL, mTarget, nullptr); + } + } +}; + +//// + +bool Intersect(int32_t srcSize, int32_t read0, int32_t readSize, + int32_t* out_intRead0, int32_t* out_intWrite0, + int32_t* out_intSize); + +uint64_t AvailGroups(uint64_t totalAvailItems, uint64_t firstItemOffset, + uint32_t groupSize, uint32_t groupStride); + +//// + +class ScopedDrawCallWrapper final { + public: + WebGLContext& mWebGL; + + explicit ScopedDrawCallWrapper(WebGLContext& webgl); + ~ScopedDrawCallWrapper(); +}; + +namespace webgl { + +class ScopedPrepForResourceClear final { + const WebGLContext& webgl; + + public: + explicit ScopedPrepForResourceClear(const WebGLContext&); + ~ScopedPrepForResourceClear(); +}; + +struct IndexedName final { + std::string name; + uint64_t index; +}; +Maybe<IndexedName> ParseIndexed(const std::string& str); + +} // namespace webgl + +webgl::LinkActiveInfo GetLinkActiveInfo( + gl::GLContext& gl, const GLuint prog, const bool webgl2, + const std::unordered_map<std::string, std::string>& nameUnmap); + +} // namespace mozilla + +#endif |