1
0
mirror of https://github.com/luanti-org/luanti.git synced 2025-10-26 13:25:27 +01:00
Files
luanti/lib/tiniergltf/tiniergltf.hpp
Lars Müller fde6384a09 Fix and clean up skeletal animation (#15722)
* Fix attachments lagging behind their parents (#14818)
* Fix animation blending (#14817)
* Bring back cool guy as another .x smoke test
* Add .x mesh loader unittest
* Do bounding box & matrix calculation at proper point in time
* Remove obsolete `SAnimatedMesh`
2025-06-01 23:21:35 +02:00

1485 lines
40 KiB
C++

#pragma once
#include <json/json.h>
#include "util/base64.h"
#include <cstdint>
#include <functional>
#include <stack>
#include <string>
#include <string_view>
#include <variant>
#include <vector>
#include <array>
#include <optional>
#include <limits>
#include <memory> // unique_ptr
#include <stdexcept>
#include <unordered_map>
#include <unordered_set>
#if JSONCPP_VERSION_HEXA < 0x01090000 /* 1.9.0 */
namespace Json {
using String = JSONCPP_STRING; // Polyfill
}
#endif
namespace tiniergltf {
static inline void check(bool cond) {
if (!cond)
throw std::runtime_error("invalid glTF");
}
template <typename T>
static inline void checkIndex(const std::optional<std::vector<T>> &vec,
const std::optional<std::size_t> &i) {
if (!i.has_value()) return;
check(vec.has_value());
check(i < vec->size());
}
template <typename T>
static inline void checkIndex(const std::vector<T> &vec,
const std::optional<std::size_t> &i) {
if (!i.has_value()) return;
check(i < vec.size());
}
template <typename T, typename F>
static inline void checkForall(const std::optional<std::vector<T>> &vec, const F &cond) {
if (!vec.has_value())
return;
for (const T &v : vec.value())
cond(v);
}
template <typename T>
static inline void checkDuplicateFree(const std::vector<T> &vec) {
check(std::unordered_set<T>(vec.begin(), vec.end()).size() == vec.size());
}
template <typename T>
static inline T as(const Json::Value &o);
template<>
bool as(const Json::Value &o) {
check(o.isBool());
return o.asBool();
}
template<>
double as (const Json::Value &o) {
check(o.isDouble());
return o.asDouble();
}
template<>
std::size_t as(const Json::Value &o) {
check(o.isUInt64());
auto u = o.asUInt64();
check(u <= std::numeric_limits<std::size_t>::max());
return u;
}
template<>
std::string as(const Json::Value &o) {
check(o.isString());
return o.asString();
}
template<typename U>
std::vector<U> asVec(const Json::Value &o) {
check(o.isArray());
std::vector<U> res;
res.reserve(o.size());
for (Json::ArrayIndex i = 0; i < o.size(); ++i) {
res.push_back(as<U>(o[i]));
}
return res;
}
template<typename U, std::size_t n>
std::array<U, n> asArr(const Json::Value &o) {
check(o.isArray());
check(o.size() == n);
std::array<U, n> res;
for (Json::ArrayIndex i = 0; i < n; ++i) {
res[i] = as<U>(o[i]);
}
return res;
}
struct AccessorSparseIndices {
std::size_t bufferView;
std::size_t byteOffset;
// as defined in the glTF specification
enum class ComponentType {
UNSIGNED_BYTE,
UNSIGNED_SHORT,
UNSIGNED_INT,
};
ComponentType componentType;
std::size_t componentSize() const {
switch (componentType) {
case ComponentType::UNSIGNED_BYTE:
return 1;
case ComponentType::UNSIGNED_SHORT:
return 2;
case ComponentType::UNSIGNED_INT:
return 4;
}
throw std::logic_error("invalid component type");
}
std::size_t elementSize() const {
return componentSize();
}
AccessorSparseIndices(const Json::Value &o)
: bufferView(as<std::size_t>(o["bufferView"]))
, byteOffset(0)
{
check(o.isObject());
if (o.isMember("byteOffset")) {
byteOffset = as<std::size_t>(o["byteOffset"]);
check(byteOffset >= 0);
}
{
static std::unordered_map<Json::UInt64, ComponentType> map = {
{5121, ComponentType::UNSIGNED_BYTE},
{5123, ComponentType::UNSIGNED_SHORT},
{5125, ComponentType::UNSIGNED_INT},
};
const auto &v = o["componentType"]; check(v.isUInt64());
componentType = map.at(v.asUInt64());
}
}
};
template<> AccessorSparseIndices as(const Json::Value &o) { return o; }
struct AccessorSparseValues {
std::size_t bufferView;
std::size_t byteOffset;
AccessorSparseValues(const Json::Value &o)
: bufferView(as<std::size_t>(o["bufferView"]))
, byteOffset(0)
{
check(o.isObject());
if (o.isMember("byteOffset")) {
byteOffset = as<std::size_t>(o["byteOffset"]);
check(byteOffset >= 0);
}
}
};
template<> AccessorSparseValues as(const Json::Value &o) { return o; }
struct AccessorSparse {
std::size_t count;
AccessorSparseIndices indices;
AccessorSparseValues values;
AccessorSparse(const Json::Value &o)
: count(as<std::size_t>(o["count"]))
, indices(as<AccessorSparseIndices>(o["indices"]))
, values(as<AccessorSparseValues>(o["values"]))
{
check(o.isObject());
check(count >= 1);
}
};
template<> AccessorSparse as(const Json::Value &o) { return o; }
struct Accessor {
std::optional<std::size_t> bufferView;
std::size_t byteOffset;
// as defined in the glTF specification
enum class ComponentType {
BYTE,
UNSIGNED_BYTE,
SHORT,
UNSIGNED_SHORT,
UNSIGNED_INT,
FLOAT,
};
ComponentType componentType;
std::size_t componentSize() const {
switch (componentType) {
case ComponentType::BYTE:
case ComponentType::UNSIGNED_BYTE:
return 1;
case ComponentType::SHORT:
case ComponentType::UNSIGNED_SHORT:
return 2;
case ComponentType::UNSIGNED_INT:
case ComponentType::FLOAT:
return 4;
}
throw std::logic_error("invalid component type");
}
std::size_t count;
std::optional<std::vector<double>> max;
std::optional<std::vector<double>> min;
std::optional<std::string> name;
bool normalized;
std::optional<AccessorSparse> sparse;
enum class Type {
MAT2,
MAT3,
MAT4,
SCALAR,
VEC2,
VEC3,
VEC4,
};
std::size_t typeCount() const {
switch (type) {
case Type::SCALAR:
return 1;
case Type::VEC2:
return 2;
case Type::VEC3:
return 3;
case Type::MAT2:
case Type::VEC4:
return 4;
case Type::MAT3:
return 9;
case Type::MAT4:
return 16;
}
throw std::logic_error("invalid type");
}
Type type;
std::size_t elementSize() const {
return componentSize() * typeCount();
}
Accessor(const Json::Value &o)
: byteOffset(0)
, count(as<std::size_t>(o["count"]))
, normalized(false)
{
check(o.isObject());
if (o.isMember("bufferView")) {
bufferView = as<std::size_t>(o["bufferView"]);
}
{
static std::unordered_map<Json::UInt64, ComponentType> map = {
{5120, ComponentType::BYTE},
{5121, ComponentType::UNSIGNED_BYTE},
{5122, ComponentType::SHORT},
{5123, ComponentType::UNSIGNED_SHORT},
{5125, ComponentType::UNSIGNED_INT},
{5126, ComponentType::FLOAT},
};
const auto &v = o["componentType"]; check(v.isUInt64());
componentType = map.at(v.asUInt64());
}
if (o.isMember("byteOffset")) {
byteOffset = as<std::size_t>(o["byteOffset"]);
check(byteOffset >= 0);
check(byteOffset % componentSize() == 0);
}
check(count >= 1);
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("normalized")) {
normalized = as<bool>(o["normalized"]);
}
if (o.isMember("sparse")) {
sparse = as<AccessorSparse>(o["sparse"]);
}
{
static std::unordered_map<Json::String, Type> map = {
{"MAT2", Type::MAT2},
{"MAT3", Type::MAT3},
{"MAT4", Type::MAT4},
{"SCALAR", Type::SCALAR},
{"VEC2", Type::VEC2},
{"VEC3", Type::VEC3},
{"VEC4", Type::VEC4},
};
const auto &v = o["type"]; check(v.isString());
type = map.at(v.asString());
}
if (o.isMember("max")) {
max = asVec<double>(o["max"]);
check(max->size() == typeCount());
}
if (o.isMember("min")) {
min = asVec<double>(o["min"]);
check(min->size() == typeCount());
}
}
};
template<> Accessor as(const Json::Value &o) { return o; }
struct AnimationChannelTarget {
std::optional<std::size_t> node;
enum class Path {
ROTATION,
SCALE,
TRANSLATION,
WEIGHTS,
};
Path path;
AnimationChannelTarget(const Json::Value &o)
{
check(o.isObject());
if (o.isMember("node")) {
node = as<std::size_t>(o["node"]);
}
{
static std::unordered_map<Json::String, Path> map = {
{"rotation", Path::ROTATION},
{"scale", Path::SCALE},
{"translation", Path::TRANSLATION},
{"weights", Path::WEIGHTS},
};
const auto &v = o["path"]; check(v.isString());
path = map.at(v.asString());
}
}
};
template<> AnimationChannelTarget as(const Json::Value &o) { return o; }
struct AnimationChannel {
std::size_t sampler;
AnimationChannelTarget target;
AnimationChannel(const Json::Value &o)
: sampler(as<std::size_t>(o["sampler"]))
, target(as<AnimationChannelTarget>(o["target"]))
{
check(o.isObject());
}
};
template<> AnimationChannel as(const Json::Value &o) { return o; }
struct AnimationSampler {
std::size_t input;
enum class Interpolation {
CUBICSPLINE,
LINEAR,
STEP,
};
Interpolation interpolation;
std::size_t output;
AnimationSampler(const Json::Value &o)
: input(as<std::size_t>(o["input"]))
, interpolation(Interpolation::LINEAR)
, output(as<std::size_t>(o["output"]))
{
check(o.isObject());
if (o.isMember("interpolation")) {
static std::unordered_map<Json::String, Interpolation> map = {
{"CUBICSPLINE", Interpolation::CUBICSPLINE},
{"LINEAR", Interpolation::LINEAR},
{"STEP", Interpolation::STEP},
};
const auto &v = o["interpolation"]; check(v.isString());
interpolation = map.at(v.asString());
}
}
};
template<> AnimationSampler as(const Json::Value &o) { return o; }
struct Animation {
std::vector<AnimationChannel> channels;
std::optional<std::string> name;
std::vector<AnimationSampler> samplers;
Animation(const Json::Value &o)
: channels(asVec<AnimationChannel>(o["channels"]))
, samplers(asVec<AnimationSampler>(o["samplers"]))
{
check(o.isObject());
check(channels.size() >= 1);
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
check(samplers.size() >= 1);
}
};
template<> Animation as(const Json::Value &o) { return o; }
struct Asset {
std::optional<std::string> copyright;
std::optional<std::string> generator;
std::optional<std::string> minVersion;
std::string version;
Asset(const Json::Value &o)
: version(as<std::string>(o["version"]))
{
check(o.isObject());
if (o.isMember("copyright")) {
copyright = as<std::string>(o["copyright"]);
}
if (o.isMember("generator")) {
generator = as<std::string>(o["generator"]);
}
if (o.isMember("minVersion")) {
minVersion = as<std::string>(o["minVersion"]);
}
}
};
template<> Asset as(const Json::Value &o) { return o; }
struct BufferView {
std::size_t buffer;
std::size_t byteLength;
std::size_t byteOffset;
std::optional<std::size_t> byteStride;
std::optional<std::string> name;
enum class Target {
ARRAY_BUFFER,
ELEMENT_ARRAY_BUFFER,
};
std::optional<Target> target;
BufferView(const Json::Value &o)
: buffer(as<std::size_t>(o["buffer"]))
, byteLength(as<std::size_t>(o["byteLength"]))
, byteOffset(0)
{
check(o.isObject());
check(byteLength >= 1);
if (o.isMember("byteOffset")) {
byteOffset = as<std::size_t>(o["byteOffset"]);
check(byteOffset >= 0);
}
if (o.isMember("byteStride")) {
byteStride = as<std::size_t>(o["byteStride"]);
check(byteStride.value() >= 4);
check(byteStride.value() <= 252);
check(byteStride.value() % 4 == 0);
}
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("target")) {
static std::unordered_map<Json::UInt64, Target> map = {
{34962, Target::ARRAY_BUFFER},
{34963, Target::ELEMENT_ARRAY_BUFFER},
};
const auto &v = o["target"]; check(v.isUInt64());
target = map.at(v.asUInt64());
}
}
};
template<> BufferView as(const Json::Value &o) { return o; }
struct Buffer {
std::size_t byteLength;
std::optional<std::string> name;
std::string data;
Buffer(const Json::Value &o,
const std::function<std::string(const std::string &uri)> &resolveURI,
std::optional<std::string> &&glbData = std::nullopt)
: byteLength(as<std::size_t>(o["byteLength"]))
{
check(o.isObject());
check(byteLength >= 1);
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (glbData.has_value()) {
check(!o.isMember("uri"));
data = *std::move(glbData);
// GLB allows padding, which need not be reflected in the JSON
check(byteLength + 3 >= data.size());
check(data.size() >= byteLength);
} else {
check(o.isMember("uri"));
bool dataURI = false;
const std::string uri = as<std::string>(o["uri"]);
for (auto &prefix : std::array<std::string, 2> {
"data:application/octet-stream;base64,",
"data:application/gltf-buffer;base64,"
}) {
if (std::string_view(uri).substr(0, prefix.length()) == prefix) {
auto view = std::string_view(uri).substr(prefix.length());
check(base64_is_valid(view));
data = base64_decode(view);
dataURI = true;
break;
}
}
if (!dataURI)
data = resolveURI(uri);
check(data.size() >= byteLength);
}
data.resize(byteLength);
}
};
struct CameraOrthographic {
double xmag;
double ymag;
double zfar;
double znear;
CameraOrthographic(const Json::Value &o)
: xmag(as<double>(o["xmag"]))
, ymag(as<double>(o["ymag"]))
, zfar(as<double>(o["zfar"]))
, znear(as<double>(o["znear"]))
{
check(o.isObject());
check(zfar > 0);
check(znear >= 0);
}
};
template<> CameraOrthographic as(const Json::Value &o) { return o; }
struct CameraPerspective {
std::optional<double> aspectRatio;
double yfov;
std::optional<double> zfar;
double znear;
CameraPerspective(const Json::Value &o)
: yfov(as<double>(o["yfov"]))
, znear(as<double>(o["znear"]))
{
check(o.isObject());
if (o.isMember("aspectRatio")) {
aspectRatio = as<double>(o["aspectRatio"]);
check(aspectRatio.value() > 0);
}
check(yfov > 0);
if (o.isMember("zfar")) {
zfar = as<double>(o["zfar"]);
check(zfar.value() > 0);
}
check(znear > 0);
}
};
template<> CameraPerspective as(const Json::Value &o) { return o; }
struct Camera {
std::optional<std::string> name;
std::optional<CameraOrthographic> orthographic;
std::optional<CameraPerspective> perspective;
enum class Type {
ORTHOGRAPHIC,
PERSPECTIVE,
};
Type type;
Camera(const Json::Value &o)
{
check(o.isObject());
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("orthographic")) {
orthographic = as<CameraOrthographic>(o["orthographic"]);
}
if (o.isMember("perspective")) {
perspective = as<CameraPerspective>(o["perspective"]);
}
{
static std::unordered_map<Json::String, Type> map = {
{"orthographic", Type::ORTHOGRAPHIC},
{"perspective", Type::PERSPECTIVE},
};
const auto &v = o["type"]; check(v.isString());
type = map.at(v.asString());
}
}
};
template<> Camera as(const Json::Value &o) { return o; }
struct Image {
std::optional<std::size_t> bufferView;
enum class MimeType {
IMAGE_JPEG,
IMAGE_PNG,
};
std::optional<MimeType> mimeType;
std::optional<std::string> name;
std::optional<std::string> uri;
Image(const Json::Value &o)
{
check(o.isObject());
if (o.isMember("bufferView")) {
bufferView = as<std::size_t>(o["bufferView"]);
}
if (o.isMember("mimeType")) {
static std::unordered_map<Json::String, MimeType> map = {
{"image/jpeg", MimeType::IMAGE_JPEG},
{"image/png", MimeType::IMAGE_PNG},
};
const auto &v = o["mimeType"]; check(v.isString());
mimeType = map.at(v.asString());
}
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("uri")) {
uri = as<std::string>(o["uri"]);
}
}
};
template<> Image as(const Json::Value &o) { return o; }
struct TextureInfo {
std::size_t index;
std::size_t texCoord;
TextureInfo(const Json::Value &o)
: index(as<std::size_t>(o["index"]))
, texCoord(0)
{
check(o.isObject());
if (o.isMember("texCoord")) {
texCoord = as<std::size_t>(o["texCoord"]);
check(texCoord >= 0);
}
}
};
template<> TextureInfo as(const Json::Value &o) { return o; }
struct MaterialNormalTextureInfo {
std::size_t index;
double scale;
std::size_t texCoord;
MaterialNormalTextureInfo(const Json::Value &o)
: index(as<std::size_t>(o["index"]))
, scale(1)
, texCoord(0)
{
check(o.isObject());
if (o.isMember("scale")) {
scale = as<double>(o["scale"]);
}
if (o.isMember("texCoord")) {
texCoord = as<std::size_t>(o["texCoord"]);
}
}
};
template<> MaterialNormalTextureInfo as(const Json::Value &o) { return o; }
struct MaterialOcclusionTextureInfo {
std::size_t index;
double strength;
std::size_t texCoord;
MaterialOcclusionTextureInfo(const Json::Value &o)
: index(as<std::size_t>(o["index"]))
, strength(1)
, texCoord(0)
{
check(o.isObject());
if (o.isMember("strength")) {
strength = as<double>(o["strength"]);
check(strength >= 0);
check(strength <= 1);
}
if (o.isMember("texCoord")) {
texCoord = as<std::size_t>(o["texCoord"]);
}
}
};
template<> MaterialOcclusionTextureInfo as(const Json::Value &o) { return o; }
struct MaterialPbrMetallicRoughness {
std::array<double, 4> baseColorFactor;
std::optional<TextureInfo> baseColorTexture;
double metallicFactor;
std::optional<TextureInfo> metallicRoughnessTexture;
double roughnessFactor;
MaterialPbrMetallicRoughness(const Json::Value &o)
: baseColorFactor{1, 1, 1, 1}
, metallicFactor(1)
, roughnessFactor(1)
{
check(o.isObject());
if (o.isMember("baseColorFactor")) {
baseColorFactor = asArr<double, 4>(o["baseColorFactor"]);
for (auto v: baseColorFactor) {
check(v >= 0);
check(v <= 1);
}
}
if (o.isMember("baseColorTexture")) {
baseColorTexture = as<TextureInfo>(o["baseColorTexture"]);
}
if (o.isMember("metallicFactor")) {
metallicFactor = as<double>(o["metallicFactor"]);
check(metallicFactor >= 0);
check(metallicFactor <= 1);
}
if (o.isMember("metallicRoughnessTexture")) {
metallicRoughnessTexture = as<TextureInfo>(o["metallicRoughnessTexture"]);
}
if (o.isMember("roughnessFactor")) {
roughnessFactor = as<double>(o["roughnessFactor"]);
check(roughnessFactor >= 0);
check(roughnessFactor <= 1);
}
}
};
template<> MaterialPbrMetallicRoughness as(const Json::Value &o) { return o; }
struct Material {
double alphaCutoff;
enum class AlphaMode {
BLEND,
MASK,
OPAQUE,
};
AlphaMode alphaMode;
bool doubleSided;
std::array<double, 3> emissiveFactor;
std::optional<TextureInfo> emissiveTexture;
std::optional<std::string> name;
std::optional<MaterialNormalTextureInfo> normalTexture;
std::optional<MaterialOcclusionTextureInfo> occlusionTexture;
std::optional<MaterialPbrMetallicRoughness> pbrMetallicRoughness;
Material(const Json::Value &o)
: alphaCutoff(0.5)
, alphaMode(AlphaMode::OPAQUE)
, doubleSided(false)
, emissiveFactor{0, 0, 0}
{
check(o.isObject());
if (o.isMember("alphaCutoff")) {
alphaCutoff = as<double>(o["alphaCutoff"]);
check(alphaCutoff >= 0);
}
if (o.isMember("alphaMode")){
static std::unordered_map<Json::String, AlphaMode> map = {
{"BLEND", AlphaMode::BLEND},
{"MASK", AlphaMode::MASK},
{"OPAQUE", AlphaMode::OPAQUE},
};
const auto &v = o["alphaMode"]; check(v.isString());
alphaMode = map.at(v.asString());
}
if (o.isMember("doubleSided")) {
doubleSided = as<bool>(o["doubleSided"]);
}
if (o.isMember("emissiveFactor")) {
emissiveFactor = asArr<double, 3>(o["emissiveFactor"]);
for (const auto &v: emissiveFactor) {
check(v >= 0);
check(v <= 1);
}
}
if (o.isMember("emissiveTexture")) {
emissiveTexture = as<TextureInfo>(o["emissiveTexture"]);
}
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("normalTexture")) {
normalTexture = as<MaterialNormalTextureInfo>(o["normalTexture"]);
}
if (o.isMember("occlusionTexture")) {
occlusionTexture = as<MaterialOcclusionTextureInfo>(o["occlusionTexture"]);
}
if (o.isMember("pbrMetallicRoughness")) {
pbrMetallicRoughness = as<MaterialPbrMetallicRoughness>(o["pbrMetallicRoughness"]);
}
}
};
template<> Material as(const Json::Value &o) { return o; }
struct MeshPrimitive {
static void enumeratedProps(const Json::Value &o, const std::string &name, std::optional<std::vector<std::size_t>> &attr) {
for (std::size_t i = 0;; ++i) {
const std::string s = name + "_" + std::to_string(i);
if (!o.isMember(s)) break;
if (i == 0) {
attr = std::vector<std::size_t>();
}
attr->push_back(as<std::size_t>(o[s]));
}
}
struct Attributes {
std::optional<std::size_t> position, normal, tangent;
std::optional<std::vector<std::size_t>> texcoord, color, joints, weights;
Attributes(const Json::Value &o) {
if (o.isMember("POSITION"))
position = as<std::size_t>(o["POSITION"]);
if (o.isMember("NORMAL"))
normal = as<std::size_t>(o["NORMAL"]);
if (o.isMember("TANGENT"))
tangent = as<std::size_t>(o["TANGENT"]);
enumeratedProps(o, "TEXCOORD", texcoord);
enumeratedProps(o, "COLOR", color);
enumeratedProps(o, "JOINTS", joints);
enumeratedProps(o, "WEIGHTS", weights);
check(joints.has_value() == weights.has_value());
if (joints.has_value()) {
check(joints->size() == weights->size());
}
check(position.has_value()
|| normal.has_value()
|| tangent.has_value()
|| texcoord.has_value()
|| color.has_value()
|| joints.has_value()
|| weights.has_value());
}
};
Attributes attributes;
std::optional<std::size_t> indices;
std::optional<std::size_t> material;
enum class Mode {
POINTS,
LINES,
LINE_LOOP,
LINE_STRIP,
TRIANGLES,
TRIANGLE_STRIP,
TRIANGLE_FAN,
};
Mode mode;
struct MorphTargets {
std::optional<std::size_t> position, normal, tangent;
std::optional<std::vector<std::size_t>> texcoord, color;
MorphTargets(const Json::Value &o) {
if (o.isMember("POSITION"))
position = as<std::size_t>(o["POSITION"]);
if (o.isMember("NORMAL"))
normal = as<std::size_t>(o["NORMAL"]);
if (o.isMember("TANGENT"))
tangent = as<std::size_t>(o["TANGENT"]);
enumeratedProps(o, "TEXCOORD", texcoord);
enumeratedProps(o, "COLOR", color);
check(position.has_value()
|| normal.has_value()
|| tangent.has_value()
|| texcoord.has_value()
|| color.has_value());
}
};
std::optional<std::vector<MorphTargets>> targets;
MeshPrimitive(const Json::Value &o)
: attributes(Attributes(o["attributes"]))
, mode(Mode::TRIANGLES)
{
check(o.isObject());
if (o.isMember("indices")) {
indices = as<std::size_t>(o["indices"]);
}
if (o.isMember("material")) {
material = as<std::size_t>(o["material"]);
}
if (o.isMember("mode")) {
static std::unordered_map<Json::UInt64, Mode> map = {
{0, Mode::POINTS},
{1, Mode::LINES},
{2, Mode::LINE_LOOP},
{3, Mode::LINE_STRIP},
{4, Mode::TRIANGLES},
{5, Mode::TRIANGLE_STRIP},
{6, Mode::TRIANGLE_FAN},
};
const auto &v = o["mode"]; check(v.isUInt64());
mode = map.at(v.asUInt64());
}
if (o.isMember("targets")) {
targets = asVec<MorphTargets>(o["targets"]);
check(targets->size() >= 1);
}
}
};
template<> MeshPrimitive::MorphTargets as(const Json::Value &o) { return o; }
template<> MeshPrimitive as(const Json::Value &o) { return o; }
struct Mesh {
std::optional<std::string> name;
std::vector<MeshPrimitive> primitives;
std::optional<std::vector<double>> weights;
Mesh(const Json::Value &o)
: primitives(asVec<MeshPrimitive>(o["primitives"]))
{
check(o.isObject());
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
check(primitives.size() >= 1);
if (o.isMember("weights")) {
weights = asVec<double>(o["weights"]);
check(weights->size() >= 1);
}
}
};
template<> Mesh as(const Json::Value &o) { return o; }
struct Node {
std::optional<std::size_t> camera;
std::optional<std::vector<std::size_t>> children;
typedef std::array<double, 16> Matrix;
struct TRS {
std::array<double, 3> translation = {0, 0, 0};
std::array<double, 4> rotation = {0, 0, 0, 1};
std::array<double, 3> scale = {1, 1, 1};
};
std::variant<Matrix, TRS> transform;
std::optional<std::size_t> mesh;
std::optional<std::string> name;
std::optional<std::size_t> skin;
std::optional<std::vector<double>> weights;
Node(const Json::Value &o)
: transform(TRS{})
{
check(o.isObject());
if (o.isMember("camera")) {
camera = as<std::size_t>(o["camera"]);
}
if (o.isMember("children")) {
children = asVec<std::size_t>(o["children"]);
check(children->size() >= 1);
checkDuplicateFree(*children);
}
bool hasTRS = o.isMember("translation") || o.isMember("rotation") || o.isMember("scale");
if (o.isMember("matrix")) {
check(!hasTRS);
transform = asArr<double, 16>(o["matrix"]);
} else if (hasTRS) {
TRS trs;
if (o.isMember("translation")) {
trs.translation = asArr<double, 3>(o["translation"]);
}
if (o.isMember("rotation")) {
trs.rotation = asArr<double, 4>(o["rotation"]);
for (auto v: trs.rotation) {
check(v >= -1);
check(v <= 1);
}
}
if (o.isMember("scale")) {
trs.scale = asArr<double, 3>(o["scale"]);
}
transform = trs;
}
if (o.isMember("mesh")) {
mesh = as<std::size_t>(o["mesh"]);
}
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("skin")) {
check(mesh.has_value());
skin = as<std::size_t>(o["skin"]);
}
if (o.isMember("weights")) {
weights = asVec<double>(o["weights"]);
check(weights->size() >= 1);
}
}
};
template<> Node as(const Json::Value &o) { return o; }
struct Sampler {
enum class MagFilter {
NEAREST,
LINEAR,
};
std::optional<MagFilter> magFilter;
enum class MinFilter {
NEAREST,
LINEAR,
NEAREST_MIPMAP_NEAREST,
LINEAR_MIPMAP_NEAREST,
NEAREST_MIPMAP_LINEAR,
LINEAR_MIPMAP_LINEAR,
};
std::optional<MinFilter> minFilter;
std::optional<std::string> name;
enum class Wrap {
REPEAT,
CLAMP_TO_EDGE,
MIRRORED_REPEAT,
};
Wrap wrapS;
Wrap wrapT;
Sampler(const Json::Value &o)
: wrapS(Wrap::REPEAT)
, wrapT(Wrap::REPEAT)
{
check(o.isObject());
if (o.isMember("magFilter")) {
static std::unordered_map<Json::UInt64, MagFilter> map = {
{9728, MagFilter::NEAREST},
{9729, MagFilter::LINEAR},
};
const auto &v = o["magFilter"]; check(v.isUInt64());
magFilter = map.at(v.asUInt64());
}
if (o.isMember("minFilter")) {
static std::unordered_map<Json::UInt64, MinFilter> map = {
{9728, MinFilter::NEAREST},
{9729, MinFilter::LINEAR},
{9984, MinFilter::NEAREST_MIPMAP_NEAREST},
{9985, MinFilter::LINEAR_MIPMAP_NEAREST},
{9986, MinFilter::NEAREST_MIPMAP_LINEAR},
{9987, MinFilter::LINEAR_MIPMAP_LINEAR},
};
const auto &v = o["minFilter"]; check(v.isUInt64());
minFilter = map.at(v.asUInt64());
}
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
static std::unordered_map<Json::UInt64, Wrap> map = {
{10497, Wrap::REPEAT},
{33071, Wrap::CLAMP_TO_EDGE},
{33648, Wrap::MIRRORED_REPEAT},
};
if (o.isMember("wrapS")) {
const auto &v = o["wrapS"]; check(v.isUInt64());
wrapS = map.at(v.asUInt64());
}
if (o.isMember("wrapT")) {
const auto &v = o["wrapT"]; check(v.isUInt64());
wrapT = map.at(v.asUInt64());
}
}
};
template<> Sampler as(const Json::Value &o) { return o; }
struct Scene {
std::optional<std::string> name;
std::optional<std::vector<std::size_t>> nodes;
Scene(const Json::Value &o)
{
check(o.isObject());
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("nodes")) {
nodes = asVec<std::size_t>(o["nodes"]);
check(nodes->size() >= 1);
checkDuplicateFree(*nodes);
}
}
};
template<> Scene as(const Json::Value &o) { return o; }
struct Skin {
std::optional<std::size_t> inverseBindMatrices;
std::vector<std::size_t> joints;
std::optional<std::string> name;
std::optional<std::size_t> skeleton;
Skin(const Json::Value &o)
: joints(asVec<std::size_t>(o["joints"]))
{
check(o.isObject());
if (o.isMember("inverseBindMatrices")) {
inverseBindMatrices = as<std::size_t>(o["inverseBindMatrices"]);
}
check(joints.size() >= 1);
checkDuplicateFree(joints);
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("skeleton")) {
skeleton = as<std::size_t>(o["skeleton"]);
}
}
};
template<> Skin as(const Json::Value &o) { return o; }
struct Texture {
std::optional<std::string> name;
std::optional<std::size_t> sampler;
std::optional<std::size_t> source;
Texture(const Json::Value &o)
{
check(o.isObject());
if (o.isMember("name")) {
name = as<std::string>(o["name"]);
}
if (o.isMember("sampler")) {
sampler = as<std::size_t>(o["sampler"]);
}
if (o.isMember("source")) {
source = as<std::size_t>(o["source"]);
}
}
};
template<> Texture as(const Json::Value &o) { return o; }
using UriResolver = std::function<std::string(const std::string &uri)>;
static inline std::string uriError(const std::string &uri) {
// only base64 data URI support by default
throw std::runtime_error("unsupported URI: " + uri);
}
struct GlTF {
std::optional<std::vector<Accessor>> accessors;
std::optional<std::vector<Animation>> animations;
Asset asset;
std::optional<std::vector<BufferView>> bufferViews;
std::optional<std::vector<Buffer>> buffers;
std::optional<std::vector<Camera>> cameras;
std::optional<std::vector<std::string>> extensionsRequired;
std::optional<std::vector<std::string>> extensionsUsed;
std::optional<std::vector<Image>> images;
std::optional<std::vector<Material>> materials;
std::optional<std::vector<Mesh>> meshes;
std::optional<std::vector<Node>> nodes;
std::optional<std::vector<Sampler>> samplers;
std::optional<std::size_t> scene;
std::optional<std::vector<Scene>> scenes;
std::optional<std::vector<Skin>> skins;
std::optional<std::vector<Texture>> textures;
GlTF(const Json::Value &o,
const UriResolver &resolveUri = uriError,
std::optional<std::string> &&glbData = std::nullopt)
: asset(as<Asset>(o["asset"]))
{
check(o.isObject());
if (o.isMember("accessors")) {
accessors = asVec<Accessor>(o["accessors"]);
check(accessors->size() >= 1);
}
if (o.isMember("animations")) {
animations = asVec<Animation>(o["animations"]);
check(animations->size() >= 1);
}
if (o.isMember("bufferViews")) {
bufferViews = asVec<BufferView>(o["bufferViews"]);
check(bufferViews->size() >= 1);
}
if (o.isMember("buffers")) {
auto b = o["buffers"];
check(b.isArray());
std::vector<Buffer> bufs;
bufs.reserve(b.size());
for (Json::ArrayIndex i = 0; i < b.size(); ++i) {
bufs.emplace_back(b[i], resolveUri,
i == 0 ? std::move(glbData) : std::nullopt);
}
check(bufs.size() >= 1);
buffers = std::move(bufs);
}
if (o.isMember("cameras")) {
cameras = asVec<Camera>(o["cameras"]);
check(cameras->size() >= 1);
}
if (o.isMember("extensionsRequired")) {
extensionsRequired = asVec<std::string>(o["extensionsRequired"]);
check(extensionsRequired->size() >= 1);
checkDuplicateFree(*extensionsRequired);
}
if (o.isMember("extensionsUsed")) {
extensionsUsed = asVec<std::string>(o["extensionsUsed"]);
check(extensionsUsed->size() >= 1);
checkDuplicateFree(*extensionsUsed);
}
if (o.isMember("images")) {
images = asVec<Image>(o["images"]);
check(images->size() >= 1);
}
if (o.isMember("materials")) {
materials = asVec<Material>(o["materials"]);
check(materials->size() >= 1);
}
if (o.isMember("meshes")) {
meshes = asVec<Mesh>(o["meshes"]);
check(meshes->size() >= 1);
}
if (o.isMember("nodes")) {
nodes = asVec<Node>(o["nodes"]);
check(nodes->size() >= 1);
// Nodes must be a forest:
// 1. Each node should have indegree 0 or 1:
std::vector<std::size_t> indeg(nodes->size());
for (std::size_t i = 0; i < nodes->size(); ++i) {
auto children = nodes->at(i).children;
if (!children.has_value()) continue;
for (auto child : children.value()) {
++indeg.at(child);
}
}
for (const auto deg : indeg) {
check(deg <= 1);
}
// 2. There should be no cycles:
std::vector<bool> visited(nodes->size());
std::stack<std::size_t, std::vector<std::size_t>> toVisit;
for (std::size_t i = 0; i < nodes->size(); ++i) {
// Only start DFS in roots.
if (indeg[i] > 0)
continue;
toVisit.push(i);
do {
std::size_t j = toVisit.top();
check(!visited.at(j));
visited[j] = true;
toVisit.pop();
auto children = nodes->at(j).children;
if (!children.has_value())
continue;
for (auto child : *children) {
toVisit.push(child);
}
} while (!toVisit.empty());
}
}
if (o.isMember("samplers")) {
samplers = asVec<Sampler>(o["samplers"]);
check(samplers->size() >= 1);
}
if (o.isMember("scene")) {
scene = as<std::size_t>(o["scene"]);
}
if (o.isMember("scenes")) {
scenes = asVec<Scene>(o["scenes"]);
check(scenes->size() >= 1);
}
if (o.isMember("skins")) {
skins = asVec<Skin>(o["skins"]);
check(skins->size() >= 1);
}
if (o.isMember("textures")) {
textures = asVec<Texture>(o["textures"]);
check(textures->size() >= 1);
}
// Validation
checkForall(bufferViews, [&](const BufferView &view) {
check(buffers.has_value());
const Buffer &buf = buffers->at(view.buffer);
// Be careful because of possible integer overflows.
check(view.byteOffset < buf.byteLength);
check(view.byteLength <= buf.byteLength);
check(view.byteOffset <= buf.byteLength - view.byteLength);
});
const auto checkAccessor = [&](const auto &accessor,
std::size_t bufferView, std::size_t byteOffset, std::size_t count) {
const BufferView &view = bufferViews->at(bufferView);
if (view.byteStride.has_value())
check(*view.byteStride % accessor.componentSize() == 0);
check(byteOffset < view.byteLength);
// Use division to avoid overflows.
const auto effective_byte_stride = view.byteStride.value_or(accessor.elementSize());
check(count <= (view.byteLength - byteOffset) / effective_byte_stride);
};
checkForall(accessors, [&](const Accessor &accessor) {
if (accessor.bufferView.has_value())
checkAccessor(accessor, *accessor.bufferView, accessor.byteOffset, accessor.count);
if (accessor.sparse.has_value()) {
const auto &indices = accessor.sparse->indices;
checkAccessor(indices, indices.bufferView, indices.byteOffset, accessor.sparse->count);
const auto &values = accessor.sparse->values;
checkAccessor(accessor, values.bufferView, values.byteOffset, accessor.sparse->count);
}
});
checkForall(images, [&](const Image &image) {
checkIndex(bufferViews, image.bufferView);
});
checkForall(meshes, [&](const Mesh &mesh) {
for (const auto &primitive : mesh.primitives) {
checkIndex(accessors, primitive.indices);
checkIndex(materials, primitive.material);
checkIndex(accessors, primitive.attributes.normal);
checkIndex(accessors, primitive.attributes.position);
checkIndex(accessors, primitive.attributes.tangent);
checkForall(primitive.attributes.texcoord, [&](const std::size_t &i) {
checkIndex(accessors, i);
});
checkForall(primitive.attributes.color, [&](const std::size_t &i) {
checkIndex(accessors, i);
});
checkForall(primitive.attributes.joints, [&](const std::size_t &i) {
checkIndex(accessors, i);
});
checkForall(primitive.attributes.weights, [&](const std::size_t &i) {
checkIndex(accessors, i);
});
if (primitive.material.has_value()) {
const Material &material = materials->at(primitive.material.value());
if (material.emissiveTexture.has_value()) {
check(primitive.attributes.texcoord.has_value());
check(material.emissiveTexture->texCoord < primitive.attributes.texcoord->size());
}
if (material.normalTexture.has_value()) {
check(primitive.attributes.texcoord.has_value());
check(material.normalTexture->texCoord < primitive.attributes.texcoord->size());
}
if (material.occlusionTexture.has_value()) {
check(primitive.attributes.texcoord.has_value());
check(material.occlusionTexture->texCoord < primitive.attributes.texcoord->size());
}
}
checkForall(primitive.targets, [&](const MeshPrimitive::MorphTargets &target) {
checkIndex(accessors, target.normal);
checkIndex(accessors, target.position);
checkIndex(accessors, target.tangent);
checkForall(target.texcoord, [&](const std::size_t &i) {
checkIndex(accessors, i);
});
checkForall(target.color, [&](const std::size_t &i) {
checkIndex(accessors, i);
});
});
}
});
checkForall(nodes, [&](const Node &node) {
checkIndex(cameras, node.camera);
checkIndex(meshes, node.mesh);
checkIndex(skins, node.skin);
});
checkForall(scenes, [&](const Scene &scene) {
checkForall(scene.nodes, [&](const size_t &i) {
checkIndex(nodes, i);
});
});
checkForall(skins, [&](const Skin &skin) {
checkIndex(accessors, skin.inverseBindMatrices);
for (const std::size_t &i : skin.joints)
checkIndex(nodes, i);
checkIndex(nodes, skin.skeleton);
});
checkForall(textures, [&](const Texture &texture) {
checkIndex(samplers, texture.sampler);
checkIndex(images, texture.source);
});
checkForall(animations, [&](const Animation &animation) {
for (const auto &sampler : animation.samplers) {
checkIndex(accessors, sampler.input);
const auto &accessor = accessors->at(sampler.input);
check(accessor.type == Accessor::Type::SCALAR);
check(accessor.componentType == Accessor::ComponentType::FLOAT);
checkIndex(accessors, sampler.output);
}
for (const auto &channel : animation.channels) {
checkIndex(nodes, channel.target.node);
checkIndex(animation.samplers, channel.sampler);
}
});
checkIndex(scenes, scene);
}
};
// std::span is C++ 20, so we roll our own little struct here.
template <typename T>
struct Span {
T *ptr;
uint32_t len;
bool empty() const {
return len == 0;
}
T *end() const {
return ptr + len;
}
template <typename U>
Span<U> cast() const {
return {(U *) ptr, len};
}
};
static Json::Value readJson(Span<const char> span) {
Json::CharReaderBuilder builder;
const std::unique_ptr<Json::CharReader> reader(builder.newCharReader());
Json::Value json;
Json::String err;
if (!reader->parse(span.ptr, span.end(), &json, &err))
throw std::runtime_error(std::string("invalid JSON: ") + err);
return json;
}
inline GlTF readGlb(const char *data, std::size_t len, const UriResolver &resolveUri = uriError) {
struct Chunk {
uint32_t type;
Span<const uint8_t> span;
};
struct Stream {
Span<const uint8_t> span;
bool eof() const {
return span.empty();
}
void advance(uint32_t n) {
span.len -= n;
span.ptr += n;
}
uint32_t readUint32() {
if (span.len < 4)
throw std::runtime_error("premature EOF");
uint32_t res = 0;
for (int i = 0; i < 4; ++i)
res += span.ptr[i] << (i * 8);
advance(4);
return res;
}
Chunk readChunk() {
const auto chunkLen = readUint32();
if (chunkLen % 4 != 0)
throw std::runtime_error("chunk length must be multiple of 4");
const auto chunkType = readUint32();
auto chunkPtr = span.ptr;
if (span.len < chunkLen)
throw std::runtime_error("premature EOF");
advance(chunkLen);
return {chunkType, {chunkPtr, chunkLen}};
}
};
constexpr uint32_t MAGIC_GLTF = 0x46546C67;
constexpr uint32_t MAGIC_JSON = 0x4E4F534A;
constexpr uint32_t MAGIC_BIN = 0x004E4942;
if (len > std::numeric_limits<uint32_t>::max())
throw std::runtime_error("too large");
Stream is{{(const uint8_t *) data, static_cast<uint32_t>(len)}};
const auto magic = is.readUint32();
if (magic != MAGIC_GLTF)
throw std::runtime_error("wrong magic number");
const auto version = is.readUint32();
if (version != 2)
throw std::runtime_error("wrong version");
const auto length = is.readUint32();
if (length != len)
throw std::runtime_error("wrong length");
const auto json = is.readChunk();
if (json.type != MAGIC_JSON)
throw std::runtime_error("expected JSON chunk");
std::optional<std::string> buffer;
if (!is.eof()) {
const auto chunk = is.readChunk();
if (chunk.type == MAGIC_BIN)
buffer = std::string((const char *) chunk.span.ptr, chunk.span.len);
else if (chunk.type == MAGIC_JSON)
throw std::runtime_error("unexpected chunk");
// Ignore all other chunks. We still want to validate that
// 1. These chunks are valid;
// 2. These chunks are *not* JSON or BIN chunks
while (!is.eof()) {
const auto type = is.readChunk().type;
if (type == MAGIC_JSON || type == MAGIC_BIN)
throw std::runtime_error("unexpected chunk");
}
}
return GlTF(readJson(json.span.cast<const char>()), resolveUri, std::move(buffer));
}
inline GlTF readGlTF(const char *data, std::size_t len, const UriResolver &resolveUri = uriError) {
if (len > std::numeric_limits<uint32_t>::max())
throw std::runtime_error("too large");
return GlTF(readJson({data, static_cast<uint32_t>(len)}), resolveUri);
}
}