2019-03-28 17:43:33 +00:00
|
|
|
/*
|
2019-04-08 02:00:49 +00:00
|
|
|
* Copyright (c) 2018-2019 Atmosphère-NX
|
2019-03-28 17:43:33 +00:00
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
#include <cstring>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <switch.h>
|
2019-03-28 21:23:34 +00:00
|
|
|
|
2019-03-28 17:43:33 +00:00
|
|
|
#include "fs_dir_utils.hpp"
|
|
|
|
#include "fs_ifile.hpp"
|
|
|
|
|
|
|
|
Result FsDirUtils::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) {
|
|
|
|
Result rc;
|
|
|
|
std::unique_ptr<IFile> src_file;
|
|
|
|
std::unique_ptr<IFile> dst_file;
|
|
|
|
const u64 file_size = dir_ent->fileSize;
|
|
|
|
|
|
|
|
/* Open source file for reading. */
|
|
|
|
if (R_FAILED((rc = src_fs->OpenFile(src_file, src_path, OpenMode_Read)))) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create and open destination file. */
|
|
|
|
{
|
|
|
|
FsPath dst_path;
|
2019-04-04 22:41:41 +00:00
|
|
|
if (static_cast<size_t>(snprintf(dst_path.str, sizeof(dst_path.str), "%s%s", dst_parent_path.str, dir_ent->name)) >= sizeof(dst_path)) {
|
2019-03-28 17:43:33 +00:00
|
|
|
/* TODO: Error code? N aborts here. */
|
|
|
|
std::abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (R_FAILED((rc = dst_fs->CreateFile(dst_path, file_size)))) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
if (R_FAILED((rc = dst_fs->OpenFile(dst_file, dst_path, OpenMode_Write)))) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Read/Write work_buf_size chunks. */
|
|
|
|
u64 offset = 0;
|
|
|
|
while (offset < file_size) {
|
|
|
|
u64 read_size;
|
|
|
|
if (R_FAILED((rc = src_file->Read(&read_size, offset, work_buf, work_buf_size)))) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
if (R_FAILED((rc = dst_file->Write(offset, work_buf, read_size)))) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
offset += read_size;
|
|
|
|
}
|
2019-04-05 20:36:38 +00:00
|
|
|
|
2019-03-29 05:39:39 +00:00
|
|
|
return ResultSuccess;
|
2019-03-28 17:43:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result FsDirUtils::CopyDirectoryRecursively(IFileSystem *dst_fs, IFileSystem *src_fs, const FsPath &dst_path, const FsPath &src_path, void *work_buf, size_t work_buf_size) {
|
|
|
|
FsPath work_path = dst_path;
|
|
|
|
|
2019-04-05 20:36:38 +00:00
|
|
|
return IterateDirectoryRecursively(src_fs, src_path,
|
2019-03-28 17:43:33 +00:00
|
|
|
[&](const FsPath &path, const FsDirectoryEntry *dir_ent) -> Result { /* On Enter Directory */
|
|
|
|
/* Update path, create new dir. */
|
|
|
|
strncat(work_path.str, dir_ent->name, sizeof(work_path) - strnlen(work_path.str, sizeof(work_path) - 1) - 1);
|
|
|
|
strncat(work_path.str, "/", sizeof(work_path) - strnlen(work_path.str, sizeof(work_path) - 1) - 1);
|
|
|
|
return dst_fs->CreateDirectory(work_path);
|
2019-04-05 20:36:38 +00:00
|
|
|
},
|
2019-03-28 17:43:33 +00:00
|
|
|
[&](const FsPath &path, const FsDirectoryEntry *dir_ent) -> Result { /* On Exit Directory */
|
|
|
|
/* Check we have a parent directory. */
|
|
|
|
const size_t work_path_len = strnlen(work_path.str, sizeof(work_path));
|
|
|
|
if (work_path_len < 2) {
|
|
|
|
return ResultFsInvalidPathFormat;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find previous separator, add NULL terminator */
|
|
|
|
char *p = &work_path.str[work_path_len - 2];
|
|
|
|
while (*p != '/' && p > work_path.str) {
|
|
|
|
p--;
|
|
|
|
}
|
|
|
|
p[1] = 0;
|
|
|
|
|
2019-03-29 05:39:39 +00:00
|
|
|
return ResultSuccess;
|
2019-04-05 20:36:38 +00:00
|
|
|
},
|
2019-03-28 17:43:33 +00:00
|
|
|
[&](const FsPath &path, const FsDirectoryEntry *dir_ent) -> Result { /* On File */
|
|
|
|
/* Just copy the file to the new fs. */
|
|
|
|
return CopyFile(dst_fs, src_fs, work_path, path, dir_ent, work_buf, work_buf_size);
|
|
|
|
});
|
2019-04-05 20:36:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result FsDirUtils::EnsureDirectoryExists(IFileSystem *fs, const FsPath &path) {
|
|
|
|
FsPath normal_path;
|
|
|
|
size_t normal_path_len;
|
|
|
|
Result rc;
|
|
|
|
|
|
|
|
/* Normalize the path. */
|
|
|
|
if (R_FAILED((rc = FsPathUtils::Normalize(normal_path.str, sizeof(normal_path.str) - 1, path.str, &normal_path_len)))) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Repeatedly call CreateDirectory on each directory leading to the target. */
|
|
|
|
for (size_t i = 1; i < normal_path_len; i++) {
|
|
|
|
/* If we detect a separator, we're done. */
|
|
|
|
if (normal_path.str[i] == '/') {
|
|
|
|
normal_path.str[i] = 0;
|
|
|
|
{
|
|
|
|
rc = fs->CreateDirectory(normal_path);
|
|
|
|
if (rc == ResultFsPathAlreadyExists) {
|
|
|
|
rc = ResultSuccess;
|
|
|
|
}
|
|
|
|
if (R_FAILED(rc)) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
normal_path.str[i] = '/';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Call CreateDirectory on the final path. */
|
|
|
|
rc = fs->CreateDirectory(normal_path);
|
|
|
|
if (rc == ResultFsPathAlreadyExists) {
|
|
|
|
rc = ResultSuccess;
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|