• STM32F4学习笔记读取芯片UID和MAC地址


    一、简介

    在嵌入式设备开发过程中有时会需要为设备设置唯一的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中地址是有差别的,如下图:
    在这里插入图片描述

    二、获取芯片UID

    uint32_t GetUid(uint8_t* pUid)
    {
        uint32_t chipId[3] = {0};
            
        //获取CPU唯一ID
        #if 0//STM32F1系列
        chipId[0] =*(volatile unsigned long *)(0x1ffff7e8); //按全字(32位)读取
        chipId[1] =*(volatile unsigned long *)(0x1ffff7ec);
        chipId[2] =*(volatile unsigned long *)(0x1ffff7f0);
        #endif
        
        #if 1//STM32F4系列
        chipId[0]=*(volatile unsigned long *)(0x1fff7a10);
        chipId[1]=*(volatile unsigned long *)(0x1fff7a14);
        chipId[2]=*(volatile unsigned long *)(0x1fff7a18);
    //  /* printf the chipid */
    //  printf("\r\n芯片的唯一ID为: %X-%X-%X\r\n",
    //              chipId[0],chipId[1],chipId[2]);
    //  printf("\r\n芯片flash的容量为: %dK \r\n", *(uint16_t *)(0X1FFF7a22));
        #endif
        
        //按字节(8位)读取
        pUid[0] = (uint8_t)(chipId[0] & 0x000000FF);
        pUid[1] = (uint8_t)((chipId[0] & 0xFF00) >>8);
        pUid[2] = (uint8_t)((chipId[0] & 0xFF0000) >>16);
        pUid[3] = (uint8_t)((chipId[0] & 0xFF000000) >>24);
        
        pUid[4] = (uint8_t)(chipId[1] & 0xFF);
        pUid[5] = (uint8_t)((chipId[1] & 0xFF00) >>8);
        pUid[6] = (uint8_t)((chipId[1] & 0xFF0000) >>16);
        pUid[7] = (uint8_t)((chipId[1] & 0xFF000000) >>24);
        
        pUid[8] = (uint8_t)(chipId[2] & 0xFF);
        pUid[9] = (uint8_t)((chipId[2] & 0xFF00) >>8);
        pUid[10] = (uint8_t)((chipId[2] & 0xFF0000) >>16);
        pUid[11] = (uint8_t)((chipId[2] & 0xFF000000) >>24);
    
        return (chipId[0]>>1)+(chipId[1]>>2)+(chipId[2]>>3);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    uint8_t uid[12] = {0};
    GetUid(uid);
    for(uint8_t i = 0; i < 12; i++)
    {
        printf("%02x", uid[i]);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    三、获取MAC地址

    /**
     @brief 获取MAC地址
     @param pMac - [out] MAC地址
     @return 无
    */
    void GetMacAddress(uint8_t *pMac)
    {
        uint32_t uid = 0;
        uint8_t chipId[15] = {0};
        int i = 0;
    
        mcuId = GetChipId(chipId);
    
        for(i = 0; i < 12; i++)         // 获取ID[12]
        {
            chipId[12] += chipId[i];    
        }
        for(i=0; i<12; i++)             // 获取ID[13]
        {
            chipId[13] ^= chipId[i];    
        }
    
        pMac[0] = (uint8_t)(uid & 0xF0);
        pMac[1] = (uint8_t)((uid & 0xFF00) >> 8);
        pMac[2] = (uint8_t)((uid & 0xFF0000) >> 16);
        pMac[3] = (uint8_t)((uid & 0xFF000000) >> 24);
        pMac[4] = chipId[12];
        pMac[5] = chipId[13];  
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    uint8_t mac[6] = {0};
    GetMacAddress(mac);
    for(uint8_t i = 0; i < 6; i++)
    {
        printf("%02x", mac[i]);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    虽然这个96位的ID是唯一的,但是MAC地址却只有48位,因为量产有不同批次,而且采购的很随机的话这个ID号也是不唯一的,比较靠谱一点的还是自己在指定FLASH位置定义一个变量,这样程序就写死去读这个地方的值,而这个地方的值我们再用别的方式去修改,如自己写个上位机用串口通信设置等。

    MAC地址的前12bit固定,后面的便可以直接如此自定义设置。

    /**
     @brief 获取MAC地址
     @param pMac - [out] MAC地址
     @return 无
    */
    void GetMacAddress(uint8_t *pMac)
    {
        pMac[0] = 0x11;    
        pMac[1] = 0x22; 
        pMac[2] = *(volatile uint8_t *)(0X800F000); 
        pMac[3] = *(volatile uint8_t *)(0X800F001);
        pMac[4] = *(volatile uint8_t *)(0X800F002);
        pMac[5] = *(volatile uint8_t *)(0X800F003);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    原创链接:https://www.jianshu.com/p/79a1bbe6786f

  • 相关阅读:
    FPGA_探针(ISSP)调试工具
    SHA256WithRSA签名算法⽀付⽹关回调签名⽅案
    Skywalking入门
    java计算机毕业设计学生心理健康信息咨询系统源码+系统+数据库+lw文档
    代码随想录训练营第III期--011--python
    Spring Boot 注解
    用Html标签和CSS3写的一个手机
    shell脚本常用语句记录--持续更新
    面试(数据库的索引结构)
    dart flutter json 转 model 常用库对比 json_serializable json_model JsonToDart
  • 原文地址:https://blog.csdn.net/weixin_41226265/article/details/133382665