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

news2025/1/11 0:19:33

本文通过按键控制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);	    //申请gpio
	gpio_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.coloradmin.cn/o/1109392.html

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

相关文章

Node.js中常用的设计模式有哪些?

本文由葡萄城技术团队首发。转载请注明出处&#xff1a;葡萄城官网&#xff0c;葡萄城为开发者提供专业的开发工具、解决方案和服务&#xff0c;赋能开发者。 设计模式简介 设计模式是由经验丰富的程序员在日积月累中抽象出的用以解决通用问题的可复用解决方案&#xff0c;它提…

谷歌浏览器修改背景色

地址栏输入&#xff1a;chrome://flags/ 改为Enabled即可把背景色改为黑色&#xff0c;亲测edge浏览器也生效

点菜MT2234

【题目】 这个题读起来真拗口&#xff0c;啥叫“有可能不存在”&#xff1f; 如果结果为yes&#xff0c;那就是说“没有三个人都选中的菜”&#xff0c; 如果结果为no&#xff0c;那就是说“有三个人都选中的菜” 换句话说&#xff0c;那就是没有一道菜被3个人都点过。 abc …

2022年亚太杯APMCM数学建模大赛D题储能系统中传热翅片的结构优化求解全过程文档及程序

2022年亚太杯APMCM数学建模大赛 D题 储能系统中传热翅片的结构优化 原题再现 高效储能技术是解决可再生能源和余热资源波动性和间歇性的核心技术。相变蓄热以其较高的储能密度和近恒温蓄热放热而得到广泛应用。固-液相变材料具有相变前后相变潜热高、体积变化小等特点&#x…

专业管理菜单的增删改、查重

1&#xff0c;点击专业管理菜单------查询所有专业信息列表 ①点击菜单&#xff0c;切换专业组件 ②切换到列表组件后&#xff0c;向后端发送请求到Servlet ③调用DAO层&#xff0c;查询数据库&#xff08;sql&#xff09;&#xff0c;封装查询到的内容 ④从后端向前端做出…

律师必备软件Alpha系统,为专业律师提供高效解决方案!

Alpha法律智能操作系统由iCourt开发&#xff0c;是一款集大数据、市场拓展、律所管理等功能于一体的智能操作系统&#xff0c;它旨在通过强大的法律科技赋能法律人&#xff0c;更快速地获取同类型案件的办理经验&#xff0c;进而减轻律师工作负担、提升工作效率。目前&#xff…

python知识:从PDF 提取文本

一、说明 PDF 到文本提取是自然语言处理和数据分析中的一项基本任务&#xff0c;它允许研究人员和数据分析师从 PDF 文件中包含的非结构化文本数据中获得见解。Python 是一种通用且广泛使用的编程语言&#xff0c;它提供了多个库和工具来促进提取过程。 二、各种PDF操作库 让我…

SPSS|正负偏态的转换方法|限值1.96|反转后处理(对数法)|正态得分法|实战小练-SPSS学习(2)

目录 学习目的软件版本参考文档基础数据正负偏态的转换方法&#xff08;引自《小白爱上SPSS》&#xff09;正偏态数据转换方法负偏态数据转换 实战数据准备数据初探输出结果分析查看峰度、偏度查看峰度标准误差、偏度标准误差计算偏度系数和峰度系数Tips&#xff1a;为什么判断…

Anthropic全球上线AI语言模型Claude 2;多模态系统:融合文本和图像的新前沿

&#x1f989; AI新闻 &#x1f680; Anthropic全球上线AI语言模型Claude 2&#xff0c;编程、数学、推理能力大幅提升 摘要&#xff1a;Anthropic在全球正式上线了AI语言模型Claude 2。相比前代版本&#xff0c;Claude 2在编程、数学、推理等方面都有大幅提升&#xff0c;支…

MySQL学习(八)——锁

文章目录 1. 锁概述2. 全局锁2.1 全局锁的必要性2.2 语法2.3 全局锁的特点 3. 表级锁3.1 表锁3.2 元数据锁3.3 意向锁3.4 自增锁 4. 行级锁4.1 介绍4.2 记录锁4.3 间隙锁4.4 临键锁 1. 锁概述 锁是计算机协调多个进程或线程并发访问某一资源的机制。在数据库中&#xff0c;除传…

