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

【经验分享】STM32F103系列(八):初识STM2标准库

[复制链接]
STMCU小助手 发布时间:2021-11-30 22:34
1、CMSIS标准及库层次关系
9 y3 ?5 c; |( F  p# Y基于Context系列芯片采用的内核都是相同的,区别主要为核外的片上外设的差异,而这些差异却导致软件在同内核、不同外设的芯片上移植困难。
( o' [8 v2 s7 k& d7 M为了解决不同的芯片厂商生产的Context微控制器软件的兼容性问题,ARM与芯片厂商建立了CMSIS标准(Context Micro Controller Software Interface Standard)。/ F" |* j6 m' o( F9 t
所谓CMSIS标准,实际是新建了一个软件抽象层,见图1-1+ N* N2 j0 A6 f0 O' g; M
20200507120021750.jpg
* {, v: f% l+ T! \$ M0 K. v
. u: G5 v5 O! q9 T2 G% i. Z$ W% M
CMSIS标准中最主要的是CMSIS核心层,它包括以下两部分。
4 [, A3 x  N* j2 r. P: n$ c& T. m- d2 F: R$ k0 r+ t
·内核函数层:其中包含用于访问内核寄存器的名称、地址定义,主要由ARM公司提供。  r( w6 G6 H: \" b# D
·设备外设访问层:提供了片上的核外外设的地址和中断定义,主要由芯片生产商提供。- |8 c0 A% g  B0 D* J) J( q2 t/ M
0 E# s9 v( }# M6 j( y$ t- c
可见CMSIS层位于硬件层与操作系统或用户层之间,提供了与芯片生产商无关的硬件抽象层,可以为接口外设、实时操作系统提供简单的处理器软件接口,屏蔽了硬件差异,这对软件的移植是有极大好处的。STM32的库就是按照CMSIS标准建立的。7 y; W: z3 Z  K$ G1 L. L* W

  @0 C' }' q+ e2 库目录、文件简介
$ f- ~8 |3 }5 |, U" |. sSTM32标准库可以从官网获得,也可以直接从论坛中的配套资料得到。本书讲解的例程全部采用3.5.0库文件。以下内容请打开STM32标准库文件配合阅读。
7 V: j+ Q/ `3 s4 n; m
- A) U( h( \! g# H解压库文件后进入其目录STM32F10x_Std Periph_Lib_V3.5.0\,软件库各文件夹的内容说明见图1-2。6 t- Q( t3 E: A( D$ r0 v, T
, f; c# ^/ L, _' L, s  G
20200507120717512.jpg

9 @8 O7 o0 X2 x/ G
图1-2 ST标准库
2 _8 j+ \* Y1 `  u
) t) x) c4 j' |  n2 U/ ^2 g1 U
·Libraries:文件夹下是驱动库的源代码及启动文件,这个文件夹非常重要,我们要使用的固件库就在这个文件夹里面。
, d. Y$ K- v; I: t. A, ?" V+ P; p·Project:文件夹下是用驱动库写的例子和工程模板,其中那些为每个外设写好的例程对我们非常有用,在学习的时候可以参考这里面的例程,非常全面,简直就是穷尽了外设的所有功能。
. h7 h0 y# |; H: {& J·Utilities:包含了基于ST官方实验板的例程,不需要用到,略过即可。
; r0 j) W3 f7 p  p2 `) B2 N; d·stm32f10x_stdperiph_lib_um.chm:库帮助文档,这个很有用,不喜欢直接看源码的用户可以在这里查询每个外设的函数说明,非常详细。这是一个已经编译好的HTML文件,主要讲述如何使用驱动库来编写自己的应用程序。说得形象一点,这个HTML就是告诉我们:ST公司已经为你写好了每个外设的驱动了,想知道如何运用这些例子就来向我求救吧。但是,这个帮助文档是英文的,这对很多英文不好的朋友来说是一个很大的障碍。但这里要告诉大家,英文仅仅是一种工具,绝对不能让它成为我们学习的障碍。
. c4 ^9 F6 K9 f7 w! ^' v- \& A) B! T
# V* o6 d; c. P! A0 L3 b在使用库开发时,我们需要把libraries目录下的库函数文件添加到工程中,并查阅库帮助文档来了解ST提供的库函数,这个文档说明了每一个库函数的使用方法。
; m; Y( f: {. o4 R4 h9 f; e
- Z3 p; t5 J. G; D  ?进入Libraries文件夹,可以看到关于内核与外设的库文件分别存放在CMSIS和STM32F10x_Std Periph_Driver文件夹中。
+ z8 K6 F; N4 Y( b
- N. m$ c1 ?$ r3 n! d. T1.CMSIS文件夹5 {7 F: W) R% C7 n; k
STM32F10x_Std Periph_Lib_V3.5.0\Libraries\CMSIS\文件夹展开内容见图1-3。- |& E6 P! Z5 l0 Y& C
, f9 C2 Y1 D/ g* \6 L, Y6 a' B
20200507120744306.jpg

9 `0 s# {8 x0 I6 V) Y. p0 a% y  N
图1-3 CMSIS文件夹内容
, o6 p5 s* Q; A+ M9 q3 t5 ?) N

