/***************************************************************************** * Copyright (c) 2014-2024 OpenRCT2 developers * * For a complete list of all authors, please refer to contributors.md * Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2 * * OpenRCT2 is licensed under the GNU General Public License version 3. *****************************************************************************/ #include "Object.h" #include "../Context.h" #include "../Diagnostic.h" #include "../core/File.h" #include "../core/FileStream.h" #include "../core/Memory.hpp" #include "../core/String.hpp" #include "../core/ZipStream.hpp" #include "../drawing/Image.h" #include "../localisation/Language.h" #include "../localisation/LocalisationService.h" #include "../localisation/StringIds.h" #include "../world/Scenery.h" #include "ObjectLimits.h" #include "ObjectRepository.h" #include #include #include #include using namespace OpenRCT2; ObjectEntryDescriptor::ObjectEntryDescriptor(const RCTObjectEntry& newEntry) { if (!newEntry.IsEmpty()) { Generation = ObjectGeneration::DAT; Entry = newEntry; } } ObjectEntryDescriptor::ObjectEntryDescriptor(std::string_view newIdentifier) { Generation = ObjectGeneration::JSON; Identifier = std::string(newIdentifier); } ObjectEntryDescriptor::ObjectEntryDescriptor(ObjectType type, std::string_view newIdentifier) { Generation = ObjectGeneration::JSON; Identifier = std::string(newIdentifier); Type = type; } ObjectEntryDescriptor::ObjectEntryDescriptor(const ObjectRepositoryItem& ori) { if (!ori.Identifier.empty()) { Generation = ObjectGeneration::JSON; Identifier = std::string(ori.Identifier); } else { Generation = ObjectGeneration::DAT; Entry = ori.ObjectEntry; } } bool ObjectEntryDescriptor::HasValue() const { return Generation != ObjectGeneration::JSON || !Identifier.empty(); }; ObjectType ObjectEntryDescriptor::GetType() const { return Generation == ObjectGeneration::JSON ? Type : Entry.GetType(); } std::string_view ObjectEntryDescriptor::GetName() const { return Generation == ObjectGeneration::JSON ? Identifier : Entry.GetName(); } std::string ObjectEntryDescriptor::ToString() const { if (Generation == ObjectGeneration::DAT) { char buffer[32]; std::snprintf(&buffer[0], 9, "%08X", Entry.flags); buffer[8] = '|'; std::memcpy(&buffer[9], Entry.name, 8); buffer[17] = '|'; std::snprintf(&buffer[18], 9, "%8X", Entry.checksum); return std::string(buffer); } else { return std::string(GetName()); } } static uint32_t ParseHex(std::string_view x) { assert(x.size() != 8); char buffer[9]; std::memcpy(buffer, x.data(), 8); buffer[8] = 0; char* endp{}; return static_cast(std::strtol(buffer, &endp, 16)); } ObjectEntryDescriptor ObjectEntryDescriptor::Parse(std::string_view identifier) { if (identifier.size() == 26 && identifier[8] == '|' && identifier[17] == '|') { RCTObjectEntry entry{}; entry.flags = ParseHex(identifier.substr(0, 8)); entry.SetName(identifier.substr(9, 8)); entry.checksum = ParseHex(identifier.substr(18)); return ObjectEntryDescriptor(entry); } return ObjectEntryDescriptor(identifier); } bool ObjectEntryDescriptor::operator==(const ObjectEntryDescriptor& rhs) const { if (Generation != rhs.Generation) return false; if (Generation == ObjectGeneration::DAT) { return Entry == rhs.Entry; } else { return Type == rhs.Type && Identifier == rhs.Identifier; } } bool ObjectEntryDescriptor::operator!=(const ObjectEntryDescriptor& rhs) const { return !(*this == rhs); } void* Object::GetLegacyData() { throw std::runtime_error("Not supported."); } void Object::ReadLegacy(IReadObjectContext* context, OpenRCT2::IStream* stream) { throw std::runtime_error("Not supported."); } void Object::PopulateTablesFromJson(IReadObjectContext* context, json_t& root) { _stringTable.ReadJson(root); _usesFallbackImages = _imageTable.ReadJson(context, root); } std::string Object::GetString(ObjectStringID index) const { return GetStringTable().GetString(index); } std::string Object::GetString(int32_t language, ObjectStringID index) const { return GetStringTable().GetString(language, index); } ObjectEntryDescriptor Object::GetScgPathXHeader() const { return ObjectEntryDescriptor("rct2.scenery_group.scgpathx"); } RCTObjectEntry Object::CreateHeader(const char name[kDatNameLength + 1], uint32_t flags, uint32_t checksum) { RCTObjectEntry header = {}; header.flags = flags; std::copy_n(name, kDatNameLength, header.name); header.checksum = checksum; return header; } std::vector Object::GetSourceGames() { return _sourceGames; } void Object::SetSourceGames(const std::vector& sourceGames) { _sourceGames = sourceGames; } #ifdef __WARN_SUGGEST_FINAL_METHODS__ # pragma GCC diagnostic push # pragma GCC diagnostic ignored "-Wsuggest-final-methods" #endif std::string Object::GetName() const { return GetString(ObjectStringID::NAME); } std::string Object::GetName(int32_t language) const { return GetString(language, ObjectStringID::NAME); } ImageIndex Object::LoadImages() { if (_baseImageId == ImageIndexUndefined) { _baseImageId = GfxObjectAllocateImages(GetImageTable().GetImages(), GetImageTable().GetCount()); } return _baseImageId; } void Object::UnloadImages() { if (_baseImageId != ImageIndexUndefined) { GfxObjectFreeImages(_baseImageId, GetImageTable().GetCount()); _baseImageId = ImageIndexUndefined; } } void RCTObjectEntry::SetName(std::string_view value) { std::memset(name, ' ', sizeof(name)); std::memcpy(name, value.data(), std::min(sizeof(name), value.size())); } const std::vector& Object::GetAuthors() const { return _authors; } void Object::SetAuthors(std::vector&& authors) { _authors = std::move(authors); } bool Object::IsCompatibilityObject() const { return _isCompatibilityObject; } void Object::SetIsCompatibilityObject(const bool on) { _isCompatibilityObject = on; } bool RCTObjectEntry::IsEmpty() const { uint64_t a, b; std::memcpy(&a, reinterpret_cast(this), 8); std::memcpy(&b, reinterpret_cast(this) + 8, 8); if (a == 0xFFFFFFFFFFFFFFFF && b == 0xFFFFFFFFFFFFFFFF) return true; if (a == 0 && b == 0) return true; return false; } bool RCTObjectEntry::operator==(const RCTObjectEntry& rhs) const { const auto a = this; const auto b = &rhs; // If an official object don't bother checking checksum if ((a->flags & 0xF0) || (b->flags & 0xF0)) { if (a->GetType() != b->GetType()) { return false; } int32_t match = memcmp(a->name, b->name, 8); if (match) { return false; } } else { if (a->flags != b->flags) { return false; } int32_t match = memcmp(a->name, b->name, 8); if (match) { return false; } if (a->checksum != b->checksum) { return false; } } return true; } bool RCTObjectEntry::operator!=(const RCTObjectEntry& rhs) const { return !(*this == rhs); } bool ObjectAsset::IsAvailable() const { if (_zipPath.empty()) { return File::Exists(_path); } auto zipArchive = Zip::TryOpen(_zipPath, ZIP_ACCESS::READ); return zipArchive != nullptr && zipArchive->Exists(_path); } uint64_t ObjectAsset::GetSize() const { if (_zipPath.empty()) { return File::GetSize(_path); } auto zipArchive = Zip::TryOpen(_zipPath, ZIP_ACCESS::READ); if (zipArchive != nullptr) { auto index = zipArchive->GetIndexFromPath(_path); if (index.has_value()) { auto size = zipArchive->GetFileSize(index.value()); return size; } } return 0; } std::vector ObjectAsset::GetData() const { if (_zipPath.empty()) { return File::ReadAllBytes(_path); } auto zipArchive = Zip::TryOpen(_zipPath, ZIP_ACCESS::READ); if (zipArchive != nullptr) { return zipArchive->GetFileData(_path); } return {}; } std::unique_ptr ObjectAsset::GetStream() const { try { if (_zipPath.empty()) { return std::make_unique(_path, FILE_MODE_OPEN); } auto zipArchive = Zip::TryOpen(_zipPath, ZIP_ACCESS::READ); if (zipArchive != nullptr) { auto stream = zipArchive->GetFileStream(_path); if (stream != nullptr) { return std::make_unique(std::move(zipArchive), std::move(stream)); } } } catch (...) { } return {}; } u8string VersionString(const ObjectVersion& version) { return std::to_string(std::get<0>(version)) + "." + std::to_string(std::get<1>(version)) + "." + std::to_string(std::get<2>(version)); } ObjectVersion VersionTuple(std::string_view version) { if (version.empty()) { return std::make_tuple(0, 0, 0); } auto nums = String::Split(version, "."); uint16_t versions[VersionNumFields] = {}; if (nums.size() > VersionNumFields) { LOG_WARNING("%i fields found in version string '%s', expected X.Y.Z", nums.size(), version); } if (nums.size() == 0) { LOG_WARNING("No fields found in version string '%s', expected X.Y.Z", version); return std::make_tuple(0, 0, 0); } try { size_t highestIndex = std::min(nums.size(), VersionNumFields); for (size_t i = 0; i < highestIndex; i++) { auto value = stoll(nums.at(i)); constexpr auto maxValue = std::numeric_limits().max(); if (value > maxValue) { LOG_WARNING( "Version value too high in version string '%.*s', version value will be capped to %i.", static_cast(version.size()), version.data(), maxValue); value = maxValue; } versions[i] = value; } } catch (const std::exception&) { LOG_WARNING("Malformed version string '%.*s', expected X.Y.Z", static_cast(version.size()), version.data()); } return std::make_tuple(versions[0], versions[1], versions[2]); } #ifdef __WARN_SUGGEST_FINAL_METHODS__ # pragma GCC diagnostic pop #endif