summaryrefslogtreecommitdiffstats
path: root/dom/canvas/WebGLIpdl.h
diff options
context:
space:
mode:
Diffstat (limited to 'dom/canvas/WebGLIpdl.h')
-rw-r--r--dom/canvas/WebGLIpdl.h644
1 files changed, 644 insertions, 0 deletions
diff --git a/dom/canvas/WebGLIpdl.h b/dom/canvas/WebGLIpdl.h
new file mode 100644
index 0000000000..748c1a30a6
--- /dev/null
+++ b/dom/canvas/WebGLIpdl.h
@@ -0,0 +1,644 @@
+/* -*- 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 WEBGLIPDL_H_
+#define WEBGLIPDL_H_
+
+#include "gfxTypes.h"
+#include "ipc/EnumSerializer.h"
+#include "ipc/IPCMessageUtils.h"
+#include "mozilla/GfxMessageUtils.h"
+#include "mozilla/ipc/IPDLParamTraits.h"
+#include "mozilla/ipc/Shmem.h"
+#include "mozilla/layers/LayersSurfaces.h"
+#include "TiedFields.h"
+#include "TupleUtils.h"
+#include "WebGLTypes.h"
+
+namespace mozilla {
+namespace webgl {
+
+// TODO: This should probably replace Shmem, or at least this should move to
+// ipc/glue.
+
+class RaiiShmem final {
+ RefPtr<mozilla::ipc::ActorLifecycleProxy> mWeakRef;
+ mozilla::ipc::Shmem mShmem = {};
+
+ public:
+ /// Returns zeroed data.
+ static RaiiShmem Alloc(mozilla::ipc::IProtocol* const allocator,
+ const size_t size) {
+ mozilla::ipc::Shmem shmem;
+ if (!allocator->AllocShmem(size, &shmem)) return {};
+ return {allocator, shmem};
+ }
+
+ static RaiiShmem AllocUnsafe(mozilla::ipc::IProtocol* const allocator,
+ const size_t size) {
+ mozilla::ipc::Shmem shmem;
+ if (!allocator->AllocUnsafeShmem(size, &shmem)) return {};
+ return {allocator, shmem};
+ }
+
+ // -
+
+ RaiiShmem() = default;
+
+ RaiiShmem(mozilla::ipc::IProtocol* const allocator,
+ const mozilla::ipc::Shmem& shmem) {
+ if (!allocator || !allocator->CanSend()) {
+ return;
+ }
+
+ // Shmems are handled by the top-level, so use that or we might leak after
+ // the actor dies.
+ mWeakRef = allocator->ToplevelProtocol()->GetLifecycleProxy();
+ mShmem = shmem;
+ if (!mWeakRef || !mWeakRef->Get() || !IsShmem()) {
+ reset();
+ }
+ }
+
+ void reset() {
+ if (IsShmem()) {
+ const auto& allocator = mWeakRef->Get();
+ if (allocator) {
+ allocator->DeallocShmem(mShmem);
+ }
+ }
+ mWeakRef = nullptr;
+ mShmem = {};
+ }
+
+ ~RaiiShmem() { reset(); }
+
+ // -
+
+ RaiiShmem(RaiiShmem&& rhs) { *this = std::move(rhs); }
+ RaiiShmem& operator=(RaiiShmem&& rhs) {
+ reset();
+ mWeakRef = rhs.mWeakRef;
+ mShmem = rhs.Extract();
+ return *this;
+ }
+
+ // -
+
+ bool IsShmem() const { return mShmem.IsReadable(); }
+
+ explicit operator bool() const { return IsShmem(); }
+
+ // -
+
+ const auto& Shmem() const {
+ MOZ_ASSERT(IsShmem());
+ return mShmem;
+ }
+
+ Range<uint8_t> ByteRange() const {
+ if (!IsShmem()) {
+ return {};
+ }
+ return {mShmem.get<uint8_t>(), mShmem.Size<uint8_t>()};
+ }
+
+ mozilla::ipc::Shmem Extract() {
+ auto ret = mShmem;
+ mShmem = {};
+ reset();
+ return ret;
+ }
+};
+
+using Int32Vector = std::vector<int32_t>;
+
+} // namespace webgl
+
+namespace ipc {
+
+template <>
+struct IPDLParamTraits<mozilla::webgl::FrontBufferSnapshotIpc> final {
+ using T = mozilla::webgl::FrontBufferSnapshotIpc;
+
+ static void Write(IPC::MessageWriter* const writer, IProtocol* actor, T& in) {
+ WriteParam(writer, in.surfSize);
+ WriteIPDLParam(writer, actor, std::move(in.shmem));
+ }
+
+ static bool Read(IPC::MessageReader* const reader, IProtocol* actor,
+ T* const out) {
+ return ReadParam(reader, &out->surfSize) &&
+ ReadIPDLParam(reader, actor, &out->shmem);
+ }
+};
+
+// -
+
+template <>
+struct IPDLParamTraits<mozilla::webgl::ReadPixelsResultIpc> final {
+ using T = mozilla::webgl::ReadPixelsResultIpc;
+
+ static void Write(IPC::MessageWriter* const writer, IProtocol* actor, T& in) {
+ WriteParam(writer, in.subrect);
+ WriteParam(writer, in.byteStride);
+ WriteIPDLParam(writer, actor, std::move(in.shmem));
+ }
+
+ static bool Read(IPC::MessageReader* const reader, IProtocol* actor,
+ T* const out) {
+ return ReadParam(reader, &out->subrect) &&
+ ReadParam(reader, &out->byteStride) &&
+ ReadIPDLParam(reader, actor, &out->shmem);
+ }
+};
+
+// -
+
+template <>
+struct IPDLParamTraits<mozilla::webgl::TexUnpackBlobDesc> final {
+ using T = mozilla::webgl::TexUnpackBlobDesc;
+
+ static void Write(IPC::MessageWriter* const writer, IProtocol* actor,
+ T&& in) {
+ WriteParam(writer, in.imageTarget);
+ WriteParam(writer, in.size);
+ WriteParam(writer, in.srcAlphaType);
+ MOZ_RELEASE_ASSERT(!in.cpuData);
+ MOZ_RELEASE_ASSERT(!in.pboOffset);
+ WriteParam(writer, in.structuredSrcSize);
+ MOZ_RELEASE_ASSERT(!in.image);
+ WriteIPDLParam(writer, actor, std::move(in.sd));
+ MOZ_RELEASE_ASSERT(!in.dataSurf);
+ WriteParam(writer, in.unpacking);
+ WriteParam(writer, in.applyUnpackTransforms);
+ }
+
+ static bool Read(IPC::MessageReader* const reader, IProtocol* actor,
+ T* const out) {
+ return ReadParam(reader, &out->imageTarget) &&
+ ReadParam(reader, &out->size) &&
+ ReadParam(reader, &out->srcAlphaType) &&
+ ReadParam(reader, &out->structuredSrcSize) &&
+ ReadIPDLParam(reader, actor, &out->sd) &&
+ ReadParam(reader, &out->unpacking) &&
+ ReadParam(reader, &out->applyUnpackTransforms);
+ }
+};
+
+} // namespace ipc
+
+namespace webgl {
+using Int32Vector = std::vector<int32_t>;
+} // namespace webgl
+} // namespace mozilla
+
+namespace IPC {
+
+// -
+
+template <class U, size_t PaddedSize>
+struct ParamTraits<mozilla::webgl::Padded<U, PaddedSize>> final {
+ using T = mozilla::webgl::Padded<U, PaddedSize>;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, *in);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &**out);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::AttribBaseType>
+ : public ContiguousEnumSerializerInclusive<
+ mozilla::webgl::AttribBaseType,
+ mozilla::webgl::AttribBaseType::Boolean,
+ mozilla::webgl::AttribBaseType::Uint> {};
+
+template <>
+struct ParamTraits<mozilla::webgl::ContextLossReason>
+ : public ContiguousEnumSerializerInclusive<
+ mozilla::webgl::ContextLossReason,
+ mozilla::webgl::ContextLossReason::None,
+ mozilla::webgl::ContextLossReason::Guilty> {};
+
+template <>
+struct ParamTraits<gfxAlphaType>
+ : public ContiguousEnumSerializerInclusive<
+ gfxAlphaType, gfxAlphaType::Opaque, gfxAlphaType::NonPremult> {};
+
+// -
+// ParamTraits_TiedFields
+
+template <class T>
+struct ParamTraits_TiedFields {
+ static_assert(mozilla::AssertTiedFieldsAreExhaustive<T>());
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ const auto& fields = mozilla::TiedFields(in);
+ MapTuple(fields, [&](const auto& field) {
+ WriteParam(writer, field);
+ return true; // ignored
+ });
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ const auto& fields = mozilla::TiedFields(*out);
+ bool ok = true;
+ MapTuple(fields, [&](auto& field) {
+ if (ok) {
+ ok &= ReadParam(reader, &field);
+ }
+ return true; // ignored
+ });
+ return ok;
+ }
+};
+
+// -
+
+template <typename T>
+bool ValidateParam(const T& val) {
+ return ParamTraits<T>::Validate(val);
+}
+
+template <typename T>
+struct ValidatedPlainOldDataSerializer : public PlainOldDataSerializer<T> {
+ static void Write(MessageWriter* const writer, const T& in) {
+ MOZ_ASSERT(ValidateParam(in));
+ PlainOldDataSerializer<T>::Write(writer, in);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ if (!PlainOldDataSerializer<T>::Read(reader, out)) return false;
+ return ValidateParam(*out);
+ }
+
+ // static bool Validate(const T&) = 0;
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::InitContextDesc> final
+ : public ValidatedPlainOldDataSerializer<mozilla::webgl::InitContextDesc> {
+ using T = mozilla::webgl::InitContextDesc;
+
+ static bool Validate(const T& val) {
+ return ValidateParam(val.options) && (val.size.x && val.size.y);
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::WebGLContextOptions> final
+ : public ValidatedPlainOldDataSerializer<mozilla::WebGLContextOptions> {
+ using T = mozilla::WebGLContextOptions;
+
+ static bool Validate(const T& val) {
+ bool ok = true;
+ ok &= ValidateParam(val.powerPreference);
+ ok &= ValidateParam(val.colorSpace);
+ return ok;
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::dom::WebGLPowerPreference> final
+ : public ValidatedPlainOldDataSerializer<
+ mozilla::dom::WebGLPowerPreference> {
+ using T = mozilla::dom::WebGLPowerPreference;
+
+ static bool Validate(const T& val) { return val <= T::High_performance; }
+};
+
+template <>
+struct ParamTraits<mozilla::dom::PredefinedColorSpace> final
+ : public ValidatedPlainOldDataSerializer<
+ mozilla::dom::PredefinedColorSpace> {
+ using T = mozilla::dom::PredefinedColorSpace;
+
+ static bool Validate(const T& val) { return val < T::EndGuard_; }
+};
+
+template <>
+struct ParamTraits<mozilla::webgl::OpaqueFramebufferOptions> final
+ : public PlainOldDataSerializer<mozilla::webgl::OpaqueFramebufferOptions> {
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::gl::GLVendor>
+ : public ContiguousEnumSerializerInclusive<mozilla::gl::GLVendor,
+ mozilla::gl::GLVendor::Intel,
+ mozilla::gl::kHighestGLVendor> {
+};
+
+template <typename T>
+struct ParamTraits<mozilla::webgl::EnumMask<T>> final
+ : public PlainOldDataSerializer<mozilla::webgl::EnumMask<T>> {};
+
+template <>
+struct ParamTraits<mozilla::webgl::InitContextResult> final
+ : public ParamTraits_TiedFields<mozilla::webgl::InitContextResult> {};
+
+template <>
+struct ParamTraits<mozilla::webgl::ExtensionBits> final
+ : public PlainOldDataSerializer<mozilla::webgl::ExtensionBits> {};
+
+template <>
+struct ParamTraits<mozilla::webgl::Limits> final
+ : public PlainOldDataSerializer<mozilla::webgl::Limits> {};
+
+template <>
+struct ParamTraits<mozilla::webgl::PixelPackingState> final
+ : public PlainOldDataSerializer<mozilla::webgl::PixelPackingState> {};
+template <>
+struct ParamTraits<mozilla::webgl::PixelUnpackStateWebgl> final
+ : public PlainOldDataSerializer<mozilla::webgl::PixelUnpackStateWebgl> {};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::ReadPixelsDesc> final {
+ using T = mozilla::webgl::ReadPixelsDesc;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.srcOffset);
+ WriteParam(writer, in.size);
+ WriteParam(writer, in.pi);
+ WriteParam(writer, in.packState);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->srcOffset) &&
+ ReadParam(reader, &out->size) && ReadParam(reader, &out->pi) &&
+ ReadParam(reader, &out->packState);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::PackingInfo> final {
+ using T = mozilla::webgl::PackingInfo;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.format);
+ WriteParam(writer, in.type);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->format) && ReadParam(reader, &out->type);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::CompileResult> final {
+ using T = mozilla::webgl::CompileResult;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.pending);
+ WriteParam(writer, in.log);
+ WriteParam(writer, in.translatedSource);
+ WriteParam(writer, in.success);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->pending) && ReadParam(reader, &out->log) &&
+ ReadParam(reader, &out->translatedSource) &&
+ ReadParam(reader, &out->success);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::LinkResult> final {
+ using T = mozilla::webgl::LinkResult;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.pending);
+ WriteParam(writer, in.log);
+ WriteParam(writer, in.success);
+ WriteParam(writer, in.active);
+ WriteParam(writer, in.tfBufferMode);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->pending) && ReadParam(reader, &out->log) &&
+ ReadParam(reader, &out->success) &&
+ ReadParam(reader, &out->active) &&
+ ReadParam(reader, &out->tfBufferMode);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::LinkActiveInfo> final {
+ using T = mozilla::webgl::LinkActiveInfo;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.activeAttribs);
+ WriteParam(writer, in.activeUniforms);
+ WriteParam(writer, in.activeUniformBlocks);
+ WriteParam(writer, in.activeTfVaryings);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->activeAttribs) &&
+ ReadParam(reader, &out->activeUniforms) &&
+ ReadParam(reader, &out->activeUniformBlocks) &&
+ ReadParam(reader, &out->activeTfVaryings);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::ActiveInfo> final {
+ using T = mozilla::webgl::ActiveInfo;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.elemType);
+ WriteParam(writer, in.elemCount);
+ WriteParam(writer, in.name);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->elemType) &&
+ ReadParam(reader, &out->elemCount) && ReadParam(reader, &out->name);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::ActiveAttribInfo> final {
+ using T = mozilla::webgl::ActiveAttribInfo;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, static_cast<const mozilla::webgl::ActiveInfo&>(in));
+ WriteParam(writer, in.location);
+ WriteParam(writer, in.baseType);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, static_cast<mozilla::webgl::ActiveInfo*>(out)) &&
+ ReadParam(reader, &out->location) &&
+ ReadParam(reader, &out->baseType);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::ActiveUniformInfo> final {
+ using T = mozilla::webgl::ActiveUniformInfo;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, static_cast<const mozilla::webgl::ActiveInfo&>(in));
+ WriteParam(writer, in.locByIndex);
+ WriteParam(writer, in.block_index);
+ WriteParam(writer, in.block_offset);
+ WriteParam(writer, in.block_arrayStride);
+ WriteParam(writer, in.block_matrixStride);
+ WriteParam(writer, in.block_isRowMajor);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, static_cast<mozilla::webgl::ActiveInfo*>(out)) &&
+ ReadParam(reader, &out->locByIndex) &&
+ ReadParam(reader, &out->block_index) &&
+ ReadParam(reader, &out->block_offset) &&
+ ReadParam(reader, &out->block_arrayStride) &&
+ ReadParam(reader, &out->block_matrixStride) &&
+ ReadParam(reader, &out->block_isRowMajor);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::ActiveUniformBlockInfo> final {
+ using T = mozilla::webgl::ActiveUniformBlockInfo;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.name);
+ WriteParam(writer, in.dataSize);
+ WriteParam(writer, in.activeUniformIndices);
+ WriteParam(writer, in.referencedByVertexShader);
+ WriteParam(writer, in.referencedByFragmentShader);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->name) && ReadParam(reader, &out->dataSize) &&
+ ReadParam(reader, &out->activeUniformIndices) &&
+ ReadParam(reader, &out->referencedByVertexShader) &&
+ ReadParam(reader, &out->referencedByFragmentShader);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::ShaderPrecisionFormat> final {
+ using T = mozilla::webgl::ShaderPrecisionFormat;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.rangeMin);
+ WriteParam(writer, in.rangeMax);
+ WriteParam(writer, in.precision);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->rangeMin) &&
+ ReadParam(reader, &out->rangeMax) &&
+ ReadParam(reader, &out->precision);
+ }
+};
+
+// -
+
+template <typename U, size_t N>
+struct ParamTraits<U[N]> final {
+ using T = U[N];
+ static constexpr size_t kByteSize = sizeof(U) * N;
+
+ static_assert(std::is_trivial<U>::value);
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ writer->WriteBytes(in, kByteSize);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ if (!reader->HasBytesAvailable(kByteSize)) {
+ return false;
+ }
+ return reader->ReadBytesInto(*out, kByteSize);
+ }
+};
+
+// -
+
+template <>
+struct ParamTraits<mozilla::webgl::GetUniformData> final {
+ using T = mozilla::webgl::GetUniformData;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ ParamTraits<decltype(in.data)>::Write(writer, in.data);
+ WriteParam(writer, in.type);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ParamTraits<decltype(out->data)>::Read(reader, &out->data) &&
+ ReadParam(reader, &out->type);
+ }
+};
+
+// -
+
+template <typename U>
+struct ParamTraits<mozilla::avec2<U>> final {
+ using T = mozilla::avec2<U>;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.x);
+ WriteParam(writer, in.y);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->x) && ReadParam(reader, &out->y);
+ }
+};
+
+// -
+
+template <typename U>
+struct ParamTraits<mozilla::avec3<U>> final {
+ using T = mozilla::avec3<U>;
+
+ static void Write(MessageWriter* const writer, const T& in) {
+ WriteParam(writer, in.x);
+ WriteParam(writer, in.y);
+ WriteParam(writer, in.z);
+ }
+
+ static bool Read(MessageReader* const reader, T* const out) {
+ return ReadParam(reader, &out->x) && ReadParam(reader, &out->y) &&
+ ReadParam(reader, &out->z);
+ }
+};
+
+} // namespace IPC
+
+#endif