linux设备驱动之阻塞与非阻塞I/O

2024-06-16 02:08

本文主要是介绍linux设备驱动之阻塞与非阻塞I/O,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

先做一下与内核阻塞有关的知识储备:

1)进程休眠:

    进程休眠,简单的说就是正在运行的进程让出CPU。休眠的进程会被内核搁置在在一边,只有当内核再次把休眠的进程唤醒,进程才会会重新在CPU运行。这是内核中的进程调度。一个CPU在同一时间只能有一个进程在运行,微观串行宏观并行,在宏观上,我们觉得是所有进程同时进行的。实际上并不是这样,内核给每个进程分配了4G的虚拟内存,并且让每个进程傻乎乎的以为自己霸占着CPU运行。同时,内核暗中的将所有的进程按一定的算法将CPU轮流的给每个进程使用,而休眠就是进程没有被运行时的一种形式。在休眠下,进程不占用CPU,等待被唤醒

2)等待队列

   等待队列是一个存放着等待某个特定事件进程链表用于存放等待唤醒的进程,等待队列结构   

  1.先看一下队列头的样子:

 /*linux/wait.h*/

 struct __wait_queue_head {

                 spinlock_t lock; //这个是自旋锁,在这里不需要理会。

                 struct list_head task_list; //这就是队列头中的核心,链表头。

 };

 typedef struct __wait_queue_head wait_queue_head_t; 

 2.定义并初始化一个链表,在这个链表添加需要等待的进程    

  1)静态定义并初始化,一个函数执行完两个操作

  DECLARE_WAIT_QUEUE_HEAD(name)

  使用:定义并初始化一个叫name的等待队列。

  2)分开两步执行。

  2.1)定义

  wait_queue_head_t test_queue;

  2.2)初始化

  init_waitqueue_head(&test_queue);

初始化函数的位置,它必须在cdev添加函数”cdev_add”。因为”cdev_add”执行成功就意味着设备可以被操作,设备被操作前当然需要把所有的事情都干完,包括等待队列的初始化。

3)进程休眠

 

  唤醒休眠进程

  void wake_up_interruptible(wait_queue_head_t *queue); //唤醒等待队列中所有可中断睡眠的进程

知识点已经介绍完,总结一下上面驱动函数的操作:

1)首先需要定义并初始化一个等待队列。

2test_read函数中,如果条件不符合,调用该函数的进程就会进入休眠。

3)每当另一个进程调用test_write函数唤醒等待队列,test_read中的函数就会再一次判断条件是否符合,如果不符合,就会继续休眠,直到哪次的唤醒时条件符合。

非阻塞实现--只需要加上判定条件

     if(filp->f_flags&O_NONBLOCK)

(下面函数实现了阻塞与非阻塞)

  阻塞

  
#include <linux/module.h>
#include <linux/init.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/mm.h>
#include <linux/uaccess.h>
#include <linux/types.h>
#include <linux/errno.h>
#include <asm/system.h>
#include <asm/io.h>
#include <linux/sched.h>
#include <linux/wait.h>
#include <linux/sched.h>
#include <linux/fcntl.h>MODULE_LICENSE("Dual BSD/GPL");#define GLOBALMEM_SIZE 0x1000
#define MEM_CLEAR 0x1
//#define GLOBALMEM_MAJOR 250static int globalmem_major=0;
static const struct file_operations globalmem_fops;
struct globalmem_dev{struct cdev cdev;unsigned char mem[GLOBALMEM_SIZE];unsigned int cur_size;wait_queue_head_t test_queue;
};static struct globalmem_dev dev;static ssize_t globalmem_read(struct file *filp,const char __user *buf,size_t count,loff_t *ppos)
{unsigned long p=*ppos;int ret=0;if(!wait_event_interruptible(dev.test_queue,dev.cur_size)){if(p>=GLOBALMEM_SIZE-p)return 0;if(count>GLOBALMEM_SIZE-p)count=GLOBALMEM_SIZE-p;if(copy_to_user(buf,(void*)(dev.mem+p),count))return -EFAULT;else{*ppos+=count;ret=count;dev.cur_size-=ret;printk(KERN_INFO "read %d bytes(s) from %d\n",\count,p);return count;}	}elsereturn -ERESTARTSYS;}
}static ssize_t globalmem_write(struct file *filp,const char __user *buf,size_t count,loff_t *ppos)
{unsigned long p=*ppos;int ret=0;if(p>=GLOBALMEM_SIZE-p)return 0;if(count>GLOBALMEM_SIZE-p)count=GLOBALMEM_SIZE-p;if(copy_from_user(dev.mem+p,buf,count))ret=-EFAULT;else{*ppos+=count;ret=count;printk(KERN_INFO "written %d bytes(s) from %d\n",count,p);}dev.cur_size+=ret;wake_up_interruptible(&dev.test_queue);return ret;	
}
static void globalmem_setup_cdev()
{int err;dev_t devno=MKDEV(globalmem_major,0);cdev_init(&dev.cdev,&globalmem_fops);dev.cdev.owner=THIS_MODULE;init_waitqueue_head(&dev.test_queue);err=cdev_add(&dev.cdev,devno,1);if(err){printk(KERN_NOTICE "Error %d adding globalmem",err);}
}static const struct file_operations globalmem_fops={.owner=THIS_MODULE,.write=globalmem_write,.read=globalmem_read,
};int globalmem_init(void)
{int result;dev_t devno=MKDEV(globalmem_major,0);if(globalmem_major){result=register_chrdev_region(devno,1,"my_globalmem");}else{result=alloc_chrdev_region(&devno,0,1,"my_globalmem");globalmem_major=MAJOR(devno);}if(result<0){return result;}globalmem_setup_cdev();return 0;
}void globalmem_exit(void)
{cdev_del(&dev.cdev);unregister_chrdev_region(MKDEV(globalmem_major,0),1);printk("leavel kernel\n");return;
}module_init(globalmem_init);
module_exit(globalmem_exit);
MODULE_AUTHOR("nw");


 

 

