第51章 设置FLASH的读写保护及解除—零死角玩转STM32-F429系列

第51章     设置FLASH的读写保护及解除

全套200集视频教程和1000PDF教程请到秉火论坛下载:www.firebbs.cn

野火视频教程优酷观看网址:http://i.youku.com/firege

 

 

 

本章参考资料:《STM32F4xx 中文参考手册》、《STM32F4xx规格书》、库说明文档《stm32f4xx_dsp_stdperiph_lib_um.chm》以及《Proprietary code read-out protection on microcontrollers》。

51.1 选项字节与读写保护

在实际发布的产品中,在STM32芯片的内部FLASH存储了控制程序,如果不作任何保护措施的话,可以使用下载器直接把内部FLASH的内容读取回来,得到binhex文件格式的代码拷贝,别有用心的厂商即可利用该代码文件山寨产品。为此,STM32芯片提供了多种方式保护内部FLASH的程序不被非法读取,但在默认情况下该保护功能是不开启的,若要开启该功能,需要改写内部FLASH选项字节(Option Bytes)中的配置。

51.1.1 选项字节的内容

选项字节是一段特殊的FLASH空间,STM32芯片会根据它的内容进行读写保护、复位电压等配置,选项字节的构成见表 511

511 选项字节的构成

地址

[63:16]

[15:0]

0x1FFF C000

保留

ROP 和用户选项字节 (RDP & USER)

0x1FFF C008

保留

扇区 0 11 的写保护 nWRP

0x1FFE C000

保留

保留

0x1FFE C008

保留

扇区 12 23 的写保护 nWRP

选项字节具体的数据位配置说明见表 512

512 选项字节具体的数据位配置说明

