You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

CMakeLists.txt 25KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737
  1. #
  2. # Rspamd - rapid antispam system
  3. #
  4. # Cmake configuration file
  5. #
  6. ############################# INITIAL SECTION #############################################
  7. CMAKE_MINIMUM_REQUIRED(VERSION 3.9 FATAL_ERROR)
  8. SET(RSPAMD_VERSION_MAJOR 2)
  9. SET(RSPAMD_VERSION_MINOR 3)
  10. # Keep two digits all the time
  11. SET(RSPAMD_VERSION_MAJOR_NUM ${RSPAMD_VERSION_MAJOR}0)
  12. SET(RSPAMD_VERSION_MINOR_NUM ${RSPAMD_VERSION_MINOR}0)
  13. IF(GIT_ID)
  14. SET(GIT_VERSION 1)
  15. SET(RSPAMD_ID "${GIT_ID}")
  16. ENDIF()
  17. SET(RSPAMD_VERSION "${RSPAMD_VERSION_MAJOR}.${RSPAMD_VERSION_MINOR}")
  18. PROJECT(rspamd VERSION "${RSPAMD_VERSION}" LANGUAGES C CXX ASM)
  19. # This is supported merely with cmake 3.1
  20. SET(CMAKE_C_STANDARD 11)
  21. SET(CMAKE_C_STANDARD_REQUIRED ON)
  22. LIST(APPEND CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake/")
  23. SET(RSPAMD_MASTER_SITE_URL "https://rspamd.com")
  24. IF(NOT RSPAMD_USER)
  25. SET(RSPAMD_USER "nobody")
  26. SET(RSPAMD_GROUP "nobody")
  27. ENDIF(NOT RSPAMD_USER)
  28. # Default for SysV Init
  29. SET(RSPAMD_WORKER_NORMAL "*:11333")
  30. SET(RSPAMD_WORKER_CONTROLLER "*:11334")
  31. SET_PROPERTY(GLOBAL PROPERTY ALLOW_DUPLICATE_CUSTOM_TARGETS 1)
  32. ############################# OPTIONS SECTION #############################################
  33. OPTION(ENABLE_LUAJIT "Link with libluajit [default: ON]" ON)
  34. OPTION(ENABLE_URL_INCLUDE "Enable urls in ucl includes (requires libcurl or libfetch) [default: OFF]" OFF)
  35. OPTION(NO_SHARED "Build internal libs static [default: ON]" ON)
  36. OPTION(INSTALL_WEBUI "Install web interface [default: ON]" ON)
  37. OPTION(WANT_SYSTEMD_UNITS "Install systemd unit files on Linux [default: OFF]" OFF)
  38. OPTION(ENABLE_SNOWBALL "Enable snowball stemmer [default: ON]" ON)
  39. OPTION(ENABLE_CLANG_PLUGIN "Enable clang static analysing plugin [default: OFF]" OFF)
  40. OPTION(ENABLE_PCRE2 "Enable pcre2 instead of pcre [default: OFF]" OFF)
  41. OPTION(ENABLE_JEMALLOC "Build rspamd with jemalloc allocator [default: OFF]" OFF)
  42. OPTION(ENABLE_UTILS "Build rspamd internal utils [default: OFF]" OFF)
  43. OPTION(ENABLE_LIBUNWIND "Use libunwind to print crash traces [default: OFF]" OFF)
  44. OPTION(ENABLE_LUA_TRACE "Trace all Lua C API invocations [default: OFF]" OFF)
  45. OPTION(ENABLE_LUA_REPL "Enables Lua repl (requires C++11 compiler) [default: ON]" ON)
  46. ############################# INCLUDE SECTION #############################################
  47. INCLUDE(CheckIncludeFiles)
  48. INCLUDE(CheckFunctionExists)
  49. INCLUDE(CheckSymbolExists)
  50. INCLUDE(CheckCSourceCompiles)
  51. INCLUDE(CheckCSourceRuns)
  52. INCLUDE(CheckLibraryExists)
  53. INCLUDE(CheckCCompilerFlag)
  54. INCLUDE(CMakeParseArguments)
  55. INCLUDE(FindArch)
  56. INCLUDE(AsmOp)
  57. INCLUDE(FindRagel)
  58. INCLUDE(FindLua)
  59. INCLUDE(ProcessPackage)
  60. IF(NOT RAGEL_FOUND)
  61. MESSAGE(FATAL_ERROR "Ragel is required to build rspamd")
  62. ENDIF()
  63. FIND_PACKAGE(PkgConfig REQUIRED)
  64. FIND_PACKAGE(Perl REQUIRED)
  65. INCLUDE(Toolset)
  66. INCLUDE(Sanitizer)
  67. INCLUDE(ArchDep)
  68. INCLUDE(Paths)
  69. IF(ENABLE_PCRE2 MATCHES "ON")
  70. SET(WITH_PCRE2 1)
  71. # For utf8 API
  72. LIST(APPEND CMAKE_REQUIRED_DEFINITIONS "-DPCRE2_CODE_UNIT_WIDTH=8")
  73. ENDIF()
  74. ############################# CONFIG SECTION #############################################
  75. # Initial set
  76. # Prefer local include dirs to system ones
  77. INCLUDE_DIRECTORIES("${CMAKE_SOURCE_DIR}/"
  78. "${CMAKE_SOURCE_DIR}/src"
  79. "${CMAKE_SOURCE_DIR}/src/libutil"
  80. "${CMAKE_SOURCE_DIR}/src/libserver"
  81. "${CMAKE_SOURCE_DIR}/src/libmime"
  82. "${CMAKE_SOURCE_DIR}/src/libstat"
  83. "${CMAKE_SOURCE_DIR}/src/libcryptobox"
  84. "${CMAKE_SOURCE_DIR}/contrib/libucl"
  85. "${CMAKE_SOURCE_DIR}/contrib/replxx/include"
  86. "${CMAKE_SOURCE_DIR}/contrib/uthash"
  87. "${CMAKE_SOURCE_DIR}/contrib/http-parser"
  88. "${CMAKE_SOURCE_DIR}/contrib/fpconv"
  89. "${CMAKE_SOURCE_DIR}/contrib/libottery"
  90. "${CMAKE_SOURCE_DIR}/contrib/xxhash"
  91. "${CMAKE_SOURCE_DIR}/contrib/cdb"
  92. "${CMAKE_SOURCE_DIR}/contrib/snowball/include"
  93. "${CMAKE_SOURCE_DIR}/contrib/librdns"
  94. "${CMAKE_SOURCE_DIR}/contrib/aho-corasick"
  95. "${CMAKE_SOURCE_DIR}/contrib/lc-btrie"
  96. "${CMAKE_SOURCE_DIR}/contrib/lua-lpeg"
  97. "${CMAKE_BINARY_DIR}/src" #Stored in the binary dir
  98. "${CMAKE_BINARY_DIR}/src/libcryptobox")
  99. SET(TAR "tar")
  100. INCLUDE(OSDep)
  101. # Now find libraries and headers
  102. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES "m")
  103. IF(ENABLE_LUAJIT MATCHES "ON")
  104. #ProcessPackage(LUAJIT luajit)
  105. SET(WITH_LUA 1)
  106. FindLua(VERSION_MAJOR "5" VERSION_MINOR "1" ROOT "${LUA_ROOT}")
  107. IF(NOT LUA_FOUND)
  108. MESSAGE(FATAL_ERROR "Lua not found, lua support is required")
  109. ELSE(NOT LUA_FOUND)
  110. SET(WITH_LUA 1)
  111. INCLUDE_DIRECTORIES("${LUA_INCLUDE_DIR}")
  112. ENDIF(NOT LUA_FOUND)
  113. ELSE(ENABLE_LUAJIT MATCHES "ON")
  114. FindLua(VERSION_MAJOR "5" VERSION_MINOR "3" ROOT "${LUA_ROOT}")
  115. IF(NOT LUA_FOUND)
  116. FindLua(VERSION_MAJOR "5" VERSION_MINOR "2" ROOT "${LUA_ROOT}")
  117. IF(NOT LUA_FOUND)
  118. FindLua(VERSION_MAJOR "5" VERSION_MINOR "1" ROOT "${LUA_ROOT}")
  119. ENDIF(NOT LUA_FOUND)
  120. ENDIF(NOT LUA_FOUND)
  121. IF(NOT LUA_FOUND)
  122. MESSAGE(FATAL_ERROR "Lua not found, lua support is required")
  123. ELSE(NOT LUA_FOUND)
  124. SET(WITH_LUA 1)
  125. INCLUDE_DIRECTORIES("${LUA_INCLUDE_DIR}")
  126. ENDIF(NOT LUA_FOUND)
  127. ENDIF(ENABLE_LUAJIT MATCHES "ON")
  128. IF(ENABLE_JEMALLOC MATCHES "ON" AND NOT SANITIZE)
  129. ProcessPackage(JEMALLOC LIBRARY jemalloc_pic jemalloc INCLUDE jemalloc.h INCLUDE_SUFFIXES include/jemalloc
  130. ROOT ${JEMALLOC_ROOT_DIR})
  131. SET(WITH_JEMALLOC "1")
  132. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES "-lpthread")
  133. ENDIF()
  134. IF(ENABLE_LIBUNWIND MATCHES "ON")
  135. ProcessPackage(LIBUNWIND LIBRARY unwind INCLUDE libunwind.h INCLUDE_SUFFIXES include/libunwind
  136. ROOT ${LIBUNWIND_ROOT_DIR} MODULES libunwind)
  137. SET(WITH_LIBUNWIND "1")
  138. ENDIF()
  139. ProcessPackage(GLIB2 LIBRARY glib-2.0 INCLUDE glib.h
  140. INCLUDE_SUFFIXES include/glib include/glib-2.0
  141. ROOT ${GLIB_ROOT_DIR} MODULES glib-2.0>=2.28)
  142. IF(ENABLE_PCRE2 MATCHES "ON")
  143. ProcessPackage(PCRE LIBRARY pcre2 pcre2-8 INCLUDE pcre2.h INCLUDE_SUFFIXES include/pcre2
  144. ROOT ${PCRE_ROOT_DIR} MODULES pcre2 pcre2-8 libpcre2 libpcre2-8)
  145. ELSE()
  146. ProcessPackage(PCRE LIBRARY pcre INCLUDE pcre.h INCLUDE_SUFFIXES include/pcre
  147. ROOT ${PCRE_ROOT_DIR} MODULES pcre libpcre pcre3 libpcre3)
  148. ENDIF()
  149. ProcessPackage(SQLITE3 LIBRARY sqlite3 INCLUDE sqlite3.h INCLUDE_SUFFIXES include/sqlite3 include/sqlite
  150. ROOT ${SQLITE3_ROOT_DIR} MODULES sqlite3 sqlite)
  151. ProcessPackage(ICUDATA LIBRARY icudata INCLUDE unicode/ucnv.h
  152. ROOT ${ICU_ROOT_DIR} MODULES icu-i18n)
  153. ProcessPackage(ICUC LIBRARY icuuc INCLUDE unicode/ucnv.h
  154. ROOT ${ICU_ROOT_DIR} MODULES icu-i18n)
  155. ProcessPackage(ICUI18N LIBRARY icui18n INCLUDE unicode/ucnv.h
  156. ROOT ${ICU_ROOT_DIR} MODULES icu-i18n)
  157. ProcessPackage(LIBCRYPT LIBRARY crypto INCLUDE openssl/evp.h
  158. ROOT ${OPENSSL_ROOT_DIR} MODULES openssl libcrypt)
  159. ProcessPackage(LIBSSL LIBRARY ssl INCLUDE openssl/ssl.h
  160. ROOT ${OPENSSL_ROOT_DIR} MODULES openssl libssl)
  161. ProcessPackage(LIBZ LIBRARY z INCLUDE zlib.h INCLUDE_SUFFIXES include/zlib
  162. ROOT ${LIBZ_ROOT_DIR} MODULES z)
  163. ProcessPackage(SODIUM LIBRARY sodium INCLUDE sodium.h
  164. INCLUDE_SUFFIXES include/libsodium include/sodium
  165. ROOT ${LIBSODIUM_ROOT_DIR} MODULES libsodium>=1.0.0)
  166. include (CompilerWarnings)
  167. include (Hyperscan)
  168. include (Openblas)
  169. IF(ENABLE_LUA_TRACE)
  170. SET(WITH_LUA_TRACE 1)
  171. ENDIF(ENABLE_LUA_TRACE)
  172. SET(CMAKE_C_FLAGS "${CMAKE_C_OPT_FLAGS} ${CMAKE_C_FLAGS}")
  173. SET(CMAKE_CXX_FLAGS "${CMAKE_C_OPT_FLAGS} ${CMAKE_CXX_FLAGS}")
  174. ADD_DEFINITIONS(-DHAVE_CONFIG_H)
  175. # Check platform specific includes
  176. CHECK_INCLUDE_FILES(sys/types.h HAVE_SYS_TYPES_H)
  177. CHECK_INCLUDE_FILES(sys/time.h HAVE_SYS_TIME_H)
  178. CHECK_INCLUDE_FILES(sys/uio.h HAVE_SYS_UIO_H)
  179. CHECK_INCLUDE_FILES(fcntl.h HAVE_FCNTL_H)
  180. CHECK_INCLUDE_FILES(math.h HAVE_MATH_H)
  181. CHECK_INCLUDE_FILES(stdio.h HAVE_STDIO_H)
  182. CHECK_INCLUDE_FILES(stdlib.h HAVE_STDLIB_H)
  183. CHECK_INCLUDE_FILES(string.h HAVE_STRING_H)
  184. CHECK_INCLUDE_FILES(strings.h HAVE_STRINGS_H)
  185. CHECK_INCLUDE_FILES(time.h HAVE_TIME_H)
  186. CHECK_INCLUDE_FILES(unistd.h HAVE_UNISTD_H)
  187. CHECK_INCLUDE_FILES(sysexits.h HAVE_SYSEXITS_H)
  188. CHECK_INCLUDE_FILES(stdint.h HAVE_STDINT_H)
  189. CHECK_INCLUDE_FILES(inttypes.h HAVE_INTTYPES_H)
  190. CHECK_INCLUDE_FILES(stdbool.h HAVE_STDBOOL_H)
  191. CHECK_INCLUDE_FILES(endian.h HAVE_ENDIAN_H)
  192. CHECK_INCLUDE_FILES(sys/endian.h HAVE_SYS_ENDIAN_H)
  193. CHECK_INCLUDE_FILES(machine/endian.h HAVE_MACHINE_ENDIAN_H)
  194. CHECK_INCLUDE_FILES(sys/queue.h HAVE_SYS_QUEUE_H)
  195. CHECK_INCLUDE_FILES(sys/socket.h HAVE_SYS_SOCKET_H)
  196. CHECK_INCLUDE_FILES(sys/mman.h HAVE_SYS_MMAN_H)
  197. CHECK_INCLUDE_FILES(sys/un.h HAVE_SYS_UN_H)
  198. CHECK_INCLUDE_FILES(sys/stat.h HAVE_SYS_STAT_H)
  199. CHECK_INCLUDE_FILES(sys/wait.h HAVE_SYS_WAIT_H)
  200. CHECK_INCLUDE_FILES(sys/param.h HAVE_SYS_PARAM_H)
  201. CHECK_INCLUDE_FILES(sys/cdefs.h HAVE_SYS_CDEFS_H)
  202. CHECK_INCLUDE_FILES(sys/file.h HAVE_SYS_FILE_H)
  203. CHECK_INCLUDE_FILES(sys/utsname.h HAVE_SYS_UTSNAME_H)
  204. CHECK_INCLUDE_FILES(sys/resource.h HAVE_SYS_RESOURCE_H)
  205. CHECK_INCLUDE_FILES(netinet/in.h HAVE_NETINET_IN_H)
  206. CHECK_INCLUDE_FILES(netinet/tcp.h HAVE_NETINET_TCP_H)
  207. CHECK_INCLUDE_FILES(arpa/inet.h HAVE_ARPA_INET_H)
  208. CHECK_INCLUDE_FILES(netdb.h HAVE_NETDB_H)
  209. CHECK_INCLUDE_FILES(syslog.h HAVE_SYSLOG_H)
  210. CHECK_INCLUDE_FILES(siginfo.h HAVE_SIGINFO_H)
  211. CHECK_INCLUDE_FILES(locale.h HAVE_LOCALE_H)
  212. CHECK_INCLUDE_FILES(libgen.h HAVE_LIBGEN_H)
  213. CHECK_INCLUDE_FILES(search.h HAVE_SEARCH_H)
  214. CHECK_INCLUDE_FILES(pwd.h HAVE_PWD_H)
  215. CHECK_INCLUDE_FILES(grp.h HAVE_GRP_H)
  216. CHECK_INCLUDE_FILES(glob.h HAVE_GLOB_H)
  217. CHECK_INCLUDE_FILES(poll.h HAVE_POLL_H)
  218. CHECK_INCLUDE_FILES(readpassphrase.h HAVE_READPASSPHRASE_H)
  219. CHECK_INCLUDE_FILES(termios.h HAVE_TERMIOS_H)
  220. CHECK_INCLUDE_FILES(paths.h HAVE_PATHS_H)
  221. CHECK_INCLUDE_FILES(ctype.h HAVE_CTYPE_H)
  222. CHECK_INCLUDE_FILES(unistd.h HAVE_UNISTD_H)
  223. CHECK_INCLUDE_FILES(cpuid.h HAVE_CPUID_H)
  224. CHECK_INCLUDE_FILES(dirent.h HAVE_DIRENT_H)
  225. CHECK_INCLUDE_FILES(stropts.h HAVE_STROPS_H)
  226. CHECK_INCLUDE_FILES(sys/ioctl.h HAVE_SYS_IOCTL_H)
  227. CHECK_INCLUDE_FILES(ucontext.h HAVE_UCONTEXT_H)
  228. CHECK_INCLUDE_FILES(sys/ucontext.h HAVE_SYS_UCONTEXT_H) # OSX specific
  229. # Check platform API
  230. CHECK_FUNCTION_EXISTS(setproctitle HAVE_SETPROCTITLE)
  231. CHECK_FUNCTION_EXISTS(getpagesize HAVE_GETPAGESIZE)
  232. CHECK_FUNCTION_EXISTS(nanosleep HAVE_NANOSLEEP)
  233. CHECK_FUNCTION_EXISTS(vfork HAVE_VFORK)
  234. CHECK_FUNCTION_EXISTS(wait4 HAVE_WAIT4)
  235. CHECK_FUNCTION_EXISTS(waitpid HAVE_WAITPID)
  236. CHECK_FUNCTION_EXISTS(flock HAVE_FLOCK)
  237. CHECK_LIBRARY_EXISTS(m tanhl "" HAVE_TANHL)
  238. CHECK_LIBRARY_EXISTS(m tanh "" HAVE_TANH)
  239. CHECK_LIBRARY_EXISTS(m expl "" HAVE_EXPL)
  240. CHECK_LIBRARY_EXISTS(m exp2l "" HAVE_EXP2L)
  241. CHECK_FUNCTION_EXISTS(sendfile HAVE_SENDFILE)
  242. CHECK_FUNCTION_EXISTS(mkstemp HAVE_MKSTEMP)
  243. CHECK_FUNCTION_EXISTS(setitimer HAVE_SETITIMER)
  244. CHECK_FUNCTION_EXISTS(inet_pton HAVE_INET_PTON)
  245. CHECK_FUNCTION_EXISTS(clock_gettime HAVE_CLOCK_GETTIME)
  246. CHECK_C_SOURCE_COMPILES(
  247. "#include <time.h>
  248. extern char *tzname[2];
  249. extern long timezone;
  250. extern int daylight;
  251. int main(int argc, char** argv) {
  252. tzset();
  253. return timezone;
  254. }" HAVE_SANE_TZSET)
  255. IF(WITH_ICONV)
  256. CHECK_C_SOURCE_COMPILES("
  257. #include <iconv.h>
  258. int main(){
  259. iconv_t conv = 0;
  260. const char* in = 0;
  261. size_t ilen = 0;
  262. char* out = 0;
  263. size_t olen = 0;
  264. iconv(conv, &in, &ilen, &out, &olen);
  265. return 0;
  266. }
  267. " ICONV_SECOND_ARGUMENT_IS_CONST)
  268. ENDIF(WITH_ICONV)
  269. # Check macros
  270. CHECK_SYMBOL_EXISTS(PATH_MAX limits.h HAVE_PATH_MAX)
  271. CHECK_SYMBOL_EXISTS(MAXPATHLEN sys/param.h HAVE_MAXPATHLEN)
  272. CHECK_SYMBOL_EXISTS(MAP_SHARED sys/mman.h HAVE_MMAP_SHARED)
  273. CHECK_SYMBOL_EXISTS(MAP_ANON sys/mman.h HAVE_MMAP_ANON)
  274. CHECK_SYMBOL_EXISTS(MAP_NOCORE sys/mman.h HAVE_MMAP_NOCORE)
  275. CHECK_SYMBOL_EXISTS(O_DIRECT fcntl.h HAVE_O_DIRECT)
  276. CHECK_SYMBOL_EXISTS(IPV6_V6ONLY "sys/socket.h;netinet/in.h" HAVE_IPV6_V6ONLY)
  277. CHECK_SYMBOL_EXISTS(posix_fadvise fcntl.h HAVE_FADVISE)
  278. CHECK_SYMBOL_EXISTS(posix_fallocate fcntl.h HAVE_POSIX_FALLOCATE)
  279. CHECK_SYMBOL_EXISTS(fallocate fcntl.h HAVE_FALLOCATE)
  280. CHECK_SYMBOL_EXISTS(fdatasync unistd.h HAVE_FDATASYNC)
  281. CHECK_SYMBOL_EXISTS(_SC_NPROCESSORS_ONLN unistd.h HAVE_SC_NPROCESSORS_ONLN)
  282. CHECK_SYMBOL_EXISTS(setbit sys/param.h PARAM_H_HAS_BITSET)
  283. CHECK_SYMBOL_EXISTS(getaddrinfo "sys/types.h;sys/socket.h;netdb.h" HAVE_GETADDRINFO)
  284. CHECK_SYMBOL_EXISTS(sched_yield "sched.h" HAVE_SCHED_YIELD)
  285. CHECK_SYMBOL_EXISTS(__get_cpuid "cpuid.h" HAVE_GET_CPUID)
  286. CHECK_SYMBOL_EXISTS(nftw "sys/types.h;ftw.h" HAVE_NFTW)
  287. IF(ENABLE_PCRE2 MATCHES "ON")
  288. LIST(APPEND CMAKE_REQUIRED_INCLUDES "${PCRE_INCLUDE}")
  289. CHECK_SYMBOL_EXISTS(PCRE2_CONFIG_JIT "pcre2.h" HAVE_PCRE_JIT)
  290. ELSE()
  291. LIST(APPEND CMAKE_REQUIRED_INCLUDES "${PCRE_INCLUDE}")
  292. CHECK_SYMBOL_EXISTS(PCRE_CONFIG_JIT "pcre.h" HAVE_PCRE_JIT)
  293. ENDIF()
  294. CHECK_SYMBOL_EXISTS(SOCK_SEQPACKET "sys/types.h;sys/socket.h" HAVE_SOCK_SEQPACKET)
  295. CHECK_SYMBOL_EXISTS(I_SETSIG "sys/types.h;sys/ioctl.h" HAVE_SETSIG)
  296. CHECK_SYMBOL_EXISTS(O_ASYNC "sys/types.h;sys/fcntl.h" HAVE_OASYNC)
  297. CHECK_SYMBOL_EXISTS(O_NOFOLLOW "sys/types.h;sys/fcntl.h" HAVE_ONOFOLLOW)
  298. CHECK_SYMBOL_EXISTS(O_CLOEXEC "sys/types.h;sys/fcntl.h" HAVE_OCLOEXEC)
  299. LIST(APPEND CMAKE_REQUIRED_INCLUDES "${LIBSSL_INCLUDE}")
  300. CHECK_SYMBOL_EXISTS(SSL_set_tlsext_host_name "openssl/ssl.h" HAVE_SSL_TLSEXT_HOSTNAME)
  301. CHECK_SYMBOL_EXISTS(dirfd "sys/types.h;unistd.h;dirent.h" HAVE_DIRFD)
  302. CHECK_SYMBOL_EXISTS(fpathconf "sys/types.h;unistd.h" HAVE_FPATHCONF)
  303. CHECK_SYMBOL_EXISTS(sigaltstack "signal.h" HAVE_SIGALTSTACK)
  304. CHECK_SYMBOL_EXISTS(open_memstream "stdio.h" HAVE_OPENMEMSTREAM)
  305. CHECK_SYMBOL_EXISTS(fmemopen "stdio.h" HAVE_FMEMOPEN)
  306. CHECK_SYMBOL_EXISTS(clock_getcpuclockid "sys/types.h;time.h" HAVE_CLOCK_GETCPUCLOCKID)
  307. CHECK_SYMBOL_EXISTS(RUSAGE_SELF "sys/types.h;sys/resource.h" HAVE_RUSAGE_SELF)
  308. IF(ENABLE_PCRE2 MATCHES "ON")
  309. IF(HAVE_PCRE_JIT)
  310. SET(HAVE_PCRE_JIT_FAST 1)
  311. ENDIF()
  312. ELSE()
  313. LIST(APPEND CMAKE_REQUIRED_INCLUDES "${PCRE_INCLUDE}")
  314. IF(PCRE_LIBRARY_PATH)
  315. SET(CMAKE_REQUIRED_LIBRARIES "${CMAKE_REQUIRED_LIBRARIES};-L${PCRE_LIBRARY_PATH};${PCRE_LIBRARY}")
  316. ELSE(PCRE_LIBRARY_PATH)
  317. SET(CMAKE_REQUIRED_LIBRARIES "${CMAKE_REQUIRED_LIBRARIES};-lpcre")
  318. ENDIF(PCRE_LIBRARY_PATH)
  319. # Some PCRE implementations are lacking of pcre_jit_exec fast path
  320. SET(_PCRE_FAST_TEST "
  321. #include \"pcre.h\"
  322. int main (void)
  323. {
  324. int rc;
  325. int ovector[30];
  326. pcre *re;
  327. pcre_extra *extra;
  328. pcre_jit_stack *jit_stack;
  329. re = pcre_compile(\"abc\", 0, NULL, NULL, NULL);
  330. extra = pcre_study(re, PCRE_STUDY_JIT_COMPILE, NULL);
  331. jit_stack = pcre_jit_stack_alloc(32*1024, 512*1024);
  332. pcre_assign_jit_stack(extra, NULL, jit_stack);
  333. rc = pcre_jit_exec(re, extra, \"abc\", 3, 0, 0, ovector, 30, jit_stack);
  334. return rc;
  335. }
  336. ")
  337. CHECK_C_SOURCE_COMPILES("${_PCRE_FAST_TEST}" HAVE_PCRE_JIT_FAST)
  338. IF(HAVE_PCRE_JIT_FAST)
  339. MESSAGE(STATUS "pcre_jit_exec is supported")
  340. ELSE(HAVE_PCRE_JIT_FAST)
  341. MESSAGE(STATUS "pcre_jit_exec is -NOT- supported")
  342. ENDIF(HAVE_PCRE_JIT_FAST)
  343. ENDIF()
  344. FILE(WRITE ${CMAKE_BINARY_DIR}/pthread_setpshared.c "
  345. #include <pthread.h>
  346. #include <stdlib.h>
  347. int main (void)
  348. {
  349. pthread_mutexattr_t mattr;
  350. if (pthread_mutexattr_init(&mattr) != 0) return 0;
  351. if (pthread_mutexattr_setpshared(&mattr, PTHREAD_PROCESS_SHARED) != 0) return 0;
  352. if (pthread_mutexattr_setrobust(&mattr, PTHREAD_MUTEX_ROBUST) != 0) return 0;
  353. return 1;
  354. }
  355. ")
  356. TRY_RUN(_CAN_RUN _CAN_COMPILE
  357. "${CMAKE_BINARY_DIR}" "${CMAKE_BINARY_DIR}/pthread_setpshared.c"
  358. CMAKE_FLAGS CMAKE_C_FLAGS="-pthread")
  359. IF(_CAN_RUN EQUAL 1)
  360. SET(HAVE_PTHREAD_PROCESS_SHARED 1 CACHE INTERNAL "")
  361. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES "-lpthread")
  362. ENDIF(_CAN_RUN EQUAL 1)
  363. IF(HAVE_PTHREAD_PROCESS_SHARED)
  364. MESSAGE(STATUS "pthread_mutexattr_setpshared is supported")
  365. ELSE(HAVE_PTHREAD_PROCESS_SHARED)
  366. MESSAGE(STATUS "pthread_mutexattr_setpshared is -NOT- supported")
  367. ENDIF(HAVE_PTHREAD_PROCESS_SHARED)
  368. IF(NOT HAVE_GETADDRINFO)
  369. MESSAGE(FATAL_ERROR "Your system does not support getaddrinfo call, please consider upgrading it to run rspamd")
  370. ENDIF(NOT HAVE_GETADDRINFO)
  371. IF(HAVE_SIGINFO_H)
  372. CHECK_SYMBOL_EXISTS(SA_SIGINFO "signal.h;siginfo.h" HAVE_SA_SIGINFO)
  373. ELSE(HAVE_SIGINFO_H)
  374. CHECK_SYMBOL_EXISTS(SA_SIGINFO "signal.h" HAVE_SA_SIGINFO)
  375. ENDIF(HAVE_SIGINFO_H)
  376. IF(NOT CMAKE_SYSTEM_NAME STREQUAL "SunOS")
  377. IF(HAVE_CLOCK_GETTIME)
  378. CHECK_SYMBOL_EXISTS(CLOCK_PROCESS_CPUTIME_ID time.h HAVE_CLOCK_PROCESS_CPUTIME_ID)
  379. CHECK_SYMBOL_EXISTS(CLOCK_VIRTUAL time.h HAVE_CLOCK_VIRTUAL)
  380. ELSE(HAVE_CLOCK_GETTIME)
  381. CHECK_INCLUDE_FILES(sys/timeb.h HAVE_SYS_TIMEB_H)
  382. ENDIF(HAVE_CLOCK_GETTIME)
  383. # Check asm pause instruction
  384. CHECK_C_SOURCE_COMPILES ("#include <sys/types.h>
  385. int main (int argc, char **argv) {
  386. __asm __volatile(\"pause\");
  387. return 0;
  388. }" HAVE_ASM_PAUSE)
  389. ENDIF(NOT CMAKE_SYSTEM_NAME STREQUAL "SunOS")
  390. CHECK_C_SOURCE_RUNS("
  391. #include <stdbool.h>
  392. int main(int argc, char **argv) {
  393. int a = 0, b = 0;
  394. if (__atomic_compare_exchange_n(&a, &b, 1, false, __ATOMIC_RELEASE, __ATOMIC_RELAXED)) {
  395. return 0;
  396. }
  397. return -1;
  398. }
  399. " HAVE_ATOMIC_BUILTINS)
  400. IF(NOT HAVE_ATOMIC_BUILTINS)
  401. MESSAGE(STATUS "atomic builtins are -NOT- supported")
  402. ELSE()
  403. MESSAGE(STATUS "atomic builtins are supported")
  404. ENDIF()
  405. CHECK_C_SOURCE_RUNS("
  406. #include <x86intrin.h>
  407. int main(int argc, char **argv) {
  408. __builtin_ia32_lfence ();
  409. if (__builtin_ia32_rdtsc()) {
  410. return 0;
  411. }
  412. return -1;
  413. }
  414. " HAVE_RDTSC)
  415. IF(NOT HAVE_RDTSC)
  416. MESSAGE(STATUS "rdtsc intrinsic is -NOT- supported")
  417. ELSE()
  418. MESSAGE(STATUS "rdtsc intrinsic is supported")
  419. ENDIF()
  420. IF(CMAKE_SYSTEM_NAME STREQUAL "Linux")
  421. # In linux, we need to mount /run/shm to test which could be unavailable
  422. # on a build system. On the other hand, we know that linux has stupid
  423. # but compatible shmem support, so we assume this macro as true
  424. SET(HAVE_SANE_SHMEM 1)
  425. ELSE()
  426. CHECK_C_SOURCE_RUNS("
  427. #include <sys/mman.h>
  428. #include <fcntl.h>
  429. #include <unistd.h>
  430. #define TEST_NAME \"/test-shmem-work\"
  431. int
  432. main (int argc, char **argv)
  433. {
  434. int fd;
  435. fd = shm_open (TEST_NAME, O_RDWR | O_CREAT | O_EXCL, 00600);
  436. if (fd == -1) {
  437. return -1;
  438. }
  439. if (ftruncate (fd, 100) == -1) {
  440. shm_unlink (TEST_NAME);
  441. close (fd);
  442. return -1;
  443. }
  444. if (ftruncate (fd, 200) == -1) {
  445. shm_unlink (TEST_NAME);
  446. close (fd);
  447. return -1;
  448. }
  449. if (ftruncate (fd, 300) == -1) {
  450. shm_unlink (TEST_NAME);
  451. close (fd);
  452. return -1;
  453. }
  454. close (fd);
  455. shm_unlink (TEST_NAME);
  456. return 0;
  457. }
  458. " HAVE_SANE_SHMEM)
  459. IF(NOT HAVE_SANE_SHMEM)
  460. MESSAGE(STATUS "shmem support is NOT compatible with POSIX")
  461. ELSE()
  462. MESSAGE(STATUS "shmem support is compatible with POSIX")
  463. ENDIF()
  464. ENDIF()
  465. # Check queue.h compatibility
  466. IF(NOT HAVE_COMPATIBLE_QUEUE_H)
  467. INCLUDE_DIRECTORIES(compat)
  468. ENDIF(NOT HAVE_COMPATIBLE_QUEUE_H)
  469. IF(ENABLE_URL_INCLUDE MATCHES "ON")
  470. FIND_LIBRARY(LIBFETCH_LIBRARY HINTS "${RSPAMD_SEARCH_PATH}"
  471. NAMES fetch PATHS PATH_SUFFIXES lib64 lib
  472. PATHS ${RSPAMD_DEFAULT_LIBRARY_PATHS}
  473. DOC "Path where the libfetch library can be found")
  474. IF(LIBFETCH_LIBRARY)
  475. FIND_FILE(HAVE_FETCH_H HINTS "${RSPAMD_SEARCH_PATH}"
  476. NAMES fetch.h
  477. PATH_SUFFIXES include
  478. PATHS ${RSPAMD_DEFAULT_INCLUDE_PATHS}
  479. DOC "Path to libfetch header")
  480. ELSE(LIBFETCH_LIBRARY)
  481. # Try to find libcurl
  482. ProcessPackage(CURL LIBRARY curl INCLUDE curl.h INCLUDE_SUFFIXES include/curl
  483. ROOT ${CURL_ROOT})
  484. IF(NOT WITH_CURL)
  485. MESSAGE(WARNING "Neither libcurl nor libfetch were found, no support of URL includes in configuration")
  486. ENDIF(NOT WITH_CURL)
  487. ENDIF(LIBFETCH_LIBRARY)
  488. ENDIF(ENABLE_URL_INCLUDE MATCHES "ON")
  489. IF(NOT DESTDIR)
  490. SET(DESTDIR $ENV{DESTDIR})
  491. ENDIF(NOT DESTDIR)
  492. ################################ SUBDIRS SECTION ###########################
  493. ADD_SUBDIRECTORY(contrib/hiredis)
  494. SET(WITH_HIREDIS 1)
  495. INCLUDE_DIRECTORIES(BEFORE "${CMAKE_SOURCE_DIR}/contrib/hiredis")
  496. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES "${CMAKE_REQUIRED_LIBRARIES}")
  497. IF(HAVE_FETCH_H)
  498. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES fetch)
  499. ENDIF(HAVE_FETCH_H)
  500. IF(WITH_DB)
  501. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES db)
  502. ENDIF(WITH_DB)
  503. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES "${LUA_LIBRARY}")
  504. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES ucl)
  505. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rdns)
  506. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES ottery)
  507. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES xxhash)
  508. IF(GLIB_COMPAT)
  509. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES glibadditions)
  510. ENDIF(GLIB_COMPAT)
  511. ADD_SUBDIRECTORY(contrib/xxhash)
  512. ADD_SUBDIRECTORY(contrib/cdb)
  513. ADD_SUBDIRECTORY(contrib/http-parser)
  514. ADD_SUBDIRECTORY(contrib/fpconv)
  515. ADD_SUBDIRECTORY(contrib/lc-btrie)
  516. ADD_SUBDIRECTORY(contrib/libottery)
  517. ADD_SUBDIRECTORY(contrib/zstd)
  518. IF(ENABLE_SNOWBALL MATCHES "ON")
  519. ADD_SUBDIRECTORY(contrib/snowball)
  520. SET(WITH_SNOWBALL 1)
  521. ENDIF()
  522. ADD_SUBDIRECTORY(contrib/libucl)
  523. ADD_SUBDIRECTORY(contrib/librdns)
  524. ADD_SUBDIRECTORY(contrib/aho-corasick)
  525. ADD_SUBDIRECTORY(contrib/lua-lpeg)
  526. ADD_SUBDIRECTORY(contrib/t1ha)
  527. ADD_SUBDIRECTORY(contrib/libev)
  528. ADD_SUBDIRECTORY(contrib/kann)
  529. ADD_SUBDIRECTORY(contrib/fastutf8)
  530. IF (NOT WITH_LUAJIT)
  531. ADD_SUBDIRECTORY(contrib/lua-bit)
  532. ENDIF()
  533. IF (ENABLE_LUA_REPL MATCHES "ON")
  534. ADD_SUBDIRECTORY(contrib/replxx)
  535. SET(WITH_LUA_REPL 1)
  536. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rspamd-replxx)
  537. ENDIF()
  538. IF (ENABLE_SNOWBALL MATCHES "ON")
  539. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES stemmer)
  540. ENDIF()
  541. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rspamd-hiredis)
  542. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rspamd-actrie)
  543. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rspamd-t1ha)
  544. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rspamd-ev)
  545. LIST(APPEND RSPAMD_REQUIRED_LIBRARIES rspamd-kann)
  546. IF(ENABLE_CLANG_PLUGIN MATCHES "ON")
  547. ADD_SUBDIRECTORY(clang-plugin)
  548. ENDIF()
  549. ADD_SUBDIRECTORY(src)
  550. ADD_SUBDIRECTORY(test)
  551. ADD_SUBDIRECTORY(utils)
  552. ############################ TARGETS SECTION ###############################
  553. CONFIGURE_FILE(config.h.in src/config.h)
  554. ##################### INSTALLATION ##########################################
  555. # Binaries
  556. # Configs
  557. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${CONFDIR})")
  558. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${SHAREDIR})")
  559. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${EXAMPLESDIR})")
  560. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${LUALIBDIR})")
  561. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${PLUGINSDIR})")
  562. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${RULESDIR})")
  563. # Install configs only if they are unchanged
  564. LIST(LENGTH CONFFILES CONFLIST_COUNT)
  565. MATH(EXPR CONFLIST_MAX ${CONFLIST_COUNT}-1)
  566. FILE(GLOB_RECURSE CONF_FILES RELATIVE "${CMAKE_SOURCE_DIR}/conf"
  567. "${CMAKE_SOURCE_DIR}/conf/*" )
  568. FOREACH(CONF_FILE ${CONF_FILES})
  569. GET_FILENAME_COMPONENT(_rp ${CONF_FILE} PATH)
  570. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${CONFDIR}/${_rp})")
  571. INSTALL(FILES "${CMAKE_CURRENT_SOURCE_DIR}/conf/${CONF_FILE}"
  572. DESTINATION ${CONFDIR}/${_rp})
  573. ENDFOREACH(CONF_FILE)
  574. # Lua plugins
  575. FILE(GLOB LUA_PLUGINS RELATIVE "${CMAKE_CURRENT_SOURCE_DIR}/src/plugins/lua"
  576. "${CMAKE_CURRENT_SOURCE_DIR}/src/plugins/lua/*.lua" )
  577. FOREACH(LUA_PLUGIN ${LUA_PLUGINS})
  578. GET_FILENAME_COMPONENT(_rp ${LUA_PLUGIN} PATH)
  579. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${PLUGINSDIR}/${_rp})")
  580. INSTALL(FILES "src/plugins/lua/${LUA_PLUGIN}" DESTINATION ${PLUGINSDIR}/${_rp})
  581. ENDFOREACH(LUA_PLUGIN)
  582. # Install TLD list
  583. INSTALL(FILES "contrib/publicsuffix/effective_tld_names.dat" DESTINATION
  584. "${SHAREDIR}")
  585. # Install languages data
  586. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${SHAREDIR}/languages)")
  587. FILE(GLOB LANGUAGES "${CMAKE_CURRENT_SOURCE_DIR}/contrib/languages-data/*.json")
  588. FOREACH(_LANG ${LANGUAGES})
  589. INSTALL(FILES "${_LANG}" DESTINATION ${SHAREDIR}/languages)
  590. ENDFOREACH()
  591. INSTALL(FILES "${CMAKE_CURRENT_SOURCE_DIR}/contrib/languages-data/stop_words" DESTINATION ${SHAREDIR}/languages)
  592. # Lua config
  593. FILE(GLOB_RECURSE LUA_CONFIGS RELATIVE "${CMAKE_CURRENT_SOURCE_DIR}/rules"
  594. "${CMAKE_CURRENT_SOURCE_DIR}/rules/*.lua")
  595. FOREACH(LUA_CONF ${LUA_CONFIGS})
  596. GET_FILENAME_COMPONENT(_rp ${LUA_CONF} PATH)
  597. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${RULESDIR}/${_rp})")
  598. INSTALL(FILES "rules/${LUA_CONF}" DESTINATION ${RULESDIR}/${_rp})
  599. ENDFOREACH(LUA_CONF)
  600. # Lua libs
  601. FILE(GLOB_RECURSE LUA_LIBS RELATIVE "${CMAKE_CURRENT_SOURCE_DIR}/lualib"
  602. "${CMAKE_CURRENT_SOURCE_DIR}/lualib/*.lua")
  603. FOREACH(LUA_LIB ${LUA_LIBS})
  604. GET_FILENAME_COMPONENT(_rp ${LUA_LIB} PATH)
  605. INSTALL(CODE "FILE(MAKE_DIRECTORY \$ENV{DESTDIR}${LUALIBDIR}/${_rp})")
  606. INSTALL(FILES "lualib/${LUA_LIB}" DESTINATION ${LUALIBDIR}/${_rp})
  607. ENDFOREACH(LUA_LIB)
  608. # Install lua fun library
  609. INSTALL(FILES "contrib/lua-fun/fun.lua" DESTINATION ${LUALIBDIR})
  610. INSTALL(FILES "contrib/lua-argparse/argparse.lua" DESTINATION ${LUALIBDIR})
  611. INSTALL(FILES "contrib/lua-tableshape/tableshape.lua" DESTINATION ${LUALIBDIR})
  612. INSTALL(FILES "contrib/lua-lupa/lupa.lua" DESTINATION ${LUALIBDIR})
  613. INSTALL(FILES "contrib/lua-lpeg/lpegre.lua" DESTINATION ${LUALIBDIR})
  614. # systemd unit
  615. IF(CMAKE_SYSTEM_NAME STREQUAL "Linux" AND WANT_SYSTEMD_UNITS MATCHES "ON")
  616. INSTALL(FILES "rspamd.service" DESTINATION ${SYSTEMDDIR})
  617. ENDIF(CMAKE_SYSTEM_NAME STREQUAL "Linux" AND WANT_SYSTEMD_UNITS MATCHES "ON")
  618. # Manual pages
  619. INSTALL(FILES "doc/rspamd.8" DESTINATION ${MANDIR}/man8)
  620. INSTALL(FILES "doc/rspamc.1" DESTINATION ${MANDIR}/man1)
  621. INSTALL(FILES "doc/rspamadm.1" DESTINATION ${MANDIR}/man1)
  622. # Utils
  623. INSTALL(PROGRAMS "utils/rspamd_stats.pl" RENAME rspamd_stats DESTINATION bin)
  624. # Install webui
  625. IF(INSTALL_WEBUI MATCHES "ON")
  626. INSTALL(DIRECTORY "interface/" DESTINATION ${WWWDIR} PATTERN ".git" EXCLUDE)
  627. ENDIF(INSTALL_WEBUI MATCHES "ON")
  628. INSTALL(DIRECTORY "contrib/elastic/" DESTINATION "${SHAREDIR}/elastic" PATTERN ".git" EXCLUDE)
  629. ADD_CUSTOM_TARGET(dist ${CMAKE_SOURCE_DIR}/dist.sh
  630. "${CMAKE_BINARY_DIR}/rspamd-${RSPAMD_VERSION}.tar.xz" "${TAR}"
  631. COMMENT "Create source distribution"
  632. WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
  633. IF(NOT DEBIAN_BUILD)
  634. ADD_CUSTOM_TARGET(check DEPENDS rspamd-test)
  635. ADD_CUSTOM_TARGET(run-test DEPENDS check COMMAND test/rspamd-test)
  636. ENDIF(NOT DEBIAN_BUILD)
  637. # PVS Studio
  638. find_program(_PVS_STUDIO "pvs-studio-analyzer")
  639. if(_PVS_STUDIO)
  640. include(PVS-Studio)
  641. pvs_studio_add_target(TARGET ${PROJECT_NAME}.analyze
  642. ANALYZE ${PROJECT_NAME} rspamd-server rspamadm rspamc
  643. OUTPUT FORMAT errorfile
  644. LOG target_${PROJECT_NAME}.err)
  645. endif()