#include "fs/archive_save_data.hpp" #include #include namespace fs = std::filesystem; HorizonResult SaveDataArchive::createFile(const FSPath& path, u64 size) { if (path.isUTF16()) { if (!isPathSafe(path)) Helpers::panic("Unsafe path in SaveData::CreateFile"); fs::path p = IOFile::getAppData() / "SaveData"; p += fs::path(path.utf16_string).make_preferred(); if (fs::exists(p)) { return Result::FS::AlreadyExists; } IOFile file(p.string().c_str(), "wb"); // If the size is 0, leave the file empty and return success if (size == 0) { file.close(); return Result::Success; } // If it is not empty, seek to size - 1 and write a 0 to create a file of size "size" else if (file.seek(size - 1, SEEK_SET) && file.writeBytes("", 1).second == 1) { file.close(); return Result::Success; } file.close(); return Result::FS::FileTooLarge; } Helpers::panic("SaveDataArchive::CreateFile: Failed"); return Result::Success; } HorizonResult SaveDataArchive::createDirectory(const FSPath& path) { if (path.isUTF16()) { if (!isPathSafe(path)) { Helpers::panic("Unsafe path in SaveData::OpenFile"); } fs::path p = IOFile::getAppData() / "SaveData"; p += fs::path(path.utf16_string).make_preferred(); if (fs::is_directory(p)) { return Result::FS::AlreadyExists; } if (fs::is_regular_file(p)) { Helpers::panic("File path passed to SaveData::CreateDirectory"); } bool success = fs::create_directory(p); return success ? Result::Success : Result::FS::UnexpectedFileOrDir; } else { Helpers::panic("Unimplemented SaveData::CreateDirectory"); } } HorizonResult SaveDataArchive::deleteFile(const FSPath& path) { if (path.isUTF16()) { if (!isPathSafe(path)) { Helpers::panic("Unsafe path in SaveData::DeleteFile"); } fs::path p = IOFile::getAppData() / "SaveData"; p += fs::path(path.utf16_string).make_preferred(); if (fs::is_directory(p)) { Helpers::panic("SaveData::DeleteFile: Tried to delete directory"); } if (!fs::is_regular_file(p)) { return Result::FS::FileNotFoundAlt; } std::error_code ec; bool success = fs::remove(p, ec); // It might still be possible for fs::remove to fail, if there's eg an open handle to a file being deleted // In this case, print a warning, but still return success for now if (!success) { Helpers::warn("SaveData::DeleteFile: fs::remove failed\n"); } return Result::Success; } Helpers::panic("SaveDataArchive::DeleteFile: Unknown path type"); return Result::Success; } FileDescriptor SaveDataArchive::openFile(const FSPath& path, const FilePerms& perms) { if (path.isUTF16()) { if (!isPathSafe(path)) { Helpers::panic("Unsafe path in SaveData::OpenFile"); } if (perms.raw == 0 || (perms.create() && !perms.write())) { Helpers::panic("[SaveData] Unsupported flags for OpenFile"); } fs::path p = IOFile::getAppData() / "SaveData"; p += fs::path(path.utf16_string).make_preferred(); const char* permString = perms.write() ? "r+b" : "rb"; if (fs::exists(p)) { // Return file descriptor if the file exists IOFile file(p.string().c_str(), permString); return file.isOpen() ? file.getHandle() : FileError; } else { // If the file is not found, create it if the create flag is on if (perms.create()) { IOFile file(p.string().c_str(), "wb"); // Create file file.close(); // Close it file.open(p.string().c_str(), permString); // Reopen with proper perms return file.isOpen() ? file.getHandle() : FileError; } else { return FileError; } } } Helpers::panic("SaveDataArchive::OpenFile: Failed"); return FileError; } Rust::Result SaveDataArchive::openDirectory(const FSPath& path) { if (path.isUTF16()) { if (!isPathSafe(path)) { Helpers::panic("Unsafe path in SaveData::OpenDirectory"); } fs::path p = IOFile::getAppData() / "SaveData"; p += fs::path(path.utf16_string).make_preferred(); if (fs::is_regular_file(p)) { printf("SaveData: OpenDirectory used with a file path"); return Err(Result::FS::UnexpectedFileOrDir); } if (fs::is_directory(p)) { return Ok(DirectorySession(this, p)); } else { return Err(Result::FS::FileNotFoundAlt); } } Helpers::panic("SaveDataArchive::OpenDirectory: Unimplemented path type"); return Err(Result::Success); } Rust::Result SaveDataArchive::getFormatInfo(const FSPath& path) { const fs::path formatInfoPath = getFormatInfoPath(); IOFile file(formatInfoPath, "rb"); // If the file failed to open somehow, we return that the archive is not formatted if (!file.isOpen()) { return Err(Result::FS::NotFormatted); } FormatInfo ret; auto [success, bytesRead] = file.readBytes(&ret, sizeof(FormatInfo)); file.close(); if (!success || bytesRead != sizeof(FormatInfo)) { Helpers::warn("SaveData::GetFormatInfo: Format file exists but was not properly read into the FormatInfo struct"); return Err(Result::FS::NotFormatted); } return Ok(ret); } void SaveDataArchive::format(const FSPath& path, const ArchiveBase::FormatInfo& info) { const fs::path saveDataPath = IOFile::getAppData() / "SaveData"; const fs::path formatInfoPath = getFormatInfoPath(); // Delete all contents by deleting the directory then recreating it fs::remove_all(saveDataPath); fs::create_directories(saveDataPath); // Write format info on disk IOFile file(formatInfoPath, "wb"); file.writeBytes(&info, sizeof(info)); file.flush(); file.close(); } Rust::Result SaveDataArchive::openArchive(const FSPath& path) { if (!path.isEmptyType()) { Helpers::panic("Unimplemented path type for SaveData archive: %d\n", path.type); return Err(Result::FS::NotFoundInvalid); } const fs::path formatInfoPath = getFormatInfoPath(); // Format info not found so the archive is not formatted if (!fs::is_regular_file(formatInfoPath)) { return Err(Result::FS::NotFormatted); } return Ok((ArchiveBase*)this); } std::optional SaveDataArchive::readFile(FileSession* file, u64 offset, u32 size, u32 dataPointer) { Helpers::panic("Unimplemented SaveData::ReadFile"); return 0; }