使用QT5_ADD_RESOURCES和使用CMake进行多线程编译时损坏的资源.cpp文件

Gui*_*ume 5 c++ qt multithreading cmake qt5

Qt的5.0版本带来了一组使用CMake构建Qt项目的简单命令.见http://qt-project.org/doc/qt-5/cmake-manual.html.需要使用命令QT5_ADD_RESOURCES包含项目的资源.

如果我的资源文件以例如Icon32.qrc命名,QT5_ADD_RESOURCES(RESOURCES Icon32.qrc)命令将自动将其转换为qrc_Icon32.cpp文件并定义$ {RESOURCES}变量,然后我将能够将其包含在正确的变量中目标.

这样做很有效,除了我在CDash中大约每20个构建一次得到编译错误.该错误通常具有以下形式:

/.../CMake/build/qrc_Icon32.cpp:272380:1: error: unknown type name 'qCleanupResources_Icon32'
Run Code Online (Sandbox Code Playgroud)

发生的事情是qrc_Icon32.cpp文件的最后一行的可变部分在通常应该在文件结尾之后重复,从而为编译器创建最后一个无意义的行.

记录CMake的作用,似乎QT5_ADD_RESOURCES的行为如下:每当它到达需要相关资源的项目时,它就会执行一个特定于编译目标的依赖 make文件,但仍会编写qrc_Icon32.cpp 在构建目录的根目录,这适用于所有目标.因此,如果并行编译两个目标,则两次调用rcc可能会同时写入同一文件,从而导致损坏.

我没有在网上找到关于此问题/功能的任何报告/讨论,所以我想知道我是否可能错过了一些内容:

有没有办法告诉CMake将生成的qrc_Icon32.cpp保存在每个目标的不同位置?更重要的是,是否有可能告诉CMake的调用RCC从主make文件只有一次,这样qrc_Icon32.cpp是后来适用于所有的目标?

我想解决方法是创建一个静态库,它将是唯一一个使用$ {RESOURCES}然后将该库链接到所有目标的库.但是,我认为CMake应该能够在使用多线程-j标志进行编译时正确管理其依赖项.


要重现此问题,请在空文件夹中创建包含以下内容的CMakeList.txt

CMAKE_MINIMUM_REQUIRED(VERSION 2.8.11)
PROJECT(SSCCE CXX)

set(CMAKE_PREFIX_PATH /usr/local/Qt-5.3.0 ${CMAKE_PREFIX_PATH})

set(CMAKE_AUTOMOC ON)
set(CMAKE_INCLUDE_CURRENT_DIR ON)

INCLUDE_DIRECTORIES(SYSTEM "/usr/local/Qt-5.3.0/include/QtCore")
find_package(Qt5Core REQUIRED)
QT5_ADD_RESOURCES(RESOURCES Icon32.qrc)

SET(LIBLIST gobject-2.0 X11-xcb Xi xcb-render-util SM ICE xcb-glx xcb-render xcb-atom xcb-property xcb-event dbus-1 xcb xcb-image xcb-icccm xcb-sync xcb-xfixes xcb-shm xcb-randr xcb-shape xcb-keysyms fontconfig freetype Xrender Xext X11 jpeg png Qt5::Core z m dl gthread-2.0 rt glib-2.0 GL pthread)

ADD_EXECUTABLE(FirstTarget Main1.cpp ${RESOURCES})
TARGET_LINK_LIBRARIES(FirstTarget ${LIBLIST})
ADD_EXECUTABLE(SecondTarget Main2.cpp ${RESOURCES})
TARGET_LINK_LIBRARIES(SecondTarget ${LIBLIST})
Run Code Online (Sandbox Code Playgroud)

然后使用创建Main1.cpp和Main2.cpp

#include <iostream>

using namespace std;

int main(int argc, char** argv) {
        std::cout<<"Hello World 1"<<std::endl;
        return 0;
}
Run Code Online (Sandbox Code Playgroud)

qrc文件是

<RCC>
    <qresource prefix="/">
        <file>Icon32/YourImage.png</file>
    </qresource>
