| 
									
										
										
										
											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 <chrono>
 | 
					
						
							|  |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | #include <boost/asio.hpp>
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | #include "input_common/udp/client.h"
 | 
					
						
							|  |  |  | #include "input_common/udp/protocol.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using boost::asio::ip::udp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace InputCommon::CemuhookUDP { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct SocketCallback { | 
					
						
							|  |  |  |     std::function<void(Response::Version)> version; | 
					
						
							|  |  |  |     std::function<void(Response::PortInfo)> port_info; | 
					
						
							|  |  |  |     std::function<void(Response::PadData)> pad_data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Socket { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using clock = std::chrono::system_clock; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |     explicit Socket(const std::string& host, u16 port, std::size_t pad_index_, u32 client_id_, | 
					
						
							|  |  |  |                     SocketCallback callback_) | 
					
						
							|  |  |  |         : callback(std::move(callback_)), timer(io_service), | 
					
						
							|  |  |  |           socket(io_service, udp::endpoint(udp::v4(), 0)), client_id(client_id_), | 
					
						
							|  |  |  |           pad_index(pad_index_) { | 
					
						
							| 
									
										
										
										
											2020-03-03 23:46:05 -07:00
										 |  |  |         boost::system::error_code ec{}; | 
					
						
							|  |  |  |         auto ipv4 = boost::asio::ip::make_address_v4(host, ec); | 
					
						
							| 
									
										
										
										
											2020-03-17 12:29:25 +01:00
										 |  |  |         if (ec.value() != boost::system::errc::success) { | 
					
						
							| 
									
										
										
										
											2020-03-03 23:46:05 -07:00
										 |  |  |             LOG_ERROR(Input, "Invalid IPv4 address \"{}\" provided to socket", host); | 
					
						
							|  |  |  |             ipv4 = boost::asio::ip::address_v4{}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         send_endpoint = {udp::endpoint(ipv4, port)}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void Stop() { | 
					
						
							|  |  |  |         io_service.stop(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Loop() { | 
					
						
							|  |  |  |         io_service.run(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void StartSend(const clock::time_point& from) { | 
					
						
							|  |  |  |         timer.expires_at(from + std::chrono::seconds(3)); | 
					
						
							|  |  |  |         timer.async_wait([this](const boost::system::error_code& error) { HandleSend(error); }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void StartReceive() { | 
					
						
							|  |  |  |         socket.async_receive_from( | 
					
						
							|  |  |  |             boost::asio::buffer(receive_buffer), receive_endpoint, | 
					
						
							|  |  |  |             [this](const boost::system::error_code& error, std::size_t bytes_transferred) { | 
					
						
							|  |  |  |                 HandleReceive(error, bytes_transferred); | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-11-15 06:22:01 -05:00
										 |  |  |     void HandleReceive(const boost::system::error_code&, std::size_t bytes_transferred) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         if (auto type = Response::Validate(receive_buffer.data(), bytes_transferred)) { | 
					
						
							|  |  |  |             switch (*type) { | 
					
						
							|  |  |  |             case Type::Version: { | 
					
						
							|  |  |  |                 Response::Version version; | 
					
						
							|  |  |  |                 std::memcpy(&version, &receive_buffer[sizeof(Header)], sizeof(Response::Version)); | 
					
						
							|  |  |  |                 callback.version(std::move(version)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Type::PortInfo: { | 
					
						
							|  |  |  |                 Response::PortInfo port_info; | 
					
						
							|  |  |  |                 std::memcpy(&port_info, &receive_buffer[sizeof(Header)], | 
					
						
							|  |  |  |                             sizeof(Response::PortInfo)); | 
					
						
							|  |  |  |                 callback.port_info(std::move(port_info)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Type::PadData: { | 
					
						
							|  |  |  |                 Response::PadData pad_data; | 
					
						
							|  |  |  |                 std::memcpy(&pad_data, &receive_buffer[sizeof(Header)], sizeof(Response::PadData)); | 
					
						
							|  |  |  |                 callback.pad_data(std::move(pad_data)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         StartReceive(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-15 06:22:01 -05:00
										 |  |  |     void HandleSend(const boost::system::error_code&) { | 
					
						
							| 
									
										
										
										
											2020-03-03 23:46:05 -07:00
										 |  |  |         boost::system::error_code _ignored{}; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         // Send a request for getting port info for the pad
 | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |         const Request::PortInfo port_info{1, {static_cast<u8>(pad_index), 0, 0, 0}}; | 
					
						
							| 
									
										
										
										
											2019-11-03 07:04:28 +01:00
										 |  |  |         const auto port_message = Request::Create(port_info, client_id); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         std::memcpy(&send_buffer1, &port_message, PORT_INFO_SIZE); | 
					
						
							| 
									
										
										
										
											2020-03-03 23:46:05 -07:00
										 |  |  |         socket.send_to(boost::asio::buffer(send_buffer1), send_endpoint, {}, _ignored); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Send a request for getting pad data for the pad
 | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |         const Request::PadData pad_data{ | 
					
						
							|  |  |  |             Request::PadData::Flags::Id, | 
					
						
							|  |  |  |             static_cast<u8>(pad_index), | 
					
						
							|  |  |  |             EMPTY_MAC_ADDRESS, | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2019-11-03 07:04:28 +01:00
										 |  |  |         const auto pad_message = Request::Create(pad_data, client_id); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         std::memcpy(send_buffer2.data(), &pad_message, PAD_DATA_SIZE); | 
					
						
							| 
									
										
										
										
											2020-03-03 23:46:05 -07:00
										 |  |  |         socket.send_to(boost::asio::buffer(send_buffer2), send_endpoint, {}, _ignored); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         StartSend(timer.expiry()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SocketCallback callback; | 
					
						
							|  |  |  |     boost::asio::io_service io_service; | 
					
						
							|  |  |  |     boost::asio::basic_waitable_timer<clock> timer; | 
					
						
							|  |  |  |     udp::socket socket; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-03 07:04:28 +01:00
										 |  |  |     u32 client_id{}; | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |     std::size_t pad_index{}; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static constexpr std::size_t PORT_INFO_SIZE = sizeof(Message<Request::PortInfo>); | 
					
						
							|  |  |  |     static constexpr std::size_t PAD_DATA_SIZE = sizeof(Message<Request::PadData>); | 
					
						
							|  |  |  |     std::array<u8, PORT_INFO_SIZE> send_buffer1; | 
					
						
							|  |  |  |     std::array<u8, PAD_DATA_SIZE> send_buffer2; | 
					
						
							|  |  |  |     udp::endpoint send_endpoint; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::array<u8, MAX_PACKET_SIZE> receive_buffer; | 
					
						
							|  |  |  |     udp::endpoint receive_endpoint; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void SocketLoop(Socket* socket) { | 
					
						
							|  |  |  |     socket->StartReceive(); | 
					
						
							|  |  |  |     socket->StartSend(Socket::clock::now()); | 
					
						
							|  |  |  |     socket->Loop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  | Client::Client() { | 
					
						
							|  |  |  |     LOG_INFO(Input, "Udp Initialization started"); | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     ReloadSockets(); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Client::~Client() { | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     Reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<Common::ParamPackage> Client::GetInputDevices() const { | 
					
						
							|  |  |  |     std::vector<Common::ParamPackage> devices; | 
					
						
							|  |  |  |     for (std::size_t client = 0; client < clients.size(); client++) { | 
					
						
							|  |  |  |         if (!DeviceConnected(client)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-09-04 23:47:56 -05:00
										 |  |  |         std::string name = fmt::format("UDP Controller {}", client); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |         devices.emplace_back(Common::ParamPackage{ | 
					
						
							|  |  |  |             {"class", "cemuhookudp"}, | 
					
						
							|  |  |  |             {"display", std::move(name)}, | 
					
						
							|  |  |  |             {"port", std::to_string(client)}, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return devices; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | bool Client::DeviceConnected(std::size_t client) const { | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     // Use last timestamp to detect if the socket has stopped sending data
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     const auto now = std::chrono::steady_clock::now(); | 
					
						
							|  |  |  |     const auto time_difference = | 
					
						
							|  |  |  |         static_cast<u64>(std::chrono::duration_cast<std::chrono::milliseconds>( | 
					
						
							|  |  |  |                              now - clients[client].last_motion_update) | 
					
						
							|  |  |  |                              .count()); | 
					
						
							|  |  |  |     return time_difference < 1000 && clients[client].active == 1; | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | void Client::ReloadSockets() { | 
					
						
							|  |  |  |     Reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::stringstream servers_ss(Settings::values.udp_input_servers); | 
					
						
							|  |  |  |     std::string server_token; | 
					
						
							|  |  |  |     std::size_t client = 0; | 
					
						
							|  |  |  |     while (std::getline(servers_ss, server_token, ',')) { | 
					
						
							|  |  |  |         if (client == max_udp_clients) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         std::stringstream server_ss(server_token); | 
					
						
							|  |  |  |         std::string token; | 
					
						
							|  |  |  |         std::getline(server_ss, token, ':'); | 
					
						
							|  |  |  |         std::string udp_input_address = token; | 
					
						
							|  |  |  |         std::getline(server_ss, token, ':'); | 
					
						
							|  |  |  |         char* temp; | 
					
						
							|  |  |  |         const u16 udp_input_port = static_cast<u16>(std::strtol(token.c_str(), &temp, 0)); | 
					
						
							|  |  |  |         if (*temp != '\0') { | 
					
						
							|  |  |  |             LOG_ERROR(Input, "Port number is not valid {}", token); | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (std::size_t pad = 0; pad < 4; ++pad) { | 
					
						
							|  |  |  |             const std::size_t client_number = | 
					
						
							|  |  |  |                 GetClientNumber(udp_input_address, udp_input_port, pad); | 
					
						
							|  |  |  |             if (client_number != max_udp_clients) { | 
					
						
							|  |  |  |                 LOG_ERROR(Input, "Duplicated UDP servers found"); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             StartCommunication(client++, udp_input_address, udp_input_port, pad, 24872); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | std::size_t Client::GetClientNumber(std::string_view host, u16 port, std::size_t pad) const { | 
					
						
							|  |  |  |     for (std::size_t client = 0; client < clients.size(); client++) { | 
					
						
							|  |  |  |         if (clients[client].active == -1) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (clients[client].host == host && clients[client].port == port && | 
					
						
							|  |  |  |             clients[client].pad_index == pad) { | 
					
						
							|  |  |  |             return client; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return max_udp_clients; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 20:50:35 -08:00
										 |  |  | void Client::OnVersion([[maybe_unused]] Response::Version data) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     LOG_TRACE(Input, "Version packet received: {}", data.version); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 20:50:35 -08:00
										 |  |  | void Client::OnPortInfo([[maybe_unused]] Response::PortInfo data) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     LOG_TRACE(Input, "PortInfo packet received: {}", data.model); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | void Client::OnPadData(Response::PadData data, std::size_t client) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     LOG_TRACE(Input, "PadData packet received"); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     if (data.packet_counter == clients[client].packet_sequence) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         LOG_WARNING( | 
					
						
							|  |  |  |             Input, | 
					
						
							|  |  |  |             "PadData packet dropped because its stale info. Current count: {} Packet count: {}", | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |             clients[client].packet_sequence, data.packet_counter); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     clients[client].active = static_cast<s8>(data.info.is_pad_active); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     clients[client].packet_sequence = data.packet_counter; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     const auto now = std::chrono::steady_clock::now(); | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |     const auto time_difference = | 
					
						
							|  |  |  |         static_cast<u64>(std::chrono::duration_cast<std::chrono::microseconds>( | 
					
						
							|  |  |  |                              now - clients[client].last_motion_update) | 
					
						
							|  |  |  |                              .count()); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     clients[client].last_motion_update = now; | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |     const Common::Vec3f raw_gyroscope = {data.gyro.pitch, data.gyro.roll, -data.gyro.yaw}; | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     clients[client].motion.SetAcceleration({data.accel.x, -data.accel.z, data.accel.y}); | 
					
						
							|  |  |  |     // Gyroscope values are not it the correct scale from better joy.
 | 
					
						
							| 
									
										
										
										
											2020-09-04 23:47:56 -05:00
										 |  |  |     // Dividing by 312 allows us to make one full turn = 1 turn
 | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     // This must be a configurable valued called sensitivity
 | 
					
						
							|  |  |  |     clients[client].motion.SetGyroscope(raw_gyroscope / 312.0f); | 
					
						
							|  |  |  |     clients[client].motion.UpdateRotation(time_difference); | 
					
						
							|  |  |  |     clients[client].motion.UpdateOrientation(time_difference); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         std::lock_guard guard(clients[client].status.update_mutex); | 
					
						
							| 
									
										
										
										
											2020-09-17 20:26:34 -05:00
										 |  |  |         clients[client].status.motion_status = clients[client].motion.GetMotion(); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // TODO: add a setting for "click" touch. Click touch refers to a device that differentiates
 | 
					
						
							|  |  |  |         // between a simple "tap" and a hard press that causes the touch screen to click.
 | 
					
						
							| 
									
										
										
										
											2019-11-03 07:04:28 +01:00
										 |  |  |         const bool is_active = data.touch_1.is_active != 0; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         float x = 0; | 
					
						
							|  |  |  |         float y = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |         if (is_active && clients[client].status.touch_calibration) { | 
					
						
							|  |  |  |             const u16 min_x = clients[client].status.touch_calibration->min_x; | 
					
						
							|  |  |  |             const u16 max_x = clients[client].status.touch_calibration->max_x; | 
					
						
							|  |  |  |             const u16 min_y = clients[client].status.touch_calibration->min_y; | 
					
						
							|  |  |  |             const u16 max_y = clients[client].status.touch_calibration->max_y; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |             x = static_cast<float>(std::clamp(static_cast<u16>(data.touch_1.x), min_x, max_x) - | 
					
						
							|  |  |  |                                    min_x) / | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |                 static_cast<float>(max_x - min_x); | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |             y = static_cast<float>(std::clamp(static_cast<u16>(data.touch_1.y), min_y, max_y) - | 
					
						
							|  |  |  |                                    min_y) / | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |                 static_cast<float>(max_y - min_y); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |         clients[client].status.touch_status = {x, y, is_active}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (configuring) { | 
					
						
							| 
									
										
										
										
											2020-09-17 20:26:34 -05:00
										 |  |  |             const Common::Vec3f gyroscope = clients[client].motion.GetGyroscope(); | 
					
						
							|  |  |  |             const Common::Vec3f accelerometer = clients[client].motion.GetAcceleration(); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |             UpdateYuzuSettings(client, accelerometer, gyroscope, is_active); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  | void Client::StartCommunication(std::size_t client, const std::string& host, u16 port, | 
					
						
							|  |  |  |                                 std::size_t pad_index, u32 client_id) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     SocketCallback callback{[this](Response::Version version) { OnVersion(version); }, | 
					
						
							|  |  |  |                             [this](Response::PortInfo info) { OnPortInfo(info); }, | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |                             [this, client](Response::PadData data) { OnPadData(data, client); }}; | 
					
						
							|  |  |  |     LOG_INFO(Input, "Starting communication with UDP input server on {}:{}:{}", host, port, | 
					
						
							|  |  |  |              pad_index); | 
					
						
							|  |  |  |     clients[client].host = host; | 
					
						
							|  |  |  |     clients[client].port = port; | 
					
						
							|  |  |  |     clients[client].pad_index = pad_index; | 
					
						
							|  |  |  |     clients[client].active = 0; | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     clients[client].socket = std::make_unique<Socket>(host, port, pad_index, client_id, callback); | 
					
						
							|  |  |  |     clients[client].thread = std::thread{SocketLoop, clients[client].socket.get()}; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     // Set motion parameters
 | 
					
						
							|  |  |  |     // SetGyroThreshold value should be dependent on GyroscopeZeroDriftMode
 | 
					
						
							|  |  |  |     // Real HW values are unknown, 0.0001 is an approximate to Standard
 | 
					
						
							|  |  |  |     clients[client].motion.SetGyroThreshold(0.0001f); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Client::Reset() { | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |     for (auto& client : clients) { | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |         if (client.thread.joinable()) { | 
					
						
							|  |  |  |             client.active = -1; | 
					
						
							|  |  |  |             client.socket->Stop(); | 
					
						
							|  |  |  |             client.thread.join(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Client::UpdateYuzuSettings(std::size_t client, const Common::Vec3<float>& acc, | 
					
						
							|  |  |  |                                 const Common::Vec3<float>& gyro, bool touch) { | 
					
						
							| 
									
										
										
										
											2020-09-26 11:32:28 +03:00
										 |  |  |     if (gyro.Length() > 0.2f) { | 
					
						
							|  |  |  |         LOG_DEBUG(Input, "UDP Controller {}: gyro=({}, {}, {}), accel=({}, {}, {}), touch={}", | 
					
						
							|  |  |  |                   client, gyro[0], gyro[1], gyro[2], acc[0], acc[1], acc[2], touch); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     UDPPadStatus pad{ | 
					
						
							|  |  |  |         .host = clients[client].host, | 
					
						
							|  |  |  |         .port = clients[client].port, | 
					
						
							|  |  |  |         .pad_index = clients[client].pad_index, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-09-04 23:47:56 -05:00
										 |  |  |     if (touch) { | 
					
						
							|  |  |  |         pad.touch = PadTouch::Click; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |         pad_queue.Push(pad); | 
					
						
							| 
									
										
										
										
											2020-09-04 23:47:56 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     for (size_t i = 0; i < 3; ++i) { | 
					
						
							| 
									
										
										
										
											2020-09-26 11:32:28 +03:00
										 |  |  |         if (gyro[i] > 5.0f || gyro[i] < -5.0f) { | 
					
						
							| 
									
										
										
										
											2020-09-04 23:47:56 -05:00
										 |  |  |             pad.motion = static_cast<PadMotion>(i); | 
					
						
							|  |  |  |             pad.motion_value = gyro[i]; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |             pad_queue.Push(pad); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-09-26 11:32:28 +03:00
										 |  |  |         if (acc[i] > 1.75f || acc[i] < -1.75f) { | 
					
						
							| 
									
										
										
										
											2020-09-04 23:47:56 -05:00
										 |  |  |             pad.motion = static_cast<PadMotion>(i + 3); | 
					
						
							|  |  |  |             pad.motion_value = acc[i]; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |             pad_queue.Push(pad); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Client::BeginConfiguration() { | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     pad_queue.Clear(); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     configuring = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Client::EndConfiguration() { | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  |     pad_queue.Clear(); | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     configuring = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | DeviceStatus& Client::GetPadState(const std::string& host, u16 port, std::size_t pad) { | 
					
						
							|  |  |  |     const std::size_t client_number = GetClientNumber(host, port, pad); | 
					
						
							|  |  |  |     if (client_number == max_udp_clients) { | 
					
						
							|  |  |  |         return clients[0].status; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return clients[client_number].status; | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | const DeviceStatus& Client::GetPadState(const std::string& host, u16 port, std::size_t pad) const { | 
					
						
							|  |  |  |     const std::size_t client_number = GetClientNumber(host, port, pad); | 
					
						
							|  |  |  |     if (client_number == max_udp_clients) { | 
					
						
							|  |  |  |         return clients[0].status; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return clients[client_number].status; | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | Common::SPSCQueue<UDPPadStatus>& Client::GetPadQueue() { | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     return pad_queue; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 22:16:29 -06:00
										 |  |  | const Common::SPSCQueue<UDPPadStatus>& Client::GetPadQueue() const { | 
					
						
							| 
									
										
										
										
											2020-09-04 21:35:42 -05:00
										 |  |  |     return pad_queue; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  | void TestCommunication(const std::string& host, u16 port, std::size_t pad_index, u32 client_id, | 
					
						
							| 
									
										
										
										
											2020-10-16 06:22:26 -04:00
										 |  |  |                        const std::function<void()>& success_callback, | 
					
						
							|  |  |  |                        const std::function<void()>& failure_callback) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     std::thread([=] { | 
					
						
							|  |  |  |         Common::Event success_event; | 
					
						
							| 
									
										
										
										
											2020-10-16 06:23:48 -04:00
										 |  |  |         SocketCallback callback{ | 
					
						
							|  |  |  |             .version = [](Response::Version) {}, | 
					
						
							|  |  |  |             .port_info = [](Response::PortInfo) {}, | 
					
						
							|  |  |  |             .pad_data = [&](Response::PadData) { success_event.Set(); }, | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2020-02-03 09:24:03 -05:00
										 |  |  |         Socket socket{host, port, pad_index, client_id, std::move(callback)}; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         std::thread worker_thread{SocketLoop, &socket}; | 
					
						
							| 
									
										
										
										
											2020-09-28 04:53:21 -04:00
										 |  |  |         const bool result = success_event.WaitFor(std::chrono::seconds(5)); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         socket.Stop(); | 
					
						
							|  |  |  |         worker_thread.join(); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         if (result) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |             success_callback(); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |             failure_callback(); | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-08-11 11:08:10 -04:00
										 |  |  |     }).detach(); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CalibrationConfigurationJob::CalibrationConfigurationJob( | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |     const std::string& host, u16 port, std::size_t pad_index, u32 client_id, | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |     std::function<void(Status)> status_callback, | 
					
						
							|  |  |  |     std::function<void(u16, u16, u16, u16)> data_callback) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-12 15:49:40 -04:00
										 |  |  |     std::thread([=, this] { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         constexpr u16 CALIBRATION_THRESHOLD = 100; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         u16 min_x{UINT16_MAX}; | 
					
						
							|  |  |  |         u16 min_y{UINT16_MAX}; | 
					
						
							|  |  |  |         u16 max_x{}; | 
					
						
							|  |  |  |         u16 max_y{}; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Status current_status{Status::Initialized}; | 
					
						
							| 
									
										
										
										
											2020-11-15 06:22:01 -05:00
										 |  |  |         SocketCallback callback{[](Response::Version) {}, [](Response::PortInfo) {}, | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |                                 [&](Response::PadData data) { | 
					
						
							|  |  |  |                                     if (current_status == Status::Initialized) { | 
					
						
							|  |  |  |                                         // Receiving data means the communication is ready now
 | 
					
						
							|  |  |  |                                         current_status = Status::Ready; | 
					
						
							|  |  |  |                                         status_callback(current_status); | 
					
						
							|  |  |  |                                     } | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |                                     if (data.touch_1.is_active == 0) { | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |                                         return; | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |                                     } | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |                                     LOG_DEBUG(Input, "Current touch: {} {}", data.touch_1.x, | 
					
						
							|  |  |  |                                               data.touch_1.y); | 
					
						
							|  |  |  |                                     min_x = std::min(min_x, static_cast<u16>(data.touch_1.x)); | 
					
						
							|  |  |  |                                     min_y = std::min(min_y, static_cast<u16>(data.touch_1.y)); | 
					
						
							|  |  |  |                                     if (current_status == Status::Ready) { | 
					
						
							|  |  |  |                                         // First touch - min data (min_x/min_y)
 | 
					
						
							|  |  |  |                                         current_status = Status::Stage1Completed; | 
					
						
							|  |  |  |                                         status_callback(current_status); | 
					
						
							|  |  |  |                                     } | 
					
						
							|  |  |  |                                     if (data.touch_1.x - min_x > CALIBRATION_THRESHOLD && | 
					
						
							|  |  |  |                                         data.touch_1.y - min_y > CALIBRATION_THRESHOLD) { | 
					
						
							|  |  |  |                                         // Set the current position as max value and finishes
 | 
					
						
							|  |  |  |                                         // configuration
 | 
					
						
							|  |  |  |                                         max_x = data.touch_1.x; | 
					
						
							|  |  |  |                                         max_y = data.touch_1.y; | 
					
						
							|  |  |  |                                         current_status = Status::Completed; | 
					
						
							|  |  |  |                                         data_callback(min_x, min_y, max_x, max_y); | 
					
						
							|  |  |  |                                         status_callback(current_status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                                         complete_event.Set(); | 
					
						
							|  |  |  |                                     } | 
					
						
							|  |  |  |                                 }}; | 
					
						
							| 
									
										
										
										
											2020-02-03 09:24:03 -05:00
										 |  |  |         Socket socket{host, port, pad_index, client_id, std::move(callback)}; | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |         std::thread worker_thread{SocketLoop, &socket}; | 
					
						
							|  |  |  |         complete_event.Wait(); | 
					
						
							|  |  |  |         socket.Stop(); | 
					
						
							|  |  |  |         worker_thread.join(); | 
					
						
							| 
									
										
										
										
											2020-08-11 11:08:10 -04:00
										 |  |  |     }).detach(); | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CalibrationConfigurationJob::~CalibrationConfigurationJob() { | 
					
						
							|  |  |  |     Stop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CalibrationConfigurationJob::Stop() { | 
					
						
							|  |  |  |     complete_event.Set(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace InputCommon::CemuhookUDP
 |