韦东山嵌入式Liunx入门驱动开发五

文章目录

      • 一、驱动程序基石
        • 1-1 休眠与唤醒
        • 1-2 POLL机制
        • 1-3 异步通知
          • (1) 异步通知程序解析
          • (2) 异步通知机制内核代码详解
        • 1-4 阻塞与非阻塞
        • 1-5 定时器
          • (1) 内核函数
          • (2) 定时器时间单位
        • 1-6 中断下半部 tasklet

本人学习完韦老师的视频,因此来复习巩固,写以笔记记之。
韦老师的课比较难,第一遍不知道在说什么,但是坚持看完一遍,再来复习,基本上就水到渠成了。
看完视频复习的同学观看最佳!
基于 IMX6ULL-PRO
参考视频 Linux快速入门到精通视频
参考资料:01_嵌入式Linux应用开发完全手册V5.1_IMX6ULL_Pro开发板.pdf

一、驱动程序基石

1-1 休眠与唤醒

当应用程序必须等待某个事件发生,比如必须等待按键被按下时, 可以使用休眠-唤醒机制
① APP调用read等函数试图读取数据,比如读取按键;
② APP进入内核态,也就是调用驱动中的对应函数,发现有数据则复制到用户空间并马上返回;
③ 如果APP在内核态,也就是在驱动程序中发现没有数据,则APP休眠;
④ 当有数据时,比如当按下按键时,驱动程序的中断服务程序被调用,它会记录数据、唤醒APP
⑤ APP继续运行它的内核态代码,也就是驱动程序中的函数,复制数据到用户空间并马上返回。
在这里插入图片描述
驱动框架
在这里插入图片描述
休眠,直到condition 为真;休眠期间是可被打断的,可以被信号打断

wait_event_interruptible(wq, condition)

唤醒wq队列中状态为“ TASK_INTERRUPTIBLE ”的线程,只唤醒其中的一个线程

wake_up_interruptible(wq)

要休眠的线程,放在wq 队列里,中断处理函数从wq队列里把它取出来唤
醒。
① 初始化wq队列
② 在驱动的read函数中,调用 wait_event_interruptible。它本身会判断
event是否为 FALSE ,如果为FASLE表示无数据,则休眠。
当从wait_event_interruptible 返回后,把数据复制回用户空间。
③ 在中断服务程序里:设置event 为TRUE,并调用wake_up_interruptible 唤醒线程。

1-2 POLL机制

使用休眠唤醒的方式等待某个事件发生时,有一个缺点:等待的时间可能
很久
。我们可以加上一个超时时间,这时就可以使用poll机制。
① APP不知道驱动程序中是否有数据,可以先调用 poll函数查询一下, poll函数可以传入超时时间
② APP进入内核态,调用驱动程序的poll函数,有数据的话立刻返回;
③ 如果发现没有数据时就休眠一段时间
④ 当有数据时,比如当按下按键时,驱动程序的中断服务程序被调用,它会记录数据、唤醒APP
⑤ 当超时时间到了之后,内核也会唤醒APP
⑥ APP根据poll函数的返回值就可以知道是否有数据,如果有数据就调用
read得到数据。

在这里插入图片描述
drv_poll函数需要做的事
① 把当前线程挂入队列wq:poll_wait
②返回设备状态:drv_poll 要返回自己的当前状态:P OLLIN | POLLRDNORM) 或 POLLOUT | POLLWRNORM) 。

static unsigned int gpio_key_drv_poll(struct file *fp, poll_table * wait)
{
	printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
	poll_wait(fp, &gpio_key_wait, wait);
	return is_key_buf_empty() ? 0 : POLLIN | POLLRDNORM;
	
}

button_test.c

int main(int argc, char **argv)
{
	int fd;
	int val;
	struct pollfd fds[1];
	int timeout_ms = 5000;
	int ret;
	
	/* 1. 判断参数 */
	if (argc != 2) 
	{
		printf("Usage: %s <dev>\n", argv[0]);
		return -1;
	}
	
	/* 2. 打开文件 */
	fd = open(argv[1], O_RDWR);
	if (fd == -1)
	{
		printf("can not open file %s\n", argv[1]);
		return -1;
	}

	fds[0].fd = fd;
	fds[0].events = POLLIN;

	while (1)
	{
		/* 3. 读文件 */
		ret = poll(fds, 1, timeout_ms);
		
		if((ret == 1) && (fds[0].revents & POLLIN)){
			read(fd, &val, 4);
			printf("get button : 0x%x\n", val);
		}
		else{
			printf("timeout\n");
		}
	}
	
	close(fd);
	return 0;
}
1-3 异步通知
(1) 异步通知程序解析

