【Linux驱动】内核模块编译 —— make modules 的使用(单模块编译、多模块编译)

编译驱动一般采用的是将驱动编译成模块(.ko 文件),然后加载到内核,这其中就用到了 make modules 命令。

一、单模块编译

1、一个 c 文件编译成一个 ko 文件

下面是最简易的单文件单模块编译,假设我们要将源文件 chrdevbase.c 编译成 ko 文件。

KERNEL_DIR  := /home/pigeon/workspace/imx6ull-kernel/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
CURRENR_DIR    := $(shell pwd)
obj-m        := chrdevbase.o 
 
build: kernel_modules
kernel_modules:
    $(MAKE) -C $(KERNEL_DIR) M=$(CURRENR_DIR) modules
 
.PHONY:clean
clean:
    $(MAKE) -C $(KERNEL_DIR) M=$(CURRENR_DIR) clean

KERNEL_DIR  := xxx

KERNEL_DIR  代表内核源码的位置,后续 make modules 会用到

CURRENR_DIR    := xxx

 CURRENR_DIR  代表当前模块源文件所在的路径

obj-m    := chrdevbase.o 

obj-m 表示把文件 chrdevbase.o 作为"模块"编译,不会编译进内核,但会生成一个独立的 ko 文件

obj-y 则是直接编译进内核,即加入到内核源码中

$(MAKE)  -C  $(KERNEL_DIR)  M=$(CURRENR_DIR)  modules

这一句我们拆成两个阶段,由简入繁分析。-C $(KERNEL_DIR) 代表切换工作目录,因为内核源码顶层的Makefile文件定义了伪目标 modules,所以要先将工作目录切换到内核源码顶层 Makefile 所在位置。

make modules -C $(KERNEL_DIR)

M=$(CURRENR_DIR) 表示回到当前路径继续执行当前的Makefile。这个可以让makefile 回到自己所指定的目录下查找模块源码,将其编译,生成 ko 文件。

make modules -C $(KERNEL_DIR) M=$(CURRENR_DIR)
2、多个文件编译成一个 ko 文件

假设我们要将 add.c、sub.c 编译成一个ko文件,只有 add.c 包含了模块初始化函数(module_init),sub.c 只是add.c 的依赖源文件。大体和上面单文件单模块类似,不同之处如下:

obj-m            +=         模块名.o
模块名-objs  +=         源文件名.o ...

如:

obj-m            +=:  calc.o
calc-objs     +=   add.o  sub.o 
KERNEL_DIR  := /home/pigeon/workspace/imx6ull-kernel/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
CURRENR_DIR    := $(shell pwd)
obj-m        += calc.o        # obj-m        += 模块名.o
calc-objs   := add.o sub.o   # 模块名-objs  += 源文件名.o ...
    
build: kernel_modules
kernel_modules:
    $(MAKE) -C $(KERNEL_DIR) M=$(CURRENR_DIR) modules
 
.PHONY:clean
clean:
    $(MAKE) -C $(KERNEL_DIR) M=$(CURRENR_DIR) clean

二、多模块编译(多文件多模块)

从上面《多文件单模块》可以了解到 obj-m 指定最终模块名,<modules_name>-objs 指定依赖源文件列表。所以如果要生成多个模块,那就需要通过 obj-m 指定,而且最终模块名要和源文件名对应。

KERNEL_DIR  := /home/pigeon/workspace/imx6ull-kernel/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
CURRENR_DIR    := $(shell pwd)
 
obj-m += hello.o  calculate.o     # 需要存在对应的 hello.c 和 calculate.c 文件
                                 
build: kernel_modules
kernel_modules:
    $(MAKE) -C $(KERNEL_DIR) M=$(CURRENR_DIR) modules
 
.PHONY:clean
clean:
    $(MAKE) -C $(KERNEL_DIR) M=$(CURRENR_DIR) clean


原文链接:https://blog.csdn.net/challenglistic/article/details/131803720

模块一般用来支持那些不经常使用的功能。例如,通常情况下你仅使用拨号网络,因此网络功能并不是任何时候都需要的,那么就应该使用可装入的模块来提供这个功能。仅在你进行拨号联接的时候,该模块才被装入。而在你断掉连接的时候它会被自动卸下。这样会使内核使用内存的量最小,减小系统的负荷。 当然,那些象硬盘访问这样时时刻刻都需要的功能,则必须作在内核里。如果你搭一台网络工作站或 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 。完成之后,就编译出一个没有调入多余模块的一个“干净的”内核映像文件了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值