forked from eden-emu/eden
		
	Merge pull request #11632 from german77/hle_cabinet
service: am: Add support for LLE Cabinet Applet
This commit is contained in:
		
						commit
						bbb8399737
					
				
					 23 changed files with 985 additions and 60 deletions
				
			
		|  | @ -698,6 +698,8 @@ add_library(core STATIC | ||||||
|     hle/service/nvnflinger/consumer_base.cpp |     hle/service/nvnflinger/consumer_base.cpp | ||||||
|     hle/service/nvnflinger/consumer_base.h |     hle/service/nvnflinger/consumer_base.h | ||||||
|     hle/service/nvnflinger/consumer_listener.h |     hle/service/nvnflinger/consumer_listener.h | ||||||
|  |     hle/service/nvnflinger/fb_share_buffer_manager.cpp | ||||||
|  |     hle/service/nvnflinger/fb_share_buffer_manager.h | ||||||
|     hle/service/nvnflinger/graphic_buffer_producer.cpp |     hle/service/nvnflinger/graphic_buffer_producer.cpp | ||||||
|     hle/service/nvnflinger/graphic_buffer_producer.h |     hle/service/nvnflinger/graphic_buffer_producer.h | ||||||
|     hle/service/nvnflinger/hos_binder_driver_server.cpp |     hle/service/nvnflinger/hos_binder_driver_server.cpp | ||||||
|  |  | ||||||
|  | @ -8,6 +8,7 @@ | ||||||
| #include "common/settings.h" | #include "common/settings.h" | ||||||
| #include "common/settings_enums.h" | #include "common/settings_enums.h" | ||||||
| #include "core/core.h" | #include "core/core.h" | ||||||
|  | #include "core/core_timing.h" | ||||||
| #include "core/file_sys/control_metadata.h" | #include "core/file_sys/control_metadata.h" | ||||||
| #include "core/file_sys/patch_manager.h" | #include "core/file_sys/patch_manager.h" | ||||||
| #include "core/file_sys/registered_cache.h" | #include "core/file_sys/registered_cache.h" | ||||||
|  | @ -19,6 +20,7 @@ | ||||||
| #include "core/hle/service/am/am.h" | #include "core/hle/service/am/am.h" | ||||||
| #include "core/hle/service/am/applet_ae.h" | #include "core/hle/service/am/applet_ae.h" | ||||||
| #include "core/hle/service/am/applet_oe.h" | #include "core/hle/service/am/applet_oe.h" | ||||||
|  | #include "core/hle/service/am/applets/applet_cabinet.h" | ||||||
| #include "core/hle/service/am/applets/applet_mii_edit_types.h" | #include "core/hle/service/am/applets/applet_mii_edit_types.h" | ||||||
| #include "core/hle/service/am/applets/applet_profile_select.h" | #include "core/hle/service/am/applets/applet_profile_select.h" | ||||||
| #include "core/hle/service/am/applets/applet_web_browser.h" | #include "core/hle/service/am/applets/applet_web_browser.h" | ||||||
|  | @ -33,11 +35,13 @@ | ||||||
| #include "core/hle/service/filesystem/filesystem.h" | #include "core/hle/service/filesystem/filesystem.h" | ||||||
| #include "core/hle/service/ipc_helpers.h" | #include "core/hle/service/ipc_helpers.h" | ||||||
| #include "core/hle/service/ns/ns.h" | #include "core/hle/service/ns/ns.h" | ||||||
|  | #include "core/hle/service/nvnflinger/fb_share_buffer_manager.h" | ||||||
| #include "core/hle/service/nvnflinger/nvnflinger.h" | #include "core/hle/service/nvnflinger/nvnflinger.h" | ||||||
| #include "core/hle/service/pm/pm.h" | #include "core/hle/service/pm/pm.h" | ||||||
| #include "core/hle/service/server_manager.h" | #include "core/hle/service/server_manager.h" | ||||||
| #include "core/hle/service/sm/sm.h" | #include "core/hle/service/sm/sm.h" | ||||||
| #include "core/hle/service/vi/vi.h" | #include "core/hle/service/vi/vi.h" | ||||||
|  | #include "core/hle/service/vi/vi_results.h" | ||||||
| #include "core/memory.h" | #include "core/memory.h" | ||||||
| 
 | 
 | ||||||
| namespace Service::AM { | namespace Service::AM { | ||||||
|  | @ -190,7 +194,7 @@ IDisplayController::IDisplayController(Core::System& system_) | ||||||
|         {4, nullptr, "UpdateCallerAppletCaptureImage"}, |         {4, nullptr, "UpdateCallerAppletCaptureImage"}, | ||||||
|         {5, nullptr, "GetLastForegroundCaptureImageEx"}, |         {5, nullptr, "GetLastForegroundCaptureImageEx"}, | ||||||
|         {6, nullptr, "GetLastApplicationCaptureImageEx"}, |         {6, nullptr, "GetLastApplicationCaptureImageEx"}, | ||||||
|         {7, nullptr, "GetCallerAppletCaptureImageEx"}, |         {7, &IDisplayController::GetCallerAppletCaptureImageEx, "GetCallerAppletCaptureImageEx"}, | ||||||
|         {8, &IDisplayController::TakeScreenShotOfOwnLayer, "TakeScreenShotOfOwnLayer"}, |         {8, &IDisplayController::TakeScreenShotOfOwnLayer, "TakeScreenShotOfOwnLayer"}, | ||||||
|         {9, nullptr, "CopyBetweenCaptureBuffers"}, |         {9, nullptr, "CopyBetweenCaptureBuffers"}, | ||||||
|         {10, nullptr, "AcquireLastApplicationCaptureBuffer"}, |         {10, nullptr, "AcquireLastApplicationCaptureBuffer"}, | ||||||
|  | @ -208,8 +212,8 @@ IDisplayController::IDisplayController(Core::System& system_) | ||||||
|         {23, nullptr, "ReleaseLastApplicationCaptureSharedBuffer"}, |         {23, nullptr, "ReleaseLastApplicationCaptureSharedBuffer"}, | ||||||
|         {24, nullptr, "AcquireLastForegroundCaptureSharedBuffer"}, |         {24, nullptr, "AcquireLastForegroundCaptureSharedBuffer"}, | ||||||
|         {25, nullptr, "ReleaseLastForegroundCaptureSharedBuffer"}, |         {25, nullptr, "ReleaseLastForegroundCaptureSharedBuffer"}, | ||||||
|         {26, nullptr, "AcquireCallerAppletCaptureSharedBuffer"}, |         {26, &IDisplayController::AcquireCallerAppletCaptureSharedBuffer, "AcquireCallerAppletCaptureSharedBuffer"}, | ||||||
|         {27, nullptr, "ReleaseCallerAppletCaptureSharedBuffer"}, |         {27, &IDisplayController::ReleaseCallerAppletCaptureSharedBuffer, "ReleaseCallerAppletCaptureSharedBuffer"}, | ||||||
|         {28, nullptr, "TakeScreenShotOfOwnLayerEx"}, |         {28, nullptr, "TakeScreenShotOfOwnLayerEx"}, | ||||||
|     }; |     }; | ||||||
|     // clang-format on
 |     // clang-format on
 | ||||||
|  | @ -219,6 +223,15 @@ IDisplayController::IDisplayController(Core::System& system_) | ||||||
| 
 | 
 | ||||||
| IDisplayController::~IDisplayController() = default; | IDisplayController::~IDisplayController() = default; | ||||||
| 
 | 
 | ||||||
|  | void IDisplayController::GetCallerAppletCaptureImageEx(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 4}; | ||||||
|  |     rb.Push(ResultSuccess); | ||||||
|  |     rb.Push(1u); | ||||||
|  |     rb.Push(0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void IDisplayController::TakeScreenShotOfOwnLayer(HLERequestContext& ctx) { | void IDisplayController::TakeScreenShotOfOwnLayer(HLERequestContext& ctx) { | ||||||
|     LOG_WARNING(Service_AM, "(STUBBED) called"); |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
| 
 | 
 | ||||||
|  | @ -226,6 +239,22 @@ void IDisplayController::TakeScreenShotOfOwnLayer(HLERequestContext& ctx) { | ||||||
|     rb.Push(ResultSuccess); |     rb.Push(ResultSuccess); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void IDisplayController::AcquireCallerAppletCaptureSharedBuffer(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 4}; | ||||||
|  |     rb.Push(ResultSuccess); | ||||||
|  |     rb.Push(1U); | ||||||
|  |     rb.Push(0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void IDisplayController::ReleaseCallerAppletCaptureSharedBuffer(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |     rb.Push(ResultSuccess); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| IDebugFunctions::IDebugFunctions(Core::System& system_) | IDebugFunctions::IDebugFunctions(Core::System& system_) | ||||||
|     : ServiceFramework{system_, "IDebugFunctions"} { |     : ServiceFramework{system_, "IDebugFunctions"} { | ||||||
|     // clang-format off
 |     // clang-format off
 | ||||||
|  | @ -285,14 +314,14 @@ ISelfController::ISelfController(Core::System& system_, Nvnflinger::Nvnflinger& | ||||||
|         {20, nullptr, "SetDesirableKeyboardLayout"}, |         {20, nullptr, "SetDesirableKeyboardLayout"}, | ||||||
|         {21, nullptr, "GetScreenShotProgramId"}, |         {21, nullptr, "GetScreenShotProgramId"}, | ||||||
|         {40, &ISelfController::CreateManagedDisplayLayer, "CreateManagedDisplayLayer"}, |         {40, &ISelfController::CreateManagedDisplayLayer, "CreateManagedDisplayLayer"}, | ||||||
|         {41, nullptr, "IsSystemBufferSharingEnabled"}, |         {41, &ISelfController::IsSystemBufferSharingEnabled, "IsSystemBufferSharingEnabled"}, | ||||||
|         {42, nullptr, "GetSystemSharedLayerHandle"}, |         {42, &ISelfController::GetSystemSharedLayerHandle, "GetSystemSharedLayerHandle"}, | ||||||
|         {43, nullptr, "GetSystemSharedBufferHandle"}, |         {43, &ISelfController::GetSystemSharedBufferHandle, "GetSystemSharedBufferHandle"}, | ||||||
|         {44, &ISelfController::CreateManagedDisplaySeparableLayer, "CreateManagedDisplaySeparableLayer"}, |         {44, &ISelfController::CreateManagedDisplaySeparableLayer, "CreateManagedDisplaySeparableLayer"}, | ||||||
|         {45, nullptr, "SetManagedDisplayLayerSeparationMode"}, |         {45, nullptr, "SetManagedDisplayLayerSeparationMode"}, | ||||||
|         {46, nullptr, "SetRecordingLayerCompositionEnabled"}, |         {46, nullptr, "SetRecordingLayerCompositionEnabled"}, | ||||||
|         {50, &ISelfController::SetHandlesRequestToDisplay, "SetHandlesRequestToDisplay"}, |         {50, &ISelfController::SetHandlesRequestToDisplay, "SetHandlesRequestToDisplay"}, | ||||||
|         {51, nullptr, "ApproveToDisplay"}, |         {51, &ISelfController::ApproveToDisplay, "ApproveToDisplay"}, | ||||||
|         {60, nullptr, "OverrideAutoSleepTimeAndDimmingTime"}, |         {60, nullptr, "OverrideAutoSleepTimeAndDimmingTime"}, | ||||||
|         {61, nullptr, "SetMediaPlaybackState"}, |         {61, nullptr, "SetMediaPlaybackState"}, | ||||||
|         {62, &ISelfController::SetIdleTimeDetectionExtension, "SetIdleTimeDetectionExtension"}, |         {62, &ISelfController::SetIdleTimeDetectionExtension, "SetIdleTimeDetectionExtension"}, | ||||||
|  | @ -491,6 +520,50 @@ void ISelfController::CreateManagedDisplayLayer(HLERequestContext& ctx) { | ||||||
|     rb.Push(*layer_id); |     rb.Push(*layer_id); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ISelfController::IsSystemBufferSharingEnabled(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |     rb.Push(this->EnsureBufferSharingEnabled()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ISelfController::GetSystemSharedLayerHandle(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 6}; | ||||||
|  |     rb.Push(this->EnsureBufferSharingEnabled()); | ||||||
|  |     rb.Push<s64>(system_shared_buffer_id); | ||||||
|  |     rb.Push<s64>(system_shared_layer_id); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ISelfController::GetSystemSharedBufferHandle(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 4}; | ||||||
|  |     rb.Push(this->EnsureBufferSharingEnabled()); | ||||||
|  |     rb.Push<s64>(system_shared_buffer_id); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result ISelfController::EnsureBufferSharingEnabled() { | ||||||
|  |     if (buffer_sharing_enabled) { | ||||||
|  |         return ResultSuccess; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (system.GetAppletManager().GetCurrentAppletId() <= Applets::AppletId::Application) { | ||||||
|  |         return VI::ResultOperationFailed; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const auto display_id = nvnflinger.OpenDisplay("Default"); | ||||||
|  |     const auto result = nvnflinger.GetSystemBufferManager().Initialize( | ||||||
|  |         &system_shared_buffer_id, &system_shared_layer_id, *display_id); | ||||||
|  | 
 | ||||||
|  |     if (result.IsSuccess()) { | ||||||
|  |         buffer_sharing_enabled = true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ISelfController::CreateManagedDisplaySeparableLayer(HLERequestContext& ctx) { | void ISelfController::CreateManagedDisplaySeparableLayer(HLERequestContext& ctx) { | ||||||
|     LOG_WARNING(Service_AM, "(STUBBED) called"); |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
| 
 | 
 | ||||||
|  | @ -516,6 +589,13 @@ void ISelfController::SetHandlesRequestToDisplay(HLERequestContext& ctx) { | ||||||
|     rb.Push(ResultSuccess); |     rb.Push(ResultSuccess); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ISelfController::ApproveToDisplay(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |     rb.Push(ResultSuccess); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ISelfController::SetIdleTimeDetectionExtension(HLERequestContext& ctx) { | void ISelfController::SetIdleTimeDetectionExtension(HLERequestContext& ctx) { | ||||||
|     IPC::RequestParser rp{ctx}; |     IPC::RequestParser rp{ctx}; | ||||||
|     idle_time_detection_extension = rp.Pop<u32>(); |     idle_time_detection_extension = rp.Pop<u32>(); | ||||||
|  | @ -686,7 +766,8 @@ void AppletMessageQueue::OperationModeChanged() { | ||||||
| 
 | 
 | ||||||
| ICommonStateGetter::ICommonStateGetter(Core::System& system_, | ICommonStateGetter::ICommonStateGetter(Core::System& system_, | ||||||
|                                        std::shared_ptr<AppletMessageQueue> msg_queue_) |                                        std::shared_ptr<AppletMessageQueue> msg_queue_) | ||||||
|     : ServiceFramework{system_, "ICommonStateGetter"}, msg_queue{std::move(msg_queue_)} { |     : ServiceFramework{system_, "ICommonStateGetter"}, msg_queue{std::move(msg_queue_)}, | ||||||
|  |       service_context{system_, "ICommonStateGetter"} { | ||||||
|     // clang-format off
 |     // clang-format off
 | ||||||
|     static const FunctionInfo functions[] = { |     static const FunctionInfo functions[] = { | ||||||
|         {0, &ICommonStateGetter::GetEventHandle, "GetEventHandle"}, |         {0, &ICommonStateGetter::GetEventHandle, "GetEventHandle"}, | ||||||
|  | @ -699,10 +780,10 @@ ICommonStateGetter::ICommonStateGetter(Core::System& system_, | ||||||
|         {7, nullptr, "GetCradleStatus"}, |         {7, nullptr, "GetCradleStatus"}, | ||||||
|         {8, &ICommonStateGetter::GetBootMode, "GetBootMode"}, |         {8, &ICommonStateGetter::GetBootMode, "GetBootMode"}, | ||||||
|         {9, &ICommonStateGetter::GetCurrentFocusState, "GetCurrentFocusState"}, |         {9, &ICommonStateGetter::GetCurrentFocusState, "GetCurrentFocusState"}, | ||||||
|         {10, nullptr, "RequestToAcquireSleepLock"}, |         {10, &ICommonStateGetter::RequestToAcquireSleepLock, "RequestToAcquireSleepLock"}, | ||||||
|         {11, nullptr, "ReleaseSleepLock"}, |         {11, nullptr, "ReleaseSleepLock"}, | ||||||
|         {12, nullptr, "ReleaseSleepLockTransiently"}, |         {12, nullptr, "ReleaseSleepLockTransiently"}, | ||||||
|         {13, nullptr, "GetAcquiredSleepLockEvent"}, |         {13, &ICommonStateGetter::GetAcquiredSleepLockEvent, "GetAcquiredSleepLockEvent"}, | ||||||
|         {14, nullptr, "GetWakeupCount"}, |         {14, nullptr, "GetWakeupCount"}, | ||||||
|         {20, nullptr, "PushToGeneralChannel"}, |         {20, nullptr, "PushToGeneralChannel"}, | ||||||
|         {30, nullptr, "GetHomeButtonReaderLockAccessor"}, |         {30, nullptr, "GetHomeButtonReaderLockAccessor"}, | ||||||
|  | @ -745,6 +826,8 @@ ICommonStateGetter::ICommonStateGetter(Core::System& system_, | ||||||
| 
 | 
 | ||||||
|     RegisterHandlers(functions); |     RegisterHandlers(functions); | ||||||
| 
 | 
 | ||||||
|  |     sleep_lock_event = service_context.CreateEvent("ICommonStateGetter::SleepLockEvent"); | ||||||
|  | 
 | ||||||
|     // Configure applets to be in foreground state
 |     // Configure applets to be in foreground state
 | ||||||
|     msg_queue->PushMessage(AppletMessageQueue::AppletMessage::FocusStateChanged); |     msg_queue->PushMessage(AppletMessageQueue::AppletMessage::FocusStateChanged); | ||||||
|     msg_queue->PushMessage(AppletMessageQueue::AppletMessage::ChangeIntoForeground); |     msg_queue->PushMessage(AppletMessageQueue::AppletMessage::ChangeIntoForeground); | ||||||
|  | @ -793,6 +876,24 @@ void ICommonStateGetter::GetCurrentFocusState(HLERequestContext& ctx) { | ||||||
|     rb.Push(static_cast<u8>(FocusState::InFocus)); |     rb.Push(static_cast<u8>(FocusState::InFocus)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ICommonStateGetter::RequestToAcquireSleepLock(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |     // Sleep lock is acquired immediately.
 | ||||||
|  |     sleep_lock_event->Signal(); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |     rb.Push(ResultSuccess); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ICommonStateGetter::GetAcquiredSleepLockEvent(HLERequestContext& ctx) { | ||||||
|  |     LOG_WARNING(Service_AM, "called"); | ||||||
|  | 
 | ||||||
|  |     IPC::ResponseBuilder rb{ctx, 2, 1}; | ||||||
|  |     rb.Push(ResultSuccess); | ||||||
|  |     rb.PushCopyObjects(sleep_lock_event->GetReadableEvent()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ICommonStateGetter::IsVrModeEnabled(HLERequestContext& ctx) { | void ICommonStateGetter::IsVrModeEnabled(HLERequestContext& ctx) { | ||||||
|     LOG_DEBUG(Service_AM, "called"); |     LOG_DEBUG(Service_AM, "called"); | ||||||
| 
 | 
 | ||||||
|  | @ -1385,7 +1486,16 @@ ILibraryAppletSelfAccessor::ILibraryAppletSelfAccessor(Core::System& system_) | ||||||
|     // clang-format on
 |     // clang-format on
 | ||||||
|     RegisterHandlers(functions); |     RegisterHandlers(functions); | ||||||
| 
 | 
 | ||||||
|  |     switch (system.GetAppletManager().GetCurrentAppletId()) { | ||||||
|  |     case Applets::AppletId::Cabinet: | ||||||
|  |         PushInShowCabinetData(); | ||||||
|  |         break; | ||||||
|  |     case Applets::AppletId::MiiEdit: | ||||||
|         PushInShowMiiEditData(); |         PushInShowMiiEditData(); | ||||||
|  |         break; | ||||||
|  |     default: | ||||||
|  |         break; | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ILibraryAppletSelfAccessor::~ILibraryAppletSelfAccessor() = default; | ILibraryAppletSelfAccessor::~ILibraryAppletSelfAccessor() = default; | ||||||
|  | @ -1431,7 +1541,7 @@ void ILibraryAppletSelfAccessor::GetLibraryAppletInfo(HLERequestContext& ctx) { | ||||||
|     LOG_WARNING(Service_AM, "(STUBBED) called"); |     LOG_WARNING(Service_AM, "(STUBBED) called"); | ||||||
| 
 | 
 | ||||||
|     const LibraryAppletInfo applet_info{ |     const LibraryAppletInfo applet_info{ | ||||||
|         .applet_id = Applets::AppletId::MiiEdit, |         .applet_id = system.GetAppletManager().GetCurrentAppletId(), | ||||||
|         .library_applet_mode = Applets::LibraryAppletMode::AllForeground, |         .library_applet_mode = Applets::LibraryAppletMode::AllForeground, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  | @ -1459,6 +1569,35 @@ void ILibraryAppletSelfAccessor::GetCallerAppletIdentityInfo(HLERequestContext& | ||||||
|     rb.PushRaw(applet_info); |     rb.PushRaw(applet_info); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ILibraryAppletSelfAccessor::PushInShowCabinetData() { | ||||||
|  |     const Applets::CommonArguments arguments{ | ||||||
|  |         .arguments_version = Applets::CommonArgumentVersion::Version3, | ||||||
|  |         .size = Applets::CommonArgumentSize::Version3, | ||||||
|  |         .library_version = static_cast<u32>(Applets::CabinetAppletVersion::Version1), | ||||||
|  |         .theme_color = Applets::ThemeColor::BasicBlack, | ||||||
|  |         .play_startup_sound = true, | ||||||
|  |         .system_tick = system.CoreTiming().GetClockTicks(), | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const Applets::StartParamForAmiiboSettings amiibo_settings{ | ||||||
|  |         .param_1 = 0, | ||||||
|  |         .applet_mode = system.GetAppletManager().GetCabinetMode(), | ||||||
|  |         .flags = Applets::CabinetFlags::None, | ||||||
|  |         .amiibo_settings_1 = 0, | ||||||
|  |         .device_handle = 0, | ||||||
|  |         .tag_info{}, | ||||||
|  |         .register_info{}, | ||||||
|  |         .amiibo_settings_3{}, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     std::vector<u8> argument_data(sizeof(arguments)); | ||||||
|  |     std::vector<u8> settings_data(sizeof(amiibo_settings)); | ||||||
|  |     std::memcpy(argument_data.data(), &arguments, sizeof(arguments)); | ||||||
|  |     std::memcpy(settings_data.data(), &amiibo_settings, sizeof(amiibo_settings)); | ||||||
|  |     queue_data.emplace_back(std::move(argument_data)); | ||||||
|  |     queue_data.emplace_back(std::move(settings_data)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ILibraryAppletSelfAccessor::PushInShowMiiEditData() { | void ILibraryAppletSelfAccessor::PushInShowMiiEditData() { | ||||||
|     struct MiiEditV3 { |     struct MiiEditV3 { | ||||||
|         Applets::MiiEditAppletInputCommon common; |         Applets::MiiEditAppletInputCommon common; | ||||||
|  | @ -2235,7 +2374,7 @@ void IProcessWindingController::GetLaunchReason(HLERequestContext& ctx) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void IProcessWindingController::OpenCallingLibraryApplet(HLERequestContext& ctx) { | void IProcessWindingController::OpenCallingLibraryApplet(HLERequestContext& ctx) { | ||||||
|     const auto applet_id = Applets::AppletId::MiiEdit; |     const auto applet_id = system.GetAppletManager().GetCurrentAppletId(); | ||||||
|     const auto applet_mode = Applets::LibraryAppletMode::AllForeground; |     const auto applet_mode = Applets::LibraryAppletMode::AllForeground; | ||||||
| 
 | 
 | ||||||
|     LOG_WARNING(Service_AM, "(STUBBED) called with applet_id={:08X}, applet_mode={:08X}", applet_id, |     LOG_WARNING(Service_AM, "(STUBBED) called with applet_id={:08X}, applet_mode={:08X}", applet_id, | ||||||
|  | @ -2256,4 +2395,5 @@ void IProcessWindingController::OpenCallingLibraryApplet(HLERequestContext& ctx) | ||||||
|     rb.Push(ResultSuccess); |     rb.Push(ResultSuccess); | ||||||
|     rb.PushIpcInterface<ILibraryAppletAccessor>(system, applet); |     rb.PushIpcInterface<ILibraryAppletAccessor>(system, applet); | ||||||
| } | } | ||||||
|  | 
 | ||||||
| } // namespace Service::AM
 | } // namespace Service::AM
 | ||||||
|  |  | ||||||
|  | @ -122,7 +122,10 @@ public: | ||||||
|     ~IDisplayController() override; |     ~IDisplayController() override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  |     void GetCallerAppletCaptureImageEx(HLERequestContext& ctx); | ||||||
|     void TakeScreenShotOfOwnLayer(HLERequestContext& ctx); |     void TakeScreenShotOfOwnLayer(HLERequestContext& ctx); | ||||||
|  |     void AcquireCallerAppletCaptureSharedBuffer(HLERequestContext& ctx); | ||||||
|  |     void ReleaseCallerAppletCaptureSharedBuffer(HLERequestContext& ctx); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class IDebugFunctions final : public ServiceFramework<IDebugFunctions> { | class IDebugFunctions final : public ServiceFramework<IDebugFunctions> { | ||||||
|  | @ -150,9 +153,13 @@ private: | ||||||
|     void SetRestartMessageEnabled(HLERequestContext& ctx); |     void SetRestartMessageEnabled(HLERequestContext& ctx); | ||||||
|     void SetOutOfFocusSuspendingEnabled(HLERequestContext& ctx); |     void SetOutOfFocusSuspendingEnabled(HLERequestContext& ctx); | ||||||
|     void SetAlbumImageOrientation(HLERequestContext& ctx); |     void SetAlbumImageOrientation(HLERequestContext& ctx); | ||||||
|  |     void IsSystemBufferSharingEnabled(HLERequestContext& ctx); | ||||||
|  |     void GetSystemSharedBufferHandle(HLERequestContext& ctx); | ||||||
|  |     void GetSystemSharedLayerHandle(HLERequestContext& ctx); | ||||||
|     void CreateManagedDisplayLayer(HLERequestContext& ctx); |     void CreateManagedDisplayLayer(HLERequestContext& ctx); | ||||||
|     void CreateManagedDisplaySeparableLayer(HLERequestContext& ctx); |     void CreateManagedDisplaySeparableLayer(HLERequestContext& ctx); | ||||||
|     void SetHandlesRequestToDisplay(HLERequestContext& ctx); |     void SetHandlesRequestToDisplay(HLERequestContext& ctx); | ||||||
|  |     void ApproveToDisplay(HLERequestContext& ctx); | ||||||
|     void SetIdleTimeDetectionExtension(HLERequestContext& ctx); |     void SetIdleTimeDetectionExtension(HLERequestContext& ctx); | ||||||
|     void GetIdleTimeDetectionExtension(HLERequestContext& ctx); |     void GetIdleTimeDetectionExtension(HLERequestContext& ctx); | ||||||
|     void ReportUserIsActive(HLERequestContext& ctx); |     void ReportUserIsActive(HLERequestContext& ctx); | ||||||
|  | @ -164,6 +171,8 @@ private: | ||||||
|     void SaveCurrentScreenshot(HLERequestContext& ctx); |     void SaveCurrentScreenshot(HLERequestContext& ctx); | ||||||
|     void SetRecordVolumeMuted(HLERequestContext& ctx); |     void SetRecordVolumeMuted(HLERequestContext& ctx); | ||||||
| 
 | 
 | ||||||
|  |     Result EnsureBufferSharingEnabled(); | ||||||
|  | 
 | ||||||
|     enum class ScreenshotPermission : u32 { |     enum class ScreenshotPermission : u32 { | ||||||
|         Inherit = 0, |         Inherit = 0, | ||||||
|         Enable = 1, |         Enable = 1, | ||||||
|  | @ -179,7 +188,10 @@ private: | ||||||
| 
 | 
 | ||||||
|     u32 idle_time_detection_extension = 0; |     u32 idle_time_detection_extension = 0; | ||||||
|     u64 num_fatal_sections_entered = 0; |     u64 num_fatal_sections_entered = 0; | ||||||
|  |     u64 system_shared_buffer_id = 0; | ||||||
|  |     u64 system_shared_layer_id = 0; | ||||||
|     bool is_auto_sleep_disabled = false; |     bool is_auto_sleep_disabled = false; | ||||||
|  |     bool buffer_sharing_enabled = false; | ||||||
|     ScreenshotPermission screenshot_permission = ScreenshotPermission::Inherit; |     ScreenshotPermission screenshot_permission = ScreenshotPermission::Inherit; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | @ -223,6 +235,8 @@ private: | ||||||
|     void GetEventHandle(HLERequestContext& ctx); |     void GetEventHandle(HLERequestContext& ctx); | ||||||
|     void ReceiveMessage(HLERequestContext& ctx); |     void ReceiveMessage(HLERequestContext& ctx); | ||||||
|     void GetCurrentFocusState(HLERequestContext& ctx); |     void GetCurrentFocusState(HLERequestContext& ctx); | ||||||
|  |     void RequestToAcquireSleepLock(HLERequestContext& ctx); | ||||||
|  |     void GetAcquiredSleepLockEvent(HLERequestContext& ctx); | ||||||
|     void GetDefaultDisplayResolutionChangeEvent(HLERequestContext& ctx); |     void GetDefaultDisplayResolutionChangeEvent(HLERequestContext& ctx); | ||||||
|     void GetOperationMode(HLERequestContext& ctx); |     void GetOperationMode(HLERequestContext& ctx); | ||||||
|     void GetPerformanceMode(HLERequestContext& ctx); |     void GetPerformanceMode(HLERequestContext& ctx); | ||||||
|  | @ -240,6 +254,8 @@ private: | ||||||
| 
 | 
 | ||||||
|     std::shared_ptr<AppletMessageQueue> msg_queue; |     std::shared_ptr<AppletMessageQueue> msg_queue; | ||||||
|     bool vr_mode_state{}; |     bool vr_mode_state{}; | ||||||
|  |     Kernel::KEvent* sleep_lock_event; | ||||||
|  |     KernelHelpers::ServiceContext service_context; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class IStorageImpl { | class IStorageImpl { | ||||||
|  | @ -311,6 +327,7 @@ private: | ||||||
|     void ExitProcessAndReturn(HLERequestContext& ctx); |     void ExitProcessAndReturn(HLERequestContext& ctx); | ||||||
|     void GetCallerAppletIdentityInfo(HLERequestContext& ctx); |     void GetCallerAppletIdentityInfo(HLERequestContext& ctx); | ||||||
| 
 | 
 | ||||||
|  |     void PushInShowCabinetData(); | ||||||
|     void PushInShowMiiEditData(); |     void PushInShowMiiEditData(); | ||||||
| 
 | 
 | ||||||
|     std::deque<std::vector<u8>> queue_data; |     std::deque<std::vector<u8>> queue_data; | ||||||
|  |  | ||||||
|  | @ -29,6 +29,15 @@ enum class CabinetAppletVersion : u32 { | ||||||
|     Version1 = 0x1, |     Version1 = 0x1, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | enum class CabinetFlags : u8 { | ||||||
|  |     None = 0, | ||||||
|  |     DeviceHandle = 1 << 0, | ||||||
|  |     TagInfo = 1 << 1, | ||||||
|  |     RegisterInfo = 1 << 2, | ||||||
|  |     All = DeviceHandle | TagInfo | RegisterInfo, | ||||||
|  | }; | ||||||
|  | DECLARE_ENUM_FLAG_OPERATORS(CabinetFlags) | ||||||
|  | 
 | ||||||
| enum class CabinetResult : u8 { | enum class CabinetResult : u8 { | ||||||
|     Cancel = 0, |     Cancel = 0, | ||||||
|     TagInfo = 1 << 1, |     TagInfo = 1 << 1, | ||||||
|  | @ -51,7 +60,7 @@ static_assert(sizeof(AmiiboSettingsStartParam) == 0x30, | ||||||
| struct StartParamForAmiiboSettings { | struct StartParamForAmiiboSettings { | ||||||
|     u8 param_1; |     u8 param_1; | ||||||
|     Service::NFP::CabinetMode applet_mode; |     Service::NFP::CabinetMode applet_mode; | ||||||
|     u8 flags; |     CabinetFlags flags; | ||||||
|     u8 amiibo_settings_1; |     u8 amiibo_settings_1; | ||||||
|     u64 device_handle; |     u64 device_handle; | ||||||
|     Service::NFP::TagInfo tag_info; |     Service::NFP::TagInfo tag_info; | ||||||
|  |  | ||||||
|  | @ -223,9 +223,9 @@ void StubApplet::Initialize() { | ||||||
| 
 | 
 | ||||||
|     const auto data = broker.PeekDataToAppletForDebug(); |     const auto data = broker.PeekDataToAppletForDebug(); | ||||||
|     system.GetReporter().SaveUnimplementedAppletReport( |     system.GetReporter().SaveUnimplementedAppletReport( | ||||||
|         static_cast<u32>(id), common_args.arguments_version, common_args.library_version, |         static_cast<u32>(id), static_cast<u32>(common_args.arguments_version), | ||||||
|         common_args.theme_color, common_args.play_startup_sound, common_args.system_tick, |         common_args.library_version, static_cast<u32>(common_args.theme_color), | ||||||
|         data.normal, data.interactive); |         common_args.play_startup_sound, common_args.system_tick, data.normal, data.interactive); | ||||||
| 
 | 
 | ||||||
|     LogCurrentStorage(broker, "Initialize"); |     LogCurrentStorage(broker, "Initialize"); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -199,6 +199,14 @@ const AppletFrontendSet& AppletManager::GetAppletFrontendSet() const { | ||||||
|     return frontend; |     return frontend; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | NFP::CabinetMode AppletManager::GetCabinetMode() const { | ||||||
|  |     return cabinet_mode; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | AppletId AppletManager::GetCurrentAppletId() const { | ||||||
|  |     return current_applet_id; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void AppletManager::SetAppletFrontendSet(AppletFrontendSet set) { | void AppletManager::SetAppletFrontendSet(AppletFrontendSet set) { | ||||||
|     if (set.cabinet != nullptr) { |     if (set.cabinet != nullptr) { | ||||||
|         frontend.cabinet = std::move(set.cabinet); |         frontend.cabinet = std::move(set.cabinet); | ||||||
|  | @ -237,6 +245,14 @@ void AppletManager::SetAppletFrontendSet(AppletFrontendSet set) { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void AppletManager::SetCabinetMode(NFP::CabinetMode mode) { | ||||||
|  |     cabinet_mode = mode; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void AppletManager::SetCurrentAppletId(AppletId applet_id) { | ||||||
|  |     current_applet_id = applet_id; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void AppletManager::SetDefaultAppletFrontendSet() { | void AppletManager::SetDefaultAppletFrontendSet() { | ||||||
|     ClearAll(); |     ClearAll(); | ||||||
|     SetDefaultAppletsIfMissing(); |     SetDefaultAppletsIfMissing(); | ||||||
|  |  | ||||||
|  | @ -34,6 +34,10 @@ class KEvent; | ||||||
| class KReadableEvent; | class KReadableEvent; | ||||||
| } // namespace Kernel
 | } // namespace Kernel
 | ||||||
| 
 | 
 | ||||||
|  | namespace Service::NFP { | ||||||
|  | enum class CabinetMode : u8; | ||||||
|  | } // namespace Service::NFP
 | ||||||
|  | 
 | ||||||
| namespace Service::AM { | namespace Service::AM { | ||||||
| 
 | 
 | ||||||
| class IStorage; | class IStorage; | ||||||
|  | @ -41,6 +45,8 @@ class IStorage; | ||||||
| namespace Applets { | namespace Applets { | ||||||
| 
 | 
 | ||||||
| enum class AppletId : u32 { | enum class AppletId : u32 { | ||||||
|  |     None = 0x00, | ||||||
|  |     Application = 0x01, | ||||||
|     OverlayDisplay = 0x02, |     OverlayDisplay = 0x02, | ||||||
|     QLaunch = 0x03, |     QLaunch = 0x03, | ||||||
|     Starter = 0x04, |     Starter = 0x04, | ||||||
|  | @ -71,6 +77,32 @@ enum class LibraryAppletMode : u32 { | ||||||
|     AllForegroundInitiallyHidden = 4, |     AllForegroundInitiallyHidden = 4, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | enum class CommonArgumentVersion : u32 { | ||||||
|  |     Version0, | ||||||
|  |     Version1, | ||||||
|  |     Version2, | ||||||
|  |     Version3, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | enum class CommonArgumentSize : u32 { | ||||||
|  |     Version3 = 0x20, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | enum class ThemeColor : u32 { | ||||||
|  |     BasicWhite = 0, | ||||||
|  |     BasicBlack = 3, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct CommonArguments { | ||||||
|  |     CommonArgumentVersion arguments_version; | ||||||
|  |     CommonArgumentSize size; | ||||||
|  |     u32 library_version; | ||||||
|  |     ThemeColor theme_color; | ||||||
|  |     bool play_startup_sound; | ||||||
|  |     u64_le system_tick; | ||||||
|  | }; | ||||||
|  | static_assert(sizeof(CommonArguments) == 0x20, "CommonArguments has incorrect size."); | ||||||
|  | 
 | ||||||
| class AppletDataBroker final { | class AppletDataBroker final { | ||||||
| public: | public: | ||||||
|     explicit AppletDataBroker(Core::System& system_, LibraryAppletMode applet_mode_); |     explicit AppletDataBroker(Core::System& system_, LibraryAppletMode applet_mode_); | ||||||
|  | @ -161,16 +193,6 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
|     struct CommonArguments { |  | ||||||
|         u32_le arguments_version; |  | ||||||
|         u32_le size; |  | ||||||
|         u32_le library_version; |  | ||||||
|         u32_le theme_color; |  | ||||||
|         bool play_startup_sound; |  | ||||||
|         u64_le system_tick; |  | ||||||
|     }; |  | ||||||
|     static_assert(sizeof(CommonArguments) == 0x20, "CommonArguments has incorrect size."); |  | ||||||
| 
 |  | ||||||
|     CommonArguments common_args{}; |     CommonArguments common_args{}; | ||||||
|     AppletDataBroker broker; |     AppletDataBroker broker; | ||||||
|     LibraryAppletMode applet_mode; |     LibraryAppletMode applet_mode; | ||||||
|  | @ -219,8 +241,12 @@ public: | ||||||
|     ~AppletManager(); |     ~AppletManager(); | ||||||
| 
 | 
 | ||||||
|     const AppletFrontendSet& GetAppletFrontendSet() const; |     const AppletFrontendSet& GetAppletFrontendSet() const; | ||||||
|  |     NFP::CabinetMode GetCabinetMode() const; | ||||||
|  |     AppletId GetCurrentAppletId() const; | ||||||
| 
 | 
 | ||||||
|     void SetAppletFrontendSet(AppletFrontendSet set); |     void SetAppletFrontendSet(AppletFrontendSet set); | ||||||
|  |     void SetCabinetMode(NFP::CabinetMode mode); | ||||||
|  |     void SetCurrentAppletId(AppletId applet_id); | ||||||
|     void SetDefaultAppletFrontendSet(); |     void SetDefaultAppletFrontendSet(); | ||||||
|     void SetDefaultAppletsIfMissing(); |     void SetDefaultAppletsIfMissing(); | ||||||
|     void ClearAll(); |     void ClearAll(); | ||||||
|  | @ -228,6 +254,9 @@ public: | ||||||
|     std::shared_ptr<Applet> GetApplet(AppletId id, LibraryAppletMode mode) const; |     std::shared_ptr<Applet> GetApplet(AppletId id, LibraryAppletMode mode) const; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  |     AppletId current_applet_id{}; | ||||||
|  |     NFP::CabinetMode cabinet_mode{}; | ||||||
|  | 
 | ||||||
|     AppletFrontendSet frontend; |     AppletFrontendSet frontend; | ||||||
|     Core::System& system; |     Core::System& system; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -23,19 +23,39 @@ public: | ||||||
|     explicit IMonitorService(Core::System& system_) : ServiceFramework{system_, "IMonitorService"} { |     explicit IMonitorService(Core::System& system_) : ServiceFramework{system_, "IMonitorService"} { | ||||||
|         // clang-format off
 |         // clang-format off
 | ||||||
|         static const FunctionInfo functions[] = { |         static const FunctionInfo functions[] = { | ||||||
|             {0, nullptr, "GetStateForMonitor"}, |             {0, &IMonitorService::GetStateForMonitor, "GetStateForMonitor"}, | ||||||
|             {1, nullptr, "GetNetworkInfoForMonitor"}, |             {1, nullptr, "GetNetworkInfoForMonitor"}, | ||||||
|             {2, nullptr, "GetIpv4AddressForMonitor"}, |             {2, nullptr, "GetIpv4AddressForMonitor"}, | ||||||
|             {3, nullptr, "GetDisconnectReasonForMonitor"}, |             {3, nullptr, "GetDisconnectReasonForMonitor"}, | ||||||
|             {4, nullptr, "GetSecurityParameterForMonitor"}, |             {4, nullptr, "GetSecurityParameterForMonitor"}, | ||||||
|             {5, nullptr, "GetNetworkConfigForMonitor"}, |             {5, nullptr, "GetNetworkConfigForMonitor"}, | ||||||
|             {100, nullptr, "InitializeMonitor"}, |             {100, &IMonitorService::InitializeMonitor, "InitializeMonitor"}, | ||||||
|             {101, nullptr, "FinalizeMonitor"}, |             {101, nullptr, "FinalizeMonitor"}, | ||||||
|         }; |         }; | ||||||
|         // clang-format on
 |         // clang-format on
 | ||||||
| 
 | 
 | ||||||
|         RegisterHandlers(functions); |         RegisterHandlers(functions); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     void GetStateForMonitor(HLERequestContext& ctx) { | ||||||
|  |         LOG_INFO(Service_LDN, "called"); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 3}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |         rb.PushEnum(state); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void InitializeMonitor(HLERequestContext& ctx) { | ||||||
|  |         LOG_INFO(Service_LDN, "called"); | ||||||
|  | 
 | ||||||
|  |         state = State::Initialized; | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     State state{State::None}; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class LDNM final : public ServiceFramework<LDNM> { | class LDNM final : public ServiceFramework<LDNM> { | ||||||
|  | @ -731,14 +751,81 @@ public: | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | class ISfMonitorService final : public ServiceFramework<ISfMonitorService> { | ||||||
|  | public: | ||||||
|  |     explicit ISfMonitorService(Core::System& system_) | ||||||
|  |         : ServiceFramework{system_, "ISfMonitorService"} { | ||||||
|  |         // clang-format off
 | ||||||
|  |         static const FunctionInfo functions[] = { | ||||||
|  |             {0, &ISfMonitorService::Initialize, "Initialize"}, | ||||||
|  |             {288, &ISfMonitorService::GetGroupInfo, "GetGroupInfo"}, | ||||||
|  |             {320, nullptr, "GetLinkLevel"}, | ||||||
|  |         }; | ||||||
|  |         // clang-format on
 | ||||||
|  | 
 | ||||||
|  |         RegisterHandlers(functions); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     void Initialize(HLERequestContext& ctx) { | ||||||
|  |         LOG_WARNING(Service_LDN, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 3}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |         rb.Push(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void GetGroupInfo(HLERequestContext& ctx) { | ||||||
|  |         LOG_WARNING(Service_LDN, "(STUBBED) called"); | ||||||
|  | 
 | ||||||
|  |         struct GroupInfo { | ||||||
|  |             std::array<u8, 0x200> info; | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         GroupInfo group_info{}; | ||||||
|  | 
 | ||||||
|  |         ctx.WriteBuffer(group_info); | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class LP2PM final : public ServiceFramework<LP2PM> { | ||||||
|  | public: | ||||||
|  |     explicit LP2PM(Core::System& system_) : ServiceFramework{system_, "lp2p:m"} { | ||||||
|  |         // clang-format off
 | ||||||
|  |         static const FunctionInfo functions[] = { | ||||||
|  |             {0, &LP2PM::CreateMonitorService, "CreateMonitorService"}, | ||||||
|  |         }; | ||||||
|  |         // clang-format on
 | ||||||
|  | 
 | ||||||
|  |         RegisterHandlers(functions); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     void CreateMonitorService(HLERequestContext& ctx) { | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         const u64 reserved_input = rp.Pop<u64>(); | ||||||
|  | 
 | ||||||
|  |         LOG_INFO(Service_LDN, "called, reserved_input={}", reserved_input); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |         rb.PushIpcInterface<ISfMonitorService>(system); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| void LoopProcess(Core::System& system) { | void LoopProcess(Core::System& system) { | ||||||
|     auto server_manager = std::make_unique<ServerManager>(system); |     auto server_manager = std::make_unique<ServerManager>(system); | ||||||
| 
 | 
 | ||||||
|     server_manager->RegisterNamedService("ldn:m", std::make_shared<LDNM>(system)); |     server_manager->RegisterNamedService("ldn:m", std::make_shared<LDNM>(system)); | ||||||
|     server_manager->RegisterNamedService("ldn:s", std::make_shared<LDNS>(system)); |     server_manager->RegisterNamedService("ldn:s", std::make_shared<LDNS>(system)); | ||||||
|     server_manager->RegisterNamedService("ldn:u", std::make_shared<LDNU>(system)); |     server_manager->RegisterNamedService("ldn:u", std::make_shared<LDNU>(system)); | ||||||
|  | 
 | ||||||
|     server_manager->RegisterNamedService("lp2p:app", std::make_shared<LP2PAPP>(system)); |     server_manager->RegisterNamedService("lp2p:app", std::make_shared<LP2PAPP>(system)); | ||||||
|     server_manager->RegisterNamedService("lp2p:sys", std::make_shared<LP2PSYS>(system)); |     server_manager->RegisterNamedService("lp2p:sys", std::make_shared<LP2PSYS>(system)); | ||||||
|  |     server_manager->RegisterNamedService("lp2p:m", std::make_shared<LP2PM>(system)); | ||||||
|  | 
 | ||||||
|     ServerManager::RunServer(std::move(server_manager)); |     ServerManager::RunServer(std::move(server_manager)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -830,11 +830,6 @@ Result NfcDevice::SetRegisterInfoPrivate(const NFP::RegisterInfoPrivate& registe | ||||||
|         return ResultWrongDeviceState; |         return ResultWrongDeviceState; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Service::Mii::StoreData store_data{}; |  | ||||||
|     Service::Mii::NfpStoreDataExtension extension{}; |  | ||||||
|     store_data.BuildBase(Mii::Gender::Male); |  | ||||||
|     extension.SetFromStoreData(store_data); |  | ||||||
| 
 |  | ||||||
|     auto& settings = tag_data.settings; |     auto& settings = tag_data.settings; | ||||||
| 
 | 
 | ||||||
|     if (tag_data.settings.settings.amiibo_initialized == 0) { |     if (tag_data.settings.settings.amiibo_initialized == 0) { | ||||||
|  | @ -843,8 +838,8 @@ Result NfcDevice::SetRegisterInfoPrivate(const NFP::RegisterInfoPrivate& registe | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     SetAmiiboName(settings, register_info.amiibo_name); |     SetAmiiboName(settings, register_info.amiibo_name); | ||||||
|     tag_data.owner_mii.BuildFromStoreData(store_data); |     tag_data.owner_mii.BuildFromStoreData(register_info.mii_store_data); | ||||||
|     tag_data.mii_extension = extension; |     tag_data.mii_extension.SetFromStoreData(register_info.mii_store_data); | ||||||
|     tag_data.unknown = 0; |     tag_data.unknown = 0; | ||||||
|     tag_data.unknown2 = {}; |     tag_data.unknown2 = {}; | ||||||
|     settings.country_code_id = 0; |     settings.country_code_id = 0; | ||||||
|  |  | ||||||
|  | @ -45,13 +45,6 @@ public: | ||||||
|         IsSharedMemMapped = 6 |         IsSharedMemMapped = 6 | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| private: |  | ||||||
|     /// Id to use for the next handle that is created.
 |  | ||||||
|     u32 next_handle = 0; |  | ||||||
| 
 |  | ||||||
|     /// Id to use for the next object that is created.
 |  | ||||||
|     u32 next_id = 0; |  | ||||||
| 
 |  | ||||||
|     struct IocCreateParams { |     struct IocCreateParams { | ||||||
|         // Input
 |         // Input
 | ||||||
|         u32_le size{}; |         u32_le size{}; | ||||||
|  | @ -113,6 +106,13 @@ private: | ||||||
|     NvResult IocParam(std::span<const u8> input, std::span<u8> output); |     NvResult IocParam(std::span<const u8> input, std::span<u8> output); | ||||||
|     NvResult IocFree(std::span<const u8> input, std::span<u8> output); |     NvResult IocFree(std::span<const u8> input, std::span<u8> output); | ||||||
| 
 | 
 | ||||||
|  | private: | ||||||
|  |     /// Id to use for the next handle that is created.
 | ||||||
|  |     u32 next_handle = 0; | ||||||
|  | 
 | ||||||
|  |     /// Id to use for the next object that is created.
 | ||||||
|  |     u32 next_id = 0; | ||||||
|  | 
 | ||||||
|     NvCore::Container& container; |     NvCore::Container& container; | ||||||
|     NvCore::NvMap& file; |     NvCore::NvMap& file; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ | ||||||
| 
 | 
 | ||||||
| namespace Service::android { | namespace Service::android { | ||||||
| 
 | 
 | ||||||
| class GraphicBuffer; | struct GraphicBuffer; | ||||||
| 
 | 
 | ||||||
| class BufferItem final { | class BufferItem final { | ||||||
| public: | public: | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ | ||||||
| 
 | 
 | ||||||
| namespace Service::android { | namespace Service::android { | ||||||
| 
 | 
 | ||||||
| class GraphicBuffer; | struct GraphicBuffer; | ||||||
| 
 | 
 | ||||||
| enum class BufferState : u32 { | enum class BufferState : u32 { | ||||||
|     Free = 0, |     Free = 0, | ||||||
|  |  | ||||||
							
								
								
									
										351
									
								
								src/core/hle/service/nvnflinger/fb_share_buffer_manager.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										351
									
								
								src/core/hle/service/nvnflinger/fb_share_buffer_manager.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,351 @@ | ||||||
|  | // SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later
 | ||||||
|  | 
 | ||||||
|  | #include <random> | ||||||
|  | 
 | ||||||
|  | #include "core/core.h" | ||||||
|  | #include "core/hle/kernel/k_process.h" | ||||||
|  | #include "core/hle/kernel/k_system_resource.h" | ||||||
|  | #include "core/hle/service/nvdrv/devices/nvmap.h" | ||||||
|  | #include "core/hle/service/nvdrv/nvdrv.h" | ||||||
|  | #include "core/hle/service/nvnflinger/buffer_queue_producer.h" | ||||||
|  | #include "core/hle/service/nvnflinger/fb_share_buffer_manager.h" | ||||||
|  | #include "core/hle/service/nvnflinger/pixel_format.h" | ||||||
|  | #include "core/hle/service/nvnflinger/ui/graphic_buffer.h" | ||||||
|  | #include "core/hle/service/vi/layer/vi_layer.h" | ||||||
|  | #include "core/hle/service/vi/vi_results.h" | ||||||
|  | 
 | ||||||
|  | namespace Service::Nvnflinger { | ||||||
|  | 
 | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | Result AllocateIoForProcessAddressSpace(Common::ProcessAddress* out_map_address, | ||||||
|  |                                         std::unique_ptr<Kernel::KPageGroup>* out_page_group, | ||||||
|  |                                         Core::System& system, u32 size) { | ||||||
|  |     using Core::Memory::YUZU_PAGESIZE; | ||||||
|  | 
 | ||||||
|  |     // Allocate memory for the system shared buffer.
 | ||||||
|  |     // FIXME: Because the gmmu can only point to cpu addresses, we need
 | ||||||
|  |     //        to map this in the application space to allow it to be used.
 | ||||||
|  |     // FIXME: Add proper smmu emulation.
 | ||||||
|  |     // FIXME: This memory belongs to vi's .data section.
 | ||||||
|  |     auto& kernel = system.Kernel(); | ||||||
|  |     auto* process = system.ApplicationProcess(); | ||||||
|  |     auto& page_table = process->GetPageTable(); | ||||||
|  | 
 | ||||||
|  |     // Hold a temporary page group reference while we try to map it.
 | ||||||
|  |     auto pg = std::make_unique<Kernel::KPageGroup>( | ||||||
|  |         kernel, std::addressof(kernel.GetSystemSystemResource().GetBlockInfoManager())); | ||||||
|  | 
 | ||||||
|  |     // Allocate memory from secure pool.
 | ||||||
|  |     R_TRY(kernel.MemoryManager().AllocateAndOpen( | ||||||
|  |         pg.get(), size / YUZU_PAGESIZE, | ||||||
|  |         Kernel::KMemoryManager::EncodeOption(Kernel::KMemoryManager::Pool::Secure, | ||||||
|  |                                              Kernel::KMemoryManager::Direction::FromBack))); | ||||||
|  | 
 | ||||||
|  |     // Get bounds of where mapping is possible.
 | ||||||
|  |     const VAddr alias_code_begin = GetInteger(page_table.GetAliasCodeRegionStart()); | ||||||
|  |     const VAddr alias_code_size = page_table.GetAliasCodeRegionSize() / YUZU_PAGESIZE; | ||||||
|  |     const auto state = Kernel::KMemoryState::Io; | ||||||
|  |     const auto perm = Kernel::KMemoryPermission::UserReadWrite; | ||||||
|  |     std::mt19937_64 rng{process->GetRandomEntropy(0)}; | ||||||
|  | 
 | ||||||
|  |     // Retry up to 64 times to map into alias code range.
 | ||||||
|  |     Result res = ResultSuccess; | ||||||
|  |     int i; | ||||||
|  |     for (i = 0; i < 64; i++) { | ||||||
|  |         *out_map_address = alias_code_begin + ((rng() % alias_code_size) * YUZU_PAGESIZE); | ||||||
|  |         res = page_table.MapPageGroup(*out_map_address, *pg, state, perm); | ||||||
|  |         if (R_SUCCEEDED(res)) { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Return failure, if necessary
 | ||||||
|  |     R_UNLESS(i < 64, res); | ||||||
|  | 
 | ||||||
|  |     // Return the mapped page group.
 | ||||||
|  |     *out_page_group = std::move(pg); | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | template <typename T> | ||||||
|  | std::span<u8> SerializeIoc(T& params) { | ||||||
|  |     return std::span(reinterpret_cast<u8*>(std::addressof(params)), sizeof(T)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result CreateNvMapHandle(u32* out_nv_map_handle, Nvidia::Devices::nvmap& nvmap, u32 size) { | ||||||
|  |     // Create a handle.
 | ||||||
|  |     Nvidia::Devices::nvmap::IocCreateParams create_in_params{ | ||||||
|  |         .size = size, | ||||||
|  |         .handle = 0, | ||||||
|  |     }; | ||||||
|  |     Nvidia::Devices::nvmap::IocCreateParams create_out_params{}; | ||||||
|  |     R_UNLESS(nvmap.IocCreate(SerializeIoc(create_in_params), SerializeIoc(create_out_params)) == | ||||||
|  |                  Nvidia::NvResult::Success, | ||||||
|  |              VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // Assign the output handle.
 | ||||||
|  |     *out_nv_map_handle = create_out_params.handle; | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result FreeNvMapHandle(Nvidia::Devices::nvmap& nvmap, u32 handle) { | ||||||
|  |     // Free the handle.
 | ||||||
|  |     Nvidia::Devices::nvmap::IocFreeParams free_in_params{ | ||||||
|  |         .handle = handle, | ||||||
|  |     }; | ||||||
|  |     Nvidia::Devices::nvmap::IocFreeParams free_out_params{}; | ||||||
|  |     R_UNLESS(nvmap.IocFree(SerializeIoc(free_in_params), SerializeIoc(free_out_params)) == | ||||||
|  |                  Nvidia::NvResult::Success, | ||||||
|  |              VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result AllocNvMapHandle(Nvidia::Devices::nvmap& nvmap, u32 handle, Common::ProcessAddress buffer, | ||||||
|  |                         u32 size) { | ||||||
|  |     // Assign the allocated memory to the handle.
 | ||||||
|  |     Nvidia::Devices::nvmap::IocAllocParams alloc_in_params{ | ||||||
|  |         .handle = handle, | ||||||
|  |         .heap_mask = 0, | ||||||
|  |         .flags = {}, | ||||||
|  |         .align = 0, | ||||||
|  |         .kind = 0, | ||||||
|  |         .address = GetInteger(buffer), | ||||||
|  |     }; | ||||||
|  |     Nvidia::Devices::nvmap::IocAllocParams alloc_out_params{}; | ||||||
|  |     R_UNLESS(nvmap.IocAlloc(SerializeIoc(alloc_in_params), SerializeIoc(alloc_out_params)) == | ||||||
|  |                  Nvidia::NvResult::Success, | ||||||
|  |              VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result AllocateHandleForBuffer(u32* out_handle, Nvidia::Module& nvdrv, | ||||||
|  |                                Common::ProcessAddress buffer, u32 size) { | ||||||
|  |     // Get the nvmap device.
 | ||||||
|  |     auto nvmap_fd = nvdrv.Open("/dev/nvmap"); | ||||||
|  |     auto nvmap = nvdrv.GetDevice<Nvidia::Devices::nvmap>(nvmap_fd); | ||||||
|  |     ASSERT(nvmap != nullptr); | ||||||
|  | 
 | ||||||
|  |     // Create a handle.
 | ||||||
|  |     R_TRY(CreateNvMapHandle(out_handle, *nvmap, size)); | ||||||
|  | 
 | ||||||
|  |     // Ensure we maintain a clean state on failure.
 | ||||||
|  |     ON_RESULT_FAILURE { | ||||||
|  |         ASSERT(R_SUCCEEDED(FreeNvMapHandle(*nvmap, *out_handle))); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     // Assign the allocated memory to the handle.
 | ||||||
|  |     R_RETURN(AllocNvMapHandle(*nvmap, *out_handle, buffer, size)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | constexpr auto SharedBufferBlockLinearFormat = android::PixelFormat::Rgba8888; | ||||||
|  | constexpr u32 SharedBufferBlockLinearBpp = 4; | ||||||
|  | 
 | ||||||
|  | constexpr u32 SharedBufferBlockLinearWidth = 1280; | ||||||
|  | constexpr u32 SharedBufferBlockLinearHeight = 768; | ||||||
|  | constexpr u32 SharedBufferBlockLinearStride = | ||||||
|  |     SharedBufferBlockLinearWidth * SharedBufferBlockLinearBpp; | ||||||
|  | constexpr u32 SharedBufferNumSlots = 7; | ||||||
|  | 
 | ||||||
|  | constexpr u32 SharedBufferWidth = 1280; | ||||||
|  | constexpr u32 SharedBufferHeight = 720; | ||||||
|  | constexpr u32 SharedBufferAsync = false; | ||||||
|  | 
 | ||||||
|  | constexpr u32 SharedBufferSlotSize = | ||||||
|  |     SharedBufferBlockLinearWidth * SharedBufferBlockLinearHeight * SharedBufferBlockLinearBpp; | ||||||
|  | constexpr u32 SharedBufferSize = SharedBufferSlotSize * SharedBufferNumSlots; | ||||||
|  | 
 | ||||||
|  | constexpr SharedMemoryPoolLayout SharedBufferPoolLayout = [] { | ||||||
|  |     SharedMemoryPoolLayout layout{}; | ||||||
|  |     layout.num_slots = SharedBufferNumSlots; | ||||||
|  | 
 | ||||||
|  |     for (u32 i = 0; i < SharedBufferNumSlots; i++) { | ||||||
|  |         layout.slots[i].buffer_offset = i * SharedBufferSlotSize; | ||||||
|  |         layout.slots[i].size = SharedBufferSlotSize; | ||||||
|  |         layout.slots[i].width = SharedBufferWidth; | ||||||
|  |         layout.slots[i].height = SharedBufferHeight; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return layout; | ||||||
|  | }(); | ||||||
|  | 
 | ||||||
|  | void MakeGraphicBuffer(android::BufferQueueProducer& producer, u32 slot, u32 handle) { | ||||||
|  |     auto buffer = std::make_shared<android::GraphicBuffer>(); | ||||||
|  |     buffer->width = SharedBufferWidth; | ||||||
|  |     buffer->height = SharedBufferHeight; | ||||||
|  |     buffer->stride = SharedBufferBlockLinearStride; | ||||||
|  |     buffer->format = SharedBufferBlockLinearFormat; | ||||||
|  |     buffer->buffer_id = handle; | ||||||
|  |     buffer->offset = slot * SharedBufferSlotSize; | ||||||
|  |     ASSERT(producer.SetPreallocatedBuffer(slot, buffer) == android::Status::NoError); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | FbShareBufferManager::FbShareBufferManager(Core::System& system, Nvnflinger& flinger, | ||||||
|  |                                            std::shared_ptr<Nvidia::Module> nvdrv) | ||||||
|  |     : m_system(system), m_flinger(flinger), m_nvdrv(std::move(nvdrv)) {} | ||||||
|  | 
 | ||||||
|  | FbShareBufferManager::~FbShareBufferManager() = default; | ||||||
|  | 
 | ||||||
|  | Result FbShareBufferManager::Initialize(u64* out_buffer_id, u64* out_layer_id, u64 display_id) { | ||||||
|  |     std::scoped_lock lk{m_guard}; | ||||||
|  | 
 | ||||||
|  |     // Ensure we have not already created a buffer.
 | ||||||
|  |     R_UNLESS(m_buffer_id == 0, VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // Allocate memory and space for the shared buffer.
 | ||||||
|  |     Common::ProcessAddress map_address; | ||||||
|  |     R_TRY(AllocateIoForProcessAddressSpace(std::addressof(map_address), | ||||||
|  |                                            std::addressof(m_buffer_page_group), m_system, | ||||||
|  |                                            SharedBufferSize)); | ||||||
|  | 
 | ||||||
|  |     // Create an nvmap handle for the buffer and assign the memory to it.
 | ||||||
|  |     R_TRY(AllocateHandleForBuffer(std::addressof(m_buffer_nvmap_handle), *m_nvdrv, map_address, | ||||||
|  |                                   SharedBufferSize)); | ||||||
|  | 
 | ||||||
|  |     // Record the display id.
 | ||||||
|  |     m_display_id = display_id; | ||||||
|  | 
 | ||||||
|  |     // Create a layer for the display.
 | ||||||
|  |     m_layer_id = m_flinger.CreateLayer(m_display_id).value(); | ||||||
|  | 
 | ||||||
|  |     // Set up the buffer.
 | ||||||
|  |     m_buffer_id = m_next_buffer_id++; | ||||||
|  | 
 | ||||||
|  |     // Get the layer.
 | ||||||
|  |     VI::Layer* layer = m_flinger.FindLayer(m_display_id, m_layer_id); | ||||||
|  |     ASSERT(layer != nullptr); | ||||||
|  | 
 | ||||||
|  |     // Get the producer and set preallocated buffers.
 | ||||||
|  |     auto& producer = layer->GetBufferQueue(); | ||||||
|  |     MakeGraphicBuffer(producer, 0, m_buffer_nvmap_handle); | ||||||
|  |     MakeGraphicBuffer(producer, 1, m_buffer_nvmap_handle); | ||||||
|  | 
 | ||||||
|  |     // Assign outputs.
 | ||||||
|  |     *out_buffer_id = m_buffer_id; | ||||||
|  |     *out_layer_id = m_layer_id; | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result FbShareBufferManager::GetSharedBufferMemoryHandleId(u64* out_buffer_size, | ||||||
|  |                                                            s32* out_nvmap_handle, | ||||||
|  |                                                            SharedMemoryPoolLayout* out_pool_layout, | ||||||
|  |                                                            u64 buffer_id, | ||||||
|  |                                                            u64 applet_resource_user_id) { | ||||||
|  |     std::scoped_lock lk{m_guard}; | ||||||
|  | 
 | ||||||
|  |     R_UNLESS(m_buffer_id > 0, VI::ResultNotFound); | ||||||
|  |     R_UNLESS(buffer_id == m_buffer_id, VI::ResultNotFound); | ||||||
|  | 
 | ||||||
|  |     *out_pool_layout = SharedBufferPoolLayout; | ||||||
|  |     *out_buffer_size = SharedBufferSize; | ||||||
|  |     *out_nvmap_handle = m_buffer_nvmap_handle; | ||||||
|  | 
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result FbShareBufferManager::GetLayerFromId(VI::Layer** out_layer, u64 layer_id) { | ||||||
|  |     // Ensure the layer id is valid.
 | ||||||
|  |     R_UNLESS(m_layer_id > 0 && layer_id == m_layer_id, VI::ResultNotFound); | ||||||
|  | 
 | ||||||
|  |     // Get the layer.
 | ||||||
|  |     VI::Layer* layer = m_flinger.FindLayer(m_display_id, layer_id); | ||||||
|  |     R_UNLESS(layer != nullptr, VI::ResultNotFound); | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     *out_layer = layer; | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result FbShareBufferManager::AcquireSharedFrameBuffer(android::Fence* out_fence, | ||||||
|  |                                                       std::array<s32, 4>& out_slot_indexes, | ||||||
|  |                                                       s64* out_target_slot, u64 layer_id) { | ||||||
|  |     std::scoped_lock lk{m_guard}; | ||||||
|  | 
 | ||||||
|  |     // Get the layer.
 | ||||||
|  |     VI::Layer* layer; | ||||||
|  |     R_TRY(this->GetLayerFromId(std::addressof(layer), layer_id)); | ||||||
|  | 
 | ||||||
|  |     // Get the producer.
 | ||||||
|  |     auto& producer = layer->GetBufferQueue(); | ||||||
|  | 
 | ||||||
|  |     // Get the next buffer from the producer.
 | ||||||
|  |     s32 slot; | ||||||
|  |     R_UNLESS(producer.DequeueBuffer(std::addressof(slot), out_fence, SharedBufferAsync != 0, | ||||||
|  |                                     SharedBufferWidth, SharedBufferHeight, | ||||||
|  |                                     SharedBufferBlockLinearFormat, 0) == android::Status::NoError, | ||||||
|  |              VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // Assign remaining outputs.
 | ||||||
|  |     *out_target_slot = slot; | ||||||
|  |     out_slot_indexes = {0, 1, -1, -1}; | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result FbShareBufferManager::PresentSharedFrameBuffer(android::Fence fence, | ||||||
|  |                                                       Common::Rectangle<s32> crop_region, | ||||||
|  |                                                       u32 transform, s32 swap_interval, | ||||||
|  |                                                       u64 layer_id, s64 slot) { | ||||||
|  |     std::scoped_lock lk{m_guard}; | ||||||
|  | 
 | ||||||
|  |     // Get the layer.
 | ||||||
|  |     VI::Layer* layer; | ||||||
|  |     R_TRY(this->GetLayerFromId(std::addressof(layer), layer_id)); | ||||||
|  | 
 | ||||||
|  |     // Get the producer.
 | ||||||
|  |     auto& producer = layer->GetBufferQueue(); | ||||||
|  | 
 | ||||||
|  |     // Request to queue the buffer.
 | ||||||
|  |     std::shared_ptr<android::GraphicBuffer> buffer; | ||||||
|  |     R_UNLESS(producer.RequestBuffer(static_cast<s32>(slot), std::addressof(buffer)) == | ||||||
|  |                  android::Status::NoError, | ||||||
|  |              VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // Queue the buffer to the producer.
 | ||||||
|  |     android::QueueBufferInput input{}; | ||||||
|  |     android::QueueBufferOutput output{}; | ||||||
|  |     input.crop = crop_region; | ||||||
|  |     input.fence = fence; | ||||||
|  |     input.transform = static_cast<android::NativeWindowTransform>(transform); | ||||||
|  |     input.swap_interval = swap_interval; | ||||||
|  |     R_UNLESS(producer.QueueBuffer(static_cast<s32>(slot), input, std::addressof(output)) == | ||||||
|  |                  android::Status::NoError, | ||||||
|  |              VI::ResultOperationFailed); | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result FbShareBufferManager::GetSharedFrameBufferAcquirableEvent(Kernel::KReadableEvent** out_event, | ||||||
|  |                                                                  u64 layer_id) { | ||||||
|  |     std::scoped_lock lk{m_guard}; | ||||||
|  | 
 | ||||||
|  |     // Get the layer.
 | ||||||
|  |     VI::Layer* layer; | ||||||
|  |     R_TRY(this->GetLayerFromId(std::addressof(layer), layer_id)); | ||||||
|  | 
 | ||||||
|  |     // Get the producer.
 | ||||||
|  |     auto& producer = layer->GetBufferQueue(); | ||||||
|  | 
 | ||||||
|  |     // Set the event.
 | ||||||
|  |     *out_event = std::addressof(producer.GetNativeHandle()); | ||||||
|  | 
 | ||||||
|  |     // We succeeded.
 | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace Service::Nvnflinger
 | ||||||
							
								
								
									
										65
									
								
								src/core/hle/service/nvnflinger/fb_share_buffer_manager.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								src/core/hle/service/nvnflinger/fb_share_buffer_manager.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,65 @@ | ||||||
|  | // SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
 | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include "common/math_util.h" | ||||||
|  | #include "core/hle/service/nvnflinger/nvnflinger.h" | ||||||
|  | #include "core/hle/service/nvnflinger/ui/fence.h" | ||||||
|  | 
 | ||||||
|  | namespace Kernel { | ||||||
|  | class KPageGroup; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | namespace Service::Nvnflinger { | ||||||
|  | 
 | ||||||
|  | struct SharedMemorySlot { | ||||||
|  |     u64 buffer_offset; | ||||||
|  |     u64 size; | ||||||
|  |     s32 width; | ||||||
|  |     s32 height; | ||||||
|  | }; | ||||||
|  | static_assert(sizeof(SharedMemorySlot) == 0x18, "SharedMemorySlot has wrong size"); | ||||||
|  | 
 | ||||||
|  | struct SharedMemoryPoolLayout { | ||||||
|  |     s32 num_slots; | ||||||
|  |     std::array<SharedMemorySlot, 0x10> slots; | ||||||
|  | }; | ||||||
|  | static_assert(sizeof(SharedMemoryPoolLayout) == 0x188, "SharedMemoryPoolLayout has wrong size"); | ||||||
|  | 
 | ||||||
|  | class FbShareBufferManager final { | ||||||
|  | public: | ||||||
|  |     explicit FbShareBufferManager(Core::System& system, Nvnflinger& flinger, | ||||||
|  |                                   std::shared_ptr<Nvidia::Module> nvdrv); | ||||||
|  |     ~FbShareBufferManager(); | ||||||
|  | 
 | ||||||
|  |     Result Initialize(u64* out_buffer_id, u64* out_layer_handle, u64 display_id); | ||||||
|  |     Result GetSharedBufferMemoryHandleId(u64* out_buffer_size, s32* out_nvmap_handle, | ||||||
|  |                                          SharedMemoryPoolLayout* out_pool_layout, u64 buffer_id, | ||||||
|  |                                          u64 applet_resource_user_id); | ||||||
|  |     Result AcquireSharedFrameBuffer(android::Fence* out_fence, std::array<s32, 4>& out_slots, | ||||||
|  |                                     s64* out_target_slot, u64 layer_id); | ||||||
|  |     Result PresentSharedFrameBuffer(android::Fence fence, Common::Rectangle<s32> crop_region, | ||||||
|  |                                     u32 transform, s32 swap_interval, u64 layer_id, s64 slot); | ||||||
|  |     Result GetSharedFrameBufferAcquirableEvent(Kernel::KReadableEvent** out_event, u64 layer_id); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     Result GetLayerFromId(VI::Layer** out_layer, u64 layer_id); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     u64 m_next_buffer_id = 1; | ||||||
|  |     u64 m_display_id = 0; | ||||||
|  |     u64 m_buffer_id = 0; | ||||||
|  |     u64 m_layer_id = 0; | ||||||
|  |     u32 m_buffer_nvmap_handle = 0; | ||||||
|  |     SharedMemoryPoolLayout m_pool_layout = {}; | ||||||
|  | 
 | ||||||
|  |     std::unique_ptr<Kernel::KPageGroup> m_buffer_page_group; | ||||||
|  | 
 | ||||||
|  |     std::mutex m_guard; | ||||||
|  |     Core::System& m_system; | ||||||
|  |     Nvnflinger& m_flinger; | ||||||
|  |     std::shared_ptr<Nvidia::Module> m_nvdrv; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace Service::Nvnflinger
 | ||||||
|  | @ -19,6 +19,7 @@ class InputParcel; | ||||||
| #pragma pack(push, 1) | #pragma pack(push, 1) | ||||||
| struct QueueBufferInput final { | struct QueueBufferInput final { | ||||||
|     explicit QueueBufferInput(InputParcel& parcel); |     explicit QueueBufferInput(InputParcel& parcel); | ||||||
|  |     explicit QueueBufferInput() = default; | ||||||
| 
 | 
 | ||||||
|     void Deflate(s64* timestamp_, bool* is_auto_timestamp_, Common::Rectangle<s32>* crop_, |     void Deflate(s64* timestamp_, bool* is_auto_timestamp_, Common::Rectangle<s32>* crop_, | ||||||
|                  NativeWindowScalingMode* scaling_mode_, NativeWindowTransform* transform_, |                  NativeWindowScalingMode* scaling_mode_, NativeWindowTransform* transform_, | ||||||
|  | @ -34,7 +35,6 @@ struct QueueBufferInput final { | ||||||
|         *fence_ = fence; |         *fence_ = fence; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: |  | ||||||
|     s64 timestamp{}; |     s64 timestamp{}; | ||||||
|     s32 is_auto_timestamp{}; |     s32 is_auto_timestamp{}; | ||||||
|     Common::Rectangle<s32> crop{}; |     Common::Rectangle<s32> crop{}; | ||||||
|  |  | ||||||
|  | @ -17,6 +17,7 @@ | ||||||
| #include "core/hle/service/nvdrv/nvdrv.h" | #include "core/hle/service/nvdrv/nvdrv.h" | ||||||
| #include "core/hle/service/nvnflinger/buffer_item_consumer.h" | #include "core/hle/service/nvnflinger/buffer_item_consumer.h" | ||||||
| #include "core/hle/service/nvnflinger/buffer_queue_core.h" | #include "core/hle/service/nvnflinger/buffer_queue_core.h" | ||||||
|  | #include "core/hle/service/nvnflinger/fb_share_buffer_manager.h" | ||||||
| #include "core/hle/service/nvnflinger/hos_binder_driver_server.h" | #include "core/hle/service/nvnflinger/hos_binder_driver_server.h" | ||||||
| #include "core/hle/service/nvnflinger/nvnflinger.h" | #include "core/hle/service/nvnflinger/nvnflinger.h" | ||||||
| #include "core/hle/service/nvnflinger/ui/graphic_buffer.h" | #include "core/hle/service/nvnflinger/ui/graphic_buffer.h" | ||||||
|  | @ -331,4 +332,14 @@ s64 Nvnflinger::GetNextTicks() const { | ||||||
|     return static_cast<s64>(speed_scale * (1000000000.f / effective_fps)); |     return static_cast<s64>(speed_scale * (1000000000.f / effective_fps)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | FbShareBufferManager& Nvnflinger::GetSystemBufferManager() { | ||||||
|  |     const auto lock_guard = Lock(); | ||||||
|  | 
 | ||||||
|  |     if (!system_buffer_manager) { | ||||||
|  |         system_buffer_manager = std::make_unique<FbShareBufferManager>(system, *this, nvdrv); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return *system_buffer_manager; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| } // namespace Service::Nvnflinger
 | } // namespace Service::Nvnflinger
 | ||||||
|  |  | ||||||
|  | @ -45,6 +45,9 @@ class BufferQueueProducer; | ||||||
| 
 | 
 | ||||||
| namespace Service::Nvnflinger { | namespace Service::Nvnflinger { | ||||||
| 
 | 
 | ||||||
|  | class FbShareBufferManager; | ||||||
|  | class HosBinderDriverServer; | ||||||
|  | 
 | ||||||
| class Nvnflinger final { | class Nvnflinger final { | ||||||
| public: | public: | ||||||
|     explicit Nvnflinger(Core::System& system_, HosBinderDriverServer& hos_binder_driver_server_); |     explicit Nvnflinger(Core::System& system_, HosBinderDriverServer& hos_binder_driver_server_); | ||||||
|  | @ -90,12 +93,16 @@ public: | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] s64 GetNextTicks() const; |     [[nodiscard]] s64 GetNextTicks() const; | ||||||
| 
 | 
 | ||||||
|  |     FbShareBufferManager& GetSystemBufferManager(); | ||||||
|  | 
 | ||||||
| private: | private: | ||||||
|     struct Layer { |     struct Layer { | ||||||
|         std::unique_ptr<android::BufferQueueCore> core; |         std::unique_ptr<android::BufferQueueCore> core; | ||||||
|         std::unique_ptr<android::BufferQueueProducer> producer; |         std::unique_ptr<android::BufferQueueProducer> producer; | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  |     friend class FbShareBufferManager; | ||||||
|  | 
 | ||||||
| private: | private: | ||||||
|     [[nodiscard]] std::unique_lock<std::mutex> Lock() const { |     [[nodiscard]] std::unique_lock<std::mutex> Lock() const { | ||||||
|         return std::unique_lock{*guard}; |         return std::unique_lock{*guard}; | ||||||
|  | @ -140,6 +147,8 @@ private: | ||||||
|     std::shared_ptr<Core::Timing::EventType> multi_composition_event; |     std::shared_ptr<Core::Timing::EventType> multi_composition_event; | ||||||
|     std::shared_ptr<Core::Timing::EventType> single_composition_event; |     std::shared_ptr<Core::Timing::EventType> single_composition_event; | ||||||
| 
 | 
 | ||||||
|  |     std::unique_ptr<FbShareBufferManager> system_buffer_manager; | ||||||
|  | 
 | ||||||
|     std::shared_ptr<std::mutex> guard; |     std::shared_ptr<std::mutex> guard; | ||||||
| 
 | 
 | ||||||
|     Core::System& system; |     Core::System& system; | ||||||
|  |  | ||||||
|  | @ -20,6 +20,9 @@ public: | ||||||
|     static constexpr Fence NoFence() { |     static constexpr Fence NoFence() { | ||||||
|         Fence fence; |         Fence fence; | ||||||
|         fence.fences[0].id = -1; |         fence.fences[0].id = -1; | ||||||
|  |         fence.fences[1].id = -1; | ||||||
|  |         fence.fences[2].id = -1; | ||||||
|  |         fence.fences[3].id = -1; | ||||||
|         return fence; |         return fence; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -12,8 +12,7 @@ | ||||||
| 
 | 
 | ||||||
| namespace Service::android { | namespace Service::android { | ||||||
| 
 | 
 | ||||||
| class GraphicBuffer final { | struct GraphicBuffer final { | ||||||
| public: |  | ||||||
|     constexpr GraphicBuffer() = default; |     constexpr GraphicBuffer() = default; | ||||||
| 
 | 
 | ||||||
|     constexpr GraphicBuffer(u32 width_, u32 height_, PixelFormat format_, u32 usage_) |     constexpr GraphicBuffer(u32 width_, u32 height_, PixelFormat format_, u32 usage_) | ||||||
|  | @ -77,7 +76,6 @@ public: | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: |  | ||||||
|     u32 magic{}; |     u32 magic{}; | ||||||
|     s32 width{}; |     s32 width{}; | ||||||
|     s32 height{}; |     s32 height{}; | ||||||
|  |  | ||||||
|  | @ -20,9 +20,12 @@ | ||||||
| #include "core/hle/kernel/k_readable_event.h" | #include "core/hle/kernel/k_readable_event.h" | ||||||
| #include "core/hle/kernel/k_thread.h" | #include "core/hle/kernel/k_thread.h" | ||||||
| #include "core/hle/service/ipc_helpers.h" | #include "core/hle/service/ipc_helpers.h" | ||||||
|  | #include "core/hle/service/nvdrv/devices/nvmap.h" | ||||||
| #include "core/hle/service/nvdrv/nvdata.h" | #include "core/hle/service/nvdrv/nvdata.h" | ||||||
|  | #include "core/hle/service/nvdrv/nvdrv.h" | ||||||
| #include "core/hle/service/nvnflinger/binder.h" | #include "core/hle/service/nvnflinger/binder.h" | ||||||
| #include "core/hle/service/nvnflinger/buffer_queue_producer.h" | #include "core/hle/service/nvnflinger/buffer_queue_producer.h" | ||||||
|  | #include "core/hle/service/nvnflinger/fb_share_buffer_manager.h" | ||||||
| #include "core/hle/service/nvnflinger/hos_binder_driver_server.h" | #include "core/hle/service/nvnflinger/hos_binder_driver_server.h" | ||||||
| #include "core/hle/service/nvnflinger/nvnflinger.h" | #include "core/hle/service/nvnflinger/nvnflinger.h" | ||||||
| #include "core/hle/service/nvnflinger/parcel.h" | #include "core/hle/service/nvnflinger/parcel.h" | ||||||
|  | @ -131,8 +134,9 @@ private: | ||||||
| 
 | 
 | ||||||
| class ISystemDisplayService final : public ServiceFramework<ISystemDisplayService> { | class ISystemDisplayService final : public ServiceFramework<ISystemDisplayService> { | ||||||
| public: | public: | ||||||
|     explicit ISystemDisplayService(Core::System& system_) |     explicit ISystemDisplayService(Core::System& system_, Nvnflinger::Nvnflinger& nvnflinger_) | ||||||
|         : ServiceFramework{system_, "ISystemDisplayService"} { |         : ServiceFramework{system_, "ISystemDisplayService"}, nvnflinger{nvnflinger_} { | ||||||
|  |         // clang-format off
 | ||||||
|         static const FunctionInfo functions[] = { |         static const FunctionInfo functions[] = { | ||||||
|             {1200, nullptr, "GetZOrderCountMin"}, |             {1200, nullptr, "GetZOrderCountMin"}, | ||||||
|             {1202, nullptr, "GetZOrderCountMax"}, |             {1202, nullptr, "GetZOrderCountMax"}, | ||||||
|  | @ -170,22 +174,126 @@ public: | ||||||
|             {3217, nullptr, "SetDisplayCmuLuma"}, |             {3217, nullptr, "SetDisplayCmuLuma"}, | ||||||
|             {3218, nullptr, "SetDisplayCrcMode"}, |             {3218, nullptr, "SetDisplayCrcMode"}, | ||||||
|             {6013, nullptr, "GetLayerPresentationSubmissionTimestamps"}, |             {6013, nullptr, "GetLayerPresentationSubmissionTimestamps"}, | ||||||
|             {8225, nullptr, "GetSharedBufferMemoryHandleId"}, |             {8225, &ISystemDisplayService::GetSharedBufferMemoryHandleId, "GetSharedBufferMemoryHandleId"}, | ||||||
|             {8250, nullptr, "OpenSharedLayer"}, |             {8250, &ISystemDisplayService::OpenSharedLayer, "OpenSharedLayer"}, | ||||||
|             {8251, nullptr, "CloseSharedLayer"}, |             {8251, nullptr, "CloseSharedLayer"}, | ||||||
|             {8252, nullptr, "ConnectSharedLayer"}, |             {8252, &ISystemDisplayService::ConnectSharedLayer, "ConnectSharedLayer"}, | ||||||
|             {8253, nullptr, "DisconnectSharedLayer"}, |             {8253, nullptr, "DisconnectSharedLayer"}, | ||||||
|             {8254, nullptr, "AcquireSharedFrameBuffer"}, |             {8254, &ISystemDisplayService::AcquireSharedFrameBuffer, "AcquireSharedFrameBuffer"}, | ||||||
|             {8255, nullptr, "PresentSharedFrameBuffer"}, |             {8255, &ISystemDisplayService::PresentSharedFrameBuffer, "PresentSharedFrameBuffer"}, | ||||||
|             {8256, nullptr, "GetSharedFrameBufferAcquirableEvent"}, |             {8256, &ISystemDisplayService::GetSharedFrameBufferAcquirableEvent, "GetSharedFrameBufferAcquirableEvent"}, | ||||||
|             {8257, nullptr, "FillSharedFrameBufferColor"}, |             {8257, nullptr, "FillSharedFrameBufferColor"}, | ||||||
|             {8258, nullptr, "CancelSharedFrameBuffer"}, |             {8258, nullptr, "CancelSharedFrameBuffer"}, | ||||||
|             {9000, nullptr, "GetDp2hdmiController"}, |             {9000, nullptr, "GetDp2hdmiController"}, | ||||||
|         }; |         }; | ||||||
|  |         // clang-format on
 | ||||||
|         RegisterHandlers(functions); |         RegisterHandlers(functions); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  |     void GetSharedBufferMemoryHandleId(HLERequestContext& ctx) { | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         const u64 buffer_id = rp.PopRaw<u64>(); | ||||||
|  | 
 | ||||||
|  |         LOG_INFO(Service_VI, "called. buffer_id={:#x}", buffer_id); | ||||||
|  | 
 | ||||||
|  |         struct OutputParameters { | ||||||
|  |             s32 nvmap_handle; | ||||||
|  |             u64 size; | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         OutputParameters out{}; | ||||||
|  |         Nvnflinger::SharedMemoryPoolLayout layout{}; | ||||||
|  |         const auto result = nvnflinger.GetSystemBufferManager().GetSharedBufferMemoryHandleId( | ||||||
|  |             &out.size, &out.nvmap_handle, &layout, buffer_id, 0); | ||||||
|  | 
 | ||||||
|  |         ctx.WriteBuffer(&layout, sizeof(layout)); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 6}; | ||||||
|  |         rb.Push(result); | ||||||
|  |         rb.PushRaw(out); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void OpenSharedLayer(HLERequestContext& ctx) { | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         const u64 layer_id = rp.PopRaw<u64>(); | ||||||
|  | 
 | ||||||
|  |         LOG_INFO(Service_VI, "(STUBBED) called. layer_id={:#x}", layer_id); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void ConnectSharedLayer(HLERequestContext& ctx) { | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         const u64 layer_id = rp.PopRaw<u64>(); | ||||||
|  | 
 | ||||||
|  |         LOG_INFO(Service_VI, "(STUBBED) called. layer_id={:#x}", layer_id); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |         rb.Push(ResultSuccess); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void GetSharedFrameBufferAcquirableEvent(HLERequestContext& ctx) { | ||||||
|  |         LOG_DEBUG(Service_VI, "called"); | ||||||
|  | 
 | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         const u64 layer_id = rp.PopRaw<u64>(); | ||||||
|  | 
 | ||||||
|  |         Kernel::KReadableEvent* event{}; | ||||||
|  |         const auto result = nvnflinger.GetSystemBufferManager().GetSharedFrameBufferAcquirableEvent( | ||||||
|  |             &event, layer_id); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2, 1}; | ||||||
|  |         rb.Push(result); | ||||||
|  |         rb.PushCopyObjects(event); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void AcquireSharedFrameBuffer(HLERequestContext& ctx) { | ||||||
|  |         LOG_DEBUG(Service_VI, "called"); | ||||||
|  | 
 | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         const u64 layer_id = rp.PopRaw<u64>(); | ||||||
|  | 
 | ||||||
|  |         struct OutputParameters { | ||||||
|  |             android::Fence fence; | ||||||
|  |             std::array<s32, 4> slots; | ||||||
|  |             s64 target_slot; | ||||||
|  |         }; | ||||||
|  |         static_assert(sizeof(OutputParameters) == 0x40, "OutputParameters has wrong size"); | ||||||
|  | 
 | ||||||
|  |         OutputParameters out{}; | ||||||
|  |         const auto result = nvnflinger.GetSystemBufferManager().AcquireSharedFrameBuffer( | ||||||
|  |             &out.fence, out.slots, &out.target_slot, layer_id); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 18}; | ||||||
|  |         rb.Push(result); | ||||||
|  |         rb.PushRaw(out); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void PresentSharedFrameBuffer(HLERequestContext& ctx) { | ||||||
|  |         LOG_DEBUG(Service_VI, "called"); | ||||||
|  | 
 | ||||||
|  |         struct InputParameters { | ||||||
|  |             android::Fence fence; | ||||||
|  |             Common::Rectangle<s32> crop_region; | ||||||
|  |             u32 window_transform; | ||||||
|  |             s32 swap_interval; | ||||||
|  |             u64 layer_id; | ||||||
|  |             s64 surface_id; | ||||||
|  |         }; | ||||||
|  |         static_assert(sizeof(InputParameters) == 0x50, "InputParameters has wrong size"); | ||||||
|  | 
 | ||||||
|  |         IPC::RequestParser rp{ctx}; | ||||||
|  |         auto input = rp.PopRaw<InputParameters>(); | ||||||
|  | 
 | ||||||
|  |         const auto result = nvnflinger.GetSystemBufferManager().PresentSharedFrameBuffer( | ||||||
|  |             input.fence, input.crop_region, input.window_transform, input.swap_interval, | ||||||
|  |             input.layer_id, input.surface_id); | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  |         rb.Push(result); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void SetLayerZ(HLERequestContext& ctx) { |     void SetLayerZ(HLERequestContext& ctx) { | ||||||
|         IPC::RequestParser rp{ctx}; |         IPC::RequestParser rp{ctx}; | ||||||
|         const u64 layer_id = rp.Pop<u64>(); |         const u64 layer_id = rp.Pop<u64>(); | ||||||
|  | @ -228,6 +336,9 @@ private: | ||||||
|         rb.PushRaw<float>(60.0f); // This wouldn't seem to be correct for 30 fps games.
 |         rb.PushRaw<float>(60.0f); // This wouldn't seem to be correct for 30 fps games.
 | ||||||
|         rb.Push<u32>(0); |         rb.Push<u32>(0); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     Nvnflinger::Nvnflinger& nvnflinger; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class IManagerDisplayService final : public ServiceFramework<IManagerDisplayService> { | class IManagerDisplayService final : public ServiceFramework<IManagerDisplayService> { | ||||||
|  | @ -453,7 +564,7 @@ private: | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |         IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(ResultSuccess); | ||||||
|         rb.PushIpcInterface<ISystemDisplayService>(system); |         rb.PushIpcInterface<ISystemDisplayService>(system, nv_flinger); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void GetManagerDisplayService(HLERequestContext& ctx) { |     void GetManagerDisplayService(HLERequestContext& ctx) { | ||||||
|  |  | ||||||
|  | @ -1551,6 +1551,14 @@ void GMainWindow::ConnectMenuEvents() { | ||||||
|     // Tools
 |     // Tools
 | ||||||
|     connect_menu(ui->action_Rederive, std::bind(&GMainWindow::OnReinitializeKeys, this, |     connect_menu(ui->action_Rederive, std::bind(&GMainWindow::OnReinitializeKeys, this, | ||||||
|                                                 ReinitializeKeyBehavior::Warning)); |                                                 ReinitializeKeyBehavior::Warning)); | ||||||
|  |     connect_menu(ui->action_Load_Cabinet_Nickname_Owner, | ||||||
|  |                  [this]() { OnCabinet(Service::NFP::CabinetMode::StartNicknameAndOwnerSettings); }); | ||||||
|  |     connect_menu(ui->action_Load_Cabinet_Eraser, | ||||||
|  |                  [this]() { OnCabinet(Service::NFP::CabinetMode::StartGameDataEraser); }); | ||||||
|  |     connect_menu(ui->action_Load_Cabinet_Restorer, | ||||||
|  |                  [this]() { OnCabinet(Service::NFP::CabinetMode::StartRestorer); }); | ||||||
|  |     connect_menu(ui->action_Load_Cabinet_Formatter, | ||||||
|  |                  [this]() { OnCabinet(Service::NFP::CabinetMode::StartFormatter); }); | ||||||
|     connect_menu(ui->action_Load_Mii_Edit, &GMainWindow::OnMiiEdit); |     connect_menu(ui->action_Load_Mii_Edit, &GMainWindow::OnMiiEdit); | ||||||
|     connect_menu(ui->action_Capture_Screenshot, &GMainWindow::OnCaptureScreenshot); |     connect_menu(ui->action_Capture_Screenshot, &GMainWindow::OnCaptureScreenshot); | ||||||
| 
 | 
 | ||||||
|  | @ -1568,6 +1576,7 @@ void GMainWindow::ConnectMenuEvents() { | ||||||
| 
 | 
 | ||||||
| void GMainWindow::UpdateMenuState() { | void GMainWindow::UpdateMenuState() { | ||||||
|     const bool is_paused = emu_thread == nullptr || !emu_thread->IsRunning(); |     const bool is_paused = emu_thread == nullptr || !emu_thread->IsRunning(); | ||||||
|  |     const bool is_firmware_available = CheckFirmwarePresence(); | ||||||
| 
 | 
 | ||||||
|     const std::array running_actions{ |     const std::array running_actions{ | ||||||
|         ui->action_Stop, |         ui->action_Stop, | ||||||
|  | @ -1578,10 +1587,22 @@ void GMainWindow::UpdateMenuState() { | ||||||
|         ui->action_Pause, |         ui->action_Pause, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  |     const std::array applet_actions{ | ||||||
|  |         ui->action_Load_Cabinet_Nickname_Owner, | ||||||
|  |         ui->action_Load_Cabinet_Eraser, | ||||||
|  |         ui->action_Load_Cabinet_Restorer, | ||||||
|  |         ui->action_Load_Cabinet_Formatter, | ||||||
|  |         ui->action_Load_Mii_Edit, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     for (QAction* action : running_actions) { |     for (QAction* action : running_actions) { | ||||||
|         action->setEnabled(emulation_running); |         action->setEnabled(emulation_running); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     for (QAction* action : applet_actions) { | ||||||
|  |         action->setEnabled(is_firmware_available && !emulation_running); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     ui->action_Capture_Screenshot->setEnabled(emulation_running && !is_paused); |     ui->action_Capture_Screenshot->setEnabled(emulation_running && !is_paused); | ||||||
| 
 | 
 | ||||||
|     if (emulation_running && is_paused) { |     if (emulation_running && is_paused) { | ||||||
|  | @ -1591,8 +1612,6 @@ void GMainWindow::UpdateMenuState() { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     multiplayer_state->UpdateNotificationStatus(); |     multiplayer_state->UpdateNotificationStatus(); | ||||||
| 
 |  | ||||||
|     ui->action_Load_Mii_Edit->setEnabled(CheckFirmwarePresence()); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void GMainWindow::OnDisplayTitleBars(bool show) { | void GMainWindow::OnDisplayTitleBars(bool show) { | ||||||
|  | @ -2103,6 +2122,8 @@ void GMainWindow::OnEmulationStopped() { | ||||||
|     OnTasStateChanged(); |     OnTasStateChanged(); | ||||||
|     render_window->FinalizeCamera(); |     render_window->FinalizeCamera(); | ||||||
| 
 | 
 | ||||||
|  |     system->GetAppletManager().SetCurrentAppletId(Service::AM::Applets::AppletId::None); | ||||||
|  | 
 | ||||||
|     // Enable all controllers
 |     // Enable all controllers
 | ||||||
|     system->HIDCore().SetSupportedStyleTag({Core::HID::NpadStyleSet::All}); |     system->HIDCore().SetSupportedStyleTag({Core::HID::NpadStyleSet::All}); | ||||||
| 
 | 
 | ||||||
|  | @ -4134,6 +4155,30 @@ void GMainWindow::OnToggleStatusBar() { | ||||||
|     statusBar()->setVisible(ui->action_Show_Status_Bar->isChecked()); |     statusBar()->setVisible(ui->action_Show_Status_Bar->isChecked()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void GMainWindow::OnCabinet(Service::NFP::CabinetMode mode) { | ||||||
|  |     constexpr u64 CabinetId = 0x0100000000001002ull; | ||||||
|  |     auto bis_system = system->GetFileSystemController().GetSystemNANDContents(); | ||||||
|  |     if (!bis_system) { | ||||||
|  |         QMessageBox::warning(this, tr("No firmware available"), | ||||||
|  |                              tr("Please install the firmware to use the Cabinet applet.")); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     auto cabinet_nca = bis_system->GetEntry(CabinetId, FileSys::ContentRecordType::Program); | ||||||
|  |     if (!cabinet_nca) { | ||||||
|  |         QMessageBox::warning(this, tr("Cabinet Applet"), | ||||||
|  |                              tr("Cabinet applet is not available. Please reinstall firmware.")); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     system->GetAppletManager().SetCurrentAppletId(Service::AM::Applets::AppletId::Cabinet); | ||||||
|  |     system->GetAppletManager().SetCabinetMode(mode); | ||||||
|  | 
 | ||||||
|  |     const auto filename = QString::fromStdString(cabinet_nca->GetFullPath()); | ||||||
|  |     UISettings::values.roms_path = QFileInfo(filename).path(); | ||||||
|  |     BootGame(filename); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void GMainWindow::OnMiiEdit() { | void GMainWindow::OnMiiEdit() { | ||||||
|     constexpr u64 MiiEditId = 0x0100000000001009ull; |     constexpr u64 MiiEditId = 0x0100000000001009ull; | ||||||
|     auto bis_system = system->GetFileSystemController().GetSystemNANDContents(); |     auto bis_system = system->GetFileSystemController().GetSystemNANDContents(); | ||||||
|  | @ -4150,6 +4195,8 @@ void GMainWindow::OnMiiEdit() { | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     system->GetAppletManager().SetCurrentAppletId(Service::AM::Applets::AppletId::MiiEdit); | ||||||
|  | 
 | ||||||
|     const auto filename = QString::fromStdString((mii_applet_nca->GetFullPath())); |     const auto filename = QString::fromStdString((mii_applet_nca->GetFullPath())); | ||||||
|     UISettings::values.roms_path = QFileInfo(filename).path(); |     UISettings::values.roms_path = QFileInfo(filename).path(); | ||||||
|     BootGame(filename); |     BootGame(filename); | ||||||
|  |  | ||||||
|  | @ -102,6 +102,10 @@ namespace Service::NFC { | ||||||
| class NfcDevice; | class NfcDevice; | ||||||
| } // namespace Service::NFC
 | } // namespace Service::NFC
 | ||||||
| 
 | 
 | ||||||
|  | namespace Service::NFP { | ||||||
|  | enum class CabinetMode : u8; | ||||||
|  | } // namespace Service::NFP
 | ||||||
|  | 
 | ||||||
| namespace Ui { | namespace Ui { | ||||||
| class MainWindow; | class MainWindow; | ||||||
| } | } | ||||||
|  | @ -365,6 +369,7 @@ private slots: | ||||||
|     void ResetWindowSize720(); |     void ResetWindowSize720(); | ||||||
|     void ResetWindowSize900(); |     void ResetWindowSize900(); | ||||||
|     void ResetWindowSize1080(); |     void ResetWindowSize1080(); | ||||||
|  |     void OnCabinet(Service::NFP::CabinetMode mode); | ||||||
|     void OnMiiEdit(); |     void OnMiiEdit(); | ||||||
|     void OnCaptureScreenshot(); |     void OnCaptureScreenshot(); | ||||||
|     void OnReinitializeKeys(ReinitializeKeyBehavior behavior); |     void OnReinitializeKeys(ReinitializeKeyBehavior behavior); | ||||||
|  |  | ||||||
|  | @ -137,6 +137,15 @@ | ||||||
|     <property name="title"> |     <property name="title"> | ||||||
|      <string>&Tools</string> |      <string>&Tools</string> | ||||||
|     </property> |     </property> | ||||||
|  |     <widget class="QMenu" name="menu_cabinet_applet"> | ||||||
|  |      <property name="title"> | ||||||
|  |       <string>&Amiibo</string> | ||||||
|  |      </property> | ||||||
|  |      <addaction name="action_Load_Cabinet_Nickname_Owner"/> | ||||||
|  |      <addaction name="action_Load_Cabinet_Eraser"/> | ||||||
|  |      <addaction name="action_Load_Cabinet_Restorer"/> | ||||||
|  |      <addaction name="action_Load_Cabinet_Formatter"/> | ||||||
|  |     </widget> | ||||||
|     <widget class="QMenu" name="menuTAS"> |     <widget class="QMenu" name="menuTAS"> | ||||||
|      <property name="title"> |      <property name="title"> | ||||||
|       <string>&TAS</string> |       <string>&TAS</string> | ||||||
|  | @ -150,6 +159,7 @@ | ||||||
|     <addaction name="action_Rederive"/> |     <addaction name="action_Rederive"/> | ||||||
|     <addaction name="action_Verify_installed_contents"/> |     <addaction name="action_Verify_installed_contents"/> | ||||||
|     <addaction name="separator"/> |     <addaction name="separator"/> | ||||||
|  |     <addaction name="menu_cabinet_applet"/> | ||||||
|     <addaction name="action_Load_Mii_Edit"/> |     <addaction name="action_Load_Mii_Edit"/> | ||||||
|     <addaction name="separator"/> |     <addaction name="separator"/> | ||||||
|     <addaction name="action_Capture_Screenshot"/> |     <addaction name="action_Capture_Screenshot"/> | ||||||
|  | @ -370,6 +380,26 @@ | ||||||
|     <string>&Capture Screenshot</string> |     <string>&Capture Screenshot</string> | ||||||
|    </property> |    </property> | ||||||
|   </action> |   </action> | ||||||
|  |   <action name="action_Load_Cabinet_Nickname_Owner"> | ||||||
|  |    <property name="text"> | ||||||
|  |     <string>&Set Nickname and Owner</string> | ||||||
|  |    </property> | ||||||
|  |   </action> | ||||||
|  |   <action name="action_Load_Cabinet_Eraser"> | ||||||
|  |    <property name="text"> | ||||||
|  |     <string>&Delete Game Data</string> | ||||||
|  |    </property> | ||||||
|  |   </action> | ||||||
|  |   <action name="action_Load_Cabinet_Restorer"> | ||||||
|  |    <property name="text"> | ||||||
|  |     <string>&Restore Amiibo</string> | ||||||
|  |    </property> | ||||||
|  |   </action> | ||||||
|  |   <action name="action_Load_Cabinet_Formatter"> | ||||||
|  |    <property name="text"> | ||||||
|  |     <string>&Format Amiibo</string> | ||||||
|  |    </property> | ||||||
|  |   </action> | ||||||
|   <action name="action_Load_Mii_Edit"> |   <action name="action_Load_Mii_Edit"> | ||||||
|     <property name="text"> |     <property name="text"> | ||||||
|       <string>Open &Mii Editor</string> |       <string>Open &Mii Editor</string> | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 liamwhite
						liamwhite