電子工程網(wǎng)
標(biāo)題: Linux字符設(shè)備驅(qū)動程序工作機理分析 [打印本頁]
作者: huangedu 時間: 2016-4-11 15:26
標(biāo)題: Linux字符設(shè)備驅(qū)動程序工作機理分析
1 本文主題
本文主要分析Linux字符設(shè)備驅(qū)動程序的工作機理。主要內(nèi)容以及代碼片段來源于《LDD3》,俺只是從另外一個角度來講述。
見過很多關(guān)于驅(qū)動程序的書,基本上都是告訴你怎么做,然后你STEP BY STEP,然后運行完后結(jié)果就出來了,可是其背后到底是如何工作的呢?雖說《LDD3》也講了很多原理性的東西,但是我覺得這個問題其描述得并不明確。
2 關(guān)于scull
scull是《LDD3》的一個字符設(shè)備驅(qū)動程序,其加載之后會在文件系統(tǒng)下生成/dev/scull文件,在shell下可以對其進行一系列的操作,例如可以使用cp、dd或者輸入輸出重定向等命令來訪問這個文件,也就是訪問這個字符設(shè)備。
3 /dev/scull是如何生成的
該文件是在驅(qū)動程序模塊加載時生成的,具體實現(xiàn)是在scull_load(可以到www.oreilly.com下載,或者《LDD3_中文》的P51)這個腳本里。
在shell下insmod了scull.ko之后,系統(tǒng)會在/proc/devices文件里生成設(shè)備名以及與之對應(yīng)的主設(shè)備號信息,scull_load腳本根據(jù)該信息mknod了/dev/scull文件節(jié)點。
注意下mknod時給的參數(shù)信息,其中注明了主設(shè)備號以及次設(shè)備號信息。
4 shell下操作/dev/scull是如何關(guān)聯(lián)到我們自己編寫的scull模塊的
這個問題一直困擾了我很久,不過現(xiàn)在我基本上算是弄明白了。
4.1 先回顧下scull模塊程序的初始化操作里做了什么事情。
搞驅(qū)動的都知道,初始化的時候需要進行設(shè)備注冊,核心代碼如下:
static void scull_setup_cdev(struct scull_dev *dev, int index)
{
int err, devno = MKDEV(scull_major, scull_minor + index);
cdev_init(&dev->cdev, &scull_fops);
dev->cdev.owner = THIS_MODULE;
dev->cdev.ops = &scull_fops;
err = cdev_add(&dev->cdev, devno, 1);
...
}
scull_fops是一個數(shù)組,其中保存了我們的scull實際的操作:
struct file_operation scull_fops = {
...
.open = scull_open,
.read = scull_read,
.write = scull_write,
.release = scull_release,
cdev是struct cdev類型(該類型由系統(tǒng)定義)的變量,其嵌入在struct scull_dev(該類型由scull模塊定義)中:
struct scull_dev{
...
struct cdev cdev;
...
}
從這句代碼
cdev_add(&dev->cdev, devno, 1);
我們可以發(fā)現(xiàn),設(shè)備號devno已經(jīng)和內(nèi)核內(nèi)部結(jié)構(gòu)cdev關(guān)聯(lián)起來了。
4.2 /dev/scull設(shè)備的打開操作
/dev/scull是文件系統(tǒng)中的一個設(shè)備文件,其在內(nèi)核中由sturct inode結(jié)構(gòu)表示。
一般說來,inode結(jié)構(gòu)下由兩個元素和驅(qū)動程序有關(guān):
dev_t i_rdev; 這個是設(shè)備號
struct cdev *i_cdev; 這里是一個指針,struct cdev是字符設(shè)備在內(nèi)核里的表示,注意我們在4.1小節(jié)里也提到了這個結(jié)構(gòu)。
也就是說,系統(tǒng)在打開/dev/scull之前僅知道inode信息。在4.1小節(jié)里已經(jīng)說過了,設(shè)備號和內(nèi)核內(nèi)部結(jié)構(gòu)cdev關(guān)聯(lián)起來了,因此cdev下的ops也和設(shè)備號關(guān)聯(lián)起來了,ops已經(jīng)初始化為scull_fops。所以我們才能夠通過inode中的設(shè)備號i_rdev定位到此時打開的是哪一個設(shè)備,從而去調(diào)用相應(yīng)設(shè)備的open操作。
看看open函數(shù)的原型吧:
int (*open)(struct inode*, struct file*);
第一個參數(shù)是struct inode類型的指針,該指針由內(nèi)核傳遞過來,是設(shè)備文件在內(nèi)核中的表示。
第二個參數(shù)struct file類型的指針,open之后,設(shè)備文件在內(nèi)核中就以struct file結(jié)構(gòu)來表示了。
看看open操作在scull里的實現(xiàn)吧:
int scull_open(struct inode *inode, struct file *filp)
{
struct scull_dev *dev;
dev = container_of(inode->i_cdev, struct scull_dev, cdev);
filp->private_data = dev;
...
}
第一句代碼通過宏container_of把包裹inode->i_cdev的結(jié)構(gòu)提取出來了,該結(jié)構(gòu)就是在scull模塊里定以的struct scull_dev,其中包含了scull模塊所需要的私有信息。
第二句代碼把該私有信息加入內(nèi)核數(shù)據(jù)結(jié)構(gòu)struct file,之前說過了,open后的文件在內(nèi)核里以struct file結(jié)構(gòu)表示。
4.3 /dev/scull的讀寫操作
先看下讀寫操作的函數(shù)原型:
ssize_t (*read)(struct file *, char __user *, size_t, loff_t *);
ssize_t (*write)(struct file*, char __user *, size_t, loff_t *);
其參數(shù)內(nèi)容我不做過多解釋,注意其第一個參數(shù)是struct file類型的指針,該指針由內(nèi)核傳入,也就是open之后的那個文件指針。
通過這個指針,能夠找到scull模塊的私有數(shù)據(jù)結(jié)構(gòu)struct scull_dev,并對其操作,我們可以從read/write操作在scull里的實現(xiàn)看到這個做法:
ssize_t scull_read(struct file *filp, char __user *buf, size_t count, loff_t *f_pos)
{
sturct scull_dev *dev = filp->private_data;
int quantum = dev->quantum, qset = dev->qset;
...
}
ssize_t scull_write(struct file *filp, char __user *buf, size_t count, loff_t *f_pos)
{
sturct scull_dev *dev = filp->private_data;
int quantum = dev->quantum, qset = dev->qset;
...
}
-
u=2429609584,2450763625&fm=21&gp=0 - 副本.jpg
(4.55 KB)
下載附件
2016-4-11 15:25 上傳
-
10671746 - 副本.gif
(36.64 KB)
下載附件
2016-4-11 15:25 上傳
| 歡迎光臨 電子工程網(wǎng) (http://m.4huy16.com/) |
Powered by Discuz! X3.4 |