#include "extract.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "current_cfw.hpp" #include "download.hpp" #include "fs.hpp" #include "main_frame.hpp" #include "progress_event.hpp" #include "utils.hpp" namespace i18n = brls::i18n; using namespace i18n::literals; constexpr size_t WRITE_BUFFER_SIZE = 0x10000; namespace extract { namespace { bool caselessCompare(const std::string& a, const std::string& b) { return strcasecmp(a.c_str(), b.c_str()) == 0; } s64 getUncompressedSize(const std::string& archivePath) { s64 size = 0; unzFile zfile = unzOpen(archivePath.c_str()); unz_global_info gi; unzGetGlobalInfo(zfile, &gi); for (uLong i = 0; i < gi.number_entry; ++i) { unz_file_info fi; unzOpenCurrentFile(zfile); unzGetCurrentFileInfo(zfile, &fi, NULL, 0, NULL, 0, NULL, 0); size += fi.uncompressed_size; unzCloseCurrentFile(zfile); unzGoToNextFile(zfile); } unzClose(zfile); return size; // in B } void preWork(const std::string& archivePath, const std::string& workingPath) { chdir(workingPath.c_str()); s64 uncompressedSize = getUncompressedSize(archivePath); s64 freeStorage; if (R_SUCCEEDED(fs::getFreeStorageSD(freeStorage))) { brls::Logger::info("Uncompressed size of archive {}: {}. Available: {}", archivePath, uncompressedSize, freeStorage); if (uncompressedSize * 1.1 > freeStorage) { brls::Application::crash("menus/errors/insufficient_storage"_i18n); std::this_thread::sleep_for(std::chrono::microseconds(2000000)); brls::Application::quit(); } } } void extractEntry(std::string filename, unzFile& zfile, bool forceCreateTree = false) { if (filename.back() == '/') { fs::createTree(filename); return; } if (forceCreateTree) { fs::createTree(filename); } void* buf = malloc(WRITE_BUFFER_SIZE); FILE* outfile; outfile = fopen(filename.c_str(), "wb"); for (int j = unzReadCurrentFile(zfile, buf, WRITE_BUFFER_SIZE); j > 0; j = unzReadCurrentFile(zfile, buf, WRITE_BUFFER_SIZE)) { fwrite(buf, 1, j, outfile); } free(buf); fclose(outfile); } } // namespace void extract(const std::string& archivePath, const std::string& workingPath, int overwriteInis, std::function func) { preWork(archivePath, workingPath); unzFile zfile = unzOpen(archivePath.c_str()); unz_global_info gi; unzGetGlobalInfo(zfile, &gi); ProgressEvent::instance().setTotalSteps(gi.number_entry); ProgressEvent::instance().setStep(0); std::set ignoreList = fs::readLineByLine(FILES_IGNORE); std::string appPath = util::getAppPath(); for (uLong i = 0; i < gi.number_entry; ++i) { char szFilename[0x301] = ""; unzOpenCurrentFile(zfile); unzGetCurrentFileInfo(zfile, NULL, szFilename, sizeof(szFilename), NULL, 0, NULL, 0); std::string filename = szFilename; if (ProgressEvent::instance().getInterupt()) { unzCloseCurrentFile(zfile); break; } if (appPath != workingPath + filename) { if ((overwriteInis == 0 && filename.substr(filename.length() - 4) == ".ini") || std::find_if(ignoreList.begin(), ignoreList.end(), [&filename, &workingPath](std::string ignored) { u8 res = (workingPath + filename).find(ignored); return (res == 0 || res == 1); }) != ignoreList.end()) { if (!std::filesystem::exists(workingPath + filename)) { extractEntry(filename, zfile); } } else { if ((filename == "atmosphere/package3") || (filename == "atmosphere/stratosphere.romfs")) { extractEntry(filename + ".aio", zfile); } else { extractEntry(filename, zfile); if (filename.substr(0, 13) == "hekate_ctcaer") { fs::copyFile(workingPath + filename, UPDATE_BIN_PATH); if (CurrentCfw::running_cfw == CFW::ams && util::showDialogBoxBlocking(fmt::format("menus/utils/set_hekate_reboot_payload"_i18n, UPDATE_BIN_PATH, REBOOT_PAYLOAD_PATH), "menus/common/yes"_i18n, "menus/common/no"_i18n) == 0) { fs::copyFile(UPDATE_BIN_PATH, REBOOT_PAYLOAD_PATH); } } } } } ProgressEvent::instance().setStep(i); unzCloseCurrentFile(zfile); unzGoToNextFile(zfile); } unzClose(zfile); ProgressEvent::instance().setStep(ProgressEvent::instance().getMax()); } std::vector getInstalledTitlesNs() { std::vector titles; NsApplicationRecord* records = new NsApplicationRecord[MaxTitleCount](); NsApplicationControlData* controlData = NULL; s32 recordCount = 0; u64 controlSize = 0; if (R_SUCCEEDED(nsListApplicationRecord(records, MaxTitleCount, 0, &recordCount))) { for (s32 i = 0; i < recordCount; i++) { controlSize = 0; free(controlData); controlData = (NsApplicationControlData*)malloc(sizeof(NsApplicationControlData)); if (controlData == NULL) { break; } else { memset(controlData, 0, sizeof(NsApplicationControlData)); } if (R_FAILED(nsGetApplicationControlData(NsApplicationControlSource_Storage, records[i].application_id, controlData, sizeof(NsApplicationControlData), &controlSize))) continue; if (controlSize < sizeof(controlData->nacp)) { continue; } titles.push_back(util::formatApplicationId(records[i].application_id)); } free(controlData); } delete[] records; std::sort(titles.begin(), titles.end()); return titles; } std::vector excludeTitles(const std::string& path, const std::vector& listedTitles) { std::vector titles; std::ifstream file(path); std::string name; if (file.is_open()) { std::string line; while (std::getline(file, line)) { std::transform(line.begin(), line.end(), line.begin(), ::toupper); for (size_t i = 0; i < listedTitles.size(); i++) { if (line == listedTitles[i]) { titles.push_back(line); break; } } } } std::sort(titles.begin(), titles.end()); std::vector diff; std::set_difference(listedTitles.begin(), listedTitles.end(), titles.begin(), titles.end(), std::inserter(diff, diff.begin())); return diff; } int computeOffset(CFW cfw) { switch (cfw) { case CFW::ams: std::filesystem::create_directory(AMS_PATH); std::filesystem::create_directory(AMS_CONTENTS); chdir(AMS_PATH); return std::string(CONTENTS_PATH).length(); break; case CFW::rnx: std::filesystem::create_directory(REINX_PATH); std::filesystem::create_directory(REINX_CONTENTS); chdir(REINX_PATH); return std::string(CONTENTS_PATH).length(); break; case CFW::sxos: std::filesystem::create_directory(SXOS_PATH); std::filesystem::create_directory(SXOS_TITLES); chdir(SXOS_PATH); return std::string(TITLES_PATH).length(); break; } return 0; } void extractCheats(const std::string& archivePath, const std::vector& titles, CFW cfw, const std::string& version, bool extractAll) { preWork(archivePath, "/"); unzFile zfile = unzOpen(archivePath.c_str()); unz_global_info gi; unzGetGlobalInfo(zfile, &gi); ProgressEvent::instance().setTotalSteps(gi.number_entry); ProgressEvent::instance().setStep(0); int offset = computeOffset(cfw); for (uLong i = 0; i < gi.number_entry; ++i) { char szFilename[0x301] = ""; unzOpenCurrentFile(zfile); unzGetCurrentFileInfo(zfile, NULL, szFilename, sizeof(szFilename), NULL, 0, NULL, 0); std::string filename = szFilename; if (ProgressEvent::instance().getInterupt()) { unzCloseCurrentFile(zfile); break; } if ((int)filename.size() > offset + 16 + 7 && caselessCompare(filename.substr(offset + 16, 7), "/cheats")) { if (extractAll) { extractEntry(filename, zfile); } else { if (std::find_if(titles.begin(), titles.end(), [&filename, offset](std::string title) { return caselessCompare((title.substr(0, 13)), filename.substr(offset, 13)); }) != titles.end()) { extractEntry(filename, zfile); } } } ProgressEvent::instance().setStep(i); unzCloseCurrentFile(zfile); unzGoToNextFile(zfile); } unzClose(zfile); if (version != "offline" && version != "") { util::saveToFile(version, CHEATS_VERSION); } ProgressEvent::instance().setStep(ProgressEvent::instance().getMax()); } void extractAllCheats(const std::string& archivePath, CFW cfw, const std::string& version) { extractCheats(archivePath, {}, cfw, version, true); } bool isBID(const std::string& bid) { for (char const& c : bid) { if (!isxdigit(c)) return false; } return true; } void writeTitlesToFile(const std::set& titles, const std::string& path) { std::ofstream updatedTitlesFile; std::set::iterator it = titles.begin(); updatedTitlesFile.open(path, std::ofstream::out | std::ofstream::trunc); if (updatedTitlesFile.is_open()) { while (it != titles.end()) { updatedTitlesFile << (*it) << std::endl; it++; } updatedTitlesFile.close(); } } void removeCheats() { std::string path = util::getContentsPath(); ProgressEvent::instance().setTotalSteps(std::distance(std::filesystem::directory_iterator(path), std::filesystem::directory_iterator()) + 1); for (const auto& entry : std::filesystem::directory_iterator(path)) { if (ProgressEvent::instance().getInterupt()) { break; } removeCheatsDirectory(entry.path().string()); ProgressEvent::instance().incrementStep(1); } std::filesystem::remove(CHEATS_VERSION); ProgressEvent::instance().setStep(ProgressEvent::instance().getMax()); } void removeOrphanedCheats() { auto path = util::getContentsPath(); std::vector titles = getInstalledTitlesNs(); ProgressEvent::instance().setTotalSteps(std::distance(std::filesystem::directory_iterator(path), std::filesystem::directory_iterator()) + 1); for (const auto& entry : std::filesystem::directory_iterator(path)) { if (ProgressEvent::instance().getInterupt()) { break; } if (std::find_if(titles.begin(), titles.end(), [&entry](std::string title) { return caselessCompare(entry.path().filename(), title); }) == titles.end()) { removeCheatsDirectory(entry.path().string()); } ProgressEvent::instance().incrementStep(1); } std::filesystem::remove(CHEATS_VERSION); ProgressEvent::instance().setStep(ProgressEvent::instance().getMax()); } bool removeCheatsDirectory(const std::string& entry) { bool res = true; std::string cheatsPath = fmt::format("{}/cheats", entry); if (std::filesystem::exists(cheatsPath)) res &= fs::removeDir(cheatsPath); if (std::filesystem::is_empty(entry)) res &= fs::removeDir(entry); return res; } } // namespace extract