如何使顶级 Makefile 调用子目录 Makefile 中的所有目标?
我的文件夹结构是这样的
/Makefile
/src/Makefile
我在 中编写所有目标/src/Makefile
,但现在我想编写另一个/Makefile
以简化我的工作。如何编写这个顶级 Makefile ?
如何使顶级 Makefile 调用子目录 Makefile 中的所有目标?
我的文件夹结构是这样的
/Makefile
/src/Makefile
我在 中编写所有目标/src/Makefile
,但现在我想编写另一个/Makefile
以简化我的工作。如何编写这个顶级 Makefile ?
你可以写在你的 TOP makefile 使用
all:
@$(MAKE) -C src
如果你不在子目录中使用 makefile,例如,你使用 somename.mk,你可以使用
all:
@$(MAKE) -C src -f somename.mk
我给你看我的例子,我的 DIR 看起来像:
TOPDIR-- Makefile
|
|-- debug
| |-- debug.c
| |-- debug.h
| |-- debug.mk
| |-- instrument.c
| `-- uart_print.c
|-- driver
| |-- driver.c
| |-- driver_ddi.c
| |-- driver_ddi.h
| |-- driver.h
| `-- driver.mk
|-- include
| `-- common.h
|-- Makefile
|-- mw
| |-- manager.c
| `-- mw.mk
|-- root
| |-- main.c
| `-- root.mk
我的 TOP makefile 看起来像:
MAKE_DIR = $(PWD)
ROOT_DIR := $(MAKE_DIR)/root
DRV_DIR := $(MAKE_DIR)/driver
INCLUDE_DIR := $(MAKE_DIR)/include
DEBUG_DIR := $(MAKE_DIR)/debug
INC_SRCH_PATH :=
INC_SRCH_PATH += -I$(ROOT_DIR)
INC_SRCH_PATH += -I$(DRV_DIR)
INC_SRCH_PATH += -I$(INCLUDE_DIR)
INC_SRCH_PATH += -I$(DEBUG_DIR)
LIB_SRCH_PATH :=
LIB_SRCH_PATH += -L$(MAKE_DIR)/libs
COLOR_ON = color
COLOR_OFF =
CC = $(COLOR_ON)gcc
#CC = $(COLOR_OFF)gcc
LD = ld
LINT = splint
LIBS := -ldriver -ldebug -lmw -lm -lpthread
CFLAGS :=
CFLAGS += $(INC_SRCH_PATH) $(LIB_SRCH_PATH)
CFLAGS += -Wall -O -ggdb -Wstrict-prototypes -Wno-pointer-sign -finstrument-functions -fdump-rtl-expand
CFLAGS += -DDEBUG -D_REENTRANT
LDFLAGS :=
export MAKE_DIR CC LD CFLAGS LDFLAGS LIBS LINT INC_SRCH_PATH
all:
@$(MAKE) -C debug -f debug.mk
@$(MAKE) -C driver -f driver.mk
@$(MAKE) -C mw -f mw.mk
@$(MAKE) -C root -f root.mk
.PHONY: clean
clean:
@$(MAKE) -C debug -f debug.mk clean
@$(MAKE) -C driver -f driver.mk clean
@$(MAKE) -C mw -f mw.mk clean
@$(MAKE) -C root -f root.mk clean
.PHONY: lint
lint:
$(MAKE) -C debug -f debug.mk lint
它会在编译期间调用 sub DIR *.mk。sub DIR makefile,我只是写了一个简单的例子供大家参考:
LIB = $(MAKE_DIR)/libs/yourmodulename.a
SRCS = $(wildcard *.c)
OBJS = $(patsubst %.c, %.o, $(SRCS))
$(LIB): $(OBJS)
@mkdir -p ../libs
@$(AR) cr $@ $^
@echo " Archive $(notdir $@)"
$(OBJS): $(SRCS)
@$(CC) $(CFLAGS) -c $^
@echo " CC $(OBJS)"
.PHONY: clean
clean:
@$(RM) -f $(LIB) $(OBJS)
@$(RM) -f *.expand
@echo " Remove Objects: $(OBJS)"
@echo " Remove Libraries: $(notdir $(LIB))"
.PHONY: lint
lint:
$(LINT) $(INC_SRCH_PATH) $(SRCS)
对于生成目标文件的makefile有点不同,因为我使用子makefile生成LIB文件,而我使用aroot.mk
来生成目标:
PROG = ../prog/DEMO
SRCS = $(wildcard *.c)
OBJS = $(patsubst %.c, %.o, $(SRCS))
$(PROG): $(SRCS)
@mkdir -p ../prog
@$(CC) $^ $(CFLAGS) -Wl,-Map=$(PROG).map $(LIBS) -o $@
@echo " Generate Program $(notdir $(PROG)) from $^"
.PHONY: clean
clean:
@$(RM) -f $(OBJS) $(PROG)
@$(RM) -f *.expand
@$(RM) -rf ../prog ../libs
@echo " Remove Objects: $(OBJS)"
@echo " Remove Libraries: $(notdir $(PROG))"
请注意,至少有两种方法可以做到这一点。
GNU Changedir 选项
一种方法是使用 GNU Make 的特定功能,该-C
选项允许更改您的编译目录并到达另一个:
all:
make -C dir
make
手册说:
-C dir, --directory=dir Change to directory dir before reading the makefiles or doing anything else. If multiple -C options are specified, each is interpreted relative to the previous one: -C / -C etc is equivalent to -C /etc. This is typically used with recursive invocations of make.
您还可以通过调用目标目录中的特定目标来组合此选项。例如,以下目标将进入src/
目录并make
与clean
目标调用:
clean:
@rm -f *.o
make -C src/ clean
POSIX方式
GNU 做事方式的问题在于它只适用于 GNU Make,而不适用于标准 Make。如果您可能使用另一个 Make(无论出于何种原因),您最好考虑以更 POSIX 的方式进行操作。
在 POSIX Make 中,您必须更多地依赖cd
命令,如下所示:
all:
cd src/ && make
请注意,我使用&&
而不是;
. 避免对make
. 实际上,cmd1 ; cmd2
将按顺序执行cmd1
,cmd2
无论每个命令的结果是什么,cmd1 && cmd2
都将按顺序执行cmd1
,并且cmd2
只有在cmd1
返回时才会执行EXIT_SUCCESS
。在我们的例子中,假设第一个cd
失败是因为目录已被删除。然后,初始的 makefile 将在无限递归循环中一次又一次地执行。
无论如何,这种 POSIX 方式,是在子目录中下降并执行其他 Makefile 的更健壮的方式。我建议您使用它而不是依赖与 GNU Make 链接的选项。
如果你想轻松地做多个目录:
SUBDIRS=dir1 dir2
all::
@echo make all
$(foreach var,$(SUBDIRS),echo $(var): ; cd $(var)/ && make $@ && cd ..;)
clean:
@echo make clean
$(foreach var,$(SUBDIRS),echo $(var): ; cd $(var)/ && make $@ && cd ..;)