From 8e41908e49f845e5937768b9631886e1de37787b Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sat, 31 Dec 2022 19:32:33 +0100 Subject: [PATCH 01/19] Partial spherical harmonics tests --- tests/ktx/src/KtxTests.cpp | 25 ++++++++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/tests/ktx/src/KtxTests.cpp b/tests/ktx/src/KtxTests.cpp index 25ab15f5db..cac9690b5c 100644 --- a/tests/ktx/src/KtxTests.cpp +++ b/tests/ktx/src/KtxTests.cpp @@ -16,7 +16,9 @@ #include #include #include - +#include "SerDes.h" +#include "KTX.h" +#include "Texture_ktx.cpp" QTEST_GUILESS_MAIN(KtxTests) @@ -31,6 +33,19 @@ QString getRootPath() { return result; } +#if 0 +ktx::Byte* serializeSPH(ktx::Byte* data, const gpu::IrradianceKTXPayload &payload) const { + *(ktx::IrradianceKTXPayload::Version*)data = IrradianceKTXPayload::CURRENT_VERSION; + data += sizeof(ktx::IrradianceKTXPayload::Version); + + memcpy(data, &payload._irradianceSH, sizeof(ktx::SphericalHarmonics)); + data += sizeof(SphericalHarmonics); + + return data + PADDING; +} +#endif + + void KtxTests::initTestCase() { } @@ -147,6 +162,14 @@ void KtxTests::testKtxSerialization() { testTexture->setKtxBacking(TEST_IMAGE_KTX.fileName().toStdString()); } + +void KtxTests::testKtxNewSerializationSphericalHarmonics() { + DataSerializer ser; + + +} + + #if 0 static const QString TEST_FOLDER { "H:/ktx_cacheold" }; From 23a5795ba9ee0c8a737db17eb7f3f6af6e9e22b0 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Fri, 10 Jun 2022 00:45:12 +0200 Subject: [PATCH 02/19] Initial version --- libraries/gpu/src/gpu/Texture.h | 56 +++- libraries/gpu/src/gpu/Texture_ktx.cpp | 47 +++- libraries/shared/src/SerDes.cpp | 70 +++++ libraries/shared/src/SerDes.h | 382 ++++++++++++++++++++++++++ tests/shared/src/SerializerTests.cpp | 107 ++++++++ tests/shared/src/SerializerTests.h | 29 ++ 6 files changed, 683 insertions(+), 8 deletions(-) create mode 100644 libraries/shared/src/SerDes.cpp create mode 100644 libraries/shared/src/SerDes.h create mode 100644 tests/shared/src/SerializerTests.cpp create mode 100644 tests/shared/src/SerializerTests.h diff --git a/libraries/gpu/src/gpu/Texture.h b/libraries/gpu/src/gpu/Texture.h index 907f9ff392..5143cc1c23 100644 --- a/libraries/gpu/src/gpu/Texture.h +++ b/libraries/gpu/src/gpu/Texture.h @@ -24,6 +24,7 @@ #include "Forward.h" #include "Resource.h" #include "Metric.h" +#include "SerDes.h" const int ABSOLUTE_MAX_TEXTURE_NUM_PIXELS = 8192 * 8192; @@ -136,7 +137,7 @@ public: uint8 _wrapModeU = WRAP_REPEAT; uint8 _wrapModeV = WRAP_REPEAT; uint8 _wrapModeW = WRAP_REPEAT; - + uint8 _mipOffset = 0; uint8 _minMip = 0; uint8 _maxMip = MAX_MIP_LEVEL; @@ -156,8 +157,24 @@ public: _minMip == other._minMip && _maxMip == other._maxMip; } + + SerDes &operator<<(SerDes &dsd) { + dsd << _borderColor; + dsd << _maxAnisotropy; + dsd << _filter; + dsd << _comparisonFunc; + dsd << _wrapModeU; + dsd << _wrapModeV; + dsd << _wrapModeW; + dsd << _mipOffset; + dsd << _minMip; + dsd << _maxMip; + return dsd; + } }; + + Sampler() {} Sampler(const Filter filter, const WrapMode wrap = WRAP_REPEAT) : _desc(filter, wrap) {} Sampler(const Desc& desc) : _desc(desc) {} @@ -193,6 +210,33 @@ protected: friend class Deserializer; }; +inline SerDes &operator<<(SerDes &ser, const Sampler::Desc &d) { + ser << d._borderColor; + ser << d._maxAnisotropy; + ser << d._filter; + ser << d._comparisonFunc; + ser << d._wrapModeU; + ser << d._wrapModeV; + ser << d._wrapModeW; + ser << d._mipOffset; + ser << d._minMip; + ser << d._maxMip; + return ser; +} + +inline SerDes &operator>>(SerDes &dsr, Sampler::Desc &d) { + dsr >> d._borderColor; + dsr >> d._maxAnisotropy; + dsr >> d._filter; + dsr >> d._comparisonFunc; + dsr >> d._wrapModeU; + dsr >> d._wrapModeV; + dsr >> d._wrapModeW; + dsr >> d._mipOffset; + dsr >> d._minMip; + dsr >> d._maxMip; + return dsr; +} enum class TextureUsageType : uint8 { RENDERBUFFER, // Used as attachments to a framebuffer RESOURCE, // Resource textures, like materials... subject to memory manipulation @@ -230,7 +274,7 @@ public: NORMAL, // Texture is a normal map ALPHA, // Texture has an alpha channel ALPHA_MASK, // Texture alpha channel is a Mask 0/1 - NUM_FLAGS, + NUM_FLAGS, }; typedef std::bitset Flags; @@ -478,7 +522,7 @@ public: uint16 evalMipDepth(uint16 level) const { return std::max(_depth >> level, 1); } // The true size of an image line or surface depends on the format, tiling and padding rules - // + // // Here are the static function to compute the different sizes from parametered dimensions and format // Tile size must be a power of 2 static uint16 evalTiledPadding(uint16 length, int tile) { int tileMinusOne = (tile - 1); return (tileMinusOne - (length + tileMinusOne) % tile); } @@ -507,7 +551,7 @@ public: uint32 evalMipFaceNumTexels(uint16 level) const { return evalMipWidth(level) * evalMipHeight(level) * evalMipDepth(level); } uint32 evalMipNumTexels(uint16 level) const { return evalMipFaceNumTexels(level) * getNumFaces(); } - // For convenience assign a source name + // For convenience assign a source name const std::string& source() const { return _source; } void setSource(const std::string& source) { _source = source; } const std::string& sourceHash() const { return _sourceHash; } @@ -633,7 +677,7 @@ protected: uint16 _maxMipLevel { 0 }; uint16 _minMip { 0 }; - + Type _type { TEX_1D }; Usage _usage; @@ -643,7 +687,7 @@ protected: bool _isIrradianceValid = false; bool _defined = false; bool _important = false; - + static TexturePointer create(TextureUsageType usageType, Type type, const Element& texelFormat, uint16 width, uint16 height, uint16 depth, uint16 numSamples, uint16 numSlices, uint16 numMips, const Sampler& sampler); Size resize(Type type, const Element& texelFormat, uint16 width, uint16 height, uint16 depth, uint16 numSamples, uint16 numSlices, uint16 numMips); diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index c4b674a917..748ec1a702 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -18,6 +18,7 @@ #include #include "GPULogging.h" +#include "SerDes.h" using namespace gpu; @@ -42,6 +43,44 @@ struct GPUKTXPayload { TextureUsageType _usageType; glm::ivec2 _originalSize { 0, 0 }; + void serialize2(SerDes &ser) { + ser << CURRENT_VERSION; + ser << _samplerDesc; + + uint32 usageData = _usage._flags.to_ulong(); + ser << usageData; + + ser << (char)_usageType; + ser << _originalSize; + ser.addPadding(PADDING); + } + + bool unserialize2(SerDes &dsr) { + Version version = 0; + uint32 usageData; + uint8_t usagetype = 0; + + dsr >> version; + + if (version > CURRENT_VERSION) { + // If we try to load a version that we don't know how to parse, + // it will render incorrectly + return false; + } + + dsr >> _samplerDesc; + dsr >> usageData; + dsr >> usagetype; + _usageType = (TextureUsageType)usagetype; + + if (version >= 2) { + dsr >> _originalSize; + } + + return true; + } + + Byte* serialize(Byte* data) const { *(Version*)data = CURRENT_VERSION; data += sizeof(Version); @@ -103,7 +142,9 @@ struct GPUKTXPayload { auto found = std::find_if(keyValues.begin(), keyValues.end(), isGPUKTX); if (found != keyValues.end()) { auto value = found->_value; - return payload.unserialize(value.data(), value.size()); + SerDes dsr(value.data(), value.size()); + return payload.unserialize2(dsr); + //return payload.unserialize(value.data(), value.size()); } return false; } @@ -467,7 +508,9 @@ ktx::KTXUniquePointer Texture::serialize(const Texture& texture, const glm::ivec gpuKeyval._originalSize = originalSize; Byte keyvalPayload[GPUKTXPayload::SIZE]; - gpuKeyval.serialize(keyvalPayload); + SerDes ser(keyvalPayload, sizeof(keyvalPayload)); + + gpuKeyval.serialize2(ser); ktx::KeyValues keyValues; keyValues.emplace_back(GPUKTXPayload::KEY, (uint32)GPUKTXPayload::SIZE, (ktx::Byte*) &keyvalPayload); diff --git a/libraries/shared/src/SerDes.cpp b/libraries/shared/src/SerDes.cpp new file mode 100644 index 0000000000..c64430f201 --- /dev/null +++ b/libraries/shared/src/SerDes.cpp @@ -0,0 +1,70 @@ +// +// SerDes.h +// +// +// Created by Dale Glass on 5/6/2022 +// Copyright 2022 Dale Glass +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#include + +#include "SerDes.h" +const int SerDes::DEFAULT_SIZE; +const char SerDes::PADDING_CHAR; + +QDebug operator<<(QDebug debug, const SerDes &ds) { + debug << "{ capacity =" << ds.capacity() << "; length = " << ds.length() << "; pos = " << ds.pos() << "}"; + debug << "\n"; + + QString literal; + QString hex; + + for(size_t i=0;i(c), 16 ); + if ( hnum.length() == 1 ) { + hnum.prepend("0"); + } + + hex.append(hnum + " "); + + if ( literal.length() == 16 || (i+1 == ds.length()) ) { + while( literal.length() < 16 ) { + literal.append(" "); + hex.append(" "); + } + + debug << literal << " " << hex << "\n"; + literal.clear(); + hex.clear(); + } + } + + return debug; +} + + +void SerDes::changeAllocation(size_t new_size) { + while ( _capacity < new_size) { + _capacity *= 2; + } + + char *new_buf = new char[_capacity]; + assert( *new_buf ); + + memcpy(new_buf, _store, _length); + char *prev_buf = _store; + _store = new_buf; + + delete []prev_buf; +} diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h new file mode 100644 index 0000000000..4651ac8353 --- /dev/null +++ b/libraries/shared/src/SerDes.h @@ -0,0 +1,382 @@ +// +// SerDes.h +// +// +// Created by Dale Glass on 5/6/2022 +// Copyright 2022 Dale Glass +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#pragma once +#include +#include +#include +#include +#include + +/** + * @brief Data serializer/deserializer + * + * When encoding, this class takes in data and encodes it into a buffer. No attempt is made to store version numbers, lengths, + * or any other metadata. It's entirely up to the user to use the class in such a way that the process can be + * correctly reversed if variable-length or optional fields are used. + * + * It can operate both on an internal, dynamically-allocated buffer, or an externally provided, fixed-size one. + * + * If an external store is used, the class will refuse to add data once capacity is reached and set the overflow flag. + * + * When decoding, this class operates on a fixed size buffer. If an attempt to read past the end is made, the read fails, + * and the overflow flag is set. + * + * The class was written for the maximum simplicity possible and inline friendliness. + */ +class SerDes { + public: + // This class is aimed at network serialization, so we assume we're going to deal + // with something MTU-sized by default. + static const int DEFAULT_SIZE = 1500; + static const char PADDING_CHAR = 0xAA; + + /** + * @brief Construct a dynamically allocated serializer + * + * If constructed this way, an internal buffer will be dynamically allocated and grown as needed. + * + * The default buffer size is 1500 bytes, based on the assumption that it will be used to construct + * network packets. + */ + SerDes() { + _capacity = DEFAULT_SIZE; + _pos = 0; + _length = 0; + _store = new char[_capacity]; + } + + /** + * @brief Construct a statically allocated serializer + * + * If constructed this way, the external buffer will be used to store data. The class will refuse to + * keep adding data if the maximum length is reached, and set the overflow flag. + * + * The flag can be read with isOverflow() + * + * @param externalStore External data store + * @param storeLength Length of the data store + */ + SerDes(char *externalStore, size_t storeLength) { + _capacity = storeLength; + _length = storeLength; + _pos = 0; + _storeIsExternal = true; + _store = externalStore; + } + + SerDes(uint8_t *externalStore, size_t storeLength) : SerDes((char*)externalStore, storeLength) { + + } + + SerDes(const SerDes &) = delete; + SerDes &operator=(const SerDes &) = delete; + + + + ~SerDes() { + if (!_storeIsExternal) { + delete[] _store; + } + } + + void addPadding(size_t bytes) { + if (!extendBy(bytes)) { + return; + } + + // Fill padding with something recognizable. Will keep valgrind happier. + memset(&_store[_pos], PADDING_CHAR, bytes); + _pos += bytes; + } + + SerDes &operator<<(uint8_t c) { + return *this << int8_t(c); + } + + SerDes &operator<<(int8_t c) { + if (!extendBy(1)) { + return *this; + } + + _store[_pos++] = c; + return *this; + } + + SerDes &operator>>(uint8_t &c) { + return *this >> reinterpret_cast(c); + } + + SerDes &operator>>(int8_t &c) { + if ( _pos < _length ) { + c = _store[_pos++]; + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading 8 bits from position " << _pos << ", length " << _length; + } + + return *this; + } + + /////////////////////////////////////////////////////////// + + SerDes &operator<<(uint16_t val) { + return *this << int16_t(val); + } + + SerDes &operator<<(int16_t val) { + if (!extendBy(sizeof(val))) { + return *this; + } + + memcpy(&_store[_pos], (char*)&val, sizeof(val)); + _pos += sizeof(val); + return *this; + } + + SerDes &operator>>(uint16_t &val) { + return *this >> reinterpret_cast(val); + } + + SerDes &operator>>(int16_t &val) { + if ( _pos + sizeof(val) <= _length ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading 16 bits from position " << _pos << ", length " << _length; + } + + return *this; + } + + /////////////////////////////////////////////////////////// + + SerDes &operator<<(uint32_t val) { + return *this << int32_t(val); + } + + SerDes &operator<<(int32_t val) { + if (!extendBy(sizeof(val))) { + return *this; + } + + memcpy(&_store[_pos], (char*)&val, sizeof(val)); + _pos += sizeof(val); + return *this; + } + + SerDes &operator>>(uint32_t &val) { + return *this >> reinterpret_cast(val); + } + + SerDes &operator>>(int32_t &val) { + if ( _pos + sizeof(val) <= _length ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading 32 bits from position " << _pos << ", length " << _length; + } + return *this; + } + + + /////////////////////////////////////////////////////////// + + SerDes &operator<<(glm::vec3 val) { + size_t sz = sizeof(val.x); + if (!extendBy(sz*3)) { + return *this; + } + + memcpy(&_store[_pos ], (char*)&val.x, sz); + memcpy(&_store[_pos + sz ], (char*)&val.y, sz); + memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); + + _pos += sz*3; + return *this; + } + + SerDes &operator>>(glm::vec3 &val) { + size_t sz = sizeof(val.x); + + if ( _pos + sz*3 <= _length ) { + memcpy((char*)&val.x, &_store[_pos ], sz); + memcpy((char*)&val.y, &_store[_pos + sz ], sz); + memcpy((char*)&val.z, &_store[_pos + sz*2], sz); + + _pos += sz*3; + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading glm::vec3 from position " << _pos << ", length " << _length; + } + return *this; + } + + /////////////////////////////////////////////////////////// + + SerDes &operator<<(glm::vec4 val) { + size_t sz = sizeof(val.x); + if (!extendBy(sz*4)) { + return *this; + } + + memcpy(&_store[_pos ], (char*)&val.x, sz); + memcpy(&_store[_pos + sz ], (char*)&val.y, sz); + memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); + memcpy(&_store[_pos + sz*3], (char*)&val.w, sz); + + _pos += sz*3; + return *this; + } + + SerDes &operator>>(glm::vec4 &val) { + size_t sz = sizeof(val.x); + + if ( _pos + sz*4 <= _length ) { + memcpy((char*)&val.x, &_store[_pos ], sz); + memcpy((char*)&val.y, &_store[_pos + sz ], sz); + memcpy((char*)&val.z, &_store[_pos + sz*2], sz); + memcpy((char*)&val.w, &_store[_pos + sz*3], sz); + + _pos += sz*3; + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading glm::vec3 from position " << _pos << ", length " << _length; + } + return *this; + } + + /////////////////////////////////////////////////////////// + + SerDes &operator<<(glm::ivec2 val) { + size_t sz = sizeof(val.x); + if (!extendBy(sz*2)) { + return *this; + } + + memcpy(&_store[_pos ], (char*)&val.x, sz); + memcpy(&_store[_pos + sz ], (char*)&val.y, sz); + + _pos += sz*2; + return *this; + } + + SerDes &operator>>(glm::ivec2 &val) { + size_t sz = sizeof(val.x); + + if ( _pos + sz*2 <= _length ) { + memcpy((char*)&val.x, &_store[_pos ], sz); + memcpy((char*)&val.y, &_store[_pos + sz ], sz); + + _pos += sz*2; + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading glm::ivec2 from position " << _pos << ", length " << _length; + } + return *this; + } + /////////////////////////////////////////////////////////// + + SerDes &operator<<(const char *val) { + size_t len = strlen(val)+1; + extendBy(len); + memcpy(&_store[_pos], val, len); + _pos += len; + return *this; + } + + SerDes &operator<<(const QString &val) { + return *this << val.toUtf8().constData(); + } + + + /////////////////////////////////////////////////////////// + + /** + * @brief Current position in the buffer. Starts at 0. + * + * @return size_t + */ + size_t pos() const { return _pos; } + + /** + * @brief Last position that was written to in the buffer. Starts at 0. + * + * @return size_t + */ + size_t length() const { return _length; } + + /** + * @brief Current capacity of the buffer. + * + * If the buffer is dynamically allocated, it can grow. + * + * If the buffer is static, this is a fixed limit. + * + * @return size_t + */ + size_t capacity() const { return _capacity; } + + /** + * @brief Whether there's any data in the buffer + * + * @return true Something has been written + * @return false The buffer is empty + */ + bool isEmpty() const { return _length == 0; } + + /** + * @brief The buffer size limit has been reached + * + * This can only return true for a statically allocated buffer. + * + * @return true Limit reached + * @return false There is still room + */ + bool isOverflow() const { return _overflow; } + + /** + * @brief Reset the serializer to the start, clear overflow bit. + * + */ + void rewind() { _pos = 0; _overflow = false; } + + friend QDebug operator<<(QDebug debug, const SerDes &ds); + + private: + bool extendBy(size_t bytes) { + //qDebug() << "Extend by" << bytes; + + if ( _capacity < _length + bytes) { + if ( _storeIsExternal ) { + _overflow = true; + return false; + } + + changeAllocation(_length + bytes); + } + + _length += bytes; + return true; + } + + // This is split up here to try to make the class as inline-friendly as possible. + void changeAllocation(size_t new_size); + + char *_store; + bool _storeIsExternal = false; + bool _overflow = false; + size_t _capacity = 0; + size_t _length = 0; + size_t _pos = 0; +}; diff --git a/tests/shared/src/SerializerTests.cpp b/tests/shared/src/SerializerTests.cpp new file mode 100644 index 0000000000..dc5ffa2160 --- /dev/null +++ b/tests/shared/src/SerializerTests.cpp @@ -0,0 +1,107 @@ +// +// SerializerTests.cpp +// +// Copyright 2022 Dale Glass +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + + +#include "SerializerTests.h" +#include +#include +#include + +QTEST_GUILESS_MAIN(SerializerTests) + + +void SerializerTests::initTestCase() { +} + +void SerializerTests::testCreate() { + SerDes s; + QCOMPARE(s.length(), 0); + QCOMPARE(s.capacity(), SerDes::DEFAULT_SIZE); + QCOMPARE(s.isEmpty(), true); +} + +void SerializerTests::testAdd() { + SerDes s; + s << (qint8)1; + QCOMPARE(s.length(), 1); + QCOMPARE(s.isEmpty(), false); + + s << (quint8)-1; + QCOMPARE(s.length(), 2); + + s << (qint16)0xaabb; + QCOMPARE(s.length(), 4); + + s << (quint16)-18000; + QCOMPARE(s.length(), 6); + + s << (qint32)0xCCDDEEFF; + QCOMPARE(s.length(), 10); + + s << (quint32)-1818000000; + QCOMPARE(s.length(), 14); + + s << "Hello, world!"; + QCOMPARE(s.length(), 28); + + glm::vec3 v{1.f,2.f,3.f}; + s << v; + QCOMPARE(s.length(), 40); + + + qDebug() << s; +} + +void SerializerTests::testAddAndRead() { + SerDes s; + glm::vec3 v{1.f, 3.1415f, 2.71828f}; + glm::vec3 v2; + + s << (qint8)1; + s << (qint16)0xaabb; + s << (qint32)0xccddeeff; + s << v; + + qint8 i8; + qint16 i16; + qint32 i32; + + s.rewind(); + + s >> i8; + s >> i16; + s >> i32; + s >> v2; + + qDebug() << s; + + QCOMPARE(i8, (qint8)1); + QCOMPARE(i16, (qint16)0xaabb); + QCOMPARE(i32, (qint32)0xccddeeff); + QCOMPARE(v, v2); +} + +void SerializerTests::testReadPastEnd() { + SerDes s; + qint8 i8; + qint16 i16; + s << (qint8)1; + s.rewind(); + s >> i8; + QCOMPARE(s.pos(), 1); + + s.rewind(); + s >> i16; + QCOMPARE(s.pos(), 0); +} + + +void SerializerTests::cleanupTestCase() { +} + diff --git a/tests/shared/src/SerializerTests.h b/tests/shared/src/SerializerTests.h new file mode 100644 index 0000000000..1320d99c57 --- /dev/null +++ b/tests/shared/src/SerializerTests.h @@ -0,0 +1,29 @@ +// +// ResourceTests.h +// +// Copyright 2015 High Fidelity, Inc. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#ifndef overte_SerializerTests_h +#define overte_SerializerTests_h + +#include +#include + +class SerializerTests : public QObject { + Q_OBJECT +private slots: + void initTestCase(); + void testCreate(); + void testAdd(); + void testAddAndRead(); + void testReadPastEnd(); + void cleanupTestCase(); +private: + +}; + +#endif // overte_SerializerTests_h From 30ced59cec1b6bde8d59b88e87e2748cb7d4777b Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sat, 11 Jun 2022 21:36:40 +0200 Subject: [PATCH 03/19] Fix vec4 serialization --- libraries/shared/src/SerDes.h | 4 ++-- tests/shared/src/SerializerTests.cpp | 20 +++++++++++++++----- 2 files changed, 17 insertions(+), 7 deletions(-) diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index 4651ac8353..b608d452ab 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -235,7 +235,7 @@ class SerDes { memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); memcpy(&_store[_pos + sz*3], (char*)&val.w, sz); - _pos += sz*3; + _pos += sz*4; return *this; } @@ -248,7 +248,7 @@ class SerDes { memcpy((char*)&val.z, &_store[_pos + sz*2], sz); memcpy((char*)&val.w, &_store[_pos + sz*3], sz); - _pos += sz*3; + _pos += sz*4; } else { _overflow = true; qCritical() << "Deserializer trying to read past end of input, reading glm::vec3 from position " << _pos << ", length " << _length; diff --git a/tests/shared/src/SerializerTests.cpp b/tests/shared/src/SerializerTests.cpp index dc5ffa2160..bc50554d96 100644 --- a/tests/shared/src/SerializerTests.cpp +++ b/tests/shared/src/SerializerTests.cpp @@ -60,13 +60,19 @@ void SerializerTests::testAdd() { void SerializerTests::testAddAndRead() { SerDes s; - glm::vec3 v{1.f, 3.1415f, 2.71828f}; - glm::vec3 v2; + glm::vec3 v3_a{1.f, 3.1415f, 2.71828f}; + glm::vec3 v3_b; + glm::vec4 v4_a{3.1415f, 2.71828f, 1.4142f, 1.6180f}; + glm::vec4 v4_b; + glm::ivec2 iv2_a{10, 24}; + glm::ivec2 iv2_b; s << (qint8)1; s << (qint16)0xaabb; s << (qint32)0xccddeeff; - s << v; + s << v3_a; + s << v4_a; + s << iv2_a; qint8 i8; qint16 i16; @@ -77,14 +83,18 @@ void SerializerTests::testAddAndRead() { s >> i8; s >> i16; s >> i32; - s >> v2; + s >> v3_b; + s >> v4_b; + s >> iv2_b; qDebug() << s; QCOMPARE(i8, (qint8)1); QCOMPARE(i16, (qint16)0xaabb); QCOMPARE(i32, (qint32)0xccddeeff); - QCOMPARE(v, v2); + QCOMPARE(v3_a, v3_b); + QCOMPARE(v4_a, v4_b); + QCOMPARE(iv2_a, iv2_b); } void SerializerTests::testReadPastEnd() { From 9c97928751f958a1424ff95e98dd26c1859cafe4 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sun, 12 Jun 2022 01:24:55 +0200 Subject: [PATCH 04/19] Add benchmarks --- tests/shared/src/SerializerTests.cpp | 73 ++++++++++++++++++++++++++++ tests/shared/src/SerializerTests.h | 3 ++ 2 files changed, 76 insertions(+) diff --git a/tests/shared/src/SerializerTests.cpp b/tests/shared/src/SerializerTests.cpp index bc50554d96..a1b3a37544 100644 --- a/tests/shared/src/SerializerTests.cpp +++ b/tests/shared/src/SerializerTests.cpp @@ -111,6 +111,79 @@ void SerializerTests::testReadPastEnd() { QCOMPARE(s.pos(), 0); } +void SerializerTests::benchmarkEncodingDynamicAlloc() { + QBENCHMARK { + SerDes s; + glm::vec3 v3_a{1.f, 3.1415f, 2.71828f}; + glm::vec3 v3_b; + glm::vec4 v4_a{3.1415f, 2.71828f, 1.4142f, 1.6180f}; + glm::vec4 v4_b; + glm::ivec2 iv2_a{10, 24}; + glm::ivec2 iv2_b; + + s << (qint8)1; + s << (qint16)0xaabb; + s << (qint32)0xccddeeff; + s << v3_a; + s << v4_a; + s << iv2_a; + } +} + +void SerializerTests::benchmarkEncodingStaticAlloc() { + char buf[1024]; + + QBENCHMARK { + SerDes s(buf, sizeof(buf)); + glm::vec3 v3_a{1.f, 3.1415f, 2.71828f}; + glm::vec3 v3_b; + glm::vec4 v4_a{3.1415f, 2.71828f, 1.4142f, 1.6180f}; + glm::vec4 v4_b; + glm::ivec2 iv2_a{10, 24}; + glm::ivec2 iv2_b; + + s << (qint8)1; + s << (qint16)0xaabb; + s << (qint32)0xccddeeff; + s << v3_a; + s << v4_a; + s << iv2_a; + } +} + + +void SerializerTests::benchmarkDecoding() { + SerDes s; + qint8 q8 = 1; + qint16 q16 = 0xaabb; + qint32 q32 = 0xccddeeff; + + glm::vec3 v3_a{1.f, 3.1415f, 2.71828f}; + glm::vec3 v3_b; + glm::vec4 v4_a{3.1415f, 2.71828f, 1.4142f, 1.6180f}; + glm::vec4 v4_b; + glm::ivec2 iv2_a{10, 24}; + glm::ivec2 iv2_b; + + s << q8; + s << q16; + s << q32; + s << v3_a; + s << v4_a; + s << iv2_a; + + + QBENCHMARK { + s.rewind(); + s >> q8; + s >> q16; + s >> q32; + s >> v3_a; + s >> v4_a; + s >> iv2_a; + } +} + void SerializerTests::cleanupTestCase() { } diff --git a/tests/shared/src/SerializerTests.h b/tests/shared/src/SerializerTests.h index 1320d99c57..3a3a3217d2 100644 --- a/tests/shared/src/SerializerTests.h +++ b/tests/shared/src/SerializerTests.h @@ -21,6 +21,9 @@ private slots: void testAdd(); void testAddAndRead(); void testReadPastEnd(); + void benchmarkEncodingDynamicAlloc(); + void benchmarkEncodingStaticAlloc(); + void benchmarkDecoding(); void cleanupTestCase(); private: From f6c203b54bad948b8068297666e2fe93831f46f1 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sun, 12 Jun 2022 01:25:38 +0200 Subject: [PATCH 05/19] Add documentation --- libraries/shared/src/SerDes.h | 239 +++++++++++++++++++++++++++++++++- 1 file changed, 232 insertions(+), 7 deletions(-) diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index b608d452ab..fe445fe854 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -24,19 +24,79 @@ * correctly reversed if variable-length or optional fields are used. * * It can operate both on an internal, dynamically-allocated buffer, or an externally provided, fixed-size one. - * * If an external store is used, the class will refuse to add data once capacity is reached and set the overflow flag. - * * When decoding, this class operates on a fixed size buffer. If an attempt to read past the end is made, the read fails, * and the overflow flag is set. * * The class was written for the maximum simplicity possible and inline friendliness. + * + * Example of encoding: + * + * @code {.cpp} + * uint8_t version = 1; + * uint16_t count = 1; + * glm::vec3 pos{1.5, 2.0, 9.0}; + * + * SerDes ser; + * ser << version; + * ser << count; + * ser << pos; + * + * // Serialized data is in ser.buffer(), ser.length() long. + * @endcode + * + * Example of decoding: + * + * @code {.cpp} + * // Incoming data has been placed in: + * // char buffer[1024]; + * + * uint8_t version; + * uint16_t count; + * glm::vec3 pos; + * + * SerDes des(buffer, sizeof(buffer)); + * des >> version; + * des >> count; + * des >> pos; + * @endcode + * + * This object should be modified directly to add support for any primitive and common datatypes in the code. To support serializing/deserializing + * classes and structures, implement a `operator<<` and `operator>>` functions for that object, eg: + * + * @code {.cpp} + * SerDes &operator<<(SerDes &ser, const Object &o) { + * ser << o._borderColor; + * ser << o._maxAnisotropy; + * ser << o._filter; + * return ser; + * } + * + * SerDes &operator>>(SerDes &des, Object &o) { + * des >> o._borderColor; + * des >> o._maxAnisotropy; + * des >> o._filter; + * return des; + * } + * + * @endcode + * */ class SerDes { public: - // This class is aimed at network serialization, so we assume we're going to deal - // with something MTU-sized by default. + /** + * @brief Default size for a dynamically allocated buffer. + * + * Since this is mostly intended to be used for networking, we default to the largest probable MTU here. + */ static const int DEFAULT_SIZE = 1500; + + /** + * @brief Character to use for padding. + * + * Padding should be ignored, so it doesn't matter what we go with here, but it can be useful to set it + * to something that would be distinctive in a dump. + */ static const char PADDING_CHAR = 0xAA; /** @@ -44,8 +104,7 @@ class SerDes { * * If constructed this way, an internal buffer will be dynamically allocated and grown as needed. * - * The default buffer size is 1500 bytes, based on the assumption that it will be used to construct - * network packets. + * The buffer is SerDes::DEFAULT_SIZE bytes by default, and doubles in size every time the limit is reached. */ SerDes() { _capacity = DEFAULT_SIZE; @@ -58,7 +117,8 @@ class SerDes { * @brief Construct a statically allocated serializer * * If constructed this way, the external buffer will be used to store data. The class will refuse to - * keep adding data if the maximum length is reached, and set the overflow flag. + * keep adding data if the maximum length is reached, write a critical message to the log, and set + * the overflow flag. * * The flag can be read with isOverflow() * @@ -73,6 +133,17 @@ class SerDes { _store = externalStore; } + /** + * @brief Construct a statically allocated serializer + * + * If constructed this way, the external buffer will be used to store data. The class will refuse to + * keep adding data if the maximum length is reached, and set the overflow flag. + * + * The flag can be read with isOverflow() + * + * @param externalStore External data store + * @param storeLength Length of the data store + */ SerDes(uint8_t *externalStore, size_t storeLength) : SerDes((char*)externalStore, storeLength) { } @@ -88,6 +159,15 @@ class SerDes { } } + /** + * @brief Adds padding to the output + * + * The bytes will be set to SerDes::PADDING_CHAR, which is a constant in the source code. + * Since padding isn't supposed to be read, it can be any value and is intended to + * be set to something that can be easily recognized in a dump. + * + * @param bytes Number of bytes to add + */ void addPadding(size_t bytes) { if (!extendBy(bytes)) { return; @@ -98,10 +178,22 @@ class SerDes { _pos += bytes; } + /** + * @brief Add an uint8_t to the output + * + * @param c Character to add + * @return SerDes& This object + */ SerDes &operator<<(uint8_t c) { return *this << int8_t(c); } + /** + * @brief Add an int8_t to the output + * + * @param c Character to add + * @return SerDes& This object + */ SerDes &operator<<(int8_t c) { if (!extendBy(1)) { return *this; @@ -111,10 +203,22 @@ class SerDes { return *this; } + /** + * @brief Read an uint8_t from the buffer + * + * @param c Character to read + * @return SerDes& This object + */ SerDes &operator>>(uint8_t &c) { return *this >> reinterpret_cast(c); } + /** + * @brief Read an int8_t from the buffer + * + * @param c Character to read + * @return SerDes& This object + */ SerDes &operator>>(int8_t &c) { if ( _pos < _length ) { c = _store[_pos++]; @@ -128,10 +232,22 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Add an uint16_t to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(uint16_t val) { return *this << int16_t(val); } + /** + * @brief Add an int16_t to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(int16_t val) { if (!extendBy(sizeof(val))) { return *this; @@ -142,10 +258,22 @@ class SerDes { return *this; } + /** + * @brief Read an uint16_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(uint16_t &val) { return *this >> reinterpret_cast(val); } + /** + * @brief Read an int16_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(int16_t &val) { if ( _pos + sizeof(val) <= _length ) { memcpy((char*)&val, &_store[_pos], sizeof(val)); @@ -160,10 +288,22 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Add an uint32_t to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(uint32_t val) { return *this << int32_t(val); } + /** + * @brief Add an int32_t to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(int32_t val) { if (!extendBy(sizeof(val))) { return *this; @@ -174,10 +314,22 @@ class SerDes { return *this; } + /** + * @brief Read an uint32_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(uint32_t &val) { return *this >> reinterpret_cast(val); } + /** + * @brief Read an int32_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(int32_t &val) { if ( _pos + sizeof(val) <= _length ) { memcpy((char*)&val, &_store[_pos], sizeof(val)); @@ -192,6 +344,12 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Add an glm::vec3 to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(glm::vec3 val) { size_t sz = sizeof(val.x); if (!extendBy(sz*3)) { @@ -206,6 +364,12 @@ class SerDes { return *this; } + /** + * @brief Read a glm::vec3 from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(glm::vec3 &val) { size_t sz = sizeof(val.x); @@ -224,6 +388,12 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Add a glm::vec4 to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(glm::vec4 val) { size_t sz = sizeof(val.x); if (!extendBy(sz*4)) { @@ -239,6 +409,12 @@ class SerDes { return *this; } + /** + * @brief Read a glm::vec4 from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(glm::vec4 &val) { size_t sz = sizeof(val.x); @@ -258,6 +434,12 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Add a glm::ivec2 to the output + * + * @param val Value to add + * @return SerDes& This object + */ SerDes &operator<<(glm::ivec2 val) { size_t sz = sizeof(val.x); if (!extendBy(sz*2)) { @@ -271,6 +453,12 @@ class SerDes { return *this; } + /** + * @brief Read a glm::ivec2 from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ SerDes &operator>>(glm::ivec2 &val) { size_t sz = sizeof(val.x); @@ -287,6 +475,14 @@ class SerDes { } /////////////////////////////////////////////////////////// + /** + * @brief Write a null-terminated string into the buffer + * + * The `\0` at the end of the string is also written. + * + * @param val Value to write + * @return SerDes& This object + */ SerDes &operator<<(const char *val) { size_t len = strlen(val)+1; extendBy(len); @@ -295,6 +491,14 @@ class SerDes { return *this; } + /** + * @brief Write a QString into the buffer + * + * The string is encoded in UTF-8 and the `\0` at the end of the string is also written. + * + * @param val Value to write + * @return SerDes& This object + */ SerDes &operator<<(const QString &val) { return *this << val.toUtf8().constData(); } @@ -302,6 +506,17 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Pointer to the start of the internal buffer. + * + * The allocated amount can be found with capacity(). + * + * The end of the stored data can be found with length(). + * + * @return Pointer to buffer + */ + char *buffer() const { return _store; } + /** * @brief Current position in the buffer. Starts at 0. * @@ -351,6 +566,16 @@ class SerDes { */ void rewind() { _pos = 0; _overflow = false; } + /** + * @brief Dump the contents of this object into QDebug + * + * This produces a dump of the internal state, and an ASCII/hex dump of + * the contents, for debugging. + * + * @param debug Qt QDebug stream + * @param ds This object + * @return QDebug + */ friend QDebug operator<<(QDebug debug, const SerDes &ds); private: From 194eebf57c4c475752a962884f660ea18c42351d Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sun, 12 Jun 2022 13:48:08 +0200 Subject: [PATCH 06/19] Finish Texture_ktx conversion to serializer --- libraries/gpu/src/gpu/Texture.h | 27 ++++++++ libraries/gpu/src/gpu/Texture_ktx.cpp | 92 +++++---------------------- libraries/shared/src/SerDes.h | 36 +++++++++++ tests/shared/src/SerializerTests.cpp | 8 +++ 4 files changed, 88 insertions(+), 75 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture.h b/libraries/gpu/src/gpu/Texture.h index 5143cc1c23..2e7f8c62ca 100644 --- a/libraries/gpu/src/gpu/Texture.h +++ b/libraries/gpu/src/gpu/Texture.h @@ -92,6 +92,33 @@ public: }; typedef std::shared_ptr< SphericalHarmonics > SHPointer; + +inline SerDes &operator<<(SerDes &ser, const SphericalHarmonics &h) { + ser << h.L00 << h.spare0; + ser << h.L1m1 << h.spare1; + ser << h.L10 << h.spare2; + ser << h.L11 << h.spare3; + ser << h.L2m2 << h.spare4; + ser << h.L2m1 << h.spare5; + ser << h.L20 << h.spare6; + ser << h.L21 << h.spare7; + ser << h.L22 << h.spare8; + return ser; +} + +inline SerDes &operator>>(SerDes &des, SphericalHarmonics &h) { + des >> h.L00 >> h.spare0; + des >> h.L1m1 >> h.spare1; + des >> h.L10 >> h.spare2; + des >> h.L11 >> h.spare3; + des >> h.L2m2 >> h.spare4; + des >> h.L2m1 >> h.spare5; + des >> h.L20 >> h.spare6; + des >> h.L21 >> h.spare7; + des >> h.L22 >> h.spare8; + return des; +} + class Sampler { public: diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index 748ec1a702..dc3d23948d 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -43,7 +43,7 @@ struct GPUKTXPayload { TextureUsageType _usageType; glm::ivec2 _originalSize { 0, 0 }; - void serialize2(SerDes &ser) { + void serialize(SerDes &ser) { ser << CURRENT_VERSION; ser << _samplerDesc; @@ -55,7 +55,7 @@ struct GPUKTXPayload { ser.addPadding(PADDING); } - bool unserialize2(SerDes &dsr) { + bool unserialize(SerDes &dsr) { Version version = 0; uint32 usageData; uint8_t usagetype = 0; @@ -80,60 +80,6 @@ struct GPUKTXPayload { return true; } - - Byte* serialize(Byte* data) const { - *(Version*)data = CURRENT_VERSION; - data += sizeof(Version); - - memcpy(data, &_samplerDesc, sizeof(Sampler::Desc)); - data += sizeof(Sampler::Desc); - - // We can't copy the bitset in Texture::Usage in a crossplateform manner - // So serialize it manually - uint32 usageData = _usage._flags.to_ulong(); - memcpy(data, &usageData, sizeof(uint32)); - data += sizeof(uint32); - - memcpy(data, &_usageType, sizeof(TextureUsageType)); - data += sizeof(TextureUsageType); - - memcpy(data, glm::value_ptr(_originalSize), sizeof(glm::ivec2)); - data += sizeof(glm::ivec2); - - return data + PADDING; - } - - bool unserialize(const Byte* data, size_t size) { - Version version = *(const Version*)data; - data += sizeof(Version); - - if (version > CURRENT_VERSION) { - // If we try to load a version that we don't know how to parse, - // it will render incorrectly - return false; - } - - memcpy(&_samplerDesc, data, sizeof(Sampler::Desc)); - data += sizeof(Sampler::Desc); - - // We can't copy the bitset in Texture::Usage in a crossplateform manner - // So unserialize it manually - uint32 usageData; - memcpy(&usageData, data, sizeof(uint32)); - _usage = Texture::Usage(usageData); - data += sizeof(uint32); - - memcpy(&_usageType, data, sizeof(TextureUsageType)); - data += sizeof(TextureUsageType); - - if (version >= 2) { - memcpy(&_originalSize, data, sizeof(glm::ivec2)); - data += sizeof(glm::ivec2); - } - - return true; - } - static bool isGPUKTX(const ktx::KeyValue& val) { return (val._key.compare(KEY) == 0); } @@ -143,8 +89,7 @@ struct GPUKTXPayload { if (found != keyValues.end()) { auto value = found->_value; SerDes dsr(value.data(), value.size()); - return payload.unserialize2(dsr); - //return payload.unserialize(value.data(), value.size()); + return payload.unserialize(dsr); } return false; } @@ -164,29 +109,24 @@ struct IrradianceKTXPayload { SphericalHarmonics _irradianceSH; - Byte* serialize(Byte* data) const { - *(Version*)data = CURRENT_VERSION; - data += sizeof(Version); - - memcpy(data, &_irradianceSH, sizeof(SphericalHarmonics)); - data += sizeof(SphericalHarmonics); - - return data + PADDING; + void serialize(SerDes &ser) const { + ser << CURRENT_VERSION; + ser << _irradianceSH; + ser.addPadding(PADDING); } - bool unserialize(const Byte* data, size_t size) { - if (size != SIZE) { + bool unserialize(SerDes &des) { + Version version; + if (des.length() != SIZE) { return false; } - Version version = *(const Version*)data; + des >> version; if (version != CURRENT_VERSION) { return false; } - data += sizeof(Version); - - memcpy(&_irradianceSH, data, sizeof(SphericalHarmonics)); + des >> _irradianceSH; return true; } @@ -198,7 +138,8 @@ struct IrradianceKTXPayload { auto found = std::find_if(keyValues.begin(), keyValues.end(), isIrradianceKTX); if (found != keyValues.end()) { auto value = found->_value; - return payload.unserialize(value.data(), value.size()); + SerDes des(value.data(), value.size()); + return payload.unserialize(des); } return false; } @@ -510,7 +451,7 @@ ktx::KTXUniquePointer Texture::serialize(const Texture& texture, const glm::ivec Byte keyvalPayload[GPUKTXPayload::SIZE]; SerDes ser(keyvalPayload, sizeof(keyvalPayload)); - gpuKeyval.serialize2(ser); + gpuKeyval.serialize(ser); ktx::KeyValues keyValues; keyValues.emplace_back(GPUKTXPayload::KEY, (uint32)GPUKTXPayload::SIZE, (ktx::Byte*) &keyvalPayload); @@ -520,7 +461,8 @@ ktx::KTXUniquePointer Texture::serialize(const Texture& texture, const glm::ivec irradianceKeyval._irradianceSH = *texture.getIrradiance(); Byte irradianceKeyvalPayload[IrradianceKTXPayload::SIZE]; - irradianceKeyval.serialize(irradianceKeyvalPayload); + SerDes ser(irradianceKeyvalPayload, sizeof(irradianceKeyvalPayload)); + irradianceKeyval.serialize(ser); keyValues.emplace_back(IrradianceKTXPayload::KEY, (uint32)IrradianceKTXPayload::SIZE, (ktx::Byte*) &irradianceKeyvalPayload); } diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index fe445fe854..356bb45522 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -344,6 +344,42 @@ class SerDes { /////////////////////////////////////////////////////////// + /** + * @brief Add an float to the output + * + * @param val Value to add + * @return SerDes& This object + */ + SerDes &operator<<(float val) { + if (!extendBy(sizeof(val))) { + return *this; + } + + memcpy(&_store[_pos], (char*)&val, sizeof(val)); + _pos += sizeof(val); + return *this; + } + + /** + * @brief Read an float from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + SerDes &operator>>(float &val) { + if ( _pos + sizeof(val) <= _length ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + } else { + _overflow = true; + qCritical() << "Deserializer trying to read past end of input, reading float from position " << _pos << ", length " << _length; + } + return *this; + } + + /////////////////////////////////////////////////////////// + + /** * @brief Add an glm::vec3 to the output * diff --git a/tests/shared/src/SerializerTests.cpp b/tests/shared/src/SerializerTests.cpp index a1b3a37544..89598a3c2a 100644 --- a/tests/shared/src/SerializerTests.cpp +++ b/tests/shared/src/SerializerTests.cpp @@ -54,6 +54,9 @@ void SerializerTests::testAdd() { s << v; QCOMPARE(s.length(), 40); + s << 1.2345f; + QCOMPARE(s.length(), 44); + qDebug() << s; } @@ -66,6 +69,8 @@ void SerializerTests::testAddAndRead() { glm::vec4 v4_b; glm::ivec2 iv2_a{10, 24}; glm::ivec2 iv2_b; + float f_a = 1.2345f; + float f_b; s << (qint8)1; s << (qint16)0xaabb; @@ -73,6 +78,7 @@ void SerializerTests::testAddAndRead() { s << v3_a; s << v4_a; s << iv2_a; + s << f_a; qint8 i8; qint16 i16; @@ -86,6 +92,7 @@ void SerializerTests::testAddAndRead() { s >> v3_b; s >> v4_b; s >> iv2_b; + s >> f_b; qDebug() << s; @@ -95,6 +102,7 @@ void SerializerTests::testAddAndRead() { QCOMPARE(v3_a, v3_b); QCOMPARE(v4_a, v4_b); QCOMPARE(iv2_a, iv2_b); + QCOMPARE(f_a, f_b); } void SerializerTests::testReadPastEnd() { From 189f91b05d6f5b9241cc4cef6c4613c36805b64f Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Thu, 23 Jun 2022 21:24:28 +0200 Subject: [PATCH 07/19] Clear blendshapes without memset Avoids warning due to glm::vec3 not being trivially copyable --- libraries/render-utils/src/Model.cpp | 4 +++- libraries/shared/src/BlendshapeConstants.h | 19 +++++++++++++++++++ 2 files changed, 22 insertions(+), 1 deletion(-) diff --git a/libraries/render-utils/src/Model.cpp b/libraries/render-utils/src/Model.cpp index 69a593ed09..6ad2534038 100644 --- a/libraries/render-utils/src/Model.cpp +++ b/libraries/render-utils/src/Model.cpp @@ -1949,7 +1949,9 @@ void Blender::run() { blendedMeshSizes.push_back(numVertsInMesh); // initialize offsets to zero - memset(unpackedBlendshapeOffsets.data(), 0, numVertsInMesh * sizeof(BlendshapeOffsetUnpacked)); + for(BlendshapeOffsetUnpacked &bou : unpackedBlendshapeOffsets) { + bou.clear(); + } // for each blendshape in this mesh, accumulate the offsets into unpackedBlendshapeOffsets. const float NORMAL_COEFFICIENT_SCALE = 0.01f; diff --git a/libraries/shared/src/BlendshapeConstants.h b/libraries/shared/src/BlendshapeConstants.h index 596e7df4ee..b741059146 100644 --- a/libraries/shared/src/BlendshapeConstants.h +++ b/libraries/shared/src/BlendshapeConstants.h @@ -122,6 +122,25 @@ struct BlendshapeOffsetUnpacked { float positionOffsetX, positionOffsetY, positionOffsetZ; float normalOffsetX, normalOffsetY, normalOffsetZ; float tangentOffsetX, tangentOffsetY, tangentOffsetZ; + + /** + * @brief Set all components of all the offsets to zero + * + * @note glm::vec3 is not trivially copyable, so it's not correct to clear it with memset. + */ + void clear() { + positionOffsetX = 0.0f; + positionOffsetY = 0.0f; + positionOffsetZ = 0.0f; + + normalOffsetX = 0.0f; + normalOffsetY = 0.0f; + normalOffsetZ = 0.0f; + + tangentOffsetX = 0.0f; + tangentOffsetY = 0.0f; + tangentOffsetZ = 0.0f; + } }; using BlendshapeOffset = BlendshapeOffsetPacked; From 201c531edbff95d9235595c891cf13b655b6e25b Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Thu, 23 Jun 2022 21:25:07 +0200 Subject: [PATCH 08/19] Split SerDes into DataSerializer and DataDeserializer Single class wasn't working well because deserialization may need to be done on const data. With the split, the deserializer part can work with const data without issues. Also cleaned things up a bit. --- libraries/gpu/src/gpu/Texture.h | 10 +- libraries/gpu/src/gpu/Texture_ktx.cpp | 16 +- libraries/octree/src/OctreePacketData.cpp | 9 +- libraries/shared/src/AACube.h | 16 + libraries/shared/src/SerDes.cpp | 8 +- libraries/shared/src/SerDes.h | 622 +++++++++++++--------- 6 files changed, 421 insertions(+), 260 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture.h b/libraries/gpu/src/gpu/Texture.h index 2e7f8c62ca..8370462c5a 100644 --- a/libraries/gpu/src/gpu/Texture.h +++ b/libraries/gpu/src/gpu/Texture.h @@ -93,7 +93,7 @@ public: typedef std::shared_ptr< SphericalHarmonics > SHPointer; -inline SerDes &operator<<(SerDes &ser, const SphericalHarmonics &h) { +inline DataSerializer &operator<<(DataSerializer &ser, const SphericalHarmonics &h) { ser << h.L00 << h.spare0; ser << h.L1m1 << h.spare1; ser << h.L10 << h.spare2; @@ -106,7 +106,7 @@ inline SerDes &operator<<(SerDes &ser, const SphericalHarmonics &h) { return ser; } -inline SerDes &operator>>(SerDes &des, SphericalHarmonics &h) { +inline DataDeserializer &operator>>(DataDeserializer &des, SphericalHarmonics &h) { des >> h.L00 >> h.spare0; des >> h.L1m1 >> h.spare1; des >> h.L10 >> h.spare2; @@ -185,7 +185,7 @@ public: _maxMip == other._maxMip; } - SerDes &operator<<(SerDes &dsd) { + DataSerializer &operator<<(DataSerializer &dsd) { dsd << _borderColor; dsd << _maxAnisotropy; dsd << _filter; @@ -237,7 +237,7 @@ protected: friend class Deserializer; }; -inline SerDes &operator<<(SerDes &ser, const Sampler::Desc &d) { +inline DataSerializer &operator<<(DataSerializer &ser, const Sampler::Desc &d) { ser << d._borderColor; ser << d._maxAnisotropy; ser << d._filter; @@ -251,7 +251,7 @@ inline SerDes &operator<<(SerDes &ser, const Sampler::Desc &d) { return ser; } -inline SerDes &operator>>(SerDes &dsr, Sampler::Desc &d) { +inline DataDeserializer &operator>>(DataDeserializer &dsr, Sampler::Desc &d) { dsr >> d._borderColor; dsr >> d._maxAnisotropy; dsr >> d._filter; diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index dc3d23948d..6845aeb55b 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -43,7 +43,7 @@ struct GPUKTXPayload { TextureUsageType _usageType; glm::ivec2 _originalSize { 0, 0 }; - void serialize(SerDes &ser) { + void serialize(DataSerializer &ser) { ser << CURRENT_VERSION; ser << _samplerDesc; @@ -55,7 +55,7 @@ struct GPUKTXPayload { ser.addPadding(PADDING); } - bool unserialize(SerDes &dsr) { + bool unserialize(DataDeserializer &dsr) { Version version = 0; uint32 usageData; uint8_t usagetype = 0; @@ -88,7 +88,7 @@ struct GPUKTXPayload { auto found = std::find_if(keyValues.begin(), keyValues.end(), isGPUKTX); if (found != keyValues.end()) { auto value = found->_value; - SerDes dsr(value.data(), value.size()); + DataDeserializer dsr(value.data(), value.size()); return payload.unserialize(dsr); } return false; @@ -109,13 +109,13 @@ struct IrradianceKTXPayload { SphericalHarmonics _irradianceSH; - void serialize(SerDes &ser) const { + void serialize(DataSerializer &ser) const { ser << CURRENT_VERSION; ser << _irradianceSH; ser.addPadding(PADDING); } - bool unserialize(SerDes &des) { + bool unserialize(DataDeserializer &des) { Version version; if (des.length() != SIZE) { return false; @@ -138,7 +138,7 @@ struct IrradianceKTXPayload { auto found = std::find_if(keyValues.begin(), keyValues.end(), isIrradianceKTX); if (found != keyValues.end()) { auto value = found->_value; - SerDes des(value.data(), value.size()); + DataDeserializer des(value.data(), value.size()); return payload.unserialize(des); } return false; @@ -449,7 +449,7 @@ ktx::KTXUniquePointer Texture::serialize(const Texture& texture, const glm::ivec gpuKeyval._originalSize = originalSize; Byte keyvalPayload[GPUKTXPayload::SIZE]; - SerDes ser(keyvalPayload, sizeof(keyvalPayload)); + DataSerializer ser(keyvalPayload, sizeof(keyvalPayload)); gpuKeyval.serialize(ser); @@ -461,7 +461,7 @@ ktx::KTXUniquePointer Texture::serialize(const Texture& texture, const glm::ivec irradianceKeyval._irradianceSH = *texture.getIrradiance(); Byte irradianceKeyvalPayload[IrradianceKTXPayload::SIZE]; - SerDes ser(irradianceKeyvalPayload, sizeof(irradianceKeyvalPayload)); + DataSerializer ser(irradianceKeyvalPayload, sizeof(irradianceKeyvalPayload)); irradianceKeyval.serialize(ser); keyValues.emplace_back(IrradianceKTXPayload::KEY, (uint32)IrradianceKTXPayload::SIZE, (ktx::Byte*) &irradianceKeyvalPayload); diff --git a/libraries/octree/src/OctreePacketData.cpp b/libraries/octree/src/OctreePacketData.cpp index c13d58226b..3745582728 100644 --- a/libraries/octree/src/OctreePacketData.cpp +++ b/libraries/octree/src/OctreePacketData.cpp @@ -17,6 +17,7 @@ #include "OctreeLogging.h" #include "NumericalConstants.h" #include +#include "SerDes.h" bool OctreePacketData::_debug = false; AtomicUIntStat OctreePacketData::_totalBytesOfOctalCodes { 0 }; @@ -847,10 +848,10 @@ int OctreePacketData::unpackDataFromBytes(const unsigned char* dataBytes, QByteA } int OctreePacketData::unpackDataFromBytes(const unsigned char* dataBytes, AACube& result) { - aaCubeData cube; - memcpy(&cube, dataBytes, sizeof(aaCubeData)); - result = AACube(cube.corner, cube.scale); - return sizeof(aaCubeData); + DataDeserializer des(dataBytes, sizeof(aaCubeData)); + des >> result; + + return des.length(); } int OctreePacketData::unpackDataFromBytes(const unsigned char* dataBytes, QRect& result) { diff --git a/libraries/shared/src/AACube.h b/libraries/shared/src/AACube.h index 66b29e3185..27c424cadb 100644 --- a/libraries/shared/src/AACube.h +++ b/libraries/shared/src/AACube.h @@ -20,6 +20,7 @@ #include #include "BoxBase.h" +#include "SerDes.h" class AABox; class Extents; @@ -80,6 +81,10 @@ private: glm::vec3 _corner; float _scale; + + friend DataSerializer& operator<<(DataSerializer &ser, const AACube &cube); + friend DataDeserializer& operator>>(DataDeserializer &des, AACube &cube); + }; inline bool operator==(const AACube& a, const AACube& b) { @@ -99,5 +104,16 @@ inline QDebug operator<<(QDebug debug, const AACube& cube) { return debug; } +inline DataSerializer& operator<<(DataSerializer &ser, const AACube &cube) { + ser << cube._corner; + ser << cube._scale; + return ser; +} + +inline DataDeserializer& operator>>(DataDeserializer &des, AACube &cube) { + des >> cube._corner; + des >> cube._scale; + return des; +} #endif // hifi_AACube_h diff --git a/libraries/shared/src/SerDes.cpp b/libraries/shared/src/SerDes.cpp index c64430f201..044ea1fad2 100644 --- a/libraries/shared/src/SerDes.cpp +++ b/libraries/shared/src/SerDes.cpp @@ -12,10 +12,10 @@ #include #include "SerDes.h" -const int SerDes::DEFAULT_SIZE; -const char SerDes::PADDING_CHAR; +const int DataSerializer::DEFAULT_SIZE; +const char DataSerializer::PADDING_CHAR; -QDebug operator<<(QDebug debug, const SerDes &ds) { +QDebug operator<<(QDebug debug, const DataSerializer &ds) { debug << "{ capacity =" << ds.capacity() << "; length = " << ds.length() << "; pos = " << ds.pos() << "}"; debug << "\n"; @@ -54,7 +54,7 @@ QDebug operator<<(QDebug debug, const SerDes &ds) { } -void SerDes::changeAllocation(size_t new_size) { +void DataSerializer::changeAllocation(size_t new_size) { while ( _capacity < new_size) { _capacity *= 2; } diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index 356bb45522..ee38a87aa8 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -17,7 +17,7 @@ #include /** - * @brief Data serializer/deserializer + * @brief Data serializer * * When encoding, this class takes in data and encodes it into a buffer. No attempt is made to store version numbers, lengths, * or any other metadata. It's entirely up to the user to use the class in such a way that the process can be @@ -37,7 +37,7 @@ * uint16_t count = 1; * glm::vec3 pos{1.5, 2.0, 9.0}; * - * SerDes ser; + * Serializer ser; * ser << version; * ser << count; * ser << pos; @@ -45,44 +45,20 @@ * // Serialized data is in ser.buffer(), ser.length() long. * @endcode * - * Example of decoding: - * - * @code {.cpp} - * // Incoming data has been placed in: - * // char buffer[1024]; - * - * uint8_t version; - * uint16_t count; - * glm::vec3 pos; - * - * SerDes des(buffer, sizeof(buffer)); - * des >> version; - * des >> count; - * des >> pos; - * @endcode - * * This object should be modified directly to add support for any primitive and common datatypes in the code. To support serializing/deserializing * classes and structures, implement a `operator<<` and `operator>>` functions for that object, eg: * * @code {.cpp} - * SerDes &operator<<(SerDes &ser, const Object &o) { + * Serializer &operator<<(Serializer &ser, const Object &o) { * ser << o._borderColor; * ser << o._maxAnisotropy; * ser << o._filter; * return ser; * } - * - * SerDes &operator>>(SerDes &des, Object &o) { - * des >> o._borderColor; - * des >> o._maxAnisotropy; - * des >> o._filter; - * return des; - * } - * * @endcode * */ -class SerDes { +class DataSerializer { public: /** * @brief Default size for a dynamically allocated buffer. @@ -106,7 +82,7 @@ class SerDes { * * The buffer is SerDes::DEFAULT_SIZE bytes by default, and doubles in size every time the limit is reached. */ - SerDes() { + DataSerializer() { _capacity = DEFAULT_SIZE; _pos = 0; _length = 0; @@ -125,7 +101,7 @@ class SerDes { * @param externalStore External data store * @param storeLength Length of the data store */ - SerDes(char *externalStore, size_t storeLength) { + DataSerializer(char *externalStore, size_t storeLength) { _capacity = storeLength; _length = storeLength; _pos = 0; @@ -144,16 +120,16 @@ class SerDes { * @param externalStore External data store * @param storeLength Length of the data store */ - SerDes(uint8_t *externalStore, size_t storeLength) : SerDes((char*)externalStore, storeLength) { + DataSerializer(uint8_t *externalStore, size_t storeLength) : DataSerializer((char*)externalStore, storeLength) { } - SerDes(const SerDes &) = delete; - SerDes &operator=(const SerDes &) = delete; + DataSerializer(const DataSerializer &) = delete; + DataSerializer &operator=(const DataSerializer &) = delete; - ~SerDes() { + ~DataSerializer() { if (!_storeIsExternal) { delete[] _store; } @@ -169,7 +145,7 @@ class SerDes { * @param bytes Number of bytes to add */ void addPadding(size_t bytes) { - if (!extendBy(bytes)) { + if (!extendBy(bytes, "padding")) { return; } @@ -184,7 +160,7 @@ class SerDes { * @param c Character to add * @return SerDes& This object */ - SerDes &operator<<(uint8_t c) { + DataSerializer &operator<<(uint8_t c) { return *this << int8_t(c); } @@ -194,8 +170,8 @@ class SerDes { * @param c Character to add * @return SerDes& This object */ - SerDes &operator<<(int8_t c) { - if (!extendBy(1)) { + DataSerializer &operator<<(int8_t c) { + if (!extendBy(1, "int8_t")) { return *this; } @@ -203,32 +179,6 @@ class SerDes { return *this; } - /** - * @brief Read an uint8_t from the buffer - * - * @param c Character to read - * @return SerDes& This object - */ - SerDes &operator>>(uint8_t &c) { - return *this >> reinterpret_cast(c); - } - - /** - * @brief Read an int8_t from the buffer - * - * @param c Character to read - * @return SerDes& This object - */ - SerDes &operator>>(int8_t &c) { - if ( _pos < _length ) { - c = _store[_pos++]; - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading 8 bits from position " << _pos << ", length " << _length; - } - - return *this; - } /////////////////////////////////////////////////////////// @@ -238,7 +188,7 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(uint16_t val) { + DataSerializer &operator<<(uint16_t val) { return *this << int16_t(val); } @@ -248,8 +198,8 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(int16_t val) { - if (!extendBy(sizeof(val))) { + DataSerializer &operator<<(int16_t val) { + if (!extendBy(sizeof(val), "int16_t")) { return *this; } @@ -258,33 +208,7 @@ class SerDes { return *this; } - /** - * @brief Read an uint16_t from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(uint16_t &val) { - return *this >> reinterpret_cast(val); - } - /** - * @brief Read an int16_t from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(int16_t &val) { - if ( _pos + sizeof(val) <= _length ) { - memcpy((char*)&val, &_store[_pos], sizeof(val)); - _pos += sizeof(val); - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading 16 bits from position " << _pos << ", length " << _length; - } - - return *this; - } /////////////////////////////////////////////////////////// @@ -294,7 +218,7 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(uint32_t val) { + DataSerializer &operator<<(uint32_t val) { return *this << int32_t(val); } @@ -304,8 +228,8 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(int32_t val) { - if (!extendBy(sizeof(val))) { + DataSerializer &operator<<(int32_t val) { + if (!extendBy(sizeof(val), "int32_t")) { return *this; } @@ -314,33 +238,6 @@ class SerDes { return *this; } - /** - * @brief Read an uint32_t from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(uint32_t &val) { - return *this >> reinterpret_cast(val); - } - - /** - * @brief Read an int32_t from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(int32_t &val) { - if ( _pos + sizeof(val) <= _length ) { - memcpy((char*)&val, &_store[_pos], sizeof(val)); - _pos += sizeof(val); - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading 32 bits from position " << _pos << ", length " << _length; - } - return *this; - } - /////////////////////////////////////////////////////////// @@ -350,33 +247,15 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(float val) { - if (!extendBy(sizeof(val))) { - return *this; - } - - memcpy(&_store[_pos], (char*)&val, sizeof(val)); - _pos += sizeof(val); - return *this; - } - - /** - * @brief Read an float from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(float &val) { - if ( _pos + sizeof(val) <= _length ) { - memcpy((char*)&val, &_store[_pos], sizeof(val)); + DataSerializer &operator<<(float val) { + if (extendBy(sizeof(val), "float")) { + memcpy(&_store[_pos], (char*)&val, sizeof(val)); _pos += sizeof(val); - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading float from position " << _pos << ", length " << _length; } return *this; } + /////////////////////////////////////////////////////////// @@ -386,38 +265,14 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(glm::vec3 val) { + DataSerializer &operator<<(glm::vec3 val) { size_t sz = sizeof(val.x); - if (!extendBy(sz*3)) { - return *this; - } - - memcpy(&_store[_pos ], (char*)&val.x, sz); - memcpy(&_store[_pos + sz ], (char*)&val.y, sz); - memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); - - _pos += sz*3; - return *this; - } - - /** - * @brief Read a glm::vec3 from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(glm::vec3 &val) { - size_t sz = sizeof(val.x); - - if ( _pos + sz*3 <= _length ) { - memcpy((char*)&val.x, &_store[_pos ], sz); - memcpy((char*)&val.y, &_store[_pos + sz ], sz); - memcpy((char*)&val.z, &_store[_pos + sz*2], sz); + if (extendBy(sz*3, "glm::vec3")) { + memcpy(&_store[_pos ], (char*)&val.x, sz); + memcpy(&_store[_pos + sz ], (char*)&val.y, sz); + memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); _pos += sz*3; - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading glm::vec3 from position " << _pos << ", length " << _length; } return *this; } @@ -430,40 +285,15 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(glm::vec4 val) { + DataSerializer &operator<<(glm::vec4 val) { size_t sz = sizeof(val.x); - if (!extendBy(sz*4)) { - return *this; - } - - memcpy(&_store[_pos ], (char*)&val.x, sz); - memcpy(&_store[_pos + sz ], (char*)&val.y, sz); - memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); - memcpy(&_store[_pos + sz*3], (char*)&val.w, sz); - - _pos += sz*4; - return *this; - } - - /** - * @brief Read a glm::vec4 from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(glm::vec4 &val) { - size_t sz = sizeof(val.x); - - if ( _pos + sz*4 <= _length ) { - memcpy((char*)&val.x, &_store[_pos ], sz); - memcpy((char*)&val.y, &_store[_pos + sz ], sz); - memcpy((char*)&val.z, &_store[_pos + sz*2], sz); - memcpy((char*)&val.w, &_store[_pos + sz*3], sz); + if (extendBy(sz*4, "glm::vec4")) { + memcpy(&_store[_pos ], (char*)&val.x, sz); + memcpy(&_store[_pos + sz ], (char*)&val.y, sz); + memcpy(&_store[_pos + sz*2], (char*)&val.z, sz); + memcpy(&_store[_pos + sz*3], (char*)&val.w, sz); _pos += sz*4; - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading glm::vec3 from position " << _pos << ", length " << _length; } return *this; } @@ -476,39 +306,18 @@ class SerDes { * @param val Value to add * @return SerDes& This object */ - SerDes &operator<<(glm::ivec2 val) { + DataSerializer &operator<<(glm::ivec2 val) { size_t sz = sizeof(val.x); - if (!extendBy(sz*2)) { - return *this; - } - - memcpy(&_store[_pos ], (char*)&val.x, sz); - memcpy(&_store[_pos + sz ], (char*)&val.y, sz); - - _pos += sz*2; - return *this; - } - - /** - * @brief Read a glm::ivec2 from the buffer - * - * @param val Value to read - * @return SerDes& This object - */ - SerDes &operator>>(glm::ivec2 &val) { - size_t sz = sizeof(val.x); - - if ( _pos + sz*2 <= _length ) { - memcpy((char*)&val.x, &_store[_pos ], sz); - memcpy((char*)&val.y, &_store[_pos + sz ], sz); + if (extendBy(sz*2, "glm::ivec2")) { + memcpy(&_store[_pos ], (char*)&val.x, sz); + memcpy(&_store[_pos + sz ], (char*)&val.y, sz); _pos += sz*2; - } else { - _overflow = true; - qCritical() << "Deserializer trying to read past end of input, reading glm::ivec2 from position " << _pos << ", length " << _length; } return *this; } + + /////////////////////////////////////////////////////////// /** @@ -519,11 +328,12 @@ class SerDes { * @param val Value to write * @return SerDes& This object */ - SerDes &operator<<(const char *val) { + DataSerializer &operator<<(const char *val) { size_t len = strlen(val)+1; - extendBy(len); - memcpy(&_store[_pos], val, len); - _pos += len; + if (extendBy(len, "string")) { + memcpy(&_store[_pos], val, len); + _pos += len; + } return *this; } @@ -535,7 +345,7 @@ class SerDes { * @param val Value to write * @return SerDes& This object */ - SerDes &operator<<(const QString &val) { + DataSerializer &operator<<(const QString &val) { return *this << val.toUtf8().constData(); } @@ -612,14 +422,15 @@ class SerDes { * @param ds This object * @return QDebug */ - friend QDebug operator<<(QDebug debug, const SerDes &ds); + friend QDebug operator<<(QDebug debug, const DataSerializer &ds); private: - bool extendBy(size_t bytes) { + bool extendBy(size_t bytes, const QString &type_name) { //qDebug() << "Extend by" << bytes; if ( _capacity < _length + bytes) { if ( _storeIsExternal ) { + qCritical() << "Serializer trying to write past end of input, writing" << bytes << "bytes for" << type_name << " from position " << _pos << ", length " << _length; _overflow = true; return false; } @@ -641,3 +452,336 @@ class SerDes { size_t _length = 0; size_t _pos = 0; }; + +/** + * @brief Data deserializer + * + * This class operates on a fixed size buffer. If an attempt to read past the end is made, the read fails, + * and the overflow flag is set. + * + * The class was written for the maximum simplicity possible and inline friendliness. + * + * Example of decoding: + * + * @code {.cpp} + * // Incoming data has been placed in: + * // char buffer[1024]; + * + * uint8_t version; + * uint16_t count; + * glm::vec3 pos; + * + * Deserializer des(buffer, sizeof(buffer)); + * des >> version; + * des >> count; + * des >> pos; + * @endcode + * + * This object should be modified directly to add support for any primitive and common datatypes in the code. To support deserializing + * classes and structures, implement an `operator>>` function for that object, eg: + * + * @code {.cpp} + * Deserializer &operator>>(Deserializer &des, Object &o) { + * des >> o._borderColor; + * des >> o._maxAnisotropy; + * des >> o._filter; + * return des; + * } + * @endcode + * + */ +class DataDeserializer { + public: + /** + * @brief Construct a Deserializer + * * + * @param externalStore External data store + * @param storeLength Length of the data store + */ + DataDeserializer(const char *externalStore, size_t storeLength) { + _length = storeLength; + _pos = 0; + _store = externalStore; + } + + /** + * @brief Construct a Deserializer + * + * @param externalStore External data store + * @param storeLength Length of the data store + */ + DataDeserializer(const uint8_t *externalStore, size_t storeLength) : DataDeserializer((const char*)externalStore, storeLength) { + + } + + /** + * @brief Construct a new Deserializer reading data from a Serializer + * + * This is a convenience function for testing. + * + * @param serializer Serializer with data + */ + DataDeserializer(const DataSerializer &serializer) : DataDeserializer(serializer.buffer(), serializer.length()) { + + } + + /** + * @brief Skips padding in the input + * + * @param bytes Number of bytes to skip + */ + void skipPadding(size_t bytes) { + if (!canAdvanceBy(bytes, "padding")) { + return; + } + + _pos += bytes; + } + + + /** + * @brief Read an uint8_t from the buffer + * + * @param c Character to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(uint8_t &c) { + return *this >> reinterpret_cast(c); + } + + /** + * @brief Read an int8_t from the buffer + * + * @param c Character to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(int8_t &c) { + if ( canAdvanceBy(1, "int8_t") ) { + c = _store[_pos++]; + } + + return *this; + } + + /////////////////////////////////////////////////////////// + + /** + * @brief Read an uint16_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(uint16_t &val) { + return *this >> reinterpret_cast(val); + } + + /** + * @brief Read an int16_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(int16_t &val) { + if ( canAdvanceBy(sizeof(val), "int16_t") ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + } + + return *this; + } + + /////////////////////////////////////////////////////////// + + /** + * @brief Read an uint32_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(uint32_t &val) { + return *this >> reinterpret_cast(val); + } + + /** + * @brief Read an int32_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(int32_t &val) { + if ( canAdvanceBy(sizeof(val), "int32_t") ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + } + return *this; + } + + + /////////////////////////////////////////////////////////// + + + /** + * @brief Read an float from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(float &val) { + if ( canAdvanceBy(sizeof(val), "float") ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + } + return *this; + } + + /////////////////////////////////////////////////////////// + + + + + /** + * @brief Read a glm::vec3 from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(glm::vec3 &val) { + size_t sz = sizeof(val.x); + + if ( canAdvanceBy(sz*3, "glm::vec3") ) { + memcpy((char*)&val.x, &_store[_pos ], sz); + memcpy((char*)&val.y, &_store[_pos + sz ], sz); + memcpy((char*)&val.z, &_store[_pos + sz*2], sz); + + _pos += sz*3; + } + + return *this; + } + + /////////////////////////////////////////////////////////// + + + /** + * @brief Read a glm::vec4 from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(glm::vec4 &val) { + size_t sz = sizeof(val.x); + + if ( canAdvanceBy(sz*4, "glm::vec4")) { + memcpy((char*)&val.x, &_store[_pos ], sz); + memcpy((char*)&val.y, &_store[_pos + sz ], sz); + memcpy((char*)&val.z, &_store[_pos + sz*2], sz); + memcpy((char*)&val.w, &_store[_pos + sz*3], sz); + + _pos += sz*4; + } + return *this; + } + + /////////////////////////////////////////////////////////// + + + /** + * @brief Read a glm::ivec2 from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(glm::ivec2 &val) { + size_t sz = sizeof(val.x); + + if ( canAdvanceBy(sz*2, "glm::ivec2") ) { + memcpy((char*)&val.x, &_store[_pos ], sz); + memcpy((char*)&val.y, &_store[_pos + sz ], sz); + + _pos += sz*2; + } + + return *this; + } + + /////////////////////////////////////////////////////////// + + /** + * @brief Pointer to the start of the internal buffer. + * + * The allocated amount can be found with capacity(). + * + * The end of the stored data can be found with length(). + * + * @return Pointer to buffer + */ + const char *buffer() const { return _store; } + + /** + * @brief Current position in the buffer. Starts at 0. + * + * @return size_t + */ + size_t pos() const { return _pos; } + + /** + * @brief Last position that was written to in the buffer. Starts at 0. + * + * @return size_t + */ + size_t length() const { return _length; } + + /** + * @brief Whether there's any data in the buffer + * + * @return true Something has been written + * @return false The buffer is empty + */ + bool isEmpty() const { return _length == 0; } + + /** + * @brief The buffer size limit has been reached + * + * This can only return true for a statically allocated buffer. + * + * @return true Limit reached + * @return false There is still room + */ + bool isOverflow() const { return _overflow; } + + /** + * @brief Reset the serializer to the start, clear overflow bit. + * + */ + void rewind() { _pos = 0; _overflow = false; } + + /** + * @brief Dump the contents of this object into QDebug + * + * This produces a dump of the internal state, and an ASCII/hex dump of + * the contents, for debugging. + * + * @param debug Qt QDebug stream + * @param ds This object + * @return QDebug + */ + friend QDebug operator<<(QDebug debug, const DataDeserializer &ds); + + private: + bool canAdvanceBy(size_t bytes, const QString &type_name) { + //qDebug() << "Checking advance by" << bytes; + + if ( _length < _pos + bytes) { + qCritical() << "Deserializer trying to read past end of input, reading" << bytes << "bytes for" << type_name << "from position " << _pos << ". Max length " << _length; + _overflow = true; + return false; + } + + return true; + } + + const char *_store; + bool _overflow = false; + size_t _length = 0; + size_t _pos = 0; +}; From 2cef749183946b9392c541ad6e48bbbc06d9b3c5 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Fri, 24 Jun 2022 14:25:20 +0200 Subject: [PATCH 09/19] Update test to work with the split serializer/deserializer --- libraries/shared/src/SerDes.cpp | 29 ++++++++++--- libraries/shared/src/SerDes.h | 4 +- tests/shared/src/SerializerTests.cpp | 65 +++++++++++++++------------- 3 files changed, 59 insertions(+), 39 deletions(-) diff --git a/libraries/shared/src/SerDes.cpp b/libraries/shared/src/SerDes.cpp index 044ea1fad2..1b30bf9f63 100644 --- a/libraries/shared/src/SerDes.cpp +++ b/libraries/shared/src/SerDes.cpp @@ -15,15 +15,13 @@ const int DataSerializer::DEFAULT_SIZE; const char DataSerializer::PADDING_CHAR; -QDebug operator<<(QDebug debug, const DataSerializer &ds) { - debug << "{ capacity =" << ds.capacity() << "; length = " << ds.length() << "; pos = " << ds.pos() << "}"; - debug << "\n"; - QString literal; +static void dumpHex(QDebug &debug, const char*buf, size_t len) { + QString literal; QString hex; - for(size_t i=0;i> i8; - s >> i16; - s >> i32; - s >> v3_b; - s >> v4_b; - s >> iv2_b; - s >> f_b; + d >> i8; + d >> i16; + d >> i32; + d >> v3_b; + d >> v4_b; + d >> iv2_b; + d >> f_b; - qDebug() << s; + qDebug() << d; QCOMPARE(i8, (qint8)1); QCOMPARE(i16, (qint16)0xaabb); @@ -106,22 +110,23 @@ void SerializerTests::testAddAndRead() { } void SerializerTests::testReadPastEnd() { - SerDes s; + DataSerializer s; qint8 i8; qint16 i16; s << (qint8)1; - s.rewind(); - s >> i8; - QCOMPARE(s.pos(), 1); - s.rewind(); - s >> i16; - QCOMPARE(s.pos(), 0); + DataDeserializer d(s); + d >> i8; + QCOMPARE(d.pos(), 1); + + d.rewind(); + d >> i16; + QCOMPARE(d.pos(), 0); } void SerializerTests::benchmarkEncodingDynamicAlloc() { QBENCHMARK { - SerDes s; + DataSerializer s; glm::vec3 v3_a{1.f, 3.1415f, 2.71828f}; glm::vec3 v3_b; glm::vec4 v4_a{3.1415f, 2.71828f, 1.4142f, 1.6180f}; @@ -142,7 +147,7 @@ void SerializerTests::benchmarkEncodingStaticAlloc() { char buf[1024]; QBENCHMARK { - SerDes s(buf, sizeof(buf)); + DataSerializer s(buf, sizeof(buf)); glm::vec3 v3_a{1.f, 3.1415f, 2.71828f}; glm::vec3 v3_b; glm::vec4 v4_a{3.1415f, 2.71828f, 1.4142f, 1.6180f}; @@ -161,7 +166,7 @@ void SerializerTests::benchmarkEncodingStaticAlloc() { void SerializerTests::benchmarkDecoding() { - SerDes s; + DataSerializer s; qint8 q8 = 1; qint16 q16 = 0xaabb; qint32 q32 = 0xccddeeff; @@ -182,13 +187,13 @@ void SerializerTests::benchmarkDecoding() { QBENCHMARK { - s.rewind(); - s >> q8; - s >> q16; - s >> q32; - s >> v3_a; - s >> v4_a; - s >> iv2_a; + DataDeserializer d(s); + d >> q8; + d >> q16; + d >> q32; + d >> v3_a; + d >> v4_a; + d >> iv2_a; } } From 3b797f57850fdf9a84c3631305b3778f0daf7d45 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sat, 9 Jul 2022 20:35:36 +0200 Subject: [PATCH 10/19] Debug code --- libraries/gpu/src/gpu/Texture_ktx.cpp | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index 6845aeb55b..a7e7844d0b 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -47,12 +47,15 @@ struct GPUKTXPayload { ser << CURRENT_VERSION; ser << _samplerDesc; + qCWarning(gpulogging) << "Offsets: " << offsetof(struct GPUKTXPayload, _samplerDesc) << offsetof(struct GPUKTXPayload, _usage) << offsetof(struct GPUKTXPayload, _usageType) << offsetof(struct GPUKTXPayload, _originalSize); uint32 usageData = _usage._flags.to_ulong(); ser << usageData; ser << (char)_usageType; ser << _originalSize; ser.addPadding(PADDING); + + assert(ser.length() == GPUKTXPayload::SIZE); } bool unserialize(DataDeserializer &dsr) { @@ -65,6 +68,9 @@ struct GPUKTXPayload { if (version > CURRENT_VERSION) { // If we try to load a version that we don't know how to parse, // it will render incorrectly + qCWarning(gpulogging) << "KTX version" << version << "is newer than our own," << CURRENT_VERSION; + qCWarning(gpulogging) << "Offsets: " << offsetof(struct GPUKTXPayload, _samplerDesc) << offsetof(struct GPUKTXPayload, _usage) << offsetof(struct GPUKTXPayload, _usageType) << offsetof(struct GPUKTXPayload, _originalSize); + qCWarning(gpulogging) << dsr; return false; } From 42d6128d9eb995ffe0d94ab8c16807d3edcef594 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Mon, 28 Nov 2022 01:28:27 +0100 Subject: [PATCH 11/19] Fix KTX issues with the serializer. * Stop trying to be compatible with the old format, and just bump the version number. * Add uint64_t support to serializer * A bit improved debug output from serializer * Add lastAdvance() function to ask the serializer how much data was added/read in the last operation. --- libraries/gpu/src/gpu/Texture_ktx.cpp | 30 +++++---- libraries/shared/src/SerDes.h | 94 +++++++++++++++++++++++++-- tests/shared/src/SerializerTests.cpp | 29 +++++++++ tests/shared/src/SerializerTests.h | 1 + 4 files changed, 136 insertions(+), 18 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index a7e7844d0b..d10fc87dd5 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -32,11 +32,8 @@ struct GPUKTXPayload { using Version = uint8; static const std::string KEY; - static const Version CURRENT_VERSION { 2 }; - static const size_t PADDING { 2 }; - static const size_t SIZE { sizeof(Version) + sizeof(Sampler::Desc) + sizeof(uint32) + sizeof(TextureUsageType) + sizeof(glm::ivec2) + PADDING }; - static_assert(GPUKTXPayload::SIZE == 44, "Packing size may differ between platforms"); - static_assert(GPUKTXPayload::SIZE % 4 == 0, "GPUKTXPayload is not 4 bytes aligned"); + static const Version CURRENT_VERSION { 3 }; + static const size_t SIZE { sizeof(Version) + sizeof(Sampler::Desc) + sizeof(uint64_t) + sizeof(TextureUsageType) + sizeof(glm::ivec2) }; Sampler::Desc _samplerDesc; Texture::Usage _usage; @@ -44,38 +41,43 @@ struct GPUKTXPayload { glm::ivec2 _originalSize { 0, 0 }; void serialize(DataSerializer &ser) { + ser << CURRENT_VERSION; + + ser << _samplerDesc; - qCWarning(gpulogging) << "Offsets: " << offsetof(struct GPUKTXPayload, _samplerDesc) << offsetof(struct GPUKTXPayload, _usage) << offsetof(struct GPUKTXPayload, _usageType) << offsetof(struct GPUKTXPayload, _originalSize); - uint32 usageData = _usage._flags.to_ulong(); + uint64_t usageData = _usage._flags.to_ulong(); ser << usageData; - - ser << (char)_usageType; + ser << ((uint8_t)_usageType); ser << _originalSize; - ser.addPadding(PADDING); + + // The +1 is here because we're adding the CURRENT_VERSION at the top, but since it's declared as a static + // const, it's not actually part of the class' size. assert(ser.length() == GPUKTXPayload::SIZE); } bool unserialize(DataDeserializer &dsr) { Version version = 0; - uint32 usageData; + uint64_t usageData = 0; uint8_t usagetype = 0; dsr >> version; - if (version > CURRENT_VERSION) { + if (version != CURRENT_VERSION) { // If we try to load a version that we don't know how to parse, // it will render incorrectly - qCWarning(gpulogging) << "KTX version" << version << "is newer than our own," << CURRENT_VERSION; - qCWarning(gpulogging) << "Offsets: " << offsetof(struct GPUKTXPayload, _samplerDesc) << offsetof(struct GPUKTXPayload, _usage) << offsetof(struct GPUKTXPayload, _usageType) << offsetof(struct GPUKTXPayload, _originalSize); + qCWarning(gpulogging) << "KTX version" << version << "is different than our own," << CURRENT_VERSION; qCWarning(gpulogging) << dsr; return false; } dsr >> _samplerDesc; + dsr >> usageData; + _usage._flags = gpu::Texture::Usage::Flags(usageData); + dsr >> usagetype; _usageType = (TextureUsageType)usagetype; diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index 365f14d79c..b0371d4b95 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -238,6 +238,33 @@ class DataSerializer { return *this; } + /////////////////////////////////////////////////////////// + + /** + * @brief Add an uint64_t to the output + * + * @param val Value to add + * @return SerDes& This object + */ + DataSerializer &operator<<(uint64_t val) { + return *this << int64_t(val); + } + + /** + * @brief Add an int64_t to the output + * + * @param val Value to add + * @return SerDes& This object + */ + DataSerializer &operator<<(int64_t val) { + if (!extendBy(sizeof(val), "int64_t")) { + return *this; + } + + memcpy(&_store[_pos], (char*)&val, sizeof(val)); + _pos += sizeof(val); + return *this; + } /////////////////////////////////////////////////////////// @@ -410,7 +437,18 @@ class DataSerializer { * @brief Reset the serializer to the start, clear overflow bit. * */ - void rewind() { _pos = 0; _overflow = false; } + void rewind() { _pos = 0; _overflow = false; _lastAdvance = 0; } + + + /** + * @brief Size of the last advance + * + * This can be used to get how many bytes were added in the last operation. + * It is reset on rewind() + * + * @return size_t + */ + size_t lastAdvance() const { return _lastAdvance; } /** * @brief Dump the contents of this object into QDebug @@ -430,7 +468,7 @@ class DataSerializer { if ( _capacity < _length + bytes) { if ( _storeIsExternal ) { - qCritical() << "Serializer trying to write past end of output buffer, writing" << bytes << "bytes for" << type_name << " from position " << _pos << ", length " << _length; + qCritical() << "Serializer trying to write past end of output buffer of" << _capacity << "bytes. Error writing" << bytes << "bytes for" << type_name << " from position " << _pos << ", length " << _length; _overflow = true; return false; } @@ -439,6 +477,7 @@ class DataSerializer { } _length += bytes; + _lastAdvance = bytes; return true; } @@ -451,6 +490,7 @@ class DataSerializer { size_t _capacity = 0; size_t _length = 0; size_t _pos = 0; + size_t _lastAdvance = 0; }; /** @@ -502,6 +542,7 @@ class DataDeserializer { _length = storeLength; _pos = 0; _store = externalStore; + _lastAdvance = 0; } /** @@ -536,6 +577,7 @@ class DataDeserializer { } _pos += bytes; + _lastAdvance = bytes; } @@ -558,6 +600,7 @@ class DataDeserializer { DataDeserializer &operator>>(int8_t &c) { if ( canAdvanceBy(1, "int8_t") ) { c = _store[_pos++]; + _lastAdvance = sizeof(c); } return *this; @@ -585,6 +628,7 @@ class DataDeserializer { if ( canAdvanceBy(sizeof(val), "int16_t") ) { memcpy((char*)&val, &_store[_pos], sizeof(val)); _pos += sizeof(val); + _lastAdvance = sizeof(val); } return *this; @@ -612,10 +656,37 @@ class DataDeserializer { if ( canAdvanceBy(sizeof(val), "int32_t") ) { memcpy((char*)&val, &_store[_pos], sizeof(val)); _pos += sizeof(val); + _lastAdvance = sizeof(val); } return *this; } + /////////////////////////////////////////////////////////// + + /** + * @brief Read an uint64_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(uint64_t &val) { + return *this >> reinterpret_cast(val); + } + + /** + * @brief Read an int64_t from the buffer + * + * @param val Value to read + * @return SerDes& This object + */ + DataDeserializer &operator>>(int64_t &val) { + if ( canAdvanceBy(sizeof(val), "int64_t") ) { + memcpy((char*)&val, &_store[_pos], sizeof(val)); + _pos += sizeof(val); + _lastAdvance = sizeof(val); + } + return *this; + } /////////////////////////////////////////////////////////// @@ -630,6 +701,7 @@ class DataDeserializer { if ( canAdvanceBy(sizeof(val), "float") ) { memcpy((char*)&val, &_store[_pos], sizeof(val)); _pos += sizeof(val); + _lastAdvance = sizeof(val); } return *this; } @@ -654,6 +726,7 @@ class DataDeserializer { memcpy((char*)&val.z, &_store[_pos + sz*2], sz); _pos += sz*3; + _lastAdvance = sz * 3; } return *this; @@ -678,6 +751,7 @@ class DataDeserializer { memcpy((char*)&val.w, &_store[_pos + sz*3], sz); _pos += sz*4; + _lastAdvance = sz*4; } return *this; } @@ -699,6 +773,7 @@ class DataDeserializer { memcpy((char*)&val.y, &_store[_pos + sz ], sz); _pos += sz*2; + _lastAdvance = sz * 2; } return *this; @@ -753,7 +828,17 @@ class DataDeserializer { * @brief Reset the serializer to the start, clear overflow bit. * */ - void rewind() { _pos = 0; _overflow = false; } + void rewind() { _pos = 0; _overflow = false; _lastAdvance = 0; } + + /** + * @brief Size of the last advance + * + * This can be used to get how many bytes were added in the last operation. + * It is reset on rewind() + * + * @return size_t + */ + size_t lastAdvance() const { return _lastAdvance; } /** * @brief Dump the contents of this object into QDebug @@ -772,7 +857,7 @@ class DataDeserializer { //qDebug() << "Checking advance by" << bytes; if ( _length < _pos + bytes) { - qCritical() << "Deserializer trying to read past end of input, reading" << bytes << "bytes for" << type_name << "from position " << _pos << ". Max length " << _length; + qCritical() << "Deserializer trying to read past end of input buffer of" << _length << "bytes, reading" << bytes << "bytes for" << type_name << "from position " << _pos; _overflow = true; return false; } @@ -784,4 +869,5 @@ class DataDeserializer { bool _overflow = false; size_t _length = 0; size_t _pos = 0; + size_t _lastAdvance = 0; }; diff --git a/tests/shared/src/SerializerTests.cpp b/tests/shared/src/SerializerTests.cpp index c52a9df69c..ae0198f573 100644 --- a/tests/shared/src/SerializerTests.cpp +++ b/tests/shared/src/SerializerTests.cpp @@ -124,6 +124,35 @@ void SerializerTests::testReadPastEnd() { QCOMPARE(d.pos(), 0); } +void SerializerTests::testWritePastEnd() { + qint8 i8 = 255; + qint16 i16 = 65535; + + + char buf[16]; + + + // 1 byte buffer, we can write 1 byte + memset(buf, 0, sizeof(buf)); + DataSerializer s1(buf, 1); + s1 << i8; + QCOMPARE(s1.pos(), 1); + QCOMPARE(s1.isOverflow(), false); + QCOMPARE(buf[0], i8); + + // 1 byte buffer, we can't write 2 bytes + memset(buf, 0, sizeof(buf)); + DataSerializer s2(buf, 1); + s2 << i16; + QCOMPARE(s2.pos(), 0); + QCOMPARE(s2.isOverflow(), true); + QCOMPARE(buf[0], 0); // We didn't write + QCOMPARE(buf[1], 0); +} + + + + void SerializerTests::benchmarkEncodingDynamicAlloc() { QBENCHMARK { DataSerializer s; diff --git a/tests/shared/src/SerializerTests.h b/tests/shared/src/SerializerTests.h index 3a3a3217d2..55da84c41a 100644 --- a/tests/shared/src/SerializerTests.h +++ b/tests/shared/src/SerializerTests.h @@ -21,6 +21,7 @@ private slots: void testAdd(); void testAddAndRead(); void testReadPastEnd(); + void testWritePastEnd(); void benchmarkEncodingDynamicAlloc(); void benchmarkEncodingStaticAlloc(); void benchmarkDecoding(); From 1c06c8652c8a3204c207ff0cff91e977be83d96b Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Fri, 2 Dec 2022 01:54:22 +0100 Subject: [PATCH 12/19] Add serializer size tracker, update documentation --- libraries/gpu/src/gpu/Texture.h | 6 +++ libraries/shared/src/SerDes.h | 86 +++++++++++++++++++++++++++++++-- 2 files changed, 89 insertions(+), 3 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture.h b/libraries/gpu/src/gpu/Texture.h index 8370462c5a..05ad70b872 100644 --- a/libraries/gpu/src/gpu/Texture.h +++ b/libraries/gpu/src/gpu/Texture.h @@ -94,6 +94,8 @@ typedef std::shared_ptr< SphericalHarmonics > SHPointer; inline DataSerializer &operator<<(DataSerializer &ser, const SphericalHarmonics &h) { + DataSerializer::SizeTracker tracker(ser); + ser << h.L00 << h.spare0; ser << h.L1m1 << h.spare1; ser << h.L10 << h.spare2; @@ -107,6 +109,8 @@ inline DataSerializer &operator<<(DataSerializer &ser, const SphericalHarmonics } inline DataDeserializer &operator>>(DataDeserializer &des, SphericalHarmonics &h) { + DataDeserializer::SizeTracker tracker(des); + des >> h.L00 >> h.spare0; des >> h.L1m1 >> h.spare1; des >> h.L10 >> h.spare2; @@ -238,6 +242,7 @@ protected: }; inline DataSerializer &operator<<(DataSerializer &ser, const Sampler::Desc &d) { + DataSerializer::SizeTracker tracker(ser); ser << d._borderColor; ser << d._maxAnisotropy; ser << d._filter; @@ -252,6 +257,7 @@ inline DataSerializer &operator<<(DataSerializer &ser, const Sampler::Desc &d) { } inline DataDeserializer &operator>>(DataDeserializer &dsr, Sampler::Desc &d) { + DataDeserializer::SizeTracker tracker(dsr); dsr >> d._borderColor; dsr >> d._maxAnisotropy; dsr >> d._filter; diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index b0371d4b95..4c97ef193b 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -49,7 +49,7 @@ * classes and structures, implement a `operator<<` and `operator>>` functions for that object, eg: * * @code {.cpp} - * Serializer &operator<<(Serializer &ser, const Object &o) { + * DataSerializer &operator<<(DataSerializer &ser, const Object &o) { * ser << o._borderColor; * ser << o._maxAnisotropy; * ser << o._filter; @@ -60,6 +60,46 @@ */ class DataSerializer { public: + + /** + * @brief RAII tracker of advance position + * + * When a custom operator<< is implemented for DataSserializer, + * this class allows to easily keep track of how much data has been added and + * adjust the parent's lastAdvance() count on this class' destruction. + * + * @code {.cpp} + * DataSerializer &operator<<(DataSerializer &ser, const Object &o) { + * DataSerializer::SizeTracker tracker(ser); + * + * ser << o._borderColor; + * ser << o._maxAnisotropy; + * ser << o._filter; + * return ser; + * } + * @endcode + */ + class SizeTracker { + public: + SizeTracker(DataSerializer &parent) : _parent(parent) { + _start_pos = _parent.pos(); + } + + ~SizeTracker() { + size_t cur_pos = _parent.pos(); + + if ( cur_pos >= _start_pos ) { + _parent._lastAdvance = cur_pos - _start_pos; + } else { + _parent._lastAdvance = 0; + } + } + + private: + DataSerializer &_parent; + size_t _start_pos = 0; + }; + /** * @brief Default size for a dynamically allocated buffer. * @@ -511,7 +551,7 @@ class DataSerializer { * uint16_t count; * glm::vec3 pos; * - * Deserializer des(buffer, sizeof(buffer)); + * DataDeserializer des(buffer, sizeof(buffer)); * des >> version; * des >> count; * des >> pos; @@ -521,7 +561,7 @@ class DataSerializer { * classes and structures, implement an `operator>>` function for that object, eg: * * @code {.cpp} - * Deserializer &operator>>(Deserializer &des, Object &o) { + * DataDeserializer &operator>>(DataDeserializer &des, Object &o) { * des >> o._borderColor; * des >> o._maxAnisotropy; * des >> o._filter; @@ -532,6 +572,46 @@ class DataSerializer { */ class DataDeserializer { public: + + /** + * @brief RAII tracker of advance position + * + * When a custom operator>> is implemented for DataDeserializer, + * this class allows to easily keep track of how much data has been added and + * adjust the parent's lastAdvance() count on this class' destruction. + * + * @code {.cpp} + * DataDeserializer &operator>>(Deserializer &des, Object &o) { + * DataDeserializer::SizeTracker tracker(des); + * + * des >> o._borderColor; + * des >> o._maxAnisotropy; + * des >> o._filter; + * return des; + * } + * @endcode + */ + class SizeTracker { + public: + SizeTracker(DataDeserializer &parent) : _parent(parent) { + _start_pos = _parent.pos(); + } + + ~SizeTracker() { + size_t cur_pos = _parent.pos(); + + if ( cur_pos >= _start_pos ) { + _parent._lastAdvance = cur_pos - _start_pos; + } else { + _parent._lastAdvance = 0; + } + } + + private: + DataDeserializer &_parent; + size_t _start_pos = 0; + }; + /** * @brief Construct a Deserializer * * From 101592357f5215f4cb5219392fa3750d742cb5da Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Fri, 2 Dec 2022 01:54:42 +0100 Subject: [PATCH 13/19] Restore KTX version to 2 This fixes compatibility with older baked assets --- libraries/gpu/src/gpu/Texture_ktx.cpp | 30 ++++++++++++++++++++------- 1 file changed, 23 insertions(+), 7 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index d10fc87dd5..67f8e9a621 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -28,12 +28,25 @@ using KtxStorage = Texture::KtxStorage; std::vector, std::shared_ptr>> KtxStorage::_cachedKtxFiles; std::mutex KtxStorage::_cachedKtxFilesMutex; + +/** + * @brief Payload for a KTX (texture) + * + * This contains a ready to use texture. This is both used for the local cache, and for baked textures. + * + * @note The usage for textures means breaking compatibility is a bad idea, and that the implementation + * should just keep on adding extra data at the bottom of the structure, and remain able to read old + * formats. In fact, version 1 KTX can be found in older baked assets. + */ struct GPUKTXPayload { using Version = uint8; static const std::string KEY; - static const Version CURRENT_VERSION { 3 }; - static const size_t SIZE { sizeof(Version) + sizeof(Sampler::Desc) + sizeof(uint64_t) + sizeof(TextureUsageType) + sizeof(glm::ivec2) }; + static const Version CURRENT_VERSION { 2 }; + static const size_t PADDING { 2 }; + static const size_t SIZE { sizeof(Version) + sizeof(Sampler::Desc) + sizeof(uint32_t) + sizeof(TextureUsageType) + sizeof(glm::ivec2) + PADDING }; + + static_assert(GPUKTXPayload::SIZE == 44, "Packing size may differ between platforms"); Sampler::Desc _samplerDesc; Texture::Usage _usage; @@ -43,13 +56,14 @@ struct GPUKTXPayload { void serialize(DataSerializer &ser) { ser << CURRENT_VERSION; - + ser.addPadding(1); ser << _samplerDesc; - uint64_t usageData = _usage._flags.to_ulong(); + uint32_t usageData = (uint32_t)_usage._flags.to_ulong(); ser << usageData; ser << ((uint8_t)_usageType); + ser.addPadding(1); ser << _originalSize; @@ -60,15 +74,16 @@ struct GPUKTXPayload { bool unserialize(DataDeserializer &dsr) { Version version = 0; - uint64_t usageData = 0; + uint32_t usageData = 0; uint8_t usagetype = 0; dsr >> version; + dsr.skipPadding(1); - if (version != CURRENT_VERSION) { + if (version > CURRENT_VERSION) { // If we try to load a version that we don't know how to parse, // it will render incorrectly - qCWarning(gpulogging) << "KTX version" << version << "is different than our own," << CURRENT_VERSION; + qCWarning(gpulogging) << "KTX version" << version << "is newer than our own," << CURRENT_VERSION; qCWarning(gpulogging) << dsr; return false; } @@ -79,6 +94,7 @@ struct GPUKTXPayload { _usage._flags = gpu::Texture::Usage::Flags(usageData); dsr >> usagetype; + dsr.skipPadding(1); _usageType = (TextureUsageType)usagetype; if (version >= 2) { From 1aa9fc0c8c910bfafca6582434b0f21ddffa9e61 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Thu, 6 Jun 2024 18:12:57 +0200 Subject: [PATCH 14/19] Updated headers --- libraries/shared/src/SerDes.cpp | 2 +- libraries/shared/src/SerDes.h | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/libraries/shared/src/SerDes.cpp b/libraries/shared/src/SerDes.cpp index 1b30bf9f63..b6adc9b0a6 100644 --- a/libraries/shared/src/SerDes.cpp +++ b/libraries/shared/src/SerDes.cpp @@ -3,7 +3,7 @@ // // // Created by Dale Glass on 5/6/2022 -// Copyright 2022 Dale Glass +// Copyright 2024 Overte e.V. // // Distributed under the Apache License, Version 2.0. // See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index 4c97ef193b..9de547313f 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -3,7 +3,7 @@ // // // Created by Dale Glass on 5/6/2022 -// Copyright 2022 Dale Glass +// Copyright 2024 Overte e.V. // // Distributed under the Apache License, Version 2.0. // See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html From f96ca65812290ff1e66d17a2a3a44154ed9a9843 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Thu, 6 Jun 2024 18:15:32 +0200 Subject: [PATCH 15/19] Fix indentation --- libraries/shared/src/SerDes.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libraries/shared/src/SerDes.cpp b/libraries/shared/src/SerDes.cpp index b6adc9b0a6..ad32d7014f 100644 --- a/libraries/shared/src/SerDes.cpp +++ b/libraries/shared/src/SerDes.cpp @@ -17,7 +17,7 @@ const char DataSerializer::PADDING_CHAR; static void dumpHex(QDebug &debug, const char*buf, size_t len) { - QString literal; + QString literal; QString hex; for(size_t i=0;i Date: Fri, 7 Jun 2024 00:45:58 +0200 Subject: [PATCH 16/19] Add more warning notes --- libraries/gpu/src/gpu/Texture_ktx.cpp | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index 67f8e9a621..f8a43bdb5b 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -53,6 +53,14 @@ struct GPUKTXPayload { TextureUsageType _usageType; glm::ivec2 _originalSize { 0, 0 }; + /** + * @brief Serialize the KTX payload + * + * @warning Be careful modifying this code, as it influences baked assets. + * Backwards compatibility must be maintained. + * + * @param ser Destination serializer + */ void serialize(DataSerializer &ser) { ser << CURRENT_VERSION; @@ -72,6 +80,16 @@ struct GPUKTXPayload { assert(ser.length() == GPUKTXPayload::SIZE); } + /** + * @brief Deserialize the KTX payload + * + * @warning Be careful modifying this code, as it influences baked assets. + * Backwards compatibility must be maintained. + * + * @param dsr Deserializer object + * @return true Successful + * @return false Version check failed + */ bool unserialize(DataDeserializer &dsr) { Version version = 0; uint32_t usageData = 0; From e1546ac3d074e284ede7e7ff9959391bc10e65b3 Mon Sep 17 00:00:00 2001 From: HifiExperiments Date: Fri, 21 Jun 2024 15:30:49 -0700 Subject: [PATCH 17/19] suggested fixes --- libraries/gpu/src/gpu/Texture_ktx.cpp | 7 ++----- libraries/render-utils/src/Model.cpp | 4 +--- libraries/shared/src/SerDes.h | 4 ++-- 3 files changed, 5 insertions(+), 10 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index f8a43bdb5b..78822ce9d7 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -64,19 +64,16 @@ struct GPUKTXPayload { void serialize(DataSerializer &ser) { ser << CURRENT_VERSION; - ser.addPadding(1); ser << _samplerDesc; uint32_t usageData = (uint32_t)_usage._flags.to_ulong(); ser << usageData; ser << ((uint8_t)_usageType); - ser.addPadding(1); ser << _originalSize; + ser.addPadding(PADDING); - // The +1 is here because we're adding the CURRENT_VERSION at the top, but since it's declared as a static - // const, it's not actually part of the class' size. assert(ser.length() == GPUKTXPayload::SIZE); } @@ -109,7 +106,7 @@ struct GPUKTXPayload { dsr >> _samplerDesc; dsr >> usageData; - _usage._flags = gpu::Texture::Usage::Flags(usageData); + _usage = gpu::Texture::Usage(usageData); dsr >> usagetype; dsr.skipPadding(1); diff --git a/libraries/render-utils/src/Model.cpp b/libraries/render-utils/src/Model.cpp index 6ad2534038..69a593ed09 100644 --- a/libraries/render-utils/src/Model.cpp +++ b/libraries/render-utils/src/Model.cpp @@ -1949,9 +1949,7 @@ void Blender::run() { blendedMeshSizes.push_back(numVertsInMesh); // initialize offsets to zero - for(BlendshapeOffsetUnpacked &bou : unpackedBlendshapeOffsets) { - bou.clear(); - } + memset(unpackedBlendshapeOffsets.data(), 0, numVertsInMesh * sizeof(BlendshapeOffsetUnpacked)); // for each blendshape in this mesh, accumulate the offsets into unpackedBlendshapeOffsets. const float NORMAL_COEFFICIENT_SCALE = 0.01f; diff --git a/libraries/shared/src/SerDes.h b/libraries/shared/src/SerDes.h index 9de547313f..f80d09a60a 100644 --- a/libraries/shared/src/SerDes.h +++ b/libraries/shared/src/SerDes.h @@ -105,7 +105,7 @@ class DataSerializer { * * Since this is mostly intended to be used for networking, we default to the largest probable MTU here. */ - static const int DEFAULT_SIZE = 1500; + static const int DEFAULT_SIZE = 1500; /** * @brief Character to use for padding. @@ -113,7 +113,7 @@ class DataSerializer { * Padding should be ignored, so it doesn't matter what we go with here, but it can be useful to set it * to something that would be distinctive in a dump. */ - static const char PADDING_CHAR = 0xAA; + static const char PADDING_CHAR = (char)0xAA; /** * @brief Construct a dynamically allocated serializer From b2665cbd4ef028ad535d3332f572d2657e7b7f79 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Mon, 24 Jun 2024 20:42:46 +0200 Subject: [PATCH 18/19] Fix warnings --- tests/ktx/src/KtxTests.cpp | 3 +-- tests/ktx/src/KtxTests.h | 1 + 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/ktx/src/KtxTests.cpp b/tests/ktx/src/KtxTests.cpp index cac9690b5c..9b104a7992 100644 --- a/tests/ktx/src/KtxTests.cpp +++ b/tests/ktx/src/KtxTests.cpp @@ -17,8 +17,7 @@ #include #include #include "SerDes.h" -#include "KTX.h" -#include "Texture_ktx.cpp" + QTEST_GUILESS_MAIN(KtxTests) diff --git a/tests/ktx/src/KtxTests.h b/tests/ktx/src/KtxTests.h index 5627dc313d..c59fc17ccc 100644 --- a/tests/ktx/src/KtxTests.h +++ b/tests/ktx/src/KtxTests.h @@ -16,6 +16,7 @@ private slots: void testKtxEvalFunctions(); void testKhronosCompressionFunctions(); void testKtxSerialization(); + void testKtxNewSerializationSphericalHarmonics(); }; From a7be389aed29df3f18bac8b77520ba0b2454ca4e Mon Sep 17 00:00:00 2001 From: HifiExperiments Date: Sat, 11 Jan 2025 20:08:31 -0800 Subject: [PATCH 19/19] fix padding, remove unused function --- libraries/gpu/src/gpu/Texture.h | 16 ---------------- libraries/gpu/src/gpu/Texture_ktx.cpp | 4 ++-- 2 files changed, 2 insertions(+), 18 deletions(-) diff --git a/libraries/gpu/src/gpu/Texture.h b/libraries/gpu/src/gpu/Texture.h index 05ad70b872..a6f527e657 100644 --- a/libraries/gpu/src/gpu/Texture.h +++ b/libraries/gpu/src/gpu/Texture.h @@ -188,24 +188,8 @@ public: _minMip == other._minMip && _maxMip == other._maxMip; } - - DataSerializer &operator<<(DataSerializer &dsd) { - dsd << _borderColor; - dsd << _maxAnisotropy; - dsd << _filter; - dsd << _comparisonFunc; - dsd << _wrapModeU; - dsd << _wrapModeV; - dsd << _wrapModeW; - dsd << _mipOffset; - dsd << _minMip; - dsd << _maxMip; - return dsd; - } }; - - Sampler() {} Sampler(const Filter filter, const WrapMode wrap = WRAP_REPEAT) : _desc(filter, wrap) {} Sampler(const Desc& desc) : _desc(desc) {} diff --git a/libraries/gpu/src/gpu/Texture_ktx.cpp b/libraries/gpu/src/gpu/Texture_ktx.cpp index 78822ce9d7..2a4d678208 100644 --- a/libraries/gpu/src/gpu/Texture_ktx.cpp +++ b/libraries/gpu/src/gpu/Texture_ktx.cpp @@ -93,7 +93,6 @@ struct GPUKTXPayload { uint8_t usagetype = 0; dsr >> version; - dsr.skipPadding(1); if (version > CURRENT_VERSION) { // If we try to load a version that we don't know how to parse, @@ -109,13 +108,14 @@ struct GPUKTXPayload { _usage = gpu::Texture::Usage(usageData); dsr >> usagetype; - dsr.skipPadding(1); _usageType = (TextureUsageType)usagetype; if (version >= 2) { dsr >> _originalSize; } + dsr.skipPadding(PADDING); + return true; }