在用戶態(tài)下編程可以通過(guò)main()的來(lái)傳遞命令行參數(shù),而編寫(xiě)一個(gè)內(nèi)核模塊則通過(guò)module_param ()
module_param宏是Linux 2.6內(nèi)核中新增的,該宏被定義在include/linux/moduleparam.h文件中,具體定義如下:
#define module_param(name, type, perm)
module_param_named(name, name, type, perm)
其中使用了 3 個(gè)參數(shù):要傳遞的參數(shù)變量名, 變量的數(shù)據(jù)類(lèi)型, 以及訪問(wèn)參數(shù)的權(quán)限。
module_param (name,type,perm);
module_param 使用了 3 個(gè)參數(shù): 變量名, 它的類(lèi)型, 以及一個(gè)權(quán)限掩碼用來(lái)做一個(gè)輔助的 sysfs 入口(啥意思). 這個(gè)宏定義應(yīng)當(dāng)放在任何函數(shù)之外, 典型地是出現(xiàn)在源文件的前面.定義如:
static char *whom = "world";
static int howmany = 1;
module_param (howmany, int, S_IRUGO);
module_param (whom, charp, S_IRUGO);
模塊參數(shù)支持許多類(lèi)型:
bool
invbool
一個(gè)布爾型( true 或者 false)值(相關(guān)的變量應(yīng)當(dāng)是 int 類(lèi)型). invbool 類(lèi)型顛倒了值, 所以真值變成 false, 反之亦然.
charp
一個(gè)字符指針值. 內(nèi)存為用戶提供的字串分配, 指針因此設(shè)置.
int
long
short
uint
ulong
ushort
基本的變長(zhǎng)整型值. 以 u 開(kāi)頭的是無(wú)符號(hào)值.
數(shù)組參數(shù), 用逗號(hào)間隔的列表提供的值, 模塊加載者也支持. 聲明一個(gè)數(shù)組參數(shù), 使用:
module_param _array(name,type,num,perm);
這里 name 是你的數(shù)組的名子(也是參數(shù)名),
type 是數(shù)組元素的類(lèi)型,
num 是一個(gè)整型變量,
perm 是通常的權(quán)限值.
如果數(shù)組參數(shù)在加載時(shí)設(shè)置, num 被設(shè)置成提供的數(shù)的個(gè)數(shù). 模塊加載者拒絕比數(shù)組能放下的多的值.
perm參數(shù)的作用是什么?
最后的 module_param 字段是一個(gè)權(quán)限值; 你應(yīng)當(dāng)使用 中定義的值. 這個(gè)值控制誰(shuí)可以存取這些模塊參數(shù)在 sysfs 中的表示.如果 perm 被設(shè)為 0, 就根本沒(méi)有 sysfs 項(xiàng). 否則, 它出現(xiàn)在 /sys/module下面, 帶有給定的權(quán)限. 使用 S_IRUGO 作為參數(shù)可以被所有人讀取, 但是不能改變; S_IRUGO|S_IWUSR 允許 root 來(lái)改變參數(shù). 注意, 如果一個(gè)參數(shù)被 sysfs 修改, 你的模塊看到的參數(shù)值也改變了, 但是你的模塊沒(méi)有任何其他的通知. 你應(yīng)當(dāng)不要使模塊參數(shù)可寫(xiě), 除非你準(zhǔn)備好檢測(cè)這個(gè)改變并且因而作出反應(yīng).
perm參數(shù)的作用是什么?
最后的 module_param 字段是一個(gè)權(quán)限值,表示此參數(shù)在sysfs文件系統(tǒng)中所對(duì)應(yīng)的文件節(jié)點(diǎn)的屬性。你應(yīng)當(dāng)使用 中定義的值. 這個(gè)值控制誰(shuí)可以存取這些模塊參數(shù)在 sysfs 中的表示.當(dāng)perm為0時(shí),表示此參數(shù)不存在 sysfs文件系統(tǒng)下對(duì)應(yīng)的文件節(jié)點(diǎn)。 否則, 模塊被加載后,在/sys/module/ 目錄下將出現(xiàn)以此模塊名命名的目錄, 帶有給定的權(quán)限.。
權(quán)限在include/linux/stat.h中有定義
比如:
#define S_IRWXU 00700
#define S_IRUSR 00400
#define S_IWUSR 00200
#define S_IXUSR 00100
#define S_IRWXG 00070
#define S_IRGRP 00040
#define S_IWGRP 00020
#define S_IXGRP 00010
#define S_IRWXO 00007
#define S_IROTH 00004
#define S_IWOTH 00002
#define S_IXOTH 00001
使用 S_IRUGO 作為參數(shù)可以被所有人讀取, 但是不能改變; S_IRUGO|S_IWUSR 允許 root 來(lái)改變參數(shù). 注意, 如果一個(gè)參數(shù)被 sysfs 修改, 你的模塊看到的參數(shù)值也改變了, 但是你的模塊沒(méi)有任何其他的通知. 你應(yīng)當(dāng)不要使模塊參數(shù)可寫(xiě), 除非你準(zhǔn)備好檢測(cè)這個(gè)改變并且因而作出反應(yīng).
通過(guò)宏MODULE_PARM_DESC()對(duì)參數(shù)進(jìn)行說(shuō)明:
static unsigned short size = 1;
module_param(size, ushort, 0644);
MODULE_PARM_DESC(size, “The size in inches of the fishing pole”
“connected to this computer.” );
module_param() 和module_param_array() 的作用就是讓那些全局變量對(duì)insmod 可見(jiàn),使模塊裝載時(shí)可重新賦值。
module_param_array() 宏的第三個(gè)參數(shù)用來(lái)記錄用戶insmod 時(shí)提供的給這個(gè)數(shù)組的元素個(gè)數(shù),NULL 表示不關(guān)心用戶提供的個(gè)數(shù)
module_param() 和module_param_array() 最后一個(gè)參數(shù)權(quán)限值不能包含讓普通用戶也有寫(xiě)權(quán)限,否則編譯報(bào)錯(cuò)。這點(diǎn)可參考linux/moduleparam.h 中__module_param_call() 宏的定義。
字符串?dāng)?shù)組中的字符串似乎不能包含逗號(hào),否則一個(gè)字符串會(huì)被解析成兩個(gè)
一個(gè)測(cè)試用例:parm_hello.c
Linux中EXPORT_SYMBOL的用法
EXPORT_SYMBOL標(biāo)簽內(nèi)定義的函數(shù)對(duì)全部?jī)?nèi)核代碼公開(kāi),不用修改內(nèi)核代碼就可以在您的內(nèi)核模塊中直接調(diào)用。您還可以手工修改內(nèi)核源代碼來(lái)導(dǎo)出另外的函數(shù),用于重新編譯并加載新內(nèi)核后的測(cè)試。
Linux symbol export method:
[1] If we want export the symbol in a module, just use the EXPORT_SYMBOL(xxxx) in the C or H file.
And compile the module by adding the compile flag -DEXPORT_SYMTAB.
Then we can use the xxxx in the other module.
[2] If we want export some symbol in Kernel that is not in a module such as xxxx in the /arch/ppc/fec.c.
Firstly, define the xxxx in the fec.c;
Secondly, make a new file which contain the "extern" define the xxxx(for example, extern int xxxx);
Lastly, in the ppc_ksyms.c we includes the new file, and add the EXPORT_SYMBOL(xxxx).
Then we can use the xxxx