Linux下使用内核源码单独编译某一模块

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)MakefileM=(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
root@ubuntu:/opt/gopath/src/github.com/kata-containers/packaging/kernel/kata-linux-5.4.60-89/drivers/android# make   CONFIG_ANDROID_BINDER_IPC=m  -C  /opt/gopath/src/github.com/kata-containers/packaging/kernel/kata-linux-5.4.60-89   M=/opt/gopath/src/github.com/kata-containers/packaging/kernel/kata-linux-5.4.60-89/drivers/android
make: Entering directory '/opt/gopath/src/github.com/kata-containers/packaging/kernel/kata-linux-5.4.60-89'
  CC       kata-linux-5.4.60-89/drivers/android/binderfs.o
  CC      /kernel/kata-linux-5.4.60-89/drivers/android/binder_alloc_selftest.o
  AR       kernel/kata-linux-5.4.60-89/drivers/android/built-in.a
  CC [M]   kata-linux-5.4.60-89/drivers/android/binder.o
  CC [M]   packaging/kernel/kata-linux-5.4.60-89/drivers/android/binder_alloc.o
cat: / kernel/kata-linux-5.4.60-89/drivers/android/modules.order: No such file or directory
  Building modules, stage 2.
  MODPOST 0 modules
sed: can't read  packaging/kernel/kata-linux-5.4.60-89/drivers/android/modules.order: No such file or directory
cat: /opt/gopath/src/github.com/kata-containers/packaging/kernel/kata-linux-5.4.60-89/drivers/android/modules.order: No such file or directory
make: Leaving directory '/ /packaging/kernel/kata-linux-5.4.60-89'

不支持modulers

make modules

The present kernel configuration has modules disabled.
Type 'make config' and enable loadable module support.
Then build a kernel with module support enabled.

Makefile:1356: recipe for target 'modules' failed
make: *** [modules] Error 1

Linux-Kconfig总结与分析

CONFIG宏变量参数

  • bool:      表示该CONFIG宏只能选择y(编译内核)或者n(不编译),不能选择m(编译为模块)
  • tristate:  表示该CONFIG宏可以设置y/m/n三种模式(tristate)
  • string:    表示该CONFIG宏可以设为一串字符,比如#define CONFIG_XXX "config test"
  • hex:       表示该CONFIG宏可以设为一个十六进制,比如#define CONFIG_XXX 0x1234
  • int:         表示该CONFIG宏可以设为一个整数,比如#define CONFIG_XXX 1234
menuconfig MY_SYMBOL_TEST    #生成一个菜单宏项
bool "MY_SYMBOL_TEST"
default y

config MY_SYMBOL1
bool "my symbol is bool"
default y
depends on MY_SYMBOL_TEST
config MY_SYMBOL2
tristate "my symbo2 is tristate"
default m
depends on MY_SYMBOL_TEST    

config MY_SYMBOL3
string "my symbo3 is string"
default "test symbo3"
depends on MY_SYMBOL2 && MY_SYMBOL_TEST

config MY_SYMBOL4
hex "my symbo4 is hex"
range 0 0x2000                  #设置hex区间
default 0x1234
depends on MY_SYMBOL2 && MY_SYMBOL_TEST

config MY_SYMBOL5
int "my symbo5 is int" 
range 0 2000                     #设置int区间
default 1234
depends on MY_SYMBOL2 && MY_SYMBOL_TEST

效果如下所示:

新手上路:内核模块入门

从最初学习使用Linux OS,到学习Linux内核,再到自己编写内核模块,顺利实现模块的装载和卸载,这是一个非常有趣的过程。下面我将内核模块的学习内容和大家分享,将学习Linux内核的快乐简单的传递。

构造和运行模块的过程

模块源代码 hds.c文件

# include <linux/module.h>  				//任何模块都必须包含,定义了可动态加载到内核的模块所需要的必要信息
# include <linux/init.h>    				//必须包含,包含了宏__init(指定初始化函数)和__exit(指定清除函数)
# include <linux/kernel.h>  				//里面包含常用的内核API,例如内核打印函数printk()
static int __init hds_init(void)    		//__init将函数hds_init()标记为初始化函数,在模块被装载到内核时调用hds_init()
{
    int sum = 0;
	int i;
    for(i = 1; i < 11; i++)		   			//函数功能为1-10累加求和
		sum +=i;
	printk(KERN_CRIT "Hello kernel
");		//注意末尾不要忘记加换行
,否则打印会出现某些小的错误
	printk(KERN_ALERT "sum is %d
",sum);   
                      //打印级别设为<1>,将求和结果立即打印,可以在插入模块后,在用户态下用命令dmesg查看打印效果
	return 0;
}  
static void __exit hds_exit(void)			//清除函数,在模块被卸载之前调用
{
	printk(KERN_ALERT "Goodbye kernel
");  //在模块卸载时,将Goodbye kernel这句话打印到日志
}	
module_init(hds_init);   					//引导内核 模块从这里进来
module_exit(hds_exit);   					//引导内核 模块从这里出去
MODULE_LICENSE("GPL");   					//(必选项) 模块许可证,缺省此句,将导致内核被污染
MODULE_AUTHOR("hds");	 					//(可选项) 描述模块作者
MODULE_DESCRIPTION("for fun");  			//(可选项) 描述模块功能 

Makefile文件

obj-m:=hds.o						#根据make的自动推导原则,make会自动将源程序hds.c编译成目标程序hds.o。
                                    #所有在配置文件中标记为-m的模块将被编译成可动态加载进内核的模块。即后缀为.ko的文件。
CURRENT_PATH:=$(shell pwd)  		#参数化,将模块源码路径保存在CURRENT_PATH中
LINUX_KERNEL:=$(shell uname -r) 	#参数化,将当前内核版本保存在LINUX_KERNEL中
LINUX_KERNEL_PATH:=/usr/src/linux-headers-$(LINUX_KERNEL) 	
                                    #参数化,将内核源代码的绝对路径保存在LINUX_KERNEL_PATH中
all:
	make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules 	#编译模块
clean:
	make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) clean  	#清理

编译模块

$ make      

为方便在当前终端查看日志打印信息,在装载模块前输入此命令

$ tail -f /var/log/kern.log &

装载模块

$ sudo insmod hds.ko

查看装载的模块

$ lsmod

卸载模块

$ sudo rmmod hds

查看模块是否已卸载

$ lsmod
原文地址:https://www.cnblogs.com/dream397/p/13984263.html