1、时钟树框图系统时钟的选择在启动时执行,即程序在上电复位时,先运行启动文件startup_stm32f0xx.s,该汇编文件中先进行了系统初始化,在系统初始化的时候进行系统时钟(SYSCLK)的选择与配置,当系统初始化完成后,才运行main函数。在复位后,内部8MHz的RC振荡器(HSI)被选为默认的CPU时钟。用户也可以选择4-32MHz的外部时钟(HSE)作为系统时钟,选择HSE作为系统时钟时,标准库函数需要用户自己进行修改,HAL库的话用户可以直接配置。当 HSE 故障的时候, 且选择 PLL 时钟作为SYSCLK, 那么当 HSE 故障的时候,不仅HSE 不能使用,连 PLL 也会被关闭,这个时候系统会自动切换 HSI 作为系统时钟,此时SYSCLK=HSI=8M,如果没有开启 CSS 和 CSS 中断的话,那么整个系统就只能在低速率运行,这是系统跟瘫痪没什么两样。如果开启了 CSS 功能的话,那么可以当 HSE 故障时,在CSS 中断里面采取补救措施,使用 HSI,并把系统时钟设置为更高的频率。允许应用程序通过几个分频器来配置 AHB 和 APB 的频率,AHB 和 APB 的最高频率为48MHz。上电复位时,启动文件运行如下程序:- ; Reset handler routine 复位处理程序
- Reset_Handler PROC
- EXPORT Reset_Handler [WEAK]
- IMPORT __main
- IMPORT SystemInit
- LDR R0, =SystemInit
- BLX R0
- LDR R0, =__main
- BX R0
- ENDP
复制代码 上电启动过程中,先调用了SystemInit()函数,再进入main()函数。SystemInit()函数在文件system_stm32f0xx.c中,它的作用是在单片机上电复位的过程中,会来调用这个函数设置系统时钟SYSCLK,默认STM32F0系列单片机时钟最大设置为48MHz,所以系统时钟默认是已经配置好的,无需用户自己进行配置,除非有特殊需要,用户可以进行修改。2、时钟系统特性系统时钟(SYSCLK)有以下三种选择:HSE(外部高速晶振):晶振范围(4MHz~32MHz)可被外部时钟源旁路,意思就是不经过分频和倍频直接将外部时钟源选为系统时钟 SYSCLK,外部时钟信号范围1MHz~32MHzHSI(内部高速RC振荡器):8MHz,出厂已校准到±1%的精度PLL经过2,3,… 16倍频(所有输出的最小频率16MHz)其他时钟源:LSI(内部低速RC振荡器):~40KHzLSE(外部低速晶振):32.768kHz可悲外部时钟源旁路,可以直接给RTCCLK提供时钟,最高频率1MHz外部晶振的驱动能力可配置(功耗VS起振能力)HSI14(内部高速14MHz RC振荡器):专用于给ADC模块提供时钟3、时钟参数一览4、时钟配置案例使用标准库进行系统时钟的修改,为什么说是修改,而不是说配置,如上所述,因为程序来到main函数之前,启动文件:startup_stm32f0xx.s 已经调用SystemInit()函数把系统时钟初始化成 48MHz了,SystemInit()在 system_stm32f0xx.c 中定义,如果用户想修改系统时钟,可参照SystemInit()函数中调用的SetSysClock() 函数,自行编写修改程序,官方文件一般不进行直接修改。以下代码为system_stm32f0xx.c原文件中的 SystemInit() 函数和 SetSysClock() 函数,源代码分析:- void SystemInit (void)
- {
- /* Set HSION bit */
- /* HSI时钟使能,开启HSI振荡器 */
- RCC->CR |= (uint32_t)0x00000001;
- #if defined(STM32F051)
- /* Reset SW[1:0], HPRE[3:0], PPRE[2:0], ADCPRE and MCOSEL[2:0] bits */
- /* 复位 系统时钟切换, AHB总线的分频因子, APB总线的分频因子, ADC的时钟频率 和微控制器时钟输出选择 */
- RCC->CFGR &= (uint32_t)0xF8FFB80C;
- #else
- /* Reset SW[1:0], HPRE[3:0], PPRE[2:0], ADCPRE, MCOSEL[2:0], MCOPRE[2:0] and PLLNODIV bits */
- /* 复位 系统时钟切换, AHB总线的分频因子, APB总线的分频因子, ADC的时钟频率 ,微控制器时钟输出选择,微控制器预分频选择,
- PLL左时钟输出时是否要分频 */
- RCC->CFGR &= (uint32_t)0x08FFB80C;
- #endif /* STM32F051 */
- /* Reset HSEON, CSSON and PLLON bits */
- /* HSE时钟使能位, 时钟安全系统使能位 和PLL使能位清0 */
- RCC->CR &= (uint32_t)0xFEF6FFFF;
- /* Reset HSEBYP bit */
- /* 外部高速时钟旁路清0 */
- RCC->CR &= (uint32_t)0xFFFBFFFF;
- /* Reset PLLSRC, PLLXTPRE and PLLMUL[3:0] bits */
- /* PLL输入时钟源位,分频HSE后作为PLL输入,PLL倍频系数清0 */
- RCC->CFGR &= (uint32_t)0xFFC0FFFF;
- /* Reset PREDIV1[3:0] bits */
- /* PREDIV1分频因子清0 */
- RCC->CFGR2 &= (uint32_t)0xFFFFFFF0;
- /* Reset USARTSW[1:0], I2CSW, CECSW and ADCSW bits */
- /* USART1时钟源选择位,I2C时钟源选择位,HDMI CEC时钟源选择位,ADC时钟源选择位清0 */
- RCC->CFGR3 &= (uint32_t)0xFFFFFEAC;
- /* Reset HSI14 bit */
- /* HSI14时钟使能位清0 */
- RCC->CR2 &= (uint32_t)0xFFFFFFFE;
- /* 禁止所有时钟中断 */
- RCC->CIR = 0x00000000;
- /* Configure the System clock frequency, AHB/APBx prescalers and Flash settings */
- /* 配置系统时钟频率,AHB/APBx分频器和闪存设置 */
- SetSysClock();
- }
- ===========================================================================================================================
- static void SetSysClock(void)
- {
- __IO uint32_t StartUpCounter = 0, HSEStatus = 0;
- /* SYSCLK, HCLK, PCLK configuration 系统时钟,AHB总线时钟,APB总线时钟配置-----------*/
- /* Enable HSE 使能HSE */
- RCC->CR |= ((uint32_t)RCC_CR_HSEON);
- /* Wait till HSE is ready and if Time out is reached exit 等待HSE稳定,如果超时,则退出 */
- do
- {
- HSEStatus = RCC->CR & RCC_CR_HSERDY;
- StartUpCounter++;
- } while((HSEStatus == 0) && (StartUpCounter != HSE_STARTUP_TIMEOUT));
- /* 到这边要么是超时了,要么就是HSE稳定了,准备好了,再次判断HSE是否准备好 */
- if ((RCC->CR & RCC_CR_HSERDY) != RESET)
- {
- HSEStatus = (uint32_t)0x01; // HSE状态标识置1(稳定了)
- }
- else
- {
- HSEStatus = (uint32_t)0x00; // HSE状态标识清0(没稳定)
- }
- if (HSEStatus == (uint32_t)0x01) //如果HSE稳定了
- {
- /* Enable Prefetch Buffer and set Flash Latency 启用 Flash 预取缓冲区 和 设置Flash延迟*/
- FLASH->ACR = FLASH_ACR_PRFTBE | FLASH_ACR_LATENCY;
- /* HCLK = SYSCLK AHB 预分频因子设置为 1 分频, HCLK = SYSCLK */
- RCC->CFGR |= (uint32_t)RCC_CFGR_HPRE_DIV1;
- /* PCLK = HCLK APB 预分频因子设置为 1 分频, PCLK = HCLK */
- RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE_DIV1;
- /* PLL configuration = HSE * 6 = 48 MHz 设置 PLL 时钟来源为 HSE(外接了8MHz),设置PLL不分频,设置 PLL 6倍频 */
- RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_PLLSRC | RCC_CFGR_PLLXTPRE | RCC_CFGR_PLLMULL));
- RCC->CFGR |= (uint32_t)(RCC_CFGR_PLLSRC_PREDIV1 | RCC_CFGR_PLLXTPRE_PREDIV1 | RCC_CFGR_PLLMULL6);
- /* Enable PLL PLL使能 */
- RCC->CR |= RCC_CR_PLLON;
- /* Wait till PLL is ready 等待PLL稳定 */
- while((RCC->CR & RCC_CR_PLLRDY) == 0)
- {
- }
- /* Select PLL as system clock source 选择PLL作为系统时钟源*/
- RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_SW));
- RCC->CFGR |= (uint32_t)RCC_CFGR_SW_PLL;
- /* Wait till PLL is used as system clock source 等待PLL作为系统时钟源稳定 */
- while ((RCC->CFGR & (uint32_t)RCC_CFGR_SWS) != (uint32_t)RCC_CFGR_SWS_PLL)
- {
- }
- }
- else
- { /* If HSE fails to start-up, the application will have wrong clock 如果HSE启动失败,应用将使用错误的配置
- configuration. User can add here some code to deal with this error 用户可以在这里添加处理错误的代码 */
- }
- }
复制代码 以上分析了标准库函数中配置系统时钟的程序逻辑,下面介绍当用户需要修改自己的时钟频率时的处理方法,4.1【标准库】的配置方式步骤:1、只要参照 SetSysClock() 函数重新编写一个自己的配置系统时钟的函数2、在main函数中进行调用,调用语句需要放在main函数的函数体第一行,以确保在他初始化前已经重新设置好了自己需要的系统时钟4.2【HAL库】的配置方式步骤:1、配置HSE系统时钟:在 STM32CubeMX 软件中选择 【System Core】→【RCC】→【选择HSE高速时钟】→【点击下拉菜单选择“晶体/陶瓷谐振器”,即电路中外接晶振的】STM32CubeMX软件可以直接生成MDK项目代码,双击打开项目工程,先进行项目编译,然后打开 main.c 查看源代码,找到 **SystemClock_Config();**函数,这个就是STM32CubeMX生成的系统时钟配置函数,代码已全部自动生成,无需用户再次修改任何内容,生成的系统时钟,就是上文配置的外部8MHz的晶振,经过6倍频后,最终生成了48MHz的系统时钟频率。进入函数体内查看源代码2、配置HSI系统时钟,按下图设置CubeMX软件,并生成代码同样打开项目中的main文件进入函数体内查看源代码至此,关于STM32F0系列的系统时钟的配置及修改方法如上所述,本人刚开始写博客,有不好或者错误的地方希望各路大神指教。此博文的主要目的是,作为自己的一个学习笔记,在自己学习的过程中做好相应的记录,方便以后回顾和巩固,也方便有需要的朋友。因为我发现只看一些教学视频的讲解,和动手练习操作一遍是很难学会的,还是要通过做笔记的方式来对知识进行巩固和累积才能帮助我真正的消化和学懂相关知识,谢谢! |