欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

【正点原子Linux连载】第十五章按键输入试验-摘自【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.0

程序员文章站 2024-03-23 22:50:16
...

1)实验平台:正点原子阿尔法Linux开发板
2)平台购买地址:https://item.taobao.com/item.htm?id=603672744434
2)全套实验源码+手册+视频下载地址:http://www.openedv.com/thread-300792-1-1.html
3)对正点原子Linux感兴趣的同学可以加群讨论:935446741
4)关注正点原子公众号,获取最新资料更新

第十五章按键输入试验

前面几章试验都是讲解如何使用I.MX6U的GPIO输出控制功能,I.MX6U的IO不仅能作为输出,而且也可以作为输入。I.MX6U-ALPHA开发板上有一个按键,按键连接了一个IO,将这个IO配置为输入功能,读取这个IO的值即可获取按键的状态(按下或松开)。本章通过这个按键来控制蜂鸣器的开关,通过本章的学习你将掌握如何将I.MX6UL的IO作为输入来使用。

15.1按键输入简介
按键就两个状态:按下或弹起,将按键连接到一个IO上,通过读取这个IO的值就知道按键是按下的还是弹起的。至于按键按下的时候是高电平还是低电平要根据实际电路来判断。前面几章我们都是讲解I.MX6U的GPIO作为输出使用,当GPIO连接按键的时候就要做为输入使用。关于I.MX6U的GPIO已经在第八章详细的讲解了,本章我们的主要工作就是配置按键所连接的IO为输入功能,然后读取这个IO的值来判断按键是否按下。
I.MX6U-ALPHA开发板上有一个按键KEY0,本章我们将会编写代码通过这个KEY0按键来控制开发板上的蜂鸣器,按一下KEY0蜂鸣器打开,再按一下蜂鸣器就关闭。
15.2 硬件原理分析
本试验我们用到的硬件有:
1) LED灯LED0。
2)蜂鸣器。
3)1个按键KEY0。
按键KEY0的原理图如图15.2.1所示:
【正点原子Linux连载】第十五章按键输入试验-摘自【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.0

图15.2.1 按键原理图
从图15.2.1可以看出,按键KEY0是连接到I.MX6U的UART1_CTS这个IO上的,KEY0接了一个10K的上拉电阻,因此KEY0没有按下的时候UART1_CTS应该是高电平,当KEY0按下以后UART1_CTS就是低电平。
15.3实验程序编写
本实验对应的例程路径为:开发板光盘-> 1、裸机例程->7_key。
本试验在上一章试验例程的基础上完成,重新创建VSCode工程,工作区名字为“key”,在工程目录的bsp文件夹中创建名为“key”和“gpio”两个文件夹。按键相关的驱动文件都放到“key”文件夹中,本章试验我们对GPIO的操作编写一个函数集合,也就是编写一个GPIO驱动文件,GPIO的驱动文件放到“gpio”文件夹里面。
新建bsp_gpio.c和bsp_gpio.h这两个文件,将这两个文件都保存到刚刚创建的bsp/gpio文件夹里面,然后在bsp_gpio.h文件夹里面输入如下内容:

