forked from eden-emu/eden
This is part of a series of PRs made in preparation for the QML rewrite. this PR specifically moves a bunch of utility functions from main.cpp into qt_common, with the biggest benefit being that QML can reuse the exact same code through ctx passthrough. Also, QtCommon::Frontend is an abstraction layer over several previously Widgets-specific stuff like QMessageBox that gets used everywhere. The idea is that once QML is implemented, these functions can have a Quick version implemented for systems that don't work well with Widgets (sun) or for those on Plasma 6+ (reduces memory usage w/o Widgets linkage) although Quick from C++ is actually anal, but whatever. Other than that this should also just kinda reduce the size of main.cpp which is a 6000-line behemoth rn, and clangd straight up gives up with it for me (likely caused by the massive amount of headers, which this DOES reduce). In the future, I probably want to create a common strings lookup table that both Qt and QML can reference--though I'm not sure how much linguist likes that--which should give us a way to keep language consistent (use frozen-map). TODO: Docs for Qt stuff Co-authored-by: MaranBr <maranbr@outlook.com> Reviewed-on: eden-emu/eden#94 Reviewed-by: MaranBr <maranbr@eden-emu.dev> Reviewed-by: Shinmegumi <shinmegumi@eden-emu.dev>
151 lines
4.8 KiB
C++
151 lines
4.8 KiB
C++
// SPDX-FileCopyrightText: Copyright 2025 Eden Emulator Project
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
#ifndef FIRMWARE_MANAGER_H
|
|
#define FIRMWARE_MANAGER_H
|
|
|
|
#include "common/common_types.h"
|
|
#include "core/core.h"
|
|
#include "core/file_sys/nca_metadata.h"
|
|
#include "core/file_sys/content_archive.h"
|
|
#include "core/file_sys/registered_cache.h"
|
|
#include "core/hle/service/filesystem/filesystem.h"
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <core/hle/service/am/frontend/applet_mii_edit.h>
|
|
#include <string>
|
|
|
|
#include "core/hle/service/set/settings_types.h"
|
|
#include "core/hle/service/set/system_settings_server.h"
|
|
#include "core/hle/result.h"
|
|
|
|
namespace FirmwareManager {
|
|
|
|
static constexpr std::array<const char *, 5> KEY_INSTALL_RESULT_STRINGS = {
|
|
"Decryption Keys were successfully installed",
|
|
"Unable to read key directory, aborting",
|
|
"One or more keys failed to copy.",
|
|
"Verify your keys file has a .keys extension and try again.",
|
|
"Decryption Keys failed to initialize. Check that your dumping tools are up to date and "
|
|
"re-dump keys.",
|
|
};
|
|
|
|
static constexpr std::array<u64, 1> FIRMWARE_REQUIRED_GAMES = {
|
|
0x0100152000022000ULL, // MK8DX
|
|
};
|
|
|
|
enum KeyInstallResult {
|
|
Success,
|
|
InvalidDir,
|
|
ErrorFailedCopy,
|
|
ErrorWrongFilename,
|
|
ErrorFailedInit,
|
|
};
|
|
|
|
/**
|
|
* @brief Installs any arbitrary set of keys for the emulator.
|
|
* @param location Where the keys are located.
|
|
* @param expected_extension What extension the file should have.
|
|
* @return A result code for the operation.
|
|
*/
|
|
KeyInstallResult InstallKeys(std::string location, std::string expected_extension);
|
|
|
|
/**
|
|
* \brief Get a string representation of a result from InstallKeys.
|
|
* \param result The result code.
|
|
* \return A string representation of the passed result code.
|
|
*/
|
|
inline constexpr const char *GetKeyInstallResultString(KeyInstallResult result)
|
|
{
|
|
return KEY_INSTALL_RESULT_STRINGS.at(static_cast<std::size_t>(result));
|
|
}
|
|
|
|
/**
|
|
* \brief Check if the specified program requires firmware to run properly.
|
|
* It is the responsibility of the frontend to properly expose this to the user.
|
|
* \param program_id The program ID to check.
|
|
* \return Whether or not the program requires firmware to run properly.
|
|
*/
|
|
inline constexpr bool GameRequiresFirmware(u64 program_id)
|
|
{
|
|
return std::find(FIRMWARE_REQUIRED_GAMES.begin(), FIRMWARE_REQUIRED_GAMES.end(), program_id)
|
|
!= FIRMWARE_REQUIRED_GAMES.end();
|
|
}
|
|
|
|
|
|
enum FirmwareCheckResult {
|
|
FirmwareGood,
|
|
ErrorFirmwareMissing,
|
|
ErrorFirmwareCorrupted,
|
|
ErrorFirmwareTooNew,
|
|
};
|
|
|
|
static constexpr std::array<const char *, 4> FIRMWARE_CHECK_STRINGS = {
|
|
"",
|
|
"Firmware missing. Firmware is required to run certain games and use the Home Menu. "
|
|
"Eden only works with firmware 19.0.1 and earlier.",
|
|
"Firmware reported as present, but was unable to be read. Check for decryption keys and "
|
|
"redump firmware if necessary.",
|
|
"Firmware is too new or could not be read. Eden only works with firmware 19.0.1 and earlier.",
|
|
};
|
|
|
|
/**
|
|
* \brief Checks for installed firmware within the system.
|
|
* \param system The system to check for firmware.
|
|
* \return Whether or not the system has installed firmware.
|
|
*/
|
|
inline bool CheckFirmwarePresence(Core::System &system)
|
|
{
|
|
constexpr u64 MiiEditId = static_cast<u64>(Service::AM::AppletProgramId::MiiEdit);
|
|
|
|
auto bis_system = system.GetFileSystemController().GetSystemNANDContents();
|
|
if (!bis_system) {
|
|
return false;
|
|
}
|
|
|
|
auto mii_applet_nca = bis_system->GetEntry(MiiEditId, FileSys::ContentRecordType::Program);
|
|
|
|
if (!mii_applet_nca) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* \brief Verifies if firmware is properly installed and is in the correct version range.
|
|
* \param system The system to check firmware on.
|
|
* \return A result code defining the status of the system's firmware.
|
|
*/
|
|
FirmwareCheckResult VerifyFirmware(Core::System &system);
|
|
|
|
/**
|
|
* \brief Get a string representation of a result from CheckFirmwareVersion.
|
|
* \param result The result code.
|
|
* \return A string representation of the passed result code.
|
|
*/
|
|
inline constexpr const char *GetFirmwareCheckString(FirmwareCheckResult result)
|
|
{
|
|
return FIRMWARE_CHECK_STRINGS.at(static_cast<std::size_t>(result));
|
|
}
|
|
|
|
/**
|
|
* @brief Get the currently installed firmware version.
|
|
* @param system The system to check firmware on.
|
|
* @return A pair of the firmware version format and result code.
|
|
*/
|
|
inline std::pair<Service::Set::FirmwareVersionFormat, Result> GetFirmwareVersion(Core::System &system)
|
|
{
|
|
Service::Set::FirmwareVersionFormat firmware_data{};
|
|
const auto result
|
|
= Service::Set::GetFirmwareVersionImpl(firmware_data,
|
|
system,
|
|
Service::Set::GetFirmwareVersionType::Version2);
|
|
|
|
return {firmware_data, result};
|
|
}
|
|
|
|
// TODO(crueter): GET AS STRING
|
|
}
|
|
|
|
#endif
|