insmod 内核模块时参数传递 应用篇

时间:2022-06-26 15:24:05

                                                                                                                内核模块参数传递

 

   在看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

staticint int_var= 0;
staticconstchar*str_var="default";
staticint 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类型的地址


 

 

staticint __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;
}

staticvoid __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的时候可以控制参数的传递。