库里面的时钟配置程序。* G6 ^* u; h' @( ^
startup 里面有个调用SystemInit()函数的过程:
+ M9 y" [( S. j9 v7 J5 P' P- Reset handler* a# r2 t$ a* {- v* q
- Reset_Handler PROC: p: G; O( n4 Y. ]
- EXPORT Reset_Handler [WEAK]
& [/ E+ j+ G& d+ R - IMPORT __main
/ d: L3 \5 F1 b& p' c - IMPORT SystemInit7 J) l# A) j7 U, [8 I7 E9 n
- LDR R0, =SystemInit) f1 s' m( s; p, f: l1 o4 J
- BLX R0 ( P# o4 z- j' c/ |' ]
- LDR R0, =__main8 J+ z ~7 U+ ]5 D$ q' L) e
- BX R0
9 B+ d( G7 c, [$ P: {/ D - ENDP
复制代码 ; ?0 y4 B% D# _# L9 l
这个SystemInit在哪里定义呢?在system_stm32f10x.c中定义。
" T5 e1 r3 S% l- c, gSystemInit()快结束时调用了SetSysClock()函数,时钟配置就由该函数具体实施。! ^! p* O. f4 S& A
------------------------------------------------------
- x- \5 _- ~5 vSetSysClock()也在system_stm32f10x.c中定义。7 v. F4 Y! G9 ?$ J, i" Y: [- P
其实该函数很短,它根据不同的宏定义来调用不同的SetSysClockToxx()函数,比如SetSysClockTo71()就是当有宏定义SYSCLK_FREQ_72MHz时就调用该函数来具体配置时钟的。这些不同的SetSysClockToxx()调用是有优先级的,频率低的大于频率高的,而直接采用HSE具有最高优先级。
( W1 h9 c) ?/ N: v( U来看看这些宏定义和调用函数之间的关系:
" ~# A N" b/ v4 t首先是定义宏:% d6 \8 X! O" t: R
- #if defined (STM32F10X_LD_VL) || (defined STM32F10X_MD_VL) || (defined STM32F10X_HD_VL)
/ I$ {$ X1 m/ R" U. M - /* #define SYSCLK_FREQ_HSE HSE_VALUE */
; L3 J' M* P1 m! c; [4 h, [8 K - #define SYSCLK_FREQ_24MHz 24000000- W9 b* x m# F" v$ x
- #else" B5 w- s T& |, N8 y
- /* #define SYSCLK_FREQ_HSE HSE_VALUE */
; m+ @. C" @/ j' t* \ - /* #define SYSCLK_FREQ_24MHz 24000000 */
- w4 r, v5 _: t. M) i9 P# J2 Q - /* #define SYSCLK_FREQ_36MHz 36000000 */' u: o- i9 r5 J/ @
- /* #define SYSCLK_FREQ_48MHz 48000000 */
( `/ E* f$ F& ^+ ~' f) J: C - /* #define SYSCLK_FREQ_56MHz 56000000 */# t+ f1 [! k" p
- #define SYSCLK_FREQ_72MHz 72000000
. R% a3 c0 F) e) D' O: a6 D - #endif
复制代码
" s* L+ m/ V7 U/ t1 w3 V看来需要根据项目需求来,不能同时使用多个宏,不用的宏需要注释掉。
, o9 x8 U) W& m, S; y6 J4 y3 F5 _然后根据定义的宏来声明所需的函数:) n! t4 `5 g; ^. v) ^. v
- [mw_shl_code=c,true]#ifdef SYSCLK_FREQ_HSE
* V/ x, u) U& V - static void SetSysClockToHSE(void);
& @) D8 e; j7 e9 {' h1 v7 F - #elif defined SYSCLK_FREQ_24MHz
6 J; ~1 L5 {9 {% Z: e7 }, I - static void SetSysClockTo24(void);
) s8 y/ S6 I/ |# i! N; g - #elif defined SYSCLK_FREQ_36MHz
# v4 ?2 g3 O# i4 d o - static void SetSysClockTo36(void);
) U7 x# A4 ^0 G! B& q - #elif defined SYSCLK_FREQ_48MHz) c2 r C% S p/ X( P# |
- static void SetSysClockTo48(void);4 k- ^8 L) G$ Z* i* n% F
- #elif defined SYSCLK_FREQ_56MHz$ V! d0 z& p4 B2 M. f F- P/ n
- static void SetSysClockTo56(void);
: |$ h9 q+ h1 m. q& U3 G* y - #elif defined SYSCLK_FREQ_72MHz V6 o3 {! {8 V: F9 O9 d
- static void SetSysClockTo72(void);
, w4 x- c# _" I3 ~5 b0 A. k5 [ - #endif/ d: ^6 v- _& ]8 m3 q9 j. O- B7 @
- [/mw_shl_code]
复制代码
8 _/ I% e2 j+ @- L. x' V4 r& p( s如果定义了多个宏会屏蔽掉低优先级的函数。在SetSysClock()里面再根据宏来调用用时钟配置函数:, c4 q- ?6 g- ?; ~" A* G, g- g4 C
- [mw_shl_code=c,true]static void SetSysClock(void)) P- _4 h/ z+ x& g/ V, ?
- {
) Z) c/ W# L& o; ?+ m - #ifdef SYSCLK_FREQ_HSE, j! B. }- A l! G$ o
- SetSysClockToHSE();6 F) B8 v9 H6 Y) C
- #elif defined SYSCLK_FREQ_24MHz/ Q1 A9 D. ?$ o) F' p f
- SetSysClockTo24();
, ^ e, a' \3 W. T3 f" p/ B - #elif defined SYSCLK_FREQ_36MHz
% d7 {% O: d/ v `8 w - SetSysClockTo36();+ R' x; J& ]% r$ I1 |
- #elif defined SYSCLK_FREQ_48MHz
4 d: f) A7 i& K/ h# [ - SetSysClockTo48();
3 d, B9 T$ y- u7 | - #elif defined SYSCLK_FREQ_56MHz7 X. J# p; f8 w8 v" @8 @$ u
- SetSysClockTo56();, J. Y! _8 {& z0 _ |+ H4 V
- #elif defined SYSCLK_FREQ_72MHz$ Q6 Y0 v4 h* y6 m! v
- SetSysClockTo72();) J# u g( f2 g- \* x
- #endif9 V o- c, b4 Y6 ?$ g
) F% s$ Q( `% Q$ W; l
& h; z5 S- e: b3 r% |- /* If none of the define above is enabled, the HSI is used as System clock
3 H6 @" m$ j5 q* D# |1 a - source (default after reset) */
6 n1 f: R" y( ]- L - }5 \" g! @; ~! K5 W @0 N( F8 C
- [/mw_shl_code]
复制代码 5 h+ c- h: Z, x3 x; M6 ~0 u
如果没有宏定义,或者说让SystemInit()函数为空函数,则系统不会配置时钟而直接采用内部高速时钟HSI作为系统时钟使用。
% Q2 t4 X4 k: e8 z7 hsystem_stm32f10x.c里面其他的代码就是关于这些SetSysClockToxx()的实现。也就是如何使用寄存器来设置倍频、分频数等等,需要了解了寄存器后才能明白。查看这些函数,有个直观的感受是里面的代码是直接使用寄存器来设置数据的,不是使用库函数。通观整个过程,如果只是学习的话,根据硬件只需要把待设置成的频率宏去掉注释而把不需要的宏注释掉就可以了,编译时会自动加入需要的时钟设置函数的。
; r6 ]% x6 [/ ~% ^
# L6 S: n: I) l- S配置好时钟后就开始执行用户程序了,在用户程序里面要使用外设首先必须让外设挂载的总线上的该设备进行时钟使能,让时钟驱动外设工作。5 {; j B7 v( N* R8 W1 F
这是通过寄存器设置来实现的。$ o: [6 x- U7 w" }( o2 G. L# B
RCC一共有10个寄存器,互联型更多。启动时配置时钟当然要是通过这些寄存器来设置的,不过进入main后使能外设时钟就只用到了三个寄存器:RCC_AHBENR、RCC_APB2ENR、RCC_APB1ENR。寄存器同样是封装在结构体里面的:RCC_TypeDef,里面的寄存器名称去掉了RCC_前缀。既然是使用库函数,我们也就不必太关心这些寄存器封装是如何使用的了,只需关心库提供了哪些函数去操作这些寄存器实现所需的功能了。
9 i3 l: X Y# v: X: L& @RCC库函数在stm32f10x_rcc.h和.c中实现。: I0 J- Q1 f9 @0 b! b. N
RCC地址宏定义,在stm32f10x.h中定义:& s3 O* q2 s' [
- [mw_shl_code=c,true]#define PERIPH_BASE ((uint32_t)0x40000000) /*!< Peripheral base address in the alias region */: ?6 u- I) q" k4 o/ e
- #define AHBPERIPH_BASE (PERIPH_BASE + 0x20000)
2 D% T; o9 U* d- c; U: m+ ^/ G - #define RCC_BASE (AHBPERIPH_BASE + 0x1000)#define RCC ((RCC_TypeDef *) RCC_BASE)
( a' q' G0 K6 Q, F% K, b - #define RCC ((RCC_TypeDef *) RCC_BASE)
- m; s+ j( _0 T( d - [/mw_shl_code]
复制代码
2 C& E c( d2 j/ L; a' ORCC寄存器原来是在AHB总线地址空间上的,这个AHB总线有些不平常。AHB、APB1、APB2总线上外设使能的函数分别是:
" x5 j3 o( @7 E- [mw_shl_code=c,true]void RCC_AHBPeriphClockCmd(uint32_t RCC_AHBPeriph, FunctionalState NewState);1 t& \, P7 w2 h, r; W6 M
- void RCC_APB2PeriphClockCmd(uint32_t RCC_APB2Periph, FunctionalState NewState);) L1 {( W$ f) ^! |/ k1 X% v1 `
- void RCC_APB1PeriphClockCmd(uint32_t RCC_APB1Periph, FunctionalState NewState);[/mw_shl_code]
复制代码 2 k! Q& K3 ]; J: G8 _2 t& O4 B
要使某个外设时钟使能,其实就是使RCC_AHBENR、RCC_APB2ENR、RCC_APB1ENR三个寄存器中对应位置1,那么这是如何在函数中实现的呢?首先是定义了这三个寄存器各个位代表的外设的宏,也就是当置1时寄存器的值。以APB2ENR寄存器各个位代表的外设为例说明:
: L8 H W W5 \' t* g' L; I8 Y9 ~- [mw_shl_code=c,true]#define RCC_APB2Periph_AFIO ((uint32_t)0x00000001)
5 Y G' l# w6 l6 E; K: b- N - #define RCC_APB2Periph_GPIOA ((uint32_t)0x00000004)
% ? D8 H: Y% o" U - #define RCC_APB2Periph_GPIOB ((uint32_t)0x00000008)4 @! N' P# t& i
- #define RCC_APB2Periph_GPIOC ((uint32_t)0x00000010)
. O$ K" n, V" Z( m: u - #define RCC_APB2Periph_GPIOD ((uint32_t)0x00000020)/ M: i* N, T; r! }& b3 x, c3 h1 Z9 O
- #define RCC_APB2Periph_GPIOE ((uint32_t)0x00000040)
, Z+ l8 ], [) W - #define RCC_APB2Periph_GPIOF ((uint32_t)0x00000080)
$ Q5 g1 S7 s% I0 s - #define RCC_APB2Periph_GPIOG ((uint32_t)0x00000100)
1 j. ]2 i& W: t3 Q8 L - #define RCC_APB2Periph_ADC1 ((uint32_t)0x00000200)/ {+ i$ i, B3 t4 i" Y% m8 D! M9 l! x' Q
- #define RCC_APB2Periph_ADC2 ((uint32_t)0x00000400)
/ h: O7 k$ L9 x: C7 b - #define RCC_APB2Periph_TIM1 ((uint32_t)0x00000800)3 {2 x( @- i; o8 x
- #define RCC_APB2Periph_SPI1 ((uint32_t)0x00001000)& f* ~% C6 l# M6 _% J
- #define RCC_APB2Periph_TIM8 ((uint32_t)0x00002000)! f4 ]7 e. I! ?/ M8 L6 ~8 Y% f2 K' `
- #define RCC_APB2Periph_USART1 ((uint32_t)0x00004000)0 `/ s6 X y& I$ ~; |* U
- #define RCC_APB2Periph_ADC3 ((uint32_t)0x00008000)
9 \" |4 i2 J3 N l' F, P - #define RCC_APB2Periph_TIM15 ((uint32_t)0x00010000)# @' ?' ]% P4 R
- #define RCC_APB2Periph_TIM16 ((uint32_t)0x00020000)
6 G- V9 B/ L0 F - #define RCC_APB2Periph_TIM17 ((uint32_t)0x00040000)3 W# |0 C" c% q* ~
- #define RCC_APB2Periph_TIM9 ((uint32_t)0x00080000)( W& C" `* |" I1 G
- #define RCC_APB2Periph_TIM10 ((uint32_t)0x00100000). J. b$ Z z+ Z
- #define RCC_APB2Periph_TIM11 ((uint32_t)0x00200000)
# J$ [/ M% E$ W2 c3 c9 }; c - [/mw_shl_code]
复制代码 8 e+ Q3 `. m c- @5 R% E! J; Y2 P
3 i' Y1 J8 m( ?4 q8 b
均是当外设位为1而其他位为0时寄存器的值。stm32中文参考手册V10.0中关于APB2ENR寄存器只有低16位有定义,高16位保留,可是,库函数里面却有高16位的几个定义,这是为什么?# B5 P1 U3 e6 R! q
而如果需要使能就使APB2ENR寄存器“或”上该值,关闭则是“与”上该值的取反值。+ j: Y9 G3 e. y
- [mw_shl_code=c,true]void RCC_APB2PeriphClockCmd(uint32_t RCC_APB2Periph, FunctionalState NewState)4 t: R! s3 O, K' Y4 m8 L2 m
- {, ]- ~" k% b* N1 h
- /* Check the parameters */
* l7 ~; S) f \, Y - assert_param(IS_RCC_APB2_PERIPH(RCC_APB2Periph));" g3 Q: V. `" e" U: D1 y. t
- assert_param(IS_FUNCTIONAL_STATE(NewState));7 a; Z" y% q( ]: f' R6 e
- if (NewState != DISABLE)' I' v7 a8 ?: C5 K' `6 x
- {
5 K- W% ?: v) f8 y) X: ^ - RCC->APB2ENR |= RCC_APB2Periph;% s( o0 S2 w3 f; `
- }8 q2 L1 }; ]0 R1 S) L- @' i
- else
' |- }: Q0 S+ s: z - {7 A/ M) a; G/ m/ }
- RCC->APB2ENR &= ~RCC_APB2Periph;
0 W! _0 D% v5 {5 F$ \+ S/ S - }) i/ i6 r# M8 U3 \) [8 j
- }
/ Z5 r( e( p7 G! Z6 a/ } - [/mw_shl_code]
复制代码 ( q1 L; k! j! v0 S) V) S- K3 K
------------------------------------------------, k0 ^6 [1 [& [; H/ u6 _
断言assert_param():4 f: J- t' `( H$ _$ @
看似是一个函数,其实是宏,在stm32f10x_conf.h中定义:. B! j/ c1 L9 K2 S0 l E
- [mw_shl_code=c,true]#ifdef USE_FULL_ASSERT1 t% i5 I- p. b& e
- #define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t *)__FILE__, __LINE__))/ e2 D! u) P+ j4 c
, h v Y' s) G' ^: D
l$ Q. V2 x/ V- void assert_failed(uint8_t* file, uint32_t line);+ i/ v1 i9 I! G) ^4 z1 {
- #else
1 x3 Q2 B/ _1 R% F - #define assert_param(expr) ((void)0)3 q3 m: A0 P$ C- z/ j5 I
- #endif /* USE_FULL_ASSERT */
' H, J& ~# P. j( H3 C - [/mw_shl_code]
复制代码
) C3 d; z+ j2 f* ~ {3 y断言用于参数检查,如果说传入的参数符合要求那么宏值为(void)0,否则执行assert_failed()函数,这个函数没有实现。如果需要可以实现assert_failed()函数,指出哪个文件的哪行出了错误。3 K( ~# }% s5 g3 X/ s0 w
断言一般用在开发阶段,必须定义USE_FULL_ASSERT宏,否则只是得到(void)0,对于C来说就相当于空语句。开发调试结束取消USE_FULL_ASSERT宏。这里需要注意:宏定义不仅仅用于预编译阶段,它甚至可以用于运行阶段。比如在运行阶段才能决定断言里面的值倒地是true还是false,从而决定断言值是(void)0还是执行assert_failed()函数。断言和错误捕捉显然不同。
+ o' R7 @5 f$ d3 C- \7 ~# P7 w---------------------------------
7 }5 C; b8 D& \ q- V" k5 H l; {来看看都做了什么检查:
* z. ^# N2 _& ]0 U& u - assert_param(IS_FUNCTIONAL_STATE(NewState));8 [' I7 n2 @2 r% H
- [mw_shl_code=c,true]#define IS_FUNCTIONAL_STATE(STATE) (((STATE) == DISABLE) || ((STATE) == ENABLE)): ]+ |# B" t0 a. M6 z! B
- [/mw_shl_code]很清楚了,如果说参数STATE为DISABLE或ENABLE则宏值为true,否则为false。
* J% m! P9 O1 C: w - 那么 assert_param(IS_RCC_APB2_PERIPH(RCC_APB2Periph))呢?[mw_shl_code=c,true]#define IS_RCC_APB2_PERIPH(PERIPH) ((((PERIPH) & 0xFFC00002) == 0x00) && ((PERIPH) != 0x00))( @% E) d6 P" o7 C6 q8 q
- [/mw_shl_code]
复制代码
1 v3 J, g) r7 t9 @9 q6 l这个比较诡异!!5 R, w* c* ~! D) p4 f# r5 n
---------------------------------------
! z4 w4 { Q# `, g3 |# W总结,void RCC_APB2PeriphClockCmd(uint32_t RCC_APB2Periph, FunctionalState NewState);中的RCC_APB2Periph参数必须是预定义的宏才能保证是有效值。
9 R1 V" T3 {- V1 _: C! Z比如:RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
" F$ @+ T, z4 c+ o- `. |8 U如果需要同时使能多个同总线上的外设就采用“或”.
" n6 O% M; b0 W7 ?如何记忆这个函数:RCC_APB2PeriphClockCmd就是“关于RCC时钟的”挂载在APB2总线上的外设时钟使能函数,参数开头也必须是RCC_APB2Periph_开头。
1 U4 |& m: N* ]————————————————
: ?; V6 ~( x9 M3 c" H% F版权声明:被大佬糊弄的只会点灯的小菜鸡
* |) H1 i; n+ f3 t, `如有侵权请联系删除; G0 O7 l _4 `- I3 b
) o. k0 |) q. ^' {. I" @: N
( A$ k5 g! B: W4 G8 {% B/ `( c: u. \/ x0 h, |' u5 ?7 k
|