forked from suyu/suyu
Merge pull request #2221 from DarkLordZach/firmware-version
set_sys: Implement GetFirmwareVersion(2) for libnx hosversion
This commit is contained in:
commit
1665b70cc6
7 changed files with 154 additions and 3 deletions
|
@ -70,6 +70,8 @@ add_library(core STATIC
|
||||||
file_sys/system_archive/ng_word.h
|
file_sys/system_archive/ng_word.h
|
||||||
file_sys/system_archive/system_archive.cpp
|
file_sys/system_archive/system_archive.cpp
|
||||||
file_sys/system_archive/system_archive.h
|
file_sys/system_archive/system_archive.h
|
||||||
|
file_sys/system_archive/system_version.cpp
|
||||||
|
file_sys/system_archive/system_version.h
|
||||||
file_sys/vfs.cpp
|
file_sys/vfs.cpp
|
||||||
file_sys/vfs.h
|
file_sys/vfs.h
|
||||||
file_sys/vfs_concat.cpp
|
file_sys/vfs_concat.cpp
|
||||||
|
|
|
@ -11,6 +11,9 @@ namespace FileSys {
|
||||||
constexpr ResultCode ERROR_PATH_NOT_FOUND{ErrorModule::FS, 1};
|
constexpr ResultCode ERROR_PATH_NOT_FOUND{ErrorModule::FS, 1};
|
||||||
constexpr ResultCode ERROR_ENTITY_NOT_FOUND{ErrorModule::FS, 1002};
|
constexpr ResultCode ERROR_ENTITY_NOT_FOUND{ErrorModule::FS, 1002};
|
||||||
constexpr ResultCode ERROR_SD_CARD_NOT_FOUND{ErrorModule::FS, 2001};
|
constexpr ResultCode ERROR_SD_CARD_NOT_FOUND{ErrorModule::FS, 2001};
|
||||||
|
constexpr ResultCode ERROR_OUT_OF_BOUNDS{ErrorModule::FS, 3005};
|
||||||
|
constexpr ResultCode ERROR_FAILED_MOUNT_ARCHIVE{ErrorModule::FS, 3223};
|
||||||
|
constexpr ResultCode ERROR_INVALID_ARGUMENT{ErrorModule::FS, 6001};
|
||||||
constexpr ResultCode ERROR_INVALID_OFFSET{ErrorModule::FS, 6061};
|
constexpr ResultCode ERROR_INVALID_OFFSET{ErrorModule::FS, 6061};
|
||||||
constexpr ResultCode ERROR_INVALID_SIZE{ErrorModule::FS, 6062};
|
constexpr ResultCode ERROR_INVALID_SIZE{ErrorModule::FS, 6062};
|
||||||
|
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
#include "core/file_sys/romfs.h"
|
#include "core/file_sys/romfs.h"
|
||||||
#include "core/file_sys/system_archive/ng_word.h"
|
#include "core/file_sys/system_archive/ng_word.h"
|
||||||
#include "core/file_sys/system_archive/system_archive.h"
|
#include "core/file_sys/system_archive/system_archive.h"
|
||||||
|
#include "core/file_sys/system_archive/system_version.h"
|
||||||
|
|
||||||
namespace FileSys::SystemArchive {
|
namespace FileSys::SystemArchive {
|
||||||
|
|
||||||
|
@ -30,7 +31,7 @@ constexpr std::array<SystemArchiveDescriptor, SYSTEM_ARCHIVE_COUNT> SYSTEM_ARCHI
|
||||||
{0x0100000000000806, "NgWord", &NgWord1},
|
{0x0100000000000806, "NgWord", &NgWord1},
|
||||||
{0x0100000000000807, "SsidList", nullptr},
|
{0x0100000000000807, "SsidList", nullptr},
|
||||||
{0x0100000000000808, "Dictionary", nullptr},
|
{0x0100000000000808, "Dictionary", nullptr},
|
||||||
{0x0100000000000809, "SystemVersion", nullptr},
|
{0x0100000000000809, "SystemVersion", &SystemVersion},
|
||||||
{0x010000000000080A, "AvatarImage", nullptr},
|
{0x010000000000080A, "AvatarImage", nullptr},
|
||||||
{0x010000000000080B, "LocalNews", nullptr},
|
{0x010000000000080B, "LocalNews", nullptr},
|
||||||
{0x010000000000080C, "Eula", nullptr},
|
{0x010000000000080C, "Eula", nullptr},
|
||||||
|
|
52
src/core/file_sys/system_archive/system_version.cpp
Normal file
52
src/core/file_sys/system_archive/system_version.cpp
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
// Copyright 2019 yuzu emulator team
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "core/file_sys/system_archive/system_version.h"
|
||||||
|
#include "core/file_sys/vfs_vector.h"
|
||||||
|
|
||||||
|
namespace FileSys::SystemArchive {
|
||||||
|
|
||||||
|
namespace SystemVersionData {
|
||||||
|
|
||||||
|
// This section should reflect the best system version to describe yuzu's HLE api.
|
||||||
|
// TODO(DarkLordZach): Update when HLE gets better.
|
||||||
|
|
||||||
|
constexpr u8 VERSION_MAJOR = 5;
|
||||||
|
constexpr u8 VERSION_MINOR = 1;
|
||||||
|
constexpr u8 VERSION_MICRO = 0;
|
||||||
|
|
||||||
|
constexpr u8 REVISION_MAJOR = 3;
|
||||||
|
constexpr u8 REVISION_MINOR = 0;
|
||||||
|
|
||||||
|
constexpr char PLATFORM_STRING[] = "NX";
|
||||||
|
constexpr char VERSION_HASH[] = "23f9df53e25709d756e0c76effcb2473bd3447dd";
|
||||||
|
constexpr char DISPLAY_VERSION[] = "5.1.0";
|
||||||
|
constexpr char DISPLAY_TITLE[] = "NintendoSDK Firmware for NX 5.1.0-3.0";
|
||||||
|
|
||||||
|
} // namespace SystemVersionData
|
||||||
|
|
||||||
|
std::string GetLongDisplayVersion() {
|
||||||
|
return SystemVersionData::DISPLAY_TITLE;
|
||||||
|
}
|
||||||
|
|
||||||
|
VirtualDir SystemVersion() {
|
||||||
|
VirtualFile file = std::make_shared<VectorVfsFile>(std::vector<u8>(0x100), "file");
|
||||||
|
file->WriteObject(SystemVersionData::VERSION_MAJOR, 0);
|
||||||
|
file->WriteObject(SystemVersionData::VERSION_MINOR, 1);
|
||||||
|
file->WriteObject(SystemVersionData::VERSION_MICRO, 2);
|
||||||
|
file->WriteObject(SystemVersionData::REVISION_MAJOR, 4);
|
||||||
|
file->WriteObject(SystemVersionData::REVISION_MINOR, 5);
|
||||||
|
file->WriteArray(SystemVersionData::PLATFORM_STRING,
|
||||||
|
std::min<u64>(sizeof(SystemVersionData::PLATFORM_STRING), 0x20ULL), 0x8);
|
||||||
|
file->WriteArray(SystemVersionData::VERSION_HASH,
|
||||||
|
std::min<u64>(sizeof(SystemVersionData::VERSION_HASH), 0x40ULL), 0x28);
|
||||||
|
file->WriteArray(SystemVersionData::DISPLAY_VERSION,
|
||||||
|
std::min<u64>(sizeof(SystemVersionData::DISPLAY_VERSION), 0x18ULL), 0x68);
|
||||||
|
file->WriteArray(SystemVersionData::DISPLAY_TITLE,
|
||||||
|
std::min<u64>(sizeof(SystemVersionData::DISPLAY_TITLE), 0x80ULL), 0x80);
|
||||||
|
return std::make_shared<VectorVfsDirectory>(std::vector<VirtualFile>{file},
|
||||||
|
std::vector<VirtualDir>{}, "data");
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace FileSys::SystemArchive
|
16
src/core/file_sys/system_archive/system_version.h
Normal file
16
src/core/file_sys/system_archive/system_version.h
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
// Copyright 2019 yuzu emulator team
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include "core/file_sys/vfs_types.h"
|
||||||
|
|
||||||
|
namespace FileSys::SystemArchive {
|
||||||
|
|
||||||
|
std::string GetLongDisplayVersion();
|
||||||
|
|
||||||
|
VirtualDir SystemVersion();
|
||||||
|
|
||||||
|
} // namespace FileSys::SystemArchive
|
|
@ -2,13 +2,88 @@
|
||||||
// Licensed under GPLv2 or any later version
|
// Licensed under GPLv2 or any later version
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "common/assert.h"
|
||||||
#include "common/logging/log.h"
|
#include "common/logging/log.h"
|
||||||
|
#include "core/file_sys/errors.h"
|
||||||
|
#include "core/file_sys/system_archive/system_version.h"
|
||||||
#include "core/hle/ipc_helpers.h"
|
#include "core/hle/ipc_helpers.h"
|
||||||
#include "core/hle/kernel/client_port.h"
|
#include "core/hle/kernel/client_port.h"
|
||||||
|
#include "core/hle/service/filesystem/filesystem.h"
|
||||||
#include "core/hle/service/set/set_sys.h"
|
#include "core/hle/service/set/set_sys.h"
|
||||||
|
|
||||||
namespace Service::Set {
|
namespace Service::Set {
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
constexpr u64 SYSTEM_VERSION_FILE_MINOR_REVISION_OFFSET = 0x05;
|
||||||
|
|
||||||
|
enum class GetFirmwareVersionType {
|
||||||
|
Version1,
|
||||||
|
Version2,
|
||||||
|
};
|
||||||
|
|
||||||
|
void GetFirmwareVersionImpl(Kernel::HLERequestContext& ctx, GetFirmwareVersionType type) {
|
||||||
|
LOG_WARNING(Service_SET, "called - Using hardcoded firmware version '{}'",
|
||||||
|
FileSys::SystemArchive::GetLongDisplayVersion());
|
||||||
|
|
||||||
|
ASSERT_MSG(ctx.GetWriteBufferSize() == 0x100,
|
||||||
|
"FirmwareVersion output buffer must be 0x100 bytes in size!");
|
||||||
|
|
||||||
|
// Instead of using the normal procedure of checking for the real system archive and if it
|
||||||
|
// doesn't exist, synthesizing one, I feel that that would lead to strange bugs because a
|
||||||
|
// used is using a really old or really new SystemVersion title. The synthesized one ensures
|
||||||
|
// consistence (currently reports as 5.1.0-0.0)
|
||||||
|
const auto archive = FileSys::SystemArchive::SystemVersion();
|
||||||
|
|
||||||
|
const auto early_exit_failure = [&ctx](const std::string& desc, ResultCode code) {
|
||||||
|
LOG_ERROR(Service_SET, "General failure while attempting to resolve firmware version ({}).",
|
||||||
|
desc.c_str());
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(code);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (archive == nullptr) {
|
||||||
|
early_exit_failure("The system version archive couldn't be synthesized.",
|
||||||
|
FileSys::ERROR_FAILED_MOUNT_ARCHIVE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto ver_file = archive->GetFile("file");
|
||||||
|
if (ver_file == nullptr) {
|
||||||
|
early_exit_failure("The system version archive didn't contain the file 'file'.",
|
||||||
|
FileSys::ERROR_INVALID_ARGUMENT);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto data = ver_file->ReadAllBytes();
|
||||||
|
if (data.size() != 0x100) {
|
||||||
|
early_exit_failure("The system version file 'file' was not the correct size.",
|
||||||
|
FileSys::ERROR_OUT_OF_BOUNDS);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the command is GetFirmwareVersion (as opposed to GetFirmwareVersion2), hardware will
|
||||||
|
// zero out the REVISION_MINOR field.
|
||||||
|
if (type == GetFirmwareVersionType::Version1) {
|
||||||
|
data[SYSTEM_VERSION_FILE_MINOR_REVISION_OFFSET] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.WriteBuffer(data);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
}
|
||||||
|
} // Anonymous namespace
|
||||||
|
|
||||||
|
void SET_SYS::GetFirmwareVersion(Kernel::HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_SET, "called");
|
||||||
|
GetFirmwareVersionImpl(ctx, GetFirmwareVersionType::Version1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SET_SYS::GetFirmwareVersion2(Kernel::HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_SET, "called");
|
||||||
|
GetFirmwareVersionImpl(ctx, GetFirmwareVersionType::Version2);
|
||||||
|
}
|
||||||
|
|
||||||
void SET_SYS::GetColorSetId(Kernel::HLERequestContext& ctx) {
|
void SET_SYS::GetColorSetId(Kernel::HLERequestContext& ctx) {
|
||||||
LOG_DEBUG(Service_SET, "called");
|
LOG_DEBUG(Service_SET, "called");
|
||||||
|
|
||||||
|
@ -33,8 +108,8 @@ SET_SYS::SET_SYS() : ServiceFramework("set:sys") {
|
||||||
{0, nullptr, "SetLanguageCode"},
|
{0, nullptr, "SetLanguageCode"},
|
||||||
{1, nullptr, "SetNetworkSettings"},
|
{1, nullptr, "SetNetworkSettings"},
|
||||||
{2, nullptr, "GetNetworkSettings"},
|
{2, nullptr, "GetNetworkSettings"},
|
||||||
{3, nullptr, "GetFirmwareVersion"},
|
{3, &SET_SYS::GetFirmwareVersion, "GetFirmwareVersion"},
|
||||||
{4, nullptr, "GetFirmwareVersion2"},
|
{4, &SET_SYS::GetFirmwareVersion2, "GetFirmwareVersion2"},
|
||||||
{5, nullptr, "GetFirmwareVersionDigest"},
|
{5, nullptr, "GetFirmwareVersionDigest"},
|
||||||
{7, nullptr, "GetLockScreenFlag"},
|
{7, nullptr, "GetLockScreenFlag"},
|
||||||
{8, nullptr, "SetLockScreenFlag"},
|
{8, nullptr, "SetLockScreenFlag"},
|
||||||
|
|
|
@ -20,6 +20,8 @@ private:
|
||||||
BasicBlack = 1,
|
BasicBlack = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void GetFirmwareVersion(Kernel::HLERequestContext& ctx);
|
||||||
|
void GetFirmwareVersion2(Kernel::HLERequestContext& ctx);
|
||||||
void GetColorSetId(Kernel::HLERequestContext& ctx);
|
void GetColorSetId(Kernel::HLERequestContext& ctx);
|
||||||
void SetColorSetId(Kernel::HLERequestContext& ctx);
|
void SetColorSetId(Kernel::HLERequestContext& ctx);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue