添加 PN532_FeliCa_THROUGH 选项,修改读写数据结构体,更新文档
This commit is contained in:
parent
03cd07ae73
commit
1617b07317
@ -81,9 +81,8 @@ enum { // 命令标记
|
||||
CMD_TO_UPDATER_MODE = 0x60,
|
||||
CMD_SEND_HEX_DATA = 0x61,
|
||||
CMD_TO_NORMAL_MODE = 0x62,
|
||||
CMD_FIRMWARE_UPDATE = 0x64,
|
||||
// CMD_SEND_BINDATA_INIT = 0x63,
|
||||
// CMD_SEND_BINDATA_EXEC = 0x64,
|
||||
CMD_SEND_BINDATA_INIT = 0x63,
|
||||
CMD_SEND_BINDATA_EXEC = 0x64,
|
||||
// FeliCa
|
||||
// CMD_FELICA_PUSH = 0x70,
|
||||
CMD_FELICA_THROUGH = 0x71,
|
||||
@ -98,6 +97,7 @@ enum { // 命令标记
|
||||
CMD_EXT_TO_NORMAL_MODE = 0xf5,
|
||||
};
|
||||
|
||||
#ifndef PN532_FeliCa_THROUGH
|
||||
enum { // FeliCa 专用,在 CMD_FELICA_THROUGH 命令使用
|
||||
FelicaPolling = 0x00,
|
||||
FelicaReqResponce = 0x04,
|
||||
@ -106,6 +106,7 @@ enum { // FeliCa 专用,在 CMD_FELICA_THROUGH 命令使用
|
||||
FelicaReqSysCode = 0x0C,
|
||||
FelicaActive2 = 0xA4,
|
||||
};
|
||||
#endif
|
||||
|
||||
enum { // 命令执行状态,res 数据包专用
|
||||
STATUS_OK = 0x00,
|
||||
@ -117,8 +118,8 @@ enum { // 命令执行状态,res 数据包专用
|
||||
STATUS_INTERNAL_ERROR = 0x06,
|
||||
STATUS_INVALID_FIRM_DATA = 0x07,
|
||||
STATUS_FIRM_UPDATE_SUCCESS = 0x08,
|
||||
STATUS_COMP_DUMMY_2ND = 0x10,
|
||||
STATUS_COMP_DUMMY_3RD = 0x20,
|
||||
STATUS_COMP_DUMMY_2ND = 0x10, // 837-15286
|
||||
STATUS_COMP_DUMMY_3RD = 0x20, // 837-15396
|
||||
};
|
||||
|
||||
typedef union { // 大小为 128 bytes 的联合体,用于存储收到的请求命令数据
|
||||
@ -138,6 +139,9 @@ typedef union { // 大小为 128 bytes 的联合体,用于存储收到的请
|
||||
};
|
||||
struct { // CMD_FELICA_THROUGH
|
||||
uint8_t encap_IDm[8];
|
||||
#ifdef PN532_FeliCa_THROUGH
|
||||
uint8_t felica_through_payload[1]; // 转发给 PN532 的命令,长度可变
|
||||
#else
|
||||
uint8_t encap_len;
|
||||
uint8_t encap_code;
|
||||
union {
|
||||
@ -156,6 +160,7 @@ typedef union { // 大小为 128 bytes 的联合体,用于存储收到的请
|
||||
};
|
||||
uint8_t felica_payload[1];
|
||||
};
|
||||
#endif
|
||||
};
|
||||
};
|
||||
};
|
||||
@ -185,6 +190,9 @@ typedef union {// 大小为 128 bytes 的联合体,用于存储读卡器准备
|
||||
};
|
||||
};
|
||||
};
|
||||
#ifdef PN532_FeliCa_THROUGH
|
||||
uint8_t felica_through_payload[1]; // 从 PN532 收到的回复,长度可变
|
||||
#else
|
||||
struct { // CMD_FELICA_THROUGH
|
||||
uint8_t encap_len;
|
||||
uint8_t encap_code;
|
||||
@ -202,6 +210,7 @@ typedef union {// 大小为 128 bytes 的联合体,用于存储读卡器准备
|
||||
uint8_t felica_payload[1];
|
||||
};
|
||||
};
|
||||
#endif
|
||||
};
|
||||
};
|
||||
} packet_response_t;
|
||||
@ -236,7 +245,7 @@ uint8_t packet_read() { // 数据包读取函数
|
||||
}
|
||||
req.bytes[++len] = r;
|
||||
if (len == req.frame_len) { // 长度正确且校验通过,则返回命令标记,否则返回 STATUS_SUM_ERROR
|
||||
if (req.cmd == CMD_FIRMWARE_UPDATE) return CMD_FIRMWARE_UPDATE; //如果命令为0x64,则不校验checksum
|
||||
if (req.cmd == CMD_SEND_BINDATA_EXEC) return req.cmd; //如果命令为0x64,则不校验checksum
|
||||
return checksum == r ? req.cmd : STATUS_SUM_ERROR;
|
||||
}
|
||||
checksum += r; // 包头后每位数据(不含转义)相加,作为校验值
|
||||
@ -355,17 +364,85 @@ void nfc_mifare_read() { // 认证成功后,读取 MIFARE 指定的 block
|
||||
}
|
||||
}
|
||||
|
||||
// 游戏发送的0x71命令后面的包实际上是完整的与Felica卡片直接通信的包,可以转发进PN532库直接用
|
||||
// response也可以直接打包转发回游戏
|
||||
#ifndef SKIP_FeliCa_THROUGH
|
||||
void nfc_felica_through() { // FeliCa 处理函数
|
||||
#ifdef PN532_FeliCa_THROUGH
|
||||
#pragma message "启用 PN532 FeliCa 直通"
|
||||
// 游戏发送的 0x71 命令后面的包实际上是完整的与 FeliCa 卡片直接通信的包,可以转发进 PN532 库直接用
|
||||
uint8_t response_length = 0xFF;
|
||||
if (nfc.inDataExchange(&req.encap_len, req.encap_len, &res.encap_len, &response_length)) {
|
||||
if (nfc.inDataExchange(req.felica_through_payload, req.felica_through_payload[0], res.felica_through_payload, &response_length)) {
|
||||
res_init(response_length);
|
||||
//如果成功的话 res.encap_len == response_length
|
||||
} else {
|
||||
res_init(0);
|
||||
res_init();
|
||||
res.status = STATUS_CARD_ERROR;
|
||||
// 数据交换失败时返回 STATUS_CARD_ERROR 触发重传
|
||||
// 不对数据交换失败时做特殊处理,直接返回 STATUS_CARD_ERROR。可能导致部分卡片(国产手机 + aicemu)读取失败。
|
||||
}
|
||||
#else // 旧 FeliCa 处理函数,留作参考
|
||||
uint16_t SystemCode;
|
||||
if (nfc.felica_Polling(0xFFFF, 0x01, res.encap_IDm, res.poll_PMm, &SystemCode, 200) == 1) {
|
||||
SystemCode = SystemCode >> 8 | SystemCode << 8; // 大小端数据翻转
|
||||
} else { // 如果读取 FeliCa 失败,则跳过后续操作
|
||||
res_init();
|
||||
res.status = STATUS_CARD_ERROR;
|
||||
return;
|
||||
}
|
||||
uint8_t code = req.encap_code;
|
||||
res.encap_code = code + 1;
|
||||
switch (code) {
|
||||
case FelicaPolling: // 作用未知,根据串口数据猜测
|
||||
{
|
||||
res_init(0x14);
|
||||
res.poll_systemCode[0] = SystemCode;
|
||||
res.poll_systemCode[1] = SystemCode >> 8;
|
||||
}
|
||||
break;
|
||||
case FelicaReqSysCode: // 作用未知,根据串口数据猜测
|
||||
{
|
||||
res_init(0x0D);
|
||||
res.felica_payload[0] = 0x01;
|
||||
res.felica_payload[1] = SystemCode;
|
||||
res.felica_payload[2] = SystemCode >> 8;
|
||||
}
|
||||
break;
|
||||
case FelicaActive2: // 作用未知,根据串口数据猜测
|
||||
{
|
||||
res_init(0x0B);
|
||||
res.felica_payload[0] = 0x00;
|
||||
}
|
||||
break;
|
||||
case FelicaReadWithoutEncryptData:
|
||||
{
|
||||
uint16_t serviceCodeList = req.serviceCodeList[1] << 8 | req.serviceCodeList[0];
|
||||
uint16_t blockList[4];
|
||||
for (uint8_t i = 0; i < req.numBlock; i++) { // 大小端数据翻转
|
||||
blockList[i] = (uint16_t)(req.blockList[i][0] << 8 | req.blockList[i][1]);
|
||||
}
|
||||
// 读取数据
|
||||
nfc.felica_ReadWithoutEncryption(1, &serviceCodeList, req.numBlock, blockList, res.blockData);
|
||||
res.RW_status[0] = 0;
|
||||
res.RW_status[1] = 0;
|
||||
res.numBlock = req.numBlock;
|
||||
res_init(0x0D + req.numBlock * 16);
|
||||
}
|
||||
break;
|
||||
case FelicaWriteWithoutEncryptData:
|
||||
{
|
||||
// 大小端数据翻转
|
||||
uint16_t serviceCodeList = req.serviceCodeList[1] << 8 | req.serviceCodeList[0];
|
||||
uint16_t blockList = (uint16_t)(req.blockList[0][0] << 8 | req.blockList[0][1]);
|
||||
// 写入数据
|
||||
nfc.felica_WriteWithoutEncryption(1, &serviceCodeList, 1, &blockList, &req.blockData);
|
||||
res_init(0x0C);
|
||||
res.RW_status[0] = 0;
|
||||
res.RW_status[1] = 0;
|
||||
}
|
||||
break;
|
||||
default: // 对于其他未知的数据默认处理方式,未确认效果
|
||||
res_init();
|
||||
res.status = STATUS_INVALID_COMMAND;
|
||||
}
|
||||
res.encap_len = res.payload_len;
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
|
@ -1,14 +1,14 @@
|
||||
#include "Aime_Reader.h"
|
||||
|
||||
// 该定义存在时,使用 115200 波特率,型号为 837-15396
|
||||
|
||||
#define high_baudrate
|
||||
|
||||
// 该定义存在时,跳过实际的 FeliCa 操作
|
||||
// 仅通过 CMD_CARD_DETECT 发送 IDm、PMm、SystemCode
|
||||
#define SKIP_FeliCa_THROUGH
|
||||
|
||||
// 该定义存在时,FeliCa 操作命令直接转发给 PN532
|
||||
#define PN532_FeliCa_THROUGH
|
||||
|
||||
#include "Aime_Reader.h"
|
||||
|
||||
void setup() {
|
||||
FastLED.addLeds<NEOPIXEL, LED_PIN>(leds, NUM_LEDS);
|
||||
@ -81,9 +81,10 @@ void loop() {
|
||||
nfc_mifare_read();
|
||||
break;
|
||||
|
||||
#ifndef SKIP_FeliCa_THROUGH
|
||||
// FeliCa 读写
|
||||
case CMD_FELICA_THROUGH:
|
||||
#ifdef SKIP_FeliCa_THROUGH
|
||||
#pragma message "跳过 FeliCa 操作"
|
||||
#else
|
||||
case CMD_FELICA_THROUGH: // FeliCa 读写
|
||||
nfc_felica_through();
|
||||
break;
|
||||
#endif
|
||||
@ -104,9 +105,10 @@ void loop() {
|
||||
case CMD_CARD_HALT:
|
||||
case CMD_EXT_TO_NORMAL_MODE:
|
||||
case CMD_TO_UPDATER_MODE:
|
||||
case CMD_SEND_BINDATA_INIT:
|
||||
res_init();
|
||||
break;
|
||||
case CMD_FIRMWARE_UPDATE:
|
||||
case CMD_SEND_BINDATA_EXEC:
|
||||
res_init();
|
||||
res.status = STATUS_FIRM_UPDATE_SUCCESS;
|
||||
// 当读卡器发送的 FW 版本与 amdaemon 要求的版本不一致时,
|
||||
|
@ -49,16 +49,16 @@
|
||||
|
||||
### 已知问题
|
||||
- 默认未启用 FeliCa 读写功能,仅在 `CMD_CARD_DETECT` 时读取 IDm 和 PMm,该设置可以通过 `SKIP_FeliCa_THROUGH` 控制
|
||||
- 启用 FeliCa 读写功能后,默认启用 `PN532_FeliCa_THROUGH`,FeliCa 操作命令直接转发给 PN532 处理,如果需要控制读写过程,可以禁用该定义,然后修改实现函数
|
||||
- 如果启用 FeliCa 读写功能,某些游戏可能不支持所有 FeliCa 卡种类,和官方读卡器行为一致
|
||||
- 因为 PN532 库支持的问题,未实现多卡同时读取,只会读到最先识别的卡片;刷数据不正确的 MIFARE 卡片(如交通卡、模拟卡)会导致游戏状态异常
|
||||
- 对于未适配的命令,默认回复 `STATUS_INVALID_COMMAND`,可能会导致游戏认为读卡器不可用
|
||||
- [此处修改](https://github.com/Sucareto/Arduino-Aime-Reader/commit/ebb2565ce4c43e8de275790f52bebee381d0bca7#diff-abc09b3d07eff9d03039f4f22ac9ac0747790237d3d00258e38644dbf13f9bbbR103) 会影响 amdaemon 对固件更新的操作,参考[此处说明](https://github.com/Sucareto/Arduino-Aime-Reader/blob/b16c63772b6d94ad9fca2c82bd89f0e900ed03d6/Arduino-Aime-Reader.ino#L114)
|
||||
|
||||
|
||||
### 版本更新情况
|
||||
- [最新](https://github.com/Sucareto/Arduino-Aime-Reader/tree/main)
|
||||
|
||||
通过 [pull/19](https://github.com/Sucareto/Arduino-Aime-Reader/pull/19) 实现了 FeliCa 的正确读写;
|
||||
通过 [QHPaeek/pull/19](https://github.com/Sucareto/Arduino-Aime-Reader/pull/19) 和 [nerimoe/pull/21](https://github.com/Sucareto/Arduino-Aime-Reader/pull/21) 实现了 FeliCa 的正确读写、固件更新处理。
|
||||
因未持有官方读卡器和框体等环境,无法进行更多测试,如没有 bug 或者新文档,将不会再更新。
|
||||
|
||||
- [v2.0](https://github.com/Sucareto/Arduino-Aime-Reader/commits/v2.0)
|
||||
@ -68,7 +68,7 @@
|
||||
- [v1.0](https://github.com/Sucareto/Arduino-Aime-Reader/commits/v1.0)
|
||||
|
||||
参考 [segatools](https://github.com/djhackersdev/segatools/blob/master/board/sg-nfc-cmd.h) 编写了基本实现;
|
||||
分析了官方读卡器串口数据,通过猜测数据结构完善了大部分功能。
|
||||
分析了官方读卡器串口数据,通过猜测数据结构完成了大部分功能。
|
||||
|
||||
|
||||
### 参考
|
||||
|
Loading…
Reference in New Issue
Block a user