选项字节(字,地址 0x1FFF C000

RDP读保护选项字节。
读保护用于保护 Flash 中存储的软件代码。

15:8

0xAA:级别 0,无保护
其它值:级别 1,存储器读保护(调试功能受限)
0xCC
:级别 2,芯片保护(禁止调试和从 RAM 启动)

USER用户选项字节
此字节用于配置以下功能:
选择看门狗事件:硬件或软件
进入停止模式时产生复位事件
进入待机模式时产生复位事件

7

nRST_STDBY
0
:进入待机模式时产生复位
1
:不产生复位

6

nRST_STOP
0
:进入停止模式时产生复位
1
:不产生复位

5

WDG_SW
0
:硬件看门狗
1
:软件看门狗

4

0x1:未使用

3:2

BOR_LEV BOR 复位级别
这些位包含释放复位信号所需达到的供电电压阈值。
通过对这些位执行写操作,可将新的 BOR 级别值编程到 Flash
00
BOR 级别 3 (VBOR3),复位阈值电压为 2.70 V 3.60 V
01
BOR 级别 2 (VBOR2),复位阈值电压为 2.40 V 2.70 V
10
BOR 级别 1 (VBOR1),复位阈值电压为 2.10 V 2.40 V
11
BOR 关闭 (VBOR0),复位阈值电压为 1.8 V 2.10 V

1:0

0x1:未使用

选项字节(字,地址 0x1FFF C008

15

SPMOD:选择nWPRi位的模式
0
nWPRi位用于写保护(默认)
1
nWPRi位用于代码读出保护(Proprietary code readout protection PCROP)

14

DB1M:设置内部FLASH1MB的产品的双Bank模式
0
:单个Bank的模式
1
:使用两个Bank的模式

13:12

0xF:未使用

nWRP Flash 写保护选项字节。
扇区 0 11 可采用写保护。

11:0

nWRPi (i值为0-11,对应0-11扇区的保护设置)
0
:开启所选扇区的写保护
1
:关闭所选扇区的写保护

选项字节(字,地址 0x1FFE C000

15:0

0xFF:未使用

选项字节(字,地址 0x1FFE C008

15:12

0xF:未使用

nWRP Flash 写保护选项字节。
扇区 12 23 可采用写保护。

11:0

nWRPi
0
:开启扇区 i 的写保护。
1
:关闭扇区 i 的写保护。

我们主要讲解选项字节配置中的RDP位和PCROP位,它们分别用于配置读保护级别及代码读出保护。

51.1.2 RDP读保护级别

修改选项字节的RDP位的值可设置内部FLASH为以下保护级别:

    0xAA:级别0,无保护

这是STM32的默认保护级别,它没有任何读保护,读取内部FLASH及"备份SRAM"的内容都没有任何限制。(注意这里说的"备份SRAM"是指STM32备份域的SRAM空间,不是指主SRAM,下同)

    其它值:级别1,使能读保护

RDP配置成除0xAA0xCC外的任意数值,都会使能级别1的读保护。在这种保护下,若使用调试功能(使用下载器、仿真器)或者从内部SRAM自举时都不能对内部FLASH及备份SRAM作任何访问(读写、擦除都被禁止);而如果STM32是从内部FLASH自举时,它允许对内部FLASH及备份SRAM的任意访问。

也就是说,在级别1模式下,任何尝试从外部访问内部FLASH内容的操作都被禁止,例如无法通过下载器读取它的内容,或编写一个从内部SRAM启动的程序,若该程序读取内部FLASH,会被禁止。而如果是芯片自己访问内部FLASH,是完全没有问题的,例如前面的"读写内部FLASH"实验中的代码自己擦写内部FLASH空间的内容,即使处于级别1的读保护,也能正常擦写。

当芯片处于级别1的时候,可以把选项字节的RDP位重新设置为0xAA,恢复级别0。在恢复到级别0前,芯片会自动擦除内部FLASH及备份SRAM的内容,即降级后原内部FLASH的代码会丢失。在级别1时使用SRAM自举的程序也可以访问选项字节进行修改,所以如果原内部FLASH的代码没有解除读保护的操作时,可以给它加载一个SRAM自举的程序进行保护降级,后面我们将会进行这样的实验。

    0xCC:级别2,禁止调试

RDP配置成0xCC值时,会进入最高级别的读保护,且设置后无法再降级,它会永久禁止用于调试的JTAG接口(相当于熔断)。在该级别中,除了具有级别1的所有保护功能外,进一步禁止了从SRAM或系统存储器的自举(即平时使用的串口ISP下载功能也失效)JTAG调试相关的功能被禁止,选项字节也不能被修改。它仅支持从内部FLASH自举时对内部FLASHSRAM的访问(读写、擦除)

由于设置了级别2后无法降级,也无法通过JTAG、串口ISP等方式更新程序,所以使用这个级别的保护时一般会在程序中预留"后门"以更新应用程序,若程序中没有预留后门,芯片就无法再更新应用程序了。所谓的"后门"是一种IAP程序(In Application Program),它通过某个通讯接口获取将要更新的程序内容,然后利用内部FLASH擦写操作把这些内容烧录到自己的内部FLASH中,实现应用程序的更新。

不同级别下的访问限制见图 511

511 不同级别下的访问限制

不同保护级别之间的状态转换见图 512

512 不同级别间的状态转换

51.1.3 PCROP代码读出保护

STM32F42xxSTM32F43xx系列的芯片中,除了可使用RDP对整片FLASH进行读保护外,还有一个专用的代码读出保护功能(Proprietary code readout protection,下面简称PCROP),它可以为内部FLASH的某几个指定扇区提供保护功能,所以它可以用于保护一些IP代码,方便提供给另一方进行二次开发,见图 513

513 PCROP保护功能

SPMOD位设置为0(默认值)nWRPi位用于指定要进行写保护的扇区,这可以防止错误的指针操作导致FLASH 内容的改变,若扇区被写保护,通过调试器也无法擦除该扇区的内容;当SPMOD位设置为1时,nWRPi位用于指定要进行PCROP保护的扇区。其中PCROP功能可以防止指定扇区的FLASH内容被读出,而写保护仅可以防止误写操作,不能被防止读出。

当要关闭PCROP功能时,必须要使芯片从读保护级别1降为级别0,同时对SPMOD位置0,才能正常关闭;若芯片原来的读保护为级别0,且使能了PCROP保护,要关闭PCROP时也要先把读保护级别设置为级别1,再在降级的同时设置SPMOD0

51.2 修改选项字节的过程

修改选项字节的内容可修改各种配置,但是,当应用程序运行时,无法直接通过选项字节的地址改写它们的内容,例如,接使用指针操作地址0x1FFFC0000的修改是无效的。要改写其内容时必须设置寄存器FLASH_OPTCRFLASH_OPTCR1中的对应数据位,寄存器的与选项字节对应位置见图 514及图 515,详细说明请查阅《STM32参考手册》。

514 FLASH_OPTCR寄存器说明(nWRP表示0-11扇区)

515 FLASH_OPTCR1寄存器说明(nWRP表示12-23扇区)

默认情况下,FLASH_OPTCR寄存器中的第0OPTLOCK值为1,它表示选项字节被上锁,需要解锁后才能进行修改,当寄存器的值设置完成后,对FLASH_OPTCR寄存器中的第1OPTSTRT值设置为1,硬件就会擦除选项字节扇区的内容,并把FLASH_OPTCR/1寄存器中包含的值写入到选项字节。

所以,修改选项字节的配置步骤如下:

(1)    解锁,在 Flash 选项密钥寄存器 (FLASH_OPTKEYR) 中写入 OPTKEY1 = 0x0819 2A3B;接着在 Flash 选项密钥寄存器 (FLASH_OPTKEYR) 中写入 OPTKEY2 = 0x4C5D 6E7F。

(2)    检查 FLASH_SR 寄存器中的 BSY 位,以确认当前未执行其它Flash 操作。

(3)    在 FLASH_OPTCR 和/或 FLASH_OPTCR1 寄存器中写入选项字节值。

(4)    将 FLASH_OPTCR 寄存器中的选项启动位 (OPTSTRT) 置 1。

(5)    等待 BSY 位清零,即写入完成。

51.3 操作选项字节的库函数

为简化编程,STM32标准库提供了一些库函数,它们封装了修改选项字节时操作寄存器的过程。

1.    选项字节解锁、上锁函数

对选项字节解锁、上锁的函数见代码清单 511

代码清单 511选项字节解锁、上锁

1

2 #define FLASH_OPT_KEY1 ((uint32_t)0x08192A3B)

3 #define FLASH_OPT_KEY2 ((uint32_t)0x4C5D6E7F)

4

5 /**

6 * @brief Unlocks the FLASH Option Control Registers access.

7 * @param None

8 * @retval None

9 */

10 void FLASH_OB_Unlock(void)

11 {

12 if((FLASH->OPTCR & FLASH_OPTCR_OPTLOCK) != RESET)

13 {

14 /* Authorizes the Option Byte register programming */

15 FLASH->OPTKEYR = FLASH_OPT_KEY1;

16 FLASH->OPTKEYR = FLASH_OPT_KEY2;

17 }

18 }

19

20 /**

21 * @brief Locks the FLASH Option Control Registers access.

22 * @param None

23 * @retval None

24 */

25 void FLASH_OB_Lock(void)

26 {

27 /* Set the OPTLOCK Bit to lock the FLASH Option Byte Registers access */

28 FLASH->OPTCR |= FLASH_OPTCR_OPTLOCK;

29 }

解锁的时候,它对FLASH_OPTCR寄存器写入两个解锁参数,上锁的时候,对FLASH_ OPTCR寄存器的FLASH_OPTCR_OPTLOCK位置1

2.    设置读保护级别

解锁后设置选项字节寄存器的RDP位可调用FLASH_OB_RDPConfig完成,见代码清单 512

代码清单 512 设置读保护级别

1 /**

2 * @brief Sets the read protection level.

3 * @param OB_RDP: specifies the read protection level.

4 * This parameter can be one of the following values:

5 * @arg OB_RDP_Level_0: No protection

6 * @arg OB_RDP_Level_1: Read protection of the memory

7 * @arg OB_RDP_Level_2: Full chip protection

8 *

9 * /!\ Warning /!\ When enabling OB_RDP level 2 it's no more possible to go back to level 1 or 0

10 *

11 * @retval None

12 */

13 void FLASH_OB_RDPConfig(uint8_t OB_RDP)

14 {

15 FLASH_Status status = FLASH_COMPLETE;

16

17 /* Check the parameters */

18 assert_param(IS_OB_RDP(OB_RDP));

19

20 status = FLASH_WaitForLastOperation();

21

22 if(status == FLASH_COMPLETE)

23 {

24 *(__IO uint8_t*)OPTCR_BYTE1_ADDRESS = OB_RDP;

25

26 }

27 }

该函数根据输入参数设置RDP寄存器位为相应的级别,其注释警告了若配置成OB_RDP_Level_2会无法恢复。类似地,配置其它选项时也有相应的库函数,如FLASH_OB_PCROP1ConfigFLASH_OB_WRP1Config分别用于设置要进行PCROP保护或WRP保护(写保护)的扇区。

3.    写入选项字节

调用上一步骤中的函数配置寄存器后,还要调用代码清单 513中的FLASH_OB_Launch函数把寄存器的内容写入到选项字节中。

代码清单 513 写入选项字节

1 /**

2 * @brief Launch the option byte loading.

3 * @param None

4* @retval FLASH Status: The returned value can be: FLASH_BUSY, FLASH_ERROR_PROGRAM,

5 * FLASH_ERROR_WRP, FLASH_ERROR_OPERATION or FLASH_COMPLETE.

6 */

7 FLASH_Status FLASH_OB_Launch(void)

8 {

9 FLASH_Status status = FLASH_COMPLETE;

10

11 /* Set the OPTSTRT bit in OPTCR register */

12 *(__IO uint8_t *)OPTCR_BYTE0_ADDRESS |= FLASH_OPTCR_OPTSTRT;

13

14 /* Wait for last operation to be completed */

15 status = FLASH_WaitForLastOperation();

16

17 return status;

18 }

该函数设置FLASH_OPTCR_OPTSTRT位后调用了FLASH_WaitForLastOperation函数等待写入完成,并返回写入状态,若操作正常,它会返回FLASH_COMPLETE

51.4 实验:设置读写保护及解除

在本实验中我们将以实例讲解如何修改选项字节的配置,更改读保护级别、设置PCROP或写保护,最后把选项字节恢复默认值。

本实验要进行的操作比较特殊,在开发和调试的过程中都是在SRAM上进行的(使用SRAM启动方式)。例如,直接使用FLASH版本的程序进行调试时,如果该程序在运行后对扇区进行了写保护而没有解除的操作或者该解除操作不正常,此时将无法再给芯片的内部FLASH下载新程序,最终还是要使用SRAM自举的方式进行解除操作。所以在本实验中为便于修改选项字节的参数,我们统一使用SRAM版本的程序进行开发和学习,当SRAM版本调试正常后再改为FLASH版本。

关于在SRAM中调试代码的相关配置,请参考前面的章节。

注意:

若您在学习的过程中想亲自修改代码进行测试,请注意备份原工程代码。当芯片的FLASH被保护导致无法下载程序到FLASH时,可以下载本工程到芯片,并使用SRAM启动运行,即可恢复芯片至默认配置。但如果修改了读保护为级别2,采用任何方法都无法恢复!(除了这个配置,其它选项都可以大胆地修改测试。)

51.4.1 硬件设计

本实验在SRAM中调试代码,因此把BOOT0BOOT1引脚都使用跳线帽连接到3.3V,使芯片从SRAM中启动。

51.4.2 软件设计

本实验的工程名称为"设置读写保护与解除",学习时请打开该工程配合阅读,它是从"RAM调试—多彩流水灯"工程改写而来的。为了方便展示及移植,我们把操作内部FLASH相关的代码都编写到"internalFlash_reset.c"及"internalFlash_reset.h"文件中,这些文件是我们自己编写的,不属于标准库的内容,可根据您的喜好命名文件。

1.    主要实验

(1)    学习配置扇区写保护;

(2)    学习配置PCROP保护;

(3)    学习配置读保护级别;

(4)    学习如何恢复选项字节到默认配置;

2.    代码分析
配置扇区写保护

我们先以代码清单 514中的设置与解除写保护过程来学习如何配置选项字节。

代码清单 514 配置扇区写保护

1

2 #define FLASH_WRP_SECTORS (OB_WRP_Sector_0|OB_WRP_Sector_1)

3 __IO uint32_t SectorsWRPStatus = 0xFFF;

4

5 /**

6 * @brief WriteProtect_Test,普通的写保护配置

7 * @param 运行本函数后会给扇区FLASH_WRP_SECTORS进行写保护,再重复一次会进行解写保护

8 * @retval None

9 */

10 void WriteProtect_Test(void)

11 {

12 FLASH_Status status = FLASH_COMPLETE;

13 {

14 /* 获取扇区的写保护状态 */

15 SectorsWRPStatus = FLASH_OB_GetWRP() & FLASH_WRP_SECTORS;

16

17 if (SectorsWRPStatus == 0x00)

18 {

19 /* 扇区已被写保护,执行解保护过程*/

20

21 /* 使能访问OPTCR寄存器 */

22 FLASH_OB_Unlock();

23

24 /* 设置对应的nWRP位,解除写保护 */

25 FLASH_OB_WRPConfig(FLASH_WRP_SECTORS, DISABLE);

26 status=FLASH_OB_Launch();

27 /* 开始对选项字节进行编程 */

28 if (status != FLASH_COMPLETE)

29 {

30 FLASH_ERROR("对选项字节编程出错,解除写保护失败,status = %x",status);

31 /* User can add here some code to deal with this error */

32 while (1)

33 {

34 }

35 }

36 /* 禁止访问OPTCR寄存器 */

37 FLASH_OB_Lock();

38

39 /* 获取扇区的写保护状态 */

40 SectorsWRPStatus = FLASH_OB_GetWRP() & FLASH_WRP_SECTORS;

41

42 /* 检查是否配置成功 */

43 if (SectorsWRPStatus == FLASH_WRP_SECTORS)

44 {

45 FLASH_INFO("解除写保护成功!");

46 }

47 else

48 {

49 FLASH_ERROR("未解除写保护!");

50 }

51 }

52 else

53 { /* 若扇区未被写保护,开启写保护配置 */

54

55 /* 使能访问OPTCR寄存器 */

56 FLASH_OB_Unlock();

57

58 /*使能 FLASH_WRP_SECTORS 扇区写保护 */

59 FLASH_OB_WRPConfig(FLASH_WRP_SECTORS, ENABLE);

60

61 status=FLASH_OB_Launch();

62 /* 开始对选项字节进行编程 */

63 if (status != FLASH_COMPLETE)

64 {

65 FLASH_ERROR("对选项字节编程出错,设置写保护失败,status = %x",status);

66 while (1)

67 {

68 }

69 }

70 /* 禁止访问OPTCR寄存器 */

71 FLASH_OB_Lock();

72

73 /* 获取扇区的写保护状态 */

74 SectorsWRPStatus = FLASH_OB_GetWRP() & FLASH_WRP_SECTORS;

75

76 /* 检查是否配置成功 */

77 if (SectorsWRPStatus == 0x00)

78 {

79 FLASH_INFO("设置写保护成功!");

80 }

81 else

82 {

83 FLASH_ERROR("设置写保护失败!");

84 }

85 }

86 }

87 }

本函数分成了两个部分,它根据目标扇区的状态进行操作,若原来扇区为非保护状态时就进行写保护,若为保护状态就解除保护。其主要操作过程如下:

    调用FLASH_OB_GetWRP函数获取目标扇区的保护状态若扇区被写保护,则开始解除保护过程,否则开始设置写保护过程;

    调用FLASH_OB_Unlock解锁选项字节的编程;

    调用FLASH_OB_WRPConfig函数配置目标扇区关闭或打开写保护;

    调用FLASH_OB_Launch函数把寄存器的配置写入到选项字节;

    调用FLASH_OB_GetWRP函数检查是否配置成功;

    调用FLASH_OB_Lock禁止修改选项字节。

配置PCROP保护

配置PCROP保护的过程与配置写保护过程稍有区别,见代码清单 515

代码清单 515 配置PCROP保护(internalFlash_reset.c文件)

1

2 /**

3 * @brief SetPCROP,设置PCROP位,用于测试解锁

4 * @note 使用有问题的串口ISP下载软件,可能会导致PCROP位置1

5 导致无法给芯片下载程序到FLASH,本函数用于把PCROP位置1

6 模拟出无法下载程序到FLASH的环境,以便用于解锁的程序调试。

7 若不了解PCROP位的作用,请不要执行此函数!!

8 * @param None

9 * @retval None

10 */

11 void SetPCROP(void)

12 {

13

14 FLASH_Status status = FLASH_COMPLETE;

15

17

18 FLASH_INFO();

19 FLASH_INFO("正在设置PCROP保护,请耐心等待...");

20 //选项字节解锁

21 FLASH_OB_Unlock();

22

23 //设置为PCROP模式

24 FLASH_OB_PCROPSelectionConfig(OB_PcROP_Enable);

25 //设置扇区0进行PCROP保护

26 FLASH_OB_PCROPConfig(OB_PCROP_Sector_10,ENABLE);

27 //把寄存器设置写入到选项字节

28 status =FLASH_OB_Launch();

29

30 if (status != FLASH_COMPLETE)

31 {

32 FLASH_INFO("设置PCROP失败!");

33 }

34 else

35 {

36 FLASH_INFO("设置PCROP成功!");

37

38 }

39 //选项字节上锁

40 FLASH_OB_Lock();

41 }

该代码在解锁选项字节后,调用FLASH_OB_PCROPSelectionConfig函数把SPMOD寄存器位配置为PCROP模式,接着调用FLASH_OB_PCROPConfig函数配置目标保护扇区。

恢复选项字节为默认值

当芯片被设置为读写保护或PCROP保护时,这时给芯片的内部FLASH下载程序时,可能会出现图 516和图 517的擦除FLASH失败的错误提示。

516 擦除失败提示

517 擦除进度条卡在开始状态

只要不是把读保护配置成了级别2保护,都可以使用SRAM启动运行代码清单 516中的函数恢复选项字节为默认状态,使得FLASH下载能正常进行。

代码清单 516 恢复选项字节为默认值

1 // @brief OPTCR register byte 0 (Bits[7:0]) base address

2 #define OPTCR_BYTE0_ADDRESS ((uint32_t)0x40023C14)

3 //@brief OPTCR register byte 1 (Bits[15:8]) base address

4 #define OPTCR_BYTE1_ADDRESS ((uint32_t)0x40023C15)

5 //@brief OPTCR register byte 2 (Bits[23:16]) base address

6 #define OPTCR_BYTE2_ADDRESS ((uint32_t)0x40023C16)

7 // @brief OPTCR register byte 3 (Bits[31:24]) base address

8 #define OPTCR_BYTE3_ADDRESS ((uint32_t)0x40023C17)

9 // @brief OPTCR1 register byte 0 (Bits[7:0]) base address

10 #define OPTCR1_BYTE2_ADDRESS ((uint32_t)0x40023C1A)

11

12 /**

13 * @brief InternalFlash_Reset,恢复内部FLASH的默认配置

14 * @param None

15 * @retval None

16 */

17 int InternalFlash_Reset(void)

18 {

19 FLASH_Status status = FLASH_COMPLETE;

20

21 /* 使能访问选项字节寄存器 */

22 FLASH_OB_Unlock();

23

24 /* 擦除用户区域 (用户区域指程序本身没有使用的空间,可以自定义)**/

25 /* 清除各种FLASH的标志位 */

26 FLASH_ClearFlag(FLASH_FLAG_EOP | FLASH_FLAG_OPERR | FLASH_FLAG_WRPERR |

27 FLASH_FLAG_PGAERR | FLASH_FLAG_PGPERR|FLASH_FLAG_PGSERR);

28

29 FLASH_INFO("\r\n");

30 FLASH_INFO("正在准备恢复的条件,请耐心等待...");

31

32 //确保把读保护级别设置为LEVEL1,以便恢复PCROP寄存器位

33 //PCROP寄存器位从设置为0时,必须是读保护级别由LEVEL1转为LEVEL0时才有效,

34 //否则修改无效

35 FLASH_OB_RDPConfig(OB_RDP_Level_1);

36

37 status=FLASH_OB_Launch();

38

39 status = FLASH_WaitForLastOperation();

40

41 //设置为LEVEL0并恢复PCROP

42

43 FLASH_INFO("\r\n");

44 FLASH_INFO("正在擦除内部FLASH的内容,请耐心等待...");

45

46 //关闭PCROP模式

47 FLASH_OB_PCROPSelectionConfig(OB_PcROP_Disable);

48 FLASH_OB_RDPConfig(OB_RDP_Level_0);

49

50 status =FLASH_OB_Launch();

51

52 //设置其它位为默认值

53 (*(__IO uint32_t *)(OPTCR_BYTE0_ADDRESS))=0x0FFFAAE9;

54 (*(__IO uint16_t *)(OPTCR1_BYTE2_ADDRESS))=0x0FFF;

55 status =FLASH_OB_Launch();

56 if (status != FLASH_COMPLETE)

57 {

58 FLASH_ERROR("恢复选项字节默认值失败,错误代码:status=%X",status);

59 }

60 else

61 {

62 FLASH_INFO("恢复选项字节默认值成功!");

63 }

64 //禁止访问

65 FLASH_OB_Lock();

66

67 return status;

68 }

这个函数进行了如下操作:

    调用FLASH_OB_Unlock解锁选项字节的编程;

    调用FLASH_ClearFlag函数清除所有FLASH异常状态标志;

    调用FLASH_OB_RDPConfig函数设置为读保护级别1,以便后面能正常关闭PCROP模式;

    调用FLASH_OB_Launch写入选项字节并等待读保护级别设置完毕;

    调用FLASH_OB_PCROPSelectionConfig函数关闭PCROP模式;

    调用FLASH_OB_RDPConfig函数把读保护级别降为0;

    调用FLASH_OB_Launch定稿选项字节并等待降级完毕,由于这个过程需要擦除内部FLASH的内容,等待的时间会比较长;

    直接操作寄存器,使用"(*(__IO uint32_t *)(OPTCR_BYTE0_ADDRESS))=0x0FFFAAE9;"和"(*(__IO uint16_t *)(OPTCR1_BYTE2_ADDRESS))=0x0FFF;"语句把OPTCR及OPTCR1寄存器与选项字节相关的位都恢复默认值;

    调用FLASH_OB_Launch函数等待上述配置被写入到选项字节;

    恢复选项字节为默认值操作完毕。

main函数

最后来看看本实验的main函数,见。代码清单 517

代码清单 517 main函数

1 /**

2 * @brief 主函数

3 * @param

4 * @retval

5 */

6 int main(void)

7 {

8 /* LED 端口初始化 */

9 LED_GPIO_Config();

10 Debug_USART_Config();

11 LED_BLUE;

12

13 FLASH_INFO("本程序将会被下载到STM32的内部SRAM运行。");

14 FLASH_INFO("下载程序前,请确认把实验板的BOOT0BOOT1引脚都接到3.3V电源处!!");

15

16 FLASH_INFO("\r\n");

17 FLASH_INFO("----这是一个STM32芯片内部FLASH解锁程序----");

18 FLASH_INFO("程序会把芯片的内部FLASH选项字节恢复为默认值");

19

20

21 #if 0 //工程调试、演示时使用,正常解除时不需要运行此函数

22 SetPCROP(); //修改PCROP位,仿真芯片被锁无法下载程序到内部FLASH的环境

23 #endif

24

25 #if 0 //工程调试、演示时使用,正常解除时不需要运行此函数

26 WriteProtect_Test(); //修改写保护位,仿真芯片扇区被设置成写保护的的环境

27 #endif

28

30

31 /*恢复选项字节到默认值,解除保护*/

32 if(InternalFlash_Reset()==FLASH_COMPLETE)

33 {

34 FLASH_INFO("选项字节恢复成功,请把BOOT0BOOT1引脚都连接到GND");

35 FLASH_INFO("然后随便找一个普通的程序,下载程序到芯片的内部FLASH进行测试");

36 LED_GREEN;

37 }

38 else

39 {

40 FLASH_INFO("选项字节恢复成功失败,请复位重试");

41 LED_RED;

42 }

43

45

46 while (1);

47 }

main函数中,主要是调用了InternalFlash_Reset函数把选项字节恢复成默认值,程序默认时没有调用SetPCROPWriteProtect_Test函数设置写保护,若您想观察实验现象,可修改条件编译的宏,使它加入到编译中。

3.    下载测试

把开发板的BOOT0和BOOT1引脚都使用跳线帽连接到3.3V电源处,使它以SRAM方式启动,然后用USB线连接开发板"USB TO UART"接口跟电脑,在电脑端打开串口调试助手,把编译好的程序下载到开发板并复位运行,在串口调试助手可看到调试信息。程序运行后,请耐心等待至开发板亮绿灯或串口调试信息提示恢复完毕再给开发板断电,否则由于恢复过程被中断,芯片内部FLASH会处于保护状态。

芯片内部FLASH处于保护状态时,可重新下载本程序到开发板以SRAM运行恢复默认配置。

51.5 每课一问

1.    分别设置芯片为读保护级别1、写保护及PCROP保护,然后给芯片的内部FLASH下载程序,观察实验现象。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/254286.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

电脑(伪)大神装B必备,来学几个windows脚本命令

作为一个经常使用电脑的人来说,掌握一些脚本命令以备在别人面前装B是很有必要的。看完这篇文章,你就学会了基础常用的装B命令,让装B轻松自如。 其实脚本命令没什么新鲜的,属于低级的操作,所谓的电脑大神也是伪大神。想…

通过腾讯云cdn进行壁外调查

引言 之前7月份叔叔更新了播放器,我以前是通过阿里云PHP函数的形势代理番剧的,速度不是很快但是能用,没有CDN这个这么麻烦,但是叔叔更新播放器之后导致PHP函数失效,照着教程改了半天也不行,于是就参考了这…

C4D R26 渲染学习笔记 建模篇(3):生成器

文章目录 前文回顾介绍篇建模篇 生成器介绍生成器变形器搭配举例 生成器详细介绍细分曲面布料曲面 未完待续 前文回顾 介绍篇 C4D R26 渲染学习笔记(1):C4D版本选择和初始UI框介绍 C4D R26 渲染学习笔记(2)&#xff…

第51章 设置FLASH的读写保护及解除

转载地址:https://www.cnblogs.com/firege/ 51.1 选项字节与读写保护 在实际发布的产品中,在STM32芯片的内部FLASH存储了控制程序,如果不作任何保护措施的话,可以使用下载器直接把内部FLASH的内容读取回来,得到bin或…

C/C++中程序内存区域划分大总结

此图来自http://www.blog.163.com/tao198352__4232/blog/static/8502064520105611157897/ 程序由代码和数据组成,其中代码存储在代码区中,数据根据类型的不同存储在不同的区域中。本文分别介绍了C和C中内存区域的划分。 C作为一款C语言的升级版本&#…

如何更优雅地使用 bilibili(b站)

苏生不惑第125 篇原创文章,将本公众号设为星标,第一时间看最新文章。 b站是我每天都要逛的网站,之前也写过了 那些我关注的 b 站 up 主 bilibili(b站)升级到BV号了,还想用av号怎么办? 那些你可能不知道的 bilibili 奇技…

B站视频下载助手使用教程

项目需要,在B站搜到对应数据,检索下载方式保存,供后续使用。工具chrome浏览器离线插件。 1、获取最新版本 bilibili哔哩哔哩B站下载助手-官方网站 2、如何手动安装crx step1、把下载的 crx 文件扩展名改为 zip,如 bilibili-hel…

Rust 笔记:有限状态机原理/状态模式 及其 在Rust 编程中的应用

Rust 笔记、设计模式 有限状态机原理及其在Rust 编程中的应用 作者:李俊才 (jcLee95):https://blog.csdn.net/qq_28550263?spm1001.2101.3001.5343 邮箱 :291148484163.com 本文地址:https://blog.csdn.ne…

Java程序设计入门教程-- switch选择语句

switch选择语句 情形 虽然if…else语句通过嵌套可以处理多分支的情况,但分支不宜太多,在Java语言中,提供了switch语句可以直接、高效地处理多分支选择的情况。 格式 switch (表达式) { case 常量表达式1&#x…

chatgpt赋能python:Python中按钮的位置摆放

Python中按钮的位置摆放 在Python应用程序中,按钮是常见的交互元素之一。按钮通常用于响应用户的操作,例如提交表单或执行某些功能。然而,在设计应用程序时,按钮的位置是一个重要的问题,因为它将直接影响用户体验和应…

MVC 接收不到参数? —— 看我如何给你安排得明明白白

文章结构 问题背景:问题处理总结 问题背景: 现有如下代码: PostMapping(value "/payment/create") ResponseBody public CommonResult create(Payment payment) {}乍眼看去是不是很好,至少没啥问题很自然&#xff0c…

js中的事件委托

1、什么是事件委托? ​ 事件委托, 他还有一个名字叫做事件代理,是JavaScript中常用的绑定事件的常用技巧,顾名思义,事件代理即是把原本需要绑定在子元素的响应事件委托给父元素,让父元素担当事件监听的职务…

有奖励!2023陕西省首台(套)重大技术装备产品项目申报条件、认定材料

本文整理了2023陕西省首台(套)重大技术装备产品项目申报条件,认定材料等相关内容,感兴趣的朋友快跟小编一起来看看吧! 一、重点支持方向及领域 重点支持方向及领域:高档工业母机、电力装备、大型矿山和冶金…

Windows下怎么更改文件扩展名

Win10如何修改文件扩展名 1、首先,我们打开我们的电脑,然后我们双击电脑上的一个文件夹; 2、进入文件夹之后,我们可以看到一个文件,此时的文件后缀名是不显示的,win10系统默认的是不显示后缀名的; 3、我们点击工具…

文件的拓展名(后缀名)怎么快速修改

最近有很多朋友在问,如何快速改名,比如将多个文件的后缀名进行修改,该如何操作呢?不知道的宝贝们,下面请随小编一起来试试吧,希望能给大家带来帮助。 所需工具 文件素材若干 操作步骤 将需要修改的所有文…

【文章学习系列之模型】SCALEFORMER

本章内容 文章概况模型结构主要方法多尺度框架跨尺度标准化模型输入编码损失函数 实验结果消融实验跨尺度标准化自适应损失函数 总结 文章概况 《SCALEFORMER: ITERATIVE MULTI-SCALE REFINING TRANSFORMERS FOR TIME SERIES FORECASTING》是2023年发表于ICLR上的一篇论文。作…

chatgpt赋能python:Python中按下某个按键的实现方法

Python中按下某个按键的实现方法 Python是一种广泛应用于各种领域的高级编程语言,可以用于编写各种类型的应用程序和工具。其中,它在游戏开发方面有着广泛的应用。而在游戏开发过程中,按键响应是至关重要的一个组成部分。本文将带领读者了解…

【第二节】变量和简单的数据类型

本节将结束python当中的变量和一部分数据类型(字符串和数字),并且介绍一些对应的方法 变量 在上一节的最后我们打印出了hello world print(hello world!)这次我们使用下面的代码实现一样的功能 message = hello world! print(message)可以看到 打印结果一模一样,而我们…

关于VUE3的一些基础知识点笔记

关于VUE3的一些基础知识点笔记 VUE3 vue2与vue3的一个重要区别是vue2是选项式API(composition API)而vue3是组合式API(option API)。 组合式API就是所有逻辑都在setup函数中,使用ref、watch等函数来组织代码&#xff…

easyX实践上手操作小项目

easyX实践上手操作小项目 效果展示主菜单的装饰玩法介绍界面开始游戏界面制作团队界面排行榜界面注:main()函数拓展数据库小结 这里我们学习过easyX的基础知识后,看看是否能实践操作一下,制作一个属于自己的游戏界面呢? 基础知识…