mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-31 11:15:51 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			159 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2019 Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| #include <switch.h>
 | |
| 
 | |
| #include "fs_path_utils.hpp"
 | |
| #include "fs_ifilesystem.hpp"
 | |
| 
 | |
| class FsDirUtils {
 | |
|     private:
 | |
|         template<typename OnEnterDir, typename OnExitDir, typename OnFile>
 | |
|         static Result IterateDirectoryRecursivelyInternal(IFileSystem *fs, FsPath &work_path, FsDirectoryEntry *ent_buf, OnEnterDir on_enter_dir, OnExitDir on_exit_dir, OnFile on_file) {
 | |
|             Result rc;
 | |
|             std::unique_ptr<IDirectory> dir;
 | |
| 
 | |
|             /* Open the directory. */
 | |
|             if (R_FAILED((rc = fs->OpenDirectory(dir, work_path, DirectoryOpenMode_All)))) {
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             const size_t parent_len = strnlen(work_path.str, sizeof(work_path.str) - 1);
 | |
| 
 | |
|             /* Read and handle entries. */
 | |
|             while (true) {
 | |
|                 /* Read a single entry. */
 | |
|                 u64 read_count;
 | |
|                 if (R_FAILED((rc = dir->Read(&read_count, ent_buf, 1)))) {
 | |
|                     return rc;
 | |
|                 }
 | |
| 
 | |
|                 /* If we're out of entries, we're done. */
 | |
|                 if (read_count == 0) {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 const size_t child_name_len = strnlen(ent_buf->name, sizeof(ent_buf->name) - 1);
 | |
|                 const bool is_dir = ent_buf->type == ENTRYTYPE_DIR;
 | |
|                 const size_t child_path_len = parent_len + child_name_len + (is_dir ? 1 : 0);
 | |
| 
 | |
|                 /* Validate child path size. */
 | |
|                 if (child_path_len >= sizeof(work_path.str)) {
 | |
|                     return ResultFsTooLongPath;
 | |
|                 }
 | |
| 
 | |
|                 strncat(work_path.str, ent_buf->name, sizeof(work_path.str) - 1 - parent_len);
 | |
|                 if (is_dir) {
 | |
|                     /* Enter directory. */
 | |
|                     if (R_FAILED((rc = on_enter_dir(work_path, ent_buf)))) {
 | |
|                         return rc;
 | |
|                     }
 | |
| 
 | |
|                     /* Append separator, recurse. */
 | |
|                     strncat(work_path.str, "/", sizeof(work_path.str) - 1 - parent_len - child_name_len);
 | |
|                     if (R_FAILED((rc = IterateDirectoryRecursivelyInternal(fs, work_path, ent_buf, on_enter_dir, on_exit_dir, on_file)))) {
 | |
|                         return rc;
 | |
|                     }
 | |
| 
 | |
|                     /* Exit directory. */
 | |
|                     if (R_FAILED((rc = on_exit_dir(work_path, ent_buf)))) {
 | |
|                         return rc;
 | |
|                     }
 | |
|                 } else {
 | |
|                     /* Call file handler. */
 | |
|                     if (R_FAILED((rc = on_file(work_path, ent_buf)))) {
 | |
|                         return rc;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 /* Restore parent path. */
 | |
|                 work_path.str[parent_len] = 0;
 | |
|             }
 | |
| 
 | |
|             return ResultSuccess;
 | |
|         }
 | |
| 
 | |
|     public:
 | |
|         template<typename OnEnterDir, typename OnExitDir, typename OnFile>
 | |
|         static Result IterateDirectoryRecursively(IFileSystem *fs, const FsPath &root_path, FsPath &work_path, FsDirectoryEntry *ent_buf, OnEnterDir on_enter_dir, OnExitDir on_exit_dir, OnFile on_file) {
 | |
|             /* Ensure valid root path. */
 | |
|             size_t root_path_len = strnlen(root_path.str, sizeof(root_path.str));
 | |
|             if (root_path_len > FS_MAX_PATH - 1 || ((root_path_len == FS_MAX_PATH - 1) && root_path.str[root_path_len-1] != '/')) {
 | |
|                 return ResultFsTooLongPath;
 | |
|             }
 | |
| 
 | |
|             /* Copy path, ensure terminating separator. */
 | |
|             memcpy(work_path.str, root_path.str, root_path_len);
 | |
|             if (work_path.str[root_path_len-1] != '/') {
 | |
|                 root_path_len++;
 | |
|                 work_path.str[root_path_len-1] = '/';
 | |
|             }
 | |
|             work_path.str[root_path_len] = 0;
 | |
| 
 | |
|             /* Actually iterate. */
 | |
|             return IterateDirectoryRecursivelyInternal(fs, work_path, ent_buf, on_enter_dir, on_exit_dir, on_file);
 | |
|         }
 | |
| 
 | |
|         /* Helper for not specifying work path/entry buffer. */
 | |
|         template<typename OnEnterDir, typename OnExitDir, typename OnFile>
 | |
|         static Result IterateDirectoryRecursively(IFileSystem *fs, const FsPath &root_path, OnEnterDir on_enter_dir, OnExitDir on_exit_dir, OnFile on_file) {
 | |
|             FsDirectoryEntry dir_ent = {0};
 | |
|             FsPath work_path = {0};
 | |
|             return IterateDirectoryRecursively(fs, root_path, work_path, &dir_ent, on_enter_dir, on_exit_dir, on_file);
 | |
|         }
 | |
| 
 | |
|         /* Helper for iterating over the filesystem root. */
 | |
|         template<typename OnEnterDir, typename OnExitDir, typename OnFile>
 | |
|         static Result IterateDirectoryRecursively(IFileSystem *fs, OnEnterDir on_enter_dir, OnExitDir on_exit_dir, OnFile on_file) {
 | |
|             return IterateDirectoryRecursively(fs, FsPathUtils::RootPath, on_enter_dir, on_exit_dir, on_file);
 | |
|         }
 | |
| 
 | |
|         /* Copy API. */
 | |
|         static Result CopyFile(IFileSystem *dst_fs, IFileSystem *src_fs, const FsPath &dst_parent_path, const FsPath &src_path, const FsDirectoryEntry *dir_ent, void *work_buf, size_t work_buf_size);
 | |
|         static Result CopyFile(IFileSystem *fs, const FsPath &dst_parent_path, const FsPath &src_path, const FsDirectoryEntry *dir_ent, void *work_buf, size_t work_buf_size) {
 | |
|             return CopyFile(fs, fs, dst_parent_path, src_path, dir_ent, work_buf, work_buf_size);
 | |
|         }
 | |
| 
 | |
|         static Result CopyDirectoryRecursively(IFileSystem *dst_fs, IFileSystem *src_fs, const FsPath &dst_path, const FsPath &src_path, void *work_buf, size_t work_buf_size);
 | |
|         static Result CopyDirectoryRecursively(IFileSystem *fs, const FsPath &dst_path, const FsPath &src_path, void *work_buf, size_t work_buf_size) {
 | |
|             return CopyDirectoryRecursively(fs, fs, dst_path, src_path, work_buf, work_buf_size);
 | |
|         }
 | |
|         
 | |
|         /* Ensure directory existence. */
 | |
|         static Result EnsureDirectoryExists(IFileSystem *fs, const FsPath &path);
 | |
| 
 | |
|         /* Other Utility. */
 | |
|         template<typename F>
 | |
|         static Result RetryUntilTargetNotLocked(F f) {
 | |
|             const size_t MaxRetries = 10;
 | |
|             Result rc = ResultSuccess;
 | |
| 
 | |
|             for (size_t i = 0; i < MaxRetries; i++) {
 | |
|                 rc = f();
 | |
| 
 | |
|                 if (rc != ResultFsTargetLocked) {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 /* If target is locked, wait 100ms and try again. */
 | |
|                 svcSleepThread(100'000'000ul);
 | |
|             }
 | |
| 
 | |
|             return rc;
 | |
|         }
 | |
| };
 |