| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | // Copyright 2018 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <cstddef>
 | 
					
						
							|  |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "input_common/udp/protocol.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace InputCommon::CemuhookUDP { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-03 07:04:28 +01:00
										 |  |  | static constexpr std::size_t GetSizeOfResponseType(Type t) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     switch (t) { | 
					
						
							|  |  |  |     case Type::Version: | 
					
						
							|  |  |  |         return sizeof(Response::Version); | 
					
						
							|  |  |  |     case Type::PortInfo: | 
					
						
							|  |  |  |         return sizeof(Response::PortInfo); | 
					
						
							|  |  |  |     case Type::PadData: | 
					
						
							|  |  |  |         return sizeof(Response::PadData); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Response { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Returns Type if the packet is valid, else none | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note: Modifies the buffer to zero out the crc (since thats the easiest way to check without | 
					
						
							|  |  |  |  * copying the buffer) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | std::optional<Type> Validate(u8* data, std::size_t size) { | 
					
						
							|  |  |  |     if (size < sizeof(Header)) { | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-11-03 07:04:28 +01:00
										 |  |  |     Header header{}; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     std::memcpy(&header, data, sizeof(Header)); | 
					
						
							|  |  |  |     if (header.magic != SERVER_MAGIC) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "UDP Packet has an unexpected magic value"); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (header.protocol_version != PROTOCOL_VERSION) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "UDP Packet protocol mismatch"); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (header.type < Type::Version || header.type > Type::PadData) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "UDP Packet is an unknown type"); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Packet size must equal sizeof(Header) + sizeof(Data)
 | 
					
						
							|  |  |  |     // and also verify that the packet info mentions the correct size. Since the spec includes the
 | 
					
						
							|  |  |  |     // type of the packet as part of the data, we need to include it in size calculations here
 | 
					
						
							|  |  |  |     // ie: payload_length == sizeof(T) + sizeof(Type)
 | 
					
						
							|  |  |  |     const std::size_t data_len = GetSizeOfResponseType(header.type); | 
					
						
							|  |  |  |     if (header.payload_length != data_len + sizeof(Type) || size < data_len + sizeof(Header)) { | 
					
						
							|  |  |  |         LOG_ERROR( | 
					
						
							|  |  |  |             Input, | 
					
						
							|  |  |  |             "UDP Packet payload length doesn't match. Received: {} PayloadLength: {} Expected: {}", | 
					
						
							|  |  |  |             size, header.payload_length, data_len + sizeof(Type)); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u32 crc32 = header.crc; | 
					
						
							|  |  |  |     boost::crc_32_type result; | 
					
						
							|  |  |  |     // zero out the crc in the buffer and then run the crc against it
 | 
					
						
							|  |  |  |     std::memset(&data[offsetof(Header, crc)], 0, sizeof(u32_le)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result.process_bytes(data, data_len + sizeof(Header)); | 
					
						
							|  |  |  |     if (crc32 != result.checksum()) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "UDP Packet CRC check failed. Offset: {}", offsetof(Header, crc)); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return header.type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } // namespace Response
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace InputCommon::CemuhookUDP
 |