mirror of
				https://github.com/Atmosphere-NX/Atmosphere-libs.git
				synced 2025-10-31 11:36:02 +01: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::ResultAllocationMemoryFailedInHostA());
 | |
| 
 | |
|             /* 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::ResultAllocationMemoryFailedInHostB());
 | |
| 
 | |
|             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::ResultAllocationMemoryFailedInHostC());
 | |
| 
 | |
|             /* 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::ResultAllocationMemoryFailedInHostC());
 | |
| 
 | |
|             /* 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);
 | |
|     }
 | |
| 
 | |
| }
 |