我已关注this tutorial。它解释了Makefile的制作,它负责依赖。我制作了以下Makefile,它按照以下目录结构工作:
folder--|Makefile
|src----|(all .c and .h files here)
|obj----|(all objects file are made here)
|bin----|(target is made here)
Makefile是:
TARGET = exec
CC = gcc
CFLAGS = -g -I.
LINKER = gcc -o
LFLAGS = -I. -lm -lpthread
BINDIR = bin
OBJDIR = obj
SRCDIR = src
INTERFACE = interface
STD = -std=c99
PROGRAMSOURCES := $(wildcard $(SRCDIR)/*.c)
PROGRAMINTERFACE:= $(wildcard $(INTERFACE)/*.h)
OBJECTS := $(PROGRAMSOURCES:$(SRCDIR)/%.c=$(OBJDIR)/%.o)
$(BINDIR)/$(TARGET) : $(OBJECTS)
$(LINKER) $@ $(LFLAGS) $(OBJECTS) $(STD)
#pull the dependencies to the .o files
-include $(OBJECTS:.o=.d)
#the -o $@ says to put the output of the compilation in the file named on the left side of the :.
#the $< is the first item in the dependencies list. Basically the name of the .c file which is to be compiled.
$(OBJECTS) : $(OBJDIR)/%.o :$(SRCDIR)/%.c
$(CC) $(CFLAGS) -c $< -o $@ $(STD)
$(CC) $(CFLAGS) -MM $< > $*.d
@mv -f $*.d $*.d.tmp #changes file name
@sed -e 's|.*:|$*.o:|' < $*.d.tmp > $*.d #Unable to understand
@sed -e 's/.*://' -e 's/\\$$//' < $*.d.tmp | fmt -1 | \
sed -e 's/^ *//' -e 's/$$/:/' >> $*.d #Unable to understand
@rm -f $*.d.tmp
.PHONY : run
run :
./$(BINDIR)/$(TARGET) ${TYPE} ${INP_FILE}
print:
@echo $(OBJECTS)
我已经理解它正在尝试预处理临时文件以生成自动依赖。我无法理解的是它是如何完成的。以下是我想解释的两行:
@sed -e 's|.*:|$*.o:|' < $*.d.tmp > $*.d #Unable to understand
@sed -e 's/.*://' -e 's/\\$$//' < $*.d.tmp | fmt -1 | \
sed -e 's/^ *//' -e 's/$$/:/' >> $*.d #Unable to understand
我之前从未使用sed
所以我遇到了问题。
任何帮助表示感谢。
答案 0 :(得分:3)
假设您要从obj/foo.o
构建src/foo.c
,因此文件foo.d.tmp
包含:
foo.o: src/foo.c src/foo.h src/bar.h
现在是第一个sed
声明:
@sed -e 's|.*:|$*.o:|' < $*.d.tmp > $*.d
“阅读foo.d.tmp
,将所有内容更改为冒号并将其更改为'foo.o',并将结果写入foo.d
。”所以现在foo.d
包含:
foo.o: src/foo.c src/foo.h src/bar.h
(在这种情况下没有变化。)现在是下一个命令:
@sed -e 's/.*://' -e 's/\\$$//' < $*.d.tmp | fmt -1 | sed -e 's/^ *//' -e 's/$$/:/' >> $*.d
“阅读foo.d.tmp
,删除所有内容(包括冒号),删除尾随\
(如果有)。取结果并将每个单词(即每个先决条件)单独放置然后,对于每一行,删除前导空格,在末尾添加一个冒号,并将结果追加到foo.d
。“所以现在foo.d
包含:
foo.o: src/foo.c src/foo.h src/bar.h
src/foo.c:
src/foo.h:
src/bar.h:
这个想法是为每个先决条件创建一个空规则,这样如果代码已经改变并且不再需要某个先决条件 - 并且不再存在 - 但它仍然列在旧{{1}中},Make不会因无法构建它而感到恐慌。