1
本文主題
本文主要分析
Linux字符設備驅動程序的工作機理。主要內容以及代碼片段來源于《LDD3》,俺只是從另外一個角度來講述。
見過很多關于驅動程序的書,基本上都是告訴你怎么做,然后你
STEP BY STEP,然后運行完后結果就出來了,可是其背后到底是如何工作的呢?雖說《LDD3》也講了很多原理性的東西,但是我覺得這個問題其描述得并不明確。
2
關于scull
scull
是《LDD3》的一個字符設備驅動程序,其加載之后會在文件系統下生成/dev/scull文件,在shell下可以對其進行一系列的操作,例如可以使用cp、dd或者輸入輸出重定向等命令來訪問這個文件,也就是訪問這個字符設備。
3 /dev/scull
是如何生成的
該文件是在驅動程序模塊加載時生成的,具體實現是在
scull_load(可以到www.oreilly.com下載,或者《
LDD3_中文》的P51)這個腳本里。
在
shell下insmod了scull.ko之后,系統會在/proc/devices文件里生成設備名以及與之對應的主設備號信息,scull_load腳本根據該信息mknod了/dev/scull文件節點。
注意下
mknod時給的參數信息,其中注明了主設備號以及次設備號信息。
4 shell
下操作/dev/scull是如何關聯到我們自己編寫的scull模塊的
這個問題一直困擾了我很久,不過現在我基本上算是弄明白了。
4.1
先回顧下scull模塊程序的初始化操作里做了什么事情。
搞驅動的都知道,初始化的時候需要進行設備注冊,核心代碼如下:
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
是一個數組,其中保存了我們的scull實際的操作:
struct file_operation scull_fops = {
...
.open = scull_open,
.read = scull_read,
.write = scull_write,
.release = scull_release,
cdev
是struct cdev類型(該類型由系統定義)的變量,其嵌入在struct scull_dev(該類型由scull模塊定義)中:
struct scull_dev{
...
struct cdev cdev;
...
}
從這句代碼
cdev_add(&dev->cdev, devno, 1);
我們可以發現,設備號
devno已經和內核內部結構cdev關聯起來了。
4.2 /dev/scull
設備的打開操作
/dev/scull
是文件系統中的一個設備文件,其在內核中由sturct inode結構表示。
一般說來,
inode結構下由兩個元素和驅動程序有關:
dev_t i_rdev;
這個是設備號
struct cdev *i_cdev;
這里是一個指針,struct cdev是字符設備在內核里的表示,注意我們在4.1小節里也提到了這個結構。
也就是說,系統在打開
/dev/scull之前僅知道inode信息。在4.1小節里已經說過了,設備號和內核內部結構cdev關聯起來了,因此cdev下的ops也和設備號關聯起來了,ops已經初始化為scull_fops。所以我們才能夠通過inode中的設備號i_rdev定位到此時打開的是哪一個設備,從而去調用相應設備的open操作。
看看
open函數的原型吧:
int (*open)(struct inode*, struct file*);
第一個參數是
struct inode類型的指針,該指針由內核傳遞過來,是設備文件在內核中的表示。
第二個參數
struct file類型的指針,open之后,設備文件在內核中就以struct file結構來表示了。
看看
open操作在scull里的實現吧:
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的結構提取出來了,該結構就是在scull模塊里定以的struct scull_dev,其中包含了scull模塊所需要的私有信息。
第二句代碼把該私有信息加入內核數據結構
struct file,之前說過了,open后的文件在內核里以struct file結構表示。
4.3 /dev/scull
的讀寫操作
先看下讀寫操作的函數原型:
ssize_t (*read)(struct file *, char __user *, size_t, loff_t *);
ssize_t (*write)(struct file*, char __user *, size_t, loff_t *);
其參數內容我不做過多解釋,注意其第一個參數是
struct file類型的指針,該指針由內核傳入,也就是open之后的那個文件指針。
通過這個指針,能夠找到
scull模塊的私有數據結構struct scull_dev,并對其操作,我們可以從read/write操作在scull里的實現看到這個做法:
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;
...
}