/ Y- g. i; O0 k8 Q1 ?. I' M& ]4 s! _: H. n
其中带阴影的文件是我们需要用到的内容,下面我们一一讲解这几个文件的作用。, R, x7 _) S5 E
7 M- G+ q. H, E1 A
(1)内核相关文件/ J9 A- W# ^) B% z* }
在Core Support文件夹中有core_cm3.c和core_cm3.h两个文件。core_cm3.h头文件里面实现了内核的寄存器映射,对应外设头文件stm32f10x.h,区别就是一个针对内核的外设,一个针对片上(内核之外)的外设。core_cm3.c文件实现了操作内核外部寄存器的函数,用得比较少。
) `3 m6 u8 n3 B% B; ]
' V7 h0 D8 U, Y" w* z" B# x' ^! V* c9 Y我们还需要了解的是core_cm3.h头文件中包含了stdint.h这个头文件,这是一个ANSI C文件,是独立于处理器之外的,就像我们熟知的C语言头文件stdio.h文件一样。它位于RVMDK这个软件的安装目录下,主要作用是提供一些类型定义,见代码清单9-1。
$ N( I* m9 m, C- b代码清单9-1 stdint.h文件中的类型定义9 o6 R7 q8 W/ |
  1.     1 /* exact-width signed integer types */
    6 t& g* X/ g, ~* d7 M8 ]; c% B! v
  2.     2 lt@span b=1>typedef   signed
    0 |$ y4 d0 d: E  d9 S1 D
  3.     char int8_t;3 {  [' ~$ G0 a- X  N
  4.     3 typedef   signed short     int int16_t;
    6 G  ]3 d+ D, N( [
  5.     4 typedef   signed            int int32_t;1 F) ~) i' q2 C% p8 c( {4 C
  6.     5 typedef   signed        __int64 int64_t;5 r+ A- I5 e7 Q* ]+ X
  7.     6* T. U5 _) Q+ ]" w1 S
  8.     7 /* exact-width unsigned integer types */0 Z) e' G1 W( c) J
  9.     8 typedef unsigned           char uint8_t;4 n4 q' b  }; T
  10.     9 typedef unsigned short     int uint16_t;0 l* E! P( Z8 b. ~% v
  11.    10 typedef unsigned            int uint32_t;. [" G0 d+ g0 N
  12.    11 typedef unsigned        __int64 uint64_t;
复制代码

6 @7 w4 L; N% M4 F( |! n这些新类型定义屏蔽了在不同芯片平台时,出现的诸如int的大小是16位,还是32位的差异。所以在我们以后的程序中,都将使用新类型,如uint8_t 、uint16_t等。
7 _' P) B6 p9 r5 ^
: W' f% t+ B2 B% E6 M0 D在稍旧版的程序中还经常会出现如u8、u16、u32这样的类型,分别表示无符号的8位、16位、32位整型。初学者碰到这样的旧类型会感觉一头雾水,它们定义的位置在STM32f10x.h文件中。建议在以后的新程序中尽量使用uint8_t 、uint16_t类型的定义。
$ i& ~- o$ N- ^% e0 _: ~
3 A4 q$ D* H- D/ s(2)启动文件! W/ _3 C! Q. k* W1 W1 ]7 E
启动文件放在startup/arm这个文件夹下面,这里面启动文件有很多个,不同型号的单片机用的启动文件不一样,有关每个启动文件的详细说明见表1-1。2 b0 v; G! \; s8 w
表1-1 各启动文件匹配的芯片类型* i" u1 e9 `8 v8 j# S. }9 J
* G( U5 x: f4 f/ }' ~9 w
20200507120849621.jpg

