当前位置:首页 > > 21ic电子网
[导读]前段时间研究了一下STM32内部的BootLoader  ,也就是STM32内嵌的一段程序,搞明白原理后,使用Qt编写的一个 测试小工具,展示效果如下。

出品 21ic论坛 一路向北lm

网站:bbs.21ic.com


前段时间研究了一下STM32内部的BootLoader  ,也就是STM32内嵌的一段程序,搞明白原理后,使用Qt编写的一个 测试小工具,展示效果如下:


很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序

可能你还不知道: STM32中内嵌了一段自举程序,可能很多人不知道。那段自举程序存放在System memory(系统存储器)中,我们通过配置启动,选择启动系统存储器就可以运行这段程序。
STM32有三种启动模式: 01.Main Flash memory:主FLASH
02.System memory:系统存储器
03.Embedded SRAM:内置SRAM
三种模式是通过不同配置来实现,一般通过BOOT引脚和BOOT位来配置启动模式。不同的芯片配置有差异,有些芯片没有BOOT1引脚,会结合BOOT位来实现。

内嵌的自举程序
STM32内嵌的自举程序存放在系统存储区,由ST在生产线上写入,用户不能修改。我们选择System memory(系统存储器)启动模式,就会进入系统存储区执行自举程序,内嵌的自举程序主要用于通过串行、IIC、SPI、USB、CAN等接口与外部通讯。不同型号MCU支持的串行接口不一样,具体请参看应用笔记 AN2606。


很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序

自举程序中UART协议
通信就会牵涉到协议,这里也说一下自举程序中 USART 协议。

USART 自举程序命令集:(具体参考官方文档AN3155)
很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序 4.介绍几个简单的命令流程
4.1 Get 命令
用户通过 Get 命令可获取自举程序的版本及支持的命令。自举程序接收到 Get 命令后,会将
自举程序版本和支持命令的代码发送给主机。
很多人不知道,STM32中内嵌了一段自举程序 具体如下:
很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序
4.2Get Version & Read Protection Status 命令GetVersion & Read Protection Status 命令用于获取自举程序版本及读保护状态。自举程序接收到此命令后,会将如下信息(版本、使能和禁止读保护的次数)发送给主机。
很多人不知道,STM32中内嵌了一段自举程序 具体如下: 很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序
4.3Get ID 命令Get ID命令用于获取芯片 ID(标识)的版本。自举程序接收到此命令后,会将产品 ID 发送给主机。 很多人不知道,STM32中内嵌了一段自举程序
具体如下: 很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序
4.4 Read Memory 命令ReadMemory 命令用于从RAM、 Flash 和信息块(系统存储器或选项字节区域)中的任何有效存储器地址(参见注释)读取数据。(读的前提是Flash没有加读保护操作) 很多人不知道,STM32中内嵌了一段自举程序
具体如下:(读取flash 0x 08000000 开始的16个字节的数据) 很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序
4.5 Go 命令Go 命令用于从应用程序指定的地址开始执行已下载的代码或其它任何代码。自举程序接收到 Go 命令后,会将 ACK 字节发送到应用程序。发送 ACK 字节后,自举程序会等待一个地址(4 个字节,字节 1 表示地址 MSB,字节 4 表示 LSB)和一个校验和字节,然后检查接收到的地址。如果接收到的地址有效且校验和正确,则自举程序将发送一个ACK字节,否则将发送一个 NACK 字节并中止此命令。
很多人不知道,STM32中内嵌了一段自举程序
具体如下:使用go 跳转到 flash地址处执行(0x08000000),此时LED已经闪烁 很多人不知道,STM32中内嵌了一段自举程序
命令太多,其它的就不再介绍了!
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序

5.自举程序上位机(自己使用Qt编写的,主要用于测试) 01. 在自举模式下可以简单实现对内部flash的数据的读出(flash不加锁) 02. 在自举模式下可以简单实现对内部sram的数据的读出(flash不加锁) 03.可以生成对应的.hex文件,使用Qt自己设计算法实现。 04.下载部分暂且没有做。
很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序
很多人不知道,STM32中内嵌了一段自举程序 很多人不知道,STM32中内嵌了一段自举程序

