// 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 #include "input_common/helpers/joycon_protocol/common_protocol.h" #include "input_common/helpers/joycon_protocol/joycon_types.h" namespace Common::Input { enum class DriverResult; } namespace InputCommon::Joycon { class NfcProtocol final : private JoyconCommonProtocol { public: explicit NfcProtocol(std::shared_ptr handle); Common::Input::DriverResult EnableNfc(); Common::Input::DriverResult DisableNfc(); Common::Input::DriverResult StartNFCPollingMode(); Common::Input::DriverResult StopNFCPollingMode(); Common::Input::DriverResult GetTagInfo(Joycon::TagInfo& tag_info); Common::Input::DriverResult ReadAmiibo(std::vector& data); Common::Input::DriverResult WriteAmiibo(std::span data); Common::Input::DriverResult ReadMifare(std::span read_request, std::span out_data); Common::Input::DriverResult WriteMifare(std::span write_request); bool HasAmiibo(); bool IsEnabled() const; bool IsPolling() const; private: // Number of times the function will be delayed until it outputs valid data static constexpr std::size_t AMIIBO_UPDATE_DELAY = 15; struct TagFoundData { u8 type; u8 uuid_size; TagUUID uuid; }; Common::Input::DriverResult WaitUntilNfcIs(NFCStatus status); Common::Input::DriverResult IsTagInRange(TagFoundData& data, std::size_t timeout_limit = 1); Common::Input::DriverResult GetAmiiboData(std::vector& data); Common::Input::DriverResult WriteAmiiboData(const TagUUID& tag_uuid, std::span data); Common::Input::DriverResult GetMifareData(const MifareUUID& tag_uuid, std::span read_request, std::span out_data); Common::Input::DriverResult WriteMifareData(const MifareUUID& tag_uuid, std::span write_request); Common::Input::DriverResult SendStartPollingRequest(MCUCommandResponse& output, bool is_second_attempt = false); Common::Input::DriverResult SendStopPollingRequest(MCUCommandResponse& output); Common::Input::DriverResult SendNextPackageRequest(MCUCommandResponse& output, u8 packet_id); Common::Input::DriverResult SendReadAmiiboRequest(MCUCommandResponse& output, NFCPages ntag_pages); Common::Input::DriverResult SendWriteAmiiboRequest(MCUCommandResponse& output, const TagUUID& tag_uuid); Common::Input::DriverResult SendWriteDataAmiiboRequest(MCUCommandResponse& output, u8 block_id, bool is_last_packet, std::span data); Common::Input::DriverResult SendReadDataMifareRequest(MCUCommandResponse& output, u8 block_id, bool is_last_packet, std::span data); std::vector SerializeWritePackage(const NFCWritePackage& package) const; std::vector SerializeMifareReadPackage(const MifareReadPackage& package) const; std::vector SerializeMifareWritePackage(const MifareWritePackage& package) const; NFCWritePackage MakeAmiiboWritePackage(const TagUUID& tag_uuid, std::span data) const; NFCDataChunk MakeAmiiboChunk(u8 page, u8 size, std::span data) const; MifareReadPackage MakeMifareReadPackage(const MifareUUID& tag_uuid, std::span read_request) const; MifareWritePackage MakeMifareWritePackage(const MifareUUID& tag_uuid, std::span read_request) const; NFCReadBlockCommand GetReadBlockCommand(NFCPages pages) const; TagUUID GetTagUUID(std::span data) const; bool is_enabled{}; bool is_polling{}; std::size_t update_counter{}; }; } // namespace InputCommon::Joycon