</RCC>
Run Code Online (Sandbox Code Playgroud)

然后创建一个名为Icon32的文件夹,并添加一个选择名称为YourImage.png的png图像.

最后,创建一个构建目录,输入并运行:

cmake -DCMAKE_CXX_COMPILER=g++-4.8 -DCMAKE_CXX_FLAGS='-std=c++11 -fPIE' ..
make -j2
Run Code Online (Sandbox Code Playgroud)

输出应该是这样的

Scanning dependencies of target FirstTarget_automoc
Scanning dependencies of target SecondTarget_automoc
[ 10%] [ 20%] Automoc for target FirstTarget
Automoc for target SecondTarget
[ 20%] [ 20%] Built target FirstTarget_automoc
Built target SecondTarget_automoc
[ 30%] [ 40%] Generating qrc_Icon32.cpp
Generating qrc_Icon32.cpp
Scanning dependencies of target SecondTarget
Scanning dependencies of target FirstTarget
[ 50%] [ 60%] Building CXX object CMakeFiles/SecondTarget.dir/Main2.cpp.o
Building CXX object CMakeFiles/FirstTarget.dir/Main1.cpp.o
[ 70%] [ 80%] Building CXX object CMakeFiles/SecondTarget.dir/qrc_Icon32.cpp.o
Building CXX object CMakeFiles/FirstTarget.dir/qrc_Icon32.cpp.o
[ 90%] [100%] Building CXX object CMakeFiles/SecondTarget.dir /SecondTarget_automoc.cpp.o
Building CXX object CMakeFiles/FirstTarget.dir/FirstTarget_automoc.cpp.o
Linking CXX executable SecondTarget
Linking CXX executable FirstTarget
Run Code Online (Sandbox Code Playgroud)

您可以看到qrc_Icon32.cpp几乎同时在构建目录的根目录下创建了两次.但是,在FirstTarget.dir和SecondTarget.dir中正确创建了qrc_Icon32.cpp.o文件,因此没有冲突.

我的观点是:1)qrc_Icon32.cpp也应该在FirstTarget.dir和SecondTarget.dir中创建,或者2)它应该在构建目录的根目录下创建,但对于所有目标只能创建一次.

ste*_*ire 6

qt5_add_resources将文件写入CMAKE_CURRENT_BINARY_DIR,而不是CMAKE_BINARY_DIR

https://qt.gitorious.org/qt/qtbase/source/d953d9a4c3bdc5ed3b8d380c4b893b51b523bc50:src/corelib/Qt5CoreMacros.cmake#L205

同上Qt 4:

http://cmake.org/gitweb?p=cmake.git;a=blob;f=Modules/Qt4Macros.cmake;h=b1b12d68b07aac076719c681fb844f4f98ba8151;hb=HEAD#l212

更新:

使用您在更新中提供的源代码,可以查看问题.

http://www.cmake.org/pipermail/cmake/2008-October/024492.html

http://cmake.org/gitweb?p=cmake.git;a=commitdiff;h=0ece8f79

http://public.kitware.com/Bug/view.php?id=12311

解决方法是添加自定义目标并添加显式取决于此.

cmake_minimum_required(VERSION 2.8.11)

project(MyTest)

find_package(Qt5Core)

qt5_add_resources(RSCS somefile.qrc)
add_custom_target(gen_qrc DEPENDS ${RSCS})

add_executable(foo foo.cpp ${RSCS})
add_dependencies(foo gen_qrc)
add_executable(bar bar.cpp ${RSCS})
add_dependencies(bar gen_qrc)
Run Code Online (Sandbox Code Playgroud)

CMake 3.0具有AUTORCC功能:

http://www.cmake.org/cmake/help/v3.0/manual/cmake-qt.7.html#autorcc

它还将生成的qrc_文件放在当前的构建目录中.CMake 3.1将把它放在一个特定于目标的目录中,这个问题就会消失:

http://cmake.org/gitweb?p=cmake.git;a=commitdiff;h=33774ca2