关于linux内核模块编程时,多个源代码文件Makefile书写的问题

本文转自:https://blog.youkuaiyun.com/u010560290/article/details/44647683

在学习内核模块编程的时候遇到了一些由于Makefile书写不正确导致的问题。一个.c源文件的Makefile按照网上的大部分资料介绍那样是没有问题的,多个源文件的内核模块编程时,就出现问题了,自己纠结了半天,最后查资料、请教大神才搞定的。把过程和解决方法贴出来,供大家参考,一起学习。

网上有很多文章讲解内核模块编程的例子,例如下面这个简单的例子:
 

#hello.c
#ifndef __KERNEL__
        #define __KERNEL__
#endif //__KERNEL__
 
#ifndef MODULE
        #define MODULE
#endif //MODULE
 
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
 
static __init int hello_init(void)
{
        printk(KERN_INFO"hello linux kernel module ...\r\n");
        return 0;
}
 
static __exit void hello_exit(void)
{
        printk(KERN_INFO"goodbye linux kernel module ...\r\n");
        return 0;
}
 
module_init(hello_init);
module_exit(hello_exit);
 
MODULE_LICENSE("GPL");

其Makefile文件编写如下:

ifneq ($(KERNELRELEASE),)
        obj-m += hello.o 
else
        CURRENT_PATH:=$(shell pwd)
        VERSION_NUM:=$(shell uname -r)
        LINUX_PATH:=/usr/src/linux-headers-$(VERSION_NUM)
 
all:
        make -C $(LINUX_PATH) M=$(CURRENT_PATH) modules
 
clean:
        make -C $(LINUX_PATH) M=$(CURRENT_PATH) clean
endif

这是一个.c源文件的情况,还是很方便的。这中书写方式中,obj-m += hello.o这里的hello.o一定要是hello.c源文件的文件名(例如:你的源文件是abc.c,那么你的Makefile中就应该写:obj-m += abc.o),否则当你make的时候会报如下错误:

*** No rule to make target`/home/share/filter/test/mod.c', needed by`/home/share/filter/test/mod.o'.  Stop.

这里的mod.o就是我随意命名的模块名,导致报错,因为当前目录下没有mod.c。

可不可以给自己的模块命名呢?当然

         同样已hello.c为例,编写随意命名模块名的Makefile,简单的指定模块建立所依赖的目标文件,其Makefile如下:

ifneq ($(KERNELRELEASE),)
        obj-m += ABC.o
        ABC -objs:=hello.o 
else
        CURRENT_PATH:=$(shell pwd)
        VERSION_NUM:=$(shell uname -r)
        LINUX_PATH:=/usr/src/linux-headers-$(VERSION_NUM)
 
 
all:
        make -C $(LINUX_PATH) M=$(CURRENT_PATH) modules
 
clean:
        make -C $(LINUX_PATH) M=$(CURRENT_PATH) clean
 
endif

这里增加了一行代码ABC -objs:=hello.o,用以指定建立ABC.o所依赖的目标文件hello.o,这样我们就可以随便给自己的模块命名了。

         我推荐第二中Makefile的书写方式,不只是因为这样可以随意命名模块名称,还与多个源文件编译一个内核模块的问题有关。

         第一种Makefile的书写方式仅能编译一个源代码文件,因为make时,默认去依赖与obj-m值相同的名称的源代码文件进行编译。但是如果当你的模块要处理的问题相对复杂的时候你可能想把你的源代码放在多个源文件中,并建立.h文件来管理。比如有如下源文件:hello.c hello1.c hello1.h;

         如果按照第一种Makefile的书写方式,可能会写成这样:obj-m += hello.o hello1.o。这样书写的Makefile在make阶段是不会报错的,但是当你要insmod ./hello.ko的时候就会如下错误:

insmod: ERROR: could not insert module./hello.ko: Unknown symbol in module

解决的方法便是如第二种Makefile书写方式一样,将模块依赖目标文件用objs一一列举出来,完整的多个源代码文件的Makefile文件编写如下:
 

ifneq ($(KERNELRELEASE),)
        obj-m += ABC.o
        ABC -objs:=hello.o hello1.h
else
        CURRENT_PATH:=$(shell pwd)
        VERSION_NUM:=$(shell uname -r)
        LINUX_PATH:=/usr/src/linux-headers-$(VERSION_NUM)
 
 
all:
        make -C $(LINUX_PATH) M=$(CURRENT_PATH) modules
 
clean:
        make -C $(LINUX_PATH) M=$(CURRENT_PATH) clean
 
endif

 

模块一般用来支持那些不经常使用的功能。例如,通常情况下你仅使用拨号网络,因此网络功能并不是任何候都需要的,那么就应该使用可装入的模块来提供这个功能。仅在你进行拨号联接的候,该模块才被装入。而在你断掉连接的候它会被自动卸下。这样会使内核使用内存的量最小,减小系统的负荷。 当然,那些象硬盘访问这样刻刻都需要的功能,则必须作在内核里。如果你搭一台网络工作站或 web服务器,那么网络功能是刻都需要的,你就应该考虑把网络功能编译到内核里。另外一个方法是在启动的候就装入网络模块。这种方法的优点是你不需要重新编译内核。而缺点是网络功能不能特别高效。 按照以上的原则,我们首先列出一张清单,看看 kernel 中哪些选项是非有不可的,也就是说,这些东西是必须被编译到内核中的。将那些非必需的模块剔除到内核以外。 第一个是root所在的硬盘配置。 哪果您的硬盘是IDE接口,就把 ide 的选项标记下来。如果是SCSI接口,请把您的接口参数及 SCSI id 记标下来。 第二个是选择使用哪一个文件系统。 Linux的默认文件系统是是 ext2 ,那么就一定要把它标记下来。如果机器中还其它的操作系统,如win98或windows NT,您还会可能选择FAT32或NTFS的支持,不过后面你可以通过手工加载的方式来加入新的模块支持。 第三个是选择Linux所支持的可执行文件格式。这里有两种格式可供选择: elf:这是当前Linux普遍支持的可执行文件格式,必须编译到内核中 。 a.out: 这是旧版的Linux的可执行文件各函数库的格式,如果你确认肯定用不到这种格式的可执行文件,那么就可以不把它编译到内核当中。 以上这些内容,是必须要编译到内核中的。其它的内容凡是所有选项中m提示的,都选择m,这样可以通过手工的方式添加该模块。 ** Loadable module support*Enable loadable module support (CONFIG_MODULES) [Y/n/?]Set version information on all symbols for modules (CONFIG_MODVERSIONS) [N/y/?]Kernel daemon support (e.g. autoload of modules) (CONFIG_KERNELD) [Y/n/?] 分别回答 Y,N,Y 。其中 CONFIG_KERNELD 的 default 值是 N, 所以要注意选择Y。 make config 完后,仍旧是 make dep; make clean。 接下来要 make zlilo 或 make zImage。 然后 make modules ; make modules_install 。完成之后,就编译出一个没有调入多余模块的一个“干净的”内核映像文件了。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值