异步通知流程如下:
在这里插入图片描述

① APP给SIGIO这个信号注册信号处理函数func,以后APP收到SIGIO信号时,这个函数会被自动调用。
② 把APP的PID(进程 ID)告诉驱动程序,这个调用不涉及驱动程序,在内核的文件系统层次记录PID。

filp->f_owner.pid = get_pid(pid);

③ 读取驱动程序文件Flag
④ 设置Flag里面的FASYNC位为1;当 FASYNC位发生变化时,会导致驱动程序的fasync被调用
⑤ 调用faync_helper ,它会根据 FAYSNC的值决定是否设置
button_async -->fa_file=filp(内含PID);open文件时,会在内核文件系统中有一个struct file *filp结构体,filp->f_owner.pid里面含有之前设置的PID 。
⑥ APP做其他事;当按下按键,发生中断,驱动程序的中断服务程序被调用,里面调用kill_fasync 发信号
⑦ APP收到信号后,它的信号处理函数被自动调用,可以在里面调用
read函数读取按键。

驱动程序中提供对应的drv_fasync函数,并在FAYNC变化时,调用fasync_helper函数,使得button_fasync->fa_file = filp或者NULL

struct fasync_struct *button_fasync;
static int gpio_key_drv_fasync(int fd, struct file *file, int on)
{
	if (fasync_helper(fd, file, on, &button_fasync) >= 0)
		return 0;
	else
		return -EIO;
}

在GPIO中断服务程序中,若button_fasync->fa_file非空,则获得PID,并发信号给上层应用

kill_fasync(&button_fasync, SIGIO, POLL_IN);

上层应用程序
在这里插入图片描述

(2) 异步通知机制内核代码详解

上层应用执行fcntl函数,内核会调用fs/fcntl.c 的如下函数。
在这里插入图片描述
进入do_fcntl函数,flag标志对应函数的cmd
在这里插入图片描述

以下分别对三种flag进行代码演示
① 当flag是F_SETOWN时,内核do_fcntl中调用f_setown函数,最终将pid给filp

fcntl(fd, F_SETOWN, getpid());

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
② 当flag是F_GETFL时,获取文件的状态标志

flags = fcntl(fd, F_GETFL);

在这里插入图片描述
③ 当flag是F_SETFL时,设置文件支持异步通知功能

fcntl(fd, F_SETFL, flags | FASYNC);

在这里插入图片描述
在这里插入图片描述
启动了FASYNC 功能的话,驱动程序的 button_fasync 就被设置了,它指向的 fasync_struct 结构体里含有 filp里含有PID

static int gpio_key_drv_fasync(int fd, struct file *file, int on)
{
	if (fasync_helper(fd, file, on, &button_fasync) >= 0)
		return 0;
	else
		return -EIO;
}

在这里插入图片描述
从button_fasync 指针中,取出 fasync_struct 结构体,从这个结构体的 fa_file 中得到接收方的PID ,然后使用 send_sigio函数发送信号。根据 PID找到进程在内核的 task_struct结构体, 修改里面的某些成员表示收到了信号。

kill_fasync(&button_fasync, SIGIO, POLL_IN);

在这里插入图片描述

1-4 阻塞与非阻塞

所谓阻塞,就是等待某件事情发生。比如调用read读取按键时,如果没有按键数据则read函数不会返回,它会让线程休眠等待。
使用poll时,传入超时时间不为0(阻塞);设置超时时间为0,没有数据立即返回(非阻塞)
如何设置阻塞与非阻塞呢?
① open时

int fd = open(/dev/xxx”, O_RDWR | O_NONBLOCK); 	/* 非阻塞方式*/
int fd = open(/dev/xxx”, O_RDWR ); 	/* 阻塞方式*/

② open后

