內核模塊參數傳遞
在看omap vout.c 的模塊時候發現了一些關於LCD和輸出的參數是由u-boot中的命令行傳遞而來。分析一下內核模塊的參數傳遞問題。究竟是如何實現的。
以下內容來自於:
http://blog.21ic.com/user1/5593/archives/2010/66420.html
對於如何向模塊傳遞參數,Linux kernel 提供了一個簡單的框架。其允許驅動程序聲明參數,並且用戶在系統啓動或模塊裝載時爲參數指定相應值,在驅動程序裏,參數的用法如同全局變量。
使用下面的宏時需要包含頭文件<linux/moduleparam.h>。
#define module_param_named(name, value, type, perm) \
param_check_##type(name, &(value)); \
module_param_call( name,
param_set_##type,
param_get_##type,
&value,
perm); \
__MODULE_PARM_TYPE(name, #type)
#define module_param(name, type, perm) \
module_param_named(name, name, type, perm)
下面介紹若干種模塊參數申明和使用的方法。
通過宏module_param()定義一個模塊參數:
module_param(name,type,perm);
name既是用戶看到的參數名,又是模塊內接受參數的變量。type表示參數的數據類型。
如:
byte, short, ushort,int, uint, long, ulong, charp, bool, invbool;
perm指定了在sysfs中相應文件的訪問權限。訪問權限與linux文件愛你訪問權限相同的方式管理,如0644,或使用stat.h中的宏如S_IRUGO表示。0表示完全關閉在sysfs中相對應的項。
這些宏不會聲明變量,因此在使用宏之前,必須聲明變量,典型
地用法如下:
static unsigned int int_var = 0;
module_param(int_var, uint, S_IRUGO);
這些必須寫在模塊源文件的開頭部分。即int_var是全局的。也可以使模塊源文件內部的變量名與外部的參數名有不同的名字,通過
module_param_named()定義。
module_param_named(name, variable, type, perm);其中name是外部可見的參數名,variable是源文件內部的全局變量名,而module_param通過module_param_named實現,只不過name與variable相同。
例如:
static unsigned int max_test = 9;
module_param_name(maximum_line_test, max_test, int, 0);
如果模塊參數是一個字符串時,通常使用char類型定義這個模塊參數。內核複製用戶提供的字符串到內存,並且相對應的變量指向這個字符串。
例如:
static char *name;
module_param(name, charp, 0);
另一種方法是通過宏module_param_string()讓內核把字符串直接複製到程序中的字符數組內。
module_param_string(name, string, len, perm);
這裏,name是外部的參數名,string是內部的變量名,len是以string命名的buffer大小(可以小於buffer的大小,但是沒有意義),perm表示sysfs的訪問權限(或者perm是零,表示完全關閉相對應的sysfs項)。
例如:
static char species[BUF_LEN];
module_param_string(specifies, species, BUF_LEN, 0);
如果需要傳遞多個參數可以通過宏module_param_array()實現。
module_param_array(name, type, nump, perm);
其中,name既是外部模塊的參數名又是程序內部的變量名,type是數據類型,perm是sysfs的訪問權限。指針nump指向一個整數,其值表示有多少個參數存放在數組name中。值得注意是name數組必須靜態分配。
例如:
static int finsh[MAX_FISH];
static int nr_fish;
module_param_array(fish, int, &nr_fish, 0444); 最終傳
遞數組元素個數存在nr_fish中通過宏module_param_array_named()使得內部的數組名與外部的參數名有不同的名字。
例如:
module_param_array_named(name, array, type, nump, perm);
通過宏MODULE_PARM_DESC()對參數進行說明:
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.” );
試驗:
下面的小模塊具體操作了如何傳遞參數的方法,麻雀雖小,五臟俱全。
在pc機上也可以試驗,並且驗證效果。
#
include
<
linux
/
module
.
h
>
#
include
<
linux
/
moduleparam
.
h
>
#
include
<
linux
/
kernel
.
h
>
#
define
MAX_ARRAY 6
static
int
int_var
=
0
;
static
const
char
*
str_var
=
"default"
;
static
int
int_array
[
6
];
int
narr
;
//模塊傳遞變量和描述module_param
(
int_var
,
int
,
0644
);
MODULE_PARM_DESC
(
int_var
,
"A
integer variable"
);
//模塊傳遞變量和描述module_param
(
str_var
,
charp
,
0644
);
MODULE_PARM_DESC
(
str_var
,
"A
string variable"
);
//模塊傳遞數組合描述module_param_array
(
int_array
,
int
,
&
narr
,
0644
);
MODULE_PARM_DESC
(
int_array
,
"A
integer array"
);
//array的傳遞時多了一個參數int類型的地址
static
int
__init hello_init
(
void
)
{
int
i
;
printk
(
KERN_ALERT
"Hello, my LKM.\n"
);
printk
(
KERN_ALERT
"int_var %d.\n"
,
int_var
);
printk
(
KERN_ALERT
"str_var %s.\n"
,
str_var
);//
這個數組參數不知道怎麼使用
for
(
i
=
0
;
i
<
narr
;
i
++){
printk
(
"int_array[%d] = %d\n"
,
i
,
int_array
[
i
]);
}
return
0
;
}
static
void
__exit hello_exit
(
void
)
{
printk
(
KERN_ALERT
"Bye, my LKM.\n"
);
}
module_init
(
hello_init
);
module_exit
(
hello_exit
);
MODULE_LICENSE
(
"GPL"
);
MODULE_AUTHOR
(
"Edward"
);
MODULE_DEION
(
"This module is a example."
);
insmod hello.ko 1 edward 00
總結:
其實在做應用層開發的時候大家多數會遇到向main函數傳遞參數的方法.比如 main(argc,argv)。內核模塊也是一樣,同樣的insmod的時候可以控制參數的傳遞。