内核模块的参数传递方式很多,第一次尝试仅仅介绍了 指定普通类型和数组类型
内核模块指定参数方式 定义内核参数变量普通变量 : module_param(变量名,变量类型,访问许可掩码)
数组变量 : module_param_array(name, type,&num,perm)
关于数组长度变量num需要说一下,num保存数组的长度变量的地址,方便别的地方调用,而数组的长度由数组初始化的时候指定。
内核支持的模块参数类型
bool 布尔型
invbool 布尔型的反转, 这两种类型和int型关联,即true=0,false=1,反转布尔型是指 装载模块的时候指定0则此变量赋值为false 否则为true
charp 字符指针型,内核会为用户提供的字符串分配内存,并相应的设置指针
int long short 有符号整形变量
uint ulong ushort 无符号整型变量
访问许可掩码,也就是sysfs的访问权限,每一个模块装载后都会在 /sys/module/下保存一个以模块名命名虚拟文件系统
模块名下的/paramters/目录内保存了以变量名命名的文件 访问许可掩码即这些文件的访问权限,看一下下面的介绍
最后的 module_param 字段是一个权限值,表示此参数在sysfs文件系统中所对应的文件节点的属性。
你应当使用 <linux/stat.h> 中定义的值. 这个值控制谁可以存取这些模块参数在 sysfs 中的表示.当perm为0时,表示此参数不存在 sysfs文件系统下对应的文件节点。 否则, 模块被加载后,在/sys/module/ 目录下将出现以此模块名命名的目录, 带有给定的权限.。
权限在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 作为参数可以被所有人读取, 但是不能改变; S_IRUGO|S_IWUSR 允许 root 来改变参数. 注意, 如果一个参数被 sysfs 修改, 你的模块看到的参数值也改变了, 但是你的模块没有任何其他的通知. 你应当不要使模块参数可写, 除非你准备好检测这个改变并且因而作出反应
普通类型 :
static int howmany=0; static char * whom="world"; module_param(howmany,int,S_IRUGO); module_param(whom,charp,S_IRUGO);insmod param_test.ko howmany=10 whom="Yuanye.Ma"
数组 :
static int len; //存储数组的长度 static int buf[]={0,0,0,0,0,0,0,0,0}; module_param_array(buf,int, &len, S_IRUGO); //此处是长度的地址
insmod parm_test.ko buf=2,2,2,2 //注意,源代码中定义了九个元素,而此处输入了4个则 len=4
insmod parm_test.ko buf=2,2,2,2 //注意,源代码中定义了九个元素,而此处输入了9个则 len=9
总之,len<=输出初始化的元素个数
目标:安装内核模块的时候通过命令行指定整型数 howmany 和 字符串 whom,模块加载进入内核的时候连续输出howmany次 “hello” whom 。
编写源代码 #include <linux/module.h> #include <linux/init.h> static char * whom = "world"; static int howmany = 0; module_param(howmany,int,S_IRUGO); module_param(whom,charp,S_IRUGO); static int __init parm_test(void) { int i = 0; for(i=0; i<howmany;i++) { printk(KERN_ALERT"hello %s\n",whom); } return 0; } static void __exit parm_exit(void) { printk(KERN_ALERT"GoodBye Kernel"); } module_init(parm_test); module_exit(parm_exit); MODULE_LICENSE("GPL"); MODULE_AUTHOR("Yuanye.Ma"); MODULE_DESCRIPTION("module param test"); 编写Makefile obj-m := parm_test.o KERNEL_PATH = "/usr/src/kernels/2.6.18-8.el5-i686/" PWD = $(shell pwd) all: make -C $(KERNEL_PATH) M=$(PWD) modules clean: make -C $(KERNEL_PATH) M=$(PWD) clean 内核信息 [root@localhost mod_parm]# modinfo parm_test.ko filename: parm_test.ko description: module param test //模块描述信息 author: Yuanye.Ma //作者名 license: GPL //内核版本信息 srcversion: F27110872EE977A467DD221 depends: vermagic: 2.6.18-8.el5 SMP mod_unload 686 REGPARM 4KSTACKS gcc-4.1 parm: howmany:int //模块参数 parm: whom:charp //模块参数 安装并检查结果 [root@localhost mod_parm]# insmod parm_test.ko howmany=10 whom="Yuanye.Ma" //传入参数 [root@localhost mod_parm]# dmesg | tail GoodBye Kernel<1>hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma hello Yuanye.Ma 实验二