共享库中的cmake预编译头

时间:2015-11-09 13:25:45

标签: gcc cmake precompiled

我尝试使用gcc和cmake预编译我的标头。

我知道像cotire或customPCH这样的脚本可以做到,但我想编写自己的脚本。

项目非常简单:有一个包含一些文件的共享库,以及在这些文件中定义的主要调用函数。 我已经设法构建库(带有预编译的头),但是在构建main(" precompiled.hpp",没有这样的文件或目录)时,我有一个奇怪的编译错误,即使预编译的头是在包含搜索路径(库使用相同的包含路径,不抱怨)

[100%] Building CXX object CMakeFiles/prog.dir/src/bin/main.cpp.o
/usr/bin/c++ -DUSE_PRECOMPILED -Isrc -Ibuild -Winvalid-pch -fPIC -o CMakeFiles/prog.dir/src/bin/main.cpp.o -c src/bin/main.cpp
src/lib/dir2/file2.hpp:7:42: fatal error: build/precompiled_header.hpp: No such file or directory
     #include "precompiled_header.hpp"

我在这里设置了一个可用的玩具示例:git clone -b toy https://bitbucket.org/mrdut/precompiled_header.git

有人有任何建议吗?

提前致谢,

以下是我项目的内容:

run.sh
CMakeLists.txt
src/
   lib/
      dir1/file1.cpp, file1.hpp
      dir2/file2.cpp, file2.hpp
   bin/main.cpp

run.sh:

mkdir build
cd build/
cmake .. && make VERBOSE=1 && ./bin/prog
cd ..

CMakeslist.txt:

cmake_minimum_required(VERSION 2.8.11)

project('PRECOMPILED' CXX)

set(CMAKE_RUNTIME_OUTPUT_DIRECTORY bin)
set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY lib)
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY lib)

add_compile_options(-Winvalid-pch)

set(use_precompiled_header 1) # <= everything is fine if not using precompiled header
set(executable_name prog)
set(library_name erty)
set(precompiled_header_name precompiled_header )

###########
# library #
###########
add_library(${library_name} SHARED src/lib/dir1/file1.cpp src/lib/dir2/file2.cpp)
target_include_directories(${library_name} PUBLIC src )

##############
# executable #
##############
add_executable(${executable_name} src/bin/main.cpp)
target_include_directories(${executable_name} PUBLIC src )
target_link_libraries(${executable_name} ${library_name})
target_compile_options(${executable_name} PRIVATE "-fPIC") # because we are building SHARED library

######################
# precompiled header #
######################
if( ${use_precompiled_header} )

    set(precompiled_header_dir ${CMAKE_CURRENT_BINARY_DIR} )
    set(precompiled_header_src ${CMAKE_CURRENT_SOURCE_DIR}/src/header.hpp )
    set(precompiled_header_dst precompiled_header.hpp.gch )

    add_custom_target( ${precompiled_header_name} ALL DEPENDS ${precompiled_header_dst} )
    add_custom_command( OUTPUT ${precompiled_header_dst}
                            COMMAND ${CMAKE_CXX_COMPILER} -fPIC -DUSE_PRECOMPILED ${precompiled_header_src} -o ${precompiled_header_dst}
                            MAIN_DEPENDENCY ${precompiled_header_src}
                            WORKING_DIRECTORY ${precompiled_header_dir}
                            VERBATIM )

    add_dependencies( ${library_name} ${precompiled_header_name} )

    target_compile_definitions(${library_name} PUBLIC -DUSE_PRECOMPILED )
    target_compile_definitions(${executable_name} PUBLIC -DUSE_PRECOMPILED )

    target_include_directories(${library_name} PUBLIC ${precompiled_header_dir})
    target_include_directories(${executable_name} PUBLIC ${precompiled_header_dir})

endif()

src / header.hpp:

#ifndef HEADER_HPP
#define HEADER_HPP
    #include <iostream>
#endif

src / bin / main.cpp:

#include "lib/dir1/file1.hpp"
#include "lib/dir2/file2.hpp"
int main(int argc, char** argv)
{
    std::cout << "\n -= " << argv[0] << " =-\n";
    libfunc1();
    libfunc2();
    return 0;
}

src / lib.dir1 / file1.cpp:

#include "lib/dir1/file1.hpp"
void libfunc1() { std::cout << "function 1" << std::endl; }

src / lib.dir1 / file1.hpp:

#ifndef LIBRARY_FILE_1_HPP
#define LIBRARY_FILE_1_HPP
    #ifndef USE_PRECOMPILED
        #include "header.hpp"
    #else
        #include "precompiled_header.hpp"
    #endif
    void libfunc1();
#endif

src / lib.dir2 / file2.cpp:

#include "lib/dir2/file2.hpp"
void libfunc2() { std::cout << "function 2" << std::endl; }

的src / lib.dir1 / file2.hpp:

#ifndef LIBRARY_FILE_2_HPP
#define LIBRARY_FILE_2_HPP
    #ifndef USE_PRECOMPILED
        #include "header.hpp"
    #else
        #include "precompiled_header.hpp"
    #endif
    void libfunc2();
#endif

2 个答案:

答案 0 :(得分:0)

由于file1.hppfile2.hpp包含了您的precompiled_header.hpp,因此您需要预先编译它们以使其有效。

来自gcc预编译标题documentation

  

一旦看到第一个C令牌,就无法使用预编译的头。您可以在预编译头之前使用预处理程序指令; 您不能在另一个标题中包含预编译的标题。

答案 1 :(得分:0)

如果我首先在main.cpp

中的另一个头之前包含预编译头,这似乎有效
#ifdef USE_PRECOMPILED
    #include "precompiled_header.hpp"
    #undef USE_PRECOMPILED
#endif