为啥我需要运行两次 makefile 来编译我的代码

Posted

技术标签:

【中文标题】为啥我需要运行两次 makefile 来编译我的代码【英文标题】:Why do I need to run my makefile twice to compile my code为什么我需要运行两次 makefile 来编译我的代码 【发布时间】:2014-11-06 02:43:23 【问题描述】:

所以我对构建自己的 makefile 还是很陌生。我遇到了一个奇怪的问题,我需要运行两次 make 才能生成我的可执行文件。我有多个文件,它们位于多个子文件夹中。每个包含文件的文件夹都有自己的 makefile。在那个 make 文件中,我生成我的目标文件,然后将目标文件推送到主目录。生成所有目标文件后,我将继续尝试创建可执行文件。

这是我运行其他 makefile 的主要 makefile。

# Declaration of variables
CC = g++
CFLAGS=-std=c++0x -Wall -lboost_system -lpthread
INCLUDE_PATH0=SocketIO/SocketServer/
INCLUDE_PATH1=SocketIO/SocketServer/SocketSession/


EXEC = run
SOURCES = $(wildcard *.cpp)
OBJECTS = $(SOURCES:.cpp=.o)

TARGETS = SocketSess SocketServ maincpp

$(EXEC): $(TARGETS) $(wildcard *.o)
  $(CC) $(wildcard *.o) $(CFLAGS) -o $(EXEC)

maincpp: $(SOURCES)
  $(CC) $(CFLAGS) -I$(INCLUDE_PATH0) -I$(INCLUDE_PATH1) -c $(SOURCES)
  ls
SocketSess:
  cd ./SocketIO/SocketServer/SocketSession ; make SocketSess

SocketServ:
  cd ./SocketIO/SocketServer ; make SocketServ

# To remove generated files
clean:
  rm -f $(EXEC) $(wildcard *.o)
  cd ./SocketIO/SocketServer/SocketSession ; make clean
  cd ./SocketIO/SocketServer ; make clean

这是我的子 makefile 之一。

  # Declaration of variables
  CC = g++
  CFLAGS=-std=c++0x -Wall -lboost_system -lpthread
  INCLUDE_PATH=SocketSession/

  #Just build object file do not link yet
  SocketServ: SocketServer.cpp SocketServer.h
    $(CC) $(CFLAGS) -I$(INCLUDE_PATH) -c SocketServer.cpp
    mv *.o ../../

  clean:
    rm -f *.o*

这是我得到的错误。

    Main.cpp  Main.o  Makefile  SocketIO  SocketServer.o  SocketSession.o  test
    g++  -std=c++0x -Wall -lboost_system -lpthread -o run
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 0 has invalid symbol index 11
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 1 has invalid symbol index 12
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 2 has invalid symbol index 2
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 3 has invalid symbol index 2
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 4 has invalid symbol index 11
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 5 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 6 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 7 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 8 has invalid symbol index 12
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 9 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 10 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 11 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 12 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 13 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 14 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 15 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 16 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 17 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 18 has invalid symbol index 13
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_info): relocation 19 has invalid symbol index 21
    /usr/bin/ld: /usr/lib/debug/usr/lib/x86_64-linux-gnu/crt1.o(.debug_line): relocation 0 has invalid symbol index 2
    /usr/lib/gcc/x86_64-linux-gnu/4.8/../../../x86_64-linux-gnu/crt1.o: In function `_start':
    (.text+0x20): undefined reference to `main'
    collect2: error: ld returned 1 exit status
    make: *** [run] Error 1

【问题讨论】:

$(wildcard *.o):第一次运行 makefile 时没有 .o 文件,所以这将是空的。也许您的意思是 $(OBJECTS),但是您的 Makefile 有各种奇怪的地方需要修复。 您希望规则的 makefile 目标与规则创建的文件名相匹配。例如,您的 SocketServ 规则不会生成该文件。它使 ../../SocketServer.o 成为规则目标。 这一行:$(CC) $(CFLAGS) -I$(INCLUDE_PATH) -c SocketServer.cpp 需要另外一个参数,可能是:-o SocketServer.o 关于 makefile(s) 中的目标,没有使用目标名称创建的实际文件,因此需要有一行:.PHONY $(TARGETS) 和 'clean' 需要虚假目标之一。 makefile 正在使用 'mv' 将目标文件移动到主目录。这将从当前目录中删除文件。然后它必须重新生成。也许使用 ''cp' 来复制文件。 【参考方案1】:

首先你必须在 makefile 中描述子目录,因为 make 不会知道你有子目录。

SUBDIR = NameSubDirecory, NameSubDirecory

其他问题请阅读RecursionMake

【讨论】:

我添加了这个,但它似乎没有太大变化。我可以在没有这个的情况下找到子目录。【参考方案2】:

查看错误输出的这一行:

g++  -std=c++0x -Wall -lboost_system -lpthread -o run

正在编译哪些文件(源文件或目标文件)以生成输出可执行文件run?看到怎么没有源文件,我怀疑这是对main的未定义引用的来源。

【讨论】:

是的,出于某种原因,它没有提取目标文件。第二次围绕它拉目标文件。我只是很困惑,如果它们是第一个生成的东西,为什么它看不到它们。

以上是关于为啥我需要运行两次 makefile 来编译我的代码的主要内容,如果未能解决你的问题,请参考以下文章

为啥相同的 CFLAGS 可以在 makefile 中重复两次?

Makefile:来自相同源的两个目标使用不同的标志编译两次

为啥 Xcode 将我的所有代码编译两次,导致任何全局变量的链接器错误?

使用 Makefile 编译后如何运行程序

为啥我的代码在完成块中运行两次

为啥我的 makefile 调用 gcc 虽然我设置了 CC=g++? [关闭]