Chinaunix首页 | 论坛 | 博客

apt

  • 博客访问: 381560
  • 博文数量: 121
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 600
  • 用 户 组: 普通用户
  • 注册时间: 2015-04-10 15:52
文章分类

全部博文(121)

文章存档

2017年(2)

2016年(23)

2015年(96)

我的朋友

分类: 其他平台

2015-08-18 14:29:08

对于如何向模块传递参数,Linux kernel 提供了一个简单的框架。其允许驱动程序声明参数,并且用户在系统启动或模块装载时为参数指定相应值,在驱动程序里,参数的用法如同全局变量。

使用下面的宏时需要包含头文件<linux/moduleparam.h>

一、过宏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中相对应的项。
    perm 字段是一个权限值,表示此参数在sysfs文件系统中所对应的文件节点的属性。你应当使用 <linux/stat.h> 中定义的值. 这个值控制谁
    可以存取这些模块参数在 sysfs 中的表示.当perm为0时,表示此参数不存在 sysfs文件系统下对应的文件节点。 否则, 模块被加载后,在
    /sys/module/ 目录下将出现以此模块名命名的目录, 带有给定的权限.。
     权限在include/linux/stat.h中有定义。比如:

点击(此处)折叠或打开

  1. #define S_IRWXU 00700
  2. #define S_IRUSR 00400
  3. #define S_IWUSR 00200
  4. #define S_IXUSR 00100
  5. #define S_IRWXG 00070
  6. #define S_IRGRP 00040
  7. #define S_IWGRP 00020
  8. #define S_IXGRP 00010
  9. #define S_IRWXO 00007
  10. #define S_IROTH 00004
  11. #define S_IWOTH 00002
  12. #define S_IXOTH 00001
    使用 S_IRUGO 参数可以被所有人读取, 但是不能改变; S_IRUGO|S_IWUSR 允许 root 来改变参数. 注意, 如果一个参数被 sysfs 修改, 你 
    的模块看到的参数值也改变了, 但是你的模块没有任何其他的通知. 你应当不要使模块参数可写, 除非你准备好检测这个改变并且因而作出反
    应.
    这些宏不会声明变量,因此在使用宏之前,必须声明变量,典型地用法如下:

点击(此处)折叠或打开

  1. static unsigned int int_var = 0;
  2. 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相同。例如:

点击(此处)折叠或打开

  1. static unsigned int max_test = 9;
  2. module_param_name(maximum_line_test, max_test, int, 0)
二、如果模块参数是一个字符串时,通常使用charp类型定义这个模块参数。内核复制用户提供的字符串到内存,并且相对应的变量指向这个 
      字符串。 例如:

点击(此处)折叠或打开

  1. static char *name;
  2. 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项)。例如:

点击(此处)折叠或打开

  1. static char species[BUF_LEN]
  2. 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数组必须静态分配。例如:

点击(此处)折叠或打开

  1. static int finsh[MAX_FISH];
  2. static int nr_fish;
  3. 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()对参数进行说明:
  

点击(此处)折叠或打开

  1. static unsigned short size = 1;

  2. module_param(size, ushort, 0644);
  3. MODULE_PARM_DESC(size, “The size in inches of the fishing pole”
  4.     “connected to this computer.)
    说明:from  http://blog.csdn.net/iczyh/archive/2008/10/26/3149727.aspx

    module_param() 和 module_param_array() 的作用就是让那些全局变量对 insmod 可见,使模块装载时可重新赋值。  
    module_param_array() 宏的第三个参数用来记录用户 insmod 时提供的给这个数组的元素个数,NULL 表示不关心用户提供的个数
    module_param() 和 module_param_array() 最后一个参数权限值不能包含让普通用户也有写权限,否则编译报错。这点可参考
    linux/moduleparam.h 中 __module_param_call() 宏的定义。

    字符串数组中的字符串似乎不能包含逗号,否则一个字符串会被解析成两个

一个测试用例:parm_hello.c

点击(此处)折叠或打开

  1. #include <linux/module.h>
  2. #include <linux/moduleparam.h>
  3. #include <linux/kernel.h>


  4. #define MAX_ARRAY 6
  5. static int int_var = 0;
  6. static const char *str_var = "default";
  7. static int int_array[6];
  8. int narr;
  9. module_param(int_var, int, 0644);
  10. MODULE_PARM_DESC(int_var, "A integer variable");


  11. module_param(str_var, charp, 0644);
  12. MODULE_PARM_DESC(str_var, "A string variable");
  13. module_param_array(int_array, int, &narr, 0644);
  14. MODULE_PARM_DESC(int_array, "A integer array");
  15.  
  16. static int __init hello_init(void)
  17. {
  18.        int i;
  19.        printk(KERN_ALERT "Hello, my LKM.\n");
  20.        printk(KERN_ALERT "int_var %d.\n", int_var);
  21.        printk(KERN_ALERT "str_var %s.\n", str_var);
  22.        for(i = 0; i < narr; i ++){
  23.                printk("int_array[%d] = %d\n", i, int_array[i]);
  24.        }
  25.        return 0;
  26. }
  27. static void __exit hello_exit(void)
  28. {
  29.        printk(KERN_ALERT "Bye, my LKM.\n");
  30. }
  31. module_init(hello_init);
  32. module_exit(hello_exit);
  33. MODULE_LICENSE("GPL");
  34. MODULE_AUTHOR("ydzhang");
  35. MODULE_DESCRIPTION("This module is a example.")
测试:insmod parm_hello.ko int_var=100 str_var=hello int_array=100,200
阅读(1053) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~