2018-01-13 21:22:39 +00:00
|
|
|
// Copyright 2018 yuzu emulator team
|
2017-10-15 02:50:04 +00:00
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-02-02 21:03:40 +00:00
|
|
|
#include "core/hle/ipc_helpers.h"
|
|
|
|
#include "core/hle/kernel/event.h"
|
2017-10-15 02:50:04 +00:00
|
|
|
#include "core/hle/service/am/am.h"
|
2018-02-02 21:03:40 +00:00
|
|
|
#include "core/hle/service/am/applet_ae.h"
|
2017-10-15 02:50:04 +00:00
|
|
|
#include "core/hle/service/am/applet_oe.h"
|
2018-02-02 21:03:40 +00:00
|
|
|
#include "core/hle/service/apm/apm.h"
|
|
|
|
#include "core/hle/service/nvflinger/nvflinger.h"
|
2017-10-15 02:50:04 +00:00
|
|
|
|
|
|
|
namespace Service {
|
|
|
|
namespace AM {
|
|
|
|
|
2018-02-02 21:03:40 +00:00
|
|
|
IWindowController::IWindowController() : ServiceFramework("IWindowController") {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{1, &IWindowController::GetAppletResourceUserId, "GetAppletResourceUserId"},
|
|
|
|
{10, &IWindowController::AcquireForegroundRights, "AcquireForegroundRights"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IWindowController::GetAppletResourceUserId(Kernel::HLERequestContext& ctx) {
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u64>(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IWindowController::AcquireForegroundRights(Kernel::HLERequestContext& ctx) {
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
IAudioController::IAudioController() : ServiceFramework("IAudioController") {}
|
|
|
|
|
|
|
|
IDisplayController::IDisplayController() : ServiceFramework("IDisplayController") {}
|
|
|
|
|
|
|
|
IDebugFunctions::IDebugFunctions() : ServiceFramework("IDebugFunctions") {}
|
|
|
|
|
|
|
|
ISelfController::ISelfController(std::shared_ptr<NVFlinger::NVFlinger> nvflinger)
|
|
|
|
: ServiceFramework("ISelfController"), nvflinger(std::move(nvflinger)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{1, &ISelfController::LockExit, "LockExit"},
|
|
|
|
{2, &ISelfController::UnlockExit, "UnlockExit"},
|
|
|
|
{11, &ISelfController::SetOperationModeChangedNotification,
|
|
|
|
"SetOperationModeChangedNotification"},
|
|
|
|
{12, &ISelfController::SetPerformanceModeChangedNotification,
|
|
|
|
"SetPerformanceModeChangedNotification"},
|
|
|
|
{13, &ISelfController::SetFocusHandlingMode, "SetFocusHandlingMode"},
|
|
|
|
{14, &ISelfController::SetRestartMessageEnabled, "SetRestartMessageEnabled"},
|
|
|
|
{16, &ISelfController::SetOutOfFocusSuspendingEnabled, "SetOutOfFocusSuspendingEnabled"},
|
|
|
|
{40, &ISelfController::CreateManagedDisplayLayer, "CreateManagedDisplayLayer"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::SetFocusHandlingMode(Kernel::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;
|
|
|
|
};
|
|
|
|
auto flags = rp.PopRaw<FocusHandlingModeParams>();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::SetRestartMessageEnabled(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::SetPerformanceModeChangedNotification(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
bool flag = rp.Pop<bool>();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called flag=%u", static_cast<u32>(flag));
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::SetOperationModeChangedNotification(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
bool flag = rp.Pop<bool>();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called flag=%u", static_cast<u32>(flag));
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::SetOutOfFocusSuspendingEnabled(Kernel::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>();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called enabled=%u", static_cast<u32>(enabled));
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::LockExit(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::UnlockExit(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ISelfController::CreateManagedDisplayLayer(Kernel::HLERequestContext& ctx) {
|
|
|
|
// TODO(Subv): Find out how AM determines the display to use, for now just create the layer
|
|
|
|
// in the Default display.
|
|
|
|
u64 display_id = nvflinger->OpenDisplay("Default");
|
|
|
|
u64 layer_id = nvflinger->CreateLayer(display_id);
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(layer_id);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ICommonStateGetter::ICommonStateGetter() : ServiceFramework("ICommonStateGetter") {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &ICommonStateGetter::GetEventHandle, "GetEventHandle"},
|
|
|
|
{1, &ICommonStateGetter::ReceiveMessage, "ReceiveMessage"},
|
|
|
|
{5, &ICommonStateGetter::GetOperationMode, "GetOperationMode"},
|
|
|
|
{6, &ICommonStateGetter::GetPerformanceMode, "GetPerformanceMode"},
|
|
|
|
{9, &ICommonStateGetter::GetCurrentFocusState, "GetCurrentFocusState"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
|
|
|
|
event = Kernel::Event::Create(Kernel::ResetType::OneShot, "ICommonStateGetter:Event");
|
|
|
|
}
|
|
|
|
|
|
|
|
void ICommonStateGetter::GetEventHandle(Kernel::HLERequestContext& ctx) {
|
|
|
|
event->Signal();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 1};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushCopyObjects(event);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ICommonStateGetter::ReceiveMessage(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u32>(15);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ICommonStateGetter::GetCurrentFocusState(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(static_cast<u8>(FocusState::InFocus));
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ICommonStateGetter::GetOperationMode(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(static_cast<u8>(OperationMode::Handheld));
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ICommonStateGetter::GetPerformanceMode(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(static_cast<u32>(APM::PerformanceMode::Handheld));
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ILibraryAppletCreator::ILibraryAppletCreator() : ServiceFramework("ILibraryAppletCreator") {}
|
|
|
|
|
|
|
|
class IStorageAccessor final : public ServiceFramework<IStorageAccessor> {
|
|
|
|
public:
|
|
|
|
explicit IStorageAccessor(std::vector<u8> buffer)
|
|
|
|
: ServiceFramework("IStorageAccessor"), buffer(std::move(buffer)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &IStorageAccessor::GetSize, "GetSize"},
|
|
|
|
{11, &IStorageAccessor::Read, "Read"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<u8> buffer;
|
|
|
|
|
|
|
|
void GetSize(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(static_cast<u64>(buffer.size()));
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_DEBUG(Service_AM, "called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Read(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
u64 offset = rp.Pop<u64>();
|
|
|
|
|
|
|
|
const auto& output_buffer = ctx.BufferDescriptorC()[0];
|
|
|
|
|
|
|
|
ASSERT(offset + output_buffer.Size() <= buffer.size());
|
|
|
|
|
|
|
|
Memory::WriteBlock(output_buffer.Address(), buffer.data() + offset, output_buffer.Size());
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_DEBUG(Service_AM, "called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class IStorage final : public ServiceFramework<IStorage> {
|
|
|
|
public:
|
|
|
|
explicit IStorage(std::vector<u8> buffer)
|
|
|
|
: ServiceFramework("IStorage"), buffer(std::move(buffer)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &IStorage::Open, "Open"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<u8> buffer;
|
|
|
|
|
|
|
|
void Open(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<AM::IStorageAccessor>(buffer);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_DEBUG(Service_AM, "called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
IApplicationFunctions::IApplicationFunctions() : ServiceFramework("IApplicationFunctions") {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{1, &IApplicationFunctions::PopLaunchParameter, "PopLaunchParameter"},
|
|
|
|
{21, &IApplicationFunctions::GetDesiredLanguage, "GetDesiredLanguage"},
|
|
|
|
{22, &IApplicationFunctions::SetTerminateResult, "SetTerminateResult"},
|
|
|
|
{66, &IApplicationFunctions::InitializeGamePlayRecording, "InitializeGamePlayRecording"},
|
|
|
|
{67, &IApplicationFunctions::SetGamePlayRecordingState, "SetGamePlayRecordingState"},
|
|
|
|
{40, &IApplicationFunctions::NotifyRunning, "NotifyRunning"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IApplicationFunctions::PopLaunchParameter(Kernel::HLERequestContext& ctx) {
|
|
|
|
constexpr u8 data[0x88] = {
|
|
|
|
0xca, 0x97, 0x94, 0xc7, // Magic
|
|
|
|
1, 0, 0, 0, // IsAccountSelected (bool)
|
|
|
|
1, 0, 0, 0, // User Id (word 0)
|
|
|
|
0, 0, 0, 0, // User Id (word 1)
|
|
|
|
0, 0, 0, 0, // User Id (word 2)
|
|
|
|
0, 0, 0, 0 // User Id (word 3)
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<u8> buffer(data, data + sizeof(data));
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<AM::IStorage>(buffer);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_DEBUG(Service_AM, "called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void IApplicationFunctions::SetTerminateResult(Kernel::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>();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called, result=0x%08X", result);
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void IApplicationFunctions::GetDesiredLanguage(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u64>(SystemLanguage::English);
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void IApplicationFunctions::InitializeGamePlayRecording(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void IApplicationFunctions::SetGamePlayRecordingState(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void IApplicationFunctions::NotifyRunning(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u8>(0); // Unknown, seems to be ignored by official processes
|
|
|
|
|
2018-02-04 21:58:12 +00:00
|
|
|
LOG_WARNING(Service_AM, "(STUBBED) called");
|
2018-02-02 21:03:40 +00:00
|
|
|
}
|
|
|
|
|
2018-01-22 18:46:36 +00:00
|
|
|
void InstallInterfaces(SM::ServiceManager& service_manager,
|
|
|
|
std::shared_ptr<NVFlinger::NVFlinger> nvflinger) {
|
2018-02-02 21:03:40 +00:00
|
|
|
std::make_shared<AppletAE>(nvflinger)->InstallAsService(service_manager);
|
2018-01-22 18:46:36 +00:00
|
|
|
std::make_shared<AppletOE>(nvflinger)->InstallAsService(service_manager);
|
2017-10-15 02:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace AM
|
|
|
|
} // namespace Service
|