Makefile变量替换显然没有完成,即使:=在声明中使用

Sha*_*baz 4 linux variables makefile kbuild

我有一个主内核模块,其他内核模块与之通信.我已经构建了这样的模块(概念上):

main module/
           |
            \drivers/
                    |
                    |\driver1
                    |\driver2
                     \driver3
Run Code Online (Sandbox Code Playgroud)

由于这些是内核模块,我需要像这样编译它们:

make -C /lib/modules/$(shell uname -r)/build M=$(shell pwd) modules
Run Code Online (Sandbox Code Playgroud)

但是,由于可以从以前的目录调用驱动程序的Makefile,我需要$(shell pwd)在调用其他make之前执行(linux的make).所以Makefile现在看起来像这样:

CURRENT_DIR := $(shell pwd)

.PHONY: all
all:
    $(MAKE) -C /lib/modules/$(shell uname -r)/build M=$(CURRENT_DIR) modules
Run Code Online (Sandbox Code Playgroud)

到目前为止它很好,它完美地工作.问题是:我有一个驱动程序需要包含的文件,所以我必须给出包含路径.我第一次尝试

EXTRA_CFLAGS += -I../..
Run Code Online (Sandbox Code Playgroud)

并立即明白为什么它不起作用(相对路径将是/ lib/module/...不是当前目录).所以我改成了:

MAIN_MODULE_HOME := $(CURRENT_DIR)/../..
EXTRA_CFLAGS += -I$(MAIN_MODULE_HOME)
Run Code Online (Sandbox Code Playgroud)

奇怪的是,这不起作用!如果我写

EXTRA_CFLAGS += -Ipath/I/get/from/pwd/../..
Run Code Online (Sandbox Code Playgroud)

手动,它编译!谁能解释我做错了什么?在调用make之前,我echo编辑$(CURRENT_DIR)并且$(MAIN_MODULE_HOME)变量是有意义的.

我知道,EXTRA_CFLAGS是不是马上进行评估,但由于CURRENT_DIRMAIN_MODULE_HOME与声明:=我不明白事情是如何得到搞砸.

(如果有人能更好地说出问题标题,请做!)

ale*_*der 8

您应该通过EXTRA_CFLAGS来制作如下:

$(MAKE) -C /lib/modules/$(shell uname -r)/build M=$(CURRENT_DIR) \
           EXTRA_CFLAGS="$(EXTRA_CFLAGS)" modules
Run Code Online (Sandbox Code Playgroud)

更新:

读取driver1/Makefile的内容两次:第一次 - 当你make在driver1目录中运行时,第二次 - 由Kbuild系统运行.

首先,CURRENT_DIR := $(shell pwd)评估类似的东西/home/users/.../main module/drivers/driver1.其次,Kbuild评估CURRENT_DIR := $(shell pwd)类似的东西/usr/src/linux-headers-2.6.32-33-generic/

LDD3,ch2,p24中描述了这种情况

诀窍是按如下方式编写makefile:

# If KERNELRELEASE is defined, we've been invoked from the
# kernel build system and can use its language.
ifneq ($(KERNELRELEASE),)
    obj-m := hello.o
# Otherwise we were called directly from the command
# line; invoke the kernel build system.
else
    KERNELDIR ?= /lib/modules/$(shell uname -r)/build
    PWD  := $(shell pwd)
default:
    $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
#endif
Run Code Online (Sandbox Code Playgroud)