Skip to content

Latest commit

 

History

History
149 lines (116 loc) · 7.8 KB

README.md

File metadata and controls

149 lines (116 loc) · 7.8 KB

CMake-Modules

This repo is a common place to store useful cmake modules, including customized Find files and utility modules. It is intended to be incorporated into projects as a subtree.

Usage

First add the cmake-module repo as a remote, so you can more easily reference it

  git remote add -f cmake-modules-repo [email protected]:leapmotion/cmake-module.git

To setup cmake-modules in your repository (only run once as a setup step):

  git subtree add --prefix cmake-modules cmake-modules-repo develop

To update the copy of cmake-modules in your repository from the latest code in the cmake-modules repo:

  git fetch cmake-modules-repo develop
  git subtree pull --prefix cmake-modules cmake-modules-repo develop

To push changes from your repository upstream into the cmake-module repo:

  git subtree push --prefix cmake-modules cmake-modules-repo <branch>
  Open a pull request to merge <branch> to develop

For more information on subtrees see Atlassian's Git Subtree Tutorial

Best Practices

Rule 1: Always check the docs & existing modules for examples

Documentation for CMake is rapidly improving. See if you can find answers in the links below.

Naming conventions
  • UPPER_CASE identifiers are public
  • Mixed_CASE identifiers are also public assuming the first word is a namespace, as with package find modules
  • _lower_idents with preceding underscores should be used for private, temporary variables
  • lower_idents should also be used for function calls
  • UPPER_CASE should be used for naming arguments to functions (see CMakeParseArguments)
General Guidelines
  1. Prefer setting target properties over global or directory properties, eg. use target_include_directories over include_directories.

  2. Prefer Generator expressions over PROPERTY_ variants wherever allowed. see Generator Expressions Manual

  3. install() is meant to define a step to package the build products of a library, and install them on the host system for use by cmake by adding it to the Package Registry. For steps such as copying .DLLs or resource files to the appropriate location so a build can be run and debugged at all, use add_custom_command( POST_BUILD ...)

  4. Never manually add a library directly in a CMakeLists.txt file (eg, via find_library or find_path). Instead, write a Find.cmake module that does that instead. See the guidelines for writing find modules further down, as well as the official Module Developer Docs

  5. Avoid the target_link_libraries signatures which use[debug|optimized|general]. Use Generator Expressions instead.

  6. Make use of INTERFACE targets and other pseudo targets to specify usage requirements. See

    • Pseudo Targets for information on what Pseudo Targets and Usage Requirements in general are.
    • Packages for information on what packages are and how to write Config.cmake files that provide IMPORT targets.
Creating New Projects

This assumes that you have added cmake-modules as a subtree in a folder at the root of your project.

Directory Structure

Root: A global CMakeLists.txt file containing global settings for all sub-projects. If you have only one sub project, you may add and configure it at the end of the root cmakelists file. Don't call find_package at the global level - use target_package (Defined in TargetImportedLibraries) instead. This has the additional benifit of setting up the appropriate post-build steps to handle copying shared libraries

Handling Resources in OS X Bundles

See CMake: Bundles and Frameworks as well as documentation for the MACOSX_PACKAGE_LOCATION

Root CMakeLists.txt
cmake_minimum_required(VERSION 3.0)
project(<Your project name>)
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake-modules")
set(CMAKE_CONFIGURATION_TYPES "Release;Debug" CACHE STRING "" FORCE) #Disables MinSizeRel & MaxSpeedRel
set(CMAKE_INCLUDE_CURRENT_DIR ON) #essentially the same as include_directories(.) in every subdir
set_property(GLOBAL PROPERTY USE_FOLDERS ON)

#Leap-Specific: We package any external libraries we're using in a single place, so add it to the default
#Search path. SDL-2.0.1 is used as a marker of something to look for.
find_path(EXTERNAL_LIBRARY_DIR "SDL-2.0.1"
  PATHS
    "$ENV{EXTERNAL_LIBRARY_DIR}"
    "$ENV{LIBRARIES_PATH}"
    "$ENV{PATH}"
    "/opt/local/Libraries"
  NO_DEFAULT_PATH)
list(INSERT CMAKE_PREFIX_PATH 0 "${EXTERNAL_LIBRARY_DIR}")

#Set any global flags for the project here, such as compiler settings you want used universally
#if(MSVC)  #Uncomment this if you want to use use /MT instead of /MD on Windows
#  add_compile_options($<$<CONFIG:debug>:/MTd> $<$<CONFIG:release>:/MT>)
#endif()

#Call add_subdirectory on relevant sub-directories

Project's CMakeLists.txt
set(<Project>_SRC <list of files here>)
set(<Project>_HEADERS <list of headers here>)

add_<executable/library>(<Project> ${<Project>_SRC} ${<Project>_HEADERS})
target_include_directories(<Project> PUBLIC .)
target_package(<Project> <Package> <Version> REQUIRED) #Repeat as nessecary

#set target's custom build settings here

#define install steps here

Writing Find Modules

For packages which do not provide Package.cmake commands, or ones that do not yet support import targets, you will have to author your own find module. This may be as simple as

include(${CMAKE_ROOT}/Modules/Find<Package>.cmake)
include(CreateImportTargetHelpers)
generate_import_target(<Package> <Library Type>)

but may also be much more complicated. See the existing Find modules in this repo for examples, and CreateImportTargetHelpers.cmake's documentation on how it works.

All Import targets generated by a Find module should use the :: naming convention where :: is an import target which will include all components specified as arguments to find_package. Private import targets should use ::_<lower_component>

Notes & TODOs

  • Better helpers and handling for modules which may be either SHARED or STATIC.

  • Proposed solution is to define SHARED_LIBRARY and STATIC_LIBRARY separately, then set LIBRARY to whichever one exists, and if both are defined then create an option so the user can choose. Alternatively, we could detect which are available, default to SHARED if both are, and expose an Xxx_IMPORT_TYPE cache variable that the user could override. This would also let us throw errors if the files for the desired type are unavailable.

  • Make a cmake module for installing executables/resources in a platform-agnostic way (e.g. Mac has a particular "bundle" format it follows).

  • The organization of the Components (with its component and library dependencies) should be implemented using a cmake module, so little redundant boilerplate is necessary.

  • Write variants of find_file and find_path which actually return ALL matches, instead of an ill-defined single match. This functionality is distinctly lacking in cmake, and causes nontrivial problems when trying to find files/paths.