微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > MCU和单片机设计讨论 > STM32 RCC管理

STM32 RCC管理

时间:10-02 整理:3721RD 点击:

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


受教了。

谢谢,学习了。

感谢分享,标记一下

感谢分享,努力学习中......

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top