mirror of
https://github.com/esphome/esphome.git
synced 2025-01-06 21:01:43 +01:00
76a6e288b6
Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com>
322 lines
12 KiB
C++
322 lines
12 KiB
C++
#include <memory>
|
|
|
|
#include "pn7150.h"
|
|
#include "esphome/core/log.h"
|
|
|
|
namespace esphome {
|
|
namespace pn7150 {
|
|
|
|
static const char *const TAG = "pn7150.mifare_classic";
|
|
|
|
uint8_t PN7150::read_mifare_classic_tag_(nfc::NfcTag &tag) {
|
|
uint8_t current_block = 4;
|
|
uint8_t message_start_index = 0;
|
|
uint32_t message_length = 0;
|
|
|
|
if (this->auth_mifare_classic_block_(current_block, nfc::MIFARE_CMD_AUTH_A, nfc::NDEF_KEY) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Tag auth failed while attempting to read tag data");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
std::vector<uint8_t> data;
|
|
|
|
if (this->read_mifare_classic_block_(current_block, data) == nfc::STATUS_OK) {
|
|
if (!nfc::decode_mifare_classic_tlv(data, message_length, message_start_index)) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
} else {
|
|
ESP_LOGE(TAG, "Failed to read block %u", current_block);
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
uint32_t index = 0;
|
|
uint32_t buffer_size = nfc::get_mifare_classic_buffer_size(message_length);
|
|
std::vector<uint8_t> buffer;
|
|
|
|
while (index < buffer_size) {
|
|
if (nfc::mifare_classic_is_first_block(current_block)) {
|
|
if (this->auth_mifare_classic_block_(current_block, nfc::MIFARE_CMD_AUTH_A, nfc::NDEF_KEY) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Block authentication failed for %u", current_block);
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
}
|
|
std::vector<uint8_t> block_data;
|
|
if (this->read_mifare_classic_block_(current_block, block_data) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Error reading block %u", current_block);
|
|
return nfc::STATUS_FAILED;
|
|
} else {
|
|
buffer.insert(buffer.end(), block_data.begin(), block_data.end());
|
|
}
|
|
|
|
index += nfc::MIFARE_CLASSIC_BLOCK_SIZE;
|
|
current_block++;
|
|
|
|
if (nfc::mifare_classic_is_trailer_block(current_block)) {
|
|
current_block++;
|
|
}
|
|
}
|
|
|
|
if (buffer.begin() + message_start_index < buffer.end()) {
|
|
buffer.erase(buffer.begin(), buffer.begin() + message_start_index);
|
|
} else {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
tag.set_ndef_message(make_unique<nfc::NdefMessage>(buffer));
|
|
|
|
return nfc::STATUS_OK;
|
|
}
|
|
|
|
uint8_t PN7150::read_mifare_classic_block_(uint8_t block_num, std::vector<uint8_t> &data) {
|
|
nfc::NciMessage rx;
|
|
nfc::NciMessage tx(nfc::NCI_PKT_MT_DATA, {XCHG_DATA_OID, nfc::MIFARE_CMD_READ, block_num});
|
|
|
|
ESP_LOGVV(TAG, "Read XCHG_DATA_REQ: %s", nfc::format_bytes(tx.get_message()).c_str());
|
|
if (this->transceive_(tx, rx) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Timeout reading tag data");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
if ((!rx.message_type_is(nfc::NCI_PKT_MT_DATA)) || (!rx.simple_status_response_is(XCHG_DATA_OID)) ||
|
|
(!rx.message_length_is(18))) {
|
|
ESP_LOGE(TAG, "MFC read block failed - block 0x%02x", block_num);
|
|
ESP_LOGV(TAG, "Read response: %s", nfc::format_bytes(rx.get_message()).c_str());
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
data.insert(data.begin(), rx.get_message().begin() + 4, rx.get_message().end() - 1);
|
|
|
|
ESP_LOGVV(TAG, " Block %u: %s", block_num, nfc::format_bytes(data).c_str());
|
|
return nfc::STATUS_OK;
|
|
}
|
|
|
|
uint8_t PN7150::auth_mifare_classic_block_(uint8_t block_num, uint8_t key_num, const uint8_t *key) {
|
|
nfc::NciMessage rx;
|
|
nfc::NciMessage tx(nfc::NCI_PKT_MT_DATA, {MFC_AUTHENTICATE_OID, this->sect_to_auth_(block_num), key_num});
|
|
|
|
switch (key_num) {
|
|
case nfc::MIFARE_CMD_AUTH_A:
|
|
tx.get_message().back() = MFC_AUTHENTICATE_PARAM_KS_A;
|
|
break;
|
|
|
|
case nfc::MIFARE_CMD_AUTH_B:
|
|
tx.get_message().back() = MFC_AUTHENTICATE_PARAM_KS_B;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (key != nullptr) {
|
|
tx.get_message().back() |= MFC_AUTHENTICATE_PARAM_EMBED_KEY;
|
|
tx.get_message().insert(tx.get_message().end(), key, key + 6);
|
|
}
|
|
|
|
ESP_LOGVV(TAG, "MFC_AUTHENTICATE_REQ: %s", nfc::format_bytes(tx.get_message()).c_str());
|
|
if (this->transceive_(tx, rx) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Sending MFC_AUTHENTICATE_REQ failed");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
if ((!rx.message_type_is(nfc::NCI_PKT_MT_DATA)) || (!rx.simple_status_response_is(MFC_AUTHENTICATE_OID)) ||
|
|
(rx.get_message()[4] != nfc::STATUS_OK)) {
|
|
ESP_LOGE(TAG, "MFC authentication failed - block 0x%02x", block_num);
|
|
ESP_LOGVV(TAG, "MFC_AUTHENTICATE_RSP: %s", nfc::format_bytes(rx.get_message()).c_str());
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
ESP_LOGV(TAG, "MFC block %u authentication succeeded", block_num);
|
|
return nfc::STATUS_OK;
|
|
}
|
|
|
|
uint8_t PN7150::sect_to_auth_(const uint8_t block_num) {
|
|
const uint8_t first_high_block = nfc::MIFARE_CLASSIC_BLOCKS_PER_SECT_LOW * nfc::MIFARE_CLASSIC_16BLOCK_SECT_START;
|
|
if (block_num >= first_high_block) {
|
|
return ((block_num - first_high_block) / nfc::MIFARE_CLASSIC_BLOCKS_PER_SECT_HIGH) +
|
|
nfc::MIFARE_CLASSIC_16BLOCK_SECT_START;
|
|
}
|
|
return block_num / nfc::MIFARE_CLASSIC_BLOCKS_PER_SECT_LOW;
|
|
}
|
|
|
|
uint8_t PN7150::format_mifare_classic_mifare_() {
|
|
std::vector<uint8_t> blank_buffer(
|
|
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00});
|
|
std::vector<uint8_t> trailer_buffer(
|
|
{0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x07, 0x80, 0x69, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF});
|
|
|
|
auto status = nfc::STATUS_OK;
|
|
|
|
for (int block = 0; block < 64; block += 4) {
|
|
if (this->auth_mifare_classic_block_(block + 3, nfc::MIFARE_CMD_AUTH_B, nfc::DEFAULT_KEY) != nfc::STATUS_OK) {
|
|
continue;
|
|
}
|
|
if (block != 0) {
|
|
if (this->write_mifare_classic_block_(block, blank_buffer) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
}
|
|
if (this->write_mifare_classic_block_(block + 1, blank_buffer) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block + 1);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(block + 2, blank_buffer) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block + 2);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(block + 3, trailer_buffer) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block + 3);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
uint8_t PN7150::format_mifare_classic_ndef_() {
|
|
std::vector<uint8_t> empty_ndef_message(
|
|
{0x03, 0x03, 0xD0, 0x00, 0x00, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00});
|
|
std::vector<uint8_t> blank_block(
|
|
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00});
|
|
std::vector<uint8_t> block_1_data(
|
|
{0x14, 0x01, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1});
|
|
std::vector<uint8_t> block_2_data(
|
|
{0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1, 0x03, 0xE1});
|
|
std::vector<uint8_t> block_3_trailer(
|
|
{0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0x78, 0x77, 0x88, 0xC1, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF});
|
|
std::vector<uint8_t> ndef_trailer(
|
|
{0xD3, 0xF7, 0xD3, 0xF7, 0xD3, 0xF7, 0x7F, 0x07, 0x88, 0x40, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF});
|
|
|
|
if (this->auth_mifare_classic_block_(0, nfc::MIFARE_CMD_AUTH_B, nfc::DEFAULT_KEY) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to authenticate block 0 for formatting");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(1, block_1_data) != nfc::STATUS_OK) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(2, block_2_data) != nfc::STATUS_OK) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(3, block_3_trailer) != nfc::STATUS_OK) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
ESP_LOGD(TAG, "Sector 0 formatted with NDEF");
|
|
|
|
auto status = nfc::STATUS_OK;
|
|
|
|
for (int block = 4; block < 64; block += 4) {
|
|
if (this->auth_mifare_classic_block_(block + 3, nfc::MIFARE_CMD_AUTH_B, nfc::DEFAULT_KEY) != nfc::STATUS_OK) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
if (block == 4) {
|
|
if (this->write_mifare_classic_block_(block, empty_ndef_message) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
} else {
|
|
if (this->write_mifare_classic_block_(block, blank_block) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
}
|
|
if (this->write_mifare_classic_block_(block + 1, blank_block) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block + 1);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(block + 2, blank_block) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write block %u", block + 2);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
if (this->write_mifare_classic_block_(block + 3, ndef_trailer) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Unable to write trailer block %u", block + 3);
|
|
status = nfc::STATUS_FAILED;
|
|
}
|
|
}
|
|
return status;
|
|
}
|
|
|
|
uint8_t PN7150::write_mifare_classic_block_(uint8_t block_num, std::vector<uint8_t> &write_data) {
|
|
nfc::NciMessage rx;
|
|
nfc::NciMessage tx(nfc::NCI_PKT_MT_DATA, {XCHG_DATA_OID, nfc::MIFARE_CMD_WRITE, block_num});
|
|
|
|
ESP_LOGVV(TAG, "Write XCHG_DATA_REQ 1: %s", nfc::format_bytes(tx.get_message()).c_str());
|
|
if (this->transceive_(tx, rx) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Sending XCHG_DATA_REQ failed");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
// write command part two
|
|
tx.set_payload({XCHG_DATA_OID});
|
|
tx.get_message().insert(tx.get_message().end(), write_data.begin(), write_data.end());
|
|
|
|
ESP_LOGVV(TAG, "Write XCHG_DATA_REQ 2: %s", nfc::format_bytes(tx.get_message()).c_str());
|
|
if (this->transceive_(tx, rx, NFCC_TAG_WRITE_TIMEOUT) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "MFC XCHG_DATA timed out waiting for XCHG_DATA_RSP during block write");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
if ((!rx.message_type_is(nfc::NCI_PKT_MT_DATA)) || (!rx.simple_status_response_is(XCHG_DATA_OID)) ||
|
|
(rx.get_message()[4] != nfc::MIFARE_CMD_ACK)) {
|
|
ESP_LOGE(TAG, "MFC write block failed - block 0x%02x", block_num);
|
|
ESP_LOGV(TAG, "Write response: %s", nfc::format_bytes(rx.get_message()).c_str());
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
|
|
return nfc::STATUS_OK;
|
|
}
|
|
|
|
uint8_t PN7150::write_mifare_classic_tag_(const std::shared_ptr<nfc::NdefMessage> &message) {
|
|
auto encoded = message->encode();
|
|
|
|
uint32_t message_length = encoded.size();
|
|
uint32_t buffer_length = nfc::get_mifare_classic_buffer_size(message_length);
|
|
|
|
encoded.insert(encoded.begin(), 0x03);
|
|
if (message_length < 255) {
|
|
encoded.insert(encoded.begin() + 1, message_length);
|
|
} else {
|
|
encoded.insert(encoded.begin() + 1, 0xFF);
|
|
encoded.insert(encoded.begin() + 2, (message_length >> 8) & 0xFF);
|
|
encoded.insert(encoded.begin() + 3, message_length & 0xFF);
|
|
}
|
|
encoded.push_back(0xFE);
|
|
|
|
encoded.resize(buffer_length, 0);
|
|
|
|
uint32_t index = 0;
|
|
uint8_t current_block = 4;
|
|
|
|
while (index < buffer_length) {
|
|
if (nfc::mifare_classic_is_first_block(current_block)) {
|
|
if (this->auth_mifare_classic_block_(current_block, nfc::MIFARE_CMD_AUTH_A, nfc::NDEF_KEY) != nfc::STATUS_OK) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
}
|
|
|
|
std::vector<uint8_t> data(encoded.begin() + index, encoded.begin() + index + nfc::MIFARE_CLASSIC_BLOCK_SIZE);
|
|
if (this->write_mifare_classic_block_(current_block, data) != nfc::STATUS_OK) {
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
index += nfc::MIFARE_CLASSIC_BLOCK_SIZE;
|
|
current_block++;
|
|
|
|
if (nfc::mifare_classic_is_trailer_block(current_block)) {
|
|
// Skipping as cannot write to trailer
|
|
current_block++;
|
|
}
|
|
}
|
|
return nfc::STATUS_OK;
|
|
}
|
|
|
|
uint8_t PN7150::halt_mifare_classic_tag_() {
|
|
nfc::NciMessage rx;
|
|
nfc::NciMessage tx(nfc::NCI_PKT_MT_DATA, {XCHG_DATA_OID, nfc::MIFARE_CMD_HALT, 0});
|
|
|
|
ESP_LOGVV(TAG, "Halt XCHG_DATA_REQ: %s", nfc::format_bytes(tx.get_message()).c_str());
|
|
if (this->transceive_(tx, rx, NFCC_TAG_WRITE_TIMEOUT) != nfc::STATUS_OK) {
|
|
ESP_LOGE(TAG, "Sending halt XCHG_DATA_REQ failed");
|
|
return nfc::STATUS_FAILED;
|
|
}
|
|
return nfc::STATUS_OK;
|
|
}
|
|
|
|
} // namespace pn7150
|
|
} // namespace esphome
|