STM32F105RCT6通过选项字节设置Flash读保护问题(基于HAL库)

STM32F105RCT6通过选项字节设置Flash读保护问题(基于HAL库)

实现函数如下:


__ramfunc HAL_StatusTypeDef chip_enter_flash_protection(void)
{
    HAL_StatusTypeDef result = HAL_OK;
    FLASH_OBProgramInitTypeDef OptionsBytesStruct = {0};

    HAL_FLASHEx_OBGetConfig(&OptionsBytesStruct);
    if (OptionsBytesStruct.RDPLevel == OB_RDP_LEVEL_0) {
      OptionsBytesStruct.OptionType = OPTIONBYTE_RDP;
      OptionsBytesStruct.RDPLevel = OB_RDP_LEVEL_1;
      HAL_FLASH_Unlock();
      HAL_FLASH_OB_Unlock();
      result = HAL_FLASHEx_OBProgram(&OptionsBytesStruct);
      if (result != HAL_OK) {
          LOG("HAL_FLASHEx_OBProgram != HAL_OK\r\n");
          HAL_FLASH_OB_Lock();
          HAL_FLASH_Lock();
          return result;
      }
      LOG("HAL_FLASH_OB_Launch 1\r\n");
      HAL_FLASH_OB_Launch();
      LOG("HAL_FLASH_OB_Launch 2\r\n");
      HAL_FLASH_OB_Lock();
      HAL_FLASH_Lock();
    }

    return result;
}

其中HAL_FLASH_OB_Launch()函数原型如下 其实就是系统软复位

/**
  * @brief  Launch the option byte loading.
  * @note   This function will reset automatically the MCU.
  * @retval None
  */
void HAL_FLASH_OB_Launch(void)
{
  /* Initiates a system reset request to launch the option byte loading */
  HAL_NVIC_SystemReset();
}

该函数执行结果:

img

我的问题(实验过程中未仿真):
1、为什么执行了HAL_FLASH_OB_Launch()(即软复位)之后,程序并未重新开始运行?并且我按下开发板的reset按键(即硬复位)程序也起不来。
2、为什么我把芯片掉电,重新上电之后程序就可以跑起来了?而且读保护功能也开启了
3、也怀疑是进入了ST的引导程序,使用工具软件STM32CubeProgrammer 通过串口并未检测到连接,应该并未进入ST的引导程序
4、也测试把此函数(chip_enter_flash_protection())放置在RAM运行,但是结果一样

芯片手册看了半天也没看出个啥,哪位朋友解答一下我的疑问,十分感谢!

设置FLash的读写保护及解除
可以借鉴下
https://www.cnblogs.com/yuanqiangfei/p/14469395.html

确保您在软复位之后正确设置程序的入口点,以便从头开始执行。
检查选项字节的配置是否正确,确保已正确设置读保护功能。
如果您怀疑进入了引导模式,可以尝试使用其他调试工具或方法来验证是否进入了引导模式。
如果您遇到问题,您可以尝试咨询STMicroelectronics的技术支持,他们可以提供更具体的帮助和指导。

针对问题一的现象解释:芯片刚开始启动会检查main flash区有没有代码,要是没有代码,就进入bootloader状态,要是有代码就从main flash区运行。
针对问题2的现象解释:芯片首次运行会进行空位检查,此时的empty 标志位(即OBL_LAUNCH)已经置位了1,当使用swd烧写完程序后,无论是软件reset还是芯片的reset管脚复位,都无法把empty标志为清0,芯片会在每次的reset后都进入bootloader状态,而不会进入main flash区。只有断电重启,empty flag才会变成0,此时芯片才会进入main flash区运行。
解决方法就是改芯片默认的启动方式,具体可以参考官方手册。

HAL_FLASH_OB_Launch()(即软复位)之后,程序并未重新开始运行问题,可以试试重新设置选项字节,或者查看芯片的时钟配置是否正确