第1课:点亮你的第一个LED灯5 a8 P. J0 ^ B) M) \
原理分析:
5 o u: R/ W' Q& w9 r1 P" M大家之前学习过51单片机,也使用过IO口。与ARM CORTEX M0的IO口配置有点区别,51不需要设置IO口为输入或者输出,而CORTEX M0的IO口有多种状态需要设置,那么下面我们一一介绍:
O" Z' v5 d! F( I. ^; p' Y首先看看IO口的模式,如下寄存器说明,IO口可以配置为4种模式:输入模式,输出模式,复用模式,模拟通道模式。由于stm32f051系列多数的IO关键复用了其外设功能,比如I2C,SPI,UART等,此时就可以设置IO口为复用模式。模拟通道则作为AD,DA的时候使用:
1 M* d3 Y* ^% L2 h3 l/ x
# d0 z$ f0 U: v4 O! }5 X0 ?) b如果大家使用库函数编程的时候,可以在 stm32f0xx_gpio.h文件中找到设置IO模式的结构体GPIOMode_TypeDef :5 W2 g: z+ J3 j% D# p
3 v. @1 O; Z, J) J下面来介绍下输入和输出模式,其中输出模式如下图所示,分为推挽输出和开漏输出。这时候大家就要回忆下模电的课程了。9 h1 W4 u0 r, }
+ E; H% r9 s; j5 A. |我使用下面一个等效图说明一下推挽输出和开漏输出。图所示:U1是输出锁存器,执行GPIO管脚写操作时,在写脉冲(Write Pulse)的作用下,数据被锁存到Q和/Q。T1和T2构成CMOS反相器,T1导通或T2导通时都表现出较低的阻抗,但T1和T2不会同时导通或同时关闭,最后形成的是推挽输出。
+ O* T0 }- e' ~+ V如图1.3所示,为GPIO管脚在开漏输出模式下的等效结构示意图。开漏输出和推挽输出相比结构基本相同,但只有下拉晶体管T1而没有上拉晶体管。同样,T1实际上也是多组可编程选择的晶体管。开漏输出的实际作用就是一个开关,输出“1”时断开、输出“0”时连接到GND(有一定内阻)。
# s! _4 R6 B! B/ ^- t1 \# v' a# f) A3 L) o2 x
% i7 ?) \5 X, G9 B& |
同样大家使用库函数编程的时候,可以在 stm32f0xx_gpio.h文件中找到设置输出模式的结构体GPIOOType_TypeDef- D' J+ h' S$ t
# {2 R% z, V8 D2 O, g1 d
同时我们可以设置IO端口输出的速度,在库函数中通过结构体来解决:
) C1 u6 O% B# N# y- ]3 y8 q7 [
( v. x6 a& \+ @9 ]4 L9 h输入的模式可以分为上拉和下拉模式,这就比较简单了,寄存器的设置如下图:
2 V6 S* v$ ~/ P6 P1 S1 u/ z. @% }; {8 e4 T* F$ j ?; C9 R+ {
同样大家使用库函数编程的时候,可以在 stm32f0xx_gpio.h文件中找到设置输入模式的结构体GPIOOPuPd_TypeDef
) |' |1 D* @' k7 h8 \" M8 B1 d# ^0 }
上面介绍完了STM32F051的IO端口有哪几种模式,现在就来点亮一个LED灯,学过51的同学可以回忆下,51是通过设置IO口输出0或者1来驱动LED灯的亮灭。在STM32F051中,我们通过设置IO端口为输出来点亮LED等。7 T3 c$ n; u$ ~$ R. @, o m2 S
硬件准备:
0 @0 r2 t6 k# G, f) z1 g: S2 s, P, C) c* \" w$ q z
: \* w+ }4 F1 ?3 b* u/ Q如上图所示: 青风STM32F051开发板上,通过管脚PA11和管脚PA12连接2个LED灯,我们下面的任务首先来点亮它。
4 b: v0 r) i7 a% m1 q* \软件准备:
; E4 v, s2 N4 N* b. {2 e按照第一章的介绍首先建立一个工程项目,采用库函数来在驱动IO口首先要添加几个驱动库,如下图所示: r7 {4 s5 j3 p! F8 b
8 Y4 e! T4 F, C' I6 h上图红色框框中的几个文件都是ST官方给我们编好的库函数。那边用户在使用中,只需要编写led.c 驱动文件和main.c主函数就OK,整个工程项目大家如果加入分层的思想那么就对之后的移植非常有利。打个比方:底层和应用程隔离。底层驱动和应用层无关,main.c使用的函数在led.c驱动中已经些好,这些才和硬件有关,这是需要移植到不同硬件时,main主函数是可以不做任何修改的,只需要修改和底层相关的led.c驱动。
- ~) f; Y) k7 t$ a9 |- ]2 l( b下面来分析下led .c的驱动编写:
% j _ W' `0 O/ n* P* @4 W0 P6 p2 \6 Q9 O3 E6 U6 V9 X
上面的函数中GPIO_ResetBits函数和GPIO_SetBits函数在stm32f0xx_gpio.c驱动文件中所定义了。分别表示复位和置位相关IO管脚。
, y+ W& o$ Z8 G* A( P+ n那么主函数的编写就比较简单了,我们需要调用下面2个头文件,才能够直接使用我们定义的子函数。如下使用LED_Open()函数就能够点亮一个LED灯了,是不是很简单。. x- h# A% e% `7 f2 N5 v8 Q8 U- ~
3 |) q/ ~+ x# w: { E
那么加入一个小的延迟delay函数和打开与关闭LED子函数相结合,就可以实现LED闪烁的功能了,函数如下所示:
1 U( a) w, b1 o) j7 \8 E/ G8 z" i& k! w! u0 j
下载到青风STM32F051开发板上运行后的效果如下图所示:7 E; p+ ^7 u8 B/ w2 F
- e# j+ s% D; J" u% y4 C
如图所示,上方的用户led灯不停闪烁。
+ o1 m0 h0 z9 ] |