2018-05-09 09:25:01 +00:00
|
|
|
/*
|
|
|
|
* This file is part of Checkpoint
|
2019-03-07 20:15:24 +00:00
|
|
|
* Copyright (C) 2017-2019 Bernardo Giordano, FlagBrew
|
2018-05-09 09:25:01 +00:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that 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/>.
|
|
|
|
*
|
|
|
|
* Additional Terms 7.b and 7.c of GPLv3 apply to this file:
|
|
|
|
* * Requiring preservation of specified reasonable legal notices or
|
|
|
|
* author attributions in that material or in the Appropriate Legal
|
|
|
|
* Notices displayed by works containing it.
|
|
|
|
* * Prohibiting misrepresentation of the origin of that material,
|
|
|
|
* or requiring that modified versions of such material be marked in
|
|
|
|
* reasonable ways as different from the original version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "io.hpp"
|
|
|
|
|
|
|
|
bool io::fileExists(const std::string& path)
|
|
|
|
{
|
|
|
|
struct stat buffer;
|
|
|
|
return (stat (path.c_str(), &buffer) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void io::copyFile(const std::string& srcPath, const std::string& dstPath)
|
|
|
|
{
|
2019-04-30 18:57:22 +00:00
|
|
|
FILE* src = fopen(srcPath.c_str(), "rb");
|
|
|
|
FILE* dst = fopen(dstPath.c_str(), "wb");
|
|
|
|
if (!src || !dst)
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2019-04-30 18:57:22 +00:00
|
|
|
|
|
|
|
fseek(src, 0, SEEK_END);
|
|
|
|
u64 sz = ftell(src);
|
|
|
|
rewind(src);
|
2018-05-09 09:25:01 +00:00
|
|
|
|
2018-09-03 19:36:28 +00:00
|
|
|
u8* buf = new u8[BUFFER_SIZE];
|
2018-05-09 09:25:01 +00:00
|
|
|
u64 offset = 0;
|
|
|
|
size_t slashpos = srcPath.rfind("/");
|
|
|
|
std::string name = srcPath.substr(slashpos + 1, srcPath.length() - slashpos - 1);
|
2018-09-03 19:36:28 +00:00
|
|
|
while (offset < sz) {
|
2019-04-30 18:57:22 +00:00
|
|
|
u32 count = fread((char*)buf, 1, BUFFER_SIZE, src);
|
|
|
|
fwrite((char*)buf, 1, count, dst);
|
|
|
|
offset += count;
|
2018-05-09 09:25:01 +00:00
|
|
|
Gui::drawCopy(name, offset, sz);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete[] buf;
|
2019-04-30 18:57:22 +00:00
|
|
|
fclose(src);
|
|
|
|
fclose(dst);
|
2018-05-11 20:20:54 +00:00
|
|
|
|
|
|
|
// commit each file to the save
|
2018-05-12 19:07:24 +00:00
|
|
|
if (dstPath.rfind("save:/", 0) == 0)
|
|
|
|
{
|
|
|
|
fsdevCommitDevice("save");
|
|
|
|
}
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result io::copyDirectory(const std::string& srcPath, const std::string& dstPath)
|
|
|
|
{
|
|
|
|
Result res = 0;
|
|
|
|
bool quit = false;
|
|
|
|
Directory items(srcPath);
|
|
|
|
|
|
|
|
if (!items.good())
|
|
|
|
{
|
|
|
|
return items.error();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0, sz = items.size(); i < sz && !quit; i++)
|
|
|
|
{
|
|
|
|
std::string newsrc = srcPath + items.entry(i);
|
|
|
|
std::string newdst = dstPath + items.entry(i);
|
|
|
|
|
|
|
|
if (items.folder(i))
|
|
|
|
{
|
|
|
|
res = io::createDirectory(newdst);
|
|
|
|
if (R_SUCCEEDED(res))
|
|
|
|
{
|
|
|
|
newsrc += "/";
|
|
|
|
newdst += "/";
|
|
|
|
res = io::copyDirectory(newsrc, newdst);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
quit = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
io::copyFile(newsrc, newdst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result io::createDirectory(const std::string& path)
|
|
|
|
{
|
|
|
|
mkdir(path.c_str(), 777);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool io::directoryExists(const std::string& path)
|
|
|
|
{
|
|
|
|
struct stat sb;
|
|
|
|
return (stat(path.c_str(), &sb) == 0 && S_ISDIR(sb.st_mode));
|
|
|
|
}
|
|
|
|
|
2018-09-03 19:36:28 +00:00
|
|
|
Result io::deleteFolderRecursively(const std::string& path)
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
2018-09-03 19:36:28 +00:00
|
|
|
Directory dir(path);
|
|
|
|
if (!dir.good())
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
2018-09-03 19:36:28 +00:00
|
|
|
return dir.error();
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
2018-09-03 19:36:28 +00:00
|
|
|
for (size_t i = 0, sz = dir.size(); i < sz; i++)
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
2018-09-03 19:36:28 +00:00
|
|
|
if (dir.folder(i))
|
|
|
|
{
|
|
|
|
std::string newpath = path + "/" + dir.entry(i) + "/";
|
|
|
|
deleteFolderRecursively(newpath);
|
|
|
|
newpath = path + dir.entry(i);
|
|
|
|
rmdir(newpath.c_str());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string newpath = path + dir.entry(i);
|
|
|
|
std::remove(newpath.c_str());
|
|
|
|
}
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
2018-09-03 19:36:28 +00:00
|
|
|
rmdir(path.c_str());
|
2018-05-09 09:25:01 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-10 09:11:07 +00:00
|
|
|
void io::backup(size_t index, u128 uid)
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
|
|
|
// check if multiple selection is enabled and don't ask for confirmation if that's the case
|
2019-04-21 09:41:51 +00:00
|
|
|
if (!MS::multipleSelectionEnabled())
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
|
|
|
if (!Gui::askForConfirmation("Backup selected save?"))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t cellIndex = Gui::index(CELLS);
|
|
|
|
const bool isNewFolder = cellIndex == 0;
|
|
|
|
Result res = 0;
|
|
|
|
|
|
|
|
Title title;
|
2018-06-10 09:11:07 +00:00
|
|
|
getTitle(title, uid, index);
|
2018-05-09 09:25:01 +00:00
|
|
|
|
|
|
|
FsFileSystem fileSystem;
|
|
|
|
res = FileSystem::mount(&fileSystem, title.id(), title.userId());
|
|
|
|
if (R_SUCCEEDED(res))
|
|
|
|
{
|
|
|
|
int ret = FileSystem::mount(fileSystem);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(-2, "Failed to mount save.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(res, "Failed to mount save.");
|
2018-07-17 16:25:07 +00:00
|
|
|
return;
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
2018-08-05 15:00:07 +00:00
|
|
|
std::string suggestion = DateTime::dateTimeStr() + " " + (StringUtils::containsInvalidChar(Account::username(title.userId())) ? "" : StringUtils::removeNotAscii(StringUtils::removeAccents(Account::username(title.userId()))));
|
2018-05-09 09:25:01 +00:00
|
|
|
std::string customPath;
|
|
|
|
|
2019-04-21 09:41:51 +00:00
|
|
|
if (MS::multipleSelectionEnabled())
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
2018-07-17 16:25:07 +00:00
|
|
|
customPath = isNewFolder ? suggestion : "";
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-08-05 14:44:55 +00:00
|
|
|
if (isNewFolder)
|
|
|
|
{
|
|
|
|
std::pair<bool, std::string> keyboardResponse = KeyboardManager::get().keyboard(suggestion);
|
|
|
|
if (keyboardResponse.first)
|
|
|
|
{
|
|
|
|
customPath = StringUtils::removeForbiddenCharacters(keyboardResponse.second);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
customPath = "";
|
|
|
|
}
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
2018-07-17 16:25:07 +00:00
|
|
|
std::string dstPath;
|
|
|
|
if (!isNewFolder)
|
|
|
|
{
|
|
|
|
// we're overriding an existing folder
|
|
|
|
dstPath = title.fullPath(cellIndex);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dstPath = title.path() + "/" + customPath;
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:25:01 +00:00
|
|
|
if (!isNewFolder || io::directoryExists(dstPath))
|
|
|
|
{
|
2018-09-03 19:36:28 +00:00
|
|
|
int ret = io::deleteFolderRecursively((dstPath + "/").c_str());
|
2018-05-09 09:25:01 +00:00
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError((Result)ret, "Failed to delete the existing backup\ndirectory recursively.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res = io::createDirectory(dstPath);
|
|
|
|
res = io::copyDirectory("save:/", dstPath + "/");
|
|
|
|
if (R_FAILED(res))
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
2018-09-03 19:36:28 +00:00
|
|
|
io::deleteFolderRecursively((dstPath + "/").c_str());
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(res, "Failed to backup save.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
refreshDirectories(title.id());
|
|
|
|
|
|
|
|
FileSystem::unmount();
|
2019-04-21 09:41:51 +00:00
|
|
|
if (!MS::multipleSelectionEnabled())
|
2019-04-12 21:39:55 +00:00
|
|
|
{
|
|
|
|
Gui::showInfo("Progress correctly saved to disk.");
|
|
|
|
}
|
|
|
|
auto systemKeyboardAvailable = KeyboardManager::get().isSystemKeyboardAvailable();
|
|
|
|
if (!systemKeyboardAvailable.first)
|
|
|
|
{
|
|
|
|
Gui::showError(systemKeyboardAvailable.second, "System keyboard applet not accessible.\nThe suggested destination folder was used\ninstead.");
|
|
|
|
}
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
2018-06-10 09:11:07 +00:00
|
|
|
void io::restore(size_t index, u128 uid)
|
2018-05-09 09:25:01 +00:00
|
|
|
{
|
|
|
|
const size_t cellIndex = Gui::index(CELLS);
|
|
|
|
if (cellIndex == 0 || !Gui::askForConfirmation("Restore selected save?"))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result res = 0;
|
|
|
|
|
|
|
|
Title title;
|
2018-06-10 09:11:07 +00:00
|
|
|
getTitle(title, uid, index);
|
2018-05-09 09:25:01 +00:00
|
|
|
|
|
|
|
FsFileSystem fileSystem;
|
2018-07-17 16:25:07 +00:00
|
|
|
res = title.systemSave() ? FileSystem::mount(&fileSystem, title.id()) : FileSystem::mount(&fileSystem, title.id(), title.userId());
|
2018-05-09 09:25:01 +00:00
|
|
|
if (R_SUCCEEDED(res))
|
|
|
|
{
|
|
|
|
int ret = FileSystem::mount(fileSystem);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(-2, "Failed to mount save.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(res, "Failed to mount save.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-17 16:25:07 +00:00
|
|
|
std::string srcPath = title.fullPath(cellIndex) + "/";
|
2018-05-09 09:25:01 +00:00
|
|
|
std::string dstPath = "save:/";
|
|
|
|
|
|
|
|
res = io::deleteFolderRecursively(dstPath.c_str());
|
|
|
|
if (R_FAILED(res))
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(res, "Failed to delete save.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = io::copyDirectory(srcPath, dstPath);
|
|
|
|
if (R_FAILED(res))
|
|
|
|
{
|
|
|
|
FileSystem::unmount();
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(res, "Failed to restore save.");
|
2018-05-09 09:25:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = fsdevCommitDevice("save");
|
|
|
|
if (R_FAILED(res))
|
|
|
|
{
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showError(res, "Failed to commit to save device.");
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-04-12 21:39:55 +00:00
|
|
|
Gui::showInfo(Gui::nameFromCell(cellIndex) + "\nhas been restored successfully.");
|
2018-05-09 09:25:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
FileSystem::unmount();
|
|
|
|
}
|