Linux驱动开发 | 异步通信+中断

本文深入探讨Linux系统中异步IO的实现机制,通过对比同步IO,解析异步IO如何使应用程序在底层数据准备就绪时自动处理数据。文章详细介绍了在应用层通过信号处理函数、设置SIGIO属主进程及IO模式为异步模式的方法,并提供了具体的代码示例。同时,展示了驱动层如何通过fasync函数和发送SIGIO信号实现异步通信。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

对于linux一切都是文件,驱动设备在应用层也是以文件的形式进行读写。之前学了阻塞、非阻塞、多路复用的方式读设备,它们都需要应用主动读取。那么应用层有没有一种方式,当底层将数据准备好了,应用程序自动处理这些数据?通过异步通信可以实现,这有写类似硬件层的中断概念

驱动层(准备好了数据) --> 发送特定信号 --> 应用程序(跳转到与之匹配的函数入口) --> 处理数据.....

异步通信:区别同步通信,当一个异步过程调用发出后,调用者不能立刻得到结果。实际处理这个调用的部件在完成后,通过状态、通知和回调来通知调用者。

参考一篇文章:简述同步IO和异步IO的区别

应用层实现

 signal将SIGIO信号和处理函数catch_signale关联;将进程设置成SIGIO属主进程,让其可以接受底层驱动的SIGIO信号;flag或上这个FASYNC宏,作用是调用FASYNC函数。

// 1,设置信号处理方法
signal(SIGIO,catch_signale);

// 2,将当前进程设置成SIGIO的属主进程
fcntl(fd, F_SETOWN, getpid());

// 3,将io模式设置成异步模式
int flags  = fcntl(fd, F_GETFL);
fcntl(fd, F_SETFL, flags | FASYNC );

//key_test.c
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <poll.h>
#include <signal.h>

#define KEY_ENTER		28



struct key_event{
	int code; // 按键的类型
	int value; // 状态
};

struct key_event event; //接收按键设备数据



static int fd;


void catch_signale(int signo)
{
	if(signo == SIGIO)
	{
		printf("we got sigal SIGIO\n");
		// 读取数据
		read(fd, &event, sizeof(struct key_event));
		if(event.code == KEY_ENTER)
		{
			if(event.value)
			{
				printf("APP__ key enter pressed\n");
			}else
			{
				printf("APP__ key enter up\n");
			}
		}
	}

}


int main(int argc, char *argv[])
{
	fd = open("/dev/key0", O_RDWR);
	if(fd < 0)
	{
		perror("open");
		exit(1);
	}
	
	// 1,设置信号处理方法
	signal(SIGIO,catch_signale);
	
	// 2,将当前进程设置成SIGIO的属主进程
	fcntl(fd, F_SETOWN, getpid());

	// 3,将io模式设置成异步模式
	int flags  = fcntl(fd, F_GETFL);
	fcntl(fd, F_SETFL, flags | FASYNC );


	while(1)
	{
		// 可以做其他的事情
		printf("I am waiting......\n");
		sleep(1);

	}



	close(fd);

	return 0;

}

驱动层

1.在file_operations中 fasync指向key_drv_fasync函数

const struct file_operations key_fops = {
	.open = key_drv_open,
	.read = key_drv_read,
	.write = key_drv_write,
	.release = key_drv_close,
	.poll = key_drv_poll,
	.fasync = key_drv_fasync,

};

2.实现接口

int key_drv_fasync(int fd, struct file *filp, int on)
{
	//只需要调用一个函数记录信号该发送给谁
	return fasync_helper(fd, filp, on,  &key_dev->faysnc);

}

3.在适当的情况(有数据),发送信号

kill_fasync(&key_dev->faysnc, SIGIO, POLLIN);

结果展示

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hinzer

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值