#include #include "content/providers/gdb_provider.hpp" #include "content/providers/file_provider.hpp" #include "content/providers/null_provider.hpp" #include "content/providers/disk_provider.hpp" #include "content/providers/intel_hex_provider.hpp" #include "content/providers/motorola_srec_provider.hpp" #include "content/providers/memory_file_provider.hpp" #include "content/providers/view_provider.hpp" #include #include "content/popups/popup_notification.hpp" #include "content/helpers/notification.hpp" #include #include #include #include #include namespace hex::plugin::builtin { void registerProviders() { ContentRegistry::Provider::add(false); ContentRegistry::Provider::add(false); #if !defined(OS_WEB) ContentRegistry::Provider::add(); #endif ContentRegistry::Provider::add(); ContentRegistry::Provider::add(); ContentRegistry::Provider::add(); ContentRegistry::Provider::add(false); ContentRegistry::Provider::add(false); #if defined(OS_WINDOWS) ||defined (OS_LINUX) ContentRegistry::Provider::add(); #endif ProjectFile::registerHandler({ .basePath = "providers", .required = true, .load = [](const std::fs::path &basePath, Tar &tar) { auto json = nlohmann::json::parse(tar.readString(basePath / "providers.json")); auto providerIds = json.at("providers").get>(); bool success = true; std::map providerWarnings; for (const auto &id : providerIds) { auto providerSettings = nlohmann::json::parse(tar.readString(basePath / hex::format("{}.json", id))); auto providerType = providerSettings.at("type").get(); auto newProvider = ImHexApi::Provider::createProvider(providerType, true, false); ON_SCOPE_EXIT { if (!success) { for (auto &task : TaskManager::getRunningTasks()) task->interrupt(); TaskManager::runWhenTasksFinished([]{ for (auto provider : ImHexApi::Provider::getProviders()) ImHexApi::Provider::remove(provider, true); }); } }; if (newProvider == nullptr) { // If a provider is not created, it will be overwritten when saving the project, // so we should prevent the project from loading at all showError(hex::format("hex.builtin.popup.error.project.load"_lang, hex::format("hex.builtin.popup.error.project.load.create_provider"_lang, providerType) )); success = false; break; } newProvider->setID(id); bool loaded = false; try { newProvider->loadSettings(providerSettings.at("settings")); loaded = true; } catch (const std::exception &e){ providerWarnings[newProvider] = e.what(); } if (loaded) { if (!newProvider->open() || !newProvider->isAvailable() || !newProvider->isReadable()) { providerWarnings[newProvider] = newProvider->getErrorMessage(); } else EventProviderOpened::post(newProvider); } } std::string warningMsg; for (const auto &warning : providerWarnings){ ImHexApi::Provider::remove(warning.first); warningMsg.append( hex::format("\n - {} : {}", warning.first->getName(), warning.second)); } // If no providers were opened, display an error with // the warnings that happened when opening them if (ImHexApi::Provider::getProviders().size() == 0) { showError(hex::format("hex.builtin.popup.error.project.load"_lang, hex::format("hex.builtin.popup.error.project.load.no_providers"_lang)) + warningMsg); return false; } else { // Else, if are warnings, still display them if (warningMsg.empty()) return true; else { showWarning( hex::format("hex.builtin.popup.error.project.load.some_providers_failed"_lang, warningMsg)); } return success; } }, .store = [](const std::fs::path &basePath, Tar &tar) { std::vector providerIds; for (const auto &provider : ImHexApi::Provider::getProviders()) { auto id = provider->getID(); providerIds.push_back(id); nlohmann::json json; json["type"] = provider->getTypeName(); json["settings"] = provider->storeSettings({}); tar.writeString(basePath / hex::format("{}.json", id), json.dump(4)); } tar.writeString(basePath / "providers.json", nlohmann::json({ { "providers", providerIds } }).dump(4) ); return true; } }); } }