示例代码15.3.1 bsp_gpio.h文件代码
1  #ifndef _BSP_GPIO_H
2  #define _BSP_GPIO_H
3  #define _BSP_KEY_H
4  #include "imx6ul.h"
5/***************************************************************
6  Copyright © zuozhongkai Co., Ltd. 1998-2019. All rights reserved.
7文件名	 : bsp_gpio.h
8作者      : 左忠凯
9版本      : V1.0
10描述      : GPIO操作文件头文件。
11其他      : 无
12论坛      : www.openedv.com
13日志      : 初版V1.0 2019/1/4 左忠凯创建
14 ***************************************************************/
15
16/* 枚举类型和结构体定义 */
17typedefenum _gpio_pin_direction
18{
19     kGPIO_DigitalInput =0U,/* 输入 */
20     kGPIO_DigitalOutput =1U,/* 输出 */
21} gpio_pin_direction_t;
22
23/* GPIO配置结构体 */
24typedefstruct _gpio_pin_config
25{
26     gpio_pin_direction_t direction;/* GPIO方向:输入还是输出		*/
27uint8_t outputLogic;	/* 如果是输出的话,默认输出电平	*/
28} gpio_pin_config_t;
29
30
31/* 函数声明 */
32void gpio_init(GPIO_Type *base,int pin, gpio_pin_config_t *config);
33int gpio_pinread(GPIO_Type *base,int pin);
34void gpio_pinwrite(GPIO_Type *base,int pin,int value);
35
36 #endif
bsp_gpio.h中定义了一个枚举类型gpio_pin_direction_t和结构体gpio_pin_config_t,枚举类型gpio_pin_direction_t表示GPIO方向,输入或输出。结构体gpio_pin_config_t是GPIO的配置结构体,里面有GPIO的方向和默认输出电平两个成员变量。在bsp_gpio.c中输入如下所示内容:
示例代码15.3.2 bsp_gpio.c文件代码
1  #include "bsp_gpio.h"
2/***************************************************************
3  Copyright © zuozhongkai Co., Ltd. 1998-2019. All rights reserved.
4文件名    : bsp_gpio.h
5作者      : 左忠凯
6版本      : V1.0
7描述      : GPIO操作文件。
8其他      : 无
9论坛      : www.openedv.com
10日志      : 初版V1.0 2019/1/4 左忠凯创建
11 ***************************************************************/
12
13/*
14  * @description  	: GPIO初始化。
15  * @param - base 	: 要初始化的GPIO组。
16  * @param - pin   	: 要初始化GPIO在组内的编号。
17  * @param - config : GPIO配置结构体。
18  * @return           : 无
19  */
20void gpio_init(GPIO_Type *base,int pin, gpio_pin_config_t *config)
21{
22	if(config->direction == kGPIO_DigitalInput)	/* 输入 */
23	{
24	base->GDIR &=~(1<< pin);
25	}
26		else			/* 输出 */
27	{
28	base->GDIR |=1<< pin;
29	gpio_pinwrite(base,pin, config->outputLogic);/* 默认输出电平 */
30	}
31}
32
33/*
34   * @description  	: 读取指定GPIO的电平值。
35   * @param – base	: 要读取的GPIO组。
36   * @param - pin  	: 要读取的GPIO脚号。
37   * @return       	: 无
38   */
39int gpio_pinread(GPIO_Type *base,int pin)
40{
41	return(((base->DR)>> pin)&0x1);
42}
43
44/*
45   * @description  	: 指定GPIO输出高或者低电平。
46   * @param – base	: 要输出的的GPIO组。
47   * @param - pin  	: 要输出的GPIO脚号。
48   * @param – value	: 要输出的电平,1 输出高电平, 0 输出低低电平
49   * @return       	: 无
50   */
51void gpio_pinwrite(GPIO_Type *base,int pin,int value)
52{
53	if(value ==0U)
54	{
55	base->DR &=~(1U<< pin);/* 输出低电平 */
56	}
57	else
58	{
59	base->DR |=(1U<< pin);/* 输出高电平 */
60	}
61}
文件bsp_gpio.c中有三个函数:gpio_init、gpio_pinread和gpio_pinwrite,函数gpio_init用于初始化指定的GPIO引脚,最终配置的是GDIR寄存器,此函数有三个参数,这三个参数的含义如下:
base:要初始化的GPIO所属于的GPIO组,比如GPIO1_IO18就属于GPIO1组。
pin:要初始化GPIO在组内的标号,比如GPIO1_IO18在组内的编号就是18。
config:要初始化的GPIO配置结构体,用来指定GPIO配置为输出还是输入。
函数gpio_pinread是读取指定的GPIO值,也就是读取DR寄存器的指定位,此函数有两个参数和一个返回值,参数含义如下:
base:要读取的GPIO所属于的GPIO组,比如GPIO1_IO18就属于GPIO1组。
pin:要读取的GPIO在组内的标号,比如GPIO1_IO18在组内的编号就是18。
返回值:读取到的GPIO值,为0或者1。
函数gpio_pinwrite是控制指定的GPIO引脚输入高电平(1)或者低电平(0),就是设置DR寄存器的指定位,此函数有三个参数,参数含义如下:
base:要设置的GPIO所属于的GPIO组,比如GPIO1_IO18就属于GPIO1组。
pin:要设置的GPIO在组内的标号,比如GPIO1_IO18在组内的编号就是18。
value:要设置的值,1(高电平)或者0(低电平)。
我们以后就可以使用函数gpio_init设置指定GPIO为输入还是输出,使用函数gpio_pinread和gpio_pinwrite来读写指定的GPIO,文件bsp_gpio.c文件就讲解到这里。
接下来编写按键驱动文件,新建bsp_key.c和bsp_key.h这两个文件,将这两个文件都保存到刚刚创建的bsp/key文件夹里面,然后在bsp_key.h文件夹里面输入如下内容:
示例代码15.3.3 bsp_key.h文件代码
1  #ifndef _BSP_KEY_H
2  #define _BSP_KEY_H
3  #include "imx6ul.h"
4/***************************************************************
5  Copyright © zuozhongkai Co., Ltd. 1998-2019. All rights reserved.
6文件名    : bsp_key.h
7作者      : 左忠凯
8版本      : V1.0
9描述      : 按键驱动头文件。
10其他      : 无
11论坛      : www.openedv.com
12日志      : 初版V1.0 2019/1/4 左忠凯创建
13 ***************************************************************/
14
15/* 定义按键值 */
16enum keyvalue{
17	KEY_NONE   =0,
18	KEY0_VALUE,
19};
20
21/* 函数声明 */
22void key_init(void);
23int key_getvalue(void);
24
25 #endif
	bsp_key.h文件中定义了一个枚举类型:keyvalue,此枚举类型表示按键值,因为I.MX6U-ALPHA开发板上只有一个按键,因此枚举类型里面只到KEY0_VALUE。在bsp_key.c中输入如下所示内容:
