| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 15:44:11 +01:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cstdio>
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "common/file_util.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-03 17:15:53 -02:00
										 |  |  | #include "common/make_unique.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "core/file_sys/disk_archive.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // FileSys namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::unique_ptr<FileBackend> DiskArchive::OpenFile(const Path& path, const Mode mode) const { | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "called path=%s mode=%01X", path.DebugStr().c_str(), mode.hex); | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto file = Common::make_unique<DiskFile>(*this, path, mode); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     if (!file->Open()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2015-01-03 17:15:53 -02:00
										 |  |  |     return std::move(file); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  | bool DiskArchive::DeleteFile(const Path& path) const { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     return FileUtil::Delete(mount_point + path.AsString()); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  | bool DiskArchive::RenameFile(const Path& src_path, const Path& dest_path) const { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     return FileUtil::Rename(mount_point + src_path.AsString(), mount_point + dest_path.AsString()); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  | bool DiskArchive::DeleteDirectory(const Path& path) const { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     return FileUtil::DeleteDir(mount_point + path.AsString()); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-20 12:43:50 -03:00
										 |  |  | ResultCode DiskArchive::CreateFile(const FileSys::Path& path, u32 size) const { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     std::string full_path = mount_point + path.AsString(); | 
					
						
							| 
									
										
										
										
											2014-12-20 12:43:50 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (FileUtil::Exists(full_path)) | 
					
						
							|  |  |  |         return ResultCode(ErrorDescription::AlreadyExists, ErrorModule::FS, ErrorSummary::NothingHappened, ErrorLevel::Info); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (size == 0) { | 
					
						
							|  |  |  |         FileUtil::CreateEmptyFile(full_path); | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FileUtil::IOFile file(full_path, "wb"); | 
					
						
							|  |  |  |     // Creates a sparse file (or a normal file on filesystems without the concept of sparse files)
 | 
					
						
							|  |  |  |     // We do this by seeking to the right size, then writing a single null byte.
 | 
					
						
							|  |  |  |     if (file.Seek(size - 1, SEEK_SET) && file.WriteBytes("", 1) == 1) | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ResultCode(ErrorDescription::TooLarge, ErrorModule::FS, ErrorSummary::OutOfResource, ErrorLevel::Info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | bool DiskArchive::CreateDirectory(const Path& path) const { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     return FileUtil::CreateDir(mount_point + path.AsString()); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  | bool DiskArchive::RenameDirectory(const Path& src_path, const Path& dest_path) const { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     return FileUtil::Rename(mount_point + src_path.AsString(), mount_point + dest_path.AsString()); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::unique_ptr<DirectoryBackend> DiskArchive::OpenDirectory(const Path& path) const { | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "called path=%s", path.DebugStr().c_str()); | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     auto directory = Common::make_unique<DiskDirectory>(*this, path); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     if (!directory->Open()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2015-01-03 17:15:53 -02:00
										 |  |  |     return std::move(directory); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-18 15:52:37 -07:00
										 |  |  | u64 DiskArchive::GetFreeBytes() const { | 
					
						
							|  |  |  |     // TODO: Stubbed to return 1GiB
 | 
					
						
							|  |  |  |     return 1024 * 1024 * 1024; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  | DiskFile::DiskFile(const DiskArchive& archive, const Path& path, const Mode mode) { | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     // TODO(Link Mauve): normalize path into an absolute path without "..", it can currently bypass
 | 
					
						
							|  |  |  |     // the root directory we set while opening the archive.
 | 
					
						
							|  |  |  |     // For example, opening /../../etc/passwd can give the emulated program your users list.
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     this->path = archive.mount_point + path.AsString(); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     this->mode.hex = mode.hex; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DiskFile::Open() { | 
					
						
							|  |  |  |     if (!mode.create_flag && !FileUtil::Exists(path)) { | 
					
						
							| 
									
										
										
										
											2014-12-18 14:06:37 +00:00
										 |  |  |         LOG_ERROR(Service_FS, "Non-existing file %s can't be open without mode create.", path.c_str()); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string mode_string; | 
					
						
							|  |  |  |     if (mode.create_flag) | 
					
						
							|  |  |  |         mode_string = "w+"; | 
					
						
							|  |  |  |     else if (mode.write_flag) | 
					
						
							|  |  |  |         mode_string = "r+"; // Files opened with Write access can be read from
 | 
					
						
							|  |  |  |     else if (mode.read_flag) | 
					
						
							|  |  |  |         mode_string = "r"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Open the file in binary mode, to avoid problems with CR/LF on Windows systems
 | 
					
						
							|  |  |  |     mode_string += "b"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-03 17:15:53 -02:00
										 |  |  |     file = Common::make_unique<FileUtil::IOFile>(path, mode_string.c_str()); | 
					
						
							| 
									
										
										
										
											2015-08-28 22:01:03 -05:00
										 |  |  |     return file->IsOpen(); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:43:34 -03:00
										 |  |  | size_t DiskFile::Read(const u64 offset, const size_t length, u8* buffer) const { | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     file->Seek(offset, SEEK_SET); | 
					
						
							|  |  |  |     return file->ReadBytes(buffer, length); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:43:34 -03:00
										 |  |  | size_t DiskFile::Write(const u64 offset, const size_t length, const bool flush, const u8* buffer) const { | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     file->Seek(offset, SEEK_SET); | 
					
						
							|  |  |  |     size_t written = file->WriteBytes(buffer, length); | 
					
						
							|  |  |  |     if (flush) | 
					
						
							|  |  |  |         file->Flush(); | 
					
						
							|  |  |  |     return written; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:43:34 -03:00
										 |  |  | u64 DiskFile::GetSize() const { | 
					
						
							|  |  |  |     return file->GetSize(); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DiskFile::SetSize(const u64 size) const { | 
					
						
							|  |  |  |     file->Resize(size); | 
					
						
							|  |  |  |     file->Flush(); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DiskFile::Close() const { | 
					
						
							|  |  |  |     return file->Close(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-25 17:03:59 -05:00
										 |  |  | DiskDirectory::DiskDirectory(const DiskArchive& archive, const Path& path) : directory() { | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     // TODO(Link Mauve): normalize path into an absolute path without "..", it can currently bypass
 | 
					
						
							|  |  |  |     // the root directory we set while opening the archive.
 | 
					
						
							|  |  |  |     // For example, opening /../../usr/bin can give the emulated program your installed programs.
 | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |     this->path = archive.mount_point + path.AsString(); | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DiskDirectory::Open() { | 
					
						
							|  |  |  |     if (!FileUtil::IsDirectory(path)) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2015-12-25 17:03:59 -05:00
										 |  |  |     unsigned size = FileUtil::ScanDirectoryTree(path, directory); | 
					
						
							|  |  |  |     directory.size = size; | 
					
						
							|  |  |  |     directory.isDirectory = true; | 
					
						
							| 
									
										
										
										
											2014-12-16 00:33:41 -05:00
										 |  |  |     children_iterator = directory.children.begin(); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u32 DiskDirectory::Read(const u32 count, Entry* entries) { | 
					
						
							|  |  |  |     u32 entries_read = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (entries_read < count && children_iterator != directory.children.cend()) { | 
					
						
							|  |  |  |         const FileUtil::FSTEntry& file = *children_iterator; | 
					
						
							|  |  |  |         const std::string& filename = file.virtualName; | 
					
						
							|  |  |  |         Entry& entry = entries[entries_read]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         LOG_TRACE(Service_FS, "File %s: size=%llu dir=%d", filename.c_str(), file.size, file.isDirectory); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // TODO(Link Mauve): use a proper conversion to UTF-16.
 | 
					
						
							|  |  |  |         for (size_t j = 0; j < FILENAME_LENGTH; ++j) { | 
					
						
							|  |  |  |             entry.filename[j] = filename[j]; | 
					
						
							|  |  |  |             if (!filename[j]) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         FileUtil::SplitFilename83(filename, entry.short_name, entry.extension); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         entry.is_directory = file.isDirectory; | 
					
						
							|  |  |  |         entry.is_hidden = (filename[0] == '.'); | 
					
						
							|  |  |  |         entry.is_read_only = 0; | 
					
						
							|  |  |  |         entry.file_size = file.size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // We emulate a SD card where the archive bit has never been cleared, as it would be on
 | 
					
						
							|  |  |  |         // most user SD cards.
 | 
					
						
							|  |  |  |         // Some homebrews (blargSNES for instance) are known to mistakenly use the archive bit as a
 | 
					
						
							|  |  |  |         // file bit.
 | 
					
						
							|  |  |  |         entry.is_archive = !file.isDirectory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ++entries_read; | 
					
						
							|  |  |  |         ++children_iterator; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return entries_read; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace FileSys
 |