内核模块参数传递
在看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的时候可以控制参数的传递。