我尝试使用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
答案 0 :(得分:0)
由于file1.hpp
和file2.hpp
包含了您的precompiled_header.hpp
,因此您需要预先编译它们以使其有效。
来自gcc预编译标题documentation:
一旦看到第一个C令牌,就无法使用预编译的头。您可以在预编译头之前使用预处理程序指令; 您不能在另一个标题中包含预编译的标题。
答案 1 :(得分:0)
如果我首先在main.cpp
中的另一个头之前包含预编译头,这似乎有效#ifdef USE_PRECOMPILED
#include "precompiled_header.hpp"
#undef USE_PRECOMPILED
#endif