Linux下等待队列、定时器、中断综合应用——按键控制LED

本文通过按键控制LED的亮灭,按键每按一次,LED的状态就发生一次变化。
等待队列是为了在按键有动作发生时再读取按键值,而不是一直读取按键的值,使得CPU的占用率很高。
定时器在本实验中引入是为了按键消抖,在键值稳定了之后再通过内核读出键值到用户端,用户端得知键值之后再将键值写入LED,LED根据写入的值就会有相应的亮或灭状态。
之前按键的实验就是通过按键按下或者松开给按键对应的GPIO赋值,本例中的按键是通过中断来实现的,按键每次有动作就会触发中断,然后在中断里完成按键值的翻转。
本文的LED驱动代码就是文章Linux下设备树、pinctrl和gpio子系统、LED灯驱动实验中使用的代码。
按键驱动代码中使用了中断,简单介绍一下和中断有关的概念和函数。
每个中断都有一个中断号,通过中断号即可区分不同的中断,在Linux内核中使用一个int变量表示中断号。
在Linux内核中要想使用某个中断是需要申请的,request_irq函数用于申请中断,request_irq函数可能会导致睡眠,因此不能在中断上下文或者其他禁止睡眠的代码段中使用request_irq函数。request_irq函数会激活(使能)中断,request_irq 函数原型如下。

int request_irq(unsigned int irq,irq_handler_t handler,unsigned long flags,const char *name,void *dev)

irq:要申请的中断号。
handler:中断处理函数,当中断发生时,执行该函数。
flags:中断标志,常用的有以下几个。
在这里插入图片描述
name:中断名字,设置后在/proc/interrupts目录下可以查看。
dev:如果flags设置为IRQF_SHARED,dev用来区分不同的中断,一般情况下,dev设置为设备结构体,它会传给中断处理函数irq_handler_t第二个参数。
free_irq函数用于释放掉相应的中断,其原型如下。

void free_irq(unsigned int irq,void *dev)

int:要释放的中断号。
dev:如果flags设置为IRQF_SHARED,dev用来区分不同的中断。
使用request_irq函数申请中断的时候需要设置中断处理函数,中断处理函数的原型如下。

irqreturn_t (*irq_handler_t) (int, void *)

第一个参数是中断号。第二个参数是一个指向void 的指针,是个通用指针,需要与request_irq函数的dev参数保持一致。用于区分共享中断的不同设备,dev也可以指向设备数据结构。中断处理函数的返回值为irqreturn_t 类型,irqreturn_t 类型定义如下。

enum irqreturn {IRQ_NONE = (0 << 0),IRQ_HANDLED = (1 << 0),IRQ_WAKE_THREAD = (1 << 1),
};typedef enum irqreturn irqreturn_t;

一般中断服务函数返回值使用如下形式。

return IRQ_RETVAL(IRQ_HANDLED)

