mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2025-01-16 06:11:37 +00:00
242 lines
10 KiB
C++
242 lines
10 KiB
C++
|
/*
|
||
|
* Copyright (c) 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>
|
||
|
#include "fsa/fs_mount_utils.hpp"
|
||
|
#include "impl/fs_file_system_proxy_service_object.hpp"
|
||
|
#include "impl/fs_file_system_service_object_adapter.hpp"
|
||
|
|
||
|
namespace ams::fs {
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
class HostRootCommonMountNameGenerator : public fsa::ICommonMountNameGenerator, public impl::Newable {
|
||
|
public:
|
||
|
explicit HostRootCommonMountNameGenerator() { /* ... */ }
|
||
|
|
||
|
virtual Result GenerateCommonMountName(char *dst, size_t dst_size) override {
|
||
|
/* Determine how much space we need. */
|
||
|
constexpr size_t RequiredNameBufferSizeSize = AMS_FS_IMPL_HOST_ROOT_FILE_SYSTEM_MOUNT_NAME_LEN + 1 + 1;
|
||
|
AMS_ASSERT(dst_size >= RequiredNameBufferSizeSize);
|
||
|
AMS_UNUSED(RequiredNameBufferSizeSize);
|
||
|
|
||
|
/* Generate the name. */
|
||
|
const auto size = util::SNPrintf(dst, dst_size, AMS_FS_IMPL_HOST_ROOT_FILE_SYSTEM_MOUNT_NAME ":");
|
||
|
AMS_ASSERT(static_cast<size_t>(size) == RequiredNameBufferSizeSize - 1);
|
||
|
AMS_UNUSED(size);
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class HostCommonMountNameGenerator : public fsa::ICommonMountNameGenerator, public impl::Newable {
|
||
|
private:
|
||
|
char m_path[fs::EntryNameLengthMax + 1];
|
||
|
public:
|
||
|
HostCommonMountNameGenerator(const char *path) {
|
||
|
util::Strlcpy<char>(m_path, path, sizeof(m_path));
|
||
|
}
|
||
|
|
||
|
virtual Result GenerateCommonMountName(char *dst, size_t dst_size) override {
|
||
|
/* Determine how much space we need. */
|
||
|
const size_t required_size = AMS_FS_IMPL_HOST_ROOT_FILE_SYSTEM_MOUNT_NAME_LEN + 1 + util::Strnlen<char>(m_path, sizeof(m_path)) + 1; /* @Host:%s */
|
||
|
R_UNLESS(dst_size >= required_size, fs::ResultTooLongPath());
|
||
|
|
||
|
|
||
|
/* Generate the name. */
|
||
|
const auto size = util::SNPrintf(dst, dst_size, AMS_FS_IMPL_HOST_ROOT_FILE_SYSTEM_MOUNT_NAME ":%s", m_path);
|
||
|
AMS_ASSERT(static_cast<size_t>(size) == required_size - 1);
|
||
|
AMS_UNUSED(size);
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Result OpenHostFileSystemImpl(std::unique_ptr<fs::fsa::IFileSystem> *out, const fssrv::sf::FspPath &path, const MountHostOption &option) {
|
||
|
/* Open the filesystem. */
|
||
|
auto fsp = impl::GetFileSystemProxyServiceObject();
|
||
|
sf::SharedPointer<fssrv::sf::IFileSystem> fs;
|
||
|
if (option != MountHostOption::None) {
|
||
|
R_TRY(fsp->OpenHostFileSystemWithOption(std::addressof(fs), path, option._value));
|
||
|
} else {
|
||
|
R_TRY(fsp->OpenHostFileSystem(std::addressof(fs), path));
|
||
|
}
|
||
|
|
||
|
/* Allocate a new filesystem wrapper. */
|
||
|
auto fsa = std::make_unique<impl::FileSystemServiceObjectAdapter>(std::move(fs));
|
||
|
R_UNLESS(fsa != nullptr, fs::ResultAllocationFailureInHostA());
|
||
|
|
||
|
/* Set the output. */
|
||
|
*out = std::move(fsa);
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
|
||
|
Result PreMountHost(std::unique_ptr<fs::HostCommonMountNameGenerator> *out, const char *name, const char *path) {
|
||
|
/* Check pre-conditions. */
|
||
|
AMS_ASSERT(out != nullptr);
|
||
|
|
||
|
/* Check the mount name. */
|
||
|
R_TRY(impl::CheckMountName(name));
|
||
|
|
||
|
/* Check that path is valid. */
|
||
|
R_UNLESS(path != nullptr, fs::ResultNullptrArgument());
|
||
|
|
||
|
/* Create a new HostCommonMountNameGenerator. */
|
||
|
*out = std::make_unique<HostCommonMountNameGenerator>(path);
|
||
|
R_UNLESS(out->get() != nullptr, fs::ResultAllocationFailureInHostB());
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
namespace impl {
|
||
|
|
||
|
Result OpenHostFileSystem(std::unique_ptr<fs::fsa::IFileSystem> *out, const char *name, const char *path, const fs::MountHostOption &option) {
|
||
|
/* Validate arguments. */
|
||
|
R_UNLESS(out != nullptr, fs::ResultNullptrArgument());
|
||
|
R_UNLESS(name != nullptr, fs::ResultNullptrArgument());
|
||
|
R_UNLESS(path != nullptr, fs::ResultNullptrArgument());
|
||
|
|
||
|
/* Check mount name isn't windows path or reserved. */
|
||
|
R_UNLESS(!fs::IsWindowsDrive(name), fs::ResultInvalidMountName());
|
||
|
R_UNLESS(!fs::impl::IsReservedMountName(name), fs::ResultInvalidMountName());
|
||
|
|
||
|
/* Convert the path for fsp. */
|
||
|
fssrv::sf::FspPath sf_path;
|
||
|
R_TRY(fs::ConvertToFspPath(std::addressof(sf_path), path));
|
||
|
|
||
|
/* Ensure that the path doesn't correspond to the root. */
|
||
|
if (sf_path.str[0] == 0) {
|
||
|
sf_path.str[0] = '.';
|
||
|
sf_path.str[1] = 0;
|
||
|
}
|
||
|
|
||
|
/* Open the host file system. */
|
||
|
R_RETURN(OpenHostFileSystemImpl(out, sf_path, option));
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
Result MountHost(const char *name, const char *root_path) {
|
||
|
/* Pre-mount host. */
|
||
|
std::unique_ptr<fs::HostCommonMountNameGenerator> generator;
|
||
|
AMS_FS_R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT_UNLESS_R_SUCCEEDED(PreMountHost(std::addressof(generator), name, root_path), name, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST(name, root_path)));
|
||
|
|
||
|
/* Open the filesystem. */
|
||
|
std::unique_ptr<fs::fsa::IFileSystem> fsa;
|
||
|
R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT_UNLESS_R_SUCCEEDED(impl::OpenHostFileSystem(std::addressof(fsa), name, root_path, MountHostOption::None), name, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST(name, root_path)));
|
||
|
|
||
|
/* Declare registration helper. */
|
||
|
auto register_impl = [&]() -> Result {
|
||
|
/* Register. */
|
||
|
R_RETURN(fsa::Register(name, std::move(fsa), std::move(generator)));
|
||
|
};
|
||
|
|
||
|
/* Mount the filesystem. */
|
||
|
AMS_FS_R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT(register_impl(), name, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST(name, root_path)));
|
||
|
|
||
|
/* Enable access logging. */
|
||
|
AMS_FS_IMPL_ACCESS_LOG_FS_ACCESSOR_ENABLE(name);
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
|
||
|
Result MountHost(const char *name, const char *root_path, const MountHostOption &option) {
|
||
|
/* Pre-mount host. */
|
||
|
std::unique_ptr<fs::HostCommonMountNameGenerator> generator;
|
||
|
AMS_FS_R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT_UNLESS_R_SUCCEEDED(PreMountHost(std::addressof(generator), name, root_path), name, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_WITH_OPTION(name, root_path, option)));
|
||
|
|
||
|
/* Open the filesystem. */
|
||
|
std::unique_ptr<fs::fsa::IFileSystem> fsa;
|
||
|
R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT_UNLESS_R_SUCCEEDED(impl::OpenHostFileSystem(std::addressof(fsa), name, root_path, option), name, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_WITH_OPTION(name, root_path, option)));
|
||
|
|
||
|
/* Declare registration helper. */
|
||
|
auto register_impl = [&]() -> Result {
|
||
|
/* Register. */
|
||
|
R_RETURN(fsa::Register(name, std::move(fsa), std::move(generator)));
|
||
|
};
|
||
|
|
||
|
/* Mount the filesystem. */
|
||
|
AMS_FS_R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT(register_impl(), name, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_WITH_OPTION(name, root_path, option)));
|
||
|
|
||
|
/* Enable access logging. */
|
||
|
AMS_FS_IMPL_ACCESS_LOG_FS_ACCESSOR_ENABLE(name);
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
|
||
|
Result MountHostRoot() {
|
||
|
/* Create host root path. */
|
||
|
fssrv::sf::FspPath sf_path;
|
||
|
sf_path.str[0] = 0;
|
||
|
|
||
|
/* Open the filesystem. */
|
||
|
std::unique_ptr<fs::fsa::IFileSystem> fsa;
|
||
|
R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT_UNLESS_R_SUCCEEDED(OpenHostFileSystemImpl(std::addressof(fsa), sf_path, MountHostOption::None), impl::HostRootFileSystemMountName, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_ROOT()));
|
||
|
|
||
|
/* Declare registration helper. */
|
||
|
auto register_impl = [&]() -> Result {
|
||
|
/* Allocate a new mountname generator. */
|
||
|
auto generator = std::make_unique<HostRootCommonMountNameGenerator>();
|
||
|
R_UNLESS(generator != nullptr, fs::ResultAllocationFailureInHostC());
|
||
|
|
||
|
/* Register. */
|
||
|
R_RETURN(fsa::Register(impl::HostRootFileSystemMountName, std::move(fsa), std::move(generator)));
|
||
|
};
|
||
|
|
||
|
/* Mount the filesystem. */
|
||
|
AMS_FS_R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT(register_impl(), impl::HostRootFileSystemMountName, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_ROOT()));
|
||
|
|
||
|
/* Enable access logging. */
|
||
|
AMS_FS_IMPL_ACCESS_LOG_FS_ACCESSOR_ENABLE(impl::HostRootFileSystemMountName);
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
|
||
|
Result MountHostRoot(const MountHostOption &option) {
|
||
|
/* Create host root path. */
|
||
|
fssrv::sf::FspPath sf_path;
|
||
|
sf_path.str[0] = 0;
|
||
|
|
||
|
/* Open the filesystem. */
|
||
|
std::unique_ptr<fs::fsa::IFileSystem> fsa;
|
||
|
R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT_UNLESS_R_SUCCEEDED(OpenHostFileSystemImpl(std::addressof(fsa), sf_path, option), impl::HostRootFileSystemMountName, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_ROOT_WITH_OPTION(option)));
|
||
|
|
||
|
/* Declare registration helper. */
|
||
|
auto register_impl = [&]() -> Result {
|
||
|
/* Allocate a new mountname generator. */
|
||
|
auto generator = std::make_unique<HostRootCommonMountNameGenerator>();
|
||
|
R_UNLESS(generator != nullptr, fs::ResultAllocationFailureInHostC());
|
||
|
|
||
|
/* Register. */
|
||
|
R_RETURN(fsa::Register(impl::HostRootFileSystemMountName, std::move(fsa), std::move(generator)));
|
||
|
};
|
||
|
|
||
|
/* Mount the filesystem. */
|
||
|
AMS_FS_R_TRY(AMS_FS_IMPL_ACCESS_LOG_MOUNT(register_impl(), impl::HostRootFileSystemMountName, AMS_FS_IMPL_ACCESS_LOG_FORMAT_MOUNT_HOST_ROOT_WITH_OPTION(option)));
|
||
|
|
||
|
/* Enable access logging. */
|
||
|
AMS_FS_IMPL_ACCESS_LOG_FS_ACCESSOR_ENABLE(impl::HostRootFileSystemMountName);
|
||
|
|
||
|
R_SUCCEED();
|
||
|
}
|
||
|
|
||
|
void UnmountHostRoot() {
|
||
|
return Unmount(impl::HostRootFileSystemMountName);
|
||
|
}
|
||
|
|
||
|
}
|