Makefile分离多个源代码的编译

时间:2015-07-14 03:40:45

标签: makefile

我编写了一个简单的Makefile来编译几种类型的源代码。它运作良好。此Makefile将找到所有protobuf来源和C++来源。所以制定该计划将如下。

  1. protoc将使用C++
  2. 生成*.proto来源文件
  3. g++将使用*.cpp*.pb.cc(由步骤1生成)编译目标文件
  4. g++(实际上是ld)会将所有对象*.obj链接到单个可执行文件
  5. 这个Makefile示例是:

    # Define the protoc generator
    PROTOC = protoc
    INCLUDE = -I$(SOMEPLACE) 
    LIB = -lxml2  -lpthread -lz
    
    # The final single outputted executable
    OUTPUT  = svrkit_adapter_v2
    
    # Define the cpp source code/object result
    SOURCES = $(wildcard *.cpp)
    OBJECTS=$(SOURCES:.cpp=.o)
    
    # Define the proto's source/generated result/object result
    PROTO_SOURCES = $(wildcard *.proto)
    PROTO_CPPSRC = $(patsubst %.proto,%.pb.cc,$(PROTO_SOURCES))
    PROTO_OBJECT = $(patsubst %.proto,%.pb.o,$(PROTO_SOURCES))
    
    # First actual making target
    all:$(OUTPUT) 
    
    # Define the rule for generating proto's cpp sources
    $(PROTO_CPPSRC) : $(PROTO_SOURCES)
      $(PROTOC) --cpp_out=. $(filter %.proto, $^) 
    
    # Define the rule for compiling generated proto cpp sources
    $(PROTO_OBJECT) : $(PROTO_CPPSRC)
      $(CXX) $(CPPFLAGS) $(INCLUDE) -c $(filter %.cc, $^) 
    
    # Define the rule for compiling other cpp sources
    $(OBJECTS): $(SOURCES) $(PROTO_OBJECT)
      $(CXX) $(CPPFLAGS) $(INCLUDE) -c $(filter %.cpp, $^) 
    
    # Define the rule for linking the final executable
    $(OUTPUT): $(OBJECTS) $(PROTO_OBJECT)
      $(CXX) $(CFLAGS) -o $@ $^ ${LIB}
    
    clean:
      rm -f *.o *.~ *.bak *.pb.*
      rm -f $(OUTPUT)
    

    然而,make执行的命令行如下(例如):

    protoc a.proto b.proto
    g++ -c a.pb.cc b.pb.cc
    g++ -c x.cpp y.cpp z.cpp
    g++ -o output a.o b.o x.o y.o z.o
    

    有时候效果很好。 但是如果源太多,重新编译一些未更改的源将花费大量时间。例如,我刚刚修改了a.proto并且Makefile也将重新编译b.proto

    我的问题是:我如何编制/生成每个源文件单独。执行的命令行应为:

    protoc a.proto
    protoc b.proto
    g++ -c a.pb.cc -o a.o
    g++ -c b.pb.cc -o b.o
    g++ -c x.cpp -o x.o
    g++ -c y.cpp -o y.o
    g++ -c z.cpp -o z.o
    g++ -o output a.o b.o x.o y.o z.o
    

    先谢谢。

2 个答案:

答案 0 :(得分:4)

您错误地将规则用于完整列表而不是单个文件,从而剥夺了make 工作在单个文件上的能力。

你的生成规则:

# Define the rule for generating proto's cpp sources
$(PROTO_CPPSRC) : $(PROTO_SOURCES)
  $(PROTOC) --cpp_out=. $(filter %.proto, $^) 

# Define the rule for compiling generated proto cpp sources
$(PROTO_OBJECT) : $(PROTO_CPPSRC)
  $(CXX) $(CPPFLAGS) $(INCLUDE) -c $(filter %.cc, $^) 

相反,请将规则用于单个文件

# Define the rule for generating proto's cpp sources
%.pb.cc: %.proto
    $(PROTOC) --cpp_out=. $<       

# Define the rule for compiling generated proto cpp sources
%.pb.o: %.pb.cc
    $(CXX) $(CPPFLAGS) $(INCLUDE) -c $< -o $@

您的C ++源代码也是如此:

# Define the rule for compiling other cpp sources
$(OBJECTS): $(SOURCES) $(PROTO_OBJECT)
  $(CXX) $(CPPFLAGS) $(INCLUDE) -c $(filter %.cpp, $^) 

(为什么$(PROTO_OBJECT)包含在依赖项中?这看起来非常错误。)

再次,将其切换为单个文件而不是整个列表:

# Define the rule for compiling other cpp sources
%.o: %.cpp
    $(CXX) $(CPPFLAGS) $(INCLUDE) -c $< -o $@

只有最终目标需要完整的对象列表:

# Define the rule for linking the final executable
$(OUTPUT): $(OBJECTS) $(PROTO_OBJECT)
  $(CXX) $(CFLAGS) -o $@ $^ ${LIB}

删除PROTO_CPPSRC,不需要它。

现在,make将查看最后一条规则,找到对象依赖项,并将单文件规则应用于需要重建的每个对象。

与您的问题无关,但在您处理此问题时,请将以下语句添加到您的Makefile中:

.PHONY: all clean

这将使&#34;所有&#34;和&#34;清洁&#34;运行始终即使存在具有该名称的文件。这通常不是问题,但是当它发生时,它可能会令人困惑。 ; - )

答案 1 :(得分:0)

我知道这不是你问题的全部答案,但我认为这可能有所帮助 您。这是我使用的makefile的一个例子:

EXEC=prog1
EXEC+=prog2

prog1_SRCS=prog1.cpp source1.cpp source2.cpp
prog2_SRCS=prog2.cpp source3.cpp

#------------------------------------------------------------------
#no need to change anything below that line (maybe add some OPTION)

CXX = g++ -std=c++11

ERRORS = -Wall -Wextra -pedantic
OPTION = 

BUILD=build

CXXFLAGS = $(ERRORS) $(OPTION)
LDFLAGS  = $(ERRORS) $(OPTION)

SRCS=$(wildcard *.cpp)

all: OPTION += -O3 -DNDEBUG
all:$(EXEC)

debug: OPTION += -ggdb 
debug:$(EXEC)

.SECONDEXPANSION:
$(EXEC): $$(patsubst %.cpp, $(BUILD)/%.o, $$($$@_SRCS)) 
    @echo Links $(notdir $^)
    $(CXX) -o $@ $^ $(LDFLAGS) 

$(BUILD)/%.o:%.cpp
    @echo Creates $(notdir $@)
    $(CXX) -MD -c $(CXXFLAGS) $< -o $@

-include $(addprefix $(BUILD)/,$(SRCS:.cpp=.d))

clean:
    rm -f $(BUILD)/* $(EXEC)

您需要一个名为build的目录来保存.o.d文件。他们会的 允许您只重新编译已修改的文件