mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-08 07:05:40 +12:00
properly handle save ids
This commit is contained in:
parent
ee4a85aa4a
commit
9ab49617ea
4 changed files with 71 additions and 22 deletions
|
@ -1,13 +1,21 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "archive_base.hpp"
|
#include "archive_base.hpp"
|
||||||
|
|
||||||
|
#pragma pack(push, 1)
|
||||||
|
struct ExtSaveDataInfo {
|
||||||
|
u8 media_type;
|
||||||
|
u8 unknown;
|
||||||
|
u16 reserved;
|
||||||
|
u64 save_id;
|
||||||
|
};
|
||||||
|
#pragma pack(pop)
|
||||||
|
|
||||||
class ExtSaveDataArchive : public ArchiveBase {
|
class ExtSaveDataArchive : public ArchiveBase {
|
||||||
public:
|
public:
|
||||||
ExtSaveDataArchive(Memory& mem, const std::string& folder, bool isShared = false) : ArchiveBase(mem),
|
ExtSaveDataArchive(Memory& mem, const std::string& folder, u64 saveId, bool isShared = false) : ArchiveBase(mem),
|
||||||
isShared(isShared), backingFolder(folder) {}
|
archiveSaveId(saveId), isShared(isShared), backingFolder(folder) {}
|
||||||
|
|
||||||
u64 getFreeBytes() override { Helpers::panic("ExtSaveData::GetFreeBytes unimplemented"); return 0; }
|
u64 getFreeBytes() override { Helpers::panic("ExtSaveData::GetFreeBytes unimplemented"); return 0; }
|
||||||
std::string name() override { return "ExtSaveData::" + backingFolder; }
|
std::string name() override { return "ExtSaveData::" + backingFolder + "::" + std::to_string(archiveSaveId); }
|
||||||
|
|
||||||
HorizonResult createDirectory(const FSPath& path) override;
|
HorizonResult createDirectory(const FSPath& path) override;
|
||||||
HorizonResult deleteDirectory(const FSPath& path) override;
|
HorizonResult deleteDirectory(const FSPath& path) override;
|
||||||
|
@ -30,7 +38,9 @@ public:
|
||||||
// Takes in a binary ExtSaveData path, outputs a combination of the backing folder with the low and high save entries of the path
|
// Takes in a binary ExtSaveData path, outputs a combination of the backing folder with the low and high save entries of the path
|
||||||
// Used for identifying the archive format info files
|
// Used for identifying the archive format info files
|
||||||
std::string getExtSaveDataPathFromBinary(const FSPath& path) const;
|
std::string getExtSaveDataPathFromBinary(const FSPath& path) const;
|
||||||
|
std::string getExtSaveDataPath() const;
|
||||||
|
|
||||||
|
u64 archiveSaveId = 0;
|
||||||
bool isShared = false;
|
bool isShared = false;
|
||||||
std::string backingFolder; // Backing folder for the archive. Can be NAND, Gamecard or SD depending on the archive path.
|
std::string backingFolder; // Backing folder for the archive. Can be NAND, Gamecard or SD depending on the archive path.
|
||||||
};
|
};
|
||||||
|
|
|
@ -33,11 +33,13 @@ class FSService {
|
||||||
UserSaveDataArchive userSaveData1;
|
UserSaveDataArchive userSaveData1;
|
||||||
UserSaveDataArchive userSaveData2;
|
UserSaveDataArchive userSaveData2;
|
||||||
|
|
||||||
ExtSaveDataArchive extSaveData_sdmc;
|
std::unordered_map<u64, ExtSaveDataArchive> extSaveData_sdmc;
|
||||||
ExtSaveDataArchive sharedExtSaveData_nand;
|
std::unordered_map<u64, ExtSaveDataArchive> sharedExtSaveData_nand;
|
||||||
SystemSaveDataArchive systemSaveData;
|
SystemSaveDataArchive systemSaveData;
|
||||||
|
|
||||||
ArchiveBase* getArchiveFromID(u32 id, const FSPath& archivePath);
|
ArchiveBase* getArchiveFromID(u32 id, const FSPath& archivePath);
|
||||||
|
ExtSaveDataArchive* getExtArchiveFromID(u64 saveId);
|
||||||
|
ExtSaveDataArchive* getSharedExtArchiveFromID(u64 saveId);
|
||||||
Rust::Result<Handle, HorizonResult> openArchiveHandle(u32 archiveID, const FSPath& path);
|
Rust::Result<Handle, HorizonResult> openArchiveHandle(u32 archiveID, const FSPath& path);
|
||||||
Rust::Result<Handle, HorizonResult> openDirectoryHandle(ArchiveBase* archive, const FSPath& path);
|
Rust::Result<Handle, HorizonResult> openDirectoryHandle(ArchiveBase* archive, const FSPath& path);
|
||||||
std::optional<Handle> openFileHandle(ArchiveBase* archive, const FSPath& path, const FSPath& archivePath, const FilePerms& perms);
|
std::optional<Handle> openFileHandle(ArchiveBase* archive, const FSPath& path, const FSPath& archivePath, const FilePerms& perms);
|
||||||
|
@ -85,7 +87,7 @@ class FSService {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FSService(Memory& mem, Kernel& kernel, const EmulatorConfig& config)
|
FSService(Memory& mem, Kernel& kernel, const EmulatorConfig& config)
|
||||||
: mem(mem), saveData(mem), sharedExtSaveData_nand(mem, "../SharedFiles/NAND", true), extSaveData_sdmc(mem, "SDMC"), sdmc(mem),
|
: mem(mem), saveData(mem), sdmc(mem),
|
||||||
sdmcWriteOnly(mem, true), selfNcch(mem), ncch(mem), userSaveData1(mem, ArchiveID::UserSaveData1),
|
sdmcWriteOnly(mem, true), selfNcch(mem), ncch(mem), userSaveData1(mem, ArchiveID::UserSaveData1),
|
||||||
userSaveData2(mem, ArchiveID::UserSaveData2), kernel(kernel), config(config), systemSaveData(mem) {}
|
userSaveData2(mem, ArchiveID::UserSaveData2), kernel(kernel), config(config), systemSaveData(mem) {}
|
||||||
|
|
||||||
|
|
|
@ -201,7 +201,7 @@ HorizonResult ExtSaveDataArchive::deleteDirectoryRecursively(const FSPath& path)
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExtSaveDataArchive::format(const FSPath& path, const FormatInfo& info) {
|
void ExtSaveDataArchive::format(const FSPath& path, const FormatInfo& info) {
|
||||||
const fs::path saveDataPath = IOFile::getAppData() / backingFolder;
|
const fs::path saveDataPath = IOFile::getAppData() / getExtSaveDataPath();
|
||||||
const fs::path formatInfoPath = getFormatInfoPath(path);
|
const fs::path formatInfoPath = getFormatInfoPath(path);
|
||||||
|
|
||||||
// Delete all contents by deleting the directory then recreating it
|
// Delete all contents by deleting the directory then recreating it
|
||||||
|
@ -222,7 +222,7 @@ void ExtSaveDataArchive::format(const FSPath& path, const FormatInfo& info) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExtSaveDataArchive::clear(const FSPath& path) const {
|
void ExtSaveDataArchive::clear(const FSPath& path) const {
|
||||||
const fs::path saveDataPath = IOFile::getAppData() / backingFolder;
|
const fs::path saveDataPath = IOFile::getAppData() / getExtSaveDataPath();
|
||||||
const fs::path formatInfoPath = getFormatInfoPath(path);
|
const fs::path formatInfoPath = getFormatInfoPath(path);
|
||||||
|
|
||||||
fs::remove_all(saveDataPath);
|
fs::remove_all(saveDataPath);
|
||||||
|
@ -234,7 +234,7 @@ std::filesystem::path ExtSaveDataArchive::getFormatInfoPath(const FSPath& path)
|
||||||
}
|
}
|
||||||
|
|
||||||
std::filesystem::path ExtSaveDataArchive::getUserDataPath() const {
|
std::filesystem::path ExtSaveDataArchive::getUserDataPath() const {
|
||||||
fs::path p = IOFile::getAppData() / backingFolder;
|
fs::path p = IOFile::getAppData() / getExtSaveDataPath();
|
||||||
if (!isShared) { // todo: "boss"?
|
if (!isShared) { // todo: "boss"?
|
||||||
p /= "user";
|
p /= "user";
|
||||||
}
|
}
|
||||||
|
@ -270,13 +270,12 @@ std::string ExtSaveDataArchive::getExtSaveDataPathFromBinary(const FSPath& path)
|
||||||
Helpers::panic("GetExtSaveDataPathFromBinary called without a Binary FSPath!");
|
Helpers::panic("GetExtSaveDataPathFromBinary called without a Binary FSPath!");
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Remove punning here
|
const ExtSaveDataInfo info = *reinterpret_cast<const ExtSaveDataInfo*>(&path.binary[0]);
|
||||||
const u32 mediaType = *(u32*)&path.binary[0];
|
return backingFolder + "_" + std::to_string(info.save_id);
|
||||||
const u32 saveLow = *(u32*)&path.binary[4];
|
}
|
||||||
const u32 saveHigh = *(u32*)&path.binary[8];
|
|
||||||
|
|
||||||
// TODO: Should the media type be used here, using it just to be safe.
|
std::string ExtSaveDataArchive::getExtSaveDataPath() const {
|
||||||
return backingFolder + std::to_string(mediaType) + "_" + std::to_string(saveLow) + "_" + std::to_string(saveHigh);
|
return backingFolder + "/" + std::to_string(archiveSaveId);
|
||||||
}
|
}
|
||||||
|
|
||||||
Rust::Result<ArchiveBase*, HorizonResult> ExtSaveDataArchive::openArchive(const FSPath& path) {
|
Rust::Result<ArchiveBase*, HorizonResult> ExtSaveDataArchive::openArchive(const FSPath& path) {
|
||||||
|
|
|
@ -84,17 +84,55 @@ void FSService::initializeFilesystem() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ExtSaveDataArchive* FSService::getExtArchiveFromID(u64 saveId) {
|
||||||
|
// todo: should we have title id in here?
|
||||||
|
if (const auto entry = extSaveData_sdmc.find(saveId); entry == extSaveData_sdmc.end()) {
|
||||||
|
extSaveData_sdmc.emplace(saveId, ExtSaveDataArchive(mem, "SDMC", saveId));
|
||||||
|
}
|
||||||
|
return &extSaveData_sdmc.at(saveId);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ExtSaveDataArchive* FSService::getSharedExtArchiveFromID(u64 saveId) {
|
||||||
|
// todo: should we have title id in here?
|
||||||
|
if (const auto entry = sharedExtSaveData_nand.find(saveId); entry == sharedExtSaveData_nand.end()) {
|
||||||
|
sharedExtSaveData_nand.emplace(saveId, ExtSaveDataArchive(mem, "../SharedFiles/NAND", saveId, true));
|
||||||
|
}
|
||||||
|
return &sharedExtSaveData_nand.at(saveId);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
ArchiveBase* FSService::getArchiveFromID(u32 id, const FSPath& archivePath) {
|
ArchiveBase* FSService::getArchiveFromID(u32 id, const FSPath& archivePath) {
|
||||||
switch (id) {
|
switch (id) {
|
||||||
case ArchiveID::SelfNCCH: return &selfNcch;
|
case ArchiveID::SelfNCCH: return &selfNcch;
|
||||||
case ArchiveID::SaveData: return &saveData;
|
case ArchiveID::SaveData: return &saveData;
|
||||||
case ArchiveID::UserSaveData2: return &userSaveData2;
|
case ArchiveID::UserSaveData2: return &userSaveData2;
|
||||||
|
|
||||||
case ArchiveID::ExtSaveData:
|
case ArchiveID::ExtSaveData: {
|
||||||
return &extSaveData_sdmc;
|
const ExtSaveDataInfo info = *reinterpret_cast<const ExtSaveDataInfo*>(&archivePath.binary[0]);
|
||||||
|
switch(info.media_type) {
|
||||||
|
case MediaType::SD: {
|
||||||
|
return getExtArchiveFromID(info.save_id);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
Helpers::panic("Unknown archive media type. ID: %d\n", info.media_type);
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
case ArchiveID::SharedExtSaveData:
|
case ArchiveID::SharedExtSaveData: {
|
||||||
return &sharedExtSaveData_nand;
|
const ExtSaveDataInfo info = *reinterpret_cast<const ExtSaveDataInfo*>(&archivePath.binary[0]);
|
||||||
|
switch(info.media_type) {
|
||||||
|
case MediaType::NAND: {
|
||||||
|
return getExtArchiveFromID(info.save_id);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
Helpers::panic("Unknown archive media type. ID: %d\n", info.media_type);
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
case ArchiveID::SystemSaveData: return &systemSaveData;
|
case ArchiveID::SystemSaveData: return &systemSaveData;
|
||||||
case ArchiveID::SDMC: return &sdmc;
|
case ArchiveID::SDMC: return &sdmc;
|
||||||
|
@ -563,8 +601,8 @@ void FSService::createExtSaveData(u32 messagePointer) {
|
||||||
|
|
||||||
ExtSaveDataArchive* selected = nullptr;
|
ExtSaveDataArchive* selected = nullptr;
|
||||||
switch(mediaType) {
|
switch(mediaType) {
|
||||||
case MediaType::NAND: selected = &sharedExtSaveData_nand; break;
|
case MediaType::NAND: selected = getSharedExtArchiveFromID(saveID); break;
|
||||||
case MediaType::SD: selected = &extSaveData_sdmc; break;
|
case MediaType::SD: selected = getExtArchiveFromID(saveID); break;
|
||||||
default: Helpers::warn("FS::CreateExtSaveData - Unhandled ExtSaveData MediaType %d", static_cast<s32>(mediaType)); break;
|
default: Helpers::warn("FS::CreateExtSaveData - Unhandled ExtSaveData MediaType %d", static_cast<s32>(mediaType)); break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue