M5Unit-RFID 0.0.2 git rev:1dc8f67
Loading...
Searching...
No Matches
unit_MFRC522.hpp
Go to the documentation of this file.
1/*
2 * SPDX-FileCopyrightText: 2024 M5Stack Technology CO LTD
3 *
4 * SPDX-License-Identifier: MIT
5 */
10#ifndef M5_UNIT_RFID_UNIT_MFRC522_HPP
11#define M5_UNIT_RFID_UNIT_MFRC522_HPP
12
13#include <M5UnitComponent.hpp>
14// M5Unit-NFC
15#include <nfc/nfc.hpp>
16#include <nfc/a/nfca.hpp>
17
18namespace m5 {
19namespace unit {
20
21namespace nfc {
22struct AdapterMFRC522; // For layer
23}
24
25namespace mfrc522 {
26
27constexpr uint16_t MAX_FIFO_DEPTH{64};
28
33enum class Command : uint8_t {
34 Idle,
35 Mem,
37 CalcCRC,
38 Transmit,
39 NoCmdChange = 0x07,
40 Receive,
41 Transceive = 0x0C,
43 MFAuthent = 0x0E,
44 SoftReset,
45};
46
51enum class ReceiverGain : uint8_t {
52 dB18,
53 dB23,
54 // 2 = dB18 (duplicated),
55 // 3 = dB23 (duplicated)
56 dB33 = 0x04,
57 dB38,
58 dB43,
59 dB48,
60};
61
66enum class Error : uint8_t {
69 ARGUMENT = 0x80,
71 REGISTER,
72 TIMEOUT,
74 CRC,
75 AUTH,
77 INTERNAL = 0xFE,
78 UNKNOWN = 0xFF
79};
80
81} // namespace mfrc522
82
89class UnitMFRC522 : public Component {
90 M5_UNIT_COMPONENT_HPP_BUILDER(UnitMFRC522, 0x28);
91
92public:
97 struct config_t {
99 m5::nfc::NFC mode{m5::nfc::NFC::A};
101 uint8_t mode_reg{0x3D};
103 bool enable_antenna{true};
105 mfrc522::ReceiverGain receiver_gain{mfrc522::ReceiverGain::dB48};
107 bool software_crc{false};
108 };
109
114 explicit UnitMFRC522(const uint8_t addr = DEFAULT_ADDRESS) : Component(addr)
115 {
116 auto ccfg = component_config();
117 ccfg.clock = 400 * 1000U;
118 component_config(ccfg);
119 }
123 virtual ~UnitMFRC522()
124 {
125 }
126
131 virtual bool begin() override;
136 virtual void update(const bool force = false) override;
137
140
142 {
143 return _cfg;
144 }
146 inline void config(const config_t& cfg)
147 {
148 _cfg = cfg;
149 }
151
156 inline m5::nfc::NFC NFCMode() const
157 {
158 return m5::nfc::NFC::A;
159 }
166 inline virtual bool configureNFCMode(const m5::nfc::NFC mode)
167 {
168 // Nop
169 return mode == m5::nfc::NFC::A;
170 }
171
177 bool softReset(const bool blocking = true);
178
184 inline bool selfTest()
185 {
186 bool ret = self_test();
187 return ret && begin();
188 }
189
192
197 bool readAntennaStatus(bool& status);
202 bool turnOnAntenna();
207 bool turnOffAntenna();
221
224
229 bool readGsN(uint8_t& value);
235 bool writeGsN(const uint8_t value);
241 bool readCWGsP(uint8_t& value);
247 bool writeCWGsP(const uint8_t value);
253 bool readModGsP(uint8_t& value);
259 bool writeModGsP(const uint8_t value);
261
264
269 bool readRxThreshold(uint8_t& value);
278 bool writeRxThreshold(const uint8_t value);
280
283
288 bool readTPrescaler(uint16_t& tprescale);
294 bool writeTPrescaler(const uint16_t tprescale);
296
299
306 bool calculateCRC(uint16_t& result, const uint8_t* buf, const uint8_t len);
314 bool calculateSoftwareCRC(uint16_t& result, const uint8_t* buf, const uint8_t len);
316
319
328 bool nfcaTransceive(uint8_t* rx, uint16_t& rx_len, const uint8_t* tx, const uint16_t tx_len,
329 const uint32_t timeout_ms);
330
336 inline bool request(uint16_t& atqa)
337 {
338 return request_wakeup(atqa, true);
339 }
345 inline bool wakeup(uint16_t& atqa)
346 {
347 return request_wakeup(atqa, false);
348 }
349
357 bool selectWithAnticollision(bool& completed, m5::nfc::a::PICC& picc, const uint8_t lv);
363 bool select(const m5::nfc::a::PICC& picc);
364
371 bool readBlock(uint8_t rx[16], const uint8_t block);
378 bool writeBlock(const uint8_t block, const uint8_t tx[16]);
383 bool hlt();
385
388
396 const m5::nfc::a::PICC& picc, const uint8_t block,
397 const m5::nfc::a::mifare::classic::Key& key = m5::nfc::a::mifare::classic::DEFAULT_KEY)
398 {
399 return mifare_classic_authenticate(m5::nfc::a::Command::AUTH_WITH_KEY_A, picc, block, key);
400 }
409 const m5::nfc::a::PICC& picc, const uint8_t block,
410 const m5::nfc::a::mifare::classic::Key& key = m5::nfc::a::mifare::classic::DEFAULT_KEY)
411 {
412 return mifare_classic_authenticate(m5::nfc::a::Command::AUTH_WITH_KEY_B, picc, block, key);
413 }
414
420
428 bool mifareClassicValueBlock(const m5::nfc::a::Command cmd, const uint8_t block, const uint32_t arg = 0);
430
431protected:
432 // register operation
433 bool modify_bit_register8(const uint8_t reg, const uint8_t set_mask, const uint8_t clear_mask);
434 bool set_bit_register8(const uint8_t reg, const uint8_t bits);
435 bool clear_bit_register8(const uint8_t reg, const uint8_t bits);
436 inline bool change_bit_register8(const uint8_t reg, const uint8_t bits, const uint8_t mask)
437 {
438 return modify_bit_register8(reg, mask & bits, mask);
439 }
440 bool read_register_with_align(const uint8_t reg, uint8_t* buf, const uint8_t len, const uint8_t align);
441 bool write_pcd_command(const mfrc522::Command cmd);
442
443 bool reset_baud_rates();
444 bool flush_fifo_buffer();
445 bool wait_comm_irq(const uint8_t irq, const uint32_t duration);
446 bool wait_div_irq(const uint8_t irq, const uint32_t duration);
447
448 // transceive
449 bool transmit_command(const mfrc522::Command cmd, const uint8_t* buf, const uint8_t len, const uint8_t txLast = 0,
450 const uint8_t rxAlign = 0);
451 bool transceive(uint8_t* rbuf, uint16_t& rlen, const uint8_t* buf, const uint16_t len, const uint32_t timeout_ms,
452 uint8_t& validBits, const uint8_t rxAlign = 0, const bool crc = false, uint8_t* err = nullptr);
453
454 bool picc_haltA();
455
456 // Read/Write
457 bool read_block(uint8_t* rbuf, uint8_t& rlen, const uint8_t addr);
458 bool write_block(const uint8_t block, const uint8_t* buf, const uint8_t len);
459 bool write_page(const uint8_t page, const uint8_t* buf, const uint8_t len);
460
461 // NFC-A
462 bool request_wakeup(uint16_t& atqa, const bool request);
463 bool anti_collision(const uint8_t cascadeLevel, uint8_t* buf);
464
465 // MIFARE classic
466 bool mifare_classic_authenticate(const m5::nfc::a::Command cmd, const m5::nfc::a::PICC& picc, const uint8_t block,
467 const m5::nfc::a::mifare::classic::Key& key);
468 bool mifare_classic_transceive(const m5::nfc::a::Command cmd, const uint8_t block, const uint32_t timeout_ms);
469 bool mifare_classic_transceive(const uint8_t* buf, const uint8_t len, const uint32_t timeout_ms,
470 const bool usingtimeout = false);
471
472 // crc
473 inline bool calculate_crc(uint16_t& result, const uint8_t* buf, const uint8_t len)
474 {
475 return (this->*(_cfg.software_crc ? &UnitMFRC522::calculateSoftwareCRC : &UnitMFRC522::calculateCRC))(result,
476 buf, len);
477 }
478
479 virtual bool self_test();
480
481protected:
482 config_t _cfg{};
483 uint16_t _tprescaler{};
484};
485
487namespace mfrc522 {
488namespace command {
489// Command and status
490constexpr uint8_t COMMAND_REG{0x01};
491constexpr uint8_t COM_IEN_REG{0x02};
492constexpr uint8_t DIV_IEN_REG{0x03};
493constexpr uint8_t COM_IRQ_REG{0x04};
494constexpr uint8_t DIV_IRQ_REG{0x05};
495constexpr uint8_t ERROR_REG{0x06};
496constexpr uint8_t STATUS2_REG{0x08};
497constexpr uint8_t FIFO_DATA_REG{0x09};
498constexpr uint8_t FIFO_LEVEL_REG{0x0A};
499constexpr uint8_t WATER_LEVEL_REG{0x0B};
500constexpr uint8_t CONTROL_REG{0x0C};
501constexpr uint8_t BIT_FRAMING_REG{0x0D};
502constexpr uint8_t COLL_REG{0x0E};
503
504// Communication
505constexpr uint8_t MODE_REG{0x11};
506constexpr uint8_t TX_MODE_REG{0x12};
507constexpr uint8_t RX_MODE_REG{0x13};
508constexpr uint8_t TX_CONTROL_REG{0x14};
509constexpr uint8_t TX_ASK_REG{0x15};
510constexpr uint8_t TX_SEL_REG{0x16};
511constexpr uint8_t RX_SEL_REG{0x17};
512constexpr uint8_t RX_THRESHOLD_REG{0x18};
513constexpr uint8_t DEMOD_REG{0x19};
514constexpr uint8_t MF_TX_REG{0x1C};
515constexpr uint8_t MF_RX_REG{0x1D};
516
517// Configuration
518constexpr uint8_t CRC_RESULT_REGH{0x21};
519constexpr uint8_t CRC_RESULT_REGL{0x22};
520constexpr uint8_t MOD_WIDTH_REG{0x24};
521constexpr uint8_t RFC_FG_REG{0x26};
522constexpr uint8_t GSN_REG{0x27};
523constexpr uint8_t CW_GSP_REG{0x28};
524constexpr uint8_t MOD_GSP_REG{0x29};
525constexpr uint8_t TMODE_REG{0x2A};
526constexpr uint8_t TPRESCALER_REG_L{0x2B};
527constexpr uint8_t TRELOAD_REG_H{0x2C};
528constexpr uint8_t TRELOAD_REG_L{0x2D};
529
530// Test
531constexpr uint8_t AUTO_TEST_REG{0x36};
532constexpr uint8_t VERSION_REG{0x37};
533
534} // namespace command
535} // namespace mfrc522
537
538} // namespace unit
539} // namespace m5
540#endif
Radio frequency identification unit.
Definition unit_MFRC522.hpp:89
config_t config()
Gets the configuration.
Definition unit_MFRC522.hpp:141
bool readAntennaStatus(bool &status)
Read the antenna status.
Definition unit_MFRC522.cpp:265
bool mifareClassicAuthenticateA(const m5::nfc::a::PICC &picc, const uint8_t block, const m5::nfc::a::mifare::classic::Key &key=m5::nfc::a::mifare::classic::DEFAULT_KEY)
Authentication using keyA of the specified block.
Definition unit_MFRC522.hpp:395
bool selectWithAnticollision(bool &completed, m5::nfc::a::PICC &picc, const uint8_t lv)
Select PICC with anti-collision.
Definition unit_MFRC522.cpp:508
bool turnOnAntenna()
Turn on the antenna.
Definition unit_MFRC522.cpp:276
virtual ~UnitMFRC522()
Destructor.
Definition unit_MFRC522.hpp:123
bool writeTPrescaler(const uint16_t tprescale)
Write the TPrescaler.
Definition unit_MFRC522.cpp:252
bool readReceiverGain(mfrc522::ReceiverGain &gain)
Gets the receiver gain.
Definition unit_MFRC522.cpp:302
virtual bool configureNFCMode(const m5::nfc::NFC mode)
Configure NFC mode.
Definition unit_MFRC522.hpp:166
bool writeModGsP(const uint8_t value)
Write the ModGsP register.
Definition unit_MFRC522.cpp:351
m5::nfc::NFC NFCMode() const
Gets the current operating mode.
Definition unit_MFRC522.hpp:156
bool select(const m5::nfc::a::PICC &picc)
Select specific PICC.
Definition unit_MFRC522.cpp:580
bool readBlock(uint8_t rx[16], const uint8_t block)
Read the 1 block / 4 page (16 bytes)
Definition unit_MFRC522.cpp:620
bool writeCWGsP(const uint8_t value)
Write the CWGsP register.
Definition unit_MFRC522.cpp:337
bool readRxThreshold(uint8_t &value)
Read the RxThreshold register.
Definition unit_MFRC522.cpp:360
bool readGsN(uint8_t &value)
Read the GsN register (N-driver conductance: CWGsN[7:4] / ModGsN[3:0])
Definition unit_MFRC522.cpp:322
bool mifareClassicAuthenticateB(const m5::nfc::a::PICC &picc, const uint8_t block, const m5::nfc::a::mifare::classic::Key &key=m5::nfc::a::mifare::classic::DEFAULT_KEY)
Authentication using keyB of the specified block.
Definition unit_MFRC522.hpp:408
UnitMFRC522(const uint8_t addr=DEFAULT_ADDRESS)
Constructor.
Definition unit_MFRC522.hpp:114
void config(const config_t &cfg)
Set the configuration.
Definition unit_MFRC522.hpp:146
bool nfcaTransceive(uint8_t *rx, uint16_t &rx_len, const uint8_t *tx, const uint16_t tx_len, const uint32_t timeout_ms)
Transceive.
Definition unit_MFRC522.cpp:184
bool selfTest()
Self test.
Definition unit_MFRC522.hpp:184
bool writeReceiverGain(const mfrc522::ReceiverGain gain)
Write the receiver gain.
Definition unit_MFRC522.cpp:312
bool writeGsN(const uint8_t value)
Write the GsN register.
Definition unit_MFRC522.cpp:327
bool calculateSoftwareCRC(uint16_t &result, const uint8_t *buf, const uint8_t len)
Calculate CRC by software.
Definition unit_MFRC522.cpp:411
bool writeRxThreshold(const uint8_t value)
Write the RxThreshold register.
Definition unit_MFRC522.cpp:365
bool calculateCRC(uint16_t &result, const uint8_t *buf, const uint8_t len)
Calculate CRC by hardware.
Definition unit_MFRC522.cpp:374
bool request(uint16_t &atqa)
Request for idle PICC.
Definition unit_MFRC522.hpp:336
bool readTPrescaler(uint16_t &tprescale)
Read the TPrescaler.
Definition unit_MFRC522.cpp:242
bool mifareClassicValueBlock(const m5::nfc::a::Command cmd, const uint8_t block, const uint32_t arg=0)
Operation for the value block.
Definition unit_MFRC522.cpp:676
bool turnOffAntenna()
Turn off the antenna.
Definition unit_MFRC522.cpp:289
virtual void update(const bool force=false) override
Update internal state.
Definition unit_MFRC522.cpp:179
bool writeBlock(const uint8_t block, const uint8_t tx[16])
Write the 1 block.
Definition unit_MFRC522.cpp:645
bool hlt()
Hlt for PICC.
Definition unit_MFRC522.cpp:654
bool wakeup(uint16_t &atqa)
Wakeup for idle/halt PICC.
Definition unit_MFRC522.hpp:345
virtual bool begin() override
Begin the unit.
Definition unit_MFRC522.cpp:139
bool softReset(const bool blocking=true)
Software reset.
Definition unit_MFRC522.cpp:220
bool readModGsP(uint8_t &value)
Read the ModGsP register (P-driver conductance during modulation)
Definition unit_MFRC522.cpp:346
bool mifareClassicStopCrypto1()
Exit from authenticated state for MIFARE classic.
Definition unit_MFRC522.cpp:671
bool readCWGsP(uint8_t &value)
Read the CWGsP register (P-driver conductance during no modulation)
Definition unit_MFRC522.cpp:332
Top level namespace of M5Stack.
Unit-related namespace.
Settings for begin.
Definition unit_MFRC522.hpp:97
bool enable_antenna
Enable antenna on begin if true.
Definition unit_MFRC522.hpp:103
uint8_t mode_reg
mode reg value. See also 9.3.2.2 ModeReg register
Definition unit_MFRC522.hpp:101
m5::nfc::NFC mode
Initial NFC mode applied at begin()
Definition unit_MFRC522.hpp:99
mfrc522::ReceiverGain receiver_gain
The receiver’s signal voltage gain factor.
Definition unit_MFRC522.hpp:105
bool software_crc
Using software CRC.
Definition unit_MFRC522.hpp:107
ReceiverGain
The receiver’s signal voltage gain factor.
Definition unit_MFRC522.hpp:51
Error
API error code.
Definition unit_MFRC522.hpp:66
@ TIMEOUT
Timeout occurs (0x83)
@ INTERNAL
Internal error.
@ CRC
CRC error (0x85)
@ CANNOT_RESOLVE_COLLISION
Collision resolution failed (0x87)
@ COMMUNICATION
Error in communication (0x81)
@ UID_NOT_COMPLETED
UID is not yet complete.
@ OCCUR_COLLISION
Collision occurs.
@ MIFARE_NACK
MIFARE NACK detection (0x84)
@ ARGUMENT
Error caused by arguments (0x80)
@ REGISTER
Error by error register value (0x82)
@ AUTH
Authentication error(0x86)
constexpr uint16_t MAX_FIFO_DEPTH
Maximum FIFO depth.
Definition unit_MFRC522.hpp:27
Command
PCD command.
Definition unit_MFRC522.hpp:33
@ SoftReset
Resets the MFRC522.
@ CalcCRC
Activates the CRC coprocessor or performs a self test.
@ NoCmdChange
No command change.
@ Receive
Activates the receiver circuits.
@ GenerateRandomID
Generates a 10-byte random ID number.
@ Transmit
Transmits data from the FIFO buffer.
@ MFAuthent
Performs the MIFARE standard authentication as a reader.
@ Mem
Stores 25 bytes into the internal buffer.
@ Idle
No action, cancels current command execution.