CMakeLists.txt 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777
  1. # Ceres Solver - A fast non-linear least squares minimizer
  2. # Copyright 2023 Google Inc. All rights reserved.
  3. # http://ceres-solver.org/
  4. #
  5. # Redistribution and use in source and binary forms, with or without
  6. # modification, are permitted provided that the following conditions are met:
  7. #
  8. # * Redistributions of source code must retain the above copyright notice,
  9. # this list of conditions and the following disclaimer.
  10. # * Redistributions in binary form must reproduce the above copyright notice,
  11. # this list of conditions and the following disclaimer in the documentation
  12. # and/or other materials provided with the distribution.
  13. # * Neither the name of Google Inc. nor the names of its contributors may be
  14. # used to endorse or promote products derived from this software without
  15. # specific prior written permission.
  16. #
  17. # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  18. # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  19. # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  20. # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
  21. # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  22. # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  23. # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  24. # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  25. # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  26. # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  27. # POSSIBILITY OF SUCH DAMAGE.
  28. #
  29. # Authors: keir@google.com (Keir Mierle)
  30. # alexs.mac@gmail.com (Alex Stewart)
  31. cmake_minimum_required(VERSION 3.16...3.27)
  32. # On macOS, add the Homebrew prefix (with appropriate suffixes) to the
  33. # respective HINTS directories (after any user-specified locations). This
  34. # handles Homebrew installations into non-standard locations (not /usr/local).
  35. # We do not use CMAKE_PREFIX_PATH for this as given the search ordering of
  36. # find_xxx(), doing so would override any user-specified HINTS locations with
  37. # the Homebrew version if it exists.
  38. if (CMAKE_SYSTEM_NAME MATCHES "Darwin")
  39. find_program(HOMEBREW_EXECUTABLE brew)
  40. mark_as_advanced(FORCE HOMEBREW_EXECUTABLE)
  41. if (HOMEBREW_EXECUTABLE)
  42. # Detected a Homebrew install, query for its install prefix.
  43. execute_process(COMMAND ${HOMEBREW_EXECUTABLE} --prefix
  44. OUTPUT_VARIABLE HOMEBREW_INSTALL_PREFIX
  45. OUTPUT_STRIP_TRAILING_WHITESPACE)
  46. message(STATUS "Detected Homebrew with install prefix: "
  47. "${HOMEBREW_INSTALL_PREFIX}, adding to CMake search paths.")
  48. list(APPEND CMAKE_PREFIX_PATH "${HOMEBREW_INSTALL_PREFIX}")
  49. endif()
  50. endif()
  51. project(Ceres C CXX)
  52. # NOTE: The following CMake variables must be applied consistently to all
  53. # targets in project to avoid visibility warnings by placing the variables at
  54. # the project top.
  55. # Always build position-independent code (PIC), even when building Ceres as a
  56. # static library so that shared libraries can link against it, not just
  57. # executables (PIC does not apply on Windows). Global variable can be overridden
  58. # by the user whereas target properties can be not.
  59. set(CMAKE_POSITION_INDEPENDENT_CODE ON)
  60. # Set the default symbol visibility to hidden to unify the behavior among
  61. # the various compilers and to get smaller binaries
  62. set(CMAKE_C_VISIBILITY_PRESET hidden)
  63. set(CMAKE_CXX_VISIBILITY_PRESET hidden)
  64. set(CMAKE_VISIBILITY_INLINES_HIDDEN ON)
  65. # NOTE: The 'generic' CMake variables CMAKE_[SOURCE/BINARY]_DIR should not be
  66. # used. Always use the project-specific variants (generated by CMake):
  67. # <PROJECT_NAME_MATCHING_CASE>_[SOURCE/BINARY]_DIR, e.g.
  68. # Ceres_SOURCE_DIR (note, *not* CERES_SOURCE_DIR) instead, as these will
  69. # always point to the correct directories for the Ceres project, even if
  70. # it is nested inside another source tree, whereas the 'generic'
  71. # CMake variables refer to the *first* project() declaration, i.e. the
  72. # top-level project, not Ceres, if Ceres is nested.
  73. # Make CMake aware of the cmake folder for local FindXXX scripts,
  74. # append rather than set in case the user has passed their own
  75. # additional paths via -D.
  76. list(APPEND CMAKE_MODULE_PATH "${Ceres_SOURCE_DIR}/cmake")
  77. include(AddCompileFlagsIfSupported)
  78. include(CheckCXXCompilerFlag)
  79. include(CheckLibraryExists)
  80. include(GNUInstallDirs)
  81. include(UpdateCacheVariable)
  82. check_cxx_compiler_flag(/bigobj HAVE_BIGOBJ)
  83. check_library_exists(m pow "" HAVE_LIBM)
  84. # Xcode 11.0-1 with macOS 10.15 (Catalina) broke alignment.
  85. include(DetectBrokenStackCheckMacOSXcodePairing)
  86. detect_broken_stack_check_macos_xcode_pairing()
  87. # Set up the git hook to make Gerrit Change-Id: lines in commit messages.
  88. include(AddGerritCommitHook)
  89. add_gerrit_commit_hook(${Ceres_SOURCE_DIR} ${Ceres_BINARY_DIR})
  90. set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${Ceres_BINARY_DIR}/bin)
  91. set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${Ceres_BINARY_DIR}/lib)
  92. set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY ${Ceres_BINARY_DIR}/lib)
  93. # Set postfixes for generated libraries based on buildtype.
  94. set(CMAKE_RELEASE_POSTFIX "")
  95. set(CMAKE_DEBUG_POSTFIX "-debug")
  96. # Read the Ceres version from the source, such that we only ever have a single
  97. # definition of the Ceres version.
  98. include(ReadCeresVersionFromSource)
  99. read_ceres_version_from_source(${Ceres_SOURCE_DIR})
  100. enable_testing()
  101. include(CMakeDependentOption)
  102. include(PrettyPrintCMakeList)
  103. option(MINIGLOG "Use a stripped down version of glog." OFF)
  104. option(GFLAGS "Enable Google Flags." ON)
  105. option(SUITESPARSE "Enable SuiteSparse." ON)
  106. if (APPLE)
  107. option(ACCELERATESPARSE
  108. "Enable use of sparse solvers in Apple's Accelerate framework." ON)
  109. option(ENABLE_BITCODE
  110. "Enable bitcode for iOS builds (disables inline optimizations for Eigen)." OFF)
  111. endif()
  112. # We can't have an option called 'CUDA' since that is a reserved word -- a
  113. # language definition.
  114. option(USE_CUDA "Enable use of CUDA linear algebra solvers." ON)
  115. option(LAPACK "Enable use of LAPACK directly within Ceres." ON)
  116. # Template specializations for the Schur complement based solvers. If
  117. # compile time, binary size or compiler performance is an issue, you
  118. # may consider disabling this.
  119. option(SCHUR_SPECIALIZATIONS "Enable fixed-size schur specializations." ON)
  120. option(CUSTOM_BLAS
  121. "Use handcoded BLAS routines (usually faster) instead of Eigen."
  122. ON)
  123. # Enable the use of Eigen as a sparse linear algebra library for
  124. # solving the nonlinear least squares problems.
  125. option(EIGENSPARSE "Enable Eigen as a sparse linear algebra library." ON)
  126. cmake_dependent_option(EIGENMETIS "Enable Eigen METIS support." ON EIGENSPARSE OFF)
  127. option(EXPORT_BUILD_DIR
  128. "Export build directory using CMake (enables external use without install)." OFF)
  129. option(BUILD_TESTING "Enable tests" ON)
  130. option(BUILD_DOCUMENTATION "Build User's Guide (html)" OFF)
  131. option(BUILD_EXAMPLES "Build examples" ON)
  132. option(BUILD_BENCHMARKS "Build Ceres benchmarking suite" ON)
  133. option(BUILD_SHARED_LIBS "Build Ceres as a shared library." OFF)
  134. option(PROVIDE_UNINSTALL_TARGET "Add a custom target to ease removal of installed targets" ON)
  135. set(SANITIZERS "" CACHE STRING "Semicolon-separated list of sanitizers to use (e.g address, memory, thread)")
  136. include(EnableSanitizer)
  137. enable_sanitizer(${SANITIZERS})
  138. if (ANDROID)
  139. option(ANDROID_STRIP_DEBUG_SYMBOLS "Strip debug symbols from Android builds (reduces file sizes)" ON)
  140. endif()
  141. # IOS is defined iff using the iOS.cmake CMake toolchain to build a static
  142. # library for iOS.
  143. if (IOS)
  144. message(STATUS "Building Ceres for iOS platform: ${IOS_PLATFORM}")
  145. # Ceres requires at least iOS 7.0+.
  146. if (IOS_DEPLOYMENT_TARGET VERSION_LESS 7.0)
  147. message(FATAL_ERROR "Unsupported iOS version: ${IOS_DEPLOYMENT_TARGET}, Ceres "
  148. "requires at least iOS version 7.0")
  149. endif()
  150. update_cache_variable(MINIGLOG ON)
  151. message(STATUS "Building for iOS: Forcing use of miniglog instead of glog.")
  152. # Apple claims that the BLAS call dsyrk_ is a private API, and will not allow
  153. # you to submit to the Apple Store if the symbol is present.
  154. update_cache_variable(LAPACK OFF)
  155. message(STATUS "Building for iOS: SuiteSparse, LAPACK, gflags "
  156. "are not available.")
  157. update_cache_variable(BUILD_EXAMPLES OFF)
  158. message(STATUS "Building for iOS: Will not build examples.")
  159. endif (IOS)
  160. unset(CERES_COMPILE_OPTIONS)
  161. # Eigen.
  162. # Eigen delivers Eigen3Config.cmake since v3.3.3
  163. find_package(Eigen3 3.3 REQUIRED)
  164. if (Eigen3_FOUND)
  165. message("-- Found Eigen version ${Eigen3_VERSION}: ${Eigen3_DIR}")
  166. if (CMAKE_SYSTEM_PROCESSOR MATCHES "^(aarch64.*|AARCH64.*)" AND
  167. Eigen3_VERSION VERSION_LESS 3.3.4)
  168. # As per issue #289: https://github.com/ceres-solver/ceres-solver/issues/289
  169. # the bundle_adjustment_test will fail for Eigen < 3.3.4 on aarch64.
  170. message(FATAL_ERROR "-- Ceres requires Eigen version >= 3.3.4 on aarch64. "
  171. "Detected version of Eigen is: ${Eigen3_VERSION}.")
  172. endif()
  173. if (EIGENSPARSE)
  174. message("-- Enabling use of Eigen as a sparse linear algebra library.")
  175. list(APPEND CERES_COMPILE_OPTIONS CERES_USE_EIGEN_SPARSE)
  176. else (EIGENSPARSE)
  177. message("-- Disabling use of Eigen as a sparse linear algebra library.")
  178. message(" This does not affect the covariance estimation algorithm ")
  179. message(" which can still use the EIGEN_SPARSE_QR algorithm.")
  180. add_definitions(-DEIGEN_MPL2_ONLY)
  181. endif (EIGENSPARSE)
  182. endif (Eigen3_FOUND)
  183. if (USE_CUDA)
  184. if (CMAKE_VERSION VERSION_LESS 3.17)
  185. # On older versions of CMake (20.04 default is 3.16) FindCUDAToolkit was
  186. # not available, but FindCUDA was deprecated. To avoid special-case handling
  187. # elsewhere, emulate the effects of FindCUDAToolkit locally in terms of the
  188. # expected CMake imported targets and defined variables. This can be removed
  189. # from as soon as the min CMake version is >= 3.17.
  190. find_package(CUDA QUIET)
  191. if (CUDA_FOUND)
  192. message("-- Found CUDA version ${CUDA_VERSION} installed in: "
  193. "${CUDA_TOOLKIT_ROOT_DIR} via legacy (< 3.17) CMake module. "
  194. "Using the legacy CMake module means that any installation of "
  195. "Ceres will require that the CUDA libraries be installed in a "
  196. "location included in the LD_LIBRARY_PATH.")
  197. enable_language(CUDA)
  198. macro(DECLARE_IMPORTED_CUDA_TARGET COMPONENT)
  199. add_library(CUDA::${COMPONENT} INTERFACE IMPORTED)
  200. target_include_directories(
  201. CUDA::${COMPONENT} INTERFACE ${CUDA_INCLUDE_DIRS})
  202. target_link_libraries(
  203. CUDA::${COMPONENT} INTERFACE ${CUDA_${COMPONENT}_LIBRARY} ${ARGN})
  204. endmacro()
  205. declare_imported_cuda_target(cublas)
  206. declare_imported_cuda_target(cusolver)
  207. declare_imported_cuda_target(cusparse)
  208. declare_imported_cuda_target(cudart ${CUDA_LIBRARIES})
  209. set(CUDAToolkit_BIN_DIR ${CUDA_TOOLKIT_ROOT_DIR}/bin)
  210. else (CUDA_FOUND)
  211. message("-- Did not find CUDA, disabling CUDA support.")
  212. update_cache_variable(USE_CUDA OFF)
  213. endif (CUDA_FOUND)
  214. else (CMAKE_VERSION VERSION_LESS 3.17)
  215. find_package(CUDAToolkit QUIET)
  216. if (CUDAToolkit_FOUND)
  217. message("-- Found CUDA version ${CUDAToolkit_VERSION} installed in: "
  218. "${CUDAToolkit_TARGET_DIR}")
  219. set(CUDAToolkit_DEPENDENCY
  220. "find_dependency(CUDAToolkit ${CUDAToolkit_VERSION})")
  221. enable_language(CUDA)
  222. if (CMAKE_VERSION VERSION_GREATER_EQUAL "3.18")
  223. # Support Maxwell, Pascal, Volta, Turing, and Ampere GPUs.
  224. set(CMAKE_CUDA_ARCHITECTURES "50;60;70;80")
  225. message("-- Setting CUDA Architecture to ${CMAKE_CUDA_ARCHITECTURES}")
  226. endif()
  227. list(APPEND CERES_CUDA_LIBRARIES
  228. CUDA::cublas
  229. CUDA::cudart
  230. CUDA::cusolver
  231. CUDA::cusparse)
  232. set(CMAKE_CUDA_RUNTIME_LIBRARY NONE)
  233. else (CUDAToolkit_FOUND)
  234. message("-- Did not find CUDA, disabling CUDA support.")
  235. update_cache_variable(USE_CUDA OFF)
  236. endif (CUDAToolkit_FOUND)
  237. endif (CMAKE_VERSION VERSION_LESS 3.17)
  238. endif (USE_CUDA)
  239. if (NOT USE_CUDA)
  240. message("-- Building without CUDA.")
  241. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_CUDA)
  242. endif (NOT USE_CUDA)
  243. if (LAPACK)
  244. find_package(LAPACK QUIET)
  245. if (LAPACK_FOUND)
  246. message("-- Found LAPACK library: ${LAPACK_LIBRARIES}")
  247. else (LAPACK_FOUND)
  248. message("-- Did not find LAPACK library, disabling LAPACK support.")
  249. update_cache_variable(LAPACK OFF)
  250. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_LAPACK)
  251. endif (LAPACK_FOUND)
  252. else (LAPACK)
  253. message("-- Building without LAPACK.")
  254. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_LAPACK)
  255. endif (LAPACK)
  256. # Set the install path for the installed CeresConfig.cmake configuration file
  257. # relative to CMAKE_INSTALL_PREFIX.
  258. set(RELATIVE_CMAKECONFIG_INSTALL_DIR ${CMAKE_INSTALL_LIBDIR}/cmake/Ceres)
  259. if (SUITESPARSE)
  260. # By default, if SuiteSparse and all dependencies are found, Ceres is
  261. # built with SuiteSparse support.
  262. # Check for SuiteSparse and dependencies.
  263. find_package(SuiteSparse 4.5.6 COMPONENTS CHOLMOD SPQR
  264. OPTIONAL_COMPONENTS Partition)
  265. if (SuiteSparse_FOUND)
  266. set(SuiteSparse_DEPENDENCY "find_dependency(SuiteSparse ${SuiteSparse_VERSION})")
  267. # By default, if all of SuiteSparse's dependencies are found, Ceres is
  268. # built with SuiteSparse support.
  269. message("-- Found SuiteSparse ${SuiteSparse_VERSION}, "
  270. "building with SuiteSparse.")
  271. if (SuiteSparse_NO_CMAKE OR NOT SuiteSparse_DIR)
  272. install(FILES ${Ceres_SOURCE_DIR}/cmake/FindSuiteSparse.cmake
  273. ${Ceres_SOURCE_DIR}/cmake/FindMETIS.cmake
  274. DESTINATION ${RELATIVE_CMAKECONFIG_INSTALL_DIR})
  275. endif (SuiteSparse_NO_CMAKE OR NOT SuiteSparse_DIR)
  276. else (SuiteSparse_FOUND)
  277. # Disable use of SuiteSparse if it cannot be found and continue.
  278. message("-- Did not find all SuiteSparse dependencies, disabling "
  279. "SuiteSparse support.")
  280. update_cache_variable(SUITESPARSE OFF)
  281. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_SUITESPARSE)
  282. endif (SuiteSparse_FOUND)
  283. else (SUITESPARSE)
  284. message("-- Building without SuiteSparse.")
  285. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_SUITESPARSE)
  286. endif (SUITESPARSE)
  287. if (NOT SuiteSparse_Partition_FOUND)
  288. list (APPEND CERES_COMPILE_OPTIONS CERES_NO_CHOLMOD_PARTITION)
  289. endif (NOT SuiteSparse_Partition_FOUND)
  290. if (EIGENMETIS)
  291. find_package (METIS)
  292. if (METIS_FOUND)
  293. # Since METIS is a private dependency of Ceres, it requires access to the
  294. # link-only METIS::METIS target to avoid undefined linker errors in projects
  295. # relying on Ceres. We do not actually need to propagate anything besides
  296. # the link libraries (such as include directories.)
  297. set(METIS_DEPENDENCY "find_dependency(METIS ${METIS_VERSION})")
  298. # METIS find module must be installed unless a package config is being used.
  299. if (NOT METIS_DIR)
  300. install(FILES ${Ceres_SOURCE_DIR}/cmake/FindMETIS.cmake
  301. DESTINATION ${RELATIVE_CMAKECONFIG_INSTALL_DIR})
  302. endif (NOT METIS_DIR)
  303. else (METIS_FOUND)
  304. message("-- Did not find METIS, disabling Eigen METIS support.")
  305. update_cache_variable(EIGENMETIS OFF)
  306. list (APPEND CERES_COMPILE_OPTIONS CERES_NO_EIGEN_METIS)
  307. endif (METIS_FOUND)
  308. else (EIGENMETIS)
  309. message("-- Building without Eigen METIS support.")
  310. list (APPEND CERES_COMPILE_OPTIONS CERES_NO_EIGEN_METIS)
  311. endif (EIGENMETIS)
  312. if (ACCELERATESPARSE)
  313. find_package(AccelerateSparse)
  314. if (AccelerateSparse_FOUND)
  315. message("-- Found Apple's Accelerate framework with sparse solvers, "
  316. "building with Accelerate sparse support.")
  317. else()
  318. message("-- Failed to find Apple's Accelerate framework with sparse solvers, "
  319. "building without Accelerate sparse support.")
  320. update_cache_variable(ACCELERATESPARSE OFF)
  321. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_ACCELERATE_SPARSE)
  322. endif()
  323. else()
  324. message("-- Building without Apple's Accelerate sparse support.")
  325. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_ACCELERATE_SPARSE)
  326. mark_as_advanced(FORCE AccelerateSparse_INCLUDE_DIR
  327. AccelerateSparse_LIBRARY)
  328. endif()
  329. # Ensure that the user understands they have disabled all sparse libraries.
  330. if (NOT SUITESPARSE AND NOT EIGENSPARSE AND NOT ACCELERATESPARSE)
  331. message(" ===============================================================")
  332. message(" Compiling without any sparse library: SuiteSparse, ")
  333. message(" EigenSparse & Apple's Accelerate are all disabled or unavailable. ")
  334. message(" No sparse linear solvers (SPARSE_NORMAL_CHOLESKY & SPARSE_SCHUR)")
  335. message(" will be available when Ceres is used.")
  336. message(" ===============================================================")
  337. endif()
  338. # ANDROID define is set by the Android CMake toolchain file.
  339. if (ANDROID)
  340. message(" ================================================================")
  341. if (ANDROID_STRIP_DEBUG_SYMBOLS)
  342. # Strip debug information unconditionally to avoid +200MB library file sizes.
  343. set( CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -s" )
  344. set( CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -s" )
  345. message(" Stripping debug information from Android build of Ceres library ")
  346. message(" to avoid +200MB library files.")
  347. else()
  348. message(" Warning: not stripping debug information from Android build of ")
  349. message(" Ceres library. This will result in a large (+200MB) library.")
  350. endif()
  351. message("")
  352. message(" You can control whether debug information is stripped via the ")
  353. message(" ANDROID_STRIP_DEBUG_SYMBOLS CMake option when configuring Ceres.")
  354. message(" ================================================================")
  355. endif()
  356. # GFlags.
  357. if (GFLAGS)
  358. # Don't search with REQUIRED as we can continue without gflags.
  359. find_package(gflags 2.2.0)
  360. if (gflags_FOUND)
  361. if (TARGET gflags)
  362. message("-- Found Google Flags (gflags) version ${gflags_VERSION}: ${gflags_DIR}")
  363. else()
  364. message("-- Detected version of gflags: ${gflags_VERSION} does not define "
  365. "expected gflags CMake target which should be exported by gflags 2.2+. "
  366. "Building without gflags.")
  367. update_cache_variable(GFLAGS OFF)
  368. endif()
  369. else (gflags_FOUND)
  370. message("-- Did not find Google Flags (gflags), Building without gflags.")
  371. update_cache_variable(GFLAGS OFF)
  372. endif (gflags_FOUND)
  373. endif()
  374. if (NOT GFLAGS)
  375. message("-- Use of gflags disabled - no tests or tools will be built!")
  376. endif()
  377. # MiniGLog.
  378. if (MINIGLOG)
  379. message("-- Compiling minimal glog substitute into Ceres.")
  380. set(GLOG_INCLUDE_DIRS internal/ceres/miniglog)
  381. set(MINIGLOG_MAX_LOG_LEVEL 2 CACHE STRING "The maximum message severity level to be logged")
  382. add_definitions("-DMAX_LOG_LEVEL=${MINIGLOG_MAX_LOG_LEVEL}")
  383. message("-- Using minimal glog substitute (include): ${GLOG_INCLUDE_DIRS}")
  384. message("-- Max log level for minimal glog substitute: ${MINIGLOG_MAX_LOG_LEVEL}")
  385. # Mark as advanced (remove from default GUI view) the glog search
  386. # variables in case user disables MINIGLOG, FindGlog did not find it, so
  387. # made search variables visible in GUI for user to set, but then user enables
  388. # MINIGLOG instead of setting them.
  389. mark_as_advanced(FORCE GLOG_INCLUDE_DIR
  390. GLOG_LIBRARY)
  391. else (MINIGLOG)
  392. unset(MINIGLOG_MAX_LOG_LEVEL CACHE)
  393. # Don't search with REQUIRED so that configuration continues if not found and
  394. # we can output an error messages explaining MINIGLOG option.
  395. find_package(Glog)
  396. if (NOT GLOG_FOUND)
  397. message(FATAL_ERROR "Can't find Google Log (glog). Please set either: "
  398. "glog_DIR (newer CMake built versions of glog) or GLOG_INCLUDE_DIR & "
  399. "GLOG_LIBRARY or enable MINIGLOG option to use minimal glog "
  400. "implementation.")
  401. endif(NOT GLOG_FOUND)
  402. # By default, assume gflags was found, updating the message if it was not.
  403. set(GLOG_GFLAGS_DEPENDENCY_MESSAGE
  404. " Assuming glog was built with gflags support as gflags was found. "
  405. "This will make gflags a public dependency of Ceres.")
  406. if (NOT gflags_FOUND)
  407. set(GLOG_GFLAGS_DEPENDENCY_MESSAGE
  408. " Assuming glog was NOT built with gflags support as gflags was "
  409. "not found. If glog was built with gflags, please set the "
  410. "gflags search locations such that it can be found by Ceres. "
  411. "Otherwise, Ceres may fail to link due to missing gflags symbols.")
  412. endif(NOT gflags_FOUND)
  413. message("-- Found Google Log (glog)." ${GLOG_GFLAGS_DEPENDENCY_MESSAGE})
  414. endif (MINIGLOG)
  415. if (NOT SCHUR_SPECIALIZATIONS)
  416. list(APPEND CERES_COMPILE_OPTIONS CERES_RESTRICT_SCHUR_SPECIALIZATION)
  417. message("-- Disabling Schur specializations (faster compiles)")
  418. endif (NOT SCHUR_SPECIALIZATIONS)
  419. if (NOT CUSTOM_BLAS)
  420. list(APPEND CERES_COMPILE_OPTIONS CERES_NO_CUSTOM_BLAS)
  421. message("-- Disabling custom blas")
  422. endif (NOT CUSTOM_BLAS)
  423. if (BUILD_BENCHMARKS)
  424. # Version 1.3 was first to provide import targets
  425. find_package(benchmark 1.3 QUIET)
  426. if (benchmark_FOUND)
  427. message("-- Found Google benchmark library. Building Ceres benchmarks.")
  428. else()
  429. message("-- Failed to find Google benchmark library, disabling build of benchmarks.")
  430. update_cache_variable(BUILD_BENCHMARKS OFF)
  431. endif()
  432. mark_as_advanced(benchmark_DIR)
  433. endif()
  434. # TODO Report features using the FeatureSummary CMake module
  435. if (BUILD_SHARED_LIBS)
  436. message("-- Building Ceres as a shared library.")
  437. else (BUILD_SHARED_LIBS)
  438. message("-- Building Ceres as a static library.")
  439. endif (BUILD_SHARED_LIBS)
  440. # Change the default build type from Debug to Release, while still
  441. # supporting overriding the build type.
  442. #
  443. # The CACHE STRING logic here and elsewhere is needed to force CMake
  444. # to pay attention to the value of these variables.
  445. if (NOT CMAKE_BUILD_TYPE)
  446. message("-- No build type specified; defaulting to CMAKE_BUILD_TYPE=Release.")
  447. set(CMAKE_BUILD_TYPE Release CACHE STRING
  448. "Choose the type of build, options are: None Debug Release RelWithDebInfo MinSizeRel."
  449. FORCE)
  450. else (NOT CMAKE_BUILD_TYPE)
  451. if (CMAKE_BUILD_TYPE STREQUAL "Debug")
  452. message("\n=================================================================================")
  453. message("\n-- Build type: Debug. Performance will be terrible!")
  454. message("-- Add -DCMAKE_BUILD_TYPE=Release to the CMake command line to get an optimized build.")
  455. message("\n=================================================================================")
  456. endif (CMAKE_BUILD_TYPE STREQUAL "Debug")
  457. endif (NOT CMAKE_BUILD_TYPE)
  458. if (MINGW)
  459. # MinGW produces code that segfaults when performing matrix multiplications
  460. # in Eigen when compiled with -O3 (see [1]), as such force the use of -O2
  461. # which works.
  462. #
  463. # [1] http://eigen.tuxfamily.org/bz/show_bug.cgi?id=556
  464. message("-- MinGW detected, forcing -O2 instead of -O3 in Release for Eigen due "
  465. "to a MinGW bug: http://eigen.tuxfamily.org/bz/show_bug.cgi?id=556")
  466. string(REPLACE "-O3" "-O2" CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE}")
  467. update_cache_variable(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE}")
  468. endif (MINGW)
  469. # After the tweaks for the compile settings, disable some warnings on MSVC.
  470. if (MSVC)
  471. # Insecure standard library functions
  472. add_compile_definitions(_CRT_SECURE_NO_WARNINGS)
  473. # std::numeric_limits<T>::has_denorm is deprecated in C++23
  474. add_compile_definitions($<$<COMPILE_LANGUAGE:CXX>:_SILENCE_CXX23_DENORM_DEPRECATION_WARNING>)
  475. # std::aligned_storage is deprecated in C++23
  476. add_compile_definitions($<$<COMPILE_LANGUAGE:CXX>:_SILENCE_CXX23_ALIGNED_STORAGE_DEPRECATION_WARNING>)
  477. # Disable signed/unsigned int conversion warnings.
  478. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/wd4018>)
  479. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/wd4267>)
  480. # Disable warning about using struct/class for the same symbol.
  481. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/wd4099>)
  482. # Disable performance warning about int-to-bool conversion.
  483. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/wd4800>)
  484. # Disable warning about int64 to int32 conversion. Disabling
  485. # this warning may not be correct; needs investigation.
  486. # TODO(keir): Investigate these warnings in more detail.
  487. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/wd4244>)
  488. # It's not possible to use STL types in DLL interfaces in a portable and
  489. # reliable way. However, that's what happens with Google Log and Google Flags
  490. # on Windows. MSVC gets upset about this and throws warnings that we can't do
  491. # much about. The real solution is to link static versions of Google Log and
  492. # Google Test, but that seems tricky on Windows. So, disable the warning.
  493. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/wd4251>)
  494. # Add bigobj flag otherwise the build would fail due to large object files
  495. # probably resulting from generated headers (like the fixed-size schur
  496. # specializations).
  497. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:/bigobj>)
  498. # Google Flags doesn't have their DLL import/export stuff set up correctly,
  499. # which results in linker warnings. This is irrelevant for Ceres, so ignore
  500. # the warnings.
  501. set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} /ignore:4049")
  502. # Tuple sizes of 10 are used by Gtest.
  503. add_definitions("-D_VARIADIC_MAX=10")
  504. endif (MSVC)
  505. if (UNIX)
  506. # Flags which we add to GCC to make it more picky about stuff
  507. # we do care about,
  508. add_cxx_compiler_flag_if_supported(CERES_STRICT_CXX_FLAGS
  509. -Wmissing-declarations)
  510. # Flags which we add to GCC to silence lots of annoying false-positives.
  511. add_cxx_compiler_flag_if_supported(CERES_STRICT_CXX_FLAGS
  512. -Wno-unknown-pragmas)
  513. add_cxx_compiler_flag_if_supported(CERES_STRICT_CXX_FLAGS
  514. -Wno-sign-compare)
  515. add_cxx_compiler_flag_if_supported(CERES_STRICT_CXX_FLAGS
  516. -Wno-unused-parameter)
  517. add_cxx_compiler_flag_if_supported(CERES_STRICT_CXX_FLAGS
  518. -Wno-missing-field-initializers)
  519. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${CERES_STRICT_CXX_FLAGS}")
  520. endif (UNIX)
  521. if (CMAKE_CXX_COMPILER_ID MATCHES "Clang") # Matches Clang & AppleClang.
  522. # Optimize for Eigen OR enable bitcode; you cannot do both since bitcode is an
  523. # intermediate representation.
  524. if (ENABLE_BITCODE)
  525. set(CMAKE_CXX_FLAGS
  526. "${CMAKE_CXX_FLAGS} -fembed-bitcode")
  527. else ()
  528. # Use a larger inlining threshold for Clang, since it hobbles Eigen,
  529. # resulting in an unreasonably slow version of the blas routines. The
  530. # -Qunused-arguments is needed because CMake passes the inline
  531. # threshold to the linker and clang complains about it and dies.
  532. set(CMAKE_CXX_FLAGS
  533. "${CMAKE_CXX_FLAGS} -Qunused-arguments -mllvm -inline-threshold=600")
  534. endif ()
  535. # Older versions of Clang (<= 2.9) do not support the 'return-type-c-linkage'
  536. # option, so check for its presence before adding it to the default flags set.
  537. check_cxx_compiler_flag("-Wno-return-type-c-linkage"
  538. HAVE_RETURN_TYPE_C_LINKAGE)
  539. if (HAVE_RETURN_TYPE_C_LINKAGE)
  540. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wno-return-type-c-linkage")
  541. endif(HAVE_RETURN_TYPE_C_LINKAGE)
  542. endif ()
  543. add_compile_definitions($<$<BOOL:${WIN32}>:NOMINMAX>)
  544. # Configure the Ceres config.h compile options header using the current
  545. # compile options and put the configured header into the Ceres build
  546. # directory. Note that the ceres/internal subdir in <build>/config where
  547. # the configured config.h is placed is important, because Ceres will be
  548. # built against this configured header, it needs to have the same relative
  549. # include path as it would if it were in the source tree (or installed).
  550. list(REMOVE_DUPLICATES CERES_COMPILE_OPTIONS)
  551. include(CreateCeresConfig)
  552. create_ceres_config("${CERES_COMPILE_OPTIONS}"
  553. ${Ceres_BINARY_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/ceres/internal)
  554. add_subdirectory(internal/ceres)
  555. if (BUILD_DOCUMENTATION)
  556. find_package (Sphinx REQUIRED COMPONENTS sphinx_rtd_theme)
  557. if (NOT Sphinx_FOUND)
  558. message("-- Failed to find Sphinx and/or its dependencies, disabling build of documentation.")
  559. update_cache_variable(BUILD_DOCUMENTATION OFF)
  560. else()
  561. # Generate the User's Guide (html).
  562. # The corresponding target is ceres_docs, but is included in ALL.
  563. message("-- Build the HTML documentation.")
  564. add_subdirectory(docs)
  565. endif()
  566. endif (BUILD_DOCUMENTATION)
  567. if (BUILD_EXAMPLES)
  568. message("-- Build the examples.")
  569. add_subdirectory(examples)
  570. else (BUILD_EXAMPLES)
  571. message("-- Do not build any example.")
  572. endif (BUILD_EXAMPLES)
  573. # Setup installation of Ceres public headers.
  574. file(GLOB CERES_HDRS ${Ceres_SOURCE_DIR}/include/ceres/*.h)
  575. install(FILES ${CERES_HDRS} DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/ceres)
  576. file(GLOB CERES_PUBLIC_INTERNAL_HDRS ${Ceres_SOURCE_DIR}/include/ceres/internal/*.h)
  577. install(FILES ${CERES_PUBLIC_INTERNAL_HDRS} DESTINATION
  578. ${CMAKE_INSTALL_INCLUDEDIR}/ceres/internal)
  579. # Also setup installation of Ceres config.h configured with the current
  580. # build options and export.h into the installed headers directory.
  581. install(DIRECTORY ${Ceres_BINARY_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/
  582. DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
  583. if (MINIGLOG)
  584. # Install miniglog header if being used as logging #includes appear in
  585. # installed public Ceres headers.
  586. install(FILES ${Ceres_SOURCE_DIR}/internal/ceres/miniglog/glog/logging.h
  587. DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/ceres/internal/miniglog/glog)
  588. endif (MINIGLOG)
  589. # Ceres supports two mechanisms by which it can be detected & imported into
  590. # client code which uses CMake via find_package(Ceres):
  591. #
  592. # 1) Installation (e.g. to /usr/local), using CMake's install() function.
  593. #
  594. # 2) (Optional) Export of the current build directory into the local CMake
  595. # package registry, using CMake's export() function. This allows use of
  596. # Ceres from other projects without requiring installation.
  597. #
  598. # In both cases, we need to generate a configured CeresConfig.cmake which
  599. # includes additional autogenerated files which in concert create an imported
  600. # target for Ceres in a client project when find_package(Ceres) is invoked.
  601. # The key distinctions are where this file is located, and whether client code
  602. # references installed copies of the compiled Ceres headers/libraries,
  603. # (option #1: installation), or the originals in the source/build directories
  604. # (option #2: export of build directory).
  605. #
  606. # NOTE: If Ceres is both exported and installed, provided that the installation
  607. # path is present in CMAKE_MODULE_PATH when find_package(Ceres) is called,
  608. # the installed version is preferred.
  609. # Build the list of Ceres components for CeresConfig.cmake from the current set
  610. # of compile options.
  611. include(CeresCompileOptionsToComponents)
  612. ceres_compile_options_to_components("${CERES_COMPILE_OPTIONS}"
  613. CERES_COMPILED_COMPONENTS)
  614. include(CMakePackageConfigHelpers)
  615. # Create a CeresConfigVersion.cmake file containing the version information,
  616. # used by both export() & install().
  617. write_basic_package_version_file("${Ceres_BINARY_DIR}/CeresConfigVersion.cmake"
  618. VERSION ${CERES_VERSION}
  619. COMPATIBILITY SameMajorVersion)
  620. # Install method #1: Put Ceres in CMAKE_INSTALL_PREFIX: /usr/local or equivalent.
  621. # This "exports" for installation all targets which have been put into the
  622. # export set "CeresExport". This generates a CeresTargets.cmake file which,
  623. # when read in by a client project as part of find_package(Ceres) creates
  624. # imported library targets for Ceres (with dependency relations) which can be
  625. # used in target_link_libraries() calls in the client project to use Ceres.
  626. install(EXPORT CeresExport
  627. NAMESPACE Ceres::
  628. DESTINATION ${RELATIVE_CMAKECONFIG_INSTALL_DIR} FILE CeresTargets.cmake)
  629. # Save the relative path from the installed CeresConfig.cmake file to the
  630. # install prefix. We do not save an absolute path in case the installed package
  631. # is subsequently relocated after installation (on Windows).
  632. file(RELATIVE_PATH INSTALL_ROOT_REL_CONFIG_INSTALL_DIR
  633. ${CMAKE_INSTALL_PREFIX}/${RELATIVE_CMAKECONFIG_INSTALL_DIR}
  634. ${CMAKE_INSTALL_PREFIX})
  635. # Configure a CeresConfig.cmake file for an installed version of Ceres from the
  636. # template, reflecting the current build options.
  637. #
  638. # NOTE: The -install suffix is necessary to distinguish the install version from
  639. # the exported version, which must be named CeresConfig.cmake in
  640. # Ceres_BINARY_DIR to be detected. The suffix is removed when
  641. # it is installed.
  642. set(SETUP_CERES_CONFIG_FOR_INSTALLATION TRUE)
  643. configure_file("${Ceres_SOURCE_DIR}/cmake/CeresConfig.cmake.in"
  644. "${Ceres_BINARY_DIR}/CeresConfig-install.cmake" @ONLY)
  645. # Install the configuration files into the same directory as the autogenerated
  646. # CeresTargets.cmake file. We include the find_package() scripts for libraries
  647. # whose headers are included in the public API of Ceres and should thus be
  648. # present in CERES_INCLUDE_DIRS.
  649. install(FILES "${Ceres_BINARY_DIR}/CeresConfig-install.cmake"
  650. RENAME CeresConfig.cmake
  651. DESTINATION ${RELATIVE_CMAKECONFIG_INSTALL_DIR})
  652. install(FILES "${Ceres_BINARY_DIR}/CeresConfigVersion.cmake"
  653. DESTINATION ${RELATIVE_CMAKECONFIG_INSTALL_DIR})
  654. if (GLOG_FOUND AND NOT FOUND_INSTALLED_GLOG_CMAKE_CONFIGURATION)
  655. # Version of glog detected was not built with CMake, install our glog module
  656. # file to enable detection in CeresConfig.
  657. install(FILES "${Ceres_SOURCE_DIR}/cmake/FindGlog.cmake"
  658. DESTINATION ${RELATIVE_CMAKECONFIG_INSTALL_DIR})
  659. endif()
  660. if (PROVIDE_UNINSTALL_TARGET)
  661. # Create an uninstall target to remove all installed files.
  662. configure_file("${Ceres_SOURCE_DIR}/cmake/uninstall.cmake.in"
  663. "${Ceres_BINARY_DIR}/cmake/uninstall.cmake"
  664. @ONLY)
  665. add_custom_target(uninstall
  666. COMMAND ${CMAKE_COMMAND} -P ${Ceres_BINARY_DIR}/cmake/uninstall.cmake)
  667. endif()
  668. # Install method #2: Put Ceres build into local CMake registry.
  669. #
  670. # Optionally export the Ceres build directory into the local CMake package
  671. # registry (~/.cmake/packages on *nix & OS X). This allows the detection &
  672. # use of Ceres without requiring that it be installed.
  673. if (EXPORT_BUILD_DIR)
  674. message("-- Export Ceres build directory to local CMake package registry.")
  675. # Save the relative path from the build directory to the source directory.
  676. file(RELATIVE_PATH INSTALL_ROOT_REL_CONFIG_INSTALL_DIR
  677. ${Ceres_BINARY_DIR}
  678. ${Ceres_SOURCE_DIR})
  679. set (Ceres_EXPORT_TARGETS ceres)
  680. if (TARGET ceres_cuda_kernels)
  681. # The target ceres depends on ceres_cuda_kernels requiring the latter to be
  682. # exported as part of the same export set.
  683. list (APPEND Ceres_EXPORT_TARGETS ceres_cuda_kernels)
  684. endif (TARGET ceres_cuda_kernels)
  685. # Analogously to install(EXPORT ...), export the Ceres target from the build
  686. # directory as a package called Ceres into the local CMake package registry.
  687. export(TARGETS ${Ceres_EXPORT_TARGETS}
  688. NAMESPACE Ceres::
  689. FILE ${Ceres_BINARY_DIR}/CeresTargets.cmake)
  690. export(PACKAGE ${CMAKE_PROJECT_NAME})
  691. unset (Ceres_EXPORT_TARGETS)
  692. # Configure a CeresConfig.cmake file for the export of the Ceres build
  693. # directory from the template, reflecting the current build options.
  694. set(SETUP_CERES_CONFIG_FOR_INSTALLATION FALSE)
  695. configure_file("${Ceres_SOURCE_DIR}/cmake/CeresConfig.cmake.in"
  696. "${Ceres_BINARY_DIR}/CeresConfig.cmake" @ONLY)
  697. endif (EXPORT_BUILD_DIR)