forked from eden-emu/eden
		
	
		
			
				
	
	
		
			2262 lines
		
	
	
	
		
			82 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			2262 lines
		
	
	
	
		
			82 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <array>
 | |
| #include <cinttypes>
 | |
| #include <cstring>
 | |
| #include "common/settings.h"
 | |
| #include "common/settings_enums.h"
 | |
| #include "core/core.h"
 | |
| #include "core/file_sys/control_metadata.h"
 | |
| #include "core/file_sys/patch_manager.h"
 | |
| #include "core/file_sys/registered_cache.h"
 | |
| #include "core/file_sys/savedata_factory.h"
 | |
| #include "core/hle/kernel/k_event.h"
 | |
| #include "core/hle/kernel/k_transfer_memory.h"
 | |
| #include "core/hle/result.h"
 | |
| #include "core/hle/service/acc/profile_manager.h"
 | |
| #include "core/hle/service/am/am.h"
 | |
| #include "core/hle/service/am/applet_ae.h"
 | |
| #include "core/hle/service/am/applet_oe.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_web_browser.h"
 | |
| #include "core/hle/service/am/applets/applets.h"
 | |
| #include "core/hle/service/am/idle.h"
 | |
| #include "core/hle/service/am/omm.h"
 | |
| #include "core/hle/service/am/spsm.h"
 | |
| #include "core/hle/service/apm/apm_controller.h"
 | |
| #include "core/hle/service/apm/apm_interface.h"
 | |
| #include "core/hle/service/bcat/backend/backend.h"
 | |
| #include "core/hle/service/caps/caps.h"
 | |
| #include "core/hle/service/filesystem/filesystem.h"
 | |
| #include "core/hle/service/ipc_helpers.h"
 | |
| #include "core/hle/service/ns/ns.h"
 | |
| #include "core/hle/service/nvnflinger/nvnflinger.h"
 | |
| #include "core/hle/service/pm/pm.h"
 | |
| #include "core/hle/service/server_manager.h"
 | |
| #include "core/hle/service/sm/sm.h"
 | |
| #include "core/hle/service/vi/vi.h"
 | |
| #include "core/memory.h"
 | |
| 
 | |
