首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >cmake共享库中的预编译头

cmake共享库中的预编译头
EN

Stack Overflow用户
提问于 2015-11-09 21:25:45
回答 2查看 1.8K关注 0票数 0

我正在尝试使用gcc和cmake预编译头文件。

我知道有一些脚本可以做到这一点,比如cotire或customPCH,但我想写自己的脚本。

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

代码语言:javascript
复制
[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克隆-b玩具https://bitbucket.org/mrdut/precompiled_header.git

有人有什么建议吗?

提前谢谢你,

以下是我的项目内容:

代码语言:javascript
复制
run.sh
CMakeLists.txt
src/
   lib/
      dir1/file1.cpp, file1.hpp
      dir2/file2.cpp, file2.hpp
   bin/main.cpp

run.sh:

代码语言:javascript
复制
mkdir build
cd build/
cmake .. && make VERBOSE=1 && ./bin/prog
cd ..

CMakeslist.txt:

代码语言:javascript
复制
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:

代码语言:javascript
复制
#ifndef HEADER_HPP
#define HEADER_HPP
    #include <iostream>
#endif

src/bin/main.cpp:

代码语言:javascript
复制
#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:

代码语言:javascript
复制
#include "lib/dir1/file1.hpp"
void libfunc1() { std::cout << "function 1" << std::endl; }

src/lib.dir1/file1.hpp:

代码语言:javascript
复制
#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:

代码语言:javascript
复制
#include "lib/dir2/file2.hpp"
void libfunc2() { std::cout << "function 2" << std::endl; }

src/lib.dir1/file2.hpp:

代码语言:javascript
复制
#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
EN

回答 2

Stack Overflow用户

发布于 2015-11-10 03:53:33

因为file1.hppfile2.hpp包含您的precompiled_header.hpp,所以您还需要预编译它们以使其正常工作。

来自gcc预编译的headers documentation

一旦看到第一个C标记,就不能使用预编译头。可以在预编译头之前使用预处理器指令;不能在另一个头中包含预编译头。

票数 0
EN

Stack Overflow用户

发布于 2015-11-10 05:29:10

如果我在main.cpp中先包含预编译头,再包含另一个头,这似乎是可行的

代码语言:javascript
复制
#ifdef USE_PRECOMPILED
    #include "precompiled_header.hpp"
    #undef USE_PRECOMPILED
#endif
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/33610143

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档