按键的驱动代码如下。

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_gpio.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#include <linux/platform_device.h>
#include <linux/of_irq.h> 
#include <linux/wait.h>
#include <linux/sched.h>
#include <linux/timer.h>#define KEY_CNT			    1		 /* 设备号个数 */
#define KEY_NAME		"gpio_key"	/* 名字 */dev_t devid;			  /* 设备号 	 */
struct cdev cdev;		  /* cdev 	*/
struct class *class;	  /* 类 */
struct device *device;	  /* 设备 	 */
int major;				  /* 主设备号	  */
int minor;				  /* 次设备号   */
struct device_node	*nd;   /* 设备节点 */
int key_gpio;			   /* key所使用的GPIO编号*/int irq;
int value = 0;
int wq_flags = 0;  //标志位
int key_count = 0;
static void timer_function(unsigned long data);
DEFINE_TIMER(key_timer,timer_function,0,0); //静态定义结构体变量并且初始化function,expires,data成员
DECLARE_WAIT_QUEUE_HEAD(key_wq);   // 定义并初始化等待队列头static void timer_function(unsigned long data)
{++key_count;printk("key trigger count: %d\r\n",key_count);
}static ssize_t key_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)
{wait_event_interruptible(key_wq,wq_flags);   //等待按键触发if(copy_to_user(buf,&value,sizeof(value))!=0){printk("copy_to_user error!\n");return -1;}wq_flags = 0;return 0;
}static struct file_operations key_fops = {.owner = THIS_MODULE,.read = key_read
};irqreturn_t key_led(int irq, void *args)
{mod_timer(&key_timer,jiffies + msecs_to_jiffies(10));value = !value;printk("key_value = %d\n", value);wq_flags = 1;  //这里只有先置1,下面的函数才能唤醒wake_up(&key_wq);  //唤醒return IRQ_RETVAL(IRQ_HANDLED);
}const struct of_device_id of_match_table_key[] = {{.compatible = "gpio_bus_key"},        //与设备树中的compatible属性匹配{}
};struct platform_driver dts_device = {    .driver = {.owner = THIS_MODULE,.name = "keygpio",.of_match_table = of_match_table_key}
};static int __init gpio_key_init(void)
{int ret;nd = of_find_node_by_path("/key");if (nd == NULL) return -EINVAL;printk("Find node key!\n");key_gpio = of_get_named_gpio(nd,"key-gpio",0);if (key_gpio < 0) {printk("of_get_named_gpio failed!\r\n");return -EINVAL;}printk("key_gpio = %d\r\n", key_gpio);gpio_request(key_gpio,KEY_NAME);	    //申请gpiogpio_direction_input(key_gpio);   	//将gpio设置为输入/*获得gpio中断号*///irq = gpio_to_irq(gpio_num); irq = irq_of_parse_and_map(nd,0); //与上面这句代码的作用相同printk("irq = %d\r\n", irq);/*申请中断*/ret = request_irq(irq,key_led,IRQF_TRIGGER_RISING,"key_led_test", NULL); if(ret < 0){printk("request_irq error!\n");return -1;}if (major){		devid = MKDEV(major, 0);register_chrdev_region(devid, KEY_CNT, KEY_NAME);}else {						alloc_chrdev_region(&devid,0,KEY_CNT,KEY_NAME);	//申请设备号major = MAJOR(devid);	   //获取分配号的主设备号 minor = MINOR(devid);	}printk("gpiokey major=%d,minor=%d\r\n",major,minor);	cdev.owner = THIS_MODULE;cdev_init(&cdev, &key_fops);cdev_add(&cdev, devid, KEY_CNT);class = class_create(THIS_MODULE, KEY_NAME);if (IS_ERR(class))return PTR_ERR(class);device = device_create(class,NULL,devid,NULL,KEY_NAME);if (IS_ERR(device))return PTR_ERR(device);platform_driver_register(&dts_device); return 0;
}static void __exit gpio_key_exit(void)
{gpio_free(key_gpio);cdev_del(&cdev);unregister_chrdev_region(devid,KEY_CNT); device_destroy(class, devid);class_destroy(class);free_irq(irq,NULL);del_timer(&key_timer);platform_driver_unregister(&dts_device);printk("driver exit!\n");
}module_init(gpio_key_init);
module_exit(gpio_key_exit);
MODULE_LICENSE("GPL");

测试代码如下。

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>int main(int argc, char *argv[])
{int fd_r,fd_w;char value[1];fd_r = open("/dev/gpio_key",O_RDWR);fd_w = open("/dev/gpioled",O_RDWR); if(fd_r < 0){perror("open key error\n"); return fd_r;}if(fd_w < 0){perror("open led error\n"); return fd_w;}while(1){read(fd_r,value,sizeof(value));write(fd_w,value,sizeof(value));}return 0;
}

测试代码完成的工作就是从按键的内核中读取键值到用户端,然后将读取到的键值写入到LED的内核中。
通过上面的代码编译出LED的驱动和按键的驱动,然后用交叉编译器编译测试文件生成一个可执行文件,将这三个文件发送到开发板进行验证。
在这里插入图片描述
通过实验结果可知,LED的亮灭是配合按键的动作的。
本文参考文档:
I.MX6U嵌入式Linux驱动开发指南V1.5——正点原子

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/110501.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

手写一个PrattParser基本运算解析器1: 编译原理概述

