1、【STM32F0系列学习】之—系统时钟(RCC)

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~32MHz
  • HSI(内部高速RC振荡器):8MHz,出厂已校准到±1%的精度
  • PLL经过2,3,… 16倍频(所有输出的最小频率16MHz)

其他时钟源:

  • LSI(内部低速RC振荡器):~40KHz
  • LSE(外部低速晶振):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函数的函数体第一行,以确保在他初始化前已经重新设置好了自己需要的系统时钟

/*========================================= 自定义 HSE 系统时钟 =========================================*/
/************************************************
函数名称 : HSE_SetSysClock
功    能 : 外部高速时钟配置
参    数 : RCC_PLLMul_x:用来设置PLL的倍频因子,在调用的时候形参可以是: RCC_PLLMul_x , x:[2,3,...16],这些宏来源于
			库函数的定义,具体功能是配置了时钟配置寄存器 CFGR 的位 21-18 PLLMUL[3:0],预先定义好倍频因子,方便调用
返 回 值 : 无
作    者 : JayYang
*************************************************/
void HSE_SetSysClock(uint32_t RCC_PLLMul_x){
	__IO uint32_t StartUpCounter = 0, HSE_StartUpStatus = 0;	//定义启动计数参数,以及外部高速时钟启动状态参数
    
    //把RCC外设初始化成复位状态,这句是必须的
    RCC_DeInit();
    
    //使能HSE,开启外部晶振,本人系统中用的是 8MHz
    RCC_HSEConfig(RCC_HSE_ON);
    
    //等待 HSE 启动稳定
    HSE_StartUpStatus = RCC_WaitForHSEStartUp();
    
    //如果 HSE 成功启动并且稳定下来
    if(HSE_StartUpStatus==SUCCESS){
        
    //-----------------------------------------------------------------//
        //使能 FLASH 预存取缓冲区
        FLASH_PrefetchBufferCmd(ENABLE);
        
        // SYSCLK 周期与闪存访问时间的比例设置,这里统一设置成 1
        // 设置成 1 的时候, SYSCLK 低于 24M 也可以工作,如果设置成 0 的时候,
        // 如果配置的 SYSCLK 超出了范围的话,则会进入硬件错误,程序就死了
        // 0: 0 < SYSCLK <= 24M
        // 1: 24< SYSCLK <= 48M
        FLASH_SetLatency(FLASH_Latency_1);
    //-----------------------------------------------------------------//
        //AHB 预分频因子设置为1分频,HCLK = SYSCLK
        RCC_HCLKConfig(RCC_SYSCLK_Div1);
        
        //APB 预分频因子设置为1分频,PCLK = HCLK
        RCC_PCLKConfig(RCC_HCLK_Div1);
    //-----------------------------------------------------------------//
    
    //-----------------设置各种频率主要就是在这里设置-------------------//
        // 设置 PLL 时钟来源为HSE,设置PLL倍频因子
        // PLLCLK = 8MHz * RCC_PLLMul_x
        RCC_PLLConfig(RCC_PLLSource_PREDIV1,RCC_PLLMul_x);
        
        //开启 PLL
        RCC_PLLCmd(ENABLE);
        
        //等待 PLL 稳定
        while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
        
        //当PLL稳定之后,把PLL时钟切换为系统时钟SYSCLK
        RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
    }
    else{
        //当HSE开启失败或者故障的时候,单片机会自动把HSI设置为系统时钟,HSI 是内部的高速时钟,8MHz
        //程序就会来到这里,用户可在这里添加出错的代码处理
        while(1){
        }
    }
}

