summaryrefslogtreecommitdiffstats
path: root/src/core/hle/service
diff options
context:
space:
mode:
Diffstat (limited to 'src/core/hle/service')
-rw-r--r--src/core/hle/service/ac/ac.cpp5
-rw-r--r--src/core/hle/service/am/am.cpp6
-rw-r--r--src/core/hle/service/apt/apt.cpp2
-rw-r--r--src/core/hle/service/apt/apt.h2
-rw-r--r--src/core/hle/service/boss/boss.cpp3
-rw-r--r--src/core/hle/service/cam/cam.cpp549
-rw-r--r--src/core/hle/service/cam/cam.h13
-rw-r--r--src/core/hle/service/cecd/cecd.cpp3
-rw-r--r--src/core/hle/service/cfg/cfg.cpp3
-rw-r--r--src/core/hle/service/csnd_snd.cpp3
-rw-r--r--src/core/hle/service/dlp/dlp_srvr.cpp1
-rw-r--r--src/core/hle/service/dsp_dsp.cpp6
-rw-r--r--src/core/hle/service/err_f.cpp4
-rw-r--r--src/core/hle/service/frd/frd.cpp5
-rw-r--r--src/core/hle/service/fs/archive.cpp19
-rw-r--r--src/core/hle/service/fs/archive.h6
-rw-r--r--src/core/hle/service/fs/fs_user.cpp40
-rw-r--r--src/core/hle/service/gsp_gpu.cpp2
-rw-r--r--src/core/hle/service/hid/hid.cpp2
-rw-r--r--src/core/hle/service/ir/ir_rst.cpp1
-rw-r--r--src/core/hle/service/ir/ir_user.cpp1
-rw-r--r--src/core/hle/service/ldr_ro/ldr_ro.cpp1
-rw-r--r--src/core/hle/service/mic_u.cpp2
-rw-r--r--src/core/hle/service/ndm/ndm.cpp2
-rw-r--r--src/core/hle/service/nfc/nfc.cpp2
-rw-r--r--src/core/hle/service/nim/nim.cpp1
-rw-r--r--src/core/hle/service/nwm/nwm_uds.cpp1
-rw-r--r--src/core/hle/service/nwm/uds_beacon.cpp7
-rw-r--r--src/core/hle/service/service.cpp43
-rw-r--r--src/core/hle/service/service.h58
-rw-r--r--src/core/hle/service/sm/sm.cpp59
-rw-r--r--src/core/hle/service/sm/sm.h49
-rw-r--r--src/core/hle/service/sm/srv.cpp (renamed from src/core/hle/service/srv.cpp)71
-rw-r--r--src/core/hle/service/sm/srv.h (renamed from src/core/hle/service/srv.h)5
-rw-r--r--src/core/hle/service/soc_u.cpp1
-rw-r--r--src/core/hle/service/ssl_c.cpp2
-rw-r--r--src/core/hle/service/y2r_u.cpp2
37 files changed, 553 insertions, 429 deletions
diff --git a/src/core/hle/service/ac/ac.cpp b/src/core/hle/service/ac/ac.cpp
index aa270a2c3..e3dd23949 100644
--- a/src/core/hle/service/ac/ac.cpp
+++ b/src/core/hle/service/ac/ac.cpp
@@ -4,11 +4,16 @@
#include <array>
+#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
+#include "core/hle/result.h"
#include "core/hle/service/ac/ac.h"
#include "core/hle/service/ac/ac_i.h"
#include "core/hle/service/ac/ac_u.h"
+#include "core/memory.h"
namespace Service {
namespace AC {
diff --git a/src/core/hle/service/am/am.cpp b/src/core/hle/service/am/am.cpp
index d344a622f..961305e9f 100644
--- a/src/core/hle/service/am/am.cpp
+++ b/src/core/hle/service/am/am.cpp
@@ -2,8 +2,12 @@
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
+#include <array>
#include <cinttypes>
+#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
+#include "core/hle/result.h"
#include "core/hle/service/am/am.h"
#include "core/hle/service/am/am_app.h"
#include "core/hle/service/am/am_net.h"
@@ -176,8 +180,6 @@ void GetTicketList(Service::Interface* self) {
}
void Init() {
- using namespace Kernel;
-
AddService(new AM_APP_Interface);
AddService(new AM_NET_Interface);
AddService(new AM_SYS_Interface);
diff --git a/src/core/hle/service/apt/apt.cpp b/src/core/hle/service/apt/apt.cpp
index 366d1eacf..4c587e3c8 100644
--- a/src/core/hle/service/apt/apt.cpp
+++ b/src/core/hle/service/apt/apt.cpp
@@ -5,6 +5,7 @@
#include "common/common_paths.h"
#include "common/file_util.h"
#include "common/logging/log.h"
+#include "core/core.h"
#include "core/hle/applets/applet.h"
#include "core/hle/kernel/event.h"
#include "core/hle/kernel/mutex.h"
@@ -74,6 +75,7 @@ void GetSharedFont(Service::Interface* self) {
LOG_ERROR(Service_APT, "shared font file missing - go dump it from your 3ds");
rb.Push<u32>(-1); // TODO: Find the right error code
rb.Skip(1 + 2, true);
+ Core::System::GetInstance().SetStatus(Core::System::ResultStatus::ErrorSharedFont);
return;
}
diff --git a/src/core/hle/service/apt/apt.h b/src/core/hle/service/apt/apt.h
index e63b61450..ee80926d2 100644
--- a/src/core/hle/service/apt/apt.h
+++ b/src/core/hle/service/apt/apt.h
@@ -4,6 +4,8 @@
#pragma once
+#include <vector>
+#include "common/common_funcs.h"
#include "common/common_types.h"
#include "common/swap.h"
#include "core/hle/kernel/kernel.h"
diff --git a/src/core/hle/service/boss/boss.cpp b/src/core/hle/service/boss/boss.cpp
index 91056189a..2bba3aff6 100644
--- a/src/core/hle/service/boss/boss.cpp
+++ b/src/core/hle/service/boss/boss.cpp
@@ -3,6 +3,9 @@
// Refer to the license.txt file included.
#include <cinttypes>
+#include "common/logging/log.h"
+#include "core/hle/ipc.h"
+#include "core/hle/result.h"
#include "core/hle/service/boss/boss.h"
#include "core/hle/service/boss/boss_p.h"
#include "core/hle/service/boss/boss_u.h"
diff --git a/src/core/hle/service/cam/cam.cpp b/src/core/hle/service/cam/cam.cpp
index 95665e754..7394c844f 100644
--- a/src/core/hle/service/cam/cam.cpp
+++ b/src/core/hle/service/cam/cam.cpp
@@ -11,13 +11,17 @@
#include "common/logging/log.h"
#include "core/core_timing.h"
#include "core/frontend/camera/factory.h"
+#include "core/hle/ipc.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/result.h"
#include "core/hle/service/cam/cam.h"
#include "core/hle/service/cam/cam_c.h"
#include "core/hle/service/cam/cam_q.h"
#include "core/hle/service/cam/cam_s.h"
#include "core/hle/service/cam/cam_u.h"
#include "core/hle/service/service.h"
+#include "core/memory.h"
#include "core/settings.h"
namespace Service {
@@ -55,7 +59,7 @@ struct PortConfig {
u16 x1; // x-coordinate of ending position for trimming
u16 y1; // y-coordinate of ending position for trimming
- u32 transfer_bytes;
+ u16 transfer_bytes;
Kernel::SharedPtr<Kernel::Event> completion_event;
Kernel::SharedPtr<Kernel::Event> buffer_error_interrupt_event;
@@ -225,8 +229,7 @@ static void ActivatePort(int port_id, int camera_id) {
template <int max_index>
class CommandParamBitSet : public BitSet8 {
public:
- explicit CommandParamBitSet(u32 command_param)
- : BitSet8(static_cast<u8>(command_param & 0xFF)) {}
+ explicit CommandParamBitSet(u8 command_param) : BitSet8(command_param) {}
bool IsValid() const {
return m_val < (1 << max_index);
@@ -244,9 +247,10 @@ using CameraSet = CommandParamBitSet<3>;
} // namespace
void StartCapture(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x01, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
@@ -267,21 +271,20 @@ void StartCapture(Service::Interface* self) {
LOG_WARNING(Service_CAM, "port %u already started", i);
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x1, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u", port_select.m_val);
}
void StopCapture(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x02, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
@@ -293,21 +296,20 @@ void StopCapture(Service::Interface* self) {
LOG_WARNING(Service_CAM, "port %u already stopped", i);
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x2, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u", port_select.m_val);
}
void IsBusy(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x03, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
if (port_select.IsValid()) {
bool is_busy = true;
@@ -315,80 +317,74 @@ void IsBusy(Service::Interface* self) {
for (int i : port_select) {
is_busy &= ports[i].is_busy;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = is_busy ? 1 : 0;
+ rb.Push(RESULT_SUCCESS);
+ rb.Push(is_busy);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.Skip(1, false);
}
- cmd_buff[0] = IPC::MakeHeader(0x3, 2, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u", port_select.m_val);
}
void ClearBuffer(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x04, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
- cmd_buff[0] = IPC::MakeHeader(0x4, 1, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
+ rb.Push(RESULT_SUCCESS);
LOG_WARNING(Service_CAM, "(STUBBED) called, port_select=%u", port_select.m_val);
}
void GetVsyncInterruptEvent(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x05, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 2);
if (port_select.IsSingle()) {
int port = *port_select.begin();
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = IPC::CopyHandleDesc();
- cmd_buff[3] = Kernel::g_handle_table.Create(ports[port].vsync_interrupt_event).MoveFrom();
+ rb.Push(RESULT_SUCCESS);
+ rb.PushCopyHandles(
+ Kernel::g_handle_table.Create(ports[port].vsync_interrupt_event).MoveFrom());
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
- cmd_buff[2] = IPC::CopyHandleDesc();
- cmd_buff[2] = 0;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.PushCopyHandles(0);
}
- cmd_buff[0] = IPC::MakeHeader(0x5, 1, 2);
-
LOG_WARNING(Service_CAM, "(STUBBED) called, port_select=%u", port_select.m_val);
}
void GetBufferErrorInterruptEvent(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x06, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 2);
if (port_select.IsSingle()) {
int port = *port_select.begin();
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = IPC::CopyHandleDesc();
- cmd_buff[3] =
- Kernel::g_handle_table.Create(ports[port].buffer_error_interrupt_event).MoveFrom();
+ rb.Push(RESULT_SUCCESS);
+ rb.PushCopyHandles(
+ Kernel::g_handle_table.Create(ports[port].buffer_error_interrupt_event).MoveFrom());
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
- cmd_buff[2] = IPC::CopyHandleDesc();
- cmd_buff[2] = 0;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.PushCopyHandles(0);
}
LOG_WARNING(Service_CAM, "(STUBBED) called, port_select=%u", port_select.m_val);
}
void SetReceiving(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const VAddr dest = cmd_buff[1];
- const PortSet port_select(cmd_buff[2]);
- const u32 image_size = cmd_buff[3];
- const u32 trans_unit = cmd_buff[4] & 0xFFFF;
-
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x07, 4, 2);
+ const VAddr dest = rp.Pop<u32>();
+ const PortSet port_select(rp.Pop<u8>());
+ const u32 image_size = rp.Pop<u32>();
+ const u16 trans_unit = rp.Pop<u16>();
+ rp.PopHandle(); // Handle to destination process. not used
+
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 2);
if (port_select.IsSingle()) {
int port_id = *port_select.begin();
PortConfig& port = ports[port_id];
@@ -403,149 +399,145 @@ void SetReceiving(Service::Interface* self) {
port.is_pending_receiving = true;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = IPC::CopyHandleDesc();
- cmd_buff[3] = Kernel::g_handle_table.Create(port.completion_event).MoveFrom();
+ rb.Push(RESULT_SUCCESS);
+ rb.PushCopyHandles(Kernel::g_handle_table.Create(port.completion_event).MoveFrom());
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.PushCopyHandles(0);
}
- cmd_buff[0] = IPC::MakeHeader(0x7, 1, 2);
-
LOG_DEBUG(Service_CAM, "called, addr=0x%X, port_select=%u, image_size=%u, trans_unit=%u", dest,
port_select.m_val, image_size, trans_unit);
}
void IsFinishedReceiving(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x08, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
if (port_select.IsSingle()) {
int port = *port_select.begin();
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = (ports[port].is_receiving || ports[port].is_pending_receiving) ? 0 : 1;
+ bool is_busy = ports[port].is_receiving || ports[port].is_pending_receiving;
+ rb.Push(RESULT_SUCCESS);
+ rb.Push(!is_busy);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.Skip(1, false);
}
- cmd_buff[0] = IPC::MakeHeader(0x8, 2, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u", port_select.m_val);
}
void SetTransferLines(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
- const u32 transfer_lines = cmd_buff[2] & 0xFFFF;
- const u32 width = cmd_buff[3] & 0xFFFF;
- const u32 height = cmd_buff[4] & 0xFFFF;
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x09, 4, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ const u16 transfer_lines = rp.Pop<u16>();
+ const u16 width = rp.Pop<u16>();
+ const u16 height = rp.Pop<u16>();
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
ports[i].transfer_bytes = transfer_lines * width * 2;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x9, 1, 0);
-
LOG_WARNING(Service_CAM, "(STUBBED) called, port_select=%u, lines=%u, width=%u, height=%u",
port_select.m_val, transfer_lines, width, height);
}
void GetMaxLines(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x0A, 2, 0);
+ const u16 width = rp.Pop<u16>();
+ const u16 height = rp.Pop<u16>();
- const u32 width = cmd_buff[1] & 0xFFFF;
- const u32 height = cmd_buff[2] & 0xFFFF;
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
// Note: the result of the algorithm below are hwtested with width < 640 and with height < 480
constexpr u32 MIN_TRANSFER_UNIT = 256;
constexpr u32 MAX_BUFFER_SIZE = 2560;
if (width * height * 2 % MIN_TRANSFER_UNIT != 0) {
- cmd_buff[1] = ERROR_OUT_OF_RANGE.raw;
+ rb.Push(ERROR_OUT_OF_RANGE);
+ rb.Skip(1, false);
} else {
u32 lines = MAX_BUFFER_SIZE / width;
if (lines > height) {
lines = height;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ ResultCode result = RESULT_SUCCESS;
while (height % lines != 0 || (lines * width * 2 % MIN_TRANSFER_UNIT != 0)) {
--lines;
if (lines == 0) {
- cmd_buff[1] = ERROR_OUT_OF_RANGE.raw;
+ result = ERROR_OUT_OF_RANGE;
break;
}
}
- cmd_buff[2] = lines;
+ rb.Push(result);
+ rb.Push(lines);
}
- cmd_buff[0] = IPC::MakeHeader(0xA, 2, 0);
-
LOG_DEBUG(Service_CAM, "called, width=%u, height=%u", width, height);
}
void SetTransferBytes(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
- const u32 transfer_bytes = cmd_buff[2] & 0xFFFF;
- const u32 width = cmd_buff[3] & 0xFFFF;
- const u32 height = cmd_buff[4] & 0xFFFF;
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x0B, 4, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ const u16 transfer_bytes = rp.Pop<u16>();
+ const u16 width = rp.Pop<u16>();
+ const u16 height = rp.Pop<u16>();
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
ports[i].transfer_bytes = transfer_bytes;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0xB, 1, 0);
-
LOG_WARNING(Service_CAM, "(STUBBED)called, port_select=%u, bytes=%u, width=%u, height=%u",
port_select.m_val, transfer_bytes, width, height);
}
void GetTransferBytes(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x0C, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
if (port_select.IsSingle()) {
int port = *port_select.begin();
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = ports[port].transfer_bytes;
+ rb.Push(RESULT_SUCCESS);
+ rb.Push(ports[port].transfer_bytes);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.Skip(1, false);
}
- cmd_buff[0] = IPC::MakeHeader(0xC, 2, 0);
-
LOG_WARNING(Service_CAM, "(STUBBED)called, port_select=%u", port_select.m_val);
}
void GetMaxBytes(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x0D, 2, 0);
+ const u16 width = rp.Pop<u16>();
+ const u16 height = rp.Pop<u16>();
- const u32 width = cmd_buff[1] & 0xFFFF;
- const u32 height = cmd_buff[2] & 0xFFFF;
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
// Note: the result of the algorithm below are hwtested with width < 640 and with height < 480
constexpr u32 MIN_TRANSFER_UNIT = 256;
constexpr u32 MAX_BUFFER_SIZE = 2560;
if (width * height * 2 % MIN_TRANSFER_UNIT != 0) {
- cmd_buff[1] = ERROR_OUT_OF_RANGE.raw;
+ rb.Push(ERROR_OUT_OF_RANGE);
+ rb.Skip(1, false);
} else {
u32 bytes = MAX_BUFFER_SIZE;
@@ -553,63 +545,59 @@ void GetMaxBytes(Service::Interface* self) {
bytes -= MIN_TRANSFER_UNIT;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = bytes;
+ rb.Push(RESULT_SUCCESS);
+ rb.Push(bytes);
}
- cmd_buff[0] = IPC::MakeHeader(0xD, 2, 0);
LOG_DEBUG(Service_CAM, "called, width=%u, height=%u", width, height);
}
void SetTrimming(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
- const bool trim = (cmd_buff[2] & 0xFF) != 0;
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x0E, 2, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ const bool trim = rp.Pop<bool>();
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
ports[i].is_trimming = trim;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0xE, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u, trim=%d", port_select.m_val, trim);
}
void IsTrimming(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x0F, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
if (port_select.IsSingle()) {
int port = *port_select.begin();
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = ports[port].is_trimming;
+ rb.Push(RESULT_SUCCESS);
+ rb.Push(ports[port].is_trimming);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.Skip(1, false);
}
- cmd_buff[0] = IPC::MakeHeader(0xF, 2, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u", port_select.m_val);
}
void SetTrimmingParams(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
- const u16 x0 = static_cast<u16>(cmd_buff[2] & 0xFFFF);
- const u16 y0 = static_cast<u16>(cmd_buff[3] & 0xFFFF);
- const u16 x1 = static_cast<u16>(cmd_buff[4] & 0xFFFF);
- const u16 y1 = static_cast<u16>(cmd_buff[5] & 0xFFFF);
-
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x10, 5, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ const u16 x0 = rp.Pop<u16>();
+ const u16 y0 = rp.Pop<u16>();
+ const u16 x1 = rp.Pop<u16>();
+ const u16 y1 = rp.Pop<u16>();
+
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
ports[i].x0 = x0;
@@ -617,49 +605,46 @@ void SetTrimmingParams(Service::Interface* self) {
ports[i].x1 = x1;
ports[i].y1 = y1;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x10, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u, x0=%u, y0=%u, x1=%u, y1=%u", port_select.m_val,
x0, y0, x1, y1);
}
void GetTrimmingParams(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x11, 1, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(5, 0);
if (port_select.IsSingle()) {
int port = *port_select.begin();
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = ports[port].x0;
- cmd_buff[3] = ports[port].y0;
- cmd_buff[4] = ports[port].x1;
- cmd_buff[5] = ports[port].y1;
+ rb.Push(RESULT_SUCCESS);
+ rb.Push(ports[port].x0);
+ rb.Push(ports[port].y0);
+ rb.Push(ports[port].x1);
+ rb.Push(ports[port].y1);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
+ rb.Skip(4, false);
}
- cmd_buff[0] = IPC::MakeHeader(0x11, 5, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u", port_select.m_val);
}
void SetTrimmingParamsCenter(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const PortSet port_select(cmd_buff[1]);
- const u16 trim_w = static_cast<u16>(cmd_buff[2] & 0xFFFF);
- const u16 trim_h = static_cast<u16>(cmd_buff[3] & 0xFFFF);
- const u16 cam_w = static_cast<u16>(cmd_buff[4] & 0xFFFF);
- const u16 cam_h = static_cast<u16>(cmd_buff[5] & 0xFFFF);
-
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x12, 5, 0);
+ const PortSet port_select(rp.Pop<u8>());
+ const u16 trim_w = rp.Pop<u16>();
+ const u16 trim_h = rp.Pop<u16>();
+ const u16 cam_w = rp.Pop<u16>();
+ const u16 cam_h = rp.Pop<u16>();
+
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (port_select.IsValid()) {
for (int i : port_select) {
ports[i].x0 = (cam_w - trim_w) / 2;
@@ -667,23 +652,21 @@ void SetTrimmingParamsCenter(Service::Interface* self) {
ports[i].x1 = ports[i].x0 + trim_w;
ports[i].y1 = ports[i].y0 + trim_h;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid port_select=%u", port_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x12, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, port_select=%u, trim_w=%u, trim_h=%u, cam_w=%u, cam_h=%u",
port_select.m_val, trim_w, trim_h, cam_w, cam_h);
}
void Activate(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x13, 1, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid()) {
if (camera_select.m_val == 0) { // deactive all
for (int i = 0; i < 2; ++i) {
@@ -694,10 +677,10 @@ void Activate(Service::Interface* self) {
}
ports[i].is_active = false;
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else if (camera_select[0] && camera_select[1]) {
LOG_ERROR(Service_CAM, "camera 0 and 1 can't be both activated");
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
} else {
if (camera_select[0]) {
ActivatePort(0, 0);
@@ -708,24 +691,22 @@ void Activate(Service::Interface* self) {
if (camera_select[2]) {
ActivatePort(1, 2);
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
}
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u", camera_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x13, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u", camera_select.m_val);
}
void SwitchContext(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
- const ContextSet context_select(cmd_buff[2]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x14, 2, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ const ContextSet context_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid() && context_select.IsSingle()) {
int context = *context_select.begin();
for (int camera : camera_select) {
@@ -736,26 +717,24 @@ void SwitchContext(Service::Interface* self) {
cameras[camera].impl->SetFormat(context_config.format);
cameras[camera].impl->SetResolution(context_config.resolution);
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x14, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
}
void FlipImage(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
- const Flip flip = static_cast<Flip>(cmd_buff[2] & 0xFF);
- const ContextSet context_select(cmd_buff[3]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x1D, 3, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ const Flip flip = static_cast<Flip>(rp.Pop<u8>());
+ const ContextSet context_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid() && context_select.IsValid()) {
for (int camera : camera_select) {
for (int context : context_select) {
@@ -765,32 +744,30 @@ void FlipImage(Service::Interface* self) {
}
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x1D, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u, flip=%d, context_select=%u",
camera_select.m_val, static_cast<int>(flip), context_select.m_val);
}
void SetDetailSize(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x1E, 8, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
Resolution resolution;
- resolution.width = static_cast<u16>(cmd_buff[2] & 0xFFFF);
- resolution.height = static_cast<u16>(cmd_buff[3] & 0xFFFF);
- resolution.crop_x0 = static_cast<u16>(cmd_buff[4] & 0xFFFF);
- resolution.crop_y0 = static_cast<u16>(cmd_buff[5] & 0xFFFF);
- resolution.crop_x1 = static_cast<u16>(cmd_buff[6] & 0xFFFF);
- resolution.crop_y1 = static_cast<u16>(cmd_buff[7] & 0xFFFF);
- const ContextSet context_select(cmd_buff[8]);
-
+ resolution.width = rp.Pop<u16>();
+ resolution.height = rp.Pop<u16>();
+ resolution.crop_x0 = rp.Pop<u16>();
+ resolution.crop_y0 = rp.Pop<u16>();
+ resolution.crop_x1 = rp.Pop<u16>();
+ resolution.crop_y1 = rp.Pop<u16>();
+ const ContextSet context_select(rp.Pop<u8>());
+
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid() && context_select.IsValid()) {
for (int camera : camera_select) {
for (int context : context_select) {
@@ -800,15 +777,13 @@ void SetDetailSize(Service::Interface* self) {
}
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x1E, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u, width=%u, height=%u, crop_x0=%u, crop_y0=%u, "
"crop_x1=%u, crop_y1=%u, context_select=%u",
camera_select.m_val, resolution.width, resolution.height, resolution.crop_x0,
@@ -816,12 +791,12 @@ void SetDetailSize(Service::Interface* self) {
}
void SetSize(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
- const u32 size = cmd_buff[2] & 0xFF;
- const ContextSet context_select(cmd_buff[3]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x1F, 3, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ const u8 size = rp.Pop<u8>();
+ const ContextSet context_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid() && context_select.IsValid()) {
for (int camera : camera_select) {
for (int context : context_select) {
@@ -831,49 +806,45 @@ void SetSize(Service::Interface* self) {
}
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x1F, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u, size=%u, context_select=%u",
camera_select.m_val, size, context_select.m_val);
}
void SetFrameRate(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
- const FrameRate frame_rate = static_cast<FrameRate>(cmd_buff[2] & 0xFF);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x20, 2, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ const FrameRate frame_rate = static_cast<FrameRate>(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid()) {
for (int camera : camera_select) {
cameras[camera].frame_rate = frame_rate;
// TODO(wwylele): consider hinting the actual camera with the expected frame rate
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u", camera_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x20, 1, 0);
-
LOG_WARNING(Service_CAM, "(STUBBED) called, camera_select=%u, frame_rate=%d",
camera_select.m_val, static_cast<int>(frame_rate));
}
void SetEffect(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
- const Effect effect = static_cast<Effect>(cmd_buff[2] & 0xFF);
- const ContextSet context_select(cmd_buff[3]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x22, 3, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ const Effect effect = static_cast<Effect>(rp.Pop<u8>());
+ const ContextSet context_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid() && context_select.IsValid()) {
for (int camera : camera_select) {
for (int context : context_select) {
@@ -883,26 +854,24 @@ void SetEffect(Service::Interface* self) {
}
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x22, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u, effect=%d, context_select=%u",
camera_select.m_val, static_cast<int>(effect), context_select.m_val);
}
void SetOutputFormat(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- const CameraSet camera_select(cmd_buff[1]);
- const OutputFormat format = static_cast<OutputFormat>(cmd_buff[2] & 0xFF);
- const ContextSet context_select(cmd_buff[3]);
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x25, 3, 0);
+ const CameraSet camera_select(rp.Pop<u8>());
+ const OutputFormat format = static_cast<OutputFormat>(rp.Pop<u8>());
+ const ContextSet context_select(rp.Pop<u8>());
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
if (camera_select.IsValid() && context_select.IsValid()) {
for (int camera : camera_select) {
for (int context : context_select) {
@@ -912,34 +881,32 @@ void SetOutputFormat(Service::Interface* self) {
}
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", camera_select.m_val,
context_select.m_val);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ rb.Push(ERROR_INVALID_ENUM_VALUE);
}
- cmd_buff[0] = IPC::MakeHeader(0x25, 1, 0);
-
LOG_DEBUG(Service_CAM, "called, camera_select=%u, format=%d, context_select=%u",
camera_select.m_val, static_cast<int>(format), context_select.m_val);
}
void SynchronizeVsyncTiming(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x29, 2, 0);
+ const u8 camera_select1 = rp.Pop<u8>();
+ const u8 camera_select2 = rp.Pop<u8>();
- const u32 camera_select1 = cmd_buff[1] & 0xFF;
- const u32 camera_select2 = cmd_buff[2] & 0xFF;
-
- cmd_buff[0] = IPC::MakeHeader(0x29, 1, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
+ rb.Push(RESULT_SUCCESS);
LOG_WARNING(Service_CAM, "(STUBBED) called, camera_select1=%u, camera_select2=%u",
camera_select1, camera_select2);
}
void GetStereoCameraCalibrationData(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestBuilder rb =
+ IPC::RequestParser(Kernel::GetCommandBuffer(), 0x2B, 0, 0).MakeBuilder(17, 0);
// Default values taken from yuriks' 3DS. Valid data is required here or games using the
// calibration get stuck in an infinite CPU loop.
@@ -958,34 +925,28 @@ void GetStereoCameraCalibrationData(Service::Interface* self) {
data.imageWidth = 640;
data.imageHeight = 480;
- cmd_buff[0] = IPC::MakeHeader(0x2B, 17, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
- memcpy(&cmd_buff[2], &data, sizeof(data));
+ rb.Push(RESULT_SUCCESS);
+ rb.PushRaw(data);
LOG_TRACE(Service_CAM, "called");
}
void SetPackageParameterWithoutContext(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x33, 11, 0);
PackageParameterWithoutContext package;
- std::memcpy(&package, cmd_buff + 1, sizeof(package));
+ rp.PopRaw(package);
- cmd_buff[0] = IPC::MakeHeader(0x33, 1, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
+ rb.Push(RESULT_SUCCESS);
LOG_WARNING(Service_CAM, "(STUBBED) called");
}
-template <typename PackageParameterType, int command_id>
-static void SetPackageParameter() {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- PackageParameterType package;
- std::memcpy(&package, cmd_buff + 1, sizeof(package));
-
- const CameraSet camera_select(static_cast<u32>(package.camera_select));
- const ContextSet context_select(static_cast<u32>(package.context_select));
+template <typename PackageParameterType>
+static ResultCode SetPackageParameter(const PackageParameterType& package) {
+ const CameraSet camera_select(package.camera_select);
+ const ContextSet context_select(package.context_select);
if (camera_select.IsValid() && context_select.IsValid()) {
for (int camera_id : camera_select) {
@@ -1002,53 +963,66 @@ static void SetPackageParameter() {
}
}
}
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ return RESULT_SUCCESS;
} else {
LOG_ERROR(Service_CAM, "invalid camera_select=%u, context_select=%u", package.camera_select,
package.context_select);
- cmd_buff[1] = ERROR_INVALID_ENUM_VALUE.raw;
+ return ERROR_INVALID_ENUM_VALUE;
}
-
- cmd_buff[0] = IPC::MakeHeader(command_id, 1, 0);
-
- LOG_DEBUG(Service_CAM, "called");
}
-Resolution PackageParameterWithContext::GetResolution() {
+Resolution PackageParameterWithContext::GetResolution() const {
return PRESET_RESOLUTION[static_cast<int>(size)];
}
void SetPackageParameterWithContext(Service::Interface* self) {
- SetPackageParameter<PackageParameterWithContext, 0x34>();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x34, 5, 0);
+
+ PackageParameterWithContext package;
+ rp.PopRaw(package);
+
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
+ ResultCode result = SetPackageParameter(package);
+ rb.Push(result);
+
+ LOG_DEBUG(Service_CAM, "called");
}
void SetPackageParameterWithContextDetail(Service::Interface* self) {
- SetPackageParameter<PackageParameterWithContextDetail, 0x35>();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x35, 7, 0);
+
+ PackageParameterWithContextDetail package;
+ rp.PopRaw(package);
+
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
+ ResultCode result = SetPackageParameter(package);
+ rb.Push(result);
+
+ LOG_DEBUG(Service_CAM, "called");
}
void GetSuitableY2rStandardCoefficient(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- cmd_buff[0] = IPC::MakeHeader(0x36, 2, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
- cmd_buff[2] = 0;
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x36, 0, 0);
+ IPC::RequestBuilder rb = rp.MakeBuilder(2, 0);
+ rb.Push(RESULT_SUCCESS);
+ rb.Push<u32>(0);
LOG_WARNING(Service_CAM, "(STUBBED) called");
}
void PlayShutterSound(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
-
- u8 sound_id = cmd_buff[1] & 0xFF;
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x38, 1, 0);
+ u8 sound_id = rp.Pop<u8>();
- cmd_buff[0] = IPC::MakeHeader(0x38, 1, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
+ rb.Push(RESULT_SUCCESS);
LOG_WARNING(Service_CAM, "(STUBBED) called, sound_id=%d", sound_id);
}
void DriverInitialize(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x39, 0, 0);
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
for (int camera_id = 0; camera_id < NumCameras; ++camera_id) {
CameraConfig& camera = cameras[camera_id];
@@ -1074,14 +1048,14 @@ void DriverInitialize(Service::Interface* self) {
port.Clear();
}
- cmd_buff[0] = IPC::MakeHeader(0x39, 1, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
LOG_DEBUG(Service_CAM, "called");
}
void DriverFinalize(Service::Interface* self) {
- u32* cmd_buff = Kernel::GetCommandBuffer();
+ IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x3A, 0, 0);
+ IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
CancelReceiving(0);
CancelReceiving(1);
@@ -1090,8 +1064,7 @@ void DriverFinalize(Service::Interface* self) {
camera.impl = nullptr;
}
- cmd_buff[0] = IPC::MakeHeader(0x3A, 1, 0);
- cmd_buff[1] = RESULT_SUCCESS.raw;
+ rb.Push(RESULT_SUCCESS);
LOG_DEBUG(Service_CAM, "called");
}
diff --git a/src/core/hle/service/cam/cam.h b/src/core/hle/service/cam/cam.h
index 34a9c8479..b6da721d8 100644
--- a/src/core/hle/service/cam/cam.h
+++ b/src/core/hle/service/cam/cam.h
@@ -184,9 +184,10 @@ struct PackageParameterWithoutContext {
s16 auto_white_balance_window_y;
s16 auto_white_balance_window_width;
s16 auto_white_balance_window_height;
+ INSERT_PADDING_WORDS(4);
};
-static_assert(sizeof(PackageParameterWithoutContext) == 28,
+static_assert(sizeof(PackageParameterWithoutContext) == 44,
"PackageParameterCameraWithoutContext structure size is wrong");
struct PackageParameterWithContext {
@@ -196,11 +197,12 @@ struct PackageParameterWithContext {
Effect effect;
Size size;
INSERT_PADDING_BYTES(3);
+ INSERT_PADDING_WORDS(3);
- Resolution GetResolution();
+ Resolution GetResolution() const;
};
-static_assert(sizeof(PackageParameterWithContext) == 8,
+static_assert(sizeof(PackageParameterWithContext) == 20,
"PackageParameterWithContext structure size is wrong");
struct PackageParameterWithContextDetail {
@@ -209,13 +211,14 @@ struct PackageParameterWithContextDetail {
Flip flip;
Effect effect;
Resolution resolution;
+ INSERT_PADDING_WORDS(3);
- Resolution GetResolution() {
+ Resolution GetResolution() const {
return resolution;
}
};
-static_assert(sizeof(PackageParameterWithContextDetail) == 16,
+static_assert(sizeof(PackageParameterWithContextDetail) == 28,
"PackageParameterWithContextDetail structure size is wrong");
/**
diff --git a/src/core/hle/service/cecd/cecd.cpp b/src/core/hle/service/cecd/cecd.cpp
index eb04273db..bd9814244 100644
--- a/src/core/hle/service/cecd/cecd.cpp
+++ b/src/core/hle/service/cecd/cecd.cpp
@@ -3,7 +3,10 @@
// Refer to the license.txt file included.
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
+#include "core/hle/result.h"
#include "core/hle/service/cecd/cecd.h"
#include "core/hle/service/cecd/cecd_ndm.h"
#include "core/hle/service/cecd/cecd_s.h"
diff --git a/src/core/hle/service/cfg/cfg.cpp b/src/core/hle/service/cfg/cfg.cpp
index caa41ded7..5a7878b31 100644
--- a/src/core/hle/service/cfg/cfg.cpp
+++ b/src/core/hle/service/cfg/cfg.cpp
@@ -13,6 +13,8 @@
#include "core/file_sys/archive_systemsavedata.h"
#include "core/file_sys/errors.h"
#include "core/file_sys/file_backend.h"
+#include "core/hle/ipc.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/result.h"
#include "core/hle/service/cfg/cfg.h"
#include "core/hle/service/cfg/cfg_i.h"
@@ -21,6 +23,7 @@
#include "core/hle/service/cfg/cfg_u.h"
#include "core/hle/service/fs/archive.h"
#include "core/hle/service/service.h"
+#include "core/memory.h"
#include "core/settings.h"
namespace Service {
diff --git a/src/core/hle/service/csnd_snd.cpp b/src/core/hle/service/csnd_snd.cpp
index 6cf62f9bc..1455f20ca 100644
--- a/src/core/hle/service/csnd_snd.cpp
+++ b/src/core/hle/service/csnd_snd.cpp
@@ -4,9 +4,12 @@
#include <cstring>
#include "common/alignment.h"
+#include "core/hle/ipc.h"
+#include "core/hle/kernel/handle_table.h"
#include "core/hle/kernel/mutex.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/service/csnd_snd.h"
+#include "core/memory.h"
namespace Service {
namespace CSND {
diff --git a/src/core/hle/service/dlp/dlp_srvr.cpp b/src/core/hle/service/dlp/dlp_srvr.cpp
index 25c07f401..32cfa2c44 100644
--- a/src/core/hle/service/dlp/dlp_srvr.cpp
+++ b/src/core/hle/service/dlp/dlp_srvr.cpp
@@ -4,6 +4,7 @@
#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/result.h"
#include "core/hle/service/dlp/dlp_srvr.h"
diff --git a/src/core/hle/service/dsp_dsp.cpp b/src/core/hle/service/dsp_dsp.cpp
index 79171a0bc..363066d14 100644
--- a/src/core/hle/service/dsp_dsp.cpp
+++ b/src/core/hle/service/dsp_dsp.cpp
@@ -3,12 +3,18 @@
// Refer to the license.txt file included.
#include <algorithm>
+#include <array>
#include <cinttypes>
#include "audio_core/hle/pipe.h"
+#include "common/assert.h"
#include "common/hash.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
+#include "core/hle/result.h"
#include "core/hle/service/dsp_dsp.h"
+#include "core/memory.h"
using DspPipe = DSP::HLE::DspPipe;
diff --git a/src/core/hle/service/err_f.cpp b/src/core/hle/service/err_f.cpp
index 9da55f328..7c8f4339f 100644
--- a/src/core/hle/service/err_f.cpp
+++ b/src/core/hle/service/err_f.cpp
@@ -6,10 +6,11 @@
#include <chrono>
#include <iomanip>
#include <sstream>
-
#include "common/bit_field.h"
#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/core.h"
+#include "core/hle/ipc.h"
#include "core/hle/result.h"
#include "core/hle/service/err_f.h"
@@ -172,6 +173,7 @@ static void ThrowFatalError(Interface* self) {
const ErrInfo* errinfo = reinterpret_cast<ErrInfo*>(&cmd_buff[1]);
LOG_CRITICAL(Service_ERR, "Fatal error type: %s",
GetErrType(errinfo->errinfo_common.specifier).c_str());
+ Core::System::GetInstance().SetStatus(Core::System::ResultStatus::ErrorUnknown);
// Generic Info
LogGenericInfo(errinfo->errinfo_common);
diff --git a/src/core/hle/service/frd/frd.cpp b/src/core/hle/service/frd/frd.cpp
index 34fdf7f53..76ecda8b7 100644
--- a/src/core/hle/service/frd/frd.cpp
+++ b/src/core/hle/service/frd/frd.cpp
@@ -2,11 +2,16 @@
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
+#include "common/assert.h"
+#include "common/logging/log.h"
#include "common/string_util.h"
+#include "core/hle/ipc.h"
+#include "core/hle/result.h"
#include "core/hle/service/frd/frd.h"
#include "core/hle/service/frd/frd_a.h"
#include "core/hle/service/frd/frd_u.h"
#include "core/hle/service/service.h"
+#include "core/memory.h"
namespace Service {
namespace FRD {
diff --git a/src/core/hle/service/fs/archive.cpp b/src/core/hle/service/fs/archive.cpp
index 632712f2c..3605ef175 100644
--- a/src/core/hle/service/fs/archive.cpp
+++ b/src/core/hle/service/fs/archive.cpp
@@ -24,7 +24,11 @@
#include "core/file_sys/directory_backend.h"
#include "core/file_sys/errors.h"
#include "core/file_sys/file_backend.h"
+#include "core/hle/ipc.h"
+#include "core/hle/kernel/client_port.h"
#include "core/hle/kernel/client_session.h"
+#include "core/hle/kernel/handle_table.h"
+#include "core/hle/kernel/server_session.h"
#include "core/hle/result.h"
#include "core/hle/service/fs/archive.h"
#include "core/hle/service/fs/fs_user.h"
@@ -83,6 +87,10 @@ File::File(std::unique_ptr<FileSys::FileBackend>&& backend, const FileSys::Path&
File::~File() {}
void File::HandleSyncRequest(Kernel::SharedPtr<Kernel::ServerSession> server_session) {
+ using Kernel::ClientSession;
+ using Kernel::ServerSession;
+ using Kernel::SharedPtr;
+
u32* cmd_buff = Kernel::GetCommandBuffer();
FileCommand cmd = static_cast<FileCommand>(cmd_buff[0]);
switch (cmd) {
@@ -161,10 +169,9 @@ void File::HandleSyncRequest(Kernel::SharedPtr<Kernel::ServerSession> server_ses
case FileCommand::OpenLinkFile: {
LOG_WARNING(Service_FS, "(STUBBED) File command OpenLinkFile %s", GetName().c_str());
- auto sessions = Kernel::ServerSession::CreateSessionPair(GetName(), shared_from_this());
- ClientConnected(std::get<Kernel::SharedPtr<Kernel::ServerSession>>(sessions));
- cmd_buff[3] = Kernel::g_handle_table
- .Create(std::get<Kernel::SharedPtr<Kernel::ClientSession>>(sessions))
+ auto sessions = ServerSession::CreateSessionPair(GetName());
+ ClientConnected(std::get<SharedPtr<ServerSession>>(sessions));
+ cmd_buff[3] = Kernel::g_handle_table.Create(std::get<SharedPtr<ClientSession>>(sessions))
.ValueOr(INVALID_HANDLE);
break;
}
@@ -258,9 +265,7 @@ ResultVal<ArchiveHandle> OpenArchive(ArchiveIdCode id_code, FileSys::Path& archi
auto itr = id_code_map.find(id_code);
if (itr == id_code_map.end()) {
- // TODO: Verify error against hardware
- return ResultCode(ErrorDescription::NotFound, ErrorModule::FS, ErrorSummary::NotFound,
- ErrorLevel::Permanent);
+ return FileSys::ERROR_NOT_FOUND;
}
CASCADE_RESULT(std::unique_ptr<ArchiveBackend> res, itr->second->Open(archive_path));
diff --git a/src/core/hle/service/fs/archive.h b/src/core/hle/service/fs/archive.h
index 2ea956e0b..3a3371c88 100644
--- a/src/core/hle/service/fs/archive.h
+++ b/src/core/hle/service/fs/archive.h
@@ -8,7 +8,7 @@
#include <string>
#include "common/common_types.h"
#include "core/file_sys/archive_backend.h"
-#include "core/hle/kernel/server_session.h"
+#include "core/hle/kernel/hle_ipc.h"
#include "core/hle/result.h"
namespace FileSys {
@@ -43,7 +43,7 @@ enum class MediaType : u32 { NAND = 0, SDMC = 1, GameCard = 2 };
typedef u64 ArchiveHandle;
-class File final : public SessionRequestHandler, public std::enable_shared_from_this<File> {
+class File final : public Kernel::SessionRequestHandler {
public:
File(std::unique_ptr<FileSys::FileBackend>&& backend, const FileSys::Path& path);
~File();
@@ -60,7 +60,7 @@ protected:
void HandleSyncRequest(Kernel::SharedPtr<Kernel::ServerSession> server_session) override;
};
-class Directory final : public SessionRequestHandler {
+class Directory final : public Kernel::SessionRequestHandler {
public:
Directory(std::unique_ptr<FileSys::DirectoryBackend>&& backend, const FileSys::Path& path);
~Directory();
diff --git a/src/core/hle/service/fs/fs_user.cpp b/src/core/hle/service/fs/fs_user.cpp
index e53a970d3..34e1783ec 100644
--- a/src/core/hle/service/fs/fs_user.cpp
+++ b/src/core/hle/service/fs/fs_user.cpp
@@ -8,8 +8,13 @@
#include "common/logging/log.h"
#include "common/scope_exit.h"
#include "common/string_util.h"
+#include "core/core.h"
#include "core/file_sys/errors.h"
+#include "core/hle/ipc.h"
+#include "core/hle/ipc_helpers.h"
+#include "core/hle/kernel/client_port.h"
#include "core/hle/kernel/client_session.h"
+#include "core/hle/kernel/server_session.h"
#include "core/hle/result.h"
#include "core/hle/service/fs/archive.h"
#include "core/hle/service/fs/fs_user.h"
@@ -18,8 +23,9 @@
////////////////////////////////////////////////////////////////////////////////////////////////////
// Namespace FS_User
-using Kernel::SharedPtr;
+using Kernel::ClientSession;
using Kernel::ServerSession;
+using Kernel::SharedPtr;
namespace Service {
namespace FS {
@@ -77,11 +83,11 @@ static void OpenFile(Service::Interface* self) {
rb.Push(file_res.Code());
if (file_res.Succeeded()) {
std::shared_ptr<File> file = *file_res;
- auto sessions = ServerSession::CreateSessionPair(file->GetName(), file);
- file->ClientConnected(std::get<Kernel::SharedPtr<Kernel::ServerSession>>(sessions));
- rb.PushMoveHandles(Kernel::g_handle_table
- .Create(std::get<Kernel::SharedPtr<Kernel::ClientSession>>(sessions))
- .MoveFrom());
+ auto sessions = ServerSession::CreateSessionPair(file->GetName());
+ file->ClientConnected(std::get<SharedPtr<ServerSession>>(sessions));
+
+ rb.PushMoveHandles(
+ Kernel::g_handle_table.Create(std::get<SharedPtr<ClientSession>>(sessions)).MoveFrom());
} else {
rb.PushMoveHandles(0);
LOG_ERROR(Service_FS, "failed to get a handle for file %s", file_path.DebugStr().c_str());
@@ -130,7 +136,7 @@ static void OpenFileDirectly(Service::Interface* self) {
ResultVal<ArchiveHandle> archive_handle = OpenArchive(archive_id, archive_path);
if (archive_handle.Failed()) {
LOG_ERROR(Service_FS,
- "failed to get a handle for archive archive_id=0x%08X archive_path=%s",
+ "Failed to get a handle for archive archive_id=0x%08X archive_path=%s",
static_cast<u32>(archive_id), archive_path.DebugStr().c_str());
cmd_buff[1] = archive_handle.Code().raw;
cmd_buff[3] = 0;
@@ -143,11 +149,11 @@ static void OpenFileDirectly(Service::Interface* self) {
cmd_buff[1] = file_res.Code().raw;
if (file_res.Succeeded()) {
std::shared_ptr<File> file = *file_res;
- auto sessions = ServerSession::CreateSessionPair(file->GetName(), file);
- file->ClientConnected(std::get<Kernel::SharedPtr<Kernel::ServerSession>>(sessions));
- cmd_buff[3] = Kernel::g_handle_table
- .Create(std::get<Kernel::SharedPtr<Kernel::ClientSession>>(sessions))
- .MoveFrom();
+ auto sessions = ServerSession::CreateSessionPair(file->GetName());
+ file->ClientConnected(std::get<SharedPtr<ServerSession>>(sessions));
+
+ cmd_buff[3] =
+ Kernel::g_handle_table.Create(std::get<SharedPtr<ClientSession>>(sessions)).MoveFrom();
} else {
cmd_buff[3] = 0;
LOG_ERROR(Service_FS, "failed to get a handle for file %s mode=%u attributes=%u",
@@ -410,11 +416,11 @@ static void OpenDirectory(Service::Interface* self) {
cmd_buff[1] = dir_res.Code().raw;
if (dir_res.Succeeded()) {
std::shared_ptr<Directory> directory = *dir_res;
- auto sessions = ServerSession::CreateSessionPair(directory->GetName(), directory);
- directory->ClientConnected(std::get<Kernel::SharedPtr<Kernel::ServerSession>>(sessions));
- cmd_buff[3] = Kernel::g_handle_table
- .Create(std::get<Kernel::SharedPtr<Kernel::ClientSession>>(sessions))
- .MoveFrom();
+ auto sessions = ServerSession::CreateSessionPair(directory->GetName());
+ directory->ClientConnected(std::get<SharedPtr<ServerSession>>(sessions));
+
+ cmd_buff[3] =
+ Kernel::g_handle_table.Create(std::get<SharedPtr<ClientSession>>(sessions)).MoveFrom();
} else {
LOG_ERROR(Service_FS, "failed to get a handle for directory type=%d size=%d data=%s",
dirname_type, dirname_size, dir_path.DebugStr().c_str());
diff --git a/src/core/hle/service/gsp_gpu.cpp b/src/core/hle/service/gsp_gpu.cpp
index 94f6b8a9c..6ff0f4812 100644
--- a/src/core/hle/service/gsp_gpu.cpp
+++ b/src/core/hle/service/gsp_gpu.cpp
@@ -5,7 +5,9 @@
#include "common/bit_field.h"
#include "common/microprofile.h"
#include "core/core.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/result.h"
#include "core/hle/service/gsp_gpu.h"
diff --git a/src/core/hle/service/hid/hid.cpp b/src/core/hle/service/hid/hid.cpp
index 64d01cdd7..5255f6dc8 100644
--- a/src/core/hle/service/hid/hid.cpp
+++ b/src/core/hle/service/hid/hid.cpp
@@ -10,7 +10,9 @@
#include "core/core_timing.h"
#include "core/frontend/emu_window.h"
#include "core/frontend/input.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/service/hid/hid.h"
#include "core/hle/service/hid/hid_spvr.h"
diff --git a/src/core/hle/service/ir/ir_rst.cpp b/src/core/hle/service/ir/ir_rst.cpp
index 53807cd91..0de698003 100644
--- a/src/core/hle/service/ir/ir_rst.cpp
+++ b/src/core/hle/service/ir/ir_rst.cpp
@@ -6,6 +6,7 @@
#include "common/bit_field.h"
#include "core/core_timing.h"
#include "core/frontend/input.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/kernel/event.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/service/hid/hid.h"
diff --git a/src/core/hle/service/ir/ir_user.cpp b/src/core/hle/service/ir/ir_user.cpp
index 369115f09..fdecdce64 100644
--- a/src/core/hle/service/ir/ir_user.cpp
+++ b/src/core/hle/service/ir/ir_user.cpp
@@ -7,6 +7,7 @@
#include <boost/optional.hpp>
#include "common/string_util.h"
#include "common/swap.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/kernel/event.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/service/ir/extra_hid.h"
diff --git a/src/core/hle/service/ldr_ro/ldr_ro.cpp b/src/core/hle/service/ldr_ro/ldr_ro.cpp
index d1e6d869f..7255ea026 100644
--- a/src/core/hle/service/ldr_ro/ldr_ro.cpp
+++ b/src/core/hle/service/ldr_ro/ldr_ro.cpp
@@ -7,6 +7,7 @@
#include "common/logging/log.h"
#include "core/arm/arm_interface.h"
#include "core/core.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/kernel/process.h"
#include "core/hle/kernel/vm_manager.h"
#include "core/hle/service/ldr_ro/cro_helper.h"
diff --git a/src/core/hle/service/mic_u.cpp b/src/core/hle/service/mic_u.cpp
index e98388560..35212b59b 100644
--- a/src/core/hle/service/mic_u.cpp
+++ b/src/core/hle/service/mic_u.cpp
@@ -3,7 +3,9 @@
// Refer to the license.txt file included.
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
#include "core/hle/kernel/kernel.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/service/mic_u.h"
diff --git a/src/core/hle/service/ndm/ndm.cpp b/src/core/hle/service/ndm/ndm.cpp
index 5eb97f0d3..096c0cdac 100644
--- a/src/core/hle/service/ndm/ndm.cpp
+++ b/src/core/hle/service/ndm/ndm.cpp
@@ -2,8 +2,10 @@
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
+#include <array>
#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/service/ndm/ndm.h"
#include "core/hle/service/ndm/ndm_u.h"
#include "core/hle/service/service.h"
diff --git a/src/core/hle/service/nfc/nfc.cpp b/src/core/hle/service/nfc/nfc.cpp
index fd3c7d9c2..b44a9f668 100644
--- a/src/core/hle/service/nfc/nfc.cpp
+++ b/src/core/hle/service/nfc/nfc.cpp
@@ -2,7 +2,9 @@
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
+#include "core/hle/ipc.h"
#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
#include "core/hle/service/nfc/nfc.h"
#include "core/hle/service/nfc/nfc_m.h"
#include "core/hle/service/nfc/nfc_u.h"
diff --git a/src/core/hle/service/nim/nim.cpp b/src/core/hle/service/nim/nim.cpp
index 63c334cb2..d5624fe54 100644
--- a/src/core/hle/service/nim/nim.cpp
+++ b/src/core/hle/service/nim/nim.cpp
@@ -4,6 +4,7 @@
#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
#include "core/hle/service/nim/nim.h"
#include "core/hle/service/nim/nim_aoc.h"
#include "core/hle/service/nim/nim_s.h"
diff --git a/src/core/hle/service/nwm/nwm_uds.cpp b/src/core/hle/service/nwm/nwm_uds.cpp
index a82cc9bd3..6c4600f25 100644
--- a/src/core/hle/service/nwm/nwm_uds.cpp
+++ b/src/core/hle/service/nwm/nwm_uds.cpp
@@ -9,6 +9,7 @@
#include "common/common_types.h"
#include "common/logging/log.h"
#include "core/core_timing.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/kernel/event.h"
#include "core/hle/kernel/shared_memory.h"
#include "core/hle/result.h"
diff --git a/src/core/hle/service/nwm/uds_beacon.cpp b/src/core/hle/service/nwm/uds_beacon.cpp
index c6e5bc5f1..6332b404c 100644
--- a/src/core/hle/service/nwm/uds_beacon.cpp
+++ b/src/core/hle/service/nwm/uds_beacon.cpp
@@ -3,14 +3,13 @@
// Refer to the license.txt file included.
#include <cstring>
-
-#include "core/hle/service/nwm/nwm_uds.h"
-#include "core/hle/service/nwm/uds_beacon.h"
-
#include <cryptopp/aes.h>
#include <cryptopp/md5.h>
#include <cryptopp/modes.h>
#include <cryptopp/sha.h>
+#include "common/assert.h"
+#include "core/hle/service/nwm/nwm_uds.h"
+#include "core/hle/service/nwm/uds_beacon.h"
namespace Service {
namespace NWM {
diff --git a/src/core/hle/service/service.cpp b/src/core/hle/service/service.cpp
index 0672ac2e3..0d443aa44 100644
--- a/src/core/hle/service/service.cpp
+++ b/src/core/hle/service/service.cpp
@@ -2,11 +2,11 @@
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
-#include <boost/range/algorithm_ext/erase.hpp>
-
#include "common/logging/log.h"
#include "common/string_util.h"
+#include "core/hle/kernel/client_port.h"
#include "core/hle/kernel/server_port.h"
+#include "core/hle/kernel/server_session.h"
#include "core/hle/service/ac/ac.h"
#include "core/hle/service/act/act.h"
#include "core/hle/service/am/am.h"
@@ -39,15 +39,15 @@
#include "core/hle/service/ptm/ptm.h"
#include "core/hle/service/qtm/qtm.h"
#include "core/hle/service/service.h"
+#include "core/hle/service/sm/sm.h"
+#include "core/hle/service/sm/srv.h"
#include "core/hle/service/soc_u.h"
-#include "core/hle/service/srv.h"
#include "core/hle/service/ssl_c.h"
#include "core/hle/service/y2r_u.h"
namespace Service {
std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
-std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
/**
* Creates a function string for logging, complete with the name (or header code, depending
@@ -66,16 +66,6 @@ static std::string MakeFunctionString(const char* name, const char* port_name,
return function_string;
}
-void SessionRequestHandler::ClientConnected(
- Kernel::SharedPtr<Kernel::ServerSession> server_session) {
- connected_sessions.push_back(server_session);
-}
-
-void SessionRequestHandler::ClientDisconnected(
- Kernel::SharedPtr<Kernel::ServerSession> server_session) {
- boost::range::remove_erase(connected_sessions, server_session);
-}
-
Interface::Interface(u32 max_sessions) : max_sessions(max_sessions) {}
Interface::~Interface() = default;
@@ -116,24 +106,27 @@ void Interface::Register(const FunctionInfo* functions, size_t n) {
// Module interface
static void AddNamedPort(Interface* interface_) {
- auto ports =
- Kernel::ServerPort::CreatePortPair(interface_->GetMaxSessions(), interface_->GetPortName(),
- std::shared_ptr<Interface>(interface_));
- auto client_port = std::get<Kernel::SharedPtr<Kernel::ClientPort>>(ports);
+ Kernel::SharedPtr<Kernel::ServerPort> server_port;
+ Kernel::SharedPtr<Kernel::ClientPort> client_port;
+ std::tie(server_port, client_port) =
+ Kernel::ServerPort::CreatePortPair(interface_->GetMaxSessions(), interface_->GetPortName());
+
+ server_port->SetHleHandler(std::shared_ptr<Interface>(interface_));
g_kernel_named_ports.emplace(interface_->GetPortName(), std::move(client_port));
}
void AddService(Interface* interface_) {
- auto ports =
- Kernel::ServerPort::CreatePortPair(interface_->GetMaxSessions(), interface_->GetPortName(),
- std::shared_ptr<Interface>(interface_));
- auto client_port = std::get<Kernel::SharedPtr<Kernel::ClientPort>>(ports);
- g_srv_services.emplace(interface_->GetPortName(), std::move(client_port));
+ auto server_port =
+ SM::g_service_manager
+ ->RegisterService(interface_->GetPortName(), interface_->GetMaxSessions())
+ .MoveFrom();
+ server_port->SetHleHandler(std::shared_ptr<Interface>(interface_));
}
/// Initialize ServiceManager
void Init() {
- AddNamedPort(new SRV::SRV);
+ SM::g_service_manager = std::make_unique<SM::ServiceManager>();
+ AddNamedPort(new SM::SRV);
AddNamedPort(new ERR::ERR_F);
FS::ArchiveInit();
@@ -194,7 +187,7 @@ void Shutdown() {
AC::Shutdown();
FS::ArchiveShutdown();
- g_srv_services.clear();
+ SM::g_service_manager = nullptr;
g_kernel_named_ports.clear();
LOG_DEBUG(Service, "shutdown OK");
}
diff --git a/src/core/hle/service/service.h b/src/core/hle/service/service.h
index e6a5f1417..8933d57cc 100644
--- a/src/core/hle/service/service.h
+++ b/src/core/hle/service/service.h
@@ -8,21 +8,19 @@
#include <string>
#include <unordered_map>
#include <boost/container/flat_map.hpp>
+#include "common/bit_field.h"
#include "common/common_types.h"
-#include "core/hle/ipc.h"
-#include "core/hle/ipc_helpers.h"
-#include "core/hle/kernel/client_port.h"
-#include "core/hle/kernel/thread.h"
-#include "core/hle/result.h"
-#include "core/memory.h"
+#include "core/hle/kernel/hle_ipc.h"
+#include "core/hle/kernel/kernel.h"
+
+////////////////////////////////////////////////////////////////////////////////////////////////////
+// Namespace Service
namespace Kernel {
+class ClientPort;
class ServerSession;
}
-////////////////////////////////////////////////////////////////////////////////////////////////////
-// Namespace Service
-
namespace Service {
static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 characters)
@@ -30,48 +28,10 @@ static const int kMaxPortSize = 8; ///< Maximum size of a port name (8 character
static const u32 DefaultMaxSessions = 10;
/**
- * Interface implemented by HLE Session handlers.
- * This can be provided to a ServerSession in order to hook into several relevant events
- * (such as a new connection or a SyncRequest) so they can be implemented in the emulator.
- */
-class SessionRequestHandler {
-public:
- /**
- * Handles a sync request from the emulated application.
- * @param server_session The ServerSession that was triggered for this sync request,
- * it should be used to differentiate which client (As in ClientSession) we're answering to.
- * TODO(Subv): Use a wrapper structure to hold all the information relevant to
- * this request (ServerSession, Originator thread, Translated command buffer, etc).
- * @returns ResultCode the result code of the translate operation.
- */
- virtual void HandleSyncRequest(Kernel::SharedPtr<Kernel::ServerSession> server_session) = 0;
-
- /**
- * Signals that a client has just connected to this HLE handler and keeps the
- * associated ServerSession alive for the duration of the connection.
- * @param server_session Owning pointer to the ServerSession associated with the connection.
- */
- void ClientConnected(Kernel::SharedPtr<Kernel::ServerSession> server_session);
-
- /**
- * Signals that a client has just disconnected from this HLE handler and releases the
- * associated ServerSession.
- * @param server_session ServerSession associated with the connection.
- */
- void ClientDisconnected(Kernel::SharedPtr<Kernel::ServerSession> server_session);
-
-protected:
- /// List of sessions that are connected to this handler.
- /// A ServerSession whose server endpoint is an HLE implementation is kept alive by this list
- // for the duration of the connection.
- std::vector<Kernel::SharedPtr<Kernel::ServerSession>> connected_sessions;
-};
-
-/**
* Framework for implementing HLE service handlers which dispatch incoming SyncRequests based on a
* table mapping header ids to handler functions.
*/
-class Interface : public SessionRequestHandler {
+class Interface : public Kernel::SessionRequestHandler {
public:
/**
* Creates an HLE interface with the specified max sessions.
@@ -149,8 +109,6 @@ void Shutdown();
/// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC.
extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
-/// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
-extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
/// Adds a service to the services table
void AddService(Interface* interface_);
diff --git a/src/core/hle/service/sm/sm.cpp b/src/core/hle/service/sm/sm.cpp
new file mode 100644
index 000000000..361f7a0a9
--- /dev/null
+++ b/src/core/hle/service/sm/sm.cpp
@@ -0,0 +1,59 @@
+// Copyright 2017 Citra Emulator Project
+// Licensed under GPLv2 or any later version
+// Refer to the license.txt file included.
+
+#include <tuple>
+#include "core/hle/kernel/client_port.h"
+#include "core/hle/kernel/client_session.h"
+#include "core/hle/kernel/server_port.h"
+#include "core/hle/result.h"
+#include "core/hle/service/sm/sm.h"
+
+namespace Service {
+namespace SM {
+
+static ResultCode ValidateServiceName(const std::string& name) {
+ if (name.size() <= 0 || name.size() > 8) {
+ return ERR_INVALID_NAME_SIZE;
+ }
+ if (name.find('\0') != std::string::npos) {
+ return ERR_NAME_CONTAINS_NUL;
+ }
+ return RESULT_SUCCESS;
+}
+
+ResultVal<Kernel::SharedPtr<Kernel::ServerPort>> ServiceManager::RegisterService(
+ std::string name, unsigned int max_sessions) {
+
+ CASCADE_CODE(ValidateServiceName(name));
+ Kernel::SharedPtr<Kernel::ServerPort> server_port;
+ Kernel::SharedPtr<Kernel::ClientPort> client_port;
+ std::tie(server_port, client_port) = Kernel::ServerPort::CreatePortPair(max_sessions, name);
+
+ registered_services.emplace(name, std::move(client_port));
+ return MakeResult<Kernel::SharedPtr<Kernel::ServerPort>>(std::move(server_port));
+}
+
+ResultVal<Kernel::SharedPtr<Kernel::ClientPort>> ServiceManager::GetServicePort(
+ const std::string& name) {
+
+ CASCADE_CODE(ValidateServiceName(name));
+ auto it = registered_services.find(name);
+ if (it == registered_services.end()) {
+ return ERR_SERVICE_NOT_REGISTERED;
+ }
+
+ return MakeResult<Kernel::SharedPtr<Kernel::ClientPort>>(it->second);
+}
+
+ResultVal<Kernel::SharedPtr<Kernel::ClientSession>> ServiceManager::ConnectToService(
+ const std::string& name) {
+
+ CASCADE_RESULT(auto client_port, GetServicePort(name));
+ return client_port->Connect();
+}
+
+std::unique_ptr<ServiceManager> g_service_manager;
+
+} // namespace SM
+} // namespace Service
diff --git a/src/core/hle/service/sm/sm.h b/src/core/hle/service/sm/sm.h
new file mode 100644
index 000000000..5fac5455c
--- /dev/null
+++ b/src/core/hle/service/sm/sm.h
@@ -0,0 +1,49 @@
+// Copyright 2017 Citra Emulator Project
+// Licensed under GPLv2 or any later version
+// Refer to the license.txt file included.
+
+#pragma once
+
+#include <string>
+#include <unordered_map>
+#include "core/hle/kernel/kernel.h"
+#include "core/hle/result.h"
+#include "core/hle/service/service.h"
+
+namespace Kernel {
+class ClientPort;
+class ClientSession;
+class ServerPort;
+class SessionRequestHandler;
+} // namespace Kernel
+
+namespace Service {
+namespace SM {
+
+constexpr ResultCode ERR_SERVICE_NOT_REGISTERED(1, ErrorModule::SRV, ErrorSummary::WouldBlock,
+ ErrorLevel::Temporary); // 0xD0406401
+constexpr ResultCode ERR_MAX_CONNECTIONS_REACHED(2, ErrorModule::SRV, ErrorSummary::WouldBlock,
+ ErrorLevel::Temporary); // 0xD0406402
+constexpr ResultCode ERR_INVALID_NAME_SIZE(5, ErrorModule::SRV, ErrorSummary::WrongArgument,
+ ErrorLevel::Permanent); // 0xD9006405
+constexpr ResultCode ERR_ACCESS_DENIED(6, ErrorModule::SRV, ErrorSummary::InvalidArgument,
+ ErrorLevel::Permanent); // 0xD8E06406
+constexpr ResultCode ERR_NAME_CONTAINS_NUL(7, ErrorModule::SRV, ErrorSummary::WrongArgument,
+ ErrorLevel::Permanent); // 0xD9006407
+
+class ServiceManager {
+public:
+ ResultVal<Kernel::SharedPtr<Kernel::ServerPort>> RegisterService(std::string name,
+ unsigned int max_sessions);
+ ResultVal<Kernel::SharedPtr<Kernel::ClientPort>> GetServicePort(const std::string& name);
+ ResultVal<Kernel::SharedPtr<Kernel::ClientSession>> ConnectToService(const std::string& name);
+
+private:
+ /// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
+ std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> registered_services;
+};
+
+extern std::unique_ptr<ServiceManager> g_service_manager;
+
+} // namespace SM
+} // namespace Service
diff --git a/src/core/hle/service/srv.cpp b/src/core/hle/service/sm/srv.cpp
index 130c9d25e..063b1b0fc 100644
--- a/src/core/hle/service/srv.cpp
+++ b/src/core/hle/service/sm/srv.cpp
@@ -6,15 +6,21 @@
#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
+#include "core/hle/kernel/client_port.h"
#include "core/hle/kernel/client_session.h"
-#include "core/hle/kernel/event.h"
+#include "core/hle/kernel/handle_table.h"
+#include "core/hle/kernel/semaphore.h"
#include "core/hle/kernel/server_session.h"
-#include "core/hle/service/srv.h"
+#include "core/hle/service/sm/sm.h"
+#include "core/hle/service/sm/srv.h"
namespace Service {
-namespace SRV {
+namespace SM {
-static Kernel::SharedPtr<Kernel::Event> event_handle;
+constexpr int MAX_PENDING_NOTIFICATIONS = 16;
+
+static Kernel::SharedPtr<Kernel::Semaphore> notification_semaphore;
/**
* SRV::RegisterClient service function
@@ -51,14 +57,13 @@ static void RegisterClient(Interface* self) {
static void EnableNotification(Interface* self) {
u32* cmd_buff = Kernel::GetCommandBuffer();
- // TODO(bunnei): Change to a semaphore once these have been implemented
- event_handle = Kernel::Event::Create(Kernel::ResetType::OneShot, "SRV:Event");
- event_handle->Clear();
+ notification_semaphore =
+ Kernel::Semaphore::Create(0, MAX_PENDING_NOTIFICATIONS, "SRV:Notification").Unwrap();
cmd_buff[0] = IPC::MakeHeader(0x2, 0x1, 0x2); // 0x20042
cmd_buff[1] = RESULT_SUCCESS.raw; // No error
cmd_buff[2] = IPC::CopyHandleDesc(1);
- cmd_buff[3] = Kernel::g_handle_table.Create(event_handle).MoveFrom();
+ cmd_buff[3] = Kernel::g_handle_table.Create(notification_semaphore).MoveFrom();
LOG_WARNING(Service_SRV, "(STUBBED) called");
}
@@ -77,25 +82,41 @@ static void GetServiceHandle(Interface* self) {
ResultCode res = RESULT_SUCCESS;
u32* cmd_buff = Kernel::GetCommandBuffer();
- std::string port_name = std::string((const char*)&cmd_buff[1], 0, Service::kMaxPortSize);
- auto it = Service::g_srv_services.find(port_name);
+ size_t name_len = cmd_buff[3];
+ if (name_len > Service::kMaxPortSize) {
+ cmd_buff[1] = ERR_INVALID_NAME_SIZE.raw;
+ LOG_ERROR(Service_SRV, "called name_len=0x%X, failed with code=0x%08X", name_len,
+ cmd_buff[1]);
+ return;
+ }
+ std::string name(reinterpret_cast<const char*>(&cmd_buff[1]), name_len);
+ bool return_port_on_failure = (cmd_buff[4] & 1) == 0;
- if (it != Service::g_srv_services.end()) {
- auto client_port = it->second;
+ // TODO(yuriks): Permission checks go here
- auto client_session = client_port->Connect();
- res = client_session.Code();
+ auto client_port = g_service_manager->GetServicePort(name);
+ if (client_port.Failed()) {
+ cmd_buff[1] = client_port.Code().raw;
+ LOG_ERROR(Service_SRV, "called service=%s, failed with code=0x%08X", name.c_str(),
+ cmd_buff[1]);
+ return;
+ }
- if (client_session.Succeeded()) {
- // Return the client session
- cmd_buff[3] = Kernel::g_handle_table.Create(*client_session).MoveFrom();
- }
- LOG_TRACE(Service_SRV, "called port=%s, handle=0x%08X", port_name.c_str(), cmd_buff[3]);
+ auto session = client_port.Unwrap()->Connect();
+ cmd_buff[1] = session.Code().raw;
+ if (session.Succeeded()) {
+ cmd_buff[3] = Kernel::g_handle_table.Create(session.MoveFrom()).MoveFrom();
+ LOG_DEBUG(Service_SRV, "called service=%s, session handle=0x%08X", name.c_str(),
+ cmd_buff[3]);
+ } else if (session.Code() == Kernel::ERR_MAX_CONNECTIONS_REACHED && return_port_on_failure) {
+ cmd_buff[1] = ERR_MAX_CONNECTIONS_REACHED.raw;
+ cmd_buff[3] = Kernel::g_handle_table.Create(client_port.MoveFrom()).MoveFrom();
+ LOG_WARNING(Service_SRV, "called service=%s, *port* handle=0x%08X", name.c_str(),
+ cmd_buff[3]);
} else {
- LOG_ERROR(Service_SRV, "(UNIMPLEMENTED) called port=%s", port_name.c_str());
- res = UnimplementedFunction(ErrorModule::SRV);
+ LOG_ERROR(Service_SRV, "called service=%s, failed with code=0x%08X", name.c_str(),
+ cmd_buff[1]);
}
- cmd_buff[1] = res.raw;
}
/**
@@ -177,12 +198,12 @@ const Interface::FunctionInfo FunctionTable[] = {
SRV::SRV() {
Register(FunctionTable);
- event_handle = nullptr;
+ notification_semaphore = nullptr;
}
SRV::~SRV() {
- event_handle = nullptr;
+ notification_semaphore = nullptr;
}
-} // namespace SRV
+} // namespace SM
} // namespace Service
diff --git a/src/core/hle/service/srv.h b/src/core/hle/service/sm/srv.h
index d3a9de879..4196ca1e2 100644
--- a/src/core/hle/service/srv.h
+++ b/src/core/hle/service/sm/srv.h
@@ -4,10 +4,11 @@
#pragma once
+#include <string>
#include "core/hle/service/service.h"
namespace Service {
-namespace SRV {
+namespace SM {
/// Interface to "srv:" service
class SRV final : public Interface {
@@ -20,5 +21,5 @@ public:
}
};
-} // namespace SRV
+} // namespace SM
} // namespace Service
diff --git a/src/core/hle/service/soc_u.cpp b/src/core/hle/service/soc_u.cpp
index 530614e6f..3d215d42d 100644
--- a/src/core/hle/service/soc_u.cpp
+++ b/src/core/hle/service/soc_u.cpp
@@ -11,6 +11,7 @@
#include "common/common_types.h"
#include "common/logging/log.h"
#include "common/scope_exit.h"
+#include "core/hle/ipc.h"
#include "core/hle/kernel/server_session.h"
#include "core/hle/result.h"
#include "core/hle/service/soc_u.h"
diff --git a/src/core/hle/service/ssl_c.cpp b/src/core/hle/service/ssl_c.cpp
index 09ced9d7a..300acca75 100644
--- a/src/core/hle/service/ssl_c.cpp
+++ b/src/core/hle/service/ssl_c.cpp
@@ -4,7 +4,9 @@
#include <random>
#include "common/common_types.h"
+#include "core/hle/ipc.h"
#include "core/hle/service/ssl_c.h"
+#include "core/memory.h"
namespace Service {
namespace SSL {
diff --git a/src/core/hle/service/y2r_u.cpp b/src/core/hle/service/y2r_u.cpp
index c0837d49d..bb7bf2d67 100644
--- a/src/core/hle/service/y2r_u.cpp
+++ b/src/core/hle/service/y2r_u.cpp
@@ -6,6 +6,8 @@
#include "common/common_funcs.h"
#include "common/common_types.h"
#include "common/logging/log.h"
+#include "core/hle/ipc.h"
+#include "core/hle/ipc_helpers.h"
#include "core/hle/kernel/event.h"
#include "core/hle/kernel/kernel.h"
#include "core/hle/service/y2r_u.h"