Merge pull request #216 from Subv/savedata
Implemented the SaveData archive and MountSaveData.
This commit is contained in:
		| @@ -7,6 +7,8 @@ add_library(core STATIC | ||||
|     core_timing.cpp | ||||
|     core_timing.h | ||||
|     file_sys/directory.h | ||||
|     file_sys/disk_filesystem.cpp | ||||
|     file_sys/disk_filesystem.h | ||||
|     file_sys/errors.h | ||||
|     file_sys/filesystem.cpp | ||||
|     file_sys/filesystem.h | ||||
| @@ -18,6 +20,8 @@ add_library(core STATIC | ||||
|     file_sys/romfs_factory.h | ||||
|     file_sys/romfs_filesystem.cpp | ||||
|     file_sys/romfs_filesystem.h | ||||
|     file_sys/savedata_factory.cpp | ||||
|     file_sys/savedata_factory.h | ||||
|     file_sys/storage.h | ||||
|     frontend/emu_window.cpp | ||||
|     frontend/emu_window.h | ||||
|   | ||||
							
								
								
									
										147
									
								
								src/core/file_sys/disk_filesystem.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								src/core/file_sys/disk_filesystem.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,147 @@ | ||||
| // Copyright 2018 yuzu emulator team | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <cstring> | ||||
| #include <memory> | ||||
| #include "common/common_types.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "core/file_sys/disk_filesystem.h" | ||||
|  | ||||
| namespace FileSys { | ||||
|  | ||||
| std::string Disk_FileSystem::GetName() const { | ||||
|     return "Disk"; | ||||
| } | ||||
|  | ||||
| ResultVal<std::unique_ptr<StorageBackend>> Disk_FileSystem::OpenFile(const std::string& path, | ||||
|                                                                      Mode mode) const { | ||||
|     ASSERT_MSG(mode == Mode::Read || mode == Mode::Write, "Other file modes are not supported"); | ||||
|  | ||||
|     std::string full_path = base_directory + path; | ||||
|     auto file = std::make_shared<FileUtil::IOFile>(full_path, mode == Mode::Read ? "rb" : "wb"); | ||||
|  | ||||
|     if (!file->IsOpen()) { | ||||
|         // TODO(Subv): Find out the correct error code. | ||||
|         return ResultCode(-1); | ||||
|     } | ||||
|  | ||||
|     return MakeResult<std::unique_ptr<StorageBackend>>( | ||||
|         std::make_unique<Disk_Storage>(std::move(file))); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::DeleteFile(const Path& path) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     // TODO(bunnei): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::RenameFile(const Path& src_path, const Path& dest_path) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     // TODO(wwylele): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::DeleteDirectory(const Path& path) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     // TODO(wwylele): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::DeleteDirectoryRecursively(const Path& path) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     // TODO(wwylele): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::CreateFile(const std::string& path, u64 size) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|  | ||||
|     std::string full_path = base_directory + path; | ||||
|     if (size == 0) { | ||||
|         FileUtil::CreateEmptyFile(full_path); | ||||
|         return RESULT_SUCCESS; | ||||
|     } | ||||
|  | ||||
|     FileUtil::IOFile file(full_path, "wb"); | ||||
|     // Creates a sparse file (or a normal file on filesystems without the concept of sparse files) | ||||
|     // We do this by seeking to the right size, then writing a single null byte. | ||||
|     if (file.Seek(size - 1, SEEK_SET) && file.WriteBytes("", 1) == 1) { | ||||
|         return RESULT_SUCCESS; | ||||
|     } | ||||
|  | ||||
|     LOG_ERROR(Service_FS, "Too large file"); | ||||
|     // TODO(Subv): Find out the correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::CreateDirectory(const Path& path) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     // TODO(wwylele): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode Disk_FileSystem::RenameDirectory(const Path& src_path, const Path& dest_path) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     // TODO(wwylele): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultVal<std::unique_ptr<DirectoryBackend>> Disk_FileSystem::OpenDirectory( | ||||
|     const Path& path) const { | ||||
|     return MakeResult<std::unique_ptr<DirectoryBackend>>(std::make_unique<Disk_Directory>()); | ||||
| } | ||||
|  | ||||
| u64 Disk_FileSystem::GetFreeSpaceSize() const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| ResultVal<FileSys::EntryType> Disk_FileSystem::GetEntryType(const std::string& path) const { | ||||
|     std::string full_path = base_directory + path; | ||||
|     if (!FileUtil::Exists(full_path)) { | ||||
|         // TODO(Subv): Find out what this actually means | ||||
|         return ResultCode(ErrorModule::FS, 1); | ||||
|     } | ||||
|  | ||||
|     // TODO(Subv): Find out the EntryType values | ||||
|     UNIMPLEMENTED_MSG("Unimplemented GetEntryType"); | ||||
| } | ||||
|  | ||||
| ResultVal<size_t> Disk_Storage::Read(const u64 offset, const size_t length, u8* buffer) const { | ||||
|     LOG_TRACE(Service_FS, "called offset=%llu, length=%zu", offset, length); | ||||
|     file->Seek(offset, SEEK_SET); | ||||
|     return MakeResult<size_t>(file->ReadBytes(buffer, length)); | ||||
| } | ||||
|  | ||||
| ResultVal<size_t> Disk_Storage::Write(const u64 offset, const size_t length, const bool flush, | ||||
|                                       const u8* buffer) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     file->Seek(offset, SEEK_SET); | ||||
|     size_t written = file->WriteBytes(buffer, length); | ||||
|     if (flush) { | ||||
|         file->Flush(); | ||||
|     } | ||||
|     return MakeResult<size_t>(written); | ||||
| } | ||||
|  | ||||
| u64 Disk_Storage::GetSize() const { | ||||
|     return file->GetSize(); | ||||
| } | ||||
|  | ||||
| bool Disk_Storage::SetSize(const u64 size) const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| u32 Disk_Directory::Read(const u32 count, Entry* entries) { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| bool Disk_Directory::Close() const { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| } // namespace FileSys | ||||
							
								
								
									
										66
									
								
								src/core/file_sys/disk_filesystem.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								src/core/file_sys/disk_filesystem.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| // Copyright 2018 yuzu emulator team | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <cstddef> | ||||
| #include <memory> | ||||
| #include <string> | ||||
| #include "common/common_types.h" | ||||
| #include "common/file_util.h" | ||||
| #include "core/file_sys/directory.h" | ||||
| #include "core/file_sys/filesystem.h" | ||||
| #include "core/file_sys/storage.h" | ||||
| #include "core/hle/result.h" | ||||
|  | ||||
| namespace FileSys { | ||||
|  | ||||
| class Disk_FileSystem : public FileSystemBackend { | ||||
| public: | ||||
|     explicit Disk_FileSystem(std::string base_directory) | ||||
|         : base_directory(std::move(base_directory)) {} | ||||
|  | ||||
|     std::string GetName() const override; | ||||
|  | ||||
|     ResultVal<std::unique_ptr<StorageBackend>> OpenFile(const std::string& path, | ||||
|                                                         Mode mode) const override; | ||||
|     ResultCode DeleteFile(const Path& path) const override; | ||||
|     ResultCode RenameFile(const Path& src_path, const Path& dest_path) const override; | ||||
|     ResultCode DeleteDirectory(const Path& path) const override; | ||||
|     ResultCode DeleteDirectoryRecursively(const Path& path) const override; | ||||
|     ResultCode CreateFile(const std::string& path, u64 size) const override; | ||||
|     ResultCode CreateDirectory(const Path& path) const override; | ||||
|     ResultCode RenameDirectory(const Path& src_path, const Path& dest_path) const override; | ||||
|     ResultVal<std::unique_ptr<DirectoryBackend>> OpenDirectory(const Path& path) const override; | ||||
|     u64 GetFreeSpaceSize() const override; | ||||
|     ResultVal<EntryType> GetEntryType(const std::string& path) const override; | ||||
|  | ||||
| protected: | ||||
|     std::string base_directory; | ||||
| }; | ||||
|  | ||||
| class Disk_Storage : public StorageBackend { | ||||
| public: | ||||
|     Disk_Storage(std::shared_ptr<FileUtil::IOFile> file) : file(std::move(file)) {} | ||||
|  | ||||
|     ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override; | ||||
|     ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) const override; | ||||
|     u64 GetSize() const override; | ||||
|     bool SetSize(u64 size) const override; | ||||
|     bool Close() const override { | ||||
|         return false; | ||||
|     } | ||||
|     void Flush() const override {} | ||||
|  | ||||
| private: | ||||
|     std::shared_ptr<FileUtil::IOFile> file; | ||||
| }; | ||||
|  | ||||
| class Disk_Directory : public DirectoryBackend { | ||||
| public: | ||||
|     u32 Read(const u32 count, Entry* entries) override; | ||||
|     bool Close() const override; | ||||
| }; | ||||
|  | ||||
| } // namespace FileSys | ||||
| @@ -27,11 +27,14 @@ enum LowPathType : u32 { | ||||
|     Wchar = 4, | ||||
| }; | ||||
|  | ||||
| union Mode { | ||||
|     u32 hex; | ||||
|     BitField<0, 1, u32> read_flag; | ||||
|     BitField<1, 1, u32> write_flag; | ||||
|     BitField<2, 1, u32> create_flag; | ||||
| enum EntryType : u32 { | ||||
|     Directory = 0, | ||||
|     File = 1, | ||||
| }; | ||||
|  | ||||
| enum class Mode : u32 { | ||||
|     Read = 1, | ||||
|     Write = 2, | ||||
| }; | ||||
|  | ||||
| class Path { | ||||
| @@ -86,7 +89,7 @@ public: | ||||
|      * @param size The size of the new file, filled with zeroes | ||||
|      * @return Result of the operation | ||||
|      */ | ||||
|     virtual ResultCode CreateFile(const Path& path, u64 size) const = 0; | ||||
|     virtual ResultCode CreateFile(const std::string& path, u64 size) const = 0; | ||||
|  | ||||
|     /** | ||||
|      * Delete a file specified by its path | ||||
| @@ -138,8 +141,8 @@ public: | ||||
|      * @param mode Mode to open the file with | ||||
|      * @return Opened file, or error code | ||||
|      */ | ||||
|     virtual ResultVal<std::unique_ptr<StorageBackend>> OpenFile(const Path& path, | ||||
|                                                                 const Mode& mode) const = 0; | ||||
|     virtual ResultVal<std::unique_ptr<StorageBackend>> OpenFile(const std::string& path, | ||||
|                                                                 Mode mode) const = 0; | ||||
|  | ||||
|     /** | ||||
|      * Open a directory specified by its path | ||||
| @@ -153,6 +156,12 @@ public: | ||||
|      * @return The number of free bytes in the archive | ||||
|      */ | ||||
|     virtual u64 GetFreeSpaceSize() const = 0; | ||||
|  | ||||
|     /** | ||||
|      * Get the type of the specified path | ||||
|      * @return The type of the specified path or error code | ||||
|      */ | ||||
|     virtual ResultVal<EntryType> GetEntryType(const std::string& path) const = 0; | ||||
| }; | ||||
|  | ||||
| class FileSystemFactory : NonCopyable { | ||||
|   | ||||
| @@ -14,8 +14,8 @@ std::string RomFS_FileSystem::GetName() const { | ||||
|     return "RomFS"; | ||||
| } | ||||
|  | ||||
| ResultVal<std::unique_ptr<StorageBackend>> RomFS_FileSystem::OpenFile(const Path& path, | ||||
|                                                                       const Mode& mode) const { | ||||
| ResultVal<std::unique_ptr<StorageBackend>> RomFS_FileSystem::OpenFile(const std::string& path, | ||||
|                                                                       Mode mode) const { | ||||
|     return MakeResult<std::unique_ptr<StorageBackend>>( | ||||
|         std::make_unique<RomFS_Storage>(romfs_file, data_offset, data_size)); | ||||
| } | ||||
| @@ -48,7 +48,7 @@ ResultCode RomFS_FileSystem::DeleteDirectoryRecursively(const Path& path) const | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultCode RomFS_FileSystem::CreateFile(const Path& path, u64 size) const { | ||||
| ResultCode RomFS_FileSystem::CreateFile(const std::string& path, u64 size) const { | ||||
|     LOG_CRITICAL(Service_FS, "Attempted to create a file in an ROMFS archive (%s).", | ||||
|                  GetName().c_str()); | ||||
|     // TODO(bunnei): Use correct error code | ||||
| @@ -79,6 +79,12 @@ u64 RomFS_FileSystem::GetFreeSpaceSize() const { | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| ResultVal<FileSys::EntryType> RomFS_FileSystem::GetEntryType(const std::string& path) const { | ||||
|     LOG_CRITICAL(Service_FS, "Called within an ROMFS archive (path %s).", path.c_str()); | ||||
|     // TODO(wwylele): Use correct error code | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultVal<size_t> RomFS_Storage::Read(const u64 offset, const size_t length, u8* buffer) const { | ||||
|     LOG_TRACE(Service_FS, "called offset=%llu, length=%zu", offset, length); | ||||
|     romfs_file->Seek(data_offset + offset, SEEK_SET); | ||||
|   | ||||
| @@ -29,17 +29,18 @@ public: | ||||
|  | ||||
|     std::string GetName() const override; | ||||
|  | ||||
|     ResultVal<std::unique_ptr<StorageBackend>> OpenFile(const Path& path, | ||||
|                                                         const Mode& mode) const override; | ||||
|     ResultVal<std::unique_ptr<StorageBackend>> OpenFile(const std::string& path, | ||||
|                                                         Mode mode) const override; | ||||
|     ResultCode DeleteFile(const Path& path) const override; | ||||
|     ResultCode RenameFile(const Path& src_path, const Path& dest_path) const override; | ||||
|     ResultCode DeleteDirectory(const Path& path) const override; | ||||
|     ResultCode DeleteDirectoryRecursively(const Path& path) const override; | ||||
|     ResultCode CreateFile(const Path& path, u64 size) const override; | ||||
|     ResultCode CreateFile(const std::string& path, u64 size) const override; | ||||
|     ResultCode CreateDirectory(const Path& path) const override; | ||||
|     ResultCode RenameDirectory(const Path& src_path, const Path& dest_path) const override; | ||||
|     ResultVal<std::unique_ptr<DirectoryBackend>> OpenDirectory(const Path& path) const override; | ||||
|     u64 GetFreeSpaceSize() const override; | ||||
|     ResultVal<EntryType> GetEntryType(const std::string& path) const override; | ||||
|  | ||||
| protected: | ||||
|     std::shared_ptr<FileUtil::IOFile> romfs_file; | ||||
|   | ||||
							
								
								
									
										42
									
								
								src/core/file_sys/savedata_factory.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								src/core/file_sys/savedata_factory.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| // Copyright 2018 yuzu emulator team | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <cinttypes> | ||||
| #include <memory> | ||||
| #include "common/common_types.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "common/string_util.h" | ||||
| #include "core/file_sys/disk_filesystem.h" | ||||
| #include "core/file_sys/savedata_factory.h" | ||||
| #include "core/hle/kernel/process.h" | ||||
|  | ||||
| namespace FileSys { | ||||
|  | ||||
| SaveData_Factory::SaveData_Factory(std::string nand_directory) | ||||
|     : nand_directory(std::move(nand_directory)) {} | ||||
|  | ||||
| ResultVal<std::unique_ptr<FileSystemBackend>> SaveData_Factory::Open(const Path& path) { | ||||
|     u64 title_id = Kernel::g_current_process->program_id; | ||||
|     // TODO(Subv): Somehow obtain this value. | ||||
|     u32 user = 0; | ||||
|     std::string save_directory = Common::StringFromFormat("%ssave/%016" PRIX64 "/%08X", | ||||
|                                                           nand_directory.c_str(), title_id, user); | ||||
|     auto archive = std::make_unique<Disk_FileSystem>(save_directory); | ||||
|     return MakeResult<std::unique_ptr<FileSystemBackend>>(std::move(archive)); | ||||
| } | ||||
|  | ||||
| ResultCode SaveData_Factory::Format(const Path& path, | ||||
|                                     const FileSys::ArchiveFormatInfo& format_info) { | ||||
|     LOG_ERROR(Service_FS, "Unimplemented Format archive %s", GetName().c_str()); | ||||
|     // TODO(bunnei): Find the right error code for this | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| ResultVal<ArchiveFormatInfo> SaveData_Factory::GetFormatInfo(const Path& path) const { | ||||
|     LOG_ERROR(Service_FS, "Unimplemented GetFormatInfo archive %s", GetName().c_str()); | ||||
|     // TODO(bunnei): Find the right error code for this | ||||
|     return ResultCode(-1); | ||||
| } | ||||
|  | ||||
| } // namespace FileSys | ||||
							
								
								
									
										31
									
								
								src/core/file_sys/savedata_factory.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								src/core/file_sys/savedata_factory.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| // Copyright 2018 yuzu emulator team | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <memory> | ||||
| #include <string> | ||||
| #include "common/common_types.h" | ||||
| #include "core/file_sys/filesystem.h" | ||||
| #include "core/hle/result.h" | ||||
|  | ||||
| namespace FileSys { | ||||
|  | ||||
| /// File system interface to the SaveData archive | ||||
| class SaveData_Factory final : public FileSystemFactory { | ||||
| public: | ||||
|     explicit SaveData_Factory(std::string nand_directory); | ||||
|  | ||||
|     std::string GetName() const override { | ||||
|         return "SaveData_Factory"; | ||||
|     } | ||||
|     ResultVal<std::unique_ptr<FileSystemBackend>> Open(const Path& path) override; | ||||
|     ResultCode Format(const Path& path, const FileSys::ArchiveFormatInfo& format_info) override; | ||||
|     ResultVal<ArchiveFormatInfo> GetFormatInfo(const Path& path) const override; | ||||
|  | ||||
| private: | ||||
|     std::string nand_directory; | ||||
| }; | ||||
|  | ||||
| } // namespace FileSys | ||||
| @@ -20,12 +20,9 @@ namespace Kernel { | ||||
| // Lists all processes that exist in the current session. | ||||
| static std::vector<SharedPtr<Process>> process_list; | ||||
|  | ||||
| SharedPtr<CodeSet> CodeSet::Create(std::string name, u64 program_id) { | ||||
| SharedPtr<CodeSet> CodeSet::Create(std::string name) { | ||||
|     SharedPtr<CodeSet> codeset(new CodeSet); | ||||
|  | ||||
|     codeset->name = std::move(name); | ||||
|     codeset->program_id = program_id; | ||||
|  | ||||
|     return codeset; | ||||
| } | ||||
|  | ||||
| @@ -34,13 +31,14 @@ CodeSet::~CodeSet() {} | ||||
|  | ||||
| u32 Process::next_process_id; | ||||
|  | ||||
| SharedPtr<Process> Process::Create(std::string&& name) { | ||||
| SharedPtr<Process> Process::Create(std::string&& name, u64 program_id) { | ||||
|     SharedPtr<Process> process(new Process); | ||||
|  | ||||
|     process->name = std::move(name); | ||||
|     process->flags.raw = 0; | ||||
|     process->flags.memory_region.Assign(MemoryRegion::APPLICATION); | ||||
|     process->status = ProcessStatus::Created; | ||||
|     process->program_id = program_id; | ||||
|  | ||||
|     process_list.push_back(process); | ||||
|     return process; | ||||
|   | ||||
| @@ -56,7 +56,7 @@ class ResourceLimit; | ||||
| struct MemoryRegionInfo; | ||||
|  | ||||
| struct CodeSet final : public Object { | ||||
|     static SharedPtr<CodeSet> Create(std::string name, u64 program_id); | ||||
|     static SharedPtr<CodeSet> Create(std::string name); | ||||
|  | ||||
|     std::string GetTypeName() const override { | ||||
|         return "CodeSet"; | ||||
| @@ -72,8 +72,6 @@ struct CodeSet final : public Object { | ||||
|  | ||||
|     /// Name of the process | ||||
|     std::string name; | ||||
|     /// Title ID corresponding to the process | ||||
|     u64 program_id; | ||||
|  | ||||
|     std::shared_ptr<std::vector<u8>> memory; | ||||
|  | ||||
| @@ -97,7 +95,7 @@ private: | ||||
|  | ||||
| class Process final : public Object { | ||||
| public: | ||||
|     static SharedPtr<Process> Create(std::string&& name); | ||||
|     static SharedPtr<Process> Create(std::string&& name, u64 program_id); | ||||
|  | ||||
|     std::string GetTypeName() const override { | ||||
|         return "Process"; | ||||
| @@ -113,6 +111,9 @@ public: | ||||
|  | ||||
|     static u32 next_process_id; | ||||
|  | ||||
|     /// Title ID corresponding to the process | ||||
|     u64 program_id; | ||||
|  | ||||
|     /// Resource limit descriptor for this process | ||||
|     SharedPtr<ResourceLimit> resource_limit; | ||||
|  | ||||
|   | ||||
| @@ -108,11 +108,11 @@ union ResultCode { | ||||
|     } | ||||
|  | ||||
|     constexpr bool IsSuccess() const { | ||||
|         return is_error.ExtractValue(raw) == 0; | ||||
|         return raw == 0; | ||||
|     } | ||||
|  | ||||
|     constexpr bool IsError() const { | ||||
|         return is_error.ExtractValue(raw) == 1; | ||||
|         return raw != 0; | ||||
|     } | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -3,7 +3,9 @@ | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <boost/container/flat_map.hpp> | ||||
| #include "common/file_util.h" | ||||
| #include "core/file_sys/filesystem.h" | ||||
| #include "core/file_sys/savedata_factory.h" | ||||
| #include "core/hle/service/filesystem/filesystem.h" | ||||
| #include "core/hle/service/filesystem/fsp_srv.h" | ||||
|  | ||||
| @@ -41,12 +43,17 @@ ResultVal<std::unique_ptr<FileSys::FileSystemBackend>> OpenFileSystem(Type type, | ||||
|     return itr->second->Open(path); | ||||
| } | ||||
|  | ||||
| void UnregisterFileSystems() { | ||||
| void RegisterFileSystems() { | ||||
|     filesystem_map.clear(); | ||||
|  | ||||
|     std::string nand_directory = FileUtil::GetUserPath(D_NAND_IDX); | ||||
|  | ||||
|     auto savedata = std::make_unique<FileSys::SaveData_Factory>(std::move(nand_directory)); | ||||
|     RegisterFileSystem(std::move(savedata), Type::SaveData); | ||||
| } | ||||
|  | ||||
| void InstallInterfaces(SM::ServiceManager& service_manager) { | ||||
|     UnregisterFileSystems(); | ||||
|     RegisterFileSystems(); | ||||
|     std::make_shared<FSP_SRV>()->InstallAsService(service_manager); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -25,6 +25,7 @@ namespace FileSystem { | ||||
| /// Supported FileSystem types | ||||
| enum class Type { | ||||
|     RomFS = 1, | ||||
|     SaveData = 2, | ||||
| }; | ||||
|  | ||||
| /** | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <cinttypes> | ||||
| #include "common/logging/log.h" | ||||
| #include "core/core.h" | ||||
| #include "core/file_sys/filesystem.h" | ||||
| @@ -65,10 +66,186 @@ private: | ||||
|     } | ||||
| }; | ||||
|  | ||||
| class IFile final : public ServiceFramework<IFile> { | ||||
| public: | ||||
|     explicit IFile(std::unique_ptr<FileSys::StorageBackend>&& backend) | ||||
|         : ServiceFramework("IFile"), backend(std::move(backend)) { | ||||
|         static const FunctionInfo functions[] = { | ||||
|             {0, &IFile::Read, "Read"}, {1, &IFile::Write, "Write"}, {2, nullptr, "Flush"}, | ||||
|             {3, nullptr, "SetSize"},   {4, nullptr, "GetSize"}, | ||||
|         }; | ||||
|         RegisterHandlers(functions); | ||||
|     } | ||||
|  | ||||
| private: | ||||
|     std::unique_ptr<FileSys::StorageBackend> backend; | ||||
|  | ||||
|     void Read(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|         const u64 unk = rp.Pop<u64>(); | ||||
|         const s64 offset = rp.Pop<s64>(); | ||||
|         const s64 length = rp.Pop<s64>(); | ||||
|  | ||||
|         LOG_DEBUG(Service_FS, "called, offset=0x%llx, length=0x%llx", offset, length); | ||||
|  | ||||
|         // Error checking | ||||
|         if (length < 0) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidLength)); | ||||
|             return; | ||||
|         } | ||||
|         if (offset < 0) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidOffset)); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Read the data from the Storage backend | ||||
|         std::vector<u8> output(length); | ||||
|         ResultVal<size_t> res = backend->Read(offset, length, output.data()); | ||||
|         if (res.Failed()) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(res.Code()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Write the data to memory | ||||
|         ctx.WriteBuffer(output); | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 4}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|         rb.Push(static_cast<u64>(*res)); | ||||
|     } | ||||
|  | ||||
|     void Write(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|         const u64 unk = rp.Pop<u64>(); | ||||
|         const s64 offset = rp.Pop<s64>(); | ||||
|         const s64 length = rp.Pop<s64>(); | ||||
|  | ||||
|         LOG_DEBUG(Service_FS, "called, offset=0x%llx, length=0x%llx", offset, length); | ||||
|  | ||||
|         // Error checking | ||||
|         if (length < 0) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidLength)); | ||||
|             return; | ||||
|         } | ||||
|         if (offset < 0) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidOffset)); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Write the data to the Storage backend | ||||
|         std::vector<u8> data = ctx.ReadBuffer(); | ||||
|         ResultVal<size_t> res = backend->Write(offset, length, true, data.data()); | ||||
|         if (res.Failed()) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(res.Code()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 2}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| class IFileSystem final : public ServiceFramework<IFileSystem> { | ||||
| public: | ||||
|     explicit IFileSystem(std::unique_ptr<FileSys::FileSystemBackend>&& backend) | ||||
|         : ServiceFramework("IFileSystem"), backend(std::move(backend)) { | ||||
|         static const FunctionInfo functions[] = { | ||||
|             {0, &IFileSystem::CreateFile, "CreateFile"}, | ||||
|             {7, &IFileSystem::GetEntryType, "GetEntryType"}, | ||||
|             {8, &IFileSystem::OpenFile, "OpenFile"}, | ||||
|             {10, &IFileSystem::Commit, "Commit"}, | ||||
|         }; | ||||
|         RegisterHandlers(functions); | ||||
|     } | ||||
|  | ||||
|     void CreateFile(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|  | ||||
|         auto file_buffer = ctx.ReadBuffer(); | ||||
|         auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0'); | ||||
|  | ||||
|         std::string name(file_buffer.begin(), end); | ||||
|  | ||||
|         u64 mode = rp.Pop<u64>(); | ||||
|         u32 size = rp.Pop<u32>(); | ||||
|  | ||||
|         LOG_DEBUG(Service_FS, "called file %s mode 0x%" PRIX64 " size 0x%08X", name.c_str(), mode, | ||||
|                   size); | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 2}; | ||||
|         rb.Push(backend->CreateFile(name, size)); | ||||
|     } | ||||
|  | ||||
|     void OpenFile(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|  | ||||
|         auto file_buffer = ctx.ReadBuffer(); | ||||
|         auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0'); | ||||
|  | ||||
|         std::string name(file_buffer.begin(), end); | ||||
|  | ||||
|         auto mode = static_cast<FileSys::Mode>(rp.Pop<u32>()); | ||||
|  | ||||
|         LOG_DEBUG(Service_FS, "called file %s mode %u", name.c_str(), static_cast<u32>(mode)); | ||||
|  | ||||
|         auto result = backend->OpenFile(name, mode); | ||||
|         if (result.Failed()) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(result.Code()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         auto file = std::move(result.Unwrap()); | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|         rb.PushIpcInterface<IFile>(std::move(file)); | ||||
|     } | ||||
|  | ||||
|     void GetEntryType(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|  | ||||
|         auto file_buffer = ctx.ReadBuffer(); | ||||
|         auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0'); | ||||
|  | ||||
|         std::string name(file_buffer.begin(), end); | ||||
|  | ||||
|         LOG_DEBUG(Service_FS, "called file %s", name.c_str()); | ||||
|  | ||||
|         auto result = backend->GetEntryType(name); | ||||
|         if (result.Failed()) { | ||||
|             IPC::ResponseBuilder rb{ctx, 2}; | ||||
|             rb.Push(result.Code()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 3}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|         rb.Push<u32>(static_cast<u32>(*result)); | ||||
|     } | ||||
|  | ||||
|     void Commit(Kernel::HLERequestContext& ctx) { | ||||
|         LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 2}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|     } | ||||
|  | ||||
| private: | ||||
|     std::unique_ptr<FileSys::FileSystemBackend> backend; | ||||
| }; | ||||
|  | ||||
| FSP_SRV::FSP_SRV() : ServiceFramework("fsp-srv") { | ||||
|     static const FunctionInfo functions[] = { | ||||
|         {1, &FSP_SRV::Initalize, "Initalize"}, | ||||
|         {18, &FSP_SRV::MountSdCard, "MountSdCard"}, | ||||
|         {51, &FSP_SRV::MountSaveData, "MountSaveData"}, | ||||
|         {200, &FSP_SRV::OpenDataStorageByCurrentProcess, "OpenDataStorageByCurrentProcess"}, | ||||
|         {202, nullptr, "OpenDataStorageByDataId"}, | ||||
|         {203, &FSP_SRV::OpenRomStorage, "OpenRomStorage"}, | ||||
| @@ -102,6 +279,17 @@ void FSP_SRV::MountSdCard(Kernel::HLERequestContext& ctx) { | ||||
|     rb.Push(RESULT_SUCCESS); | ||||
| } | ||||
|  | ||||
| void FSP_SRV::MountSaveData(Kernel::HLERequestContext& ctx) { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|  | ||||
|     FileSys::Path unused; | ||||
|     auto filesystem = OpenFileSystem(Type::SaveData, unused).Unwrap(); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||
|     rb.Push(RESULT_SUCCESS); | ||||
|     rb.PushIpcInterface<IFileSystem>(std::move(filesystem)); | ||||
| } | ||||
|  | ||||
| void FSP_SRV::GetGlobalAccessLogMode(Kernel::HLERequestContext& ctx) { | ||||
|     LOG_WARNING(Service_FS, "(STUBBED) called"); | ||||
|  | ||||
|   | ||||
| @@ -24,6 +24,7 @@ private: | ||||
|  | ||||
|     void Initalize(Kernel::HLERequestContext& ctx); | ||||
|     void MountSdCard(Kernel::HLERequestContext& ctx); | ||||
|     void MountSaveData(Kernel::HLERequestContext& ctx); | ||||
|     void GetGlobalAccessLogMode(Kernel::HLERequestContext& ctx); | ||||
|     void OpenDataStorageByCurrentProcess(Kernel::HLERequestContext& ctx); | ||||
|     void OpenRomStorage(Kernel::HLERequestContext& ctx); | ||||
|   | ||||
| @@ -110,8 +110,6 @@ ResultStatus AppLoader_DeconstructedRomDirectory::Load( | ||||
|         return ResultStatus::Error; | ||||
|     } | ||||
|  | ||||
|     process = Kernel::Process::Create("main"); | ||||
|  | ||||
|     const std::string directory = filepath.substr(0, filepath.find_last_of("/\\")) + DIR_SEP; | ||||
|     const std::string npdm_path = directory + DIR_SEP + "main.npdm"; | ||||
|  | ||||
| @@ -121,13 +119,15 @@ ResultStatus AppLoader_DeconstructedRomDirectory::Load( | ||||
|     } | ||||
|     metadata.Print(); | ||||
|  | ||||
|     process = Kernel::Process::Create("main", metadata.GetTitleID()); | ||||
|  | ||||
|     // Load NSO modules | ||||
|     VAddr next_load_addr{Memory::PROCESS_IMAGE_VADDR}; | ||||
|     for (const auto& module : {"rtld", "main", "subsdk0", "subsdk1", "subsdk2", "subsdk3", | ||||
|                                "subsdk4", "subsdk5", "subsdk6", "subsdk7", "sdk"}) { | ||||
|         const std::string path = directory + DIR_SEP + module; | ||||
|         const VAddr load_addr = next_load_addr; | ||||
|         next_load_addr = AppLoader_NSO::LoadModule(path, load_addr, metadata.GetTitleID()); | ||||
|         next_load_addr = AppLoader_NSO::LoadModule(path, load_addr); | ||||
|         if (next_load_addr) { | ||||
|             LOG_DEBUG(Loader, "loaded module %s @ 0x%" PRIx64, module, load_addr); | ||||
|         } else { | ||||
|   | ||||
| @@ -300,7 +300,7 @@ SharedPtr<CodeSet> ElfReader::LoadInto(u32 vaddr) { | ||||
|     std::vector<u8> program_image(total_image_size); | ||||
|     size_t current_image_position = 0; | ||||
|  | ||||
|     SharedPtr<CodeSet> codeset = CodeSet::Create("", 0); | ||||
|     SharedPtr<CodeSet> codeset = CodeSet::Create(""); | ||||
|  | ||||
|     for (unsigned int i = 0; i < header->e_phnum; ++i) { | ||||
|         Elf32_Phdr* p = &segments[i]; | ||||
| @@ -406,7 +406,7 @@ ResultStatus AppLoader_ELF::Load(Kernel::SharedPtr<Kernel::Process>& process) { | ||||
|     SharedPtr<CodeSet> codeset = elf_reader.LoadInto(Memory::PROCESS_IMAGE_VADDR); | ||||
|     codeset->name = filename; | ||||
|  | ||||
|     process = Kernel::Process::Create("main"); | ||||
|     process = Kernel::Process::Create("main", 0); | ||||
|     process->LoadModule(codeset, codeset->entrypoint); | ||||
|     process->svc_access_mask.set(); | ||||
|     process->address_mappings = default_address_mappings; | ||||
|   | ||||
| @@ -83,7 +83,7 @@ bool AppLoader_NRO::LoadNro(const std::string& path, VAddr load_base) { | ||||
|     } | ||||
|  | ||||
|     // Build program image | ||||
|     Kernel::SharedPtr<Kernel::CodeSet> codeset = Kernel::CodeSet::Create("", 0); | ||||
|     Kernel::SharedPtr<Kernel::CodeSet> codeset = Kernel::CodeSet::Create(""); | ||||
|     std::vector<u8> program_image; | ||||
|     program_image.resize(PageAlignSize(nro_header.file_size)); | ||||
|     file.Seek(0, SEEK_SET); | ||||
| @@ -125,7 +125,7 @@ ResultStatus AppLoader_NRO::Load(Kernel::SharedPtr<Kernel::Process>& process) { | ||||
|         return ResultStatus::Error; | ||||
|     } | ||||
|  | ||||
|     process = Kernel::Process::Create("main"); | ||||
|     process = Kernel::Process::Create("main", 0); | ||||
|  | ||||
|     // Load NRO | ||||
|     static constexpr VAddr base_addr{Memory::PROCESS_IMAGE_VADDR}; | ||||
|   | ||||
| @@ -92,7 +92,7 @@ static constexpr u32 PageAlignSize(u32 size) { | ||||
|     return (size + Memory::PAGE_MASK) & ~Memory::PAGE_MASK; | ||||
| } | ||||
|  | ||||
| VAddr AppLoader_NSO::LoadModule(const std::string& path, VAddr load_base, u64 tid) { | ||||
| VAddr AppLoader_NSO::LoadModule(const std::string& path, VAddr load_base) { | ||||
|     FileUtil::IOFile file(path, "rb"); | ||||
|     if (!file.IsOpen()) { | ||||
|         return {}; | ||||
| @@ -109,7 +109,7 @@ VAddr AppLoader_NSO::LoadModule(const std::string& path, VAddr load_base, u64 ti | ||||
|     } | ||||
|  | ||||
|     // Build program image | ||||
|     Kernel::SharedPtr<Kernel::CodeSet> codeset = Kernel::CodeSet::Create("", tid); | ||||
|     Kernel::SharedPtr<Kernel::CodeSet> codeset = Kernel::CodeSet::Create(""); | ||||
|     std::vector<u8> program_image; | ||||
|     for (int i = 0; i < nso_header.segments.size(); ++i) { | ||||
|         std::vector<u8> data = | ||||
| @@ -155,10 +155,10 @@ ResultStatus AppLoader_NSO::Load(Kernel::SharedPtr<Kernel::Process>& process) { | ||||
|         return ResultStatus::Error; | ||||
|     } | ||||
|  | ||||
|     process = Kernel::Process::Create("main"); | ||||
|     process = Kernel::Process::Create("main", 0); | ||||
|  | ||||
|     // Load module | ||||
|     LoadModule(filepath, Memory::PROCESS_IMAGE_VADDR, 0); | ||||
|     LoadModule(filepath, Memory::PROCESS_IMAGE_VADDR); | ||||
|     LOG_DEBUG(Loader, "loaded module %s @ 0x%" PRIx64, filepath.c_str(), | ||||
|               Memory::PROCESS_IMAGE_VADDR); | ||||
|  | ||||
|   | ||||
| @@ -29,7 +29,7 @@ public: | ||||
|         return IdentifyType(file, filepath); | ||||
|     } | ||||
|  | ||||
|     static VAddr LoadModule(const std::string& path, VAddr load_base, u64 tid); | ||||
|     static VAddr LoadModule(const std::string& path, VAddr load_base); | ||||
|  | ||||
|     ResultStatus Load(Kernel::SharedPtr<Kernel::Process>& process) override; | ||||
|  | ||||
|   | ||||
| @@ -15,7 +15,7 @@ static Memory::PageTable* page_table = nullptr; | ||||
| TestEnvironment::TestEnvironment(bool mutable_memory_) | ||||
|     : mutable_memory(mutable_memory_), test_memory(std::make_shared<TestMemory>(this)) { | ||||
|  | ||||
|     Kernel::g_current_process = Kernel::Process::Create(""); | ||||
|     Kernel::g_current_process = Kernel::Process::Create("", 0); | ||||
|     page_table = &Kernel::g_current_process->vm_manager.page_table; | ||||
|  | ||||
|     page_table->pointers.fill(nullptr); | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
|  | ||||
| TEST_CASE("Memory::IsValidVirtualAddress", "[core][memory][!hide]") { | ||||
|     SECTION("these regions should not be mapped on an empty process") { | ||||
|         auto process = Kernel::Process::Create(""); | ||||
|         auto process = Kernel::Process::Create("", 0); | ||||
|         CHECK(Memory::IsValidVirtualAddress(*process, Memory::PROCESS_IMAGE_VADDR) == false); | ||||
|         CHECK(Memory::IsValidVirtualAddress(*process, Memory::HEAP_VADDR) == false); | ||||
|         CHECK(Memory::IsValidVirtualAddress(*process, Memory::LINEAR_HEAP_VADDR) == false); | ||||
| @@ -20,14 +20,14 @@ TEST_CASE("Memory::IsValidVirtualAddress", "[core][memory][!hide]") { | ||||
|     } | ||||
|  | ||||
|     SECTION("CONFIG_MEMORY_VADDR and SHARED_PAGE_VADDR should be valid after mapping them") { | ||||
|         auto process = Kernel::Process::Create(""); | ||||
|         auto process = Kernel::Process::Create("", 0); | ||||
|         Kernel::MapSharedPages(process->vm_manager); | ||||
|         CHECK(Memory::IsValidVirtualAddress(*process, Memory::CONFIG_MEMORY_VADDR) == true); | ||||
|         CHECK(Memory::IsValidVirtualAddress(*process, Memory::SHARED_PAGE_VADDR) == true); | ||||
|     } | ||||
|  | ||||
|     SECTION("special regions should be valid after mapping them") { | ||||
|         auto process = Kernel::Process::Create(""); | ||||
|         auto process = Kernel::Process::Create("", 0); | ||||
|         SECTION("VRAM") { | ||||
|             Kernel::HandleSpecialMapping(process->vm_manager, | ||||
|                                          {Memory::VRAM_VADDR, Memory::VRAM_SIZE, false, false}); | ||||
| @@ -48,7 +48,7 @@ TEST_CASE("Memory::IsValidVirtualAddress", "[core][memory][!hide]") { | ||||
|     } | ||||
|  | ||||
|     SECTION("Unmapping a VAddr should make it invalid") { | ||||
|         auto process = Kernel::Process::Create(""); | ||||
|         auto process = Kernel::Process::Create("", 0); | ||||
|         Kernel::MapSharedPages(process->vm_manager); | ||||
|         process->vm_manager.UnmapRange(Memory::CONFIG_MEMORY_VADDR, Memory::CONFIG_MEMORY_SIZE); | ||||
|         CHECK(Memory::IsValidVirtualAddress(*process, Memory::CONFIG_MEMORY_VADDR) == false); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user