多级目录makefile例子

时间:2022-01-25 12:43:57

先看一下make前后的目录结构对比

多级目录makefile例子   多级目录makefile例子

makefile会自动生成需要的文件夹:各个模块下src目录下的obj、target目录

各个模块下src/obj负责存放生成的 .o 文件;

target用于存储生成的可执行文件;

lib用于存储生成的静态库(libModuleA.a)和引用的静态库文件(libtest.a);

include文件夹用于存储引用库的头文件(test.h);

module_a模块会生成一个静态库,在生成可执行文件时链接这个静态库。

a.c中是一个函数实现,a.h中声明,main.c中调用。

main函数会调用module_a生成静态库中的函数和引用库中的函数;

以下是各级makefile的实现。

最外层makefile实现:

#最外层的makefile负责创建相关目录和进入每个子目录去make

#后边里目录要根据前面一个的目录为基准
SUBDIRS = ./module_a ../main

.PHONY:all
all:
mkdir -p ./target
mkdir -p ./module_a/src/obj
mkdir -p ./main/src/obj

@list='$(SUBDIRS)';for subdir in $$list; do \
cd $$subdir && make; \
done

.PHONY:clean
clean:
@list='$(SUBDIRS)'; for subdir in $$list; do \
echo "Clean in $$subdir";\
cd $$subdir && make clean;\
done
rm -rf ./target

module_a目录下makefile实现:

CC := gcc
CFLAGS := -fPIC

INC_DIR := ./inc
SRC_DIR := ./src
OBJ_DIR := $(SRC_DIR)/obj

INSTALL_LIB_PATH := ../lib
LIB_TARGET := libmoduleA.a

#包含需要的头文件路径
CFLAGS += -I$(INC_DIR)

#搜索指定目录下的所有以.c结尾的文件(搜索结果保存在SOURCE,结果中包含路径相关信息)
SOURCE = $(wildcard $(SRC_DIR)/*.c)
#去除结果中的路径信息
SOURCER = $(notdir $(SOURCE))
#把.c换成.o
OBJS = $(patsubst %.c,%.o,$(SOURCER))

#把所有.o加上想要的路径信息
OBJS := $(foreach X,$(OBJS),$(OBJ_DIR)/$(X))
#OBJS := $(OBJ_DIR)/*.o
#OBJS := $(OBJ_DIR)/a.o \
#$(OBJ_DIR)/main.o
#OBJS := $(foreach x,$(OBJS),$(x).o)
$(OBJ_DIR)/%.o: $(SRC_DIR)/%.c
$(CC) $(CFLAGS) -c -o $@ $<

#$(TARGET):$(OBJS)
#$(CC) $(CFLAGS) -o $(TARGET_DIR)/$@ $(OBJS)
#用.o生成静态库文件.a,并放到指定目录下
$(LIB_TARGET):$(OBJS)
ar crus $(LIB_TARGET) $^
install -m 0755 $(LIB_TARGET) $(INSTALL_LIB_PATH)
rm -rf $(LIB_TARGET)

.PHONY:clean
clean:
rm -rf $(OBJ_DIR)
rm -rf $(INSTALL_LIB_PATH)/$(LIB_TARGET)

main目录下makefile实现:

CC := gcc
CFLAGS := -fPIC

INC_DIR := ../module_a/inc ../include
SRC_DIR := ./src
OBJ_DIR := $(SRC_DIR)/obj

TARGET_DIR := ../target
TARGET := main
INSTALL_LIB_PATH := ../lib

#包含所有以.a结尾的文件
LIB_TARGET := *.a

CFLAGS += -I$(INC_DIR)

SOURCE = $(wildcard $(SRC_DIR)/*.c)
SOURCER = $(notdir $(SOURCE))
OBJS = $(patsubst %.c,%.o,$(SOURCER))
OBJS := $(foreach X,$(OBJS),$(OBJ_DIR)/$(X))

$(OBJ_DIR)/%.o: $(SRC_DIR)/%.c
$(CC) $(CFLAGS) -c -o $@ $<

#生成可执行文件
$(TARGET):$(OBJS)
$(CC) -o $(TARGET_DIR)/$(TARGET) $(OBJS) \
$(INSTALL_LIB_PATH)/$(LIB_TARGET)

@echo "------------------------ MAKEFILE COMPLETE -----------------------"


.PHONY:clean
clean:
rm -rf $(OBJ_DIR)