| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "common/swap.h"
 | 
					
						
							|  |  |  | #include "common/thread.h"
 | 
					
						
							|  |  |  | #include "input_common/helpers/joycon_driver.h"
 | 
					
						
							| 
									
										
										
										
											2022-12-22 01:07:46 -06:00
										 |  |  | #include "input_common/helpers/joycon_protocol/calibration.h"
 | 
					
						
							|  |  |  | #include "input_common/helpers/joycon_protocol/generic_functions.h"
 | 
					
						
							|  |  |  | #include "input_common/helpers/joycon_protocol/nfc.h"
 | 
					
						
							|  |  |  | #include "input_common/helpers/joycon_protocol/poller.h"
 | 
					
						
							|  |  |  | #include "input_common/helpers/joycon_protocol/ringcon.h"
 | 
					
						
							|  |  |  | #include "input_common/helpers/joycon_protocol/rumble.h"
 | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace InputCommon::Joycon { | 
					
						
							|  |  |  | JoyconDriver::JoyconDriver(std::size_t port_) : port{port_} { | 
					
						
							|  |  |  |     hidapi_handle = std::make_shared<JoyconHandle>(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | JoyconDriver::~JoyconDriver() { | 
					
						
							|  |  |  |     Stop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void JoyconDriver::Stop() { | 
					
						
							|  |  |  |     is_connected = false; | 
					
						
							|  |  |  |     input_thread = {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::RequestDeviceAccess(SDL_hid_device_info* device_info) { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handle_device_type = ControllerType::None; | 
					
						
							|  |  |  |     GetDeviceType(device_info, handle_device_type); | 
					
						
							|  |  |  |     if (handle_device_type == ControllerType::None) { | 
					
						
							|  |  |  |         return DriverResult::UnsupportedControllerType; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hidapi_handle->handle = | 
					
						
							|  |  |  |         SDL_hid_open(device_info->vendor_id, device_info->product_id, device_info->serial_number); | 
					
						
							|  |  |  |     std::memcpy(&handle_serial_number, device_info->serial_number, 15); | 
					
						
							|  |  |  |     if (!hidapi_handle->handle) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "Yuzu can't gain access to this device: ID {:04X}:{:04X}.", | 
					
						
							|  |  |  |                   device_info->vendor_id, device_info->product_id); | 
					
						
							|  |  |  |         return DriverResult::HandleInUse; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SDL_hid_set_nonblocking(hidapi_handle->handle, 1); | 
					
						
							|  |  |  |     return DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::InitializeDevice() { | 
					
						
							|  |  |  |     if (!hidapi_handle->handle) { | 
					
						
							|  |  |  |         return DriverResult::InvalidHandle; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     disable_input_thread = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reset Counters
 | 
					
						
							|  |  |  |     error_counter = 0; | 
					
						
							|  |  |  |     hidapi_handle->packet_counter = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 19:10:42 -06:00
										 |  |  |     // Reset external device status
 | 
					
						
							|  |  |  |     starlink_connected = false; | 
					
						
							|  |  |  |     ring_connected = false; | 
					
						
							|  |  |  |     amiibo_detected = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     // Set HW default configuration
 | 
					
						
							|  |  |  |     vibration_enabled = true; | 
					
						
							|  |  |  |     motion_enabled = true; | 
					
						
							|  |  |  |     hidbus_enabled = false; | 
					
						
							|  |  |  |     nfc_enabled = false; | 
					
						
							|  |  |  |     passive_enabled = false; | 
					
						
							| 
									
										
										
										
											2022-12-20 19:10:42 -06:00
										 |  |  |     irs_enabled = false; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     gyro_sensitivity = Joycon::GyroSensitivity::DPS2000; | 
					
						
							|  |  |  |     gyro_performance = Joycon::GyroPerformance::HZ833; | 
					
						
							|  |  |  |     accelerometer_sensitivity = Joycon::AccelerometerSensitivity::G8; | 
					
						
							|  |  |  |     accelerometer_performance = Joycon::AccelerometerPerformance::HZ100; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize HW Protocols
 | 
					
						
							| 
									
										
										
										
											2022-12-20 18:09:59 -06:00
										 |  |  |     calibration_protocol = std::make_unique<CalibrationProtocol>(hidapi_handle); | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     generic_protocol = std::make_unique<GenericProtocol>(hidapi_handle); | 
					
						
							| 
									
										
										
										
											2022-12-22 01:07:46 -06:00
										 |  |  |     nfc_protocol = std::make_unique<NfcProtocol>(hidapi_handle); | 
					
						
							| 
									
										
										
										
											2022-12-20 19:10:42 -06:00
										 |  |  |     ring_protocol = std::make_unique<RingConProtocol>(hidapi_handle); | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |     rumble_protocol = std::make_unique<RumbleProtocol>(hidapi_handle); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Get fixed joycon info
 | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     generic_protocol->GetVersionNumber(version); | 
					
						
							|  |  |  |     generic_protocol->GetColor(color); | 
					
						
							|  |  |  |     if (handle_device_type == ControllerType::Pro) { | 
					
						
							|  |  |  |         // Some 3rd party controllers aren't pro controllers
 | 
					
						
							|  |  |  |         generic_protocol->GetControllerType(device_type); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         device_type = handle_device_type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     generic_protocol->GetSerialNumber(serial_number); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     supported_features = GetSupportedFeatures(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get Calibration data
 | 
					
						
							| 
									
										
										
										
											2022-12-20 18:09:59 -06:00
										 |  |  |     calibration_protocol->GetLeftJoyStickCalibration(left_stick_calibration); | 
					
						
							|  |  |  |     calibration_protocol->GetRightJoyStickCalibration(right_stick_calibration); | 
					
						
							|  |  |  |     calibration_protocol->GetImuCalibration(motion_calibration); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set led status
 | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     generic_protocol->SetLedBlinkPattern(static_cast<u8>(1 + port)); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Apply HW configuration
 | 
					
						
							|  |  |  |     SetPollingMode(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |     // Initialize joycon poller
 | 
					
						
							|  |  |  |     joycon_poller = std::make_unique<JoyconPoller>(device_type, left_stick_calibration, | 
					
						
							|  |  |  |                                                    right_stick_calibration, motion_calibration); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     // Start pooling for data
 | 
					
						
							|  |  |  |     is_connected = true; | 
					
						
							|  |  |  |     if (!input_thread_running) { | 
					
						
							|  |  |  |         input_thread = | 
					
						
							|  |  |  |             std::jthread([this](std::stop_token stop_token) { InputThread(stop_token); }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     disable_input_thread = false; | 
					
						
							|  |  |  |     return DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void JoyconDriver::InputThread(std::stop_token stop_token) { | 
					
						
							|  |  |  |     LOG_INFO(Input, "JC Adapter input thread started"); | 
					
						
							|  |  |  |     Common::SetCurrentThreadName("JoyconInput"); | 
					
						
							|  |  |  |     input_thread_running = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Max update rate is 5ms, ensure we are always able to read a bit faster
 | 
					
						
							|  |  |  |     constexpr int ThreadDelay = 2; | 
					
						
							|  |  |  |     std::vector<u8> buffer(MaxBufferSize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (!stop_token.stop_requested()) { | 
					
						
							|  |  |  |         int status = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!IsInputThreadValid()) { | 
					
						
							|  |  |  |             input_thread.request_stop(); | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // By disabling the input thread we can ensure custom commands will succeed as no package is
 | 
					
						
							|  |  |  |         // skipped
 | 
					
						
							|  |  |  |         if (!disable_input_thread) { | 
					
						
							|  |  |  |             status = SDL_hid_read_timeout(hidapi_handle->handle, buffer.data(), buffer.size(), | 
					
						
							|  |  |  |                                           ThreadDelay); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             std::this_thread::sleep_for(std::chrono::milliseconds(ThreadDelay)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (IsPayloadCorrect(status, buffer)) { | 
					
						
							|  |  |  |             OnNewData(buffer); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::this_thread::yield(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     is_connected = false; | 
					
						
							|  |  |  |     input_thread_running = false; | 
					
						
							|  |  |  |     LOG_INFO(Input, "JC Adapter input thread stopped"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void JoyconDriver::OnNewData(std::span<u8> buffer) { | 
					
						
							|  |  |  |     const auto report_mode = static_cast<InputReport>(buffer[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |     // Packages can be a litte bit inconsistent. Average the delta time to provide a smoother motion
 | 
					
						
							|  |  |  |     // experience
 | 
					
						
							|  |  |  |     switch (report_mode) { | 
					
						
							|  |  |  |     case InputReport::STANDARD_FULL_60HZ: | 
					
						
							|  |  |  |     case InputReport::NFC_IR_MODE_60HZ: | 
					
						
							|  |  |  |     case InputReport::SIMPLE_HID_MODE: { | 
					
						
							|  |  |  |         const auto now = std::chrono::steady_clock::now(); | 
					
						
							|  |  |  |         const auto new_delta_time = static_cast<u64>( | 
					
						
							|  |  |  |             std::chrono::duration_cast<std::chrono::microseconds>(now - last_update).count()); | 
					
						
							|  |  |  |         delta_time = ((delta_time * 8) + (new_delta_time * 2)) / 10; | 
					
						
							|  |  |  |         last_update = now; | 
					
						
							|  |  |  |         joycon_poller->UpdateColor(color); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const MotionStatus motion_status{ | 
					
						
							|  |  |  |         .is_enabled = motion_enabled, | 
					
						
							|  |  |  |         .delta_time = delta_time, | 
					
						
							|  |  |  |         .gyro_sensitivity = gyro_sensitivity, | 
					
						
							|  |  |  |         .accelerometer_sensitivity = accelerometer_sensitivity, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 19:10:42 -06:00
										 |  |  |     // TODO: Remove this when calibration is properly loaded and not calculated
 | 
					
						
							|  |  |  |     if (ring_connected && report_mode == InputReport::STANDARD_FULL_60HZ) { | 
					
						
							|  |  |  |         InputReportActive data{}; | 
					
						
							|  |  |  |         memcpy(&data, buffer.data(), sizeof(InputReportActive)); | 
					
						
							|  |  |  |         calibration_protocol->GetRingCalibration(ring_calibration, data.ring_input); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const RingStatus ring_status{ | 
					
						
							|  |  |  |         .is_enabled = ring_connected, | 
					
						
							|  |  |  |         .default_value = ring_calibration.default_value, | 
					
						
							|  |  |  |         .max_value = ring_calibration.max_value, | 
					
						
							|  |  |  |         .min_value = ring_calibration.min_value, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 01:07:46 -06:00
										 |  |  |     if (nfc_protocol->IsEnabled()) { | 
					
						
							|  |  |  |         if (amiibo_detected) { | 
					
						
							|  |  |  |             if (!nfc_protocol->HasAmiibo()) { | 
					
						
							|  |  |  |                 joycon_poller->updateAmiibo({}); | 
					
						
							|  |  |  |                 amiibo_detected = false; | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!amiibo_detected) { | 
					
						
							|  |  |  |             std::vector<u8> data(0x21C); | 
					
						
							|  |  |  |             const auto result = nfc_protocol->ScanAmiibo(data); | 
					
						
							|  |  |  |             if (result == DriverResult::Success) { | 
					
						
							|  |  |  |                 joycon_poller->updateAmiibo(data); | 
					
						
							|  |  |  |                 amiibo_detected = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     switch (report_mode) { | 
					
						
							|  |  |  |     case InputReport::STANDARD_FULL_60HZ: | 
					
						
							| 
									
										
										
										
											2022-12-20 19:10:42 -06:00
										 |  |  |         joycon_poller->ReadActiveMode(buffer, motion_status, ring_status); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case InputReport::NFC_IR_MODE_60HZ: | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |         joycon_poller->ReadNfcIRMode(buffer, motion_status); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case InputReport::SIMPLE_HID_MODE: | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |         joycon_poller->ReadPassiveMode(buffer); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     case InputReport::SUBCMD_REPLY: | 
					
						
							|  |  |  |         LOG_DEBUG(Input, "Unhandled command reply"); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     default: | 
					
						
							|  |  |  |         LOG_ERROR(Input, "Report mode not Implemented {}", report_mode); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void JoyconDriver::SetPollingMode() { | 
					
						
							|  |  |  |     disable_input_thread = true; | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |     rumble_protocol->EnableRumble(vibration_enabled && supported_features.vibration); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     if (motion_enabled && supported_features.motion) { | 
					
						
							|  |  |  |         generic_protocol->EnableImu(true); | 
					
						
							|  |  |  |         generic_protocol->SetImuConfig(gyro_sensitivity, gyro_performance, | 
					
						
							|  |  |  |                                        accelerometer_sensitivity, accelerometer_performance); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         generic_protocol->EnableImu(false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 01:07:46 -06:00
										 |  |  |     if (nfc_protocol->IsEnabled()) { | 
					
						
							|  |  |  |         amiibo_detected = false; | 
					
						
							|  |  |  |         nfc_protocol->DisableNfc(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (nfc_enabled && supported_features.nfc) { | 
					
						
							|  |  |  |         auto result = nfc_protocol->EnableNfc(); | 
					
						
							|  |  |  |         if (result == DriverResult::Success) { | 
					
						
							|  |  |  |             result = nfc_protocol->StartNFCPollingMode(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (result == DriverResult::Success) { | 
					
						
							|  |  |  |             disable_input_thread = false; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         nfc_protocol->DisableNfc(); | 
					
						
							|  |  |  |         LOG_ERROR(Input, "Error enabling NFC"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 19:10:42 -06:00
										 |  |  |     if (ring_protocol->IsEnabled()) { | 
					
						
							|  |  |  |         ring_connected = false; | 
					
						
							|  |  |  |         ring_protocol->DisableRingCon(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hidbus_enabled && supported_features.hidbus) { | 
					
						
							|  |  |  |         auto result = ring_protocol->EnableRingCon(); | 
					
						
							|  |  |  |         if (result == DriverResult::Success) { | 
					
						
							|  |  |  |             result = ring_protocol->StartRingconPolling(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (result == DriverResult::Success) { | 
					
						
							|  |  |  |             ring_connected = true; | 
					
						
							|  |  |  |             disable_input_thread = false; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ring_connected = false; | 
					
						
							|  |  |  |         ring_protocol->DisableRingCon(); | 
					
						
							|  |  |  |         LOG_ERROR(Input, "Error enabling Ringcon"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     if (passive_enabled && supported_features.passive) { | 
					
						
							|  |  |  |         const auto result = generic_protocol->EnablePassiveMode(); | 
					
						
							|  |  |  |         if (result == DriverResult::Success) { | 
					
						
							|  |  |  |             disable_input_thread = false; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         LOG_ERROR(Input, "Error enabling passive mode"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Default Mode
 | 
					
						
							|  |  |  |     const auto result = generic_protocol->EnableActiveMode(); | 
					
						
							|  |  |  |     if (result != DriverResult::Success) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "Error enabling active mode"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     disable_input_thread = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | JoyconDriver::SupportedFeatures JoyconDriver::GetSupportedFeatures() { | 
					
						
							|  |  |  |     SupportedFeatures features{ | 
					
						
							|  |  |  |         .passive = true, | 
					
						
							|  |  |  |         .motion = true, | 
					
						
							|  |  |  |         .vibration = true, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (device_type == ControllerType::Right) { | 
					
						
							|  |  |  |         features.nfc = true; | 
					
						
							|  |  |  |         features.irs = true; | 
					
						
							|  |  |  |         features.hidbus = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (device_type == ControllerType::Pro) { | 
					
						
							|  |  |  |         features.nfc = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return features; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool JoyconDriver::IsInputThreadValid() const { | 
					
						
							|  |  |  |     if (!is_connected) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (hidapi_handle->handle == nullptr) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // Controller is not responding. Terminate connection
 | 
					
						
							|  |  |  |     if (error_counter > MaxErrorCount) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool JoyconDriver::IsPayloadCorrect(int status, std::span<const u8> buffer) { | 
					
						
							|  |  |  |     if (status <= -1) { | 
					
						
							|  |  |  |         error_counter++; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // There's no new data
 | 
					
						
							|  |  |  |     if (status == 0) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // No reply ever starts with zero
 | 
					
						
							|  |  |  |     if (buffer[0] == 0x00) { | 
					
						
							|  |  |  |         error_counter++; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     error_counter = 0; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::SetVibration(const VibrationValue& vibration) { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     if (disable_input_thread) { | 
					
						
							|  |  |  |         return DriverResult::HandleInUse; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  |     return rumble_protocol->SendVibration(vibration); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::SetLedConfig(u8 led_pattern) { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     if (disable_input_thread) { | 
					
						
							|  |  |  |         return DriverResult::HandleInUse; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return generic_protocol->SetLedPattern(led_pattern); | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::SetPasiveMode() { | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     motion_enabled = false; | 
					
						
							|  |  |  |     hidbus_enabled = false; | 
					
						
							|  |  |  |     nfc_enabled = false; | 
					
						
							|  |  |  |     passive_enabled = true; | 
					
						
							|  |  |  |     SetPollingMode(); | 
					
						
							|  |  |  |     return DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::SetActiveMode() { | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     motion_enabled = true; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     hidbus_enabled = false; | 
					
						
							|  |  |  |     nfc_enabled = false; | 
					
						
							|  |  |  |     passive_enabled = false; | 
					
						
							|  |  |  |     SetPollingMode(); | 
					
						
							|  |  |  |     return DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::SetNfcMode() { | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2022-12-22 20:47:51 -06:00
										 |  |  |     motion_enabled = true; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     hidbus_enabled = false; | 
					
						
							|  |  |  |     nfc_enabled = true; | 
					
						
							|  |  |  |     passive_enabled = false; | 
					
						
							|  |  |  |     SetPollingMode(); | 
					
						
							|  |  |  |     return DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DriverResult JoyconDriver::SetRingConMode() { | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |     motion_enabled = true; | 
					
						
							|  |  |  |     hidbus_enabled = true; | 
					
						
							|  |  |  |     nfc_enabled = false; | 
					
						
							|  |  |  |     passive_enabled = false; | 
					
						
							|  |  |  |     SetPollingMode(); | 
					
						
							|  |  |  |     return DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool JoyconDriver::IsConnected() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return is_connected; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool JoyconDriver::IsVibrationEnabled() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return vibration_enabled; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FirmwareVersion JoyconDriver::GetDeviceVersion() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return version; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Color JoyconDriver::GetDeviceColor() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return color; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::size_t JoyconDriver::GetDevicePort() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return port; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ControllerType JoyconDriver::GetDeviceType() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							| 
									
										
										
										
											2022-12-20 14:30:03 -06:00
										 |  |  |     return device_type; | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ControllerType JoyconDriver::GetHandleDeviceType() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return handle_device_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SerialNumber JoyconDriver::GetSerialNumber() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return serial_number; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SerialNumber JoyconDriver::GetHandleSerialNumber() const { | 
					
						
							|  |  |  |     std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |     return handle_serial_number; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 20:27:34 -06:00
										 |  |  | void JoyconDriver::SetCallbacks(const Joycon::JoyconCallbacks& callbacks) { | 
					
						
							|  |  |  |     joycon_poller->SetCallbacks(callbacks); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  | Joycon::DriverResult JoyconDriver::GetDeviceType(SDL_hid_device_info* device_info, | 
					
						
							|  |  |  |                                                  ControllerType& controller_type) { | 
					
						
							| 
									
										
										
										
											2022-12-23 08:32:02 -06:00
										 |  |  |     static constexpr std::array<std::pair<u32, Joycon::ControllerType>, 4> supported_devices{ | 
					
						
							| 
									
										
										
										
											2022-12-20 11:34:33 -06:00
										 |  |  |         std::pair<u32, Joycon::ControllerType>{0x2006, Joycon::ControllerType::Left}, | 
					
						
							|  |  |  |         {0x2007, Joycon::ControllerType::Right}, | 
					
						
							|  |  |  |         {0x2009, Joycon::ControllerType::Pro}, | 
					
						
							|  |  |  |         {0x200E, Joycon::ControllerType::Grip}, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     constexpr u16 nintendo_vendor_id = 0x057e; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     controller_type = Joycon::ControllerType::None; | 
					
						
							|  |  |  |     if (device_info->vendor_id != nintendo_vendor_id) { | 
					
						
							|  |  |  |         return Joycon::DriverResult::UnsupportedControllerType; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& [product_id, type] : supported_devices) { | 
					
						
							|  |  |  |         if (device_info->product_id == static_cast<u16>(product_id)) { | 
					
						
							|  |  |  |             controller_type = type; | 
					
						
							|  |  |  |             return Joycon::DriverResult::Success; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return Joycon::DriverResult::UnsupportedControllerType; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Joycon::DriverResult JoyconDriver::GetSerialNumber(SDL_hid_device_info* device_info, | 
					
						
							|  |  |  |                                                    Joycon::SerialNumber& serial_number) { | 
					
						
							|  |  |  |     if (device_info->serial_number == nullptr) { | 
					
						
							|  |  |  |         return Joycon::DriverResult::Unknown; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     std::memcpy(&serial_number, device_info->serial_number, 15); | 
					
						
							|  |  |  |     return Joycon::DriverResult::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace InputCommon::Joycon
 |