int flags = fcntl(fd, F_GETFL);
fcntl(fd, F_SETFL, flags | O_NONBLOCK); 	/* 非阻塞方式*/
fcntl(fd, F_SETFL, flags & ~O_NONBLOCK); 	/* 阻塞方式*/

驱动程序中,当APP打开某个驱动时,在内核中会有一个struct file 结构体的f _flags对应打开文件时的标记位;可以设置f _flasgs 的O_NONBLOCK 位,表示非阻塞;也可以清除这个位表示阻塞。

/* 实现对应的open/read/write等函数,填入file_operations结构体*/                
static ssize_t gpio_key_drv_read (struct file *file, char __user *buf, size_t size, loff_t *offset)
{
	//printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
	int err;
	int key;

	if(is_key_buf_empty() && (file->f_flags & O_NONBLOCK))
		return -EAGAIN;
	else{
		wait_event_interruptible(gpio_key_wait, !is_key_buf_empty());
		key = get_key();
		err = copy_to_user(buf, &key, 4);
		return 4;
	}
}
1-5 定时器
(1) 内核函数

所谓定时器,就是闹钟,时间到后你就要做某些事。有2个要素:时间、做事;换成程序员的话就是:超时时间、函数。
内核源码:include\linux\timer.h

1、设置定时器,主要是初始化timer_list结构体,设置其中的函数、参数。

setup_timer(timer, fn, data);

2、向内核添加定时器。 timer–>expires 表示超时时间。
当超时时间到达,内核就会调用这个函数:timer->function(timer -->data) 。

void add_timer(struct timer_list *timer)

在这里插入图片描述
3、修改定时器的超时时间

int mod_timer(struct timer_list *timer, unsigned long expires):

4、删除定时器

int del_timer(struct timer_list *timer)
(2) 定时器时间单位

这表示内核每秒中会发生100次系统滴答中断 (tick),这是Linux系统的心跳。每发生一次tick中断,全局变量jiffies累加1。即:每个滴答是10ms。

CONFIG_HZ=100

按键触发中断,进入中断处理函数,若不断发生机械振动,会不断进入中断处理函数更新定时器超时时间,时间到后进入定时器处理函数,打印GPIO端口信息
probe函数设置定时器

static int gpio_key_probe(struct platform_device *pdev)
{
	/* 设置定时器*/
	setup_timer(&gpio_keys_100ask[i].key_timer, key_timer_expire, &gpio_keys_100ask[i]);

	/*设置超时时间*/
	gpio_keys_100ask[i].key_timer.expires = ~0;
	add_timer(&gpio_keys_100ask[i].key_timer);
}

中断处理函数修改定时器超时时间

static irqreturn_t gpio_key_isr(int irq, void *dev_id)
{
	struct gpio_key *gpio_key = dev_id;
	printk("gpio_key_isr %d irq happened\n", gpio_key->gpio);
	mod_timer(&gpio_key->key_timer, jiffies + HZ/50); //20ms  HZ = 1s
	return IRQ_HANDLED;
}

定时器处理函数中打印GPIO信息和唤醒线程

struct timer_list key_timer;
/*定时器处理函数*/
static void key_timer_expire(unsigned long data)
{
	/*data ==> gpio*/
	struct gpio_key *gpio_key = data;
	int val;
	int key;
	val = gpiod_get_value(gpio_key->gpiod);
	
	printk("key_timer_expire %d %d\n", gpio_key->gpio, val);
	key = (gpio_key->gpio << 8) | val;
	put_key(key);
	wake_up_interruptible(&gpio_key_wait); 			/*唤醒线程*/
	kill_fasync(&button_fasync, SIGIO, POLL_IN);	/*发信号*/

}
1-6 中断下半部 tasklet

在这里插入图片描述
在上半部处理紧急的事情时,在处理过程中,中断是被禁止的;在下半部处理耗时的事情时,在处理过程中,中断是使能的。
内核源码:include\linux\interrupt.h

struct tasklet_struct
{
	struct tasklet_struct *next;
	unsigned long state;
	atomic_t count;
	void (*func)(unsigned long);
	unsigned long data;
};

