| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QDir>
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  | #include <QFile>
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | #include <QFileInfo>
 | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | #include <QSettings>
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | #include "common/fs/fs.h"
 | 
					
						
							|  |  |  | #include "common/fs/path_util.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  | #include "core/core.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/card_image.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | #include "core/file_sys/content_archive.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/control_metadata.h"
 | 
					
						
							| 
									
										
										
										
											2024-01-18 21:31:41 +01:00
										 |  |  | #include "core/file_sys/fs_filesystem.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | #include "core/file_sys/nca_metadata.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/patch_manager.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/registered_cache.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  | #include "core/file_sys/submission_package.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-09 19:09:37 -04:00
										 |  |  | #include "yuzu/compatibility_list.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | #include "yuzu/game_list.h"
 | 
					
						
							|  |  |  | #include "yuzu/game_list_p.h"
 | 
					
						
							|  |  |  | #include "yuzu/game_list_worker.h"
 | 
					
						
							| 
									
										
										
										
											2019-07-29 16:06:33 -04:00
										 |  |  | #include "yuzu/uisettings.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | QString GetGameListCachedObject(const std::string& filename, const std::string& ext, | 
					
						
							|  |  |  |                                 const std::function<QString()>& generator) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |     if (!UISettings::values.cache_game_list || filename == "0000000000000000") { | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |         return generator(); | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     const auto path = | 
					
						
							| 
									
										
										
										
											2025-05-08 22:16:07 +00:00
										 |  |  |         Common::FS::PathToUTF8String(Common::FS::GetEdenPath(Common::FS::EdenPath::CacheDir) / | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |                                      "game_list" / fmt::format("{}.{}", filename, ext)); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     void(Common::FS::CreateParentDirs(path)); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-15 08:33:16 -04:00
										 |  |  |     if (!Common::FS::Exists(path)) { | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |         const auto str = generator(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |         QFile file{QString::fromStdString(path)}; | 
					
						
							|  |  |  |         if (file.open(QFile::WriteOnly)) { | 
					
						
							|  |  |  |             file.write(str.toUtf8()); | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return str; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |     QFile file{QString::fromStdString(path)}; | 
					
						
							|  |  |  |     if (file.open(QFile::ReadOnly)) { | 
					
						
							|  |  |  |         return QString::fromUtf8(file.readAll()); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return generator(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::pair<std::vector<u8>, std::string> GetGameListCachedObject( | 
					
						
							|  |  |  |     const std::string& filename, const std::string& ext, | 
					
						
							|  |  |  |     const std::function<std::pair<std::vector<u8>, std::string>()>& generator) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |     if (!UISettings::values.cache_game_list || filename == "0000000000000000") { | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |         return generator(); | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     const auto game_list_dir = | 
					
						
							| 
									
										
										
										
											2025-05-08 22:16:07 +00:00
										 |  |  |         Common::FS::GetEdenPath(Common::FS::EdenPath::CacheDir) / "game_list"; | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     const auto jpeg_name = fmt::format("{}.jpeg", filename); | 
					
						
							|  |  |  |     const auto app_name = fmt::format("{}.appname.txt", filename); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     const auto path1 = Common::FS::PathToUTF8String(game_list_dir / jpeg_name); | 
					
						
							|  |  |  |     const auto path2 = Common::FS::PathToUTF8String(game_list_dir / app_name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void(Common::FS::CreateParentDirs(path1)); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-15 08:33:16 -04:00
										 |  |  |     if (!Common::FS::Exists(path1) || !Common::FS::Exists(path2)) { | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |         const auto [icon, nacp] = generator(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |         QFile file1{QString::fromStdString(path1)}; | 
					
						
							|  |  |  |         if (!file1.open(QFile::WriteOnly)) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |             LOG_ERROR(Frontend, "Failed to open cache file."); | 
					
						
							|  |  |  |             return generator(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |         if (!file1.resize(icon.size())) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |             LOG_ERROR(Frontend, "Failed to resize cache file to necessary size."); | 
					
						
							|  |  |  |             return generator(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 15:59:23 -04:00
										 |  |  |         if (file1.write(reinterpret_cast<const char*>(icon.data()), icon.size()) != | 
					
						
							|  |  |  |             s64(icon.size())) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |             LOG_ERROR(Frontend, "Failed to write data to cache file."); | 
					
						
							|  |  |  |             return generator(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |         QFile file2{QString::fromStdString(path2)}; | 
					
						
							|  |  |  |         if (file2.open(QFile::WriteOnly)) { | 
					
						
							|  |  |  |             file2.write(nacp.data(), nacp.size()); | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return std::make_pair(icon, nacp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |     QFile file1(QString::fromStdString(path1)); | 
					
						
							|  |  |  |     QFile file2(QString::fromStdString(path2)); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |     if (!file1.open(QFile::ReadOnly)) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |         LOG_ERROR(Frontend, "Failed to open cache file for reading."); | 
					
						
							|  |  |  |         return generator(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |     if (!file2.open(QFile::ReadOnly)) { | 
					
						
							| 
									
										
										
										
											2019-05-26 17:14:09 -04:00
										 |  |  |         LOG_ERROR(Frontend, "Failed to open cache file for reading."); | 
					
						
							|  |  |  |         return generator(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |     std::vector<u8> vec(file1.size()); | 
					
						
							|  |  |  |     if (file1.read(reinterpret_cast<char*>(vec.data()), vec.size()) != | 
					
						
							|  |  |  |         static_cast<s64>(vec.size())) { | 
					
						
							|  |  |  |         return generator(); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 22:13:38 -04:00
										 |  |  |     const auto data = file2.readAll(); | 
					
						
							|  |  |  |     return std::make_pair(vec, data.toStdString()); | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-09 14:22:31 -04:00
										 |  |  | void GetMetadataFromControlNCA(const FileSys::PatchManager& patch_manager, const FileSys::NCA& nca, | 
					
						
							|  |  |  |                                std::vector<u8>& icon, std::string& name) { | 
					
						
							| 
									
										
										
										
											2019-05-30 18:56:04 -04:00
										 |  |  |     std::tie(icon, name) = GetGameListCachedObject( | 
					
						
							| 
									
										
										
										
											2019-04-23 09:08:38 -04:00
										 |  |  |         fmt::format("{:016X}", patch_manager.GetTitleID()), {}, [&patch_manager, &nca] { | 
					
						
							|  |  |  |             const auto [nacp, icon_f] = patch_manager.ParseControlNCA(nca); | 
					
						
							|  |  |  |             return std::make_pair(icon_f->ReadAllBytes(), nacp->GetApplicationName()); | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool HasSupportedFileExtension(const std::string& file_name) { | 
					
						
							|  |  |  |     const QFileInfo file = QFileInfo(QString::fromStdString(file_name)); | 
					
						
							|  |  |  |     return GameList::supported_file_extensions.contains(file.suffix(), Qt::CaseInsensitive); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool IsExtractedNCAMain(const std::string& file_name) { | 
					
						
							| 
									
										
										
										
											2019-05-20 15:07:56 -04:00
										 |  |  |     return QFileInfo(QString::fromStdString(file_name)).fileName() == QStringLiteral("main"); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString FormatGameName(const std::string& physical_name) { | 
					
						
							|  |  |  |     const QString physical_name_as_qstring = QString::fromStdString(physical_name); | 
					
						
							|  |  |  |     const QFileInfo file_info(physical_name_as_qstring); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (IsExtractedNCAMain(physical_name)) { | 
					
						
							|  |  |  |         return file_info.dir().path(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return physical_name_as_qstring; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-25 09:21:06 -04:00
										 |  |  | QString FormatPatchNameVersions(const FileSys::PatchManager& patch_manager, | 
					
						
							|  |  |  |                                 Loader::AppLoader& loader, bool updatable = true) { | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |     QString out; | 
					
						
							| 
									
										
										
										
											2018-09-25 09:21:06 -04:00
										 |  |  |     FileSys::VirtualFile update_raw; | 
					
						
							|  |  |  |     loader.ReadUpdateRaw(update_raw); | 
					
						
							| 
									
										
										
										
											2024-01-19 16:37:34 -05:00
										 |  |  |     for (const auto& patch : patch_manager.GetPatches(update_raw)) { | 
					
						
							|  |  |  |         const bool is_update = patch.name == "Update"; | 
					
						
							| 
									
										
										
										
											2018-10-24 11:25:55 -04:00
										 |  |  |         if (!updatable && is_update) { | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2018-10-24 11:25:55 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-19 16:37:34 -05:00
										 |  |  |         const QString type = | 
					
						
							|  |  |  |             QString::fromStdString(patch.enabled ? patch.name : "[D] " + patch.name); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-19 16:37:34 -05:00
										 |  |  |         if (patch.version.empty()) { | 
					
						
							| 
									
										
										
										
											2018-10-24 11:25:55 -04:00
										 |  |  |             out.append(QStringLiteral("%1\n").arg(type)); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2024-01-19 16:37:34 -05:00
										 |  |  |             auto ver = patch.version; | 
					
						
							| 
									
										
										
										
											2018-09-25 14:07:13 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Display container name for packed updates
 | 
					
						
							| 
									
										
										
										
											2018-10-24 11:25:55 -04:00
										 |  |  |             if (is_update && ver == "PACKED") { | 
					
						
							| 
									
										
										
										
											2018-09-25 14:07:13 -04:00
										 |  |  |                 ver = Loader::GetFileTypeString(loader.GetFileType()); | 
					
						
							| 
									
										
										
										
											2018-10-24 11:25:55 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-09-25 14:07:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-24 11:25:55 -04:00
										 |  |  |             out.append(QStringLiteral("%1 (%2)\n").arg(type, QString::fromStdString(ver))); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out.chop(1); | 
					
						
							|  |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  | QList<QStandardItem*> MakeGameListEntry(const std::string& path, | 
					
						
							|  |  |  |                                         const std::string& name, | 
					
						
							|  |  |  |                                         const std::size_t size, | 
					
						
							|  |  |  |                                         const std::vector<u8>& icon, | 
					
						
							|  |  |  |                                         Loader::AppLoader& loader, | 
					
						
							|  |  |  |                                         u64 program_id, | 
					
						
							| 
									
										
										
										
											2023-08-24 01:16:19 +02:00
										 |  |  |                                         const CompatibilityList& compatibility_list, | 
					
						
							| 
									
										
										
										
											2023-08-27 18:41:42 -04:00
										 |  |  |                                         const PlayTime::PlayTimeManager& play_time_manager, | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |                                         const FileSys::PatchManager& patch, | 
					
						
							|  |  |  |                                         const bool cached) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  |     const auto it = FindMatchingCompatibilityEntry(compatibility_list, program_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // The game list uses this as compatibility number for untested games
 | 
					
						
							| 
									
										
										
										
											2019-05-20 15:07:56 -04:00
										 |  |  |     QString compatibility{QStringLiteral("99")}; | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  |     if (it != compatibility_list.end()) { | 
					
						
							|  |  |  |         compatibility = it->second.first; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:47:03 -05:00
										 |  |  |     const auto file_type = loader.GetFileType(); | 
					
						
							|  |  |  |     const auto file_type_string = QString::fromStdString(Loader::GetFileTypeString(file_type)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  |     QList<QStandardItem*> list{ | 
					
						
							| 
									
										
										
										
											2018-12-05 17:47:03 -05:00
										 |  |  |         new GameListItemPath(FormatGameName(path), icon, QString::fromStdString(name), | 
					
						
							|  |  |  |                              file_type_string, program_id), | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  |         new GameListItemCompat(compatibility), | 
					
						
							| 
									
										
										
										
											2018-12-05 17:47:03 -05:00
										 |  |  |         new GameListItem(file_type_string), | 
					
						
							| 
									
										
										
										
											2023-08-24 01:16:19 +02:00
										 |  |  |         new GameListItemSize(size), | 
					
						
							| 
									
										
										
										
											2023-08-27 18:41:42 -04:00
										 |  |  |         new GameListItemPlayTime(play_time_manager.GetPlayTime(program_id)), | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |     QString patch_versions; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cached) { | 
					
						
							|  |  |  |         patch_versions = GetGameListCachedObject( | 
					
						
							|  |  |  |             fmt::format("{:016X}", patch.GetTitleID()), "pv.txt", [&patch, &loader] { | 
					
						
							|  |  |  |                 return FormatPatchNameVersions(patch, loader, loader.IsRomFSUpdatable()); | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         patch_versions = FormatPatchNameVersions(patch, loader, loader.IsRomFSUpdatable()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-02 06:27:08 +01:00
										 |  |  |     list.insert(2, new GameListItem(patch_versions)); | 
					
						
							| 
									
										
										
										
											2018-12-01 23:23:02 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return list; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-13 16:52:19 -04:00
										 |  |  | GameListWorker::GameListWorker(FileSys::VirtualFilesystem vfs_, | 
					
						
							|  |  |  |                                FileSys::ManualContentProvider* provider_, | 
					
						
							|  |  |  |                                QVector<UISettings::GameDir>& game_dirs_, | 
					
						
							| 
									
										
										
										
											2023-08-27 18:41:42 -04:00
										 |  |  |                                const CompatibilityList& compatibility_list_, | 
					
						
							|  |  |  |                                const PlayTime::PlayTimeManager& play_time_manager_, | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |                                Core::System& system_, | 
					
						
							|  |  |  |                                const bool cached_) | 
					
						
							|  |  |  |     : vfs{std::move(vfs_)} | 
					
						
							|  |  |  |     , provider{provider_} | 
					
						
							|  |  |  |     , game_dirs{game_dirs_} | 
					
						
							|  |  |  |     , compatibility_list{compatibility_list_} | 
					
						
							|  |  |  |     , play_time_manager{play_time_manager_} | 
					
						
							|  |  |  |     , system{system_} | 
					
						
							|  |  |  |     , cached{cached_} | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |     // We want the game list to manage our lifetime.
 | 
					
						
							|  |  |  |     setAutoDelete(false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GameListWorker::~GameListWorker() { | 
					
						
							|  |  |  |     this->disconnect(); | 
					
						
							|  |  |  |     stop_requested.store(true); | 
					
						
							|  |  |  |     processing_completed.Wait(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GameListWorker::ProcessEvents(GameList* game_list) { | 
					
						
							|  |  |  |     while (true) { | 
					
						
							|  |  |  |         std::function<void(GameList*)> func; | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Lock queue to protect concurrent modification.
 | 
					
						
							|  |  |  |             std::scoped_lock lk(lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // If we can't pop a function, return.
 | 
					
						
							|  |  |  |             if (queued_events.empty()) { | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Pop a function.
 | 
					
						
							|  |  |  |             func = std::move(queued_events.back()); | 
					
						
							|  |  |  |             queued_events.pop_back(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Run the function.
 | 
					
						
							|  |  |  |         func(game_list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename F> | 
					
						
							|  |  |  | void GameListWorker::RecordEvent(F&& func) { | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Lock queue to protect concurrent modification.
 | 
					
						
							|  |  |  |         std::scoped_lock lk(lock); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |         // Add the function into the front of the queue.
 | 
					
						
							|  |  |  |         queued_events.emplace_front(std::move(func)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Data now available.
 | 
					
						
							|  |  |  |     emit DataAvailable(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  | void GameListWorker::AddTitlesToGameList(GameListDir* parent_dir) { | 
					
						
							|  |  |  |     using namespace FileSys; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-21 19:22:14 -04:00
										 |  |  |     const auto& cache = system.GetContentProviderUnion(); | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |     auto installed_games = cache.ListEntriesFilterOrigin(std::nullopt, TitleType::Application, | 
					
						
							|  |  |  |                                                          ContentRecordType::Program); | 
					
						
							| 
									
										
										
										
											2019-05-05 03:07:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parent_dir->type() == static_cast<int>(GameListItemType::SdmcDir)) { | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |         installed_games = cache.ListEntriesFilterOrigin( | 
					
						
							|  |  |  |             ContentProviderUnionSlot::SDMC, TitleType::Application, ContentRecordType::Program); | 
					
						
							| 
									
										
										
										
											2019-05-05 03:07:09 +02:00
										 |  |  |     } else if (parent_dir->type() == static_cast<int>(GameListItemType::UserNandDir)) { | 
					
						
							|  |  |  |         installed_games = cache.ListEntriesFilterOrigin( | 
					
						
							|  |  |  |             ContentProviderUnionSlot::UserNAND, TitleType::Application, ContentRecordType::Program); | 
					
						
							|  |  |  |     } else if (parent_dir->type() == static_cast<int>(GameListItemType::SysNandDir)) { | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |         installed_games = cache.ListEntriesFilterOrigin( | 
					
						
							|  |  |  |             ContentProviderUnionSlot::SysNAND, TitleType::Application, ContentRecordType::Program); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  |     for (const auto& [slot, game] : installed_games) { | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         if (slot == ContentProviderUnionSlot::FrontendManual) { | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const auto file = cache.GetEntryUnparsed(game.title_id, game.type); | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         std::unique_ptr<Loader::AppLoader> loader = Loader::GetLoader(system, file); | 
					
						
							|  |  |  |         if (!loader) { | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         std::vector<u8> icon; | 
					
						
							|  |  |  |         std::string name; | 
					
						
							|  |  |  |         u64 program_id = 0; | 
					
						
							| 
									
										
										
										
											2023-07-24 08:05:47 -06:00
										 |  |  |         const auto result = loader->ReadProgramId(program_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (result != Loader::ResultStatus::Success) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         const PatchManager patch{program_id, system.GetFileSystemController(), | 
					
						
							|  |  |  |                                  system.GetContentProvider()}; | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |         LOG_INFO(Frontend, "PatchManager initiated for id {:X}", program_id); | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |         const auto control = cache.GetEntry(game.title_id, ContentRecordType::Control); | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         if (control != nullptr) { | 
					
						
							| 
									
										
										
										
											2018-10-09 14:22:31 -04:00
										 |  |  |             GetMetadataFromControlNCA(patch, *control, icon, name); | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |         auto entry = MakeGameListEntry(file->GetFullPath(), | 
					
						
							|  |  |  |                                        name, | 
					
						
							|  |  |  |                                        file->GetSize(), | 
					
						
							|  |  |  |                                        icon, | 
					
						
							|  |  |  |                                        *loader, | 
					
						
							|  |  |  |                                        program_id, | 
					
						
							|  |  |  |                                        compatibility_list, | 
					
						
							|  |  |  |                                        play_time_manager, | 
					
						
							|  |  |  |                                        patch, | 
					
						
							|  |  |  |                                        cached); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |         RecordEvent([=](GameList* game_list) { game_list->AddEntry(entry, parent_dir); }); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | void GameListWorker::ScanFileSystem(ScanTarget target, const std::string& dir_path, bool deep_scan, | 
					
						
							|  |  |  |                                     GameListDir* parent_dir) { | 
					
						
							| 
									
										
										
										
											2021-07-30 09:32:26 -04:00
										 |  |  |     const auto callback = [this, target, parent_dir](const std::filesystem::path& path) -> bool { | 
					
						
							| 
									
										
										
										
											2023-10-12 21:07:49 -04:00
										 |  |  |         if (stop_requested) { | 
					
						
							| 
									
										
										
										
											2018-12-04 18:39:32 -05:00
										 |  |  |             // Breaks the callback loop.
 | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |         const auto physical_name = Common::FS::PathToUTF8String(path); | 
					
						
							|  |  |  |         const auto is_dir = Common::FS::IsDir(path); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |         if (!is_dir && | 
					
						
							|  |  |  |             (HasSupportedFileExtension(physical_name) || IsExtractedNCAMain(physical_name))) { | 
					
						
							| 
									
										
										
										
											2024-01-18 21:31:41 +01:00
										 |  |  |             const auto file = vfs->OpenFile(physical_name, FileSys::OpenMode::Read); | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |             if (!file) { | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 07:53:10 -05:00
										 |  |  |             auto loader = Loader::GetLoader(system, file); | 
					
						
							| 
									
										
										
										
											2018-12-05 17:58:11 -05:00
										 |  |  |             if (!loader) { | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |                 return true; | 
					
						
							| 
									
										
										
										
											2018-12-05 17:58:11 -05:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const auto file_type = loader->GetFileType(); | 
					
						
							| 
									
										
										
										
											2020-02-09 11:30:02 -05:00
										 |  |  |             if (file_type == Loader::FileType::Unknown || file_type == Loader::FileType::Error) { | 
					
						
							| 
									
										
										
										
											2018-12-05 17:58:11 -05:00
										 |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |             u64 program_id = 0; | 
					
						
							|  |  |  |             const auto res2 = loader->ReadProgramId(program_id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  |             if (target == ScanTarget::FillManualContentProvider) { | 
					
						
							|  |  |  |                 if (res2 == Loader::ResultStatus::Success && file_type == Loader::FileType::NCA) { | 
					
						
							|  |  |  |                     provider->AddEntry(FileSys::TitleType::Application, | 
					
						
							|  |  |  |                                        FileSys::GetCRTypeFromNCAType(FileSys::NCA{file}.GetType()), | 
					
						
							|  |  |  |                                        program_id, file); | 
					
						
							|  |  |  |                 } else if (res2 == Loader::ResultStatus::Success && | 
					
						
							|  |  |  |                            (file_type == Loader::FileType::XCI || | 
					
						
							|  |  |  |                             file_type == Loader::FileType::NSP)) { | 
					
						
							|  |  |  |                     const auto nsp = file_type == Loader::FileType::NSP | 
					
						
							|  |  |  |                                          ? std::make_shared<FileSys::NSP>(file) | 
					
						
							|  |  |  |                                          : FileSys::XCI{file}.GetSecurePartitionNSP(); | 
					
						
							|  |  |  |                     for (const auto& title : nsp->GetNCAs()) { | 
					
						
							|  |  |  |                         for (const auto& entry : title.second) { | 
					
						
							|  |  |  |                             provider->AddEntry(entry.first.first, entry.first.second, title.first, | 
					
						
							|  |  |  |                                                entry.second->GetBaseFile()); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2021-07-20 13:10:05 +08:00
										 |  |  |                 std::vector<u64> program_ids; | 
					
						
							|  |  |  |                 loader->ReadProgramIds(program_ids); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (res2 == Loader::ResultStatus::Success && program_ids.size() > 1 && | 
					
						
							|  |  |  |                     (file_type == Loader::FileType::XCI || file_type == Loader::FileType::NSP)) { | 
					
						
							|  |  |  |                     for (const auto id : program_ids) { | 
					
						
							|  |  |  |                         loader = Loader::GetLoader(system, file, id); | 
					
						
							|  |  |  |                         if (!loader) { | 
					
						
							|  |  |  |                             continue; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         std::vector<u8> icon; | 
					
						
							|  |  |  |                         [[maybe_unused]] const auto res1 = loader->ReadIcon(icon); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 13:10:05 +08:00
										 |  |  |                         std::string name = " "; | 
					
						
							|  |  |  |                         [[maybe_unused]] const auto res3 = loader->ReadTitle(name); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 13:10:05 +08:00
										 |  |  |                         const FileSys::PatchManager patch{id, system.GetFileSystemController(), | 
					
						
							|  |  |  |                                                           system.GetContentProvider()}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |                         auto entry = MakeGameListEntry(physical_name, | 
					
						
							|  |  |  |                                                        name, | 
					
						
							|  |  |  |                                                        Common::FS::GetSize(physical_name), | 
					
						
							|  |  |  |                                                        icon, | 
					
						
							|  |  |  |                                                        *loader, | 
					
						
							|  |  |  |                                                        id, | 
					
						
							|  |  |  |                                                        compatibility_list, | 
					
						
							|  |  |  |                                                        play_time_manager, | 
					
						
							|  |  |  |                                                        patch, | 
					
						
							|  |  |  |                                                        cached); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         RecordEvent( | 
					
						
							|  |  |  |                             [=](GameList* game_list) { game_list->AddEntry(entry, parent_dir); }); | 
					
						
							| 
									
										
										
										
											2021-07-20 13:10:05 +08:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     std::vector<u8> icon; | 
					
						
							|  |  |  |                     [[maybe_unused]] const auto res1 = loader->ReadIcon(icon); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 13:10:05 +08:00
										 |  |  |                     std::string name = " "; | 
					
						
							|  |  |  |                     [[maybe_unused]] const auto res3 = loader->ReadTitle(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     const FileSys::PatchManager patch{program_id, system.GetFileSystemController(), | 
					
						
							|  |  |  |                                                       system.GetContentProvider()}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-29 08:20:11 +00:00
										 |  |  |                     auto entry = MakeGameListEntry(physical_name, | 
					
						
							|  |  |  |                                                    name, | 
					
						
							|  |  |  |                                                    Common::FS::GetSize(physical_name), | 
					
						
							|  |  |  |                                                    icon, | 
					
						
							|  |  |  |                                                    *loader, | 
					
						
							|  |  |  |                                                    program_id, | 
					
						
							|  |  |  |                                                    compatibility_list, | 
					
						
							|  |  |  |                                                    play_time_manager, | 
					
						
							|  |  |  |                                                    patch, | 
					
						
							|  |  |  |                                                    cached); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     RecordEvent( | 
					
						
							|  |  |  |                         [=](GameList* game_list) { game_list->AddEntry(entry, parent_dir); }); | 
					
						
							| 
									
										
										
										
											2021-07-20 13:10:05 +08:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-03-04 12:40:53 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |         } else if (is_dir) { | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  |             watch_list.append(QString::fromStdString(physical_name)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  |     if (deep_scan) { | 
					
						
							|  |  |  |         Common::FS::IterateDirEntriesRecursively(dir_path, callback, | 
					
						
							|  |  |  |                                                  Common::FS::DirEntryFilter::All); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         Common::FS::IterateDirEntries(dir_path, callback, Common::FS::DirEntryFilter::File); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GameListWorker::run() { | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |     watch_list.clear(); | 
					
						
							| 
									
										
										
										
											2020-07-29 07:51:42 -04:00
										 |  |  |     provider->ClearAllEntries(); | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |     const auto DirEntryReady = [&](GameListDir* game_list_dir) { | 
					
						
							|  |  |  |         RecordEvent([=](GameList* game_list) { game_list->AddDirEntry(game_list_dir); }); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |     for (UISettings::GameDir& game_dir : game_dirs) { | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |         if (stop_requested) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-12 02:03:01 -05:00
										 |  |  |         if (game_dir.path == std::string("SDMC")) { | 
					
						
							| 
									
										
										
										
											2019-05-05 03:07:09 +02:00
										 |  |  |             auto* const game_list_dir = new GameListDir(game_dir, GameListItemType::SdmcDir); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |             DirEntryReady(game_list_dir); | 
					
						
							| 
									
										
										
										
											2019-05-05 03:07:09 +02:00
										 |  |  |             AddTitlesToGameList(game_list_dir); | 
					
						
							| 
									
										
										
										
											2023-11-12 02:03:01 -05:00
										 |  |  |         } else if (game_dir.path == std::string("UserNAND")) { | 
					
						
							| 
									
										
										
										
											2019-05-05 03:07:09 +02:00
										 |  |  |             auto* const game_list_dir = new GameListDir(game_dir, GameListItemType::UserNandDir); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |             DirEntryReady(game_list_dir); | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |             AddTitlesToGameList(game_list_dir); | 
					
						
							| 
									
										
										
										
											2023-11-12 02:03:01 -05:00
										 |  |  |         } else if (game_dir.path == std::string("SysNAND")) { | 
					
						
							| 
									
										
										
										
											2019-05-05 03:07:09 +02:00
										 |  |  |             auto* const game_list_dir = new GameListDir(game_dir, GameListItemType::SysNandDir); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |             DirEntryReady(game_list_dir); | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |             AddTitlesToGameList(game_list_dir); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2023-11-12 02:03:01 -05:00
										 |  |  |             watch_list.append(QString::fromStdString(game_dir.path)); | 
					
						
							| 
									
										
										
										
											2019-05-03 19:21:57 +02:00
										 |  |  |             auto* const game_list_dir = new GameListDir(game_dir); | 
					
						
							| 
									
										
										
										
											2023-10-23 22:09:29 -04:00
										 |  |  |             DirEntryReady(game_list_dir); | 
					
						
							| 
									
										
										
										
											2023-11-12 02:03:01 -05:00
										 |  |  |             ScanFileSystem(ScanTarget::FillManualContentProvider, game_dir.path, game_dir.deep_scan, | 
					
						
							|  |  |  |                            game_list_dir); | 
					
						
							|  |  |  |             ScanFileSystem(ScanTarget::PopulateGameList, game_dir.path, game_dir.deep_scan, | 
					
						
							|  |  |  |                            game_list_dir); | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-08-15 11:14:09 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-01 23:21:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 22:57:47 +00:00
										 |  |  |     RecordEvent([this](GameList* game_list) { game_list->DonePopulating(watch_list); }); | 
					
						
							| 
									
										
										
										
											2023-10-12 21:07:49 -04:00
										 |  |  |     processing_completed.Set(); | 
					
						
							| 
									
										
										
										
											2018-09-07 16:08:08 -04:00
										 |  |  | } |