点击查看 基于Swift的PrattParser项目 编译原理概述 编译原理是我们每一个程序猿必须要了解的技能, 编译原理实际上并没有啥高深的技术, 我们如果在做业务开发, 也很少会用到编译开发的知识, 但是编译原理又是我们必备的基础知识之一. 所以我们需要对编译原理的内容有一个大概的…

【PXIE301-211】基于PXIE总线的16路并行LVDS数据采集、4路低速、2路隔离RS422数据处理平台

板卡概述 PXIE301-211A是一款基于PXIE总线架构的16路高速LVDS、4路低速LVDS采集、2路隔离RS422数据处理平台&#xff0c;该平台板卡采用Xilinx的高性能Kintex 7系列FPGA XC7K325T作为实时处理器&#xff0c;实现各个接口之间的互联。板载1组64位的DDR3 SDRAM用作数据缓存。板卡…

[Hive] explode

在 Hive 中&#xff0c;explode 函数用于将数组&#xff08;Array&#xff09;或者Map类型的列拆分成多行&#xff0c; 每个元素或键值对为一行。这允许我们在查询中对数组或 Map 进行扁平化操作。 下面是使用 explode 函数的示例&#xff1a; 假设我们有一个包含数组字段的表…

pycharm操作git

pycharm操作git 之前用命令做的所有操作&#xff0c;使用pychrm点点就可以完成 克隆代码 上方工具栏Git ⇢ \dashrightarrow ⇢ Clone ⇢ \dashrightarrow ⇢ 填写地址&#xff08;http、ssh&#xff09; 提交到暂存区&#xff0c;提交到版本库&#xff0c;推送到远程 直接…

HTTP和HTTPS

目录 HTTP协议 1.HTTP协议 2.HTTP请求 URL 方法 GET请求 post请求 header报头 请求正文&#xff08;body&#xff09; 3.HTTP响应 结构 常见的状态码 4.form表单构造HTTP请求 5.通过ajax构造HTTP请求 6.使用工具postman ​编辑 HTTPS 1.对称加密 2.非对称加密…

填充颜色游戏

无语死了这题。 题目描述 小明最近迷上下面一款游戏。游戏开始时&#xff0c; 系统将随机生成一个 N N 的 正方形棋盘&#xff0c; 棋盘的每个格子都由六种颜色中的一种绘制。在每个步骤中&#xff0c; 玩家选择一种颜色&#xff0c; 并将与左上角连接的所有网格更改为该特…

jenkins 安装与使用、用户权限划分

jenkins 安装与使用 安装插件&#xff1a; 开启该插件功能 验证用户管理 创建web01~02 使用web01登录 用户权限划分 安装 Role-Based Strategy 插件后&#xff0c;系统管理 中多了如图下所示的一个功能&#xff0c;用户权限的划分就是靠他来做的 创建角色 重新访问 创建项目…

Zabbix“专家坐诊”第207期问答汇总

问题一 Q&#xff1a;不小心把host表删除了&#xff0c;怎么处理&#xff1f;现在使用的zabbix 4.0.3的server&#xff0c;agent是4.2.1&#xff0c;能不能不动agent的情况下升级server版本&#xff0c;重新部署&#xff1f; A&#xff1a;数据库有备份话恢复即可&#xff0c;…

SSTI模板注入(flask) 学习总结

文章目录 Flask-jinja2 SSTI 一般利用姿势SSTI 中常用的魔术方法内建函数 利用 SSTI 读取文件Python 2Python 3 利用 SSTI 执行命令寻找内建函数 eval 执行命令寻找 os 模块执行命令寻找 popen 函数执行命令寻找 importlib 类执行命令寻找 linecache 函数执行命令寻找 subproce…

windows中elasticsearch7中添加用户名密码验证

1.找到elsatic的bin目录输入cmd 2.生成ca证书 输入 elasticsearch-certutil ca 在es7根目录生成ca证书&#xff0c;输入密码时直接回车即可&#xff0c;否则后面会报错 Please enter the desired output file [elastic-stack-ca.p12]: #这里直接回车即可 Enter password for…

