如何在加载模块时向模块传递参数值

1. 现象描述

Linux系统中,动态加载的模块往往需要实现:

1)在加载模块时,向模块传递一个参数值,且该参数值在模块运行过程中不能对其进行修改;

2)在加载模块时,向模块传递一个参数值,且该参数值在模块运行过程中根据需求对其进行动态修改。

2. 关键过程

1Linux系统2.6内核下,可以通过宏module_param(name, type, perm)将参数name声明为模块参数,该参数值可以在加载模块时指定,否则为模块内定义的缺省值;

2、我们需要关注permperm表示此参数在sysfs文件系统中所对应的文件节点的属性:

1)当perm0时,表示此参数不存在在sysfs文件系统下对应的文件节点;

2)模块被加载后,在/sys/module/目录下将出现以此模块名命名的目录;

3)如果此模块存在perm不为0的命令行参数,在此模块的目录下将出现parameters目录,包含一系列以参数命名的文件节点;

4)这些文件的权限值等于perm,文件的内容为参数的值。

3perm有四种不同属性的模块参数:

1)可读写参数:此参数可以实时查看和动态修改其值;其属性为S_IRUGO|S_IWUSR

2)可读参数:此参数只能实时查看,不能动态修改;其属性为S_IRUGO

3)可写参数:此参数不能实时查看,但可以动态修改;其属性为S_IWUSR

4)不可读写参数:此参数不能在sysfs文件系统中显示,也不能实时查看和动态修改其值;其属性值为0

由此我们可以通过设置不同的perm值,实现不同参数不同的控制。

3. 案例实现

3.1. Demo说明

1demo程序定义了2各模块参数,分别为uidperiod。加载模块时,可以向模块传递这两个参数的值,否则这两个参数值将为默认值;模块运行过程中,两者均可被用户、用户所在组及其他组读取,后者还可被用户修改。

2demo程序使用到的perm权限值含义:

S_IRUGOS_IRUSR|S_IRGRP|S_IROTH,表示可以被用户、用户所在组及其他组读取且只读;

S_IWUSR:表示可以被用户(能动态加载模块的用户必须是超级用户)修改。

3demo程序根据uidperiod打印不同的信息;当period值变化时,会再次打印信息。

4demo程序使用说明如下:

1)加载模块时:insmod demo.ko uid=0 period =0

2)模块运行过程中,读取uid值:cat /sys/module/demo/parameters/uid

3) 模块运行过程中,修改period值:echo 1 > /sys/module/demo/parameters/period

3.2. 源文件

 #include <linux/kernel.h> #include <linux/types.h> #include <linux/spinlock.h> #include <linux/interrupt.h> #include <linux/delay.h> #include <linux/blkdev.h> #include <linux/list.h> #include <linux/timer.h> #include <linux/version.h> #include <linux/moduleparam.h> #include <linux/device.h> #include <linux/module.h> #include <linux/string.h> #include <linux/dma-mapping.h> #include <asm/semaphore.h> #include <asm/page.h> typedef uint8_t bool_t; #define MORNING 0 #define AFTERNOON 1 #define EVENING 2 #define UNKNOWN 3 #define FALSE 0 #define TRUE 1 #define USER_NUM 3 /******** module information ***************/ MODULE_AUTHOR("H3C Corporation"); MODULE_DESCRIPTION("Demo function"); MODULE_LICENSE("GPL"); /******** module parameters ***************/ static uint uid = 0; module_param(uid, uint, S_IRUGO); static uint period = MORNING; module_param(period, uint, S_IRUGO|S_IWUSR); /******** global parameters ***************/ char user_name[USER_NUM][20] = {{"wuwu"}, {"pikaqiu"}, {"lala"}}; uint prev_period = UNKNOWN; bool_t demo_quit = FALSE; struct completion com_demo; int demo_thread(void* unuse) { daemonize("demo_thread"); for(;;) { if(demo_quit) { break; } if(prev_period != period) { if(MORNING == period) { printk(KERN_EMERG "Good morning, %s/n", user_name[uid]); } else if(AFTERNOON == period) { printk(KERN_EMERG "Good afternoon, %s/n", user_name[uid]); } else if(EVENING == period) { printk(KERN_EMERG "Good evening, %s/n", user_name[uid]); } else { printk(KERN_EMERG "Parameter period is invalid! Please choose from follows:/n"); printk(KERN_EMERG "0: means morning/n"); printk(KERN_EMERG "1: means afternoon/n"); printk(KERN_EMERG "2: means evening/n"); printk(KERN_EMERG "/n"); } prev_period = period; } msleep(10); } complete_and_exit(&com_demo, 0); } static int __init demo_init(void) { pid_t pid = 0; uint32_t i = 0; if(uid >= USER_NUM) { printk(KERN_EMERG "Parameter uid is invalid, please choose from follows:/n"); for(i=0; i<USER_NUM; i++) { printk(KERN_EMERG "i: named %s/n", user_name[i]); } printk(KERN_EMERG "/n"); return -1; } init_completion(&com_demo); pid = kernel_thread(demo_thread, NULL, CLONE_KERNEL); if(pid < 0) { printk(KERN_EMERG "demo thread create failed!/n"); return -1; } printk(KERN_EMERG "Welcome to the demo!/n"); return 0; } static void __exit demo_exit(void) { demo_quit = TRUE; wait_for_completion(&com_demo); printk(KERN_EMERG "Byebye, %s/n", user_name[uid]); return; } module_init(demo_init); module_exit(demo_exit);

 

4. 经验总结

1、根据对不同用户权限要求设置perm位;

2、对于只需在加载模块时传递参数值、而不希望用户看到该参数,建议将perm位置为0

你可能感兴趣的:(thread,linux,Module,user,Parameters,h3c)