forked from eden-emu/eden
		
	Merge pull request #5187 from Morph1984/revert-stdfs
fs: Revert all std::filesystem changes
This commit is contained in:
		
						commit
						bf0fea0cfc
					
				
					 3 changed files with 396 additions and 142 deletions
				
			
		|  | @ -3,7 +3,6 @@ | ||||||
| // Refer to the license.txt file included.
 | // Refer to the license.txt file included.
 | ||||||
| 
 | 
 | ||||||
| #include <array> | #include <array> | ||||||
| #include <filesystem> |  | ||||||
| #include <limits> | #include <limits> | ||||||
| #include <memory> | #include <memory> | ||||||
| #include <sstream> | #include <sstream> | ||||||
|  | @ -68,122 +67,290 @@ | ||||||
| #include <algorithm> | #include <algorithm> | ||||||
| #include <sys/stat.h> | #include <sys/stat.h> | ||||||
| 
 | 
 | ||||||
|  | #ifndef S_ISDIR | ||||||
|  | #define S_ISDIR(m) (((m)&S_IFMT) == S_IFDIR) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | // This namespace has various generic functions related to files and paths.
 | ||||||
|  | // The code still needs a ton of cleanup.
 | ||||||
|  | // REMEMBER: strdup considered harmful!
 | ||||||
