小男孩‘自慰网亚洲一区二区,亚洲一级在线播放毛片,亚洲中文字幕av每天更新,黄aⅴ永久免费无码,91成人午夜在线精品,色网站免费在线观看,亚洲欧洲wwwww在线观看

分享

Linux驅(qū)動(dòng)技術(shù)(四)

 豫龍晏子 2017-02-13

異步通知的全稱是'信號(hào)驅(qū)動(dòng)的異步IO',通過(guò)'信號(hào)'的方式,放期望獲取的資源可用時(shí),驅(qū)動(dòng)會(huì)主動(dòng)通知指定的應(yīng)用程序,和應(yīng)用層的'信號(hào)'相對(duì)應(yīng),這里使用的是信號(hào)'SIGIO'。操作步驟是

  1. 應(yīng)用層程序?qū)⒆约鹤?cè)為接收來(lái)自設(shè)備文件的SIGIO信號(hào)的進(jìn)程
  2. 驅(qū)動(dòng)實(shí)現(xiàn)相應(yīng)的接口,以期具有向所有注冊(cè)接收這個(gè)設(shè)備驅(qū)動(dòng)SIGIO信號(hào)的應(yīng)用程序發(fā)SIGIO信號(hào)的能力。
  3. 驅(qū)動(dòng)在適當(dāng)?shù)奈恢谜{(diào)用發(fā)送函數(shù),應(yīng)用程序即可接收到SIGIO信號(hào)。

整個(gè)機(jī)制的框架:

Linux驅(qū)動(dòng)技術(shù)(四) _異步通知技術(shù)

應(yīng)用層接收SIGIO

和其他信號(hào)一樣,應(yīng)用層需要注冊(cè)一個(gè)信號(hào)處理函數(shù),

注冊(cè)的方式還是使用signalsigaction

此外,應(yīng)用層還需要把自己加入到驅(qū)動(dòng)的通知鏈表中,加入的代碼如下

fcntl(dev_fd,F_SETOWN,getpid);int oflags = fcntl(dev_fd,F_GETFL);fcntl(dev_fd,F(xiàn)_SETFL,oflags|FASYNC);...while(1);

完成了上面的工作,應(yīng)用層的程序就可以靜待SIGIO的到來(lái)了。

驅(qū)動(dòng)發(fā)送SIGIO

應(yīng)用層注冊(cè)好了,最終的發(fā)送還是看設(shè)備驅(qū)動(dòng)的處理方式,為了使設(shè)備支持異步通知機(jī)制,參照應(yīng)用層的接口,驅(qū)動(dòng)程序中涉及3項(xiàng)工作。

  1. 支持

    F_SETOWN

  2. 支持

    F_SETFL

    FASYNC

    fasync

    驅(qū)動(dòng)中要實(shí)現(xiàn)fasync
  3. 當(dāng)設(shè)備資源可用時(shí),

    通過(guò)kill_fasync發(fā)送SIGIO

為了在內(nèi)核中實(shí)現(xiàn)上面這三個(gè)功能,驅(qū)動(dòng)需要使用1個(gè)結(jié)構(gòu)+2個(gè)API,結(jié)構(gòu)是struct fasync_struct,函數(shù)是fasync_helperkill_fasync

struct fasync_struct { spinlock_t fa_lock; int magic; int fa_fd; struct fasync_struct *fa_next; /* singly linked list */ struct file *fa_file; struct rcu_head fa_rcu;};

fasync_helper的作用是將一個(gè)fasync_struct的對(duì)象注冊(cè)進(jìn)內(nèi)核,應(yīng)用層執(zhí)行fcntl(dev_fd,F(xiàn)_SETFL,oflags|FASYNC)時(shí)會(huì)回調(diào)驅(qū)動(dòng)的fops.fasync,所以通常將fasync_helper放到fasync的實(shí)現(xiàn)中。

/** *fasync_helper - 將一個(gè)fasync_struct對(duì)象注冊(cè)進(jìn)內(nèi)核 *@fd:文件描述符,由fasync傳入 *@filp:file指針,由fasync傳入 *@sig:信號(hào)類(lèi)型,通常使用的就是SIGIO *@dev_fasync:事前準(zhǔn)備的fasync_struct對(duì)象指針的指針 */int fasync_helper(int fd, struct file * filp, int sig, struct fasync_struct ** dev_fasync);

下面這個(gè)API就是釋放SIGIO,根據(jù)需求的不同放到不同的位置。

/** *kill_fasync - 釋放一個(gè)信號(hào) *@dev_fasync:事前使用fasync_helper注冊(cè)進(jìn)內(nèi)核的fasync_struct對(duì)象指針的指針 *@filp:file指針,由fasync傳入 *@sig:信號(hào)類(lèi)型,通常使用的就是SIGIO *@flag:標(biāo)志,通常,如果資源可讀用POLLIN,如果資源可寫(xiě)用POLLOUT */void kill_fasync(struct fasync_struct **dev_fasync, int sig, int flag);
驅(qū)動(dòng)模板

下面這個(gè)驅(qū)動(dòng)模板針對(duì)在硬件中斷到來(lái)(資源可用)的時(shí)候向應(yīng)用層發(fā)信號(hào),實(shí)際的操作中表明資源可用的情境還有很多

static struct fasync_struct *fasync = NULL;static irqreturn_t handler(int irq, void *dev){ kill_fasync(&fasync, SIGIO, POLLIN); return IRQ_HANDLED;}static int demo_fasync(int fd, struct file *filp, int mode){ return fasync_helper(fd, filp, mode, &fasync);}struct file_operations fops = { ... .fasync = demo_fasync, ...}static int __init demo_init(void){ ... request_irq(irq, handler, IRQF_TRIGGER_RISING, 'demo', NULL); ...}

    本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購(gòu)買(mǎi)等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊一鍵舉報(bào)。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評(píng)論

    發(fā)表

    請(qǐng)遵守用戶 評(píng)論公約

    類(lèi)似文章 更多