mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-12 15:54:47 +00:00
569 lines
22 KiB
C++
569 lines
22 KiB
C++
/*
|
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
|
|
namespace ams::fs {
|
|
|
|
namespace {
|
|
|
|
Result CheckSharedName(const char *path, int len) {
|
|
if (len == 1) {
|
|
R_UNLESS(path[0] != StringTraits::Dot, fs::ResultInvalidPathFormat());
|
|
} else if (len == 2) {
|
|
R_UNLESS(path[0] != StringTraits::Dot || path[1] != StringTraits::Dot, fs::ResultInvalidPathFormat());
|
|
}
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result ParseWindowsPath(const char **out_path, char *out, size_t *out_windows_path_len, bool *out_normalized, const char *path, size_t max_out_size, bool has_mount_name) {
|
|
/* Prepare to parse. */
|
|
const char * const path_start = path;
|
|
if (out_normalized != nullptr) {
|
|
*out_normalized = true;
|
|
}
|
|
|
|
/* Handle start of path. */
|
|
bool skipped_mount = false;
|
|
auto prefix_len = 0;
|
|
if (has_mount_name) {
|
|
if (PathNormalizer::IsSeparator(path[0]) && path[1] == StringTraits::AlternateDirectorySeparator && path[2] == StringTraits::AlternateDirectorySeparator) {
|
|
path += 1;
|
|
prefix_len = 1;
|
|
} else {
|
|
/* Advance past separators. */
|
|
while (PathNormalizer::IsSeparator(path[0])) {
|
|
++path;
|
|
}
|
|
if (path != path_start) {
|
|
if (path - path_start == 1 || IsWindowsDrive(path)) {
|
|
prefix_len = 1;
|
|
} else {
|
|
if (path - path_start > 2 && out_normalized != nullptr) {
|
|
*out_normalized = false;
|
|
return ResultSuccess();
|
|
}
|
|
path -= 2;
|
|
skipped_mount = true;
|
|
}
|
|
}
|
|
}
|
|
} else if (PathNormalizer::IsSeparator(path[0]) && !IsUnc(path)) {
|
|
path += 1;
|
|
prefix_len = 1;
|
|
}
|
|
|
|
|
|
/* Parse the path. */
|
|
const char *trimmed_path = path_start;
|
|
if (IsWindowsDrive(path)) {
|
|
/* Find the first separator. */
|
|
int i;
|
|
for (i = 2; !PathNormalizer::IsNullTerminator(path[i]); ++i) {
|
|
if (PathNormalizer::IsAnySeparator(path[i])) {
|
|
break;
|
|
}
|
|
}
|
|
trimmed_path = path + i;
|
|
|
|
const size_t win_path_len = trimmed_path - path_start;
|
|
if (out != nullptr) {
|
|
R_UNLESS(win_path_len <= max_out_size, fs::ResultTooLongPath());
|
|
std::memcpy(out, path_start, win_path_len);
|
|
}
|
|
*out_path = trimmed_path;
|
|
*out_windows_path_len = win_path_len;
|
|
} else if (IsUnc(path)) {
|
|
if (PathNormalizer::IsAnySeparator(path[2])) {
|
|
AMS_ASSERT(!has_mount_name);
|
|
return fs::ResultInvalidPathFormat();
|
|
}
|
|
|
|
int cur_part_ofs = 0;
|
|
bool needs_sep_fix = false;
|
|
for (auto i = 2; !PathNormalizer::IsNullTerminator(path[i]); ++i) {
|
|
if (cur_part_ofs == 0 && path[i] == StringTraits::AlternateDirectorySeparator) {
|
|
needs_sep_fix = true;
|
|
if (out_normalized != nullptr) {
|
|
*out_normalized = false;
|
|
return ResultSuccess();
|
|
}
|
|
}
|
|
if (PathNormalizer::IsAnySeparator(path[i])) {
|
|
if (path[i] == StringTraits::AlternateDirectorySeparator) {
|
|
needs_sep_fix = true;
|
|
}
|
|
|
|
if (cur_part_ofs != 0) {
|
|
break;
|
|
}
|
|
|
|
R_UNLESS(!PathNormalizer::IsSeparator(path[i + 1]), fs::ResultInvalidPathFormat());
|
|
|
|
R_TRY(CheckSharedName(path + 2, i - 2));
|
|
|
|
cur_part_ofs = i + 1;
|
|
}
|
|
if (path[i] == '$' || path[i] == StringTraits::DriveSeparator) {
|
|
R_UNLESS(cur_part_ofs != 0, fs::ResultInvalidCharacter());
|
|
R_UNLESS(PathNormalizer::IsAnySeparator(path[i + 1]) || PathNormalizer::IsNullTerminator(path[i + 1]), fs::ResultInvalidPathFormat());
|
|
trimmed_path = path + i + 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (trimmed_path == path_start) {
|
|
int tr_part_ofs = 0;
|
|
int i;
|
|
for (i = 2; !PathNormalizer::IsNullTerminator(path[i]); ++i) {
|
|
if (PathNormalizer::IsAnySeparator(path[i])) {
|
|
if (tr_part_ofs != 0) {
|
|
R_TRY(CheckSharedName(path + tr_part_ofs, i - tr_part_ofs));
|
|
trimmed_path = path + i;
|
|
break;
|
|
}
|
|
|
|
R_UNLESS(!PathNormalizer::IsSeparator(path[i + 1]), fs::ResultInvalidPathFormat());
|
|
|
|
R_TRY(CheckSharedName(path + 2, i - 2));
|
|
|
|
cur_part_ofs = i + 1;
|
|
}
|
|
}
|
|
|
|
if (tr_part_ofs != 0 && trimmed_path == path_start) {
|
|
R_TRY(CheckSharedName(path + tr_part_ofs, i - tr_part_ofs));
|
|
trimmed_path = path + i;
|
|
}
|
|
}
|
|
|
|
const size_t win_path_len = trimmed_path - path;
|
|
const bool prepend_sep = prefix_len != 0 || skipped_mount;
|
|
if (out != nullptr) {
|
|
R_UNLESS(win_path_len <= max_out_size, fs::ResultTooLongPath());
|
|
if (prepend_sep) {
|
|
*(out++) = StringTraits::DirectorySeparator;
|
|
}
|
|
std::memcpy(out, path, win_path_len);
|
|
out[0] = StringTraits::AlternateDirectorySeparator;
|
|
out[1] = StringTraits::AlternateDirectorySeparator;
|
|
if (needs_sep_fix) {
|
|
for (size_t i = 2; i < win_path_len; ++i) {
|
|
if (PathNormalizer::IsSeparator(out[i])) {
|
|
out[i] = StringTraits::AlternateDirectorySeparator;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
*out_path = trimmed_path;
|
|
*out_windows_path_len = win_path_len + (prepend_sep ? 1 : 0);
|
|
} else {
|
|
*out_path = trimmed_path;
|
|
}
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result SkipWindowsPath(const char **out_path, bool *out_normalized, const char *path, bool has_mount_name) {
|
|
size_t windows_path_len;
|
|
return ParseWindowsPath(out_path, nullptr, std::addressof(windows_path_len), out_normalized, path, 0, has_mount_name);
|
|
}
|
|
|
|
Result ParseMountName(const char **out_path, char *out, size_t *out_mount_name_len, const char *path, size_t max_out_size) {
|
|
/* Decide on a start. */
|
|
const char *start = PathNormalizer::IsSeparator(path[0]) ? path + 1 : path;
|
|
|
|
/* Find the end of the mount name. */
|
|
const char *cur;
|
|
for (cur = start; cur < start + MountNameLengthMax + 1; ++cur) {
|
|
if (*cur == StringTraits::DriveSeparator) {
|
|
++cur;
|
|
break;
|
|
}
|
|
if (PathNormalizer::IsSeparator(*cur)) {
|
|
*out_path = path;
|
|
*out_mount_name_len = 0;
|
|
return ResultSuccess();
|
|
}
|
|
}
|
|
R_UNLESS(start < cur - 1, fs::ResultInvalidPathFormat());
|
|
R_UNLESS(cur[-1] == StringTraits::DriveSeparator, fs::ResultInvalidPathFormat());
|
|
|
|
/* Check the mount name doesn't contain a dot. */
|
|
if (cur != start) {
|
|
for (const char *p = start; p < cur; ++p) {
|
|
R_UNLESS(*p != StringTraits::Dot, fs::ResultInvalidCharacter());
|
|
}
|
|
}
|
|
|
|
const size_t mount_name_len = cur - path;
|
|
if (out != nullptr) {
|
|
R_UNLESS(mount_name_len <= max_out_size, fs::ResultTooLongPath());
|
|
std::memcpy(out, path, mount_name_len);
|
|
}
|
|
*out_path = cur;
|
|
*out_mount_name_len = mount_name_len;
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result SkipMountName(const char **out_path, const char *path) {
|
|
size_t mount_name_len;
|
|
return ParseMountName(out_path, nullptr, std::addressof(mount_name_len), path, 0);
|
|
}
|
|
|
|
bool IsParentDirectoryPathReplacementNeeded(const char *path) {
|
|
if (!PathNormalizer::IsAnySeparator(path[0])) {
|
|
return false;
|
|
}
|
|
|
|
for (auto i = 0; !PathNormalizer::IsNullTerminator(path[i]); ++i) {
|
|
if (path[i + 0] == StringTraits::AlternateDirectorySeparator &&
|
|
path[i + 1] == StringTraits::Dot &&
|
|
path[i + 2] == StringTraits::Dot &&
|
|
(PathNormalizer::IsAnySeparator(path[i + 3]) || PathNormalizer::IsNullTerminator(path[i + 3])))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (PathNormalizer::IsAnySeparator(path[i + 0]) &&
|
|
path[i + 1] == StringTraits::Dot &&
|
|
path[i + 2] == StringTraits::Dot &&
|
|
path[i + 3] == StringTraits::AlternateDirectorySeparator)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void ReplaceParentDirectoryPath(char *dst, const char *src) {
|
|
dst[0] = StringTraits::DirectorySeparator;
|
|
|
|
int i = 1;
|
|
while (!PathNormalizer::IsNullTerminator(src[i])) {
|
|
if (PathNormalizer::IsAnySeparator(src[i - 1]) &&
|
|
src[i + 0] == StringTraits::Dot &&
|
|
src[i + 1] == StringTraits::Dot &&
|
|
PathNormalizer::IsAnySeparator(src[i + 2]))
|
|
{
|
|
dst[i - 1] = StringTraits::DirectorySeparator;
|
|
dst[i + 0] = StringTraits::Dot;
|
|
dst[i + 1] = StringTraits::Dot;
|
|
dst[i - 2] = StringTraits::DirectorySeparator;
|
|
i += 3;
|
|
} else {
|
|
if (src[i - 1] == StringTraits::AlternateDirectorySeparator &&
|
|
src[i + 0] == StringTraits::Dot &&
|
|
src[i + 1] == StringTraits::Dot &&
|
|
PathNormalizer::IsNullTerminator(src[i + 2]))
|
|
{
|
|
dst[i - 1] = StringTraits::DirectorySeparator;
|
|
dst[i + 0] = StringTraits::Dot;
|
|
dst[i + 1] = StringTraits::Dot;
|
|
i += 2;
|
|
break;
|
|
}
|
|
|
|
dst[i] = src[i];
|
|
++i;
|
|
}
|
|
}
|
|
|
|
dst[i] = StringTraits::NullTerminator;
|
|
}
|
|
|
|
}
|
|
|
|
Result PathNormalizer::Normalize(char *out, size_t *out_len, const char *path, size_t max_out_size, bool unc_preserved, bool has_mount_name) {
|
|
/* Check pre-conditions. */
|
|
AMS_ASSERT(out != nullptr);
|
|
AMS_ASSERT(out_len != nullptr);
|
|
AMS_ASSERT(path != nullptr);
|
|
|
|
/* If we should, handle the mount name. */
|
|
size_t prefix_len = 0;
|
|
if (has_mount_name) {
|
|
size_t mount_name_len = 0;
|
|
R_TRY(ParseMountName(std::addressof(path), out, std::addressof(mount_name_len), path, max_out_size));
|
|
prefix_len += mount_name_len;
|
|
}
|
|
|
|
/* Deal with unc. */
|
|
bool is_unc_path = false;
|
|
if (unc_preserved) {
|
|
const char * const path_start = path;
|
|
size_t windows_path_len = 0;
|
|
R_TRY(ParseWindowsPath(std::addressof(path), out + prefix_len, std::addressof(windows_path_len), nullptr, path, max_out_size, has_mount_name));
|
|
prefix_len += windows_path_len;
|
|
is_unc_path = path != path_start;
|
|
}
|
|
|
|
/* Paths must start with / */
|
|
R_UNLESS(prefix_len != 0 || IsSeparator(path[0]), fs::ResultInvalidPathFormat());
|
|
|
|
/* Check if parent directory path replacement is needed. */
|
|
std::unique_ptr<char[], fs::impl::Deleter> replacement_path;
|
|
if (IsParentDirectoryPathReplacementNeeded(path)) {
|
|
/* Allocate a buffer to hold the replacement path. */
|
|
replacement_path = fs::impl::MakeUnique<char[]>(EntryNameLengthMax + 1);
|
|
R_UNLESS(replacement_path != nullptr, fs::ResultAllocationFailureInNew());
|
|
|
|
/* Replace the path. */
|
|
ReplaceParentDirectoryPath(replacement_path.get(), path);
|
|
|
|
/* Set path to be the replacement path. */
|
|
path = replacement_path.get();
|
|
}
|
|
|
|
bool skip_next_sep = false;
|
|
size_t i = 0;
|
|
size_t len = prefix_len;
|
|
|
|
while (!IsNullTerminator(path[i])) {
|
|
if (IsSeparator(path[i])) {
|
|
/* Swallow separators. */
|
|
while (IsSeparator(path[++i])) { }
|
|
if (IsNullTerminator(path[i])) {
|
|
break;
|
|
}
|
|
|
|
/* Handle skip if needed */
|
|
if (!skip_next_sep) {
|
|
if (len + 1 == max_out_size) {
|
|
out[len] = StringTraits::NullTerminator;
|
|
*out_len = len;
|
|
return fs::ResultTooLongPath();
|
|
}
|
|
|
|
out[len++] = StringTraits::DirectorySeparator;
|
|
}
|
|
skip_next_sep = false;
|
|
}
|
|
|
|
/* See length of current dir. */
|
|
size_t dir_len = 0;
|
|
while (!IsSeparator(path[i + dir_len]) && !IsNullTerminator(path[i + dir_len])) {
|
|
++dir_len;
|
|
}
|
|
|
|
if (IsCurrentDirectory(path + i)) {
|
|
skip_next_sep = true;
|
|
} else if (IsParentDirectory(path + i)) {
|
|
AMS_ASSERT(IsSeparator(out[len - 1]));
|
|
if (!is_unc_path) {
|
|
AMS_ASSERT(IsSeparator(out[prefix_len]));
|
|
}
|
|
|
|
/* Walk up a directory. */
|
|
if (len == prefix_len + 1) {
|
|
R_UNLESS(is_unc_path, fs::ResultDirectoryUnobtainable());
|
|
--len;
|
|
} else {
|
|
len -= 2;
|
|
do {
|
|
if (IsSeparator(out[len])) {
|
|
break;
|
|
}
|
|
--len;
|
|
} while (len != prefix_len);
|
|
}
|
|
|
|
if (!is_unc_path) {
|
|
AMS_ASSERT(IsSeparator(out[prefix_len]));
|
|
}
|
|
|
|
AMS_ASSERT(len < max_out_size);
|
|
} else {
|
|
/* Copy, possibly truncating. */
|
|
if (len + dir_len + 1 <= max_out_size) {
|
|
for (size_t j = 0; j < dir_len; ++j) {
|
|
out[len++] = path[i+j];
|
|
}
|
|
} else {
|
|
const size_t copy_len = max_out_size - 1 - len;
|
|
for (size_t j = 0; j < copy_len; ++j) {
|
|
out[len++] = path[i+j];
|
|
}
|
|
out[len] = StringTraits::NullTerminator;
|
|
*out_len = len;
|
|
return fs::ResultTooLongPath();
|
|
}
|
|
}
|
|
|
|
i += dir_len;
|
|
}
|
|
|
|
if (skip_next_sep) {
|
|
--len;
|
|
}
|
|
|
|
if (!is_unc_path && len == prefix_len && max_out_size > len) {
|
|
out[len++] = StringTraits::DirectorySeparator;
|
|
}
|
|
|
|
R_UNLESS(max_out_size >= len - 1, fs::ResultTooLongPath());
|
|
|
|
/* Null terminate. */
|
|
out[len] = StringTraits::NullTerminator;
|
|
*out_len = len;
|
|
|
|
/* Assert normalized. */
|
|
{
|
|
bool normalized = false;
|
|
const auto is_norm_result = IsNormalized(std::addressof(normalized), out, unc_preserved, has_mount_name);
|
|
AMS_ASSERT(R_SUCCEEDED(is_norm_result));
|
|
AMS_ASSERT(normalized);
|
|
}
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result PathNormalizer::IsNormalized(bool *out, const char *path, bool unc_preserved, bool has_mount_name) {
|
|
AMS_ASSERT(out != nullptr);
|
|
AMS_ASSERT(path != nullptr);
|
|
|
|
/* Save the start of the path. */
|
|
const char *path_start = path;
|
|
|
|
/* If we should, skip the mount name. */
|
|
if (has_mount_name) {
|
|
R_TRY(SkipMountName(std::addressof(path), path));
|
|
R_UNLESS(IsSeparator(*path), fs::ResultInvalidPathFormat());
|
|
}
|
|
|
|
/* If we should, handle unc. */
|
|
bool is_unc_path = false;
|
|
if (unc_preserved) {
|
|
path_start = path;
|
|
|
|
/* Skip the windows path. */
|
|
bool normalized_windows = false;
|
|
R_TRY(SkipWindowsPath(std::addressof(path), std::addressof(normalized_windows), path, has_mount_name));
|
|
|
|
/* If we're not windows-normalized, we're not normalized. */
|
|
if (!normalized_windows) {
|
|
*out = false;
|
|
return ResultSuccess();
|
|
}
|
|
|
|
/* Handle the case where we're dealing with a unc path. */
|
|
if (path != path_start) {
|
|
is_unc_path = true;
|
|
if (IsSeparator(path_start[0]) && IsSeparator(path_start[1])) {
|
|
*out = false;
|
|
return ResultSuccess();
|
|
}
|
|
|
|
if (IsNullTerminator(path[0])) {
|
|
*out = true;
|
|
return ResultSuccess();
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Check if parent directory path replacement is needed. */
|
|
if (IsParentDirectoryPathReplacementNeeded(path)) {
|
|
*out = false;
|
|
return ResultSuccess();
|
|
}
|
|
|
|
/* Nintendo uses a state machine here. */
|
|
enum class PathState {
|
|
Start,
|
|
Normal,
|
|
FirstSeparator,
|
|
Separator,
|
|
CurrentDir,
|
|
ParentDir,
|
|
};
|
|
|
|
PathState state = PathState::Start;
|
|
for (const char *cur = path; *cur != StringTraits::NullTerminator; ++cur) {
|
|
const char c = *cur;
|
|
switch (state) {
|
|
case PathState::Start:
|
|
if (IsSeparator(c)) {
|
|
state = PathState::FirstSeparator;
|
|
} else {
|
|
R_UNLESS(path != path_start, fs::ResultInvalidPathFormat());
|
|
if (c == StringTraits::Dot) {
|
|
state = PathState::CurrentDir;
|
|
} else {
|
|
state = PathState::Normal;
|
|
}
|
|
}
|
|
break;
|
|
case PathState::Normal:
|
|
if (IsSeparator(c)) {
|
|
state = PathState::Separator;
|
|
}
|
|
break;
|
|
case PathState::FirstSeparator:
|
|
case PathState::Separator:
|
|
if (IsSeparator(c)) {
|
|
*out = false;
|
|
return ResultSuccess();
|
|
} else if (c == StringTraits::Dot) {
|
|
state = PathState::CurrentDir;
|
|
} else {
|
|
state = PathState::Normal;
|
|
}
|
|
break;
|
|
case PathState::CurrentDir:
|
|
if (IsSeparator(c)) {
|
|
*out = false;
|
|
return ResultSuccess();
|
|
} else if (c == StringTraits::Dot) {
|
|
state = PathState::ParentDir;
|
|
} else {
|
|
state = PathState::Normal;
|
|
}
|
|
break;
|
|
case PathState::ParentDir:
|
|
if (IsSeparator(c)) {
|
|
*out = false;
|
|
return ResultSuccess();
|
|
} else {
|
|
state = PathState::Normal;
|
|
}
|
|
break;
|
|
AMS_UNREACHABLE_DEFAULT_CASE();
|
|
}
|
|
}
|
|
|
|
switch (state) {
|
|
case PathState::Start:
|
|
return fs::ResultInvalidPathFormat();
|
|
case PathState::Normal:
|
|
*out = true;
|
|
break;
|
|
case PathState::FirstSeparator:
|
|
*out = !is_unc_path;
|
|
break;
|
|
case PathState::CurrentDir:
|
|
case PathState::ParentDir:
|
|
case PathState::Separator:
|
|
*out = false;
|
|
break;
|
|
AMS_UNREACHABLE_DEFAULT_CASE();
|
|
}
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
}
|