在嵌入式设备开发过程中有时会需要为设备设置唯一的ID用以标识设备唯一,比如要求同一总线上的所有设备ID不能重复,要求设备具体唯一的MAC地址等等。每个STM32微控制器都自带一个96位
的唯一ID,这个ID在任何情况下都是唯一且不允许修改
的,这96位的ID可以以字节(8位)为单位读取,也可以以半字(16位)或全字(32位)读取。不同型号的STM32芯片首地址不同,UID首地址也不同
。
在ST的相关资料中,对其功能的描述有3各方面:
- 用作序列号(例如 USB 字符串序列号或其它终端应用程序)
- 在对内部 Flash 进行编程前将唯一 ID 与软件加密原语和协议结合使用时用作安全密钥以提高 Flash 中代码的安全性
- 激活安全自举过程等
由上图可知,在STM32F1xx的数据手册中关于UID的描述有(从0x1FFFF7E8
地址开始的12个字节96bit)
在不同系列的MCU中地址是有差别的,如下图:
uint32_t GetUID(uint8_t* pMcuID)
{
uint32 CpuID[3] = {0};
//获取CPU唯一ID
#if 0//STM32F1系列
CpuID[0] =*(vu32*)(0x1ffff7e8); //按全字(32位)读取
CpuID[1] =*(vu32*)(0x1ffff7ec);
CpuID[2] =*(vu32*)(0x1ffff7f0);
#endif
#if 1//STM32F4系列
CpuID[0]=*(vu32*)(0x1fff7a10);
CpuID[1]=*(vu32*)(0x1fff7a14);
CpuID[2]=*(vu32*)(0x1fff7a18);
// /* printf the chipid */
// rt_kprintf("\r\n芯片的唯一ID为: %X-%X-%X\r\n",
// CpuID[0],CpuID[1],CpuID[2]);
// rt_kprintf("\r\n芯片flash的容量为: %dK \r\n", *(rt_uint16_t *)(0X1FFF7a22));
#endif
//按字节(8位)读取
pMcuID[0] = (uint8_t)(CpuID[0] & 0x000000FF);
pMcuID[1] = (uint8_t)((CpuID[0] & 0xFF00) >>8);
pMcuID[2] = (uint8_t)((CpuID[0] & 0xFF0000) >>16);
pMcuID[3] = (uint8_t)((CpuID[0] & 0xFF000000) >>24);
pMcuID[4] = (uint8_t)(CpuID[1] & 0xFF);
pMcuID[5] = (uint8_t)((CpuID[1] & 0xFF00) >>8);
pMcuID[6] = (uint8_t)((CpuID[1] & 0xFF0000) >>16);
pMcuID[7] = (uint8_t)((CpuID[1] & 0xFF000000) >>24);
pMcuID[8] = (uint8_t)(CpuID[2] & 0xFF);
pMcuID[9] = (uint8_t)((CpuID[2] & 0xFF00) >>8);
pMcuID[10] = (uint8_t)((CpuID[2] & 0xFF0000) >>16);
pMcuID[11] = (uint8_t)((CpuID[2] & 0xFF000000) >>24);
return (CpuID[0]>>1)+(CpuID[1]>>2)+(CpuID[2]>>3);
}
• 由 Leung 写于 2022 年 7 月 29 日
• 参考:STM32 进阶教程 9 - 芯片维一码(UID)读取
如何获取STM32 MCU的唯一ID
读取STM32芯片的唯一ID和MAC地址