STM32各个系列时钟调高时出现异常案例

发布者:sigma28最新更新时间:2023-09-22 来源: elecfans关键字:STM32 手机看文章 扫描二维码
随时随地手机看文章

STM32用户反馈,使用STM32F103内部时钟,把系统时钟配置成64MHz单片机就不跑了,配置成36MHz程序就正常妥妥的,频率稍高点就容易导致死机。他贴出的代码如下:void RCC_Configuration(void)

{

RCC_DeInit();//将外设 RCC寄存器重设为缺省值

RCC_HSICmd(ENABLE);//使能HSI

while(RCC_GetFlagStatus(RCC_FLAG_HSIRDY) == RESET);//等待HSI使能成功

//FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

//FLASH_SetLatency(FLASH_Latency_2);

RCC_HCLKConfig(RCC_SYSCLK_Div1);

RCC_PCLK1Config(RCC_HCLK_Div2);

RCC_PCLK2Config(RCC_HCLK_Div1);

//设置 PLL 时钟源及倍频系数

RCC_PLLConfig(RCC_PLLSource_HSI_Div2, RCC_PLLMul_16);

。。。。。。

结合他的问题描述及他贴出来的代码,大致可以判断很可能是因为他屏蔽了指令预取和flash读取等待延迟的参数配置而导致的。即上面两条红色标注出来的代码。

后来我明确地提醒他这点后,他似乎并没及时反应过来,还折腾了几下才开启了上述配置,问题最终得以解决。

其实,关于这个问题经常有人遇到,尤其是那些基于STM32标准固件库进行开发或自行创建工程的新手更容易碰到这个问题。主要原因是因为他们对上述两行代码的功能不了解,导致有意或无意的将库例程中相关代码屏蔽掉无视掉而不做配置、或者配置不正确。

这里将这个问题再次分享出来,并对那两行代码简单做些解释。希望更多人对此有所知晓,少在这个地方走弯路。

这句 FLASH_PrefetchBufferCmd(); 用来开启或禁用flash指令预取功能。

现有STM32各个系列都是基于ARM cortexM内核的微处理器,它们采用多级流水线的哈佛结构,即一条指令的执行分割为几个阶段,如取指、译码、执行等,使得当前指令的取指操作完成后就可以开始后续指令的取指、译码等操作,程序指令就这样像流水一样执行下去,大大提高了指令的执行效率。

具体到STM32各系列单片机,这个指令预取功能的开启或关闭可以软件配置,一般配置为开启。要注意的是,复位后不同的系列该功能有的默认为开启有的则默认为关闭。比方STM32F1系列的flash指令预取功能就是默认打开的,当然你也可以关闭。其中,明确要求打开的情景就是当那个AHB时钟预分频系数不等于1时。

再比如STM32F4系列,它的指令预取功能在芯片复位后是默认关闭的,你可以自行打开。但明确要求关闭的场景就是芯片的供电电压低于2.1V时。

其实,STM32F4的预取功能与STM32F1不尽一样,STM32F4、STM32F2、STM32L4、STM32F7等系列芯片使用了ST的专利技术ART存储加速器【Adaptive real-time memory accelerator】。该加速器使用指令预取队列和分支跳转缓存技术,从而提高 Flash 程序代码执行速度,使得CPU即使在其最高主频下也能完美实现0 等待执行flash程序指令。

上面大致介绍了指令预取功能,预取主要是为了实现指令读取和执行的高效性。具体细节请参考STM32和ARM cortex内核的相关技术手册。

我们知道CPU的运行速度可调、可以很快,通常使用高速总线访问FLASH接口控制器,FLASH控制器收到来自CPU的取指指令后然后去读取相应地址的指令或数据。Flash控制器自身的读取速度相比CPU的高速请求来说可能会出现滞后,往往需要CPU做相应的延时等待。为了让CPU准确及时读取Flash 数据,我们须根据 CPU 时钟频率以及器件供电情况在Flash存取控制寄存器 (FLASH_ACR)中正确地编程等待周期数(LATENCY),类似上面提到另外一句代码:

FLASH_SetLatency(FLASH_Latency_n);

这里的等待周期数视不同的STM32系列也各有差异,不妨以STM32F4为例:

下面是STM32F4系列部分产品线的LATENCY设置的表格。从表格中可以看出LATENCY参数的设置与CPU的时钟、电源电压都有关系。另外,当电源电压在2.1V以下时要关闭预取。

在设置上面的等待周期参数时,选择合适的就好。不难理解,设置太大了影响CPU性能的充分发挥,太小了容易导致异常。

具体回到开头所提案例,它出现死机问题极可能是因为没有合理配置等待周期参数导致异常,因为它屏蔽了例程中那两句配置代码,即使用其默认功能,对于STM32F1,指令预取功能默认为开启。而STM32F1系列芯片的latency默认值即为0,无等待。这样的话,当他把时钟调高到一定程度时出现死机就不难理解了。

另外,当他反馈时钟调高产生异常时,我还给他提醒了注意检查VDDA的电源情况。我碰到有人遇到因VDDA没接好使得PLL不正常的情况。我们知道,对于STM32芯片,调高其工作时钟,往往借助于锁相环。而PLL的供电来自VDDA,如果PLL没有被正常供电,也是个非常隐蔽的麻烦。曾经有个客户为此折腾好久,才愿沉下心来检查其“坏品”的电源脚,结果发现有个VDDA脚虚焊。一直以芯片低频没问题,频率高了就异常为由怀疑芯片品质问题而耽误时间。

最后给点建议,做STM32开发的话,尤其是新手,如果参照ST的官方例程的话,有些配置在没看懂的情况下不要轻易屏蔽或修改。我碰到多个类似本案随意屏蔽例程中的初始化配置代码或断言代码出现异常,自己又找不到方向的。另外,尽可能使用ST官方的stm32cubeMx图形配置工具做基本的配置,通过它来生成初始化配置文件,这样方便省事很多。当然,即使使用STM32CUBEMX配置也不是万能的。比方:曾经有人使用STM32F0开发产品,用CUBEMX配置初始化文件,刚开始配置时时钟选择得比较低, STM32CubeMx自然根据他选择的时钟做了相关参数配置。后来他自己在用户代码里手动调高了时钟,而不知相应调整跟FLASH读取等待有关的参数,也是发生跟本案同样的情况。


关键字:STM32 引用地址:STM32各个系列时钟调高时出现异常案例

上一篇:为什么STM32单片机编程时需要使能时钟
下一篇:使用STM32WB设计BLE应用时影响功耗的内容

推荐阅读最新更新时间:2024-11-04 16:12

stm32中编码器模式读出“负数”的问题
最近在调试平衡小车,在网上找到的关于直流电机编码器的使用源码。查看stm32使用手册可以看到如下的配置,和图片。基本上程序也是这样去设置的相关寄存器的。 如果TI1和TI2分别接电机的A相和B相的话,那么,当电机正转的时候,如下图计数器回想上计数,反转的时候会向下计数,注意了这个向下计数并不会出现负的值,依旧是从(0-ARR)计数。 这种模式的好处:1.上升沿和下降沿都会计数,所以被软件4倍频了。2.当某一相有毛刺的时候,计数器会在硬件上停止计数。 下面看网上找到的代码。 int Read_Encoder(u8 TIMX) { int Encoder_TIM; switch(TIMX)
[单片机]
STM32 Cotex-M3处理器系列编程】串口调试
#include stm32f10x.h //#include stm32f10x_lib.h void Delay(unsigned int x); void UART_Init(void); int main(void) { while (1) { Delay(300000); UART_Init(); //初始化串口 USART_SendData(USART1,0x1A); //从串口发送数据到计算机 while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);//RESET是0,等待发送完毕
[单片机]
STM32系统初始化函数
keil是两家私人公司联合运营,美国和德国 2005年被ARM公司收购,所有他对ARM内核支持应该是很棒的 ARM为stm32的芯片内核方案,所以keil对stm的芯片支持也很到位 STM32F10x_StdPeriph_Lib_V3.5.0 这个是ST公司专门为10x系列提供的软件开发固件库 首先打开KEIL软件,当前用的是v4.10 建立好工程后,点击魔术棒工具 options for target 在C/C++选项,在符号预处理器里定义三个符号 USE_STDPERIPH_DRIVER, 允许使用标准外设驱动 ------------------基本不变 STM32F10X_CL, 主芯片为互联网型芯片,105
[单片机]
STM32输出与输入概念区别
最近在看数据手册的时候,发现在Cortex-M3里,对于GPIO的配置种类有8种之多: (1)GPIO_Mode_AIN 模拟输入 (2)GPIO_Mode_IN_FLOATING 浮空输入 (3)GPIO_Mode_IPD 下拉输入 (4)GPIO_Mode_IPU 上拉输入 (5)GPIO_Mode_Out_OD 开漏输出 (6)GPIO_Mode_Out_PP 推挽输出 (7)GPIO_Mode_AF_OD 复用开漏输出 (8)GPIO_Mode_AF_PP 复用推挽输出 对于刚入门的新手,我想这几个概念是必须得搞清楚的,平时接触的最多的也就是推挽输出、开漏输出、上拉输入这三种,但一直未曾对这些做过归纳。因此,在这里做一
[单片机]
<font color='red'>STM32</font>输出与输入概念区别
stm32—按键(双模式,双函数选择)
不使用CubeMX 一、新建方法不说了,记得将key和led文件封装进去 二、ked.c代码: 1、按键初始化部分 2、第一个按键函数 注意: 这个函数提供两种模式(是否连按) 注意这个函数存在响应优先级 3、第二个按键函数 三、key.h部分: 注意: 有两种方式读取电位,一种用位带操作读取,一种用HAL库读取 声明了.c文件里的三个函数,宏定义了一些特殊常量 四、main.c中: 通过这个KEY_Scan();内的数字来选择扫描模式 0和1 使用CubeMX 一、首先记得先把delay函数和KEY函数封装好 二、端口初始化部分都在gpio.c里面
[单片机]
<font color='red'>stm32</font>—按键(双模式,双函数选择)
STM32 BIT_BAND 位段位带别名区使用入门
1. 什么是位段、位带别名区? 2. 它有什么好处? 答1: 是这样的,记得MCS51吗? MCS51就是有位操作,以一位(BIT)为数据对象的操作, MCS51可以简单的将P1口的第2位独立操作: P1.2=0;P1.2=1 ; 就是这样把P1口的第三个脚(BIT2)置0置。 而现在STM32的位段、位带别名区就为了实现这样的功能。 对象可以是SRAM,I/O外设空间。实现对这些地方的某一位的操作。 它是这样的。在寻址空间(32位地址是 4GB )另一地方,取个别名区空间,从这地址开始处,每一个字(32BIT) 就对应SRAM或I/O的一位。 这样呢,1MB SRAM就
[单片机]
<font color='red'>STM32</font> BIT_BAND 位段位带别名区使用入门
基于stm32的电子秤方案
  什么是电子秤   电子秤(英文名:electronic balance)是衡器的一种,是利用胡克定律或力的杠杆平衡原理测定物体质量的工具。按结构原理可分为机械秤、电子秤、机电结合秤三大类。   电子秤主要由承重系统(如秤盘、秤体)、传力转换系统(如杠杆传力系统、传感器)和示值系统(如刻度盘、电子显示仪表)3部分组成。      基于stm32的电子秤方案大全(一)   基于STM32的多功能电子秤设计   题目要求:   1、基本部分:   (1)能用键盘设置单价,称重后能同时显示重量、单价和总额;电子计价秤:最大称重为10.000公斤,重量误差不大于±0.1%;   (2)具有TFT液晶屏显示,显示重量、单价、总额等信息
[单片机]
基于<font color='red'>stm32</font>的电子秤方案
STM32芯片存储器映射概述
STM32系统构架 当你拿到一款芯片后,怎样可以快速了解到它具有哪些功能?了解单片机的系统架构是使用款芯片的基础,这些信息在芯片手册上都会有详细的说明,并且它们会被安排在最前面的章节,目的就是让用户最开始就能够关注到该款芯片具有哪些功能,特点。 STM32F103系统结构 从上图我们可以了解到STM32F103系列的主要系统组成单元: CPU核:使用ARM Cortex-M3内核; 存储器: STM32F103系列微控制器采用经典的哈佛架构,程序存储和数据存储采用独立的存储器空间; ICode:指令代码,Flash存储器用于存储程序; DCode:数据代码,SRAM用于存储数据; 外设: GPIO;
[单片机]
<font color='red'>STM32</font>芯片存储器映射概述
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved