Большинство ответов на удивление сложны или ошибочны. Однако простые и надежные примеры были опубликованы в другом месте [ codereview ]. По общему признанию, параметры, предоставляемые препроцессором GNU, немного сбивают с толку. Однако удаление всех каталогов из целевой сборки с помощью -MM
задокументировано и не является ошибкой [ gpp ]:
По умолчанию CPP принимает имя основного входного файла, удаляет все
компоненты каталога и любой суффикс файла, такой как '.c', и добавляет обычный суффикс объекта платформы.
-MMD
Возможно, вам нужен вариант (несколько более новый) . Для полноты картины приведен пример файла makefile, который поддерживает несколько директорий src и директории сборки с некоторыми комментариями. Простую версию без директорий сборки см. В [ 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)
Этот метод работает, потому что, если существует несколько строк зависимостей для одной цели, зависимости просто объединяются, например:
a.o: a.h
a.o: a.c
./cmd
эквивалентно:
a.o: a.c a.h
./cmd
как упоминалось в: Makefile нескольких строк зависимостей для одной цели?