diff options
Diffstat (limited to '')
-rw-r--r-- | dom/canvas/WebGLTexture.cpp | 1112 |
1 files changed, 1112 insertions, 0 deletions
diff --git a/dom/canvas/WebGLTexture.cpp b/dom/canvas/WebGLTexture.cpp new file mode 100644 index 0000000000..7b50e85ed3 --- /dev/null +++ b/dom/canvas/WebGLTexture.cpp @@ -0,0 +1,1112 @@ +/* -*- Mode: C++; tab-width: 20; 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/. */ + +#include "WebGLTexture.h" + +#include <algorithm> +#include "GLContext.h" +#include "mozilla/Casting.h" +#include "mozilla/dom/WebGLRenderingContextBinding.h" +#include "mozilla/gfx/Logging.h" +#include "mozilla/IntegerRange.h" +#include "mozilla/MathAlgorithms.h" +#include "mozilla/Scoped.h" +#include "mozilla/ScopeExit.h" +#include "mozilla/Unused.h" +#include "ScopedGLHelpers.h" +#include "WebGLContext.h" +#include "WebGLContextUtils.h" +#include "WebGLFormats.h" +#include "WebGLFramebuffer.h" +#include "WebGLSampler.h" +#include "WebGLTexelConversions.h" + +namespace mozilla { +namespace webgl { + +/*static*/ const ImageInfo ImageInfo::kUndefined; + +size_t ImageInfo::MemoryUsage() const { + if (!IsDefined()) return 0; + + size_t samples = mSamples; + if (!samples) { + samples = 1; + } + + const size_t bytesPerTexel = mFormat->format->estimatedBytesPerPixel; + return size_t(mWidth) * size_t(mHeight) * size_t(mDepth) * samples * + bytesPerTexel; +} + +Maybe<ImageInfo> ImageInfo::NextMip(const GLenum target) const { + MOZ_ASSERT(IsDefined()); + + auto next = *this; + + if (target == LOCAL_GL_TEXTURE_3D) { + if (mWidth <= 1 && mHeight <= 1 && mDepth <= 1) { + return {}; + } + + next.mDepth = std::max(uint32_t(1), next.mDepth / 2); + } else { + // TEXTURE_2D_ARRAY may have depth != 1, but that's normal. + if (mWidth <= 1 && mHeight <= 1) { + return {}; + } + } + if (next.mUninitializedSlices) { + next.mUninitializedSlices = Some(std::vector<bool>(next.mDepth, true)); + } + + next.mWidth = std::max(uint32_t(1), next.mWidth / 2); + next.mHeight = std::max(uint32_t(1), next.mHeight / 2); + return Some(next); +} + +} // namespace webgl + +//////////////////////////////////////// + +WebGLTexture::WebGLTexture(WebGLContext* webgl, GLuint tex) + : WebGLContextBoundObject(webgl), + mGLName(tex), + mTarget(LOCAL_GL_NONE), + mFaceCount(0), + mImmutable(false), + mImmutableLevelCount(0), + mBaseMipmapLevel(0), + mMaxMipmapLevel(1000) {} + +WebGLTexture::~WebGLTexture() { + for (auto& cur : mImageInfoArr) { + cur = webgl::ImageInfo(); + } + InvalidateCaches(); + + if (!mContext) return; + mContext->gl->fDeleteTextures(1, &mGLName); +} + +size_t WebGLTexture::MemoryUsage() const { + size_t accum = 0; + for (const auto& cur : mImageInfoArr) { + accum += cur.MemoryUsage(); + } + return accum; +} + +// --------------------------- + +void WebGLTexture::PopulateMipChain(const uint32_t maxLevel) { + // Used by GenerateMipmap and TexStorage. + // Populates based on mBaseMipmapLevel. + + auto ref = BaseImageInfo(); + MOZ_ASSERT(ref.mWidth && ref.mHeight && ref.mDepth); + + for (auto level = mBaseMipmapLevel; level <= maxLevel; ++level) { + // GLES 3.0.4, p161 + // "A cube map texture is mipmap complete if each of the six texture images, + // considered individually, is mipmap complete." + + for (uint8_t face = 0; face < mFaceCount; face++) { + auto& cur = ImageInfoAtFace(face, level); + cur = ref; + } + + const auto next = ref.NextMip(mTarget.get()); + if (!next) break; + ref = next.ref(); + } + InvalidateCaches(); +} + +static bool ZeroTextureData(const WebGLContext* webgl, GLuint tex, + TexImageTarget target, uint32_t level, + const webgl::ImageInfo& info); + +bool WebGLTexture::IsMipAndCubeComplete(const uint32_t maxLevel, + const bool ensureInit, + bool* const out_initFailed) const { + *out_initFailed = false; + + // Reference dimensions based on baseLevel. + auto ref = BaseImageInfo(); + MOZ_ASSERT(ref.mWidth && ref.mHeight && ref.mDepth); + + for (auto level = mBaseMipmapLevel; level <= maxLevel; ++level) { + // GLES 3.0.4, p161 + // "A cube map texture is mipmap complete if each of the six texture images, + // considered individually, is mipmap complete." + + for (uint8_t face = 0; face < mFaceCount; face++) { + auto& cur = ImageInfoAtFace(face, level); + + // "* The set of mipmap arrays `level_base` through `q` (where `q` + // is defined the "Mipmapping" discussion of section 3.8.10) were + // each specified with the same effective internal format." + + // "* The dimensions of the arrays follow the sequence described in + // the "Mipmapping" discussion of section 3.8.10." + + if (cur.mWidth != ref.mWidth || cur.mHeight != ref.mHeight || + cur.mDepth != ref.mDepth || cur.mFormat != ref.mFormat) { + return false; + } + + if (MOZ_UNLIKELY(ensureInit && cur.mUninitializedSlices)) { + auto imageTarget = mTarget.get(); + if (imageTarget == LOCAL_GL_TEXTURE_CUBE_MAP) { + imageTarget = LOCAL_GL_TEXTURE_CUBE_MAP_POSITIVE_X + face; + } + if (!ZeroTextureData(mContext, mGLName, imageTarget, level, cur)) { + mContext->ErrorOutOfMemory("Failed to zero tex image data."); + *out_initFailed = true; + return false; + } + cur.mUninitializedSlices = Nothing(); + } + } + + const auto next = ref.NextMip(mTarget.get()); + if (!next) break; + ref = next.ref(); + } + + return true; +} + +Maybe<const WebGLTexture::CompletenessInfo> WebGLTexture::CalcCompletenessInfo( + const bool ensureInit, const bool skipMips) const { + Maybe<CompletenessInfo> ret = Some(CompletenessInfo()); + + // - + + const auto level_base = Es3_level_base(); + if (level_base > kMaxLevelCount - 1) { + ret->incompleteReason = "`level_base` too high."; + return ret; + } + + // Texture completeness is established at GLES 3.0.4, p160-161. + // "[A] texture is complete unless any of the following conditions hold true:" + + // "* Any dimension of the `level_base` array is not positive." + const auto& baseImageInfo = ImageInfoAtFace(0, level_base); + if (!baseImageInfo.IsDefined()) { + // In case of undefined texture image, we don't print any message because + // this is a very common and often legitimate case (asynchronous texture + // loading). + ret->incompleteReason = nullptr; + return ret; + } + + if (!baseImageInfo.mWidth || !baseImageInfo.mHeight || + !baseImageInfo.mDepth) { + ret->incompleteReason = + "The dimensions of `level_base` are not all positive."; + return ret; + } + + // "* The texture is a cube map texture, and is not cube complete." + bool initFailed = false; + if (!IsMipAndCubeComplete(level_base, ensureInit, &initFailed)) { + if (initFailed) return {}; + + // Can only fail if not cube-complete. + ret->incompleteReason = "Cubemaps must be \"cube complete\"."; + return ret; + } + ret->levels = 1; + ret->usage = baseImageInfo.mFormat; + RefreshSwizzle(); + + ret->powerOfTwo = mozilla::IsPowerOfTwo(baseImageInfo.mWidth) && + mozilla::IsPowerOfTwo(baseImageInfo.mHeight); + if (mTarget == LOCAL_GL_TEXTURE_3D) { + ret->powerOfTwo &= mozilla::IsPowerOfTwo(baseImageInfo.mDepth); + } + + // - + + if (!mContext->IsWebGL2() && !ret->powerOfTwo) { + // WebGL 1 mipmaps require POT. + ret->incompleteReason = "Mipmapping requires power-of-two sizes."; + return ret; + } + + // "* `level_base <= level_max`" + + const auto level_max = Es3_level_max(); + const auto maxLevel_aka_q = Es3_q(); + if (level_base > level_max) { // `level_max` not `q`! + ret->incompleteReason = "`level_base > level_max`."; + return ret; + } + + if (skipMips) return ret; + + if (!IsMipAndCubeComplete(maxLevel_aka_q, ensureInit, &initFailed)) { + if (initFailed) return {}; + + ret->incompleteReason = "Bad mipmap dimension or format."; + return ret; + } + ret->levels = AutoAssertCast(maxLevel_aka_q - level_base + 1); + ret->mipmapComplete = true; + + // - + + return ret; +} + +Maybe<const webgl::SampleableInfo> WebGLTexture::CalcSampleableInfo( + const WebGLSampler* const sampler) const { + Maybe<webgl::SampleableInfo> ret = Some(webgl::SampleableInfo()); + + const bool ensureInit = true; + const auto completeness = CalcCompletenessInfo(ensureInit); + if (!completeness) return {}; + + ret->incompleteReason = completeness->incompleteReason; + + if (!completeness->levels) return ret; + + const auto* sampling = &mSamplingState; + if (sampler) { + sampling = &sampler->State(); + } + const auto isDepthTex = bool(completeness->usage->format->d); + ret->isDepthTexCompare = isDepthTex & bool(sampling->compareMode.get()); + // Because if it's not a depth texture, we always ignore compareMode. + + const auto& minFilter = sampling->minFilter; + const auto& magFilter = sampling->magFilter; + + // - + + const bool needsMips = (minFilter == LOCAL_GL_NEAREST_MIPMAP_NEAREST || + minFilter == LOCAL_GL_NEAREST_MIPMAP_LINEAR || + minFilter == LOCAL_GL_LINEAR_MIPMAP_NEAREST || + minFilter == LOCAL_GL_LINEAR_MIPMAP_LINEAR); + if (needsMips & !completeness->mipmapComplete) return ret; + + const bool isMinFilteringNearest = + (minFilter == LOCAL_GL_NEAREST || + minFilter == LOCAL_GL_NEAREST_MIPMAP_NEAREST); + const bool isMagFilteringNearest = (magFilter == LOCAL_GL_NEAREST); + const bool isFilteringNearestOnly = + (isMinFilteringNearest && isMagFilteringNearest); + if (!isFilteringNearestOnly) { + bool isFilterable = completeness->usage->isFilterable; + + // "* The effective internal format specified for the texture arrays is a + // sized internal depth or depth and stencil format, the value of + // TEXTURE_COMPARE_MODE is NONE[1], and either the magnification filter + // is not NEAREST, or the minification filter is neither NEAREST nor + // NEAREST_MIPMAP_NEAREST." + // [1]: This sounds suspect, but is explicitly noted in the change log for + // GLES 3.0.1: + // "* Clarify that a texture is incomplete if it has a depth component, + // no shadow comparison, and linear filtering (also Bug 9481)." + // In short, depth formats are not filterable, but shadow-samplers are. + if (ret->isDepthTexCompare) { + isFilterable = true; + + if (mContext->mWarnOnce_DepthTexCompareFilterable) { + mContext->mWarnOnce_DepthTexCompareFilterable = false; + mContext->GenerateWarning( + "Depth texture comparison requests (e.g. `LINEAR`) Filtering, but" + " behavior is implementation-defined, and so on some systems will" + " sometimes behave as `NEAREST`. (warns once)"); + } + } + + // "* The effective internal format specified for the texture arrays is a + // sized internal color format that is not texture-filterable, and either + // the magnification filter is not NEAREST or the minification filter is + // neither NEAREST nor NEAREST_MIPMAP_NEAREST." + // Since all (GLES3) unsized color formats are filterable just like their + // sized equivalents, we don't have to care whether its sized or not. + if (!isFilterable) { + ret->incompleteReason = + "Minification or magnification filtering is not" + " NEAREST or NEAREST_MIPMAP_NEAREST, and the" + " texture's format is not \"texture-filterable\"."; + return ret; + } + } + + // Texture completeness is effectively (though not explicitly) amended for + // GLES2 by the "Texture Access" section under $3.8 "Fragment Shaders". This + // also applies to vertex shaders, as noted on GLES 2.0.25, p41. + if (!mContext->IsWebGL2() && !completeness->powerOfTwo) { + // GLES 2.0.25, p87-88: + // "Calling a sampler from a fragment shader will return (R,G,B,A)=(0,0,0,1) + // if + // any of the following conditions are true:" + + // "* A two-dimensional sampler is called, the minification filter is one + // that requires a mipmap[...], and the sampler's associated texture + // object is not complete[.]" + // (already covered) + + // "* A two-dimensional sampler is called, the minification filter is + // not one that requires a mipmap (either NEAREST nor[sic] LINEAR), and + // either dimension of the level zero array of the associated texture + // object is not positive." + // (already covered) + + // "* A two-dimensional sampler is called, the corresponding texture + // image is a non-power-of-two image[...], and either the texture wrap + // mode is not CLAMP_TO_EDGE, or the minification filter is neither + // NEAREST nor LINEAR." + + // "* A cube map sampler is called, any of the corresponding texture + // images are non-power-of-two images, and either the texture wrap mode + // is not CLAMP_TO_EDGE, or the minification filter is neither NEAREST + // nor LINEAR." + // "either the texture wrap mode is not CLAMP_TO_EDGE" + if (sampling->wrapS != LOCAL_GL_CLAMP_TO_EDGE || + sampling->wrapT != LOCAL_GL_CLAMP_TO_EDGE) { + ret->incompleteReason = + "Non-power-of-two textures must have a wrap mode of" + " CLAMP_TO_EDGE."; + return ret; + } + + // "* A cube map sampler is called, and either the corresponding cube + // map texture image is not cube complete, or TEXTURE_MIN_FILTER is one + // that requires a mipmap and the texture is not mipmap cube complete." + // (already covered) + } + + // Mark complete. + ret->incompleteReason = + nullptr; // NB: incompleteReason is also null for undefined + ret->levels = completeness->levels; // textures. + if (!needsMips && ret->levels) { + ret->levels = 1; + } + ret->usage = completeness->usage; + return ret; +} + +const webgl::SampleableInfo* WebGLTexture::GetSampleableInfo( + const WebGLSampler* const sampler) const { + auto itr = mSamplingCache.Find(sampler); + if (!itr) { + const auto info = CalcSampleableInfo(sampler); + if (!info) return nullptr; + + auto entry = mSamplingCache.MakeEntry(sampler, info.value()); + entry->AddInvalidator(*this); + if (sampler) { + entry->AddInvalidator(*sampler); + } + itr = mSamplingCache.Insert(std::move(entry)); + } + return itr; +} + +// --------------------------- + +uint32_t WebGLTexture::Es3_q() const { + const auto& imageInfo = BaseImageInfo(); + if (!imageInfo.IsDefined()) return mBaseMipmapLevel; + + uint32_t largestDim = std::max(imageInfo.mWidth, imageInfo.mHeight); + if (mTarget == LOCAL_GL_TEXTURE_3D) { + largestDim = std::max(largestDim, imageInfo.mDepth); + } + if (!largestDim) return mBaseMipmapLevel; + + // GLES 3.0.4, 3.8 - Mipmapping: `floor(log2(largest_of_dims)) + 1` + const auto numLevels = FloorLog2Size(largestDim) + 1; + + const auto maxLevelBySize = mBaseMipmapLevel + numLevels - 1; + return std::min<uint32_t>(maxLevelBySize, mMaxMipmapLevel); +} + +// - + +static void SetSwizzle(gl::GLContext* gl, TexTarget target, + const GLint* swizzle) { + static const GLint kNoSwizzle[4] = {LOCAL_GL_RED, LOCAL_GL_GREEN, + LOCAL_GL_BLUE, LOCAL_GL_ALPHA}; + if (!swizzle) { + swizzle = kNoSwizzle; + } else if (!gl->IsSupported(gl::GLFeature::texture_swizzle)) { + MOZ_CRASH("GFX: Needs swizzle feature to swizzle!"); + } + + gl->fTexParameteri(target.get(), LOCAL_GL_TEXTURE_SWIZZLE_R, swizzle[0]); + gl->fTexParameteri(target.get(), LOCAL_GL_TEXTURE_SWIZZLE_G, swizzle[1]); + gl->fTexParameteri(target.get(), LOCAL_GL_TEXTURE_SWIZZLE_B, swizzle[2]); + gl->fTexParameteri(target.get(), LOCAL_GL_TEXTURE_SWIZZLE_A, swizzle[3]); +} + +void WebGLTexture::RefreshSwizzle() const { + const auto& imageInfo = BaseImageInfo(); + const auto& swizzle = imageInfo.mFormat->textureSwizzleRGBA; + + if (swizzle != mCurSwizzle) { + const gl::ScopedBindTexture scopeBindTexture(mContext->gl, mGLName, + mTarget.get()); + SetSwizzle(mContext->gl, mTarget, swizzle); + mCurSwizzle = swizzle; + } +} + +bool WebGLTexture::EnsureImageDataInitialized(const TexImageTarget target, + const uint32_t level) { + auto& imageInfo = ImageInfoAt(target, level); + if (!imageInfo.IsDefined()) return true; + + if (!imageInfo.mUninitializedSlices) return true; + + if (!ZeroTextureData(mContext, mGLName, target, level, imageInfo)) { + return false; + } + imageInfo.mUninitializedSlices = Nothing(); + return true; +} + +static bool ClearDepthTexture(const WebGLContext& webgl, const GLuint tex, + const TexImageTarget imageTarget, + const uint32_t level, + const webgl::ImageInfo& info) { + const auto& gl = webgl.gl; + const auto& usage = info.mFormat; + const auto& format = usage->format; + + // Depth resources actually clear to 1.0f, not 0.0f! + // They are also always renderable. + MOZ_ASSERT(usage->IsRenderable()); + MOZ_ASSERT(info.mUninitializedSlices); + + GLenum attachPoint = LOCAL_GL_DEPTH_ATTACHMENT; + GLbitfield clearBits = LOCAL_GL_DEPTH_BUFFER_BIT; + + if (format->s) { + attachPoint = LOCAL_GL_DEPTH_STENCIL_ATTACHMENT; + clearBits |= LOCAL_GL_STENCIL_BUFFER_BIT; + } + + // - + + gl::ScopedFramebuffer scopedFB(gl); + const gl::ScopedBindFramebuffer scopedBindFB(gl, scopedFB.FB()); + const webgl::ScopedPrepForResourceClear scopedPrep(webgl); + + const auto fnAttach = [&](const uint32_t z) { + switch (imageTarget.get()) { + case LOCAL_GL_TEXTURE_3D: + case LOCAL_GL_TEXTURE_2D_ARRAY: + gl->fFramebufferTextureLayer(LOCAL_GL_FRAMEBUFFER, attachPoint, tex, + level, z); + break; + default: + if (attachPoint == LOCAL_GL_DEPTH_STENCIL_ATTACHMENT) { + gl->fFramebufferTexture2D(LOCAL_GL_FRAMEBUFFER, + LOCAL_GL_DEPTH_ATTACHMENT, + imageTarget.get(), tex, level); + gl->fFramebufferTexture2D(LOCAL_GL_FRAMEBUFFER, + LOCAL_GL_STENCIL_ATTACHMENT, + imageTarget.get(), tex, level); + } else { + gl->fFramebufferTexture2D(LOCAL_GL_FRAMEBUFFER, attachPoint, + imageTarget.get(), tex, level); + } + break; + } + }; + + for (const auto z : IntegerRange(info.mDepth)) { + if ((*info.mUninitializedSlices)[z]) { + fnAttach(z); + gl->fClear(clearBits); + } + } + const auto& status = gl->fCheckFramebufferStatus(LOCAL_GL_FRAMEBUFFER); + const bool isComplete = (status == LOCAL_GL_FRAMEBUFFER_COMPLETE); + MOZ_ASSERT(isComplete); + return isComplete; +} + +static bool ZeroTextureData(const WebGLContext* webgl, GLuint tex, + TexImageTarget target, uint32_t level, + const webgl::ImageInfo& info) { + // This has one usecase: + // Lazy zeroing of uninitialized textures: + // a. Before draw. + // b. Before partial upload. (TexStorage + TexSubImage) + + // We have no sympathy for this case. + + // "Doctor, it hurts when I do this!" "Well don't do that!" + MOZ_ASSERT(info.mUninitializedSlices); + + const auto targetStr = EnumString(target.get()); + webgl->GenerateWarning( + "Tex image %s level %u is incurring lazy initialization.", + targetStr.c_str(), level); + + gl::GLContext* gl = webgl->GL(); + const auto& width = info.mWidth; + const auto& height = info.mHeight; + const auto& depth = info.mDepth; + const auto& usage = info.mFormat; + + GLenum scopeBindTarget; + switch (target.get()) { + case LOCAL_GL_TEXTURE_CUBE_MAP_POSITIVE_X: + case LOCAL_GL_TEXTURE_CUBE_MAP_NEGATIVE_X: + case LOCAL_GL_TEXTURE_CUBE_MAP_POSITIVE_Y: + case LOCAL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Y: + case LOCAL_GL_TEXTURE_CUBE_MAP_POSITIVE_Z: + case LOCAL_GL_TEXTURE_CUBE_MAP_NEGATIVE_Z: + scopeBindTarget = LOCAL_GL_TEXTURE_CUBE_MAP; + break; + default: + scopeBindTarget = target.get(); + break; + } + const gl::ScopedBindTexture scopeBindTexture(gl, tex, scopeBindTarget); + const auto& compression = usage->format->compression; + if (compression) { + auto sizedFormat = usage->format->sizedFormat; + MOZ_RELEASE_ASSERT(sizedFormat, "GFX: texture sized format not set"); + + const auto fnSizeInBlocks = [](CheckedUint32 pixels, + uint8_t pixelsPerBlock) { + return RoundUpToMultipleOf(pixels, pixelsPerBlock) / pixelsPerBlock; + }; + + const auto widthBlocks = fnSizeInBlocks(width, compression->blockWidth); + const auto heightBlocks = fnSizeInBlocks(height, compression->blockHeight); + + CheckedUint32 checkedByteCount = compression->bytesPerBlock; + checkedByteCount *= widthBlocks; + checkedByteCount *= heightBlocks; + + if (!checkedByteCount.isValid()) return false; + + const size_t sliceByteCount = checkedByteCount.value(); + + const auto zeros = UniqueBuffer::Take(calloc(1u, sliceByteCount)); + if (!zeros) return false; + + // Don't bother with striding it well. + // TODO: We shouldn't need to do this for CompressedTexSubImage. + webgl::PixelPackingState{}.AssertCurrentUnpack(*gl, webgl->IsWebGL2()); + gl->fPixelStorei(LOCAL_GL_UNPACK_ALIGNMENT, 1); + const auto revert = MakeScopeExit( + [&]() { gl->fPixelStorei(LOCAL_GL_UNPACK_ALIGNMENT, 4); }); + + GLenum error = 0; + for (const auto z : IntegerRange(depth)) { + if ((*info.mUninitializedSlices)[z]) { + error = DoCompressedTexSubImage(gl, target.get(), level, 0, 0, z, width, + height, 1, sizedFormat, sliceByteCount, + zeros.get()); + if (error) break; + } + } + return !error; + } + + const auto driverUnpackInfo = usage->idealUnpack; + MOZ_RELEASE_ASSERT(driverUnpackInfo, "GFX: ideal unpack info not set."); + + if (usage->format->d) { + // ANGLE_depth_texture does not allow uploads, so we have to clear. + // (Restriction because of D3D9) + // Also, depth resources are cleared to 1.0f and are always renderable, so + // just use FB clears. + return ClearDepthTexture(*webgl, tex, target, level, info); + } + + const webgl::PackingInfo packing = driverUnpackInfo->ToPacking(); + + const auto bytesPerPixel = webgl::BytesPerPixel(packing); + + CheckedUint32 checkedByteCount = bytesPerPixel; + checkedByteCount *= width; + checkedByteCount *= height; + + if (!checkedByteCount.isValid()) return false; + + const size_t sliceByteCount = checkedByteCount.value(); + + const auto zeros = UniqueBuffer::Take(calloc(1u, sliceByteCount)); + if (!zeros) return false; + + // Don't bother with striding it well. + webgl::PixelPackingState{}.AssertCurrentUnpack(*gl, webgl->IsWebGL2()); + gl->fPixelStorei(LOCAL_GL_UNPACK_ALIGNMENT, 1); + const auto revert = + MakeScopeExit([&]() { gl->fPixelStorei(LOCAL_GL_UNPACK_ALIGNMENT, 4); }); + + GLenum error = 0; + for (const auto z : IntegerRange(depth)) { + if ((*info.mUninitializedSlices)[z]) { + error = DoTexSubImage(gl, target, level, 0, 0, z, width, height, 1, + packing, zeros.get()); + if (error) break; + } + } + return !error; +} + +template <typename T, typename R> +static constexpr R Clamp(const T val, const R min, const R max) { + if (val < min) return min; + if (val > max) return max; + return static_cast<R>(val); +} + +void WebGLTexture::ClampLevelBaseAndMax() { + if (!mImmutable) return; + + // GLES 3.0.4, p158: + // "For immutable-format textures, `level_base` is clamped to the range + // `[0, levels-1]`, `level_max` is then clamped to the range ` + // `[level_base, levels-1]`, where `levels` is the parameter passed to + // TexStorage* for the texture object." + MOZ_ASSERT(mImmutableLevelCount > 0); + const auto oldBase = mBaseMipmapLevel; + const auto oldMax = mMaxMipmapLevel; + mBaseMipmapLevel = Clamp(mBaseMipmapLevel, 0u, mImmutableLevelCount - 1u); + mMaxMipmapLevel = + Clamp(mMaxMipmapLevel, mBaseMipmapLevel, mImmutableLevelCount - 1u); + if (oldBase != mBaseMipmapLevel && + mBaseMipmapLevelState != MIPMAP_LEVEL_DEFAULT) { + mBaseMipmapLevelState = MIPMAP_LEVEL_DIRTY; + } + if (oldMax != mMaxMipmapLevel && + mMaxMipmapLevelState != MIPMAP_LEVEL_DEFAULT) { + mMaxMipmapLevelState = MIPMAP_LEVEL_DIRTY; + } + + // Note: This means that immutable textures are *always* texture-complete! +} + +////////////////////////////////////////////////////////////////////////////////////////// +// GL calls + +bool WebGLTexture::BindTexture(TexTarget texTarget) { + const bool isFirstBinding = !mTarget; + if (!isFirstBinding && mTarget != texTarget) { + mContext->ErrorInvalidOperation( + "bindTexture: This texture has already been bound" + " to a different target."); + return false; + } + + mTarget = texTarget; + + mContext->gl->fBindTexture(mTarget.get(), mGLName); + + if (isFirstBinding) { + mFaceCount = IsCubeMap() ? 6 : 1; + + gl::GLContext* gl = mContext->gl; + + // Thanks to the WebKit people for finding this out: GL_TEXTURE_WRAP_R + // is not present in GLES 2, but is present in GL and it seems as if for + // cube maps we need to set it to GL_CLAMP_TO_EDGE to get the expected + // GLES behavior. + // If we are WebGL 2 though, we'll want to leave it as REPEAT. + const bool hasWrapR = gl->IsSupported(gl::GLFeature::texture_3D); + if (IsCubeMap() && hasWrapR && !mContext->IsWebGL2()) { + gl->fTexParameteri(texTarget.get(), LOCAL_GL_TEXTURE_WRAP_R, + LOCAL_GL_CLAMP_TO_EDGE); + } + } + + return true; +} + +static constexpr GLint ClampMipmapLevelForDriver(uint32_t level) { + return Clamp(level, uint8_t{0}, WebGLTexture::kMaxLevelCount); +} + +void WebGLTexture::GenerateMipmap() { + // GLES 3.0.4 p160: + // "Mipmap generation replaces texel array levels level base + 1 through q + // with arrrays derived from the level base array, regardless of their + // previous contents. All other mipmap arrays, including the level base + // array, are left unchanged by this computation." + // But only check and init the base level. + const bool ensureInit = true; + const bool skipMips = true; + const auto completeness = CalcCompletenessInfo(ensureInit, skipMips); + if (!completeness || !completeness->levels) { + mContext->ErrorInvalidOperation( + "The texture's base level must be complete."); + return; + } + const auto& usage = completeness->usage; + const auto& format = usage->format; + if (!mContext->IsWebGL2()) { + if (!completeness->powerOfTwo) { + mContext->ErrorInvalidOperation( + "The base level of the texture does not" + " have power-of-two dimensions."); + return; + } + if (format->isSRGB) { + mContext->ErrorInvalidOperation( + "EXT_sRGB forbids GenerateMipmap with" + " sRGB."); + return; + } + } + + if (format->compression) { + mContext->ErrorInvalidOperation( + "Texture data at base level is compressed."); + return; + } + + if (format->d) { + mContext->ErrorInvalidOperation("Depth textures are not supported."); + return; + } + + // OpenGL ES 3.0.4 p160: + // If the level base array was not specified with an unsized internal format + // from table 3.3 or a sized internal format that is both color-renderable and + // texture-filterable according to table 3.13, an INVALID_OPERATION error + // is generated. + bool canGenerateMipmap = (usage->IsRenderable() && usage->isFilterable); + switch (usage->format->effectiveFormat) { + case webgl::EffectiveFormat::Luminance8: + case webgl::EffectiveFormat::Alpha8: + case webgl::EffectiveFormat::Luminance8Alpha8: + // Non-color-renderable formats from Table 3.3. + canGenerateMipmap = true; + break; + default: + break; + } + + if (!canGenerateMipmap) { + mContext->ErrorInvalidOperation( + "Texture at base level is not unsized" + " internal format or is not" + " color-renderable or texture-filterable."); + return; + } + + if (usage->IsRenderable() && !usage->IsExplicitlyRenderable()) { + mContext->WarnIfImplicit(usage->GetExtensionID()); + } + + // Done with validation. Do the operation. + + gl::GLContext* gl = mContext->gl; + + if (gl->WorkAroundDriverBugs()) { + // If we first set GL_TEXTURE_BASE_LEVEL to a number such as 20, then set + // MGL_TEXTURE_MAX_LEVEL to a smaller number like 8, our copy of the + // base level will be lowered, but we havn't yet updated the driver, we + // should do so now, before calling glGenerateMipmap(). + if (mBaseMipmapLevelState == MIPMAP_LEVEL_DIRTY) { + gl->fTexParameteri(mTarget.get(), LOCAL_GL_TEXTURE_BASE_LEVEL, + ClampMipmapLevelForDriver(mBaseMipmapLevel)); + mBaseMipmapLevelState = MIPMAP_LEVEL_CLEAN; + } + if (mMaxMipmapLevelState == MIPMAP_LEVEL_DIRTY) { + gl->fTexParameteri(mTarget.get(), LOCAL_GL_TEXTURE_MAX_LEVEL, + ClampMipmapLevelForDriver(mMaxMipmapLevel)); + mMaxMipmapLevelState = MIPMAP_LEVEL_CLEAN; + } + + // bug 696495 - to work around failures in the texture-mips.html test on + // various drivers, we set the minification filter before calling + // glGenerateMipmap. This should not carry a significant performance + // overhead so we do it unconditionally. + // + // note that the choice of GL_NEAREST_MIPMAP_NEAREST really matters. See + // Chromium bug 101105. + gl->fTexParameteri(mTarget.get(), LOCAL_GL_TEXTURE_MIN_FILTER, + LOCAL_GL_NEAREST_MIPMAP_NEAREST); + gl->fGenerateMipmap(mTarget.get()); + gl->fTexParameteri(mTarget.get(), LOCAL_GL_TEXTURE_MIN_FILTER, + mSamplingState.minFilter.get()); + } else { + gl->fGenerateMipmap(mTarget.get()); + } + + // Record the results. + + const auto maxLevel = Es3_q(); + PopulateMipChain(maxLevel); +} + +Maybe<double> WebGLTexture::GetTexParameter(GLenum pname) const { + GLint i = 0; + GLfloat f = 0.0f; + + switch (pname) { + case LOCAL_GL_TEXTURE_BASE_LEVEL: + return Some(mBaseMipmapLevel); + + case LOCAL_GL_TEXTURE_MAX_LEVEL: + return Some(mMaxMipmapLevel); + + case LOCAL_GL_TEXTURE_IMMUTABLE_FORMAT: + return Some(mImmutable); + + case LOCAL_GL_TEXTURE_IMMUTABLE_LEVELS: + return Some(uint32_t(mImmutableLevelCount)); + + case LOCAL_GL_TEXTURE_MIN_FILTER: + case LOCAL_GL_TEXTURE_MAG_FILTER: + case LOCAL_GL_TEXTURE_WRAP_S: + case LOCAL_GL_TEXTURE_WRAP_T: + case LOCAL_GL_TEXTURE_WRAP_R: + case LOCAL_GL_TEXTURE_COMPARE_MODE: + case LOCAL_GL_TEXTURE_COMPARE_FUNC: { + MOZ_ASSERT(mTarget); + const gl::ScopedBindTexture autoTex(mContext->gl, mGLName, mTarget.get()); + mContext->gl->fGetTexParameteriv(mTarget.get(), pname, &i); + return Some(i); + } + + case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT: + case LOCAL_GL_TEXTURE_MAX_LOD: + case LOCAL_GL_TEXTURE_MIN_LOD: { + MOZ_ASSERT(mTarget); + const gl::ScopedBindTexture autoTex(mContext->gl, mGLName, mTarget.get()); + mContext->gl->fGetTexParameterfv(mTarget.get(), pname, &f); + return Some(f); + } + + default: + MOZ_CRASH("GFX: Unhandled pname."); + } +} + +// Here we have to support all pnames with both int and float params. +// See this discussion: +// https://www.khronos.org/webgl/public-mailing-list/archives/1008/msg00014.html +void WebGLTexture::TexParameter(TexTarget texTarget, GLenum pname, + const FloatOrInt& param) { + bool isPNameValid = false; + switch (pname) { + // GLES 2.0.25 p76: + case LOCAL_GL_TEXTURE_WRAP_S: + case LOCAL_GL_TEXTURE_WRAP_T: + case LOCAL_GL_TEXTURE_MIN_FILTER: + case LOCAL_GL_TEXTURE_MAG_FILTER: + isPNameValid = true; + break; + + // GLES 3.0.4 p149-150: + case LOCAL_GL_TEXTURE_BASE_LEVEL: + case LOCAL_GL_TEXTURE_COMPARE_MODE: + case LOCAL_GL_TEXTURE_COMPARE_FUNC: + case LOCAL_GL_TEXTURE_MAX_LEVEL: + case LOCAL_GL_TEXTURE_MAX_LOD: + case LOCAL_GL_TEXTURE_MIN_LOD: + case LOCAL_GL_TEXTURE_WRAP_R: + if (mContext->IsWebGL2()) isPNameValid = true; + break; + + case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT: + if (mContext->IsExtensionEnabled( + WebGLExtensionID::EXT_texture_filter_anisotropic)) + isPNameValid = true; + break; + } + + if (!isPNameValid) { + mContext->ErrorInvalidEnumInfo("texParameter: pname", pname); + return; + } + + //////////////// + // Validate params and invalidate if needed. + + bool paramBadEnum = false; + bool paramBadValue = false; + + switch (pname) { + case LOCAL_GL_TEXTURE_BASE_LEVEL: + case LOCAL_GL_TEXTURE_MAX_LEVEL: + paramBadValue = (param.i < 0); + break; + + case LOCAL_GL_TEXTURE_COMPARE_MODE: + paramBadValue = (param.i != LOCAL_GL_NONE && + param.i != LOCAL_GL_COMPARE_REF_TO_TEXTURE); + break; + + case LOCAL_GL_TEXTURE_COMPARE_FUNC: + switch (param.i) { + case LOCAL_GL_LEQUAL: + case LOCAL_GL_GEQUAL: + case LOCAL_GL_LESS: + case LOCAL_GL_GREATER: + case LOCAL_GL_EQUAL: + case LOCAL_GL_NOTEQUAL: + case LOCAL_GL_ALWAYS: + case LOCAL_GL_NEVER: + break; + + default: + paramBadValue = true; + break; + } + break; + + case LOCAL_GL_TEXTURE_MIN_FILTER: + switch (param.i) { + case LOCAL_GL_NEAREST: + case LOCAL_GL_LINEAR: + case LOCAL_GL_NEAREST_MIPMAP_NEAREST: + case LOCAL_GL_LINEAR_MIPMAP_NEAREST: + case LOCAL_GL_NEAREST_MIPMAP_LINEAR: + case LOCAL_GL_LINEAR_MIPMAP_LINEAR: + break; + + default: + paramBadEnum = true; + break; + } + break; + + case LOCAL_GL_TEXTURE_MAG_FILTER: + switch (param.i) { + case LOCAL_GL_NEAREST: + case LOCAL_GL_LINEAR: + break; + + default: + paramBadEnum = true; + break; + } + break; + + case LOCAL_GL_TEXTURE_WRAP_S: + case LOCAL_GL_TEXTURE_WRAP_T: + case LOCAL_GL_TEXTURE_WRAP_R: + switch (param.i) { + case LOCAL_GL_CLAMP_TO_EDGE: + case LOCAL_GL_MIRRORED_REPEAT: + case LOCAL_GL_REPEAT: + break; + + default: + paramBadEnum = true; + break; + } + break; + + case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT: + if (param.f < 1.0f) paramBadValue = true; + + break; + } + + if (paramBadEnum) { + if (!param.isFloat) { + mContext->ErrorInvalidEnum( + "pname 0x%04x: Invalid param" + " 0x%04x.", + pname, param.i); + } else { + mContext->ErrorInvalidEnum("pname 0x%04x: Invalid param %g.", pname, + param.f); + } + return; + } + + if (paramBadValue) { + if (!param.isFloat) { + mContext->ErrorInvalidValue( + "pname 0x%04x: Invalid param %i" + " (0x%x).", + pname, param.i, param.i); + } else { + mContext->ErrorInvalidValue("pname 0x%04x: Invalid param %g.", pname, + param.f); + } + return; + } + + //////////////// + // Store any needed values + + FloatOrInt clamped = param; + bool invalidate = true; + switch (pname) { + case LOCAL_GL_TEXTURE_BASE_LEVEL: { + mBaseMipmapLevel = clamped.i; + mBaseMipmapLevelState = MIPMAP_LEVEL_CLEAN; + ClampLevelBaseAndMax(); + clamped = FloatOrInt(ClampMipmapLevelForDriver(mBaseMipmapLevel)); + break; + } + + case LOCAL_GL_TEXTURE_MAX_LEVEL: { + mMaxMipmapLevel = clamped.i; + mMaxMipmapLevelState = MIPMAP_LEVEL_CLEAN; + ClampLevelBaseAndMax(); + clamped = FloatOrInt(ClampMipmapLevelForDriver(mMaxMipmapLevel)); + break; + } + + case LOCAL_GL_TEXTURE_MIN_FILTER: + mSamplingState.minFilter = clamped.i; + break; + + case LOCAL_GL_TEXTURE_MAG_FILTER: + mSamplingState.magFilter = clamped.i; + break; + + case LOCAL_GL_TEXTURE_WRAP_S: + mSamplingState.wrapS = clamped.i; + break; + + case LOCAL_GL_TEXTURE_WRAP_T: + mSamplingState.wrapT = clamped.i; + break; + + case LOCAL_GL_TEXTURE_COMPARE_MODE: + mSamplingState.compareMode = clamped.i; + break; + + default: + invalidate = false; // Texture completeness will not change. + break; + } + + if (invalidate) { + InvalidateCaches(); + } + + //////////////// + + if (!clamped.isFloat) { + mContext->gl->fTexParameteri(texTarget.get(), pname, clamped.i); + } else { + mContext->gl->fTexParameterf(texTarget.get(), pname, clamped.f); + } +} + +void WebGLTexture::Truncate() { + for (auto& cur : mImageInfoArr) { + cur = {}; + } + InvalidateCaches(); +} + +} // namespace mozilla |