塊設(shè)備和字符設(shè)備驅(qū)動(dòng)比較
掃描二維碼
隨時(shí)隨地手機(jī)看文章
Linux中I/O設(shè)備分為兩類:塊設(shè)備和字符設(shè)備。兩種設(shè)備本身沒(méi)有嚴(yán)格限制,但是,基于不同的功能進(jìn)行了分類。
(1) 字符設(shè)備:提供連續(xù)的數(shù)據(jù)流,應(yīng)用程序可以順序讀取,通常不支持隨機(jī)存取。相反,此類設(shè)備支持按字節(jié)/字符來(lái)讀寫(xiě)數(shù)據(jù)。舉例來(lái)說(shuō),調(diào)制解調(diào)器是典型的字符設(shè)備。
(2) 塊設(shè)備:應(yīng)用程序可以隨機(jī)訪問(wèn)設(shè)備數(shù)據(jù),程序可自行確定讀取數(shù)據(jù)的位置。硬盤(pán)是典型的塊設(shè)備,應(yīng)用程序可以尋址磁盤(pán)上的任何位置,并由此讀取數(shù)據(jù)。此外,數(shù)據(jù)的讀寫(xiě)只能以塊(通常是512B)的倍數(shù)進(jìn)行。與字符設(shè)備不同,塊設(shè)備并不支持基于字符的尋址。
兩種設(shè)備本身并沒(méi)用嚴(yán)格的區(qū)分,主要是字符設(shè)備和塊設(shè)備驅(qū)動(dòng)程序提供的訪問(wèn)接口(file I/O API)是不一樣的。本文主要就數(shù)據(jù)接口、訪問(wèn)接口和設(shè)備注冊(cè)方法對(duì)兩種設(shè)備進(jìn)行比較。
1、數(shù)據(jù)結(jié)構(gòu)
1.1字符設(shè)備數(shù)據(jù)結(jié)構(gòu)
struct file;
struct inode;
file定義于 , 是設(shè)備驅(qū)動(dòng)中第二個(gè)最重要的數(shù)據(jù)結(jié)構(gòu). 文件結(jié)構(gòu)代表一個(gè)打開(kāi)的文件. 它由內(nèi)核在 open 時(shí)創(chuàng)建, 并傳遞給在文件上操作的任何函數(shù), 直到最后的關(guān)閉. 在文件的所有實(shí)例都關(guān)閉后, 內(nèi)核釋放這個(gè)數(shù)據(jù)結(jié)構(gòu)。
inode 結(jié)構(gòu)由內(nèi)核在內(nèi)部用來(lái)表示文件.inode 結(jié)構(gòu)包含大量關(guān)于文件的信息其中dev_t i_rdev成員包含實(shí)際的設(shè)備編號(hào).struct cdev *i_cdev中struct cdev 是內(nèi)核的內(nèi)部結(jié)構(gòu), 代表字符設(shè)備。
1.2塊設(shè)備數(shù)據(jù)結(jié)構(gòu)
struct gendisk (定義于 ) 是單獨(dú)一個(gè)磁盤(pán)驅(qū)動(dòng)器的內(nèi)核表示. 事實(shí)上, 內(nèi)核還使用 gendisk 來(lái)表示分區(qū)。
2、設(shè)備訪問(wèn)接口
2.1字符設(shè)備訪問(wèn)接口
struct file_operations 其中file_operation 結(jié)構(gòu)中的每個(gè)成員必須指向驅(qū)動(dòng)中的函數(shù), 這些函數(shù)實(shí)現(xiàn)一個(gè)特別的操作, 或者對(duì)于不支持的操作留置為 NULL. 當(dāng)指定為 NULL 指針時(shí)內(nèi)核的確切的行為是每個(gè)函數(shù)不同的,該結(jié)構(gòu)中主要函數(shù)如下:
ssize_t read(struct file *filp, char __user *buff, size_t count, loff_t *offp);
ssize_t write(struct file *filp, const char __user *buff, size_t count, loff_t *offp);
filp 是文件指針, count 是請(qǐng)求的傳輸數(shù)據(jù)大小. buff 參數(shù)指向持有被寫(xiě)入數(shù)據(jù)的緩存, 或者放入新數(shù)據(jù)的空緩存. 最后, offp 是一個(gè)指針指向一個(gè)"long offset type"對(duì)象, 它指出用戶正在存取的文件位置. 返回值是一個(gè)"signed size type"。
2.2塊設(shè)備訪問(wèn)接口
字符設(shè)備通過(guò) file_ 操作結(jié)構(gòu)使它們的操作對(duì)系統(tǒng)可用. 一個(gè)類似的結(jié)構(gòu)用在塊設(shè)備上; 它是 struct block_device_operations, 定義在 ,其主要操作方法如下:
int (*open)(struct inode *inode, struct file *filp);
int (*release)(struct inode *inode, struct file *filp);
就像它們的字符驅(qū)動(dòng)對(duì)等體一樣工作的函數(shù); 無(wú)論何時(shí)設(shè)備被打開(kāi)和關(guān)閉都調(diào)用它們. 一個(gè)字符驅(qū)動(dòng)可能通過(guò)啟動(dòng)設(shè)備或者鎖住門(mén)(為可移出的介質(zhì))來(lái)響應(yīng)一個(gè) open 調(diào)用. 如果你將介質(zhì)鎖入設(shè)備, 你當(dāng)然應(yīng)當(dāng)在 release 方法中解鎖。
int (*ioctl)(struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg);
實(shí)現(xiàn) ioctl 系統(tǒng)調(diào)用的方法. 但是, 塊層首先解釋大量的標(biāo)準(zhǔn)請(qǐng)求; 因此大部分的塊驅(qū)動(dòng) ioctl 方法相當(dāng)短。
3、設(shè)備注冊(cè)
3.1字符設(shè)備注冊(cè)
int register_chrdev_region(dev_t first, unsigned int count, char *name)
int alloc_chrdev_region(dev_t *dev, unsigned int firstminor, unsigned int count, char *name)
void unregister_chrdev_region(dev_t first, unsigned int count);
允許驅(qū)動(dòng)分配和釋放設(shè)備編號(hào)的范圍的函數(shù). register_chrdev_region 應(yīng)當(dāng)用在事先知道需要的主編號(hào)時(shí); 對(duì)于動(dòng)態(tài)分配, 使用 alloc_chrdev_region 代替.
3.2塊設(shè)備注冊(cè)
int register_blkdev(unsigned int major, const char *name);
int unregister_blkdev(unsigned int major, const char *name);
register_blkdev 注冊(cè)一個(gè)塊驅(qū)動(dòng)到內(nèi)核, 并且, 可選地, 獲得一個(gè)主編號(hào). 一個(gè)驅(qū)動(dòng)可被注銷, 使用 unregister_blkdev。