Outils pour utilisateurs

Outils du site


prog:cmake

Différences

Ci-dessous, les différences entre deux révisions de la page.

Lien vers cette vue comparative

Les deux révisions précédentesRévision précédente
Prochaine révision
Révision précédente
prog:cmake [2024/03/25 09:34] – Création de "Rustines à appliquer systématiquement" rootprog:cmake [2024/12/12 11:33] (Version actuelle) – [Compilation] : ajout des options de compilation root
Ligne 12: Ligne 12:
 git clone https://gitlab.kitware.com/cmake/cmake git clone https://gitlab.kitware.com/cmake/cmake
 cd cmake cd cmake
-cmake -B build+cmake -B build -DCMAKE_USE_OPENSSL:BOOL=ON -DBUILD_TESTING:BOOL=OFF -DCMAKE_BUILD_TYPE=Release
 cmake --build build --parallel 16 cmake --build build --parallel 16
 sudo cmake --install build sudo cmake --install build
Ligne 26: Ligne 26:
  
 <code bash> <code bash>
 +# Etape de configuration.
 CXXFLAGS="..." cmake -S . -B build -DCMAKE_BUILD_TYPE="Release" CXXFLAGS="..." cmake -S . -B build -DCMAKE_BUILD_TYPE="Release"
 +# Etape de compilation.
 cmake --build build/ --target all --parallel --config "Release" cmake --build build/ --target all --parallel --config "Release"
 </code> </code>
Ligne 37: Ligne 39:
 En compilant pour ''Visual Studio'', ''-DCMAKE_CXX_FLAGS'' écrase la valeur qu'aurait dû générer ''cmake'' (''/DWIN32 /D_WINDOWS /W3 /GR /EHsc'' par défaut). En compilant pour ''Visual Studio'', ''-DCMAKE_CXX_FLAGS'' écrase la valeur qu'aurait dû générer ''cmake'' (''/DWIN32 /D_WINDOWS /W3 /GR /EHsc'' par défaut).
  
