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)))))
我有这个生成文件:
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)))))