非阻塞

 

 

 

 

#include <linux/module.h>
#include <linux/init.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/mm.h>
#include <linux/uaccess.h>
#include <linux/types.h>
#include <linux/errno.h>
#include <asm/system.h>
#include <asm/io.h>
#include <linux/sched.h>
#include <linux/wait.h>
#include <linux/sched.h>
#include <linux/fcntl.h>MODULE_LICENSE("Dual BSD/GPL");#define GLOBALMEM_SIZE 0x1000
#define MEM_CLEAR 0x1
//#define GLOBALMEM_MAJOR 250static int globalmem_major=0;
static const struct file_operations globalmem_fops;
struct globalmem_dev{struct cdev cdev;unsigned char mem[GLOBALMEM_SIZE];unsigned int cur_size;wait_queue_head_t test_queue;
};static struct globalmem_dev dev;static ssize_t globalmem_read(struct file *filp,const char __user *buf,size_t count,loff_t *ppos)
{unsigned long p=*ppos;int ret=0;if(filp->f_flags&O_NONBLOCK){if(dev.cur_size>0){if(!wait_event_interruptible(dev.test_queue,dev.cur_size)){if(p>=GLOBALMEM_SIZE-p)return 0;if(count>GLOBALMEM_SIZE-p)count=GLOBALMEM_SIZE-p;if(copy_to_user(buf,(void*)(dev.mem+p),count))return -EFAULT;else{*ppos+=count;ret=count;dev.cur_size-=ret;printk(KERN_INFO "read %d bytes(s) from %d\n",\count,p);return count;}	}elsereturn -ERESTARTSYS;}elsereturn -EAGAIN;}
}static ssize_t globalmem_write(struct file *filp,const char __user *buf,size_t count,loff_t *ppos)
{unsigned long p=*ppos;int ret=0;if(p>=GLOBALMEM_SIZE-p)return 0;if(count>GLOBALMEM_SIZE-p)count=GLOBALMEM_SIZE-p;if(copy_from_user(dev.mem+p,buf,count))ret=-EFAULT;else{*ppos+=count;ret=count;printk(KERN_INFO "written %d bytes(s) from %d\n",count,p);}dev.cur_size+=ret;wake_up_interruptible(&dev.test_queue);return ret;	
}
static void globalmem_setup_cdev()
{int err;dev_t devno=MKDEV(globalmem_major,0);cdev_init(&dev.cdev,&globalmem_fops);dev.cdev.owner=THIS_MODULE;init_waitqueue_head(&dev.test_queue);err=cdev_add(&dev.cdev,devno,1);if(err){printk(KERN_NOTICE "Error %d adding globalmem",err);}
}static const struct file_operations globalmem_fops={.owner=THIS_MODULE,.write=globalmem_write,.read=globalmem_read,
};int globalmem_init(void)
{int result;dev_t devno=MKDEV(globalmem_major,0);if(globalmem_major){result=register_chrdev_region(devno,1,"my_globalmem");}else{result=alloc_chrdev_region(&devno,0,1,"my_globalmem");globalmem_major=MAJOR(devno);}if(result<0){return result;}globalmem_setup_cdev();return 0;
}void globalmem_exit(void)
{cdev_del(&dev.cdev);unregister_chrdev_region(MKDEV(globalmem_major,0),1);printk("leavel kernel\n");return;
}module_init(globalmem_init);
module_exit(globalmem_exit);
MODULE_AUTHOR("nw");

