| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | // Copyright 2016 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <tuple>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  | #include "core/hle/ipc_helpers.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/client_port.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | #include "core/hle/kernel/client_session.h"
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | #include "core/hle/kernel/handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/hle_ipc.h"
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | #include "core/hle/kernel/server_session.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/session.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 22:58:02 -05:00
										 |  |  | ServerSession::ServerSession() = default; | 
					
						
							| 
									
										
										
										
											2016-12-08 15:01:10 -05:00
										 |  |  | ServerSession::~ServerSession() { | 
					
						
							| 
									
										
										
										
											2016-12-14 12:33:49 -05:00
										 |  |  |     // This destructor will be called automatically when the last ServerSession handle is closed by
 | 
					
						
							|  |  |  |     // the emulated application.
 | 
					
						
							| 
									
										
										
										
											2017-01-04 23:23:17 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Decrease the port's connection count.
 | 
					
						
							|  |  |  |     if (parent->port) | 
					
						
							|  |  |  |         parent->port->active_sessions--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO(Subv): Wake up all the ClientSession's waiting threads and set
 | 
					
						
							|  |  |  |     // the SendSyncRequest result to 0xC920181A.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parent->server = nullptr; | 
					
						
							| 
									
										
										
										
											2016-12-08 15:01:10 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  | ResultVal<SharedPtr<ServerSession>> ServerSession::Create(std::string name) { | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  |     SharedPtr<ServerSession> server_session(new ServerSession); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     server_session->name = std::move(name); | 
					
						
							| 
									
										
										
										
											2017-01-04 23:23:17 -05:00
										 |  |  |     server_session->parent = nullptr; | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  |     return MakeResult(std::move(server_session)); | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | bool ServerSession::ShouldWait(Thread* thread) const { | 
					
						
							| 
									
										
										
										
											2017-06-20 17:33:28 -05:00
										 |  |  |     // Closed sessions should never wait, an error will be returned from svcReplyAndReceive.
 | 
					
						
							|  |  |  |     if (parent->client == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     // Wait if we have no pending requests, or if we're currently handling a request.
 | 
					
						
							|  |  |  |     return pending_requesting_threads.empty() || currently_handling != nullptr; | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | void ServerSession::Acquire(Thread* thread) { | 
					
						
							|  |  |  |     ASSERT_MSG(!ShouldWait(thread), "object unavailable!"); | 
					
						
							| 
									
										
										
										
											2017-06-20 17:33:28 -05:00
										 |  |  |     // We are now handling a request, pop it from the stack.
 | 
					
						
							|  |  |  |     // TODO(Subv): What happens if the client endpoint is closed before any requests are made?
 | 
					
						
							|  |  |  |     ASSERT(!pending_requesting_threads.empty()); | 
					
						
							|  |  |  |     currently_handling = pending_requesting_threads.back(); | 
					
						
							|  |  |  |     pending_requesting_threads.pop_back(); | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-20 17:33:28 -05:00
										 |  |  | ResultCode ServerSession::HandleSyncRequest(SharedPtr<Thread> thread) { | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  |     // The ServerSession received a sync request, this means that there's new data available
 | 
					
						
							| 
									
										
										
										
											2016-12-14 12:33:49 -05:00
										 |  |  |     // from its ClientSession, so wake up any threads that may be waiting on a svcReplyAndReceive or
 | 
					
						
							|  |  |  |     // similar.
 | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |     Kernel::HLERequestContext context(this); | 
					
						
							|  |  |  |     u32* cmd_buf = (u32*)Memory::GetPointer(thread->GetTLSAddress()); | 
					
						
							|  |  |  |     context.PopulateFromIncomingCommandBuffer(cmd_buf, *Kernel::g_current_process, | 
					
						
							|  |  |  |                                               Kernel::g_handle_table); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the session has been converted to a domain, handle the doomain request
 | 
					
						
							|  |  |  |     if (IsDomain()) { | 
					
						
							|  |  |  |         auto& domain_message_header = context.GetDomainMessageHeader(); | 
					
						
							|  |  |  |         if (domain_message_header) { | 
					
						
							|  |  |  |             // If there is a DomainMessageHeader, then this is CommandType "Request"
 | 
					
						
							|  |  |  |             const u32 object_id{context.GetDomainMessageHeader()->object_id}; | 
					
						
							|  |  |  |             switch (domain_message_header->command) { | 
					
						
							|  |  |  |             case IPC::DomainMessageHeader::CommandType::SendMessage: | 
					
						
							|  |  |  |                 return domain_request_handlers[object_id - 1]->HandleSyncRequest(context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             case IPC::DomainMessageHeader::CommandType::CloseVirtualHandle: { | 
					
						
							|  |  |  |                 LOG_DEBUG(IPC, "CloseVirtualHandle, object_id=0x%08X", object_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 domain_request_handlers[object_id - 1] = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |                 IPC::ResponseBuilder rb{context, 2}; | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |                 rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |                 return RESULT_SUCCESS; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             LOG_CRITICAL(IPC, "Unknown domain command=%d", domain_message_header->command.Value()); | 
					
						
							| 
									
										
										
										
											2018-01-23 22:00:17 -05:00
										 |  |  |             ASSERT(false); | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-23 22:00:17 -05:00
										 |  |  |         // If there is no domain header, the regular session handler is used
 | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  |     // If this ServerSession has an associated HLE handler, forward the request to it.
 | 
					
						
							| 
									
										
										
										
											2017-10-31 19:30:05 -04:00
										 |  |  |     ResultCode result{RESULT_SUCCESS}; | 
					
						
							| 
									
										
										
										
											2016-12-09 12:52:12 -05:00
										 |  |  |     if (hle_handler != nullptr) { | 
					
						
							|  |  |  |         // Attempt to translate the incoming request's command buffer.
 | 
					
						
							| 
									
										
										
										
											2017-10-31 19:30:05 -04:00
										 |  |  |         ResultCode translate_result = TranslateHLERequest(this); | 
					
						
							|  |  |  |         if (translate_result.IsError()) | 
					
						
							|  |  |  |             return translate_result; | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         result = hle_handler->HandleSyncRequest(context); | 
					
						
							| 
									
										
										
										
											2017-06-20 17:33:28 -05:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // Add the thread to the list of threads that have issued a sync request with this
 | 
					
						
							|  |  |  |         // server.
 | 
					
						
							|  |  |  |         pending_requesting_threads.push_back(std::move(thread)); | 
					
						
							| 
									
										
										
										
											2016-12-09 12:52:12 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-11-30 22:50:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 12:33:49 -05:00
										 |  |  |     // If this ServerSession does not have an HLE implementation, just wake up the threads waiting
 | 
					
						
							|  |  |  |     // on it.
 | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  |     WakeupAllWaitingThreads(); | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Handle scenario when ConvertToDomain command was issued, as we must do the conversion at the
 | 
					
						
							|  |  |  |     // end of the command such that only commands following this one are handled as domains
 | 
					
						
							|  |  |  |     if (convert_to_domain) { | 
					
						
							|  |  |  |         ASSERT_MSG(domain_request_handlers.empty(), "already a domain"); | 
					
						
							| 
									
										
										
										
											2018-01-23 22:00:17 -05:00
										 |  |  |         domain_request_handlers = {hle_handler}; | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |         convert_to_domain = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-31 19:30:05 -04:00
										 |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  | ServerSession::SessionPair ServerSession::CreateSessionPair(const std::string& name, | 
					
						
							|  |  |  |                                                             SharedPtr<ClientPort> port) { | 
					
						
							| 
									
										
										
										
											2017-06-18 19:03:15 -07:00
										 |  |  |     auto server_session = ServerSession::Create(name + "_Server").Unwrap(); | 
					
						
							| 
									
										
										
										
											2017-05-21 18:52:42 -05:00
										 |  |  |     SharedPtr<ClientSession> client_session(new ClientSession); | 
					
						
							|  |  |  |     client_session->name = name + "_Client"; | 
					
						
							| 
									
										
										
										
											2017-01-04 23:23:17 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::shared_ptr<Session> parent(new Session); | 
					
						
							|  |  |  |     parent->client = client_session.get(); | 
					
						
							|  |  |  |     parent->server = server_session.get(); | 
					
						
							|  |  |  |     parent->port = port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client_session->parent = parent; | 
					
						
							|  |  |  |     server_session->parent = parent; | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-05 11:02:08 -05:00
										 |  |  |     return std::make_tuple(std::move(server_session), std::move(client_session)); | 
					
						
							| 
									
										
										
										
											2016-06-14 18:03:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-09 12:52:12 -05:00
										 |  |  | ResultCode TranslateHLERequest(ServerSession* server_session) { | 
					
						
							|  |  |  |     // TODO(Subv): Implement this function once multiple concurrent processes are supported.
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-06-20 17:33:28 -05:00
										 |  |  | } // namespace Kernel
 |