构建一个树外的Linux内核模块,它们与导出的符号共享目标文件

Dim*_*mak 5 linux makefile kbuild

想象一个项目,需要构建两个Linux内核模块,具有以下源树布局:

modules/
|--common/
|  `--common_data.c
|--mod1/
|  `--mod1_main.c
`--mod2/
   `--mod2_main.c
Run Code Online (Sandbox Code Playgroud)

common_data.o对象文件需要包含在两个模块中.所以这个项目的Kbuild文件看起来像这样:

obj-m  := mod1.o mod2.o
mod1-y := mod1/mod1_main.o common/common_data.o
mod2-y := mod2/mod2_main.o common/common_data.o
Run Code Online (Sandbox Code Playgroud)

如果common_data.c文件中没有导出的符号(即EXPORT_SYMBOL),它构建正常.但是,如果有一些导出符号(例如some_func),那么MODPOST将产生如下警告:

WARNING: mod2: 'some_func' exported twice. Previous export was in mod1.ko
Run Code Online (Sandbox Code Playgroud)

这是因为两个模块共享单个Module.symvers文件.可以在mod1 /和mod2 /目录中创建两个单独的Kbuild文件以避免此警告,因为这将有两个单独的Module.symvers文件.但这将导致另一个问题 - commond_data.o对象将被构建两次 - 第一次编译mod1.ko时第二次编译mod2.ko.

有什么方法可以在使用单个Kbuild文件时抑制此警告,或者在使用两个单独的Kbuild文件时阻止再次重建common_data.o文件?

Dim*_*mak 0

在对 Kbuild 系统进行一些修改之后,我最终得出结论,处理这种情况的最直接方法是从make外部 makefile 的输出中过滤掉警告消息:

$(MAKE) -C $(KDIR) M=$$PWD 2>&1 \
| grep -v '^WARNING:.*exported twice\. Previous export was in'
Run Code Online (Sandbox Code Playgroud)

或者不牺牲 STDERR 并将其压缩到 STDOUT 中,但它需要bash

bash -c "$(MAKE) -C $(KDIR) M=$$PWD 2> >( grep -v '^WARNING:.*exported twice\. Previous export was in' )"
Run Code Online (Sandbox Code Playgroud)