mirror of
https://github.com/yuzu-emu/yuzu.git
synced 2024-12-26 20:00:05 +00:00
input_common: Implement joycon nfc
This commit is contained in:
parent
6e33731f29
commit
6d6b7bdbc3
@ -144,7 +144,8 @@ void EmulatedController::LoadDevices() {
|
|||||||
battery_params[RightIndex].Set("battery", true);
|
battery_params[RightIndex].Set("battery", true);
|
||||||
|
|
||||||
camera_params = Common::ParamPackage{"engine:camera,camera:1"};
|
camera_params = Common::ParamPackage{"engine:camera,camera:1"};
|
||||||
nfc_params = Common::ParamPackage{"engine:virtual_amiibo,nfc:1"};
|
nfc_params = right_joycon;
|
||||||
|
nfc_params.Set("nfc", true);
|
||||||
ring_params = Common::ParamPackage{"engine:joycon,axis_x:100,axis_y:101"};
|
ring_params = Common::ParamPackage{"engine:joycon,axis_x:100,axis_y:101"};
|
||||||
|
|
||||||
output_params[LeftIndex] = left_joycon;
|
output_params[LeftIndex] = left_joycon;
|
||||||
|
@ -64,6 +64,8 @@ if (ENABLE_SDL2)
|
|||||||
helpers/joycon_protocol/generic_functions.cpp
|
helpers/joycon_protocol/generic_functions.cpp
|
||||||
helpers/joycon_protocol/generic_functions.h
|
helpers/joycon_protocol/generic_functions.h
|
||||||
helpers/joycon_protocol/joycon_types.h
|
helpers/joycon_protocol/joycon_types.h
|
||||||
|
helpers/joycon_protocol/nfc.cpp
|
||||||
|
helpers/joycon_protocol/nfc.h
|
||||||
helpers/joycon_protocol/poller.cpp
|
helpers/joycon_protocol/poller.cpp
|
||||||
helpers/joycon_protocol/poller.h
|
helpers/joycon_protocol/poller.h
|
||||||
helpers/joycon_protocol/ringcon.cpp
|
helpers/joycon_protocol/ringcon.cpp
|
||||||
|
@ -388,7 +388,9 @@ void Joycons::OnRingConUpdate(f32 ring_data) {
|
|||||||
|
|
||||||
void Joycons::OnAmiiboUpdate(std::size_t port, const std::vector<u8>& amiibo_data) {
|
void Joycons::OnAmiiboUpdate(std::size_t port, const std::vector<u8>& amiibo_data) {
|
||||||
const auto identifier = GetIdentifier(port, Joycon::ControllerType::Right);
|
const auto identifier = GetIdentifier(port, Joycon::ControllerType::Right);
|
||||||
SetNfc(identifier, {Common::Input::NfcState::NewAmiibo, amiibo_data});
|
const auto nfc_state = amiibo_data.empty() ? Common::Input::NfcState::AmiiboRemoved
|
||||||
|
: Common::Input::NfcState::NewAmiibo;
|
||||||
|
SetNfc(identifier, {nfc_state, amiibo_data});
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<Joycon::JoyconDriver> Joycons::GetHandle(PadIdentifier identifier) const {
|
std::shared_ptr<Joycon::JoyconDriver> Joycons::GetHandle(PadIdentifier identifier) const {
|
||||||
|
@ -5,6 +5,12 @@
|
|||||||
#include "common/swap.h"
|
#include "common/swap.h"
|
||||||
#include "common/thread.h"
|
#include "common/thread.h"
|
||||||
#include "input_common/helpers/joycon_driver.h"
|
#include "input_common/helpers/joycon_driver.h"
|
||||||
|
#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"
|
||||||
|
|
||||||
namespace InputCommon::Joycon {
|
namespace InputCommon::Joycon {
|
||||||
JoyconDriver::JoyconDriver(std::size_t port_) : port{port_} {
|
JoyconDriver::JoyconDriver(std::size_t port_) : port{port_} {
|
||||||
@ -72,6 +78,7 @@ DriverResult JoyconDriver::InitializeDevice() {
|
|||||||
// Initialize HW Protocols
|
// Initialize HW Protocols
|
||||||
calibration_protocol = std::make_unique<CalibrationProtocol>(hidapi_handle);
|
calibration_protocol = std::make_unique<CalibrationProtocol>(hidapi_handle);
|
||||||
generic_protocol = std::make_unique<GenericProtocol>(hidapi_handle);
|
generic_protocol = std::make_unique<GenericProtocol>(hidapi_handle);
|
||||||
|
nfc_protocol = std::make_unique<NfcProtocol>(hidapi_handle);
|
||||||
ring_protocol = std::make_unique<RingConProtocol>(hidapi_handle);
|
ring_protocol = std::make_unique<RingConProtocol>(hidapi_handle);
|
||||||
rumble_protocol = std::make_unique<RumbleProtocol>(hidapi_handle);
|
rumble_protocol = std::make_unique<RumbleProtocol>(hidapi_handle);
|
||||||
|
|
||||||
@ -193,6 +200,25 @@ void JoyconDriver::OnNewData(std::span<u8> buffer) {
|
|||||||
.min_value = ring_calibration.min_value,
|
.min_value = ring_calibration.min_value,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
switch (report_mode) {
|
switch (report_mode) {
|
||||||
case InputReport::STANDARD_FULL_60HZ:
|
case InputReport::STANDARD_FULL_60HZ:
|
||||||
joycon_poller->ReadActiveMode(buffer, motion_status, ring_status);
|
joycon_poller->ReadActiveMode(buffer, motion_status, ring_status);
|
||||||
@ -225,6 +251,24 @@ void JoyconDriver::SetPollingMode() {
|
|||||||
generic_protocol->EnableImu(false);
|
generic_protocol->EnableImu(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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");
|
||||||
|
}
|
||||||
|
|
||||||
if (ring_protocol->IsEnabled()) {
|
if (ring_protocol->IsEnabled()) {
|
||||||
ring_connected = false;
|
ring_connected = false;
|
||||||
ring_protocol->DisableRingCon();
|
ring_protocol->DisableRingCon();
|
||||||
|
@ -8,14 +8,15 @@
|
|||||||
#include <span>
|
#include <span>
|
||||||
#include <thread>
|
#include <thread>
|
||||||
|
|
||||||
#include "input_common/helpers/joycon_protocol/calibration.h"
|
|
||||||
#include "input_common/helpers/joycon_protocol/generic_functions.h"
|
|
||||||
#include "input_common/helpers/joycon_protocol/joycon_types.h"
|
#include "input_common/helpers/joycon_protocol/joycon_types.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"
|
|
||||||
|
|
||||||
namespace InputCommon::Joycon {
|
namespace InputCommon::Joycon {
|
||||||
|
class CalibrationProtocol;
|
||||||
|
class GenericProtocol;
|
||||||
|
class NfcProtocol;
|
||||||
|
class JoyconPoller;
|
||||||
|
class RingConProtocol;
|
||||||
|
class RumbleProtocol;
|
||||||
|
|
||||||
class JoyconDriver final {
|
class JoyconDriver final {
|
||||||
public:
|
public:
|
||||||
@ -84,17 +85,18 @@ private:
|
|||||||
SupportedFeatures GetSupportedFeatures();
|
SupportedFeatures GetSupportedFeatures();
|
||||||
|
|
||||||
// Protocol Features
|
// Protocol Features
|
||||||
std::unique_ptr<CalibrationProtocol> calibration_protocol = nullptr;
|
std::unique_ptr<CalibrationProtocol> calibration_protocol;
|
||||||
std::unique_ptr<GenericProtocol> generic_protocol = nullptr;
|
std::unique_ptr<GenericProtocol> generic_protocol;
|
||||||
std::unique_ptr<JoyconPoller> joycon_poller = nullptr;
|
std::unique_ptr<NfcProtocol> nfc_protocol;
|
||||||
std::unique_ptr<RingConProtocol> ring_protocol = nullptr;
|
std::unique_ptr<JoyconPoller> joycon_poller;
|
||||||
std::unique_ptr<RumbleProtocol> rumble_protocol = nullptr;
|
std::unique_ptr<RingConProtocol> ring_protocol;
|
||||||
|
std::unique_ptr<RumbleProtocol> rumble_protocol;
|
||||||
|
|
||||||
// Connection status
|
// Connection status
|
||||||
bool is_connected{};
|
bool is_connected{};
|
||||||
u64 delta_time;
|
u64 delta_time;
|
||||||
std::size_t error_counter{};
|
std::size_t error_counter{};
|
||||||
std::shared_ptr<JoyconHandle> hidapi_handle = nullptr;
|
std::shared_ptr<JoyconHandle> hidapi_handle;
|
||||||
std::chrono::time_point<std::chrono::steady_clock> last_update;
|
std::chrono::time_point<std::chrono::steady_clock> last_update;
|
||||||
|
|
||||||
// External device status
|
// External device status
|
||||||
|
414
src/input_common/helpers/joycon_protocol/nfc.cpp
Normal file
414
src/input_common/helpers/joycon_protocol/nfc.cpp
Normal file
@ -0,0 +1,414 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include <thread>
|
||||||
|
#include "common/logging/log.h"
|
||||||
|
#include "input_common/helpers/joycon_protocol/nfc.h"
|
||||||
|
|
||||||
|
namespace InputCommon::Joycon {
|
||||||
|
|
||||||
|
NfcProtocol::NfcProtocol(std::shared_ptr<JoyconHandle> handle) : JoyconCommonProtocol(handle) {}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::EnableNfc() {
|
||||||
|
LOG_INFO(Input, "Enable NFC");
|
||||||
|
DriverResult result{DriverResult::Success};
|
||||||
|
SetBlocking();
|
||||||
|
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = SetReportMode(ReportMode::NFC_IR_MODE_60HZ);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = EnableMCU(true);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = WaitSetMCUMode(ReportMode::NFC_IR_MODE_60HZ, MCUMode::Standby);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
const MCUConfig config{
|
||||||
|
.command = MCUCommand::ConfigureMCU,
|
||||||
|
.sub_command = MCUSubCommand::SetMCUMode,
|
||||||
|
.mode = MCUMode::NFC,
|
||||||
|
.crc = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
result = ConfigureMCU(config);
|
||||||
|
}
|
||||||
|
|
||||||
|
SetNonBlocking();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::DisableNfc() {
|
||||||
|
LOG_DEBUG(Input, "Disable NFC");
|
||||||
|
DriverResult result{DriverResult::Success};
|
||||||
|
SetBlocking();
|
||||||
|
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = EnableMCU(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
is_enabled = false;
|
||||||
|
|
||||||
|
SetNonBlocking();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::StartNFCPollingMode() {
|
||||||
|
LOG_DEBUG(Input, "Start NFC pooling Mode");
|
||||||
|
DriverResult result{DriverResult::Success};
|
||||||
|
TagFoundData tag_data{};
|
||||||
|
SetBlocking();
|
||||||
|
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = WaitSetMCUMode(ReportMode::NFC_IR_MODE_60HZ, MCUMode::NFC);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = WaitUntilNfcIsReady();
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
is_enabled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SetNonBlocking();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::ScanAmiibo(std::vector<u8>& data) {
|
||||||
|
LOG_DEBUG(Input, "Start NFC pooling Mode");
|
||||||
|
DriverResult result{DriverResult::Success};
|
||||||
|
TagFoundData tag_data{};
|
||||||
|
SetBlocking();
|
||||||
|
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = StartPolling(tag_data);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = ReadTag(tag_data);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = WaitUntilNfcIsReady();
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = StartPolling(tag_data);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = GetAmiiboData(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
SetNonBlocking();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool NfcProtocol::HasAmiibo() {
|
||||||
|
DriverResult result{DriverResult::Success};
|
||||||
|
TagFoundData tag_data{};
|
||||||
|
SetBlocking();
|
||||||
|
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = StartPolling(tag_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
SetNonBlocking();
|
||||||
|
return result == DriverResult::Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::WaitUntilNfcIsReady() {
|
||||||
|
constexpr std::size_t timeout_limit = 10;
|
||||||
|
std::vector<u8> output;
|
||||||
|
std::size_t tries = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
auto result = SendStartWaitingRecieveRequest(output);
|
||||||
|
|
||||||
|
if (result != DriverResult::Success) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
if (tries++ > timeout_limit) {
|
||||||
|
return DriverResult::Timeout;
|
||||||
|
}
|
||||||
|
} while (output[49] != 0x2a || (output[51] << 8) + output[50] != 0x0500 || output[55] != 0x31 ||
|
||||||
|
output[56] != 0x00);
|
||||||
|
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::StartPolling(TagFoundData& data) {
|
||||||
|
LOG_DEBUG(Input, "Start Polling for tag");
|
||||||
|
constexpr std::size_t timeout_limit = 7;
|
||||||
|
std::vector<u8> output;
|
||||||
|
std::size_t tries = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
const auto result = SendStartPollingRequest(output);
|
||||||
|
if (result != DriverResult::Success) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
if (tries++ > timeout_limit) {
|
||||||
|
return DriverResult::Timeout;
|
||||||
|
}
|
||||||
|
} while (output[49] != 0x2a || (output[51] << 8) + output[50] != 0x0500 || output[56] != 0x09);
|
||||||
|
|
||||||
|
data.type = output[62];
|
||||||
|
data.uuid.resize(output[64]);
|
||||||
|
memcpy(data.uuid.data(), output.data() + 65, data.uuid.size());
|
||||||
|
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::ReadTag(const TagFoundData& data) {
|
||||||
|
constexpr std::size_t timeout_limit = 10;
|
||||||
|
std::vector<u8> output;
|
||||||
|
std::size_t tries = 0;
|
||||||
|
|
||||||
|
std::string uuid_string = "";
|
||||||
|
for (auto& content : data.uuid) {
|
||||||
|
uuid_string += " " + fmt::format("{:02x}", content);
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_INFO(Input, "Tag detected, type={}, uuid={}", data.type, uuid_string);
|
||||||
|
|
||||||
|
tries = 0;
|
||||||
|
std::size_t ntag_pages = 0;
|
||||||
|
// Read Tag data
|
||||||
|
loop1:
|
||||||
|
while (true) {
|
||||||
|
auto result = SendReadAmiiboRequest(output, ntag_pages);
|
||||||
|
|
||||||
|
int attempt = 0;
|
||||||
|
while (1) {
|
||||||
|
if (attempt != 0) {
|
||||||
|
result = GetMCUDataResponse(ReportMode::NFC_IR_MODE_60HZ, output);
|
||||||
|
}
|
||||||
|
if ((output[49] == 0x3a || output[49] == 0x2a) && output[56] == 0x07) {
|
||||||
|
return DriverResult::ErrorReadingData;
|
||||||
|
}
|
||||||
|
if (output[49] == 0x3a && output[51] == 0x07 && output[52] == 0x01) {
|
||||||
|
if (data.type != 2) {
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
switch (output[74]) {
|
||||||
|
case 0:
|
||||||
|
ntag_pages = 135;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
ntag_pages = 45;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
ntag_pages = 231;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return DriverResult::ErrorReadingData;
|
||||||
|
}
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
if (output[49] == 0x2a && output[56] == 0x04) {
|
||||||
|
// finished
|
||||||
|
SendStopPollingRequest(output);
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
if (output[49] == 0x2a) {
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
if (attempt++ > 6) {
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result != DriverResult::Success) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
if (tries++ > timeout_limit) {
|
||||||
|
return DriverResult::Timeout;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::GetAmiiboData(std::vector<u8>& ntag_data) {
|
||||||
|
constexpr std::size_t timeout_limit = 10;
|
||||||
|
std::vector<u8> output;
|
||||||
|
std::size_t tries = 0;
|
||||||
|
|
||||||
|
std::size_t ntag_pages = 135;
|
||||||
|
std::size_t ntag_buffer_pos = 0;
|
||||||
|
// Read Tag data
|
||||||
|
loop1:
|
||||||
|
while (true) {
|
||||||
|
auto result = SendReadAmiiboRequest(output, ntag_pages);
|
||||||
|
|
||||||
|
int attempt = 0;
|
||||||
|
while (1) {
|
||||||
|
if (attempt != 0) {
|
||||||
|
result = GetMCUDataResponse(ReportMode::NFC_IR_MODE_60HZ, output);
|
||||||
|
}
|
||||||
|
if ((output[49] == 0x3a || output[49] == 0x2a) && output[56] == 0x07) {
|
||||||
|
return DriverResult::ErrorReadingData;
|
||||||
|
}
|
||||||
|
if (output[49] == 0x3a && output[51] == 0x07) {
|
||||||
|
std::size_t payload_size = (output[54] << 8 | output[55]) & 0x7FF;
|
||||||
|
if (output[52] == 0x01) {
|
||||||
|
memcpy(ntag_data.data() + ntag_buffer_pos, output.data() + 116,
|
||||||
|
payload_size - 60);
|
||||||
|
ntag_buffer_pos += payload_size - 60;
|
||||||
|
} else {
|
||||||
|
memcpy(ntag_data.data() + ntag_buffer_pos, output.data() + 56, payload_size);
|
||||||
|
}
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
if (output[49] == 0x2a && output[56] == 0x04) {
|
||||||
|
LOG_INFO(Input, "Finished reading amiibo");
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
if (output[49] == 0x2a) {
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
if (attempt++ > 4) {
|
||||||
|
goto loop1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result != DriverResult::Success) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
if (tries++ > timeout_limit) {
|
||||||
|
return DriverResult::Timeout;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::SendStartPollingRequest(std::vector<u8>& output) {
|
||||||
|
NFCRequestState request{
|
||||||
|
.sub_command = MCUSubCommand::ReadDeviceMode,
|
||||||
|
.command_argument = NFCReadCommand::StartPolling,
|
||||||
|
.packet_id = 0x0,
|
||||||
|
.packet_flag = MCUPacketFlag::LastCommandPacket,
|
||||||
|
.data_length = sizeof(NFCPollingCommandData),
|
||||||
|
.nfc_polling =
|
||||||
|
{
|
||||||
|
.enable_mifare = 0x01,
|
||||||
|
.unknown_1 = 0x00,
|
||||||
|
.unknown_2 = 0x00,
|
||||||
|
.unknown_3 = 0x2c,
|
||||||
|
.unknown_4 = 0x01,
|
||||||
|
},
|
||||||
|
.crc = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<u8> request_data(sizeof(NFCRequestState));
|
||||||
|
memcpy(request_data.data(), &request, sizeof(NFCRequestState));
|
||||||
|
request_data[37] = CalculateMCU_CRC8(request_data.data() + 1, 36);
|
||||||
|
return SendMCUData(ReportMode::NFC_IR_MODE_60HZ, SubCommand::STATE, request_data, output);
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::SendStopPollingRequest(std::vector<u8>& output) {
|
||||||
|
NFCRequestState request{
|
||||||
|
.sub_command = MCUSubCommand::ReadDeviceMode,
|
||||||
|
.command_argument = NFCReadCommand::StopPolling,
|
||||||
|
.packet_id = 0x0,
|
||||||
|
.packet_flag = MCUPacketFlag::LastCommandPacket,
|
||||||
|
.data_length = 0,
|
||||||
|
.raw_data = {},
|
||||||
|
.crc = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<u8> request_data(sizeof(NFCRequestState));
|
||||||
|
memcpy(request_data.data(), &request, sizeof(NFCRequestState));
|
||||||
|
request_data[37] = CalculateMCU_CRC8(request_data.data() + 1, 36);
|
||||||
|
return SendMCUData(ReportMode::NFC_IR_MODE_60HZ, SubCommand::STATE, request_data, output);
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::SendStartWaitingRecieveRequest(std::vector<u8>& output) {
|
||||||
|
NFCRequestState request{
|
||||||
|
.sub_command = MCUSubCommand::ReadDeviceMode,
|
||||||
|
.command_argument = NFCReadCommand::StartWaitingRecieve,
|
||||||
|
.packet_id = 0x0,
|
||||||
|
.packet_flag = MCUPacketFlag::LastCommandPacket,
|
||||||
|
.data_length = 0,
|
||||||
|
.raw_data = {},
|
||||||
|
.crc = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<u8> request_data(sizeof(NFCRequestState));
|
||||||
|
memcpy(request_data.data(), &request, sizeof(NFCRequestState));
|
||||||
|
request_data[37] = CalculateMCU_CRC8(request_data.data() + 1, 36);
|
||||||
|
return SendMCUData(ReportMode::NFC_IR_MODE_60HZ, SubCommand::STATE, request_data, output);
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::SendReadAmiiboRequest(std::vector<u8>& output, std::size_t ntag_pages) {
|
||||||
|
NFCRequestState request{
|
||||||
|
.sub_command = MCUSubCommand::ReadDeviceMode,
|
||||||
|
.command_argument = NFCReadCommand::Ntag,
|
||||||
|
.packet_id = 0x0,
|
||||||
|
.packet_flag = MCUPacketFlag::LastCommandPacket,
|
||||||
|
.data_length = sizeof(NFCReadCommandData),
|
||||||
|
.nfc_read =
|
||||||
|
{
|
||||||
|
.unknown = 0xd0,
|
||||||
|
.uuid_length = 0x07,
|
||||||
|
.unknown_2 = 0x00,
|
||||||
|
.uid = {},
|
||||||
|
.tag_type = NFCTagType::AllTags,
|
||||||
|
.read_block = GetReadBlockCommand(ntag_pages),
|
||||||
|
},
|
||||||
|
.crc = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<u8> request_data(sizeof(NFCRequestState));
|
||||||
|
memcpy(request_data.data(), &request, sizeof(NFCRequestState));
|
||||||
|
request_data[37] = CalculateMCU_CRC8(request_data.data() + 1, 36);
|
||||||
|
return SendMCUData(ReportMode::NFC_IR_MODE_60HZ, SubCommand::STATE, request_data, output);
|
||||||
|
}
|
||||||
|
|
||||||
|
NFCReadBlockCommand NfcProtocol::GetReadBlockCommand(std::size_t pages) const {
|
||||||
|
if (pages == 0) {
|
||||||
|
return {
|
||||||
|
.block_count = 1,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pages == 45) {
|
||||||
|
return {
|
||||||
|
.block_count = 1,
|
||||||
|
.blocks =
|
||||||
|
{
|
||||||
|
NFCReadBlock{0x00, 0x2C},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pages == 135) {
|
||||||
|
return {
|
||||||
|
.block_count = 3,
|
||||||
|
.blocks =
|
||||||
|
{
|
||||||
|
NFCReadBlock{0x00, 0x3b},
|
||||||
|
{0x3c, 0x77},
|
||||||
|
{0x78, 0x86},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pages == 231) {
|
||||||
|
return {
|
||||||
|
.block_count = 4,
|
||||||
|
.blocks =
|
||||||
|
{
|
||||||
|
NFCReadBlock{0x00, 0x3b},
|
||||||
|
{0x3c, 0x77},
|
||||||
|
{0x78, 0x83},
|
||||||
|
{0xb4, 0xe6},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
bool NfcProtocol::IsEnabled() {
|
||||||
|
return is_enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace InputCommon::Joycon
|
61
src/input_common/helpers/joycon_protocol/nfc.h
Normal file
61
src/input_common/helpers/joycon_protocol/nfc.h
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
// Based on dkms-hid-nintendo implementation, CTCaer joycon toolkit and dekuNukem reverse
|
||||||
|
// engineering https://github.com/nicman23/dkms-hid-nintendo/blob/master/src/hid-nintendo.c
|
||||||
|
// https://github.com/CTCaer/jc_toolkit
|
||||||
|
// https://github.com/dekuNukem/Nintendo_Switch_Reverse_Engineering
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "input_common/helpers/joycon_protocol/common_protocol.h"
|
||||||
|
#include "input_common/helpers/joycon_protocol/joycon_types.h"
|
||||||
|
|
||||||
|
namespace InputCommon::Joycon {
|
||||||
|
|
||||||
|
class NfcProtocol final : private JoyconCommonProtocol {
|
||||||
|
public:
|
||||||
|
NfcProtocol(std::shared_ptr<JoyconHandle> handle);
|
||||||
|
|
||||||
|
DriverResult EnableNfc();
|
||||||
|
|
||||||
|
DriverResult DisableNfc();
|
||||||
|
|
||||||
|
DriverResult StartNFCPollingMode();
|
||||||
|
|
||||||
|
DriverResult ScanAmiibo(std::vector<u8>& data);
|
||||||
|
|
||||||
|
bool HasAmiibo();
|
||||||
|
|
||||||
|
bool IsEnabled();
|
||||||
|
|
||||||
|
private:
|
||||||
|
struct TagFoundData {
|
||||||
|
u8 type;
|
||||||
|
std::vector<u8> uuid;
|
||||||
|
};
|
||||||
|
|
||||||
|
DriverResult WaitUntilNfcIsReady();
|
||||||
|
|
||||||
|
DriverResult StartPolling(TagFoundData& data);
|
||||||
|
|
||||||
|
DriverResult ReadTag(const TagFoundData& data);
|
||||||
|
|
||||||
|
DriverResult GetAmiiboData(std::vector<u8>& data);
|
||||||
|
|
||||||
|
DriverResult SendStartPollingRequest(std::vector<u8>& output);
|
||||||
|
|
||||||
|
DriverResult SendStopPollingRequest(std::vector<u8>& output);
|
||||||
|
|
||||||
|
DriverResult SendStartWaitingRecieveRequest(std::vector<u8>& output);
|
||||||
|
|
||||||
|
DriverResult SendReadAmiiboRequest(std::vector<u8>& output, std::size_t ntag_pages);
|
||||||
|
|
||||||
|
NFCReadBlockCommand GetReadBlockCommand(std::size_t pages) const;
|
||||||
|
|
||||||
|
bool is_enabled{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace InputCommon::Joycon
|
@ -74,6 +74,10 @@ void JoyconPoller::UpdateColor(const Color& color) {
|
|||||||
callbacks.on_color_data(color);
|
callbacks.on_color_data(color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void JoyconPoller::updateAmiibo(const std::vector<u8>& amiibo_data) {
|
||||||
|
callbacks.on_amiibo_data(amiibo_data);
|
||||||
|
}
|
||||||
|
|
||||||
void JoyconPoller::UpdateRing(s16 value, const RingStatus& ring_status) {
|
void JoyconPoller::UpdateRing(s16 value, const RingStatus& ring_status) {
|
||||||
float normalized_value = static_cast<float>(value - ring_status.default_value);
|
float normalized_value = static_cast<float>(value - ring_status.default_value);
|
||||||
if (normalized_value > 0) {
|
if (normalized_value > 0) {
|
||||||
|
@ -36,6 +36,7 @@ public:
|
|||||||
|
|
||||||
void UpdateColor(const Color& color);
|
void UpdateColor(const Color& color);
|
||||||
void UpdateRing(s16 value, const RingStatus& ring_status);
|
void UpdateRing(s16 value, const RingStatus& ring_status);
|
||||||
|
void updateAmiibo(const std::vector<u8>& amiibo_data);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void UpdateActiveLeftPadInput(const InputReportActive& input,
|
void UpdateActiveLeftPadInput(const InputReportActive& input,
|
||||||
|
Loading…
Reference in New Issue
Block a user