示例代码15.3.4 bsp_key.c文件代码
1  #include "bsp_key.h"
2  #include "bsp_gpio.h"
3  #include "bsp_delay.h"
4/***************************************************************
5  Copyright © zuozhongkai Co., Ltd. 1998-2019. All rights reserved.
6文件名    : bsp_key.c
7作者      : 左忠凯
8版本      : V1.0
9描述      : 按键驱动文件。
10其他      : 无
11论坛      : www.openedv.com
12 日志      : 初版V1.0 2019/1/4 左忠凯创建
13 ***************************************************************/
14
15/*
16  * @description  	: 初始化按键
17  * @param        	: 无
18  * @return       	: 无
19  */
20void key_init(void)
21{
22	gpio_pin_config_t key_config;
23
24	/* 1、初始化IO复用, 复用为GPIO1_IO18 */
25	IOMUXC_SetPinMux(IOMUXC_UART1_CTS_B_GPIO1_IO18, 0);
26
27	/* 2、、配置UART1_CTS_B的IO属性
28	 *bit 16:0 HYS关闭
29	 *bit [15:14]: 11 默认22K上拉
30	 *bit [13]: 1 pull功能
31	 *bit [12]: 1 pull/keeper使能
32	 *bit [11]: 0 关闭开路输出
33	 *bit [7:6]: 10 速度100Mhz
34	 *bit [5:3]: 000 关闭输出
35	 *bit [0]: 0 低转换率
36	*/
37	IOMUXC_SetPinConfig(IOMUXC_UART1_CTS_B_GPIO1_IO18, 0xF080);
38
39	/* 3、初始化GPIO GPIO1_IO18设置为输入*/
40	key_config.direction = kGPIO_DigitalInput;
41	gpio_init(GPIO1,18,&key_config);
42
43}
44
45 /*
46  * @description  	: 获取按键值
47  * @param        	: 无
48  * @return       	: 0 没有按键按下,其他值:对应的按键值
49  */
50int key_getvalue(void)
51{
52	int ret =0;
53	staticunsignedchar release =1;	/* 按键松开 */
54
55	if((release==1)&&(gpio_pinread(GPIO1,18)==0))/* KEY0按下 */
56	{
57	delay(10);/* 延时消抖	*/
58	release =0;/* 标记按键按下 */
59	if(gpio_pinread(GPIO1,18)==0)
60		ret = KEY0_VALUE;
61	}
62	elseif(gpio_pinread(GPIO1,18)==1)		/* KEY0未按下 */
63	{
64	ret =0;
65	release =1;/* 标记按键释放 */
66	}
67
68	return ret;
69}
bsp_key.c中一共有两个函数:key_init和key_getvalue,key_init是按键初始化函数,用来初始化按键所使用的UART1_CTS这个IO。函数key_init先设置UART1_CTS复用为GPIO1_IO18,然后配置UART1_CTS这个IO为速度为100MHz,默认22K上拉。最后调用函数gpio_init来设置GPIO1_IO18为输入功能。
函数key_getvalue用于获取按键值,此函数没有参数,只有一个返回值,返回值表示按键值,返回值为0的话就表示没有按键按下,如果返回其他值的话就表示对应的按键按下了。获取按键值其实就是不断的读取GPIO1_IO18的值,如果按键按下的话相应的IO被拉低,那么GPIO1_IO18值就为0,如果按键未按下的话GPIO1_IO18的值就为1。此函数中静态局部变量release表示按键是否释放。

“示例代码15.3.4”中的57行是按键消抖延时函数,延时时间大约为10ms,用于消除按键抖动。理想型的按键电压变化过程如图15.3.1所示:
【正点原子Linux连载】第十五章按键输入试验-摘自【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.0

图15.3.1 理想的按键电压变化过程
在图15.3.1中,按键没有按下的时候按键值为1,当按键在t1时刻按键被按下以后按键值就变为0,这是最理想的状态。但是实际的按键是机械结构,加上刚按下去的一瞬间人手可能也有抖动,实际的按键电压变化过程如图15.3.2所示:

