// Copyright 2022, FOSS-VG Developers and Contributers // // This program is free software: you can redistribute it and/or modify it // under the terms of the GNU Affero General Public License as published // by the Free Software Foundation, version 3. // // This program 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // version 3 along with this program. // If not, see https://www.gnu.org/licenses/agpl-3.0.en.html // information taken from https://wiki.vg/NBT // This is an attempt at creating a uniform model for all NBT tags to allow for a uniform interface based on subclassing a single NBT tag super class. // NBT tags have a type, optionally a name which consists of the name size and the name string, optionally content type, and optionally a payload which can consist of optionally content type, optionally a content size, // and the stored content. The format in which they are stored is as follows: . All numbers are stored in big endian representation. // All tag types: // generic representation: Tag(uint8:tag_type, String:name, uint16:name_size, byte[]:content, int32:size) // None (compound end): Tag( 0, "", 0, None, 0) => used to determine the end of a compound tag, only the type gets stored // int8: Tag( 1, String:name, uint16:name_size, int8:content, 1) => a single signed byte, size not stored // int16: Tag( 2, String:name, uint16:name_size, int16:content, 2) => 16 bit signed integer, size not stored // int32: Tag( 3, String:name, uint16:name_size, int32:content, 4) => 32 bit signed integer, size not stored // int64: Tag( 4, String:name, uint16:name_size, int64:content, 8) => 64 bit signed integer, size not stored // float: Tag( 5, String:name, uint16:name_size, float:content, 4) => 32 bit IEEE754 floating point number, size not stored // double: Tag( 6, String:name, uint16:name_size, double:content, 8) => 64 bit IEEE754 floating point number, size not stored // int8[]: Tag( 7, String:name, uint16:name_size, int8[]:content, int32:size) => content stored prefixed with size // String: Tag( 8, String:name, uint16:name_size, String:content, uint16:size) => Java style modified UTF-8 string, content stored prefixed with size // Tag[] (list): Tag( 9, String:name, uint16:name_size, Tag[]:content, int32:size) => list of tags of the same type with tag type and name information omitted prefixed by (in order) content type and size // Tag[] (compound): Tag(10, String:name, uint16:name_size, Tag[]:content, int32:size) => list of tags, last tag is always an end tag, size not stored // int32[]: Tag(11, String:name, uint16:name_size, int32[]:content,int32:size) => list of 32 bit signed integers prefixed with its size, endianness not verified at this point // int64[]: Tag(12, String:name, uint16:name_size, int64[]:content,int32:size) => list of 64 bit signed integers prefixed with its size, endianness not verified at this point #pragma once #include #include #include #include #include "error.hpp" namespace NBT { namespace Helper { ErrorOr readInt8(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr readInt16(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr readInt32(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr readInt64(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr readFloat(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr readDouble(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr> readInt8Array(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr readString(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr> readInt32Array(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr> readInt64Array(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); void writeInt8(std::vector* destination, int8_t data); void writeInt16(std::vector* destination, int16_t data); void writeInt32(std::vector* destination, int32_t data); void writeInt64(std::vector* destination, int64_t data); void writeFloat(std::vector* destination, float data); void writeDouble(std::vector* destination, double data); void writeInt8Array(std::vector* destination, std::vector data); void writeInt8Array(std::vector* destination, int8_t data[], uint32_t dataSize); ErrorOrVoid writeString(std::vector* destination, tiny_utf8::string data); void writeInt32Array(std::vector* destination, std::vector data); void writeInt32Array(std::vector* destination, int32_t data[], uint32_t dataSize); void writeInt64Array(std::vector* destination, std::vector data); void writeInt64Array(std::vector* destination, int64_t data[], uint32_t dataSize); ErrorOr totalTagSize(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); ErrorOr containedDataLength(uint8_t data[], uint64_t dataSize, uint64_t currentPosition); } namespace TagType { const uint8_t END = 0; const uint8_t INT8 = 1; const uint8_t INT16 = 2; const uint8_t INT32 = 3; const uint8_t INT64 = 4; const uint8_t FLOAT = 5; const uint8_t DOUBLE = 6; const uint8_t INT8_ARRAY = 7; const uint8_t STRING = 8; const uint8_t LIST = 9; const uint8_t COMPOUND = 10; const uint8_t INT32_ARRAY= 11; const uint8_t INT64_ARRAY= 12; // This is a workaround that's not part of the spec. const uint8_t INVALID = 255; } namespace Tag { class Generic { protected: std::mutex mutex; uint8_t type; public: tiny_utf8::string name; Generic(); virtual ~Generic(); virtual ErrorOrVoid toRawData(std::vector* rawData); uint8_t getTagType(); }; class End: public Generic { public: End(); ErrorOrVoid toRawData(std::vector* rawData) override; }; class Int8: public Generic { private: int8_t value; public: Int8(); Int8(tiny_utf8::string name, int8_t value); ErrorOrVoid toRawData(std::vector* rawData) override; int8_t getValue(); void setValue(int8_t value); }; class Int16: public Generic { private: int16_t value; public: Int16(); Int16(tiny_utf8::string name, int16_t value); ErrorOrVoid toRawData(std::vector* rawData) override; int16_t getValue(); void setValue(int16_t value); }; class Int32: public Generic { private: int32_t value; public: Int32(); Int32(tiny_utf8::string name, int32_t value); ErrorOrVoid toRawData(std::vector* rawData) override; int32_t getValue(); void setValue(int32_t value); }; class Int64: public Generic { private: int64_t value; public: Int64(); Int64(tiny_utf8::string name, int64_t value); ErrorOrVoid toRawData(std::vector* rawData) override; int64_t getValue(); void setValue(int64_t value); }; class Float: public Generic { private: float value; public: Float(); Float(tiny_utf8::string name, float value); ErrorOrVoid toRawData(std::vector* rawData) override; float getValue(); void setValue(float value); }; class Double: public Generic { private: double value; public: Double(); Double(tiny_utf8::string name, double value); ErrorOrVoid toRawData(std::vector* rawData) override; double getValue(); void setValue(double value); }; } bool validateRawNBTData(uint8_t data[], uint64_t dataSize, uint64_t initialPosition=0, uint64_t* processedDataSize=nullptr); }