mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-16 02:59:11 +01:00
8317a66ea5
Since all FS access will go through the new FS interface (PR #6421) in order to keep track of metadata properly, there is no need to return absolute paths anymore. In fact, returning host paths is a roadblock to using the FS interface. This starts the migration work by adding a way to get paths that are relative to the Wii NAND instead of always getting absolute paths on the host FS. To prepare for future changes, this commit also makes returned paths canonical by removing the trailing slash when it's unneeded. Eventually, once everything has been migrated to the new interface, we can remove the "from" parameter.
149 lines
4.5 KiB
C++
149 lines
4.5 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include "Common/NandPaths.h"
|
|
|
|
#include <algorithm>
|
|
#include <string>
|
|
#include <unordered_set>
|
|
#include <vector>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/FileUtil.h"
|
|
#include "Common/StringUtil.h"
|
|
|
|
namespace Common
|
|
{
|
|
std::string RootUserPath(FromWhichRoot from)
|
|
{
|
|
int idx = from == FROM_CONFIGURED_ROOT ? D_WIIROOT_IDX : D_SESSION_WIIROOT_IDX;
|
|
return File::GetUserPath(idx);
|
|
}
|
|
|
|
static std::string RootUserPath(std::optional<FromWhichRoot> from)
|
|
{
|
|
return from ? RootUserPath(*from) : "";
|
|
}
|
|
|
|
std::string GetImportTitlePath(u64 title_id, std::optional<FromWhichRoot> from)
|
|
{
|
|
return RootUserPath(from) + StringFromFormat("/import/%08x/%08x",
|
|
static_cast<u32>(title_id >> 32),
|
|
static_cast<u32>(title_id));
|
|
}
|
|
|
|
std::string GetTicketFileName(u64 title_id, std::optional<FromWhichRoot> from)
|
|
{
|
|
return StringFromFormat("%s/ticket/%08x/%08x.tik", RootUserPath(from).c_str(),
|
|
static_cast<u32>(title_id >> 32), static_cast<u32>(title_id));
|
|
}
|
|
|
|
std::string GetTitlePath(u64 title_id, std::optional<FromWhichRoot> from)
|
|
{
|
|
return StringFromFormat("%s/title/%08x/%08x", RootUserPath(from).c_str(),
|
|
static_cast<u32>(title_id >> 32), static_cast<u32>(title_id));
|
|
}
|
|
|
|
std::string GetTitleDataPath(u64 title_id, std::optional<FromWhichRoot> from)
|
|
{
|
|
return GetTitlePath(title_id, from) + "/data";
|
|
}
|
|
|
|
std::string GetTitleContentPath(u64 title_id, std::optional<FromWhichRoot> from)
|
|
{
|
|
return GetTitlePath(title_id, from) + "/content";
|
|
}
|
|
|
|
std::string GetTMDFileName(u64 title_id, std::optional<FromWhichRoot> from)
|
|
{
|
|
return GetTitleContentPath(title_id, from) + "/title.tmd";
|
|
}
|
|
|
|
bool IsTitlePath(const std::string& path, std::optional<FromWhichRoot> from, u64* title_id)
|
|
{
|
|
std::string expected_prefix = RootUserPath(from) + "/title/";
|
|
if (!StringBeginsWith(path, expected_prefix))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Try to find a title ID in the remaining path.
|
|
std::string subdirectory = path.substr(expected_prefix.size());
|
|
std::vector<std::string> components = SplitString(subdirectory, '/');
|
|
if (components.size() < 2)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
u32 title_id_high, title_id_low;
|
|
if (!AsciiToHex(components[0], title_id_high) || !AsciiToHex(components[1], title_id_low))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (title_id != nullptr)
|
|
{
|
|
*title_id = (static_cast<u64>(title_id_high) << 32) | title_id_low;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
std::string EscapeFileName(const std::string& filename)
|
|
{
|
|
// Prevent paths from containing special names like ., .., ..., ...., and so on
|
|
if (std::all_of(filename.begin(), filename.end(), [](char c) { return c == '.'; }))
|
|
return ReplaceAll(filename, ".", "__2e__");
|
|
|
|
// Escape all double underscores since we will use double underscores for our escape sequences
|
|
std::string filename_with_escaped_double_underscores = ReplaceAll(filename, "__", "__5f____5f__");
|
|
|
|
// Escape all other characters that need to be escaped
|
|
static const std::unordered_set<char> chars_to_replace = {'\"', '*', '/', ':', '<',
|
|
'>', '?', '\\', '|', '\x7f'};
|
|
std::string result;
|
|
result.reserve(filename_with_escaped_double_underscores.size());
|
|
for (char c : filename_with_escaped_double_underscores)
|
|
{
|
|
if ((c >= 0 && c <= 0x1F) || chars_to_replace.find(c) != chars_to_replace.end())
|
|
result.append(StringFromFormat("__%02x__", c));
|
|
else
|
|
result.push_back(c);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
std::string EscapePath(const std::string& path)
|
|
{
|
|
const std::vector<std::string> split_strings = SplitString(path, '/');
|
|
|
|
std::vector<std::string> escaped_split_strings;
|
|
escaped_split_strings.reserve(split_strings.size());
|
|
for (const std::string& split_string : split_strings)
|
|
escaped_split_strings.push_back(EscapeFileName(split_string));
|
|
|
|
return JoinStrings(escaped_split_strings, "/");
|
|
}
|
|
|
|
std::string UnescapeFileName(const std::string& filename)
|
|
{
|
|
std::string result = filename;
|
|
size_t pos = 0;
|
|
|
|
// Replace escape sequences of the format "__3f__" with the ASCII
|
|
// character defined by the escape sequence's two hex digits.
|
|
while ((pos = result.find("__", pos)) != std::string::npos)
|
|
{
|
|
u32 character;
|
|
if (pos + 6 <= result.size() && result[pos + 4] == '_' && result[pos + 5] == '_')
|
|
if (AsciiToHex(result.substr(pos + 2, 2), character))
|
|
result.replace(pos, 6, {static_cast<char>(character)});
|
|
|
|
++pos;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|