Merge pull request #1618 from DarkLordZach/dump-nso
patch_manager: Add support for dumping uncompressed NSOs
This commit is contained in:
		| @@ -8,8 +8,9 @@ | ||||
|  | ||||
| namespace FileSys { | ||||
|  | ||||
| BISFactory::BISFactory(VirtualDir nand_root_, VirtualDir load_root_) | ||||
| BISFactory::BISFactory(VirtualDir nand_root_, VirtualDir load_root_, VirtualDir dump_root_) | ||||
|     : nand_root(std::move(nand_root_)), load_root(std::move(load_root_)), | ||||
|       dump_root(std::move(dump_root_)), | ||||
|       sysnand_cache(std::make_unique<RegisteredCache>( | ||||
|           GetOrCreateDirectoryRelative(nand_root, "/system/Contents/registered"))), | ||||
|       usrnand_cache(std::make_unique<RegisteredCache>( | ||||
| @@ -32,4 +33,10 @@ VirtualDir BISFactory::GetModificationLoadRoot(u64 title_id) const { | ||||
|     return GetOrCreateDirectoryRelative(load_root, fmt::format("/{:016X}", title_id)); | ||||
| } | ||||
|  | ||||
| VirtualDir BISFactory::GetModificationDumpRoot(u64 title_id) const { | ||||
|     if (title_id == 0) | ||||
|         return nullptr; | ||||
|     return GetOrCreateDirectoryRelative(dump_root, fmt::format("/{:016X}", title_id)); | ||||
| } | ||||
|  | ||||
| } // namespace FileSys | ||||
|   | ||||
| @@ -17,17 +17,19 @@ class RegisteredCache; | ||||
| /// registered caches. | ||||
| class BISFactory { | ||||
| public: | ||||
|     explicit BISFactory(VirtualDir nand_root, VirtualDir load_root); | ||||
|     explicit BISFactory(VirtualDir nand_root, VirtualDir load_root, VirtualDir dump_root); | ||||
|     ~BISFactory(); | ||||
|  | ||||
|     RegisteredCache* GetSystemNANDContents() const; | ||||
|     RegisteredCache* GetUserNANDContents() const; | ||||
|  | ||||
|     VirtualDir GetModificationLoadRoot(u64 title_id) const; | ||||
|     VirtualDir GetModificationDumpRoot(u64 title_id) const; | ||||
|  | ||||
| private: | ||||
|     VirtualDir nand_root; | ||||
|     VirtualDir load_root; | ||||
|     VirtualDir dump_root; | ||||
|  | ||||
|     std::unique_ptr<RegisteredCache> sysnand_cache; | ||||
|     std::unique_ptr<RegisteredCache> usrnand_cache; | ||||
|   | ||||
| @@ -19,6 +19,7 @@ | ||||
| #include "core/file_sys/vfs_vector.h" | ||||
| #include "core/hle/service/filesystem/filesystem.h" | ||||
| #include "core/loader/loader.h" | ||||
| #include "core/settings.h" | ||||
|  | ||||
| namespace FileSys { | ||||
|  | ||||
| @@ -119,6 +120,18 @@ std::vector<u8> PatchManager::PatchNSO(const std::vector<u8>& nso) const { | ||||
|     const auto build_id_raw = Common::HexArrayToString(header.build_id); | ||||
|     const auto build_id = build_id_raw.substr(0, build_id_raw.find_last_not_of('0') + 1); | ||||
|  | ||||
|     if (Settings::values.dump_nso) { | ||||
|         LOG_INFO(Loader, "Dumping NSO for build_id={}, title_id={:016X}", build_id, title_id); | ||||
|         const auto dump_dir = Service::FileSystem::GetModificationDumpRoot(title_id); | ||||
|         if (dump_dir != nullptr) { | ||||
|             const auto nso_dir = GetOrCreateDirectoryRelative(dump_dir, "/nso"); | ||||
|             const auto file = nso_dir->CreateFile(fmt::format("{}.nso", build_id)); | ||||
|  | ||||
|             file->Resize(nso.size()); | ||||
|             file->WriteBytes(nso); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     LOG_INFO(Loader, "Patching NSO for build_id={}", build_id); | ||||
|  | ||||
|     const auto load_dir = Service::FileSystem::GetModificationLoadRoot(title_id); | ||||
|   | ||||
| @@ -370,6 +370,15 @@ FileSys::VirtualDir GetModificationLoadRoot(u64 title_id) { | ||||
|     return bis_factory->GetModificationLoadRoot(title_id); | ||||
| } | ||||
|  | ||||
| FileSys::VirtualDir GetModificationDumpRoot(u64 title_id) { | ||||
|     LOG_TRACE(Service_FS, "Opening mod dump root for tid={:016X}", title_id); | ||||
|  | ||||
|     if (bis_factory == nullptr) | ||||
|         return nullptr; | ||||
|  | ||||
|     return bis_factory->GetModificationDumpRoot(title_id); | ||||
| } | ||||
|  | ||||
| void CreateFactories(FileSys::VfsFilesystem& vfs, bool overwrite) { | ||||
|     if (overwrite) { | ||||
|         bis_factory = nullptr; | ||||
| @@ -383,14 +392,22 @@ void CreateFactories(FileSys::VfsFilesystem& vfs, bool overwrite) { | ||||
|                                           FileSys::Mode::ReadWrite); | ||||
|     auto load_directory = vfs.OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::LoadDir), | ||||
|                                             FileSys::Mode::ReadWrite); | ||||
|     auto dump_directory = vfs.OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::DumpDir), | ||||
|                                             FileSys::Mode::ReadWrite); | ||||
|  | ||||
|     if (bis_factory == nullptr) | ||||
|         bis_factory = std::make_unique<FileSys::BISFactory>(nand_directory, load_directory); | ||||
|     if (save_data_factory == nullptr) | ||||
|     if (bis_factory == nullptr) { | ||||
|         bis_factory = | ||||
|             std::make_unique<FileSys::BISFactory>(nand_directory, load_directory, dump_directory); | ||||
|     } | ||||
|  | ||||
|     if (save_data_factory == nullptr) { | ||||
|         save_data_factory = std::make_unique<FileSys::SaveDataFactory>(std::move(nand_directory)); | ||||
|     if (sdmc_factory == nullptr) | ||||
|     } | ||||
|  | ||||
|     if (sdmc_factory == nullptr) { | ||||
|         sdmc_factory = std::make_unique<FileSys::SDMCFactory>(std::move(sd_directory)); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void InstallInterfaces(SM::ServiceManager& service_manager, FileSys::VfsFilesystem& vfs) { | ||||
|     romfs_factory = nullptr; | ||||
|   | ||||
| @@ -55,6 +55,7 @@ FileSys::RegisteredCache* GetUserNANDContents(); | ||||
| FileSys::RegisteredCache* GetSDMCContents(); | ||||
|  | ||||
| FileSys::VirtualDir GetModificationLoadRoot(u64 title_id); | ||||
| FileSys::VirtualDir GetModificationDumpRoot(u64 title_id); | ||||
|  | ||||
| // Creates the SaveData, SDMC, and BIS Factories. Should be called once and before any function | ||||
| // above is called. | ||||
|   | ||||
| @@ -154,7 +154,7 @@ std::optional<VAddr> AppLoader_NSO::LoadModule(const FileSys::VfsFile& file, VAd | ||||
|     program_image.resize(image_size); | ||||
|  | ||||
|     // Apply patches if necessary | ||||
|     if (pm && pm->HasNSOPatch(nso_header.build_id)) { | ||||
|     if (pm && (pm->HasNSOPatch(nso_header.build_id) || Settings::values.dump_nso)) { | ||||
|         std::vector<u8> pi_header(program_image.size() + 0x100); | ||||
|         std::memcpy(pi_header.data(), &nso_header, sizeof(NsoHeader)); | ||||
|         std::memcpy(pi_header.data() + 0x100, program_image.data(), program_image.size()); | ||||
|   | ||||
| @@ -159,6 +159,7 @@ struct Values { | ||||
|     bool use_gdbstub; | ||||
|     u16 gdbstub_port; | ||||
|     std::string program_args; | ||||
|     bool dump_nso; | ||||
|  | ||||
|     // WebService | ||||
|     bool enable_telemetry; | ||||
|   | ||||
| @@ -153,6 +153,7 @@ void Config::ReadValues() { | ||||
|     Settings::values.use_gdbstub = qt_config->value("use_gdbstub", false).toBool(); | ||||
|     Settings::values.gdbstub_port = qt_config->value("gdbstub_port", 24689).toInt(); | ||||
|     Settings::values.program_args = qt_config->value("program_args", "").toString().toStdString(); | ||||
|     Settings::values.dump_nso = qt_config->value("dump_nso", false).toBool(); | ||||
|     qt_config->endGroup(); | ||||
|  | ||||
|     qt_config->beginGroup("WebService"); | ||||
| @@ -295,6 +296,7 @@ void Config::SaveValues() { | ||||
|     qt_config->setValue("use_gdbstub", Settings::values.use_gdbstub); | ||||
|     qt_config->setValue("gdbstub_port", Settings::values.gdbstub_port); | ||||
|     qt_config->setValue("program_args", QString::fromStdString(Settings::values.program_args)); | ||||
|     qt_config->setValue("dump_nso", Settings::values.dump_nso); | ||||
|     qt_config->endGroup(); | ||||
|  | ||||
|     qt_config->beginGroup("WebService"); | ||||
|   | ||||
| @@ -34,6 +34,7 @@ void ConfigureDebug::setConfiguration() { | ||||
|     ui->toggle_console->setChecked(UISettings::values.show_console); | ||||
|     ui->log_filter_edit->setText(QString::fromStdString(Settings::values.log_filter)); | ||||
|     ui->homebrew_args_edit->setText(QString::fromStdString(Settings::values.program_args)); | ||||
|     ui->dump_decompressed_nso->setChecked(Settings::values.dump_nso); | ||||
| } | ||||
|  | ||||
| void ConfigureDebug::applyConfiguration() { | ||||
| @@ -42,6 +43,7 @@ void ConfigureDebug::applyConfiguration() { | ||||
|     UISettings::values.show_console = ui->toggle_console->isChecked(); | ||||
|     Settings::values.log_filter = ui->log_filter_edit->text().toStdString(); | ||||
|     Settings::values.program_args = ui->homebrew_args_edit->text().toStdString(); | ||||
|     Settings::values.dump_nso = ui->dump_decompressed_nso->isChecked(); | ||||
|     Debugger::ToggleConsole(); | ||||
|     Log::Filter filter; | ||||
|     filter.ParseFilterString(Settings::values.log_filter); | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
|     <x>0</x> | ||||
|     <y>0</y> | ||||
|     <width>400</width> | ||||
|     <height>300</height> | ||||
|     <height>357</height> | ||||
|    </rect> | ||||
|   </property> | ||||
|   <property name="windowTitle"> | ||||
| @@ -129,6 +129,25 @@ | ||||
|      </layout> | ||||
|     </widget> | ||||
|    </item> | ||||
|    <item> | ||||
|     <widget class="QGroupBox" name="groupBox_4"> | ||||
|      <property name="title"> | ||||
|       <string>Dump</string> | ||||
|      </property> | ||||
|      <layout class="QVBoxLayout" name="verticalLayout_4"> | ||||
|       <item> | ||||
|        <widget class="QCheckBox" name="dump_decompressed_nso"> | ||||
|         <property name="whatsThis"> | ||||
|          <string>When checked, any NSO yuzu tries to load or patch will be copied decompressed to the yuzu/dump directory.</string> | ||||
|         </property> | ||||
|         <property name="text"> | ||||
|          <string>Dump Decompressed NSOs</string> | ||||
|         </property> | ||||
|        </widget> | ||||
|       </item> | ||||
|      </layout> | ||||
|     </widget> | ||||
|    </item> | ||||
|    <item> | ||||
|     <spacer name="verticalSpacer"> | ||||
|      <property name="orientation"> | ||||
|   | ||||
| @@ -148,6 +148,7 @@ void Config::ReadValues() { | ||||
|     Settings::values.gdbstub_port = | ||||
|         static_cast<u16>(sdl2_config->GetInteger("Debugging", "gdbstub_port", 24689)); | ||||
|     Settings::values.program_args = sdl2_config->Get("Debugging", "program_args", ""); | ||||
|     Settings::values.dump_nso = sdl2_config->GetBoolean("Debugging", "dump_nso", false); | ||||
|  | ||||
|     // Web Service | ||||
|     Settings::values.enable_telemetry = | ||||
|   | ||||
| @@ -206,6 +206,8 @@ log_filter = *:Trace | ||||
| # Port for listening to GDB connections. | ||||
| use_gdbstub=false | ||||
| gdbstub_port=24689 | ||||
| # Determines whether or not yuzu will dump all NSOs it attempts to load while loading them | ||||
| dump_nso=false | ||||
|  | ||||
| [WebService] | ||||
| # Whether or not to enable telemetry | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 bunnei
					bunnei