| namespace Common::FS { | namespace Common::FS { | ||||||
| namespace fs = std::filesystem; |  | ||||||
| 
 | 
 | ||||||
| bool Exists(const fs::path& path) { | // Remove any ending forward slashes from directory paths
 | ||||||
|     std::error_code ec; | // Modifies argument.
 | ||||||
|     return fs::exists(path, ec); | static void StripTailDirSlashes(std::string& fname) { | ||||||
|  |     if (fname.length() <= 1) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::size_t i = fname.length(); | ||||||
|  |     while (i > 0 && fname[i - 1] == DIR_SEP_CHR) { | ||||||
|  |         --i; | ||||||
|  |     } | ||||||
|  |     fname.resize(i); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool IsDirectory(const fs::path& path) { | bool Exists(const std::string& filename) { | ||||||
|     std::error_code ec; |     struct stat file_info; | ||||||
|     return fs::is_directory(path, ec); | 
 | ||||||
|  |     std::string copy(filename); | ||||||
|  |     StripTailDirSlashes(copy); | ||||||
|  | 
 | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     // Windows needs a slash to identify a driver root
 | ||||||
|  |     if (copy.size() != 0 && copy.back() == ':') | ||||||
|  |         copy += DIR_SEP_CHR; | ||||||
|  | 
 | ||||||
|  |     int result = _wstat64(Common::UTF8ToUTF16W(copy).c_str(), &file_info); | ||||||
|  | #else | ||||||
|  |     int result = stat(copy.c_str(), &file_info); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |     return (result == 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool Delete(const fs::path& path) { | bool IsDirectory(const std::string& filename) { | ||||||
|     LOG_TRACE(Common_Filesystem, "file {}", path.string()); |     struct stat file_info; | ||||||
|  | 
 | ||||||
|  |     std::string copy(filename); | ||||||
|  |     StripTailDirSlashes(copy); | ||||||
|  | 
 | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     // Windows needs a slash to identify a driver root
 | ||||||
|  |     if (copy.size() != 0 && copy.back() == ':') | ||||||
|  |         copy += DIR_SEP_CHR; | ||||||
|  | 
 | ||||||
|  |     int result = _wstat64(Common::UTF8ToUTF16W(copy).c_str(), &file_info); | ||||||
|  | #else | ||||||
|  |     int result = stat(copy.c_str(), &file_info); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |     if (result < 0) { | ||||||
|  |         LOG_DEBUG(Common_Filesystem, "stat failed on {}: {}", filename, GetLastErrorMsg()); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return S_ISDIR(file_info.st_mode); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Delete(const std::string& filename) { | ||||||
|  |     LOG_TRACE(Common_Filesystem, "file {}", filename); | ||||||
| 
 | 
 | ||||||
|     // Return true because we care about the file no
 |     // Return true because we care about the file no
 | ||||||
|     // being there, not the actual delete.
 |     // being there, not the actual delete.
 | ||||||
|     if (!Exists(path)) { |     if (!Exists(filename)) { | ||||||
|         LOG_DEBUG(Common_Filesystem, "{} does not exist", path.string()); |         LOG_DEBUG(Common_Filesystem, "{} does not exist", filename); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     std::error_code ec; |     // We can't delete a directory
 | ||||||
|     return fs::remove(path, ec); |     if (IsDirectory(filename)) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "Failed: {} is a directory", filename); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     if (!DeleteFileW(Common::UTF8ToUTF16W(filename).c_str())) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "DeleteFile failed on {}: {}", filename, GetLastErrorMsg()); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | #else | ||||||
|  |     if (unlink(filename.c_str()) == -1) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "unlink failed on {}: {}", filename, GetLastErrorMsg()); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool CreateDir(const fs::path& path) { | bool CreateDir(const std::string& path) { | ||||||
|     LOG_TRACE(Common_Filesystem, "directory {}", path.string()); |     LOG_TRACE(Common_Filesystem, "directory {}", path); | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     if (::CreateDirectoryW(Common::UTF8ToUTF16W(path).c_str(), nullptr)) | ||||||
|  |         return true; | ||||||
|  |     DWORD error = GetLastError(); | ||||||
|  |     if (error == ERROR_ALREADY_EXISTS) { | ||||||
|  |         LOG_DEBUG(Common_Filesystem, "CreateDirectory failed on {}: already exists", path); | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |     LOG_ERROR(Common_Filesystem, "CreateDirectory failed on {}: {}", path, error); | ||||||
|  |     return false; | ||||||
|  | #else | ||||||
|  |     if (mkdir(path.c_str(), 0755) == 0) | ||||||
|  |         return true; | ||||||
| 
 | 
 | ||||||
|     if (Exists(path)) { |     int err = errno; | ||||||
|         LOG_DEBUG(Common_Filesystem, "path exists {}", path.string()); | 
 | ||||||
|  |     if (err == EEXIST) { | ||||||
|  |         LOG_DEBUG(Common_Filesystem, "mkdir failed on {}: already exists", path); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     std::error_code ec; |     LOG_ERROR(Common_Filesystem, "mkdir failed on {}: {}", path, strerror(err)); | ||||||
|     const bool success = fs::create_directory(path, ec); |     return false; | ||||||
| 
 | #endif | ||||||
|     if (!success) { |  | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to create directory: {}", ec.message()); |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return true; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool CreateDirs(const fs::path& path) { | bool CreateFullPath(const std::string& fullPath) { | ||||||
|     LOG_TRACE(Common_Filesystem, "path {}", path.string()); |     int panicCounter = 100; | ||||||
|  |     LOG_TRACE(Common_Filesystem, "path {}", fullPath); | ||||||
| 
 | 
 | ||||||
|     if (Exists(path)) { |     if (Exists(fullPath)) { | ||||||
|         LOG_DEBUG(Common_Filesystem, "path exists {}", path.string()); |         LOG_DEBUG(Common_Filesystem, "path exists {}", fullPath); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     std::error_code ec; |     std::size_t position = 0; | ||||||
|     const bool success = fs::create_directories(path, ec); |     while (true) { | ||||||
|  |         // Find next sub path
 | ||||||
|  |         position = fullPath.find(DIR_SEP_CHR, position); | ||||||
| 
 | 
 | ||||||
|     if (!success) { |         // we're done, yay!
 | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to create directories: {}", ec.message()); |         if (position == fullPath.npos) | ||||||
|  |             return true; | ||||||
|  | 
 | ||||||
|  |         // Include the '/' so the first call is CreateDir("/") rather than CreateDir("")
 | ||||||
|  |         std::string const subPath(fullPath.substr(0, position + 1)); | ||||||
|  |         if (!IsDirectory(subPath) && !CreateDir(subPath)) { | ||||||
|  |             LOG_ERROR(Common, "CreateFullPath: directory creation failed"); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // A safety check
 | ||||||
|  |         panicCounter--; | ||||||
|  |         if (panicCounter <= 0) { | ||||||
|  |             LOG_ERROR(Common, "CreateFullPath: directory structure is too deep"); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         position++; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool DeleteDir(const std::string& filename) { | ||||||
|  |     LOG_TRACE(Common_Filesystem, "directory {}", filename); | ||||||
|  | 
 | ||||||
|  |     // check if a directory
 | ||||||
|  |     if (!IsDirectory(filename)) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "Not a directory {}", filename); | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return true; | #ifdef _WIN32 | ||||||
|  |     if (::RemoveDirectoryW(Common::UTF8ToUTF16W(filename).c_str())) | ||||||
|  |         return true; | ||||||
|  | #else | ||||||
|  |     if (rmdir(filename.c_str()) == 0) | ||||||
|  |         return true; | ||||||
|  | #endif | ||||||
|  |     LOG_ERROR(Common_Filesystem, "failed {}: {}", filename, GetLastErrorMsg()); | ||||||
|  | 
 | ||||||
|  |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool CreateFullPath(const fs::path& path) { | bool Rename(const std::string& srcFilename, const std::string& destFilename) { | ||||||
|     LOG_TRACE(Common_Filesystem, "path {}", path); |     LOG_TRACE(Common_Filesystem, "{} --> {}", srcFilename, destFilename); | ||||||
| 
 | #ifdef _WIN32 | ||||||
|     if (path.has_extension()) { |     if (_wrename(Common::UTF8ToUTF16W(srcFilename).c_str(), | ||||||
|         return CreateDirs(path.parent_path()); |                  Common::UTF8ToUTF16W(destFilename).c_str()) == 0) | ||||||
|     } else { |         return true; | ||||||
|         return CreateDirs(path); | #else | ||||||
|     } |     if (rename(srcFilename.c_str(), destFilename.c_str()) == 0) | ||||||
|  |         return true; | ||||||
|  | #endif | ||||||
|  |     LOG_ERROR(Common_Filesystem, "failed {} --> {}: {}", srcFilename, destFilename, | ||||||
|  |               GetLastErrorMsg()); | ||||||
|  |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool Rename(const fs::path& src, const fs::path& dst) { | bool Copy(const std::string& srcFilename, const std::string& destFilename) { | ||||||
|     LOG_TRACE(Common_Filesystem, "{} --> {}", src.string(), dst.string()); |     LOG_TRACE(Common_Filesystem, "{} --> {}", srcFilename, destFilename); | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     if (CopyFileW(Common::UTF8ToUTF16W(srcFilename).c_str(), | ||||||
|  |                   Common::UTF8ToUTF16W(destFilename).c_str(), FALSE)) | ||||||
|  |         return true; | ||||||
| 
 | 
 | ||||||
|     std::error_code ec; |     LOG_ERROR(Common_Filesystem, "failed {} --> {}: {}", srcFilename, destFilename, | ||||||
|     fs::rename(src, dst, ec); |               GetLastErrorMsg()); | ||||||
|  |     return false; | ||||||
|  | #else | ||||||
|  |     using CFilePointer = std::unique_ptr<FILE, decltype(&std::fclose)>; | ||||||
| 
 | 
 | ||||||
|     if (ec) { |     // Open input file
 | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to rename file from {} to {}: {}", src.string(), |     CFilePointer input{fopen(srcFilename.c_str(), "rb"), std::fclose}; | ||||||
|                   dst.string(), ec.message()); |     if (!input) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "opening input failed {} --> {}: {}", srcFilename, | ||||||
|  |                   destFilename, GetLastErrorMsg()); | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return true; |     // open output file
 | ||||||
| } |     CFilePointer output{fopen(destFilename.c_str(), "wb"), std::fclose}; | ||||||
| 
 |     if (!output) { | ||||||
| bool Copy(const fs::path& src, const fs::path& dst) { |         LOG_ERROR(Common_Filesystem, "opening output failed {} --> {}: {}", srcFilename, | ||||||
|     LOG_TRACE(Common_Filesystem, "{} --> {}", src.string(), dst.string()); |                   destFilename, GetLastErrorMsg()); | ||||||
| 
 |  | ||||||
|     std::error_code ec; |  | ||||||
|     const bool success = fs::copy_file(src, dst, fs::copy_options::overwrite_existing, ec); |  | ||||||
| 
 |  | ||||||
|     if (!success) { |  | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to copy file {} to {}: {}", src.string(), dst.string(), |  | ||||||
|                   ec.message()); |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     // copy loop
 | ||||||
|  |     std::array<char, 1024> buffer; | ||||||
|  |     while (!feof(input.get())) { | ||||||
|  |         // read input
 | ||||||
|  |         std::size_t rnum = fread(buffer.data(), sizeof(char), buffer.size(), input.get()); | ||||||
|  |         if (rnum != buffer.size()) { | ||||||
|  |             if (ferror(input.get()) != 0) { | ||||||
|  |                 LOG_ERROR(Common_Filesystem, "failed reading from source, {} --> {}: {}", | ||||||
|  |                           srcFilename, destFilename, GetLastErrorMsg()); | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // write output
 | ||||||
|  |         std::size_t wnum = fwrite(buffer.data(), sizeof(char), rnum, output.get()); | ||||||
|  |         if (wnum != rnum) { | ||||||
|  |             LOG_ERROR(Common_Filesystem, "failed writing to output, {} --> {}: {}", srcFilename, | ||||||
|  |                       destFilename, GetLastErrorMsg()); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     return true; |     return true; | ||||||
|  | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u64 GetSize(const fs::path& path) { | u64 GetSize(const std::string& filename) { | ||||||
|     std::error_code ec; |     if (!Exists(filename)) { | ||||||
|     const auto size = fs::file_size(path, ec); |         LOG_ERROR(Common_Filesystem, "failed {}: No such file", filename); | ||||||
| 
 |  | ||||||
|     if (ec) { |  | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to retrieve file size ({}): {}", path.string(), |  | ||||||
|                   ec.message()); |  | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return size; |     if (IsDirectory(filename)) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "failed {}: is a directory", filename); | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     struct stat buf; | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     if (_wstat64(Common::UTF8ToUTF16W(filename).c_str(), &buf) == 0) | ||||||
|  | #else | ||||||
|  |     if (stat(filename.c_str(), &buf) == 0) | ||||||
|  | #endif | ||||||
|  |     { | ||||||
|  |         LOG_TRACE(Common_Filesystem, "{}: {}", filename, buf.st_size); | ||||||
|  |         return buf.st_size; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     LOG_ERROR(Common_Filesystem, "Stat failed {}: {}", filename, GetLastErrorMsg()); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u64 GetSize(const int fd) { | ||||||
|  |     struct stat buf; | ||||||
|  |     if (fstat(fd, &buf) != 0) { | ||||||
|  |         LOG_ERROR(Common_Filesystem, "GetSize: stat failed {}: {}", fd, GetLastErrorMsg()); | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |     return buf.st_size; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u64 GetSize(FILE* f) { | u64 GetSize(FILE* f) { | ||||||
|  | @ -233,7 +400,7 @@ bool ForeachDirectoryEntry(u64* num_entries_out, const std::string& directory, | ||||||
|     } |     } | ||||||
|     // windows loop
 |     // windows loop
 | ||||||
|     do { |     do { | ||||||
|         const std::string virtual_name = std::filesystem::path(ffd.cFileName).string(); |         const std::string virtual_name(Common::UTF16ToUTF8(ffd.cFileName)); | ||||||
| #else | #else | ||||||
|     DIR* dirp = opendir(directory.c_str()); |     DIR* dirp = opendir(directory.c_str()); | ||||||
|     if (!dirp) |     if (!dirp) | ||||||
|  | @ -271,58 +438,132 @@ bool ForeachDirectoryEntry(u64* num_entries_out, const std::string& directory, | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool DeleteDirRecursively(const fs::path& path) { | u64 ScanDirectoryTree(const std::string& directory, FSTEntry& parent_entry, | ||||||
|     std::error_code ec; |                       unsigned int recursion) { | ||||||
|     fs::remove_all(path, ec); |     const auto callback = [recursion, &parent_entry](u64* num_entries_out, | ||||||
|  |                                                      const std::string& directory, | ||||||
|  |                                                      const std::string& virtual_name) -> bool { | ||||||
|  |         FSTEntry entry; | ||||||
|  |         entry.virtualName = virtual_name; | ||||||
|  |         entry.physicalName = directory + DIR_SEP + virtual_name; | ||||||
| 
 | 
 | ||||||
|     if (ec) { |         if (IsDirectory(entry.physicalName)) { | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to completely delete directory {}: {}", path.string(), |             entry.isDirectory = true; | ||||||
|                   ec.message()); |             // is a directory, lets go inside if we didn't recurse to often
 | ||||||
|  |             if (recursion > 0) { | ||||||
|  |                 entry.size = ScanDirectoryTree(entry.physicalName, entry, recursion - 1); | ||||||
|  |                 *num_entries_out += entry.size; | ||||||
|  |             } else { | ||||||
|  |                 entry.size = 0; | ||||||
|  |             } | ||||||
|  |         } else { // is a file
 | ||||||
|  |             entry.isDirectory = false; | ||||||
|  |             entry.size = GetSize(entry.physicalName); | ||||||
|  |         } | ||||||
|  |         (*num_entries_out)++; | ||||||
|  | 
 | ||||||
|  |         // Push into the tree
 | ||||||
|  |         parent_entry.children.push_back(std::move(entry)); | ||||||
|  |         return true; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     u64 num_entries; | ||||||
|  |     return ForeachDirectoryEntry(&num_entries, directory, callback) ? num_entries : 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool DeleteDirRecursively(const std::string& directory, unsigned int recursion) { | ||||||
|  |     const auto callback = [recursion](u64*, const std::string& directory, | ||||||
|  |                                       const std::string& virtual_name) { | ||||||
|  |         const std::string new_path = directory + DIR_SEP_CHR + virtual_name; | ||||||
|  | 
 | ||||||
|  |         if (IsDirectory(new_path)) { | ||||||
|  |             if (recursion == 0) { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |             return DeleteDirRecursively(new_path, recursion - 1); | ||||||
|  |         } | ||||||
|  |         return Delete(new_path); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     if (!ForeachDirectoryEntry(nullptr, directory, callback)) | ||||||
|         return false; |         return false; | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|  |     // Delete the outermost directory
 | ||||||
|  |     DeleteDir(directory); | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void CopyDir(const fs::path& src, const fs::path& dst) { | void CopyDir([[maybe_unused]] const std::string& source_path, | ||||||
|     constexpr auto copy_flags = fs::copy_options::skip_existing | fs::copy_options::recursive; |              [[maybe_unused]] const std::string& dest_path) { | ||||||
|  | #ifndef _WIN32 | ||||||
|  |     if (source_path == dest_path) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     if (!Exists(source_path)) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     if (!Exists(dest_path)) { | ||||||
|  |         CreateFullPath(dest_path); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     std::error_code ec; |     DIR* dirp = opendir(source_path.c_str()); | ||||||
|     fs::copy(src, dst, copy_flags, ec); |     if (!dirp) { | ||||||
| 
 |  | ||||||
|     if (ec) { |  | ||||||
|         LOG_ERROR(Common_Filesystem, "Error copying directory {} to {}: {}", src.string(), |  | ||||||
|                   dst.string(), ec.message()); |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     LOG_TRACE(Common_Filesystem, "Successfully copied directory."); |     while (struct dirent* result = readdir(dirp)) { | ||||||
|  |         const std::string virtualName(result->d_name); | ||||||
|  |         // check for "." and ".."
 | ||||||
|  |         if (((virtualName[0] == '.') && (virtualName[1] == '\0')) || | ||||||
|  |             ((virtualName[0] == '.') && (virtualName[1] == '.') && (virtualName[2] == '\0'))) { | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         std::string source, dest; | ||||||
|  |         source = source_path + virtualName; | ||||||
|  |         dest = dest_path + virtualName; | ||||||
|  |         if (IsDirectory(source)) { | ||||||
|  |             source += '/'; | ||||||
|  |             dest += '/'; | ||||||
|  |             if (!Exists(dest)) { | ||||||
|  |                 CreateFullPath(dest); | ||||||
|  |             } | ||||||
|  |             CopyDir(source, dest); | ||||||
|  |         } else if (!Exists(dest)) { | ||||||
|  |             Copy(source, dest); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     closedir(dirp); | ||||||
|  | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::optional<fs::path> GetCurrentDir() { | std::optional<std::string> GetCurrentDir() { | ||||||
|     std::error_code ec; | // Get the current working directory (getcwd uses malloc)
 | ||||||
|     auto path = fs::current_path(ec); | #ifdef _WIN32 | ||||||
| 
 |     wchar_t* dir = _wgetcwd(nullptr, 0); | ||||||
|     if (ec) { |     if (!dir) { | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to retrieve current working directory: {}", | #else | ||||||
|                   ec.message()); |     char* dir = getcwd(nullptr, 0); | ||||||
|  |     if (!dir) { | ||||||
|  | #endif | ||||||
|  |         LOG_ERROR(Common_Filesystem, "GetCurrentDirectory failed: {}", GetLastErrorMsg()); | ||||||
|         return std::nullopt; |         return std::nullopt; | ||||||
|     } |     } | ||||||
| 
 | #ifdef _WIN32 | ||||||
|     return {std::move(path)}; |     std::string strDir = Common::UTF16ToUTF8(dir); | ||||||
|  | #else | ||||||
|  |     std::string strDir = dir; | ||||||
|  | #endif | ||||||
|  |     free(dir); | ||||||
|  |     return strDir; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool SetCurrentDir(const fs::path& path) { | bool SetCurrentDir(const std::string& directory) { | ||||||
|     std::error_code ec; | #ifdef _WIN32 | ||||||
|     fs::current_path(path, ec); |     return _wchdir(Common::UTF8ToUTF16W(directory).c_str()) == 0; | ||||||
| 
 | #else | ||||||
|     if (ec) { |     return chdir(directory.c_str()) == 0; | ||||||
|         LOG_ERROR(Common_Filesystem, "Unable to set {} as working directory: {}", path.string(), | #endif | ||||||
|                   ec.message()); |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return true; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|  |  | ||||||
|  | @ -6,7 +6,6 @@ | ||||||
| 
 | 
 | ||||||
| #include <array> | #include <array> | ||||||
| #include <cstdio> | #include <cstdio> | ||||||
| #include <filesystem> |  | ||||||
| #include <fstream> | #include <fstream> | ||||||
| #include <functional> | #include <functional> | ||||||
| #include <limits> | #include <limits> | ||||||
|  | @ -39,40 +38,48 @@ enum class UserPath { | ||||||
|     UserDir, |     UserDir, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Returns true if the path exists
 | // FileSystem tree node/
 | ||||||
| [[nodiscard]] bool Exists(const std::filesystem::path& path); | struct FSTEntry { | ||||||
|  |     bool isDirectory; | ||||||
|  |     u64 size;                 // file length or number of entries from children
 | ||||||
|  |     std::string physicalName; // name on disk
 | ||||||
|  |     std::string virtualName;  // name in FST names table
 | ||||||
|  |     std::vector<FSTEntry> children; | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| // Returns true if path is a directory
 | // Returns true if file filename exists
 | ||||||
| [[nodiscard]] bool IsDirectory(const std::filesystem::path& path); | [[nodiscard]] bool Exists(const std::string& filename); | ||||||
|  | 
 | ||||||
|  | // Returns true if filename is a directory
 | ||||||
|  | [[nodiscard]] bool IsDirectory(const std::string& filename); | ||||||
| 
 | 
 | ||||||
| // Returns the size of filename (64bit)
 | // Returns the size of filename (64bit)
 | ||||||
| [[nodiscard]] u64 GetSize(const std::filesystem::path& path); | [[nodiscard]] u64 GetSize(const std::string& filename); | ||||||
|  | 
 | ||||||
|  | // Overloaded GetSize, accepts file descriptor
 | ||||||
|  | [[nodiscard]] u64 GetSize(int fd); | ||||||
| 
 | 
 | ||||||
| // Overloaded GetSize, accepts FILE*
 | // Overloaded GetSize, accepts FILE*
 | ||||||
| [[nodiscard]] u64 GetSize(FILE* f); | [[nodiscard]] u64 GetSize(FILE* f); | ||||||
| 
 | 
 | ||||||
| // Returns true if successful, or path already exists.
 | // Returns true if successful, or path already exists.
 | ||||||
| bool CreateDir(const std::filesystem::path& path); | bool CreateDir(const std::string& filename); | ||||||
| 
 | 
 | ||||||
| // Create all directories in path
 | // Creates the full path of fullPath returns true on success
 | ||||||
| // Returns true if successful, or path already exists.
 | bool CreateFullPath(const std::string& fullPath); | ||||||
| [[nodiscard("Directory creation can fail and must be tested")]] bool CreateDirs( |  | ||||||
|     const std::filesystem::path& path); |  | ||||||
| 
 | 
 | ||||||
| // Creates directories in path. Returns true on success.
 | // Deletes a given filename, return true on success
 | ||||||
| [[deprecated("This function is deprecated, use CreateDirs")]] bool CreateFullPath( | // Doesn't supports deleting a directory
 | ||||||
|     const std::filesystem::path& path); | bool Delete(const std::string& filename); | ||||||
| 
 | 
 | ||||||
| // Deletes a given file at the path.
 | // Deletes a directory filename, returns true on success
 | ||||||
| // This will also delete empty directories.
 | bool DeleteDir(const std::string& filename); | ||||||
| // Return true on success
 |  | ||||||
| bool Delete(const std::filesystem::path& path); |  | ||||||
| 
 | 
 | ||||||
| // Renames file src to dst, returns true on success
 | // renames file srcFilename to destFilename, returns true on success
 | ||||||
| bool Rename(const std::filesystem::path& src, const std::filesystem::path& dst); | bool Rename(const std::string& srcFilename, const std::string& destFilename); | ||||||
| 
 | 
 | ||||||
| // copies file src to dst, returns true on success
 | // copies file srcFilename to destFilename, returns true on success
 | ||||||
| bool Copy(const std::filesystem::path& src, const std::filesystem::path& dst); | bool Copy(const std::string& srcFilename, const std::string& destFilename); | ||||||
| 
 | 
 | ||||||
| // creates an empty file filename, returns true on success
 | // creates an empty file filename, returns true on success
 | ||||||
| bool CreateEmptyFile(const std::string& filename); | bool CreateEmptyFile(const std::string& filename); | ||||||
|  | @ -99,17 +106,27 @@ using DirectoryEntryCallable = std::function<bool( | ||||||
| bool ForeachDirectoryEntry(u64* num_entries_out, const std::string& directory, | bool ForeachDirectoryEntry(u64* num_entries_out, const std::string& directory, | ||||||
|                            DirectoryEntryCallable callback); |                            DirectoryEntryCallable callback); | ||||||
| 
 | 
 | ||||||
| // Deletes the given path and anything under it. Returns true on success.
 | /**
 | ||||||
| bool DeleteDirRecursively(const std::filesystem::path& path); |  * Scans the directory tree, storing the results. | ||||||
|  |  * @param directory the parent directory to start scanning from | ||||||
|  |  * @param parent_entry FSTEntry where the filesystem tree results will be stored. | ||||||
|  |  * @param recursion Number of children directories to read before giving up. | ||||||
|  |  * @return the total number of files/directories found | ||||||
|  |  */ | ||||||
|  | u64 ScanDirectoryTree(const std::string& directory, FSTEntry& parent_entry, | ||||||
|  |                       unsigned int recursion = 0); | ||||||
|  | 
 | ||||||
|  | // deletes the given directory and anything under it. Returns true on success.
 | ||||||
|  | bool DeleteDirRecursively(const std::string& directory, unsigned int recursion = 256); | ||||||
| 
 | 
 | ||||||
| // Returns the current directory
 | // Returns the current directory
 | ||||||
| [[nodiscard]] std::optional<std::filesystem::path> GetCurrentDir(); | [[nodiscard]] std::optional<std::string> GetCurrentDir(); | ||||||
| 
 | 
 | ||||||
| // Create directory and copy contents (does not overwrite existing files)
 | // Create directory and copy contents (does not overwrite existing files)
 | ||||||
| void CopyDir(const std::filesystem::path& src, const std::filesystem::path& dst); | void CopyDir(const std::string& source_path, const std::string& dest_path); | ||||||
| 
 | 
 | ||||||
| // Set the current directory to given path
 | // Set the current directory to given directory
 | ||||||
| bool SetCurrentDir(const std::filesystem::path& path); | bool SetCurrentDir(const std::string& directory); | ||||||
| 
 | 
 | ||||||
| // Returns a pointer to a string with a yuzu data dir in the user's home
 | // Returns a pointer to a string with a yuzu data dir in the user's home
 | ||||||
| // directory. To be used in "multi-user" mode (that is, installed).
 | // directory. To be used in "multi-user" mode (that is, installed).
 | ||||||
|  |  | ||||||
|  | @ -94,13 +94,9 @@ VirtualFile RealVfsFilesystem::OpenFile(std::string_view path_, Mode perms) { | ||||||
| 
 | 
 | ||||||
| VirtualFile RealVfsFilesystem::CreateFile(std::string_view path_, Mode perms) { | VirtualFile RealVfsFilesystem::CreateFile(std::string_view path_, Mode perms) { | ||||||
|     const auto path = FS::SanitizePath(path_, FS::DirectorySeparator::PlatformDefault); |     const auto path = FS::SanitizePath(path_, FS::DirectorySeparator::PlatformDefault); | ||||||
|     const auto parent_path = FS::GetParentPath(path); |     const auto path_fwd = FS::SanitizePath(path, FS::DirectorySeparator::ForwardSlash); | ||||||
| 
 |  | ||||||
|     if (!FS::Exists(path)) { |     if (!FS::Exists(path)) { | ||||||
|         if (!FS::CreateDirs(parent_path)) { |         FS::CreateFullPath(path_fwd); | ||||||
|             return nullptr; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (!FS::CreateEmptyFile(path)) { |         if (!FS::CreateEmptyFile(path)) { | ||||||
|             return nullptr; |             return nullptr; | ||||||
|         } |         } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei