FM17550读写器实战:从零开始玩转S50卡(附完整代码)
FM17550读写器实战从零开始玩转S50卡附完整代码第一次接触RFID技术时我被那个隔空取物般的神奇体验震撼到了——不需要任何物理接触卡片靠近读写器就能完成数据交换。作为物联网领域最基础的感知技术之一RFID在门禁、物流、支付等场景中无处不在。本文将带你深入FM17550读写器与S50卡的实战开发从硬件连接到完整代码实现解决初学者最头疼的芯片手册看不懂、防冲突机制不会用等实际问题。1. 硬件准备与环境搭建1.1 硬件选型与连接FM17550是复旦微电子推出的13.56MHz RFID读写芯片兼容MFRC522但性价比更高。典型硬件连接如下表所示FM17550引脚连接目标备注VCC3.3V严禁接5VGND地线与MCU共地RSTGPIO硬件复位引脚IRQ悬空中断输出(本教程未使用)MISOMCU_MISOSPI从机输出MOSIMCU_MOSISPI从机输入SCKMCU_SCKSPI时钟SDA(SS)GPIO片选信号注意FM17550工作电压为2.5V-3.3V直接接5V会烧毁芯片建议使用电平转换模块与5V单片机通信。1.2 开发环境配置以STM32 HAL库为例需要配置SPI接口// SPI初始化代码示例 hspi1.Instance SPI1; hspi1.Init.Mode SPI_MODE_MASTER; hspi1.Init.Direction SPI_DIRECTION_2LINES; hspi1.Init.DataSize SPI_DATASIZE_8BIT; hspi1.Init.CLKPolarity SPI_POLARITY_LOW; hspi1.Init.CLKPhase SPI_PHASE_1EDGE; hspi1.Init.NSS SPI_NSS_SOFT; hspi1.Init.BaudRatePrescaler SPI_BAUDRATEPRESCALER_32; hspi1.Init.FirstBit SPI_FIRSTBIT_MSB; hspi1.Init.TIMode SPI_TIMODE_DISABLE; hspi1.Init.CRCCalculation SPI_CRCCALCULATION_DISABLE; hspi1.Init.CRCPolynomial 10; if (HAL_SPI_Init(hspi1) ! HAL_OK) { Error_Handler(); }关键参数说明BaudRatePrescalerSPI时钟分频建议初始设为32约1MHzNSS必须设为软件控制SPI_NSS_SOFTCLKPolarity/CLKPhase模式0POL0,PHA0或模式3POL1,PHA12. FM17550寄存器配置详解2.1 关键寄存器映射FM17550采用分页寄存器架构主要寄存器页如下寄存器页功能描述Page 0命令与状态控制Page 1通信控制Page 2发送配置Page 3接收配置常用寄存器操作函数// 写入寄存器 void FM17550_WriteReg(uint8_t addr, uint8_t val) { HAL_GPIO_WritePin(FM17550_SS_GPIO_Port, FM17550_SS_Pin, GPIO_PIN_RESET); HAL_SPI_Transmit(hspi1, addr, 1, 100); HAL_SPI_Transmit(hspi1, val, 1, 100); HAL_GPIO_WritePin(FM17550_SS_GPIO_Port, FM17550_SS_Pin, GPIO_PIN_SET); } // 读取寄存器 uint8_t FM17550_ReadReg(uint8_t addr) { uint8_t val; addr | 0x80; // 设置读标志位 HAL_GPIO_WritePin(FM17550_SS_GPIO_Port, FM17550_SS_Pin, GPIO_PIN_RESET); HAL_SPI_Transmit(hspi1, addr, 1, 100); HAL_SPI_Receive(hspi1, val, 1, 100); HAL_GPIO_WritePin(FM17550_SS_GPIO_Port, FM17550_SS_Pin, GPIO_PIN_SET); return val; }2.2 初始化流程完整的芯片初始化应包括以下步骤硬件复位拉低RST引脚至少50ms软件复位写入CommandReg的SoftReset命令定时器配置FM17550_WriteReg(TModeReg, 0x80); // 定时器自动重启 FM17550_WriteReg(TPrescalerReg, 0xA9);// 定时器预分频 FM17550_WriteReg(TReloadRegH, 0x03); // 重载值高位 FM17550_WriteReg(TReloadRegL, 0xE8); // 重载值低位RF配置FM17550_WriteReg(TxASKReg, 0x40); // 100% ASK调制 FM17550_WriteReg(ModeReg, 0x3D); // 定义发送器行为天线开启FM17550_WriteReg(TxControlReg, FM17550_ReadReg(TxControlReg) | 0x03); // 开启TX1/TX23. S50卡操作全流程3.1 卡片通信状态机S50卡遵循严格的状态转换机制POWER OFF → IDLE → READY → ACTIVE → HALT完整操作流程对应的状态转换复位POWER OFF → IDLE寻卡IDLE → READY防冲突获取完整UID选卡READY → ACTIVE数据操作读写/验证密码等挂起ACTIVE → HALT3.2 核心操作代码实现寻卡与防冲突uint8_t FM17550_Request(uint8_t req_mode, uint16_t *tag_type) { uint8_t status; FM17550_WriteReg(BitFramingReg, 0x07); // 最后一个字节的位数 FM17550_WriteReg(CommandReg, PCD_IDLE); // 停止当前命令 if (req_mode PICC_REQIDL) { FM17550_WriteReg(FIFOLevelReg, 0x26); // 写入REQA命令 } else { FM17550_WriteReg(FIFOLevelReg, 0x52); // 写入WUPA命令 } FM17550_WriteReg(CommandReg, PCD_TRANSCEIVE); FM17550_WriteReg(BitFramingReg, 0x80); // 启动发送 // 等待命令完成 uint16_t i 2000; while (1) { uint8_t n FM17550_ReadReg(ComIrqReg); if (n 0x30) break; // 收到数据或超时 if (--i 0) return STATUS_TIMEOUT; } // 检查错误标志 if (FM17550_ReadReg(ErrorReg) 0x13) { return STATUS_ERROR; } // 获取返回数据 uint8_t len FM17550_ReadReg(FIFOLevelReg); if (len ! 2) return STATUS_ERROR; *tag_type FM17550_ReadReg(FIFODataReg); *tag_type 8; *tag_type | FM17550_ReadReg(FIFODataReg); return STATUS_OK; }密码验证与数据读取uint8_t FM17550_Auth(uint8_t auth_mode, uint8_t block_addr, uint8_t *sector_key, uint8_t *ser_num) { uint8_t buff[12]; // 构造认证指令 buff[0] auth_mode; buff[1] block_addr; memcpy(buff[2], sector_key, 6); memcpy(buff[8], ser_num, 4); // 执行认证 return FM17550_ToCard(PCD_AUTHENT, buff, 12, NULL, NULL); } uint8_t FM17550_ReadBlock(uint8_t block_addr, uint8_t *recv_data) { uint8_t buff[2] {PICC_READ, block_addr}; uint8_t status FM17550_ToCard(PCD_TRANSCEIVE, buff, 2, recv_data, NULL); if (status ! STATUS_OK || recv_data[0] ! 0x0A) { return STATUS_ERROR; } return STATUS_OK; }4. 实战案例门禁系统设计4.1 系统架构设计[FM17550读写器] ←SPI→ [STM32 MCU] ←UART→ [上位机] ↑ [继电器控制] ↓ [电磁锁装置]4.2 核心业务逻辑实现void DoorLock_Control(uint8_t* uid) { // 1. 检查白名单 if (!Check_WhiteList(uid)) { Buzzer_Beep(3); // 非法卡提示音 return; } // 2. 验证扇区1密码 uint8_t default_key[6] {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}; if (FM17550_Auth(PICC_AUTHENT1A, 4, default_key, uid) ! STATUS_OK) { Buzzer_Beep(2); // 认证失败提示音 return; } // 3. 读取用户权限 uint8_t block_data[16]; if (FM17550_ReadBlock(4, block_data) STATUS_OK) { if (block_data[0] 0x01) { // 检查权限位 Relay_On(); // 开锁 Delay_ms(3000); Relay_Off(); // 闭锁 Log_Access(uid, 1); // 记录成功日志 } else { Buzzer_Beep(1); // 权限不足提示音 } } }4.3 常见问题排查问题1寻卡失败返回STATUS_TIMEOUT检查天线连接是否正常确认晶振起振测量OSCIN引脚应有13.56MHz信号调整RxGain寄存器值0x27地址建议初始值0x70问题2密码验证失败确认使用的密钥与卡片匹配检查防冲突过程是否获取了正确的UID验证前确保卡片处于ACTIVE状态问题3数据写入后读取异常S50卡每个扇区最后一块存储密钥禁止普通写入写入前必须通过密码验证注意块地址范围是0-6316扇区×4块在完成第一个RFID项目后我强烈建议将常用操作封装成库函数。比如把寻卡、验证、读写等流程做成API这样后续开发可以直接调用效率能提升数倍。实际项目中还要考虑多卡同时出现的防冲突处理以及针对不同扇区设置不同权限等高级功能。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2448122.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!