你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32的时钟树深入详解

[复制链接]
Bonner 发布时间:2017-10-13 16:28
STM32上如果不使用外部晶振,OSC_INOSC_OUT的接法
如果使用内部RC振荡器而不使用外部晶振,请按照下面方法处理:
' ~9 q; m0 o1 w/ {) x* W
1)对于100脚或144脚的产品,OSC_IN应接地,OSC_OUT应悬空。" b" w4 ?9 y' T" M
2)对于少于100脚的产品,有2种接法:5 K4 d6 s7 f  \% ]4 z* f5 s' q
   2.1)OSC_IN和OSC_OUT分别通过10K电阻接地。此方法可提高EMC性能。
' M0 Z4 a1 q1 t4 k4 A7 p2 g% g   2.2)分别重映射OSC_IN和OSC_OUT至PD0和PD1,再配置PD0和PD1为推挽输出并输出'0'。此方法可以减小功耗并(相对上面2.1)节省2个外部电阻。
   HSI内部8MHz的RC振荡器的误差在1%左右,内部RC振荡器的精度通常比用HSE(外部晶振)要差上十倍以上。STM32的ISP就是用(HSI)内部RC振荡器。
. v0 T; }# R# X2 b2 D
STM32时钟系统
     在STM32中,有五个时钟源,为HSIHSELSILSEPLL
HSI是高速内部时钟,RC振荡器,频率为8MHz。
HSE是高速外部时钟,可接石英/陶瓷谐振器,或者接外部时钟源,频率范围为4MHz~16MHz。
LSI是低速内部时钟,RC振荡器,频率为40kHz。
LSE是低速外部时钟,接频率为32.768kHz的石英晶体。
PLL为锁相环倍频输出,其时钟输入源可选择为HSI/2、HSE或者HSE/2。倍频可选择为2~16倍,但是其输出频率最大不得超过72MHz。
9 J* z# s! T$ X* s6 I
    用户可通过多个预分频器配置AHB总线、高速APB2总线和低速APB1总线的频率。AHB和APB2域的最大频率是72MHZ。APB1域的最大允许频率是36MHZ。SDIO接口的时钟频率固定为HCLK/2。
; l- X3 l0 ?  c) f    40kHz的LSI供独立看门狗IWDG使用,另外它还可以被选择为实时时钟RTC的时钟源。另外,实时时钟RTC的时钟源还可以选择LSE,或者是HSE的128分频。RTC的时钟源通过RTCSEL[1:0]来选择。% p' y! [" ?5 i$ \/ y, E( U
    STM32中有一个全速功能的USB模块,其串行接口引擎需要一个频率为48MHz的时钟源。该时钟源只能从PLL输出端获取,可以选择为1.5分频或者1分频,也就是,当需要使用USB模块时,PLL必须使能,并且时钟频率配置为48MHz或72MHz。: a+ G! m- K0 _3 w! o- z
    另外,STM32还可以选择一个PLL输出的2分频、HSI、HSE、或者系统时钟SYSCLK输出到MCO脚(PA8)上。系统时钟SYSCLK,是供STM32中绝大部分部件工作的时钟源,它可选择为PLL输出、HSI或者HSE,(一般程序中采用PLL倍频到72Mhz)在选择时钟源前注意要判断目标时钟源是否已经稳定振荡。Max=72MHz,它分为2路,1路送给I2S2、I2S3使用的I2S2CLK,I2S3CLK;另外1路通过AHB分频器分频(1/2/4/8/16/64/128/256/512)分频后送给以下8大模块使用:2 q. z8 q. d/ T  K1 T
