CMakeLists.txt 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967
  1. project(minetest)
  2. INCLUDE(CheckTypeSize)
  3. INCLUDE(CheckIncludeFiles)
  4. INCLUDE(CheckLibraryExists)
  5. check_type_size(int SIZEOF_INT BUILTIN_TYPES_ONLY LANGUAGE CXX)
  6. if(SIZEOF_INT LESS 4)
  7. message(FATAL_ERROR "Minetest will not work with int less than 32 bits wide.")
  8. endif()
  9. check_type_size(size_t SIZEOF_SIZE_T LANGUAGE CXX)
  10. if(SIZEOF_SIZE_T LESS 4)
  11. message(FATAL_ERROR "Minetest will not work with size_t less than 32 bits wide.")
  12. endif()
  13. # Add custom SemiDebug build mode
  14. set(CMAKE_CXX_FLAGS_SEMIDEBUG "-O1 -g -Wall" CACHE STRING
  15. "Flags used by the C++ compiler during semidebug builds."
  16. FORCE
  17. )
  18. set(CMAKE_C_FLAGS_SEMIDEBUG "-O1 -g -Wall -pedantic" CACHE STRING
  19. "Flags used by the C compiler during semidebug builds."
  20. FORCE
  21. )
  22. mark_as_advanced(
  23. CMAKE_CXX_FLAGS_SEMIDEBUG
  24. CMAKE_C_FLAGS_SEMIDEBUG
  25. )
  26. set(CMAKE_BUILD_TYPE "${CMAKE_BUILD_TYPE}" CACHE STRING
  27. "Choose the type of build. Options are: None Debug SemiDebug RelWithDebInfo MinSizeRel."
  28. FORCE
  29. )
  30. # Set some random things default to not being visible in the GUI
  31. mark_as_advanced(EXECUTABLE_OUTPUT_PATH LIBRARY_OUTPUT_PATH)
  32. if(NOT (BUILD_CLIENT OR BUILD_SERVER))
  33. message(WARNING "Neither BUILD_CLIENT nor BUILD_SERVER is set! Setting BUILD_SERVER=true")
  34. set(BUILD_SERVER TRUE)
  35. endif()
  36. option(ENABLE_CURL "Enable cURL support for fetching media" TRUE)
  37. set(USE_CURL FALSE)
  38. if(ENABLE_CURL)
  39. find_package(CURL 7.28.0)
  40. if (CURL_FOUND)
  41. message(STATUS "cURL support enabled.")
  42. set(USE_CURL TRUE)
  43. endif()
  44. else()
  45. mark_as_advanced(CLEAR CURL_LIBRARY CURL_INCLUDE_DIR)
  46. endif()
  47. if(NOT USE_CURL)
  48. if(BUILD_CLIENT)
  49. message(WARNING "cURL is required to load the server list")
  50. endif()
  51. if(BUILD_SERVER)
  52. message(WARNING "cURL is required to announce to the server list")
  53. endif()
  54. endif()
  55. option(ENABLE_GETTEXT "Use GetText for internationalization" ${BUILD_CLIENT})
  56. set(USE_GETTEXT FALSE)
  57. if(ENABLE_GETTEXT)
  58. find_package(GettextLib)
  59. if(GETTEXTLIB_FOUND)
  60. if(WIN32)
  61. message(STATUS "GetText library: ${GETTEXT_LIBRARY}")
  62. message(STATUS "GetText DLL(s): ${GETTEXT_DLL}")
  63. endif()
  64. set(USE_GETTEXT TRUE)
  65. message(STATUS "GetText enabled; locales found: ${GETTEXT_AVAILABLE_LOCALES}")
  66. endif(GETTEXTLIB_FOUND)
  67. else()
  68. mark_as_advanced(GETTEXT_INCLUDE_DIR GETTEXT_LIBRARY GETTEXT_MSGFMT)
  69. message(STATUS "GetText disabled.")
  70. endif()
  71. option(ENABLE_SOUND "Enable sound" TRUE)
  72. set(USE_SOUND FALSE)
  73. if(BUILD_CLIENT AND ENABLE_SOUND)
  74. # Sound libraries
  75. find_package(OpenAL)
  76. find_package(Vorbis)
  77. if(NOT OPENAL_FOUND)
  78. message(STATUS "Sound enabled, but OpenAL not found!")
  79. mark_as_advanced(CLEAR OPENAL_LIBRARY OPENAL_INCLUDE_DIR)
  80. endif()
  81. if(NOT VORBIS_FOUND)
  82. message(STATUS "Sound enabled, but Vorbis libraries not found!")
  83. mark_as_advanced(CLEAR OGG_INCLUDE_DIR VORBIS_INCLUDE_DIR OGG_LIBRARY VORBIS_LIBRARY VORBISFILE_LIBRARY)
  84. endif()
  85. if(OPENAL_FOUND AND VORBIS_FOUND)
  86. set(USE_SOUND TRUE)
  87. message(STATUS "Sound enabled.")
  88. else()
  89. message(FATAL_ERROR "Sound enabled, but cannot be used.\n"
  90. "To continue, either fill in the required paths or disable sound. (-DENABLE_SOUND=0)")
  91. endif()
  92. endif()
  93. option(ENABLE_TOUCH "Enable touchscreen by default" FALSE)
  94. if(ENABLE_TOUCH)
  95. message(STATUS "Touchscreen support enabled by default.")
  96. add_definitions(-DENABLE_TOUCH)
  97. endif()
  98. if(BUILD_CLIENT)
  99. find_package(Freetype REQUIRED)
  100. endif()
  101. option(ENABLE_CURSES "Enable ncurses console" TRUE)
  102. set(USE_CURSES FALSE)
  103. if(ENABLE_CURSES)
  104. find_package(Ncursesw)
  105. if(CURSES_FOUND)
  106. set(USE_CURSES TRUE)
  107. message(STATUS "ncurses console enabled.")
  108. include_directories(${CURSES_INCLUDE_DIRS})
  109. else()
  110. message(STATUS "ncurses not found!")
  111. endif()
  112. endif(ENABLE_CURSES)
  113. option(ENABLE_POSTGRESQL "Enable PostgreSQL backend" TRUE)
  114. set(USE_POSTGRESQL FALSE)
  115. if(ENABLE_POSTGRESQL)
  116. if(CMAKE_VERSION VERSION_LESS "3.20")
  117. find_package(PostgreSQL QUIET)
  118. # Before CMake 3.20 FindPostgreSQL.cmake always looked for server includes
  119. # but we don't need them, so continue anyway if only those are missing.
  120. if(PostgreSQL_INCLUDE_DIR AND PostgreSQL_LIBRARY)
  121. set(PostgreSQL_FOUND TRUE)
  122. set(PostgreSQL_INCLUDE_DIRS ${PostgreSQL_INCLUDE_DIR})
  123. set(PostgreSQL_LIBRARIES ${PostgreSQL_LIBRARY})
  124. endif()
  125. else()
  126. find_package(PostgreSQL)
  127. endif()
  128. if(PostgreSQL_FOUND)
  129. set(USE_POSTGRESQL TRUE)
  130. message(STATUS "PostgreSQL backend enabled")
  131. # This variable is case sensitive, don't try to change it to POSTGRESQL_INCLUDE_DIR
  132. message(STATUS "PostgreSQL includes: ${PostgreSQL_INCLUDE_DIRS}")
  133. include_directories(${PostgreSQL_INCLUDE_DIRS})
  134. else()
  135. message(STATUS "PostgreSQL not found!")
  136. endif()
  137. endif(ENABLE_POSTGRESQL)
  138. option(ENABLE_LEVELDB "Enable LevelDB backend" TRUE)
  139. set(USE_LEVELDB FALSE)
  140. if(ENABLE_LEVELDB)
  141. find_library(LEVELDB_LIBRARY NAMES leveldb libleveldb)
  142. find_path(LEVELDB_INCLUDE_DIR db.h PATH_SUFFIXES leveldb)
  143. if(LEVELDB_LIBRARY AND LEVELDB_INCLUDE_DIR)
  144. set(USE_LEVELDB TRUE)
  145. message(STATUS "LevelDB backend enabled.")
  146. include_directories(${LEVELDB_INCLUDE_DIR})
  147. else()
  148. message(STATUS "LevelDB not found!")
  149. endif()
  150. endif(ENABLE_LEVELDB)
  151. OPTION(ENABLE_REDIS "Enable Redis backend" TRUE)
  152. set(USE_REDIS FALSE)
  153. if(ENABLE_REDIS)
  154. find_library(REDIS_LIBRARY hiredis)
  155. find_path(REDIS_INCLUDE_DIR hiredis.h PATH_SUFFIXES hiredis)
  156. if(REDIS_LIBRARY AND REDIS_INCLUDE_DIR)
  157. set(USE_REDIS TRUE)
  158. message(STATUS "Redis backend enabled.")
  159. include_directories(${REDIS_INCLUDE_DIR})
  160. else(REDIS_LIBRARY AND REDIS_INCLUDE_DIR)
  161. message(STATUS "Redis not found!")
  162. endif(REDIS_LIBRARY AND REDIS_INCLUDE_DIR)
  163. endif(ENABLE_REDIS)
  164. find_package(SQLite3 REQUIRED)
  165. OPTION(ENABLE_PROMETHEUS "Enable prometheus client support" FALSE)
  166. set(USE_PROMETHEUS FALSE)
  167. if(ENABLE_PROMETHEUS)
  168. find_path(PROMETHEUS_CPP_INCLUDE_DIR NAMES prometheus/counter.h)
  169. find_library(PROMETHEUS_PULL_LIBRARY NAMES prometheus-cpp-pull)
  170. find_library(PROMETHEUS_CORE_LIBRARY NAMES prometheus-cpp-core)
  171. if(PROMETHEUS_CPP_INCLUDE_DIR AND PROMETHEUS_PULL_LIBRARY AND PROMETHEUS_CORE_LIBRARY)
  172. set(PROMETHEUS_LIBRARIES ${PROMETHEUS_PULL_LIBRARY} ${PROMETHEUS_CORE_LIBRARY})
  173. set(USE_PROMETHEUS TRUE)
  174. include_directories(${PROMETHEUS_CPP_INCLUDE_DIR})
  175. endif(PROMETHEUS_CPP_INCLUDE_DIR AND PROMETHEUS_PULL_LIBRARY AND PROMETHEUS_CORE_LIBRARY)
  176. endif(ENABLE_PROMETHEUS)
  177. if(USE_PROMETHEUS)
  178. message(STATUS "Prometheus client enabled.")
  179. else(USE_PROMETHEUS)
  180. message(STATUS "Prometheus client disabled.")
  181. endif(USE_PROMETHEUS)
  182. OPTION(ENABLE_SPATIAL "Enable SpatialIndex AreaStore backend" TRUE)
  183. set(USE_SPATIAL FALSE)
  184. if(ENABLE_SPATIAL)
  185. find_library(SPATIAL_LIBRARY spatialindex)
  186. find_path(SPATIAL_INCLUDE_DIR spatialindex/SpatialIndex.h)
  187. if(SPATIAL_LIBRARY AND SPATIAL_INCLUDE_DIR)
  188. set(USE_SPATIAL TRUE)
  189. message(STATUS "SpatialIndex AreaStore backend enabled.")
  190. include_directories(${SPATIAL_INCLUDE_DIR})
  191. else(SPATIAL_LIBRARY AND SPATIAL_INCLUDE_DIR)
  192. message(STATUS "SpatialIndex not found!")
  193. endif(SPATIAL_LIBRARY AND SPATIAL_INCLUDE_DIR)
  194. endif(ENABLE_SPATIAL)
  195. find_package(ZLIB REQUIRED)
  196. find_package(Zstd REQUIRED)
  197. if(NOT MSVC)
  198. set(USE_GPROF FALSE CACHE BOOL "Use -pg flag for g++")
  199. endif()
  200. # Haiku endian support
  201. if(HAIKU)
  202. set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -D_BSD_SOURCE")
  203. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -D_BSD_SOURCE")
  204. endif()
  205. # Use cmake_config.h
  206. add_definitions(-DUSE_CMAKE_CONFIG_H)
  207. set(THREADS_PREFER_PTHREAD_FLAG ON)
  208. find_package(Threads REQUIRED)
  209. set(PLATFORM_LIBS Threads::Threads)
  210. if(WIN32)
  211. # Windows
  212. if(MSVC) # MSVC Specifics
  213. set(PLATFORM_LIBS dbghelp.lib ${PLATFORM_LIBS})
  214. # Surpress some useless warnings
  215. add_definitions ( /D "_CRT_SECURE_NO_DEPRECATE" /W1 )
  216. # Get M_PI to work
  217. add_definitions(/D "_USE_MATH_DEFINES")
  218. # Don't define min/max macros in minwindef.h
  219. add_definitions(/D "NOMINMAX")
  220. endif()
  221. set(PLATFORM_LIBS ws2_32.lib version.lib shlwapi.lib winmm.lib ${PLATFORM_LIBS})
  222. set(EXTRA_DLL "" CACHE FILEPATH "Optional paths to additional DLLs that should be packaged")
  223. # DLLs are automatically copied to the output directory by vcpkg when VCPKG_APPLOCAL_DEPS=ON
  224. if(NOT VCPKG_APPLOCAL_DEPS)
  225. set(ZLIB_DLL "" CACHE FILEPATH "Path to Zlib DLL for installation (optional)")
  226. set(ZSTD_DLL "" CACHE FILEPATH "Path to Zstd DLL for installation (optional)")
  227. if(BUILD_CLIENT)
  228. set(PNG_DLL "" CACHE FILEPATH "Path to libpng DLL for installation (optional)")
  229. set(JPEG_DLL "" CACHE FILEPATH "Path to libjpeg DLL for installation (optional)")
  230. set(SDL2_DLL "" CACHE FILEPATH "Path to SDL2 DLL for installation (optional)")
  231. endif()
  232. if(BUILD_CLIENT AND ENABLE_SOUND)
  233. set(OPENAL_DLL "" CACHE FILEPATH "Path to OpenAL DLL for installation (optional)")
  234. set(OGG_DLL "" CACHE FILEPATH "Path to Ogg DLL for installation (optional)")
  235. set(VORBIS_DLL "" CACHE FILEPATH "Path to Vorbis DLLs for installation (optional)")
  236. endif()
  237. if(USE_GETTEXT)
  238. set(GETTEXT_DLL "" CACHE FILEPATH "Path to Intl/Iconv DLLs for installation (optional)")
  239. endif()
  240. if(USE_LUAJIT)
  241. set(LUA_DLL "" CACHE FILEPATH "Path to luajit-5.1.dll for installation (optional)")
  242. endif()
  243. endif()
  244. else()
  245. # Unix probably
  246. set(PLATFORM_LIBS ${PLATFORM_LIBS} ${CMAKE_DL_LIBS})
  247. if(APPLE)
  248. set(PLATFORM_LIBS "-framework CoreFoundation" ${PLATFORM_LIBS})
  249. else()
  250. check_library_exists(rt clock_gettime "" HAVE_LIBRT)
  251. if (HAVE_LIBRT)
  252. set(PLATFORM_LIBS -lrt ${PLATFORM_LIBS})
  253. endif(HAVE_LIBRT)
  254. endif(APPLE)
  255. find_library(ICONV_LIBRARY iconv)
  256. mark_as_advanced(ICONV_LIBRARY)
  257. if (ICONV_LIBRARY)
  258. set(PLATFORM_LIBS ${PLATFORM_LIBS} ${ICONV_LIBRARY})
  259. endif()
  260. if (HAIKU)
  261. set(PLATFORM_LIBS ${PLATFORM_LIBS} network)
  262. endif()
  263. if (ANDROID)
  264. set(PLATFORM_LIBS ${PLATFORM_LIBS} android log)
  265. endif()
  266. endif()
  267. # On clang and gcc, some functionalities of std::atomic require -latomic.
  268. # See <https://en.cppreference.com/w/cpp/atomic/atomic#Notes>.
  269. # Note that find_library does not reliably find it so we have to resort to this.
  270. # Also, passing -latomic is not always the same as adding atomic to the library list.
  271. include(CheckCSourceCompiles)
  272. set(CMAKE_REQUIRED_LIBRARIES "-latomic")
  273. check_c_source_compiles("int main(){}" HAVE_LINK_ATOMIC)
  274. set(CMAKE_REQUIRED_LIBRARIES "")
  275. if(HAVE_LINK_ATOMIC)
  276. set(PLATFORM_LIBS ${PLATFORM_LIBS} "-latomic")
  277. endif()
  278. include(CheckSymbolExists)
  279. check_symbol_exists(strlcpy "string.h" HAVE_STRLCPY)
  280. check_include_files(endian.h HAVE_ENDIAN_H)
  281. configure_file(
  282. "${PROJECT_SOURCE_DIR}/cmake_config.h.in"
  283. "${PROJECT_BINARY_DIR}/cmake_config.h"
  284. )
  285. # Add a target that always rebuilds cmake_config_githash.h
  286. add_custom_target(GenerateVersion
  287. COMMAND ${CMAKE_COMMAND}
  288. -D "GENERATE_VERSION_SOURCE_DIR=${CMAKE_CURRENT_SOURCE_DIR}"
  289. -D "GENERATE_VERSION_BINARY_DIR=${CMAKE_CURRENT_BINARY_DIR}"
  290. -D "VERSION_STRING=${VERSION_STRING}"
  291. -D "DEVELOPMENT_BUILD=${DEVELOPMENT_BUILD}"
  292. -P "${CMAKE_SOURCE_DIR}/cmake/Modules/GenerateVersion.cmake"
  293. WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}")
  294. add_subdirectory(threading)
  295. add_subdirectory(content)
  296. add_subdirectory(database)
  297. add_subdirectory(gui)
  298. add_subdirectory(mapgen)
  299. add_subdirectory(network)
  300. add_subdirectory(script)
  301. add_subdirectory(unittest)
  302. add_subdirectory(benchmark)
  303. add_subdirectory(util)
  304. add_subdirectory(irrlicht_changes)
  305. add_subdirectory(server)
  306. set(common_SRCS
  307. ${database_SRCS}
  308. ${mapgen_SRCS}
  309. ${server_SRCS}
  310. ${content_SRCS}
  311. chat.cpp
  312. clientdynamicinfo.cpp
  313. collision.cpp
  314. content_mapnode.cpp
  315. content_nodemeta.cpp
  316. convert_json.cpp
  317. craftdef.cpp
  318. debug.cpp
  319. defaultsettings.cpp
  320. emerge.cpp
  321. environment.cpp
  322. face_position_cache.cpp
  323. filesys.cpp
  324. gettext.cpp
  325. httpfetch.cpp
  326. hud.cpp
  327. inventory.cpp
  328. inventorymanager.cpp
  329. itemdef.cpp
  330. itemstackmetadata.cpp
  331. light.cpp
  332. lighting.cpp
  333. log.cpp
  334. main.cpp
  335. map.cpp
  336. map_settings_manager.cpp
  337. mapblock.cpp
  338. mapnode.cpp
  339. mapsector.cpp
  340. metadata.cpp
  341. modchannels.cpp
  342. nameidmapping.cpp
  343. nodedef.cpp
  344. nodemetadata.cpp
  345. nodetimer.cpp
  346. noise.cpp
  347. objdef.cpp
  348. object_properties.cpp
  349. particles.cpp
  350. pathfinder.cpp
  351. player.cpp
  352. porting.cpp
  353. profiler.cpp
  354. raycast.cpp
  355. reflowscan.cpp
  356. remoteplayer.cpp
  357. rollback_interface.cpp
  358. serialization.cpp
  359. server.cpp
  360. serverenvironment.cpp
  361. servermap.cpp
  362. settings.cpp
  363. staticobject.cpp
  364. terminal_chat_console.cpp
  365. texture_override.cpp
  366. tileanimation.cpp
  367. tool.cpp
  368. translation.cpp
  369. version.cpp
  370. voxel.cpp
  371. voxelalgorithms.cpp
  372. hud.cpp
  373. ${common_network_SRCS}
  374. ${JTHREAD_SRCS}
  375. ${common_SCRIPT_SRCS}
  376. ${UTIL_SRCS}
  377. )
  378. if(ANDROID)
  379. set(common_SRCS ${common_SRCS} porting_android.cpp)
  380. endif()
  381. if(BUILD_UNITTESTS)
  382. set(common_SRCS ${common_SRCS} ${UNITTEST_SRCS})
  383. endif()
  384. if(BUILD_BENCHMARKS)
  385. set(common_SRCS ${common_SRCS} ${BENCHMARK_SRCS})
  386. endif()
  387. # This gives us the icon and file version information
  388. if(WIN32)
  389. set(WINRESOURCE_FILE "${CMAKE_CURRENT_SOURCE_DIR}/../misc/winresource.rc")
  390. set(MINETEST_EXE_MANIFEST_FILE "${CMAKE_CURRENT_SOURCE_DIR}/../misc/minetest.exe.manifest")
  391. if(MINGW)
  392. if(NOT CMAKE_RC_COMPILER)
  393. set(CMAKE_RC_COMPILER "windres.exe")
  394. endif()
  395. ADD_CUSTOM_COMMAND(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/winresource_rc.o
  396. COMMAND ${CMAKE_RC_COMPILER} -I${CMAKE_CURRENT_SOURCE_DIR} -I${CMAKE_CURRENT_BINARY_DIR}
  397. -i${WINRESOURCE_FILE}
  398. -o ${CMAKE_CURRENT_BINARY_DIR}/winresource_rc.o
  399. WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
  400. DEPENDS ${WINRESOURCE_FILE} ${MINETEST_EXE_MANIFEST_FILE})
  401. SET(extra_windows_SRCS ${CMAKE_CURRENT_BINARY_DIR}/winresource_rc.o)
  402. else(MINGW) # Probably MSVC
  403. set(extra_windows_SRCS ${WINRESOURCE_FILE} ${MINETEST_EXE_MANIFEST_FILE})
  404. endif(MINGW)
  405. endif()
  406. # Client sources
  407. if (BUILD_CLIENT)
  408. add_subdirectory(client)
  409. endif(BUILD_CLIENT)
  410. set(client_SRCS
  411. ${client_SRCS}
  412. ${common_SRCS}
  413. ${gui_SRCS}
  414. ${client_network_SRCS}
  415. ${client_irrlicht_changes_SRCS}
  416. ${client_SCRIPT_SRCS}
  417. )
  418. if(BUILD_UNITTESTS)
  419. set(client_SRCS ${client_SRCS} ${UNITTEST_CLIENT_SRCS})
  420. endif()
  421. if(BUILD_BENCHMARKS)
  422. set(client_SRCS ${client_SRCS} ${BENCHMARK_CLIENT_SRCS})
  423. endif()
  424. list(SORT client_SRCS)
  425. # Server sources
  426. set(server_SRCS
  427. ${common_SRCS}
  428. )
  429. list(SORT server_SRCS)
  430. # Avoid source_group on broken CMake version.
  431. # see issue #7074 #7075
  432. if (CMAKE_VERSION VERSION_GREATER 3.8.1)
  433. source_group(TREE ${PROJECT_SOURCE_DIR} PREFIX "Source Files" FILES ${client_SRCS})
  434. source_group(TREE ${PROJECT_SOURCE_DIR} PREFIX "Source Files" FILES ${server_SRCS})
  435. endif()
  436. include_directories(
  437. ${PROJECT_BINARY_DIR}
  438. ${PROJECT_SOURCE_DIR}
  439. ${PROJECT_SOURCE_DIR}/script
  440. )
  441. include_directories(SYSTEM
  442. ${ZLIB_INCLUDE_DIR}
  443. ${ZSTD_INCLUDE_DIR}
  444. ${SQLITE3_INCLUDE_DIR}
  445. ${LUA_INCLUDE_DIR}
  446. ${GMP_INCLUDE_DIR}
  447. ${JSON_INCLUDE_DIR}
  448. ${LUA_BIT_INCLUDE_DIR}
  449. # on Android, Minetest depends on SDL2 directly
  450. # on other platforms, only IrrlichtMt depends on SDL2
  451. "$<$<PLATFORM_ID:Android>:${SDL2_INCLUDE_DIRS}>"
  452. )
  453. if(USE_GETTEXT)
  454. include_directories(${GETTEXT_INCLUDE_DIR})
  455. endif()
  456. if(BUILD_CLIENT)
  457. include_directories(SYSTEM
  458. ${FREETYPE_INCLUDE_DIRS}
  459. ${SOUND_INCLUDE_DIRS}
  460. )
  461. endif()
  462. if(USE_CURL)
  463. include_directories(${CURL_INCLUDE_DIR})
  464. endif()
  465. # When cross-compiling assume the user doesn't want to run the executable anyway,
  466. # otherwise place it in <source dir>/bin/ since Minetest can only run from there.
  467. if(NOT CMAKE_CROSSCOMPILING)
  468. set(EXECUTABLE_OUTPUT_PATH "${CMAKE_SOURCE_DIR}/bin")
  469. endif()
  470. if(BUILD_CLIENT)
  471. if(ANDROID)
  472. add_library(${PROJECT_NAME} SHARED ${client_SRCS})
  473. else()
  474. add_executable(${PROJECT_NAME} ${client_SRCS} ${extra_windows_SRCS})
  475. endif()
  476. add_dependencies(${PROJECT_NAME} GenerateVersion)
  477. target_link_libraries(
  478. ${PROJECT_NAME}
  479. ${ZLIB_LIBRARIES}
  480. IrrlichtMt::IrrlichtMt
  481. ${ZSTD_LIBRARY}
  482. ${SOUND_LIBRARIES}
  483. ${SQLITE3_LIBRARY}
  484. ${LUA_LIBRARY}
  485. ${GMP_LIBRARY}
  486. ${JSON_LIBRARY}
  487. ${LUA_BIT_LIBRARY}
  488. ${FREETYPE_LIBRARY}
  489. ${PLATFORM_LIBS}
  490. # on Android, Minetest depends on SDL2 directly
  491. # on other platforms, only IrrlichtMt depends on SDL2
  492. "$<$<PLATFORM_ID:Android>:${SDL2_LIBRARIES}>"
  493. )
  494. if(NOT USE_LUAJIT)
  495. set_target_properties(${PROJECT_NAME} PROPERTIES
  496. # This is necessary for dynamic Lua modules
  497. # to work when Lua is statically linked (issue #10806)
  498. ENABLE_EXPORTS 1
  499. )
  500. endif()
  501. if(USE_GETTEXT)
  502. target_link_libraries(
  503. ${PROJECT_NAME}
  504. ${GETTEXT_LIBRARY}
  505. )
  506. endif()
  507. if(USE_CURL)
  508. target_link_libraries(
  509. ${PROJECT_NAME}
  510. ${CURL_LIBRARY}
  511. )
  512. endif()
  513. if(FREETYPE_PKGCONFIG_FOUND)
  514. set_target_properties(${PROJECT_NAME}
  515. PROPERTIES
  516. COMPILE_FLAGS "${FREETYPE_CFLAGS_STR}"
  517. )
  518. endif()
  519. if (USE_CURSES)
  520. target_link_libraries(${PROJECT_NAME} ${CURSES_LIBRARIES})
  521. endif()
  522. if (USE_POSTGRESQL)
  523. target_link_libraries(${PROJECT_NAME} ${PostgreSQL_LIBRARIES})
  524. endif()
  525. if (USE_LEVELDB)
  526. target_link_libraries(${PROJECT_NAME} ${LEVELDB_LIBRARY})
  527. endif()
  528. if (USE_REDIS)
  529. target_link_libraries(${PROJECT_NAME} ${REDIS_LIBRARY})
  530. endif()
  531. if (USE_PROMETHEUS)
  532. target_link_libraries(${PROJECT_NAME} ${PROMETHEUS_LIBRARIES})
  533. endif()
  534. if (USE_SPATIAL)
  535. target_link_libraries(${PROJECT_NAME} ${SPATIAL_LIBRARY})
  536. endif()
  537. if(BUILD_BENCHMARKS)
  538. target_link_libraries(${PROJECT_NAME} catch2)
  539. endif()
  540. endif(BUILD_CLIENT)
  541. if(BUILD_SERVER)
  542. add_executable(${PROJECT_NAME}server ${server_SRCS} ${extra_windows_SRCS})
  543. add_dependencies(${PROJECT_NAME}server GenerateVersion)
  544. get_target_property(
  545. IRRLICHT_INCLUDES IrrlichtMt::IrrlichtMt INTERFACE_INCLUDE_DIRECTORIES)
  546. # Doesn't work without PRIVATE/PUBLIC/INTERFACE mode specified.
  547. target_include_directories(${PROJECT_NAME}server PRIVATE ${IRRLICHT_INCLUDES})
  548. target_link_libraries(
  549. ${PROJECT_NAME}server
  550. ${ZLIB_LIBRARIES}
  551. ${ZSTD_LIBRARY}
  552. ${SQLITE3_LIBRARY}
  553. ${JSON_LIBRARY}
  554. ${LUA_LIBRARY}
  555. ${LUA_BIT_LIBRARY}
  556. ${GMP_LIBRARY}
  557. ${PLATFORM_LIBS}
  558. )
  559. set_target_properties(${PROJECT_NAME}server PROPERTIES
  560. COMPILE_DEFINITIONS "SERVER")
  561. if(NOT USE_LUAJIT)
  562. set_target_properties(${PROJECT_NAME}server PROPERTIES
  563. # This is necessary for dynamic Lua modules
  564. # to work when Lua is statically linked (issue #10806)
  565. ENABLE_EXPORTS 1
  566. )
  567. endif()
  568. if (USE_GETTEXT)
  569. target_link_libraries(${PROJECT_NAME}server ${GETTEXT_LIBRARY})
  570. endif()
  571. if (USE_CURSES)
  572. target_link_libraries(${PROJECT_NAME}server ${CURSES_LIBRARIES})
  573. endif()
  574. if (USE_POSTGRESQL)
  575. target_link_libraries(${PROJECT_NAME}server ${PostgreSQL_LIBRARIES})
  576. endif()
  577. if (USE_LEVELDB)
  578. target_link_libraries(${PROJECT_NAME}server ${LEVELDB_LIBRARY})
  579. endif()
  580. if (USE_REDIS)
  581. target_link_libraries(${PROJECT_NAME}server ${REDIS_LIBRARY})
  582. endif()
  583. if (USE_PROMETHEUS)
  584. target_link_libraries(${PROJECT_NAME}server ${PROMETHEUS_LIBRARIES})
  585. endif()
  586. if (USE_SPATIAL)
  587. target_link_libraries(${PROJECT_NAME}server ${SPATIAL_LIBRARY})
  588. endif()
  589. if(USE_CURL)
  590. target_link_libraries(
  591. ${PROJECT_NAME}server
  592. ${CURL_LIBRARY}
  593. )
  594. endif()
  595. if(BUILD_BENCHMARKS)
  596. target_link_libraries(${PROJECT_NAME}server catch2)
  597. endif()
  598. endif(BUILD_SERVER)
  599. # See issue #4638
  600. FILE(READ "${CMAKE_SOURCE_DIR}/src/unsupported_language_list.txt" GETTEXT_BLACKLISTED_LOCALES)
  601. STRING(REGEX REPLACE "\n" ";" GETTEXT_BLACKLISTED_LOCALES "${GETTEXT_BLACKLISTED_LOCALES}")
  602. option(APPLY_LOCALE_BLACKLIST "Use a blacklist to avoid known broken locales" TRUE)
  603. if (GETTEXTLIB_FOUND AND APPLY_LOCALE_BLACKLIST)
  604. set(GETTEXT_USED_LOCALES "")
  605. foreach(LOCALE ${GETTEXT_AVAILABLE_LOCALES})
  606. if (NOT "${LOCALE}" IN_LIST GETTEXT_BLACKLISTED_LOCALES)
  607. list(APPEND GETTEXT_USED_LOCALES ${LOCALE})
  608. endif()
  609. endforeach()
  610. message(STATUS "Locale blacklist applied; Locales used: ${GETTEXT_USED_LOCALES}")
  611. elseif (GETTEXTLIB_FOUND)
  612. set(GETTEXT_USED_LOCALES ${GETTEXT_AVAILABLE_LOCALES})
  613. endif()
  614. # Set some optimizations and tweaks
  615. include(CheckCSourceCompiles)
  616. set(CMAKE_REQUIRED_INCLUDES ${LUA_INCLUDE_DIR})
  617. if(USE_LUAJIT)
  618. # libm usually required if statically linking
  619. set(CMAKE_REQUIRED_LIBRARIES ${LUA_LIBRARY} m)
  620. # LuaJIT provides exactly zero ways to determine how recent it is (the version
  621. # is unchanged since 2017), however it happens that string buffers were added
  622. # after the changes which we care about so that works as an indicator.
  623. # (https://github.com/LuaJIT/LuaJIT/commit/4c6b669 March 2021)
  624. # Note: This is no longer true as of August 2023, but we're keeping the old check.
  625. unset(HAVE_RECENT_LJ CACHE)
  626. check_symbol_exists(luaopen_string_buffer "lualib.h" HAVE_RECENT_LJ)
  627. if(NOT HAVE_RECENT_LJ)
  628. string(CONCAT explanation_msg
  629. "You are using a relatively old version of LuaJIT. We recommend "
  630. "running a recent version (from git) as older ones are known not "
  631. "to build/work correctly in all cases.\n"
  632. "THIS APPLIES ESPECIALLY ON macOS OR Linux/aarch64!")
  633. message(WARNING ${explanation_msg})
  634. endif()
  635. elseif(NOT MSVC)
  636. set(CMAKE_REQUIRED_LIBRARIES "")
  637. unset(HAVE_ATCCALL CACHE)
  638. # Note: we need to check the function without having the library
  639. # available for linking, so check_symbol_exists won't work.
  640. # Incidentally this doesn't seem to work on MSVC...
  641. check_c_source_compiles("#include <lua.h>\nint main(){return sizeof(lua_atccall);}" HAVE_ATCCALL)
  642. if(NOT HAVE_ATCCALL)
  643. string(CONCAT explanation_msg
  644. "It looks like you're trying to build Minetest using a system-wide "
  645. "Lua installation. This is no longer supported because PUC Lua "
  646. "cannot interoperate with C++ correctly. Read src/unittest/test_lua.cpp "
  647. " for technical details.")
  648. message(FATAL_ERROR ${explanation_msg})
  649. endif()
  650. endif()
  651. if(MSVC)
  652. # Visual Studio
  653. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /D _WIN32_WINNT=0x0601 /D WIN32_LEAN_AND_MEAN")
  654. # EHa enables SEH exceptions (used for catching segfaults)
  655. set(CMAKE_CXX_FLAGS_RELEASE "/EHa /Ox /MD /GS- /Zi /fp:fast /D NDEBUG /D _HAS_ITERATOR_DEBUGGING=0")
  656. if(CMAKE_SIZEOF_VOID_P EQUAL 4)
  657. set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /arch:SSE")
  658. endif()
  659. set(CMAKE_EXE_LINKER_FLAGS_RELEASE "/INCREMENTAL:NO /DEBUG /OPT:REF /OPT:ICF /SUBSYSTEM:WINDOWS /ENTRY:mainCRTStartup")
  660. set(CMAKE_CXX_FLAGS_SEMIDEBUG "/MDd /Zi /Ob0 /O1 /RTC1")
  661. # Debug build doesn't catch exceptions by itself
  662. # Add some optimizations because otherwise it's VERY slow
  663. set(CMAKE_CXX_FLAGS_DEBUG "/MDd /Zi /Ob0 /Od /RTC1")
  664. # Flags for C files (sqlite)
  665. # /MD = dynamically link to MSVCRxxx.dll
  666. set(CMAKE_C_FLAGS_RELEASE "/O2 /Ob2 /MD")
  667. # Flags that cannot be shared between cl and clang-cl
  668. # https://clang.llvm.org/docs/UsersManual.html#clang-cl
  669. if(CMAKE_CXX_COMPILER_ID STREQUAL "Clang")
  670. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fuse-ld=lld")
  671. # Disable pragma-pack warning
  672. set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -Wno-pragma-pack")
  673. else()
  674. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /MP")
  675. set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /TP /FD /GL")
  676. set(CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG")
  677. endif()
  678. else()
  679. # GCC or compatible compilers such as Clang
  680. set(WARNING_FLAGS "-Wall -Wextra")
  681. set(WARNING_FLAGS "${WARNING_FLAGS} -Wno-unused-parameter")
  682. if(WARN_ALL)
  683. set(RELEASE_WARNING_FLAGS "${WARNING_FLAGS}")
  684. else()
  685. set(RELEASE_WARNING_FLAGS "")
  686. endif()
  687. if(MINGW)
  688. if(CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
  689. set(OTHER_FLAGS "${OTHER_FLAGS} -mthreads")
  690. endif()
  691. set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -D_WIN32_WINNT=0x0601 -DWIN32_LEAN_AND_MEAN")
  692. endif()
  693. # Use a safe subset of flags to speed up math calculations:
  694. # - we don't need errno or math exceptions
  695. # - we don't deal with signed zero
  696. set(MATH_FLAGS "-fno-math-errno -fno-trapping-math -fno-signed-zeros")
  697. # Enable SSE for floating point math on 32-bit x86 by default
  698. # reasoning see minetest issue #11810 and https://gcc.gnu.org/wiki/FloatingPointMath
  699. if(CMAKE_SIZEOF_VOID_P EQUAL 4)
  700. check_c_source_compiles("#ifndef __i686__\n#error\n#endif\nint main(){}" IS_I686)
  701. if(IS_I686)
  702. message(STATUS "Detected Intel x86: using SSE instead of x87 FPU")
  703. set(OTHER_FLAGS "${OTHER_FLAGS} -mfpmath=sse -msse")
  704. endif()
  705. endif()
  706. set(CMAKE_CXX_FLAGS_RELEASE "-DNDEBUG ${RELEASE_WARNING_FLAGS} ${OTHER_FLAGS} -pipe -funroll-loops -O3 -fomit-frame-pointer")
  707. if(CMAKE_SYSTEM_NAME STREQUAL "Linux"
  708. AND CMAKE_CXX_COMPILER_ID STREQUAL "Clang"
  709. AND CMAKE_CXX_COMPILER_VERSION MATCHES "^9\\.")
  710. # Clang 9 has broken -ffast-math on glibc
  711. else()
  712. set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} ${MATH_FLAGS}")
  713. endif()
  714. set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELEASE} -g")
  715. set(CMAKE_CXX_FLAGS_SEMIDEBUG "-g -O1 ${WARNING_FLAGS} ${OTHER_FLAGS}")
  716. set(CMAKE_CXX_FLAGS_DEBUG "-g -O0 ${WARNING_FLAGS} ${OTHER_FLAGS}")
  717. if(UNIX)
  718. # enable assertions for libstdc++ or libc++
  719. set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -Wp,-D_GLIBCXX_ASSERTIONS -Wp,-D_LIBCPP_ENABLE_ASSERTIONS=1")
  720. endif()
  721. if(USE_GPROF)
  722. set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -pg")
  723. endif()
  724. if(MINGW)
  725. set(CMAKE_EXE_LINKER_FLAGS_RELEASE "-mwindows")
  726. endif()
  727. endif()
  728. # Installation
  729. if(WIN32)
  730. if(EXTRA_DLL)
  731. install(FILES ${EXTRA_DLL} DESTINATION ${BINDIR})
  732. endif()
  733. if(VCPKG_APPLOCAL_DEPS)
  734. # Collect the dll's from the output path
  735. install(DIRECTORY ${EXECUTABLE_OUTPUT_PATH}/Release/
  736. DESTINATION ${BINDIR}
  737. CONFIGURATIONS Release
  738. FILES_MATCHING PATTERN "*.dll")
  739. install(DIRECTORY ${EXECUTABLE_OUTPUT_PATH}/Debug/
  740. DESTINATION ${BINDIR}
  741. CONFIGURATIONS Debug
  742. FILES_MATCHING PATTERN "*.dll")
  743. install(DIRECTORY ${EXECUTABLE_OUTPUT_PATH}/RelWithDebInfo/
  744. DESTINATION ${BINDIR}
  745. CONFIGURATIONS RelWithDebInfo
  746. FILES_MATCHING PATTERN "*.dll")
  747. install(DIRECTORY ${EXECUTABLE_OUTPUT_PATH}/MinSizeRel/
  748. DESTINATION ${BINDIR}
  749. CONFIGURATIONS MinSizeRel
  750. FILES_MATCHING PATTERN "*.dll")
  751. else()
  752. # Use the old-style way to install dll's
  753. if(BUILD_CLIENT AND USE_SOUND)
  754. if(OPENAL_DLL)
  755. install(FILES ${OPENAL_DLL} DESTINATION ${BINDIR})
  756. endif()
  757. if(OGG_DLL)
  758. install(FILES ${OGG_DLL} DESTINATION ${BINDIR})
  759. endif()
  760. if(VORBIS_DLL)
  761. install(FILES ${VORBIS_DLL} DESTINATION ${BINDIR})
  762. endif()
  763. endif()
  764. if(BUILD_CLIENT)
  765. if(PNG_DLL)
  766. install(FILES ${PNG_DLL} DESTINATION ${BINDIR})
  767. endif()
  768. if(JPEG_DLL)
  769. install(FILES ${JPEG_DLL} DESTINATION ${BINDIR})
  770. endif()
  771. if(SDL2_DLL)
  772. install(FILES ${SDL2_DLL} DESTINATION ${BINDIR})
  773. endif()
  774. if(FREETYPE_DLL)
  775. install(FILES ${FREETYPE_DLL} DESTINATION ${BINDIR})
  776. endif()
  777. endif()
  778. if(CURL_DLL)
  779. install(FILES ${CURL_DLL} DESTINATION ${BINDIR})
  780. endif()
  781. if(ZLIB_DLL)
  782. install(FILES ${ZLIB_DLL} DESTINATION ${BINDIR})
  783. endif()
  784. if(ZSTD_DLL)
  785. install(FILES ${ZSTD_DLL} DESTINATION ${BINDIR})
  786. endif()
  787. if(SQLITE3_DLL)
  788. install(FILES ${SQLITE3_DLL} DESTINATION ${BINDIR})
  789. endif()
  790. if(LEVELDB_DLL)
  791. install(FILES ${LEVELDB_DLL} DESTINATION ${BINDIR})
  792. endif()
  793. if(LUA_DLL)
  794. install(FILES ${LUA_DLL} DESTINATION ${BINDIR})
  795. endif()
  796. if(BUILD_CLIENT AND USE_GETTEXT AND GETTEXT_DLL)
  797. install(FILES ${GETTEXT_DLL} DESTINATION ${BINDIR})
  798. endif()
  799. endif()
  800. endif()
  801. if(BUILD_CLIENT AND NOT ANDROID)
  802. install(TARGETS ${PROJECT_NAME}
  803. RUNTIME DESTINATION ${BINDIR}
  804. LIBRARY DESTINATION ${BINDIR}
  805. ARCHIVE DESTINATION ${BINDIR}
  806. BUNDLE DESTINATION .
  807. )
  808. if(APPLE)
  809. install(CODE "
  810. set(BU_CHMOD_BUNDLE_ITEMS ON)
  811. include(BundleUtilities)
  812. fixup_bundle(\"\${CMAKE_INSTALL_PREFIX}/${BUNDLE_PATH}\" \"\" \"\${CMAKE_INSTALL_PREFIX}/${BINDIR}\")
  813. " COMPONENT Runtime)
  814. endif()
  815. if(USE_GETTEXT)
  816. foreach(LOCALE ${GETTEXT_USED_LOCALES})
  817. set_mo_paths(MO_BUILD_PATH MO_DEST_PATH ${LOCALE})
  818. set(MO_BUILD_PATH "${MO_BUILD_PATH}/${PROJECT_NAME}.mo")
  819. install(FILES ${MO_BUILD_PATH} DESTINATION ${MO_DEST_PATH})
  820. endforeach()
  821. endif()
  822. install(DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/../fonts" DESTINATION "${SHAREDIR}"
  823. FILES_MATCHING PATTERN "*.ttf" PATTERN "*.txt")
  824. endif()
  825. if(BUILD_SERVER)
  826. install(TARGETS ${PROJECT_NAME}server DESTINATION ${BINDIR})
  827. endif()
  828. if (ANDROID)
  829. # Android does this manually in app/build.gradle -> prepareAssets
  830. # for now!
  831. elseif (USE_GETTEXT)
  832. set(MO_FILES)
  833. foreach(LOCALE ${GETTEXT_USED_LOCALES})
  834. set(PO_FILE_PATH "${GETTEXT_PO_PATH}/${LOCALE}/${PROJECT_NAME}.po")
  835. set_mo_paths(MO_BUILD_PATH MO_DEST_PATH ${LOCALE})
  836. set(MO_FILE_PATH "${MO_BUILD_PATH}/${PROJECT_NAME}.mo")
  837. add_custom_command(OUTPUT ${MO_BUILD_PATH}
  838. COMMAND ${CMAKE_COMMAND} -E make_directory ${MO_BUILD_PATH}
  839. COMMENT "mo-update [${LOCALE}]: Creating locale directory.")
  840. add_custom_command(
  841. OUTPUT ${MO_FILE_PATH}
  842. COMMAND ${GETTEXT_MSGFMT} -o ${MO_FILE_PATH} ${PO_FILE_PATH}
  843. DEPENDS ${MO_BUILD_PATH} ${PO_FILE_PATH}
  844. WORKING_DIRECTORY "${GETTEXT_PO_PATH}/${LOCALE}"
  845. COMMENT "mo-update [${LOCALE}]: Creating mo file."
  846. )
  847. set(MO_FILES ${MO_FILES} ${MO_FILE_PATH})
  848. endforeach()
  849. add_custom_target(translations ALL COMMENT "mo update" DEPENDS ${MO_FILES})
  850. endif()