1
0

PrivateSdlFunctions.cmake 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. # This file is shared amongst SDL_image/SDL_mixer/SDL_ttf
  2. include(CheckCCompilerFlag)
  3. include(CheckCSourceCompiles)
  4. include(CMakePushCheckState)
  5. macro(sdl_calculate_derived_version_variables MAJOR MINOR MICRO)
  6. set(SO_VERSION_MAJOR "0")
  7. set(SO_VERSION_MINOR "${MINOR_VERSION}")
  8. set(SO_VERSION_MICRO "${MICRO_VERSION}")
  9. set(SO_VERSION "${SO_VERSION_MAJOR}.${SO_VERSION_MINOR}.${SO_VERSION_MICRO}")
  10. if(MINOR MATCHES "[02468]$")
  11. math(EXPR DYLIB_COMPAT_VERSION_MAJOR "100 * ${MINOR} + 1")
  12. set(DYLIB_COMPAT_VERSION_MINOR "0")
  13. math(EXPR DYLIB_CURRENT_VERSION_MAJOR "${DYLIB_COMPAT_VERSION_MAJOR}")
  14. set(DYLIB_CURRENT_VERSION_MINOR "${MICRO}")
  15. else()
  16. math(EXPR DYLIB_COMPAT_VERSION_MAJOR "100 * ${MINOR} + ${MICRO} + 1")
  17. set(DYLIB_COMPAT_VERSION_MINOR "0")
  18. math(EXPR DYLIB_CURRENT_VERSION_MAJOR "${DYLIB_COMPAT_VERSION_MAJOR}")
  19. set(DYLIB_CURRENT_VERSION_MINOR "0")
  20. endif()
  21. set(DYLIB_COMPAT_VERSION_MICRO "0")
  22. set(DYLIB_CURRENT_VERSION_MICRO "0")
  23. set(DYLIB_CURRENT_VERSION "${DYLIB_CURRENT_VERSION_MAJOR}.${DYLIB_CURRENT_VERSION_MINOR}.${DYLIB_CURRENT_VERSION_MICRO}")
  24. set(DYLIB_COMPAT_VERSION "${DYLIB_COMPAT_VERSION_MAJOR}.${DYLIB_COMPAT_VERSION_MINOR}.${DYLIB_COMPAT_VERSION_MICRO}")
  25. endmacro()
  26. function(read_absolute_symlink DEST PATH)
  27. file(READ_SYMLINK "${PATH}" p)
  28. if(NOT IS_ABSOLUTE "${p}")
  29. get_filename_component(pdir "${PATH}" DIRECTORY)
  30. set(p "${pdir}/${p}")
  31. endif()
  32. get_filename_component(p "${p}" ABSOLUTE)
  33. set("${DEST}" "${p}" PARENT_SCOPE)
  34. endfunction()
  35. function(win32_implib_identify_dll DEST IMPLIB)
  36. cmake_parse_arguments(ARGS "NOTFATAL" "" "" ${ARGN})
  37. if(CMAKE_DLLTOOL)
  38. execute_process(
  39. COMMAND "${CMAKE_DLLTOOL}" --identify "${IMPLIB}"
  40. RESULT_VARIABLE retcode
  41. OUTPUT_VARIABLE stdout
  42. ERROR_VARIABLE stderr)
  43. if(NOT retcode EQUAL 0)
  44. if(NOT ARGS_NOTFATAL)
  45. message(FATAL_ERROR "${CMAKE_DLLTOOL} failed.")
  46. else()
  47. set("${DEST}" "${DEST}-NOTFOUND" PARENT_SCOPE)
  48. return()
  49. endif()
  50. endif()
  51. string(STRIP "${stdout}" result)
  52. set(${DEST} "${result}" PARENT_SCOPE)
  53. elseif(MSVC)
  54. get_filename_component(CMAKE_C_COMPILER_DIRECTORY "${CMAKE_C_COMPILER}" DIRECTORY CACHE)
  55. find_program(CMAKE_DUMPBIN NAMES dumpbin PATHS "${CMAKE_C_COMPILER_DIRECTORY}")
  56. if(CMAKE_DUMPBIN)
  57. execute_process(
  58. COMMAND "${CMAKE_DUMPBIN}" "-headers" "${IMPLIB}"
  59. RESULT_VARIABLE retcode
  60. OUTPUT_VARIABLE stdout
  61. ERROR_VARIABLE stderr)
  62. if(NOT retcode EQUAL 0)
  63. if(NOT ARGS_NOTFATAL)
  64. message(FATAL_ERROR "dumpbin failed.")
  65. else()
  66. set(${DEST} "${DEST}-NOTFOUND" PARENT_SCOPE)
  67. return()
  68. endif()
  69. endif()
  70. string(REGEX MATCH "DLL name[ ]+:[ ]+([^\n]+)\n" match "${stdout}")
  71. if(NOT match)
  72. if(NOT ARGS_NOTFATAL)
  73. message(FATAL_ERROR "dumpbin did not find any associated dll for ${IMPLIB}.")
  74. else()
  75. set(${DEST} "${DEST}-NOTFOUND" PARENT_SCOPE)
  76. return()
  77. endif()
  78. endif()
  79. set(result "${CMAKE_MATCH_1}")
  80. set(${DEST} "${result}" PARENT_SCOPE)
  81. else()
  82. message(FATAL_ERROR "Cannot find dumpbin, please set CMAKE_DUMPBIN cmake variable")
  83. endif()
  84. else()
  85. if(NOT ARGS_NOTFATAL)
  86. message(FATAL_ERROR "Don't know how to identify dll from import library. Set CMAKE_DLLTOOL (for mingw) or CMAKE_DUMPBIN (for MSVC)")
  87. else()
  88. set(${DEST} "${DEST}-NOTFOUND")
  89. endif()
  90. endif()
  91. endfunction()
  92. function(get_actual_target)
  93. set(dst "${ARGV0}")
  94. set(target "${${dst}}")
  95. set(input "${target}")
  96. get_target_property(alias "${target}" ALIASED_TARGET)
  97. while(alias)
  98. set(target "${alias}")
  99. get_target_property(alias "${target}" ALIASED_TARGET)
  100. endwhile()
  101. message(DEBUG "get_actual_target(\"${input}\") -> \"${target}\"")
  102. set("${dst}" "${target}" PARENT_SCOPE)
  103. endfunction()
  104. function(target_get_dynamic_library DEST TARGET)
  105. set(result)
  106. get_actual_target(TARGET)
  107. if(WIN32)
  108. # Use the target dll of the import library
  109. set(props_to_check IMPORTED_IMPLIB)
  110. if(CMAKE_BUILD_TYPE)
  111. list(APPEND props_to_check IMPORTED_IMPLIB_${CMAKE_BUILD_TYPE})
  112. endif()
  113. list(APPEND props_to_check IMPORTED_LOCATION)
  114. if(CMAKE_BUILD_TYPE)
  115. list(APPEND props_to_check IMPORTED_LOCATION_${CMAKE_BUILD_TYPE})
  116. endif()
  117. foreach (config_type ${CMAKE_CONFIGURATION_TYPES} RELEASE DEBUG RELWITHDEBINFO MINSIZEREL)
  118. list(APPEND props_to_check IMPORTED_IMPLIB_${config_type})
  119. list(APPEND props_to_check IMPORTED_LOCATION_${config_type})
  120. endforeach()
  121. foreach(prop_to_check ${props_to_check})
  122. if(NOT result)
  123. get_target_property(propvalue "${TARGET}" ${prop_to_check})
  124. if(propvalue AND EXISTS "${propvalue}")
  125. win32_implib_identify_dll(result "${propvalue}" NOTFATAL)
  126. endif()
  127. endif()
  128. endforeach()
  129. else()
  130. # 1. find the target library a file might be symbolic linking to
  131. # 2. find all other files in the same folder that symolic link to it
  132. # 3. sort all these files, and select the 1st item on Linux, and last on Macos
  133. set(location_properties IMPORTED_LOCATION)
  134. if(CMAKE_BUILD_TYPE)
  135. list(APPEND location_properties IMPORTED_LOCATION_${CMAKE_BUILD_TYPE})
  136. endif()
  137. foreach (config_type ${CMAKE_CONFIGURATION_TYPES} RELEASE DEBUG RELWITHDEBINFO MINSIZEREL)
  138. list(APPEND location_properties IMPORTED_LOCATION_${config_type})
  139. endforeach()
  140. if(APPLE)
  141. set(valid_shared_library_regex "\\.[0-9]+\\.dylib$")
  142. else()
  143. set(valid_shared_library_regex "\\.so\\.([0-9.]+)?[0-9]")
  144. endif()
  145. foreach(location_property ${location_properties})
  146. if(NOT result)
  147. get_target_property(library_path "${TARGET}" ${location_property})
  148. message(DEBUG "get_target_property(${TARGET} ${location_propert}) -> ${library_path}")
  149. if(EXISTS "${library_path}")
  150. get_filename_component(library_path "${library_path}" ABSOLUTE)
  151. while (IS_SYMLINK "${library_path}")
  152. read_absolute_symlink(library_path "${library_path}")
  153. endwhile()
  154. message(DEBUG "${TARGET} -> ${library_path}")
  155. get_filename_component(libdir "${library_path}" DIRECTORY)
  156. file(GLOB subfiles "${libdir}/*")
  157. set(similar_files "${library_path}")
  158. foreach(subfile ${subfiles})
  159. if(IS_SYMLINK "${subfile}")
  160. read_absolute_symlink(subfile_target "${subfile}")
  161. while(IS_SYMLINK "${subfile_target}")
  162. read_absolute_symlink(subfile_target "${subfile_target}")
  163. endwhile()
  164. get_filename_component(subfile_target "${subfile_target}" ABSOLUTE)
  165. if(subfile_target STREQUAL library_path AND subfile MATCHES "${valid_shared_library_regex}")
  166. list(APPEND similar_files "${subfile}")
  167. endif()
  168. endif()
  169. endforeach()
  170. list(SORT similar_files)
  171. message(DEBUG "files that are similar to \"${library_path}\"=${similar_files}")
  172. if(APPLE)
  173. list(REVERSE similar_files)
  174. endif()
  175. list(GET similar_files 0 item)
  176. get_filename_component(result "${item}" NAME)
  177. endif()
  178. endif()
  179. endforeach()
  180. endif()
  181. if(result)
  182. string(TOLOWER "${result}" result_lower)
  183. if(WIN32 OR OS2)
  184. if(NOT result_lower MATCHES ".*dll")
  185. message(FATAL_ERROR "\"${result}\" is not a .dll library")
  186. endif()
  187. elseif(APPLE)
  188. if(NOT result_lower MATCHES ".*dylib.*")
  189. message(FATAL_ERROR "\"${result}\" is not a .dylib shared library")
  190. endif()
  191. else()
  192. if(NOT result_lower MATCHES ".*so.*")
  193. message(FATAL_ERROR "\"${result}\" is not a .so shared library")
  194. endif()
  195. endif()
  196. else()
  197. get_target_property(target_type ${TARGET} TYPE)
  198. if(target_type MATCHES "SHARED_LIBRARY|MODULE_LIBRARY")
  199. # OK
  200. elseif(target_type MATCHES "STATIC_LIBRARY|OBJECT_LIBRARY|INTERFACE_LIBRARY|EXECUTABLE")
  201. message(SEND_ERROR "${TARGET} is not a shared library, but has type=${target_type}")
  202. else()
  203. message(WARNING "Unable to extract dynamic library from target=${TARGET}, type=${target_type}.")
  204. endif()
  205. # TARGET_SONAME_FILE is not allowed for DLL target platforms.
  206. if(WIN32)
  207. set(result "$<TARGET_FILE_NAME:${TARGET}>")
  208. else()
  209. set(result "$<TARGET_SONAME_FILE_NAME:${TARGET}>")
  210. endif()
  211. endif()
  212. set(${DEST} ${result} PARENT_SCOPE)
  213. endfunction()
  214. function(sdl_check_project_in_subfolder relative_subfolder name vendored_option)
  215. cmake_parse_arguments(ARG "" "FILE" "" ${ARGN})
  216. if(NOT ARG_FILE)
  217. set(ARG_FILE "CMakeLists.txt")
  218. endif()
  219. if(NOT EXISTS "${PROJECT_SOURCE_DIR}/${relative_subfolder}/${ARG_FILE}")
  220. message(FATAL_ERROR "Could not find ${ARG_FILE} for ${name} in ${relative_subfolder}.\n"
  221. "Run the download script in the external folder, or re-configure with -D${vendored_option}=OFF to use system packages.")
  222. endif()
  223. endfunction()
  224. macro(sdl_check_linker_flag flag var)
  225. # FIXME: Use CheckLinkerFlag module once cmake minimum version >= 3.18
  226. cmake_push_check_state(RESET)
  227. set(CMAKE_REQUIRED_LINK_OPTIONS "${flag}")
  228. check_c_source_compiles("int main() { return 0; }" ${var} FAIL_REGEX "(unsupported|syntax error|unrecognized option)")
  229. cmake_pop_check_state()
  230. endmacro()
  231. function(SDL_detect_linker)
  232. if(CMAKE_VERSION VERSION_LESS 3.29)
  233. if(NOT DEFINED SDL_CMAKE_C_COMPILER_LINKER_ID)
  234. execute_process(COMMAND ${CMAKE_LINKER} -v OUTPUT_VARIABLE LINKER_OUTPUT ERROR_VARIABLE LINKER_OUTPUT)
  235. string(REGEX REPLACE "[\r\n]" " " LINKER_OUTPUT "${LINKER_OUTPUT}")
  236. if(LINKER_OUTPUT MATCHES ".*Microsoft.*")
  237. set(linker MSVC)
  238. else()
  239. set(linker GNUlike)
  240. endif()
  241. message(STATUS "Linker identification: ${linker}")
  242. set(SDL_CMAKE_C_COMPILER_LINKER_ID "${linker}" CACHE STRING "Linker identification")
  243. mark_as_advanced(SDL_CMAKE_C_COMPILER_LINKER_ID)
  244. endif()
  245. set(CMAKE_C_COMPILER_LINKER_ID "${SDL_CMAKE_C_COMPILER_LINKER_ID}" PARENT_SCOPE)
  246. endif()
  247. endfunction()
  248. function(check_linker_support_version_script VAR)
  249. SDL_detect_linker()
  250. if(CMAKE_C_COMPILER_LINKER_ID MATCHES "^(MSVC)$")
  251. set(LINKER_SUPPORTS_VERSION_SCRIPT FALSE)
  252. else()
  253. cmake_push_check_state(RESET)
  254. file(WRITE "${CMAKE_CURRENT_BINARY_DIR}/dummy.sym" "n_0 {\n global:\n func;\n local: *;\n};\n")
  255. list(APPEND CMAKE_REQUIRED_LINK_OPTIONS "-Wl,--version-script=${CMAKE_CURRENT_BINARY_DIR}/dummy.sym")
  256. 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)")
  257. cmake_pop_check_state()
  258. endif()
  259. set(${VAR} "${LINKER_SUPPORTS_VERSION_SCRIPT}" PARENT_SCOPE)
  260. endfunction()
  261. function(sdl_target_link_options_no_undefined TARGET)
  262. if(NOT MSVC AND NOT CMAKE_SYSTEM_NAME MATCHES ".*OpenBSD.*")
  263. if(CMAKE_C_COMPILER_ID MATCHES "AppleClang")
  264. target_link_options(${TARGET} PRIVATE "-Wl,-undefined,error")
  265. else()
  266. sdl_check_linker_flag("-Wl,--no-undefined" HAVE_WL_NO_UNDEFINED)
  267. if(HAVE_WL_NO_UNDEFINED AND NOT ((CMAKE_C_COMPILER_ID MATCHES "Clang") AND WIN32))
  268. target_link_options(${TARGET} PRIVATE "-Wl,--no-undefined")
  269. endif()
  270. endif()
  271. endif()
  272. endfunction()
  273. function(sdl_target_link_option_version_file TARGET VERSION_SCRIPT)
  274. check_linker_support_version_script(HAVE_WL_VERSION_SCRIPT)
  275. if(HAVE_WL_VERSION_SCRIPT)
  276. target_link_options(${TARGET} PRIVATE "-Wl,--version-script=${VERSION_SCRIPT}")
  277. set_property(TARGET ${TARGET} APPEND PROPERTY LINK_DEPENDS "${VERSION_SCRIPT}")
  278. else()
  279. if(LINUX OR ANDROID)
  280. message(FATAL_ERROR "Linker does not support '-Wl,--version-script=xxx.sym'. This is required on the current host platform.")
  281. endif()
  282. endif()
  283. endfunction()
  284. function(sdl_add_warning_options TARGET)
  285. cmake_parse_arguments(ARGS "" "WARNING_AS_ERROR" "" ${ARGN})
  286. if(MSVC)
  287. target_compile_options(${TARGET} PRIVATE /W2)
  288. else()
  289. target_compile_options(${TARGET} PRIVATE -Wall -Wextra -Wno-unused-parameter)
  290. endif()
  291. if(ARGS_WARNING_AS_ERROR)
  292. if(MSVC)
  293. target_compile_options(${TARGET} PRIVATE /WX)
  294. else()
  295. target_compile_options(${TARGET} PRIVATE -Werror)
  296. endif()
  297. endif()
  298. endfunction()
  299. function(sdl_no_deprecated_errors TARGET)
  300. check_c_compiler_flag(-Wno-error=deprecated-declarations HAVE_WNO_ERROR_DEPRECATED_DECLARATIONS)
  301. if(HAVE_WNO_ERROR_DEPRECATED_DECLARATIONS)
  302. target_compile_options(${TARGET} PRIVATE "-Wno-error=deprecated-declarations")
  303. endif()
  304. endfunction()
  305. function(sdl_get_git_revision_hash VARNAME)
  306. set("${VARNAME}" "" CACHE STRING "${PROJECT_NAME} revision")
  307. set(revision "${${VARNAME}}")
  308. if(NOT revision)
  309. if(EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/VERSION.txt")
  310. # If VERSION.txt exists, it contains the SDL version
  311. file(READ "${CMAKE_CURRENT_SOURCE_DIR}/VERSION.txt" revision_version)
  312. string(STRIP "${revision_version}" revision_version)
  313. else()
  314. # If VERSION.txt does not exist, use git to calculate a version
  315. git_describe(revision_version)
  316. if(NOT revision_version)
  317. set(revision_version "${PROJECT_VERSION}-no-vcs")
  318. endif()
  319. endif()
  320. set(revision "${revision_version}")
  321. endif()
  322. set("${VARNAME}" "${revision}" PARENT_SCOPE)
  323. endfunction()
  324. function(SDL_install_pdb TARGET DIRECTORY)
  325. get_property(type TARGET ${TARGET} PROPERTY TYPE)
  326. if(type MATCHES "^(SHARED_LIBRARY|EXECUTABLE)$")
  327. install(FILES $<TARGET_PDB_FILE:${TARGET}> DESTINATION "${DIRECTORY}" OPTIONAL)
  328. elseif(type STREQUAL "STATIC_LIBRARY")
  329. # FIXME: Use $<TARGET_COMPILE_PDB_FILE:${TARGET} once it becomes available (https://gitlab.kitware.com/cmake/cmake/-/issues/25244)
  330. if(CMAKE_GENERATOR MATCHES "^Visual Studio.*")
  331. install(CODE "file(INSTALL DESTINATION \"\${CMAKE_INSTALL_PREFIX}/${DIRECTORY}\" TYPE FILE OPTIONAL FILES \"${CMAKE_CURRENT_BINARY_DIR}/\${CMAKE_INSTALL_CONFIG_NAME}/${TARGET}.pdb\")")
  332. else()
  333. install(CODE "file(INSTALL DESTINATION \"\${CMAKE_INSTALL_PREFIX}/${DIRECTORY}\" TYPE FILE OPTIONAL FILES \"${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/${TARGET}.dir/${TARGET}.pdb\")")
  334. endif()
  335. endif()
  336. endfunction()