1. 现象描述
Linux系统中,动态加载的模块往往需要实现:
1)在加载模块时,向模块传递一个参数值,且该参数值在模块运行过程中不能对其进行修改;
2)在加载模块时,向模块传递一个参数值,且该参数值在模块运行过程中根据需求对其进行动态修改。
2. 关键过程
1、Linux系统2.6内核下,可以通过宏module_param(name, type, perm)将参数name声明为模块参数,该参数值可以在加载模块时指定,否则为模块内定义的缺省值;
2、我们需要关注perm,perm表示此参数在sysfs文件系统中所对应的文件节点的属性:
1)当perm为0时,表示此参数不存在在sysfs文件系统下对应的文件节点;
2)模块被加载后,在/sys/module/目录下将出现以此模块名命名的目录;
3)如果此模块存在perm不为0的命令行参数,在此模块的目录下将出现parameters目录,包含一系列以参数命名的文件节点;
4)这些文件的权限值等于perm,文件的内容为参数的值。
3、perm有四种不同属性的模块参数:
1)可读写参数:此参数可以实时查看和动态修改其值;其属性为S_IRUGO|S_IWUSR;
2)可读参数:此参数只能实时查看,不能动态修改;其属性为S_IRUGO;
3)可写参数:此参数不能实时查看,但可以动态修改;其属性为S_IWUSR;
4)不可读写参数:此参数不能在sysfs文件系统中显示,也不能实时查看和动态修改其值;其属性值为0;
由此我们可以通过设置不同的perm值,实现不同参数不同的控制。
3. 案例实现
3.1. Demo说明
1、demo程序定义了2各模块参数,分别为uid、period。加载模块时,可以向模块传递这两个参数的值,否则这两个参数值将为默认值;模块运行过程中,两者均可被用户、用户所在组及其他组读取,后者还可被用户修改。
2、demo程序使用到的perm权限值含义:
S_IRUGO:S_IRUSR|S_IRGRP|S_IROTH,表示可以被用户、用户所在组及其他组读取且只读;
S_IWUSR:表示可以被用户(能动态加载模块的用户必须是超级用户)修改。
3、demo程序根据uid和period打印不同的信息;当period值变化时,会再次打印信息。
4、demo程序使用说明如下:
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。
(责任编辑:IT) |