一、主設備號和次設備號
主設備號表示設備對應的驅動程序;次設備號由內核使用,用於正確確定設備文件所指的設備。
內核用dev_t類型()來保存設備編號,dev_t是一個32位的數,12位表示主設備號,20爲表示次設備號。
在實際使用中,是通過中定義的宏來轉換格式。
(dev_t)-->主設備號、次設備號 | MAJOR(dev_t dev) MINOR(dev_t dev) |
主設備號、次設備號-->(dev_t) | MKDEV(int major,int minor) |
建立一個字符設備之前,驅動程序首先要做的事情就是獲得設備編號。其這主要函數在中聲明:
|
分配之設備號的最佳方式是:默認採用動態分配,同時保留在加載甚至是編譯時指定主設備號的餘地。
以下是在scull.c中用來獲取主設備好的代碼:
|
在這部分中,比較重要的是在用函數獲取設備編號後,其中的參數name是和該編號範圍關聯的設備名稱,它將出現在/proc/devices和sysfs中。
看到這裏,就可以理解爲什麼mdev和udev可以動態、自動地生成當前系統需要的設備文件。udev就是通過讀取sysfs下的信息來識別硬件設備的.
(請看《理解和認識udev》
URL:http://blog.chinaunix.net/u/6541/showart_396425.html)
二、一些重要的數據結構
大部分基本的驅動程序操作涉及及到三個重要的內核數據結構,分別是file_operations、file和inode,它們的定義都在。
三、字符設備的註冊
內核內部使用struct cdev結構來表示字符設備。在內核調用設備的操作之前,必須分配並註冊一個或多個struct
cdev。代碼應包含,它定義了struct
cdev以及與其相關的一些輔助函數。
註冊一個獨立的cdev設備的基本過程如下:
1、爲struct cdev 分配空間(如果已經將struct cdev 嵌入到自己的設備的特定結構體中,並分配了空間,這步略過!)
struct cdev *my_cdev = cdev_alloc();
2、初始化struct cdev
void cdev_init(struct cdev *cdev, const struct file_operations *fops)
3、初始化cdev.owner
cdev.owner = THIS_MODULE;
4、cdev設置完成,通知內核struct cdev的信息(在執行這步之前必須確定你對struct cdev的以上設置已經完成!)
int cdev_add(struct cdev *p, dev_t dev, unsigned count)
從系統中移除一個字符設備:void cdev_del(struct cdev *p)
以下是scull中的初始化代碼(之前已經爲struct
scull_dev 分配了空間):
|
四、scull模型的內存使用
以下是scull模型的結構體:
|
scull驅動程序引入了兩個Linux內核中用於內存管理的核心函數,它們的定義都在:
|
以下是scull模塊中的一個釋放整個數據區的函數(類似清零),將在scull以寫方式打開和scull_cleanup_module中被調用:
|
以下是scull模塊中的一個沿鏈表前行得到正確scull_set指針的函數,將在read和write方法中被調用:
|
其實這個函數的實質是:如果已經存在這個scull_set,就返回這個scull_set的指針。如果不存在這個scull_set,一邊沿鏈表爲scull_set分配空間一邊沿鏈表前行,直到所需要的scull_set被分配到空間並初始化爲止,就返回這個scull_set的指針。
五、open和release
open方法提供給驅動程序以初始化的能力,爲以後的操作作準備。應完成的工作如下:
(1)檢查設備特定的錯誤(如設備未就緒或硬件問題);
(2)如果設備是首次打開,則對其進行初始化;
(3)如有必要,更新f_op指針;
(4)分配並填寫置於filp->private_data裏的數據結構。
而根據scull的實際情況,他的open函數只要完成第四步(將初始化過的struct scull_dev dev的指針傳遞到filp->private_data裏,以備後用)就好了,所以open函數很簡單。但是其中用到了定義在中的container_of宏,源碼如下:
|
其實從源碼可以看出,其作用就是:通過指針ptr,獲得包含ptr所指向數據(是member結構體)的type結構體的指針。即是用指針得到另外一個指針。
release方法提供釋放內存,關閉設備的功能。應完成的工作如下:
(1)釋放由open分配的、保存在file->private_data中的所有內容;
(2)在最後一次關閉操作時關閉設備。
由於前面定義了scull是一個全局且持久的內存區,所以他的release什麼都不做。
六、read和write
read和write方法的主要作用就是實現內核與用戶空間之間的數據拷貝。因爲Linux的內核空間和用戶空間隔離的,所以要實現數據拷貝就必須使用在中定義的:
|
而值得一提的是以上兩個函數和
|
之間的關係:通過源碼可知,前者調用後者,但前者在調用前對用戶空間指針進行了檢查。
至於read和write 的具體函數比較簡單,就在實驗中驗證好了。
七、模塊實驗
這次模塊實驗的使用是友善之臂SBC2440V4,使用Linux2.6.22.2內核。
模塊程序鏈接:scull模塊源程序
模塊測試程序鏈接:模塊測試程序
測試結果:
量子大小爲6: [Tekkaman2440@SBC2440V4]#cd /lib/modules/ [Tekkaman2440@SBC2440V4]#insmod scull.koscull_quantum=6 [Tekkaman2440@SBC2440V4]#cat /proc/devices Block devices:
啓動測試程序 [Tekkaman2440@SBC2440V4]#./scull_test write error! code=6 write error! code=6 write error! code=6 write ok! code=2 read error! code=6 read error! code=6 read error! code=6 read ok! code=2 [0]=0 [1]=1 [2]=2 [3]=3 [4]=4 [5]=5 [6]=6 [7]=7 [8]=8 [9]=9 [10]=10 [11]=11 [12]=12 [13]=13 [14]=14 [15]=15 [16]=16 [17]=17 [18]=18 [19]=19
改變量子大小爲默認值4000: [Tekkaman2440@SBC2440V4]#cd /lib/modules/ [Tekkaman2440@SBC2440V4]#rmmod scull [Tekkaman2440@SBC2440V4]#insmod scull.ko
啓動測試程序 [Tekkaman2440@SBC2440V4]#./scull_test write ok! code=20 read ok! code=20 [0]=0 [1]=1 [2]=2 [3]=3 [4]=4 [5]=5 [6]=6 [7]=7 [8]=8 [9]=9 [10]=10 [11]=11 [12]=12 [13]=13 [14]=14 [15]=15 [16]=16 [17]=17 [18]=18 [19]=19
[Tekkaman2440@SBC2440V4]# 改變量子大小爲6,量子集大小爲2: [Tekkaman2440@SBC2440V4]#cd /lib/modules/ [Tekkaman2440@SBC2440V4]#rmmod scull [Tekkaman2440@SBC2440V4]#insmod scull.ko scull_quantum=6 scull_qset=2
啓動測試程序 [Tekkaman2440@SBC2440V4]#./scull_test write error! code=6 write error! code=6 write error! code=6 write ok! code=2 read error! code=6 read error! code=6 read error! code=6 read ok! code=2 [0]=0 [1]=1 [2]=2 [3]=3 [4]=4 [5]=5 [6]=6 [7]=7 [8]=8 [9]=9 [10]=10 [11]=11 [12]=12 [13]=13 [14]=14 [15]=15 [16]=16 [17]=17 [18]=18 [19]=19
|
實驗不僅測試了模塊的讀寫能力,還測試了量子讀寫是否有效