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

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

 
 

阻塞:

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

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

例如:调用 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 为 非阻塞方式
  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 函数中同样查看:

  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,以阻塞方式运行:

后台执行应用程序,进程处于睡眠状态,按下按键,立马打印按键号;
  1. int main(int argc, char **argv)
  2. {
  3. unsigned char key_val;
  4. int Oflags;
  5. fd = open("/dev/buttons", O_RDWR );
  6. if (fd < 0)
  7. {
  8. printf("can't open!\n");
  9. return -1;
  10. }
  11. while (1)
  12. {
  13. read(fd, &key_val, 1);
  14. printf("key_val: 0x%x\n", key_val);
  15. }
  16. return 0;
  17. }

2,以非阻塞方式运行:

open 驱动程序的时候,传入标志 O_NONBLOCK 非阻塞;
后台执行应用程序:
  1. int main(int argc, char **argv)
  2. {
  3. unsigned char key_val;
  4. int ret;
  5. int Oflags;
  6. fd = open("/dev/buttons", O_RDWR | O_NONBLOCK);
  7. if (fd < 0)
  8. {
  9. printf("can't open!\n");
  10. return -1;
  11. }
  12. while (1)
  13. {
  14. ret = read(fd, &key_val, 1);
  15. printf("key_val: 0x%x, ret = %d\n", key_val, ret);
  16. sleep(5);
  17. }
  18. return 0;
  19. }
 

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

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

  1. 【转】linux设备驱动程序之简单字符设备驱动

    原文网址:http://www.cnblogs.com/geneil/archive/2011/12/03/2272869.html 一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用 ...

  2. Linux 驱动框架---cdev字符设备驱动和misc杂项设备驱动

    字符设备 Linux中设备常见分类是字符设备,块设备.网络设备,其中字符设备也是Linux驱动中最常用的设备类型.因此开发Linux设备驱动肯定是要先学习一下字符设备的抽象的.在内核中使用struct ...

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

    驱动开发   字符设备,块设备,网络设备   字符设备 以字节流的方式访问, 不能随机访问 有例外,显卡.EEPROM可以随机访问   EEPROM可以擦写1亿次,是一种字符设备,可以随机访问 读写是 ...

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

    /*********************************************************************************** * * 字符设备驱动基本操作及 ...

  5. LCD驱动分析(一)字符设备驱动框架分析

    参考:S3C2440 LCD驱动(FrameBuffer)实例开发<一>   S3C2440 LCD驱动(FrameBuffer)实例开发<二> LCD驱动也是字符设备驱动,也 ...

  6. Linux驱动开发2——字符设备驱动

    1.申请设备号 #include <linux/fs.h> int register_chrdev_region(dev_t first, unsigned int count, char ...

  7. 【linux驱动笔记】字符设备驱动相关数据结构与算法

    欢迎转载,转载时需保留作者信息,谢谢. 邮箱:tangzhongp@163.com 博客园地址:http://www.cnblogs.com/embedded-tzp Csdn博客地址:http:// ...

  8. 初入android驱动开发之字符设备(一)

    大学毕业,初入公司,招进去的是android驱动开发工程师的岗位,那时候刚进去,首先学到的就是如何搭建kernel.android的编译环境,然后就是了解如何刷设备以及一些最基本的工具.如adb.fa ...

  9. Linux驱动开发之字符设备模板

    /***************************** ** 驱动程序模板* 版本:V1* 使用方法(末行模式下):* :%s/xxx/"你的驱动名称"/g********* ...

随机推荐

  1. 理解线程3 c语言示例线程基本操作

    Table of Contents 1. 基本线程的动作 1.1. 设置线程属性 1.1.1. 设置脱离状态 1.1.2. 设置调度属性 1.2. 取消线程 1.3. 主线程创建多个线程示例 2. 了 ...

  2. SpringMVC---简单登录例子

    所需jar包aopalliance-1.0.jar.commons-logging-1.2.jar.spring-aop-5.0.0.RELEASE.jar.spring-beans-5.0.0.RE ...

  3. [Python3.x]python3.5实现socket通讯(TCP)

    TCP连接: tcp是面向连接的一个协议,意味着,客户端和服务器开发发送数据之前,需要先握手创建一个TCP连接.TCP连接的一端与客户端套接字相互联系,另一端与服务器套接字相联系.当创建该TCP连接的 ...

  4. stm32--free modbus 1.5.0移植(作为从机)

    添加文件 获取原始free modbus library(官网) 将...\freemodbus-v1.5.0\demo\BARE中的所有文件复制到...\freemodbus-v1.5.0\modb ...

  5. ElasticSearch学习笔记(二)-- mapping

    1. mapping简介 2. 自定义 mapping 3. mapping演示 创建索引,设置mapping,可以新增字段 get一下mapping 设置索引的字段不可新增 为索引添加字段,发现报错 ...

  6. 为什么rows这么大,在mysql explain中---写在去acumg听讲座的前一夜

    这周五下班前,发现了一个奇怪问题,大概是这个背景 一张表,结构为 Create Table: CREATE TABLE `out_table` ( `id` ) NOT NULL AUTO_INCRE ...

  7. Ajax异步与JavaScript的一些初浅认识

    向服务器请求数据的技术 有以下五种常用技术用于向服务器请求数据 XMLHttpRequest(XHR) Dynamic script tag insertion(动态脚本标签插入) iframes C ...

  8. CodeForces Round #515 Div.3 D. Boxes Packing

    http://codeforces.com/contest/1066/problem/D Maksim has nn objects and mm boxes, each box has size e ...

  9. Centos安装后的一些必要处理工作

    1永久关闭selinux,修改成permissive或者disabled(建议),修改完需重启 2配置network 3.禁止ping(可选,一般不需要禁止)(默认为0位启用ICMP协议,1为禁止), ...

  10. Linq 聚合函数

    var WReserve = (from m in _db.W_RESERVE select m).ToList().LastOrDefault(); 必须ToList() 之后才能 last fir ...