| 
									
										
										
										
											2018-01-13 16:22:39 -05:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 08:06:09 -04:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include <array>
 | 
					
						
							|  |  |  | #include <sstream>
 | 
					
						
							| 
									
										
										
										
											2018-07-18 19:02:47 -04:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include <boost/range/algorithm_ext/erase.hpp>
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | #include "core/hle/ipc_helpers.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:55:18 -07:00
										 |  |  | #include "core/hle/kernel/handle_table.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/hle_ipc.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 12:21:34 -04:00
										 |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-01 22:40:00 -04:00
										 |  |  | #include "core/hle/kernel/object.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  | #include "core/hle/kernel/readable_event.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | #include "core/hle/kernel/server_session.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-31 18:09:41 -05:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  | #include "core/hle/kernel/writable_event.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-31 18:09:41 -05:00
										 |  |  | SessionRequestHandler::SessionRequestHandler() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SessionRequestHandler::~SessionRequestHandler() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | void SessionRequestHandler::ClientConnected(SharedPtr<ServerSession> server_session) { | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  |     server_session->SetHleHandler(shared_from_this()); | 
					
						
							| 
									
										
										
										
											2018-07-18 19:02:47 -04:00
										 |  |  |     connected_sessions.push_back(std::move(server_session)); | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 19:02:47 -04:00
										 |  |  | void SessionRequestHandler::ClientDisconnected(const SharedPtr<ServerSession>& server_session) { | 
					
						
							| 
									
										
										
										
											2017-06-05 22:39:26 -07:00
										 |  |  |     server_session->SetHleHandler(nullptr); | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  |     boost::range::remove_erase(connected_sessions, server_session); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  | SharedPtr<WritableEvent> HLERequestContext::SleepClientThread( | 
					
						
							|  |  |  |     SharedPtr<Thread> thread, const std::string& reason, u64 timeout, WakeupCallback&& callback, | 
					
						
							| 
									
										
										
										
											2018-11-29 09:14:12 -05:00
										 |  |  |     SharedPtr<WritableEvent> writable_event) { | 
					
						
							| 
									
										
										
										
											2018-03-18 20:22:46 -04:00
										 |  |  |     // Put the client thread to sleep until the wait event is signaled or the timeout expires.
 | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     thread->SetWakeupCallback([context = *this, callback]( | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |                                   ThreadWakeupReason reason, SharedPtr<Thread> thread, | 
					
						
							|  |  |  |                                   SharedPtr<WaitObject> object, std::size_t index) mutable -> bool { | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |         ASSERT(thread->GetStatus() == ThreadStatus::WaitHLEEvent); | 
					
						
							| 
									
										
										
										
											2018-03-18 20:22:46 -04:00
										 |  |  |         callback(thread, context, reason); | 
					
						
							|  |  |  |         context.WriteToOutgoingCommandBuffer(*thread); | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-03-18 20:22:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  |     auto& kernel = Core::System::GetInstance().Kernel(); | 
					
						
							| 
									
										
										
										
											2018-11-29 09:14:12 -05:00
										 |  |  |     if (!writable_event) { | 
					
						
							| 
									
										
										
										
											2018-07-17 00:05:13 -04:00
										 |  |  |         // Create event if not provided
 | 
					
						
							| 
									
										
										
										
											2018-11-27 09:18:29 -05:00
										 |  |  |         const auto pair = WritableEvent::CreateEventPair(kernel, Kernel::ResetType::OneShot, | 
					
						
							|  |  |  |                                                          "HLE Pause Event: " + reason); | 
					
						
							|  |  |  |         writable_event = pair.writable; | 
					
						
							| 
									
										
										
										
											2018-07-17 00:05:13 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-29 09:14:12 -05:00
										 |  |  |     const auto readable_event{writable_event->GetReadableEvent()}; | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  |     writable_event->Clear(); | 
					
						
							| 
									
										
										
										
											2018-10-03 18:47:57 -04:00
										 |  |  |     thread->SetStatus(ThreadStatus::WaitHLEEvent); | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  |     thread->SetWaitObjects({readable_event}); | 
					
						
							|  |  |  |     readable_event->AddWaitingThread(thread); | 
					
						
							| 
									
										
										
										
											2018-03-18 20:22:46 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (timeout > 0) { | 
					
						
							|  |  |  |         thread->WakeAfterDelay(timeout); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-26 18:32:52 -05:00
										 |  |  |     return writable_event; | 
					
						
							| 
									
										
										
										
											2018-03-18 20:22:46 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | HLERequestContext::HLERequestContext(SharedPtr<Kernel::ServerSession> server_session) | 
					
						
							|  |  |  |     : server_session(std::move(server_session)) { | 
					
						
							| 
									
										
										
										
											2017-06-18 16:05:12 -07:00
										 |  |  |     cmd_buf[0] = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-06 21:20:52 -07:00
										 |  |  | HLERequestContext::~HLERequestContext() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  | void HLERequestContext::ParseCommandBuffer(const HandleTable& handle_table, u32_le* src_cmdbuf, | 
					
						
							|  |  |  |                                            bool incoming) { | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     IPC::RequestParser rp(src_cmdbuf); | 
					
						
							| 
									
										
										
										
											2018-03-18 20:18:42 -04:00
										 |  |  |     command_header = std::make_shared<IPC::CommandHeader>(rp.PopRaw<IPC::CommandHeader>()); | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     if (command_header->type == IPC::CommandType::Close) { | 
					
						
							|  |  |  |         // Close does not populate the rest of the IPC header
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     // If handle descriptor is present, add size of it
 | 
					
						
							|  |  |  |     if (command_header->enable_handle_descriptor) { | 
					
						
							|  |  |  |         handle_descriptor_header = | 
					
						
							| 
									
										
										
										
											2018-03-18 20:18:42 -04:00
										 |  |  |             std::make_shared<IPC::HandleDescriptorHeader>(rp.PopRaw<IPC::HandleDescriptorHeader>()); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         if (handle_descriptor_header->send_current_pid) { | 
					
						
							|  |  |  |             rp.Skip(2, false); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-07 09:22:20 -05:00
										 |  |  |         if (incoming) { | 
					
						
							|  |  |  |             // Populate the object lists with the data in the IPC request.
 | 
					
						
							|  |  |  |             for (u32 handle = 0; handle < handle_descriptor_header->num_handles_to_copy; ++handle) { | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |                 copy_objects.push_back(handle_table.GetGeneric(rp.Pop<Handle>())); | 
					
						
							| 
									
										
										
										
											2018-01-07 09:22:20 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             for (u32 handle = 0; handle < handle_descriptor_header->num_handles_to_move; ++handle) { | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |                 move_objects.push_back(handle_table.GetGeneric(rp.Pop<Handle>())); | 
					
						
							| 
									
										
										
										
											2018-01-07 09:22:20 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             // For responses we just ignore the handles, they're empty and will be populated when
 | 
					
						
							|  |  |  |             // translating the response.
 | 
					
						
							|  |  |  |             rp.Skip(handle_descriptor_header->num_handles_to_copy, false); | 
					
						
							|  |  |  |             rp.Skip(handle_descriptor_header->num_handles_to_move, false); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     for (unsigned i = 0; i < command_header->num_buf_x_descriptors; ++i) { | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |         buffer_x_desciptors.push_back(rp.PopRaw<IPC::BufferDescriptorX>()); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     for (unsigned i = 0; i < command_header->num_buf_a_descriptors; ++i) { | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |         buffer_a_desciptors.push_back(rp.PopRaw<IPC::BufferDescriptorABW>()); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     for (unsigned i = 0; i < command_header->num_buf_b_descriptors; ++i) { | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |         buffer_b_desciptors.push_back(rp.PopRaw<IPC::BufferDescriptorABW>()); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |     for (unsigned i = 0; i < command_header->num_buf_w_descriptors; ++i) { | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |         buffer_w_desciptors.push_back(rp.PopRaw<IPC::BufferDescriptorABW>()); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     buffer_c_offset = rp.GetCurrentOffset() + command_header->data_size; | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-18 21:38:01 -04:00
										 |  |  |     // Padding to align to 16 bytes
 | 
					
						
							|  |  |  |     rp.AlignWithPadding(); | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-17 14:03:52 -07:00
										 |  |  |     if (Session()->IsDomain() && ((command_header->type == IPC::CommandType::Request || | 
					
						
							|  |  |  |                                    command_header->type == IPC::CommandType::RequestWithContext) || | 
					
						
							|  |  |  |                                   !incoming)) { | 
					
						
							| 
									
										
										
										
											2017-12-29 00:36:22 -05:00
										 |  |  |         // If this is an incoming message, only CommandType "Request" has a domain header
 | 
					
						
							| 
									
										
										
										
											2018-02-20 19:27:49 +03:00
										 |  |  |         // All outgoing domain messages have the domain header, if only incoming has it
 | 
					
						
							|  |  |  |         if (incoming || domain_message_header) { | 
					
						
							|  |  |  |             domain_message_header = | 
					
						
							| 
									
										
										
										
											2018-03-18 20:18:42 -04:00
										 |  |  |                 std::make_shared<IPC::DomainMessageHeader>(rp.PopRaw<IPC::DomainMessageHeader>()); | 
					
						
							| 
									
										
										
										
											2018-02-20 22:51:54 +03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             if (Session()->IsDomain()) | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |                 LOG_WARNING(IPC, "Domain request has no DomainMessageHeader!"); | 
					
						
							| 
									
										
										
										
											2018-02-20 19:27:49 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     data_payload_header = | 
					
						
							| 
									
										
										
										
											2018-03-18 20:18:42 -04:00
										 |  |  |         std::make_shared<IPC::DataPayloadHeader>(rp.PopRaw<IPC::DataPayloadHeader>()); | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 01:16:55 -05:00
										 |  |  |     data_payload_offset = rp.GetCurrentOffset(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 00:48:02 -07:00
										 |  |  |     if (domain_message_header && domain_message_header->command == | 
					
						
							|  |  |  |                                      IPC::DomainMessageHeader::CommandType::CloseVirtualHandle) { | 
					
						
							| 
									
										
										
										
											2018-01-17 01:16:55 -05:00
										 |  |  |         // CloseVirtualHandle command does not have SFC* or any data
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     if (incoming) { | 
					
						
							|  |  |  |         ASSERT(data_payload_header->magic == Common::MakeMagic('S', 'F', 'C', 'I')); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         ASSERT(data_payload_header->magic == Common::MakeMagic('S', 'F', 'C', 'O')); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  |     rp.SetCurrentOffset(buffer_c_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // For Inline buffers, the response data is written directly to buffer_c_offset
 | 
					
						
							|  |  |  |     // and in this case we don't have any BufferDescriptorC on the request.
 | 
					
						
							|  |  |  |     if (command_header->buf_c_descriptor_flags > | 
					
						
							|  |  |  |         IPC::CommandHeader::BufferDescriptorCFlag::InlineDescriptor) { | 
					
						
							|  |  |  |         if (command_header->buf_c_descriptor_flags == | 
					
						
							|  |  |  |             IPC::CommandHeader::BufferDescriptorCFlag::OneDescriptor) { | 
					
						
							|  |  |  |             buffer_c_desciptors.push_back(rp.PopRaw<IPC::BufferDescriptorC>()); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             unsigned num_buf_c_descriptors = | 
					
						
							|  |  |  |                 static_cast<unsigned>(command_header->buf_c_descriptor_flags.Value()) - 2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // This is used to detect possible underflows, in case something is broken
 | 
					
						
							|  |  |  |             // with the two ifs above and the flags value is == 0 || == 1.
 | 
					
						
							|  |  |  |             ASSERT(num_buf_c_descriptors < 14); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (unsigned i = 0; i < num_buf_c_descriptors; ++i) { | 
					
						
							|  |  |  |                 buffer_c_desciptors.push_back(rp.PopRaw<IPC::BufferDescriptorC>()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rp.SetCurrentOffset(data_payload_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     command = rp.Pop<u32_le>(); | 
					
						
							| 
									
										
										
										
											2018-01-06 23:19:42 -05:00
										 |  |  |     rp.Skip(1, false); // The command is actually an u64, but we don't use the high part.
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  | ResultCode HLERequestContext::PopulateFromIncomingCommandBuffer(const HandleTable& handle_table, | 
					
						
							|  |  |  |                                                                 u32_le* src_cmdbuf) { | 
					
						
							|  |  |  |     ParseCommandBuffer(handle_table, src_cmdbuf, true); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:59:31 -05:00
										 |  |  |     if (command_header->type == IPC::CommandType::Close) { | 
					
						
							|  |  |  |         // Close does not populate the rest of the IPC header
 | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     // The data_size already includes the payload header, the padding and the domain header.
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     std::size_t size = data_payload_offset + command_header->data_size - | 
					
						
							|  |  |  |                        sizeof(IPC::DataPayloadHeader) / sizeof(u32) - 4; | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     if (domain_message_header) | 
					
						
							|  |  |  |         size -= sizeof(IPC::DomainMessageHeader) / sizeof(u32); | 
					
						
							|  |  |  |     std::copy_n(src_cmdbuf, size, cmd_buf.begin()); | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  | ResultCode HLERequestContext::WriteToOutgoingCommandBuffer(Thread& thread) { | 
					
						
							|  |  |  |     auto& owner_process = *thread.GetOwnerProcess(); | 
					
						
							|  |  |  |     auto& handle_table = owner_process.GetHandleTable(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-18 20:17:06 -04:00
										 |  |  |     std::array<u32, IPC::COMMAND_BUFFER_LENGTH> dst_cmdbuf; | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  |     Memory::ReadBlock(owner_process, thread.GetTLSAddress(), dst_cmdbuf.data(), | 
					
						
							| 
									
										
										
										
											2018-03-18 20:17:06 -04:00
										 |  |  |                       dst_cmdbuf.size() * sizeof(u32)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     // The header was already built in the internal command buffer. Attempt to parse it to verify
 | 
					
						
							|  |  |  |     // the integrity and then copy it over to the target command buffer.
 | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  |     ParseCommandBuffer(handle_table, cmd_buf.data(), false); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // The data_size already includes the payload header, the padding and the domain header.
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     std::size_t size = data_payload_offset + command_header->data_size - | 
					
						
							|  |  |  |                        sizeof(IPC::DataPayloadHeader) / sizeof(u32) - 4; | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     if (domain_message_header) | 
					
						
							|  |  |  |         size -= sizeof(IPC::DomainMessageHeader) / sizeof(u32); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-18 20:17:06 -04:00
										 |  |  |     std::copy_n(cmd_buf.begin(), size, dst_cmdbuf.data()); | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     if (command_header->enable_handle_descriptor) { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         ASSERT_MSG(!move_objects.empty() || !copy_objects.empty(), | 
					
						
							|  |  |  |                    "Handle descriptor bit set but no handles to translate"); | 
					
						
							|  |  |  |         // We write the translated handles at a specific offset in the command buffer, this space
 | 
					
						
							|  |  |  |         // was already reserved when writing the header.
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |         std::size_t current_offset = | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |             (sizeof(IPC::CommandHeader) + sizeof(IPC::HandleDescriptorHeader)) / sizeof(u32); | 
					
						
							|  |  |  |         ASSERT_MSG(!handle_descriptor_header->send_current_pid, "Sending PID is not implemented"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 23:13:16 +10:00
										 |  |  |         ASSERT(copy_objects.size() == handle_descriptor_header->num_handles_to_copy); | 
					
						
							|  |  |  |         ASSERT(move_objects.size() == handle_descriptor_header->num_handles_to_move); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // We don't make a distinction between copy and move handles when translating since HLE
 | 
					
						
							|  |  |  |         // services don't deal with handles directly. However, the guest applications might check
 | 
					
						
							|  |  |  |         // for specific values in each of these descriptors.
 | 
					
						
							|  |  |  |         for (auto& object : copy_objects) { | 
					
						
							|  |  |  |             ASSERT(object != nullptr); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |             dst_cmdbuf[current_offset++] = handle_table.Create(object).Unwrap(); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (auto& object : move_objects) { | 
					
						
							|  |  |  |             ASSERT(object != nullptr); | 
					
						
							| 
									
										
										
										
											2018-08-28 12:30:33 -04:00
										 |  |  |             dst_cmdbuf[current_offset++] = handle_table.Create(object).Unwrap(); | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |     // TODO(Subv): Translate the X/A/B/W buffers.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-20 21:59:58 +03:00
										 |  |  |     if (Session()->IsDomain() && domain_message_header) { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         ASSERT(domain_message_header->num_objects == domain_objects.size()); | 
					
						
							|  |  |  |         // Write the domain objects to the command buffer, these go after the raw untranslated data.
 | 
					
						
							|  |  |  |         // TODO(Subv): This completely ignores C buffers.
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |         std::size_t domain_offset = size - domain_message_header->num_objects; | 
					
						
							| 
									
										
										
										
											2018-01-23 18:03:09 -05:00
										 |  |  |         auto& request_handlers = server_session->domain_request_handlers; | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (auto& object : domain_objects) { | 
					
						
							|  |  |  |             request_handlers.emplace_back(object); | 
					
						
							| 
									
										
										
										
											2018-01-19 18:01:41 -05:00
										 |  |  |             dst_cmdbuf[domain_offset++] = static_cast<u32_le>(request_handlers.size()); | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-03-18 20:17:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Copy the translated command buffer back into the thread's command buffer area.
 | 
					
						
							| 
									
										
										
										
											2018-10-20 14:34:41 -04:00
										 |  |  |     Memory::WriteBlock(owner_process, thread.GetTLSAddress(), dst_cmdbuf.data(), | 
					
						
							| 
									
										
										
										
											2018-03-18 20:17:06 -04:00
										 |  |  |                        dst_cmdbuf.size() * sizeof(u32)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-09 05:23:13 -07:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							| 
									
										
										
										
											2017-06-08 23:55:18 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  | std::vector<u8> HLERequestContext::ReadBuffer(int buffer_index) const { | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     std::vector<u8> buffer; | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |     const bool is_buffer_a{BufferDescriptorA().size() && BufferDescriptorA()[buffer_index].Size()}; | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (is_buffer_a) { | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |         buffer.resize(BufferDescriptorA()[buffer_index].Size()); | 
					
						
							|  |  |  |         Memory::ReadBlock(BufferDescriptorA()[buffer_index].Address(), buffer.data(), | 
					
						
							|  |  |  |                           buffer.size()); | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |         buffer.resize(BufferDescriptorX()[buffer_index].Size()); | 
					
						
							|  |  |  |         Memory::ReadBlock(BufferDescriptorX()[buffer_index].Address(), buffer.data(), | 
					
						
							|  |  |  |                           buffer.size()); | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return buffer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | std::size_t HLERequestContext::WriteBuffer(const void* buffer, std::size_t size, | 
					
						
							|  |  |  |                                            int buffer_index) const { | 
					
						
							| 
									
										
										
										
											2018-06-22 11:18:23 +03:00
										 |  |  |     if (size == 0) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_WARNING(Core, "skip empty buffer write"); | 
					
						
							| 
									
										
										
										
											2018-06-22 11:18:23 +03:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |     const bool is_buffer_b{BufferDescriptorB().size() && BufferDescriptorB()[buffer_index].Size()}; | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     const std::size_t buffer_size{GetWriteBufferSize(buffer_index)}; | 
					
						
							| 
									
										
										
										
											2018-03-31 16:05:19 -04:00
										 |  |  |     if (size > buffer_size) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Core, "size ({:016X}) is greater than buffer_size ({:016X})", size, | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |                      buffer_size); | 
					
						
							| 
									
										
										
										
											2018-03-31 16:05:19 -04:00
										 |  |  |         size = buffer_size; // TODO(bunnei): This needs to be HW tested
 | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-14 00:14:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     if (is_buffer_b) { | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |         Memory::WriteBlock(BufferDescriptorB()[buffer_index].Address(), buffer, size); | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |         Memory::WriteBlock(BufferDescriptorC()[buffer_index].Address(), buffer, size); | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-13 23:53:18 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return size; | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | std::size_t HLERequestContext::GetReadBufferSize(int buffer_index) const { | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |     const bool is_buffer_a{BufferDescriptorA().size() && BufferDescriptorA()[buffer_index].Size()}; | 
					
						
							|  |  |  |     return is_buffer_a ? BufferDescriptorA()[buffer_index].Size() | 
					
						
							|  |  |  |                        : BufferDescriptorX()[buffer_index].Size(); | 
					
						
							| 
									
										
										
										
											2018-02-14 00:14:17 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | std::size_t HLERequestContext::GetWriteBufferSize(int buffer_index) const { | 
					
						
							| 
									
										
										
										
											2018-05-01 13:28:36 -07:00
										 |  |  |     const bool is_buffer_b{BufferDescriptorB().size() && BufferDescriptorB()[buffer_index].Size()}; | 
					
						
							|  |  |  |     return is_buffer_b ? BufferDescriptorB()[buffer_index].Size() | 
					
						
							|  |  |  |                        : BufferDescriptorC()[buffer_index].Size(); | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 17:22:11 +03:00
										 |  |  | std::string HLERequestContext::Description() const { | 
					
						
							|  |  |  |     if (!command_header) { | 
					
						
							|  |  |  |         return "No command header available"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     std::ostringstream s; | 
					
						
							|  |  |  |     s << "IPC::CommandHeader: Type:" << static_cast<u32>(command_header->type.Value()); | 
					
						
							|  |  |  |     s << ", X(Pointer):" << command_header->num_buf_x_descriptors; | 
					
						
							|  |  |  |     if (command_header->num_buf_x_descriptors) { | 
					
						
							|  |  |  |         s << '['; | 
					
						
							|  |  |  |         for (u64 i = 0; i < command_header->num_buf_x_descriptors; ++i) { | 
					
						
							|  |  |  |             s << "0x" << std::hex << BufferDescriptorX()[i].Size(); | 
					
						
							|  |  |  |             if (i < command_header->num_buf_x_descriptors - 1) | 
					
						
							|  |  |  |                 s << ", "; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         s << ']'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     s << ", A(Send):" << command_header->num_buf_a_descriptors; | 
					
						
							|  |  |  |     if (command_header->num_buf_a_descriptors) { | 
					
						
							|  |  |  |         s << '['; | 
					
						
							|  |  |  |         for (u64 i = 0; i < command_header->num_buf_a_descriptors; ++i) { | 
					
						
							|  |  |  |             s << "0x" << std::hex << BufferDescriptorA()[i].Size(); | 
					
						
							|  |  |  |             if (i < command_header->num_buf_a_descriptors - 1) | 
					
						
							|  |  |  |                 s << ", "; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         s << ']'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     s << ", B(Receive):" << command_header->num_buf_b_descriptors; | 
					
						
							|  |  |  |     if (command_header->num_buf_b_descriptors) { | 
					
						
							|  |  |  |         s << '['; | 
					
						
							|  |  |  |         for (u64 i = 0; i < command_header->num_buf_b_descriptors; ++i) { | 
					
						
							|  |  |  |             s << "0x" << std::hex << BufferDescriptorB()[i].Size(); | 
					
						
							|  |  |  |             if (i < command_header->num_buf_b_descriptors - 1) | 
					
						
							|  |  |  |                 s << ", "; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         s << ']'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     s << ", C(ReceiveList):" << BufferDescriptorC().size(); | 
					
						
							|  |  |  |     if (!BufferDescriptorC().empty()) { | 
					
						
							|  |  |  |         s << '['; | 
					
						
							|  |  |  |         for (u64 i = 0; i < BufferDescriptorC().size(); ++i) { | 
					
						
							|  |  |  |             s << "0x" << std::hex << BufferDescriptorC()[i].Size(); | 
					
						
							|  |  |  |             if (i < BufferDescriptorC().size() - 1) | 
					
						
							|  |  |  |                 s << ", "; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         s << ']'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     s << ", data_size:" << command_header->data_size.Value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return s.str(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 21:52:19 -07:00
										 |  |  | } // namespace Kernel
 |