JAVA学习日记1——JAVA简介及第一个java程序

简单记忆 JAVA SE &#xff1a;标准版&#xff0c;核心基础 JAVA EE&#xff1a;企业版&#xff0c;进阶 JDK&#xff1a;Java Development Kit&#xff0c;Java开发工具包&#xff0c;包含JRE JRE&#xff1a;Java Runtime Environment&#xff0c;Java运行时环境&#xff…

手撕 视觉slam14讲 ch7 / pose_estimation_3d2d.cpp (2)

上一篇文章中: 手撕ch7/pose_estimation_3d2d&#xff08;1&#xff09;&#xff0c;我们调用了epnp的方法进行位姿估计&#xff0c;这里我们使用非线性优化的方法来求解位姿&#xff0c;使用g2o进行BA优化 首先介绍g2o&#xff1a;可参考&#xff1a;g2o详细介绍 1.构建g2o图…

加权平均、EMD、小波等方法去噪效果对比

加权平均、EMD、小波等方法去噪效果对比 代码 整体代码如下 %% clear all; clc;load(data_filter120Hz.mat); %可自己生成随机噪声 fs1000;%采样频率是1000Hz %% %生成正弦波信号 tlinspace(0, length(data)/fs-1/fs, length(data)); y1 15*sin(2*pi* 2.8 *t);%生成频率为2.…

Android之使用QBadgeView给TabLayout顶部栏设置数量角标,数值可更新

TabLayout搭配ViewPager、Fragement使用可看另一篇文章&#xff1a; Android中TabLayoutViewPagerFragment实现顶部导航栏 本文主要描述给TabLayout的某一栏添加角标&#xff0c;数值可更新&#xff1a; 一、效果 二、TabLayout使用 1、xml文件中 <com.google.android.m…

通讯协议学习之路:QSPI协议理论

通讯协议之路主要分为两部分&#xff0c;第一部分从理论上面讲解各类协议的通讯原理以及通讯格式&#xff0c;第二部分从具体运用上讲解各类通讯协议的具体应用方法。 后续文章会同时发表在个人博客(jason1016.club)、CSDN&#xff1b;视频会发布在bilibili(UID:399951374) 一、…

【Django 01】环境搭配与项目配置

1. 介绍 https://github.com/Joe-2002/sweettalk-django4.2#readme Django 是一个使用 Python 编写的开源 Web 应用程序框架&#xff0c;它提供了一套用于快速开发安全、 可扩展和高效的 Web 应用程序的工具和功能。Django 基于 MVC&#xff08;Model-View-Controller&#xf…

Windows11家庭版没有本地组策略编辑器解决

1. 新建一个文本文件将下面代码粘到里面&#xff0c;保存后修改后缀为.cmd或者.bat echo off pushd "%~dp0"dir /b C:\Windows\servicing\Packages\Microsoft-Windows-GroupPolicy-ClientExtensions-Package~3*.mum >List.txt dir /b C:\Windows\servicing\Packa…

redis(普通连接和连接池、字符串类型、hash类型、列表类型)

1 redis普通连接和连接池 1.1 普通连接 1.2 连接池 2 redis字符串类型 3 redis hash类型 4 redis列表类型 1 redis普通连接和连接池 #1 python 代码作为客户端---》连接# 2 安装模块&#xff1a;pip install redis1.1 普通连接 from redis import Redisconn Redis(host&qu…

Selenium浏览器自动化怎么上传文件

Selenium 封装了现成的文件上传操作。但是随着现代前端框架的发展&#xff0c;文件上传的方式越来越多样。而有一些文件上传的控件&#xff0c;要做自动化控制会更复杂一些&#xff0c;这篇文章主要讨论在复杂情况下&#xff0c;如何通过自动化完成文件上传。 1. input 元素上传…

蓝桥杯每日一题2023.10.17

迷宫 - 蓝桥云课 (lanqiao.cn) 题目描述 样例&#xff1a; 01010101001011001001010110010110100100001000101010 00001000100000101010010000100000001001100110100101 01111011010010001000001101001011100011000000010000 0100000000101010001101000010100000101010101100…