| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:42:59 +02:00
										 |  |  | #include <memory>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:42:59 +02:00
										 |  |  | #include "common/bit_field.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 05:03:17 -02:00
										 |  |  | #include "core/file_sys/file_backend.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-15 04:59:29 -02:00
										 |  |  | #include "core/file_sys/directory_backend.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  | #include "core/mem_map.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // FileSys namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  | // Path string type
 | 
					
						
							|  |  |  | enum LowPathType : u32 { | 
					
						
							|  |  |  |     Invalid = 0, | 
					
						
							|  |  |  |     Empty   = 1, | 
					
						
							|  |  |  |     Binary  = 2, | 
					
						
							|  |  |  |     Char    = 3, | 
					
						
							|  |  |  |     Wchar   = 4 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:42:59 +02:00
										 |  |  | union Mode { | 
					
						
							|  |  |  |     u32 hex; | 
					
						
							|  |  |  |     BitField<0, 1, u32> read_flag; | 
					
						
							|  |  |  |     BitField<1, 1, u32> write_flag; | 
					
						
							|  |  |  |     BitField<2, 1, u32> create_flag; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  | class Path { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     Path() : type(Invalid) { | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     Path(const char* path) : type(Char), string(path) { | 
					
						
							| 
									
										
										
										
											2014-12-18 23:35:24 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-29 13:04:37 -05:00
										 |  |  |     Path(std::vector<u8> binary_data) : type(Binary), binary(std::move(binary_data)) { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     Path(LowPathType type, u32 size, u32 pointer) : type(type) { | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |         switch (type) { | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |         case Binary: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             u8* data = Memory::GetPointer(pointer); | 
					
						
							|  |  |  |             binary = std::vector<u8>(data, data + size); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case Char: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const char* data = reinterpret_cast<const char*>(Memory::GetPointer(pointer)); | 
					
						
							|  |  |  |             string = std::string(data, size - 1); // Data is always null-terminated.
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case Wchar: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const char16_t* data = reinterpret_cast<const char16_t*>(Memory::GetPointer(pointer)); | 
					
						
							|  |  |  |             u16str = std::u16string(data, size/2 - 1); // Data is always null-terminated.
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LowPathType GetType() const { | 
					
						
							|  |  |  |         return type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 19:26:33 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Gets the string representation of the path for debugging | 
					
						
							|  |  |  |      * @return String representation of the path for debugging | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     const std::string DebugStr() const { | 
					
						
							|  |  |  |         switch (GetType()) { | 
					
						
							|  |  |  |         case Invalid: | 
					
						
							| 
									
										
										
										
											2015-01-01 12:39:27 -05:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-11-13 19:26:33 -05:00
										 |  |  |             return "[Invalid]"; | 
					
						
							|  |  |  |         case Empty: | 
					
						
							|  |  |  |             return "[Empty]"; | 
					
						
							|  |  |  |         case Binary: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             std::stringstream res; | 
					
						
							|  |  |  |             res << "[Binary: "; | 
					
						
							|  |  |  |             for (unsigned byte : binary) | 
					
						
							|  |  |  |                 res << std::hex << std::setw(2) << std::setfill('0') << byte; | 
					
						
							|  |  |  |             res << ']'; | 
					
						
							|  |  |  |             return res.str(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Char: | 
					
						
							|  |  |  |             return "[Char: " + AsString() + ']'; | 
					
						
							|  |  |  |         case Wchar: | 
					
						
							|  |  |  |             return "[Wchar: " + AsString() + ']'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |     const std::string AsString() const { | 
					
						
							|  |  |  |         switch (GetType()) { | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |         case Char: | 
					
						
							|  |  |  |             return string; | 
					
						
							|  |  |  |         case Wchar: | 
					
						
							|  |  |  |             return Common::UTF16ToUTF8(u16str); | 
					
						
							|  |  |  |         case Empty: | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         case Invalid: | 
					
						
							|  |  |  |         case Binary: | 
					
						
							| 
									
										
										
										
											2015-01-01 12:39:27 -05:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |             // TODO(yuriks): Add assert
 | 
					
						
							|  |  |  |             LOG_ERROR(Service_FS, "LowPathType cannot be converted to string!"); | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const std::u16string AsU16Str() const { | 
					
						
							|  |  |  |         switch (GetType()) { | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |         case Char: | 
					
						
							|  |  |  |             return Common::UTF8ToUTF16(string); | 
					
						
							|  |  |  |         case Wchar: | 
					
						
							|  |  |  |             return u16str; | 
					
						
							|  |  |  |         case Empty: | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         case Invalid: | 
					
						
							|  |  |  |         case Binary: | 
					
						
							|  |  |  |             // TODO(yuriks): Add assert
 | 
					
						
							|  |  |  |             LOG_ERROR(Service_FS, "LowPathType cannot be converted to u16string!"); | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const std::vector<u8> AsBinary() const { | 
					
						
							|  |  |  |         switch (GetType()) { | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |         case Binary: | 
					
						
							|  |  |  |             return binary; | 
					
						
							|  |  |  |         case Char: | 
					
						
							|  |  |  |             return std::vector<u8>(string.begin(), string.end()); | 
					
						
							|  |  |  |         case Wchar: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // use two u8 for each character of u16str
 | 
					
						
							|  |  |  |             std::vector<u8> to_return(u16str.size() * 2); | 
					
						
							|  |  |  |             for (size_t i = 0; i < u16str.size(); ++i) { | 
					
						
							|  |  |  |                 u16 tmp_char = u16str.at(i); | 
					
						
							|  |  |  |                 to_return[i*2] = (tmp_char & 0xFF00) >> 8; | 
					
						
							|  |  |  |                 to_return[i*2 + 1] = (tmp_char & 0x00FF); | 
					
						
							| 
									
										
										
										
											2014-12-19 22:16:34 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |             return to_return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Empty: | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         case Invalid: | 
					
						
							| 
									
										
										
										
											2015-01-01 12:39:27 -05:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |             // TODO(yuriks): Add assert
 | 
					
						
							|  |  |  |             LOG_ERROR(Service_FS, "LowPathType cannot be converted to binary!"); | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2014-11-10 14:36:32 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     LowPathType type; | 
					
						
							|  |  |  |     std::vector<u8> binary; | 
					
						
							|  |  |  |     std::string string; | 
					
						
							|  |  |  |     std::u16string u16str; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 02:51:38 -02:00
										 |  |  | class ArchiveBackend : NonCopyable { | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     virtual ~ArchiveBackend() { | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2014-12-15 02:44:04 -02:00
										 |  |  |      * Get a descriptive name for the archive (e.g. "RomFS", "SaveData", etc.) | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-15 02:44:04 -02:00
										 |  |  |     virtual std::string GetName() const = 0; | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:42:59 +02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Open a file specified by its path, using the specified mode | 
					
						
							|  |  |  |      * @param path Path relative to the archive | 
					
						
							|  |  |  |      * @param mode Mode to open the file with | 
					
						
							|  |  |  |      * @return Opened file, or nullptr | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-15 05:03:17 -02:00
										 |  |  |     virtual std::unique_ptr<FileBackend> OpenFile(const Path& path, const Mode mode) const = 0; | 
					
						
							| 
									
										
										
										
											2014-09-12 00:42:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Delete a file specified by its path | 
					
						
							|  |  |  |      * @param path Path relative to the archive | 
					
						
							|  |  |  |      * @return Whether the file could be deleted | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     virtual bool DeleteFile(const Path& path) const = 0; | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Rename a File specified by its path | 
					
						
							|  |  |  |      * @param src_path Source path relative to the archive | 
					
						
							|  |  |  |      * @param dest_path Destination path relative to the archive | 
					
						
							|  |  |  |      * @return Whether rename succeeded | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     virtual bool RenameFile(const Path& src_path, const Path& dest_path) const = 0; | 
					
						
							| 
									
										
										
										
											2014-11-23 23:20:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Delete a directory specified by its path | 
					
						
							|  |  |  |      * @param path Path relative to the archive | 
					
						
							|  |  |  |      * @return Whether the directory could be deleted | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     virtual bool DeleteDirectory(const Path& path) const = 0; | 
					
						
							| 
									
										
										
										
											2014-11-11 10:37:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-20 12:43:50 -03:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Create a file specified by its path | 
					
						
							|  |  |  |      * @param path Path relative to the Archive | 
					
						
							|  |  |  |      * @param size The size of the new file, filled with zeroes | 
					
						
							|  |  |  |      * @return File creation result code | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     virtual ResultCode CreateFile(const Path& path, u32 size) const = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 22:52:56 -07:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Create a directory specified by its path | 
					
						
							|  |  |  |      * @param path Path relative to the archive | 
					
						
							|  |  |  |      * @return Whether the directory could be created | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-11-11 19:27:35 -05:00
										 |  |  |     virtual bool CreateDirectory(const Path& path) const = 0; | 
					
						
							| 
									
										
										
										
											2014-10-28 22:52:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Rename a Directory specified by its path | 
					
						
							|  |  |  |      * @param src_path Source path relative to the archive | 
					
						
							|  |  |  |      * @param dest_path Destination path relative to the archive | 
					
						
							|  |  |  |      * @return Whether rename succeeded | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |     virtual bool RenameDirectory(const Path& src_path, const Path& dest_path) const = 0; | 
					
						
							| 
									
										
										
										
											2014-11-24 01:12:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 00:47:05 +02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Open a directory specified by its path | 
					
						
							|  |  |  |      * @param path Path relative to the archive | 
					
						
							|  |  |  |      * @return Opened directory, or nullptr | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-12-15 04:59:29 -02:00
										 |  |  |     virtual std::unique_ptr<DirectoryBackend> OpenDirectory(const Path& path) const = 0; | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | class ArchiveFactory : NonCopyable { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     virtual ~ArchiveFactory() { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Get a descriptive name for the archive (e.g. "RomFS", "SaveData", etc.) | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     virtual std::string GetName() const = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Tries to open the archive of this type with the specified path | 
					
						
							|  |  |  |      * @param path Path to the archive | 
					
						
							|  |  |  |      * @return An ArchiveBackend corresponding operating specified archive path. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     virtual ResultVal<std::unique_ptr<ArchiveBackend>> Open(const Path& path) = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Deletes the archive contents and then re-creates the base folder | 
					
						
							|  |  |  |      * @param path Path to the archive | 
					
						
							|  |  |  |      * @return ResultCode of the operation, 0 on success | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     virtual ResultCode Format(const Path& path) = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | } // namespace FileSys
 |