使用Spire.PDF for Python插件从PDF文件提取文字和图片信息

目录 一、Spire.PDF插件的安装 二、从PDF文件提取文字信息 三、从PDF文件提取图片信息 四、提取图片和文字信息的进阶应用 总结 在Python中&#xff0c;提取PDF文件的文字和图片信息是一种常见的需求。为了满足这个需求&#xff0c;许多开发者会选择使用Spire.PDF插件&…

13.2测试用例

一.测试用例的四个重要要素 1.测试环境 2.操作步骤 3.测试数据 4.预期结果 注意:不需要执行结果. 5.为什么要有测试用例 a.提高测试效率,节约测试设计. b.测试用例是自动化测试的前提. 二.设计方法(黑盒测试) a.基于需求的设计方法(4,0.50) b.等价类: 依据需求,将输…

软件开发团队的成本分布

软件开发团队的成本分布通常涵盖了各种因素和人员&#xff0c;这些因素在项目的不同阶段会有所变化。以下是一个一般性的软件开发团队成本分布示例&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合作。 开发…

Pysied6 ComboBox

Pysied6 ComboBox Pysied6 ComboBoxComboBox常用函数ComboBox常用信号例程ComboBox添加选项程序设置界面设置 返回选项信息添加删除选项完整程序界面程序主程序 Pyside6的ComboBox下拉列表框&#xff0c;可以给用户提供一系列的选项&#xff0c;下面就来简单了解一下Pysied6 Co…

SpringCloud-Sentinel

一、介绍 &#xff08;1&#xff09;提供界面配置配置服务限流、服务降级、服务熔断 &#xff08;2&#xff09;SentinelResource的blockHandler只处理后台配置的异常&#xff0c;运行时异常fallBack处理&#xff0c;且资源名为value时才生效&#xff0c;走兜底方法 二、安装…

数据分析:密度图

目前拥有的数据如图&#xff0c;三列分别对应瑕疵种类&#xff0c;对应的置信 度&#xff0c;x方向坐标。 现在想要做的事是观看瑕疵种类和置信度之间的关系。 要显示数据分布的集中程度&#xff0c;可以使用以下几种常见的图形来观察&#xff1a; 1、箱线图&#xff08;Box P…

【工具】电脑网络连接正常,但是有些页面无法登录,如何解决?

目录 0.环境 1.背景 2.具体操作 0.环境 windows 11 64位 1.背景 我们公司的大楼无法进移动联通这种网络的线&#xff0c;所以网络用的是小厂的&#xff0c;有些小厂为了提高网络速度&#xff0c;会屏蔽一些网站&#xff0c;这就导致在工作中有些网站直接通过WIFI连接是无法访…

头歌平台——基于结构体的学生信息管理系统

第1关&#xff1a;学生信息管理系统 任务描述 本关任务&#xff1a;编写一个基于结构体得学生成绩信息管理系统。主要功能如下&#xff1a; 1. 用结构体存放所有数据。 2. 每个功能都用函数实现。 3. 输入10个学生的学号和三门课程的成绩。 4. 计算每个学生的总分。 5. 按总分…

头歌平台——C语言之数学运算强化练习题

第1关&#xff1a;逐一挑出来 任务描述 本关需要完成一个输出某个三位数各个位上的数字的小程序。 相关知识 算数运算符 %是取模运算符&#xff0c;获得整除后的余数。 /是通过分子除以分母&#xff0c;获得整除后的整数部分。 floor floor()是返回一个浮点型的整数部分…

腾讯云服务器带宽下载速度表(附上行带宽计算方法)

腾讯云服务器公网带宽下载速度计算&#xff0c;1M公网带宽下载速度是128KB/秒&#xff0c;5M带宽下载速度是512KB/s&#xff0c;腾讯云10M带宽下载速度是1.25M/秒&#xff0c;腾讯云百科txybk.com来详细说下腾讯云服务器不同公网带宽实际下载速度以及对应的上传速度对照表&…