summaryrefslogtreecommitdiffstats
path: root/dom/canvas/WebGLBuffer.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'dom/canvas/WebGLBuffer.cpp')
-rw-r--r--dom/canvas/WebGLBuffer.cpp439
1 files changed, 439 insertions, 0 deletions
diff --git a/dom/canvas/WebGLBuffer.cpp b/dom/canvas/WebGLBuffer.cpp
new file mode 100644
index 0000000000..86ba035195
--- /dev/null
+++ b/dom/canvas/WebGLBuffer.cpp
@@ -0,0 +1,439 @@
+/* -*- 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 "WebGLBuffer.h"
+
+#include "GLContext.h"
+#include "mozilla/dom/WebGLRenderingContextBinding.h"
+#include "WebGLContext.h"
+
+namespace mozilla {
+
+WebGLBuffer::WebGLBuffer(WebGLContext* webgl, GLuint buf)
+ : WebGLContextBoundObject(webgl), mGLName(buf) {}
+
+WebGLBuffer::~WebGLBuffer() {
+ mByteLength = 0;
+ mFetchInvalidator.InvalidateCaches();
+
+ mIndexCache.reset();
+ mIndexRanges.clear();
+
+ if (!mContext) return;
+ mContext->gl->fDeleteBuffers(1, &mGLName);
+}
+
+void WebGLBuffer::SetContentAfterBind(GLenum target) {
+ if (mContent != Kind::Undefined) return;
+
+ switch (target) {
+ case LOCAL_GL_ELEMENT_ARRAY_BUFFER:
+ mContent = Kind::ElementArray;
+ break;
+
+ case LOCAL_GL_ARRAY_BUFFER:
+ case LOCAL_GL_PIXEL_PACK_BUFFER:
+ case LOCAL_GL_PIXEL_UNPACK_BUFFER:
+ case LOCAL_GL_UNIFORM_BUFFER:
+ case LOCAL_GL_TRANSFORM_FEEDBACK_BUFFER:
+ case LOCAL_GL_COPY_READ_BUFFER:
+ case LOCAL_GL_COPY_WRITE_BUFFER:
+ mContent = Kind::OtherData;
+ break;
+
+ default:
+ MOZ_CRASH("GFX: invalid target");
+ }
+}
+
+////////////////////////////////////////
+
+static bool ValidateBufferUsageEnum(WebGLContext* webgl, GLenum usage) {
+ switch (usage) {
+ case LOCAL_GL_STREAM_DRAW:
+ case LOCAL_GL_STATIC_DRAW:
+ case LOCAL_GL_DYNAMIC_DRAW:
+ return true;
+
+ case LOCAL_GL_DYNAMIC_COPY:
+ case LOCAL_GL_DYNAMIC_READ:
+ case LOCAL_GL_STATIC_COPY:
+ case LOCAL_GL_STATIC_READ:
+ case LOCAL_GL_STREAM_COPY:
+ case LOCAL_GL_STREAM_READ:
+ if (MOZ_LIKELY(webgl->IsWebGL2())) return true;
+ break;
+
+ default:
+ break;
+ }
+
+ webgl->ErrorInvalidEnumInfo("usage", usage);
+ return false;
+}
+
+void WebGLBuffer::BufferData(const GLenum target, const uint64_t size,
+ const void* const maybeData, const GLenum usage,
+ bool allowUninitialized) {
+ // The driver knows only GLsizeiptr, which is int32_t on 32bit!
+ bool sizeValid = CheckedInt<GLsizeiptr>(size).isValid();
+
+ if (mContext->gl->WorkAroundDriverBugs()) {
+ // Bug 790879
+#if defined(XP_MACOSX) || defined(MOZ_WIDGET_GTK)
+ sizeValid &= CheckedInt<int32_t>(size).isValid();
+#endif
+
+ // Bug 1610383
+ if (mContext->gl->IsANGLE()) {
+ // While ANGLE seems to support up to `unsigned int`, UINT32_MAX-4 causes
+ // GL_OUT_OF_MEMORY in glFlush??
+ sizeValid &= CheckedInt<int32_t>(size).isValid();
+ }
+ }
+
+ if (!sizeValid) {
+ mContext->ErrorOutOfMemory("Size not valid for platform: %" PRIu64, size);
+ return;
+ }
+
+ // -
+
+ if (!ValidateBufferUsageEnum(mContext, usage)) return;
+
+ const void* uploadData = maybeData;
+ UniqueBuffer maybeCalloc;
+ if (!uploadData && !allowUninitialized) {
+ maybeCalloc = UniqueBuffer::Take(calloc(1, AssertedCast<size_t>(size)));
+ if (!maybeCalloc) {
+ mContext->ErrorOutOfMemory("Failed to alloc zeros.");
+ return;
+ }
+ uploadData = maybeCalloc.get();
+ }
+ MOZ_ASSERT(uploadData || allowUninitialized);
+
+ UniqueBuffer newIndexCache;
+ const bool needsIndexCache = mContext->mNeedsIndexValidation ||
+ mContext->mMaybeNeedsLegacyVertexAttrib0Handling;
+ if (target == LOCAL_GL_ELEMENT_ARRAY_BUFFER && needsIndexCache) {
+ newIndexCache = UniqueBuffer::Take(malloc(AssertedCast<size_t>(size)));
+ if (!newIndexCache) {
+ mContext->ErrorOutOfMemory("Failed to alloc index cache.");
+ return;
+ }
+ // memcpy out of SharedArrayBuffers can be racey, and should generally use
+ // memcpySafeWhenRacy. But it's safe here:
+ // * We only memcpy in one place.
+ // * We only read out of the single copy, and only after copying.
+ // * If we get data value corruption from racing read-during-write, that's
+ // fine.
+ memcpy(newIndexCache.get(), uploadData, size);
+ uploadData = newIndexCache.get();
+ }
+
+ const auto& gl = mContext->gl;
+ const ScopedLazyBind lazyBind(gl, target, this);
+
+ const bool sizeChanges = (size != ByteLength());
+ if (sizeChanges) {
+ gl::GLContext::LocalErrorScope errorScope(*gl);
+ gl->fBufferData(target, size, uploadData, usage);
+ const auto error = errorScope.GetError();
+
+ if (error) {
+ MOZ_ASSERT(error == LOCAL_GL_OUT_OF_MEMORY);
+ mContext->ErrorOutOfMemory("Error from driver: 0x%04x", error);
+
+ // Truncate
+ mByteLength = 0;
+ mFetchInvalidator.InvalidateCaches();
+ mIndexCache.reset();
+ return;
+ }
+ } else {
+ gl->fBufferData(target, size, uploadData, usage);
+ }
+
+ mContext->OnDataAllocCall();
+
+ mUsage = usage;
+ mByteLength = size;
+ mFetchInvalidator.InvalidateCaches();
+ mIndexCache = std::move(newIndexCache);
+
+ if (mIndexCache) {
+ if (!mIndexRanges.empty()) {
+ mContext->GeneratePerfWarning("[%p] Invalidating %u ranges.", this,
+ uint32_t(mIndexRanges.size()));
+ mIndexRanges.clear();
+ }
+ }
+
+ ResetLastUpdateFenceId();
+}
+
+void WebGLBuffer::BufferSubData(GLenum target, uint64_t rawDstByteOffset,
+ uint64_t rawDataLen, const void* data,
+ bool unsynchronized) const {
+ if (!ValidateRange(rawDstByteOffset, rawDataLen)) return;
+
+ const CheckedInt<GLintptr> dstByteOffset = rawDstByteOffset;
+ const CheckedInt<GLsizeiptr> dataLen = rawDataLen;
+ if (!dstByteOffset.isValid() || !dataLen.isValid()) {
+ return mContext->ErrorOutOfMemory("offset or size too large for platform.");
+ }
+
+ ////
+
+ if (!rawDataLen) return; // With validation successful, nothing else to do.
+
+ const void* uploadData = data;
+ if (mIndexCache) {
+ auto* const cachedDataBegin =
+ (uint8_t*)mIndexCache.get() + rawDstByteOffset;
+ memcpy(cachedDataBegin, data, dataLen.value());
+ uploadData = cachedDataBegin;
+
+ InvalidateCacheRange(dstByteOffset.value(), dataLen.value());
+ }
+
+ ////
+
+ const auto& gl = mContext->gl;
+ const ScopedLazyBind lazyBind(gl, target, this);
+
+ void* mapping = nullptr;
+ // Repeated calls to glMapBufferRange is slow on ANGLE, so fall back to the
+ // glBufferSubData path. See bug 1827047.
+ if (unsynchronized && gl->IsSupported(gl::GLFeature::map_buffer_range) &&
+ !gl->IsANGLE()) {
+ GLbitfield access = LOCAL_GL_MAP_WRITE_BIT |
+ LOCAL_GL_MAP_UNSYNCHRONIZED_BIT |
+ LOCAL_GL_MAP_INVALIDATE_RANGE_BIT;
+ // On some devices there are known performance issues with the combination
+ // of GL_MAP_UNSYNCHRONIZED_BIT and GL_MAP_INVALIDATE_RANGE_BIT, so omit the
+ // latter.
+ if (gl->Renderer() == gl::GLRenderer::MaliT ||
+ gl->Vendor() == gl::GLVendor::Qualcomm) {
+ access &= ~LOCAL_GL_MAP_INVALIDATE_RANGE_BIT;
+ }
+ mapping = gl->fMapBufferRange(target, dstByteOffset.value(),
+ dataLen.value(), access);
+ }
+
+ if (mapping) {
+ memcpy(mapping, uploadData, dataLen.value());
+ gl->fUnmapBuffer(target);
+ } else {
+ gl->fBufferSubData(target, dstByteOffset.value(), dataLen.value(),
+ uploadData);
+ }
+
+ ResetLastUpdateFenceId();
+}
+
+bool WebGLBuffer::ValidateRange(size_t byteOffset, size_t byteLen) const {
+ auto availLength = mByteLength;
+ if (byteOffset > availLength) {
+ mContext->ErrorInvalidValue("Offset passes the end of the buffer.");
+ return false;
+ }
+ availLength -= byteOffset;
+
+ if (byteLen > availLength) {
+ mContext->ErrorInvalidValue("Offset+size passes the end of the buffer.");
+ return false;
+ }
+
+ return true;
+}
+
+////////////////////////////////////////
+
+static uint8_t IndexByteSizeByType(GLenum type) {
+ switch (type) {
+ case LOCAL_GL_UNSIGNED_BYTE:
+ return 1;
+ case LOCAL_GL_UNSIGNED_SHORT:
+ return 2;
+ case LOCAL_GL_UNSIGNED_INT:
+ return 4;
+ default:
+ MOZ_CRASH();
+ }
+}
+
+void WebGLBuffer::InvalidateCacheRange(uint64_t byteOffset,
+ uint64_t byteLength) const {
+ MOZ_ASSERT(mIndexCache);
+
+ std::vector<IndexRange> invalids;
+ const uint64_t updateBegin = byteOffset;
+ const uint64_t updateEnd = updateBegin + byteLength;
+ for (const auto& cur : mIndexRanges) {
+ const auto& range = cur.first;
+ const auto& indexByteSize = IndexByteSizeByType(range.type);
+ const auto rangeBegin = range.byteOffset * indexByteSize;
+ const auto rangeEnd =
+ rangeBegin + uint64_t(range.indexCount) * indexByteSize;
+ if (rangeBegin >= updateEnd || rangeEnd <= updateBegin) continue;
+ invalids.push_back(range);
+ }
+
+ if (!invalids.empty()) {
+ mContext->GeneratePerfWarning("[%p] Invalidating %u/%u ranges.", this,
+ uint32_t(invalids.size()),
+ uint32_t(mIndexRanges.size()));
+
+ for (const auto& cur : invalids) {
+ mIndexRanges.erase(cur);
+ }
+ }
+}
+
+size_t WebGLBuffer::SizeOfIncludingThis(
+ mozilla::MallocSizeOf mallocSizeOf) const {
+ size_t size = mallocSizeOf(this);
+ if (mIndexCache) {
+ size += mByteLength;
+ }
+ return size;
+}
+
+template <typename T>
+static Maybe<uint32_t> MaxForRange(const void* const start,
+ const uint32_t count,
+ const Maybe<uint32_t>& untypedIgnoredVal) {
+ const Maybe<T> ignoredVal =
+ (untypedIgnoredVal ? Some(T(untypedIgnoredVal.value())) : Nothing());
+ Maybe<uint32_t> maxVal;
+
+ auto itr = (const T*)start;
+ const auto end = itr + count;
+
+ for (; itr != end; ++itr) {
+ const auto& val = *itr;
+ if (ignoredVal && val == ignoredVal.value()) continue;
+
+ if (maxVal && val <= maxVal.value()) continue;
+
+ maxVal = Some(val);
+ }
+
+ return maxVal;
+}
+
+static const uint32_t kMaxIndexRanges = 256;
+
+Maybe<uint32_t> WebGLBuffer::GetIndexedFetchMaxVert(
+ const GLenum type, const uint64_t byteOffset,
+ const uint32_t indexCount) const {
+ if (!mIndexCache) return Nothing();
+
+ const IndexRange range = {type, byteOffset, indexCount};
+ auto res = mIndexRanges.insert({range, Nothing()});
+ if (mIndexRanges.size() > kMaxIndexRanges) {
+ mContext->GeneratePerfWarning(
+ "[%p] Clearing mIndexRanges after exceeding %u.", this,
+ kMaxIndexRanges);
+ mIndexRanges.clear();
+ res = mIndexRanges.insert({range, Nothing()});
+ }
+
+ const auto& itr = res.first;
+ const auto& didInsert = res.second;
+
+ auto& maxFetchIndex = itr->second;
+ if (didInsert) {
+ const auto& data = mIndexCache.get();
+
+ const auto start = (const uint8_t*)data + byteOffset;
+
+ Maybe<uint32_t> ignoredVal;
+ if (mContext->IsWebGL2()) {
+ ignoredVal = Some(UINT32_MAX);
+ }
+
+ switch (type) {
+ case LOCAL_GL_UNSIGNED_BYTE:
+ maxFetchIndex = MaxForRange<uint8_t>(start, indexCount, ignoredVal);
+ break;
+ case LOCAL_GL_UNSIGNED_SHORT:
+ maxFetchIndex = MaxForRange<uint16_t>(start, indexCount, ignoredVal);
+ break;
+ case LOCAL_GL_UNSIGNED_INT:
+ maxFetchIndex = MaxForRange<uint32_t>(start, indexCount, ignoredVal);
+ break;
+ default:
+ MOZ_CRASH();
+ }
+ const auto displayMaxVertIndex =
+ maxFetchIndex ? int64_t(maxFetchIndex.value()) : -1;
+ mContext->GeneratePerfWarning("[%p] New range #%u: (0x%04x, %" PRIu64
+ ", %u):"
+ " %" PRIi64,
+ this, uint32_t(mIndexRanges.size()),
+ range.type, range.byteOffset,
+ range.indexCount, displayMaxVertIndex);
+ }
+
+ return maxFetchIndex;
+}
+
+////
+
+bool WebGLBuffer::ValidateCanBindToTarget(GLenum target) {
+ /* https://www.khronos.org/registry/webgl/specs/latest/2.0/#5.1
+ *
+ * In the WebGL 2 API, buffers have their WebGL buffer type
+ * initially set to undefined. Calling bindBuffer, bindBufferRange
+ * or bindBufferBase with the target argument set to any buffer
+ * binding point except COPY_READ_BUFFER or COPY_WRITE_BUFFER will
+ * then set the WebGL buffer type of the buffer being bound
+ * according to the table above.
+ *
+ * Any call to one of these functions which attempts to bind a
+ * WebGLBuffer that has the element array WebGL buffer type to a
+ * binding point that falls under other data, or bind a
+ * WebGLBuffer which has the other data WebGL buffer type to
+ * ELEMENT_ARRAY_BUFFER will generate an INVALID_OPERATION error,
+ * and the state of the binding point will remain untouched.
+ */
+
+ if (mContent == WebGLBuffer::Kind::Undefined) return true;
+
+ switch (target) {
+ case LOCAL_GL_COPY_READ_BUFFER:
+ case LOCAL_GL_COPY_WRITE_BUFFER:
+ return true;
+
+ case LOCAL_GL_ELEMENT_ARRAY_BUFFER:
+ if (mContent == WebGLBuffer::Kind::ElementArray) return true;
+ break;
+
+ case LOCAL_GL_ARRAY_BUFFER:
+ case LOCAL_GL_PIXEL_PACK_BUFFER:
+ case LOCAL_GL_PIXEL_UNPACK_BUFFER:
+ case LOCAL_GL_TRANSFORM_FEEDBACK_BUFFER:
+ case LOCAL_GL_UNIFORM_BUFFER:
+ if (mContent == WebGLBuffer::Kind::OtherData) return true;
+ break;
+
+ default:
+ MOZ_CRASH();
+ }
+
+ const auto dataType =
+ (mContent == WebGLBuffer::Kind::OtherData) ? "other" : "element";
+ mContext->ErrorInvalidOperation("Buffer already contains %s data.", dataType);
+ return false;
+}
+
+void WebGLBuffer::ResetLastUpdateFenceId() const {
+ mLastUpdateFenceId = mContext->mNextFenceId;
+}
+
+} // namespace mozilla