| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  | // Copyright 2016 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | #include "common/swap.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-21 00:11:36 -07:00
										 |  |  | #include "core/hle/kernel/errors.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace IPC { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-08 23:52:30 -07:00
										 |  |  | /// Size of the command buffer area, in 32-bit words.
 | 
					
						
							|  |  |  | constexpr size_t COMMAND_BUFFER_LENGTH = 0x100 / sizeof(u32); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-21 00:11:36 -07:00
										 |  |  | // These errors are commonly returned by invalid IPC translations, so alias them here for
 | 
					
						
							|  |  |  | // convenience.
 | 
					
						
							|  |  |  | // TODO(yuriks): These will probably go away once translation is implemented inside the kernel.
 | 
					
						
							|  |  |  | constexpr auto ERR_INVALID_HANDLE = Kernel::ERR_INVALID_HANDLE_OS; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | enum class ControlCommand : u32 { | 
					
						
							|  |  |  |     ConvertSessionToDomain = 0, | 
					
						
							|  |  |  |     ConvertDomainToSession = 1, | 
					
						
							|  |  |  |     DuplicateSession = 2, | 
					
						
							|  |  |  |     QueryPointerBufferSize = 3, | 
					
						
							|  |  |  |     DuplicateSessionEx = 4, | 
					
						
							|  |  |  |     Unspecified, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class CommandType : u32 { | 
					
						
							| 
									
										
										
										
											2018-05-30 20:09:21 +02:00
										 |  |  |     Invalid = 0, | 
					
						
							|  |  |  |     LegacyRequest = 1, | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     Close = 2, | 
					
						
							| 
									
										
										
										
											2018-05-30 20:09:21 +02:00
										 |  |  |     LegacyControl = 3, | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     Request = 4, | 
					
						
							|  |  |  |     Control = 5, | 
					
						
							| 
									
										
										
										
											2018-05-17 14:03:52 -07:00
										 |  |  |     RequestWithContext = 6, | 
					
						
							|  |  |  |     ControlWithContext = 7, | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     Unspecified, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct CommandHeader { | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |         u32_le raw_low; | 
					
						
							|  |  |  |         BitField<0, 16, CommandType> type; | 
					
						
							|  |  |  |         BitField<16, 4, u32_le> num_buf_x_descriptors; | 
					
						
							|  |  |  |         BitField<20, 4, u32_le> num_buf_a_descriptors; | 
					
						
							|  |  |  |         BitField<24, 4, u32_le> num_buf_b_descriptors; | 
					
						
							|  |  |  |         BitField<28, 4, u32_le> num_buf_w_descriptors; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum class BufferDescriptorCFlag : u32 { | 
					
						
							|  |  |  |         Disabled = 0, | 
					
						
							| 
									
										
										
										
											2018-01-07 21:25:01 -05:00
										 |  |  |         InlineDescriptor = 1, | 
					
						
							|  |  |  |         OneDescriptor = 2, | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |         u32_le raw_high; | 
					
						
							|  |  |  |         BitField<0, 10, u32_le> data_size; | 
					
						
							|  |  |  |         BitField<10, 4, BufferDescriptorCFlag> buf_c_descriptor_flags; | 
					
						
							|  |  |  |         BitField<31, 1, u32_le> enable_handle_descriptor; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(CommandHeader) == 8, "CommandHeader size is incorrect"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | union HandleDescriptorHeader { | 
					
						
							|  |  |  |     u32_le raw_high; | 
					
						
							|  |  |  |     BitField<0, 1, u32_le> send_current_pid; | 
					
						
							|  |  |  |     BitField<1, 4, u32_le> num_handles_to_copy; | 
					
						
							|  |  |  |     BitField<5, 4, u32_le> num_handles_to_move; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(HandleDescriptorHeader) == 4, "HandleDescriptorHeader size is incorrect"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BufferDescriptorX { | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |         BitField<0, 6, u32_le> counter_bits_0_5; | 
					
						
							|  |  |  |         BitField<6, 3, u32_le> address_bits_36_38; | 
					
						
							|  |  |  |         BitField<9, 3, u32_le> counter_bits_9_11; | 
					
						
							|  |  |  |         BitField<12, 4, u32_le> address_bits_32_35; | 
					
						
							|  |  |  |         BitField<16, 16, u32_le> size; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32_le address_bits_0_31; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32_le Counter() const { | 
					
						
							| 
									
										
										
										
											2018-01-16 18:05:21 +00:00
										 |  |  |         u32_le counter{counter_bits_0_5}; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         counter |= counter_bits_9_11 << 9; | 
					
						
							|  |  |  |         return counter; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VAddr Address() const { | 
					
						
							| 
									
										
										
										
											2018-01-16 18:05:21 +00:00
										 |  |  |         VAddr address{address_bits_0_31}; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         address |= static_cast<VAddr>(address_bits_32_35) << 32; | 
					
						
							|  |  |  |         address |= static_cast<VAddr>(address_bits_36_38) << 36; | 
					
						
							|  |  |  |         return address; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-13 21:41:20 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     u64 Size() const { | 
					
						
							|  |  |  |         return static_cast<u64>(size); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(BufferDescriptorX) == 8, "BufferDescriptorX size is incorrect"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BufferDescriptorABW { | 
					
						
							|  |  |  |     u32_le size_bits_0_31; | 
					
						
							|  |  |  |     u32_le address_bits_0_31; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |         BitField<0, 2, u32_le> flags; | 
					
						
							|  |  |  |         BitField<2, 3, u32_le> address_bits_36_38; | 
					
						
							|  |  |  |         BitField<24, 4, u32_le> size_bits_32_35; | 
					
						
							|  |  |  |         BitField<28, 4, u32_le> address_bits_32_35; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VAddr Address() const { | 
					
						
							| 
									
										
										
										
											2018-01-16 18:05:21 +00:00
										 |  |  |         VAddr address{address_bits_0_31}; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         address |= static_cast<VAddr>(address_bits_32_35) << 32; | 
					
						
							|  |  |  |         address |= static_cast<VAddr>(address_bits_36_38) << 36; | 
					
						
							|  |  |  |         return address; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u64 Size() const { | 
					
						
							| 
									
										
										
										
											2018-01-16 18:05:21 +00:00
										 |  |  |         u64 size{size_bits_0_31}; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         size |= static_cast<u64>(size_bits_32_35) << 32; | 
					
						
							|  |  |  |         return size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(BufferDescriptorABW) == 12, "BufferDescriptorABW size is incorrect"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BufferDescriptorC { | 
					
						
							|  |  |  |     u32_le address_bits_0_31; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |         BitField<0, 16, u32_le> address_bits_32_47; | 
					
						
							|  |  |  |         BitField<16, 16, u32_le> size; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VAddr Address() const { | 
					
						
							| 
									
										
										
										
											2018-01-16 18:05:21 +00:00
										 |  |  |         VAddr address{address_bits_0_31}; | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  |         address |= static_cast<VAddr>(address_bits_32_47) << 32; | 
					
						
							|  |  |  |         return address; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-18 16:54:34 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     u64 Size() const { | 
					
						
							|  |  |  |         return static_cast<u64>(size); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(BufferDescriptorC) == 8, "BufferDescriptorC size is incorrect"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct DataPayloadHeader { | 
					
						
							|  |  |  |     u32_le magic; | 
					
						
							|  |  |  |     INSERT_PADDING_WORDS(1); | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-09-13 11:59:23 -05:00
										 |  |  | static_assert(sizeof(DataPayloadHeader) == 8, "DataPayloadHeader size is incorrect"); | 
					
						
							| 
									
										
										
										
											2017-10-14 22:18:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | struct DomainMessageHeader { | 
					
						
							| 
									
										
										
										
											2018-01-17 01:16:55 -05:00
										 |  |  |     enum class CommandType : u32_le { | 
					
						
							|  |  |  |         SendMessage = 1, | 
					
						
							|  |  |  |         CloseVirtualHandle = 2, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     union { | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |         // Used when responding to an IPC request, Server -> Client.
 | 
					
						
							|  |  |  |         struct { | 
					
						
							|  |  |  |             u32_le num_objects; | 
					
						
							|  |  |  |             INSERT_PADDING_WORDS(3); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Used when performing an IPC request, Client -> Server.
 | 
					
						
							|  |  |  |         struct { | 
					
						
							|  |  |  |             union { | 
					
						
							| 
									
										
										
										
											2018-01-17 01:16:55 -05:00
										 |  |  |                 BitField<0, 8, CommandType> command; | 
					
						
							| 
									
										
										
										
											2018-04-30 23:24:31 -04:00
										 |  |  |                 BitField<8, 8, u32_le> input_object_count; | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  |                 BitField<16, 16, u32_le> size; | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |             u32_le object_id; | 
					
						
							|  |  |  |             INSERT_PADDING_WORDS(2); | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2017-10-15 01:24:22 -04:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-01-07 01:50:55 -05:00
										 |  |  | static_assert(sizeof(DomainMessageHeader) == 16, "DomainMessageHeader size is incorrect"); | 
					
						
							| 
									
										
										
										
											2016-12-08 11:06:19 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | } // namespace IPC
 |