A maioria das respostas é surpreendentemente complicada ou errônea. No entanto, exemplos simples e robustos foram postados em outro lugar [ codereview ]. É certo que as opções fornecidas pelo pré-processador GNU são um pouco confusas. No entanto, a remoção de todos os diretórios do destino de construção com -MM
está documentada e não é um bug [ gpp ]:
Por padrão, o CPP recebe o nome do arquivo de entrada principal, exclui quaisquer
componentes do diretório e qualquer sufixo de arquivo, como '.c', e anexa o sufixo de objeto usual da plataforma.
A opção (um pouco mais recente) -MMD
é provavelmente o que você deseja. Para completar, um exemplo de um makefile que suporta vários diretórios src e diretórios de construção com alguns comentários. Para uma versão simples sem diretórios de compilação, consulte [ codereview ].
CXX = clang++
CXX_FLAGS = -Wfatal-errors -Wall -Wextra -Wpedantic -Wconversion -Wshadow
# Final binary
BIN = mybin
# Put all auto generated stuff to this build dir.
BUILD_DIR = ./build
# List of all .cpp source files.
CPP = main.cpp $(wildcard dir1/*.cpp) $(wildcard dir2/*.cpp)
# All .o files go to build dir.
OBJ = $(CPP:%.cpp=$(BUILD_DIR)/%.o)
# Gcc/Clang will create these .d files containing dependencies.
DEP = $(OBJ:%.o=%.d)
# Default target named after the binary.
$(BIN) : $(BUILD_DIR)/$(BIN)
# Actual target of the binary - depends on all .o files.
$(BUILD_DIR)/$(BIN) : $(OBJ)
# Create build directories - same structure as sources.
mkdir -p $(@D)
# Just link all the object files.
$(CXX) $(CXX_FLAGS) $^ -o $@
# Include all .d files
-include $(DEP)
# Build target for every single object file.
# The potential dependency on header files is covered
# by calling `-include $(DEP)`.
$(BUILD_DIR)/%.o : %.cpp
mkdir -p $(@D)
# The -MMD flags additionaly creates a .d file with
# the same name as the .o file.
$(CXX) $(CXX_FLAGS) -MMD -c $< -o $@
.PHONY : clean
clean :
# This should remove all generated files.
-rm $(BUILD_DIR)/$(BIN) $(OBJ) $(DEP)
Este método funciona porque se houver várias linhas de dependência para um único destino, as dependências serão simplesmente unidas, por exemplo:
a.o: a.h
a.o: a.c
./cmd
é equivalente a:
a.o: a.c a.h
./cmd
conforme mencionado em: Makefile várias linhas de dependência para um único destino?