在装载内核模块时,用户可以向模块传递参数,形式为:insmod/modprobe 模块名 参数名 = 参数值。
如果不传递,参数将使用模块内定义的缺省值。
我们可以使用以下方法为模块定义一个参数:
module_param(参数名,参数类型,参数读/写权限);
如下代码定义了一个整型参数:
static int num = 4000 ;
module_param(num, int , S_IRUGO);
参数可以是以下类型:
byte
short
ushort
int
uint
long
ulong
char(字符指针)
bool
invbool(布尔的反,invbool 类型颠倒了值, 所以真值变成 false,)
在模块被编译时会将module_param中声明的类型与变量定义的类型进行比较,判断是否一致。
模块被加载后,在/sys/module/目录下将出现以此模块命名的目录。
当“参数读/写权限”为0时,表示此参数不存在sysfs文件系统下对应的文件节点。
当“参数读/写权限”不为0时,在此模块的目录下还将出现parameters目录,包含一系列以参数名命名的文件节点,这些文件的权限值就是传入module_param( )的“参数读写权限”,而这些文件的内容为参数的值。
除此之外,模块也可以拥有参数数组,形式为“module_param_array(数组名,数组类型,数组长,参数读/写权限)”。
从2.6.0~2.6.10版本,需要将数组长变量名赋给“数组长”。从2.6.10版本开始,需将数组长变量的指针赋给“数组长”,当不需要保存实际输入的数组元素个数时,可以设置“数组长”为NULL。
运行insmod或modprobe命令时,使用逗号分割输入的数组元素。
一下是实验例子:
- #include <linux/init.h>
- #include <linux/module.h>
- #include <linux/moduleparam.h>
-
- MODULE_LICENSE("Dual BSD/GPL");
- char *msg_buf = "Hello world!";
- int n_arr[] = {1,2,3,4,5};
- int n = 7;
-
- module_param_array(n_arr, int, &n, S_IRUSR);
- module_param(msg_buf, charp, S_IRUSR);
-
- static __init int hello_init(void)
- {
- int i;
- printk("%s/n", msg_buf);
- for (i=0; i<n; i++)
- {
- printk("n_arr[%d]=%d/n", i, n_arr[i]);
- }
- return 0;
- }
-
- static __exit void hello_exit(void)
- {
- printk("Goodbye, kernel!/n");
- }
- module_init(hello_init);
- module_exit(hello_exit);
Makefile文件:
- ### Makefile
- obj-m := hello.o
- KERNEL_DIR := /lib/modules/$(shell uname -r)/build
- PWD := $(shell pwd)
-
- default:
- $(MAKE) -C $(KERNEL_DIR) M=$(PWD) modules
- clean:
- $(MAKE) -C $(KERNEL_DIR) M=$(PWD) clean
运行命令:
sudo insmod hello.ko msg_buf=veryCD
然后使用dmesg可以查看到printk的输出:
[35983.685059] veryCD
[35983.685067] n_arr[0]=1
[35983.685072] n_arr[1]=2
[35983.685075] n_arr[2]=3
[35983.685079] n_arr[3]=4
[35983.685083] n_arr[4]=5
[35983.685087] n_arr[5]=7
[35983.685091] n_arr[6]=1
可以看出,实现n_arr的长度应该为5,而n为7,驱动里面并没有检测出,n_arr[5],n_arr[6]已经越界了,,,
但是insmod的时候却是会检测n_arr的长度的,,
输入命令:
sudo insmod hello.ko msg_buf=veryCD n_arr=1,2,3,4,5,6
但是提示出错了,:
insmod: error inserting 'hello.ko': -1 Invalid parameters
因为n_arr的数组长度为5,当输入的数组长度小于等于5的时候,insmod可以加载模块成功,
sudo insmod hello.ko msg_buf=veryCD n_arr=1,2,3
[36315.732903] veryCD
[36315.732908] n_arr[0]=1
[36315.732909] n_arr[1]=2
[36315.732911] n_arr[2]=3
可以看出,module_param_array中的nump的值为实际的输入数组参数长度。
然后当使用insmod加载内核模块,并传递数据参数的时候,系统会自动检测数组的长度,当输入的数组长度小于模块的数组长度是,insmod才会成功。