当前位置: 首页 > news >正文

初学STM32系统时钟设置

  资料来自正点原子

         在学习江科大教程示例的时候默认系统时钟是72MHZ,但是这个系统时钟是怎么过来的呢,通过时钟树以及相关的资料的学习可知,系统时钟它可以是内部RC时钟HSI 8MHZ通过锁相环倍频而来,也可以是外部晶振4-16MHZ通过锁相环倍频而来。但是我们在写程序的过程中好像并没有配置过程。

    通过deepseek得知:

  • STM32 芯片上电后,首先执行启动文件(如startup_stm32f10x.s),该文件会调用 SystemInit() 函数。
  • SystemInit() 是标准库(如 STM32F10x Standard Peripheral Library)提供的函数,其默认配置会将外部 8MHz 晶振通过 PLL 倍频到 72MHz。

但如果我们不想用外部时钟做为系统时钟的来源,而是内部时钟作为系统时钟该怎么设置呢?比如用内部时钟8MHZ设置输出36MHZ的系统时钟

配置函数

#include "stm32f10x.h"
#include "stm32f10x_rcc.h"

void SetSysClockToHSI_36MHz(void) {
   
    
    // 1. 将 RCC 配置复位到默认状态
    RCC_DeInit();
    
    // 2. 使能 HSI(内部 8MHz 时钟)
    RCC_HSICmd(ENABLE);
    
    // 3. 等待 HSI 稳定
    while (RCC_GetFlagStatus(RCC_FLAG_HSIRDY) == RESET);
    
    // 4. 配置 PLL:HSI/2 = 4MHz -> PLL倍频9倍 -> 36MHz
    RCC_PLLConfig(RCC_PLLSource_HSI_Div2, RCC_PLLMul_9);
    
    // 5. 使能 PLL
    RCC_PLLCmd(ENABLE);
    
    // 6. 等待 PLL 就绪
    while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
    
    // 7. 配置总线分频系数(可选)
    RCC_HCLKConfig(RCC_SYSCLK_Div1);    // AHB 时钟 = 36MHz
    RCC_PCLK1Config(RCC_HCLK_Div2);    // APB1时钟 = 18MHz(最大36MHz)
    RCC_PCLK2Config(RCC_HCLK_Div1);     // APB2时钟 = 36MHz
    
    // 8. 切换系统时钟到 PLL
    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
    
    // 9. 等待系统时钟切换成功
    while (RCC_GetSYSCLKSource() != 0x08); // 0x08表示PLL作为系统时钟
}

//int main(void) {
//    // 调用自定义时钟配置函数
//    SetSysClockToHSI_36MHz();
//    
//    // 后续初始化代码(开启外设时钟、GPIO初始化等)
//    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
//    // ... 其他初始化代码
//    
//    while(1) {
//        // 主循环代码
//    }
//}

该配置能设置为内部时钟倍频后得到的36MHZ的系统时钟。

内部时钟设置

  •   HSI(内部高速时钟信号) 的 PLL (锁相环)输入限制

STM32F10x 系列中,当选择 HSI 作为 PLL 时钟源 时,必须对 HSI 进行 2 分频(即 HSI/2 = 4MHz)。可以通过库函数发现没有其它选项
PLL 的倍频系数范围为 2~16 倍,因此最高输出频率:无法通过 HSI 达到 72MHz。最大是64MHZ,因此如果想达到72MHZ必须要使用外部时钟源。

  • APB1最大能达到的速度是36MHZ,因此当系统时钟的速度大于36MHZ的时候,APB1的时钟都需要分频设置一下。
  • 为什么都设置为72MHZ作为内部系统时钟呢,全速USB的系统时钟需要48MHZ,72MHZ通过1.5倍分频后刚好是48MHZ。
  • 为什么不常用内部时钟做系统时钟呢?因为内部时钟温漂大,它的震动频率会随着温度变化幅度大。


以外部晶振作为系统时钟如果不想设置为72MHZ,那么该怎么配置这个函数呢?

