FindMPI.cmake 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677
  1. #.rst:
  2. # FindMPI
  3. # -------
  4. #
  5. # Find a Message Passing Interface (MPI) implementation
  6. #
  7. # The Message Passing Interface (MPI) is a library used to write
  8. # high-performance distributed-memory parallel applications, and is
  9. # typically deployed on a cluster. MPI is a standard interface (defined
  10. # by the MPI forum) for which many implementations are available. All
  11. # of them have somewhat different include paths, libraries to link
  12. # against, etc., and this module tries to smooth out those differences.
  13. #
  14. # Variables
  15. # ^^^^^^^^^
  16. #
  17. # This module will set the following variables per language in your
  18. # project, where <lang> is one of C, CXX, or Fortran:
  19. #
  20. # ::
  21. #
  22. # MPI_<lang>_FOUND TRUE if FindMPI found MPI flags for <lang>
  23. # MPI_<lang>_COMPILER MPI Compiler wrapper for <lang>
  24. # MPI_<lang>_COMPILE_FLAGS Compilation flags for MPI programs
  25. # MPI_<lang>_INCLUDE_PATH Include path(s) for MPI header
  26. # MPI_<lang>_LINK_FLAGS Linking flags for MPI programs
  27. # MPI_<lang>_LIBRARIES All libraries to link MPI programs against
  28. #
  29. # Additionally, FindMPI sets the following variables for running MPI
  30. # programs from the command line:
  31. #
  32. # ::
  33. #
  34. # MPIEXEC Executable for running MPI programs
  35. # MPIEXEC_NUMPROC_FLAG Flag to pass to MPIEXEC before giving
  36. # it the number of processors to run on
  37. # MPIEXEC_PREFLAGS Flags to pass to MPIEXEC directly
  38. # before the executable to run.
  39. # MPIEXEC_POSTFLAGS Flags to pass to MPIEXEC after other flags
  40. #
  41. # Usage
  42. # ^^^^^
  43. #
  44. # To use this module, simply call FindMPI from a CMakeLists.txt file, or
  45. # run ``find_package(MPI)``, then run CMake. If you are happy with the
  46. # auto-detected configuration for your language, then you're done. If
  47. # not, you have two options:
  48. #
  49. # ::
  50. #
  51. # 1. Set MPI_<lang>_COMPILER to the MPI wrapper (mpicc, etc.) of your
  52. # choice and reconfigure. FindMPI will attempt to determine all the
  53. # necessary variables using THAT compiler's compile and link flags.
  54. # 2. If this fails, or if your MPI implementation does not come with
  55. # a compiler wrapper, then set both MPI_<lang>_LIBRARIES and
  56. # MPI_<lang>_INCLUDE_PATH. You may also set any other variables
  57. # listed above, but these two are required. This will circumvent
  58. # autodetection entirely.
  59. #
  60. # When configuration is successful, ``MPI_<lang>_COMPILER`` will be set to
  61. # the compiler wrapper for <lang>, if it was found. ``MPI_<lang>_FOUND``
  62. # and other variables above will be set if any MPI implementation was
  63. # found for <lang>, regardless of whether a compiler was found.
  64. #
  65. # When using ``MPIEXEC`` to execute MPI applications, you should typically
  66. # use all of the ``MPIEXEC`` flags as follows:
  67. #
  68. # ::
  69. #
  70. # ${MPIEXEC} ${MPIEXEC_NUMPROC_FLAG} PROCS
  71. # ${MPIEXEC_PREFLAGS} EXECUTABLE ${MPIEXEC_POSTFLAGS} ARGS
  72. #
  73. # where ``PROCS`` is the number of processors on which to execute the
  74. # program, ``EXECUTABLE`` is the MPI program, and ``ARGS`` are the arguments to
  75. # pass to the MPI program.
  76. #
  77. # Backward Compatibility
  78. # ^^^^^^^^^^^^^^^^^^^^^^
  79. #
  80. # For backward compatibility with older versions of FindMPI, these
  81. # variables are set, but deprecated:
  82. #
  83. # ::
  84. #
  85. # MPI_FOUND MPI_COMPILER MPI_LIBRARY
  86. # MPI_COMPILE_FLAGS MPI_INCLUDE_PATH MPI_EXTRA_LIBRARY
  87. # MPI_LINK_FLAGS MPI_LIBRARIES
  88. #
  89. # In new projects, please use the ``MPI_<lang>_XXX`` equivalents.
  90. #=============================================================================
  91. # Copyright 2001-2011 Kitware, Inc.
  92. # Copyright 2010-2011 Todd Gamblin tgamblin@llnl.gov
  93. # Copyright 2001-2009 Dave Partyka
  94. #
  95. # Distributed under the OSI-approved BSD License (the "License");
  96. # see accompanying file Copyright.txt for details.
  97. #
  98. # This software is distributed WITHOUT ANY WARRANTY; without even the
  99. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  100. # See the License for more information.
  101. #=============================================================================
  102. # (To distribute this file outside of CMake, substitute the full
  103. # License text for the above reference.)
  104. # include this to handle the QUIETLY and REQUIRED arguments
  105. include(${CMAKE_CURRENT_LIST_DIR}/FindPackageHandleStandardArgs.cmake)
  106. #
  107. # This part detects MPI compilers, attempting to wade through the mess of compiler names in
  108. # a sensible way.
  109. #
  110. # The compilers are detected in this order:
  111. #
  112. # 1. Try to find the most generic available MPI compiler, as this is usually set up by
  113. # cluster admins. e.g., if plain old mpicc is available, we'll use it and assume it's
  114. # the right compiler.
  115. #
  116. # 2. If a generic mpicc is NOT found, then we attempt to find one that matches
  117. # CMAKE_<lang>_COMPILER_ID. e.g. if you are using XL compilers, we'll try to find mpixlc
  118. # and company, but not mpiicc. This hopefully prevents toolchain mismatches.
  119. #
  120. # If you want to force a particular MPI compiler other than what we autodetect (e.g. if you
  121. # want to compile regular stuff with GNU and parallel stuff with Intel), you can always set
  122. # your favorite MPI_<lang>_COMPILER explicitly and this stuff will be ignored.
  123. #
  124. # Start out with the generic MPI compiler names, as these are most commonly used.
  125. set(_MPI_C_COMPILER_NAMES mpicc mpcc mpicc_r mpcc_r)
  126. set(_MPI_CXX_COMPILER_NAMES mpicxx mpiCC mpcxx mpCC mpic++ mpc++
  127. mpicxx_r mpiCC_r mpcxx_r mpCC_r mpic++_r mpc++_r)
  128. set(_MPI_Fortran_COMPILER_NAMES mpif95 mpif95_r mpf95 mpf95_r
  129. mpif90 mpif90_r mpf90 mpf90_r
  130. mpif77 mpif77_r mpf77 mpf77_r)
  131. # GNU compiler names
  132. set(_MPI_GNU_C_COMPILER_NAMES mpigcc mpgcc mpigcc_r mpgcc_r)
  133. set(_MPI_GNU_CXX_COMPILER_NAMES mpig++ mpg++ mpig++_r mpg++_r)
  134. set(_MPI_GNU_Fortran_COMPILER_NAMES mpigfortran mpgfortran mpigfortran_r mpgfortran_r
  135. mpig77 mpig77_r mpg77 mpg77_r)
  136. # Intel MPI compiler names
  137. set(_MPI_Intel_C_COMPILER_NAMES mpiicc)
  138. set(_MPI_Intel_CXX_COMPILER_NAMES mpiicpc mpiicxx mpiic++ mpiiCC)
  139. set(_MPI_Intel_Fortran_COMPILER_NAMES mpiifort mpiif95 mpiif90 mpiif77)
  140. # PGI compiler names
  141. set(_MPI_PGI_C_COMPILER_NAMES mpipgcc mppgcc)
  142. set(_MPI_PGI_CXX_COMPILER_NAMES mpipgCC mppgCC)
  143. set(_MPI_PGI_Fortran_COMPILER_NAMES mpipgf95 mpipgf90 mppgf95 mppgf90 mpipgf77 mppgf77)
  144. # XLC MPI Compiler names
  145. set(_MPI_XL_C_COMPILER_NAMES mpxlc mpxlc_r mpixlc mpixlc_r)
  146. set(_MPI_XL_CXX_COMPILER_NAMES mpixlcxx mpixlC mpixlc++ mpxlcxx mpxlc++ mpixlc++ mpxlCC
  147. mpixlcxx_r mpixlC_r mpixlc++_r mpxlcxx_r mpxlc++_r mpixlc++_r mpxlCC_r)
  148. set(_MPI_XL_Fortran_COMPILER_NAMES mpixlf95 mpixlf95_r mpxlf95 mpxlf95_r
  149. mpixlf90 mpixlf90_r mpxlf90 mpxlf90_r
  150. mpixlf77 mpixlf77_r mpxlf77 mpxlf77_r
  151. mpixlf mpixlf_r mpxlf mpxlf_r)
  152. # append vendor-specific compilers to the list if we either don't know the compiler id,
  153. # or if we know it matches the regular compiler.
  154. foreach (lang C CXX Fortran)
  155. foreach (id GNU Intel PGI XL)
  156. if (NOT CMAKE_${lang}_COMPILER_ID OR CMAKE_${lang}_COMPILER_ID STREQUAL id)
  157. list(APPEND _MPI_${lang}_COMPILER_NAMES ${_MPI_${id}_${lang}_COMPILER_NAMES})
  158. endif()
  159. unset(_MPI_${id}_${lang}_COMPILER_NAMES) # clean up the namespace here
  160. endforeach()
  161. endforeach()
  162. # Names to try for MPI exec
  163. set(_MPI_EXEC_NAMES mpiexec mpirun lamexec srun)
  164. # Grab the path to MPI from the registry if we're on windows.
  165. set(_MPI_PREFIX_PATH)
  166. if(WIN32)
  167. # MSMPI
  168. file(TO_CMAKE_PATH "$ENV{MSMPI_BIN}" msmpi_bin_path) # The default path ends with a '\' and doesn't mix with ';' when appending.
  169. list(APPEND _MPI_PREFIX_PATH "${msmpi_bin_path}")
  170. unset(msmpi_bin_path)
  171. list(APPEND _MPI_PREFIX_PATH "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\MPI;InstallRoot]/Bin")
  172. list(APPEND _MPI_PREFIX_PATH "$ENV{MSMPI_INC}/..") # The SDK is installed separately from the runtime
  173. # MPICH
  174. list(APPEND _MPI_PREFIX_PATH "[HKEY_LOCAL_MACHINE\\SOFTWARE\\MPICH\\SMPD;binary]/..")
  175. list(APPEND _MPI_PREFIX_PATH "[HKEY_LOCAL_MACHINE\\SOFTWARE\\MPICH2;Path]")
  176. list(APPEND _MPI_PREFIX_PATH "$ENV{ProgramW6432}/MPICH2/")
  177. endif()
  178. # Build a list of prefixes to search for MPI.
  179. foreach(SystemPrefixDir ${CMAKE_SYSTEM_PREFIX_PATH})
  180. foreach(MpiPackageDir ${_MPI_PREFIX_PATH})
  181. if(EXISTS ${SystemPrefixDir}/${MpiPackageDir})
  182. list(APPEND _MPI_PREFIX_PATH "${SystemPrefixDir}/${MpiPackageDir}")
  183. endif()
  184. endforeach()
  185. endforeach()
  186. function (_mpi_check_compiler compiler options cmdvar resvar)
  187. execute_process(
  188. COMMAND "${compiler}" ${options}
  189. OUTPUT_VARIABLE cmdline OUTPUT_STRIP_TRAILING_WHITESPACE
  190. ERROR_VARIABLE cmdline ERROR_STRIP_TRAILING_WHITESPACE
  191. RESULT_VARIABLE success)
  192. # Intel MPI 5.0.1 will return a zero return code even when the
  193. # argument to the MPI compiler wrapper is unknown. Attempt to
  194. # catch this case.
  195. if(cmdline MATCHES "undefined reference" OR cmdline MATCHES "unrecognized")
  196. set(success 255 )
  197. endif()
  198. set(${cmdvar} "${cmdline}" PARENT_SCOPE)
  199. set(${resvar} "${success}" PARENT_SCOPE)
  200. endfunction()
  201. #
  202. # interrogate_mpi_compiler(lang try_libs)
  203. #
  204. # Attempts to extract compiler and linker args from an MPI compiler. The arguments set
  205. # by this function are:
  206. #
  207. # MPI_<lang>_INCLUDE_PATH MPI_<lang>_LINK_FLAGS MPI_<lang>_FOUND
  208. # MPI_<lang>_COMPILE_FLAGS MPI_<lang>_LIBRARIES
  209. #
  210. # MPI_<lang>_COMPILER must be set beforehand to the absolute path to an MPI compiler for
  211. # <lang>. Additionally, MPI_<lang>_INCLUDE_PATH and MPI_<lang>_LIBRARIES may be set
  212. # to skip autodetection.
  213. #
  214. # If try_libs is TRUE, this will also attempt to find plain MPI libraries in the usual
  215. # way. In general, this is not as effective as interrogating the compilers, as it
  216. # ignores language-specific flags and libraries. However, some MPI implementations
  217. # (Windows implementations) do not have compiler wrappers, so this approach must be used.
  218. #
  219. function (interrogate_mpi_compiler lang try_libs)
  220. # MPI_${lang}_NO_INTERROGATE will be set to a compiler name when the *regular* compiler was
  221. # discovered to be the MPI compiler. This happens on machines like the Cray XE6 that use
  222. # modules to set cc, CC, and ftn to the MPI compilers. If the user force-sets another MPI
  223. # compiler, MPI_${lang}_COMPILER won't be equal to MPI_${lang}_NO_INTERROGATE, and we'll
  224. # inspect that compiler anew. This allows users to set new compilers w/o rm'ing cache.
  225. string(COMPARE NOTEQUAL "${MPI_${lang}_NO_INTERROGATE}" "${MPI_${lang}_COMPILER}" interrogate)
  226. # If MPI is set already in the cache, don't bother with interrogating the compiler.
  227. if (interrogate AND ((NOT MPI_${lang}_INCLUDE_PATH) OR (NOT MPI_${lang}_LIBRARIES)))
  228. if (MPI_${lang}_COMPILER)
  229. # Check whether the -showme:compile option works. This indicates that we have either OpenMPI
  230. # or a newer version of LAM-MPI, and implies that -showme:link will also work.
  231. _mpi_check_compiler("${MPI_${lang}_COMPILER}" "-showme:compile" MPI_COMPILE_CMDLINE MPI_COMPILER_RETURN)
  232. if (MPI_COMPILER_RETURN EQUAL 0)
  233. # If we appear to have -showme:compile, then we should
  234. # also have -showme:link. Try it.
  235. execute_process(
  236. COMMAND ${MPI_${lang}_COMPILER} -showme:link
  237. OUTPUT_VARIABLE MPI_LINK_CMDLINE OUTPUT_STRIP_TRAILING_WHITESPACE
  238. ERROR_VARIABLE MPI_LINK_CMDLINE ERROR_STRIP_TRAILING_WHITESPACE
  239. RESULT_VARIABLE MPI_COMPILER_RETURN)
  240. if (MPI_COMPILER_RETURN EQUAL 0)
  241. # We probably have -showme:incdirs and -showme:libdirs as well,
  242. # so grab that while we're at it.
  243. execute_process(
  244. COMMAND ${MPI_${lang}_COMPILER} -showme:incdirs
  245. OUTPUT_VARIABLE MPI_INCDIRS OUTPUT_STRIP_TRAILING_WHITESPACE
  246. ERROR_VARIABLE MPI_INCDIRS ERROR_STRIP_TRAILING_WHITESPACE)
  247. execute_process(
  248. COMMAND ${MPI_${lang}_COMPILER} -showme:libdirs
  249. OUTPUT_VARIABLE MPI_LIBDIRS OUTPUT_STRIP_TRAILING_WHITESPACE
  250. ERROR_VARIABLE MPI_LIBDIRS ERROR_STRIP_TRAILING_WHITESPACE)
  251. else()
  252. # reset things here if something went wrong.
  253. set(MPI_COMPILE_CMDLINE)
  254. set(MPI_LINK_CMDLINE)
  255. endif()
  256. endif ()
  257. # Older versions of LAM-MPI have "-showme". Try to find that.
  258. if (NOT MPI_COMPILER_RETURN EQUAL 0)
  259. _mpi_check_compiler("${MPI_${lang}_COMPILER}" "-showme" MPI_COMPILE_CMDLINE MPI_COMPILER_RETURN)
  260. endif()
  261. # MVAPICH uses -compile-info and -link-info. Try them.
  262. if (NOT MPI_COMPILER_RETURN EQUAL 0)
  263. _mpi_check_compiler("${MPI_${lang}_COMPILER}" "-compile-info" MPI_COMPILE_CMDLINE MPI_COMPILER_RETURN)
  264. # If we have compile-info, also have link-info.
  265. if (MPI_COMPILER_RETURN EQUAL 0)
  266. execute_process(
  267. COMMAND ${MPI_${lang}_COMPILER} -link-info
  268. OUTPUT_VARIABLE MPI_LINK_CMDLINE OUTPUT_STRIP_TRAILING_WHITESPACE
  269. ERROR_VARIABLE MPI_LINK_CMDLINE ERROR_STRIP_TRAILING_WHITESPACE
  270. RESULT_VARIABLE MPI_COMPILER_RETURN)
  271. endif()
  272. # make sure we got compile and link. Reset vars if something's wrong.
  273. if (NOT MPI_COMPILER_RETURN EQUAL 0)
  274. set(MPI_COMPILE_CMDLINE)
  275. set(MPI_LINK_CMDLINE)
  276. endif()
  277. endif()
  278. # MPICH just uses "-show". Try it.
  279. if (NOT MPI_COMPILER_RETURN EQUAL 0)
  280. _mpi_check_compiler("${MPI_${lang}_COMPILER}" "-show" MPI_COMPILE_CMDLINE MPI_COMPILER_RETURN)
  281. endif()
  282. if (MPI_COMPILER_RETURN EQUAL 0)
  283. # We have our command lines, but we might need to copy MPI_COMPILE_CMDLINE
  284. # into MPI_LINK_CMDLINE, if we didn't find the link line.
  285. if (NOT MPI_LINK_CMDLINE)
  286. set(MPI_LINK_CMDLINE ${MPI_COMPILE_CMDLINE})
  287. endif()
  288. else()
  289. message(STATUS "Unable to determine MPI from MPI driver ${MPI_${lang}_COMPILER}")
  290. set(MPI_COMPILE_CMDLINE)
  291. set(MPI_LINK_CMDLINE)
  292. endif()
  293. # Here, we're done with the interrogation part, and we'll try to extract args we care
  294. # about from what we learned from the compiler wrapper scripts.
  295. # If interrogation came back with something, extract our variable from the MPI command line
  296. if (MPI_COMPILE_CMDLINE OR MPI_LINK_CMDLINE)
  297. # Extract compile flags from the compile command line.
  298. string(REGEX MATCHALL "(^| )-[Df]([^\" ]+|\"[^\"]+\")" MPI_ALL_COMPILE_FLAGS "${MPI_COMPILE_CMDLINE}")
  299. set(MPI_COMPILE_FLAGS_WORK)
  300. foreach(FLAG ${MPI_ALL_COMPILE_FLAGS})
  301. if (MPI_COMPILE_FLAGS_WORK)
  302. set(MPI_COMPILE_FLAGS_WORK "${MPI_COMPILE_FLAGS_WORK} ${FLAG}")
  303. else()
  304. set(MPI_COMPILE_FLAGS_WORK ${FLAG})
  305. endif()
  306. endforeach()
  307. # Extract include paths from compile command line
  308. string(REGEX MATCHALL "(^| )-I([^\" ]+|\"[^\"]+\")" MPI_ALL_INCLUDE_PATHS "${MPI_COMPILE_CMDLINE}")
  309. foreach(IPATH ${MPI_ALL_INCLUDE_PATHS})
  310. string(REGEX REPLACE "^ ?-I" "" IPATH ${IPATH})
  311. string(REPLACE "//" "/" IPATH ${IPATH})
  312. list(APPEND MPI_INCLUDE_PATH_WORK ${IPATH})
  313. endforeach()
  314. # try using showme:incdirs if extracting didn't work.
  315. if (NOT MPI_INCLUDE_PATH_WORK)
  316. set(MPI_INCLUDE_PATH_WORK ${MPI_INCDIRS})
  317. separate_arguments(MPI_INCLUDE_PATH_WORK)
  318. endif()
  319. # If all else fails, just search for mpi.h in the normal include paths.
  320. if (NOT MPI_INCLUDE_PATH_WORK)
  321. set(MPI_HEADER_PATH "MPI_HEADER_PATH-NOTFOUND" CACHE FILEPATH "Cleared" FORCE)
  322. find_path(MPI_HEADER_PATH mpi.h
  323. HINTS ${_MPI_BASE_DIR} ${_MPI_PREFIX_PATH}
  324. PATH_SUFFIXES include)
  325. set(MPI_INCLUDE_PATH_WORK ${MPI_HEADER_PATH})
  326. endif()
  327. # Extract linker paths from the link command line
  328. string(REGEX MATCHALL "(^| |-Wl,)(-L|/LIBPATH:)([^\" ]+|\"[^\"]+\")" MPI_ALL_LINK_PATHS "${MPI_LINK_CMDLINE}")
  329. set(MPI_LINK_PATH)
  330. foreach(LPATH ${MPI_ALL_LINK_PATHS})
  331. string(REGEX REPLACE "^(| |-Wl,)(-L|/LIBPATH:)" "" LPATH ${LPATH})
  332. string(REPLACE "//" "/" LPATH ${LPATH})
  333. list(APPEND MPI_LINK_PATH ${LPATH})
  334. endforeach()
  335. # try using showme:libdirs if extracting didn't work.
  336. if (NOT MPI_LINK_PATH)
  337. set(MPI_LINK_PATH ${MPI_LIBDIRS})
  338. separate_arguments(MPI_LINK_PATH)
  339. endif()
  340. # Extract linker flags from the link command line
  341. string(REGEX MATCHALL "(^| )(-Wl,|-Xlinker )([^\" ]+|\"[^\"]+\")" MPI_ALL_LINK_FLAGS "${MPI_LINK_CMDLINE}")
  342. set(MPI_LINK_FLAGS_WORK)
  343. foreach(FLAG ${MPI_ALL_LINK_FLAGS})
  344. if (MPI_LINK_FLAGS_WORK)
  345. set(MPI_LINK_FLAGS_WORK "${MPI_LINK_FLAGS_WORK} ${FLAG}")
  346. else()
  347. set(MPI_LINK_FLAGS_WORK ${FLAG})
  348. endif()
  349. endforeach()
  350. # Extract the set of libraries to link against from the link command
  351. # line
  352. string(REGEX MATCHALL "(^| )-l([^\" ]+|\"[^\"]+\")" MPI_LIBNAMES "${MPI_LINK_CMDLINE}")
  353. if(WIN32)
  354. # The intel wrappers on windows link against static versions of the MPI libraries.
  355. # The static libraries are simply listed on the command line without -l.
  356. # For instance: " icl ... impi.lib "
  357. string(REGEX MATCHALL "(^| )([^\" ]+)\\.lib" tmp "${MPI_LINK_CMDLINE}")
  358. list(APPEND MPI_LIBNAMES ${tmp})
  359. endif()
  360. # add the compiler implicit directories because some compilers
  361. # such as the intel compiler have libraries that show up
  362. # in the showme list that can only be found in the implicit
  363. # link directories of the compiler.
  364. if (DEFINED CMAKE_${lang}_IMPLICIT_LINK_DIRECTORIES)
  365. set(MPI_LINK_PATH
  366. "${MPI_LINK_PATH};${CMAKE_${lang}_IMPLICIT_LINK_DIRECTORIES}")
  367. endif ()
  368. # Determine full path names for all of the libraries that one needs
  369. # to link against in an MPI program
  370. foreach(LIB ${MPI_LIBNAMES})
  371. string(REGEX REPLACE "^ ?-l" "" LIB ${LIB})
  372. if(WIN32)
  373. string(REGEX REPLACE "\\.lib$" "" LIB ${LIB})
  374. endif()
  375. string(STRIP ${LIB} LIB)
  376. # MPI_LIB is cached by find_library, but we don't want that. Clear it first.
  377. set(MPI_LIB "MPI_LIB-NOTFOUND" CACHE FILEPATH "Cleared" FORCE)
  378. find_library(MPI_LIB NAMES ${LIB} HINTS ${MPI_LINK_PATH})
  379. if (MPI_LIB)
  380. list(APPEND MPI_LIBRARIES_WORK ${MPI_LIB})
  381. elseif (NOT MPI_FIND_QUIETLY)
  382. message(WARNING "Unable to find MPI library ${LIB}")
  383. endif()
  384. endforeach()
  385. # Sanity check MPI_LIBRARIES to make sure there are enough libraries
  386. list(LENGTH MPI_LIBRARIES_WORK MPI_NUMLIBS)
  387. list(LENGTH MPI_LIBNAMES MPI_NUMLIBS_EXPECTED)
  388. if (NOT MPI_NUMLIBS EQUAL MPI_NUMLIBS_EXPECTED)
  389. set(MPI_LIBRARIES_WORK "MPI_${lang}_LIBRARIES-NOTFOUND")
  390. endif()
  391. endif()
  392. elseif(try_libs)
  393. # If we didn't have an MPI compiler script to interrogate, attempt to find everything
  394. # with plain old find functions. This is nasty because MPI implementations have LOTS of
  395. # different library names, so this section isn't going to be very generic. We need to
  396. # make sure it works for MS MPI, though, since there are no compiler wrappers for that.
  397. find_path(MPI_HEADER_PATH mpi.h
  398. HINTS ${_MPI_BASE_DIR} ${_MPI_PREFIX_PATH}
  399. PATH_SUFFIXES include Inc)
  400. set(MPI_INCLUDE_PATH_WORK ${MPI_HEADER_PATH})
  401. # Decide between 32-bit and 64-bit libraries for Microsoft's MPI
  402. if("${CMAKE_SIZEOF_VOID_P}" EQUAL 8)
  403. set(MS_MPI_ARCH_DIR x64)
  404. set(MS_MPI_ARCH_DIR2 amd64)
  405. else()
  406. set(MS_MPI_ARCH_DIR x86)
  407. set(MS_MPI_ARCH_DIR2 i386)
  408. endif()
  409. set(MPI_LIB "MPI_LIB-NOTFOUND" CACHE FILEPATH "Cleared" FORCE)
  410. find_library(MPI_LIB
  411. NAMES mpi mpich mpich2 msmpi
  412. HINTS ${_MPI_BASE_DIR} ${_MPI_PREFIX_PATH}
  413. PATH_SUFFIXES lib lib/${MS_MPI_ARCH_DIR} Lib Lib/${MS_MPI_ARCH_DIR} Lib/${MS_MPI_ARCH_DIR2})
  414. set(MPI_LIBRARIES_WORK ${MPI_LIB})
  415. # Right now, we only know about the extra libs for C++.
  416. # We could add Fortran here (as there is usually libfmpich, etc.), but
  417. # this really only has to work with MS MPI on Windows.
  418. # Assume that other MPI's are covered by the compiler wrappers.
  419. if (${lang} STREQUAL CXX)
  420. set(MPI_LIB "MPI_LIB-NOTFOUND" CACHE FILEPATH "Cleared" FORCE)
  421. find_library(MPI_LIB
  422. NAMES mpi++ mpicxx cxx mpi_cxx
  423. HINTS ${_MPI_BASE_DIR} ${_MPI_PREFIX_PATH}
  424. PATH_SUFFIXES lib)
  425. if (MPI_LIBRARIES_WORK AND MPI_LIB)
  426. list(APPEND MPI_LIBRARIES_WORK ${MPI_LIB})
  427. endif()
  428. endif()
  429. if (NOT MPI_LIBRARIES_WORK)
  430. set(MPI_LIBRARIES_WORK "MPI_${lang}_LIBRARIES-NOTFOUND")
  431. endif()
  432. endif()
  433. # If we found MPI, set up all of the appropriate cache entries
  434. set(MPI_${lang}_COMPILE_FLAGS ${MPI_COMPILE_FLAGS_WORK} CACHE STRING "MPI ${lang} compilation flags" FORCE)
  435. set(MPI_${lang}_INCLUDE_PATH ${MPI_INCLUDE_PATH_WORK} CACHE STRING "MPI ${lang} include path" FORCE)
  436. set(MPI_${lang}_LINK_FLAGS ${MPI_LINK_FLAGS_WORK} CACHE STRING "MPI ${lang} linking flags" FORCE)
  437. set(MPI_${lang}_LIBRARIES ${MPI_LIBRARIES_WORK} CACHE STRING "MPI ${lang} libraries to link against" FORCE)
  438. mark_as_advanced(MPI_${lang}_COMPILE_FLAGS MPI_${lang}_INCLUDE_PATH MPI_${lang}_LINK_FLAGS MPI_${lang}_LIBRARIES)
  439. # clear out our temporary lib/header detectionv variable here.
  440. set(MPI_LIB "MPI_LIB-NOTFOUND" CACHE INTERNAL "Scratch variable for MPI lib detection" FORCE)
  441. set(MPI_HEADER_PATH "MPI_HEADER_PATH-NOTFOUND" CACHE INTERNAL "Scratch variable for MPI header detection" FORCE)
  442. endif()
  443. # finally set a found variable for each MPI language
  444. if (MPI_${lang}_INCLUDE_PATH AND MPI_${lang}_LIBRARIES)
  445. set(MPI_${lang}_FOUND TRUE PARENT_SCOPE)
  446. else()
  447. set(MPI_${lang}_FOUND FALSE PARENT_SCOPE)
  448. endif()
  449. endfunction()
  450. # This function attempts to compile with the regular compiler, to see if MPI programs
  451. # work with it. This is a last ditch attempt after we've tried interrogating mpicc and
  452. # friends, and after we've tried to find generic libraries. Works on machines like
  453. # Cray XE6, where the modules environment changes what MPI version cc, CC, and ftn use.
  454. function(try_regular_compiler lang success)
  455. set(scratch_directory ${CMAKE_CURRENT_BINARY_DIR}${CMAKE_FILES_DIRECTORY})
  456. if (${lang} STREQUAL Fortran)
  457. set(test_file ${scratch_directory}/cmake_mpi_test.f90)
  458. file(WRITE ${test_file}
  459. "program hello\n"
  460. "include 'mpif.h'\n"
  461. "integer ierror\n"
  462. "call MPI_INIT(ierror)\n"
  463. "call MPI_FINALIZE(ierror)\n"
  464. "end\n")
  465. else()
  466. if (${lang} STREQUAL CXX)
  467. set(test_file ${scratch_directory}/cmake_mpi_test.cpp)
  468. else()
  469. set(test_file ${scratch_directory}/cmake_mpi_test.c)
  470. endif()
  471. file(WRITE ${test_file}
  472. "#include <mpi.h>\n"
  473. "int main(int argc, char **argv) {\n"
  474. " MPI_Init(&argc, &argv);\n"
  475. " MPI_Finalize();\n"
  476. "}\n")
  477. endif()
  478. try_compile(compiler_has_mpi ${scratch_directory} ${test_file})
  479. if (compiler_has_mpi)
  480. set(MPI_${lang}_NO_INTERROGATE ${CMAKE_${lang}_COMPILER} CACHE STRING "Whether to interrogate MPI ${lang} compiler" FORCE)
  481. set(MPI_${lang}_COMPILER ${CMAKE_${lang}_COMPILER} CACHE STRING "MPI ${lang} compiler" FORCE)
  482. set(MPI_${lang}_COMPILE_FLAGS "" CACHE STRING "MPI ${lang} compilation flags" FORCE)
  483. set(MPI_${lang}_INCLUDE_PATH "" CACHE STRING "MPI ${lang} include path" FORCE)
  484. set(MPI_${lang}_LINK_FLAGS "" CACHE STRING "MPI ${lang} linking flags" FORCE)
  485. set(MPI_${lang}_LIBRARIES "" CACHE STRING "MPI ${lang} libraries to link against" FORCE)
  486. endif()
  487. set(${success} ${compiler_has_mpi} PARENT_SCOPE)
  488. unset(compiler_has_mpi CACHE)
  489. endfunction()
  490. # End definitions, commence real work here.
  491. # Most mpi distros have some form of mpiexec which gives us something we can reliably look for.
  492. find_program(MPIEXEC
  493. NAMES ${_MPI_EXEC_NAMES}
  494. HINTS ${MPI_HOME} $ENV{MPI_HOME}
  495. PATHS ${_MPI_PREFIX_PATH}
  496. PATH_SUFFIXES bin
  497. DOC "Executable for running MPI programs.")
  498. # call get_filename_component twice to remove mpiexec and the directory it exists in (typically bin).
  499. # This gives us a fairly reliable base directory to search for /bin /lib and /include from.
  500. get_filename_component(_MPI_BASE_DIR "${MPIEXEC}" PATH)
  501. get_filename_component(_MPI_BASE_DIR "${_MPI_BASE_DIR}" PATH)
  502. set(MPIEXEC_NUMPROC_FLAG "-np" CACHE STRING "Flag used by MPI to specify the number of processes for MPIEXEC; the next option will be the number of processes.")
  503. set(MPIEXEC_PREFLAGS "" CACHE STRING "These flags will be directly before the executable that is being run by MPIEXEC.")
  504. set(MPIEXEC_POSTFLAGS "" CACHE STRING "These flags will come after all flags given to MPIEXEC.")
  505. set(MPIEXEC_MAX_NUMPROCS "2" CACHE STRING "Maximum number of processors available to run MPI applications.")
  506. mark_as_advanced(MPIEXEC MPIEXEC_NUMPROC_FLAG MPIEXEC_PREFLAGS MPIEXEC_POSTFLAGS MPIEXEC_MAX_NUMPROCS)
  507. #=============================================================================
  508. # Backward compatibility input hacks. Propagate the FindMPI hints to C and
  509. # CXX if the respective new versions are not defined. Translate the old
  510. # MPI_LIBRARY and MPI_EXTRA_LIBRARY to respective MPI_${lang}_LIBRARIES.
  511. #
  512. # Once we find the new variables, we translate them back into their old
  513. # equivalents below.
  514. foreach (lang C CXX)
  515. # Old input variables.
  516. set(_MPI_OLD_INPUT_VARS COMPILER COMPILE_FLAGS INCLUDE_PATH LINK_FLAGS)
  517. # Set new vars based on their old equivalents, if the new versions are not already set.
  518. foreach (var ${_MPI_OLD_INPUT_VARS})
  519. if (NOT MPI_${lang}_${var} AND MPI_${var})
  520. set(MPI_${lang}_${var} "${MPI_${var}}")
  521. endif()
  522. endforeach()
  523. # Special handling for MPI_LIBRARY and MPI_EXTRA_LIBRARY, which we nixed in the
  524. # new FindMPI. These need to be merged into MPI_<lang>_LIBRARIES
  525. if (NOT MPI_${lang}_LIBRARIES AND (MPI_LIBRARY OR MPI_EXTRA_LIBRARY))
  526. set(MPI_${lang}_LIBRARIES ${MPI_LIBRARY} ${MPI_EXTRA_LIBRARY})
  527. endif()
  528. endforeach()
  529. #=============================================================================
  530. # This loop finds the compilers and sends them off for interrogation.
  531. foreach (lang C CXX Fortran)
  532. if (CMAKE_${lang}_COMPILER_WORKS)
  533. # If the user supplies a compiler *name* instead of an absolute path, assume that we need to find THAT compiler.
  534. if (MPI_${lang}_COMPILER)
  535. if (NOT IS_ABSOLUTE "${MPI_${lang}_COMPILER}")
  536. # Get rid of our default list of names and just search for the name the user wants.
  537. set(_MPI_${lang}_COMPILER_NAMES ${MPI_${lang}_COMPILER})
  538. set(MPI_${lang}_COMPILER "MPI_${lang}_COMPILER-NOTFOUND" CACHE FILEPATH "Cleared" FORCE)
  539. endif()
  540. # If the user specifies a compiler, we don't want to try to search libraries either.
  541. set(try_libs FALSE)
  542. else()
  543. set(try_libs TRUE)
  544. endif()
  545. find_program(MPI_${lang}_COMPILER
  546. NAMES ${_MPI_${lang}_COMPILER_NAMES}
  547. HINTS ${_MPI_BASE_DIR}/bin
  548. PATHS ${_MPI_PREFIX_PATH}
  549. )
  550. interrogate_mpi_compiler(${lang} ${try_libs})
  551. mark_as_advanced(MPI_${lang}_COMPILER)
  552. # last ditch try -- if nothing works so far, just try running the regular compiler and
  553. # see if we can create an MPI executable.
  554. set(regular_compiler_worked 0)
  555. if (NOT MPI_${lang}_LIBRARIES OR NOT MPI_${lang}_INCLUDE_PATH)
  556. try_regular_compiler(${lang} regular_compiler_worked)
  557. endif()
  558. set(MPI_${lang}_FIND_QUIETLY ${MPI_FIND_QUIETLY})
  559. set(MPI_${lang}_FIND_REQUIRED ${MPI_FIND_REQUIRED})
  560. set(MPI_${lang}_FIND_VERSION ${MPI_FIND_VERSION})
  561. set(MPI_${lang}_FIND_VERSION_EXACT ${MPI_FIND_VERSION_EXACT})
  562. if (regular_compiler_worked)
  563. find_package_handle_standard_args(MPI_${lang} DEFAULT_MSG MPI_${lang}_COMPILER)
  564. else()
  565. find_package_handle_standard_args(MPI_${lang} DEFAULT_MSG MPI_${lang}_LIBRARIES MPI_${lang}_INCLUDE_PATH)
  566. endif()
  567. endif()
  568. endforeach()
  569. #=============================================================================
  570. # More backward compatibility stuff
  571. #
  572. # Bare MPI sans ${lang} vars are set to CXX then C, depending on what was found.
  573. # This mimics the behavior of the old language-oblivious FindMPI.
  574. set(_MPI_OLD_VARS FOUND COMPILER INCLUDE_PATH COMPILE_FLAGS LINK_FLAGS LIBRARIES)
  575. if (MPI_CXX_FOUND)
  576. foreach (var ${_MPI_OLD_VARS})
  577. set(MPI_${var} ${MPI_CXX_${var}})
  578. endforeach()
  579. elseif (MPI_C_FOUND)
  580. foreach (var ${_MPI_OLD_VARS})
  581. set(MPI_${var} ${MPI_C_${var}})
  582. endforeach()
  583. else()
  584. # Note that we might still have found Fortran, but you'll need to use MPI_Fortran_FOUND
  585. set(MPI_FOUND FALSE)
  586. endif()
  587. # Chop MPI_LIBRARIES into the old-style MPI_LIBRARY and MPI_EXTRA_LIBRARY, and set them in cache.
  588. if (MPI_LIBRARIES)
  589. list(GET MPI_LIBRARIES 0 MPI_LIBRARY_WORK)
  590. set(MPI_LIBRARY ${MPI_LIBRARY_WORK} CACHE FILEPATH "MPI library to link against" FORCE)
  591. else()
  592. set(MPI_LIBRARY "MPI_LIBRARY-NOTFOUND" CACHE FILEPATH "MPI library to link against" FORCE)
  593. endif()
  594. list(LENGTH MPI_LIBRARIES MPI_NUMLIBS)
  595. if (MPI_NUMLIBS GREATER 1)
  596. set(MPI_EXTRA_LIBRARY_WORK ${MPI_LIBRARIES})
  597. list(REMOVE_AT MPI_EXTRA_LIBRARY_WORK 0)
  598. set(MPI_EXTRA_LIBRARY ${MPI_EXTRA_LIBRARY_WORK} CACHE STRING "Extra MPI libraries to link against" FORCE)
  599. else()
  600. set(MPI_EXTRA_LIBRARY "MPI_EXTRA_LIBRARY-NOTFOUND" CACHE STRING "Extra MPI libraries to link against" FORCE)
  601. endif()
  602. #=============================================================================
  603. # unset these vars to cleanup namespace
  604. unset(_MPI_OLD_VARS)
  605. unset(_MPI_PREFIX_PATH)
  606. unset(_MPI_BASE_DIR)
  607. foreach (lang C CXX Fortran)
  608. unset(_MPI_${lang}_COMPILER_NAMES)
  609. endforeach()