我正在尝试使用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克隆-b玩具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.cpprun.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>
#endifsrc/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();
#endifsrc/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发布于 2015-11-10 03:53:33
因为file1.hpp和file2.hpp包含您的precompiled_header.hpp,所以您还需要预编译它们以使其正常工作。
来自gcc预编译的headers documentation
一旦看到第一个C标记,就不能使用预编译头。可以在预编译头之前使用预处理器指令;不能在另一个头中包含预编译头。
发布于 2015-11-10 05:29:10
如果我在main.cpp中先包含预编译头,再包含另一个头,这似乎是可行的
#ifdef USE_PRECOMPILED
#include "precompiled_header.hpp"
#undef USE_PRECOMPILED
#endifhttps://stackoverflow.com/questions/33610143
复制相似问题