summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/core/CMakeLists.txt6
-rw-r--r--src/core/hle/service/service.cpp19
-rw-r--r--src/core/hle/service/service.h2
-rw-r--r--src/core/hle/service/sm/sm.cpp58
-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)68
-rw-r--r--src/core/hle/service/sm/srv.h (renamed from src/core/hle/service/srv.h)5
7 files changed, 166 insertions, 41 deletions
diff --git a/src/core/CMakeLists.txt b/src/core/CMakeLists.txt
index 51ee80bc4..6e602b0c5 100644
--- a/src/core/CMakeLists.txt
+++ b/src/core/CMakeLists.txt
@@ -156,8 +156,9 @@ set(SRCS
156 hle/service/qtm/qtm_sp.cpp 156 hle/service/qtm/qtm_sp.cpp
157 hle/service/qtm/qtm_u.cpp 157 hle/service/qtm/qtm_u.cpp
158 hle/service/service.cpp 158 hle/service/service.cpp
159 hle/service/sm/sm.cpp
160 hle/service/sm/srv.cpp
159 hle/service/soc_u.cpp 161 hle/service/soc_u.cpp
160 hle/service/srv.cpp
161 hle/service/ssl_c.cpp 162 hle/service/ssl_c.cpp
162 hle/service/y2r_u.cpp 163 hle/service/y2r_u.cpp
163 hle/shared_page.cpp 164 hle/shared_page.cpp
@@ -352,8 +353,9 @@ set(HEADERS
352 hle/service/qtm/qtm_sp.h 353 hle/service/qtm/qtm_sp.h
353 hle/service/qtm/qtm_u.h 354 hle/service/qtm/qtm_u.h
354 hle/service/service.h 355 hle/service/service.h
356 hle/service/sm/sm.h
357 hle/service/sm/srv.h
355 hle/service/soc_u.h 358 hle/service/soc_u.h
356 hle/service/srv.h
357 hle/service/ssl_c.h 359 hle/service/ssl_c.h
358 hle/service/y2r_u.h 360 hle/service/y2r_u.h
359 hle/shared_page.h 361 hle/shared_page.h
diff --git a/src/core/hle/service/service.cpp b/src/core/hle/service/service.cpp
index 8f7c97d54..1b64ee77d 100644
--- a/src/core/hle/service/service.cpp
+++ b/src/core/hle/service/service.cpp
@@ -38,15 +38,15 @@
38#include "core/hle/service/ptm/ptm.h" 38#include "core/hle/service/ptm/ptm.h"
39#include "core/hle/service/qtm/qtm.h" 39#include "core/hle/service/qtm/qtm.h"
40#include "core/hle/service/service.h" 40#include "core/hle/service/service.h"
41#include "core/hle/service/sm/sm.h"
42#include "core/hle/service/sm/srv.h"
41#include "core/hle/service/soc_u.h" 43#include "core/hle/service/soc_u.h"
42#include "core/hle/service/srv.h"
43#include "core/hle/service/ssl_c.h" 44#include "core/hle/service/ssl_c.h"
44#include "core/hle/service/y2r_u.h" 45#include "core/hle/service/y2r_u.h"
45 46
46namespace Service { 47namespace Service {
47 48
48std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports; 49std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
49std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
50 50
51/** 51/**
52 * Creates a function string for logging, complete with the name (or header code, depending 52 * Creates a function string for logging, complete with the name (or header code, depending
@@ -115,18 +115,17 @@ static void AddNamedPort(Interface* interface_) {
115} 115}
116 116
117void AddService(Interface* interface_) { 117void AddService(Interface* interface_) {
118 Kernel::SharedPtr<Kernel::ServerPort> server_port; 118 auto server_port =
119 Kernel::SharedPtr<Kernel::ClientPort> client_port; 119 SM::g_service_manager
120 std::tie(server_port, client_port) = 120 ->RegisterService(interface_->GetPortName(), interface_->GetMaxSessions())
121 Kernel::ServerPort::CreatePortPair(interface_->GetMaxSessions(), interface_->GetPortName()); 121 .MoveFrom();
122
123 server_port->SetHleHandler(std::shared_ptr<Interface>(interface_)); 122 server_port->SetHleHandler(std::shared_ptr<Interface>(interface_));
124 g_srv_services.emplace(interface_->GetPortName(), std::move(client_port));
125} 123}
126 124
127/// Initialize ServiceManager 125/// Initialize ServiceManager
128void Init() { 126void Init() {
129 AddNamedPort(new SRV::SRV); 127 SM::g_service_manager = std::make_unique<SM::ServiceManager>();
128 AddNamedPort(new SM::SRV);
130 AddNamedPort(new ERR::ERR_F); 129 AddNamedPort(new ERR::ERR_F);
131 130
132 FS::ArchiveInit(); 131 FS::ArchiveInit();
@@ -187,7 +186,7 @@ void Shutdown() {
187 AC::Shutdown(); 186 AC::Shutdown();
188 FS::ArchiveShutdown(); 187 FS::ArchiveShutdown();
189 188
190 g_srv_services.clear(); 189 SM::g_service_manager = nullptr;
191 g_kernel_named_ports.clear(); 190 g_kernel_named_ports.clear();
192 LOG_DEBUG(Service, "shutdown OK"); 191 LOG_DEBUG(Service, "shutdown OK");
193} 192}
diff --git a/src/core/hle/service/service.h b/src/core/hle/service/service.h
index a5fe843f6..7010b116b 100644
--- a/src/core/hle/service/service.h
+++ b/src/core/hle/service/service.h
@@ -107,8 +107,6 @@ void Shutdown();
107 107
108/// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC. 108/// Map of named ports managed by the kernel, which can be retrieved using the ConnectToPort SVC.
109extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports; 109extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_kernel_named_ports;
110/// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
111extern std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> g_srv_services;
112 110
113/// Adds a service to the services table 111/// Adds a service to the services table
114void AddService(Interface* interface_); 112void 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..40df0f0dd
--- /dev/null
+++ b/src/core/hle/service/sm/sm.cpp
@@ -0,0 +1,58 @@
1// Copyright 2017 Citra Emulator Project
2// Licensed under GPLv2 or any later version
3// Refer to the license.txt file included.
4
5#include <tuple>
6#include "core/hle/kernel/client_session.h"
7#include "core/hle/kernel/server_port.h"
8#include "core/hle/result.h"
9#include "core/hle/service/sm/sm.h"
10
11namespace Service {
12namespace SM {
13
14static ResultCode ValidateServiceName(const std::string& name) {
15 if (name.size() <= 0 || name.size() > 8) {
16 return ERR_INVALID_NAME_SIZE;
17 }
18 if (name.find('\0') != std::string::npos) {
19 return ERR_NAME_CONTAINS_NUL;
20 }
21 return RESULT_SUCCESS;
22}
23
24ResultVal<Kernel::SharedPtr<Kernel::ServerPort>> ServiceManager::RegisterService(
25 std::string name, unsigned int max_sessions) {
26
27 CASCADE_CODE(ValidateServiceName(name));
28 Kernel::SharedPtr<Kernel::ServerPort> server_port;
29 Kernel::SharedPtr<Kernel::ClientPort> client_port;
30 std::tie(server_port, client_port) = Kernel::ServerPort::CreatePortPair(max_sessions, name);
31
32 registered_services.emplace(name, std::move(client_port));
33 return MakeResult<Kernel::SharedPtr<Kernel::ServerPort>>(std::move(server_port));
34}
35
36ResultVal<Kernel::SharedPtr<Kernel::ClientPort>> ServiceManager::GetServicePort(
37 const std::string& name) {
38
39 CASCADE_CODE(ValidateServiceName(name));
40 auto it = registered_services.find(name);
41 if (it == registered_services.end()) {
42 return ERR_SERVICE_NOT_REGISTERED;
43 }
44
45 return MakeResult<Kernel::SharedPtr<Kernel::ClientPort>>(it->second);
46}
47
48ResultVal<Kernel::SharedPtr<Kernel::ClientSession>> ServiceManager::ConnectToService(
49 const std::string& name) {
50
51 CASCADE_RESULT(auto client_port, GetServicePort(name));
52 return client_port->Connect();
53}
54
55std::unique_ptr<ServiceManager> g_service_manager;
56
57} // namespace SM
58} // 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 @@
1// Copyright 2017 Citra Emulator Project
2// Licensed under GPLv2 or any later version
3// Refer to the license.txt file included.
4
5#pragma once
6
7#include <string>
8#include <unordered_map>
9#include "core/hle/kernel/kernel.h"
10#include "core/hle/result.h"
11#include "core/hle/service/service.h"
12
13namespace Kernel {
14class ClientPort;
15class ClientSession;
16class ServerPort;
17class SessionRequestHandler;
18} // namespace Kernel
19
20namespace Service {
21namespace SM {
22
23constexpr ResultCode ERR_SERVICE_NOT_REGISTERED(1, ErrorModule::SRV, ErrorSummary::WouldBlock,
24 ErrorLevel::Temporary); // 0xD0406401
25constexpr ResultCode ERR_MAX_CONNECTIONS_REACHED(2, ErrorModule::SRV, ErrorSummary::WouldBlock,
26 ErrorLevel::Temporary); // 0xD0406402
27constexpr ResultCode ERR_INVALID_NAME_SIZE(5, ErrorModule::SRV, ErrorSummary::WrongArgument,
28 ErrorLevel::Permanent); // 0xD9006405
29constexpr ResultCode ERR_ACCESS_DENIED(6, ErrorModule::SRV, ErrorSummary::InvalidArgument,
30 ErrorLevel::Permanent); // 0xD8E06406
31constexpr ResultCode ERR_NAME_CONTAINS_NUL(7, ErrorModule::SRV, ErrorSummary::WrongArgument,
32 ErrorLevel::Permanent); // 0xD9006407
33
34class ServiceManager {
35public:
36 ResultVal<Kernel::SharedPtr<Kernel::ServerPort>> RegisterService(std::string name,
37 unsigned int max_sessions);
38 ResultVal<Kernel::SharedPtr<Kernel::ClientPort>> GetServicePort(const std::string& name);
39 ResultVal<Kernel::SharedPtr<Kernel::ClientSession>> ConnectToService(const std::string& name);
40
41private:
42 /// Map of services registered with the "srv:" service, retrieved using GetServiceHandle.
43 std::unordered_map<std::string, Kernel::SharedPtr<Kernel::ClientPort>> registered_services;
44};
45
46extern std::unique_ptr<ServiceManager> g_service_manager;
47
48} // namespace SM
49} // namespace Service
diff --git a/src/core/hle/service/srv.cpp b/src/core/hle/service/sm/srv.cpp
index 130c9d25e..34166289c 100644
--- a/src/core/hle/service/srv.cpp
+++ b/src/core/hle/service/sm/srv.cpp
@@ -7,14 +7,17 @@
7#include "common/common_types.h" 7#include "common/common_types.h"
8#include "common/logging/log.h" 8#include "common/logging/log.h"
9#include "core/hle/kernel/client_session.h" 9#include "core/hle/kernel/client_session.h"
10#include "core/hle/kernel/event.h" 10#include "core/hle/kernel/semaphore.h"
11#include "core/hle/kernel/server_session.h" 11#include "core/hle/kernel/server_session.h"
12#include "core/hle/service/srv.h" 12#include "core/hle/service/sm/sm.h"
13#include "core/hle/service/sm/srv.h"
13 14
14namespace Service { 15namespace Service {
15namespace SRV { 16namespace SM {
16 17
17static Kernel::SharedPtr<Kernel::Event> event_handle; 18constexpr int MAX_PENDING_NOTIFICATIONS = 16;
19
20static Kernel::SharedPtr<Kernel::Semaphore> notification_semaphore;
18 21
19/** 22/**
20 * SRV::RegisterClient service function 23 * SRV::RegisterClient service function
@@ -51,14 +54,13 @@ static void RegisterClient(Interface* self) {
51static void EnableNotification(Interface* self) { 54static void EnableNotification(Interface* self) {
52 u32* cmd_buff = Kernel::GetCommandBuffer(); 55 u32* cmd_buff = Kernel::GetCommandBuffer();
53 56
54 // TODO(bunnei): Change to a semaphore once these have been implemented 57 notification_semaphore =
55 event_handle = Kernel::Event::Create(Kernel::ResetType::OneShot, "SRV:Event"); 58 Kernel::Semaphore::Create(0, MAX_PENDING_NOTIFICATIONS, "SRV:Notification").Unwrap();
56 event_handle->Clear();
57 59
58 cmd_buff[0] = IPC::MakeHeader(0x2, 0x1, 0x2); // 0x20042 60 cmd_buff[0] = IPC::MakeHeader(0x2, 0x1, 0x2); // 0x20042
59 cmd_buff[1] = RESULT_SUCCESS.raw; // No error 61 cmd_buff[1] = RESULT_SUCCESS.raw; // No error
60 cmd_buff[2] = IPC::CopyHandleDesc(1); 62 cmd_buff[2] = IPC::CopyHandleDesc(1);
61 cmd_buff[3] = Kernel::g_handle_table.Create(event_handle).MoveFrom(); 63 cmd_buff[3] = Kernel::g_handle_table.Create(notification_semaphore).MoveFrom();
62 LOG_WARNING(Service_SRV, "(STUBBED) called"); 64 LOG_WARNING(Service_SRV, "(STUBBED) called");
63} 65}
64 66
@@ -77,25 +79,41 @@ static void GetServiceHandle(Interface* self) {
77 ResultCode res = RESULT_SUCCESS; 79 ResultCode res = RESULT_SUCCESS;
78 u32* cmd_buff = Kernel::GetCommandBuffer(); 80 u32* cmd_buff = Kernel::GetCommandBuffer();
79 81
80 std::string port_name = std::string((const char*)&cmd_buff[1], 0, Service::kMaxPortSize); 82 size_t name_len = cmd_buff[3];
81 auto it = Service::g_srv_services.find(port_name); 83 if (name_len > Service::kMaxPortSize) {
84 cmd_buff[1] = ERR_INVALID_NAME_SIZE.raw;
85 LOG_ERROR(Service_SRV, "called name_len=0x%X, failed with code=0x%08X", name_len,
86 cmd_buff[1]);
87 return;
88 }
89 std::string name(reinterpret_cast<const char*>(&cmd_buff[1]), name_len);
90 bool return_port_on_failure = (cmd_buff[4] & 1) == 0;
82 91
83 if (it != Service::g_srv_services.end()) { 92 // TODO(yuriks): Permission checks go here
84 auto client_port = it->second;
85 93
86 auto client_session = client_port->Connect(); 94 auto client_port = g_service_manager->GetServicePort(name);
87 res = client_session.Code(); 95 if (client_port.Failed()) {
96 cmd_buff[1] = client_port.Code().raw;
97 LOG_ERROR(Service_SRV, "called service=%s, failed with code=0x%08X", name.c_str(),
98 cmd_buff[1]);
99 return;
100 }
88 101
89 if (client_session.Succeeded()) { 102 auto session = client_port.Unwrap()->Connect();
90 // Return the client session 103 cmd_buff[1] = session.Code().raw;
91 cmd_buff[3] = Kernel::g_handle_table.Create(*client_session).MoveFrom(); 104 if (session.Succeeded()) {
92 } 105 cmd_buff[3] = Kernel::g_handle_table.Create(session.MoveFrom()).MoveFrom();
93 LOG_TRACE(Service_SRV, "called port=%s, handle=0x%08X", port_name.c_str(), cmd_buff[3]); 106 LOG_DEBUG(Service_SRV, "called service=%s, session handle=0x%08X", name.c_str(),
107 cmd_buff[3]);
108 } else if (session.Code() == Kernel::ERR_MAX_CONNECTIONS_REACHED && return_port_on_failure) {
109 cmd_buff[1] = ERR_MAX_CONNECTIONS_REACHED.raw;
110 cmd_buff[3] = Kernel::g_handle_table.Create(client_port.MoveFrom()).MoveFrom();
111 LOG_WARNING(Service_SRV, "called service=%s, *port* handle=0x%08X", name.c_str(),
112 cmd_buff[3]);
94 } else { 113 } else {
95 LOG_ERROR(Service_SRV, "(UNIMPLEMENTED) called port=%s", port_name.c_str()); 114 LOG_ERROR(Service_SRV, "called service=%s, failed with code=0x%08X", name.c_str(),
96 res = UnimplementedFunction(ErrorModule::SRV); 115 cmd_buff[1]);
97 } 116 }
98 cmd_buff[1] = res.raw;
99} 117}
100 118
101/** 119/**
@@ -177,12 +195,12 @@ const Interface::FunctionInfo FunctionTable[] = {
177 195
178SRV::SRV() { 196SRV::SRV() {
179 Register(FunctionTable); 197 Register(FunctionTable);
180 event_handle = nullptr; 198 notification_semaphore = nullptr;
181} 199}
182 200
183SRV::~SRV() { 201SRV::~SRV() {
184 event_handle = nullptr; 202 notification_semaphore = nullptr;
185} 203}
186 204
187} // namespace SRV 205} // namespace SM
188} // namespace Service 206} // 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 @@
4 4
5#pragma once 5#pragma once
6 6
7#include <string>
7#include "core/hle/service/service.h" 8#include "core/hle/service/service.h"
8 9
9namespace Service { 10namespace Service {
10namespace SRV { 11namespace SM {
11 12
12/// Interface to "srv:" service 13/// Interface to "srv:" service
13class SRV final : public Interface { 14class SRV final : public Interface {
@@ -20,5 +21,5 @@ public:
20 } 21 }
21}; 22};
22 23
23} // namespace SRV 24} // namespace SM
24} // namespace Service 25} // namespace Service