| 
									
										
										
										
											2018-01-13 16:22:39 -05:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							| 
									
										
										
										
											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-10-14 22:18:42 -04:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-16 21:32:59 -05:00
										 |  |  | #include "core/hle/service/acc/acc.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:50:04 -04:00
										 |  |  | #include "core/hle/service/am/am.h"
 | 
					
						
							| 
									
										
										
										
											2017-12-28 15:24:05 -05:00
										 |  |  | #include "core/hle/service/aoc/aoc_u.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:50:04 -04:00
										 |  |  | #include "core/hle/service/apm/apm.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-14 21:29:11 -05:00
										 |  |  | #include "core/hle/service/audio/audio.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-16 19:20:12 -08:00
										 |  |  | #include "core/hle/service/filesystem/filesystem.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"
 | 
					
						
							| 
									
										
										
										
											2018-01-07 21:27:58 -05:00
										 |  |  | #include "core/hle/service/nvdrv/nvdrv.h"
 | 
					
						
							| 
									
										
										
										
											2017-12-28 23:44:39 -05:00
										 |  |  | #include "core/hle/service/pctl/pctl.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/service/service.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-19 21:44:58 +01:00
										 |  |  | #include "core/hle/service/set/set.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"
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:35:03 +01:00
										 |  |  | #include "core/hle/service/sockets/sockets.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  | #include "core/hle/service/time/time.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-07 21:27:58 -05:00
										 |  |  | #include "core/hle/service/vi/vi.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::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; | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  |     std::tie(server_port, client_port) = | 
					
						
							|  |  |  |         Kernel::ServerPort::CreatePortPair(max_sessions, service_name); | 
					
						
							| 
									
										
										
										
											2017-10-18 21:34:19 -04:00
										 |  |  |     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-12-28 15:24:59 -05:00
										 |  |  |     std::string function_name = info == nullptr ? fmt::format("{}", 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()); | 
					
						
							| 
									
										
										
										
											2017-12-28 15:24:59 -05:00
										 |  |  |     UNIMPLEMENTED(); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | ResultCode ServiceFrameworkBase::HandleSyncRequest(Kernel::HLERequestContext& context) { | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2017-10-31 19:30:05 -04:00
										 |  |  |         return ResultCode(ErrorModule::HIPC, ErrorDescription::RemoteProcessDead); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											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-12-29 00:36:22 -05:00
										 |  |  |     u32* cmd_buf = (u32*)Memory::GetPointer(Kernel::GetCurrentThread()->GetTLSAddress()); | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |     context.WriteToOutgoingCommandBuffer(cmd_buf, *Kernel::g_current_process, | 
					
						
							|  |  |  |                                          Kernel::g_handle_table); | 
					
						
							| 
									
										
										
										
											2017-10-31 19:30:05 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							| 
									
										
										
										
											2018-01-22 13:40:02 -05:00
										 |  |  |     // NVFlinger needs to be accessed by several services like Vi and AppletOE so we instantiate it
 | 
					
						
							|  |  |  |     // here and pass it into the respective InstallInterfaces functions.
 | 
					
						
							|  |  |  |     auto nv_flinger = std::make_shared<NVFlinger::NVFlinger>(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:25:28 -07:00
										 |  |  |     SM::g_service_manager = std::make_shared<SM::ServiceManager>(); | 
					
						
							|  |  |  |     SM::ServiceManager::InstallInterfaces(SM::g_service_manager); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 21:32:59 -05:00
										 |  |  |     Account::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-22 13:46:36 -05:00
										 |  |  |     AM::InstallInterfaces(*SM::g_service_manager, nv_flinger); | 
					
						
							| 
									
										
										
										
											2017-12-28 15:24:05 -05:00
										 |  |  |     AOC::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:50:04 -04:00
										 |  |  |     APM::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-14 21:29:11 -05:00
										 |  |  |     Audio::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-16 19:20:12 -08:00
										 |  |  |     FileSystem::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-14 21:24:50 -05:00
										 |  |  |     HID::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     LM::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-15 17:39:00 -05:00
										 |  |  |     Nvidia::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2017-12-28 23:44:39 -05:00
										 |  |  |     PCTL::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-18 20:35:03 +01:00
										 |  |  |     Sockets::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  |     Time::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-22 13:40:02 -05:00
										 |  |  |     VI::InstallInterfaces(*SM::g_service_manager, nv_flinger); | 
					
						
							| 
									
										
										
										
											2018-01-19 21:44:58 +01:00
										 |  |  |     Set::InstallInterfaces(*SM::g_service_manager); | 
					
						
							| 
									
										
										
										
											2017-10-10 17:32:14 -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() { | 
					
						
							| 
									
										
										
										
											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
 |