// Copyright (c) 2025 Quinn // Licensed under the MIT Licence. See LICENSE for details #pragma once #include #include #include #include "../util/atrb.h" #include "../util/compat/endian.h" #include "../util/types.h" /* NBT (named binary tag) is a tree data structure. Tags have a numeric type ID, name and a payload. * NBT files are a compressed `compound` tag. GZip is the compression used in most cases, * in some (rare) cases it's stored uncompressed. * A tag is an individual part of the data tree. * The first byte is the ID, followed by a `u16`, for the length of the name. * Then the name as `UTF-8`. (end tag does not contain name) */ /* specifies the NBT tag IDs. * NOTE: every type is stored as BE (big-endian) in the file. */ enum nbt_tagid { NBT_END = 0x00, // signifies the end of a compound tag NBT_I8 = 0x01, // next byte is for an 8 bit signed integer. NBT_I16 = 0x02, // next 2 bytes are for a 16 bit signed integer NBT_I32 = 0x03, // next 4 bytes are for a 32 bit signed integer NBT_I64 = 0x04, // next 8 bytes are for a 64 bit signed integer NBT_F32 = 0x05, // next 4 bytes are for a single-precision floating-point NBT_F64 = 0x06, // next 8 bytes are for a double-precision floating-point NBT_ARR_I8 = 0x07, // starts with a i32, denoting size, followed by the i8 data NBT_STR = 0x08, // starts with a u16, denoting size, followed by the UTF-8 data NBT_LIST = 0x09, // starts with an ID, followed by a 32 bit signed integer denoting the size NBT_COMPOUND = 0x0A, // compound tag, contains tags and is delimited by `NBT_END` NBT_ARR_I32 = 0x0B, // starts with a i32, denoting size, followed by the i32 data NBT_ARR_I64 = 0x0C, // starts with a i32, denoting size, followed by the u32 data }; struct nbt_path { const char **restrict pat; // specifies the NBT path components as separate elements i16 len; // specifies the length of the NBT elements }; /* gets the tag size of primitive types, returns `>0` on success, `<0` on failure */ CONST int nbt_primsize(u8 tag); /* checks whether the tag is a primitive data tag. (not recommended for filtering tags, use a `switch`) * returns a boolean value. */ CONST int nbt_isprim(u8 tag); /* returns the name length of a specific tag. `buf` is the pointer to start of the tag */ atrb_pure atrb_nonnull(1) static inline u16 nbt_namelen(const u8 *restrict buf) { assert(*buf != NBT_END); return be16toh(*(u16 *)(buf + 1)); } /* returns the (expected) pointer of the tag following this one. * `NULL` is returned if anything went wrong. */ atrb_pure atrb_nonnull(1) const u8 *nbt_nexttag(const u8 *restrict buf, u16 naml); /* gets the byte size of an NBT tag's data (excluding id and name), returns `0` upon error. */ atrb_pure size_t nbt_tagdatlen(const u8 *buf); /* processes the uncompressed `NBT` data in `buf`, with a size of `len`. */ atrb_nonnull(1, 3) int nbt_proc(struct nbt_path const *restrict paths, uint npaths, const u8 *restrict buf, size_t len);