7

设置

我有一个使用 CMake 构建和运行良好的项目。我的项目设置是这样的:

├── CMakeLists.txt
|
├── include/
│   └── standalone/
│       └── x.hpp
|
├── src/
    └── standalone/
        └── main.cpp

我的标题的内容是这样的:

// ------x.hpp--------
#pragma once
#include <iostream>

class X
{
public:
  void hello()
  {
    std::cout << "Hello World!\n"; // needs <iostream>
  }
};

// -------main.cpp-------
#include <standalone/x.hpp>

int main()
{
  X x;
  x.hello();
}

我使用以下CMakeLists.txt

cmake_minimum_required(VERSION 2.8)
project(standalone)

###############################################################
# Compiler settings
###############################################################

# use C++11 features
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")

# set warning level
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Werror -pedantic -pedantic-errors -Wall -Wextra")

###############################################################
# header dependencies
###############################################################

# compile against project headers
include_directories(${PROJECT_SOURCE_DIR}/include)

# the header files
file(GLOB_RECURSE header_files FOLLOW_SYMLINKS "include/*.hpp")

# the source files
file(GLOB_RECURSE source_files FOLLOW_SYMLINKS "src/*.cpp")

###############################################################
# build target
###############################################################

# the program depends on the header files and the source files
add_executable(main ${header_files} ${source_files})

命令序列mkdir build, cd build, cmake ..,make./main正确打印“Hello World!” 没有警告。

问题

上面的设置是正确的。但是假设<iostream>不包括在内,x.hpp而是在main.cpp代替。那么程序仍然会正确构建,但x.hpp不会是独立的标题。所以我想测试我的头文件的自给自足,即我想为每个头文件编译一个小测试程序

#include "some_header.hpp"
int main() {}

但是,如果我将以下部分添加到CMakeList.txt

###############################################################
# header self-sufficiency
###############################################################

set(CMAKE_REQUIRED_FLAGS ${CMAKE_CXX_FLAGS})
message("REQUIRED_FLAGS=${CMAKE_REQUIRED_FLAGS}")
set(CMAKE_REQUIRED_INCLUDES ${CMAKE_SOURCE_DIR}/include)
message("REQUIRED_INCLUDES=${CMAKE_REQUIRED_INCLUDES}")
include(CheckIncludeFiles)
check_include_files("${header_files}" IS_STANDALONE)

${header_files}正确地展开到标头x.hpp,但check_include_files()命令没有正确编译它

REQUIRED_FLAGS= -std=c++11 -Werror -pedantic -pedantic-errors -Wall -Wextra
REQUIRED_INCLUDES=/home/rein/projects/standalone/include
-- Looking for include file /home/rein/projects/standalone/include/standalone/x.hpp
-- Looking for include file /home/rein/projects/standalone/include/standalone/x.hpp - not found.

问题

显然我缺少一些配置变量,让 CMake 在正确的位置进行搜索。即使对于正确的标题,check_include_files()也不起作用。我需要做什么才能完成这项工作?只有当正确的标题被认为是正确的,我才能继续测试不正确的标题。

注意除非绝对必要,否则我对直接调用的循环TRY_COMPILE或类似的shell脚本或精心制作的CMake不感兴趣。AFAICS,这就是CheckIncludeFiles模块的用途,如果可能的话,我想知道如何正确配置它。

4

2 回答 2

6

对于 C++ 头文件而不是 C 头文件,您需要使用CheckIncludeFileCXX. 此模块的不同之处还在于CheckIncludeFiles您一次只能将一个文件传递给此宏。

也可能是您需要设置CMAKE_REQUIRED_INCLUDES或文档中列出的其他变量之一。

例如,如果您的某些标题相互引用(如 中#include "h1.hpp"),那么您将需要:

set(CMAKE_REQUIRED_FLAGS ${CMAKE_CXX_FLAGS})
set(CMAKE_REQUIRED_INCLUDES ${CMAKE_SOURCE_DIR}/include)
include(CheckIncludeFileCXX)
foreach(project_header ${project_headers})
  get_filename_component(header_name ${project_header} NAME_WE)
  check_include_file_cxx("${project_header}" ${header_name}_IS_STANDALONE)
endforeach()

CMake wiki 文章How To Write Platform Checks中提供了更多信息。

于 2013-05-07T22:31:52.037 回答
1

浏览CMakeFiles/CMakeError.log输出后,似乎有两个基本限制check_include_files()无法通过允许的配置选项克服(似乎对这些点进行了一些确认):

  1. 总是调用C 编译器,而不是 C++ 编译器。这意味着这-std=C++11不是要传递的有效命令行选项。
  2. 即使使用手动添加,标准系统包含路径(/usr/include等)也会被忽略set(CMAKE_REQUIRED_INCLUDES ${CMAKE_SOURCE_DIR}/include "/usr/include")。因此无法正确找到#include <iostream>内部。x.hpp

错误日志有这个片段:

/usr/bin/gcc   -I/home/rein/projects/standalone/include    -o CMakeFiles/cmTryCompileExec4052324189.dir/CheckIncludeFiles.c.o   -c /home/rein/projects/standalone/build/CMakeFiles/CMakeTmp/CheckIncludeFiles.c
In file included from /home/rein/projects/standalone/build/CMakeFiles/CMakeTmp/CheckIncludeFiles.c:2:0:
/home/rein/projects/standalone/include/standalone/x.hpp:2:20: fatal error: iostream: No such file or directory
compilation terminated.

这本质上将检查自给自足的想法限制在系统 C 头文件中。看来我将不得不寻找shell 脚本替代品,例如关于这个SO question的那个。

更新:感谢@Fraser 的更新答案,我的问题现在通过check_include_file_cxx使用编译器标志和包含预先设置的路径来解决。

于 2013-05-08T18:15:12.997 回答