Linux 设备驱动--- 阻塞型字符设备驱动 --- O_NONBLOCK --- 非阻塞标志【转】

转自:http://blog.csdn.net/yikai2009/article/details/8653697

版权声明:本文为博主原创文章,未经博主允许不得转载。

 

目录(?)[-]

  1. 阻塞
    1. 阻塞操作
    2. 非阻塞操作
  2. 阻塞方式-read- 实现
  3. 阻塞方式-write- 实现
  4. 非阻塞方式的读写操作
  5. 实例 --- 读阻塞的实现
  6. 实例 --- 按键驱动阻塞实现
    1. 1在 open 函数 查看看是 阻塞方式 还是 非阻塞方式
    2. 2在 read 函数中同样查看
    3. 3应用程序中
      1. 1以阻塞方式运行
      2. 2以非阻塞方式运行

 

阻塞:

          在设计简单字符驱动程序时,要注意一个重要问题.

          当一个设备无法立刻满足用户的读写请求时应当如何处理?

          例如:调用 read 时没有数据可读,但以后可能会有;

          或者一个进程试图向设备写入数据,但是设备暂时没有准备好接收数据.

          应用程序通常不关心这种问题,应用程序只是调用 read 或 write 并得到返回值.

          驱动程序应当 ( 缺省地 ) 阻塞进程使它进入睡眠直到请求可以得到满足.

阻塞操作:

          是指在执行设备操作时,若不能获得资源则挂起进程,直到满足可操作的条件后进行操作,

          被挂起的进程进入睡眠状态,被从调度器的运行队列移走,直到等待的条件被满足.

非阻塞操作:

          进程不能进行设备操作时并不挂起,他或者放弃,或者不停的查询,直到可以进行操作为止.

 

阻塞方式-read- 实现:

          在阻塞型驱动程序中,read 实现方式如下:

          如果进程调用 read ,但设备 没有数据 或 数据不足,进程阻塞.

          当新数据到达后,唤醒被阻塞进程.

 

阻塞方式-write- 实现:

          在阻塞型驱动程序中,write 实现方式如下:

          如果进程调用了 write ,但设备 没有足够的空间供其写入数据,进程阻塞.

          当设备中的数据被读走后,缓冲区中空出部分空间,则唤醒进程.

 

非阻塞方式的读写操作:

          阻塞方式是文件读写操作的默认方式,但是应用程序员可通过使用O_NONBLOCK 标志来人为

          的设置读写操作为非阻塞方式 .( 该标志定义在 < linux/fcntl.h > 中,在打开文件时指定 ) .

 

          如果设置了 O_NONBLOCK 标志,read 和 write 的行为是不同的 ,如果进程没有数据就绪时调用了 read ,

          或者在缓冲区没有空间时调用了 write ,系统只是简单的返回 -EAGAIN,而不会阻塞进程.

 

实例 --- 读阻塞的实现:

          

          

 

用 while 是因为可能别的信号唤醒了睡眠,我们要通过while 重新检测是否真有数据了....

 

 

          

 

 

实例 --- 按键驱动阻塞实现:

1,在 open 函数 查看看是 阻塞方式 还是 非阻塞方式:

         file 结构体中含有 f_flags 标志位,看是 阻塞方式 还是 非阻塞方式:

         O_NONBLOCK 为 非阻塞方式

[cpp] view plain copy

 

  1. if (file->f_flags & O_NONBLOCK)  /* 非 阻塞操作 */  
  2. {  
  3.     if (down_trylock(&button_lock))   /* 无法获取信号量,down_trylock 立马返回 一个 非零值 */  
  4.         return -EBUSY;  
  5. }  
  6. else                             /* 阻塞操作 */  
  7. {  
  8.     /* 获取信号量 */  
  9.     down(&button_lock);   /* 获取不到  睡眠 */  
  10. }  

2,在 read 函数中同样查看:

[cpp] view plain copy

 

  1. if (file->f_flags & O_NONBLOCK)       /* 非 阻塞操作 */  
  2. {  
  3.     if (!ev_press)                 /* ev_press 为 1 表示有按键按下,为 0 if 成立 ,没有按键按下, */  
  4.         return -EAGAIN;        /* 返回 -EAGAIN 让再次来执行 */  
  5. }  
  6. else                                   /* 阻塞操作 */  
  7. {  
  8.     /* 如果没有按键动作, 休眠 */  
  9.     wait_event_interruptible(button_waitq, ev_press);  
  10. }  

3,应用程序中:

1,以阻塞方式运行:

后台执行应用程序,进程处于睡眠状态,按下按键,立马打印按键号;

[cpp] view plain copy

 

  1. int main(int argc, char **argv)  
  2. {  
  3.     unsigned char key_val;  
  4.     int Oflags;  
  5.                                                      
  6.     fd = open("/dev/buttons", O_RDWR );  
  7.     if (fd < 0)  
  8.     {  
  9.         printf("can't open!\n");  
  10.         return -1;  
  11.     }  
  12.   
  13.     while (1)  
  14.     {  
  15.         read(fd, &key_val, 1);  
  16.         printf("key_val: 0x%x\n", key_val);  
  17.     }  
  18.       
  19.     return 0;  
  20. }  

2,以非阻塞方式运行:

open 驱动程序的时候,传入标志 O_NONBLOCK 非阻塞;

后台执行应用程序:

