四种IO模型在底层驱动的实现方法

姓名:李浩然

学号:16030410020

转自:http://blog.csdn.net/Dreaming_My_Dreams/article/details/8272877(有删改)

【嵌牛导读】:为了区分IO的五种模型,下面先来看看同步与异步、阻塞与非阻塞的概念差别。

同步:所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回。按照这个定义,其实绝大多数函数都是同步调用(例如sin, isdigit等)。但是一般而言,我们在说同步、异步的时候,特指那些需要其他部件协作或者需要一定时间完成的任务。最常见的例子就是 SendMessage。该函数发送一个消息给某个窗口,在对方处理完消息之前,这个函数不返回。当对方处理完毕以后,该函数才把消息处理函数所返回的 LRESULT值返回给调用者。

异步:异步的概念和同步相对。当一个异步功能调用发出后,调用者不能立刻得到结果。当该异步功能完成后,通过状态、通知或回调来通知调用者。

【嵌牛鼻子】:非阻塞IO、阻塞IO、轮询、异步通知、同步、异步

【嵌牛提问】:4种IO模型是什么?都有哪四种IO模型?这些模型各有什么特点?在底层是怎么实现的。

【嵌牛正文】:一、非阻塞IO

当设备没有要操作的资源时,它会立即返回。这种操作,一般要在应用层循环检测,比较占用cpu资源。

二、阻塞IO

这种实现方法需要借助于等待队列

对待buf有两种队列,读和写,可以在设备的私有资源中定义两个队列:wait_queue_head_t    rq,wq;当读写时,没有相应的资源,并且是以阻塞方式读写设备的,那么,就可以把当前任务(应用层的进程,这些任务都有一个task_struct来表示)放到等待队列中,等待有资源或中断时唤醒。

睡眠方式有两种:手动睡眠、自动睡眠

手动睡眠:

自动睡眠:wait_event_interruptible(rq/wq,condition)

这种方式是可以被中断唤醒的睡眠方式,不管何种唤醒,醒来首先会检查condition是否满足,若不满足会继续睡眠,若condition满足,那么,就会再次去申请资源(因为同时唤醒这个等待队列的所有任务,会产生竞态),没有申请到,会继续睡眠,申请到,就会去完成读写。

三、轮询(这种方法的select和poll之间的机制还有点迷糊,稍后补上)

轮询,需要借助于底层驱动的poll函数和等待队列来实现

在应用层中,我们把需要监控的fd放入fd_set中,然后调用select函数,最终就会调用到驱动中的poll 函数。

这些系统调用功能相同:允许进程来决定它是否可读或写一个或多 个文件而不阻塞。这些调用也可阻塞进程直到任何一个给定集合的文件描述符可用来读或写。(也就是说,在应用程序中的select中会一直阻塞(还是睡眠),直至至少有一个设备可以操作,所以说,在应用层次中,其实不是轮询执行select函数,这点是以前的误解)

四、异步通知

在设备驱动中使用异步通知可以使得对设备的访问 可进行时,由驱动主动通知应用程序进行访问,这样,使用无阻塞IO的应用程序无须轮询设备是否可访问,而阻塞访问 也可被类似“中断”的异步通知所取代。

概念:一旦设备就绪,则主动通知应用程序,这样应用程序根本就不需要查询设备状态,这一点非常类似硬件上 “中断”的概念,可称之为“信号驱使的异步IO”。

应用程序:

(1)首先应用程序指定进程问文件的属主,就是把pid保存在file中

fcntl(fd,F_SETOWN,getpid());

(2) 要在设备中设置FASYNC标志

oflag = fcntl(fd,F_GETFL)

fcntl(fd,F_SETFL,oflag | FASYNC)

设置完上面一句话后,应用程序会自动调用驱动中的fasync函数,

驱动中:

(1) 在设备结构体中定义一个异步结构体指针:struct fasync_struct  * async_queue;

(2)实现fasync函数,调用fasync_helper函数,把file放入或删除异步通知队列中

(3)当设备资源就绪时,发送信号,可以发一次,也可以一直发,直至资源被消耗;这完全取决于驱动怎么写

kill_fsaync(&dev->async_queue,SIGIO,POLL_IN)

(4)当进程关闭设备时,要在release函数中把进程从异步通知队列中删除.

[cpp]view plaincopy

structkey_device{

structcdev cdev;

structfasync_struct *async_queue;//定义一个队列

}key_device;

intkey_fasync(intfd,structfile *file,intmode)

{

structkey_device *device = file->private_data;

returnfasync_helper(fd,file,mode,&device->async_queue);//此函数是用来把进程加入或删除一个队列,这是根据参数mode来区别的,当应用层fcntl(fd,F_SETFL,oflag | FASYNC)时,mode经过VFS层后自动为1,则此函数是用来加入队列//在release中,mode传递为0,则是把任务从队列中删除

}

structkey_release(structinode *,structfile *)

{

key_fasync(-1,file,0)//删除

}

structfile_operations fops={

.fasync  = key_fasync,

};

if(key_device->async_queue)

kill_fasync(&((structkey_device*)dev)->async_queue,SIGIO,POLL_IN);    //向队列中的所有任务发送SIGIO信号,

}




structkey_device{

structcdev cdev;

structfasync_struct *async_queue;//定义一个队列

}key_device;

intkey_fasync(intfd,structfile *file,intmode)

{

structkey_device *device = file->private_data;

returnfasync_helper(fd,file,mode,&device->async_queue);//此函数是用来把进程加入或删除一个队列,这是根据参数mode来区别的,当应用层fcntl(fd,F_SETFL,oflag | FASYNC)时,mode经过VFS层后自动为1,则此函数是用来加入队列//在release中,mode传递为0,则是把任务从队列中删除

}

structkey_release(structinode *,structfile *)

{

key_fasync(-1,file,0)//删除

}

structfile_operations fops={

.fasync  = key_fasync,

};

if(key_device->async_queue)

kill_fasync(&((structkey_device*)dev)->async_queue,SIGIO,POLL_IN);    //向队列中的所有任务发送SIGIO信号,

}





阻塞IO意味着一直等待设备可访问后再访问。

非阻塞IO中使用poll意味着查询设备是否可访问。

异步通知则意味着设备通知自身可访问,实现了异步IO。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 196,165评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,503评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,295评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,589评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,439评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,342评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,749评论 3 387
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,397评论 0 255
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,700评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,740评论 2 313
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,523评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,364评论 3 314
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,755评论 3 300
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,024评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,297评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,721评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,918评论 2 336

推荐阅读更多精彩内容