先看一下make前后的目录结构对比
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)