我有以下makefile似乎运行得很好:
HDRS = include/config.h include/opcode_def.h include/x86lib.h include/x86lib_internal.h
CXX ?= g++
AR ?= ar
TEST_CC ?= i386-elf-gcc
TEST_CFLAGS ?= -fdata-sections -ffunction-sections
CXX_VM_SRC = vm/x86lib.cpp vm/modrm.cpp vm/device_manager.cpp vm/cpu_helpers.cpp vm/ops/strings.cpp vm/ops/store.cpp vm/ops/maths.cpp \
vm/ops/groups.cpp vm/ops/flow.cpp vm/ops/flags.cpp vm/ops/etc.cpp
CXX_VM_OBJS = $(subst .cpp,.o,$(CXX_VM_SRC))
CXX_TESTBENCH_SRC = testbench/testbench.cpp
CXX_TESTBENCH_OBJS = $(subst .cpp,.o,$(CXX_TESTBENCH_SRC))
CXXFLAGS ?= -Wall -g3 -fexceptions -fPIC -Wall
CXXFLAGS += -DX86LIB_BUILD -I./include
VERSION=1.1
VM_OUTPUTS = libx86lib.a libx86lib.so.$(VERSION)
OUTPUTS = $(VM_OUTPUTS) x86testbench
default: build
build: $(OUTPUTS)
libx86lib.a: $(CXX_VM_OBJS) $(CXX_VM_SRC)
ar crs libx86lib.a $(CXX_VM_OBJS)
libx86lib.so.$(VERSION): $(CXX_VM_OBJS) $(CXX_VM_SRC)
$(CXX) -shared $(CXX_VM_OBJS) -o libx86lib.so.$(VERSION)
x86testbench: $(CXX_TESTBENCH_OBJS) $(VM_OUTPUTS)
$(CXX) $(CXXFLAGS) -static -o x86testbench $(CXX_TESTBENCH_OBJS) -lx86lib -L.
$(CXX_TESTBENCH_OBJS): $(HDRS)
$(CXX) $(CXXFLAGS) -c $*.cpp -o $@
$(CXX_VM_OBJS): $(HDRS)
$(CXX) $(CXXFLAGS) -c $*.cpp -o $@
clean:
rm $(CXX_VM_OBJS) $(OUTPUTS) $(CXX_TESTBENCH_OBJS)
我的问题是,如果我更改文件,例如" vm / x86lib.cpp"然后它将重新链接最终输出等,但它不会重新编译vm / x86lib.o。我不知道如何向$(CXX_VM_OBJS)
构建规则表达此重新编译要求。
如何修复此问题,以便重新编译源文件并重新链接它们?
注意:如果我将$(CXX_VM_SRC)
作为$(CXX_VM_OBJS)
的依赖项,那么它会在单个文件发生更改时随时重新编译每个文件。另外,我只关心GNU Make和OSX / Linux环境。
答案 0 :(得分:4)
正如其他人所指出的那样,当源更改时,makefile不会重建任何目标文件的原因是您的规则:
$(CXX_VM_OBJS): $(HDRS)
...
没有提到源文件是先决条件。以及建议添加它们的解决方案的原因:
$(CXX_VM_OBJS): $(HDRS) $(CXX_VM_SRC)
...
当任何源文件发生更改时,会导致Make重建所有源文件,这正是此规则所要求的。它使所有源文件成为任何目标文件的先决条件。
处理此问题的正确方法是使用static pattern rule:
$(CXX_VM_OBJS): %.o: %.cpp $(HDRS)
...
我会为你的$(CXX_TESTBENCH_OBJS)
规则推荐相同的内容;即使它现在只适用于一个目标文件,但很明显你想要在以后添加其他文件。
答案 1 :(得分:0)
在CXX_VM_OBJS的依赖项列表中,尝试添加希望对象依赖的源文件名,例如:
$(CXX_VM_OBJS): $(HDRS) vm/x86lib.cpp
$(CXX) $(CXXFLAGS) -c $*.cpp -o $@