/*=========================== main.c主函数调用 ===========================*/
int main(void){
	HSE_SetSysClock(RCC_PLLMul_9); 		//设置系统时钟为: 8MHZ * 9 =72MHZ。
}
 
 
/*========================================= 自定义 HSI 系统时钟 =========================================*/
/************************************************
函数名称 : HSI_SetSysClk
功    能 : 外部高速时钟配置
参    数 : RCC_PLLMul_x:用来设置PLL的倍频因子,在调用的时候形参可以是: RCC_PLLMul_x , x:[2,3,...16],这些宏来源于
			库函数的定义,具体功能是配置了时钟配置寄存器 CFGR 的位 21-18 PLLMUL[3:0],预先定义好倍频因子,方便调用
返 回 值 : 无
作    者 : JayYang
*************************************************/
void HSI_SetSysClk( uint32_t RCC_PLLMul_x)
{
	__IO uint32_t HSI_StartUpStatus = 0;		//定义内部高速时钟启动状态
	
	// 把 RCC 外设初始化成复位状态,这句是必须的
	RCC_DeInit();	

	// 使能 HSI 
	RCC_HSICmd(ENABLE);
	
	// 等待 HSI 就绪
	HSI_StartUpStatus = RCC->CR & RCC_CR_HSIRDY;
	
	// 只有 HSI 就绪之后则继续往下执行
	if( HSI_StartUpStatus== RCC_CR_HSIRDY )
	{
	//-----------------------------------------------------------------//
        //使能 FLASH 预存取缓冲区
		FLASH_PrefetchBufferCmd(ENABLE);
        
        // SYSCLK 周期与闪存访问时间的比例设置,这里统一设置成 1
        // 设置成 1 的时候, SYSCLK 低于 24M 也可以工作,如果设置成 0 的时候,
        // 如果配置的 SYSCLK 超出了范围的话,则会进入硬件错误,程序就死了
        // 0: 0 < SYSCLK <= 24M
        // 1: 24< SYSCLK <= 48M
		FLASH_SetLatency(FLASH_Latency_1);
		
        //AHB 预分频因子设置为1分频,HCLK = SYSCLK
		RCC_HCLKConfig(RCC_SYSCLK_Div1);
        
        //APB 预分频因子设置为1分频,PCLK = HCLK
		RCC_PCLKConfig(RCC_HCLK_Div1);
    //-----------------------------------------------------------------//
    
    //-----------------设置各种频率主要就是在这里设置-------------------//
		// 配置 PLLCLK = HSI * RCC_PLLMul_x
        RCC_PLLConfig(RCC_PLLSource_HSI_Div2, RCC_PLLMul_x);
		
        // 使能PLL
		RCC_PLLCmd(ENABLE);
		
		// 等待PLL稳定
		while( RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET );
		
        //当PLL稳定之后,把PLL时钟切换为系统时钟SYSCLK
		RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
		
        //读取时钟切换状态位,确保 PLLCLK 被选为系统时钟
        while( RCC_GetSYSCLKSource() != 0x08 );
	}
	else
    {
		//当HSI开启失败或者故障的时候,程序就会来到这里,用户可在这里添加出错的代码处理
        while(1){
        }
	}
}

/*=========================== main.c主函数调用 ===========================*/
int main(void){
	HSI_SetSysClock(RCC_PLLMul_12); 	//设置系统时钟为: 4MHZ * 12 =48MHZ。
}

4.2【HAL库】的配置方式

步骤:
1、配置HSE系统时钟:在 STM32CubeMX 软件中选择 【System Core】→【RCC】→【选择HSE高速时钟】→【点击下拉菜单选择“晶体/陶瓷谐振器”,即电路中外接晶振的】
RCC-HSE1
RCC-HSE2
RCC-HSE3

RCC-HSE4
STM32CubeMX软件可以直接生成MDK项目代码,双击打开项目工程,先进行项目编译,然后打开 main.c 查看源代码,找到 **SystemClock_Config();**函数,这个就是STM32CubeMX生成的系统时钟配置函数,代码已全部自动生成,无需用户再次修改任何内容,生成的系统时钟,就是上文配置的外部8MHz的晶振,经过6倍频后,最终生成了48MHz的系统时钟频率。
RCC-HSE5
进入函数体内查看源代码
RCC-HSE6

2、配置HSI系统时钟,按下图设置CubeMX软件,并生成代码
RCC-HSI1
RCC-HSI2
同样打开项目中的main文件
RCC-HSI3
进入函数体内查看源代码
RCC-HSI4
至此,关于STM32F0系列的系统时钟的配置及修改方法如上所述,本人刚开始写博客,有不好或者错误的地方希望各路大神指教。此博文的主要目的是,作为自己的一个学习笔记,在自己学习的过程中做好相应的记录,方便以后回顾和巩固,也方便有需要的朋友。因为我发现只看一些教学视频的讲解,和动手练习操作一遍是很难学会的,还是要通过做笔记的方式来对知识进行巩固和累积才能帮助我真正的消化和学懂相关知识,谢谢!

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Geek@Yang

码字不易,来点鼓励~~~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值