/****************** * 高級(jí)字符設(shè)備驅(qū)動(dòng) ******************/
(1)ioctl
除了讀取和寫(xiě)入設(shè)備外,大部分驅(qū)動(dòng)程序還需要另外一種能力,即通過(guò)設(shè)備驅(qū)動(dòng)程序執(zhí)行各種類型的硬件控制。比如彈出介質(zhì),改變波特率等等。這些操作通過(guò)ioctl方法支持,該方法實(shí)現(xiàn)了同名的系統(tǒng)調(diào)用。
在用戶空間,ioctl系統(tǒng)調(diào)用的原型是:
驅(qū)動(dòng)程序的ioctl方法原型和用戶空間的版本有一些不同:
int (*ioctl) (struct inode *inode,
struct file *filp,
unsigned int cmd,
unsigned long arg);
inode/filp: 對(duì)應(yīng)用戶空間的fd
cmd: 對(duì)應(yīng)用戶空間傳來(lái)的cmd
arg: 對(duì)應(yīng)傳來(lái)的cmd參數(shù)
大多數(shù)ioctl的實(shí)現(xiàn)中都包括一個(gè)switch語(yǔ)句,用于根據(jù)cmd參數(shù)選擇對(duì)應(yīng)的操作。用戶空間和內(nèi)核空間的命令號(hào)要一致。
(2)選擇ioctl的命令號(hào)
在編寫(xiě)ioctl的代碼之前,要選擇對(duì)應(yīng)不同命令的編號(hào)。不能簡(jiǎn)單地從0或1開(kāi)始選擇編號(hào),因?yàn)閘inux要求這個(gè)命令號(hào)應(yīng)該在系統(tǒng)范圍內(nèi)唯一。linux內(nèi)核采用約定方法為驅(qū)動(dòng)程序選擇ioctl號(hào),可以參考include/asm/ioctl.h和Documentation/ioctl-number.txt。
一個(gè)ioctl號(hào)為32位,linux將其分成4個(gè)部分,構(gòu)建一個(gè)ioctl號(hào)碼所需要的宏都定義在<linux/ioctl.h>:
可以采用<linux/ioctl.h>中的宏構(gòu)建一個(gè)ioctl號(hào)
返回值
對(duì)于系統(tǒng)調(diào)用來(lái)說(shuō),正的返回值是首保護(hù)的,而負(fù)值被認(rèn)為是一個(gè)錯(cuò)誤,并被用來(lái)設(shè)置用戶空間的error變量。如果在調(diào)用ioctl方法時(shí)傳入了沒(méi)有定義的ioctl號(hào),則系統(tǒng)返回的錯(cuò)誤值為-ENVAL和-ENOTTY
(3)阻塞和非阻塞型操作
對(duì)于read和write等操作,默認(rèn)的操作是阻塞型的,其特性是:
*如果一個(gè)進(jìn)程調(diào)用了read但還沒(méi)有數(shù)據(jù)可讀,則此進(jìn)程必須阻塞。數(shù)據(jù)到達(dá)時(shí)進(jìn)程被喚醒,并把數(shù)據(jù)返回給調(diào)用者,即使數(shù)據(jù)數(shù)目少于count參數(shù)指定的數(shù)據(jù)也會(huì)返回。
*如果一個(gè)進(jìn)程調(diào)用了write但緩沖區(qū)沒(méi)有空間,則此進(jìn)程必須阻塞,而且必須休眠在與讀進(jìn)程不同的等待隊(duì)列上。當(dāng)向硬件設(shè)備寫(xiě)入一些數(shù)據(jù),從而騰出了部分輸出緩沖區(qū)后,進(jìn)程即被喚醒,write調(diào)用成功。
有時(shí)我們希望改變這一特性,將其改為非阻塞型的,這樣,無(wú)論設(shè)備是否有數(shù)據(jù)可讀寫(xiě),read/write方法都馬上返回。
如果希望設(shè)定某個(gè)文件是非阻塞的,則應(yīng)設(shè)定filp->f_flags的O_NONBLOCK標(biāo)志。處理非阻塞型文件時(shí),應(yīng)用程序調(diào)用stdio函數(shù)必須非常小心,因?yàn)楹苋菀装岩粋€(gè)非阻塞型的返回誤認(rèn)為是EOF,所以必須始終檢查errno。
(4)異步通知
a.異步通知的作用
大多數(shù)時(shí)候阻塞型和非阻塞型操作的組合以及select方法可以有效查詢?cè)O(shè)備,但有時(shí)候用這種技術(shù)效率就不高了。在面對(duì)某些隨機(jī)或很少出現(xiàn)的情況時(shí)(如通過(guò)鍵盤(pán)輸入CTRL+C),則需要采用異步通知(asynchronous notification)。
b.用戶空間程序如何啟動(dòng)異步通知
為了啟動(dòng)文件的異步通知機(jī)制,用戶程序必須執(zhí)行兩個(gè)步驟:
(5)驅(qū)動(dòng)程序中如何實(shí)現(xiàn)異步通知
a.用戶空間操作在內(nèi)核的對(duì)應(yīng)
b.在設(shè)備結(jié)構(gòu)體中加入fasync_struct的指針
該結(jié)構(gòu)在<linux/fs.h>中定義:
struct fasync_struct { int magic; int fa_fd; struct fasync_struct *fa_next; struct file *fa_file; };
c.驅(qū)動(dòng)要調(diào)用的兩個(gè)函數(shù)
這兩個(gè)函數(shù)在<linux/fs.h>中聲明。
定義在/fs/fcntl.c中。
原型如下:
當(dāng)一個(gè)打開(kāi)文件的FASYNC標(biāo)志被修改,調(diào)用fasync_helper以便從相關(guān)的進(jìn)程列表中增加或刪除文件,而kill_fasync在數(shù)據(jù)到達(dá)時(shí)通知所有相關(guān)進(jìn)程。
d.例子
01.在設(shè)備類型中定義fasync_struct動(dòng)態(tài)數(shù)據(jù)結(jié)構(gòu)
struct my_pipe { struct fasync_struct *async_queue; /* 異步讀取結(jié)構(gòu) */ ...... };
02.驅(qū)動(dòng)中的fasync函數(shù)調(diào)用fasync_helper
int my_fasync(fasync_file fd, struct file *filp, int mode) { my_pipe *dev = filp->private_data; return fasync_helper(fd, filp, mode, &dev->async_queue); }
03.符合異步通知條件時(shí)調(diào)用kill_fasync
異步通知的是一個(gè)讀進(jìn)程,所以要用write發(fā)送kill_fasync。
調(diào)用kill_fasync向所有注冊(cè)在設(shè)備上的異步隊(duì)列async_queue中的進(jìn)程發(fā)送信號(hào)SIGIO。
ssize_t my_write(struct file *filp, const char *buf, size_t count, loff_t *f_pos) { ...... if (dev->async_queue) kill_fasync(&dev->async_queue, SIGIO, POLL_IN); ...... }
04.關(guān)閉文件時(shí)必須調(diào)用fasync方法
當(dāng)關(guān)閉文件時(shí)必須調(diào)用fasync方法,以便從活動(dòng)的異步讀進(jìn)程列表中刪除該文件。
在release中調(diào)用:scull_p_fasync(-1, filp, 0);
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
標(biāo)簽:蕪湖 撫順 孝感 巴中 綏化 晉城 南充 福州
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Linux內(nèi)核設(shè)備驅(qū)動(dòng)之高級(jí)字符設(shè)備驅(qū)動(dòng)筆記整理》,本文關(guān)鍵詞 Linux,內(nèi)核,設(shè)備驅(qū)動(dòng),之,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。