macros.cmake 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442
  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(PrintEnabledBackends _SUBSYS _REGEXP)
  320. get_cmake_property(_ALLVARS VARIABLES)
  321. foreach(_VAR IN LISTS _ALLVARS)
  322. if(_VAR AND _VAR MATCHES "${_REGEXP}")
  323. string(TOLOWER "${CMAKE_MATCH_1}" _LOWERED)
  324. if(NOT _LOWERED MATCHES "available|default|dynamic") # a little hack
  325. if(${_VAR}_DYNAMIC)
  326. list(APPEND _ENABLED_BACKENDS "${_LOWERED}(dynamic)")
  327. else()
  328. list(APPEND _ENABLED_BACKENDS "${_LOWERED}")
  329. endif()
  330. endif()
  331. endif()
  332. endforeach()
  333. if(_ENABLED_BACKENDS STREQUAL "")
  334. set(_SPACEDOUT "(none)")
  335. else()
  336. string(REPLACE ";" " " _SPACEDOUT "${_ENABLED_BACKENDS}")
  337. endif()
  338. message(STATUS " ${_SUBSYS}: ${_SPACEDOUT}")
  339. endfunction()
  340. function(SDL_PrintSummary)
  341. ##### Info output #####
  342. message(STATUS "")
  343. message(STATUS "SDL3 was configured with the following options:")
  344. message(STATUS "")
  345. message(STATUS "Platform: ${CMAKE_SYSTEM}")
  346. message(STATUS "64-bit: ${ARCH_64}")
  347. message(STATUS "Compiler: ${CMAKE_C_COMPILER}")
  348. message(STATUS "Revision: ${SDL_REVISION}")
  349. message(STATUS "Vendor: ${SDL_VENDOR_INFO}")
  350. message(STATUS "")
  351. message(STATUS "Subsystems:")
  352. find_stringlength_longest_item(SDL_SUBSYSTEMS maxLength)
  353. foreach(_SUB IN LISTS SDL_SUBSYSTEMS)
  354. string(LENGTH ${_SUB} _SUBLEN)
  355. math(EXPR _PADLEN "(${maxLength} + 1) - ${_SUBLEN}")
  356. string(RANDOM LENGTH ${_PADLEN} ALPHABET " " _PADDING)
  357. string(TOUPPER ${_SUB} _OPT)
  358. message_bool_option(${_SUB} SDL_${_OPT} ${_PADDING})
  359. endforeach()
  360. message(STATUS "")
  361. message(STATUS "Options:")
  362. list(SORT ALLOPTIONS)
  363. message_dictlist(ALLOPTIONS)
  364. message(STATUS "")
  365. message(STATUS " Build Shared Library: ${SDL_SHARED}")
  366. message(STATUS " Build Static Library: ${SDL_STATIC}")
  367. if(APPLE)
  368. message(STATUS " Build libraries as Apple Framework: ${SDL_FRAMEWORK}")
  369. endif()
  370. message(STATUS "")
  371. message(STATUS "Enabled backends:")
  372. PrintEnabledBackends("Video drivers" "^SDL_VIDEO_DRIVER_([A-Z0-9]*)$")
  373. if(SDL_VIDEO_DRIVER_X11)
  374. PrintEnabledBackends("X11 libraries" "^SDL_VIDEO_DRIVER_X11_([A-Z0-9]*)$")
  375. endif()
  376. PrintEnabledBackends("Render drivers" "^SDL_VIDEO_RENDER_([A-Z0-9_]*)$")
  377. PrintEnabledBackends("GPU drivers" "^SDL_GPU_([A-Z0-9]*)$")
  378. PrintEnabledBackends("Audio drivers" "^SDL_AUDIO_DRIVER_([A-Z0-9]*)$")
  379. PrintEnabledBackends("Joystick drivers" "^SDL_JOYSTICK_([A-Z0-9]*)$")
  380. message(STATUS "")
  381. if(UNIX)
  382. message(STATUS "If something was not detected, although the libraries")
  383. message(STATUS "were installed, then make sure you have set the")
  384. message(STATUS "CMAKE_C_FLAGS and CMAKE_PREFIX_PATH CMake variables correctly.")
  385. message(STATUS "")
  386. endif()
  387. if(UNIX AND NOT (ANDROID OR APPLE OR EMSCRIPTEN OR HAIKU OR RISCOS))
  388. if(NOT (HAVE_X11 OR HAVE_WAYLAND))
  389. if(NOT SDL_UNIX_CONSOLE_BUILD)
  390. message(FATAL_ERROR
  391. "SDL could not find X11 or Wayland development libraries on your system. "
  392. "This means SDL will not be able to create windows on a typical unix operating system. "
  393. "Most likely, this is not wanted."
  394. "\n"
  395. "On Linux, install the packages listed at "
  396. "https://github.com/libsdl-org/SDL/blob/main/docs/README-linux.md#build-dependencies "
  397. "\n"
  398. "If you really don't need desktop windows, the documentation tells you how to skip this check. "
  399. "https://github.com/libsdl-org/SDL/blob/main/docs/README-cmake.md#cmake-fails-to-build-without-x11-or-wayland-support\n"
  400. )
  401. endif()
  402. endif()
  403. endif()
  404. endfunction()
  405. function(SDL_install_pdb TARGET DIRECTORY)
  406. get_property(type TARGET ${TARGET} PROPERTY TYPE)
  407. if(type MATCHES "^(SHARED_LIBRARY|EXECUTABLE)$")
  408. install(FILES $<TARGET_PDB_FILE:${TARGET}> DESTINATION "${DIRECTORY}" OPTIONAL)
  409. elseif(type STREQUAL "STATIC_LIBRARY")
  410. # FIXME: Use $<TARGET_COMPILE_PDB_FILE:${TARGET} once it becomes available (https://gitlab.kitware.com/cmake/cmake/-/issues/25244)
  411. if(CMAKE_GENERATOR MATCHES "^Visual Studio.*")
  412. install(CODE "file(INSTALL DESTINATION \"\${CMAKE_INSTALL_PREFIX}/${DIRECTORY}\" TYPE FILE OPTIONAL FILES \"${CMAKE_CURRENT_BINARY_DIR}/\${CMAKE_INSTALL_CONFIG_NAME}/${TARGET}.pdb\")")
  413. else()
  414. install(CODE "file(INSTALL DESTINATION \"\${CMAKE_INSTALL_PREFIX}/${DIRECTORY}\" TYPE FILE OPTIONAL FILES \"${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/${TARGET}.dir/${TARGET}.pdb\")")
  415. endif()
  416. endif()
  417. endfunction()