Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to best handle data files with CMake?

Tags:

I've got a CMake project that contains code and a few data files (images to be precise).

My directory structure is like this:

  • src
  • data

src contains the source code, data the data files. CMake suggests out of source builds, so when I invoke make, I have the executable program, but not the data files, thus I cannot execute the program.

Of course, make install would copy my data files to the required location and make it work, therefore I develop like this right now:

  1. cmake -DCMAKE_INSTALL_DIR=dist
  2. <edit source code>
  3. make install
  4. dist/myprogram.exe

That's okay if I'm working with the command line and an editor, but I recently decided to move to Eclipse CDT. Generating an Eclipse project from CMake works great, but manually executing the install target from Eclipse is not so nice.

How do you people tackle this problem? Does your program have some clever algorithms to try and find its data directory even if it's not where the binary is? Or do you not use out of source builds?

like image 729
theone Avatar asked Aug 07 '10 16:08

theone


People also ask

Where do I put CMake files?

You can install them into /usr/share/cmake/SomeProject/ folder, for example.

What is CMake good for?

CMake handles the difficult aspects of building software such as cross-platform builds, system introspection, and user customized builds, in a simple manner that allows users to easily tailor builds for complex hardware and software systems.

How do CMake files work?

CMake is a meta build system that uses scripts called CMakeLists to generate build files for a specific environment (for example, makefiles on Unix machines). When you create a new CMake project in CLion, a CMakeLists. txt file is automatically generated under the project root.


3 Answers

configure_file should solve that problem.

I have a CMakeLists.txt file in my data directory which contains the following:

configure_file(data_file ${CMAKE_CURRENT_BINARY_DIR}/data_file COPYONLY)

This copies the specified file into the build directory when cmake is invoked, so it is available in the same location even in out of source builds.

configure_file does not support directories however while the file command does:

file(COPY assets DESTINATION ${CMAKE_CURRENT_BINARY_DIR})
like image 146
fhd Avatar answered Oct 12 '22 06:10

fhd


And if copying the files takes too much time (they are images...) you could make it even better by creating a "custom" data_header.h with configure_file which contains the paths to the data still in your source-directory.

This is what I do: I have a file "global_build_config.h.in" in my source, containing the following:

const char* const global_testdatapath = "@Test_Data_Path@";

and then use configure_file in CMake:

# Assume CMake knows a variable Test_Data_Path, it will be filled in automatically
# in the generated config/Global_Build_Config.h
configure_file( Global_Build_Config.h.in ${CMAKE_BINARY_DIR}/config/Global_Build_Config.h )
# The config directory should be added as a include-searchpath
include_directories( ${CMAKE_BINARY_DIR}/config/ )

I can then #include "Global_Build_Config.h" in my cpp files and refer to the fixed path.

like image 26
André Avatar answered Oct 12 '22 05:10

André


Your question is a bit old, but in case you're still interested (or someone else), I have a similar scenario where I copy testdata for a unit-test target:

add_custom_command( TARGET ${UTEST_EXE_NAME}
    POST_BUILD
    COMMAND ${CMAKE_COMMAND} -E echo "Copying unit test data.."
    COMMAND ${CMAKE_COMMAND} -E copy_directory ${CMAKE_HOME_DIRECTORY}/utest/testdata ${CMAKE_BINARY_DIR}
    )

So the main idea is to use a post-build target, and it is executed after each build. For me, it's not much data, and the filesystem caches it, so I don't feel the copy process at all. You could probably enhance this by copying with copy_if_different. In that case, however, you have to create a list of your image files and write a loop, because the command is file based. With the GLOB command, this shouldn't be hard to do if you need to.

like image 20
sebkraemer Avatar answered Oct 12 '22 07:10

sebkraemer