制作文件:没有规则来制定目标:如何设置正确的规则?

Makefile: no rule to make target: how to set the correct rules?

本文关键字:设置 规则 文件 有规则 目标 何设置      更新时间:2023-10-16

我有一个项目,我正在尝试为其创建一个Makefile。文件夹结构如下所示:

|-- bin
|-- build
|-- Makefile
|-- src
|   |-- some_files.cpp
|   |-- some_files.h
|   |-- ForestFactory
|   |-- MMParSet
|   |-- century
|       |-- century_carbon.cpp
|       |-- century_carbon.h
|   `-- grass
`-- src_multithread
|-- __history

我尝试使用此引用来创建 Makefile。 我拥有的制作文件是

CC := g++
SRCDIR := src
BUILDDIR := build
TARGETDIR := bin
INSTALLBINDIR := /usr/local/bin
TARGET := bin/formind
# CPP files
SOURCES := $(shell find $(SRCDIR) -type f -name *.cpp)
# remove the src path and add the build path and the .o extension
OBJECTS := $(addprefix $(BUILDDIR)/,$(patsubst %.cpp,%.o,$(notdir $(SOURCES))))
CFLAGS := -std=c++11 -stdlib=libc++
INC := -I/usr/local/include -I $(SRCDIR) -I $(SRCDIR)/century -I $(SRCDIR)/grass -I $(SRCDIR)/MMParSet -I $(SRCDIR)/ForestFactory
$(TARGET): $(OBJECTS)
@mkdir -p $(TARGETDIR)
@echo " Linking..."
@echo " $(CC) $^ -o $(TARGET)"
$(CC) $^ -o $(TARGET)
$(BUILDDIR)/%.o: $(SRCDIR)/%.cpp
@mkdir -p $(BUILDDIR)
@echo " $(CC) $(CFLAGS) $(INC) -c -o $@ $<"
$(CC) $(CFLAGS) $(INC) -c -o $@ $<
clean:
@echo " Cleaning...";
@echo " $(RM) -r $(BUILDDIR) $(TARGET)"
$(RM) -r $(BUILDDIR) $(TARGET)

.PHONY: clean

当我尝试make时,我收到以下错误消息:

make: *** No rule to make target `build/century_carbon.o', needed by `bin/formind'.  Stop.

源文件来自 src 中的一组嵌套子目录。我的理解是OBJECTS变量应该包含最终对象文件的完整路径,这样我就有类似的东西

$SOURCES = src/century/century_carbon.cpp ... src/for_branching.cpp ... src/grass/grass_log.cpp ...
$OBJECTS = build/century_carbon.o ... build/for_branching.o ... build/grass_log.o ...

我编写编译规则的方式有问题吗?

问题出在嵌套的源目录century上。如果添加到生成文件:

info:
@echo " Info..."
@echo " SOURCES = $(SOURCES)"
@echo " OBJECTS = $(OBJECTS)"

然后make info给出:

Info...
SOURCES = src/century/century_carbon.cpp src/some_files.cpp
OBJECTS = build/century_carbon.o build/some_files.o

此检查表明这些宏的构造是正常的。正如您所报告的那样,它还表明make随后会告诉我们:

make: *** No rule to make target 'build/century_carbon.o', needed by 'bin/formind'.  Stop.

如果添加其他规则,并指定目录century

$(BUILDDIR)/%.o: $(SRCDIR)/century/%.cpp
@mkdir -p $(BUILDDIR)
$(CC) $(CFLAGS) $(INC) -c -o $@ $<

然后,您的构建将继续。


另一种解决方案是使用VPATH.添加行:

VPATH = src:src/century

并从规则中删除$(SRCDIR)/,现在变为:

$(BUILDDIR)/%.o: %.cpp
@mkdir -p $(BUILDDIR)
$(CC) $(CFLAGS) $(INC) -c -o $@ $<

更新:上述答案可以通过两种方式改进:

  • 使用vpath而不是VPATH,以更精确地控制搜索。

  • 自动生成vpath/VPATH的参数。

这非常简单:只需将我的硬编码VPATH行替换为:

vpath %.cpp $(sort $(dir $(SOURCES)))