Merge pull request #10344 from german77/pro-amiibo
input_common: Fix pro controller amiibo support
This commit is contained in:
commit
1e398e6c36
6 changed files with 79 additions and 112 deletions
|
@ -195,8 +195,8 @@ void Joycons::RegisterNewDevice(SDL_hid_device_info* device_info) {
|
||||||
OnMotionUpdate(port, type, id, value);
|
OnMotionUpdate(port, type, id, value);
|
||||||
}},
|
}},
|
||||||
.on_ring_data = {[this](f32 ring_data) { OnRingConUpdate(ring_data); }},
|
.on_ring_data = {[this](f32 ring_data) { OnRingConUpdate(ring_data); }},
|
||||||
.on_amiibo_data = {[this, port](const std::vector<u8>& amiibo_data) {
|
.on_amiibo_data = {[this, port, type](const std::vector<u8>& amiibo_data) {
|
||||||
OnAmiiboUpdate(port, amiibo_data);
|
OnAmiiboUpdate(port, type, amiibo_data);
|
||||||
}},
|
}},
|
||||||
.on_camera_data = {[this, port](const std::vector<u8>& camera_data,
|
.on_camera_data = {[this, port](const std::vector<u8>& camera_data,
|
||||||
Joycon::IrsResolution format) {
|
Joycon::IrsResolution format) {
|
||||||
|
@ -398,8 +398,9 @@ void Joycons::OnRingConUpdate(f32 ring_data) {
|
||||||
SetAxis(identifier, 100, ring_data);
|
SetAxis(identifier, 100, ring_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Joycons::OnAmiiboUpdate(std::size_t port, const std::vector<u8>& amiibo_data) {
|
void Joycons::OnAmiiboUpdate(std::size_t port, Joycon::ControllerType type,
|
||||||
const auto identifier = GetIdentifier(port, Joycon::ControllerType::Right);
|
const std::vector<u8>& amiibo_data) {
|
||||||
|
const auto identifier = GetIdentifier(port, type);
|
||||||
const auto nfc_state = amiibo_data.empty() ? Common::Input::NfcState::AmiiboRemoved
|
const auto nfc_state = amiibo_data.empty() ? Common::Input::NfcState::AmiiboRemoved
|
||||||
: Common::Input::NfcState::NewAmiibo;
|
: Common::Input::NfcState::NewAmiibo;
|
||||||
SetNfc(identifier, {nfc_state, amiibo_data});
|
SetNfc(identifier, {nfc_state, amiibo_data});
|
||||||
|
|
|
@ -81,7 +81,8 @@ private:
|
||||||
void OnMotionUpdate(std::size_t port, Joycon::ControllerType type, int id,
|
void OnMotionUpdate(std::size_t port, Joycon::ControllerType type, int id,
|
||||||
const Joycon::MotionData& value);
|
const Joycon::MotionData& value);
|
||||||
void OnRingConUpdate(f32 ring_data);
|
void OnRingConUpdate(f32 ring_data);
|
||||||
void OnAmiiboUpdate(std::size_t port, const std::vector<u8>& amiibo_data);
|
void OnAmiiboUpdate(std::size_t port, Joycon::ControllerType type,
|
||||||
|
const std::vector<u8>& amiibo_data);
|
||||||
void OnCameraUpdate(std::size_t port, const std::vector<u8>& camera_data,
|
void OnCameraUpdate(std::size_t port, const std::vector<u8>& camera_data,
|
||||||
Joycon::IrsResolution format);
|
Joycon::IrsResolution format);
|
||||||
|
|
||||||
|
|
|
@ -265,7 +265,7 @@ DriverResult JoyconCommonProtocol::SendMCUData(ReportMode report_mode, MCUSubCom
|
||||||
|
|
||||||
DriverResult JoyconCommonProtocol::WaitSetMCUMode(ReportMode report_mode, MCUMode mode) {
|
DriverResult JoyconCommonProtocol::WaitSetMCUMode(ReportMode report_mode, MCUMode mode) {
|
||||||
MCUCommandResponse output{};
|
MCUCommandResponse output{};
|
||||||
constexpr std::size_t MaxTries{8};
|
constexpr std::size_t MaxTries{16};
|
||||||
std::size_t tries{};
|
std::size_t tries{};
|
||||||
|
|
||||||
do {
|
do {
|
||||||
|
|
|
@ -577,8 +577,8 @@ static_assert(sizeof(NFCPollingCommandData) == 0x05, "NFCPollingCommandData is a
|
||||||
|
|
||||||
struct NFCRequestState {
|
struct NFCRequestState {
|
||||||
NFCReadCommand command_argument;
|
NFCReadCommand command_argument;
|
||||||
u8 packet_id;
|
|
||||||
INSERT_PADDING_BYTES(0x1);
|
INSERT_PADDING_BYTES(0x1);
|
||||||
|
u8 packet_id;
|
||||||
MCUPacketFlag packet_flag;
|
MCUPacketFlag packet_flag;
|
||||||
u8 data_length;
|
u8 data_length;
|
||||||
union {
|
union {
|
||||||
|
|
|
@ -64,6 +64,20 @@ DriverResult NfcProtocol::StartNFCPollingMode() {
|
||||||
if (result == DriverResult::Success) {
|
if (result == DriverResult::Success) {
|
||||||
result = WaitUntilNfcIsReady();
|
result = WaitUntilNfcIsReady();
|
||||||
}
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
MCUCommandResponse output{};
|
||||||
|
result = SendStopPollingRequest(output);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = WaitUntilNfcIsReady();
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
MCUCommandResponse output{};
|
||||||
|
result = SendStartPollingRequest(output);
|
||||||
|
}
|
||||||
|
if (result == DriverResult::Success) {
|
||||||
|
result = WaitUntilNfcIsPolling();
|
||||||
|
}
|
||||||
if (result == DriverResult::Success) {
|
if (result == DriverResult::Success) {
|
||||||
is_enabled = true;
|
is_enabled = true;
|
||||||
}
|
}
|
||||||
|
@ -77,24 +91,21 @@ DriverResult NfcProtocol::ScanAmiibo(std::vector<u8>& data) {
|
||||||
}
|
}
|
||||||
update_counter = 0;
|
update_counter = 0;
|
||||||
|
|
||||||
LOG_DEBUG(Input, "Start NFC pooling Mode");
|
LOG_DEBUG(Input, "Scan for amiibos");
|
||||||
ScopedSetBlocking sb(this);
|
ScopedSetBlocking sb(this);
|
||||||
DriverResult result{DriverResult::Success};
|
DriverResult result{DriverResult::Success};
|
||||||
TagFoundData tag_data{};
|
TagFoundData tag_data{};
|
||||||
|
|
||||||
if (result == DriverResult::Success) {
|
if (result == DriverResult::Success) {
|
||||||
result = StartPolling(tag_data);
|
result = IsTagInRange(tag_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (result == DriverResult::Success) {
|
if (result == DriverResult::Success) {
|
||||||
result = ReadTag(tag_data);
|
std::string uuid_string;
|
||||||
|
for (auto& content : tag_data.uuid) {
|
||||||
|
uuid_string += fmt::format(" {:02x}", content);
|
||||||
}
|
}
|
||||||
if (result == DriverResult::Success) {
|
LOG_INFO(Input, "Tag detected, type={}, uuid={}", tag_data.type, uuid_string);
|
||||||
result = WaitUntilNfcIsReady();
|
|
||||||
}
|
|
||||||
if (result == DriverResult::Success) {
|
|
||||||
result = StartPolling(tag_data, 7);
|
|
||||||
}
|
|
||||||
if (result == DriverResult::Success) {
|
|
||||||
result = GetAmiiboData(data);
|
result = GetAmiiboData(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,12 +113,17 @@ DriverResult NfcProtocol::ScanAmiibo(std::vector<u8>& data) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool NfcProtocol::HasAmiibo() {
|
bool NfcProtocol::HasAmiibo() {
|
||||||
|
if (update_counter++ < AMIIBO_UPDATE_DELAY) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
update_counter = 0;
|
||||||
|
|
||||||
ScopedSetBlocking sb(this);
|
ScopedSetBlocking sb(this);
|
||||||
DriverResult result{DriverResult::Success};
|
DriverResult result{DriverResult::Success};
|
||||||
TagFoundData tag_data{};
|
TagFoundData tag_data{};
|
||||||
|
|
||||||
if (result == DriverResult::Success) {
|
if (result == DriverResult::Success) {
|
||||||
result = StartPolling(tag_data);
|
result = IsTagInRange(tag_data, 7);
|
||||||
}
|
}
|
||||||
|
|
||||||
return result == DriverResult::Success;
|
return result == DriverResult::Success;
|
||||||
|
@ -119,7 +135,7 @@ DriverResult NfcProtocol::WaitUntilNfcIsReady() {
|
||||||
std::size_t tries = 0;
|
std::size_t tries = 0;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
auto result = SendStartWaitingRecieveRequest(output);
|
auto result = SendNextPackageRequest(output, {});
|
||||||
|
|
||||||
if (result != DriverResult::Success) {
|
if (result != DriverResult::Success) {
|
||||||
return result;
|
return result;
|
||||||
|
@ -134,13 +150,14 @@ DriverResult NfcProtocol::WaitUntilNfcIsReady() {
|
||||||
return DriverResult::Success;
|
return DriverResult::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
DriverResult NfcProtocol::StartPolling(TagFoundData& data, std::size_t timeout_limit) {
|
DriverResult NfcProtocol::WaitUntilNfcIsPolling() {
|
||||||
LOG_DEBUG(Input, "Start Polling for tag");
|
constexpr std::size_t timeout_limit = 10;
|
||||||
MCUCommandResponse output{};
|
MCUCommandResponse output{};
|
||||||
std::size_t tries = 0;
|
std::size_t tries = 0;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
const auto result = SendStartPollingRequest(output);
|
auto result = SendNextPackageRequest(output, {});
|
||||||
|
|
||||||
if (result != DriverResult::Success) {
|
if (result != DriverResult::Success) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -149,7 +166,26 @@ DriverResult NfcProtocol::StartPolling(TagFoundData& data, std::size_t timeout_l
|
||||||
}
|
}
|
||||||
} while (output.mcu_report != MCUReport::NFCState ||
|
} while (output.mcu_report != MCUReport::NFCState ||
|
||||||
(output.mcu_data[1] << 8) + output.mcu_data[0] != 0x0500 ||
|
(output.mcu_data[1] << 8) + output.mcu_data[0] != 0x0500 ||
|
||||||
output.mcu_data[6] != 0x09);
|
output.mcu_data[5] != 0x31 || output.mcu_data[6] != 0x01);
|
||||||
|
|
||||||
|
return DriverResult::Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
DriverResult NfcProtocol::IsTagInRange(TagFoundData& data, std::size_t timeout_limit) {
|
||||||
|
MCUCommandResponse output{};
|
||||||
|
std::size_t tries = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
const auto result = SendNextPackageRequest(output, {});
|
||||||
|
if (result != DriverResult::Success) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
if (tries++ > timeout_limit) {
|
||||||
|
return DriverResult::Timeout;
|
||||||
|
}
|
||||||
|
} while (output.mcu_report != MCUReport::NFCState ||
|
||||||
|
(output.mcu_data[1] << 8) + output.mcu_data[0] != 0x0500 ||
|
||||||
|
(output.mcu_data[6] != 0x09 && output.mcu_data[6] != 0x04));
|
||||||
|
|
||||||
data.type = output.mcu_data[12];
|
data.type = output.mcu_data[12];
|
||||||
data.uuid.resize(output.mcu_data[14]);
|
data.uuid.resize(output.mcu_data[14]);
|
||||||
|
@ -158,85 +194,22 @@ DriverResult NfcProtocol::StartPolling(TagFoundData& data, std::size_t timeout_l
|
||||||
return DriverResult::Success;
|
return DriverResult::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
DriverResult NfcProtocol::ReadTag(const TagFoundData& data) {
|
DriverResult NfcProtocol::GetAmiiboData(std::vector<u8>& ntag_data) {
|
||||||
constexpr std::size_t timeout_limit = 10;
|
constexpr std::size_t timeout_limit = 60;
|
||||||
MCUCommandResponse output{};
|
MCUCommandResponse output{};
|
||||||
std::size_t tries = 0;
|
std::size_t tries = 0;
|
||||||
|
|
||||||
std::string uuid_string;
|
u8 package_index = 0;
|
||||||
for (auto& content : data.uuid) {
|
std::size_t ntag_buffer_pos = 0;
|
||||||
uuid_string += fmt::format(" {:02x}", content);
|
auto result = SendReadAmiiboRequest(output, NFCPages::Block135);
|
||||||
}
|
|
||||||
|
|
||||||
LOG_INFO(Input, "Tag detected, type={}, uuid={}", data.type, uuid_string);
|
|
||||||
|
|
||||||
tries = 0;
|
|
||||||
NFCPages ntag_pages = NFCPages::Block0;
|
|
||||||
// Read Tag data
|
|
||||||
while (true) {
|
|
||||||
auto result = SendReadAmiiboRequest(output, ntag_pages);
|
|
||||||
const auto nfc_status = static_cast<NFCStatus>(output.mcu_data[6]);
|
|
||||||
|
|
||||||
if (result != DriverResult::Success) {
|
if (result != DriverResult::Success) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((output.mcu_report == MCUReport::NFCReadData ||
|
|
||||||
output.mcu_report == MCUReport::NFCState) &&
|
|
||||||
nfc_status == NFCStatus::TagLost) {
|
|
||||||
return DriverResult::ErrorReadingData;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (output.mcu_report == MCUReport::NFCReadData && output.mcu_data[1] == 0x07 &&
|
|
||||||
output.mcu_data[2] == 0x01) {
|
|
||||||
if (data.type != 2) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
switch (output.mcu_data[24]) {
|
|
||||||
case 0:
|
|
||||||
ntag_pages = NFCPages::Block135;
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
ntag_pages = NFCPages::Block45;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
ntag_pages = NFCPages::Block231;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return DriverResult::ErrorReadingData;
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (output.mcu_report == MCUReport::NFCState && nfc_status == NFCStatus::LastPackage) {
|
|
||||||
// finished
|
|
||||||
SendStopPollingRequest(output);
|
|
||||||
return DriverResult::Success;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ignore other state reports
|
|
||||||
if (output.mcu_report == MCUReport::NFCState) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
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;
|
|
||||||
MCUCommandResponse output{};
|
|
||||||
std::size_t tries = 0;
|
|
||||||
|
|
||||||
NFCPages ntag_pages = NFCPages::Block135;
|
|
||||||
std::size_t ntag_buffer_pos = 0;
|
|
||||||
// Read Tag data
|
// Read Tag data
|
||||||
while (true) {
|
while (tries++ < timeout_limit) {
|
||||||
auto result = SendReadAmiiboRequest(output, ntag_pages);
|
result = SendNextPackageRequest(output, package_index);
|
||||||
const auto nfc_status = static_cast<NFCStatus>(output.mcu_data[6]);
|
const auto nfc_status = static_cast<NFCStatus>(output.mcu_data[6]);
|
||||||
|
|
||||||
if (result != DriverResult::Success) {
|
if (result != DriverResult::Success) {
|
||||||
|
@ -259,6 +232,7 @@ DriverResult NfcProtocol::GetAmiiboData(std::vector<u8>& ntag_data) {
|
||||||
memcpy(ntag_data.data() + ntag_buffer_pos, output.mcu_data.data() + 6,
|
memcpy(ntag_data.data() + ntag_buffer_pos, output.mcu_data.data() + 6,
|
||||||
payload_size);
|
payload_size);
|
||||||
}
|
}
|
||||||
|
package_index++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -266,19 +240,10 @@ DriverResult NfcProtocol::GetAmiiboData(std::vector<u8>& ntag_data) {
|
||||||
LOG_INFO(Input, "Finished reading amiibo");
|
LOG_INFO(Input, "Finished reading amiibo");
|
||||||
return DriverResult::Success;
|
return DriverResult::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ignore other state reports
|
|
||||||
if (output.mcu_report == MCUReport::NFCState) {
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tries++ > timeout_limit) {
|
|
||||||
return DriverResult::Timeout;
|
return DriverResult::Timeout;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return DriverResult::Success;
|
|
||||||
}
|
|
||||||
|
|
||||||
DriverResult NfcProtocol::SendStartPollingRequest(MCUCommandResponse& output) {
|
DriverResult NfcProtocol::SendStartPollingRequest(MCUCommandResponse& output) {
|
||||||
NFCRequestState request{
|
NFCRequestState request{
|
||||||
|
@ -321,10 +286,10 @@ DriverResult NfcProtocol::SendStopPollingRequest(MCUCommandResponse& output) {
|
||||||
output);
|
output);
|
||||||
}
|
}
|
||||||
|
|
||||||
DriverResult NfcProtocol::SendStartWaitingRecieveRequest(MCUCommandResponse& output) {
|
DriverResult NfcProtocol::SendNextPackageRequest(MCUCommandResponse& output, u8 packet_id) {
|
||||||
NFCRequestState request{
|
NFCRequestState request{
|
||||||
.command_argument = NFCReadCommand::StartWaitingRecieve,
|
.command_argument = NFCReadCommand::StartWaitingRecieve,
|
||||||
.packet_id = 0x0,
|
.packet_id = packet_id,
|
||||||
.packet_flag = MCUPacketFlag::LastCommandPacket,
|
.packet_flag = MCUPacketFlag::LastCommandPacket,
|
||||||
.data_length = 0,
|
.data_length = 0,
|
||||||
.raw_data = {},
|
.raw_data = {},
|
||||||
|
|
|
@ -42,9 +42,9 @@ private:
|
||||||
|
|
||||||
DriverResult WaitUntilNfcIsReady();
|
DriverResult WaitUntilNfcIsReady();
|
||||||
|
|
||||||
DriverResult StartPolling(TagFoundData& data, std::size_t timeout_limit = 1);
|
DriverResult WaitUntilNfcIsPolling();
|
||||||
|
|
||||||
DriverResult ReadTag(const TagFoundData& data);
|
DriverResult IsTagInRange(TagFoundData& data, std::size_t timeout_limit = 1);
|
||||||
|
|
||||||
DriverResult GetAmiiboData(std::vector<u8>& data);
|
DriverResult GetAmiiboData(std::vector<u8>& data);
|
||||||
|
|
||||||
|
@ -52,7 +52,7 @@ private:
|
||||||
|
|
||||||
DriverResult SendStopPollingRequest(MCUCommandResponse& output);
|
DriverResult SendStopPollingRequest(MCUCommandResponse& output);
|
||||||
|
|
||||||
DriverResult SendStartWaitingRecieveRequest(MCUCommandResponse& output);
|
DriverResult SendNextPackageRequest(MCUCommandResponse& output, u8 packet_id);
|
||||||
|
|
||||||
DriverResult SendReadAmiiboRequest(MCUCommandResponse& output, NFCPages ntag_pages);
|
DriverResult SendReadAmiiboRequest(MCUCommandResponse& output, NFCPages ntag_pages);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue