// 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 #include #include #include #include #include "nbt.hpp" #include "error.hpp" #include "javacompat.hpp" #include "../../.endianness" #ifdef FOSSVG_ENDIAN_BIG_WORD #error "Honeywell-316-style endianness is not supported. If you feel like it should, feel free to participate in the project to maintain it." #endif #ifdef FOSSVG_ENDIAN_LITTLE_WORD #error "PDP-11-style endianness is not supported. If you feel like it should, feel free to participate in the project to maintain it." #endif #ifdef FOSSVG_ENDIAN_UNKNOWN #error "The endianness of your system could not be determined. Please set it manually. FOSS-VG is currently implemented using some endian-specific functions." #endif namespace NBT { namespace Helper { ErrorOr readInt8(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { if (currentPosition>=dataSize) return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); return ErrorOr((int8_t) data[currentPosition]); } ErrorOr readInt16(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { if (currentPosition>=dataSize) return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); if (dataSize(true, ErrorCodes::OVERRUN); return ErrorOr((int16_t) ((static_cast(data[currentPosition]) << 8) | static_cast(data[currentPosition+1]))); } ErrorOr readInt32(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { if (currentPosition>=dataSize) return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); if (dataSize(true, ErrorCodes::OVERRUN); return ErrorOr((int32_t) ( (static_cast(data[currentPosition ]) << 24) | (static_cast(data[currentPosition+1]) << 16) | (static_cast(data[currentPosition+2]) << 8) | static_cast(data[currentPosition+3]) )); } ErrorOr readInt64(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { if (currentPosition>=dataSize) return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); if (dataSize(true, ErrorCodes::OVERRUN); return ErrorOr((int64_t) ( (static_cast(data[currentPosition ]) << 56) | (static_cast(data[currentPosition+1]) << 48) | (static_cast(data[currentPosition+2]) << 40) | (static_cast(data[currentPosition+3]) << 32) | (static_cast(data[currentPosition+4]) << 24) | (static_cast(data[currentPosition+5]) << 16) | (static_cast(data[currentPosition+6]) << 8) | static_cast(data[currentPosition+7]) )); } //FIXME: endian-dependent implementations ErrorOr readFloat(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { float* value = new float; uint8_t* valueAsBytes = reinterpret_cast(value); if (dataSize<=currentPosition) return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); if (dataSize(true, ErrorCodes::OVERRUN); #ifdef FOSSVG_BIG_ENDIAN *valueAsBytes = data[currentPosition]; *(valueAsBytes+1) = data[currentPosition+1]; *(valueAsBytes+2) = data[currentPosition+2]; *(valueAsBytes+3) = data[currentPosition+3]; #else #ifdef FOSSVG_LITTLE_ENDIAN *valueAsBytes = data[currentPosition+3]; *(valueAsBytes+1) = data[currentPosition+2]; *(valueAsBytes+2) = data[currentPosition+1]; *(valueAsBytes+3) = data[currentPosition]; #else #error "NBT::Helper::readFloat: An implementation for your endianness is unavailable." #endif #endif float dereferencedValue = *value; delete value; return ErrorOr(dereferencedValue); } //FIXME: endian-dependent implementations ErrorOr readDouble(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { double* value = new double; uint8_t* valueAsBytes = reinterpret_cast(value); if (dataSize<=currentPosition) return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); if (dataSize(true, ErrorCodes::OVERRUN); #ifdef FOSSVG_BIG_ENDIAN *valueAsBytes = data[currentPosition]; *(valueAsBytes+1) = data[currentPosition+1]; *(valueAsBytes+2) = data[currentPosition+2]; *(valueAsBytes+3) = data[currentPosition+3]; *(valueAsBytes+4) = data[currentPosition+4]; *(valueAsBytes+5) = data[currentPosition+5]; *(valueAsBytes+6) = data[currentPosition+6]; *(valueAsBytes+7) = data[currentPosition+7]; #else #ifdef FOSSVG_LITTLE_ENDIAN *valueAsBytes = data[currentPosition+7]; *(valueAsBytes+1) = data[currentPosition+6]; *(valueAsBytes+2) = data[currentPosition+5]; *(valueAsBytes+3) = data[currentPosition+4]; *(valueAsBytes+4) = data[currentPosition+3]; *(valueAsBytes+5) = data[currentPosition+2]; *(valueAsBytes+6) = data[currentPosition+1]; *(valueAsBytes+7) = data[currentPosition]; #else #error "NBT::Helper::readDouble: An implementation for your endianness is unavailable." #endif #endif double dereferencedValue = *value; delete value; return ErrorOr(dereferencedValue); } ErrorOr> readInt8Array(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { ErrorOr size = readInt32(data, dataSize, currentPosition); if (size.isError) { // It is okay to pass up the error code here without further // processing because both OVERRUN and OUT_OF_RANGE errors will // still apply. return ErrorOr>(true, size.errorCode); } // get content if (currentPosition+4+size.value > dataSize) return ErrorOr>(true, ErrorCodes::OVERRUN); std::vector result = std::vector(); for (int i=0; i>(result); } ErrorOr readString(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { ErrorOr stringSize = readInt16(data, dataSize, currentPosition); if (stringSize.isError) { // It is okay to pass up the error code here without further // processing because both OVERRUN and OUT_OF_RANGE errors will // still apply. return ErrorOr(true, stringSize.errorCode); } if (currentPosition + (uint64_t) stringSize.value + 2 > dataSize) { return ErrorOr(true, ErrorCodes::OVERRUN); } ErrorOr output = JavaCompat::importJavaString(data+currentPosition, stringSize.value); if(output.isError){ return ErrorOr(true, output.errorCode); } return output; } ErrorOr> readInt32Array(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { ErrorOr size = readInt32(data, dataSize, currentPosition); if (size.isError) { // It is okay to pass up the error code here without further // processing because both OVERRUN and OUT_OF_RANGE errors will // still apply. return ErrorOr>(true, size.errorCode); } // get content if (currentPosition+4+(size.value*4) > dataSize) return ErrorOr>(true, ErrorCodes::OVERRUN); std::vector result = std::vector(); for (int i=0; i nextInt32 = readInt32(data, dataSize, currentPosition+4+(i*4)); if (nextInt32.isError) return ErrorOr>(true, nextInt32.errorCode); result.push_back(nextInt32.value); } return ErrorOr>(result); } ErrorOr> readInt64Array(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { ErrorOr size = readInt32(data, dataSize, currentPosition); if (size.isError) { // It is okay to pass up the error code here without further // processing because both OVERRUN and OUT_OF_RANGE errors will // still apply. return ErrorOr>(true, size.errorCode); } // get content if (currentPosition+4+(size.value*8) > dataSize) return ErrorOr>(true, ErrorCodes::OVERRUN); std::vector result = std::vector(); for (int i=0; i nextInt64 = readInt64(data, dataSize, currentPosition+4+(i*8)); if (nextInt64.isError) return ErrorOr>(true, nextInt64.errorCode); result.push_back(nextInt64.value); } return ErrorOr>(result); } void writeInt8(std::vector* destination, int8_t data) { destination->push_back((uint8_t) data); } //FIXME: endian dependent implementation void writeInt16(std::vector* destination, int16_t data) { int16_t* value = new int16_t; uint8_t* valueAsBytes = reinterpret_cast(value); *value = data; #ifdef FOSSVG_BIG_ENDIAN destination->push_back(*valueAsBytes); destination->push_back(*(valueAsBytes+1)); #else #ifdef FOSSVG_LITTLE_ENDIAN destination->push_back(*(valueAsBytes+1)); destination->push_back(*valueAsBytes); #else #error "NBT::Helper::writeInt16: An implementation for your endianness is unavailable." #endif #endif delete value; } //FIXME: endian dependent implementation void writeInt32(std::vector* destination, int32_t data) { int32_t* value = new int32_t; uint8_t* valueAsBytes = reinterpret_cast(value); *value = data; #ifdef FOSSVG_BIG_ENDIAN destination->push_back(*valueAsBytes); destination->push_back(*(valueAsBytes+1)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+3)); #else #ifdef FOSSVG_LITTLE_ENDIAN destination->push_back(*(valueAsBytes+3)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+1)); destination->push_back(*valueAsBytes); #else #error "NBT::Helper::writeInt16: An implementation for your endianness is unavailable." #endif #endif delete value; } //FIXME: endian dependent implementation void writeInt64(std::vector* destination, int64_t data) { int64_t* value = new int64_t; uint8_t* valueAsBytes = reinterpret_cast(value); *value = data; #ifdef FOSSVG_BIG_ENDIAN destination->push_back(*valueAsBytes); destination->push_back(*(valueAsBytes+1)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+3)); destination->push_back(*(valueAsBytes+4)); destination->push_back(*(valueAsBytes+5)); destination->push_back(*(valueAsBytes+6)); destination->push_back(*(valueAsBytes+7)); #else #ifdef FOSSVG_LITTLE_ENDIAN destination->push_back(*(valueAsBytes+7)); destination->push_back(*(valueAsBytes+6)); destination->push_back(*(valueAsBytes+5)); destination->push_back(*(valueAsBytes+4)); destination->push_back(*(valueAsBytes+3)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+1)); destination->push_back(*valueAsBytes); #else #error "NBT::Helper::writeInt16: An implementation for your endianness is unavailable." #endif #endif delete value; } //FIXME: endian-specific implementations void writeFloat(std::vector* destination, float data) { float* value = new float; uint8_t* valueAsBytes = reinterpret_cast(value); *value = data; #ifdef FOSSVG_BIG_ENDIAN destination->push_back(*valueAsBytes); destination->push_back(*(valueAsBytes+1)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+3)); #else #ifdef FOSSVG_LITTLE_ENDIAN destination->push_back(*(valueAsBytes+3)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+1)); destination->push_back(*valueAsBytes); #else #error "NBT::Helper::writeInt16: An implementation for your endianness is unavailable." #endif #endif delete value; } //FIXME: endian-specific implementations void writeDouble(std::vector* destination, double data) { double* value = new double; uint8_t* valueAsBytes = reinterpret_cast(value); *value = data; #ifdef FOSSVG_BIG_ENDIAN destination->push_back(*valueAsBytes); destination->push_back(*(valueAsBytes+1)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+3)); destination->push_back(*(valueAsBytes+4)); destination->push_back(*(valueAsBytes+5)); destination->push_back(*(valueAsBytes+6)); destination->push_back(*(valueAsBytes+7)); #else #ifdef FOSSVG_LITTLE_ENDIAN destination->push_back(*(valueAsBytes+7)); destination->push_back(*(valueAsBytes+6)); destination->push_back(*(valueAsBytes+5)); destination->push_back(*(valueAsBytes+4)); destination->push_back(*(valueAsBytes+3)); destination->push_back(*(valueAsBytes+2)); destination->push_back(*(valueAsBytes+1)); destination->push_back(*valueAsBytes); #else #error "NBT::Helper::writeInt16: An implementation for your endianness is unavailable." #endif #endif delete value; } void writeInt8Array(std::vector* destination, std::vector data) { writeInt32(destination, data.size()); for(int8_t datum: data){ destination->push_back(datum); } } void writeInt8Array(std::vector* destination, int8_t data[], uint32_t dataSize) { writeInt32(destination, dataSize); for(uint32_t i=0; i < dataSize; i++){ destination->push_back(data[i]); } } ErrorOrVoid writeString(std::vector* destination, tiny_utf8::string data) { ErrorOr> exportedString = JavaCompat::exportJavaString(data); if(exportedString.isError){ return ErrorOrVoid(true, ErrorCodes::OVERRUN); } *destination = exportedString.value; return ErrorOrVoid(); } void writeInt32Array(std::vector* destination, std::vector data) { writeInt32(destination, data.size()); for(int32_t element: data){ writeInt32(destination, element); } } void writeInt32Array(std::vector* destination, int32_t data[], uint32_t dataSize) { writeInt32(destination, dataSize); for(uint32_t i = 0; i* destination, std::vector data) { writeInt32(destination, data.size()); for(int64_t element: data){ writeInt64(destination, element); } } void writeInt64Array(std::vector* destination, int64_t data[], uint32_t dataSize) { writeInt32(destination, dataSize); for(uint32_t i = 0; i totalTagSize(uint8_t data[], uint64_t dataSize, uint64_t currentPosition) { uint8_t nextTag; if (dataSize <= currentPosition) { return ErrorOr(true, ErrorCodes::OVERRUN); } else { nextTag = data[currentPosition]; } // deal with compound tags and lists separately if (nextTag == TagType::COMPOUND || nextTag == TagType::LIST) return ErrorOr(true, ErrorCodes::NOT_YET_KNOWN); // deal with end tag before trying to access the name if (nextTag == TagType::END) return ErrorOr(1); ErrorOr nameSize = Helper::readInt16(data, dataSize, currentPosition+1); if (nameSize.isError) { return ErrorOr(true, nameSize.errorCode); } // add type byte and name size bytes uint64_t prefixSize = (uint64_t) nameSize.value + 3; switch (nextTag) { case TagType::INT8: return ErrorOr(prefixSize+1); case TagType::INT16: return ErrorOr(prefixSize+2); case TagType::INT32: return ErrorOr(prefixSize+4); case TagType::INT64: return ErrorOr(prefixSize+8); case TagType::FLOAT: return ErrorOr(prefixSize+4); case TagType::DOUBLE: return ErrorOr(prefixSize+8); case TagType::INT8_ARRAY: { ErrorOr arrayLength = Helper::readInt32(data, dataSize, currentPosition+prefixSize); if (arrayLength.isError) { return ErrorOr(true, arrayLength.errorCode); } return ErrorOr((uint64_t) arrayLength.value + prefixSize + 4); } case TagType::STRING: { ErrorOr stringSize = Helper::readInt16(data, dataSize, currentPosition+prefixSize); if (stringSize.isError) { return ErrorOr(true, stringSize.errorCode); } return ErrorOr((uint64_t) stringSize.value + prefixSize + 2); } case TagType::INT32_ARRAY: { ErrorOr arrayLength = Helper::readInt32(data, dataSize, currentPosition+prefixSize); if (arrayLength.isError) { return ErrorOr(true, arrayLength.errorCode); } return ErrorOr((uint64_t) arrayLength.value*4 + prefixSize + 4); } case TagType::INT64_ARRAY: { ErrorOr arrayLength = Helper::readInt32(data, dataSize, currentPosition+prefixSize); if (arrayLength.isError) { return ErrorOr(true, arrayLength.errorCode); } return ErrorOr((uint64_t) arrayLength.value*8 + prefixSize + 4); } // unknown tag or parsing error default: return ErrorOr(true, ErrorCodes::UNKNOWN); } } //FIXME: instead of blindly passing the error code upwards, choose // one that is applicable to the situation (for example replace // OUT_OF_RANGE with OVERRUN where appropriate) // // Length is the number of stored elements, not to be confused with size // which is the size in bytes. ErrorOr containedDataLength(uint8_t data[], uint64_t dataSize, uint64_t currentPosition){ uint8_t nextTag; if (dataSize <= currentPosition) { return ErrorOr(true, ErrorCodes::OVERRUN); } else { nextTag = data[currentPosition]; } // deal with compound tags separately if (nextTag == TagType::COMPOUND) { return ErrorOr(true, ErrorCodes::NOT_YET_KNOWN); } // deal with end tag before trying to access the name if (nextTag == TagType::END) { return ErrorOr(0); } // tags that only ever hold one value if (nextTag == TagType::INT8 || nextTag == TagType::INT16 || nextTag == TagType::INT32 || nextTag == TagType::INT64 || nextTag == TagType::FLOAT || nextTag == TagType::DOUBLE) { return ErrorOr(1); } ErrorOr nameSize = Helper::readInt16(data, dataSize, currentPosition+1); if (nameSize.isError) { return ErrorOr(true, nameSize.errorCode); } // add type byte and name size bytes uint64_t prefixSize = (uint64_t) nameSize.value + 3; switch (nextTag) { case TagType::INT8_ARRAY: { return Helper::readInt32(data, dataSize, currentPosition+prefixSize); } case TagType::STRING: { ErrorOr stringSize = Helper::readInt16(data, dataSize, currentPosition+prefixSize); if (stringSize.isError) { return ErrorOr(true, stringSize.errorCode); } return ErrorOr((int32_t) stringSize.value); } case TagType::LIST: { // add an additional byte for the contained data type return Helper::readInt32(data, dataSize, currentPosition+prefixSize+1); } case TagType::INT32_ARRAY: { return Helper::readInt32(data, dataSize, currentPosition+prefixSize); } case TagType::INT64_ARRAY: { return Helper::readInt32(data, dataSize, currentPosition+prefixSize); } default: // unknown tag or parsing error return ErrorOr(true, ErrorCodes::UNKNOWN); } } } namespace Tag { Generic::Generic() { this->type = TagType::INVALID; } Generic::~Generic() {} ErrorOrVoid Generic::toRawData([[maybe_unused]] std::vector* rawData) { return ErrorOrVoid(true, ErrorCodes::INVALID_TYPE); } uint8_t Generic::getTagType(){ return this->type; } End::End() { this->type = TagType::END; } ErrorOrVoid End::toRawData(std::vector* rawData) { rawData->push_back(this->type); return ErrorOrVoid(); } Int8::Int8() { this->type = TagType::INT8; } Int8::Int8(tiny_utf8::string name, int8_t value) { this->type = TagType::INT8; this->name = name; this->value = value; } ErrorOrVoid Int8::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeInt8(rawData, this->value); return ErrorOrVoid(); } int8_t Int8::getValue() { this->mutex.lock(); int8_t value = this->value; this->mutex.unlock(); return value; } void Int8::setValue(int8_t value) { this->mutex.lock(); this->value = value; this->mutex.unlock(); } Int16::Int16() { this->type = TagType::INT16; } Int16::Int16(tiny_utf8::string name, int16_t value) { this->type = TagType::INT16; this->name = name; this->value = value; } ErrorOrVoid Int16::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeInt16(rawData, this->value); return ErrorOrVoid(); } int16_t Int16::getValue() { this->mutex.lock(); int16_t value = this->value; this->mutex.unlock(); return value; } void Int16::setValue(int16_t value) { this->mutex.lock(); this->value = value; this->mutex.unlock(); } Int32::Int32() { this->type = TagType::INT32; } Int32::Int32(tiny_utf8::string name, int32_t value) { this->type = TagType::INT32; this->name = name; this->value = value; } ErrorOrVoid Int32::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeInt32(rawData, this->value); return ErrorOrVoid(); } int32_t Int32::getValue() { this->mutex.lock(); int32_t value = this->value; this->mutex.unlock(); return value; } void Int32::setValue(int32_t value) { this->mutex.lock(); this->value = value; this->mutex.unlock(); } Int64::Int64() { this->type = TagType::INT64; } Int64::Int64(tiny_utf8::string name, int64_t value) { this->type = TagType::INT64; this->name = name; this->value = value; } ErrorOrVoid Int64::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeInt64(rawData, this->value); return ErrorOrVoid(); } int64_t Int64::getValue() { this->mutex.lock(); int64_t value = this->value; this->mutex.unlock(); return value; } void Int64::setValue(int64_t value) { this->mutex.lock(); this->value = value; this->mutex.unlock(); } Float::Float() { this->type = TagType::FLOAT; } Float::Float(tiny_utf8::string name, float value) { this->type = TagType::FLOAT; this->name = name; this->value = value; } ErrorOrVoid Float::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeFloat(rawData, this->value); return ErrorOrVoid(); } float Float::getValue() { this->mutex.lock(); float value = this->value; this->mutex.unlock(); return value; } void Float::setValue(float value) { this->mutex.lock(); this->value = value; this->mutex.unlock(); } Double::Double() { this->type = TagType::DOUBLE; } Double::Double(tiny_utf8::string name, double value) { this->type = TagType::DOUBLE; this->name = name; this->value = value; } ErrorOrVoid Double::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeDouble(rawData, this->value); return ErrorOrVoid(); } double Double::getValue() { this->mutex.lock(); double value = this->value; this->mutex.unlock(); return value; } void Double::setValue(double value) { this->mutex.lock(); this->value = value; this->mutex.unlock(); } Int8Array::Int8Array() { this->type = TagType::INT8_ARRAY; } Int8Array::Int8Array(tiny_utf8::string name, std::vector data) { this->type = TagType::INT8_ARRAY; this->name = name; this->data = data; } Int8Array::Int8Array(tiny_utf8::string name, uint64_t length, int8_t data[]){ this->type = TagType::INT8_ARRAY; this->name = name; this->data = std::vector(data, data+length); } ErrorOrVoid Int8Array::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } Helper::writeInt8Array(rawData, this->data); return ErrorOrVoid(); } std::vector Int8Array::getData() { return this->data; } ErrorOr Int8Array::getValue(uint64_t position) { if (this->data.size() <= position) { return ErrorOr(this->data.at(position)); } return ErrorOr(true, ErrorCodes::OUT_OF_RANGE); } void Int8Array::setData(std::vector newData) { this->data = newData; } ErrorOrVoid Int8Array::setValue(uint64_t position, int8_t value) { if (this->data.size() <= position) { return ErrorOrVoid(true, ErrorCodes::OUT_OF_RANGE); } this->data[position] = value; return ErrorOrVoid(); } uint64_t Int8Array::length() { return this->data.size(); } void Int8Array::addElement(int8_t element) { this->data.push_back(element); } ErrorOrVoid Int8Array::removeElement(uint64_t position) { #pragma message("TODO: implement") //this->data.erase(position); return ErrorOrVoid(true, ErrorCodes::UNIMPLEMENTED); } String::String() { this->type = TagType::STRING; } String::String(tiny_utf8::string name, tiny_utf8::string value) { this->type = TagType::STRING; this->name = name; this->value = value; } ErrorOrVoid String::toRawData(std::vector* rawData) { rawData->push_back(this->type); if (Helper::writeString(rawData, this->name).isError) { return ErrorOrVoid(true, ErrorCodes::OVERRUN); } return Helper::writeString(rawData, this->value); } tiny_utf8::string String::getValue() { return this->value; } void String::setValue(tiny_utf8::string value) { this->value = value; } List::List() { this->type = TagType::LIST; this->containedType = TagType::INVALID; } List::List(tiny_utf8::string name, uint8_t type) { this->type = TagType::LIST; this->name = name; this->containedType = type; } // WARNING: The pointers inside the vector are automatically cleaned // up upon deletion of the List object. Do not retain a copy of them // elsewhere and especially do not delete them externally. List::List(tiny_utf8::string name, std::vector data) { this->type = TagType::LIST; this->name = name; this->tags = data; if (data.size() == 0) { this->containedType = TagType::END; } else { this->containedType = data.at(0)->getTagType(); } } List::~List() { for (uint64_t i=0; itags.size(); i++) { delete this->tags.at(i); } } ErrorOrVoid toRawData(std::vector* rawData) { #pragma message("TODO: Implement.") return ErrorOrVoid(true, ErrorCodes::UNIMPLEMENTED); } ErrorOr getElementPointer(uint64_t position) { #pragma message("TODO: Implement.") return ErrorOr(true, ErrorCodes::UNIMPLEMENTED); } ErrorOrVoid setElementPointerAt(uint64_t position, Generic*) { #pragma message("TODO: Implement.") return ErrorOrVoid(true, ErrorCodes::UNIMPLEMENTED); } ErrorOrVoid appendPointer(Generic*) { #pragma message("TODO: Implement.") return ErrorOrVoid(true, ErrorCodes::UNIMPLEMENTED); } ErrorOrVoid deleteElement(uint64_t position) { #pragma message("TODO: Implement.") return ErrorOrVoid(true, ErrorCodes::UNIMPLEMENTED); } uint64_t length() { #pragma message("TODO: Implement.") return 0; } } bool validateRawListContents(uint8_t data[], uint64_t dataSize, uint64_t initialPosition, uint64_t* processedDataSize) { // get contained data length by reading it manually because // the function that does it normally can't deal with // headerless tags // // add one byte to position to skip the type byte ErrorOr elementCount = Helper::readInt32(data, dataSize, initialPosition+1); if (elementCount.isError) { return false; } uint8_t contentType = data[initialPosition]; // contained type byte + 4 length bytes = 5 *processedDataSize = 5; switch (contentType) { case TagType::END: // everything except content has been touched at this point // and a list of end tags has no content return true; case TagType::INT8: { *processedDataSize += (uint64_t) elementCount.value; return initialPosition + *processedDataSize < dataSize; } case TagType::INT16: { *processedDataSize += (uint64_t) elementCount.value * 2; return initialPosition + *processedDataSize < dataSize; } case TagType::INT32: case TagType::FLOAT: { *processedDataSize += (uint64_t) elementCount.value * 4; return initialPosition + *processedDataSize < dataSize; } case TagType::INT64: case TagType::DOUBLE: { *processedDataSize += (uint64_t) elementCount.value * 8; return initialPosition + *processedDataSize < dataSize; } case TagType::INT8_ARRAY: { for (int32_t i=0; i> nextArray = Helper::readInt8Array(data, dataSize, initialPosition+*processedDataSize); if (nextArray.isError) { return false; } *processedDataSize += (uint64_t) nextArray.value.size(); } return true; } case TagType::STRING: { for (int32_t i=0; i nextString = Helper::readString(data, dataSize, initialPosition+*processedDataSize); if (nextString.isError) { return false; } // this cannot be an error because it just got checked int16_t nextStringSize = Helper::readInt16(data, dataSize, initialPosition+*processedDataSize).value; *processedDataSize += (uint64_t) nextStringSize + 2; } return true; } case TagType::LIST: { uint64_t* containedDataSize = new uint64_t; for (int32_t i=0; i> nextArray = Helper::readInt32Array(data, dataSize, initialPosition+*processedDataSize); if (nextArray.isError) { return false; } *processedDataSize += (uint64_t) nextArray.value.size() * 4; } return true; } case TagType::INT64_ARRAY: { for (int32_t i=0; i> nextArray = Helper::readInt64Array(data, dataSize, initialPosition+*processedDataSize); if (nextArray.isError) { return false; } *processedDataSize += (uint64_t) nextArray.value.size() * 8; } return true; } default: return false; } } bool validateRawNBTData(uint8_t data[], uint64_t dataSize, uint64_t initialPosition, uint64_t* processedDataSize){ if (initialPosition >= dataSize) { // Yes, this *could* return an instance of ErrorOr with // ErrorCodes::OVERRUN but we only care to know if what is // at that position is valid NBT which it clearly isn't according // to the original spec. if (processedDataSize!=nullptr) *processedDataSize=0; return false; // An interesting question at this point is whether we should // consider empty input valid or invalid NBT data. // // The original spec says that the top-most tag is always a // compound (or in more recent times, the Microsoft-commercialized // in-game-purchase-enabling version also allows list tags) // which automatically means that no data is invalid data... // I don't see a reason why having a different tag as the top-most // tag shouldn't be valid NBT in which case we have to face the // question whether no data is invalid or just empty NBT data. // // This seems like a reasonable extension to the spec to me and // it should be backwards compatible AFAIK. // // - BodgeMaster } uint64_t currentPosition = initialPosition; #define return if (processedDataSize!=nullptr) *processedDataSize = currentPosition-initialPosition; return while (currentPosition nextTagSize = Helper::totalTagSize(data, dataSize, currentPosition); if (nextTagSize.isError) { if (nextTagSize.errorCode == ErrorCodes::NOT_YET_KNOWN) { // attempt parsing the name ErrorOr tagName = Helper::readString(data, dataSize, currentPosition+1); if (tagName.isError) { return false; } // used seek to the start of the list's/compound’s contents // // there is no way this is an error bc it gets // checked while trying to parse the string above int16_t nameSize = Helper::readInt16(data, dataSize, currentPosition+1).value; uint64_t* processedTagSize = new uint64_t; *processedTagSize = 0; if (data[currentPosition]==TagType::LIST) { // type byte + two name size bytes = 3 if (!validateRawListContents(data, dataSize, currentPosition + (uint64_t) nameSize + 3, processedTagSize)) { delete processedTagSize; return false; } *processedTagSize += (uint64_t) nameSize + 3; } if (data[currentPosition]==TagType::COMPOUND) { // type byte + two name size bytes = 3 if (!validateRawNBTData(data, dataSize, currentPosition + (uint64_t) nameSize + 3, processedTagSize)) { delete processedTagSize; return false; } *processedTagSize += (uint64_t) nameSize + 3; } currentPosition += *processedTagSize; delete processedTagSize; continue; } return false; } if (currentPosition + nextTagSize.value > dataSize) { return false; } // recursion abort condition if (data[currentPosition]==TagType::END) { currentPosition++; return true; } // nameSize cannot be an error here bc it got checked in // nextTagSize() already int16_t nameSize = Helper::readInt16(data, dataSize, currentPosition+1).value; // attempt parsing the name // // This shouldn't matter too much here as the only error condition // the parser function deals with rn is an overrun which is already // being guarded against with // if (currentPosition + nextTagSize.value > dataSize) return false; // It might, however, turn out to be a useful check in the future. ErrorOr name = Helper::readString(data, dataSize, currentPosition+1); if (name.isError) { return false; } switch (data[currentPosition]) { case TagType::INT8: case TagType::INT16: case TagType::INT32: case TagType::INT64: case TagType::FLOAT: case TagType::DOUBLE: case TagType::INT8_ARRAY: break; case TagType::STRING: { // attempt parsing the content // // This shouldn't matter too much here as the only // error condition the parser function deals with rn is // an overrun which is already being guarded against with // if (currentPosition + nextTagSize.value > dataSize) return false; // It might, however, turn out to be a useful check // in the future. // // type byte + two name size bytes = 3 ErrorOr content = Helper::readString(data, dataSize, currentPosition+nameSize+3); if (content.isError) { return false; } break; } case TagType::INT32_ARRAY: case TagType::INT64_ARRAY: break; default: return false; } currentPosition += nextTagSize.value; } return true; #undef return } }