Linux下使用内核源码单独编译某一模块
初衷:由于ubuntu1604安装版本没有 IGMP Snooping的模块,所以研究了一下此功能。
首先下载需要的内核源码,参看上一篇文章(https://blog.csdn.net/bingyu9875/article/details/95946968)
本文章只介绍如何编译某个内核的模块:
1. 查看config配置项
进入需要编译的目录,我的目录是 ./net/bridge,
sean@sean:/media/sean/b55f4db0-2560-4807-b8bf-b29a66db54e1/home/sean/work/tmp/kernel/linux-4.8/net/bridge$ ls
br.c br_fdb.o bridge.mod.c br_if.o br_ioctl.o br_multicast.o br_netfilter_ipv6.o br_netfilter.o br_nf_core.o br_stp_bpdu.c br_stp_if.o br_sysfs_br.c br_vlan.c Makefile
br_device.c br_forward.c bridge.mod.o br_input.c br_mdb.c br_netfilter_hooks.c br_netfilter.ko br_netlink.c br.o br_stp_bpdu.o br_stp.o br_sysfs_br.o br_vlan.o modules.order
br_device.o br_forward.o bridge.o br_input.o br_mdb.o br_netfilter_hooks.o br_netfilter.mod.c br_netlink.o br_private.h br_stp.c br_stp_timer.c br_sysfs_if.c built-in.o Module.symvers
br_fdb.c bridge.ko br_if.c br_ioctl.c br_multicast.c br_netfilter_ipv6.c br_netfilter.mod.o br_nf_core.c br_private_stp.h br_stp_if.c br_stp_timer.o br_sysfs_if.o Kconfig netfilter
sean@sean:/media/sean/b55f4db0-2560-4807-b8bf-b29a66db54e1/home/sean/work/tmp/kernel/linux-4.8/net/bridge$
查看Makefile,找到需要编译的文件,并确认编译的config参数,如下:
2. 编译
命令如下:
make CONFIG_BRIDGE_IGMP_SNOOPING=m -C /home/sean/kernel/linux-4.8 M=/home/sean/kernel/linux-4.8/net/bridge modules
或者
cd /home/sean/kernel/linux-4.8/net/bridge
make CONFIG_BRIDGE_IGMP_SNOOPING=m -C /home/sean/kernel/linux-4.8 M=`pwd` modules
然后手动将生成的*.ko拷贝到/lib/modules/2.6.19/kernel/对应的目录即可。(由于我需要的文件只能编译为.o,所以不需要拷贝)
运行depmod -a重新配置依赖关系,以后就可以通过modprobe fuse来加载fuse模块了。
3. make 参数的说明:
$(MAKE) -C $(KDIR) M=$(PWD) modules
-C:后面的参数为linux内核的顶层目录
M:后面的参数为需要编译文件的目录
4. 例子
1)模块Makefile
ifneq ($(KERNELRELEASE),)
obj-m := mytest.o
mytest-objs := file1.o file2.o file3.o
else
KDIR := /lib/modules/$(shell uname -r)/build
PWD := $(shell pwd)
default:
$(MAKE) -C $(KDIR) M=$(PWD) modules
endif
KERNELRELEASE 是在内核源码的顶层Makefile中定义的一个变量,在第一次读取执行此Makefile时,KERNELRELEASE没有被定义,所以make将读取执行else之后的内容。如果make的目标是clean,直接执行clean操作,然后结束。
当make的目标为modules或modules_install时,-C ( K E R N E L D I R ) 指 明 跳 转 到 内 核 源 码 目 录 下 读 取 那 里 的 M a k e f i l e ; M = (KERNELDIR)指明跳转到内核源码目录下读取那里的Makefile;M= (KERNELDIR)指明跳转到内核源码目录下读取那里的Makefile;M=(PWD) 表明然后返回到当前目录继续读入、执行当前的Makefile。
当从内核源码目录返回时,KERNELRELEASE已被被定义,kbuild也被启动去解析kbuild语法的语句,make将继续读取else之前的内容。else之前的内容为kbuild语法的语句, 指明模块源码中各文件的依赖关系,以及要生成的目标模块名。
2)Kconfig与内核Makefile
-
a. 使用Kconfig以及内核的Makefile可以在内核中添加自己的源代码,并且可以添加内核配置选项,是否编进内核,是否以模块的方式等;
-
b. 在内核某个目录的Kconfig文件中可以配置各个选项的含义;在Makefile中指定如果配置了,该如何编译;
如要在/driver/char中增加一个配置选项CONFIG_FISHING_POLE选项;
在driver/char/Kconfig文件中增加对该选项的说明:
config FISHING_POLE
tristate “简单说明” //tristate代表有三种方式,如为bool代表不能变为模块
default n //默认是否选择
help
**** //一些帮助信息
在driver/char/Makefile中增加:
obj-$(CONFIG_FISHING_POLE) += fishing.o
如果有多个源文件:
obj-$(CONFIG_FISHING_POLE) += fishing.o
fishing-objs := fishing-main.o fishing-line.o