图15.3.2实际的按键电压变化过程
在图15.3.2中t1时刻按键被按下,但是由于抖动的原因,直到t2时刻才稳定下来,t1到t2这段时间就是抖动。一般这段时间就是十几ms左右,从图15.3.2中可以看出在抖动期间会有多次触发,如果不消除这段抖动的话软件就会误判,本来按键就按下了一次,结果软件读取IO值发现电平多次跳变以为按下了多次。所以我们需要跳过这段抖动时间再去读取按键的IO值,也就是至少要在t2时刻以后再去读IO值。在“示例代码15.3.4”中的57行是延时了大约10ms后再去读取GPIO1_IO18的IO值,如果此时按键的值依旧是0,那么就表示这是一次有效的按键触发。
按键驱动就讲解到这里,最后就是main.c文件的内容了,在main.c中输入如下代码:

示例代码15.3.5 main.c文件代码
/**************************************************************
Copyright © zuozhongkai Co., Ltd. 1998-2019. All rights reserved.
文件名   :    mian.c
作者     : 左忠凯
版本     : V1.0
描述     : I.MX6U开发板裸机实验7 按键输入实验
其他     : 本实验主要学习如何配置I.MX6U的GPIO作为输入来使用,通过
开发板上的按键控制蜂鸣器的开关。
论坛     : www.openedv.com
日志     : 初版V1.0 2019/1/4 左忠凯创建
**************************************************************/
1  #include "bsp_clk.h"
2  #include "bsp_delay.h"
3  #include "bsp_led.h"
4  #include "bsp_beep.h"
5  #include "bsp_key.h"
6
7/*
8   * @description  	: main函数
9   * @param        	: 无
10  * @return       	: 无
11  */
12int main(void)
13{
14	int i =0;
15	int keyvalue =0;
16	unsignedchar led_state = OFF;
17	unsignedchar beep_state = OFF;
18
19	clk_enable();/* 使能所有的时钟	*/
20	led_init();/* 初始化led     	*/
21	beep_init();/* 初始化beep    	*/
22	key_init();/* 初始化key     	*/
23
24	while(1)
25	{
26	keyvalue = key_getvalue();
27	if(keyvalue)
28	{
29		switch(keyvalue)
30		{
31			case KEY0_VALUE:
32		beep_state =!beep_state;
33		beep_switch(beep_state);
34		break;
35		}
36	}
37	i++;
38	if(i==50)
39	{
40		i =0;
41		led_state =!led_state;
42		led_switch(LED0, led_state);
43	}
44		delay(10);
45		}
46	return0;
47}
main.c函数先初始化led灯、蜂鸣器和按键,然后在while(1)循环中不断的调用函数key_getvalue来读取按键值,如果KEY0按下的话就打开/关闭蜂鸣器。LED0作为系统提示指示灯闪烁,闪烁周期大约为500ms。本章例程的软件编写就到这里结束了,接下来就是编译下载验证了。

15.4编译下载验证
15.4.1编写Makefile和链接脚本
Makefile使用第十三章编写的通用Makefile,修改变量TARGET为key,在变量INCDIRS和SRCDIRS中追加“bsp/gpio”和“bsp/key”,修改完成以后如下所示:

示例代码15.4.1.1  Makefile文件代码
1  CROSS_COMPILE	?= arm-linux-gnueabihf-
2  TARGET          	?=key
3
4/* 省略掉其它代码...... */
5
6  INCDIRS	:= imx6ul \
7		bsp/clk \
8		bsp/led \
9		bsp/delay  \
10 		bsp/beep \
11		bsp/gpio \
12		bsp/key
13
14 SRCDIRS	:=	project \
15		bsp/clk \
16		bsp/led \
17		bsp/delay \
18		bsp/beep \
19		bsp/gpio \
20		bsp/key
21
22/* 省略掉其它代码...... */
23
24 clean:
25  rm -rf $(TARGET).elf $(TARGET).dis $(TARGET).bin $(COBJS)$(SOBJS)
第2行修改变量TARGET为“key”,也就是目标名称为“key”。
第11、12行在变量INCDIRS中添加GPIO和按键驱动头文件(.h)路径。
第19、20行在变量SRCDIRS中添加GPIO和按键驱动文件(.c)路径。

链接脚本就使用第十三章试验中的链接脚本文件imx6ul.lds即可。
15.4.2编译下载
使用Make命令编译代码,编译成功以后使用软件imxdownload将编译完成的key.bin文件下载到SD卡中,命令如下:
chmod 777 imxdownload //给予imxdownload可执行权限,一次即可
./imxdownload key.bin /dev/sdd //烧写到SD卡中
烧写成功以后将SD卡插到开发板的SD卡槽中,然后复位开发板。如果代码运行正常的话LED0会以大约500ms周期闪烁,按下开发板上的KEY0按键,蜂鸣器打开,再按下KEY0按键,蜂鸣器关闭。