9

相关:目标特定变量作为 Makefile 中的先决条件

我正在尝试制作一个 Makefile,它使用特定于目标的变量来指定目标文件和最终可执行文件的输出目录。这个想法是维护两个独立的二进制版本,一个“发布”版本和一个带有额外调试信息的“调试”版本。

我的问题是'make'每次都会进行干净的构建,即使我没有改变任何东西。我很确定这是因为'make'在'debug'或'release'目标的先决条件中的变量声明之前评估目标'corewars'的先决条件。

Makefile 如下所示。

CXX=g++
LD=g++
LDFLAGS=
CXXFLAGS=-Iinclude -Wall -Wextra
OBJECTS=main.o Machine.o Core.o ProcessQueue.o Instruction.o
OUTPUT_DIR:=Test/

.PHONY: default
.PHONY: all
.PHONY: release
default: release
all: release
release: OUTPUT_DIR:=Release/
release: corewars

.PHONY: debug
debug: CXXFLAGS+=-DDEBUG -g
debug: OUTPUT_DIR:=Debug/
debug: corewars

corewars: $(OUTPUT_DIR) $(addprefix $(OUTPUT_DIR),$(OBJECTS))
    $(LD) -o $(addprefix $(OUTPUT_DIR),corewars) $(addprefix $(OUTPUT_DIR),$(OBJECTS))

Release:
    mkdir -p $@
Debug:
    mkdir -p $@

%.o: %.cpp include/%.h
    $(CXX) -c $(CXXFLAGS) $< -o $(OUTPUT_DIR)$@


.PHONY: clean
clean:
    $(RM) -r Release
    $(RM) -r Debug
4

2 回答 2

15

首先,非虚假配方必须创建一个目标,$@而不是$(OUTPUT_DIR)$@。还要考虑将目录依赖项转换为仅订单的先决条件。

为了在$(OUTPUT_DIR)先决条件列表中获得适当的值,您必须使用辅助扩展,因为否则,在主要扩展期间,将OUTPUT_DIR:=Test/使用全局定义而不是特定于目标的定义。

不幸的是,我想不出一种理智的方法来使用目标特定变量使其工作,而不求助于二次扩展和 vpath 魔法。就我个人而言,我宁愿先设置环境(找出 的值OUTPUT_DIR等),然后使用正确的值重新执行 Make。

ifndef OUTPUT_DIR

.PHONY: default all release debug

default all: release

release: export OUTPUT_DIR := Release/
debug:   export OUTPUT_DIR := Debug/
debug:   export EXTRA_CXXFLAGS := -DDEBUG -g

release debug:
    @$(MAKE)

else

# ...
CXXFLAGS := -Iinclude -Wall -Wextra $(EXTRA_CXXFLAGS)

PROGRAM := $(OUTPUT_DIR)corewars
OBJECTS := $(addprefix $(OUTPUT_DIR), \
    main.o Machine.o Core.o ProcessQueue.o Instruction.o)

# Default target.
$(PROGRAM): $(OBJECTS) | $(OUTPUT_DIR)
    $(LD) -o $@ $<

$(OUTPUT_DIR)%.o: %.cpp | $(OUTPUT_DIR)
    $(CXX) -c $(CXXFLAGS) $< -o $@

$(OUTPUT_DIR):
    mkdir -p $@

endif # OUTPUT_DIR

这两部分可以拆分为单独的 makefile,根(启动器)一个,以及做实际工作的一个,以使整个事情更易于管理。

于 2012-05-30T22:28:19.053 回答
1

特定于目标的变量仅在目标配方及其递归先决条件的上下文中可用。也就是说,特定于目标的变量不能用作目标或先决条件

一种解决方法是那里的makefile 。

于 2020-06-12T18:35:23.700 回答