#include "stm32f10x.h"
#include "stm32f10x_rcc.h"

void SetSysClockToHSE_72MHz(void) {
    // 复位RCC配置
    RCC_DeInit();
    
    // 使能HSE
    RCC_HSEConfig(RCC_HSE_ON);
    while (RCC_GetFlagStatus(RCC_FLAG_HSERDY) == RESET);

    // 配置PLL:HSE=8MHz → 倍频9 → 72MHz
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
    RCC_PLLCmd(ENABLE);
    while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);

    // 配置总线分频
    RCC_HCLKConfig(RCC_SYSCLK_Div1);    // AHB = 72MHz
    RCC_PCLK1Config(RCC_HCLK_Div2);     // APB1 = 36MHz(最大36MHz)
    RCC_PCLK2Config(RCC_HCLK_Div1);     // APB2 = 72MHz

    // 切换系统时钟到PLL
    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
    while (RCC_GetSYSCLKSource() != 0x08);
}

int main(void) {
    SetSysClockToHSE_72MHz();
    // 后续初始化代码...
}

该函数要放在主函数最前面。可以通过修改参数调整系统时钟比如:

 // 配置PLL:HSE=8/2MHz → 倍频9 → 36MHz
    RCC_PLLConfig(RCC_PLLSource_HSE_Div2, RCC_PLLMul_9);

如此就修改了默认系统时钟为36MHZ


      在学习TIM2的时候发现它是APB1总线上的设备,但是APB1的总线频率是36MHZ,为什么在设置TIM2的时钟频率的时候还是72MHZ呢,这是因为APB1上的定时器时钟又被硬件自动倍频了。使得APB1上的定时器时钟能够允许在较快时钟频率下工作。但是APB1上的其它外设还是在36MHZ的频率下工作。

看图:

关键区别总结

特性APB1 Peripheral Clock (PCLK1)APB1 Time Clock(定时器时钟)
作用对象APB1总线上所有外设的寄存器访问仅APB1总线上的定时器模块
频率决定因素直接由HCLK和APB1预分频系数决定由PCLK1决定,可能自动倍频(×2)
典型场景低速外设通信(如I2C、UART)定时器高精度操作(如PWM、输入捕获)
配置影响修改APB1分频系数直接影响PCLK1频率修改APB1分频系数可能触发定时器时钟倍频

相关文章:

  • Vue 组件 - Slot 内容分发
  • Windows搭建AI大模型应用开发环境以及踩过的坑
  • 软件测试(2):selenium 4.0 特点以及新特性
  • 数据库权限获取
  • MySQL基本查询
  • LeetCode[15]三数之和
  • OpenAI重磅回归开源!首发推理模型不限商用,直面DeepSeek挑战
  • 操作系统高频(六)linux内核
  • 交叉熵损失
  • leetcode25.k个一组翻转链表
  • (二十六)Dart 中泛型的使用与优势
  • WEB安全--SQL注入--无列名注入
  • 本地合并多个仓库,保留Commit历史
  • MyBatis choose when otherwise
  • 算法设计学习2
  • 【FreeRtos】任务调度器可以被挂起吗?
  • 【配电网】基于差分进化算法的含DG配电网无功优化模型
  • python技巧:自动控制高低温箱,通过串口输入命令,生成16进制字符串,并计算CRC16。
  • 4.1-3 模拟器
  • C#调用ACCESS数据库,解决“Microsoft.ACE.OLEDB.12.0”未注册问题
  • 证券日报:降准今日正式落地,年内或还有降准空间
  • 万科再获深铁集团借款,今年已累计获股东借款近120亿元
  • 上海145家博物馆、73家美术馆将减免费开放
  • 优化营商环境,服务上海“五个中心”建设,北外滩有何举措?
  • 人民日报钟声:通过平等对话协商解决分歧的重要一步
  • 从600名外到跻身大满贯,孙发京:走过的路成就了现在的我