第二章 模块参数
有些模块在加载的时候可能需要根据不同的参数来产生不同的功能。这些参数的值可以直接由insmod或者modprobe命令在加载的时候指定。当然,你的代码在使用模块参数前,必须定义这些参数,并让这些参数可用。一个使用了模块参数的设备驱动模块源码如下,这个代码是在原来最基础的hello world的基础上修改的:
1 #include <linux/init.h>
2 #include <linux/module.h>
3 MODULE_LICENSE("Dual BSD/GPL");
4 static char *whom = "world";
5 static int howmany = 1;
6 module_param(howmany, int, S_IRUGO);
7 module_param(whom, charp, S_IRUGO);
8 static int hello_init(void)
9 {
10 int i;
11 for(i=0;i<=howmany;i++)
12 {
13 printk(KERN_ALERT "Hello %c\n",whom);
14 }
15 return 0;
16 }
17 static void hello_exit(void)
18 {
19 printk(KERN_ALERT "Goodbye, world\n");
20 }
21 module_init(hello_init);
22 module_exit(hello_exit);
将上面的模块编译,然后只需按照insmod (hello.ko的路径) howmany=X whom=X的格式加载设备驱动模块就行了,输出的效果的话,从代码中就能很明显的看出了,跟设置的参数有关,输出howmany次“hello 'whom'”。
module_param宏相当于把模块参数声明,第一个参数是要声明的参数,第二个是参数类型,第三个是权限。
附带说一点我自己在实现时遇到的一个问题,如果我想往常一样写循环 for(int i=0;i<=howmany;i++)在编译的时候就会出现loop initial declarations are only allowed in C99 mode的错误。把循环标志位拿出来定义之后,问题就解决了。看了下linux的源码,基本上也是把循环标志的定义拿出来的,看来linux下的c编写还是要跟通用的c标准分开。
这样,一个支持模块参数的helloworld就有了。当然,一个真正的设备驱动模块是不可能仅仅这么简单的,helloworld实例只包括了最根本的初始化函数和退出函数,其他一些函数如模块的打开,关闭,读取,写入,通信等,以后会慢慢提及的。