| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | #include <fmt/format.h>
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-15 22:40:19 -04:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | #include "core/hle/ipc.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | #include "core/hle/ipc_helpers.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-06 01:29:46 -07:00
										 |  |  | #include "core/hle/kernel/client_port.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  | #include "core/hle/kernel/handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | #include "core/hle/kernel/server_port.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/server_session.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:50:04 -04:00
										 |  |  | #include "core/hle/service/am/am.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/apm/apm.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include "core/hle/service/dsp_dsp.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/gsp_gpu.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "core/hle/service/hid/hid.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | #include "core/hle/service/lm/lm.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/service/service.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | #include "core/hle/service/sm/controller.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-05 23:31:59 -07:00
										 |  |  | #include "core/hle/service/sm/sm.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | using Kernel::ClientPort; | 
					
						
							|  |  |  | using Kernel::ServerPort; | 
					
						
							|  |  |  | using Kernel::ServerSession; | 
					
						
							|  |  |  | using Kernel::SharedPtr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | namespace Service { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:22:28 -07:00
										 |  |  | std::unordered_map<std::string, SharedPtr<ClientPort>> g_kernel_named_ports; | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  |  * Creates a function string for logging, complete with the name (or header code, depending | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  |  * on what's passed in) the port name, and all the cmd_buff arguments. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | static std::string MakeFunctionString(const char* name, const char* port_name, | 
					
						
							|  |  |  |                                       const u32* cmd_buff) { | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  |     // Number of params == bits 0-5 + bits 6-11
 | 
					
						
							|  |  |  |     int num_params = (cmd_buff[0] & 0x3F) + ((cmd_buff[0] >> 6) & 0x3F); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     std::string function_string = | 
					
						
							|  |  |  |         Common::StringFromFormat("function '%s': port=%s", name, port_name); | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  |     for (int i = 1; i <= num_params; ++i) { | 
					
						
							| 
									
										
										
										
											2015-07-19 13:34:38 -07:00
										 |  |  |         function_string += Common::StringFromFormat(", cmd_buff[%i]=0x%X", i, cmd_buff[i]); | 
					
						
							| 
									
										
										
										
											2015-04-14 15:22:09 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     return function_string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ServiceFrameworkBase::ServiceFrameworkBase(const char* service_name, u32 max_sessions, | 
					
						
							|  |  |  |                                            InvokerFn* handler_invoker) | 
					
						
							|  |  |  |     : service_name(service_name), max_sessions(max_sessions), handler_invoker(handler_invoker) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ServiceFrameworkBase::~ServiceFrameworkBase() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ServiceFrameworkBase::InstallAsService(SM::ServiceManager& service_manager) { | 
					
						
							|  |  |  |     ASSERT(port == nullptr); | 
					
						
							|  |  |  |     port = service_manager.RegisterService(service_name, max_sessions).Unwrap(); | 
					
						
							|  |  |  |     port->SetHleHandler(shared_from_this()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ServiceFrameworkBase::InstallAsNamedPort() { | 
					
						
							|  |  |  |     ASSERT(port == nullptr); | 
					
						
							|  |  |  |     SharedPtr<ServerPort> server_port; | 
					
						
							|  |  |  |     SharedPtr<ClientPort> client_port; | 
					
						
							|  |  |  |     std::tie(server_port, client_port) = ServerPort::CreatePortPair(max_sessions, service_name); | 
					
						
							|  |  |  |     server_port->SetHleHandler(shared_from_this()); | 
					
						
							|  |  |  |     AddNamedPort(service_name, std::move(client_port)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-18 21:34:19 -04:00
										 |  |  | Kernel::SharedPtr<Kernel::ClientPort> ServiceFrameworkBase::CreatePort() { | 
					
						
							|  |  |  |     ASSERT(port == nullptr); | 
					
						
							|  |  |  |     Kernel::SharedPtr<Kernel::ServerPort> server_port; | 
					
						
							|  |  |  |     Kernel::SharedPtr<Kernel::ClientPort> client_port; | 
					
						
							|  |  |  |     std::tie(server_port, client_port) = Kernel::ServerPort::CreatePortPair(max_sessions, service_name); | 
					
						
							|  |  |  |     port = MakeResult<Kernel::SharedPtr<Kernel::ServerPort>>(std::move(server_port)).Unwrap(); | 
					
						
							|  |  |  |     port->SetHleHandler(shared_from_this()); | 
					
						
							|  |  |  |     return client_port; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | void ServiceFrameworkBase::RegisterHandlersBase(const FunctionInfoBase* functions, size_t n) { | 
					
						
							|  |  |  |     handlers.reserve(handlers.size() + n); | 
					
						
							|  |  |  |     for (size_t i = 0; i < n; ++i) { | 
					
						
							|  |  |  |         // Usually this array is sorted by id already, so hint to insert at the end
 | 
					
						
							|  |  |  |         handlers.emplace_hint(handlers.cend(), functions[i].expected_header, functions[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  | void ServiceFrameworkBase::ReportUnimplementedFunction(Kernel::HLERequestContext& ctx, | 
					
						
							|  |  |  |                                                        const FunctionInfoBase* info) { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     auto cmd_buf = ctx.CommandBuffer(); | 
					
						
							| 
									
										
										
										
											2017-10-17 18:02:08 -04:00
										 |  |  |     std::string function_name = info == nullptr ? fmt::format("{:#08x}", ctx.GetCommand()) : info->name; | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fmt::MemoryWriter w; | 
					
						
							|  |  |  |     w.write("function '{}': port='{}' cmd_buf={{[0]={:#x}", function_name, service_name, | 
					
						
							|  |  |  |             cmd_buf[0]); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     for (int i = 1; i <= 8; ++i) { | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |         w.write(", [{}]={:#x}", i, cmd_buf[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     w << '}'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_ERROR(Service, "unknown / unimplemented %s", w.c_str()); | 
					
						
							|  |  |  |     // TODO(bunnei): Hack - ignore error
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     IPC::RequestBuilder rb{ctx, 1}; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | void ServiceFrameworkBase::InvokeRequest(Kernel::HLERequestContext& ctx) { | 
					
						
							|  |  |  |     auto itr = handlers.find(ctx.GetCommand()); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     const FunctionInfoBase* info = itr == handlers.end() ? nullptr : &itr->second; | 
					
						
							|  |  |  |     if (info == nullptr || info->handler_callback == nullptr) { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         return ReportUnimplementedFunction(ctx, info); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     LOG_TRACE( | 
					
						
							|  |  |  |         Service, "%s", | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         MakeFunctionString(info->name, GetServiceName().c_str(), ctx.CommandBuffer()).c_str()); | 
					
						
							|  |  |  |     handler_invoker(this, info->handler_callback, ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ServiceFrameworkBase::HandleSyncRequest(SharedPtr<ServerSession> server_session) { | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     u32* cmd_buf = (u32*)Memory::GetPointer(Kernel::GetCurrentThread()->GetTLSAddress()); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  |     // TODO(yuriks): The kernel should be the one handling this as part of translation after
 | 
					
						
							|  |  |  |     // everything else is migrated
 | 
					
						
							| 
									
										
										
										
											2017-06-18 16:05:12 -07:00
										 |  |  |     Kernel::HLERequestContext context(std::move(server_session)); | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |     context.PopulateFromIncomingCommandBuffer(cmd_buf, *Kernel::g_current_process, | 
					
						
							|  |  |  |                                               Kernel::g_handle_table); | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     switch (context.GetCommandType()) { | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     case IPC::CommandType::Close: { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         IPC::RequestBuilder rb{context, 1}; | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     case IPC::CommandType::Control: { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         SM::g_service_manager->InvokeControlRequest(context); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     case IPC::CommandType::Request: { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         InvokeRequest(context); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         UNIMPLEMENTED_MSG("command_type=%d", context.GetCommandType()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |     context.WriteToOutgoingCommandBuffer(cmd_buf, *Kernel::g_current_process, | 
					
						
							|  |  |  |                                          Kernel::g_handle_table); | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-13 00:38:48 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							| 
									
										
										
										
											2015-01-30 16:07:04 -02:00
										 |  |  | // Module interface
 | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | // TODO(yuriks): Move to kernel
 | 
					
						
							|  |  |  | void AddNamedPort(std::string name, SharedPtr<ClientPort> port) { | 
					
						
							|  |  |  |     g_kernel_named_ports.emplace(std::move(name), std::move(port)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | /// Initialize ServiceManager
 | 
					
						
							|  |  |  | void Init() { | 
					
						
							| 
									
										
										
										
											2017-06-06 21:25:28 -07:00
										 |  |  |     SM::g_service_manager = std::make_shared<SM::ServiceManager>(); | 
					
						
							|  |  |  |     SM::ServiceManager::InstallInterfaces(SM::g_service_manager); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:50:04 -04:00
										 |  |  |     AM::InstallInterfaces(*SM::g_service_manager); | 
					
						
							|  |  |  |     APM::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     LM::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2017-10-10 17:32:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     HID::Init(); | 
					
						
							| 
									
										
										
										
											2014-04-16 00:03:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |     LOG_DEBUG(Service, "initialized OK"); | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Shutdown ServiceManager
 | 
					
						
							|  |  |  | void Shutdown() { | 
					
						
							| 
									
										
										
										
											2016-12-08 02:43:27 -05:00
										 |  |  |     HID::Shutdown(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-05 23:31:59 -07:00
										 |  |  |     SM::g_service_manager = nullptr; | 
					
						
							| 
									
										
										
										
											2015-01-30 16:07:04 -02:00
										 |  |  |     g_kernel_named_ports.clear(); | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |     LOG_DEBUG(Service, "shutdown OK"); | 
					
						
							| 
									
										
										
										
											2014-04-12 21:55:36 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  | } // namespace Service
 |