makefile循环依赖和覆盖命令

时间:2012-09-16 05:39:38

标签: c build makefile

有一个参考makefile,我正在慢慢编辑和使用哪个吐出这两个错误

Makefile:25: warning: overriding commands for target `build/semanticHash'
Makefile:21: warning: ignoring old commands for target `build/semanticHash'
make: Circular build/semanticHash <- build/semanticHash dependency dropped.
cc -g -ldl  -lgsl -lgslcblas -lzmq  -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG  -fPIC   -c -o src/semanticHash/rmb.o src/semanticHash/rmb.c

我是makefile语法和规则的新手,所以我谷歌常见的错误,但在这种情况下我自己没有运气。

所以问题是,我在哪里导致这些错误,是否有我应该在当前的makefile中避免的模式?

CFLAGS= -g $(LIBS) -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG $(OPTFLAGS)
LIBS= -ldl  $(OPTLIBS)
PREFIX?=/usr/local
OPTLIBS= -lgsl -lgslcblas -lzmq 

SOURCES=$(wildcard src/**/*.c src/*.c)
OBJECTS=$(patsubst %.c,%.o,$(SOURCES))

TEST_SRC=$(wildcard tests/*_tests.c)
TESTS=$(patsubst %.c,%,$(TEST_SRC))

TARGET=build/semanticHash  # Rename to library !!!!!
SO_TARGET=$(patsubst %.a,%.so,$(TARGET))

# The Target Build
all: $(TARGET) tests


$(TARGET): CFLAGS += -fPIC
$(TARGET): build $(OBJECTS)
    ar rcs $@ $(OBJECTS)
    ranlib $@

$(SO_TARGET): $(TARGET) $(OBJECTS) 
    $(CC) -shared -o $@ $(OBJECTS) 

build:
    @mkdir -p build
    @mkdir -p bin

# The Unit Tests
.PHONY: tests
tests: CFLAGS += $(TARGET)
tests: $(TESTS)
    sh ./tests/runtests.sh

valgrind:
    VALGRIND="valgrind --log-file=/tmp/valgrind-%p.log" $(MAKE)

# The Cleaner
clean:
    rm -rf build $(OBJECTS) $(TESTS)
    rm -f tests/tests.log 
    find . -name "*.gc*" -exec rm {} \;
    rm -rf `find . -name "*.dSYM" -print`

# The Install
install: all
    install -d $(DESTDIR)/$(PREFIX)/lib/
    install $(TARGET) $(DESTDIR)/$(PREFIX)/lib/

1 个答案:

答案 0 :(得分:1)

您将SO_TARGET设为与TARGET相同:

TARGET=build/semanticHash  # Rename to library !!!!!
SO_TARGET=$(patsubst %.a,%.so,$(TARGET))

如所见TARGET .a结尾,因此不会替换任何内容使SO_TARGET相同。

稍后你有

$(SO_TARGET): $(TARGET) $(OBJECTS) 

由于SO_TARGETTARGET都相同,因此您具有循环依赖关系。

另外两个警告也是因为这个问题,因为你有相同目标的冲突命令。

在一个不相关的说明中,你不应该像你一样混合编译器和链接器标志。编译器标志用于编译,用于链接的链接器标志。您还应该更改链接器选项的顺序,并将库放在目标文件之后与链接。这是因为如果没有任何内容依赖于它们,GNU链接器不会加载库,并且在加载目标文件之前不会加载依赖项。