STM32 RCC管理
STM32 的RCC看起来还是蛮复杂的,英文全称是Reset Clock Config,复位时钟配置,以下基本网上贴的,整理了一下,非原创哈
一、时钟概述
STM32有多个时钟源,分别是:
HSI:上电默认启动,因精度不高所以先不采用,以后如果需要再使用
HSE:外部高速时钟,系统时钟一般采用它,经过PLL倍频作为系统同时钟
LSE:外部低速时钟,一般专门用于RTC,等到RTC模块时再使用
LSI:内部低速时钟,精度不高,一般用于IWDGCLK
二、配置流程
1.将RCC寄存器重新设置为默认值 RCC_DeInit
2.打开外部高速时钟晶振HSE RCC_HSEConfig(RCC_HSE_ON);
3.等待外部高速时钟晶振工作 HSEStartUpStatus = RCC_WaitForHSEStartUp();
4.设置AHB时钟 RCC_HCLKConfig;
5.设置高速APB2时钟 RCC_PCLK2Config;
6.设置低速速APB1时钟 RCC_PCLK1Config
7.设置PLL RCC_PLLConfig
8.打开PLL RCC_PLLCmd(ENABLE);
9.等待PLL工作 while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)
10.设置系统时钟 RCC_SYSCLKConfig
11.判断是否PLL是系统时钟 while(RCC_GetSYSCLKSource() != 0x08)
至此系统时钟已经配置完成,STM32的每个模块都有自己的时钟,如果要使用某个模块,必须使能这个模块的时钟。使能对应模块的时钟,STM32有个库函数,RCC_APB2PeriphClockCmd(对应模块,ENABLE)/RCC_APB1PeriphClockCmd(对应模块,ENABLE);其中不同的模块有不同的时钟源,下面列举各个模块的时钟源:
AHB2 时钟用于以下资源:
RCC_APB2Periph_AFIO 功能复用 IO时钟
RCC_APB2Periph_GPIOA GPIOA 时钟
RCC_APB2Periph_GPIOB GPIOB 时钟
RCC_APB2Periph_GPIOC GPIOC 时钟
RCC_APB2Periph_GPIOD GPIOD 时钟
RCC_APB2Periph_GPIOE GPIOE 时钟
RCC_APB2Periph_ADC1 ADC1 时钟
RCC_APB2Periph_ADC2 ADC2 时钟
RCC_APB2Periph_TIM1 TIM1 时钟
RCC_APB2Periph_SPI1 SPI1 时钟
RCC_APB2Periph_USART1 USART1 时钟
RCC_APB2Periph_ALL 全部 APB2外设时钟
AHB1 时钟用于以下资源:
RCC_APB1Periph_TIM2 TIM2 时钟
RCC_APB1Periph_TIM3 TIM3 时钟
RCC_APB1Periph_TIM4 TIM4 时钟
RCC_APB1Periph_WWDG WWDG时钟
RCC_APB1Periph_SPI2 SPI2 时钟
RCC_APB1Periph_USART2 USART2 时钟
RCC_APB1Periph_USART3 USART3 时钟
RCC_APB1Periph_I2C1 I2C1 时钟
RCC_APB1Periph_I2C2 I2C2 时钟
RCC_APB1Periph_USB USB 时钟
RCC_APB1Periph_CAN CAN时钟
RTC 时钟来源:
RCC_RTCCLKSource_LSE 选择 LSE 作为 RTC 时钟
RCC_RTCCLKSource_LSI 选择 LSI 作为 RTC 时钟
RCC_RTCCLKSource_HSE_Div128 选择 HSE 时钟频率除以 128 作为 RTC时钟
ADC 时钟来源:
该时钟源自 APB2 时钟(PCLK2)
RCC_PCLK2_Div2 ADC 时钟 = PCLK / 2
RCC_PCLK2_Div4 ADC 时钟 = PCLK / 4
RCC_PCLK2_Div6 ADC 时钟 = PCLK / 6
RCC_PCLK2_Div8 ADC 时钟 = PCLK / 8
USB 时钟来源:
该时钟来源于PLLCLK时钟的预分频
三、RCC配置实例代码,与解析
void RCC_Configuration(void)
{
ErrorStatus HSEStartUpStatus;
RCC_DeInit(); //复位RCC模块的寄存器,复位成缺省值
RCC_HSEConfig(RCC_HSE_ON); //开启HSE时钟,咱是用HSE的时钟作为PLL的时钟源
HSEStartUpStatus = RCC_WaitForHSEStartUp(); //获取HSE启动状态
if(HSEStartUpStatus == SUCCESS) //如果HSE启动成功
{
FLASH_PrefetchBufferCmd(ENABLE); //开启FLASH的预取功能
FLASH_SetLatency(FLASH_Latency_2); //FLASH延迟2个周期, 和STM32超频比较相关,Ourdev上有个超频的帖子
//,蛮有意思的有兴趣的可以看看
//附上网址http://www.ourdev.cn/bbs/bbs_content_all.jsp?bbs_sn=3554410
RCC_HCLKConfig(RCC_SYSCLK_Div1); //配置HCLK,PCLK2,PCLK1,PLL
RCC_PCLK2Config(RCC_HCLK_Div1);
RCC_PCLK1Config(RCC_HCLK_Div2);
RCC_PLLConfig(RCC_PLLSource_HSE_Div1,RCC_PLLMul_9);
RCC_PLLCmd(ENABLE); //启动PLL
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY)==RESET)
{} //等待PLL启动完成
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //配置系统时钟
while(RCC_GetSYSCLKSource() !=0x80) //检查是否将HSE 9倍频后作为系统时钟
{}
}
}
总之,STM32的时钟看起来比较复杂,但是可以不用去深究,使用STM32的库函数还是很好用的,虽然效率低了点,但是其实只使用很少的次数,无所谓了,要想STM32跑起来,按照上述配置就好了,千万不要忘记为使用的模块分配时钟,不要像我,跑个LED,忘记给IO分配时钟,还在纳闷呢,为什么不亮.(摘自网络).
总结:我比这位原作者好点,我第一次让STM32跑两个灯一下就顺利了.就是第一次KEIL联不上STM32有点郁闷,在网上升级了,STLINK还是不行.
后来得知,原来STLINK升级版本好几个了,找了新的版本总算跟4.12联上了.我的STLINK版本号J11
受教了。
谢谢,学习了。
感谢分享,标记一下
感谢分享,努力学习中......