[cpp] view plain copy

 

  1. int main(int argc, char **argv)  
  2. {  
  3.     unsigned char key_val;  
  4.     int ret;  
  5.     int Oflags;  
  6.   
  7.     fd = open("/dev/buttons", O_RDWR | O_NONBLOCK);  
  8.     if (fd < 0)  
  9.     {  
  10.         printf("can't open!\n");  
  11.         return -1;  
  12.     }  
  13.   
  14.     while (1)  
  15.     {  
  16.         ret = read(fd, &key_val, 1);  
  17.         printf("key_val: 0x%x, ret = %d\n", key_val, ret);  
  18.         sleep(5);  
  19.     }  
  20.       
  21.     return 0;  
  22. }  

 

非阻塞方式,没有按键值按下,程序立马返回;
read 返回值 为 -1;

时间: 2024-10-10 16:27:43

Linux 设备驱动--- 阻塞型字符设备驱动 --- O_NONBLOCK --- 非阻塞标志【转】的相关文章

驱动开发--【字符设备、块设备简介】【sky原创】

驱动开发   字符设备,块设备,网络设备   字符设备 以字节流的方式访问, 不能随机访问 有例外,显卡.EEPROM可以随机访问   EEPROM可以擦写1亿次,是一种字符设备,可以随机访问 读写是直接访问硬件的 flash 擦写次数有限,一百万次,容易有坏块   块设备 能随机访问 以"块"为单位进行访问 块大小一般为512字节 块的大小由硬件决定 是内核进行数据传输的基本单位     硬盘结构: 格式化分区是以柱面为单位的,即硬盘的柱面 如果有10个盘面,就有十个柱面   对于嵌

Samsung_tiny4412(驱动笔记03)----字符设备驱动基本操作及调用流程

/*********************************************************************************** * * 字符设备驱动基本操作及调用流程 * * 声明: * 1. 本系列文档是在vim下编辑,请尽量是用vim来阅读,在其它编辑器下可能会 * 不对齐,从而影响阅读. * 2. 以下所有的shell命令都是在root权限下运行的; * * 2015-3-7 阴 深圳 尚观 Sbin 曾剑锋 *******************

异步 非阻塞-求帮忙设计一个异步非阻塞服务器的程序

问题描述 求帮忙设计一个异步非阻塞服务器的程序 伪代码也可以,尽量使用linux下的函数.我不太明白,select到底实现的是异步还是同步,有的地方说异步,有的地方说同步,还有一种看似权威的说法是实质是同步,是一种伪异步而已.如果select算做异步的话,那么非阻塞又要如何实现呢?求大神!

【Linux驱动】字符设备驱动

一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 1.字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流的设备,常见的字符设备有鼠标.键盘.串口.控制台和LED设备等. 2.块设备:是指可以从设备的任意位置读取一定长度数据的设备.块设备包括硬盘.磁盘.U盘和SD卡等. 每一个字符设备或块设备都在/dev目录下对应一个设备文件.linux用户程序通过设备文件(或称设备节点)来使用驱动程序操作字符设备

Linux字符设备驱动编写基本流程

  ---简介 Linux下的MISC简单字符设备驱动虽然使用简单,但却不灵活. 只能建立主设备号为10的设备文件.字符设备比较容易理解,同时也能够满足大多数简单的硬件设备,字符设备通过文件系          统中的名字来读取.这些名字就是文件系统中的特殊文件或者称为设备文件.文件系统的简单结点,一般位于/dev/目录下          使用ls进行查看会显示以C开头证明这是字符设备文件crw--w---- 1 root tty 4, 0 4月 14 11:05 tty0.第一个数字是主设备

Linux字符设备驱动框架

字符设备是Linux三大设备之一(另外两种是块设备,网络设备),字符设备就是字节流形式通讯的I/O设备,绝大部分设备都是字符设备,常见的字符设备包括鼠标.键盘.显示器.串口等等,当我们执行ls -l /dev的时候,就能看到大量的设备文件,c就是字符设备,b就是块设备,网络设备没有对应的设备文件.编写一个外部模块的字符设备驱动,除了要实现编写一个模块所需要的代码之外,还需要编写作为一个字符设备的代码. 驱动模型 Linux一切皆文件,那么作为一个设备文件,它的操作方法接口封装在struct fi

Linux 字符设备驱动框架详细介绍_Linux

Linux 字符设备驱动框架 字符设备是Linux三大设备之一(另外两种是块设备,网络设备),字符设备就是字节流形式通讯的I/O设备,绝大部分设备都是字符设备,常见的字符设备包括鼠标.键盘.显示器.串口等等,当我们执行ls -l /dev的时候,就能看到大量的设备文件,c就是字符设备,b就是块设备,网络设备没有对应的设备文件.编写一个外部模块的字符设备驱动,除了要实现编写一个模块所需要的代码之外,还需要编写作为一个字符设备的代码. 驱动模型 Linux一切皆文件,那么作为一个设备文件,它的操作方

Linux字符设备驱动编写流程

驱动程序编写基本流程: 1.首先是一些版本信息,没什么用,但是不能少 #define __NO_VERSION__ #include <linux/modules.h> #include <linux/version.h> char kernel_version[] = UTS_RELEASE; 2.为了把系统调用和驱动程序关联起来,需要一个非常关键的数据结构:struct file_operations.file_operations结构的每一个成员的名字都对应着一个系统调用.用

浅谈Linux 中字符设备的注册

Linux中字符设备的注册过程是比较简单的.我们通常可以调用misc_register()函数来注册一个字符设备.Misc设备是一种字符设备,通过该设备可以将fops请求转发给注册的misc设备,从而实现字符设备的功能.用户调用该接口注册Misc字符设备时,可以动态分配设备Minor号,当获取Minor号之后调用class_simple_device_add()或者device_create()函数完成字符设备的创建.Misc字符设备注册函数如下所示: int misc_register(str