Makefile 将源列表编译到自定义目录

Makefile to compile a list of sources to custom directory

我有这个生成文件:

IDIR = include
SDIR = src
ODIR = obj
BDIR = build

DEPS = $(shell find $(IDIR)/ -name '*.hpp')
SRCS = $(shell find $(SDIR)/ -name '*.cpp')
OBJS = $(patsubst %.cpp,$(ODIR)/%.o,$(notdir $(SRCS)))
BIN = main

CPPC = g++
CFLAGS = -Wall -c

all: dir $(BDIR)/$(BIN)
    @echo Finished compiling $(BIN)

dir:
    mkdir -p $(BDIR)
    mkdir -p $(ODIR)

$(BDIR)/$(BIN): $(OBJS)
    $(CPPC) $^ -o $@

$(OBJS): $(SRCS)
    $(CPPC) $(CFLAGS) $^ -o $@

clean:
    rm -rf $(BDIR) $(ODIR)

当我尝试制作时,出现以下错误:

mkdir -p build
mkdir -p obj
g++ -Wall -c src/sdk/tcp/Tcp.cpp src/sdk/socket/Socket.cpp src/Main.cpp -o obj/Tcp.o
g++: fatal error: cannot specify ‘-o’ with ‘-c’, ‘-S’ or ‘-E’ with multiple files
compilation terminated.
make: *** [Makefile:27: obj/Tcp.o] Error 1

我的问题是,是否可以使用此生成文件实现我正在尝试的目标?遍历 $(SRCS) 中的每个源文件,并直接在 obj 目录中仅使用基本名称编译目标文件。编译成功后obj目录示例:

         obj
  /       |        \
 /        |         \
Tcp.o   Socket.o    Main.o

你的 $(OBJS) 规则是错误的。

有(至少)两种方法可以做到这一点。

您可以写一个 pattern rule and use vpath 来定位来源:

vpath %.cpp $(dir $(SRCS))

$(OBJS): obj/%.o: %.cpp
    $(CPPC) $(CFLAGS) $^ -o $@

或者您可以为每个对象生成一条规则:

define template
$(patsubst %,obj/%.o,$(notdir $(1))): $(addsuffix .cpp,$(1))
    echo $(CPPC) $(CFLAGS) $$^ -o $$@
endef

$(foreach src,$(SRCS),$(eval $(call template,$(basename $(src)))))