具有多个.c和.h依赖项的单个目标的Makefile

时间:2013-09-29 14:01:29

标签: c makefile

我有这个hello-world.c,我想编译成hello-world二进制文件。但hello-world.c取决于../helpers/a.c../helpers/b.c中定义的一些函数,而且每个帮助程序分别包含../helpers/a.h../helpers/b.h

我当前的Makefile看起来像

CC      =   @gcc
CFLAGS  =   -g -Wall -Wextra -Werror
CFLAGS  +=  

LDLIBS  =   
LDLIBS  +=  

OBJS    =   ../helpers/a.o ../helpers/b.o

SOURCES =   hello-world.c
DESTS   =   hello-world

new: clean all

clean:
    @rm -rf *.o */*.o $(DESTS)

all: $(OBJS) $(DESTS)

%.o: %.c
    $(CC) $(CFLAGS) -c $< -o $@

%: %.c
    $(CC) $(CFLAGS) -o $@ $<

但它不起作用,返回make: *** No rule to make target `../helpers/a.o', needed by `all'. Stop.

据我所知,Makefile似乎没有看到%.o的规则,但我不明白为什么。

编辑:Makefile debug:

alexandernst@stupidbox:/media/sf_procmon/procmon$ make --debug=b
GNU Make 3.81
Copyright (C) 2006  Free Software Foundation, Inc.
This is free software; see the source for copying conditions.
There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A
PARTICULAR PURPOSE.

This program built for x86_64-pc-linux-gnu
Reading makefiles...
Updating goal targets....
 File `new' does not exist.
   File `clean' does not exist.
  Must remake target `clean'.
  Successfully remade target file `clean'.
   File `all' does not exist.
     File `../helpers/a.o' does not exist.
    Must remake target `../helpers/a.o'.
make: *** No rule to make target `../helpers/a.o', needed by `all'.  Stop.

2 个答案:

答案 0 :(得分:0)

如果将Makefile放在父目录中,这将更容易实现。然后你可以这样写:

CC        = gcc
CFLAGS    = -g -Wall -Wextra -Werror
CPPFLAGS  = -Ihelpers

DESTS       = hello-world/hello-world
OBJS        = helpers/a.o helpers/b.o hello-world/hello-world.o

# Do not make 'all' depend directly on object files.
all: $(DESTS)

# Clean rules should always be prefixed with '-' to avoid
# errors when files do not exist.  Do not use 'rm -r' when
# there shouldn't be directories to delete; do not delete
# wildcards when you can use explicit lists instead.
# Never use '@'.
clean:
        -rm -f $(OBJS) $(DESTS)

# An explicit linkage rule is needed for each entry in DESTS.
hello-world/hello-world: hello-world/hello-world.o \
                         helpers/a.o helpers/b.o
        $(CC) $(CFLAGS) $(LDFLAGS) -o $@ $^ $(LIBS)

# The main function of these dependency lists is to prevent
# Make from deleting object files after each build.  We also
# take the opportunity to specify header dependencies.
# We rely on the built-in %.o:%.c rule for commands.
hello-world/hello-world.o: hello-world/hello-world.c \
                           helpers/a.h helpers/b.h
helpers/a.o: helpers/a.c helpers/a.h
helpers/b.o: helpers/b.c helpers/b.h

# This tells Make that 'all' and 'clean' are not files to be created.
.PHONY: all clean

有关此技术的进一步说明,请参阅分水岭论文“Recursive Make Considered Harmful”以及相关的实施说明。

答案 1 :(得分:0)

我设法修复了我的Makefile:

CC      =   gcc
CFLAGS  =   -g -Wall -Wextra -Werror
CFLAGS  +=  

LDLIBS  =   
LDLIBS  +=  

OBJS    =   ../helpers/a.o ../helpers/b.o hello-world.o

SOURCES =   hello-world.c
DESTS   =   hello-world

new: clean all

clean:
    @rm -rf *.o */*.o ../helpers/*.o $(DESTS)

all: $(DESTS)

hello-world: $(OBJS)
    $(CC) $(CFLAGS) $(LDFLAGS) -o $@ $^ $(LIBS)

其中一个问题是,我写的一个糟糕的清理规则已经擦除了a.c和b.c,并且Makefile因此而起作用。