Crafter.Asset/implementations/Crafter.Asset-Compression.cpp

136 lines
5.5 KiB
C++
Raw Normal View History

2026-05-11 18:37:30 +02:00
/*
Crafter®.Asset
Copyright (C) 2026 Catcrafts®
catcrafts.net
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License version 3.0 as published by the Free Software Foundation;
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
module;
// Vendored GDeflate (Microsoft DirectStorage reference, Apache-2.0). Headers
// pull libdeflate (MIT) via -Ilib/gdeflate/libdeflate. The C++ wrappers are
// pulled inline because Crafter.Build's cFiles only handles .c TUs — folding
// them into this module impl avoids adding a parallel cppFiles channel.
#include "../lib/gdeflate/GDeflate.h"
#include "../lib/gdeflate/GDeflateCompress.cpp"
#include "../lib/gdeflate/GDeflateDecompress.cpp"
module Crafter.Asset;
import std;
namespace Crafter::Compression {
CompressedBlob CompressStreams(std::span<const std::span<const std::byte>> streams) {
CompressedBlob blob;
blob.regions.reserve(streams.size());
// First pass: compress each stream into its own buffer so we know the
// exact final size. GDeflate::CompressBound is an upper bound; we'd
// waste capacity if we appended the bound directly into a single
// shared buffer.
std::vector<std::vector<std::byte>> compressed;
compressed.reserve(streams.size());
std::uint64_t totalSize = 0;
for (const std::span<const std::byte>& stream : streams) {
if (stream.empty()) {
compressed.emplace_back();
continue;
}
std::size_t boundSize = GDeflate::CompressBound(stream.size());
std::vector<std::byte> out(boundSize);
std::size_t actualSize = boundSize;
bool ok = GDeflate::Compress(
reinterpret_cast<std::uint8_t*>(out.data()),
&actualSize,
reinterpret_cast<const std::uint8_t*>(stream.data()),
stream.size(),
GDeflate::MaximumCompressionLevel,
0);
if (!ok) {
2026-05-18 22:31:28 +02:00
Fatal("GDeflate::Compress failed");
2026-05-11 18:37:30 +02:00
}
out.resize(actualSize);
totalSize += actualSize;
compressed.push_back(std::move(out));
}
// Second pass: concatenate and build the region table.
blob.bytes.reserve(totalSize);
for (std::size_t i = 0; i < streams.size(); ++i) {
RegionMeta r {
.srcOffset = blob.bytes.size(),
.compressedSize = compressed[i].size(),
.decompressedSize = streams[i].size(),
};
blob.regions.push_back(r);
blob.bytes.insert(blob.bytes.end(), compressed[i].begin(), compressed[i].end());
}
return blob;
}
void WriteBlob(std::ostream& file, const CompressedBlob& blob) {
std::uint32_t regionCount = static_cast<std::uint32_t>(blob.regions.size());
file.write(reinterpret_cast<const char*>(&regionCount), sizeof(regionCount));
if (regionCount > 0) {
file.write(reinterpret_cast<const char*>(blob.regions.data()),
regionCount * sizeof(RegionMeta));
}
std::uint64_t payloadSize = blob.bytes.size();
file.write(reinterpret_cast<const char*>(&payloadSize), sizeof(payloadSize));
if (payloadSize > 0) {
file.write(reinterpret_cast<const char*>(blob.bytes.data()), payloadSize);
}
}
CompressedBlob ReadBlob(std::istream& file) {
CompressedBlob blob;
std::uint32_t regionCount = 0;
file.read(reinterpret_cast<char*>(&regionCount), sizeof(regionCount));
blob.regions.resize(regionCount);
if (regionCount > 0) {
file.read(reinterpret_cast<char*>(blob.regions.data()),
regionCount * sizeof(RegionMeta));
}
std::uint64_t payloadSize = 0;
file.read(reinterpret_cast<char*>(&payloadSize), sizeof(payloadSize));
blob.bytes.resize(payloadSize);
if (payloadSize > 0) {
file.read(reinterpret_cast<char*>(blob.bytes.data()), payloadSize);
}
return blob;
}
void DecompressCPU(const CompressedBlob& blob, std::span<const std::span<std::byte>> outputs) {
if (outputs.size() != blob.regions.size()) {
2026-05-18 22:31:28 +02:00
Fatal("DecompressCPU: outputs.size() != regions.size()");
2026-05-11 18:37:30 +02:00
}
for (std::size_t i = 0; i < blob.regions.size(); ++i) {
const RegionMeta& r = blob.regions[i];
const std::span<std::byte>& out = outputs[i];
if (out.size() != r.decompressedSize) {
2026-05-18 22:31:28 +02:00
Fatal("DecompressCPU: output size mismatch");
2026-05-11 18:37:30 +02:00
}
if (r.decompressedSize == 0) continue;
bool ok = GDeflate::Decompress(
reinterpret_cast<std::uint8_t*>(out.data()),
r.decompressedSize,
reinterpret_cast<const std::uint8_t*>(blob.bytes.data() + r.srcOffset),
r.compressedSize,
/*numWorkers=*/1);
if (!ok) {
2026-05-18 22:31:28 +02:00
Fatal("GDeflate::Decompress failed");
2026-05-11 18:37:30 +02:00
}
}
}
}