forked from mirrors/gecko-dev
This updates the version wpf-gpu-raster which adds support for GPUs/drivers that use truncation instead of rounding when converting vertices to fixed point. It also adds the GL vendor to InitContextResult so that we can detect AMD on macOS and tell wpf-gpu-raster that truncation is going to happen. Differential Revision: https://phabricator.services.mozilla.com/D167503
613 lines
17 KiB
C++
613 lines
17 KiB
C++
/* -*- 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 "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 <>
|
|
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> {};
|
|
|
|
// -
|
|
|
|
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 {
|
|
using T = mozilla::webgl::InitContextResult;
|
|
|
|
static void Write(MessageWriter* const writer, const T& in) {
|
|
WriteParam(writer, in.error);
|
|
WriteParam(writer, in.options);
|
|
WriteParam(writer, in.limits);
|
|
WriteParam(writer, in.uploadableSdTypes);
|
|
WriteParam(writer, in.vendor);
|
|
}
|
|
|
|
static bool Read(MessageReader* const reader, T* const out) {
|
|
return ReadParam(reader, &out->error) && ReadParam(reader, &out->options) &&
|
|
ReadParam(reader, &out->limits) &&
|
|
ReadParam(reader, &out->uploadableSdTypes) &&
|
|
ReadParam(reader, &out->vendor);
|
|
}
|
|
};
|
|
|
|
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
|