有个项目要用到STM32F207Vx单片机,找到网上的例子照猫画虎的写了几个例子,比如ADC,可是到了ADC多通道转换的时候就有点傻眼了,这里面的时钟跑的到底是多少M呢?单片机外挂的时钟是25M,由于该单片机时钟系统较为复杂,有内部高/低、外部高/低 、PLL锁相环时钟,又有AHB总线时钟、APB1/2时钟,而例子中很少讲到系统时钟的默认配置是怎么配置呢?那么就发点时间研究下这个单片机内部的复杂时钟系统吧。 下图是STM32F2系列的时钟树结构图: 1、内部高速时钟HSI、外部高速时钟HSE和PLL时钟PLLCLK时钟都接到了SW开关处,通过SW选择哪一路作为SYSCLK,SYSCLK经过AHB分频器进行分频得到HCLK,APB1和APB2是挂在总线AHB上的,通过APB1和APB2分频得出fpclk1和fpclk2。 2、PLL输入时钟源主要是靠外部高速时钟和内部高速时钟作为时钟源,通过PLLCFGR寄存器的bit22来选择具体哪一路作为时钟源。选择好了时钟源进入/M分频器,也就是PLLM进行分频,送入VCO,在通过xN,进行倍频,也就是PLLN1)通过/P进行分频(PLLP)得到PLLCLK;(2)通过/Q分频(PLLQ),得到PLL48CK。 然后边看代码边对照结构图进行分析,看软件如何给单片机配置系统时钟的。 % V: L0 ]; g7 s+ t# [; F
! S9 X+ E& V3 h! R6 N
然后找到启动代码“startup_stmf32xx.s”,该代码是用汇编写的,可以看到,在调用main函数之前,是先调用了SystemInit函数的,该函数是在“system_stm32f2xx.c”中 - ; Reset handlerReset_Handler PROC! ~% [; l) u' G% m& Y! [
- EXPORT Reset_Handler [WEAK]
* c2 F6 W* d% Z+ j - IMPORT SystemInit
5 K* e+ ]5 q6 }4 g8 b4 G9 d - IMPORT __main
$ Q+ D0 L$ j/ i; Y5 `. `1 V: V1 P. _ - LDR R0, =SystemInit: N: |' X& a3 F# J! D7 |2 b
- BLX R0
0 a: t' {4 G4 }& q& A - LDR R0, =__main2 U9 |0 N* e; [ E9 @% B6 p
- BX R0 l- a/ O$ F5 ]) ?5 b. {
- ENDP
复制代码 1 a2 d, C4 G7 W+ q0 n! H
代码如下,变量直接赋个16进制的数,都不知道是啥意思,目的是干什么的,不知道,所以看下面代码时最好对照STM32F2x用户手册。当然这个只是一个初始化,待会主要看SetSysClock();这个函数,在调用该函数之前,我们知道单片机是先启用了内部高速时钟等一些配置。 $ M; C. B3 W3 t& r, p) L
- void SystemInit(void)+ N2 ~5 s6 q q8 f. k( D* q' R
- {9 J9 E+ i0 w+ C" C
- /* Reset the RCC clock configuration to the default reset state ------------*/
& `* h! S; \' c7 d5 B* A& z - /* Set HSION bit */
1 T! I# A7 r0 u+ o: f2 c, q# R - RCC->CR |= (uint32_t)0x00000001; //RCC_CR复位值0x0000_xx83,内部高速时钟使能,也就是说上电开始就使用内部高速时钟,16MHZ% U V7 s: A% P4 N" m6 w( k+ O [
- - V, E, O$ H5 O; S0 l! `
- /* Reset CFGR register */
; _9 }2 `- `* E2 V5 h1 f- e2 J2 y - RCC->CFGR = 0x00000000; //通过开关SW选择内部高速时钟作为系统时钟16MHZ1 [: z" ]6 s: f- ^# K
- //AHB prescaler 不分频% u9 [% B& A4 k
- //APB Low speed prescaler (APB1) 不分频,fplck1 = 16MHZ
! g; l0 b- [9 o1 E* Q5 d& A - //APB high-speed prescaler (APB2)不分频,fplck2 = 16MHZ& L2 C# ~2 {8 \: t) t
- //MCO1和MCO2时钟输出等配置可参考用户手册Page95
v. l$ c0 J A+ u/ }0 I - ' ]( `( y( {& a6 C7 { ~. \
- /* Reset HSEON, CSSON and PLLON bits */
( w6 ?) h2 H9 s: P8 s - RCC->CR &= (uint32_t)0xFEF6FFFF;9 ^! b9 i- O1 W# ?6 q1 X
- 0 V1 l @, ?3 W+ v7 t4 {
- /* Reset PLLCFGR register */
# ^3 a' w% q2 N$ v1 {5 `) F - RCC->PLLCFGR = 0x24003010; //RCC_CFGR复位值是0x2400_30106 C% m9 b8 K/ L
- ; n% Z2 ^/ K. n* c% O( V
- /* Reset HSEBYP bit */# D, q% ^* T4 |* {( D
- RCC->CR &= (uint32_t)0xFFFBFFFF; //对bit18 HSEBYP 设置为0,外部高速时钟被禁止
4 d" t0 D. U+ B. ` - . s# H% k9 F7 u+ ^. a
- /* Disable all interrupts */, X$ G2 U/ M* O6 J4 \) t( t
- RCC->CIR = 0x00000000; //所有时钟中断都被禁止- c9 O& T5 @ ^0 p
- ~3 ]8 |& }& s f3 r" @
- #ifdef DATA_IN_ExtSRAM
4 P' L: k9 J( e1 l- t v - SystemInit_ExtMemCtl(); 9 _7 m2 f/ ^$ p2 G% P" i7 F+ O
- #endif /* DATA_IN_ExtSRAM */* h) p3 L9 b5 Z6 V7 }' a4 a+ n
- : }1 s7 J( `3 S, C+ Y# m7 U
- /* Configure the System clock source, PLL Multiplier and Divider factors, % w+ ]0 P2 F2 i( i/ X
- AHB/APBx prescalers and Flash settings ----------------------------------*/3 I1 `' p% _) L; O
- SetSysClock();* `% S& B; [9 n, @5 t* i
0 P) z( E" ]. a& V+ G, W% t% `6 H- /* Configure the Vector Table location add offset address ------------------*/
3 J( O7 ]2 j& [$ \6 I - #ifdef VECT_TAB_SRAM+ h6 q0 f+ ^, u2 W* y7 o
- SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal SRAM */- m3 y. ?% ], c% U4 \* l
- #else
* `' X z) O4 k5 k( O# J/ N( B( } - SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal FLASH */
% i2 e7 d* o" P$ E+ o0 c - #endif& `9 p0 a8 \$ w8 K
- }
复制代码9 {! v0 P2 M; w* \, m
在SystemInit(void)函数中在配置完一些参数后,还调用了SetSysClock()函数,该函数代码如下 - static void SetSysClock(void)5 G% @0 z: ?, Z
- {2 [& x& \8 {$ i7 k$ }3 e
- /******************************************************************************/
0 y' _1 O \: a - /* PLL (clocked by HSE) used as System clock source */* a4 V/ @0 w% J0 S0 Q) [
- /******************************************************************************/
# u3 p) \- A1 D) \$ c7 c" P) G - __IO uint32_t StartUpCounter = 0, HSEStatus = 0;
& D2 E- V2 V4 ?$ T6 b6 n! P -
8 V3 Q7 S0 G* U) P7 l - /* Enable HSE */
/ C ~+ F4 R E& H1 y - RCC->CR |= ((uint32_t)RCC_CR_HSEON); //外部高速时钟使能,25MHZ
% l+ N& q9 Z' e |" b# G -
9 k0 ]1 ~( o: i5 ` - /* Wait till HSE is ready and if Time out is reached exit */ //外部时钟使能后,得需要一点时间到达各个端口
1 J6 `) @! z9 a0 v& V - do
! ^% F+ x" s3 d5 x! H$ D - {
& M2 H; r1 L7 n- r/ o- Z- q, A - HSEStatus = RCC->CR & RCC_CR_HSERDY; //如果RCC_CR_HSERDY为0,说明外部时钟还没准备好,1说明外部时钟已准备好
d2 K3 A0 O$ S' M; N - StartUpCounter++;//对读的次数进行累加,当累加次数到达1280次时,就意味着启动时间超时
$ Z3 ^2 G1 g+ g5 _, N( _5 x) g - } while((HSEStatus == 0) && (StartUpCounter != HSE_STARTUP_TIMEOUT));
. u& V5 X7 U3 s7 S - . G* v+ [# L1 N" N
- if ((RCC->CR & RCC_CR_HSERDY) != RESET)//这里判断上面的do while循环是因哪个条件结束的1 @% l4 H3 ]) w1 q6 r: \. h) E
- {
! ]4 S; A/ i5 m0 D6 @: ^ - HSEStatus = (uint32_t)0x01; //说明时钟已准备好了,才结束do whlie循环
4 S: W! O* F2 m" k# W5 d5 k - }
5 r [( U5 Z2 {7 @) U$ J - else
! N- ~3 X$ r: q5 C/ g8 s - {, b: b" L3 `4 l6 C2 k8 L5 D; n
- HSEStatus = (uint32_t)0x00; //说明是因为超时了而退出do while循环7 Z% I; R# b D8 E/ W+ W: V
- }
. j, Q1 r. d1 s6 j$ n/ s
( [/ s, f, e/ x2 r- if (HSEStatus == (uint32_t)0x01)
4 J- B+ ^- B# C, s2 D; C - {
0 g& S" l5 _1 \) f. L5 x+ _) Z# {1 U - /* HCLK = SYSCLK / 1*/
" Q2 X. K" {- A. b) O* X - RCC->CFGR |= RCC_CFGR_HPRE_DIV1; //AHB不分频,AHB出来后时钟就是sysclk=120M
, @6 d' T& q H3 U& Z+ l$ A - 4 x# M. P9 C$ @
- /* PCLK2 = HCLK / 2*/4 }0 J: V- E' t1 M6 Q9 @) E
- RCC->CFGR |= RCC_CFGR_PPRE2_DIV2;//APB2 2分频,fpclk2 = sysclk/2 = 60M
" P; F) k+ I* T' H -
# y1 r( N1 f9 z3 M; s) ? N# t- p - /* PCLK1 = HCLK / 4*/* w' P: p* Z# n. g
- RCC->CFGR |= RCC_CFGR_PPRE1_DIV4; //APB1 4分频,fplck1 = sysclk/4 = 30M
- ]1 M- k/ \/ f' i - ) b9 x9 @& F; @
- /* Configure the main PLL */ //主要对PLL和PPI2S 进行配置$ L0 T) W4 J: a9 [1 E! }! d' o( F
- RCC->PLLCFGR = PLL_M | (PLL_N << 6) | (((PLL_P >> 1) -1) << 16) |
A( Z( `# D/ o0 A - (RCC_PLLCFGR_PLLSRC_HSE) | (PLL_Q << 24);5 q0 X5 N1 q1 h: n
- //配置完后RCC->PLLCFGR = 0x05403c19,然后对照寄存器RCC_PLLCFGR查看哪些位对应哪些功能- k9 H! N- o# y* F( a% H
- //通过bit5~bit0可计算出PLLM=25,那么input VCO = PLL input clock /PLLM = 25M/25 = 1M,对应时钟结构图中的/M
- t) U; W; Z4 y. F3 d# u- N8 {- G; f& P - //通过bit14~bit6可计算出倍频因子PLLN = 240,那么VCO output clock = PLLN * input VCO = 240 * 1 = 240M,对应时钟结构图中的xN/ w( ?2 q4 s4 K1 `9 R; w+ C
- //bit17~bit16可计算出分频因子PLLP = 2,那么PLLCLK = VCO output clock /PLLP = 240/2 = 120M
& Q, C( N% q2 R0 [0 b8 b3 M - //bit22是选择给PLL输入的时钟源,输入时钟源有外部和内部高速时钟,这里选择的是外部高速时钟即PLL input clock = HSE =25M- O3 d5 X7 t8 \6 z) t9 _
- //bit27~24可计算出分频因子PLLQ = 5,那么PLL48CK = VCO output clock/PLLQ = 240/5 = 48M5 y6 ^, L. t& Z8 i1 H: {4 I) E1 H( i
-
4 t4 s% i, Q1 ]7 l9 s& B4 G - /* Enable the main PLL */ y" \2 v5 W6 T3 }1 \, n
- RCC->CR |= RCC_CR_PLLON; //使能PLL+ h% R. a: }) I7 Z1 _! B. o
. I. u2 S* ?, h9 [9 e3 O- /* Wait till the main PLL is ready */
- k% c0 M$ Q0 }2 X b6 ^. O - while((RCC->CR & RCC_CR_PLLRDY) == 0)" Q* B( ^4 @) s; _0 i
- {
: e; B1 u5 K9 p: }8 u - }
& X" B. W5 T" C6 j4 Y1 M - //到这里RCC->CR已配置完,最终值0x03036783; t% _9 x$ ] \6 t+ _) _
- //通过查看用户手册知道,内部高速时钟、外部高速时钟、PLL时钟都已开启
! {& Z( q S+ T- x0 w -
0 h. }4 ?. U4 `2 n - /* Configure Flash prefetch, Instruction cache, Data cache and wait state */6 @4 _2 r9 T( H9 D( W: W( D4 U
- FLASH->ACR = FLASH_ACR_PRFTEN | FLASH_ACR_ICEN | FLASH_ACR_DCEN | FLASH_ACR_LATENCY_3WS;! X0 G1 p; @/ V- @+ Y/ F. Q) _
- ' W$ m7 p8 B* E
- /* Select the main PLL as system clock source */4 i- D/ r/ n+ U1 m1 x) Y
- RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_SW));$ n4 Q; _9 M+ o) o5 I( c
- RCC->CFGR |= RCC_CFGR_SW_PLL;3 d: M- V! s' w# W: T
- 6 l% A' y2 _5 i3 J
- //到这里RCC->CFGR已配置完,最终值是0x0000_940A$ o; \- R$ X, H/ O
- //通过查看用户手册,知道,PLL时钟作为系统时钟即120M
, |% f3 L Q$ L - //AHB不分频,即HCLK = 120M p/ E! X# c# ~3 G7 S
- //APB1 4分频,即fpclk1 = 120/4=30M, D* h+ `! s+ b& Q1 u
- //APB2 2分频,即fpclk2 = 120/2=60M8 y4 ]% G7 Z: H" d8 V8 `* b
- 0 o) Q+ D* C$ g$ N/ M' z
- /* Wait till the main PLL is used as system clock source */& q. |( R, m* I- K3 a( v" }
- while ((RCC->CFGR & (uint32_t)RCC_CFGR_SWS ) != RCC_CFGR_SWS_PLL);+ o% D8 i$ `2 Z. }1 N, i
- {
2 j9 V l; A8 {. J - }" k! D4 n0 a, Z9 A
- }" k3 G- N9 s$ [+ m! W$ N( X; r
- else8 M4 l+ p+ N, Z+ d D8 H* O- [" u4 E
- { /* If HSE fails to start-up, the application will have wrong clock; l1 ]2 h) B0 w3 ^$ u
- configuration. User can add here some code to deal with this error */# H- ]7 ?# N9 G& u2 U0 g, t0 Z
- }
5 B, D& |& p, m. U6 i- c i/ I - . r( d# M' S/ m s
- }
复制代码- s; I5 X6 z$ \ `
OK,分析完这段代码后,调用系统固件函数后,现在知道了时钟树结构图中右边出来的时钟是多少MHz了吧。 总结: 1、使用的是外部时钟25MHZ,通过PLL进行分频倍频分频得到PLLCLK 120M,PLLCLK作为系统时钟SYSCLK。 2、APB1出来是30M,也就是FPCLK1。 3、APB2出来是60M,也就是FPCLK2。 4 u$ H' B1 x. h3 b4 s8 f3 e) k
t- {4 R+ P! J' t4 }
, h1 A2 j/ s: n- S) h. [" I5 p0 ?8 x3 U j& W- r: Y1 X& O9 K
|