98 lines
3.1 KiB
C++
98 lines
3.1 KiB
C++
#include "compressed_chunks.hpp"
|
|
|
|
#include "coders/rle.hpp"
|
|
#include "coders/gzip.hpp"
|
|
|
|
#include "files/WorldFiles.hpp"
|
|
|
|
inline constexpr int HAS_VOXELS = 0x1;
|
|
inline constexpr int HAS_METADATA = 0x2;
|
|
|
|
std::vector<ubyte> compressed_chunks::encode(
|
|
const ubyte* data,
|
|
const BlocksMetadata& metadata,
|
|
util::Buffer<ubyte>& rleBuffer
|
|
) {
|
|
size_t rleCompressedSize =
|
|
extrle::encode16(data, CHUNK_DATA_LEN, rleBuffer.data());
|
|
|
|
const auto gzipCompressedData = gzip::compress(
|
|
rleBuffer.data(), rleCompressedSize
|
|
);
|
|
auto metadataBytes = metadata.serialize();
|
|
|
|
ByteBuilder builder(2 + 8 + gzipCompressedData.size() + metadataBytes.size());
|
|
builder.put(HAS_VOXELS | HAS_METADATA); // flags
|
|
builder.put(0); // reserved
|
|
builder.putInt32(gzipCompressedData.size());
|
|
builder.put(gzipCompressedData.data(), gzipCompressedData.size());
|
|
builder.putInt32(metadataBytes.size());
|
|
builder.put(metadataBytes.data(), metadataBytes.size());
|
|
return builder.build();
|
|
}
|
|
|
|
std::vector<ubyte> compressed_chunks::encode(const Chunk& chunk) {
|
|
auto data = chunk.encode();
|
|
|
|
/// world.get_chunk_data is only available in the main Lua state
|
|
static util::Buffer<ubyte> rleBuffer(CHUNK_DATA_LEN * 2);
|
|
return encode(data.get(), chunk.blocksMetadata, rleBuffer);
|
|
}
|
|
|
|
static void read_voxel_data(ByteReader& reader, util::Buffer<ubyte>& dst) {
|
|
size_t gzipCompressedSize = reader.getInt32();
|
|
|
|
auto rleData = gzip::decompress(reader.pointer(), gzipCompressedSize);
|
|
reader.skip(gzipCompressedSize);
|
|
|
|
extrle::decode16(rleData.data(), rleData.size(), dst.data());
|
|
}
|
|
|
|
void compressed_chunks::decode(Chunk& chunk, const ubyte* src, size_t size) {
|
|
ByteReader reader(src, size);
|
|
|
|
ubyte flags = reader.get();
|
|
reader.skip(1); // reserved byte
|
|
|
|
if (flags & HAS_VOXELS) {
|
|
/// world.get_chunk_data is only available in the main Lua state
|
|
static util::Buffer<ubyte> voxelData (CHUNK_DATA_LEN);
|
|
read_voxel_data(reader, voxelData);
|
|
chunk.decode(voxelData.data());
|
|
chunk.updateHeights();
|
|
}
|
|
if (flags & HAS_METADATA) {
|
|
size_t metadataSize = reader.getInt32();
|
|
chunk.blocksMetadata.deserialize(reader.pointer(), metadataSize);
|
|
reader.skip(metadataSize);
|
|
}
|
|
chunk.setModifiedAndUnsaved();
|
|
}
|
|
|
|
void compressed_chunks::save(
|
|
int x, int z, std::vector<ubyte> bytes, WorldRegions& regions
|
|
) {
|
|
ByteReader reader(bytes.data(), bytes.size());
|
|
|
|
ubyte flags = reader.get();
|
|
reader.skip(1); // reserved byte
|
|
if (flags & HAS_VOXELS) {
|
|
util::Buffer<ubyte> voxelData (CHUNK_DATA_LEN);
|
|
read_voxel_data(reader, voxelData);
|
|
regions.put(
|
|
x, z, REGION_LAYER_VOXELS, voxelData.release(), CHUNK_DATA_LEN
|
|
);
|
|
}
|
|
if (flags & HAS_METADATA) {
|
|
size_t metadataSize = reader.getInt32();
|
|
regions.put(
|
|
x,
|
|
z,
|
|
REGION_LAYER_BLOCKS_DATA,
|
|
util::Buffer<ubyte>(reader.pointer(), metadataSize).release(),
|
|
metadataSize
|
|
);
|
|
reader.skip(metadataSize);
|
|
}
|
|
}
|