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

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

[复制链接]
STMCU小助手 发布时间:2021-11-30 22:34
1、CMSIS标准及库层次关系
, i( e/ `: `% H9 v4 ^基于Context系列芯片采用的内核都是相同的,区别主要为核外的片上外设的差异,而这些差异却导致软件在同内核、不同外设的芯片上移植困难。' e$ [  B9 o/ x0 t& s  X% Z) B9 D
为了解决不同的芯片厂商生产的Context微控制器软件的兼容性问题,ARM与芯片厂商建立了CMSIS标准(Context Micro Controller Software Interface Standard)。
+ Q( R  c" \/ p8 s4 q所谓CMSIS标准,实际是新建了一个软件抽象层,见图1-17 M: s5 k3 g  b5 I
20200507120021750.jpg

  Z0 @+ R" |* o, t2 i8 S2 _& V5 W
( u4 V4 E' T4 A! z: ?: CCMSIS标准中最主要的是CMSIS核心层,它包括以下两部分。* {4 k. u5 C* s# E

% x! g# s: p  A# J5 H9 {·内核函数层:其中包含用于访问内核寄存器的名称、地址定义,主要由ARM公司提供。
& H5 Q8 z5 q+ S# g& A·设备外设访问层:提供了片上的核外外设的地址和中断定义,主要由芯片生产商提供。
# Z- c+ C/ ?: z* Q
$ Y0 c; P4 v- e* t9 M3 {可见CMSIS层位于硬件层与操作系统或用户层之间,提供了与芯片生产商无关的硬件抽象层,可以为接口外设、实时操作系统提供简单的处理器软件接口,屏蔽了硬件差异,这对软件的移植是有极大好处的。STM32的库就是按照CMSIS标准建立的。! j8 Z# b1 O5 O( z: c* A& k7 ^1 c
; O+ P- l; Y; A' V
2 库目录、文件简介& q$ j- R. F5 ~" @
STM32标准库可以从官网获得,也可以直接从论坛中的配套资料得到。本书讲解的例程全部采用3.5.0库文件。以下内容请打开STM32标准库文件配合阅读。
+ X$ y% l* G6 g. Y) @# T
5 S0 \4 K! R$ O: K/ j+ F解压库文件后进入其目录STM32F10x_Std Periph_Lib_V3.5.0\,软件库各文件夹的内容说明见图1-2。
% d- U3 ~' A1 U* H% C
$ ~( K" Y& [2 ^
20200507120717512.jpg

. O& A* {  Z" o8 V
图1-2 ST标准库
& Y! K  E$ N, j- ]: X* c6 K9 B) O

7 H* F3 P8 Y0 K; t3 n·Libraries:文件夹下是驱动库的源代码及启动文件,这个文件夹非常重要,我们要使用的固件库就在这个文件夹里面。
6 e  f5 u5 \/ V1 H9 G2 k. E·Project:文件夹下是用驱动库写的例子和工程模板,其中那些为每个外设写好的例程对我们非常有用,在学习的时候可以参考这里面的例程,非常全面,简直就是穷尽了外设的所有功能。- q  F( r& C$ ]1 T. z/ r
·Utilities:包含了基于ST官方实验板的例程,不需要用到,略过即可。
. [* v5 }  t; L) ~" X+ e·stm32f10x_stdperiph_lib_um.chm:库帮助文档,这个很有用,不喜欢直接看源码的用户可以在这里查询每个外设的函数说明,非常详细。这是一个已经编译好的HTML文件,主要讲述如何使用驱动库来编写自己的应用程序。说得形象一点,这个HTML就是告诉我们:ST公司已经为你写好了每个外设的驱动了,想知道如何运用这些例子就来向我求救吧。但是,这个帮助文档是英文的,这对很多英文不好的朋友来说是一个很大的障碍。但这里要告诉大家,英文仅仅是一种工具,绝对不能让它成为我们学习的障碍。
. S; N1 e9 `# b  T; D5 g- z- Y' L% f) |( k! o4 A  z
在使用库开发时,我们需要把libraries目录下的库函数文件添加到工程中,并查阅库帮助文档来了解ST提供的库函数,这个文档说明了每一个库函数的使用方法。- N, K  Q( B1 s8 {5 u

" B1 v+ `/ n8 ~- h进入Libraries文件夹,可以看到关于内核与外设的库文件分别存放在CMSIS和STM32F10x_Std Periph_Driver文件夹中。" }) _/ R& c2 I+ m

* H4 x$ X2 b# U& d2 l% B1.CMSIS文件夹( S0 Z8 b- ?1 X  E9 d0 L
STM32F10x_Std Periph_Lib_V3.5.0\Libraries\CMSIS\文件夹展开内容见图1-3。4 t5 T8 O( E2 k8 s  N8 B

  Z  a" E1 J+ x1 W! W1 `8 b
20200507120744306.jpg
) j5 B2 w$ }: O2 p( L% y% Y; {" o
图1-3 CMSIS文件夹内容

/ v$ [3 @1 ?# v7 G9 W6 R( |7 w* p  @6 ~' q' B
- h( I# k! S: V$ r+ D
其中带阴影的文件是我们需要用到的内容,下面我们一一讲解这几个文件的作用。
) R$ C4 @1 o. Z: r) H9 m' ~5 X& R% \* T% y) L
(1)内核相关文件1 V7 l2 C1 B; b( r, i0 g
在Core Support文件夹中有core_cm3.c和core_cm3.h两个文件。core_cm3.h头文件里面实现了内核的寄存器映射,对应外设头文件stm32f10x.h,区别就是一个针对内核的外设,一个针对片上(内核之外)的外设。core_cm3.c文件实现了操作内核外部寄存器的函数,用得比较少。. c: t4 p7 o2 m9 v7 r6 n8 Z
/ z, _4 K( X+ A! u4 R* |+ w6 D
我们还需要了解的是core_cm3.h头文件中包含了stdint.h这个头文件,这是一个ANSI C文件,是独立于处理器之外的,就像我们熟知的C语言头文件stdio.h文件一样。它位于RVMDK这个软件的安装目录下,主要作用是提供一些类型定义,见代码清单9-1。) O. o( S$ `- S
代码清单9-1 stdint.h文件中的类型定义
% ]  I; \. n3 q& o' H) P6 [
  1.     1 /* exact-width signed integer types */
    , W5 f# r: C7 ]- F" o+ _5 x/ C4 q( {% N
  2.     2 lt@span b=1>typedef   signed# E' R! g+ P( P
  3.     char int8_t;3 \( o7 O# r- J& l
  4.     3 typedef   signed short     int int16_t;" Y5 d( X) |+ ]" ^/ U" I& w
  5.     4 typedef   signed            int int32_t;
    6 K  k# \( i: I9 b2 f$ F
  6.     5 typedef   signed        __int64 int64_t;( N. _+ {. q8 Z9 _. b% [
  7.     66 j$ o* S6 u: w, `1 b1 }
  8.     7 /* exact-width unsigned integer types */8 W8 T% X' Q/ }( k+ V- p. ~
  9.     8 typedef unsigned           char uint8_t;
    + e/ i8 p0 C/ p; ?
  10.     9 typedef unsigned short     int uint16_t;- X4 c. G" y- C: t/ {9 c' q% h2 b
  11.    10 typedef unsigned            int uint32_t;
    - h/ j& c% Z! A/ Y$ A8 p- R$ f
  12.    11 typedef unsigned        __int64 uint64_t;
复制代码
# f$ K7 h9 F/ c3 {
这些新类型定义屏蔽了在不同芯片平台时,出现的诸如int的大小是16位,还是32位的差异。所以在我们以后的程序中,都将使用新类型,如uint8_t 、uint16_t等。
* X3 K/ h' Q: v$ q3 ^: H1 Z' [3 ^# j7 d% f: m5 o4 f
在稍旧版的程序中还经常会出现如u8、u16、u32这样的类型,分别表示无符号的8位、16位、32位整型。初学者碰到这样的旧类型会感觉一头雾水,它们定义的位置在STM32f10x.h文件中。建议在以后的新程序中尽量使用uint8_t 、uint16_t类型的定义。1 {8 Y& g2 e& |1 e  K

; k8 Z. W% \$ x, }(2)启动文件
5 O" Q" U" Y2 e1 Q3 G启动文件放在startup/arm这个文件夹下面,这里面启动文件有很多个,不同型号的单片机用的启动文件不一样,有关每个启动文件的详细说明见表1-1。
- k0 a8 {0 W& r& y7 W: G  g表1-1 各启动文件匹配的芯片类型3 A/ Z7 E) `7 k7 o* i
! Q7 e3 K* N# e  A+ K! Z8 `
20200507120849621.jpg
9 E' J, _  \6 i% y9 _; w

0 ?/ D( _; C1 @( t我们开发板中用的STM32F103VET6或者STM32F103ZET6中的Flash都是512k,属于基本型的大容量产品,启动文件统一选择startup_stm32f10x_hd.s。
: d7 o% k' s! Q7 e: B" k
8 @. ?/ t3 G* N4 V# X* |(3)Stm32f10x.h
3 [9 u5 y6 g. k$ s& T这个头文件实现了片上外设的所以寄存器的映射,是一个非常重要的头文件,在内核中与之相对应的头文件是core_cm3.h。8 z8 Z) A0 D% ~" }' @6 L

8 N( W5 u/ d/ `- J9 q/ _(4)system_stm32f10x.c: \4 \! v* x( @: p
system_stm32f10x.c文件实现了STM32的时钟配置,操作的是片上的RCC这个外设。系统在上电之后,首先会执行由汇编编写的启动文件,启动文件中的复位函数中调用的System Init函数就在这个文件里面定义。调用完之后,系统的时钟就被初始化成72M。如果后面我们需要重新配置系统时钟,我们就可以参考这个函数重写。为了维持库的完整性,我们不会直接在这个文件里面修改时钟配置函数。
. L; l5 P. a, p; b& O4 O2 ^, [
: n$ T" p, I$ ], `0 L  p4 e3.STM32F10x_Std Periph_Driver文件夹. s: a7 _/ P" H" `$ j% ?
libraries目录下的STM32F10x_Std Periph_Driver文件夹见图1-4。2 V" F8 d0 Y% B1 i  n1 l
图1-4 外设驱动" {& M; T! Z& ~+ z2 w" g

* `* [1 M4 l# R. f1 m3 ]6 H
20200507121543617.jpg
  B* s5 {8 r: n: r2 g4 Y
' a. L  @- d& D1 E$ ?
STM32F10x_Std Periph_Driver文件夹下有inc(include的缩写)与src(source的缩写)这两个文件夹,这里的文件属于CMSIS之外的、芯片片上的外设部分。src里面是每个设备外设的驱动源程序,inc则是相对应的外设头文件。src及inc文件夹是ST标准库的主要内容,不少人甚至认为ST标准库就是指这些文件,可见其重要性。' K- K2 E' x" {$ N

3 Z& h+ T; Y1 O% @- F# X- G; p在src和inc文件夹里的就是ST公司针对每个STM32外设而编写的库函数文件,每个外设对应一个.c和.h后缀的文件。我们把这类外设文件统称为stm32f10x_ppp.c或stm32f10x_ppp.h文件,ppp表示外设名称。如在上一章中我们自建的stm32f10x_gpio.c及stm32f10x_gpio.h文件,就属于这一类。, |2 P+ f' X6 z; F: ?' V
+ H' D( T- V9 g& @# n: U+ `5 D. N! G
如针对模数转换(ADC)外设,在src文件夹下有一个stm32f10x_adc.c源文件,在inc文件夹下有一个stm32f10x_adc.h头文件,若我们开发的工程中用到了STM32内部的ADC,则至少要把这两个文件包含到工程里,见图1-5。
2 _1 k- Y  L! j) l& c. s
) m' y2 @7 S- Y8 a# U
2020050712145141.jpg

$ ^7 m4 H2 T2 O4 k7 h& h' A9 {$ r" T- i$ Z
图1-5 驱动的源文件及头文件$ E5 ?% C0 X8 y6 _
6 |4 E& Z& R& h4 _$ O2 X
这两个文件夹中,还有一个很特别的misc.c文件,这个文件提供了外设对内核中的NVIC(中断向量控制器)的访问函数,在配置中断时,必须把这个文件添加到工程中。
  |1 E7 e' K9 L1 h+ Y. q! {& y8 @- R$ m
3.stm32f10x_it.c、stm32f10x_conf.h和system_stm32f10x.c文件
, {. i5 d9 h7 q$ c在文件目录STM32F10x_Std Periph_Lib_V3.5.0\Project\STM32F10x_Std Periph_Template下,存放了官方的一个库工程模板,在用库建立一个完整的工程时,还需要添加这个目录下的stm32f10x_it.c、stm32f10x_conf.h和system_stm32f10x.c这4个文件。
! D5 D, U- j  X3 L( v5 p8 H& `4 G7 D
(1)stm32f10x_it.c( M' w( N/ O5 [# b  T% K
这个文件是专门用来编写中断服务函数的,在我们修改前,这个文件已经定义了一些系统异常(特殊中断)的接口,其他普通中断服务函数由我们自己添加。但是我们怎么知道这些中断服务函数的接口如何写呢?是不是可以自定义呢?答案当然不是,这些都可以在汇编启动文件中找到,在学**断和启动文件的时候会详细介绍。9 B7 @& N" w/ S# M6 V) \) L* R
, c- i# _0 |6 J
(2)system_stm32f10x.c+ l' C5 i! P% m' N" D0 {
这个文件包含了STM32芯片上电后初始化系统时钟、扩展外部存储器用的函数,例如我们前两章提到供启动文件调用的System Init函数,用于上电后初始化时钟,该函数的定义就存储在system_stm32f10x.c文件中。STM32F103系列的芯片,调用库的这个System Init函数后,系统时钟被初始化为72MHz,如需要可以修改这个文件的内容,设置成自己所需的时钟频率。但鉴于保持库的完整性,我们在做系统时钟配置的时候会另外重写时钟配置函数。
+ X% p' y% [' Z6 G; p" ]
. p; u" m: g; @) D6 Y(3)stm32f10x_conf.h
6 t# Y( T) `1 z) y3 o0 E' K2 E这个文件被包含进stm32f10x.h文件。当使用固件库编程的时候,如果需要某个外设的驱动库,就需要包含该外设的头文件:stm32f10x_ppp.h。包含一个还好,如果用了多个外设,就需要包含多个头文件,这不仅影响代码美观,而且也不好管理。现我们用一个头文件stm32f10x_conf.h把这些外设的头文件都包含在里面,让这个配置头文件统一管理这些外设的头文件,我们在应用程序中只需要包含这个配置头文件即可。我们又知道这个头文件在stm32f10x.h的最后被包含,所以最终我们只需要包含stm32f10x.h这个头文件即可,非常方便。Stm32f10x_conf.h见代码清单9-2。默认情况下是所有头文件都被包含,没有被注释掉。也可以把不要的都注释掉,只留下需要使用的即可。
; d; ?9 C4 C- n5 [' b
9 Z- z* {# @  D  Y6 P$ `" D# T代码清单1-2 stm32f10x_conf.h文件配置软件库
1 J7 o0 N; a; q1 K( j. g' a
  1.    1 #include“stm32f10x_adc.h”
    3 \" p" \( \( e- K
  2.    2 #include“stm32f10x_bkp.h”
    1 {2 x3 o" u2 ]8 r$ v  S3 X) x
  3.    3 #include“stm32f10x_can.h”
    3 J5 S9 R) J3 X! n
  4.    4 #include“stm32f10x_cec.h”' s0 A" x; i+ ^- e, o: P
  5.    5 #include“stm32f10x_crc.h”
    5 }! H1 O! S: ]  z
  6.    6 #include“stm32f10x_dac.h”! e8 g: A  E$ `) g* z  y
  7.    7 #include“stm32f10x_dbgmcu.h”- \. x. J+ i/ {4 I% q# M9 I
  8.    8 #include“stm32f10x_dma.h”% ~. F9 t9 u7 p" N
  9.    9 #include“stm32f10x_exti.h”2 M$ D, {+ A5 `
  10.   10 #include“stm32f10x_flash.h”3 \7 Z) f+ [$ ^7 D9 i) L/ s
  11.   11 #include“stm32f10x_fsmc.h”0 _2 I+ L! {' _5 U) H7 ~
  12.   12 #include“stm32f10x_gpio.h”0 g7 \: H0 |, T) U+ B6 v
  13.   13 #include“stm32f10x_i2c.h”
      q$ h0 o. c& c; P: W
  14.   14 #include“stm32f10x_iwdg.h”7 C4 U/ r0 ]* l/ ?; a6 q" H! q
  15.   15 #include“stm32f10x_pwr.h”
    5 r6 x# C2 ^% W( W; t! a
  16.   16 #include“stm32f10x_rcc.h”
    " C) o- X) K  s
  17.   17 #include“stm32f10x_rtc.h”+ i4 L0 y) j; S, S1 ]" y
  18.   18 #include“stm32f10x_sdio.h”
    4 W( M7 E: H0 `9 O
  19.   19 #include“stm32f10x_spi.h”" o; O' v* J) g3 O4 o0 m9 X0 f
  20.   20 #include“stm32f10x_tim.h”/ j5 f9 {$ D) D1 P
  21.   21 #include“stm32f10x_usart.h”
    % B9 ^" T6 T& Y5 C3 Z
  22.   22 #include“stm32f10x_wwdg.h”1 u3 W" M5 {+ C0 H$ K: t& {
  23.   23 #include“misc.h”
复制代码

0 E+ f9 C1 z6 j7 g8 y2 tstm32f10x_conf.h这个文件还可配置是否使用“断言”编译选项,见代码清单1-3。
7 T# j! _/ ?! M/ E, p
% L, ]7 ^" `4 `$ r/ e# s代码清单1-3 断言配置
7 O& q/ L+ C! q& H  S
  1.    1 #ifdef  USE_FULL_ASSERT
    4 g- t0 q$ r. C4 _1 q# x( r
  2.    2 /**
    ! z& R+ R# N  j
  3.    3     * @brief  The assert_param macro is used for  parameters check.( ?" g$ B& t) y& W; @
  4.    4     * @param  expr: If expr is false, it calls assert_failed function
    , P$ a4 d! x! s  c
  5.    5     *   which reports the name of the source file and the source
    4 d0 ~0 e5 N! C8 a2 |
  6.    6     *   line number of the call that failed.
    3 Q% m4 t2 z3 T6 ?9 L
  7.    7     *   If expr is true, it returns no value.( _# E9 h, R4 N/ S+ e
  8.    8     * @retval None' a" f. F- l, v2 k
  9.    9     */
    0 |" q8 D# {6 K8 W- W; S' B
  10.   10 #define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t
    0 P) Q# H& [5 j# {) o. X! N
  11.   11 *)__FILE__, __LINE__))
    1 d3 N: x2 c% F  [, J, q
  12.   12 /* Exported functions ---------------------------------- */5 c, E* |4 R$ r8 W) v. C
  13.   13 void assert_failed(uint8_t* file, uint32_t line);
    4 z" T8 e& a( A
  14.   14 #else
    , X5 p' W) P7 B. h! h/ j, p
  15.   15 #define assert_param(expr) ((void)0)8 G2 X9 Z" v: m! [7 x& w9 u8 b
  16.   16 #endif /* USE_FULL_ASSERT */
复制代码

1 C8 i/ b  f' f8 N在ST标准库的函数中,一般会包含输入参数检查,即上述代码中的assert_param宏,当参数不符合要求时,会调用assert_failed函数,这个函数默认是空的。6 y! U6 p- ^5 X6 b' M) i) W

( i4 h; q4 h! ~! M: b, d5 `实际开发中使用断言时,先通过定义USE_FULL_ASSERT宏来使能断言,然后定义assert_failed函数,通常会让它调用printf函数输出错误说明。使能断言后,程序运行时会检查函数的输入参数,当软件经过测试可发布时,会取消USE_FULL_ASSERT宏来去掉断言功能,使程序全速运行。
2 V7 D0 }7 Y" j; h; ]% ]; _  Q1 u2 s1 l6 `, i) @4 _  e
4 库各文件间的关系
8 ?4 L3 i% S  s4 P! m8 I
前面简单介绍了各个库文件的作用,库文件直接包含进工程即可,丝毫不用修改,而有的文件就要我们在使用的时候根据具体的需要进行配置。接下来从整体上把握一下各个文件在库工程中的层次或关系,这些文件对应到CMSIS标准架构上,见图1-6。
7 N0 p0 W" H" \7 c' ]
' x) ]. ?2 I" c- [+ j. h
20200507121415550.jpg
; T# G7 f5 g: a' t6 Q) K1 Y0 }; Q0 ^
图1-6 库各文件关系
# f; X0 R2 `( b: @

' Z8 b" Y9 ~. j+ J& [2 H. ]图1-6描述了STM32库各文件之间的调用关系,在实际使用库开发工程的过程中,我们把位于CMSIS层的文件包含进工程,除了特殊系统时钟需要修改system_stm32f10x.c,其他文件丝毫不用修改,也不建议修改。4 _) l) O: B( m; Q$ |) H
$ f& ]; M! o5 ?  E  t, e; I' P
对于位于用户层的几个文件,就是我们在使用库的时候,针对不同的应用对库文件进行增删(用条件编译的方法增删)和改动的文件。
* g, u0 j. r0 [5 f
- Z+ b  @3 z( I; f: x1 h3 g5 使用帮助文档. I4 @$ a* Y5 j
俗话说,授之以鱼不如授之以渔。官方资料是所有关于STM32知识的源头,所以本节介绍如何使用官方资料。官方的帮助手册是最好的教程,几乎包含了所有在开发过程中会遇到的问题。这些资料可以到秉火论坛下载。" G  }  e" j! f5 _
2 Z/ \- q9 x! ~
4.1 常用官方资料
0 q1 X* \8 M& G! U+ V1.《STM32F10X-中文参考手册》- d3 P6 K2 a1 |# p/ _( D; u
这个手册全方位介绍了STM32芯片的各种片上外设,它把STM32的时钟、存储器架构,以及各种外设、寄存器都描述得清清楚楚。当我们对STM32的外设感到困惑时,可查阅这个手册。以直接配置寄存器方式开发的话,查阅这个文档寄存器部分的频率会相当高,但这样开发效率太低了。
7 E7 k, Z6 M+ m6 i5 K% m4 i  S7 Q, r  a
2.《STM32规格书》
6 z5 ^6 |- l" X' }' `& r7 o. C" b5 w2 p本文档相当于STM32的数据手册,包含了STM32芯片所有的引脚功能说明,以及存储器架构、芯片外设架构说明。后面我们使用STM32其他外设时,常常需要查找这个文档,了解外设对应到STM32的哪个GPIO引脚。' E* A# b- e0 ~  \5 S  x& a
. T: C+ j4 M8 f; }: |
3.《Cortex-M3内核编程手册》8 J5 R$ E* R0 _8 \$ G
本手册由ST公司提供,主要讲解STM32内核寄存器相关的说明,例如系统定时器、NVIC等核外设的寄存器。这部分的内容是对《STM32F10X-中文参考手册》没涉及的内核部分的补充。相对来说,本文档虽然介绍了内核寄存器,但不如以下两个文档详细,要了解内核时,可作为以下两个手册的配合资料使用。9 r/ d7 |. F$ ^. v3 M
/ ^; T- B8 a7 s
4.《Cortex-M3权威指南》
8 a( B. z1 _# E! c这个手册是由ARM公司提供的,它详细讲解了Cortex内核的架构和特性,要深入了解Cortex-M内核,这是首选,是经典中的经典。这个手册已被翻译成中文,出版发行,我们配套的资料里面提供中文版的电子版。8 |- }* w& X* c. m) l9 P

  u. \+ i" Y9 @, j5 b! y5.《stm32f10x_stdperiph_lib_um.chm》
8 a/ p3 g- A" p! \: s. s这个就是本章提到的库的帮助文档,在使用库函数时,我们最好通过查阅此文件来了解标准库提供了哪些外设、函数原型或库函数的调用的方法,也可以直接阅读源码里面的函数说明。. T( S( N8 q$ {4 i: X+ O$ Y, J
& V9 S8 o4 D: n3 n% R8 k
9.2.2 初识库函数
' ]8 D4 g+ M. g0 L1 T3 O* J所谓库函数,就是STM32的库文件中为我们编写好驱动外设的函数接口,只要调用这些库函数,就可以对STM32进行配置,达到控制的目的。我们可以不知道库函数是如何实现的,但调用函数时必须知道函数的功能、可传入的参数及其意义和函数的返回值。
* L5 L, \3 u9 K有读者可能会问:那么多函数我怎么记呀?回答是:会查就行了,哪个人记得了那么多。所以学会查阅库帮助文档是很有必要的。9 C' u# `4 c. e# @% ]6 _
打开库帮助文档《stm32f10x_stdperiph_lib_um.chm》,见图1-7。
: R7 C4 }: a2 S. t
) g+ T. Q8 H0 a, u9 \
20200507121345363.jpg
) L. w6 q' V! ^8 P' _5 Q
图1-7 库帮

9 o& |9 X* l' z# w
助文档
! V& N5 q+ }# X" j$ H
层层打开文档的目录标签Modules\STM32F10x_Std Periph_Driver\,可看到STM32F10x_Std Periph_Driver标签下有很多外设驱动文件的名字:MISC、ADC、BKP、CAN等。
) Y2 a$ G: C5 C8 `  e
3 v  @- e( T  p- i7 T. J& J我们试着查看GPIO的“位设置函数GPIO_Set Bits”,打开标签Modules\STM32F10x_Std Periph_Driver\GPIO\Functions\GPIO_Set Bits,见图1-8。! o  k9 {; X  i0 e$ [$ p+ t4 ~

* s7 I8 a+ s; a8 `- l3 F9 @
20200507121305863.jpg

3 f, P- M* z, |% P# J: T& T% I
图1-8 库帮助文档的函数说明
  q; L" ?- ?6 W7 |9 J

, S9 i& ^% M1 m6 y利用这个文档,我们即使不去看它的具体源代码,也知道怎么利用它了。
% q+ [5 n5 u/ g" B  @$ y; q+ P+ b& L& b* Q% v# g5 w
如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 d2 T$ |7 u# Y9 j: o. _
其中输入的参数GPIOx为ST标准库中定义的自定义数据类型,这两个传入参数均为结构体指针。初学时,我们并不知道像GPIO_Type Def这样的类型是什么意思,单击函数原型中带下划线的GPIO_Type Def就可以查看这个类型的声明了。
( h& U6 g* o: g" j9 c0 `0 a
% D0 ]1 K0 X# a' G8 Q/ F; q0 k! V就这样初步了解一下库函数,就可以发现STM32的库写得很优美。每个函数和数据类型都符合见名知义的原则,当然,这样的名称写起来特别长,而且对于中国人来说要输入这么长的英文,很容易出错,所以在开发软件的时候,在用到库函数的地方,直接把库帮助文档中的函数名称复制并粘贴到工程文件中就可以了。而且,配合MDK软件的代码自动补全功能,可以减少输入量。
3 d& }% |! W5 k9 `7 Q. L' U7 G0 T+ Y% n, t+ Q% s. r4 v3 a
有的用户觉得使用库文档麻烦,也可以直接查阅STM32标准库的源码,库帮助文档的说明都是根据源码生成的,所以直接看源码也可以了解函数功能。  x1 Z" H2 p( ^5 {9 ?4 y- n

1 i1 D( K/ q1 V( i
# m' ]+ ?: v# M6 q+ _$ o
  ]8 |% o1 y. X$ l7 u0 o
收藏 1 评论0 发布时间:2021-11-30 22:34

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版