state用于表示 tasklet 的状态,一共有2位。
bit0 表示 TASKLET_STATE_SCHED
等于1时,表示已经执行了 tasklet_schedule把tasklet放入队列;
bit1 表示 TASKLET_STATE_RUN
等于1时,表示正在运行 tasklet 中的func函数;函数执行完后内核会把该位清0。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2023年第十四届蓝桥杯大赛软件类省赛C/C++大学A组真题

2023年第十四届蓝桥杯大赛软件类省赛C/C大学A组部分真题和题解分享 文章目录 蓝桥杯2023年第十四届省赛真题-平方差思路题解 蓝桥杯2023年第十四届省赛真题-更小的数思路题解 蓝桥杯2023年第十四届省赛真题-颜色平衡树思路题解 蓝桥杯2023年第十四届省赛真题-买瓜思路题解 蓝桥…

c/c++ | 静态链接、动态链接

正如标题所见&#xff0c;我们就来讲讲开发时遇到的一些问题&#xff0c;以及解决方案 这里不介绍动态库、静态库的生成与调用&#xff0c; 无论是静态库还是动态库&#xff0c;都是在编译项目的时候链接器会根据编译命令去调用的 如果直接把库&#xff08;动态、静态不论&…

自己本地模拟内存数据库增删改查

目录 学习初衷准备代码实现结果感谢阅读 学习初衷 用于满足自己的测试要求&#xff0c;不连接数据库&#xff0c;也不在意数据丢失 准备 maven依赖 org.springframework.boot spring-boot-starter-test test 代码实现 内存数据库&#xff08;InMemoryDatabase&#xff0…

玩转SpringBoot:动态排除Starter配置,轻松部署

引言 在软件开发中&#xff0c;进行本地单元测试是一项常规且必要的任务。然而&#xff0c;在进行单元测试时&#xff0c;有时需要启动一些中间件服务&#xff0c;如Kafka、Elasticjob等。举例来说&#xff0c;我曾经遇到过一个问题&#xff1a;项目中使用了Redisson锁&#x…

试手一下CameraX(APP)

书接上回。 首先还是看谷歌的官方文档&#xff1a; https://developer.android.com/media/camera/camerax?hlzh-cn https://developer.android.com/codelabs/camerax-getting-started?hlzh-cn#1 注&#xff1a;这里大部分内容也来自谷歌文档。 官方文档用的是Kotlin&…

JavaWeb之 创建 Web项目,使用Tomcat 部署项目,使用 Maven 构建Web项目(一万八千字详解)

目录 前言3.1 Tomcat 简介3.1.1 什么是 Web服务器3.1.2 Tomcat 是什么3.1.3 小结 3.2 Tomcat 的基本使用3.2.1 下载 Tomcat3.2.2 安装 Tomcat3.2.3 卸载 Tomcat3.2.4 启动 Tomcat3.2.5 关闭 Tomcat3.2.6 配置 Tomcat3.2.7 在 Tomcat 中部署 Web项目 3.3 在 IDEA 中创建 Web 项目…

探索前景:机器学习中常见优化算法的比较分析

目录 一、介绍 二、技术背景 三、相关代码 四、结论 一、介绍 优化算法在机器学习和深度学习中至关重要&#xff0c;可以最小化损失函数&#xff0c;从而改善模型的预测。每个优化器都有其独特的方法来导航损失函数的复杂环境以找到最小值。本文探讨了一些最常见的优化算法&…

Python爬虫——解析常用三大方式之JsonPath

目录 JsonPath 安装 使用 我们的json数据 基本使用 案例 总结 JsonPath 主要适用于解析一些json的数据 安装 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ jsonpath 使用 obj json.load(open( json文件 , r , encoding utf-8 ) )ret jsonpath.…

一些C语言知识

C语言的内置类型&#xff1a; char short int long float double C99中引入了bool类型&#xff0c;用来表示真假的变量类型&#xff0c;包含true&#xff0c;false。 这个代码的执行结果是什么&#xff1f;好好想想哦&#xff0c;坑挺多的。 #include <stdio.h>int mai…

EdgeX Foundry 安全模式安装部署

文章目录 一、安装准备1.官方文档2. 克隆服务器3.安装 Docker4.安装 docker-compose 二、安装部署1.docker-comepse2.启动 EdgeX Foundry3.访问 UI3.1. consul3.2. EdgeX Console EdgeX Foundry # EdgeX Foundryhttps://iothub.org.cn/docs/edgex/ https://iothub.org.cn/docs…