调试app函数略


这篇关于linux设备驱动之阻塞与非阻塞I/O的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1065215

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

Linux 安装、配置Tomcat 的HTTPS

Linux 安装 、配置Tomcat的HTTPS 安装Tomcat 这里选择的是 tomcat 10.X ,需要Java 11及更高版本 Binary Distributions ->Core->选择 tar.gz包 下载、上传到内网服务器 /opt 目录tar -xzf 解压将解压的根目录改名为 tomat-10 并移动到 /opt 下, 形成个人习惯的路径 /opt/tomcat-10

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

【Linux进阶】UNIX体系结构分解——操作系统,内核,shell

1.什么是操作系统? 从严格意义上说,可将操作系统定义为一种软件,它控制计算机硬件资源,提供程序运行环境。我们通常将这种软件称为内核(kerel),因为它相对较小,而且位于环境的核心。  从广义上说,操作系统包括了内核和一些其他软件,这些软件使得计算机能够发挥作用,并使计算机具有自己的特生。这里所说的其他软件包括系统实用程序(system utility)、应用程序、shell以及公用函数库等

Windows/macOS/Linux 安装 Redis 和 Redis Desktop Manager 可视化工具

本文所有安装都在macOS High Sierra 10.13.4进行,Windows安装相对容易些,Linux安装与macOS类似,文中会做区分讲解 1. Redis安装 1.下载Redis https://redis.io/download 把下载的源码更名为redis-4.0.9-source,我喜欢跟maven、Tomcat放在一起,就放到/Users/zhan/Documents

Linux系统稳定性的奥秘:探究其背后的机制与哲学

在计算机操作系统的世界里,Linux以其卓越的稳定性和可靠性著称,成为服务器、嵌入式系统乃至个人电脑用户的首选。那么,是什么造就了Linux如此之高的稳定性呢?本文将深入解析Linux系统稳定性的几个关键因素,揭示其背后的技术哲学与实践。 1. 开源协作的力量Linux是一个开源项目,意味着任何人都可以查看、修改和贡献其源代码。这种开放性吸引了全球成千上万的开发者参与到内核的维护与优化中,形成了

Linux 下的Vim命令宝贝

vim 命令详解(转自:https://www.cnblogs.com/usergaojie/p/4583796.html) vi: Visual Interface 可视化接口 vim: VI iMproved VI增强版 全屏编辑器,模式化编辑器 vim模式: 编辑模式(命令模式)输入模式末行模式 模式转换: 编辑-->输入: i: 在当前光标所在字符的前面,转为输入模式

Linux和Mac分卷压缩

使用 zip 命令压缩文件 使用 zip 命令压缩文件,并结合 split 命令来分卷: zip - largefile | split -b 500k 举例: zip - ./tomcat.dmg |split -b 500k 上述命令将文件 largefile 压缩成 zip 包并分卷成不超过 500k 的文件,分解后文件名默认是 x* ,后缀为 2 位a-z 字母,如 aa、ab。

Linux文本三剑客sed

sed和awk grep就是查找文本当中的内容,最强大的功能就是使用扩展正则表达式 sed sed是一种流编辑器,一次处理一行内容。 如果只是展示,会放在缓冲区(模式空间),展示结束后,会从模式空间把结果删除 一行行处理,处理完当前行,才会处理下一行。直到文件的末尾。 sed的命令格式和操作选项: sed -e '操作符 ' -e '操作符' 文件1 文件2 -e表示可以跟多个操作

WDF驱动开发-WDF总线枚举(一)

支持在总线驱动程序中进行 PnP 和电源管理 某些设备永久插入系统,而其他设备可以在系统运行时插入和拔出电源。 总线驱动 必须识别并报告连接到其总线的设备,并且他们必须发现并报告系统中设备的到达和离开情况。 总线驱动程序标识和报告的设备称为总线的 子设备。 标识和报告子设备的过程称为 总线枚举。 在总线枚举期间,总线驱动程序会为其子 设备创建设备对象 。  总线驱动程序本质上是同时处理总线枚