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系统: Debian操作系统常用命令指南

《高效管理你的Linux系统:Debian操作系统常用命令指南》在Debian操作系统中,了解和掌握常用命令对于提高工作效率和系统管理至关重要,本文将详细介绍Debian的常用命令,帮助读者更好地使... Debian是一个流行的linux发行版,它以其稳定性、强大的软件包管理和丰富的社区资源而闻名。在使用

Linux Mint Xia 22.1重磅发布: 重要更新一览

《LinuxMintXia22.1重磅发布:重要更新一览》Beta版LinuxMint“Xia”22.1发布,新版本基于Ubuntu24.04,内核版本为Linux6.8,这... linux Mint 22.1「Xia」正式发布啦!这次更新带来了诸多优化和改进,进一步巩固了 Mint 在 Linux 桌面

LinuxMint怎么安装? Linux Mint22下载安装图文教程

《LinuxMint怎么安装?LinuxMint22下载安装图文教程》LinuxMint22发布以后,有很多新功能,很多朋友想要下载并安装,该怎么操作呢?下面我们就来看看详细安装指南... linux Mint 是一款基于 Ubuntu 的流行发行版,凭借其现代、精致、易于使用的特性,深受小伙伴们所喜爱。对

什么是 Linux Mint? 适合初学者体验的桌面操作系统

《什么是LinuxMint?适合初学者体验的桌面操作系统》今天带你全面了解LinuxMint,包括它的历史、功能、版本以及独特亮点,话不多说,马上开始吧... linux Mint 是一款基于 Ubuntu 和 Debian 的知名发行版,它的用户体验非常友好,深受广大 Linux 爱好者和日常用户的青睐,

Linux(Centos7)安装Mysql/Redis/MinIO方式

《Linux(Centos7)安装Mysql/Redis/MinIO方式》文章总结:介绍了如何安装MySQL和Redis,以及如何配置它们为开机自启,还详细讲解了如何安装MinIO,包括配置Syste... 目录安装mysql安装Redis安装MinIO总结安装Mysql安装Redis搜索Red

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

Linux磁盘分区、格式化和挂载方式

《Linux磁盘分区、格式化和挂载方式》本文详细介绍了Linux系统中磁盘分区、格式化和挂载的基本操作步骤和命令,包括MBR和GPT分区表的区别、fdisk和gdisk命令的使用、常见的文件系统格式以... 目录一、磁盘分区表分类二、fdisk命令创建分区1、交互式的命令2、分区主分区3、创建扩展分区,然后

Linux中chmod权限设置方式

《Linux中chmod权限设置方式》本文介绍了Linux系统中文件和目录权限的设置方法,包括chmod、chown和chgrp命令的使用,以及权限模式和符号模式的详细说明,通过这些命令,用户可以灵活... 目录设置基本权限命令:chmod1、权限介绍2、chmod命令常见用法和示例3、文件权限详解4、ch

Linux内核之内核裁剪详解

《Linux内核之内核裁剪详解》Linux内核裁剪是通过移除不必要的功能和模块,调整配置参数来优化内核,以满足特定需求,裁剪的方法包括使用配置选项、模块化设计和优化配置参数,图形裁剪工具如makeme... 目录简介一、 裁剪的原因二、裁剪的方法三、图形裁剪工具四、操作说明五、make menuconfig

Linux使用nohup命令在后台运行脚本

《Linux使用nohup命令在后台运行脚本》在Linux或类Unix系统中,后台运行脚本是一项非常实用的技能,尤其适用于需要长时间运行的任务或服务,本文我们来看看如何使用nohup命令在后台... 目录nohup 命令简介基本用法输出重定向& 符号的作用后台进程的特点注意事项实际应用场景长时间运行的任务服