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 and add_subdirectory:
    • Adds dependencies with a CMakeLists.txt file.
    • Can add any 3rd party dependency having CMakeLists.txt file.
    • Build every target locally.
  • FetchContent or ExternalProject_Add:
    • Similar to git submodule and add_subdirectory.
    • Eliminates the need for git submodule management.
    • Can add non-CMake projects, providing build scripts.

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)

Resources