| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | #include "common/file_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  | #include "common/math_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | #include "core/file_sys/archive.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | #include "core/file_sys/archive_sdmc.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | #include "core/file_sys/directory.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | #include "core/hle/service/service.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | #include "core/hle/kernel/archive.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Kernel namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-09-12 00:48:04 +02:00
										 |  |  | // Command to access directory
 | 
					
						
							|  |  |  | enum class DirectoryCommand : u32 { | 
					
						
							|  |  |  |     Dummy1          = 0x000100C6, | 
					
						
							|  |  |  |     Control         = 0x040100C4, | 
					
						
							|  |  |  |     Read            = 0x08010042, | 
					
						
							|  |  |  |     Close           = 0x08020000, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | class Archive : public Object { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     std::string GetTypeName() const override { return "Archive"; } | 
					
						
							|  |  |  |     std::string GetName() const override { return name; } | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static Kernel::HandleType GetStaticHandleType() { return HandleType::Archive; } | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Kernel::HandleType GetHandleType() const override { return HandleType::Archive; } | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |     std::string name;           ///< Name of archive (optional)
 | 
					
						
							|  |  |  |     FileSys::Archive* backend;  ///< Archive backend interface
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Synchronize kernel object  | 
					
						
							|  |  |  |      * @param wait Boolean wait set if current thread should wait as a result of sync operation | 
					
						
							|  |  |  |      * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Result SyncRequest(bool* wait) override { | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |         u32* cmd_buff = Service::GetCommandBuffer(); | 
					
						
							|  |  |  |         FileCommand cmd = static_cast<FileCommand>(cmd_buff[0]); | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |         switch (cmd) { | 
					
						
							|  |  |  |         // Read from archive...
 | 
					
						
							|  |  |  |         case FileCommand::Read: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |             u64 offset  = cmd_buff[1] | ((u64)cmd_buff[2] << 32); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |             u32 length  = cmd_buff[3]; | 
					
						
							|  |  |  |             u32 address = cmd_buff[5]; | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Number of bytes read
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |             cmd_buff[2] = backend->Read(offset, length, Memory::GetPointer(address)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |         // Write to archive...
 | 
					
						
							|  |  |  |         case FileCommand::Write: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u64 offset  = cmd_buff[1] | ((u64)cmd_buff[2] << 32); | 
					
						
							|  |  |  |             u32 length  = cmd_buff[3]; | 
					
						
							|  |  |  |             u32 flush   = cmd_buff[4]; | 
					
						
							|  |  |  |             u32 address = cmd_buff[6]; | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |             // Number of bytes written
 | 
					
						
							|  |  |  |             cmd_buff[2] = backend->Write(offset, length, flush, Memory::GetPointer(address)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case FileCommand::GetSize: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u64 filesize = (u64) backend->GetSize(); | 
					
						
							|  |  |  |             cmd_buff[2]  = (u32) filesize;         // Lower word
 | 
					
						
							|  |  |  |             cmd_buff[3]  = (u32) (filesize >> 32); // Upper word
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case FileCommand::SetSize: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             backend->SetSize(cmd_buff[1] | ((u64)cmd_buff[2] << 32)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |         case FileCommand::Close: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "Close %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							|  |  |  |             Kernel::g_object_pool.Destroy<Archive>(GetHandle()); | 
					
						
							|  |  |  |             CloseArchive(backend->GetIdCode()); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |         // Unknown command...
 | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |             ERROR_LOG(KERNEL, "Unknown command=0x%08X!", cmd); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |         cmd_buff[1] = 0; // No error
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Wait for kernel object to synchronize | 
					
						
							|  |  |  |      * @param wait Boolean wait set if current thread should wait as a result of sync operation | 
					
						
							|  |  |  |      * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Result WaitSynchronization(bool* wait) override { | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |         // TODO(bunnei): ImplementMe
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |         ERROR_LOG(OSHLE, "(UNIMPLEMENTED)"); | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | class File : public Object { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     std::string GetTypeName() const override { return "File"; } | 
					
						
							|  |  |  |     std::string GetName() const override { return path; } | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static Kernel::HandleType GetStaticHandleType() { return HandleType::File; } | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Kernel::HandleType GetHandleType() const override { return HandleType::File; } | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::string path; ///< Path of the file
 | 
					
						
							|  |  |  |     std::unique_ptr<FileSys::File> backend; ///< File backend interface
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Synchronize kernel object | 
					
						
							|  |  |  |      * @param wait Boolean wait set if current thread should wait as a result of sync operation | 
					
						
							|  |  |  |      * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Result SyncRequest(bool* wait) override { | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |         u32* cmd_buff = Service::GetCommandBuffer(); | 
					
						
							|  |  |  |         FileCommand cmd = static_cast<FileCommand>(cmd_buff[0]); | 
					
						
							|  |  |  |         switch (cmd) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Read from file...
 | 
					
						
							|  |  |  |         case FileCommand::Read: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u64 offset = cmd_buff[1] | ((u64) cmd_buff[2]) << 32; | 
					
						
							|  |  |  |             u32 length  = cmd_buff[3]; | 
					
						
							|  |  |  |             u32 address = cmd_buff[5]; | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "Read %s %s: offset=0x%x length=%d address=0x%x", | 
					
						
							|  |  |  |                       GetTypeName().c_str(), GetName().c_str(), offset, length, address); | 
					
						
							|  |  |  |             cmd_buff[2] = backend->Read(offset, length, Memory::GetPointer(address)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Write to file...
 | 
					
						
							|  |  |  |         case FileCommand::Write: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u64 offset  = cmd_buff[1] | ((u64) cmd_buff[2]) << 32; | 
					
						
							|  |  |  |             u32 length  = cmd_buff[3]; | 
					
						
							|  |  |  |             u32 flush   = cmd_buff[4]; | 
					
						
							|  |  |  |             u32 address = cmd_buff[6]; | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "Write %s %s: offset=0x%x length=%d address=0x%x, flush=0x%x", | 
					
						
							|  |  |  |                       GetTypeName().c_str(), GetName().c_str(), offset, length, address, flush); | 
					
						
							|  |  |  |             cmd_buff[2] = backend->Write(offset, length, flush, Memory::GetPointer(address)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case FileCommand::GetSize: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "GetSize %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							|  |  |  |             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); | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "SetSize %s %s size=%d", GetTypeName().c_str(), GetName().c_str(), size); | 
					
						
							|  |  |  |             backend->SetSize(size); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |         case FileCommand::Close: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "Close %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							|  |  |  |             Kernel::g_object_pool.Destroy<File>(GetHandle()); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |         // Unknown command...
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             ERROR_LOG(KERNEL, "Unknown command=0x%08X!", cmd); | 
					
						
							|  |  |  |             cmd_buff[1] = -1; // TODO(Link Mauve): use the correct error code for that.
 | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         cmd_buff[1] = 0; // No error
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Wait for kernel object to synchronize | 
					
						
							|  |  |  |      * @param wait Boolean wait set if current thread should wait as a result of sync operation | 
					
						
							|  |  |  |      * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Result WaitSynchronization(bool* wait) override { | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |         // TODO(bunnei): ImplementMe
 | 
					
						
							|  |  |  |         ERROR_LOG(OSHLE, "(UNIMPLEMENTED)"); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | class Directory : public Object { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     std::string GetTypeName() const override { return "Directory"; } | 
					
						
							|  |  |  |     std::string GetName() const override { return path; } | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static Kernel::HandleType GetStaticHandleType() { return HandleType::Directory; } | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Kernel::HandleType GetHandleType() const override { return HandleType::Directory; } | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::string path; ///< Path of the directory
 | 
					
						
							|  |  |  |     std::unique_ptr<FileSys::Directory> backend; ///< File backend interface
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Synchronize kernel object | 
					
						
							|  |  |  |      * @param wait Boolean wait set if current thread should wait as a result of sync operation | 
					
						
							|  |  |  |      * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Result SyncRequest(bool* wait) override { | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  |         u32* cmd_buff = Service::GetCommandBuffer(); | 
					
						
							|  |  |  |         DirectoryCommand cmd = static_cast<DirectoryCommand>(cmd_buff[0]); | 
					
						
							|  |  |  |         switch (cmd) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Read from directory...
 | 
					
						
							|  |  |  |         case DirectoryCommand::Read: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u32 count = cmd_buff[1]; | 
					
						
							|  |  |  |             u32 address = cmd_buff[3]; | 
					
						
							|  |  |  |             FileSys::Entry* entries = reinterpret_cast<FileSys::Entry*>(Memory::GetPointer(address)); | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "Read %s %s: count=%d", GetTypeName().c_str(), GetName().c_str(), count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Number of entries actually read
 | 
					
						
							|  |  |  |             cmd_buff[2] = backend->Read(count, entries); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  |         case DirectoryCommand::Close: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             DEBUG_LOG(KERNEL, "Close %s %s", GetTypeName().c_str(), GetName().c_str()); | 
					
						
							|  |  |  |             Kernel::g_object_pool.Destroy<Directory>(GetHandle()); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  |         // Unknown command...
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             ERROR_LOG(KERNEL, "Unknown command=0x%08X!", cmd); | 
					
						
							|  |  |  |             cmd_buff[1] = -1; // TODO(Link Mauve): use the correct error code for that.
 | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         cmd_buff[1] = 0; // No error
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Wait for kernel object to synchronize | 
					
						
							|  |  |  |      * @param wait Boolean wait set if current thread should wait as a result of sync operation | 
					
						
							|  |  |  |      * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     Result WaitSynchronization(bool* wait) override { | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  |         // TODO(bunnei): ImplementMe
 | 
					
						
							|  |  |  |         ERROR_LOG(OSHLE, "(UNIMPLEMENTED)"); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::map<FileSys::Archive::IdCode, Handle> g_archive_map; ///< Map of file archives by IdCode
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Opens an archive | 
					
						
							|  |  |  |  * @param id_code IdCode of the archive to open | 
					
						
							|  |  |  |  * @return Handle to archive if it exists, otherwise a null handle (0) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Handle OpenArchive(FileSys::Archive::IdCode id_code) { | 
					
						
							|  |  |  |     auto itr = g_archive_map.find(id_code); | 
					
						
							|  |  |  |     if (itr == g_archive_map.end()) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return itr->second; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-14 11:52:52 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Closes an archive | 
					
						
							|  |  |  |  * @param id_code IdCode of the archive to open | 
					
						
							|  |  |  |  * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Result CloseArchive(FileSys::Archive::IdCode id_code) { | 
					
						
							|  |  |  |     if (1 != g_archive_map.erase(id_code)) { | 
					
						
							|  |  |  |         ERROR_LOG(KERNEL, "Cannot close archive %d", (int) id_code); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     INFO_LOG(KERNEL, "Closed archive %d", (int) id_code); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Mounts an archive | 
					
						
							|  |  |  |  * @param archive Pointer to the archive to mount | 
					
						
							|  |  |  |  * @return Result of operation, 0 on success, otherwise error code | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Result MountArchive(Archive* archive) { | 
					
						
							|  |  |  |     FileSys::Archive::IdCode id_code = archive->backend->GetIdCode(); | 
					
						
							|  |  |  |     if (0 != OpenArchive(id_code)) { | 
					
						
							|  |  |  |         ERROR_LOG(KERNEL, "Cannot mount two archives with the same ID code! (%d)", (int) id_code); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     g_archive_map[id_code] = archive->GetHandle(); | 
					
						
							| 
									
										
										
										
											2014-08-17 23:03:22 -04:00
										 |  |  |     INFO_LOG(KERNEL, "Mounted archive %s", archive->GetName().c_str()); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Creates an Archive | 
					
						
							|  |  |  |  * @param handle Handle to newly created archive object | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |  * @param backend File system backend interface to the archive | 
					
						
							|  |  |  |  * @param name Optional name of Archive | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |  * @return Newly created Archive object | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | Archive* CreateArchive(Handle& handle, FileSys::Archive* backend, const std::string& name) { | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |     Archive* archive = new Archive; | 
					
						
							|  |  |  |     handle = Kernel::g_object_pool.Create(archive); | 
					
						
							|  |  |  |     archive->name = name; | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |     archive->backend = backend; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MountArchive(archive); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |     return archive; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Creates an Archive | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |  * @param backend File system backend interface to the archive | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |  * @param name Optional name of Archive | 
					
						
							|  |  |  |  * @return Handle to newly created Archive object | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | Handle CreateArchive(FileSys::Archive* backend, const std::string& name) { | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |     Handle handle; | 
					
						
							| 
									
										
										
										
											2014-08-20 22:03:31 -07:00
										 |  |  |     CreateArchive(handle, backend, name); | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  |     return handle; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Open a File from an Archive | 
					
						
							|  |  |  |  * @param archive_handle Handle to an open Archive object | 
					
						
							|  |  |  |  * @param path Path to the File inside of the Archive | 
					
						
							|  |  |  |  * @param mode Mode under which to open the File | 
					
						
							|  |  |  |  * @return Opened File object | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Handle OpenFileFromArchive(Handle archive_handle, const std::string& path, const FileSys::Mode mode) { | 
					
						
							|  |  |  |     File* file = new File; | 
					
						
							|  |  |  |     Handle handle = Kernel::g_object_pool.Create(file); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Archive* archive = Kernel::g_object_pool.GetFast<Archive>(archive_handle); | 
					
						
							|  |  |  |     file->path = path; | 
					
						
							|  |  |  |     file->backend = archive->backend->OpenFile(path, mode); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-27 19:21:48 +00:00
										 |  |  |     if (!file->backend) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |     return handle; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:48:04 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Open a Directory from an Archive | 
					
						
							|  |  |  |  * @param archive_handle Handle to an open Archive object | 
					
						
							|  |  |  |  * @param path Path to the Directory inside of the Archive | 
					
						
							|  |  |  |  * @return Opened Directory object | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Handle OpenDirectoryFromArchive(Handle archive_handle, const std::string& path) { | 
					
						
							|  |  |  |     Directory* directory = new Directory; | 
					
						
							|  |  |  |     Handle handle = Kernel::g_object_pool.Create(directory); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Archive* archive = Kernel::g_object_pool.GetFast<Archive>(archive_handle); | 
					
						
							|  |  |  |     directory->path = path; | 
					
						
							|  |  |  |     directory->backend = archive->backend->OpenDirectory(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return handle; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 13:38:12 -04:00
										 |  |  | /// Initialize archives
 | 
					
						
							|  |  |  | void ArchiveInit() { | 
					
						
							|  |  |  |     g_archive_map.clear(); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO(Link Mauve): Add the other archive types (see here for the known types:
 | 
					
						
							|  |  |  |     // http://3dbrew.org/wiki/FS:OpenArchive#Archive_idcodes).  Currently the only half-finished
 | 
					
						
							|  |  |  |     // archive type is SDMC, so it is the only one getting exposed.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-20 15:11:38 -07:00
										 |  |  |     std::string sdmc_directory = FileUtil::GetUserPath(D_SDMC_IDX); | 
					
						
							| 
									
										
										
										
											2014-09-12 00:45:40 +02:00
										 |  |  |     auto archive = new FileSys::Archive_SDMC(sdmc_directory); | 
					
						
							|  |  |  |     if (archive->Initialize()) | 
					
						
							|  |  |  |         CreateArchive(archive, "SDMC"); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         ERROR_LOG(KERNEL, "Can't instantiate SDMC archive with path %s", sdmc_directory.c_str()); | 
					
						
							| 
									
										
										
										
											2014-07-04 13:38:12 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown archives
 | 
					
						
							|  |  |  | void ArchiveShutdown() { | 
					
						
							|  |  |  |     g_archive_map.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 18:15:35 -04:00
										 |  |  | } // namespace Kernel
 |