| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <unordered_map>
 | 
					
						
							| 
									
										
										
										
											2014-12-14 03:30:11 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | #include <boost/container/flat_map.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | #include "common/file_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-20 03:21:23 -02:00
										 |  |  | #include "common/make_unique.h"
 | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  | #include "common/math_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 02:51:38 -02:00
										 |  |  | #include "core/file_sys/archive_backend.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-31 19:36:50 -05:00
										 |  |  | #include "core/file_sys/archive_extsavedata.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/archive_romfs.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/archive_savedata.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-01 12:39:27 -05:00
										 |  |  | #include "core/file_sys/archive_savedatacheck.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | #include "core/file_sys/archive_sdmc.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-24 18:02:40 -05:00
										 |  |  | #include "core/file_sys/archive_systemsavedata.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-15 04:59:29 -02:00
										 |  |  | #include "core/file_sys/directory_backend.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "core/hle/service/service.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-14 04:32:45 -02:00
										 |  |  | #include "core/hle/service/fs/archive.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "core/hle/service/fs/fs_user.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  | #include "core/hle/result.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 22:06:23 -02:00
										 |  |  | // Specializes std::hash for ArchiveIdCode, so that we can use it in std::unordered_map.
 | 
					
						
							|  |  |  | // Workaroung for libstdc++ bug: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=60970
 | 
					
						
							|  |  |  | namespace std { | 
					
						
							|  |  |  |     template <> | 
					
						
							|  |  |  |     struct hash<Service::FS::ArchiveIdCode> { | 
					
						
							|  |  |  |         typedef Service::FS::ArchiveIdCode argument_type; | 
					
						
							|  |  |  |         typedef std::size_t result_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         result_type operator()(const argument_type& id_code) const { | 
					
						
							|  |  |  |             typedef std::underlying_type<argument_type>::type Type; | 
					
						
							|  |  |  |             return std::hash<Type>()(static_cast<Type>(id_code)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 15:02:30 -05:00
										 |  |  | /// TODO(Subv): Confirm length of these strings
 | 
					
						
							| 
									
										
										
										
											2015-01-04 09:10:27 -05:00
										 |  |  | const std::string SYSTEM_ID = "00000000000000000000000000000000"; | 
					
						
							|  |  |  | const std::string SDCARD_ID = "00000000000000000000000000000000"; | 
					
						
							| 
									
										
										
										
											2015-01-03 20:46:05 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 05:55:11 -02:00
										 |  |  | namespace Service { | 
					
						
							|  |  |  | namespace FS { | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  | // TODO: Verify code
 | 
					
						
							|  |  |  | /// Returned when a function is passed an invalid handle.
 | 
					
						
							|  |  |  | const ResultCode ERR_INVALID_HANDLE(ErrorDescription::InvalidHandle, ErrorModule::FS, | 
					
						
							|  |  |  |         ErrorSummary::InvalidArgument, ErrorLevel::Permanent); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | // Command to access archive file
 | 
					
						
							|  |  |  | enum class FileCommand : u32 { | 
					
						
							|  |  |  |     Dummy1          = 0x000100C6, | 
					
						
							|  |  |  |     Control         = 0x040100C4, | 
					
						
							|  |  |  |     OpenSubFile     = 0x08010100, | 
					
						
							|  |  |  |     Read            = 0x080200C2, | 
					
						
							|  |  |  |     Write           = 0x08030102, | 
					
						
							|  |  |  |     GetSize         = 0x08040000, | 
					
						
							|  |  |  |     SetSize         = 0x08050080, | 
					
						
							|  |  |  |     GetAttributes   = 0x08060000, | 
					
						
							|  |  |  |     SetAttributes   = 0x08070040, | 
					
						
							|  |  |  |     Close           = 0x08080000, | 
					
						
							|  |  |  |     Flush           = 0x08090000, | 
					
						
							| 
									
										
										
										
											2014-12-31 19:36:50 -05:00
										 |  |  |     SetPriority     = 0x080A0040, | 
					
						
							|  |  |  |     GetPriority     = 0x080B0000, | 
					
						
							|  |  |  |     OpenLinkFile    = 0x080C0000, | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | // Command to access directory
 | 
					
						
							|  |  |  | enum class DirectoryCommand : u32 { | 
					
						
							|  |  |  |     Dummy1          = 0x000100C6, | 
					
						
							|  |  |  |     Control         = 0x040100C4, | 
					
						
							|  |  |  |     Read            = 0x08010042, | 
					
						
							|  |  |  |     Close           = 0x08020000, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 02:15:46 -03:00
										 |  |  | File::File(std::unique_ptr<FileSys::FileBackend>&& backend, const FileSys::Path & path) | 
					
						
							|  |  |  |     : path(path), priority(0), backend(std::move(backend)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | File::~File() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  | ResultVal<bool> File::SyncRequest() { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  |     FileCommand cmd = static_cast<FileCommand>(cmd_buff[0]); | 
					
						
							|  |  |  |     switch (cmd) { | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Read from file...
 | 
					
						
							|  |  |  |         case FileCommand::Read: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |             u64 offset = cmd_buff[1] | ((u64)cmd_buff[2]) << 32; | 
					
						
							|  |  |  |             u32 length = cmd_buff[3]; | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |             u32 address = cmd_buff[5]; | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "Read %s %s: offset=0x%llx length=%d address=0x%x", | 
					
						
							| 
									
										
										
										
											2015-02-01 15:31:21 -05:00
										 |  |  |                       GetTypeName().c_str(), GetName().c_str(), offset, length, address); | 
					
						
							|  |  |  |             cmd_buff[2] = static_cast<u32>(backend->Read(offset, length, Memory::GetPointer(address))); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Write to file...
 | 
					
						
							|  |  |  |         case FileCommand::Write: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |             u64 offset = cmd_buff[1] | ((u64)cmd_buff[2]) << 32; | 
					
						
							|  |  |  |             u32 length = cmd_buff[3]; | 
					
						
							|  |  |  |             u32 flush = cmd_buff[4]; | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |             u32 address = cmd_buff[6]; | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "Write %s %s: offset=0x%llx length=%d address=0x%x, flush=0x%x", | 
					
						
							| 
									
										
										
										
											2015-02-01 15:31:21 -05:00
										 |  |  |                       GetTypeName().c_str(), GetName().c_str(), offset, length, address, flush); | 
					
						
							|  |  |  |             cmd_buff[2] = static_cast<u32>(backend->Write(offset, length, flush, Memory::GetPointer(address))); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case FileCommand::GetSize: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "GetSize %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |             u64 size = backend->GetSize(); | 
					
						
							|  |  |  |             cmd_buff[2] = (u32)size; | 
					
						
							|  |  |  |             cmd_buff[3] = size >> 32; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-27 19:16:51 +00:00
										 |  |  |         case FileCommand::SetSize: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u64 size = cmd_buff[1] | ((u64)cmd_buff[2] << 32); | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "SetSize %s %s size=%llu", | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |                 GetTypeName().c_str(), GetName().c_str(), size); | 
					
						
							| 
									
										
										
										
											2014-09-27 19:16:51 +00:00
										 |  |  |             backend->SetSize(size); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |         case FileCommand::Close: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "Close %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |             backend->Close(); | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |         case FileCommand::Flush: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             LOG_TRACE(Service_FS, "Flush"); | 
					
						
							|  |  |  |             backend->Flush(); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-31 19:36:50 -05:00
										 |  |  |         case FileCommand::OpenLinkFile: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             LOG_WARNING(Service_FS, "(STUBBED) File command OpenLinkFile %s", GetName().c_str()); | 
					
						
							| 
									
										
										
										
											2015-01-31 10:37:32 -02:00
										 |  |  |             cmd_buff[3] = Kernel::g_handle_table.Create(this).ValueOr(INVALID_HANDLE); | 
					
						
							| 
									
										
										
										
											2014-12-31 19:36:50 -05:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case FileCommand::SetPriority: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             priority = cmd_buff[1]; | 
					
						
							|  |  |  |             LOG_TRACE(Service_FS, "SetPriority %u", priority); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case FileCommand::GetPriority: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             cmd_buff[2] = priority; | 
					
						
							|  |  |  |             LOG_TRACE(Service_FS, "GetPriority"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |         // Unknown command...
 | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_ERROR(Service_FS, "Unknown command=0x%08X!", cmd); | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  |             ResultCode error = UnimplementedFunction(ErrorModule::FS); | 
					
						
							|  |  |  |             cmd_buff[1] = error.raw; // TODO(Link Mauve): use the correct error code for that.
 | 
					
						
							|  |  |  |             return error; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; // No error
 | 
					
						
							|  |  |  |     return MakeResult<bool>(false); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 02:15:46 -03:00
										 |  |  | Directory::Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend, const FileSys::Path & path) | 
					
						
							|  |  |  |     : path(path), backend(std::move(backend)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Directory::~Directory() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  | ResultVal<bool> Directory::SyncRequest() { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  |     DirectoryCommand cmd = static_cast<DirectoryCommand>(cmd_buff[0]); | 
					
						
							|  |  |  |     switch (cmd) { | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Read from directory...
 | 
					
						
							|  |  |  |         case DirectoryCommand::Read: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u32 count = cmd_buff[1]; | 
					
						
							|  |  |  |             u32 address = cmd_buff[3]; | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  |             auto entries = reinterpret_cast<FileSys::Entry*>(Memory::GetPointer(address)); | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "Read %s %s: count=%d", | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |                 GetTypeName().c_str(), GetName().c_str(), count); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Number of entries actually read
 | 
					
						
							|  |  |  |             cmd_buff[2] = backend->Read(count, entries); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |         case DirectoryCommand::Close: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_TRACE(Service_FS, "Close %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							| 
									
										
										
										
											2014-12-21 10:04:08 -02:00
										 |  |  |             backend->Close(); | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  |         // Unknown command...
 | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |             LOG_ERROR(Service_FS, "Unknown command=0x%08X!", cmd); | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  |             ResultCode error = UnimplementedFunction(ErrorModule::FS); | 
					
						
							|  |  |  |             cmd_buff[1] = error.raw; // TODO(Link Mauve): use the correct error code for that.
 | 
					
						
							| 
									
										
										
										
											2014-12-14 03:30:11 -02:00
										 |  |  |             return MakeResult<bool>(false); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; // No error
 | 
					
						
							|  |  |  |     return MakeResult<bool>(false); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  | using FileSys::ArchiveBackend; | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | using FileSys::ArchiveFactory; | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Map of registered archives, identified by id code. Once an archive is registered here, it is | 
					
						
							|  |  |  |  * never removed until the FS service is shut down. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | static boost::container::flat_map<ArchiveIdCode, std::unique_ptr<ArchiveFactory>> id_code_map; | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Map of active archive handles. Values are pointers to the archives in `idcode_map`. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | static std::unordered_map<ArchiveHandle, std::unique_ptr<ArchiveBackend>> handle_map; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | static ArchiveHandle next_handle; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  | static ArchiveBackend* GetArchive(ArchiveHandle handle) { | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     auto itr = handle_map.find(handle); | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     return (itr == handle_map.end()) ? nullptr : itr->second.get(); | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  | ResultVal<ArchiveHandle> OpenArchive(ArchiveIdCode id_code, FileSys::Path& archive_path) { | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     LOG_TRACE(Service_FS, "Opening archive with id code 0x%08X", id_code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto itr = id_code_map.find(id_code); | 
					
						
							|  |  |  |     if (itr == id_code_map.end()) { | 
					
						
							|  |  |  |         // TODO: Verify error against hardware
 | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  |         return ResultCode(ErrorDescription::NotFound, ErrorModule::FS, | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |                           ErrorSummary::NotFound, ErrorLevel::Permanent); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     CASCADE_RESULT(std::unique_ptr<ArchiveBackend> res, itr->second->Open(archive_path)); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     // This should never even happen in the first place with 64-bit handles, 
 | 
					
						
							|  |  |  |     while (handle_map.count(next_handle) != 0) { | 
					
						
							|  |  |  |         ++next_handle; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     handle_map.emplace(next_handle, std::move(res)); | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     return MakeResult<ArchiveHandle>(next_handle++); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | ResultCode CloseArchive(ArchiveHandle handle) { | 
					
						
							|  |  |  |     if (handle_map.erase(handle) == 0) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     else | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | // TODO(yuriks): This might be what the fs:REG service is for. See the Register/Unregister calls in
 | 
					
						
							|  |  |  | // http://3dbrew.org/wiki/Filesystem_services#ProgramRegistry_service_.22fs:REG.22
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | ResultCode RegisterArchiveType(std::unique_ptr<FileSys::ArchiveFactory>&& factory, ArchiveIdCode id_code) { | 
					
						
							|  |  |  |     auto result = id_code_map.emplace(id_code, std::move(factory)); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     bool inserted = result.second; | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  |     ASSERT_MSG(inserted, "Tried to register more than one archive with same id code"); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     auto& archive = result.first->second; | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Registered archive %s with id code 0x%08X", archive->GetName().c_str(), id_code); | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  | ResultVal<Kernel::SharedPtr<File>> OpenFileFromArchive(ArchiveHandle archive_handle, | 
					
						
							| 
									
										
										
										
											2015-01-30 12:32:46 -02:00
										 |  |  |         const FileSys::Path& path, const FileSys::Mode mode) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* archive = GetArchive(archive_handle); | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     if (archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     std::unique_ptr<FileSys::FileBackend> backend = archive->OpenFile(path, mode); | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     if (backend == nullptr) { | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |         return ResultCode(ErrorDescription::FS_NotFound, ErrorModule::FS, | 
					
						
							|  |  |  |                           ErrorSummary::NotFound, ErrorLevel::Status); | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-09-27 19:21:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 12:32:46 -02:00
										 |  |  |     auto file = Kernel::SharedPtr<File>(new File(std::move(backend), path)); | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |     return MakeResult<Kernel::SharedPtr<File>>(std::move(file)); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | ResultCode DeleteFileFromArchive(ArchiveHandle archive_handle, const FileSys::Path& path) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* archive = GetArchive(archive_handle); | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  |     if (archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     if (archive->DeleteFile(path)) | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     return ResultCode(ErrorDescription::NoData, ErrorModule::FS, // TODO: verify description
 | 
					
						
							|  |  |  |                       ErrorSummary::Canceled, ErrorLevel::Status); | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | ResultCode RenameFileBetweenArchives(ArchiveHandle src_archive_handle, const FileSys::Path& src_path, | 
					
						
							|  |  |  |                                      ArchiveHandle dest_archive_handle, const FileSys::Path& dest_path) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* src_archive = GetArchive(src_archive_handle); | 
					
						
							|  |  |  |     ArchiveBackend* dest_archive = GetArchive(dest_archive_handle); | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  |     if (src_archive == nullptr || dest_archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  |     if (src_archive == dest_archive) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |         if (src_archive->RenameFile(src_path, dest_path)) | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |             return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // TODO: Implement renaming across archives
 | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |         return UnimplementedFunction(ErrorModule::FS); | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO(yuriks): This code probably isn't right, it'll return a Status even if the file didn't
 | 
					
						
							|  |  |  |     // exist or similar. Verify.
 | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |     return ResultCode(ErrorDescription::NoData, ErrorModule::FS, // TODO: verify description
 | 
					
						
							|  |  |  |                       ErrorSummary::NothingHappened, ErrorLevel::Status); | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | ResultCode DeleteDirectoryFromArchive(ArchiveHandle archive_handle, const FileSys::Path& path) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* archive = GetArchive(archive_handle); | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  |     if (archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     if (archive->DeleteDirectory(path)) | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     return ResultCode(ErrorDescription::NoData, ErrorModule::FS, // TODO: verify description
 | 
					
						
							|  |  |  |                       ErrorSummary::Canceled, ErrorLevel::Status); | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-01 12:39:27 -05:00
										 |  |  | ResultCode CreateFileInArchive(ArchiveHandle archive_handle, const FileSys::Path& path, u32 file_size) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* archive = GetArchive(archive_handle); | 
					
						
							| 
									
										
										
										
											2014-12-20 12:43:50 -03:00
										 |  |  |     if (archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-20 12:43:50 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     return archive->CreateFile(path, file_size); | 
					
						
							| 
									
										
										
										
											2014-12-20 12:43:50 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | ResultCode CreateDirectoryFromArchive(ArchiveHandle archive_handle, const FileSys::Path& path) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* archive = GetArchive(archive_handle); | 
					
						
							| 
									
										
										
										
											2014-10-28 22:52:56 -07:00
										 |  |  |     if (archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     if (archive->CreateDirectory(path)) | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     return ResultCode(ErrorDescription::NoData, ErrorModule::FS, // TODO: verify description
 | 
					
						
							|  |  |  |                       ErrorSummary::Canceled, ErrorLevel::Status); | 
					
						
							| 
									
										
										
										
											2014-10-28 22:52:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | ResultCode RenameDirectoryBetweenArchives(ArchiveHandle src_archive_handle, const FileSys::Path& src_path, | 
					
						
							|  |  |  |                                           ArchiveHandle dest_archive_handle, const FileSys::Path& dest_path) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* src_archive = GetArchive(src_archive_handle); | 
					
						
							|  |  |  |     ArchiveBackend* dest_archive = GetArchive(dest_archive_handle); | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  |     if (src_archive == nullptr || dest_archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  |     if (src_archive == dest_archive) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |         if (src_archive->RenameDirectory(src_path, dest_path)) | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |             return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // TODO: Implement renaming across archives
 | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |         return UnimplementedFunction(ErrorModule::FS); | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO(yuriks): This code probably isn't right, it'll return a Status even if the file didn't
 | 
					
						
							|  |  |  |     // exist or similar. Verify.
 | 
					
						
							| 
									
										
										
										
											2014-11-24 15:45:20 -08:00
										 |  |  |     return ResultCode(ErrorDescription::NoData, ErrorModule::FS, // TODO: verify description
 | 
					
						
							|  |  |  |                       ErrorSummary::NothingHappened, ErrorLevel::Status); | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  | ResultVal<Kernel::SharedPtr<Directory>> OpenDirectoryFromArchive(ArchiveHandle archive_handle, | 
					
						
							| 
									
										
										
										
											2015-01-30 12:32:46 -02:00
										 |  |  |         const FileSys::Path& path) { | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     ArchiveBackend* archive = GetArchive(archive_handle); | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     if (archive == nullptr) | 
					
						
							| 
									
										
										
										
											2015-01-23 03:44:52 -02:00
										 |  |  |         return ERR_INVALID_HANDLE; | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 22:32:15 -02:00
										 |  |  |     std::unique_ptr<FileSys::DirectoryBackend> backend = archive->OpenDirectory(path); | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     if (backend == nullptr) { | 
					
						
							| 
									
										
										
										
											2014-12-07 14:40:27 -08:00
										 |  |  |         return ResultCode(ErrorDescription::NotFound, ErrorModule::FS, | 
					
						
							|  |  |  |                           ErrorSummary::NotFound, ErrorLevel::Permanent); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 12:32:46 -02:00
										 |  |  |     auto directory = Kernel::SharedPtr<Directory>(new Directory(std::move(backend), path)); | 
					
						
							| 
									
										
										
										
											2015-02-07 13:06:48 -05:00
										 |  |  |     return MakeResult<Kernel::SharedPtr<Directory>>(std::move(directory)); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 13:31:34 -05:00
										 |  |  | ResultCode FormatArchive(ArchiveIdCode id_code, const FileSys::Path& path) { | 
					
						
							|  |  |  |     auto archive_itr = id_code_map.find(id_code); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  |     if (archive_itr == id_code_map.end()) { | 
					
						
							|  |  |  |         return UnimplementedFunction(ErrorModule::FS); // TODO(Subv): Find the right error
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 13:31:34 -05:00
										 |  |  |     return archive_itr->second->Format(path); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-14 12:00:01 -05:00
										 |  |  | ResultCode CreateExtSaveData(MediaType media_type, u32 high, u32 low) { | 
					
						
							| 
									
										
										
										
											2015-01-13 23:56:00 -05:00
										 |  |  |     // Construct the binary path to the archive first
 | 
					
						
							| 
									
										
										
										
											2015-03-14 12:00:01 -05:00
										 |  |  |     FileSys::Path path = FileSys::ConstructExtDataBinaryPath(static_cast<u32>(media_type), high, low); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string media_type_directory; | 
					
						
							|  |  |  |     if (media_type == MediaType::NAND) { | 
					
						
							|  |  |  |         media_type_directory = FileUtil::GetUserPath(D_NAND_IDX); | 
					
						
							|  |  |  |     } else if (media_type == MediaType::SDMC) { | 
					
						
							|  |  |  |         media_type_directory = FileUtil::GetUserPath(D_SDMC_IDX); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         LOG_ERROR(Service_FS, "Unsupported media type %u", media_type); | 
					
						
							|  |  |  |         return ResultCode(-1); // TODO(Subv): Find the right error code
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string base_path = FileSys::GetExtDataContainerPath(media_type_directory, media_type == MediaType::NAND); | 
					
						
							| 
									
										
										
										
											2015-01-13 23:56:00 -05:00
										 |  |  |     std::string extsavedata_path = FileSys::GetExtSaveDataPath(base_path, path); | 
					
						
							|  |  |  |     if (!FileUtil::CreateFullPath(extsavedata_path)) | 
					
						
							|  |  |  |         return ResultCode(-1); // TODO(Subv): Find the right error code
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-14 12:00:01 -05:00
										 |  |  | ResultCode DeleteExtSaveData(MediaType media_type, u32 high, u32 low) { | 
					
						
							|  |  |  |     // Construct the binary path to the archive first
 | 
					
						
							|  |  |  |     FileSys::Path path = FileSys::ConstructExtDataBinaryPath(static_cast<u32>(media_type), high, low); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string media_type_directory; | 
					
						
							|  |  |  |     if (media_type == MediaType::NAND) { | 
					
						
							|  |  |  |         media_type_directory = FileUtil::GetUserPath(D_NAND_IDX); | 
					
						
							|  |  |  |     } else if (media_type == MediaType::SDMC) { | 
					
						
							|  |  |  |         media_type_directory = FileUtil::GetUserPath(D_SDMC_IDX); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         LOG_ERROR(Service_FS, "Unsupported media type %u", media_type); | 
					
						
							|  |  |  |         return ResultCode(-1); // TODO(Subv): Find the right error code
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string base_path = FileSys::GetExtDataContainerPath(media_type_directory, media_type == MediaType::NAND); | 
					
						
							|  |  |  |     std::string extsavedata_path = FileSys::GetExtSaveDataPath(base_path, path); | 
					
						
							|  |  |  |     if (!FileUtil::DeleteDirRecursively(extsavedata_path)) | 
					
						
							|  |  |  |         return ResultCode(-1); // TODO(Subv): Find the right error code
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode DeleteSystemSaveData(u32 high, u32 low) { | 
					
						
							|  |  |  |     // Construct the binary path to the archive first
 | 
					
						
							|  |  |  |     FileSys::Path path = FileSys::ConstructSystemSaveDataBinaryPath(high, low); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string nand_directory = FileUtil::GetUserPath(D_NAND_IDX); | 
					
						
							|  |  |  |     std::string base_path = FileSys::GetSystemSaveDataContainerPath(nand_directory); | 
					
						
							|  |  |  |     std::string systemsavedata_path = FileSys::GetSystemSaveDataPath(base_path, path); | 
					
						
							|  |  |  |     if (!FileUtil::DeleteDirRecursively(systemsavedata_path)) | 
					
						
							|  |  |  |         return ResultCode(-1); // TODO(Subv): Find the right error code
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode CreateSystemSaveData(u32 high, u32 low) { | 
					
						
							|  |  |  |     // Construct the binary path to the archive first
 | 
					
						
							|  |  |  |     FileSys::Path path = FileSys::ConstructSystemSaveDataBinaryPath(high, low); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string nand_directory = FileUtil::GetUserPath(D_NAND_IDX); | 
					
						
							|  |  |  |     std::string base_path = FileSys::GetSystemSaveDataContainerPath(nand_directory); | 
					
						
							|  |  |  |     std::string systemsavedata_path = FileSys::GetSystemSaveDataPath(base_path, path); | 
					
						
							|  |  |  |     if (!FileUtil::CreateFullPath(systemsavedata_path)) | 
					
						
							|  |  |  |         return ResultCode(-1); // TODO(Subv): Find the right error code
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 13:38:12 -04:00
										 |  |  | /// Initialize archives
 | 
					
						
							|  |  |  | void ArchiveInit() { | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     next_handle = 1; | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     AddService(new FS::Interface); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 15:34:37 -05:00
										 |  |  |     // TODO(Subv): Add the other archive types (see here for the known types:
 | 
					
						
							| 
									
										
										
										
											2015-01-06 15:02:30 -05:00
										 |  |  |     // http://3dbrew.org/wiki/FS:OpenArchive#Archive_idcodes).
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-20 15:11:38 -07:00
										 |  |  |     std::string sdmc_directory = FileUtil::GetUserPath(D_SDMC_IDX); | 
					
						
							| 
									
										
										
										
											2015-01-03 20:46:05 -05:00
										 |  |  |     std::string nand_directory = FileUtil::GetUserPath(D_NAND_IDX); | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto sdmc_factory = Common::make_unique<FileSys::ArchiveFactory_SDMC>(sdmc_directory); | 
					
						
							|  |  |  |     if (sdmc_factory->Initialize()) | 
					
						
							|  |  |  |         RegisterArchiveType(std::move(sdmc_factory), ArchiveIdCode::SDMC); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |         LOG_ERROR(Service_FS, "Can't instantiate SDMC archive with path %s", sdmc_directory.c_str()); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  |      | 
					
						
							|  |  |  |     // Create the SaveData archive
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto savedata_factory = Common::make_unique<FileSys::ArchiveFactory_SaveData>(sdmc_directory); | 
					
						
							|  |  |  |     RegisterArchiveType(std::move(savedata_factory), ArchiveIdCode::SaveData); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto extsavedata_factory = Common::make_unique<FileSys::ArchiveFactory_ExtSaveData>(sdmc_directory, false); | 
					
						
							|  |  |  |     if (extsavedata_factory->Initialize()) | 
					
						
							|  |  |  |         RegisterArchiveType(std::move(extsavedata_factory), ArchiveIdCode::ExtSaveData); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |         LOG_ERROR(Service_FS, "Can't instantiate ExtSaveData archive with path %s", extsavedata_factory->GetMountPoint().c_str()); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto sharedextsavedata_factory = Common::make_unique<FileSys::ArchiveFactory_ExtSaveData>(nand_directory, true); | 
					
						
							|  |  |  |     if (sharedextsavedata_factory->Initialize()) | 
					
						
							|  |  |  |         RegisterArchiveType(std::move(sharedextsavedata_factory), ArchiveIdCode::SharedExtSaveData); | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  |     else | 
					
						
							|  |  |  |         LOG_ERROR(Service_FS, "Can't instantiate SharedExtSaveData archive with path %s",  | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |             sharedextsavedata_factory->GetMountPoint().c_str()); | 
					
						
							| 
									
										
										
										
											2014-12-31 19:36:50 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Create the SaveDataCheck archive, basically a small variation of the RomFS archive
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto savedatacheck_factory = Common::make_unique<FileSys::ArchiveFactory_SaveDataCheck>(nand_directory); | 
					
						
							|  |  |  |     RegisterArchiveType(std::move(savedatacheck_factory), ArchiveIdCode::SaveDataCheck); | 
					
						
							| 
									
										
										
										
											2015-02-24 18:02:40 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto systemsavedata_factory = Common::make_unique<FileSys::ArchiveFactory_SystemSaveData>(nand_directory); | 
					
						
							|  |  |  |     RegisterArchiveType(std::move(systemsavedata_factory), ArchiveIdCode::SystemSaveData); | 
					
						
							| 
									
										
										
										
											2014-07-04 13:38:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown archives
 | 
					
						
							|  |  |  | void ArchiveShutdown() { | 
					
						
							| 
									
										
										
										
											2014-12-15 06:41:02 -02:00
										 |  |  |     handle_map.clear(); | 
					
						
							|  |  |  |     id_code_map.clear(); | 
					
						
							| 
									
										
										
										
											2014-07-04 13:38:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 05:55:11 -02:00
										 |  |  | } // namespace FS
 | 
					
						
							|  |  |  | } // namespace Service
 |