首页 > 解决方案 > Makefile标记依赖、编译和链接

问题描述

我有一个在 openmp 中完成的障碍的模块化实现。我有一个我想链接的测试工具。该目录如下所示:

Project/
  -Tests
     -openmp_tournament
        -test_openmp_tournament.cpp
  -OpenMP
     -tournament.cpp .   //barrier implementation
     -tournament.h

现在,这两个 .cpp 文件都在 Project/OpenMp/ 中包含了针对锦标赛.h 的标头。这个想法是编译两者,然后以模块化方式将它们链接在一起(因为我有各种我想测试的实现)。我的生成文件如下所示:

PATH_TO_OPENMP = ../../OpenMP/

#OpenMP Flags Etc.
OMPFLAGS = -fopenmp -DLEVEL1_DCACHE_LINESIZE=`getconf LEVEL1_DCACHE_LINESIZE`
OMPLIBS = -lgomp

CC = g++
CPPFLAGS = -MMD -MP # enables automatic dependency tracking
CFLAGS = -Wall -Wextra -I$(PATH_TO_OPENMP) $(OMPFLAGS) -std=c++11
LDLIBS = $(OMPLIBS)

COMPILE = $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDLIBS) -o $@
COMPILE_NOLINK = $(CC) -c $(CPPFLAGS) $(CFLAGS) $< $(LDLIBS) -o $@

all: tournament_openmp.a

# link all files into one binary
tournament_openmp.a: tournament.o test_openmp_tournament.o 
    $(COMPILE)

# compile only each source file and mark as dependent
%.o: $(PATH_TO_OPENMP)%.cpp %.d
    $(COMPILE_NOLINK)

# Empty pattern rule to match dependency (*.d) files (i.e. makefiles),
# so make won't fail if dependency doesn't exist
%.d: ;

# Mark dependency files as precious so they won't be deleted as intermediates
.PRECIOUS: %.d

# The list of all source files I want to track dependencies for
SOURCES=$(wildcard *.cpp)

# Include any dependency files that exist, and
# suppress error message for ones that don't yet (via hyphen)
-include $(SOURCES:.cpp=.d)

.PHONY: clean
clean:
    rm -rf *.o *.d *.a *_openmp *_mpi

当我运行 Make 时,我收到以下错误,我不确定如何解决它。它找不到 test_openmp_tournament.cpp 文件的锦标赛.h:

a@ubuntu:~/Documents/Project/Tests/openmp_tournament$ make all
g++ -c -MMD -MP  -Wall -Wextra -I../../OpenMP/ -fopenmp -DLEVEL1_DCACHE_LINESIZE=`getconf LEVEL1_DCACHE_LINESIZE` -std=c++11 ../../OpenMP/tournament.cpp -lgomp -o tournament.o
g++  -MMD -MP   -c -o test_openmp_tournament.o test_openmp_tournament.cpp
test_openmp_tournament.cpp:51:10: fatal error: tournament.h: No such file or directory
   51 | #include "tournament.h"

标签: c++makefilelinker

解决方案


为什么编译器找不到你的头文件?好吧,我们应该始终假设编译器是正确的,而您做错了什么:)。所以,看看有效的编译行:

g++ -c -MMD -MP  -Wall -Wextra -I../../OpenMP/ -fopenmp ...

然后是失败的那个:

g++  -MMD -MP   -c ...

很明显为什么编译器找不到你的头文件,对吧?因为,您没有给它正确的编译器标志。

那么问题就变成了,为什么 make 不运行我们期望的编译命令呢?make 这样做的唯一原因是因为它使用两个不同的规则来构建两个目标。所以让我们看看你的规则:

%.o: $(PATH_TO_OPENMP)%.cpp %.d
        $(COMPILE_NOLINK)

好吧,这告诉 make 如何xxx.o../../OpenMP/xxx.cpp任何xxx. 这对于../../OpenMP目录中的源文件非常有用,但当然,此规则不能用于构建任何在../../OpenMP.

那么,如果您的规则不匹配怎么办?它将查看自己的内置规则,果然,有一个内置规则知道如何从当前目录中的源文件构建目标文件。

为什么会失败?它失败是因为您使用的是非标准变量,因此 make 使用的内置规则没有使用您的自定义变量。

特别是,该CPPFLAGS变量旨在保存 C/C++ 预处理器标志,因此从技术上讲,该-I选项(这是一个预处理器选项)应该出现在该变量中。其次,C++ 编译器的标准变量用于编译CXX器(not CC,它是 C 编译器)和CXXFLAGS标志(not CFLAGS,它用于 C 编译器)。

因此,您应该至少执行以下一项,并且可能同时执行以下两项:

将您的变量名称修复为标准:

CXX = g++
CPPFLAGS = -MMD -MP -I$(PATH_TO_OPENMP)
CXXFLAGS = -Wall -Wextra $(OMPFLAGS) -std=c++11

创建一个规则来构建本地对象,而不是默认规则:

%.o: %.cpp %.d
        $(COMPILE_NOLINK)

(精明的人会注意到其中也OMPFLAGS包含一个预处理器标志:-D选项......是否要尝试梳理它取决于您)。


推荐阅读