【问题标题】:Makefile to compile a list of sources to custom directoryMakefile 将源列表编译到自定义目录
【发布时间】:2020-12-24 18:33:43
【问题描述】:

我有这个makefile:

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

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

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

【问题讨论】:

    标签: c++ c linux makefile wildcard


    【解决方案1】:

    您的$(OBJS) 规则错误。

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

    您可以写一个pattern rule 并使用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)))))
    

    【讨论】:

    • 非常感谢!不知道 vpath 选项!我还有一个问题。如何使用 vpath 选项检查 $(DEPS) 文件中的修改?
    • @CataCata:这是一个棘手的问题。一个粗略的解决方案是制作头文件order-only prerequisites:$(OBJS): obj/%.o: %.cpp | $(DEPS)。我所知道的最佳解决方案在Mad Scientist's page on Auto-Dependency Generation 上有说明;这是一项复杂的技术,需要仔细考虑。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-11-02
    • 1970-01-01
    • 2015-12-31
    相关资源
    最近更新 更多