146 lines
4.4 KiB
C++
146 lines
4.4 KiB
C++
#include "instance_data.h"
|
|
#include "../util/array_slice.h"
|
|
#include "../util/serialization.h"
|
|
#include "../util/utility.h"
|
|
#include "../voxel_constants.h"
|
|
|
|
namespace {
|
|
const uint32_t TRAILING_MAGIC = 0x900df00d;
|
|
}
|
|
|
|
inline uint8_t norm_to_u8(float x) {
|
|
return clamp(static_cast<int>(128.f * x + 128.f), 0, 0xff);
|
|
}
|
|
|
|
inline float u8_to_norm(uint8_t v) {
|
|
return (static_cast<real_t>(v) - 0x7f) * VoxelConstants::INV_0x7f;
|
|
}
|
|
|
|
struct CompressedQuaternion4b {
|
|
uint8_t x;
|
|
uint8_t y;
|
|
uint8_t z;
|
|
uint8_t w;
|
|
|
|
static CompressedQuaternion4b from_quat(Quat q) {
|
|
CompressedQuaternion4b c;
|
|
c.x = norm_to_u8(q.x);
|
|
c.y = norm_to_u8(q.y);
|
|
c.z = norm_to_u8(q.z);
|
|
c.w = norm_to_u8(q.w);
|
|
return c;
|
|
}
|
|
|
|
Quat to_quat() const {
|
|
Quat q;
|
|
q.x = u8_to_norm(x);
|
|
q.y = u8_to_norm(y);
|
|
q.z = u8_to_norm(z);
|
|
q.w = u8_to_norm(w);
|
|
return q.normalized();
|
|
}
|
|
};
|
|
|
|
void serialize_instance_block_data(const VoxelInstanceBlockData &src, std::vector<uint8_t> &dst) {
|
|
const uint8_t version = 0;
|
|
const uint8_t instance_format = 0;
|
|
|
|
VoxelUtility::MemoryWriter w(dst, VoxelUtility::ENDIANESS_BIG_ENDIAN);
|
|
|
|
w.store_8(version);
|
|
w.store_8(src.layers.size());
|
|
w.store_float(src.position_range);
|
|
|
|
// TODO Introduce a margin to position coordinates, stuff can spawn offset from the ground.
|
|
// Or just compute the ranges
|
|
const float pos_norm_scale = 1.f / src.position_range;
|
|
|
|
for (size_t i = 0; i < src.layers.size(); ++i) {
|
|
const VoxelInstanceBlockData::LayerData &layer = src.layers[i];
|
|
|
|
w.store_16(layer.id);
|
|
w.store_16(layer.instances.size());
|
|
w.store_float(layer.scale_min);
|
|
w.store_float(layer.scale_max);
|
|
w.store_8(instance_format);
|
|
|
|
const float scale_norm_scale = 1.f / (layer.scale_max - layer.scale_min);
|
|
|
|
for (size_t j = 0; j < layer.instances.size(); ++j) {
|
|
const VoxelInstanceBlockData::InstanceData &instance = layer.instances[j];
|
|
|
|
w.store_16(static_cast<uint16_t>(pos_norm_scale * instance.transform.origin.x * 0xffff));
|
|
w.store_16(static_cast<uint16_t>(pos_norm_scale * instance.transform.origin.y * 0xffff));
|
|
w.store_16(static_cast<uint16_t>(pos_norm_scale * instance.transform.origin.z * 0xffff));
|
|
|
|
const float scale = instance.transform.get_basis().get_scale().y;
|
|
w.store_8(static_cast<uint8_t>(scale_norm_scale * (scale - layer.scale_min) * 0xff));
|
|
|
|
const Quat q = instance.transform.get_basis().get_rotation_quat();
|
|
const CompressedQuaternion4b cq = CompressedQuaternion4b::from_quat(q);
|
|
w.store_8(cq.x);
|
|
w.store_8(cq.y);
|
|
w.store_8(cq.z);
|
|
w.store_8(cq.w);
|
|
}
|
|
}
|
|
|
|
w.store_32(TRAILING_MAGIC);
|
|
}
|
|
|
|
bool deserialize_instance_block_data(VoxelInstanceBlockData &dst, ArraySlice<const uint8_t> src) {
|
|
const uint8_t expected_version = 0;
|
|
const uint8_t expected_instance_format = 0;
|
|
|
|
VoxelUtility::MemoryReader r(src, VoxelUtility::ENDIANESS_BIG_ENDIAN);
|
|
|
|
const uint8_t version = r.get_8();
|
|
ERR_FAIL_COND_V(version != expected_version, false);
|
|
|
|
const uint8_t layers_count = r.get_8();
|
|
dst.layers.resize(layers_count);
|
|
|
|
dst.position_range = r.get_float();
|
|
|
|
for (size_t i = 0; i < dst.layers.size(); ++i) {
|
|
VoxelInstanceBlockData::LayerData &layer = dst.layers[i];
|
|
|
|
layer.id = r.get_16();
|
|
|
|
const uint16_t instance_count = r.get_16();
|
|
layer.instances.resize(instance_count);
|
|
|
|
layer.scale_min = r.get_float();
|
|
layer.scale_max = r.get_float();
|
|
ERR_FAIL_COND_V(layer.scale_max < layer.scale_min, false);
|
|
const float scale_range = layer.scale_max - layer.scale_min;
|
|
|
|
const uint8_t instance_format = r.get_8();
|
|
ERR_FAIL_COND_V(instance_format != expected_instance_format, false);
|
|
|
|
for (size_t j = 0; j < layer.instances.size(); ++j) {
|
|
const float x = (static_cast<float>(r.get_16()) / 0xffff) * dst.position_range;
|
|
const float y = (static_cast<float>(r.get_16()) / 0xffff) * dst.position_range;
|
|
const float z = (static_cast<float>(r.get_16()) / 0xffff) * dst.position_range;
|
|
|
|
const float s = (static_cast<float>(r.get_8()) / 0xff) * scale_range + layer.scale_min;
|
|
|
|
CompressedQuaternion4b cq;
|
|
cq.x = r.get_8();
|
|
cq.y = r.get_8();
|
|
cq.z = r.get_8();
|
|
cq.w = r.get_8();
|
|
const Quat q = cq.to_quat();
|
|
|
|
VoxelInstanceBlockData::InstanceData &instance = layer.instances[j];
|
|
instance.transform = Transform(Basis(q).scaled(Vector3(s, s, s)), Vector3(x, y, z));
|
|
}
|
|
}
|
|
|
|
const uint32_t control_end = r.get_32();
|
|
ERR_FAIL_COND_V_MSG(control_end != TRAILING_MAGIC, false,
|
|
String("Expected {0}, found {1}").format(varray(TRAILING_MAGIC, control_end)));
|
|
|
|
return true;
|
|
}
|