3 J7 I; P# x  A3 K' f7 t% o2 ]+ X& n$ j: ]4 |+ h2 @) L7 {' U
我们开发板中用的STM32F103VET6或者STM32F103ZET6中的Flash都是512k,属于基本型的大容量产品,启动文件统一选择startup_stm32f10x_hd.s。& v- U5 w% \1 a- D  B; C1 v

  r+ A  S% f7 x8 V+ t' g& L: Q* i(3)Stm32f10x.h
* v6 r3 b2 v# N3 t9 v% r+ g+ u这个头文件实现了片上外设的所以寄存器的映射,是一个非常重要的头文件,在内核中与之相对应的头文件是core_cm3.h。% a, N0 C$ p2 g( `" x

7 @! L" B8 }# z/ A5 g  ~: q(4)system_stm32f10x.c" x+ r" r' v* _9 s# f
system_stm32f10x.c文件实现了STM32的时钟配置,操作的是片上的RCC这个外设。系统在上电之后,首先会执行由汇编编写的启动文件,启动文件中的复位函数中调用的System Init函数就在这个文件里面定义。调用完之后,系统的时钟就被初始化成72M。如果后面我们需要重新配置系统时钟,我们就可以参考这个函数重写。为了维持库的完整性,我们不会直接在这个文件里面修改时钟配置函数。6 [6 m6 W" T2 g' b) q. u3 H+ p" M3 h
" `7 v9 O4 n' K* E
3.STM32F10x_Std Periph_Driver文件夹
6 j9 E; P- m; ?, C7 r+ o' E4 Clibraries目录下的STM32F10x_Std Periph_Driver文件夹见图1-4。  p  L9 v" M0 C6 d1 y: z: j0 k
图1-4 外设驱动4 _4 M6 j" n6 A0 X9 U% n
2 F, i  G2 @! y' i7 Y
20200507121543617.jpg

. s8 N# w5 d4 `! V& J1 V" ^( C4 T0 L
STM32F10x_Std Periph_Driver文件夹下有inc(include的缩写)与src(source的缩写)这两个文件夹,这里的文件属于CMSIS之外的、芯片片上的外设部分。src里面是每个设备外设的驱动源程序,inc则是相对应的外设头文件。src及inc文件夹是ST标准库的主要内容,不少人甚至认为ST标准库就是指这些文件,可见其重要性。6 U6 H" b) `1 {6 C

% N9 t6 m) K' c4 B在src和inc文件夹里的就是ST公司针对每个STM32外设而编写的库函数文件,每个外设对应一个.c和.h后缀的文件。我们把这类外设文件统称为stm32f10x_ppp.c或stm32f10x_ppp.h文件,ppp表示外设名称。如在上一章中我们自建的stm32f10x_gpio.c及stm32f10x_gpio.h文件,就属于这一类。: N- C" ~4 H# w5 l: d
5 ~$ o0 j7 |3 x
如针对模数转换(ADC)外设,在src文件夹下有一个stm32f10x_adc.c源文件,在inc文件夹下有一个stm32f10x_adc.h头文件,若我们开发的工程中用到了STM32内部的ADC,则至少要把这两个文件包含到工程里,见图1-5。
2 A( ~0 @1 M; z6 O- W" Y* G. ]- p3 `  y
2020050712145141.jpg
# X+ q4 Q0 L' C# G* u' a- @: }
  N5 z% g0 N7 i* x. N
图1-5 驱动的源文件及头文件; l3 T% J: N. m" e. a

- S- A* C1 w. S% q这两个文件夹中,还有一个很特别的misc.c文件,这个文件提供了外设对内核中的NVIC(中断向量控制器)的访问函数,在配置中断时,必须把这个文件添加到工程中。
! m+ l9 p( ~, N# V# @, A# d
% ^+ N" @) n( X8 @# t% n* b) F3.stm32f10x_it.c、stm32f10x_conf.h和system_stm32f10x.c文件6 e' y( V3 `2 Y
在文件目录STM32F10x_Std Periph_Lib_V3.5.0\Project\STM32F10x_Std Periph_Template下,存放了官方的一个库工程模板,在用库建立一个完整的工程时,还需要添加这个目录下的stm32f10x_it.c、stm32f10x_conf.h和system_stm32f10x.c这4个文件。3 ]' C7 q) x% p- G' u5 y% S

5 b4 B6 Z) Z- m5 _( t& A9 I(1)stm32f10x_it.c
0 A6 k, V" Y, d这个文件是专门用来编写中断服务函数的,在我们修改前,这个文件已经定义了一些系统异常(特殊中断)的接口,其他普通中断服务函数由我们自己添加。但是我们怎么知道这些中断服务函数的接口如何写呢?是不是可以自定义呢?答案当然不是,这些都可以在汇编启动文件中找到,在学**断和启动文件的时候会详细介绍。
2 I2 _/ T; T  H9 ^5 {1 F, Z$ ?+ o' V1 @4 ^% {9 z
(2)system_stm32f10x.c0 e- w" X3 [4 h8 j0 @2 O
这个文件包含了STM32芯片上电后初始化系统时钟、扩展外部存储器用的函数,例如我们前两章提到供启动文件调用的System Init函数,用于上电后初始化时钟,该函数的定义就存储在system_stm32f10x.c文件中。STM32F103系列的芯片,调用库的这个System Init函数后,系统时钟被初始化为72MHz,如需要可以修改这个文件的内容,设置成自己所需的时钟频率。但鉴于保持库的完整性,我们在做系统时钟配置的时候会另外重写时钟配置函数。1 [4 P# t& H4 V  o

% H/ e% t; [: x" k( f( S(3)stm32f10x_conf.h& a5 C, s  p4 N; K# [& I
这个文件被包含进stm32f10x.h文件。当使用固件库编程的时候,如果需要某个外设的驱动库,就需要包含该外设的头文件:stm32f10x_ppp.h。包含一个还好,如果用了多个外设,就需要包含多个头文件,这不仅影响代码美观,而且也不好管理。现我们用一个头文件stm32f10x_conf.h把这些外设的头文件都包含在里面,让这个配置头文件统一管理这些外设的头文件,我们在应用程序中只需要包含这个配置头文件即可。我们又知道这个头文件在stm32f10x.h的最后被包含,所以最终我们只需要包含stm32f10x.h这个头文件即可,非常方便。Stm32f10x_conf.h见代码清单9-2。默认情况下是所有头文件都被包含,没有被注释掉。也可以把不要的都注释掉,只留下需要使用的即可。9 p1 _: |0 F8 [. N, J) n

- P2 I! o9 |# C+ g代码清单1-2 stm32f10x_conf.h文件配置软件库
( u4 M+ e. P( p0 j7 E
  1.    1 #include“stm32f10x_adc.h”
    ( ?; p' K; `5 G+ s% e2 K
  2.    2 #include“stm32f10x_bkp.h”
    . @: S( t: f. [# h& y
  3.    3 #include“stm32f10x_can.h”4 x- ^8 j4 S! N$ |8 W" Y* p& V7 \( a
  4.    4 #include“stm32f10x_cec.h”' @- g* q; k( y
  5.    5 #include“stm32f10x_crc.h”
    2 N% {: H2 w+ H' [% r3 b5 V
  6.    6 #include“stm32f10x_dac.h”
    # U/ i3 Q7 A7 Y" m5 ]; H8 s% {
  7.    7 #include“stm32f10x_dbgmcu.h”
    1 [8 p$ z1 t9 o4 ~0 b8 K6 I
  8.    8 #include“stm32f10x_dma.h”2 `9 w7 b! Q) ]/ `" B' ^5 Y$ m* J
  9.    9 #include“stm32f10x_exti.h”6 @; r5 z' n9 T/ `* J
  10.   10 #include“stm32f10x_flash.h”
    # D; e3 |$ y- i
  11.   11 #include“stm32f10x_fsmc.h”
    8 T. J3 \9 F" V4 x$ Y
  12.   12 #include“stm32f10x_gpio.h”
    + i, K* X7 K! {+ V# l7 W
  13.   13 #include“stm32f10x_i2c.h”: q0 K) M7 g9 C; _7 B/ ]8 V2 d
  14.   14 #include“stm32f10x_iwdg.h”
    / b" a+ \" w' u
  15.   15 #include“stm32f10x_pwr.h”
    ; ^  m( \# w* P9 V" w- @. @4 Q
  16.   16 #include“stm32f10x_rcc.h”
    6 O1 S7 F, C" [! W) X5 b8 j7 C* R
  17.   17 #include“stm32f10x_rtc.h”- {: X8 D. |- l! L  |" L/ Z! k* s; L
  18.   18 #include“stm32f10x_sdio.h”  j- ?1 w3 Z( i+ _, e+ {
  19.   19 #include“stm32f10x_spi.h”
    , P) w6 N+ Q  W
  20.   20 #include“stm32f10x_tim.h”
    2 p8 z& r) @8 \/ S2 [
  21.   21 #include“stm32f10x_usart.h”9 [3 \( A" ~1 W; I4 P5 `
  22.   22 #include“stm32f10x_wwdg.h”
    $ [; w. N+ D1 a& ?2 K6 V4 E6 `
  23.   23 #include“misc.h”
复制代码

* t$ [- n6 y" p: q$ Wstm32f10x_conf.h这个文件还可配置是否使用“断言”编译选项,见代码清单1-3。! C; t$ L+ E+ T* T: @% N. n' W

0 Y+ j8 H/ E; o4 }* n代码清单1-3 断言配置
! P0 T- d1 g9 |
  1.    1 #ifdef  USE_FULL_ASSERT
    9 ~5 a% B: s- C% r4 `
  2.    2 /*** `0 M: `, v7 W( B- |6 Z) Y
  3.    3     * @brief  The assert_param macro is used for  parameters check.
    " T* L7 n% s& \$ j* @# `
  4.    4     * @param  expr: If expr is false, it calls assert_failed function# C: W4 H% P! [/ m
  5.    5     *   which reports the name of the source file and the source
    * \9 X+ f4 S( l
  6.    6     *   line number of the call that failed.
    7 J7 f& f* J3 B" y, d
  7.    7     *   If expr is true, it returns no value.
    $ r( \9 O; [3 D
  8.    8     * @retval None
    + x2 I+ E) w2 x+ V
  9.    9     */
    7 }/ w: |1 _9 Q1 ~4 u9 w
  10.   10 #define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t" p9 W( k) \4 r: e$ j9 S- g, x7 ]
  11.   11 *)__FILE__, __LINE__))
    # @4 v& ?& h3 n8 w# _" D
  12.   12 /* Exported functions ---------------------------------- */1 Y& s0 \4 r5 v& O2 E+ v
  13.   13 void assert_failed(uint8_t* file, uint32_t line);$ I. V+ v9 P" d0 e
  14.   14 #else
    / a% e* S0 k- \" h
  15.   15 #define assert_param(expr) ((void)0)# P+ f( B' a+ Q- R; W: K4 ^
  16.   16 #endif /* USE_FULL_ASSERT */
复制代码
4 d7 y* `; I3 H" E  V$ K* E  }
在ST标准库的函数中,一般会包含输入参数检查,即上述代码中的assert_param宏,当参数不符合要求时,会调用assert_failed函数,这个函数默认是空的。
7 K5 f7 H( E$ u# L2 w4 q
6 i0 [8 G: A! d8 F  u实际开发中使用断言时,先通过定义USE_FULL_ASSERT宏来使能断言,然后定义assert_failed函数,通常会让它调用printf函数输出错误说明。使能断言后,程序运行时会检查函数的输入参数,当软件经过测试可发布时,会取消USE_FULL_ASSERT宏来去掉断言功能,使程序全速运行。, w% @7 R) t* V' z6 C1 }2 X6 g
# c0 \* _* b3 I" `$ Q
4 库各文件间的关系

1 O: Y6 y" K% ?7 ^& E7 J, N) ^前面简单介绍了各个库文件的作用,库文件直接包含进工程即可,丝毫不用修改,而有的文件就要我们在使用的时候根据具体的需要进行配置。接下来从整体上把握一下各个文件在库工程中的层次或关系,这些文件对应到CMSIS标准架构上,见图1-6。
! d. a6 r; \2 F5 {- M& G3 N6 p$ ]; ^) Z7 k( _0 J/ n
20200507121415550.jpg

$ r, B) J4 o3 j& |+ }$ s( f" D) N
图1-6 库各文件关系
( z! i8 N) Z, t' c' V* v1 }8 h

! e9 p' J- q6 ~3 N" K$ n- z* d图1-6描述了STM32库各文件之间的调用关系,在实际使用库开发工程的过程中,我们把位于CMSIS层的文件包含进工程,除了特殊系统时钟需要修改system_stm32f10x.c,其他文件丝毫不用修改,也不建议修改。
6 {1 I7 i' E2 x( H5 Y& d! p- o( u' i2 c" \3 j# c  K$ N( Q
对于位于用户层的几个文件,就是我们在使用库的时候,针对不同的应用对库文件进行增删(用条件编译的方法增删)和改动的文件。) M; o; R7 F3 F. x7 T4 D: H, M7 q
1 y6 C- X; X7 S. v* I  g  t
5 使用帮助文档
4 ~3 V# b% [# d* _* Y+ J俗话说,授之以鱼不如授之以渔。官方资料是所有关于STM32知识的源头,所以本节介绍如何使用官方资料。官方的帮助手册是最好的教程,几乎包含了所有在开发过程中会遇到的问题。这些资料可以到秉火论坛下载。5 p: C* A2 G; S$ j$ N2 j
( [  {, Q) y- o; w2 }5 w( r( {
4.1 常用官方资料8 k& m6 x5 {  F" _( ~& ]$ U
1.《STM32F10X-中文参考手册》; }7 k9 [" F" D# \# D2 I
这个手册全方位介绍了STM32芯片的各种片上外设,它把STM32的时钟、存储器架构,以及各种外设、寄存器都描述得清清楚楚。当我们对STM32的外设感到困惑时,可查阅这个手册。以直接配置寄存器方式开发的话,查阅这个文档寄存器部分的频率会相当高,但这样开发效率太低了。
0 h& f: U4 g% B3 f& }& n2 B! s  y, F; u8 b' x# k
2.《STM32规格书》# a( h# T$ a6 J$ K. Y( N$ I5 x
本文档相当于STM32的数据手册,包含了STM32芯片所有的引脚功能说明,以及存储器架构、芯片外设架构说明。后面我们使用STM32其他外设时,常常需要查找这个文档,了解外设对应到STM32的哪个GPIO引脚。) Q' x/ y: k% Y6 M! h( Y" F

3 u5 j0 S5 r/ A) P3.《Cortex-M3内核编程手册》
4 j, C1 T' ]7 q+ M( y本手册由ST公司提供,主要讲解STM32内核寄存器相关的说明,例如系统定时器、NVIC等核外设的寄存器。这部分的内容是对《STM32F10X-中文参考手册》没涉及的内核部分的补充。相对来说,本文档虽然介绍了内核寄存器,但不如以下两个文档详细,要了解内核时,可作为以下两个手册的配合资料使用。
% o; o9 Q% q- v0 b. l1 p! U: A/ U
4.《Cortex-M3权威指南》
# m  @% s' L; N% e' @这个手册是由ARM公司提供的,它详细讲解了Cortex内核的架构和特性,要深入了解Cortex-M内核,这是首选,是经典中的经典。这个手册已被翻译成中文,出版发行,我们配套的资料里面提供中文版的电子版。0 q# i/ y) I" E
5 f/ m7 v2 f2 V2 T- V+ H
5.《stm32f10x_stdperiph_lib_um.chm》8 {5 y4 k6 X. p" c
这个就是本章提到的库的帮助文档,在使用库函数时,我们最好通过查阅此文件来了解标准库提供了哪些外设、函数原型或库函数的调用的方法,也可以直接阅读源码里面的函数说明。8 p  ^+ d. I+ s- Q

7 U5 w4 w6 ~5 A9.2.2 初识库函数" J. O7 V! b6 w7 {! {
所谓库函数,就是STM32的库文件中为我们编写好驱动外设的函数接口,只要调用这些库函数,就可以对STM32进行配置,达到控制的目的。我们可以不知道库函数是如何实现的,但调用函数时必须知道函数的功能、可传入的参数及其意义和函数的返回值。) u+ J. v- A9 j; l) D9 {5 i& Y. ]
有读者可能会问:那么多函数我怎么记呀?回答是:会查就行了,哪个人记得了那么多。所以学会查阅库帮助文档是很有必要的。+ [4 U6 r. c# j
打开库帮助文档《stm32f10x_stdperiph_lib_um.chm》,见图1-7。0 \8 W5 E0 M$ t% x" a  g& ?

/ C, O  n9 R* V. a
20200507121345363.jpg

  z3 o1 m* g) g
图1-7 库帮

' i6 ]! Q2 _1 _7 M3 w; y0 B. Y
助文档

( S. ?% w; ]4 e6 j层层打开文档的目录标签Modules\STM32F10x_Std Periph_Driver\,可看到STM32F10x_Std Periph_Driver标签下有很多外设驱动文件的名字:MISC、ADC、BKP、CAN等。
0 J+ d, `' |7 |5 {- x! E3 m9 M& ?! N+ t2 s  i3 @0 o
我们试着查看GPIO的“位设置函数GPIO_Set Bits”,打开标签Modules\STM32F10x_Std Periph_Driver\GPIO\Functions\GPIO_Set Bits,见图1-8。. C5 b* j. \, s2 v6 S/ d, x/ f

) E' @6 E0 Y- ^1 I( G
20200507121305863.jpg
9 [3 Z' V8 ?  P& U
图1-8 库帮助文档的函数说明

5 Q9 m2 M. o1 _9 s/ j4 h7 r; \6 M* |& I& K
利用这个文档,我们即使不去看它的具体源代码,也知道怎么利用它了。
/ U* R# {/ n" x+ Y6 R6 T/ H+ O2 k7 e3 ]( E$ E* T+ v
如GPIO_Set Bits,函数的原型为void GPIO_Set Bits(GPIO_Type Def * GPIOx , uint16_t GPIO_Pin)。它的功能是:输入一个类型为GPIO_Type Def的指针GPIOx参数,选定要控制的GPIO端口;输入GPIO_Pin_x宏,其中x指端口的引脚号,指定要控制的引脚。
3 e% e: f8 [* [+ c# v0 ^  y其中输入的参数GPIOx为ST标准库中定义的自定义数据类型,这两个传入参数均为结构体指针。初学时,我们并不知道像GPIO_Type Def这样的类型是什么意思,单击函数原型中带下划线的GPIO_Type Def就可以查看这个类型的声明了。
. u% C! p% r" O1 X' i  \/ M
! L5 ]/ i$ e& q2 I; s5 C就这样初步了解一下库函数,就可以发现STM32的库写得很优美。每个函数和数据类型都符合见名知义的原则,当然,这样的名称写起来特别长,而且对于中国人来说要输入这么长的英文,很容易出错,所以在开发软件的时候,在用到库函数的地方,直接把库帮助文档中的函数名称复制并粘贴到工程文件中就可以了。而且,配合MDK软件的代码自动补全功能,可以减少输入量。$ D* K6 L  f0 {9 e! W2 [0 x
# t/ d6 k% v1 _
有的用户觉得使用库文档麻烦,也可以直接查阅STM32标准库的源码,库帮助文档的说明都是根据源码生成的,所以直接看源码也可以了解函数功能。+ b* F6 ^& _% ^$ e' n1 Z% E+ ~

3 O# U9 o4 x1 A. ^' o# G, T% Y$ |' M9 a+ a: w
9 a: r! X  x8 w, J  Y3 f
收藏 1 评论0 发布时间:2021-11-30 22:34

举报

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