CMake Without the Headache
As C++ developers, we’re often stuck with CMake, whether we like it or not. Sure, there are better alternatives out there, that are even using real programming languages, but I’m not here to whine about CMake. I use it for all of my C++ projects and often recommend it to teams I work with. Maybe it’s Stockholm syndrome, clinging to the “sliver of standardization” in the land of C++!
While there are many great and comprehensive resources already written on the topic, the goal of this post is to offer a quick and pain-free guide to use Modern CMake features in your project. For more in-depth exploration, I highly recommend you to consume all the content by Craig Scott on CMake. A list of resources is at the end of this post.
Now, let’s dive into how to make CMake work for you, without the headaches.
Modern CMake
Here are the essentials for Modern CMake:
- Think of everything in terms of “Compilation Targets”:
add_executable
add_library
- Use target-based CMake commands:
target_sources
target_include_directories
target_compile_options
target_link_options
target_link_libraries
target_compile_definitions
target_precompile_headers
set_target_properties
- Avoid using non-target based, global CMake commands like:
include_directories
link_libraries
Focusing on these target-based commands will inevitably make your build system modular and target based.
How find_package
works?
One of the most important CMake functions to understand is find_package
. It’s used to locate and link external libraries to your project.
find_package(SDL2_ttf REQUIRED)
target_link_libraries(myapp PRIVATE SDL2_ttf::SDL2_ttf)
But how does this magic work? When you call find_package
, CMake searches for corresponding “Finder Scripts” on your system. These scripts tell CMake how to locate and configure packages. Let’s find the source of magic. To see the scripts available on your system, run:
find / -name "*-config.cmake" 2>/dev/null
find / -name "Find*.cmake" 2>/dev/null
You might see paths like:
/usr/share/cmake/bash-completion/bash-completion-config.cmake
/usr/lib/x86_64-linux-gnu/cmake/harfbuzz/harfbuzz-config.cmake
/usr/lib/x86_64-linux-gnu/cmake/SDL2_ttf/sdl2_ttf-config.cmake
/usr/lib/x86_64-linux-gnu/cmake/SDL2/sdl2-config.cmake
...
These “Finder Scripts” are what make find_package
work. File name of these scripts can either be in the form of mypkg-config.cmake
or Findmypkg.cmake
.
Finder Scripts
CMake provides built-in finder scripts for many common libraries. Some older scripts, like FindLua
, define global variables (e.g., LUA_LIBRARIES
) that you use in your target_link_libraries
call. You can read the docs to see the list of variables it defines.
find_package(Lua REQUIRED)
target_link_libraries(myapp PRIVATE ${LUA_LIBRARIES})
Newer scripts, like FindSDL
, create CMake targets you can use directly:
find_package(SDL2 REQUIRED CONFIG REQUIRED COMPONENTS SDL2)
target_link_libraries(myapp PRIVATE SDL2::SDL2)
You can see the full list of built-in finder scripts from CMake Docs.
Adding Dependencies
There are a few ways to manage dependencies in CMake:
find_package
:- Locates system or distro-provided packages.
- Requires package to be installed and have a finder script.
git submodule
andadd_subdirectory
:- Adds dependencies with a
CMakeLists.txt
file. - Can add any 3rd party dependency having
CMakeLists.txt
file. - Build every target locally.
- Adds dependencies with a
FetchContent
orExternalProject_Add
:- Similar to
git submodule
andadd_subdirectory
. - Eliminates the need for
git submodule
management. - Can add non-CMake projects, providing build scripts.
- Similar to
There are also package managers you can use. These generally rely on find_package
and work with a custom CMake toolchain provided by package manager. They make your dependencies distro/system-free. Most popular ones are vcpkg and Conan.
Personally, I use CPM (CMake Package Manager) for dependencies that are compatible with it and fall back to system-provided packages for everything else. The popularity of package managers is often overblown, especially when you have Docker containers and the ability to use your distro’s package manager across different environments.
My pio_add_target
Helper CMake Function
To streamline my CMake workflow, I created a helper function, pio_add_target
, which simplifies adding targets with common configurations. Here’s how it works:
pio_add_target(
NAME "myapp"
IS_EXE
SRC_PRIVATE ${MYAPP_SRC_FILES}
INCLUDE_PUBLIC "include"
INCLUDE_PRIVATE "src"
LINK_PRIVATE "SDL2::SDL2;SDL2_ttf::SDL2_ttf")
It organizes your target setup into a single function call. It takes care of:
- Adding sanitizer support
- Setting compiler options
- Supporting both executables and libraries
- Enforcing C++17 and C11 standards
- Enabling
clang-tidy
- Passing remaining arguments to relevant CMake commands
It uses cmake_parse_arguments
to take in named arguments just like other CMake commands. Figuring how to use the cmake_parse_arguments
causes the programmer Agonizing Pain I must say… Stuff like compiler options and sanitizers are hard-coded but you can easily change that to your use case if you wish.
Here’s the full function implementation:
function(pio_add_target)
cmake_parse_arguments(
PARSED_ARGS
"IS_EXE"
"NAME"
"INCLUDE_PUBLIC;INCLUDE_PRIVATE;LINK_PUBLIC;LINK_PRIVATE;SRC_PUBLIC;SRC_PRIVATE"
${ARGN})
set(cc_sanitizers
-fsanitize=address
# -fsanitize=memory -fsanitize=thread
-fsanitize=undefined)
set(cc_opts
${cc_sanitizers}
# -w
-fno-rtti -fno-exceptions -Wall -Wextra
-Wshadow)
if(${PARSED_ARGS_IS_EXE})
pio_log("adding executable: ${PARSED_ARGS_NAME}")
add_executable(${PARSED_ARGS_NAME})
else()
pio_log("adding library: ${PARSED_ARGS_NAME}")
add_library(${PARSED_ARGS_NAME})
endif()
# set language standards
target_compile_features(${PARSED_ARGS_NAME} PRIVATE cxx_std_17)
set_target_properties(${PARSED_ARGS_NAME} PROPERTIES CXX_EXTENSIONS OFF)
set_target_properties(${PARSED_ARGS_NAME} PROPERTIES C_STANDARD 11)
set_target_properties(${PARSED_ARGS_NAME} PROPERTIES C_EXTENSIONS OFF)
# compiler options
target_compile_options(${PARSED_ARGS_NAME} PRIVATE ${cc_opts})
# sources
target_sources(${PARSED_ARGS_NAME} PUBLIC ${PARSED_ARGS_SRC_PUBLIC})
target_sources(${PARSED_ARGS_NAME} PRIVATE ${PARSED_ARGS_SRC_PRIVATE})
# includes
target_include_directories(${PARSED_ARGS_NAME}
PUBLIC ${PARSED_ARGS_INCLUDE_PUBLIC})
target_include_directories(${PARSED_ARGS_NAME}
PRIVATE ${PARSED_ARGS_INCLUDE_PRIVATE})
# links
target_link_libraries(${PARSED_ARGS_NAME} PUBLIC ${PARSED_ARGS_LINK_PUBLIC})
target_link_libraries(${PARSED_ARGS_NAME} PRIVATE ${PARSED_ARGS_LINK_PRIVATE})
# clang-tidy
set_target_properties(${PARSED_ARGS_NAME} PROPERTIES CXX_CLANG_TIDY
"clang-tidy")
set_target_properties(${PARSED_ARGS_NAME} PROPERTIES C_CLANG_TIDY
"clang-tidy")
# sanitizers
target_link_options(${PARSED_ARGS_NAME} PRIVATE ${cc_sanitizers})
endfunction(pio_add_target)