16.1 初学者重要提示$ ?% ^2 @' z# ^. f
对于一些常用的函数,大家一定要熟练的掌握都实现了什么功能,比如HAL_Init,HAL_RCC_OscConfig,HAL_RCC_ClockConfig等。最好的办法是把这些函数的源码读一遍。' M+ m; i7 v1 J! L+ X
16.2 那些是必备的API" U# }! B, {2 W7 D' Y: N
这里我们通过一个简单的初始化流程来了解STM32H7的工程模板所必备的库文件和API:
7 |7 \% _ ~2 Z' Y2 \- v& T$ X" L
* Q9 I( m) q' q) `) p1 h 第1步:系统上电复位,进入启动文件startup_stm32h743xx.s,在这个文件里面执行复位中断服务程序。
- a4 k! s5 T9 G3 l0 `
9 z- O( K5 K% ^8 U; Z 在复位中断服务程序里面执行函数SystemInit,此函数在文件system_stm32h7xx.c里面。& `' b0 r+ n0 w" e7 Z
之后是调用编译器封装好的函数,比如用于MDK的启动文件是调用__main,最终进入到main函数。
- L% c6 J) @6 @. X7 [: ~: [ 第2步:进入到main函数就可以开始用户应用程序编程了。在这个函数里面要做几个重要的初始化,依次是:
# A/ T& V1 H% @5 D2 r
8 }4 a7 k/ u2 |) F% U U MPU初始化,需要用到库文件stm32h7xx_hal_cortex.c和stm32h7xx_hal_cortex.h。& D. ?/ b* q* ^5 v6 U( T
Cache初始化,需要用到core_cm7.h文件。- ^0 K; D: p D: {0 b" v0 l
HAL库初始化函数HAL_Init,需要用到文件stm32h7xx_hal.c。3 [, @5 c c3 s' O
系统时钟初始化,需要用到库文件stm32h7xx_hal_rcc.c。
5 a, v- [8 O" b: Z; _0 }8 u
7 W& W) t' U. ^, _: c- D z3 E
3 A+ o0 t/ n5 Y' ^6 R+ h/ L前面的两步完成后,就可以开始做用户需要的按键、串口等方面的初始化和应用代码的实现了。这里把我们需要学习的几个库文件整理出来,依次有:. g6 Y. [0 \- V1 [. J
startup_stm32h743xx.s' \7 q! X( w% |# x* C/ f0 z
system_stm32h7xx.c0 k* R2 l5 O+ {' t# p
stm32h7xx_hal.c
& X% g$ k1 n- a$ n: ~ stm32h7xx_hal_cortex.c
4 V0 S+ Z6 i2 |' G; k stm32h7xx_hal_rcc.c
. I& f U+ H# E, U- ] core_cm7.h
- _ g) k0 U# ^( Y其中startup_stm32h743xx.s和system_stm32h7xx.c已经在第13章为大家讲解过,这里不再赘述。而MPU和Cache涉及到的文件core_cm7.h在第23和24章为大家讲解。本章教程重点为大家讲解文件stm32h7xx_hal.c、stm32h7xx_hal_cortex.c和sm32h7xx_hal_rcc.c。
) Y- M: N, F' ?6 ^7 c2 }; y% z6 ?4 r7 s; x3 `4 b. m+ Z8 I
16.3 源文件stm32h7xx_hal.c(重要)
5 n6 u6 M# `) V这个文件比较杂,像基准电压大小配置,EXTI配置,IO补偿配置等都在这个文件里面设置。学习这个文件注意事项:
5 B, {# u* ]# }3 ~5 Z; s3 b$ R" ?0 I) [* [6 y) Y' g7 X
HAL库中各个外设驱动里面的延迟实现是基于此文件提供的时间基准,而这个时间基准既可以使用滴答定时器实现也可以使用通用的定时器实现,默认情况下是用的滴答定时器。
) ?, { n: b2 H- d5 h/ G 函数HAL_Init里面会调用时间基准初始化函数HAL_InitTick,而调用函数HAL_RCC_ClockConfig也会调用时间基准初始化函数HAL_InitTick。
$ z: x1 H! }* J" }1 p1 L 如果在中断服务程序里面调用延迟函数HAL_Delay要特别注意,因为这个函数的时间基准是基于滴答定时器或者其他通用定时器实现,实现方式是滴答定时器或者其他通用定时器里面做了个变量计数。如此一来,结果是显而易见的,如果其他中断服务程序调用了此函数,且中断优先级高于滴答定时器,会导致滴答定时器中断服务程序一直得不到执行,从而卡死在里面。所以滴答定时器的中断优先级一定要比它们高。$ R1 X+ r* T' f& M. m
1 g* ?8 d# ~& }( T16.3.1 函数HAL_Init
( }, E$ d) q& @, F函数原型:
8 W0 `" k& R' _- c
- A% j8 I9 t4 M6 l' o: n7 s- HAL_StatusTypeDef HAL_Init(void)
7 `, [- {" M7 [& R, T3 p' r - {
* } U# \% ^8 ]0 { - /* 设置中断优先级分组 */
0 X5 M* E; L! p. |6 y9 l - HAL_NVIC_SetPriorityGrouping(NVIC_PRIORITYGROUP_4);! b0 a6 d; k K2 R
- & Q7 C" @2 b$ A7 X( n* q
- /* 使用滴答定时器做为默认时基,配置为1ms滴答,另外系统上电后默认使用的HIS时钟 */
9 j) Y+ j3 F/ W+ Y* \) d - if(HAL_InitTick(TICK_INT_PRIORITY) != HAL_OK)
, r8 u, D# w! U( p5 f - {9 A p. M% L5 K+ t4 X; t
- return HAL_ERROR;* U4 Z0 y+ g3 D' i& X$ q
- }; }; w' H G+ ~) j% W! G
- # ^7 W$ s' J" N+ d" H
- /* 初始化底层硬件 */' B" ]7 o* m! G$ Z% ?2 M
- HAL_MspInit();# Z* z; P& S) ~. r. r
- {7 U- _5 A) s4 N+ I6 j- /* 返回函数状态 */: B- g5 Z7 f$ T' P
- return HAL_OK;" v5 N. J5 a9 y* S. H; S5 h
- }
复制代码 , Q' b. C$ g9 K) P7 i6 s5 B
函数描述:
# ^# }1 ?' _8 P- j; h i
: _9 i4 e& D [3 V6 q% k此函数用于初始化HAL库,此函数主要实现如下功能:( X# ]! M2 `. v6 q$ U% H" w
; M' @% e+ T; R* ^; }2 X 设置NVIC优先级分组是4。
. l2 d& }! p* m 设置滴答定时器的每1ms中断一次。! H6 d8 {$ z6 k8 L
HAL库不像之前的标准库,在系统启动函数SystemInit里面做了RCC初始化,HAL库是没有做的,所以进入到main函数后,系统还在用内部高速时钟HSI,对于H7来说,HSI主频是64MHz。, r2 F6 V1 r% U2 {+ I
函数HAL_Init里面调用的HAL_MspInit一般在文件stm32h7xx_hal_msp.c里面做具体实现,主要用于底层初始化。当前此函数也在文件stm32h7xx_hal.c里面,只是做了弱定义。
9 O2 M+ h. |( t3 `. t% o6 b5 h& v& J- @5 p$ ?5 a
函数参数:3 [1 k. ` v2 h! r, U
返回值,返回HAL_ERROR表示参数错误,HAL_OK表示发送成功,HAL_BUSY表示忙,正在使用中。, ~* a: g/ ^: P
* b8 [$ C1 X2 N) Z
注意事项:' _! s+ |) `- ^$ c# P
必须在main函数里面优先调用此函数。
2 Y# G: P) l9 H1 g9 h! U! I用户务必保证每1ms一次滴答中断。
7 P: b: l* E- A- k9 ?) s+ A0 t关于优先级分组的设置可以看第21章节。
* W: F1 \$ n" T9 G4 T0 X! r$ \: a* {' p/ q, c
使用举例:
% c/ V# U' K& Y- B1 o9 ?/ X此函数的使用比较简单,上电后优先调用即可。
: |1 Q: v5 q. ?- L Q, u, u$ I& q9 M. i& J
16.3.2 函数HAL_DeInit
+ B5 c% t h* \函数原型:" b8 i% B+ k% A7 K: O. R
* k: ?$ o0 ^8 B8 B9 x( n8 i: [# q- HAL_StatusTypeDef HAL_DeInit(void)
. _9 p( [: \4 F/ l - {4 B7 z o/ |* q8 y( N! E
- /* 复位所有外设 */__set_PRIMASK
! K3 [" G$ h; a - __HAL_RCC_AHB3_FORCE_RESET();4 }+ o% L+ t" F5 Z
- __HAL_RCC_AHB3_RELEASE_RESET();
1 |3 q( v; u/ t' F - ! _& x4 _: c* ] c: _# p
- /* 省略未写 */8 @. ]) [' {- ]& ^7 j1 `0 {. q
3 p/ ^. j9 W/ W! c8 s3 E5 B- __HAL_RCC_APB4_FORCE_RESET();
4 M9 j* _- E+ @1 V- h) l0 j - __HAL_RCC_APB4_RELEASE_RESET();" o H$ v1 K M+ m& x4 l# S( }- I/ b
7 E1 P R8 ~/ C. }, N- /* 复位底层硬件初始化 */# E! X* A7 F! M, o9 ?( R
- HAL_MspDeInit();
. E8 z, a+ T" H. k* F8 ^+ W9 a
1 ^# O- J) i/ S# L- /* 返回值 */
, R, k: C# X0 W' l4 `' E - return HAL_OK;
. `8 x' P1 f( ? ? - }
复制代码 ( q' @1 R$ z. u
函数描述:
" @& K8 h8 V* B# A J( m5 E- Z/ D: n) w6 n+ H5 Z0 Z. Z
此函数用于复位HAL库和滴答时钟。
) k: ^+ {/ s' H& m9 a 复位了AHB1,2,3,4的时钟以及APB1L,APB1H,APB2,3,4的时钟。
( j$ t: g5 ~% ^; v2 l7 O 函数HAL_DeInit里面调用的HAL_MspDeInit一般在文件stm32h7xx_hal_msp.c里面做具体实现,主要用于底层初始化,跟函数HAL_Init里面调用的HAL_MspInit是一对。当前此函数也在文件stm32h7xx_hal.c里面,只是做了弱定义。 ]" ?: \. d/ f- ?9 v! \3 j
) J% H% J: e& y
9 Y+ X! I1 \0 M. ]) P) }6 d使用举例:
$ `0 Z4 C) E1 A* W" n此函数的使用比较简单,需要调用的时候直接调用即可。
* C1 x; a- @. I g2 |2 u0 ^" D0 s2 k1 l r$ ~; ^
16.3.3 函数HAL_InitTick
# M) G- {) ?) |函数原型:
6 l! q% T9 i! D: f2 O4 e
' G7 k7 P* g% Z) G; o- __weak HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority)- w1 i3 ?. V& I+ Z( m2 H
- {' S' U6 g( Q1 {: E( y1 Q
- /* Configure the SysTick to have interrupt in 1ms time basis*/
. B3 b7 _4 N- D* ~2 T - if (HAL_SYSTICK_Config(SystemCoreClock / (1000U / uwTickFreq)) > 0U)
% F- ?, a0 s! _ - {
( R2 i& c5 c- k. ^ - return HAL_ERROR;1 n1 {" W& A; O7 U3 u. _
- }
% \# u( ^2 Q, c% z0 G' U% y7 A
: s& J; \9 m- ^- D6 X- /* Configure the SysTick IRQ priority */ ]0 O# V# B+ f& A: T( T
- if (TickPriority < (1UL << __NVIC_PRIO_BITS))
$ J* K0 M) S7 ~5 z" ^ - {
7 L/ i5 a" ]) P1 c - HAL_NVIC_SetPriority(SysTick_IRQn, TickPriority, 0U);! S% r6 W# j* }
- uwTickPrio = TickPriority;
; q/ I4 l/ D$ c% `6 T, f4 t- A& ] - }
" T7 V/ V/ v5 f: n - else; n: {# V% h& |' Q! F* j
- {
" E7 `: c; G9 @ - return HAL_ERROR;
$ c9 e- w+ V8 ]0 P: k - }5 `3 [( }. ?+ P$ a2 {
- 8 ~, `3 t# s) u
- /* Return function status */7 v0 `$ l6 y- _, {6 v
- return HAL_OK;- t0 J: Q8 x$ P5 P) ?7 b
- }
复制代码
8 w. e/ S# ^5 j+ s x7 V函数描述:
. l/ d" P- K: ?; _/ i0 D" n此函数用于初始化滴答时钟,此函数相关问题如下:- k) A* P3 u" P2 V- G* x, o8 ? I( }
此函数有个前缀__weak ,表示弱定义,用户可以重定义。% ~5 ~/ q3 F" I3 [/ V
此函数用于初始化滴答时钟1ms中断一次,并且为滴答中断配置一个用户指定的优先级。
8 U8 ]) j( ~( l3 h3 f' _+ z 此函数由HAL_Init调用,或者任何其它地方调用函数HAL_RCC_ClockConfig配置RCC的时候也会调用HAL_InitTick。, w5 Y( T6 u) C# M7 h, a& T
调用基于此函数实现的HAL_Delay要特别注意,因为这个函数的时间基准是基于滴答定时器或者其他通用定时器实现,实现方式是滴答定时器或者其他通用定时器里面做了个变量计数。如此一来,结果是显而易见的,如果其他中断服务程序调用了此函数,且中断优先级高于滴答定时器,会导致滴答定时器中断服务程序一直得不到执行,从而卡死在里面。所以滴答定时器的中断优先级一定要比它们高。! k7 n& b8 B- \. v7 u
+ p" v( [5 ]! X5 U' w! l" n: B" I- P
函数参数:
) S$ o3 L/ J! R3 I; E6 _' E 形参TickPriority用于设置滴答定时器优先级。
! w- A O6 n* `3 c8 z/ I 返回值,返回HAL_ERROR表示参数错误,HAL_OK表示发送成功,HAL_BUSY表示忙,正在使用中。
4 s+ Q$ Q0 ?' l/ y5 d' g3 l, n5 O$ n9 Y C
6 f `' H1 }7 C3 W. d* |& ]使用举例:
" d! d, ^7 ]- M) X. S2 a此函数由HAL_Init调用,无需用户操作,除非需要重定义。
1 t) P* p" F b* M) Z3 v2 H8 e6 }: A+ P7 `8 i$ Z" ]' v, @# }" ~
16.3.4 Systick的相关函数
' t1 F0 f) R! c9 I调用了函数HAL_Init后,Systick相关的函数就可以使用了。这些函数如下:" B8 v$ G6 p2 E. a, p
! z! x6 b. a0 q6 k函数原型:
`& _# x# v# r$ \7 p
* E9 _ H+ N" \* O( X- ~- O: m- __weak void HAL_IncTick(void)
0 Q+ C* x: e% f; U( c5 c# K# X0 A( | - __weak uint32_t HAL_GetTick(void)
" q* A# C; `! l0 t- S# J& ~8 D - uint32_t HAL_GetTickPrio(void)
$ A# J6 i% b2 r( O - HAL_StatusTypeDef HAL_SetTickFreq(HAL_TickFreqTypeDef Freq)
+ \/ y! K, T% ]( a- L A0 ^ - HAL_TickFreqTypeDef HAL_GetTickFreq(void)9 N* W1 S) T3 C0 Y) {6 Y
- __weak void HAL_Delay(uint32_t Delay)1 o7 a2 V4 f$ q+ @
- __weak void HAL_SuspendTick(void)4 ?2 r0 U! F% w
- __weak void HAL_ResumeTick(void)
复制代码
, J2 H7 U" u, ]% l$ I& X: b函数描述:7 d" L/ f' \2 o2 L* O4 W
这些函数就比较简单了,下面把这些函数实现的功能做个简单的说明:* r. E' B: |7 z5 [1 g
函数HAL_IncTick在滴答定时器中断里面被调用,实现一个简单的计数功能,因为一般滴答定时器中断都是配置的1ms,所以计数全局变量uwTick每毫秒加1。
# g4 F: Y+ A7 n8 Q9 i9 C6 h/ y 函数HAL_GetTick用于获取全局变量uwTick当前的计数。8 E, ` v- S/ C0 P; ?; M% }
函数HAL_GetTickPrio用于获取滴答时钟优先级。+ r- R7 Z" t5 M, m: N) u- `5 R
函数HAL_SetTickFreq和HAL_GetTickFreq是一对,前者用于设置滴答中断频率,后再用于获取滴答中断频率。5 f6 ^, L7 U x
函数HAL_Delay用于阻塞式延迟,默认单位是ms。
- o; n/ A* z5 v8 R# y* T 函数HAL_SuspendTick和HAL_ResumeTick是一对,前者用于挂起滴答定时器,后者用于恢复。
G/ x% \: V) c
$ _3 i* K0 P, H8 i+ x4 g
+ P- O, v/ [( J( T/ d) [9 ]# g+ ]' e注意事项:
! j) ?5 h8 p- x+ _, J" @ 函数有个前缀__weak ,表示弱定义,用户可以重定义。2 S6 M+ v( i: P* w2 ^) b
. H7 u6 J9 _, f/ s* z) Y* L3 C* W7 [- [, T, R. U. r2 S5 N
使用举例:" A0 ^1 |" k( m; @, N/ J) c! j% O
这些函数都比较简单,这里就不举例了。需要的时候,直接调用即可。
. x( ?$ x D9 e6 X6 h; P* W: F- D+ D9 @+ E
16.3.5 函数HAL_SYSCFG_VREFBUF_VoltageScalingConfig
" m* `: Z5 Y+ q' e, _函数原型:
, Z7 B* G3 h5 y0 w8 v
' J# u0 y5 V4 x- void HAL_SYSCFG_VREFBUF_VoltageScalingConfig(uint32_t VoltageScaling)
复制代码
& R- n6 Q+ w- z* j. S函数描述:
/ a9 W$ z0 R* R j R3 j此函数用于配置STM32H7内部电压基准大小。6 Z( m4 E% v; Z7 n& P$ V5 F( y- }
n1 Z4 C' M P+ \+ o
当形参VoltageScaling = SYSCFG_VREFBUF_VOLTAGE_SCALE0时3 a% m1 c0 }4 p3 ~7 B- u5 }
输出基准是2.048 V,条件是VDDA >= 2.4V。
. U; e5 x4 W; h5 w1 X# K+ ?9 F6 {
" T5 [1 I/ L* ] C$ o5 ? 当形参VoltageScaling = SYSCFG_VREFBUF_VOLTAGE_SCALE1时
4 D- e- `( @' O% d; U: d; \# {- j输出基准是2.5 V,条件是VDDA >= 2.8V。
, U' b2 K3 J# n7 u. e7 A: \5 i: {1 m+ M" |0 p& z
当形参VoltageScaling = SYSCFG_VREFBUF_VOLTAGE_SCALE2时
% W$ j3 f& j2 T( P, _3 _- z 输出基准是1.5 V,条件是VDDA >= 1.8V。) N# c: ?/ z$ _+ `
+ C) f" ~. H0 ?) w: p8 t- T 当形参VoltageScaling = SYSCFG_VREFBUF_VOLTAGE_SCALE3时
+ i' ~& s0 Q5 Z9 E* s输出基准是1.8 V,条件是VDDA >= 2.1V。
$ K; s3 z2 W* O- f
3 R5 |) m$ r& W# A16.3.6 H7自带电压基准相关函数
8 o& _+ q( ^8 I函数原型:8 m; g4 ~# Q$ J1 V+ o+ v
- void HAL_SYSCFG_VREFBUF_VoltageScalingConfig(uint32_t VoltageScaling)7 O2 h) D! J5 w; B4 g1 Q# ?4 g1 d) j& h
- void HAL_SYSCFG_VREFBUF_TrimmingConfig(uint32_t TrimmingValue): W7 B; P0 {7 T" B3 U+ c
- HAL_StatusTypeDef HAL_SYSCFG_EnableVREFBUF(void)
1 v0 o! I0 r' C K2 [ - void HAL_SYSCFG_DisableVREFBUF(void)
复制代码
% U* m/ J% Z* d" _: a4 x函数描述:
2 e. z1 R' P- S8 J2 l) x6 x' `" J; J" x7 n
函数HAL_SYSCFG_VREFBUF_VoltageScalingConfig
8 o+ G% `5 o( n# w. r此函数用于配置STM32H7内部电压基准是否在芯片内部与VREF+引脚接通。
6 R' u5 M5 P* d: a! G2 H0 B( P" w; O6 i4 y! F8 U
形参为SYSCFG_VREFBUF_HIGH_IMPEDANCE_DISABLE时,表示导通。5 E8 ~/ \9 D$ T% T3 V" t) m. ]& m
形参为SYSCFG_VREFBUF_HIGH_IMPEDANCE_ENABLE时,表示高阻,即不导通。1 Y: s) v V' J0 v
函数HAL_SYSCFG_VREFBUF_TrimmingConfig5 _3 c3 e5 L/ O& e
此函数用于内部电压基准的校准调节。, J6 q$ }6 N" T; Y9 g
, ?5 {7 f& @/ m! E+ u+ A9 e
函数HAL_SYSCFG_EnableVREFBUF和HAL_SYSCFG_DisableVREFBUF是一对,分别用于内部电压参考基准的禁止和使能。 b# d& `/ I- E( i
% z2 q9 o9 _. ?! e1 J16.3.7 函数HAL_SYSCFG_AnalogSwitchConfig- T/ d8 G `% L
函数原型:0 w4 C# F1 f$ U$ m
- void HAL_SYSCFG_AnalogSwitchConfig(uint32_t SYSCFG_AnalogSwitch , uint32_t SYSCFG_SwitchState )
复制代码 8 ~) m2 j# Q3 j. g1 t/ a5 g0 o
函数描述:, Z" u' U, R, r- k
引脚PA0,PA1,PC2,PC3用于ADC时,还有一组对应的可选引脚PA0_C,PA1_C,PC2_C和PC3_C。此函数的作用就是切换可选引脚。关于这个问题的详情可看此贴:http://www.armbbs.cn/forum.php?mod=viewthread&tid=87707。: t! j; I3 M7 z6 Q
16.3.8 BOOST的使能和禁止(用于ADC) ]8 ~& g% U4 D4 ] ^1 t
! l) |: ~) O% o- r4 D" l: s! S. n7 O5 L% d+ S2 b T# j& `' o
函数原型:8 E5 [7 a; A& M1 w) q
- void HAL_SYSCFG_EnableBOOST(void)
/ ?! a- o6 v: r$ v' r- R - void HAL_SYSCFG_DisableBOOST(void)
复制代码 % G/ o2 @' \9 l
函数描述:
1 T6 y) D, m, M% l0 F/ a, c& |8 I 这两个函数用于使能或者禁止Booster。如果使能了booster的话,在供电电压低于2.7V时,可以减少模拟开关总的谐波失真,这样的话,模拟开关的性能跟正常供电电压时的全范围测量一样。
9 E0 i1 W2 n7 ]" ^& W& ]9 d8 m1 H& {$ D
4 m) I5 m: e& a2 U& G1 a
16.3.9 函数HAL_SYSCFG_CM7BootAddConfig
0 ?* n! D; Q: e3 t& d+ Q函数原型:8 n( X, @+ m+ r2 f T/ o1 }
- void HAL_SYSCFG_CM7BootAddConfig(uint32_t BootRegister, uint32_t BootAddress)
复制代码
/ v$ D1 r5 O, m, t' C0 o) Y函数描述:2 M6 n/ ~1 V# h R, h
: E) J" j& p& Z9 x+ x3 t+ @用于配置BOOT = 0或者BOOT = 1时的启动地址,详情可以看第13章的13.4小节。
. d2 F$ I. E0 e7 Y+ S( Z6 e" \/ D1 i
, G& s5 V5 ]# z% Z- h- O+ |5 ~16.3.10 IO补偿相关函数8 ?' r# g a0 s& n3 J) r
函数原型:
; U# Z X/ B3 n* m! V
+ T$ T% c% {' U5 I7 o) o# e- void HAL_EnableCompensationCell(void) Z3 z5 |- h/ M; F
- void HAL_DisableCompensationCell(void)
% w5 U& S0 o! z2 Z/ T - void HAL_SYSCFG_EnableIOSpeedOptimize(void)
# l# x) g: m% ^, }7 l; g - void HAL_SYSCFG_DisableIOSpeedOptimize(void)
# {& l2 C& ?' F' h o - void HAL_SYSCFG_CompensationCodeSelect(uint32_t SYSCFG_CompCode)
( p7 X8 [5 f5 K$ ?! y - void HAL_SYSCFG_CompensationCodeConfig(uint32_t SYSCFG_PMOSCode, uint32_t SYSCFG_NMOSCode )
复制代码 ' O3 ` f1 {+ h. H/ S
函数描述:
2 ~/ X4 z2 G( u7 t m6 t1 P. |1 U5 t- }' v' r; R- ?
函数HAL_EnableCompensationCell和HAL_DisableCompensationCell
$ a A: |* n9 P: G分别用于使能或者禁止IO补偿,只有在供电电压范围是2.4V到3.6V之间时,使用此功能才有意义。+ \3 U/ W9 S( Y, _( p: v
! u. s/ C/ |2 z) H 函数HAL_SYSCFG_EnableIOSpeedOptimize和HAL_SYSCFG_DisableIOSpeedOptimize. a8 _& p# ]9 k0 x5 X% [
分别用于优化IO速度或者禁止优化,不过仅在供电电压低于2.5V时可用,高于2.5V是不可以使用的,另外使用这个功能的前提是用户使能了PRODUCT_BELOW_25V(是可选字节配置选项里面的一个bit)。7 [5 I2 D4 ?( ?# T1 ]
! P$ a: K( u& T e& a& K
函数HAL_SYSCFG_CompensationCodeSelect
; P% v- m1 e, i# E; Z0 M/ a/ KIO补偿单元的选择,函数形参可以是SYSCFG_CELL_CODE,即寄存器SYSCFG_CCVR,也可以是SYSCFG_REGISTER_CODE,即寄存器SYSCFG_CCCR。" ^9 G% \$ X, l2 z. A5 W, W( X
& h$ n) w4 q |2 X2 j: I. C
函数HAL_SYSCFG_CompensationCodeConfig
+ O8 D8 C: Q6 w% d Y+ {用于设置GPIO内部构造中NMOS和PMOS的补偿值,两个形参的范围都是0-15。根据用户调用函数HAL_SYSCFG_CompensationCodeSelect选择的寄存器,这里仅有一个形参的设置是有效的。+ H& G# ~3 r4 ~# ^
; q* H) ^+ W% z+ E$ p$ {+ ]( o16.3.11 低功耗和EXTI相关函数
% K1 ]. E. ?* C' p4 |4 s低功耗和EXTI相关的函数暂时不做讲解,后面章节用到时候再说明。0 n5 Q$ k: O; @& B X& s9 ?
5 {7 C& _# E/ l' h/ a
16.4 源文件stm32h7xx_hal_rcc.c, I, L6 _4 S" a# Q. ]# I) t
这个文件主要是实现内部和外部时钟(HSE、HSI、LSE、CSI、LSI、HSI48、PLL、CSS、MCO)以及总线时钟(SYSCLK、AHB3、 AHB1、AHB2、AHB4、APB3、APB1L、APB1H、APB2、 APB4)的配置。8 J& Q7 K6 V/ `# u
, i9 O2 }- R) x& x( Z
学习这个文件注意事项:9 H1 g% x3 l$ N( }( I% i( \
+ F1 m0 }1 k' b8 X5 t- e, c
系统上电复位后,通过内部高速时钟HSI运行(主频64MHz),Flash工作在0等待周期,所有外设除了SRAM、Flash、JTAG 和 PWR,时钟都是关闭的。这里特别注意SRAM,官方手册里面说的太笼统,这个SRAM到底是指的哪个SRAM。
) u$ R$ [& Y" A& J' {4 L, d
$ D5 ], L* A j' e7 XAHB和APB总线无分频,所有挂载这两类总线上的外设都是以HSI频率运行。) o1 f& S( n9 |0 y3 M @1 N& b
所有的GPIO都是模拟模式,除了JTAG相关的几个引脚。
5 O( U4 ]9 E6 f) l- k 系统上电复位后,用户需要完成以下工作:
$ j2 k- G( ~% K 选择用于驱动系统时钟的时钟源。
- K/ {( @4 C; w! _ x7 y% q 配置系统时钟频率和Flash设置。 j( |: n3 R9 @1 u* Z
配置分频器。
0 L9 W7 D- \, Q" A1 ~; Q* G, ~ 使能外设时钟。. |. e" Z+ [. I' o: s- L9 _+ X
配置外设时钟源,部分外设的时钟可以不来自系统时钟,此时通过配置寄存器RCC_D1CCIPR、RCC_D2CCIP1R、RCC_D2CCIP2R 和 RCC_D3CCIPR 实现。
- B' g0 q/ b+ r1 d8 W
' t# i3 t& l. ?/ |$ L+ {2 V; ~! T- d) O. x$ s
RCC局限性:
# S# M1 R/ y$ R. z5 F
; N U- R2 y" l+ D2 c使能了外设时钟后,不能立即操作对应的寄存器,要加延迟。不同外设延迟不同:
9 V) y1 v j' T* t; t
5 N4 }7 G" J: O. }8 l( V" @ 如果是AHB的外设,使能了时钟后,需要等待2个AHB时钟周期才可以操作这个外设的寄存器。
6 l5 T: A' c& A+ W( ]1 @ 如果是APB的外设,使能了时钟后,需要等待2个APB时钟周期才可以操作这个外设的寄存器。
7 ^0 g" I4 {" p# g* ]9 i6 S
/ d& c. {1 A# H( d" q0 [当前HAL库的解决方案是在使能了外设时钟后,再搞一个读操作,算是当做延迟用。
* U6 E# z, P/ o$ V: r; j" [ Z- C' N5 K* e' x1 H5 E3 \2 ]/ m# o
比如下面使能GPIOA的时钟:; D: l; E2 a2 g+ s
1 ~% I3 H U% A) c$ x; a- #define __HAL_RCC_GPIOA_CLK_ENABLE() do { \
$ }0 w( P9 w6 w - __IO uint32_t tmpreg; \# ^5 K4 M, Q) ^7 K0 \% z
- SET_BIT(RCC->AHB4ENR, RCC_AHB4ENR_GPIOAEN);\0 g- f! m0 \. D! j
- /* Delay after an RCC peripheral clock enabling */ \' L9 R/ T: x+ E$ I
- tmpreg = READ_BIT(RCC->AHB4ENR, RCC_AHB4ENR_GPIOAEN);\
: S7 O2 [, q9 C8 e; c. v* E - UNUSED(tmpreg); \. `+ v' w8 I; T% L* _8 `* i# W5 m5 a$ M
- } while(0)
复制代码 - h6 Q/ E9 m) ?) j- C5 u
关于时钟方面的知识点补充! O$ h7 Y0 G7 L! b; a( Y
1、正确理解HCLK1,2,3,4以及PCLK1,2,3,4对应哪些总线的时钟,下面一张图可以说明问题:
3 D$ F; U5 E* ~
2 F$ s M5 n' L. o7 z
* Q! w# O( W( j* T6 P6 | Z. Y* N
( [, |* _* j! i2 }由上图可以得出:
v- r7 T& B2 j3 a2 ?1 R: I/ M# P9 \2 e, ^& c' m
HCLK1,2,3,4对应的是AHB总线AHB1,AHB2,AHB3和AHB4时钟。7 i& B( q3 b0 y+ H1 l
: x/ P' l) X' N% L' M! @
PCLK1、2、3、4对应的是APB总线APB1,APB2,APB3和APB4时钟。
6 l6 T- p( H( O" f" O5 @& F+ y' w! K! K0 P) c
2、内部和外部时钟配置:0 Z C# p% J' _5 @7 _+ T
6 y% X0 A4 f0 f
HSI (high-speed internal)
3 y3 J; ?2 {! n高速内部RC振荡器,可以直接或者通过PLL倍频后做系统时钟源。缺点是精度差些,即使经过校准。+ x' n2 k! x5 ~" Q* }3 C
& |. m7 ^/ G. \: G( a1 d
CSI (Low-power internal oscillator)5 W1 t) u4 @$ F. t; O$ T
低功耗内部RC振荡器,作用同HSI,不过主要用于低功耗。/ m* H( a6 _9 h) w
; {9 y& q9 l) p7 X: p
LSI (low-speed internal)8 X* F1 a0 j* d9 j) w& {
低速内部时钟,主要用于独立看门狗和RTC的时钟源。. C* m9 i0 G% V# A7 m# K
( \7 m; O- V& m$ C
HSE (high-speed external)' E5 p* \. I# c6 c! ~1 k
高速外部晶振,可接4 - 48MHz的晶振,可以直接或者通过PLL倍频后做系统时钟源,也可以做RTC的是时钟源。
8 B/ [& s* z3 g. H' r2 a
+ _2 I) k/ |; A7 E! m LSE (low-speed external)
4 x0 @* L! _: u1 v; h( X低速外部晶振,主要用于RTC。( s9 F& e, ?5 `
3 \4 B- x) \/ M) ?# ?, i
CSS (Clock security system)2 [% `; |% P' i( ]
时钟安全系统,一旦使能后,如果HSE启动失败(不管是直接作为系统时钟源还是通过PLL输出后做系统时钟源),系统时钟将切换到HSI。如果使能了中断的话,将进入不可屏蔽中断NMI。0 K8 }! g. J* b8 A
. s$ d5 t D; h
MCO1 (micro controller clock output)
# S7 _# }1 i; B8 z6 D' W, D可以在PA8引脚输出HSI, LSE, HSE, PLL1(PLL1_Q)或HSI48 时钟。
7 e/ c5 S* M6 p2 U1 `0 t9 ]3 ]6 L
* d }* f) a; \+ b MCO2 (micro controller clock output); q+ W! @* p r2 X; X0 A5 }
可以在PC9引脚输出HSE, PLL2(PLL2_P), SYSCLK, LSI, CSI或PLL1(PLL1_P) 时钟。
" P' K( k8 y, N% s& s1 K
8 p, P* ]5 q. a X7 B% q PLL锁相环,时钟输入来自HSI , HSE 或者CSI, u8 Q6 N1 q2 R, U; _& T" t
主锁相环PLL1,用于给CPU和部分外设提供时钟。3 k6 r8 N# t K6 M+ |
' H1 I3 U; n6 d6 B- V专用锁相环PLL2和PLL3,用于给部分外设提供时钟。/ v U( Y( V; ]
o# e4 i* w, P& c5 h* Q& e; ?' e- \- P# e5 `2 h
' i& U9 Z9 T8 n. a, T0 @5 R- M+ U
3、System, AHB 和 APB总线时钟配置: j1 s2 M5 L/ Y3 S( g$ W3 T
* i( i8 p/ C7 M
系统总线时钟源可以来自CSI,HIS,HSE 或 PLL。
5 x5 l2 Y0 m/ m7 T AHB总线时钟是通过系统时钟分频产生的,用于给AXI,AHB1,2,3,4和APB1,2,3,4九个总线上的外设提供时钟。另外用户可以通过函数HAL_RCC_GetSysClockFreq获取系统时钟。
^+ O8 \8 f, d( b7 U 所有外设时钟都可以通过系统时钟获取,并且部分外设还支持多个时钟源,比如下面串口1,6) Y% r5 _- R0 q2 } q9 v7 a
" t# k L, V" R- F0 \% b$ P! ~1 n g6 V$ m# @9 U1 U
& r+ i; Q0 F) y% k& T( D16.4.1 函数HAL_RCC_DeInit1 Y9 m! {5 Z* X6 B* J& N* \6 ~
函数原型:
5 b; @: X M$ G8 \$ r6 U& p4 r/ l* l/ @, n: @5 j* U8 \3 I- K8 g6 X2 B
- void HAL_RCC_DeInit(void)
4 M4 m& e, C% w9 L* Z {0 s) z - {
( |, @7 z: ~! Y5 X - /* Set HSION bit */3 H; A# d. { a
- SET_BIT(RCC->CR, RCC_CR_HSION);2 A" Y1 E! e! V. h
- 0 H1 Y$ u/ ]( s g
- /* Reset CFGR register */
- K# k! }! ~* ~. \ - CLEAR_REG(RCC->CFGR);! P3 K8 h% x" |# l2 V R& C* A; L
- 0 q1 X* c' D6 c% W9 h+ k: h
- /* 省略未写 */" r5 `& M- T$ U: z
6 ?1 W7 [1 p8 \- N N1 H0 G# [0 l- /* Reset HSEBYP bit */+ C4 v! G5 C( b8 `& h" S
- CLEAR_BIT(RCC->CR, RCC_CR_HSEBYP);
) w. a( d j9 h e h. q( \ - l3 ~/ s$ X/ ~" L4 ^
- /* Disable all interrupts */
. X1 H9 \0 K2 `6 L: _ - CLEAR_REG(RCC->CICR);
- r+ a: c5 S* O/ q( ]$ J' ] - }
复制代码
$ ~* m8 I: r5 O" [6 T函数描述:
! q: q9 j& h) d' U3 b* s. N/ m% f/ g# u3 I) x4 t
此函数用于RCC复位函数,主要实现如下功能:. h2 X: a7 {1 |( {" J
HSI 打开作为系统时钟。" G" C% A7 G Z
HSE, PLL1, PLL2 和 PLL3 关闭。% Z/ |9 F+ `8 Y5 U# |( I1 w7 z4 r1 v
AHB, APB 总线无分频 。
2 n! X% E, r9 D5 z" n CSS, MCO1 和 MCO2 关闭。
# [; v0 _; b) Q( A1 \! H 所有中断关闭。
4 |; v, ~) M: `7 m
r2 I% K! c0 G1 O& d
, D9 ~ G% m# _5 U- @注意事项:
6 a, F/ p) H2 }7 J, f' h! p此函数不会修改外设时钟,LSI、LSE和RTC时钟。: Q2 s- W% K. |; L
# y# r0 p# v- g/ r$ E( r7 _$ W% T% H+ K9 W ], b& g0 `' c2 A1 H
使用举例:
# V3 f: f. E5 f, o此函数的使用比较简单,需要调用的时候直接调用即可。* `- r0 y9 i6 j4 T. }5 Z
# ^; q- Y. s! l16.4.2 函数HAL_RCC_OscConfig2 L! ^; L3 Y2 x- g7 I) S
函数原型:) }# E; O1 A1 ~& S. \
' G7 O) k8 N+ t6 m: v' t. X+ Z
/ w3 {2 k, O4 b! T. i/ B
+ Z$ R3 M" D% F& g函数描述:
' s( f' F' p: S/ u# ?
, {+ o' E- b" Z2 I) Y5 K* v/ w通过上面函数原型,我们可以一目了然的看出此函数的作用,配置了HSE、HSI、CSI、LSI、HSI48、LSE和PLL。
# U9 \& \: v% h! D5 @$ y9 J; z& X+ Q' Z
函数参数:
! W$ w# L! c/ g% `! L) h. u. `5 z- j9 A6 R1 ^
函数的形参是RCC_OscInitTypeDef类型结构体变量,这个结构体的定义如下(主要是HSE、HSI、CSI、LSI、HSI48、LSE和PLL的配置变量):
/ H# G2 ^4 [. _( i2 r/ R8 b- /**
2 h [6 U t7 O3 [ Q$ _' B - * @brief RCC Internal/External Oscillator (HSE, HSI, CSI, LSE and LSI) configuration structure definition& J1 \6 ~. o# V6 J P
- */
" @% P% w& c# x n - typedef struct
1 N6 k* [ Z3 Z" D7 h - {$ `: ^$ n. ^: k v
- uint32_t OscillatorType;
+ S5 l" s: o, j' ~: } - uint32_t HSEState; 0 A* n3 x) W0 f1 N: r- }: `
- uint32_t LSEState;
. v- b5 d6 i, w1 l( H - uint32_t HSIState; , N. L8 b) ]. _! B( b7 ]
- uint32_t HSICalibrationValue;
, C1 h& Z. U! w( I, a. s7 j - uint32_t LSIState;
! K& T: j% e7 m( a* ]: M - uint32_t HSI48State;
% q }& [. z4 d# c - uint32_t CSIState;
/ x0 }, K1 E! t' D8 X7 T. Z/ _ - uint32_t CSICalibrationValue; 8 p$ \7 d @8 [$ m' g# C
- RCC_PLLInitTypeDef PLL; . ?! F* r% l: c$ U. s6 E3 q6 R
- }RCC_OscInitTypeDef;
复制代码
5 c& x) x$ o$ y3 r1 o注意事项:, S( `# ^ H' K4 w. ^
1 q( }# d8 _0 H5 ^# ULSE Bypass 切换到 LSE On 或者 LSE On切换到 LSE Bypass都不支持,用需要先关闭LSE,然后才可以切换到LSE Bypass 或者LSE On。8 @7 ]0 {- V( W
HSE Bypass 切换到 HSE On 或者 LSE On切换到 LSE Bypass都不支持,用需要先关闭HSE,然后才可以切换到HSE Bypass 或者HSE On。
9 w7 p" E, j! Z' n+ A) C使用举例:
$ l4 _) S6 U' F v( z# c
: ?7 g. H. e3 e- RCC_OscInitTypeDef RCC_OscInitStruct;
9 p( B- u& ?0 K+ [" r( i& Z% t - HAL_StatusTypeDef ret = HAL_OK;# x/ w) v( K9 t- k3 o# f4 e
. _% T9 k# }. p7 b& e9 {- /* 使能HSE,并选择HSE作为PLL时钟源 */
" k4 U; f3 U8 h9 \8 H - RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
c! L6 w$ S* P' R8 T5 T - RCC_OscInitStruct.HSEState = RCC_HSE_ON;
+ d+ s- v9 H$ X6 [- g - RCC_OscInitStruct.HSIState = RCC_HSI_OFF;( a W6 q3 J: _9 Q' J3 ]
- RCC_OscInitStruct.CSIState = RCC_CSI_OFF;
# f+ E# f9 p9 N) B - RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;; {, [! ]; Y* r' Y8 C2 B: y
- RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
+ e8 `$ U9 p- Y7 k9 [- Q -
& i9 U. |& w5 Q2 m$ c+ I - RCC_OscInitStruct.PLL.PLLM = 5;' Y9 E$ | _" `8 _- ?9 c' h
- RCC_OscInitStruct.PLL.PLLN = 160;3 R# a* F) k. R( {9 E4 R
- RCC_OscInitStruct.PLL.PLLP = 2;
9 E* |3 e, ?- a. `# R6 K2 g' d - RCC_OscInitStruct.PLL.PLLR = 2;7 P) e' W" U4 q4 p! k9 a
- RCC_OscInitStruct.PLL.PLLQ = 4; 5 _$ r4 X [2 E% q3 h0 P& v
-
9 w! E1 c0 n! w3 n. {: ?+ ^+ q4 T - RCC_OscInitStruct.PLL.PLLVCOSEL = RCC_PLL1VCOWIDE;
# }, {, `# L3 h - RCC_OscInitStruct.PLL.PLLRGE = RCC_PLL1VCIRANGE_2;
, D; C/ W) i( z# _9 [ - ret = HAL_RCC_OscConfig(&RCC_OscInitStruct); J6 v4 f, l+ n( U1 g6 C3 b' A
- if(ret != HAL_OK), A A9 G5 k1 ^" v
- {, T, Y( u. b3 C% G% H8 P' @
- Error_Handler(__FILE__, __LINE__);% J% L) v0 T; }$ O. @
- }
复制代码 # L+ g7 N! L, y
16.4.3 函数HAL_RCC_ClockConfig/ u% L' ]% h, x8 C* i$ P: N
函数原型: ]4 h- ]& m8 }3 V3 n4 B7 o
7 Q# p2 D8 u7 c! H% h0 C# [! A/ b) ~. B
2 w H! ?, I4 ]9 P$ O9 K/ _函数描述:5 t$ P+ M- x5 B( r. W4 Z- q# i
通过上面函数原型,我们可以一目了然的看出此函数的作用,配置了CPU、AHB和APB的所有总线时钟,即HCLK,SYSTICK,D1PCLK1,PCLK1,PCLK2和D3PCLK1时钟。这里的D1PCLK1其实就是PCLK3,而D3PCLK1是PCLK4。
1 a- y% W6 D& B4 j1 d: v% r8 f. R& h
$ h' g3 Y; N- a9 I- P函数参数:. f6 q& ~- G8 G1 v3 a6 D
第1个参数是RCC_ClkInitTypeDef类型的结构体变量。
- f# ]7 ^; X! e% w 第2个参数是Flash的延迟设置。
% L+ m) B; E9 y9 _0 T9 ^" ?$ N3 y: H 返回值,返回HAL_ERROR表示参数错误,HAL_OK表示发送成功,HAL_BUSY表示忙,正在使用中。- K. b. r8 R- O' U8 c
4 D0 @6 J- m- `0 r; s, ?; ^
7 m/ \* p. V2 e6 N! T! ?# M3 `4 U注意事项:# U6 m5 z: l1 q* O' y: ]- t
此函数会更新全局变量SystemCoreClock的主频值,并且会再次调用函数HAL_InitTick更新系统滴答时钟,这点要特别注意。
. y" N# O4 O3 o4 r7 E' h系统上电复位或者从停机、待机模式唤醒后,使用的是HSI作为系统时钟。以防使用HSE直接或者通过PLL输出后做系统时钟时失败(如果使能了CSS)。
% Y- T+ X) P" K' T+ r目标时钟就绪后才可以从当前的时钟源往这个目标时钟源切换,如果目标时钟源没有就绪,就会等待直到时钟源就绪才可以切换。9 t! f: J' ~# d, X2 ~: z! N
根据设备的供电范围,必须正确设置D1CPRE[3:0]位的范围,防止超过允许的最大频率。
" {+ s' z3 I, w4 k' s% C: }( r( c- b# W2 C" q
( n/ a6 C! m. @2 [
使用举例:
" o2 E- I9 W. B, o8 ^7 a9 x1 K1 l8 v4 l
- RCC_ClkInitTypeDef RCC_ClkInitStruct;+ O j% f; C( n0 |
- HAL_StatusTypeDef ret = HAL_OK;
6 t- A9 ?9 C7 d1 H4 Z
/ C' ^ `4 |# R. `- /*
/ k2 R+ Y) y2 [, s, k; z& ? - 选择PLL的输出作为系统时钟$ g/ j& C' C- S6 H7 W0 R) d/ Z
- 配置RCC_CLOCKTYPE_SYSCLK系统时钟
5 p3 y( A5 D3 }1 n - 配置RCC_CLOCKTYPE_HCLK 时钟,对应AHB1,AHB2,AHB3和AHB4总线3 S- D" }2 Z$ ~6 K5 ~+ P2 {
- 配置RCC_CLOCKTYPE_PCLK1时钟,对应APB1总线
, [3 }7 E( c& Q& ^ - 配置RCC_CLOCKTYPE_PCLK2时钟,对应APB2总线
& W5 r1 D" E# M: P' k7 F: { - 配置RCC_CLOCKTYPE_D1PCLK1时钟,对应APB3总线
4 i9 {4 E% e. m8 D% }# a - 配置RCC_CLOCKTYPE_D3PCLK1时钟,对应APB4总线
0 Z6 T& |+ S1 s" m - */
! L8 `( r/ D: I& X1 Y b - RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_D1PCLK1 |
4 E. F* v8 S# m( ? M6 x1 C, j' a - RCC_CLOCKTYPE_PCLK1 | \
1 n- ?! o: ]) e2 _. b - RCC_CLOCKTYPE_PCLK2 | RCC_CLOCKTYPE_D3PCLK1);! G3 a4 w! O& B. F+ M$ T
- 1 M9 z+ J" I* l
- RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;% w7 @% W* `9 N
- RCC_ClkInitStruct.SYSCLKDivider = RCC_SYSCLK_DIV1;
+ Z$ r! {5 J8 A) y$ e; C - RCC_ClkInitStruct.AHBCLKDivider = RCC_HCLK_DIV2;3 _% ?3 R* J3 l$ k
- RCC_ClkInitStruct.APB3CLKDivider = RCC_APB3_DIV2;
2 R# @7 y: H$ D/ w - RCC_ClkInitStruct.APB1CLKDivider = RCC_APB1_DIV2; 9 _- \3 d9 z( Q, w
- RCC_ClkInitStruct.APB2CLKDivider = RCC_APB2_DIV2; 0 A! r0 y0 V. ?, W& o
- RCC_ClkInitStruct.APB4CLKDivider = RCC_APB4_DIV2; & N2 C$ c4 R: u
- / s h2 e5 V* D/ J3 w/ G0 l
- /* 此函数会更新SystemCoreClock,并重新配置HAL_InitTick */! M9 V# M6 B0 e+ h! G0 {
- ret = HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_4);( ~: e" [) O- X# b% B, m
- if(ret != HAL_OK)
4 |& |& E7 U5 ]$ ?# \4 k - {
$ x1 F9 e! E; y - Error_Handler(__FILE__, __LINE__);/ f) B* s0 @: r" s4 H! g& D' d7 ?# t
- }
复制代码
" h4 K2 [9 x/ [8 W! X! C3 z16.4.4 函数HAL_RCC_MCOConfig
/ G" m; r6 o; o( U0 T0 o函数原型:: A( i! N$ F0 P; Q
7 e \" T0 a9 Z, ~+ V- ]! F- void HAL_RCC_MCOConfig(uint32_t RCC_MCOx, uint32_t RCC_MCOSource, uint32_t RCC_MCODiv)/ P/ l$ ^# z9 z5 N
- {
% r L) }7 p a, l( a$ [. u5 L - GPIO_InitTypeDef GPIO_InitStruct;1 e! c& @7 {9 E. i0 f/ w$ A' U
- /* Check the parameters */
0 U; z; D- ^- m" n2 G - assert_param(IS_RCC_MCO(RCC_MCOx));! j) z2 v. l) d" p6 w
- assert_param(IS_RCC_MCODIV(RCC_MCODiv));
( n& F$ ]/ o2 F7 l4 S
' ?( I( i" q3 `% {1 h- /* RCC_MCO1 */
8 Q8 X! y2 W) e) H2 T3 x2 g1 X$ } - if(RCC_MCOx == RCC_MCO1)
% v0 Z9 N# g1 k1 ]0 ? ^ - {
. r! j4 q0 t1 A2 Q- r - /* 省略未写 */4 P" r% p6 m! s; m" m) j8 ^( d
- }
- C0 B4 o& U# R) |6 j - else! l+ g& l$ G5 M/ ^3 @1 t0 W% T
- {
' {3 p" y5 s. F# [ - /* 省略未写 */9 o G9 _' L/ }8 M E
- }$ j1 d1 Z# C5 L* d, @
- }
复制代码
d8 V5 T$ G6 H9 D% [* I* @7 f函数描述:, m- j: z# T: _/ d u; ]
此函数的作用是配置MCO1(PA8引脚)和MCO2(PC9引脚)的时钟输出以及选择的时钟源,通过下面的截图可以很好的说明此函数的作用:
8 d& z; b6 j; U
0 C9 S8 c! N; j+ Y |* V5 |: V7 E- Z* A' D- s( \! u% c
" X( j$ ~' u7 K0 M# b
函数参数:
. K9 _- G" Z' V) N I! t 第1个形参用于选择输出的引脚,可选择RCC_MCO1(PA8引脚)或者RCC_MCO2(PC9引脚)。3 g( K& _9 h( y8 M, I
第2个形参用于选择输出的时钟源,MCO1可选择的时钟源如下:' n" }4 g7 _8 K2 y
RCC_MCO1SOURCE_HSI& Y$ O/ Q( D6 k
RCC_MCO1SOURCE_LSE* A4 `% N z$ F: Q+ q4 ?5 T
RCC_MCO1SOURCE_HSE( `8 V: B8 E9 x
RCC_MCO1SOURCE_PLL1QCLK- d. j# B' E' H( M0 C7 A
RCC_MCO1SOURCE_HSI485 J1 O" U8 `8 j1 ?
. y/ ? o( r5 G2 q
( q( E/ M/ A) {6 H' m; d5 }MCO1可选择的时钟源如下:3 N$ i! g" b4 u: [7 M5 R' F
RCC_MCO2SOURCE_SYSCLK& y% ]5 R) n2 Y$ ~2 K- ^
RCC_MCO2SOURCE_PLL2PCLK! h- Z% E) Y) N/ U! [; D( |* v' ]* ^
RCC_MCO2SOURCE_HSE7 \- [5 R8 L, W( Q. h3 U4 ~
RCC_MCO2SOURCE_PLLCLK
( v/ j- ^0 T; l) j: T6 b RCC_MCO2SOURCE_CSICLK/ I, F5 u+ w$ M8 ~' |1 N: Z
RCC_MCO2SOURCE_LSICLK5 g, E1 Q3 `* i) W; ]( z% z: Q
1 ~8 ]' g" i* O4 b第3个参数用于设置输出分频,范围从RCC_MCODIV_1到RCC_MCODIV_15。+ r4 |& R7 k" i$ h9 A
- m. i0 t9 e* X u; A4 D
2 j3 H |1 [3 G+ P; }
使用举例:# k+ z& i ~+ l. S* y6 M; } X w
此函数的使用比较简单,需要在MCO引脚输出时钟的时候,直接调用即可。
$ P* w) N& m% {+ O. _
; C6 T/ |& g5 C! O0 Y1 {1 M16.4.5 RCC相关函数
' ~8 \2 Z/ n, W1 m- F函数原型:
! N7 Z6 r7 _# p4 ?6 s. e
6 p6 V$ |0 t( V- void HAL_RCC_EnableCSS(void); Q& g. A/ v; x
- uint32_t HAL_RCC_GetSysClockFreq(void)
+ K4 ?8 T. Y- c3 k5 ?: v0 S2 ^ - uint32_t HAL_RCC_GetHCLKFreq(void)5 j: |0 ~( Q% c; m) {
- uint32_t HAL_RCC_GetPCLK1Freq(void)
3 ~0 z2 b9 l' b - uint32_t HAL_RCC_GetPCLK2Freq(void); }: T/ d5 d E; W. t
- void HAL_RCC_GetOscConfig(RCC_OscInitTypeDef *RCC_OscInitStruct)+ ]- l. J( A6 Y) z: a7 w
- void HAL_RCC_GetClockConfig(RCC_ClkInitTypeDef *RCC_ClkInitStruct, uint32_t *pFLatency)
复制代码 1 W+ y7 I! m4 _, Y. `: o% E7 Z8 D
剩下的这些函数都比较简单了,我们这里就不做讲解了。
( |0 P" |4 _7 ]* B& H1 y. l
4 ?# v2 J) F- w' l1 P N16.5 源文件stm32h7xx_hal_cortex.c
8 {& l9 o) d5 B" T9 v9 S/ J这个库文件主要功能是NVIC,MPU和Systick的配置。此文件有个臃肿的地方,里面的API其实就是将ARM的CMSIS库各种API重新封装了一遍。这么做的好处是保证了HAL的API都是以字母HAL开头。
# F: m7 ~/ Y( B+ D. I0 B1 w' [8 x" h
更多NVIC相关知识需要大家看第21章节。4 P" o8 D6 ^ [8 I. u2 W
% E0 c3 p% L- n% y
更多SysTick相关知识需要大家看第22章节。
+ a! {/ a0 m/ a5 q8 Q) a) I+ t) ~2 Q2 S
3 c2 C4 A j6 v5 G更多MPU相关知识需要大家看第23章节。/ x% _6 V2 B) e7 k/ T1 P' R
: m* x+ B R( d K1 \$ z
16.5.1 NVIC相关函数1 _* o3 q( M2 _! F8 h9 X/ O
函数原型:4 o- u' H/ l2 A4 J8 H U9 R1 a, {
- void HAL_NVIC_SetPriorityGrouping(uint32_t PriorityGroup)
( ? `. h, a, V; V- m; i- N - void HAL_NVIC_SetPriority(IRQn_Type IRQn, uint32_t PreemptPriority, uint32_t SubPriority)$ V5 Q) M. r1 f3 {
- void HAL_NVIC_EnableIRQ(IRQn_Type IRQn)6 q' _: ~0 |: k9 V" f% M7 {
- void HAL_NVIC_DisableIRQ(IRQn_Type IRQn)1 \: `/ w8 a6 T- l. r3 b' w
- void HAL_NVIC_SystemReset(void)- D7 }+ ?4 i( x, |
- uint32_t HAL_NVIC_GetPriorityGrouping(void)
" |* f3 M, x" v( K6 X" N* E a2 l - void HAL_NVIC_GetPriority(IRQn_Type IRQn, uint32_t PriorityGroup, uint32_t *pPreemptPriority, uint32_t *pSubPriority)
7 }* b4 }" i0 n8 r) l4 r6 y' _: } - void HAL_NVIC_SetPendingIRQ(IRQn_Type IRQn)
! Q7 I6 ?% K1 @1 }, c - uint32_t HAL_NVIC_GetPendingIRQ(IRQn_Type IRQn)
* G9 @. N) t# t2 p2 n' I* t6 m - void HAL_NVIC_ClearPendingIRQ(IRQn_Type IRQn)8 z1 W% W3 x5 ~
- uint32_t HAL_NVIC_GetActive(IRQn_Type IRQn)
复制代码
* l% r( A2 G- p; Q# j- p% ^1 s( p这些函数的操作就比较容易了,这里就不做讲解了,用到时调用即可。: t% G7 ]$ j) R2 Z+ |
) s' w+ J9 p& N+ i/ P- v16.5.2 SysTick相关函数3 F) `0 }+ {0 [& O9 b3 i8 r/ x+ ~
函数原型:
+ \- T: I- E; y$ K6 H! U8 U- uint32_t HAL_SYSTICK_Config(uint32_t TicksNumb)6 B, A Y2 h5 r( p
- void HAL_SYSTICK_CLKSourceConfig(uint32_t CLKSource)
* ^' I/ ]& r2 y; I* \" r - void HAL_SYSTICK_IRQHandler(void)6 C/ Z4 u6 p9 N8 p2 F! p2 G
- __weak void HAL_SYSTICK_Callback(void)
复制代码
4 v, e/ D4 e5 p) Z9 y- K- `这些函数的操作就比较容易了,这里就不做讲解了,用到时调用即可。6 Y6 \9 \$ C& m) D3 S1 _$ |
1 y' P5 S+ S, h1 e: P( I16.5.3 函数HAL_MPU_ConfigRegion4 p: J# N; Q5 Z. n6 I1 b
函数原型:
' V9 w2 _, p; x- E; P
" Y0 c/ U1 y( l6 L: e; X- void HAL_MPU_ConfigRegion(MPU_Region_InitTypeDef *MPU_Init)8 |# h6 H s! y: W4 ~# ^+ l& ?
- {8 f* N5 U, X9 m. q
- /* 部分省略未写 */
4 u: N* P* i1 C) p) n) s6 T# Z - , A2 r! O, Y( w
- /* Set the Region number */
' D! U, J) [4 S/ A, q - MPU->RNR = MPU_Init->Number;
" I/ U$ H: d8 @* l" X* p+ h) W
* A! _+ v9 ]6 s% _7 N$ L) Y- if ((MPU_Init->Enable) != RESET)* F; V' T5 v; Y. W, m; }+ Z! e8 \
- {& f/ v' _& |) r3 u( c
- /* 部分省略未写 */4 t6 q. T9 P' U% ]3 ?0 ~3 U
- 7 b) i V2 i- H8 {3 A* h3 {* ]
- MPU->RBAR = MPU_Init->BaseAddress;6 w9 j: e! z# |0 t9 E2 x1 ^
- MPU->RASR = ((uint32_t)MPU_Init->DisableExec << MPU_RASR_XN_Pos) |5 G1 s! o! n' t3 e- B5 v t: y& z- E
- ((uint32_t)MPU_Init->AccessPermission << MPU_RASR_AP_Pos) |
3 r8 D @6 t- G0 H' { }) M - ((uint32_t)MPU_Init->TypeExtField << MPU_RASR_TEX_Pos) |; e( ?8 r, R' o6 ^4 p1 M4 U4 z* W
- ((uint32_t)MPU_Init->IsShareable << MPU_RASR_S_Pos) |2 v+ g6 v8 w0 N3 R2 Y
- ((uint32_t)MPU_Init->IsCacheable << MPU_RASR_C_Pos) |
: X3 y. D! V# I+ o; p - ((uint32_t)MPU_Init->IsBufferable << MPU_RASR_B_Pos) |4 D# {, ^6 D2 Z* \8 `( u. p- J
- ((uint32_t)MPU_Init->SubRegionDisable << MPU_RASR_SRD_Pos) |
5 A7 G0 I L0 y* ~2 ] - ((uint32_t)MPU_Init->Size << MPU_RASR_SIZE_Pos) |
3 A) T' p* }9 O' b - ((uint32_t)MPU_Init->Enable << MPU_RASR_ENABLE_Pos);
/ J( G# U- k. Q6 C - }
/ v- ^3 }/ N! H B$ T/ } - else
- J. \: X% Y, g. y5 }# C0 f9 @% b( S - {
7 z: q6 C( _6 Y& u - MPU->RBAR = 0x00;" U: H* q: y7 {# c, z
- MPU->RASR = 0x00;
1 Q) B6 b/ b7 {5 V/ M - }0 z. J1 C" s R
- }
复制代码
5 j6 ^; \/ ^: }; W) e* z. [' D此函数在本教程第23章有专门的讲解。
. \# w% C8 x2 v9 O
5 O" Z4 A8 H) Y; k1 ?/ ]7 h* [3 D16.6 总结- z! a% f0 i4 C1 @
本章节就为大家讲解这么多,对于一些常用的函数,望大家务必要掌握。
- _% N4 N5 s2 H
5 U7 ^5 A( Y$ w8 {9 e
9 N& b* x5 V; z. n* @9 A w6 I9 l) V3 k9 y! K
3 T, }1 {) [" Y+ w
5 m; A9 s, | W) t- X |