-En utilisant ''CXXFLAGS'', les flags sont ajoutés au début (''%%...%% /DWIN32 /D_WINDOWS /W3 /GR /EHsc'').+En utilisant la variable d'environnement ''CXXFLAGS'', les flags sont ajoutés au début (''%%...%% /DWIN32 /D_WINDOWS /W3 /GR /EHsc'').
  
 [[https://stackoverflow.com/questions/44284275/passing-compiler-options-cmake|Passing compiler options cmake]] {{ :prog:cmake:passing_compiler_options_cmake_-_stack_overflow_05_10_2022_11_12_25_.html |Archive du 31/05/2017 le 05/10/2022}} [[https://stackoverflow.com/questions/44284275/passing-compiler-options-cmake|Passing compiler options cmake]] {{ :prog:cmake:passing_compiler_options_cmake_-_stack_overflow_05_10_2022_11_12_25_.html |Archive du 31/05/2017 le 05/10/2022}}
 </WRAP> </WRAP>
  
 +  * Effacer le cache et regénérer les CMakeLists.txt
 +
 +Il faut effacer le fichier ''CMakeCache.txt'' et le dossier ''CMakeFiles'' dans le dossier de compilation. Cela permet de conserver les binaires / fichiers objets déjà compilés.
  
   * Différences entre les générateurs   * Différences entre les générateurs
  
-En fonction des générateurs (''Unix Makefiles'' pour Linux et ''Visual Studio 17 2022'' pour Windows), il existe plusieurs types de build (''CMAKE_BUILD_TYPE'') :+Certaines générateurs (''Unix Makefiles'' et ''Ninja''ne supportent qu'un seul type de compilation à la fois. Il faut définir le type de compilation avec ''CMAKE_BUILD_TYPE''.
  
-  - Debug +D'autres générateurs (''Visual Studio'', ''Xcode'', ''Ninja Multi-Config'') acceptent plusieurs types de compilation. Il faut passer par ''CMAKE_CONFIGURATION_TYPES'' qui vaut par défaut ''Debug;Release;MinSizeRel;RelWithDebInfo''.
-  - Release +
-  - RelWithDebInfo+
-  - MinSizeRel+
  
-''Debug'' est évidemment ''Debug'' et les 3 autres sont dans la catégorie ''Release''.+''CMAKE_BUILD_TYPE'' est indispensable lors de la génération des CMakeLists.txt pour les générateurs mono-build.
  
-''CMAKE_BUILD_TYPE'' doit être passé dans le premier appel au ''cmake -S . -B build''. C'est indispensable pour ''Unix Makefiles'' car le générateur ne supporte qu'un seul type de build à la fois. C'est facultatif pour ''Visual Studio XX YYYY'' car il supporte plusieurs configurations à la fois. Cependant, il est quand même conseillé de l'utiliser à chaque fois pour la cohérence. +''%%--%%config'' est indispensable lors de la compilation pour les générateurs multi-build.
- +
-''%%--%%config'' doit être passé pour chaque ''%%cmake --build build%%'' pour ''Visual Studio XX YYYY'' mais est facultatif pour ''Unix Makefiles''. Il est aussi conseillé de toujours le spécifier.+
  
 ====Cross compilation pour Android==== ====Cross compilation pour Android====
Ligne 71: Ligne 71:
 </code> </code>
  
-La version de ''CMAKE_SYSTEM_VERSION'' doit correspondre à minSdkVersion du fichier ''build.gradle''.+La version de ''CMAKE_SYSTEM_VERSION'' doit correspondre à ''minSdkVersion'' du fichier ''build.gradle''.
  
 [[https://developer.android.com/ndk/guides/cmake|CMake]] {{ :prog:cmake:cmake_android_ndk_android_developers_26_09_2022_13_53_29_.html |Archive du 25/04/2022 le 26/09/2022}} [[https://developer.android.com/ndk/guides/cmake|CMake]] {{ :prog:cmake:cmake_android_ndk_android_developers_26_09_2022_13_53_29_.html |Archive du 25/04/2022 le 26/09/2022}}
 +
 =====CMakeLists.txt===== =====CMakeLists.txt=====
 ====Cas courants==== ====Cas courants====
 +
 +Voir une version à jour avec tous les commentaires dans [[https://github.com/bansan85/cmake-library/tree/main/library/lib|cmake-library]].
  
 ===add_library=== ===add_library===
Ligne 83: Ligne 86:
 <code cmake> <code cmake>
 add_library(Librairie sources.cpp sources.hpp CMakeLists.txt) add_library(Librairie sources.cpp sources.hpp CMakeLists.txt)
 +add_library(NameSpace::Librairie ALIAS Librairie)
 +set(Librairie_SRC ${CMAKE_CURRENT_SOURCE_DIR}/sources.cpp
 +                  ${CMAKE_CURRENT_SOURCE_DIR}/CMakeLists.txt)
 +set(Librairie_HEADER ${CMAKE_CURRENT_SOURCE_DIR}/sources.h)
 +target_sources(
 +  Librairie
 +  PRIVATE ${Librairie_SRC}
 +  PUBLIC FILE_SET HEADERS BASE_DIRS ${CMAKE_CURRENT_SOURCE_DIR}/ FILES ${Librairie_HEADER})
 </code> </code>
  
Ligne 99: Ligne 110:
 ''INTERFACE'' : ''Librairie2'' n'est pas nécessaire pour compiler ''Librairie'' mais est nécessaire pour qu'un exécutable puisse se lier à ''Librairie''. Cela est surtout nécessaire pour faire une librairie header-only. ''INTERFACE'' : ''Librairie2'' n'est pas nécessaire pour compiler ''Librairie'' mais est nécessaire pour qu'un exécutable puisse se lier à ''Librairie''. Cela est surtout nécessaire pour faire une librairie header-only.
  
-  * Depuis un binaire+  * Depuis une librairie dynamique précompilée
  
 <code cmake> <code cmake>
Ligne 121: Ligne 132:
  
 ''IMPORTED_IMPLIB'' : uniquement pour Windows et des librairies ''SHARED''. Le fichier ''.dll'' ne contient pas les symboles. Il lui faut donc le ''.lib'' associé. Attention, le fichier ''.lib'' associé au ''.dll'' (''SHARED'') n'est pas le même que le ''.lib'' compilé en ''STATIC''. ''IMPORTED_IMPLIB'' : uniquement pour Windows et des librairies ''SHARED''. Le fichier ''.dll'' ne contient pas les symboles. Il lui faut donc le ''.lib'' associé. Attention, le fichier ''.lib'' associé au ''.dll'' (''SHARED'') n'est pas le même que le ''.lib'' compilé en ''STATIC''.
 +
 ===Librairie header-only=== ===Librairie header-only===
  
-Follow guide:+Suivre les guides :
  
 Exemple sans ''find_package'' : [[http://mariobadr.com/creating-a-header-only-library-with-cmake.html|Creating a header only library with cmake]] {{ :prog:cmake:mario_badr_creating_a_header-only_library_with_cmake_2021-07-11_08_48_29_.html |Archive du 2017 le 11/07/2021}} Exemple sans ''find_package'' : [[http://mariobadr.com/creating-a-header-only-library-with-cmake.html|Creating a header only library with cmake]] {{ :prog:cmake:mario_badr_creating_a_header-only_library_with_cmake_2021-07-11_08_48_29_.html |Archive du 2017 le 11/07/2021}}
Ligne 141: Ligne 153:
  
 ===Activation de l'optimisation global du lieur=== ===Activation de l'optimisation global du lieur===
 +
 Le LTCG (Link Time Code Generation) est identique à l'option lto (Link Time Optimization) de gcc. Le LTCG (Link Time Code Generation) est identique à l'option lto (Link Time Optimization) de gcc.
  
Ligne 169: Ligne 182:
  
 <code bash> <code bash>
--DCMAKE_C_COMPILER=/usr/bin/clang -DCMAKE_CXX_COMPILER=/usr/bin/clang++ -DCMAKE_AR=/usr/bin/llvm-ar -DCMAKE_AS=/usr/bin/llvm-as -DCMAKE_RANLIB=/usr/bin/llvm-ranlib+-DCMAKE_C_COMPILER=/usr/bin/clang -DCMAKE_CXX_COMPILER=/usr/bin/clang++ -DCMAKE_AR=/usr/bin/llvm-ar -DCMAKE_AS=/usr/bin/llvm-as -DCMAKE_RANLIB=/usr/bin/llvm-ranlib -DCMAKE_LINKER_TYPE=LLD
 </code> </code>
 +
 +  * ''DWARF error: invalid or unhandled FORM value: 0x23''
 +
 +Ne pas spécifier ''CMAKE_LINKER_TYPE'' va entrainer cette erreur. [[https://github.com/llvm/llvm-project/issues/56994| C++17 leads to linker error with ld due to some strange debug info?]] {{ :prog:cmake:c_17_leads_to_linker_error_with_ld_due_to_some_strange_debug_info_issue_56994_llvm_llvm-project_5_31_2024_12_40_25_pm_.html |Archive du 08/08/2022 le 31/05/2024}}
 +
 +Sinon, il est aussi possible de mettre l'option de compilation ''-gdwarf-4'' pour les flags ''C'' et ''CXX''.
  
 ===FetchContent=== ===FetchContent===
Ligne 237: Ligne 256:
 </code> </code>
  
-====Cas spécifiques====+====Environnement de compilation et de sortie====
  
-===Exécuter une action qui n'est rattachée à aucun programme/target===+===Environnement de sortie===
  
-Il suffit de passer par une ''target'' intermédiaire.+<code cmake> 
 +if(WIN32) 
 +  set(OS "Windows"
 +elseif(UNIX AND NOT APPLE) 
 +  set(OS "Linux/Unix"
 +elseif(APPLE) 
 +  set(OS "macOS"
 +else() 
 +  set(OS "Unknown"
 +endif() 
 + 
 +if(CMAKE_SIZEOF_VOID_P EQUAL 8) 
 +  set(ARCHITECTURE "64-bit"
 +elseif(CMAKE_SIZEOF_VOID_P EQUAL 4) 
 +  set(ARCHITECTURE "32-bit"
 +else() 
 +  set(ARCHITECTURE "Unknown"
 +endif() 
 +</code> 
 + 
 +====Exécution d'une commande==== 
 + 
 +===En passant par une nouvelle target===
  
 <code cmake> <code cmake>
-# On crée une target qui sera ajoutée à la commande make par défaut. +add_custom_target(clang_fmt ALL COMMAND xxxxxx)
-add_custom_target(clang_fmt ALL+
-# On définir une commande à exécuter. +
-add_custom_command(TARGET clang_fmt COMMAND xxxxxx)+
 </code> </code>
  
-===Exécuter uniquement le préprocesseur===+===Associé une commande à une target existante===
  
-Il faut créer une target compilable en l'excluant de ''all''. Cela va créer des target intermédiaire, notamment ''xxx.cpp.i''. Il suffit alors de l'exécuter explicitement.+  * En passant par une target intermédiaire
  
 <code cmake> <code cmake>
-add_library(file_obj OBJECT EXCLUDE_FROM_ALL file.cpp) +add_custom_target(clang_fmt_dep COMMAND ...) 
-add_custom_target(file ALL COMMAND make file.cpp.i)+ 
 +add_dependencies(clang_fmt clang_fmt_dep)
 </code> </code>
  
-[[http://anadoxin.org/blog/generating-preprocessed-sources-in-cmake-projects.html|Generating preprocessed sources in CMake projects]{{ :prog:cmake:generating_preprocessed_sources_in_cmake_projects_-_antek_s_tech_blog_2020-07-21_18_01_14_.html |Archive du 21/11/2018 le 21/07/2020}}+  * Sans passer par une target intermédiaire 
 + 
 +<code cmake> 
 +add_custom_command(TARGET clang_fmt PRE_BUILD COMMAND ...) 
 +</code> 
 + 
 +  * En passant par un fichier qui sera généré 
 + 
 +<code cmake> 
 +add_custom_command(OUTPUT file COMMAND ...) 
 + 
 +add_dependencies(clang_fmt file) 
 +</code> 
 + 
 +===Spécificité au générateur Visual Studio=== 
 + 
 +Normalement, une commande échoue si elle ne renvoie pas 0. 
 + 
 +Mais Visual Studio va également analyser la sortie standard. Si le contenu de la sortie standard est faussement considéré comme une erreur, il faut rediriger la sortie standard vers NUL. 
 + 
 +<code cmake> 
 +if(CMAKE_GENERATOR MATCHES "Visual Studio"
 +  set(IGNORE_ERROR_IN_LOG ">" "NUL" "2>&1"
 +else() 
 +  set(IGNORE_ERROR_IN_LOG ""
 +endif() 
 + 
 +  add_custom_target(targ COMMAND ... ${IGNORE_ERROR_IN_LOG}) 
 +
 +</code> 
 + 
 +Ce problème peut se produire en lançant la commande `python -m ensurepip` car elle lance la sous-commande `pip install` et cette commande peut générer l'erreur : 
 + 
 +<code> 
 +"...\target.vcxproj" (default target) (5) -
 +(CustomBuild target) ->  
 +  CUSTOMBUILD : error : pip's dependency resolver does not currently take into account all the packages that are installedThis behaviour is the source of the following dependency conflicts. [...\target.vcxproj] 
 +  C:\Program Files\Microsoft Visual Studio\2022\Community\MSBuild\Microsoft\VC\v170\Microsoft.CppCommon.targets(254,5)error MSB8066Custom build for '...\file.rule;...\target.rule;...\CMakeLists.txt' exited with code -1[...\target.vcxproj] 
 + 
 +    15 Warning(s) 
 +    2 Error(s) 
 +</code>
  
 ===Dépendances dans un sous-dossier=== ===Dépendances dans un sous-dossier===
Ligne 315: Ligne 395:
   DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}-${version})   DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}-${version})
 </code> </code>
 +
 +====Divers====
 +
 +===Exécuter uniquement le préprocesseur===
 +
 +Il faut créer une target compilable en l'excluant de ''all''. Cela va créer des target intermédiaire, notamment ''xxx.cpp.i''. Il suffit alors de l'exécuter explicitement.
 +
 +<code cmake>
 +add_library(file_obj OBJECT EXCLUDE_FROM_ALL file.cpp)
 +add_custom_target(file ALL COMMAND make file.cpp.i)
 +</code>
 +
 +[[http://anadoxin.org/blog/generating-preprocessed-sources-in-cmake-projects.html|Generating preprocessed sources in CMake projects]] {{ :prog:cmake:generating_preprocessed_sources_in_cmake_projects_-_antek_s_tech_blog_2020-07-21_18_01_14_.html |Archive du 21/11/2018 le 21/07/2020}}
 +
  
 ====Tests==== ====Tests====
Ligne 462: Ligne 556:
 [[https://stackoverflow.com/questions/9298278/cmake-print-out-all-accessible-variables-in-a-script|CMake: Print out all accessible variables in a script]] {{ :prog:cmake:cmake_print_out_all_accessible_variables_in_a_script_-_stack_overflow_18_08_2022_14_09_34_.html |Archive du 15/02/2012 le 18/08/2022}} [[https://stackoverflow.com/questions/9298278/cmake-print-out-all-accessible-variables-in-a-script|CMake: Print out all accessible variables in a script]] {{ :prog:cmake:cmake_print_out_all_accessible_variables_in_a_script_-_stack_overflow_18_08_2022_14_09_34_.html |Archive du 15/02/2012 le 18/08/2022}}
  
-  * Afficher toutes les étapes, line par line, d'un run ''CMake'' (configuration uniquement)+  * Afficher toutes les étapes, ligne par ligne, d'un run ''CMake'' (configuration uniquement)
  
 Ajouter l'option ''%%--%%trace-expand''. Ajouter l'option ''%%--%%trace-expand''.
prog/cmake.1711355668.txt.gz · Dernière modification : 2024/03/25 09:34 de root