| namespace Service::AM {
 | |
| 
 | |
| constexpr Result ResultNoDataInChannel{ErrorModule::AM, 2};
 | |
| constexpr Result ResultNoMessages{ErrorModule::AM, 3};
 | |
| constexpr Result ResultInvalidOffset{ErrorModule::AM, 503};
 | |
| 
 | |
| enum class LaunchParameterKind : u32 {
 | |
|     UserChannel = 1,
 | |
|     AccountPreselectedUser = 2,
 | |
| };
 | |
| 
 | |
| constexpr u32 LAUNCH_PARAMETER_ACCOUNT_PRESELECTED_USER_MAGIC = 0xC79497CA;
 | |
| 
 | |
| struct LaunchParameterAccountPreselectedUser {
 | |
|     u32_le magic;
 | |
|     u32_le is_account_selected;
 | |
|     Common::UUID current_user;
 | |
|     INSERT_PADDING_BYTES(0x70);
 | |
| };
 | |
| static_assert(sizeof(LaunchParameterAccountPreselectedUser) == 0x88);
 | |
| 
 | |
| IWindowController::IWindowController(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IWindowController"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, nullptr, "CreateWindow"},
 | |
|         {1, &IWindowController::GetAppletResourceUserId, "GetAppletResourceUserId"},
 | |
|         {2, nullptr, "GetAppletResourceUserIdOfCallerApplet"},
 | |
|         {10, &IWindowController::AcquireForegroundRights, "AcquireForegroundRights"},
 | |
|         {11, nullptr, "ReleaseForegroundRights"},
 | |
|         {12, nullptr, "RejectToChangeIntoBackground"},
 | |
|         {20, nullptr, "SetAppletWindowVisibility"},
 | |
|         {21, nullptr, "SetAppletGpuTimeSlice"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IWindowController::~IWindowController() = default;
 | |
| 
 | |
| void IWindowController::GetAppletResourceUserId(HLERequestContext& ctx) {
 | |
|     const u64 process_id = system.ApplicationProcess()->GetProcessId();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called. Process ID=0x{:016X}", process_id);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u64>(process_id);
 | |
| }
 | |
| 
 | |
| void IWindowController::AcquireForegroundRights(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| IAudioController::IAudioController(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IAudioController"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, &IAudioController::SetExpectedMasterVolume, "SetExpectedMasterVolume"},
 | |
|         {1, &IAudioController::GetMainAppletExpectedMasterVolume, "GetMainAppletExpectedMasterVolume"},
 | |
|         {2, &IAudioController::GetLibraryAppletExpectedMasterVolume, "GetLibraryAppletExpectedMasterVolume"},
 | |
|         {3, &IAudioController::ChangeMainAppletMasterVolume, "ChangeMainAppletMasterVolume"},
 | |
|         {4, &IAudioController::SetTransparentAudioRate, "SetTransparentVolumeRate"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IAudioController::~IAudioController() = default;
 | |
| 
 | |
| void IAudioController::SetExpectedMasterVolume(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const float main_applet_volume_tmp = rp.Pop<float>();
 | |
|     const float library_applet_volume_tmp = rp.Pop<float>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called. main_applet_volume={}, library_applet_volume={}",
 | |
|               main_applet_volume_tmp, library_applet_volume_tmp);
 | |
| 
 | |
|     // Ensure the volume values remain within the 0-100% range
 | |
|     main_applet_volume = std::clamp(main_applet_volume_tmp, min_allowed_volume, max_allowed_volume);
 | |
|     library_applet_volume =
 | |
|         std::clamp(library_applet_volume_tmp, min_allowed_volume, max_allowed_volume);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IAudioController::GetMainAppletExpectedMasterVolume(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called. main_applet_volume={}", main_applet_volume);
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(main_applet_volume);
 | |
| }
 | |
| 
 | |
| void IAudioController::GetLibraryAppletExpectedMasterVolume(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called. library_applet_volume={}", library_applet_volume);
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(library_applet_volume);
 | |
| }
 | |
| 
 | |
| void IAudioController::ChangeMainAppletMasterVolume(HLERequestContext& ctx) {
 | |
|     struct Parameters {
 | |
|         float volume;
 | |
|         s64 fade_time_ns;
 | |
|     };
 | |
|     static_assert(sizeof(Parameters) == 16);
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto parameters = rp.PopRaw<Parameters>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called. volume={}, fade_time_ns={}", parameters.volume,
 | |
|               parameters.fade_time_ns);
 | |
| 
 | |
|     main_applet_volume = std::clamp(parameters.volume, min_allowed_volume, max_allowed_volume);
 | |
|     fade_time_ns = std::chrono::nanoseconds{parameters.fade_time_ns};
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IAudioController::SetTransparentAudioRate(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const float transparent_volume_rate_tmp = rp.Pop<float>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called. transparent_volume_rate={}", transparent_volume_rate_tmp);
 | |
| 
 | |
|     // Clamp volume range to 0-100%.
 | |
|     transparent_volume_rate =
 | |
|         std::clamp(transparent_volume_rate_tmp, min_allowed_volume, max_allowed_volume);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| IDisplayController::IDisplayController(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IDisplayController"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, nullptr, "GetLastForegroundCaptureImage"},
 | |
|         {1, nullptr, "UpdateLastForegroundCaptureImage"},
 | |
|         {2, nullptr, "GetLastApplicationCaptureImage"},
 | |
|         {3, nullptr, "GetCallerAppletCaptureImage"},
 | |
|         {4, nullptr, "UpdateCallerAppletCaptureImage"},
 | |
|         {5, nullptr, "GetLastForegroundCaptureImageEx"},
 | |
|         {6, nullptr, "GetLastApplicationCaptureImageEx"},
 | |
|         {7, nullptr, "GetCallerAppletCaptureImageEx"},
 | |
|         {8, &IDisplayController::TakeScreenShotOfOwnLayer, "TakeScreenShotOfOwnLayer"},
 | |
|         {9, nullptr, "CopyBetweenCaptureBuffers"},
 | |
|         {10, nullptr, "AcquireLastApplicationCaptureBuffer"},
 | |
|         {11, nullptr, "ReleaseLastApplicationCaptureBuffer"},
 | |
|         {12, nullptr, "AcquireLastForegroundCaptureBuffer"},
 | |
|         {13, nullptr, "ReleaseLastForegroundCaptureBuffer"},
 | |
|         {14, nullptr, "AcquireCallerAppletCaptureBuffer"},
 | |
|         {15, nullptr, "ReleaseCallerAppletCaptureBuffer"},
 | |
|         {16, nullptr, "AcquireLastApplicationCaptureBufferEx"},
 | |
|         {17, nullptr, "AcquireLastForegroundCaptureBufferEx"},
 | |
|         {18, nullptr, "AcquireCallerAppletCaptureBufferEx"},
 | |
|         {20, nullptr, "ClearCaptureBuffer"},
 | |
|         {21, nullptr, "ClearAppletTransitionBuffer"},
 | |
|         {22, nullptr, "AcquireLastApplicationCaptureSharedBuffer"},
 | |
|         {23, nullptr, "ReleaseLastApplicationCaptureSharedBuffer"},
 | |
|         {24, nullptr, "AcquireLastForegroundCaptureSharedBuffer"},
 | |
|         {25, nullptr, "ReleaseLastForegroundCaptureSharedBuffer"},
 | |
|         {26, nullptr, "AcquireCallerAppletCaptureSharedBuffer"},
 | |
|         {27, nullptr, "ReleaseCallerAppletCaptureSharedBuffer"},
 | |
|         {28, nullptr, "TakeScreenShotOfOwnLayerEx"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IDisplayController::~IDisplayController() = default;
 | |
| 
 | |
| void IDisplayController::TakeScreenShotOfOwnLayer(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| IDebugFunctions::IDebugFunctions(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IDebugFunctions"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, nullptr, "NotifyMessageToHomeMenuForDebug"},
 | |
|         {1, nullptr, "OpenMainApplication"},
 | |
|         {10, nullptr, "PerformSystemButtonPressing"},
 | |
|         {20, nullptr, "InvalidateTransitionLayer"},
 | |
|         {30, nullptr, "RequestLaunchApplicationWithUserAndArgumentForDebug"},
 | |
|         {31, nullptr, "RequestLaunchApplicationByApplicationLaunchInfoForDebug"},
 | |
|         {40, nullptr, "GetAppletResourceUsageInfo"},
 | |
|         {50, nullptr, "AddSystemProgramIdAndAppletIdForDebug"},
 | |
|         {51, nullptr, "AddOperationConfirmedLibraryAppletIdForDebug"},
 | |
|         {100, nullptr, "SetCpuBoostModeForApplet"},
 | |
|         {101, nullptr, "CancelCpuBoostModeForApplet"},
 | |
|         {110, nullptr, "PushToAppletBoundChannelForDebug"},
 | |
|         {111, nullptr, "TryPopFromAppletBoundChannelForDebug"},
 | |
|         {120, nullptr, "AlarmSettingNotificationEnableAppEventReserve"},
 | |
|         {121, nullptr, "AlarmSettingNotificationDisableAppEventReserve"},
 | |
|         {122, nullptr, "AlarmSettingNotificationPushAppEventNotify"},
 | |
|         {130, nullptr, "FriendInvitationSetApplicationParameter"},
 | |
|         {131, nullptr, "FriendInvitationClearApplicationParameter"},
 | |
|         {132, nullptr, "FriendInvitationPushApplicationParameter"},
 | |
|         {140, nullptr, "RestrictPowerOperationForSecureLaunchModeForDebug"},
 | |
|         {200, nullptr, "CreateFloatingLibraryAppletAccepterForDebug"},
 | |
|         {300, nullptr, "TerminateAllRunningApplicationsForDebug"},
 | |
|         {900, nullptr, "GetGrcProcessLaunchedSystemEvent"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IDebugFunctions::~IDebugFunctions() = default;
 | |
| 
 | |
| ISelfController::ISelfController(Core::System& system_, Nvnflinger::Nvnflinger& nvnflinger_)
 | |
|     : ServiceFramework{system_, "ISelfController"}, nvnflinger{nvnflinger_},
 | |
|       service_context{system, "ISelfController"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, &ISelfController::Exit, "Exit"},
 | |
|         {1, &ISelfController::LockExit, "LockExit"},
 | |
|         {2, &ISelfController::UnlockExit, "UnlockExit"},
 | |
|         {3, &ISelfController::EnterFatalSection, "EnterFatalSection"},
 | |
|         {4, &ISelfController::LeaveFatalSection, "LeaveFatalSection"},
 | |
|         {9, &ISelfController::GetLibraryAppletLaunchableEvent, "GetLibraryAppletLaunchableEvent"},
 | |
|         {10, &ISelfController::SetScreenShotPermission, "SetScreenShotPermission"},
 | |
|         {11, &ISelfController::SetOperationModeChangedNotification, "SetOperationModeChangedNotification"},
 | |
|         {12, &ISelfController::SetPerformanceModeChangedNotification, "SetPerformanceModeChangedNotification"},
 | |
|         {13, &ISelfController::SetFocusHandlingMode, "SetFocusHandlingMode"},
 | |
|         {14, &ISelfController::SetRestartMessageEnabled, "SetRestartMessageEnabled"},
 | |
|         {15, nullptr, "SetScreenShotAppletIdentityInfo"},
 | |
|         {16, &ISelfController::SetOutOfFocusSuspendingEnabled, "SetOutOfFocusSuspendingEnabled"},
 | |
|         {17, nullptr, "SetControllerFirmwareUpdateSection"},
 | |
|         {18, nullptr, "SetRequiresCaptureButtonShortPressedMessage"},
 | |
|         {19, &ISelfController::SetAlbumImageOrientation, "SetAlbumImageOrientation"},
 | |
|         {20, nullptr, "SetDesirableKeyboardLayout"},
 | |
|         {21, nullptr, "GetScreenShotProgramId"},
 | |
|         {40, &ISelfController::CreateManagedDisplayLayer, "CreateManagedDisplayLayer"},
 | |
|         {41, nullptr, "IsSystemBufferSharingEnabled"},
 | |
|         {42, nullptr, "GetSystemSharedLayerHandle"},
 | |
|         {43, nullptr, "GetSystemSharedBufferHandle"},
 | |
|         {44, &ISelfController::CreateManagedDisplaySeparableLayer, "CreateManagedDisplaySeparableLayer"},
 | |
|         {45, nullptr, "SetManagedDisplayLayerSeparationMode"},
 | |
|         {46, nullptr, "SetRecordingLayerCompositionEnabled"},
 | |
|         {50, &ISelfController::SetHandlesRequestToDisplay, "SetHandlesRequestToDisplay"},
 | |
|         {51, nullptr, "ApproveToDisplay"},
 | |
|         {60, nullptr, "OverrideAutoSleepTimeAndDimmingTime"},
 | |
|         {61, nullptr, "SetMediaPlaybackState"},
 | |
|         {62, &ISelfController::SetIdleTimeDetectionExtension, "SetIdleTimeDetectionExtension"},
 | |
|         {63, &ISelfController::GetIdleTimeDetectionExtension, "GetIdleTimeDetectionExtension"},
 | |
|         {64, nullptr, "SetInputDetectionSourceSet"},
 | |
|         {65, &ISelfController::ReportUserIsActive, "ReportUserIsActive"},
 | |
|         {66, nullptr, "GetCurrentIlluminance"},
 | |
|         {67, nullptr, "IsIlluminanceAvailable"},
 | |
|         {68, &ISelfController::SetAutoSleepDisabled, "SetAutoSleepDisabled"},
 | |
|         {69, &ISelfController::IsAutoSleepDisabled, "IsAutoSleepDisabled"},
 | |
|         {70, nullptr, "ReportMultimediaError"},
 | |
|         {71, nullptr, "GetCurrentIlluminanceEx"},
 | |
|         {72, nullptr, "SetInputDetectionPolicy"},
 | |
|         {80, nullptr, "SetWirelessPriorityMode"},
 | |
|         {90, &ISelfController::GetAccumulatedSuspendedTickValue, "GetAccumulatedSuspendedTickValue"},
 | |
|         {91, &ISelfController::GetAccumulatedSuspendedTickChangedEvent, "GetAccumulatedSuspendedTickChangedEvent"},
 | |
|         {100, &ISelfController::SetAlbumImageTakenNotificationEnabled, "SetAlbumImageTakenNotificationEnabled"},
 | |
|         {110, nullptr, "SetApplicationAlbumUserData"},
 | |
|         {120, &ISelfController::SaveCurrentScreenshot, "SaveCurrentScreenshot"},
 | |
|         {130, &ISelfController::SetRecordVolumeMuted, "SetRecordVolumeMuted"},
 | |
|         {1000, nullptr, "GetDebugStorageChannel"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| 
 | |
|     launchable_event = service_context.CreateEvent("ISelfController:LaunchableEvent");
 | |
| 
 | |
|     // This event is created by AM on the first time GetAccumulatedSuspendedTickChangedEvent() is
 | |
|     // called. Yuzu can just create it unconditionally, since it doesn't need to support multiple
 | |
|     // ISelfControllers. The event is signaled on creation, and on transition from suspended -> not
 | |
|     // suspended if the event has previously been created by a call to
 | |
|     // GetAccumulatedSuspendedTickChangedEvent.
 | |
| 
 | |
|     accumulated_suspended_tick_changed_event =
 | |
|         service_context.CreateEvent("ISelfController:AccumulatedSuspendedTickChangedEvent");
 | |
|     accumulated_suspended_tick_changed_event->Signal();
 | |
| }
 | |
| 
 | |
| ISelfController::~ISelfController() {
 | |
|     service_context.CloseEvent(launchable_event);
 | |
|     service_context.CloseEvent(accumulated_suspended_tick_changed_event);
 | |
| }
 | |
| 
 | |
| void ISelfController::Exit(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| 
 | |
|     system.Exit();
 | |
| }
 | |
| 
 | |
| void ISelfController::LockExit(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     system.SetExitLocked(true);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::UnlockExit(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     system.SetExitLocked(false);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| 
 | |
|     if (system.GetExitRequested()) {
 | |
|         system.Exit();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ISelfController::EnterFatalSection(HLERequestContext& ctx) {
 | |
|     ++num_fatal_sections_entered;
 | |
|     LOG_DEBUG(Service_AM, "called. Num fatal sections entered: {}", num_fatal_sections_entered);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::LeaveFatalSection(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called.");
 | |
| 
 | |
|     // Entry and exit of fatal sections must be balanced.
 | |
|     if (num_fatal_sections_entered == 0) {
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(Result{ErrorModule::AM, 512});
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     --num_fatal_sections_entered;
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::GetLibraryAppletLaunchableEvent(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     launchable_event->Signal();
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(launchable_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| void ISelfController::SetScreenShotPermission(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto permission = rp.PopEnum<ScreenshotPermission>();
 | |
|     LOG_DEBUG(Service_AM, "called, permission={}", permission);
 | |
| 
 | |
|     screenshot_permission = permission;
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetOperationModeChangedNotification(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     bool flag = rp.Pop<bool>();
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called flag={}", flag);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetPerformanceModeChangedNotification(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     bool flag = rp.Pop<bool>();
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called flag={}", flag);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetFocusHandlingMode(HLERequestContext& ctx) {
 | |
|     // Takes 3 input u8s with each field located immediately after the previous
 | |
|     // u8, these are bool flags. No output.
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     struct FocusHandlingModeParams {
 | |
|         u8 unknown0;
 | |
|         u8 unknown1;
 | |
|         u8 unknown2;
 | |
|     };
 | |
|     const auto flags = rp.PopRaw<FocusHandlingModeParams>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called. unknown0={}, unknown1={}, unknown2={}",
 | |
|                 flags.unknown0, flags.unknown1, flags.unknown2);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetRestartMessageEnabled(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetOutOfFocusSuspendingEnabled(HLERequestContext& ctx) {
 | |
|     // Takes 3 input u8s with each field located immediately after the previous
 | |
|     // u8, these are bool flags. No output.
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     bool enabled = rp.Pop<bool>();
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called enabled={}", enabled);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetAlbumImageOrientation(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::CreateManagedDisplayLayer(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     // TODO(Subv): Find out how AM determines the display to use, for now just
 | |
|     // create the layer in the Default display.
 | |
|     const auto display_id = nvnflinger.OpenDisplay("Default");
 | |
|     const auto layer_id = nvnflinger.CreateLayer(*display_id);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(*layer_id);
 | |
| }
 | |
| 
 | |
| void ISelfController::CreateManagedDisplaySeparableLayer(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     // TODO(Subv): Find out how AM determines the display to use, for now just
 | |
|     // create the layer in the Default display.
 | |
|     // This calls nn::vi::CreateRecordingLayer() which creates another layer.
 | |
|     // Currently we do not support more than 1 layer per display, output 1 layer id for now.
 | |
|     // Outputting 1 layer id instead of the expected 2 has not been observed to cause any adverse
 | |
|     // side effects.
 | |
|     // TODO: Support multiple layers
 | |
|     const auto display_id = nvnflinger.OpenDisplay("Default");
 | |
|     const auto layer_id = nvnflinger.CreateLayer(*display_id);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(*layer_id);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetHandlesRequestToDisplay(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetIdleTimeDetectionExtension(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     idle_time_detection_extension = rp.Pop<u32>();
 | |
|     LOG_DEBUG(Service_AM, "(STUBBED) called idle_time_detection_extension={}",
 | |
|               idle_time_detection_extension);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::GetIdleTimeDetectionExtension(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u32>(idle_time_detection_extension);
 | |
| }
 | |
| 
 | |
| void ISelfController::ReportUserIsActive(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetAutoSleepDisabled(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     is_auto_sleep_disabled = rp.Pop<bool>();
 | |
| 
 | |
|     // On the system itself, if the previous state of is_auto_sleep_disabled
 | |
|     // differed from the current value passed in, it'd signify the internal
 | |
|     // window manager to update (and also increment some statistics like update counts)
 | |
|     //
 | |
|     // It'd also indicate this change to an idle handling context.
 | |
|     //
 | |
|     // However, given we're emulating this behavior, most of this can be ignored
 | |
|     // and it's sufficient to simply set the member variable for querying via
 | |
|     // IsAutoSleepDisabled().
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called. is_auto_sleep_disabled={}", is_auto_sleep_disabled);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::IsAutoSleepDisabled(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called.");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(is_auto_sleep_disabled);
 | |
| }
 | |
| 
 | |
| void ISelfController::GetAccumulatedSuspendedTickValue(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called.");
 | |
| 
 | |
|     // This command returns the total number of system ticks since ISelfController creation
 | |
|     // where the game was suspended. Since Yuzu doesn't implement game suspension, this command
 | |
|     // can just always return 0 ticks.
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u64>(0);
 | |
| }
 | |
| 
 | |
| void ISelfController::GetAccumulatedSuspendedTickChangedEvent(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called.");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(accumulated_suspended_tick_changed_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| void ISelfController::SetAlbumImageTakenNotificationEnabled(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     // This service call sets an internal flag whether a notification is shown when an image is
 | |
|     // captured. Currently we do not support capturing images via the capture button, so this can be
 | |
|     // stubbed for now.
 | |
|     const bool album_image_taken_notification_enabled = rp.Pop<bool>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called. album_image_taken_notification_enabled={}",
 | |
|                 album_image_taken_notification_enabled);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SaveCurrentScreenshot(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const auto album_report_option = rp.PopEnum<Capture::AlbumReportOption>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called. album_report_option={}", album_report_option);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ISelfController::SetRecordVolumeMuted(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const auto is_record_volume_muted = rp.Pop<bool>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called. is_record_volume_muted={}", is_record_volume_muted);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| AppletMessageQueue::AppletMessageQueue(Core::System& system)
 | |
|     : service_context{system, "AppletMessageQueue"} {
 | |
|     on_new_message = service_context.CreateEvent("AMMessageQueue:OnMessageReceived");
 | |
|     on_operation_mode_changed = service_context.CreateEvent("AMMessageQueue:OperationModeChanged");
 | |
| }
 | |
| 
 | |
| AppletMessageQueue::~AppletMessageQueue() {
 | |
|     service_context.CloseEvent(on_new_message);
 | |
|     service_context.CloseEvent(on_operation_mode_changed);
 | |
| }
 | |
| 
 | |
| Kernel::KReadableEvent& AppletMessageQueue::GetMessageReceiveEvent() {
 | |
|     return on_new_message->GetReadableEvent();
 | |
| }
 | |
| 
 | |
| Kernel::KReadableEvent& AppletMessageQueue::GetOperationModeChangedEvent() {
 | |
|     return on_operation_mode_changed->GetReadableEvent();
 | |
| }
 | |
| 
 | |
| void AppletMessageQueue::PushMessage(AppletMessage msg) {
 | |
|     messages.push(msg);
 | |
|     on_new_message->Signal();
 | |
| }
 | |
| 
 | |
| AppletMessageQueue::AppletMessage AppletMessageQueue::PopMessage() {
 | |
|     if (messages.empty()) {
 | |
|         on_new_message->Clear();
 | |
|         return AppletMessage::None;
 | |
|     }
 | |
|     auto msg = messages.front();
 | |
|     messages.pop();
 | |
|     if (messages.empty()) {
 | |
|         on_new_message->Clear();
 | |
|     }
 | |
|     return msg;
 | |
| }
 | |
| 
 | |
| std::size_t AppletMessageQueue::GetMessageCount() const {
 | |
|     return messages.size();
 | |
| }
 | |
| 
 | |
| void AppletMessageQueue::RequestExit() {
 | |
|     PushMessage(AppletMessage::Exit);
 | |
| }
 | |
| 
 | |
| void AppletMessageQueue::RequestResume() {
 | |
|     PushMessage(AppletMessage::Resume);
 | |
| }
 | |
| 
 | |
| void AppletMessageQueue::FocusStateChanged() {
 | |
|     PushMessage(AppletMessage::FocusStateChanged);
 | |
| }
 | |
| 
 | |
| void AppletMessageQueue::OperationModeChanged() {
 | |
|     PushMessage(AppletMessage::OperationModeChanged);
 | |
|     PushMessage(AppletMessage::PerformanceModeChanged);
 | |
|     on_operation_mode_changed->Signal();
 | |
| }
 | |
| 
 | |
| ICommonStateGetter::ICommonStateGetter(Core::System& system_,
 | |
|                                        std::shared_ptr<AppletMessageQueue> msg_queue_)
 | |
|     : ServiceFramework{system_, "ICommonStateGetter"}, msg_queue{std::move(msg_queue_)} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, &ICommonStateGetter::GetEventHandle, "GetEventHandle"},
 | |
|         {1, &ICommonStateGetter::ReceiveMessage, "ReceiveMessage"},
 | |
|         {2, nullptr, "GetThisAppletKind"},
 | |
|         {3, nullptr, "AllowToEnterSleep"},
 | |
|         {4, nullptr, "DisallowToEnterSleep"},
 | |
|         {5, &ICommonStateGetter::GetOperationMode, "GetOperationMode"},
 | |
|         {6, &ICommonStateGetter::GetPerformanceMode, "GetPerformanceMode"},
 | |
|         {7, nullptr, "GetCradleStatus"},
 | |
|         {8, &ICommonStateGetter::GetBootMode, "GetBootMode"},
 | |
|         {9, &ICommonStateGetter::GetCurrentFocusState, "GetCurrentFocusState"},
 | |
|         {10, nullptr, "RequestToAcquireSleepLock"},
 | |
|         {11, nullptr, "ReleaseSleepLock"},
 | |
|         {12, nullptr, "ReleaseSleepLockTransiently"},
 | |
|         {13, nullptr, "GetAcquiredSleepLockEvent"},
 | |
|         {14, nullptr, "GetWakeupCount"},
 | |
|         {20, nullptr, "PushToGeneralChannel"},
 | |
|         {30, nullptr, "GetHomeButtonReaderLockAccessor"},
 | |
|         {31, nullptr, "GetReaderLockAccessorEx"},
 | |
|         {32, nullptr, "GetWriterLockAccessorEx"},
 | |
|         {40, nullptr, "GetCradleFwVersion"},
 | |
|         {50, &ICommonStateGetter::IsVrModeEnabled, "IsVrModeEnabled"},
 | |
|         {51, &ICommonStateGetter::SetVrModeEnabled, "SetVrModeEnabled"},
 | |
|         {52, &ICommonStateGetter::SetLcdBacklighOffEnabled, "SetLcdBacklighOffEnabled"},
 | |
|         {53, &ICommonStateGetter::BeginVrModeEx, "BeginVrModeEx"},
 | |
|         {54, &ICommonStateGetter::EndVrModeEx, "EndVrModeEx"},
 | |
|         {55, nullptr, "IsInControllerFirmwareUpdateSection"},
 | |
|         {59, nullptr, "SetVrPositionForDebug"},
 | |
|         {60, &ICommonStateGetter::GetDefaultDisplayResolution, "GetDefaultDisplayResolution"},
 | |
|         {61, &ICommonStateGetter::GetDefaultDisplayResolutionChangeEvent, "GetDefaultDisplayResolutionChangeEvent"},
 | |
|         {62, nullptr, "GetHdcpAuthenticationState"},
 | |
|         {63, nullptr, "GetHdcpAuthenticationStateChangeEvent"},
 | |
|         {64, nullptr, "SetTvPowerStateMatchingMode"},
 | |
|         {65, nullptr, "GetApplicationIdByContentActionName"},
 | |
|         {66, &ICommonStateGetter::SetCpuBoostMode, "SetCpuBoostMode"},
 | |
|         {67, nullptr, "CancelCpuBoostMode"},
 | |
|         {68, nullptr, "GetBuiltInDisplayType"},
 | |
|         {80, &ICommonStateGetter::PerformSystemButtonPressingIfInFocus, "PerformSystemButtonPressingIfInFocus"},
 | |
|         {90, nullptr, "SetPerformanceConfigurationChangedNotification"},
 | |
|         {91, nullptr, "GetCurrentPerformanceConfiguration"},
 | |
|         {100, nullptr, "SetHandlingHomeButtonShortPressedEnabled"},
 | |
|         {110, nullptr, "OpenMyGpuErrorHandler"},
 | |
|         {120, nullptr, "GetAppletLaunchedHistory"},
 | |
|         {200, nullptr, "GetOperationModeSystemInfo"},
 | |
|         {300, &ICommonStateGetter::GetSettingsPlatformRegion, "GetSettingsPlatformRegion"},
 | |
|         {400, nullptr, "ActivateMigrationService"},
 | |
|         {401, nullptr, "DeactivateMigrationService"},
 | |
|         {500, nullptr, "DisableSleepTillShutdown"},
 | |
|         {501, nullptr, "SuppressDisablingSleepTemporarily"},
 | |
|         {502, nullptr, "IsSleepEnabled"},
 | |
|         {503, nullptr, "IsDisablingSleepSuppressed"},
 | |
|         {900, &ICommonStateGetter::SetRequestExitToLibraryAppletAtExecuteNextProgramEnabled, "SetRequestExitToLibraryAppletAtExecuteNextProgramEnabled"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| 
 | |
|     // Configure applets to be in foreground state
 | |
|     msg_queue->PushMessage(AppletMessageQueue::AppletMessage::FocusStateChanged);
 | |
|     msg_queue->PushMessage(AppletMessageQueue::AppletMessage::ChangeIntoForeground);
 | |
| }
 | |
| 
 | |
| ICommonStateGetter::~ICommonStateGetter() = default;
 | |
| 
 | |
| void ICommonStateGetter::GetBootMode(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u8>(static_cast<u8>(Service::PM::SystemBootMode::Normal)); // Normal boot mode
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetEventHandle(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(msg_queue->GetMessageReceiveEvent());
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::ReceiveMessage(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     const auto message = msg_queue->PopMessage();
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
| 
 | |
|     if (message == AppletMessageQueue::AppletMessage::None) {
 | |
|         LOG_ERROR(Service_AM, "Message queue is empty");
 | |
|         rb.Push(AM::ResultNoMessages);
 | |
|         rb.PushEnum<AppletMessageQueue::AppletMessage>(message);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushEnum<AppletMessageQueue::AppletMessage>(message);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetCurrentFocusState(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(static_cast<u8>(FocusState::InFocus));
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::IsVrModeEnabled(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(vr_mode_state);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::SetVrModeEnabled(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     vr_mode_state = rp.Pop<bool>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "VR Mode is {}", vr_mode_state ? "on" : "off");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::SetLcdBacklighOffEnabled(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto is_lcd_backlight_off_enabled = rp.Pop<bool>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called. is_lcd_backlight_off_enabled={}",
 | |
|                 is_lcd_backlight_off_enabled);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::BeginVrModeEx(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::EndVrModeEx(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetDefaultDisplayResolutionChangeEvent(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(msg_queue->GetOperationModeChangedEvent());
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetDefaultDisplayResolution(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
| 
 | |
|     if (Settings::IsDockedMode()) {
 | |
|         rb.Push(static_cast<u32>(Service::VI::DisplayResolution::DockedWidth));
 | |
|         rb.Push(static_cast<u32>(Service::VI::DisplayResolution::DockedHeight));
 | |
|     } else {
 | |
|         rb.Push(static_cast<u32>(Service::VI::DisplayResolution::UndockedWidth));
 | |
|         rb.Push(static_cast<u32>(Service::VI::DisplayResolution::UndockedHeight));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::SetCpuBoostMode(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called, forwarding to APM:SYS");
 | |
| 
 | |
|     const auto& sm = system.ServiceManager();
 | |
|     const auto apm_sys = sm.GetService<APM::APM_Sys>("apm:sys");
 | |
|     ASSERT(apm_sys != nullptr);
 | |
| 
 | |
|     apm_sys->SetCpuBoostMode(ctx);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::PerformSystemButtonPressingIfInFocus(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto system_button{rp.PopEnum<SystemButtonType>()};
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called, system_button={}", system_button);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetSettingsPlatformRegion(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushEnum(SysPlatformRegion::Global);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::SetRequestExitToLibraryAppletAtExecuteNextProgramEnabled(
 | |
|     HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| IStorageImpl::~IStorageImpl() = default;
 | |
| 
 | |
| class StorageDataImpl final : public IStorageImpl {
 | |
| public:
 | |
|     explicit StorageDataImpl(std::vector<u8>&& buffer_) : buffer{std::move(buffer_)} {}
 | |
| 
 | |
|     std::vector<u8>& GetData() override {
 | |
|         return buffer;
 | |
|     }
 | |
| 
 | |
|     const std::vector<u8>& GetData() const override {
 | |
|         return buffer;
 | |
|     }
 | |
| 
 | |
|     std::size_t GetSize() const override {
 | |
|         return buffer.size();
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     std::vector<u8> buffer;
 | |
| };
 | |
| 
 | |
| IStorage::IStorage(Core::System& system_, std::vector<u8>&& buffer)
 | |
|     : ServiceFramework{system_, "IStorage"}, impl{std::make_shared<StorageDataImpl>(
 | |
|                                                  std::move(buffer))} {
 | |
|     Register();
 | |
| }
 | |
| 
 | |
| void IStorage::Register() {
 | |
|     // clang-format off
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &IStorage::Open, "Open"},
 | |
|             {1, nullptr, "OpenTransferStorage"},
 | |
|         };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IStorage::~IStorage() = default;
 | |
| 
 | |
| void IStorage::Open(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
| 
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<IStorageAccessor>(system, *this);
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetOperationMode(HLERequestContext& ctx) {
 | |
|     const bool use_docked_mode{Settings::IsDockedMode()};
 | |
|     LOG_DEBUG(Service_AM, "called, use_docked_mode={}", use_docked_mode);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(static_cast<u8>(use_docked_mode ? OperationMode::Docked : OperationMode::Handheld));
 | |
| }
 | |
| 
 | |
| void ICommonStateGetter::GetPerformanceMode(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushEnum(system.GetAPMController().GetCurrentPerformanceMode());
 | |
| }
 | |
| 
 | |
| class ILibraryAppletAccessor final : public ServiceFramework<ILibraryAppletAccessor> {
 | |
| public:
 | |
|     explicit ILibraryAppletAccessor(Core::System& system_, std::shared_ptr<Applets::Applet> applet_)
 | |
|         : ServiceFramework{system_, "ILibraryAppletAccessor"}, applet{std::move(applet_)} {
 | |
|         // clang-format off
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &ILibraryAppletAccessor::GetAppletStateChangedEvent, "GetAppletStateChangedEvent"},
 | |
|             {1, &ILibraryAppletAccessor::IsCompleted, "IsCompleted"},
 | |
|             {10, &ILibraryAppletAccessor::Start, "Start"},
 | |
|             {20, &ILibraryAppletAccessor::RequestExit, "RequestExit"},
 | |
|             {25, nullptr, "Terminate"},
 | |
|             {30, &ILibraryAppletAccessor::GetResult, "GetResult"},
 | |
|             {50, nullptr, "SetOutOfFocusApplicationSuspendingEnabled"},
 | |
|             {60, &ILibraryAppletAccessor::PresetLibraryAppletGpuTimeSliceZero, "PresetLibraryAppletGpuTimeSliceZero"},
 | |
|             {100, &ILibraryAppletAccessor::PushInData, "PushInData"},
 | |
|             {101, &ILibraryAppletAccessor::PopOutData, "PopOutData"},
 | |
|             {102, nullptr, "PushExtraStorage"},
 | |
|             {103, &ILibraryAppletAccessor::PushInteractiveInData, "PushInteractiveInData"},
 | |
|             {104, &ILibraryAppletAccessor::PopInteractiveOutData, "PopInteractiveOutData"},
 | |
|             {105, &ILibraryAppletAccessor::GetPopOutDataEvent, "GetPopOutDataEvent"},
 | |
|             {106, &ILibraryAppletAccessor::GetPopInteractiveOutDataEvent, "GetPopInteractiveOutDataEvent"},
 | |
|             {110, nullptr, "NeedsToExitProcess"},
 | |
|             {120, nullptr, "GetLibraryAppletInfo"},
 | |
|             {150, nullptr, "RequestForAppletToGetForeground"},
 | |
|             {160, &ILibraryAppletAccessor::GetIndirectLayerConsumerHandle, "GetIndirectLayerConsumerHandle"},
 | |
|         };
 | |
|         // clang-format on
 | |
| 
 | |
|         RegisterHandlers(functions);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void GetAppletStateChangedEvent(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.PushCopyObjects(applet->GetBroker().GetStateChangedEvent());
 | |
|     }
 | |
| 
 | |
|     void IsCompleted(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 3};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.Push<u32>(applet->TransactionComplete());
 | |
|     }
 | |
| 
 | |
|     void GetResult(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(applet->GetStatus());
 | |
|     }
 | |
| 
 | |
|     void PresetLibraryAppletGpuTimeSliceZero(HLERequestContext& ctx) {
 | |
|         LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultSuccess);
 | |
|     }
 | |
| 
 | |
|     void Start(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         ASSERT(applet != nullptr);
 | |
| 
 | |
|         applet->Initialize();
 | |
|         applet->Execute();
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultSuccess);
 | |
|     }
 | |
| 
 | |
|     void RequestExit(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         ASSERT(applet != nullptr);
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(applet->RequestExit());
 | |
|     }
 | |
| 
 | |
|     void PushInData(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::RequestParser rp{ctx};
 | |
|         applet->GetBroker().PushNormalDataFromGame(rp.PopIpcInterface<IStorage>().lock());
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultSuccess);
 | |
|     }
 | |
| 
 | |
|     void PopOutData(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         auto storage = applet->GetBroker().PopNormalDataToGame();
 | |
|         if (storage == nullptr) {
 | |
|             LOG_DEBUG(Service_AM,
 | |
|                       "storage is a nullptr. There is no data in the current normal channel");
 | |
|             IPC::ResponseBuilder rb{ctx, 2};
 | |
|             rb.Push(AM::ResultNoDataInChannel);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.PushIpcInterface<IStorage>(std::move(storage));
 | |
|     }
 | |
| 
 | |
|     void PushInteractiveInData(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::RequestParser rp{ctx};
 | |
|         applet->GetBroker().PushInteractiveDataFromGame(rp.PopIpcInterface<IStorage>().lock());
 | |
| 
 | |
|         ASSERT(applet->IsInitialized());
 | |
|         applet->ExecuteInteractive();
 | |
|         applet->Execute();
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultSuccess);
 | |
|     }
 | |
| 
 | |
|     void PopInteractiveOutData(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         auto storage = applet->GetBroker().PopInteractiveDataToGame();
 | |
|         if (storage == nullptr) {
 | |
|             LOG_DEBUG(Service_AM,
 | |
|                       "storage is a nullptr. There is no data in the current interactive channel");
 | |
|             IPC::ResponseBuilder rb{ctx, 2};
 | |
|             rb.Push(AM::ResultNoDataInChannel);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.PushIpcInterface<IStorage>(std::move(storage));
 | |
|     }
 | |
| 
 | |
|     void GetPopOutDataEvent(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.PushCopyObjects(applet->GetBroker().GetNormalDataEvent());
 | |
|     }
 | |
| 
 | |
|     void GetPopInteractiveOutDataEvent(HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.PushCopyObjects(applet->GetBroker().GetInteractiveDataEvent());
 | |
|     }
 | |
| 
 | |
|     void GetIndirectLayerConsumerHandle(HLERequestContext& ctx) {
 | |
|         LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|         // We require a non-zero handle to be valid. Using 0xdeadbeef allows us to trace if this is
 | |
|         // actually used anywhere
 | |
|         constexpr u64 handle = 0xdeadbeef;
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 4};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.Push(handle);
 | |
|     }
 | |
| 
 | |
|     std::shared_ptr<Applets::Applet> applet;
 | |
| };
 | |
| 
 | |
| IStorageAccessor::IStorageAccessor(Core::System& system_, IStorage& backing_)
 | |
|     : ServiceFramework{system_, "IStorageAccessor"}, backing{backing_} {
 | |
|     // clang-format off
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &IStorageAccessor::GetSize, "GetSize"},
 | |
|             {10, &IStorageAccessor::Write, "Write"},
 | |
|             {11, &IStorageAccessor::Read, "Read"},
 | |
|         };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IStorageAccessor::~IStorageAccessor() = default;
 | |
| 
 | |
| void IStorageAccessor::GetSize(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
| 
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(static_cast<u64>(backing.GetSize()));
 | |
| }
 | |
| 
 | |
| void IStorageAccessor::Write(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const u64 offset{rp.Pop<u64>()};
 | |
|     const auto data{ctx.ReadBuffer()};
 | |
|     const std::size_t size{std::min<u64>(data.size(), backing.GetSize() - offset)};
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called, offset={}, size={}", offset, size);
 | |
| 
 | |
|     if (offset > backing.GetSize()) {
 | |
|         LOG_ERROR(Service_AM,
 | |
|                   "offset is out of bounds, backing_buffer_sz={}, data_size={}, offset={}",
 | |
|                   backing.GetSize(), size, offset);
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(AM::ResultInvalidOffset);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     std::memcpy(backing.GetData().data() + offset, data.data(), size);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IStorageAccessor::Read(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const u64 offset{rp.Pop<u64>()};
 | |
|     const std::size_t size{std::min<u64>(ctx.GetWriteBufferSize(), backing.GetSize() - offset)};
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called, offset={}, size={}", offset, size);
 | |
| 
 | |
|     if (offset > backing.GetSize()) {
 | |
|         LOG_ERROR(Service_AM, "offset is out of bounds, backing_buffer_sz={}, size={}, offset={}",
 | |
|                   backing.GetSize(), size, offset);
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(AM::ResultInvalidOffset);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     ctx.WriteBuffer(backing.GetData().data() + offset, size);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| ILibraryAppletCreator::ILibraryAppletCreator(Core::System& system_)
 | |
|     : ServiceFramework{system_, "ILibraryAppletCreator"} {
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, &ILibraryAppletCreator::CreateLibraryApplet, "CreateLibraryApplet"},
 | |
|         {1, nullptr, "TerminateAllLibraryApplets"},
 | |
|         {2, nullptr, "AreAnyLibraryAppletsLeft"},
 | |
|         {10, &ILibraryAppletCreator::CreateStorage, "CreateStorage"},
 | |
|         {11, &ILibraryAppletCreator::CreateTransferMemoryStorage, "CreateTransferMemoryStorage"},
 | |
|         {12, &ILibraryAppletCreator::CreateHandleStorage, "CreateHandleStorage"},
 | |
|     };
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| ILibraryAppletCreator::~ILibraryAppletCreator() = default;
 | |
| 
 | |
| void ILibraryAppletCreator::CreateLibraryApplet(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const auto applet_id = rp.PopRaw<Applets::AppletId>();
 | |
|     const auto applet_mode = rp.PopRaw<Applets::LibraryAppletMode>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called with applet_id={:08X}, applet_mode={:08X}", applet_id,
 | |
|               applet_mode);
 | |
| 
 | |
|     const auto& applet_manager{system.GetAppletManager()};
 | |
|     const auto applet = applet_manager.GetApplet(applet_id, applet_mode);
 | |
| 
 | |
|     if (applet == nullptr) {
 | |
|         LOG_ERROR(Service_AM, "Applet doesn't exist! applet_id={}", applet_id);
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
| 
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<ILibraryAppletAccessor>(system, applet);
 | |
| }
 | |
| 
 | |
| void ILibraryAppletCreator::CreateStorage(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const s64 size{rp.Pop<s64>()};
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called, size={}", size);
 | |
| 
 | |
|     if (size <= 0) {
 | |
|         LOG_ERROR(Service_AM, "size is less than or equal to 0");
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     std::vector<u8> buffer(size);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<IStorage>(system, std::move(buffer));
 | |
| }
 | |
| 
 | |
| void ILibraryAppletCreator::CreateTransferMemoryStorage(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     struct Parameters {
 | |
|         u8 permissions;
 | |
|         s64 size;
 | |
|     };
 | |
| 
 | |
|     const auto parameters{rp.PopRaw<Parameters>()};
 | |
|     const auto handle{ctx.GetCopyHandle(0)};
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called, permissions={}, size={}, handle={:08X}", parameters.permissions,
 | |
|               parameters.size, handle);
 | |
| 
 | |
|     if (parameters.size <= 0) {
 | |
|         LOG_ERROR(Service_AM, "size is less than or equal to 0");
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     auto transfer_mem =
 | |
|         system.ApplicationProcess()->GetHandleTable().GetObject<Kernel::KTransferMemory>(handle);
 | |
| 
 | |
|     if (transfer_mem.IsNull()) {
 | |
|         LOG_ERROR(Service_AM, "transfer_mem is a nullptr for handle={:08X}", handle);
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     std::vector<u8> memory(transfer_mem->GetSize());
 | |
|     system.ApplicationMemory().ReadBlock(transfer_mem->GetSourceAddress(), memory.data(),
 | |
|                                          memory.size());
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<IStorage>(system, std::move(memory));
 | |
| }
 | |
| 
 | |
| void ILibraryAppletCreator::CreateHandleStorage(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
| 
 | |
|     const s64 size{rp.Pop<s64>()};
 | |
|     const auto handle{ctx.GetCopyHandle(0)};
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called, size={}, handle={:08X}", size, handle);
 | |
| 
 | |
|     if (size <= 0) {
 | |
|         LOG_ERROR(Service_AM, "size is less than or equal to 0");
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     auto transfer_mem =
 | |
|         system.ApplicationProcess()->GetHandleTable().GetObject<Kernel::KTransferMemory>(handle);
 | |
| 
 | |
|     if (transfer_mem.IsNull()) {
 | |
|         LOG_ERROR(Service_AM, "transfer_mem is a nullptr for handle={:08X}", handle);
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     std::vector<u8> memory(transfer_mem->GetSize());
 | |
|     system.ApplicationMemory().ReadBlock(transfer_mem->GetSourceAddress(), memory.data(),
 | |
|                                          memory.size());
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<IStorage>(system, std::move(memory));
 | |
| }
 | |
| 
 | |
| ILibraryAppletSelfAccessor::ILibraryAppletSelfAccessor(Core::System& system_)
 | |
|     : ServiceFramework{system_, "ILibraryAppletSelfAccessor"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, &ILibraryAppletSelfAccessor::PopInData, "PopInData"},
 | |
|         {1, &ILibraryAppletSelfAccessor::PushOutData, "PushOutData"},
 | |
|         {2, nullptr, "PopInteractiveInData"},
 | |
|         {3, nullptr, "PushInteractiveOutData"},
 | |
|         {5, nullptr, "GetPopInDataEvent"},
 | |
|         {6, nullptr, "GetPopInteractiveInDataEvent"},
 | |
|         {10, &ILibraryAppletSelfAccessor::ExitProcessAndReturn, "ExitProcessAndReturn"},
 | |
|         {11, &ILibraryAppletSelfAccessor::GetLibraryAppletInfo, "GetLibraryAppletInfo"},
 | |
|         {12, nullptr, "GetMainAppletIdentityInfo"},
 | |
|         {13, nullptr, "CanUseApplicationCore"},
 | |
|         {14, &ILibraryAppletSelfAccessor::GetCallerAppletIdentityInfo, "GetCallerAppletIdentityInfo"},
 | |
|         {15, nullptr, "GetMainAppletApplicationControlProperty"},
 | |
|         {16, nullptr, "GetMainAppletStorageId"},
 | |
|         {17, nullptr, "GetCallerAppletIdentityInfoStack"},
 | |
|         {18, nullptr, "GetNextReturnDestinationAppletIdentityInfo"},
 | |
|         {19, nullptr, "GetDesirableKeyboardLayout"},
 | |
|         {20, nullptr, "PopExtraStorage"},
 | |
|         {25, nullptr, "GetPopExtraStorageEvent"},
 | |
|         {30, nullptr, "UnpopInData"},
 | |
|         {31, nullptr, "UnpopExtraStorage"},
 | |
|         {40, nullptr, "GetIndirectLayerProducerHandle"},
 | |
|         {50, nullptr, "ReportVisibleError"},
 | |
|         {51, nullptr, "ReportVisibleErrorWithErrorContext"},
 | |
|         {60, nullptr, "GetMainAppletApplicationDesiredLanguage"},
 | |
|         {70, nullptr, "GetCurrentApplicationId"},
 | |
|         {80, nullptr, "RequestExitToSelf"},
 | |
|         {90, nullptr, "CreateApplicationAndPushAndRequestToLaunch"},
 | |
|         {100, nullptr, "CreateGameMovieTrimmer"},
 | |
|         {101, nullptr, "ReserveResourceForMovieOperation"},
 | |
|         {102, nullptr, "UnreserveResourceForMovieOperation"},
 | |
|         {110, nullptr, "GetMainAppletAvailableUsers"},
 | |
|         {120, nullptr, "GetLaunchStorageInfoForDebug"},
 | |
|         {130, nullptr, "GetGpuErrorDetectedSystemEvent"},
 | |
|         {140, nullptr, "SetApplicationMemoryReservation"},
 | |
|         {150, nullptr, "ShouldSetGpuTimeSliceManually"},
 | |
|     };
 | |
|     // clang-format on
 | |
|     RegisterHandlers(functions);
 | |
| 
 | |
|     PushInShowMiiEditData();
 | |
| }
 | |
| 
 | |
| ILibraryAppletSelfAccessor::~ILibraryAppletSelfAccessor() = default;
 | |
| void ILibraryAppletSelfAccessor::PopInData(HLERequestContext& ctx) {
 | |
|     LOG_INFO(Service_AM, "called");
 | |
| 
 | |
|     if (queue_data.empty()) {
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultNoDataInChannel);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     auto data = queue_data.front();
 | |
|     queue_data.pop_front();
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<IStorage>(system, std::move(data));
 | |
| }
 | |
| 
 | |
| void ILibraryAppletSelfAccessor::PushOutData(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ILibraryAppletSelfAccessor::ExitProcessAndReturn(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     system.Exit();
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void ILibraryAppletSelfAccessor::GetLibraryAppletInfo(HLERequestContext& ctx) {
 | |
|     struct LibraryAppletInfo {
 | |
|         Applets::AppletId applet_id;
 | |
|         Applets::LibraryAppletMode library_applet_mode;
 | |
|     };
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     const LibraryAppletInfo applet_info{
 | |
|         .applet_id = Applets::AppletId::MiiEdit,
 | |
|         .library_applet_mode = Applets::LibraryAppletMode::AllForeground,
 | |
|     };
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushRaw(applet_info);
 | |
| }
 | |
| 
 | |
| void ILibraryAppletSelfAccessor::GetCallerAppletIdentityInfo(HLERequestContext& ctx) {
 | |
|     struct AppletIdentityInfo {
 | |
|         Applets::AppletId applet_id;
 | |
|         INSERT_PADDING_BYTES(0x4);
 | |
|         u64 application_id;
 | |
|     };
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     const AppletIdentityInfo applet_info{
 | |
|         .applet_id = Applets::AppletId::QLaunch,
 | |
|         .application_id = 0x0100000000001000ull,
 | |
|     };
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 6};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushRaw(applet_info);
 | |
| }
 | |
| 
 | |
| void ILibraryAppletSelfAccessor::PushInShowMiiEditData() {
 | |
|     struct MiiEditV3 {
 | |
|         Applets::MiiEditAppletInputCommon common;
 | |
|         Applets::MiiEditAppletInputV3 input;
 | |
|     };
 | |
|     static_assert(sizeof(MiiEditV3) == 0x100, "MiiEditV3 has incorrect size.");
 | |
| 
 | |
|     MiiEditV3 mii_arguments{
 | |
|         .common =
 | |
|             {
 | |
|                 .version = Applets::MiiEditAppletVersion::Version3,
 | |
|                 .applet_mode = Applets::MiiEditAppletMode::ShowMiiEdit,
 | |
|             },
 | |
|         .input{},
 | |
|     };
 | |
| 
 | |
|     std::vector<u8> argument_data(sizeof(mii_arguments));
 | |
|     std::memcpy(argument_data.data(), &mii_arguments, sizeof(mii_arguments));
 | |
| 
 | |
|     queue_data.emplace_back(std::move(argument_data));
 | |
| }
 | |
| 
 | |
| IAppletCommonFunctions::IAppletCommonFunctions(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IAppletCommonFunctions"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, nullptr, "SetTerminateResult"},
 | |
|         {10, nullptr, "ReadThemeStorage"},
 | |
|         {11, nullptr, "WriteThemeStorage"},
 | |
|         {20, nullptr, "PushToAppletBoundChannel"},
 | |
|         {21, nullptr, "TryPopFromAppletBoundChannel"},
 | |
|         {40, nullptr, "GetDisplayLogicalResolution"},
 | |
|         {42, nullptr, "SetDisplayMagnification"},
 | |
|         {50, nullptr, "SetHomeButtonDoubleClickEnabled"},
 | |
|         {51, nullptr, "GetHomeButtonDoubleClickEnabled"},
 | |
|         {52, nullptr, "IsHomeButtonShortPressedBlocked"},
 | |
|         {60, nullptr, "IsVrModeCurtainRequired"},
 | |
|         {61, nullptr, "IsSleepRequiredByHighTemperature"},
 | |
|         {62, nullptr, "IsSleepRequiredByLowBattery"},
 | |
|         {70, &IAppletCommonFunctions::SetCpuBoostRequestPriority, "SetCpuBoostRequestPriority"},
 | |
|         {80, nullptr, "SetHandlingCaptureButtonShortPressedMessageEnabledForApplet"},
 | |
|         {81, nullptr, "SetHandlingCaptureButtonLongPressedMessageEnabledForApplet"},
 | |
|         {90, nullptr, "OpenNamedChannelAsParent"},
 | |
|         {91, nullptr, "OpenNamedChannelAsChild"},
 | |
|         {100, nullptr, "SetApplicationCoreUsageMode"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IAppletCommonFunctions::~IAppletCommonFunctions() = default;
 | |
| 
 | |
| void IAppletCommonFunctions::SetCpuBoostRequestPriority(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| IApplicationFunctions::IApplicationFunctions(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IApplicationFunctions"}, service_context{system,
 | |
|                                                                           "IApplicationFunctions"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {1, &IApplicationFunctions::PopLaunchParameter, "PopLaunchParameter"},
 | |
|         {10, nullptr, "CreateApplicationAndPushAndRequestToStart"},
 | |
|         {11, nullptr, "CreateApplicationAndPushAndRequestToStartForQuest"},
 | |
|         {12, nullptr, "CreateApplicationAndRequestToStart"},
 | |
|         {13, &IApplicationFunctions::CreateApplicationAndRequestToStartForQuest, "CreateApplicationAndRequestToStartForQuest"},
 | |
|         {14, nullptr, "CreateApplicationWithAttributeAndPushAndRequestToStartForQuest"},
 | |
|         {15, nullptr, "CreateApplicationWithAttributeAndRequestToStartForQuest"},
 | |
|         {20, &IApplicationFunctions::EnsureSaveData, "EnsureSaveData"},
 | |
|         {21, &IApplicationFunctions::GetDesiredLanguage, "GetDesiredLanguage"},
 | |
|         {22, &IApplicationFunctions::SetTerminateResult, "SetTerminateResult"},
 | |
|         {23, &IApplicationFunctions::GetDisplayVersion, "GetDisplayVersion"},
 | |
|         {24, nullptr, "GetLaunchStorageInfoForDebug"},
 | |
|         {25, &IApplicationFunctions::ExtendSaveData, "ExtendSaveData"},
 | |
|         {26, &IApplicationFunctions::GetSaveDataSize, "GetSaveDataSize"},
 | |
|         {27, &IApplicationFunctions::CreateCacheStorage, "CreateCacheStorage"},
 | |
|         {28, &IApplicationFunctions::GetSaveDataSizeMax, "GetSaveDataSizeMax"},
 | |
|         {29, nullptr, "GetCacheStorageMax"},
 | |
|         {30, &IApplicationFunctions::BeginBlockingHomeButtonShortAndLongPressed, "BeginBlockingHomeButtonShortAndLongPressed"},
 | |
|         {31, &IApplicationFunctions::EndBlockingHomeButtonShortAndLongPressed, "EndBlockingHomeButtonShortAndLongPressed"},
 | |
|         {32, &IApplicationFunctions::BeginBlockingHomeButton, "BeginBlockingHomeButton"},
 | |
|         {33, &IApplicationFunctions::EndBlockingHomeButton, "EndBlockingHomeButton"},
 | |
|         {34, nullptr, "SelectApplicationLicense"},
 | |
|         {35, nullptr, "GetDeviceSaveDataSizeMax"},
 | |
|         {36, nullptr, "GetLimitedApplicationLicense"},
 | |
|         {37, nullptr, "GetLimitedApplicationLicenseUpgradableEvent"},
 | |
|         {40, &IApplicationFunctions::NotifyRunning, "NotifyRunning"},
 | |
|         {50, &IApplicationFunctions::GetPseudoDeviceId, "GetPseudoDeviceId"},
 | |
|         {60, nullptr, "SetMediaPlaybackStateForApplication"},
 | |
|         {65, &IApplicationFunctions::IsGamePlayRecordingSupported, "IsGamePlayRecordingSupported"},
 | |
|         {66, &IApplicationFunctions::InitializeGamePlayRecording, "InitializeGamePlayRecording"},
 | |
|         {67, &IApplicationFunctions::SetGamePlayRecordingState, "SetGamePlayRecordingState"},
 | |
|         {68, nullptr, "RequestFlushGamePlayingMovieForDebug"},
 | |
|         {70, nullptr, "RequestToShutdown"},
 | |
|         {71, nullptr, "RequestToReboot"},
 | |
|         {72, nullptr, "RequestToSleep"},
 | |
|         {80, nullptr, "ExitAndRequestToShowThanksMessage"},
 | |
|         {90, &IApplicationFunctions::EnableApplicationCrashReport, "EnableApplicationCrashReport"},
 | |
|         {100, &IApplicationFunctions::InitializeApplicationCopyrightFrameBuffer, "InitializeApplicationCopyrightFrameBuffer"},
 | |
|         {101, &IApplicationFunctions::SetApplicationCopyrightImage, "SetApplicationCopyrightImage"},
 | |
|         {102, &IApplicationFunctions::SetApplicationCopyrightVisibility, "SetApplicationCopyrightVisibility"},
 | |
|         {110, &IApplicationFunctions::QueryApplicationPlayStatistics, "QueryApplicationPlayStatistics"},
 | |
|         {111, &IApplicationFunctions::QueryApplicationPlayStatisticsByUid, "QueryApplicationPlayStatisticsByUid"},
 | |
|         {120, &IApplicationFunctions::ExecuteProgram, "ExecuteProgram"},
 | |
|         {121, &IApplicationFunctions::ClearUserChannel, "ClearUserChannel"},
 | |
|         {122, &IApplicationFunctions::UnpopToUserChannel, "UnpopToUserChannel"},
 | |
|         {123, &IApplicationFunctions::GetPreviousProgramIndex, "GetPreviousProgramIndex"},
 | |
|         {124, nullptr, "EnableApplicationAllThreadDumpOnCrash"},
 | |
|         {130, &IApplicationFunctions::GetGpuErrorDetectedSystemEvent, "GetGpuErrorDetectedSystemEvent"},
 | |
|         {131, nullptr, "SetDelayTimeToAbortOnGpuError"},
 | |
|         {140, &IApplicationFunctions::GetFriendInvitationStorageChannelEvent, "GetFriendInvitationStorageChannelEvent"},
 | |
|         {141, &IApplicationFunctions::TryPopFromFriendInvitationStorageChannel, "TryPopFromFriendInvitationStorageChannel"},
 | |
|         {150, &IApplicationFunctions::GetNotificationStorageChannelEvent, "GetNotificationStorageChannelEvent"},
 | |
|         {151, nullptr, "TryPopFromNotificationStorageChannel"},
 | |
|         {160, &IApplicationFunctions::GetHealthWarningDisappearedSystemEvent, "GetHealthWarningDisappearedSystemEvent"},
 | |
|         {170, nullptr, "SetHdcpAuthenticationActivated"},
 | |
|         {180, nullptr, "GetLaunchRequiredVersion"},
 | |
|         {181, nullptr, "UpgradeLaunchRequiredVersion"},
 | |
|         {190, nullptr, "SendServerMaintenanceOverlayNotification"},
 | |
|         {200, nullptr, "GetLastApplicationExitReason"},
 | |
|         {500, nullptr, "StartContinuousRecordingFlushForDebug"},
 | |
|         {1000, nullptr, "CreateMovieMaker"},
 | |
|         {1001, &IApplicationFunctions::PrepareForJit, "PrepareForJit"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| 
 | |
|     gpu_error_detected_event =
 | |
|         service_context.CreateEvent("IApplicationFunctions:GpuErrorDetectedSystemEvent");
 | |
|     friend_invitation_storage_channel_event =
 | |
|         service_context.CreateEvent("IApplicationFunctions:FriendInvitationStorageChannelEvent");
 | |
|     notification_storage_channel_event =
 | |
|         service_context.CreateEvent("IApplicationFunctions:NotificationStorageChannelEvent");
 | |
|     health_warning_disappeared_system_event =
 | |
|         service_context.CreateEvent("IApplicationFunctions:HealthWarningDisappearedSystemEvent");
 | |
| }
 | |
| 
 | |
| IApplicationFunctions::~IApplicationFunctions() {
 | |
|     service_context.CloseEvent(gpu_error_detected_event);
 | |
|     service_context.CloseEvent(friend_invitation_storage_channel_event);
 | |
|     service_context.CloseEvent(notification_storage_channel_event);
 | |
|     service_context.CloseEvent(health_warning_disappeared_system_event);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::EnableApplicationCrashReport(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::InitializeApplicationCopyrightFrameBuffer(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::SetApplicationCopyrightImage(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::SetApplicationCopyrightVisibility(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto is_visible = rp.Pop<bool>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called, is_visible={}", is_visible);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::BeginBlockingHomeButtonShortAndLongPressed(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::EndBlockingHomeButtonShortAndLongPressed(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::BeginBlockingHomeButton(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::EndBlockingHomeButton(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::PopLaunchParameter(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto kind = rp.PopEnum<LaunchParameterKind>();
 | |
| 
 | |
|     LOG_INFO(Service_AM, "called, kind={:08X}", kind);
 | |
| 
 | |
|     if (kind == LaunchParameterKind::UserChannel) {
 | |
|         auto channel = system.GetUserChannel();
 | |
|         if (channel.empty()) {
 | |
|             LOG_ERROR(Service_AM, "Attempted to load launch parameter but none was found!");
 | |
|             IPC::ResponseBuilder rb{ctx, 2};
 | |
|             rb.Push(AM::ResultNoDataInChannel);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         auto data = channel.back();
 | |
|         channel.pop_back();
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
|         rb.PushIpcInterface<IStorage>(system, std::move(data));
 | |
|     } else if (kind == LaunchParameterKind::AccountPreselectedUser &&
 | |
|                !launch_popped_account_preselect) {
 | |
|         // TODO: Verify this is hw-accurate
 | |
|         LaunchParameterAccountPreselectedUser params{};
 | |
| 
 | |
|         params.magic = LAUNCH_PARAMETER_ACCOUNT_PRESELECTED_USER_MAGIC;
 | |
|         params.is_account_selected = 1;
 | |
| 
 | |
|         Account::ProfileManager profile_manager{};
 | |
|         const auto uuid = profile_manager.GetUser(static_cast<s32>(Settings::values.current_user));
 | |
|         ASSERT(uuid.has_value() && uuid->IsValid());
 | |
|         params.current_user = *uuid;
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|         rb.Push(ResultSuccess);
 | |
| 
 | |
|         std::vector<u8> buffer(sizeof(LaunchParameterAccountPreselectedUser));
 | |
|         std::memcpy(buffer.data(), ¶ms, buffer.size());
 | |
| 
 | |
|         rb.PushIpcInterface<IStorage>(system, std::move(buffer));
 | |
|         launch_popped_account_preselect = true;
 | |
|     } else {
 | |
|         LOG_ERROR(Service_AM, "Unknown launch parameter kind.");
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(AM::ResultNoDataInChannel);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::CreateApplicationAndRequestToStartForQuest(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::EnsureSaveData(HLERequestContext& ctx) {
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     u128 user_id = rp.PopRaw<u128>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called, uid={:016X}{:016X}", user_id[1], user_id[0]);
 | |
| 
 | |
|     FileSys::SaveDataAttribute attribute{};
 | |
|     attribute.title_id = system.GetApplicationProcessProgramID();
 | |
|     attribute.user_id = user_id;
 | |
|     attribute.type = FileSys::SaveDataType::SaveData;
 | |
| 
 | |
|     FileSys::VirtualDir save_data{};
 | |
|     const auto res = system.GetFileSystemController().CreateSaveData(
 | |
|         &save_data, FileSys::SaveDataSpaceId::NandUser, attribute);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(res);
 | |
|     rb.Push<u64>(0);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::SetTerminateResult(HLERequestContext& ctx) {
 | |
|     // Takes an input u32 Result, no output.
 | |
|     // For example, in some cases official apps use this with error 0x2A2 then
 | |
|     // uses svcBreak.
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     u32 result = rp.Pop<u32>();
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called, result=0x{:08X}", result);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetDisplayVersion(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     std::array<u8, 0x10> version_string{};
 | |
| 
 | |
|     const auto res = [this] {
 | |
|         const auto title_id = system.GetApplicationProcessProgramID();
 | |
| 
 | |
|         const FileSys::PatchManager pm{title_id, system.GetFileSystemController(),
 | |
|                                        system.GetContentProvider()};
 | |
|         auto metadata = pm.GetControlMetadata();
 | |
|         if (metadata.first != nullptr) {
 | |
|             return metadata;
 | |
|         }
 | |
| 
 | |
|         const FileSys::PatchManager pm_update{FileSys::GetUpdateTitleID(title_id),
 | |
|                                               system.GetFileSystemController(),
 | |
|                                               system.GetContentProvider()};
 | |
|         return pm_update.GetControlMetadata();
 | |
|     }();
 | |
| 
 | |
|     if (res.first != nullptr) {
 | |
|         const auto& version = res.first->GetVersionString();
 | |
|         std::copy(version.begin(), version.end(), version_string.begin());
 | |
|     } else {
 | |
|         static constexpr char default_version[]{"1.0.0"};
 | |
|         std::memcpy(version_string.data(), default_version, sizeof(default_version));
 | |
|     }
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 6};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushRaw(version_string);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetDesiredLanguage(HLERequestContext& ctx) {
 | |
|     // TODO(bunnei): This should be configurable
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     // Get supported languages from NACP, if possible
 | |
|     // Default to 0 (all languages supported)
 | |
|     u32 supported_languages = 0;
 | |
| 
 | |
|     const auto res = [this] {
 | |
|         const auto title_id = system.GetApplicationProcessProgramID();
 | |
| 
 | |
|         const FileSys::PatchManager pm{title_id, system.GetFileSystemController(),
 | |
|                                        system.GetContentProvider()};
 | |
|         auto metadata = pm.GetControlMetadata();
 | |
|         if (metadata.first != nullptr) {
 | |
|             return metadata;
 | |
|         }
 | |
| 
 | |
|         const FileSys::PatchManager pm_update{FileSys::GetUpdateTitleID(title_id),
 | |
|                                               system.GetFileSystemController(),
 | |
|                                               system.GetContentProvider()};
 | |
|         return pm_update.GetControlMetadata();
 | |
|     }();
 | |
| 
 | |
|     if (res.first != nullptr) {
 | |
|         supported_languages = res.first->GetSupportedLanguages();
 | |
|     }
 | |
| 
 | |
|     // Call IApplicationManagerInterface implementation.
 | |
|     auto& service_manager = system.ServiceManager();
 | |
|     auto ns_am2 = service_manager.GetService<NS::NS>("ns:am2");
 | |
|     auto app_man = ns_am2->GetApplicationManagerInterface();
 | |
| 
 | |
|     // Get desired application language
 | |
|     u8 desired_language{};
 | |
|     const auto res_lang =
 | |
|         app_man->GetApplicationDesiredLanguage(&desired_language, supported_languages);
 | |
|     if (res_lang != ResultSuccess) {
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(res_lang);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // Convert to settings language code.
 | |
|     u64 language_code{};
 | |
|     const auto res_code =
 | |
|         app_man->ConvertApplicationLanguageToLanguageCode(&language_code, desired_language);
 | |
|     if (res_code != ResultSuccess) {
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(res_code);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "got desired_language={:016X}", language_code);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(language_code);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::IsGamePlayRecordingSupported(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     constexpr bool gameplay_recording_supported = false;
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(gameplay_recording_supported);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::InitializeGamePlayRecording(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::SetGamePlayRecordingState(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::NotifyRunning(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u8>(0); // Unknown, seems to be ignored by official processes
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetPseudoDeviceId(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 6};
 | |
|     rb.Push(ResultSuccess);
 | |
| 
 | |
|     // Returns a 128-bit UUID
 | |
|     rb.Push<u64>(0);
 | |
|     rb.Push<u64>(0);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::ExtendSaveData(HLERequestContext& ctx) {
 | |
|     struct Parameters {
 | |
|         FileSys::SaveDataType type;
 | |
|         u128 user_id;
 | |
|         u64 new_normal_size;
 | |
|         u64 new_journal_size;
 | |
|     };
 | |
|     static_assert(sizeof(Parameters) == 40);
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto [type, user_id, new_normal_size, new_journal_size] = rp.PopRaw<Parameters>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM,
 | |
|               "called with type={:02X}, user_id={:016X}{:016X}, new_normal={:016X}, "
 | |
|               "new_journal={:016X}",
 | |
|               static_cast<u8>(type), user_id[1], user_id[0], new_normal_size, new_journal_size);
 | |
| 
 | |
|     system.GetFileSystemController().WriteSaveDataSize(
 | |
|         type, system.GetApplicationProcessProgramID(), user_id,
 | |
|         {new_normal_size, new_journal_size});
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 4};
 | |
|     rb.Push(ResultSuccess);
 | |
| 
 | |
|     // The following value is used upon failure to help the system recover.
 | |
|     // Since we always succeed, this should be 0.
 | |
|     rb.Push<u64>(0);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetSaveDataSize(HLERequestContext& ctx) {
 | |
|     struct Parameters {
 | |
|         FileSys::SaveDataType type;
 | |
|         u128 user_id;
 | |
|     };
 | |
|     static_assert(sizeof(Parameters) == 24);
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto [type, user_id] = rp.PopRaw<Parameters>();
 | |
| 
 | |
|     LOG_DEBUG(Service_AM, "called with type={:02X}, user_id={:016X}{:016X}", type, user_id[1],
 | |
|               user_id[0]);
 | |
| 
 | |
|     const auto size = system.GetFileSystemController().ReadSaveDataSize(
 | |
|         type, system.GetApplicationProcessProgramID(), user_id);
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 6};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(size.normal);
 | |
|     rb.Push(size.journal);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::CreateCacheStorage(HLERequestContext& ctx) {
 | |
|     struct InputParameters {
 | |
|         u16 index;
 | |
|         s64 size;
 | |
|         s64 journal_size;
 | |
|     };
 | |
|     static_assert(sizeof(InputParameters) == 24);
 | |
| 
 | |
|     struct OutputParameters {
 | |
|         u32 storage_target;
 | |
|         u64 required_size;
 | |
|     };
 | |
|     static_assert(sizeof(OutputParameters) == 16);
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto params = rp.PopRaw<InputParameters>();
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called with index={}, size={:#x}, journal_size={:#x}",
 | |
|                 params.index, params.size, params.journal_size);
 | |
| 
 | |
|     const OutputParameters resp{
 | |
|         .storage_target = 1,
 | |
|         .required_size = 0,
 | |
|     };
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 6};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushRaw(resp);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetSaveDataSizeMax(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     constexpr u64 size_max_normal = 0xFFFFFFF;
 | |
|     constexpr u64 size_max_journal = 0xFFFFFFF;
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 6};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push(size_max_normal);
 | |
|     rb.Push(size_max_journal);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::QueryApplicationPlayStatistics(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u32>(0);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::QueryApplicationPlayStatisticsByUid(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<u32>(0);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::ExecuteProgram(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     [[maybe_unused]] const auto unk_1 = rp.Pop<u32>();
 | |
|     [[maybe_unused]] const auto unk_2 = rp.Pop<u32>();
 | |
|     const auto program_index = rp.Pop<u64>();
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| 
 | |
|     system.ExecuteProgram(program_index);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::ClearUserChannel(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     system.GetUserChannel().clear();
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::UnpopToUserChannel(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::RequestParser rp{ctx};
 | |
|     const auto storage = rp.PopIpcInterface<IStorage>().lock();
 | |
|     if (storage) {
 | |
|         system.GetUserChannel().push_back(storage->GetData());
 | |
|     }
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetPreviousProgramIndex(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.Push<s32>(previous_program_index);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetGpuErrorDetectedSystemEvent(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(gpu_error_detected_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetFriendInvitationStorageChannelEvent(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(friend_invitation_storage_channel_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::TryPopFromFriendInvitationStorageChannel(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(AM::ResultNoDataInChannel);
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetNotificationStorageChannelEvent(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(notification_storage_channel_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::GetHealthWarningDisappearedSystemEvent(HLERequestContext& ctx) {
 | |
|     LOG_DEBUG(Service_AM, "called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(health_warning_disappeared_system_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| void IApplicationFunctions::PrepareForJit(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void LoopProcess(Nvnflinger::Nvnflinger& nvnflinger, Core::System& system) {
 | |
|     auto message_queue = std::make_shared<AppletMessageQueue>(system);
 | |
|     // Needed on game boot
 | |
|     message_queue->PushMessage(AppletMessageQueue::AppletMessage::FocusStateChanged);
 | |
| 
 | |
|     auto server_manager = std::make_unique<ServerManager>(system);
 | |
| 
 | |
|     server_manager->RegisterNamedService(
 | |
|         "appletAE", std::make_shared<AppletAE>(nvnflinger, message_queue, system));
 | |
|     server_manager->RegisterNamedService(
 | |
|         "appletOE", std::make_shared<AppletOE>(nvnflinger, message_queue, system));
 | |
|     server_manager->RegisterNamedService("idle:sys", std::make_shared<IdleSys>(system));
 | |
|     server_manager->RegisterNamedService("omm", std::make_shared<OMM>(system));
 | |
|     server_manager->RegisterNamedService("spsm", std::make_shared<SPSM>(system));
 | |
|     ServerManager::RunServer(std::move(server_manager));
 | |
| }
 | |
| 
 | |
| IHomeMenuFunctions::IHomeMenuFunctions(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IHomeMenuFunctions"}, service_context{system,
 | |
|                                                                        "IHomeMenuFunctions"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {10, &IHomeMenuFunctions::RequestToGetForeground, "RequestToGetForeground"},
 | |
|         {11, nullptr, "LockForeground"},
 | |
|         {12, nullptr, "UnlockForeground"},
 | |
|         {20, nullptr, "PopFromGeneralChannel"},
 | |
|         {21, &IHomeMenuFunctions::GetPopFromGeneralChannelEvent, "GetPopFromGeneralChannelEvent"},
 | |
|         {30, nullptr, "GetHomeButtonWriterLockAccessor"},
 | |
|         {31, nullptr, "GetWriterLockAccessorEx"},
 | |
|         {40, nullptr, "IsSleepEnabled"},
 | |
|         {41, nullptr, "IsRebootEnabled"},
 | |
|         {50, nullptr, "LaunchSystemApplet"},
 | |
|         {51, nullptr, "LaunchStarter"},
 | |
|         {100, nullptr, "PopRequestLaunchApplicationForDebug"},
 | |
|         {110, nullptr, "IsForceTerminateApplicationDisabledForDebug"},
 | |
|         {200, nullptr, "LaunchDevMenu"},
 | |
|         {1000, nullptr, "SetLastApplicationExitReason"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| 
 | |
|     pop_from_general_channel_event =
 | |
|         service_context.CreateEvent("IHomeMenuFunctions:PopFromGeneralChannelEvent");
 | |
| }
 | |
| 
 | |
| IHomeMenuFunctions::~IHomeMenuFunctions() {
 | |
|     service_context.CloseEvent(pop_from_general_channel_event);
 | |
| }
 | |
| 
 | |
| void IHomeMenuFunctions::RequestToGetForeground(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2};
 | |
|     rb.Push(ResultSuccess);
 | |
| }
 | |
| 
 | |
| void IHomeMenuFunctions::GetPopFromGeneralChannelEvent(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushCopyObjects(pop_from_general_channel_event->GetReadableEvent());
 | |
| }
 | |
| 
 | |
| IGlobalStateController::IGlobalStateController(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IGlobalStateController"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, nullptr, "RequestToEnterSleep"},
 | |
|         {1, nullptr, "EnterSleep"},
 | |
|         {2, nullptr, "StartSleepSequence"},
 | |
|         {3, nullptr, "StartShutdownSequence"},
 | |
|         {4, nullptr, "StartRebootSequence"},
 | |
|         {9, nullptr, "IsAutoPowerDownRequested"},
 | |
|         {10, nullptr, "LoadAndApplyIdlePolicySettings"},
 | |
|         {11, nullptr, "NotifyCecSettingsChanged"},
 | |
|         {12, nullptr, "SetDefaultHomeButtonLongPressTime"},
 | |
|         {13, nullptr, "UpdateDefaultDisplayResolution"},
 | |
|         {14, nullptr, "ShouldSleepOnBoot"},
 | |
|         {15, nullptr, "GetHdcpAuthenticationFailedEvent"},
 | |
|         {30, nullptr, "OpenCradleFirmwareUpdater"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IGlobalStateController::~IGlobalStateController() = default;
 | |
| 
 | |
| IApplicationCreator::IApplicationCreator(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IApplicationCreator"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, nullptr, "CreateApplication"},
 | |
|         {1, nullptr, "PopLaunchRequestedApplication"},
 | |
|         {10, nullptr, "CreateSystemApplication"},
 | |
|         {100, nullptr, "PopFloatingApplicationForDevelopment"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IApplicationCreator::~IApplicationCreator() = default;
 | |
| 
 | |
| IProcessWindingController::IProcessWindingController(Core::System& system_)
 | |
|     : ServiceFramework{system_, "IProcessWindingController"} {
 | |
|     // clang-format off
 | |
|     static const FunctionInfo functions[] = {
 | |
|         {0, &IProcessWindingController::GetLaunchReason, "GetLaunchReason"},
 | |
|         {11, &IProcessWindingController::OpenCallingLibraryApplet, "OpenCallingLibraryApplet"},
 | |
|         {21, nullptr, "PushContext"},
 | |
|         {22, nullptr, "PopContext"},
 | |
|         {23, nullptr, "CancelWindingReservation"},
 | |
|         {30, nullptr, "WindAndDoReserved"},
 | |
|         {40, nullptr, "ReserveToStartAndWaitAndUnwindThis"},
 | |
|         {41, nullptr, "ReserveToStartAndWait"},
 | |
|     };
 | |
|     // clang-format on
 | |
| 
 | |
|     RegisterHandlers(functions);
 | |
| }
 | |
| 
 | |
| IProcessWindingController::~IProcessWindingController() = default;
 | |
| 
 | |
| void IProcessWindingController::GetLaunchReason(HLERequestContext& ctx) {
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called");
 | |
| 
 | |
|     struct AppletProcessLaunchReason {
 | |
|         u8 flag;
 | |
|         INSERT_PADDING_BYTES(3);
 | |
|     };
 | |
|     static_assert(sizeof(AppletProcessLaunchReason) == 0x4,
 | |
|                   "AppletProcessLaunchReason is an invalid size");
 | |
| 
 | |
|     AppletProcessLaunchReason reason{
 | |
|         .flag = 0,
 | |
|     };
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 3};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushRaw(reason);
 | |
| }
 | |
| 
 | |
| void IProcessWindingController::OpenCallingLibraryApplet(HLERequestContext& ctx) {
 | |
|     const auto applet_id = Applets::AppletId::MiiEdit;
 | |
|     const auto applet_mode = Applets::LibraryAppletMode::AllForeground;
 | |
| 
 | |
|     LOG_WARNING(Service_AM, "(STUBBED) called with applet_id={:08X}, applet_mode={:08X}", applet_id,
 | |
|                 applet_mode);
 | |
| 
 | |
|     const auto& applet_manager{system.GetAppletManager()};
 | |
|     const auto applet = applet_manager.GetApplet(applet_id, applet_mode);
 | |
| 
 | |
|     if (applet == nullptr) {
 | |
|         LOG_ERROR(Service_AM, "Applet doesn't exist! applet_id={}", applet_id);
 | |
| 
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(ResultUnknown);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(ResultSuccess);
 | |
|     rb.PushIpcInterface<ILibraryAppletAccessor>(system, applet);
 | |
| }
 | |
| } // namespace Service::AM
 | 
