From 27910952fb4a7894992f4a3e97119f9ed4e9e824 Mon Sep 17 00:00:00 2001 From: sean Date: Sat, 24 Dec 2022 23:44:03 +0100 Subject: [PATCH] Fix: Keep copied JSON data alive for GLBs --- src/fastgltf.cpp | 26 ++++++++++++++------------ src/fastgltf_parser.hpp | 1 + 2 files changed, 15 insertions(+), 12 deletions(-) diff --git a/src/fastgltf.cpp b/src/fastgltf.cpp index e94b04a47..11d5878d6 100644 --- a/src/fastgltf.cpp +++ b/src/fastgltf.cpp @@ -54,8 +54,7 @@ namespace fastgltf { constexpr auto hashTextures = crc32("textures"); struct ParserData { - // Can simdjson not store this data itself? - std::vector bytes; + simdjson::padded_string json; // Only used to keep the copied json data with GLBs alive. simdjson::ondemand::document doc; simdjson::ondemand::object root; @@ -64,8 +63,9 @@ namespace fastgltf { void* userPointer = nullptr; }; - // ASCII for "glTF". - constexpr uint32_t binaryGltfHeaderMagic = 0x46546C67; + constexpr uint32_t binaryGltfHeaderMagic = 0x46546C67; // ASCII for "glTF". + constexpr uint32_t binaryGltfJsonChunkMagic = 0x4E4F534A; + constexpr uint32_t binaryGltfDataChunkMagic = 0x004E4942; struct BinaryGltfHeader { uint32_t magic; @@ -1957,22 +1957,24 @@ std::unique_ptr fg::Parser::loadBinaryGLTF(GltfDataBuffer* buffer, fs: // 2. BIN chunk (optional) BinaryGltfChunk jsonChunk = {}; read(&jsonChunk, sizeof jsonChunk); - if (jsonChunk.chunkType != 0x4E4F534A) { + if (jsonChunk.chunkType != binaryGltfJsonChunkMagic) { errorCode = Error::InvalidGLB; return nullptr; } - // TODO: Keep this alive somehow - simdjson::padded_string jsonString(jsonChunk.chunkLength); // This adds the padding itself. - read(jsonString.data(), jsonChunk.chunkLength); - if (hasBit(options, Options::DontUseSIMD)) { simdjson::get_active_implementation() = simdjson::get_available_implementations()["fallback"]; } - // The 'false' indicates that simdjson doesn't have to copy the data internally. auto data = std::make_unique(); - if (jsonParser->iterate(jsonString).get(data->doc) != SUCCESS) { + + { + simdjson::padded_string glbJson(jsonChunk.chunkLength); // This adds the padding itself. + read(glbJson.data(), jsonChunk.chunkLength); + data->json = std::move(glbJson); + } + + if (jsonParser->iterate(simdjson::padded_string_view(data->json)).get(data->doc) != SUCCESS) { errorCode = Error::InvalidJson; return nullptr; } @@ -1991,7 +1993,7 @@ std::unique_ptr fg::Parser::loadBinaryGLTF(GltfDataBuffer* buffer, fs: BinaryGltfChunk binaryChunk = {}; read(&binaryChunk, sizeof binaryChunk); - if (binaryChunk.chunkType != 0x004E4942) { + if (binaryChunk.chunkType != binaryGltfDataChunkMagic) { errorCode = Error::InvalidGLB; return nullptr; } diff --git a/src/fastgltf_parser.hpp b/src/fastgltf_parser.hpp index 6bba80a08..ac0145147 100644 --- a/src/fastgltf_parser.hpp +++ b/src/fastgltf_parser.hpp @@ -230,6 +230,7 @@ namespace fastgltf { void parseScenes(simdjson::fallback::ondemand::array array); void parseSkins(simdjson::fallback::ondemand::array array); void parseTextures(simdjson::fallback::ondemand::array array); + public: explicit glTF(const glTF& scene) = delete; glTF& operator=(const glTF& scene) = delete;