读取Hex文件  和拼Hex文件代码开源给大家
  1. // 读取芯片内部flash数据

  2. // 地址范围:0x0800 0000 ~ 0x0801 ffff   128K Byte

  3. // 循环读取,每次读取16个字节,读取128K Byte /16Byte = 8000 次

  4. int base = 0x08000000;

  5. for(int i =0;i<8000;i++)

  6. {

  7. //1.发送读内存命令

  8. UartSendHex("11ee");

  9. Sleep(5);

  10. //2.发送地址和校验和

  11. int offer = base + i*16;

  12. QString offer_str = QString::number(offer,16);

  13. if(offer_str.size()<8)

  14. offer_str.insert(0,"0");

  15. QString check_str;

  16. check_str = getXORresult(offer_str.mid(0,2), offer_str.mid(2,2));

  17. check_str = getXORresult(check_str, offer_str.mid(4,2));

  18. check_str = getXORresult(check_str, offer_str.mid(6,2));


  19. UartSendHex(offer_str+check_str);

  20. Sleep(5);

  21. //3.发送要读取的字节数和效验和

  22. QByteArray ByteNum = "0ff0";

  23. UartSendHex(ByteNum);

  24. Sleep(5);

  25. ui->progressBar->setFormat(QString::fromLocal8Bit("%1%").arg(QString::number((i+1)/10, 'f', 1)));

  26. ui->progressBar->setValue((i+1)/10);

  27. }



  28. // Qt flash数据 转 hex文件算法


  29. int t = 0,g = 0;

  30. QString text = ui->textEdit_Recv->toPlainText();

  31. QStringList number_list = text.split("\n");

  32. QStringList result_list;

  33. // 遍历字符串列表,滤除全ffff.........结尾,记录行数

  34. for(int i = 0; i < number_list.size(); ++i)

  35. {

  36. if(number_list.at(i) == "ffffffffffffffffffffffffffffffff")

  37. {

  38. t = i;

  39. break;

  40. }

  41. }

  42. // 遍历字符串列表斩断行,滤除改行的 ffff.........

  43. text.clear();

  44. text = number_list.at(t-1);

  45. for(int i =0;i

  46. {

  47. if((text.at(i) == 'f')&&(text.at(i+1) == 'f')&&(i%2 == 0))

  48. {

  49. g = i;

  50. break;

  51. }

  52. }

  53. // 拼接新的全数据的字符串列表

  54. for(int i =0;i

  55. {

  56. result_list<

  57. }

  58. if(g!=0)

  59. {

  60. text = text.remove(g,32-g);


  61. result_list<

  62. }

  63. //生成HEX文件

  64. //1.添加 10 字节数

  65. text.clear();

  66. QStringList result_list1;

  67. for(int i = 0; i < t; ++i)

  68. {

  69. text = result_list.at(i);

  70. if(i!=t-1)

  71. text = text.insert(0,"10");

  72. else

  73. {

  74. QString str = QString::number(text.size()/2,16);

  75. if(str.size()<2)

  76. str.insert(0,"0");

  77. text = text.insert(0,str);

  78. }

  79. result_list1 << text;

  80. }

  81. //2.添加地址偏移

  82. QStringList result_list2;

  83. int offer = 0x0000;

  84. QString offer_str;

  85. text.clear();

  86. for(int i = 0; i < t; ++i)

  87. {

  88. offer = i*16;

  89. offer_str = QString::number(offer,16);

  90. if(offer_str.size()<4)

  91. {

  92. if(offer_str.size() == 1)

  93. offer_str.insert(0,"000");

  94. if(offer_str.size() == 2)

  95. offer_str.insert(0,"00");

  96. if(offer_str.size() == 3)

  97. offer_str.insert(0,"0");

  98. }

  99. text = result_list1.at(i);

  100. text = text.insert(2,offer_str);

  101. result_list2 << text;

  102. }

  103. //3.添加效验和

  104. // 以020000040003F7为例子

  105. //具体算法为 0x100-((0x02+0x00+0x00+0x04+0x00+0x03)%256)= 0xF7

  106. text.clear();

  107. bool ok;

  108. int check;

  109. QString check_str;

  110. QStringList result_list3;


  111. for(int i = 0; i < t; ++i)

  112. {

  113. text = result_list2.at(i);

  114. check = 0;

  115. for(int j=0;j

  116. {

  117. check += text.mid(j*2,2).toInt(&ok,16);

  118. }

  119. check = (0x100 - check%256);

  120. check_str = QString::number(check,16);

  121. if(check_str.size()<2)

  122. check_str = check_str.insert(0,"0");

  123. text = text.insert(text.size(),check_str);

  124. result_list3<

  125. }

  126. //4.添加冒号 :

  127. QStringList result_list4;

  128. text.clear();

  129. for(int i = 0; i < t; ++i)

  130. {

  131. text = result_list3.at(i);

  132. text = text.insert(0,":");

  133. result_list4<

  134. }

  135. //5.添加HEX头部

  136. QString hear=":020000040800f2";

  137. //QStringList result_list5;

  138. result_list4.insert(0,hear);

  139. //6.添加HEX尾部

  140. QString tail1=":04000005080001cd21";

  141. QString tail2=":00000001ff";

  142. result_list4.insert(t+2,tail1);

  143. result_list4.insert(t+3,tail2);

  144. //7.保存数据为.hex文件

  145. ...............


本文系21ic论坛网友一路向北lm原创
 

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

21ic电子网

扫描二维码,关注更多精彩内容

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