zl程序教程

您现在的位置是:首页 >  系统

当前栏目

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

Linux驱动嵌入式输入开发 指南 原子 连载
2023-09-11 14:20:39 时间

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所示:
在这里插入图片描述

图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所示:
在这里插入图片描述

图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按键,蜂鸣器关闭。