① 送给SDIO使用的SDIOCLK时钟。
- P' Q3 t7 t; N# B1 [) {" \② 送给FSMC使用的FSMCCLK时钟。+ {: V) y9 l8 \5 z
③ 送给AHB总线、内核、内存和DMA使用的HCLK时钟。
5 v( v( `% R' S& M④ 通过8分频后送给Cortex的系统定时器时钟(SysTick)。( @) }3 {, u" N0 x- z2 E# t/ v. b
⑤ 直接送给Cortex的空闲运行时钟FCLK。
: @1 q6 x3 K1 b" N5 Z⑥ 送给APB1分频器。APB1分频器可选择1、2、4、8、16分频,其输出一路供APB1外设使用(PCLK1,最大频率36MHz),另一路送给定时器(Timer2-7)2、3、4倍频器使用。该倍频器可选择1或者2倍频,时钟输出供定时器2、3、4、5、6、7使用。
$ T+ F: L/ ]5 t! g% h. V' p( I3 q⑦ 送给APB2分频器。APB2分频器可选择1、2、4、8、16分频,其输出一路供APB2外设使用(PCLK2,最大频率72MHz),另一路送给定时器(Timer1、Timer8)1、2倍频器使用。该倍频器可选择1或者2倍频,时钟输出供定时器1和定时器8使用。另外,APB2分频器还有一路输出供ADC分频器使用,分频后得到ADCCLK时钟送给ADC模块使用。ADC分频器可选择为2、4、6、8分频。4 `3 O* _0 k' }
⑧ 2分频后送给SDIO AHB接口使用(HCLK/2)。

1 f0 d0 A7 l: L, J8 a) `- ~" A/ K' r  \时钟输出的使能控制
, l2 Q, U+ r1 w/ j" R
    在以上的时钟输出中有很多是带使能控制的,如AHB总线时钟、内核时钟、各种APB1外设、APB2外设等。1 E2 g/ x: [3 x; q9 h
当需要使用某模块时,必需先使能对应的时钟。需要注意的是定时器的倍频器,当APB的分频为1时,它的倍频值为1,否则它的倍频值就为2。
: `5 m+ s+ E2 ~. I, W   连接在APB1(低速外设)上的设备有:电源接口、备份接口、CAN、USB、I2C1、I2C2、UART2、UART3、SPI2、窗口看门狗、 Timer2、Timer3、Timer4。注意USB模块虽然需要一个单独的48MHz时钟信号,但它应该不是供USB模块工作的时钟,而只是提供给串行接口引擎(SIE)使用的时钟。USB模块工作的时钟应该是由APB1提供的。
% I4 ~+ U9 n5 ?' h" N! z
   连接在APB2(高速外设)上的设备有:GPIO_A-E、USART1、ADC1、ADC2、ADC3、TIM1、TIM8、SPI1、AFIO

/ A8 g$ [# @2 h2 D$ O) h
11.jpg

; k  S. A; s2 W3 `4 j/ }4 N' I

( q+ k* d7 Z% |
使用HSE时钟,程序设置时钟参数流程:
! h+ Y! {% ?( I- m# N' K, z# B1、将RCC寄存器重新设置为默认值   RCC_DeInit;4 ~  N0 n: @, s8 Z
2、打开外部高速时钟晶振HSE       RCC_HSEConfig(RCC_HSE_ON);4 @0 o3 ]5 v) D- H2 w
3、等待外部高速时钟晶振工作      HSEStartUpStatus = RCC_WaitForHSEStartUp();
3 o5 |0 d8 ?) R4 C; {- o% C4、设置AHB时钟         RCC_HCLKConfig;! h$ Y+ _# Q  W2 }
5、设置高速AHB时钟     RCC_PCLK2Config;+ O  w1 _, G- U. L, T+ {: H9 B6 V$ r6 O
6、设置低速速AHB时钟   RCC_PCLK1Config;
! }% ?8 c5 ?( x9 D  y7 D7、设置PLL              RCC_PLLConfig;
2 X, m: h/ |! o4 k8、打开PLL              RCC_PLLCmd(ENABLE);( [6 ]! N* p; |$ G/ A: l6 b
9、等待PLL工作          while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)
: M$ V+ B  Y4 |10、设置系统时钟        RCC_SYSCLKConfig;$ B" G0 V% F2 Q. X  E! _
11、判断是否PLL是系统时钟     while(RCC_GetSYSCLKSource() != 0x08)" L: _0 f1 @- W- b7 ?
12、打开要使用的外设时钟      RCC_APB2PeriphClockCmd()/RCC_APB1PeriphClockCmd()

. d' j  Y- L4 m' h3 M! ~
下面是STM32软件固件库的程序中对RCC的配置函数(使用外部8MHz晶振)
: {# k: o0 c9 i; o# ^  h6 V
void RCC_Configuration(void)
{
  RCC_DeInit();
  RCC_HSEConfig(RCC_HSE_ON);   //RCC_HSE_ON——HSE晶振打开(ON)
  HSEStartUpStatus = RCC_WaitForHSEStartUp();
  if(HSEStartUpStatus == SUCCESS)        //SUCCESS:HSE晶振稳定且就绪
  {   
    RCC_HCLKConfig(RCC_SYSCLK_Div1);  //RCC_SYSCLK_Div1——AHB时钟 = 系统时钟
    RCC_PCLK2Config(RCC_HCLK_Div1);   //RCC_HCLK_Div1——APB2时钟 = HCLK
    RCC_PCLK1Config(RCC_HCLK_Div2);   //RCC_HCLK_Div2——APB1时钟 = HCLK / 2
    FLASH_SetLatency(FLASH_Latency_2);    //FLASH_Latency_2  2延时周期
    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);       // 预取指缓存使能
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);   
   // PLL的输入时钟 = HSE时钟频率;RCC_PLLMul_9——PLL输入时钟x 9
    RCC_PLLCmd(ENABLE);
    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET) ;   
    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
   //RCC_SYSCLKSource_PLLCLK——选择PLL作为系统时钟
    while(RCC_GetSYSCLKSource() != 0x08);        //0x08:PLL作为系统时钟
  }
& w/ e& G2 g1 Z, b% o- l  P, |! H
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB |
  RCC_APB2Periph_GPIOC , ENABLE);
//RCC_APB2Periph_GPIOA    GPIOA时钟
//RCC_APB2Periph_GPIOB    GPIOB时钟
//RCC_APB2Periph_GPIOC    GPIOC时钟
//RCC_APB2Periph_GPIOD    GPIOD时钟
}

, S1 @% g. i, Y4 m, P& C3 {: L

2 H1 A$ T; A! T' T
下面是用mdk仿真GPIO的实验截图:
<span]1 x4 \# z0 z8 m: T7 R; v9 q3 y" r
12.jpg

, u* K! e! Q, H  m

评分

参与人数 1 ST金币 +6 收起 理由
MrJiu + 6 赞一个!

查看全部评分

收藏 4 评论6 发布时间:2017-10-13 16:28

举报

6个回答
alisa123 回答时间:2017-10-13 17:00:07
楼主介绍很详细,谢谢分享,先收藏了!
stary666 回答时间:2017-10-13 20:43:43
MrJiu 回答时间:2017-10-14 08:38:58
支持!!!
epochal 回答时间:2017-10-14 09:48:49
谢谢分享!
斜阳 回答时间:2017-10-14 09:54:48
好详细
pythonworld 回答时间:2017-10-14 11:28:22
谢谢分享!

所属标签

相似分享

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版