机器学习|KNN和Kmeans

KNN和Kmeans KNN KNN-K个最近的邻居&#xff0c;而K是可人先预设出来的。 所谓近朱者赤&#xff0c;近墨者黑。 可以选取离当前最近的K个样本来作为辅助判断&#xff0c;因为本样本和最近的K个样本应该是处于一种相似的状态。 以下是一个苹果和梨的识别任务。 图上会出现一个未…

立式学习灯值得买吗?五款立式学习灯真实测评

现在人们注重健康生活&#xff0c;特别是在面对目前青少年严峻的近视情况&#xff0c;大路灯作为补充光线的照明电器&#xff0c;在市场热度持续高涨&#xff0c;但其负面评价也屡见不鲜。有人反映使用后眼睛更容易疲劳、酸疼等不适症状。作为一名专业测评师&#xff0c;我提醒…

pdf编辑软件哪个好用?5款PDF编辑器分享

pdf编辑软件哪个好用&#xff1f;PDF编辑软件在现代办公和学术研究中发挥着举足轻重的作用&#xff0c;它们不仅具备基础的编辑和修改功能&#xff0c;还能够支持多种注释工具&#xff0c;帮助我们高效地管理和整理PDF文件。无论是需要调整文档布局、添加文本或图像&#xff0c…

Linux系统——LNMP架构

目录 一、LNMP架构定义 1.LNMP定义 1.1LNMP工作原理 2.FASTCGI 2.1CGI的由来 2.2为什么会有FastCGI 3.PHP 3.1什么是PHP-FPM 3.2PHP配置 3.1.1对配置文件的修改生效方法 3.1.2/etc/php.ini配置文件格式 3.1.3注释符&#xff1a; 3.1.4php.ini配置参考文档 3.1.5…

【Linux取经路】文件系统——inode与软硬链接

文章目录 一、前言二、认识硬件——磁盘2.1 磁盘的存储构成2.2 磁盘的逻辑抽象 三、操作系统对磁盘的使用3.1 再来理解创建文件3.2 再来理解删除文件3.3 再来理解目录 四、硬链接五、软链接六、结语 一、前言 在之前的【Linux取经路】文件系统之被打开的文件——文件描述符的引…

自动驾驶加速落地,激光雷达放量可期(上)

1 激光雷达应用广泛&#xff0c;汽车有望成最大催化 激光雷达&#xff08;LiDAR&#xff09;是一种主动遥感技术&#xff0c;通过测定传感器发出的激光在传感器与目标物体之间的传播距离&#xff0c;来分析目标地物表面的反射能量大小、反射波谱的幅度、频率和相位等信息&#…

python基础使用之记录日志模块

我们在编写Python 程序时&#xff0c;记录日志信息是一种非常重要的需求&#xff0c;日志可以帮助调试和跟踪程序的执行过程。那么Python中提供了内置的logging模块&#xff0c;用于记录各种级别的日志信息。本文主要介绍Python日志信息输出的实现过程。 1. 导入 logging 模块…

C++入门全集(4):类与对象【下】

一、再谈构造函数 1.1 构造函数体内赋值 我们知道&#xff0c;在创建对象时&#xff0c;编译器会自动调用构造函数给对象中的各个成员变量一个合适的初始值 class Date { public:Date(int year, int month, int day){_year year;_month month;_day day;}private:int _yea…

开源项目:智能化图像分类技术在新能源发电监控中的应用与实践

一、引言 在当今世界&#xff0c;能源的转型和升级是推动社会可持续发展的关键因素。随着技术的进步&#xff0c;新能源发电逐渐成为能源结构调整的重要力量。在众多发电方式中&#xff0c;新能源发电技术如风力、太阳能等因其清洁、可再生的特性而备受青睐。然而&#xff0c;…

百度文库旋转验证码识别

最近研究了一下图像识别&#xff0c;一直找到很好的应用场景&#xff0c;今天我就发现可以用百度的旋转验证码来做一个实验。没想到效果还挺好&#xff0c;下面就是实际的识别效果。 1、效果演示 2、如何识别 2.1准备数据集 首先需要使用爬虫&#xff0c;对验证码图片进行采…