macros.cmake 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. macro(add_to_alloptions _NEWNAME)
  2. list(APPEND ALLOPTIONS ${_NEWNAME})
  3. endmacro()
  4. macro(set_option _NAME _DESC)
  5. add_to_alloptions(${_NAME})
  6. if(${ARGC} EQUAL 3)
  7. set(_DEFLT ${ARGV2})
  8. else()
  9. set(_DEFLT OFF)
  10. endif()
  11. option(${_NAME} ${_DESC} ${_DEFLT})
  12. endmacro()
  13. macro(dep_option _NAME _DESC _DEFLT _DEPTEST _FAILDFLT)
  14. add_to_alloptions("${_NAME}")
  15. cmake_dependent_option("${_NAME}" "${_DESC}" "${_DEFLT}" "${_DEPTEST}" "${_FAILDFLT}")
  16. endmacro()
  17. macro(option_string _NAME _DESC _VALUE)
  18. add_to_alloptions(${_NAME})
  19. set(${_NAME} ${_VALUE} CACHE STRING "${_DESC}")
  20. set(HAVE_${_NAME} ${_VALUE})
  21. ENDMACRO()
  22. macro(message_bool_option _NAME _VALUE)
  23. set(_PAD "\t")
  24. if(${ARGC} EQUAL 3)
  25. set(_PAD ${ARGV2})
  26. endif()
  27. if(${_VALUE})
  28. message(STATUS " ${_NAME}:${_PAD}ON")
  29. else()
  30. message(STATUS " ${_NAME}:${_PAD}OFF")
  31. endif()
  32. endmacro()
  33. macro(message_tested_option _NAME)
  34. set(_REQVALUE ${${_NAME}})
  35. set(_PAD " ")
  36. if(${ARGC} EQUAL 2)
  37. set(_PAD ${ARGV1})
  38. endif()
  39. string(SUBSTRING "${_NAME}" 0 4 _NAMESTART)
  40. if(_NAMESTART STREQUAL "SDL_")
  41. string(SUBSTRING "${_NAME}" 4 -1 _STRIPPEDNAME)
  42. else()
  43. set(_STRIPPEDNAME "${_NAME}")
  44. endif()
  45. if(NOT HAVE_${_STRIPPEDNAME})
  46. set(HAVE_${_STRIPPEDNAME} OFF)
  47. elseif("${HAVE_${_STRIPPEDNAME}}" MATCHES "1|TRUE|YES|Y")
  48. set(HAVE_${_STRIPPEDNAME} ON)
  49. endif()
  50. message(STATUS " ${_NAME}${_PAD}(Wanted: ${_REQVALUE}): ${HAVE_${_STRIPPEDNAME}}")
  51. endmacro()
  52. function(find_stringlength_longest_item inList outLength)
  53. set(maxLength 0)
  54. foreach(item IN LISTS ${inList})
  55. string(LENGTH "${item}" slen)
  56. if(slen GREATER maxLength)
  57. set(maxLength ${slen})
  58. endif()
  59. endforeach()
  60. set("${outLength}" ${maxLength} PARENT_SCOPE)
  61. endfunction()
  62. function(message_dictlist inList)
  63. find_stringlength_longest_item(${inList} maxLength)
  64. foreach(name IN LISTS ${inList})
  65. # Get the padding
  66. string(LENGTH ${name} nameLength)
  67. math(EXPR padLength "(${maxLength} + 1) - ${nameLength}")
  68. string(RANDOM LENGTH ${padLength} ALPHABET " " padding)
  69. message_tested_option(${name} ${padding})
  70. endforeach()
  71. endfunction()
  72. if(APPLE)
  73. include(CheckOBJCSourceCompiles)
  74. enable_language(OBJC)
  75. endif()
  76. function(SDL_detect_linker)
  77. if(CMAKE_VERSION VERSION_LESS 3.29)
  78. if(NOT DEFINED SDL_CMAKE_C_COMPILER_LINKER_ID)
  79. execute_process(COMMAND ${CMAKE_LINKER} -v OUTPUT_VARIABLE LINKER_OUTPUT ERROR_VARIABLE LINKER_OUTPUT)
  80. string(REGEX REPLACE "[\r\n]" " " LINKER_OUTPUT "${LINKER_OUTPUT}")
  81. if(LINKER_OUTPUT MATCHES ".*Microsoft.*")
  82. set(linker MSVC)
  83. else()
  84. set(linker GNUlike)
  85. endif()
  86. message(STATUS "Linker identification: ${linker}")
  87. set(SDL_CMAKE_C_COMPILER_LINKER_ID "${linker}" CACHE STRING "Linker identification")
  88. mark_as_advanced(SDL_CMAKE_C_COMPILER_LINKER_ID)
  89. endif()
  90. set(CMAKE_C_COMPILER_LINKER_ID "${SDL_CMAKE_C_COMPILER_LINKER_ID}" PARENT_SCOPE)
  91. endif()
  92. endfunction()
  93. function(read_absolute_symlink DEST PATH)
  94. file(READ_SYMLINK "${PATH}" p)
  95. if(NOT IS_ABSOLUTE "${p}")
  96. get_filename_component(pdir "${PATH}" DIRECTORY)
  97. set(p "${pdir}/${p}")
  98. endif()
  99. get_filename_component(p "${p}" ABSOLUTE)
  100. set("${DEST}" "${p}" PARENT_SCOPE)
  101. endfunction()
  102. function(win32_implib_identify_dll DEST IMPLIB)
  103. cmake_parse_arguments(ARGS "NOTFATAL" "" "" ${ARGN})
  104. if(CMAKE_DLLTOOL)
  105. execute_process(
  106. COMMAND "${CMAKE_DLLTOOL}" --identify "${IMPLIB}"
  107. RESULT_VARIABLE retcode
  108. OUTPUT_VARIABLE stdout
  109. ERROR_VARIABLE stderr)
  110. if(NOT retcode EQUAL 0)
  111. if(NOT ARGS_NOTFATAL)
  112. message(FATAL_ERROR "${CMAKE_DLLTOOL} failed.")
  113. else()
  114. set("${DEST}" "${DEST}-NOTFOUND" PARENT_SCOPE)
  115. return()
  116. endif()
  117. endif()
  118. string(STRIP "${stdout}" result)
  119. set(${DEST} "${result}" PARENT_SCOPE)
  120. elseif(MSVC)
  121. get_filename_component(CMAKE_C_COMPILER_DIRECTORY "${CMAKE_C_COMPILER}" DIRECTORY CACHE)
  122. find_program(CMAKE_DUMPBIN NAMES dumpbin PATHS "${CMAKE_C_COMPILER_DIRECTORY}")
  123. if(CMAKE_DUMPBIN)
  124. execute_process(
  125. COMMAND "${CMAKE_DUMPBIN}" "-headers" "${IMPLIB}"
  126. RESULT_VARIABLE retcode
  127. OUTPUT_VARIABLE stdout
  128. ERROR_VARIABLE stderr)
  129. if(NOT retcode EQUAL 0)
  130. if(NOT ARGS_NOTFATAL)
  131. message(FATAL_ERROR "dumpbin failed.")
  132. else()
  133. set(${DEST} "${DEST}-NOTFOUND" PARENT_SCOPE)
  134. return()
  135. endif()
  136. endif()
  137. string(REGEX MATCH "DLL name[ ]+:[ ]+([^\n]+)\n" match "${stdout}")
  138. if(NOT match)
  139. if(NOT ARGS_NOTFATAL)
  140. message(FATAL_ERROR "dumpbin did not find any associated dll for ${IMPLIB}.")
  141. else()
  142. set(${DEST} "${DEST}-NOTFOUND" PARENT_SCOPE)
  143. return()
  144. endif()
  145. endif()
  146. set(result "${CMAKE_MATCH_1}")
  147. set(${DEST} "${result}" PARENT_SCOPE)
  148. else()
  149. message(FATAL_ERROR "Cannot find dumpbin, please set CMAKE_DUMPBIN cmake variable")
  150. endif()
  151. else()
  152. if(NOT ARGS_NOTFATAL)
  153. message(FATAL_ERROR "Don't know how to identify dll from import library. Set CMAKE_DLLTOOL (for mingw) or CMAKE_DUMPBIN (for MSVC)")
  154. else()
  155. set(${DEST} "${DEST}-NOTFOUND")
  156. endif()
  157. endif()
  158. endfunction()
  159. function(get_actual_target)
  160. set(dst "${ARGV0}")
  161. set(target "${${dst}}")
  162. set(input "${target}")
  163. get_target_property(alias "${target}" ALIASED_TARGET)
  164. while(alias)
  165. set(target "${alias}")
  166. get_target_property(alias "${target}" ALIASED_TARGET)
  167. endwhile()
  168. message(DEBUG "get_actual_target(\"${input}\") -> \"${target}\"")
  169. set("${dst}" "${target}" PARENT_SCOPE)
  170. endfunction()
  171. function(target_get_dynamic_library DEST TARGET)
  172. set(result)
  173. get_actual_target(TARGET)
  174. if(WIN32)
  175. # Use the target dll of the import library
  176. set(props_to_check IMPORTED_IMPLIB)
  177. if(CMAKE_BUILD_TYPE)
  178. list(APPEND props_to_check IMPORTED_IMPLIB_${CMAKE_BUILD_TYPE})
  179. endif()
  180. list(APPEND props_to_check IMPORTED_LOCATION)
  181. if(CMAKE_BUILD_TYPE)
  182. list(APPEND props_to_check IMPORTED_LOCATION_${CMAKE_BUILD_TYPE})
  183. endif()
  184. foreach (config_type ${CMAKE_CONFIGURATION_TYPES} RELEASE DEBUG RELWITHDEBINFO MINSIZEREL)
  185. list(APPEND props_to_check IMPORTED_IMPLIB_${config_type})
  186. list(APPEND props_to_check IMPORTED_LOCATION_${config_type})
  187. endforeach()
  188. foreach(prop_to_check ${props_to_check})
  189. if(NOT result)
  190. get_target_property(propvalue "${TARGET}" ${prop_to_check})
  191. if(propvalue AND EXISTS "${propvalue}")
  192. win32_implib_identify_dll(result "${propvalue}" NOTFATAL)
  193. endif()
  194. endif()
  195. endforeach()
  196. else()
  197. # 1. find the target library a file might be symbolic linking to
  198. # 2. find all other files in the same folder that symbolic link to it
  199. # 3. sort all these files, and select the 1st item on Linux, and last on macOS
  200. set(location_properties IMPORTED_LOCATION)
  201. if(CMAKE_BUILD_TYPE)
  202. list(APPEND location_properties IMPORTED_LOCATION_${CMAKE_BUILD_TYPE})
  203. endif()
  204. foreach (config_type ${CMAKE_CONFIGURATION_TYPES} RELEASE DEBUG RELWITHDEBINFO MINSIZEREL)
  205. list(APPEND location_properties IMPORTED_LOCATION_${config_type})
  206. endforeach()
  207. if(APPLE)
  208. set(valid_shared_library_regex "\\.[0-9]+\\.dylib$")
  209. else()
  210. set(valid_shared_library_regex "\\.so\\.([0-9.]+)?[0-9]")
  211. endif()
  212. foreach(location_property ${location_properties})
  213. if(NOT result)
  214. get_target_property(library_path "${TARGET}" ${location_property})
  215. message(DEBUG "get_target_property(${TARGET} ${location_property}) -> ${library_path}")
  216. if(EXISTS "${library_path}")
  217. get_filename_component(library_path "${library_path}" ABSOLUTE)
  218. while (IS_SYMLINK "${library_path}")
  219. read_absolute_symlink(library_path "${library_path}")
  220. endwhile()
  221. message(DEBUG "${TARGET} -> ${library_path}")
  222. get_filename_component(libdir "${library_path}" DIRECTORY)
  223. file(GLOB subfiles "${libdir}/*")
  224. set(similar_files "${library_path}")
  225. foreach(subfile ${subfiles})
  226. if(IS_SYMLINK "${subfile}")
  227. read_absolute_symlink(subfile_target "${subfile}")
  228. while(IS_SYMLINK "${subfile_target}")
  229. read_absolute_symlink(subfile_target "${subfile_target}")
  230. endwhile()
  231. get_filename_component(subfile_target "${subfile_target}" ABSOLUTE)
  232. if(subfile_target STREQUAL library_path AND subfile MATCHES "${valid_shared_library_regex}")
  233. list(APPEND similar_files "${subfile}")
  234. endif()
  235. endif()
  236. endforeach()
  237. list(SORT similar_files)
  238. message(DEBUG "files that are similar to \"${library_path}\"=${similar_files}")
  239. if(APPLE)
  240. list(REVERSE similar_files)
  241. endif()
  242. list(GET similar_files 0 item)
  243. get_filename_component(result "${item}" NAME)
  244. endif()
  245. endif()
  246. endforeach()
  247. endif()
  248. if(result)
  249. string(TOLOWER "${result}" result_lower)
  250. if(WIN32 OR OS2)
  251. if(NOT result_lower MATCHES ".*dll")
  252. message(FATAL_ERROR "\"${result}\" is not a .dll library")
  253. endif()
  254. elseif(APPLE)
  255. if(NOT result_lower MATCHES ".*dylib.*")
  256. message(FATAL_ERROR "\"${result}\" is not a .dylib shared library")
  257. endif()
  258. else()
  259. if(NOT result_lower MATCHES ".*so.*")
  260. message(FATAL_ERROR "\"${result}\" is not a .so shared library")
  261. endif()
  262. endif()
  263. else()
  264. get_target_property(target_type ${TARGET} TYPE)
  265. if(target_type MATCHES "SHARED_LIBRARY|MODULE_LIBRARY")
  266. # OK
  267. elseif(target_type MATCHES "STATIC_LIBRARY|OBJECT_LIBRARY|INTERFACE_LIBRARY|EXECUTABLE")
  268. message(SEND_ERROR "${TARGET} is not a shared library, but has type=${target_type}")
  269. else()
  270. message(WARNING "Unable to extract dynamic library from target=${TARGET}, type=${target_type}.")
  271. endif()
  272. # TARGET_SONAME_FILE is not allowed for DLL target platforms.
  273. if(WIN32)
  274. set(result "$<TARGET_FILE_NAME:${TARGET}>")
  275. else()
  276. set(result "$<TARGET_SONAME_FILE_NAME:${TARGET}>")
  277. endif()
  278. endif()
  279. set(${DEST} ${result} PARENT_SCOPE)
  280. endfunction()
  281. function(check_linker_supports_version_file VAR)
  282. SDL_detect_linker()
  283. if(CMAKE_C_COMPILER_LINKER_ID MATCHES "^(MSVC)$")
  284. set(LINKER_SUPPORTS_VERSION_SCRIPT FALSE)
  285. else()
  286. cmake_push_check_state(RESET)
  287. file(WRITE "${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/dummy.sym" "n_0 {\n global:\n func;\n local: *;\n};\n")
  288. list(APPEND CMAKE_REQUIRED_LINK_OPTIONS "-Wl,--version-script=${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/dummy.sym")
  289. check_c_source_compiles("int func(void) {return 0;} int main(int argc,char*argv[]){(void)argc;(void)argv;return func();}" LINKER_SUPPORTS_VERSION_SCRIPT FAIL_REGEX "(unsupported|syntax error|unrecognized option)")
  290. cmake_pop_check_state()
  291. endif()
  292. set(${VAR} "${LINKER_SUPPORTS_VERSION_SCRIPT}" PARENT_SCOPE)
  293. endfunction()
  294. if(CMAKE_VERSION VERSION_LESS 3.18)
  295. function(check_linker_flag LANG FLAG VAR)
  296. cmake_push_check_state(RESET)
  297. list(APPEND CMAKE_REQUIRED_LINK_OPTIONS ${FLAG})
  298. if(LANG STREQUAL "C")
  299. include(CheckCSourceCompiles)
  300. check_c_source_compiles("int main(int argc,char*argv[]){(void)argc;(void)argv;return 0;}" ${VAR} FAIL_REGEX "(unsupported|syntax error)")
  301. elseif(LANG STREQUAL "CXX")
  302. include(CheckCXXSourceCompiles)
  303. check_cxx_source_compiles("int main(int argc,char*argv[]){(void)argc;(void)argv;return 0;}" ${VAR} FAIL_REGEX "(unsupported|syntax error)")
  304. else()
  305. message(FATAL_ERROR "Unsupported language: ${LANG}")
  306. endif()
  307. cmake_pop_check_state()
  308. endfunction()
  309. else()
  310. cmake_policy(SET CMP0057 NEW) # Support new if() IN_LIST operator. (used inside check_linker_flag, used in CMake 3.18)
  311. include(CheckLinkerFlag)
  312. endif()
  313. if(APPLE)
  314. check_language(OBJC)
  315. if(NOT CMAKE_OBJC_COMPILER)
  316. message(WARNING "Cannot find working OBJC compiler.")
  317. endif()
  318. endif()
  319. function(SDL_PrintSummary)
  320. ##### Info output #####
  321. message(STATUS "")
  322. message(STATUS "SDL3 was configured with the following options:")
  323. message(STATUS "")
  324. message(STATUS "Platform: ${CMAKE_SYSTEM}")
  325. message(STATUS "64-bit: ${ARCH_64}")
  326. message(STATUS "Compiler: ${CMAKE_C_COMPILER}")
  327. message(STATUS "Revision: ${SDL_REVISION}")
  328. message(STATUS "Vendor: ${SDL_VENDOR_INFO}")
  329. message(STATUS "")
  330. message(STATUS "Subsystems:")
  331. find_stringlength_longest_item(SDL_SUBSYSTEMS maxLength)
  332. foreach(_SUB IN LISTS SDL_SUBSYSTEMS)
  333. string(LENGTH ${_SUB} _SUBLEN)
  334. math(EXPR _PADLEN "(${maxLength} + 1) - ${_SUBLEN}")
  335. string(RANDOM LENGTH ${_PADLEN} ALPHABET " " _PADDING)
  336. string(TOUPPER ${_SUB} _OPT)
  337. message_bool_option(${_SUB} SDL_${_OPT} ${_PADDING})
  338. endforeach()
  339. message(STATUS "")
  340. message(STATUS "Options:")
  341. list(SORT ALLOPTIONS)
  342. message_dictlist(ALLOPTIONS)
  343. message(STATUS "")
  344. message(STATUS " Build Shared Library: ${SDL_SHARED}")
  345. message(STATUS " Build Static Library: ${SDL_STATIC}")
  346. if(APPLE)
  347. message(STATUS " Build libraries as Apple Framework: ${SDL_FRAMEWORK}")
  348. endif()
  349. message(STATUS "")
  350. if(UNIX)
  351. message(STATUS "If something was not detected, although the libraries")
  352. message(STATUS "were installed, then make sure you have set the")
  353. message(STATUS "CMAKE_C_FLAGS and CMAKE_PREFIX_PATH CMake variables correctly.")
  354. message(STATUS "")
  355. endif()
  356. if(UNIX AND NOT (ANDROID OR APPLE OR EMSCRIPTEN OR HAIKU OR RISCOS))
  357. if(NOT (HAVE_X11 OR HAVE_WAYLAND))
  358. if(NOT SDL_UNIX_CONSOLE_BUILD)
  359. message(FATAL_ERROR
  360. "SDL could not find X11 or Wayland development libraries on your system. "
  361. "This means SDL will not be able to create windows on a typical unix operating system. "
  362. "Most likely, this is not wanted."
  363. "\n"
  364. "On Linux, install the packages listed at "
  365. "https://github.com/libsdl-org/SDL/blob/main/docs/README-linux.md#build-dependencies "
  366. "\n"
  367. "If you really don't need desktop windows, the documentation tells you how to skip this check. "
  368. "https://github.com/libsdl-org/SDL/blob/main/docs/README-cmake.md#cmake-fails-to-build-without-x11-or-wayland-support\n"
  369. )
  370. endif()
  371. endif()
  372. endif()
  373. endfunction()
  374. function(SDL_install_pdb TARGET DIRECTORY)
  375. get_property(type TARGET ${TARGET} PROPERTY TYPE)
  376. if(type MATCHES "^(SHARED_LIBRARY|EXECUTABLE)$")
  377. install(FILES $<TARGET_PDB_FILE:${TARGET}> DESTINATION "${DIRECTORY}" OPTIONAL)
  378. elseif(type STREQUAL "STATIC_LIBRARY")
  379. # FIXME: Use $<TARGET_COMPILE_PDB_FILE:${TARGET} once it becomes available (https://gitlab.kitware.com/cmake/cmake/-/issues/25244)
  380. if(CMAKE_GENERATOR MATCHES "^Visual Studio.*")
  381. install(CODE "file(INSTALL DESTINATION \"\${CMAKE_INSTALL_PREFIX}/${DIRECTORY}\" TYPE FILE OPTIONAL FILES \"${CMAKE_CURRENT_BINARY_DIR}/\${CMAKE_INSTALL_CONFIG_NAME}/${TARGET}.pdb\")")
  382. else()
  383. install(CODE "file(INSTALL DESTINATION \"\${CMAKE_INSTALL_PREFIX}/${DIRECTORY}\" TYPE FILE OPTIONAL FILES \"${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/${TARGET}.dir/${TARGET}.pdb\")")
  384. endif()
  385. endif()
  386. endfunction()