forked from eden-emu/eden
		
	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…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei