From 748962d676fb3ab3dc1286b0b3e1f189e3decf1c Mon Sep 17 00:00:00 2001 From: Luis Michaelis Date: Sun, 9 Jun 2024 08:10:31 +0200 Subject: [PATCH] fix(Vfs): always write catalog directly after header for compatibility --- src/Vfs.cc | 20 +++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/src/Vfs.cc b/src/Vfs.cc index 9166024d..8873c264 100644 --- a/src/Vfs.cc +++ b/src/Vfs.cc @@ -260,12 +260,26 @@ namespace zenkit { return const_cast(const_cast(this)->find(name)); } + static uint32_t count_nodes(VfsNode const* node) { + uint32_t count = 1; /* self */ + + if (node->type() == VfsNodeType::DIRECTORY) { + for (auto& child : node->children()) { + count += count_nodes(&child); + } + } + + return count; + } + void Vfs::save(Write* w, GameVersion version) const { std::vector catalog; auto write_catalog = Write::to(&catalog); // Skip the header, we'll write it at the end. - w->seek(256 + 16 + 6 * 4, Whence::BEG); + unsigned header_size = 256 + 16 + 6 * 4; + unsigned catalog_size = (count_nodes(&_m_root) - 1) * (64 + 4 * 4); // -1 because the root node is not counted + w->seek(header_size + catalog_size, Whence::BEG); std::vector cache; std::string name; @@ -334,9 +348,9 @@ namespace zenkit { w->write_uint(index); w->write_uint(0); w->write_uint(off + catalog.size()); - w->write_uint(off); + w->write_uint(header_size); w->write_uint(80); - w->seek(static_cast(off), Whence::BEG); + w->seek(static_cast(header_size), Whence::BEG); w->write(catalog.data(), catalog.size()); }