]> The Tcpdump Group git mirrors - tcpdump/blob - CMakeLists.txt
Use C99 macros to define 64-bit constants and maximum 64-bit values.
[tcpdump] / CMakeLists.txt
1 if(WIN32)
2 #
3 # We need 3.12 or later, so that we can set policy CMP0074; see
4 # below.
5 #
6 cmake_minimum_required(VERSION 3.12)
7 else(WIN32)
8 #
9 # For now:
10 #
11 # if this is a version of CMake less than 3.5, require only
12 # 2.8.12, just in case somebody is configuring with CMake
13 # on a "long-term support" version # of some OS and that
14 # version supplies an older version of CMake;
15 #
16 # otherwise, if it's a version less than 3.10, require only
17 # 3.5, just in case somebody is configuring with CMake
18 # on a "long-term support" version # of some OS and that
19 # version supplies an older version of CMake;
20 #
21 # otherwise, require 3.10, so we don't get messages warning
22 # that support for versions of CMake lower than 3.10 is
23 # deprecated.
24 #
25 if(CMAKE_VERSION VERSION_LESS "3.5")
26 cmake_minimum_required(VERSION 2.8.12)
27 elseif(CMAKE_VERSION VERSION_LESS "3.10")
28 cmake_minimum_required(VERSION 3.5)
29 else()
30 cmake_minimum_required(VERSION 3.10)
31 endif()
32 endif(WIN32)
33
34 #
35 # We want find_path() and find_library() to honor {packagename}_ROOT,
36 # as that appears to be the standard way to say "hey, look here for
37 # this package" from the command line.
38 #
39 if(POLICY CMP0074)
40 cmake_policy(SET CMP0074 NEW)
41 endif()
42
43 #
44 # OK, this is a pain.
45 #
46 # When building on NetBSD, with a libpcap installed from pkgsrc,
47 # a -Wl,-rpath,/usr/pkg/lib option is added to the options when
48 # linking tcpdump. This puts /usr/pkg/lib into the run-time path.
49 #
50 # However, by default, CMake adds a rule to the install CMake script
51 # a CMake command (using an undocumented subcommand of file()) that
52 # strips /usr/pkg/lib *out* of the run-time path; the message in the
53 # output for the "install" target is
54 #
55 # -- Set runtime path of "{target-directory}/tcpdump" to ""
56 #
57 # I am not certain what the rationale is for doing this, but a
58 # *consequence* of this is that, when you run the installed tcpdump,
59 # it fails to find libpcap.so:
60 #
61 # $ {target-directory}/tcpdump -h
62 # {target-directory}/tcpdump: Shared object "libpcap.so.0" not found
63 #
64 # It also appears to be the case that, on Ubuntu 22.04, FreeBSD 12,
65 # DragonFly BSD 5.8, OpenBSD 6.6, and Solaris 11.4,
66 #
67 # On Ubuntu and Solaris, even if you have a libpcap in /usr/local, you
68 # have to provide not only -I/usr/local/include and -L/usr/local/lib,
69 # you also must provide -Wl,-rpath,/usr/local/lib in order to have
70 # the run-time linker look in /usr/local/lib for libpcap. If it's not
71 # specified, then, if the shared library major version number of the
72 # libpcap in /usr/lib is the same as the shared major version number
73 # of the libpcap in /usr/local/lib, the run-time linker will find the
74 # libpcap in /usr/lib; if the versions are different, the run-time
75 # linker will fail to find the libpcap in /usr/lib, so the program will
76 # fail to run.
77 #
78 # We suppress this by setting CMAKE_INSTALL_RPATH_USE_LINK_PATH to TRUE;
79 # as the documentation for that variable says:
80 #
81 # Add paths to linker search and installed rpath.
82 #
83 # CMAKE_INSTALL_RPATH_USE_LINK_PATH is a boolean that if set to True
84 # will append to the runtime search path (rpath) of installed
85 # binaries any directories outside the project that are in the linker
86 # search path or contain linked library files. The directories are
87 # appended after the value of the INSTALL_RPATH target property.
88 #
89 # If, for whatever reason, directories in which we search for external
90 # libraries, other than the standard system library directories, are
91 # added to the executable's rpath in the build process, we most
92 # definitely want them in the installed image's rpath if they are
93 # necessary in order to find the libraries at run time.
94 #
95 set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE)
96
97 set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake/Modules)
98
99 #
100 # We explicitly indicate what languages are used in tcpdump to avoid
101 # checking for a C++ compiler.
102 #
103 # One reason to avoid that check is that there's no need to waste
104 # configuration time performing it.
105 #
106 # Another reason is that:
107 #
108 # CMake will try to determine the sizes of some data types, including
109 # void *, early in the process of configuration; apparently, it's done
110 # as part of processing the project() command.
111 #
112 # At least as of CMake 2.8.6, it does so by checking the size of
113 # "void *" in C, setting CMAKE_C_SIZEOF_DATA_PTR based on that,
114 # setting CMAKE_SIZEOF_VOID_P to that, and then checking the size
115 # of "void *" in C++, setting CMAKE_CXX_SIZEOF_DATA_PTR based on
116 # that, and then setting CMAKE_SIZEOF_VOID_P to *that*.
117 #
118 # The compile tests include whatever C flags may have been provided
119 # to CMake in the CFLAGS and CXXFLAGS environment variables.
120 #
121 # If you set an architecture flag such as -m32 or -m64 in CFLAGS
122 # but *not* in CXXFLAGS, the size for C++ will win, and hilarity
123 # will ensue.
124 #
125 # Or if, at least on Solaris, you have a newer version of GCC
126 # installed, but *not* a newer version of G++, and you have Oracle
127 # Studio installed, it will find GCC, which will default to building
128 # 64-bit, and Oracle Studio's C++ compiler, which will default to
129 # building 32-bit, the size for C++ will win, and, again, hilarity
130 # will ensue.
131 #
132 project(tcpdump C)
133
134 #
135 # Export the size of void * as SIZEOF_VOID_P so that it can be
136 # tested with #if.
137 #
138 set(SIZEOF_VOID_P "${CMAKE_SIZEOF_VOID_P}")
139
140 #
141 # Show the bit width for which we're compiling.
142 # This can help debug problems if you're dealing with a compiler that
143 # defaults to generating 32-bit code even when running on a 64-bit
144 # platform, and where that platform may provide only 64-bit versions of
145 # libraries that we might use (looking at *you*, Oracle Studio!).
146 #
147 if(CMAKE_SIZEOF_VOID_P EQUAL 4)
148 message(STATUS "Building 32-bit")
149 elseif(CMAKE_SIZEOF_VOID_P EQUAL 8)
150 message(STATUS "Building 64-bit")
151 endif()
152
153 #
154 # Solaris pkg-config is annoying. For at least one package (D-Bus, I'm
155 # looking at *you*!), there are separate include files for 32-bit and
156 # 64-bit builds (I guess using "unsigned long long" as a 64-bit integer
157 # type on a 64-bit build is like crossing the beams or something), and
158 # there are two separate .pc files, so if we're doing a 32-bit build we
159 # should make sure we look in /usr/lib/pkgconfig for .pc files and if
160 # we're doing a 64-bit build we should make sure we look in
161 # /usr/lib/amd64/pkgconfig for .pc files.
162 #
163 if(CMAKE_SYSTEM_NAME STREQUAL "SunOS" AND CMAKE_SYSTEM_VERSION MATCHES "5[.][0-9.]*")
164 #
165 # Note: string(REPLACE) does not appear to support using ENV{...}
166 # as an argument, so we set a variable and then use set() to set
167 # the environment variable.
168 #
169 if(CMAKE_SIZEOF_VOID_P EQUAL 8)
170 #
171 # 64-bit build. If /usr/lib/pkgconfig appears in the path,
172 # prepend /usr/lib/amd64/pkgconfig to it; otherwise,
173 # put /usr/lib/amd64 at the end.
174 #
175 if((NOT DEFINED ENV{PKG_CONFIG_PATH}) OR "$ENV{PKG_CONFIG_PATH}" EQUAL "")
176 #
177 # Not set, or empty. Set it to /usr/lib/amd64/pkgconfig.
178 #
179 set(fixed_path "/usr/lib/amd64/pkgconfig")
180 elseif("$ENV{PKG_CONFIG_PATH}" MATCHES "/usr/lib/pkgconfig")
181 #
182 # It contains /usr/lib/pkgconfig. Prepend
183 # /usr/lib/amd64/pkgconfig to /usr/lib/pkgconfig.
184 #
185 string(REPLACE "/usr/lib/pkgconfig"
186 "/usr/lib/amd64/pkgconfig:/usr/lib/pkgconfig"
187 fixed_path "$ENV{PKG_CONFIG_PATH}")
188 else()
189 #
190 # Not empty, but doesn't contain /usr/lib/pkgconfig.
191 # Append /usr/lib/amd64/pkgconfig to it.
192 #
193 set(fixed_path "$ENV{PKG_CONFIG_PATH}:/usr/lib/amd64/pkgconfig")
194 endif()
195 set(ENV{PKG_CONFIG_PATH} "${fixed_path}")
196 elseif(CMAKE_SIZEOF_VOID_P EQUAL 4)
197 #
198 # 32-bit build. If /usr/amd64/lib/pkgconfig appears in the path,
199 # prepend /usr/lib/pkgconfig to it.
200 #
201 if("$ENV{PKG_CONFIG_PATH}" MATCHES "/usr/lib/amd64/pkgconfig")
202 #
203 # It contains /usr/lib/amd64/pkgconfig. Prepend
204 # /usr/lib/pkgconfig to /usr/lib/amd64/pkgconfig.
205 #
206 string(REPLACE "/usr/lib/amd64/pkgconfig"
207 "/usr/lib/pkgconfig:/usr/lib/amd64/pkgconfig"
208 fixed_path "$ENV{PKG_CONFIG_PATH}")
209 set(ENV{PKG_CONFIG_PATH} "${fixed_path}")
210 endif()
211 endif()
212 endif()
213
214 #
215 # For checking if a compiler flag works and adding it if it does.
216 #
217 include(CheckCCompilerFlag)
218 macro(check_and_add_compiler_option _option)
219 message(STATUS "Checking C compiler flag ${_option}")
220 string(REPLACE "=" "-" _temp_option_variable ${_option})
221 string(REGEX REPLACE "^-" "" _option_variable ${_temp_option_variable})
222 check_c_compiler_flag("${_option}" ${_option_variable})
223 if(${${_option_variable}})
224 set(C_ADDITIONAL_FLAGS "${C_ADDITIONAL_FLAGS} ${_option}")
225 endif()
226 endmacro()
227
228 #
229 # If we're building with Visual Studio, we require Visual Studio 2015,
230 # in order to get sufficient C99 compatibility. Check for that.
231 #
232 # If not, try the appropriate flag for the compiler to enable C99
233 # features.
234 #
235 set(C_ADDITIONAL_FLAGS "")
236 if(MSVC)
237 if(MSVC_VERSION LESS 1900)
238 message(FATAL_ERROR "Visual Studio 2015 or later is required")
239 endif()
240
241 #
242 # Treat source files as being in UTF-8 with MSVC if it's not using
243 # the Clang front end.
244 # We assume that UTF-8 source is OK with other compilers and with
245 # MSVC if it's using the Clang front end.
246 #
247 if(NOT ${CMAKE_C_COMPILER} MATCHES "clang*")
248 set(C_ADDITIONAL_FLAGS "${C_ADDITIONAL_FLAGS} /utf-8")
249 endif(NOT ${CMAKE_C_COMPILER} MATCHES "clang*")
250 else(MSVC)
251 #
252 # Try to enable as many C99 features as we can.
253 # At minimum, we want C++/C99-style // comments.
254 #
255 # Newer versions of compilers might default to supporting C99, but
256 # older versions may require a special flag.
257 #
258 # Prior to CMake 3.1, setting CMAKE_C_STANDARD will not have any effect,
259 # so, unless and until we require CMake 3.1 or later, we have to do it
260 # ourselves on pre-3.1 CMake, so we just do it ourselves on all versions
261 # of CMake.
262 #
263 # Note: with CMake 3.1 through 3.5, the only compilers for which CMake
264 # handles CMAKE_C_STANDARD are GCC and Clang. 3.6 adds support only
265 # for Intel C; 3.9 adds support for PGI C, Sun C, and IBM XL C, and
266 # 3.10 adds support for Cray C and IAR C, but no version of CMake has
267 # support for HP C. Therefore, even if we use CMAKE_C_STANDARD with
268 # compilers for which CMake supports it, we may still have to do it
269 # ourselves on other compilers.
270 #
271 # See the CMake documentation for the CMAKE_<LANG>_COMPILER_ID variables
272 # for a list of compiler IDs.
273 #
274 # XXX - this just tests whether the option works and adds it if it does.
275 # We don't test whether it's necessary in order to get the C99 features
276 # that we use; if we ever have a user who tries to compile with a compiler
277 # that can't be made to support those features, we can add a test to make
278 # sure we actually *have* C99 support.
279 #
280 if(CMAKE_C_COMPILER_ID MATCHES "GNU" OR
281 CMAKE_C_COMPILER_ID MATCHES "Clang")
282 check_and_add_compiler_option("-std=gnu99")
283 elseif(CMAKE_C_COMPILER_ID MATCHES "XL")
284 #
285 # We want support for extensions picked up for GNU C compatibility,
286 # so we use -qlanglvl=extc99.
287 #
288 check_and_add_compiler_option("-qlanglvl=extc99")
289 elseif(CMAKE_C_COMPILER_ID MATCHES "HP")
290 check_and_add_compiler_option("-AC99")
291 elseif(CMAKE_C_COMPILER_ID MATCHES "Sun")
292 check_and_add_compiler_option("-xc99")
293 elseif(CMAKE_C_COMPILER_ID MATCHES "Intel")
294 check_and_add_compiler_option("-c99")
295 endif()
296 endif(MSVC)
297
298 set(LIBRARY_NAME netdissect)
299
300 ###################################################################
301 # Parameters
302 ###################################################################
303
304 option(WITH_SMI "Build with libsmi, if available" ON)
305 option(WITH_CRYPTO "Build with OpenSSL/libressl libcrypto, if available" ON)
306 option(WITH_CAPSICUM "Build with Capsicum security functions, if available" ON)
307 option(WITH_CAP_NG "Use libcap-ng, if available" ON)
308 option(ENABLE_SMB "Build with the SMB dissector" OFF)
309
310 #
311 # String parameters. Neither of them are set, initially; only if the
312 # user explicitly configures them are they set.
313 #
314 # WITH_CHROOT is STRING, not PATH, as the directory need not exist
315 # when CMake is run.
316 #
317 set(WITH_CHROOT CACHE STRING
318 "Directory to which to chroot when dropping privileges")
319 set(WITH_USER CACHE STRING
320 "User to whom to set the UID when dropping privileges")
321
322 #
323 # By default, build universal with the appropriate set of architectures
324 # for the OS on which we're doing the build.
325 #
326 if(APPLE AND "${CMAKE_OSX_ARCHITECTURES}" STREQUAL "")
327 #
328 # Get the major version of Darwin.
329 #
330 string(REGEX MATCH "^([0-9]+)" SYSTEM_VERSION_MAJOR "${CMAKE_SYSTEM_VERSION}")
331
332 if(SYSTEM_VERSION_MAJOR EQUAL 9)
333 #
334 # Leopard. Build for x86 and 32-bit PowerPC, with
335 # x86 first. (That's what Apple does.)
336 #
337 set(CMAKE_OSX_ARCHITECTURES "i386;ppc")
338 elseif(SYSTEM_VERSION_MAJOR EQUAL 10)
339 #
340 # Snow Leopard. Build for x86-64 and x86, with
341 # x86-64 first. (That's what Apple does.)
342 #
343 set(CMAKE_OSX_ARCHITECTURES "x86_64;i386")
344 endif()
345 endif()
346
347 ###################################################################
348 # Versioning
349 ###################################################################
350
351 # Get, parse, format and set tcpdump's version string from
352 # [tcpdump_root]/VERSION for later use.
353
354 # Get MAJOR, MINOR, PATCH & SUFFIX
355 file(STRINGS ${tcpdump_SOURCE_DIR}/VERSION
356 PACKAGE_VERSION
357 LIMIT_COUNT 1 # Read only the first line
358 )
359
360 ######################################
361 # Project settings
362 ######################################
363
364 include_directories(
365 ${CMAKE_CURRENT_BINARY_DIR}
366 ${tcpdump_SOURCE_DIR}
367 )
368
369 if(MSVC)
370 add_definitions(-D__STDC__)
371 add_definitions(-D_CRT_SECURE_NO_WARNINGS)
372 endif(MSVC)
373
374 if(MSVC)
375 if (USE_STATIC_RT)
376 MESSAGE(STATUS "Use STATIC runtime")
377 set(NAME_RT MT)
378 set (CMAKE_CXX_FLAGS_MINSIZEREL "${CMAKE_CXX_FLAGS_MINSIZEREL} /MT")
379 set (CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} /MT")
380 set (CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MT")
381 set (CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd")
382
383 set (CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL} /MT")
384 set (CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO} /MT")
385 set (CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /MT")
386 set (CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /MTd")
387 else (USE_STATIC_RT)
388 MESSAGE(STATUS "Use DYNAMIC runtime")
389 set(NAME_RT MD)
390 set (CMAKE_CXX_FLAGS_MINSIZEREL "${CMAKE_CXX_FLAGS_MINSIZEREL} /MD")
391 set (CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} /MD")
392 set (CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MD")
393 set (CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MDd")
394
395 set (CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL} /MD")
396 set (CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO} /MD")
397 set (CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /MD")
398 set (CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /MDd")
399 endif (USE_STATIC_RT)
400 endif(MSVC)
401
402 #
403 # CMake's definition of "cross-compiling" appears to be "compiling
404 # for an *operating system* other than the one on which the build
405 # is being done*.
406 #
407 # This is an inadequate definition, as people build for the same
408 # operating system but a different instruction set, e.g. building
409 # on an IA-32 or x86-64 Linux box for an Arm embedded Linux box,
410 # or building Arm code on an IA-32 or x86-64 Windows box.
411 #
412 # At least for the Windows case, people may do those builds by
413 # setting the target with th -A flag to CMake; that causes
414 # CMAKE_GENERATOR_PLATFORM to be set to the target. If
415 # CMAKE_GENERATOR_PLATFORM is set, compare it with
416 # CMAKE_HOST_SYSTEM_PROCESSOR and, if they're not equal, set
417 # CMAKE_CROSSCOMPILING to TRUE.
418 #
419 if (CMAKE_GENERATOR_PLATFORM AND
420 NOT CMAKE_GENERATOR_PLATFORM STREQUAL CMAKE_HOST_SYSTEM_PROCESSOR)
421 set(CMAKE_CROSSCOMPILING TRUE)
422 endif()
423
424 ###################################################################
425 # Detect available platform features
426 ###################################################################
427
428 include(CMakePushCheckState)
429 include(CheckIncludeFile)
430 include(CheckIncludeFiles)
431 include(CheckFunctionExists)
432 include(CheckLibraryExists)
433 include(CheckSymbolExists)
434 include(CheckStructHasMember)
435 include(CheckVariableExists)
436 include(CheckTypeSize)
437
438 #
439 # Get the size of a time_t, to know whether it's 32-bit or 64-bit.
440 #
441 cmake_push_check_state()
442 set(CMAKE_EXTRA_INCLUDE_FILES time.h)
443 check_type_size("time_t" SIZEOF_TIME_T)
444 cmake_pop_check_state()
445
446 #
447 # Header files.
448 #
449 check_include_file(rpc/rpc.h HAVE_RPC_RPC_H)
450 if(HAVE_RPC_RPC_H)
451 check_include_files("rpc/rpc.h;rpc/rpcent.h" HAVE_RPC_RPCENT_H)
452 endif(HAVE_RPC_RPC_H)
453
454 #
455 # Functions.
456 #
457 check_function_exists(strlcat HAVE_STRLCAT)
458 check_function_exists(strlcpy HAVE_STRLCPY)
459 check_function_exists(strsep HAVE_STRSEP)
460
461 #
462 # Find library needed for gethostbyaddr.
463 # NOTE: if you hand check_library_exists as its last argument a variable
464 # that's been set, it skips the test, so we need different variables.
465 #
466 set(TCPDUMP_LINK_LIBRARIES "")
467 if(WIN32)
468 #
469 # We need winsock2.h and ws2tcpip.h.
470 #
471 cmake_push_check_state()
472 set(CMAKE_REQUIRED_LIBRARIES ws2_32)
473 check_symbol_exists(gethostbyaddr "winsock2.h;ws2tcpip.h" LIBWS2_32_HAS_GETHOSTBYADDR)
474 cmake_pop_check_state()
475 if(LIBWS2_32_HAS_GETHOSTBYADDR)
476 set(TCPDUMP_LINK_LIBRARIES ws2_32 ${TCPDUMP_LINK_LIBRARIES})
477 else(LIBWS2_32_HAS_GETHOSTBYADDR)
478 message(FATAL_ERROR "gethostbyaddr is required, but wasn't found")
479 endif(LIBWS2_32_HAS_GETHOSTBYADDR)
480 else(WIN32)
481 check_function_exists(gethostbyaddr STDLIBS_HAVE_GETHOSTBYADDR)
482 if(NOT STDLIBS_HAVE_GETHOSTBYADDR)
483 check_library_exists(socket gethostbyaddr "" LIBSOCKET_HAS_GETHOSTBYADDR)
484 if(LIBSOCKET_HAS_GETHOSTBYADDR)
485 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} socket)
486 else(LIBSOCKET_HAS_GETHOSTBYADDR)
487 check_library_exists(nsl gethostbyaddr "" LIBNSL_HAS_GETHOSTBYADDR)
488 if(LIBNSL_HAS_GETHOSTBYADDR)
489 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} nsl)
490 else(LIBNSL_HAS_GETHOSTBYADDR)
491 check_library_exists(network gethostbyaddr "" LIBNETWORK_HAS_GETHOSTBYADDR)
492 if(LIBNETWORK_HAS_GETHOSTBYADDR)
493 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} network)
494 else(LIBNETWORK_HAS_GETHOSTBYADDR)
495 message(FATAL_ERROR "gethostbyaddr is required, but wasn't found")
496 endif(LIBNETWORK_HAS_GETHOSTBYADDR)
497 endif(LIBNSL_HAS_GETHOSTBYADDR)
498 endif(LIBSOCKET_HAS_GETHOSTBYADDR)
499 endif(NOT STDLIBS_HAVE_GETHOSTBYADDR)
500 endif(WIN32)
501
502 #
503 # This may require additional libraries.
504 #
505 cmake_push_check_state()
506 set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES})
507 check_function_exists(getservent STDLIBS_HAVE_GETSERVENT)
508 if(STDLIBS_HAVE_GETSERVENT)
509 set(HAVE_GETSERVENT TRUE)
510 else(STDLIBS_HAVE_GETSERVENT)
511 #
512 # Some platforms may need -lsocket for getservent.
513 #
514 set(CMAKE_REQUIRED_LIBRARIES socket ${TCPDUMP_LINK_LIBRARIES})
515 check_function_exists(getservent LIBSOCKET_HAS_GETSERVENT)
516 if(LIBSOCKET_HAS_GETSERVENT)
517 set(HAVE_GETSERVENT TRUE)
518 set(TCPDUMP_LINK_LIBRARIES socket ${TCPDUMP_LINK_LIBRARIES})
519 endif(LIBSOCKET_HAS_GETSERVENT)
520 endif(STDLIBS_HAVE_GETSERVENT)
521 cmake_pop_check_state()
522
523 if (NOT CMAKE_CROSSCOMPILING)
524 #
525 # Require a proof of suitable snprintf(3), same as in Autoconf.
526 #
527 include(CheckCSourceRuns)
528 check_c_source_runs("
529 #include <stdio.h>
530 #include <string.h>
531 #include <inttypes.h>
532 #include <sys/types.h>
533
534 int main()
535 {
536 char buf[100];
537 uint64_t t = (uint64_t)1 << 32;
538
539 snprintf(buf, sizeof(buf), \"%zu\", sizeof(buf));
540 if (strncmp(buf, \"100\", sizeof(buf)))
541 return 1;
542
543 snprintf(buf, sizeof(buf), \"%zd\", -sizeof(buf));
544 if (strncmp(buf, \"-100\", sizeof(buf)))
545 return 2;
546
547 snprintf(buf, sizeof(buf), \"%\" PRId64, -t);
548 if (strncmp(buf, \"-4294967296\", sizeof(buf)))
549 return 3;
550
551 snprintf(buf, sizeof(buf), \"0o%\" PRIo64, t);
552 if (strncmp(buf, \"0o40000000000\", sizeof(buf)))
553 return 4;
554
555 snprintf(buf, sizeof(buf), \"0x%\" PRIx64, t);
556 if (strncmp(buf, \"0x100000000\", sizeof(buf)))
557 return 5;
558
559 snprintf(buf, sizeof(buf), \"%\" PRIu64, t);
560 if (strncmp(buf, \"4294967296\", sizeof(buf)))
561 return 6;
562
563 return 0;
564 }
565
566 "
567 SUITABLE_SNPRINTF
568 )
569 if(NOT SUITABLE_SNPRINTF)
570 message(FATAL_ERROR
571 "The snprintf(3) implementation in this libc is not suitable,
572 tcpdump would not work correctly even if it managed to compile."
573 )
574 endif()
575 else()
576 message(STATUS "Skipped SUITABLE_SNPRINTF because cross-compiling.")
577 endif()
578
579 check_function_exists(getopt_long HAVE_GETOPT_LONG)
580 #
581 # For Windows, don't need to waste time checking for fork() or vfork().
582 #
583 if(NOT WIN32)
584 check_function_exists(fork HAVE_FORK)
585 check_function_exists(vfork HAVE_VFORK)
586 endif(NOT WIN32)
587
588 #
589 # Some platforms may need -lnsl for getrpcbynumber.
590 #
591 cmake_push_check_state()
592 set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES})
593 check_function_exists(getrpcbynumber STDLIBS_HAVE_GETRPCBYNUMBER)
594 if(STDLIBS_HAVE_GETRPCBYNUMBER)
595 set(HAVE_GETRPCBYNUMBER TRUE)
596 else(STDLIBS_HAVE_GETRPCBYNUMBER)
597 set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} nsl)
598 check_function_exists(getrpcbynumber LIBNSL_HAS_GETRPCBYNUMBER)
599 if(LIBNSL_HAS_GETRPCBYNUMBER)
600 set(HAVE_GETRPCBYNUMBER TRUE)
601 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} nsl)
602 endif(LIBNSL_HAS_GETRPCBYNUMBER)
603 endif(STDLIBS_HAVE_GETRPCBYNUMBER)
604 cmake_pop_check_state()
605
606 #
607 # This requires the libraries we require, as ether_ntohost might be
608 # in one of those libraries. That means we have to do this after
609 # we check for those libraries.
610 #
611 # You are in a twisty little maze of UN*Xes, all different.
612 # Some might not have ether_ntohost().
613 # Some might have it and declare it in <net/ethernet.h>.
614 # Some might have it and declare it in <netinet/ether.h>
615 # Some might have it and declare it in <sys/ethernet.h>.
616 # Some might have it and declare it in <arpa/inet.h>.
617 # Some might have it and declare it in <netinet/if_ether.h>.
618 # Some might have it and not declare it in any header file.
619 #
620 # Before you is a C compiler.
621 #
622 cmake_push_check_state()
623 set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES})
624 check_function_exists(ether_ntohost HAVE_ETHER_NTOHOST)
625 if(HAVE_ETHER_NTOHOST)
626 #
627 # OK, we have ether_ntohost(). We don't check whether it's buggy,
628 # as we assume any system that has CMake is likely to be new enough
629 # that, if it has ether_ntohost(), whatever bug is checked for in
630 # autotools is fixed; we just decide to use it.
631 #
632 set(USE_ETHER_NTOHOST TRUE)
633
634 #
635 # Is it declared in <net/ethernet.h>?
636 #
637 # This test fails if we don't have <net/ethernet.h> or if we do
638 # but it doesn't declare ether_ntohost().
639 #
640 check_symbol_exists(ether_ntohost net/ethernet.h NET_ETHERNET_H_DECLARES_ETHER_NTOHOST)
641 if(NET_ETHERNET_H_DECLARES_ETHER_NTOHOST)
642 #
643 # Yes - we have it declared.
644 #
645 set(HAVE_DECL_ETHER_NTOHOST TRUE)
646 endif()
647 #
648 # Did that succeed?
649 #
650 if(NOT HAVE_DECL_ETHER_NTOHOST)
651 #
652 # No - how about <netinet/ether.h>, as on Linux?
653 #
654 # This test fails if we don't have <netinet/ether.h>
655 # or if we do but it doesn't declare ether_ntohost().
656 #
657 check_symbol_exists(ether_ntohost netinet/ether.h NETINET_ETHER_H_DECLARES_ETHER_NTOHOST)
658 if(NETINET_ETHER_H_DECLARES_ETHER_NTOHOST)
659 #
660 # Yes - we have it declared.
661 #
662 set(HAVE_DECL_ETHER_NTOHOST TRUE)
663 endif()
664 endif()
665 #
666 # Did that succeed?
667 #
668 if(NOT HAVE_DECL_ETHER_NTOHOST)
669 #
670 # No - how about <sys/ethernet.h>, as on Solaris 10 and later?
671 #
672 # This test fails if we don't have <sys/ethernet.h>
673 # or if we do but it doesn't declare ether_ntohost().
674 #
675 check_symbol_exists(ether_ntohost sys/ethernet.h SYS_ETHERNET_H_DECLARES_ETHER_NTOHOST)
676 if(SYS_ETHERNET_H_DECLARES_ETHER_NTOHOST)
677 #
678 # Yes - we have it declared.
679 #
680 set(HAVE_DECL_ETHER_NTOHOST TRUE)
681 endif()
682 endif()
683 #
684 # Did that succeed?
685 #
686 if(NOT HAVE_DECL_ETHER_NTOHOST)
687 #
688 # No, how about <arpa/inet.h>, as on AIX?
689 #
690 # This test fails if we don't have <arpa/inet.h>
691 # or if we do but it doesn't declare ether_ntohost().
692 #
693 check_symbol_exists(ether_ntohost arpa/inet.h ARPA_INET_H_DECLARES_ETHER_NTOHOST)
694 if(ARPA_INET_H_DECLARES_ETHER_NTOHOST)
695 #
696 # Yes - we have it declared.
697 #
698 set(HAVE_DECL_ETHER_NTOHOST TRUE)
699 endif()
700 endif()
701 #
702 # Did that succeed?
703 #
704 if(NOT HAVE_DECL_ETHER_NTOHOST)
705 #
706 # No, how about <netinet/if_ether.h>?
707 # On some platforms, it requires <net/if.h> and
708 # <netinet/in.h>, and we always include it with
709 # both of them, so test it with both of them.
710 #
711 # This test fails if we don't have <netinet/if_ether.h>
712 # and the headers we include before it, or if we do but
713 # <netinet/if_ether.h> doesn't declare ether_ntohost().
714 #
715 check_symbol_exists(ether_ntohost "sys/types.h;sys/socket.h;net/if.h;netinet/in.h;netinet/if_ether.h" NETINET_IF_ETHER_H_DECLARES_ETHER_NTOHOST)
716 if(NETINET_IF_ETHER_H_DECLARES_ETHER_NTOHOST)
717 #
718 # Yes - we have it declared.
719 #
720 set(HAVE_DECL_ETHER_NTOHOST TRUE)
721 endif()
722 endif()
723 #
724 # After all that, is ether_ntohost() declared?
725 #
726 if(NOT HAVE_DECL_ETHER_NTOHOST)
727 #
728 # No, we'll have to declare it ourselves.
729 # Do we have "struct ether_addr" if we include<netinet/if_ether.h>?
730 #
731 check_struct_has_member("struct ether_addr" octet "sys/types.h;sys/socket.h;net/if.h;netinet/in.h;netinet/if_ether.h" HAVE_STRUCT_ETHER_ADDR)
732 endif()
733 endif()
734 cmake_pop_check_state()
735
736 #
737 # Data types.
738 #
739 # XXX - there's no check_struct() macro that's like check_struct_has_member()
740 # except that it only checks for the existence of the structure type,
741 # so we use check_struct_has_member() and look for ss_family.
742 #
743
744 ######################################
745 # External dependencies
746 ######################################
747
748 #
749 # libpcap/WinPcap/Npcap.
750 # First, find it.
751 #
752 find_package(PCAP REQUIRED)
753 include_directories(${PCAP_INCLUDE_DIRS})
754
755 cmake_push_check_state()
756
757 #
758 # Now check headers.
759 #
760 set(CMAKE_REQUIRED_INCLUDES ${PCAP_INCLUDE_DIRS})
761
762 #
763 # Check whether we have pcap/pcap-inttypes.h.
764 # If we do, we use that to get the C99 types defined.
765 #
766 check_include_file(pcap/pcap-inttypes.h HAVE_PCAP_PCAP_INTTYPES_H)
767
768 #
769 # Check for various functions in libpcap/WinPcap/Npcap.
770 #
771 cmake_push_check_state()
772 set(CMAKE_REQUIRED_LIBRARIES ${PCAP_LIBRARIES})
773
774 #
775 # Do we have the new open API? Check for pcap_create() and for
776 # pcap_statustostr(), and assume that, if we have both of them,
777 # we also have pcap_activate() and the other new routines
778 # introduced in libpcap 1.0.
779 #
780 # We require those routines, so fail if we don't find it.
781 #
782 check_function_exists(pcap_create HAVE_PCAP_CREATE)
783 if(NOT HAVE_PCAP_CREATE)
784 if(WIN32)
785 MESSAGE(FATAL_ERROR "libpcap is too old; 1.0 or later is required")
786 else(WIN32)
787 MESSAGE(FATAL_ERROR "libpcap is too old; 1.0 or later is required - if you're using WinPcap, try Npcap")
788 endif(WIN32)
789 endif(NOT HAVE_PCAP_CREATE)
790
791 if(WIN32)
792 #
793 # We check for pcap_statustostr() as well, because WinPcap 4.1.3
794 # screwed up and exported pcap_create() but not other routines
795 # such as pcap_statustostr(), even though it defined them and
796 # even though you really want pcap_statustostr() to get strings
797 # corresponding to some of the status returns from the new routines.)
798 #
799 check_function_exists(pcap_statustostr HAVE_PCAP_STATUSTOSTR)
800 if(NOT HAVE_PCAP_STATUSTOSTR)
801 MESSAGE(FATAL_ERROR "pcap_create is available, but pcap_statustostr isn't - if you're using WinPcap, try Npcap ")
802 endif(NOT HAVE_PCAP_STATUSTOSTR)
803 endif(WIN32)
804
805 #
806 # OK, do we have pcap_set_tstamp_type? If so, assume we have
807 # pcap_list_tstamp_types and pcap_free_tstamp_types as well.
808 #
809 check_function_exists(pcap_set_tstamp_type HAVE_PCAP_SET_TSTAMP_TYPE)
810
811 #
812 # And do we have pcap_set_tstamp_precision? If so, we assume
813 # we also have pcap_open_offline_with_tstamp_precision.
814 #
815 check_function_exists(pcap_set_tstamp_precision HAVE_PCAP_SET_TSTAMP_PRECISION)
816
817 #
818 # Check for a miscellaneous collection of functions which we use
819 # if we have them.
820 #
821 check_function_exists(pcap_set_immediate_mode HAVE_PCAP_SET_IMMEDIATE_MODE)
822 check_function_exists(pcap_dump_ftell64 HAVE_PCAP_DUMP_FTELL64)
823 #
824 # macOS Sonoma's libpcap includes stub versions of the remote-
825 # capture APIs. They are exported as "weakly linked symbols".
826 #
827 # Xcode 15 offers only a macOS Sonoma SDK, which has a .tbd
828 # file for libpcap that claims it includes those APIs. (Newer
829 # versions of macOS don't provide the system shared libraries,
830 # they only provide the dyld shared cache containing those
831 # libraries, so the OS provides SDKs that include a .tbd file
832 # to use when linking.)
833 #
834 # This means that check_function_exists() will think that
835 # the remote-capture APIs are present, including pcap_open()
836 # and pcap_findalldevs_ex().
837 #
838 # However, they are *not* present in macOS Ventura and earlier,
839 # which means that building on Ventura with Xcode 15 produces
840 # executables that fail to start because one of those APIs
841 # isn't found in the system libpcap.
842 #
843 # Protecting calls to those APIs with __builtin_available()
844 # does not prevent this, because the libpcap header files
845 # in the Sonoma SDK mark them as being first available
846 # in macOS 10.13, just like all the other routines introduced
847 # in libpcap 1.9, even though they're only available if libpcap
848 # is built with remote capture enabled or stub routines are
849 # provided. (A fix to enable this has been checked into the
850 # libpcap repository, and may end up in a later version of
851 # the SDK.)
852 #
853 # Given all that, and given that the versions of the
854 # remote-capture APIs in Sonoma are stubs that always fail,
855 # there doesn't seem to be any point in checking for pcap_open()
856 # and pcap_findalldevs_ex() if we're linking against the Apple libpcap.
857 #
858 # However, if we're *not* linking against the Apple libpcap,
859 # we should check for it, so that we can use it if it's present.
860 #
861 # So we check for pcap_open() and pcap_findalldevs_ex() if 1) this isn't
862 # macOS or 2) the the libpcap we found is not a system library, meaning
863 # that its path begins neither with /usr/lib (meaning it's a system
864 # dylib) nor /Application/Xcode.app (meaning it's a file in
865 # the Xcode SDK).
866 #
867 if(NOT APPLE OR NOT
868 (PCAP_LIBRARIES MATCHES "/usr/lib/.*" OR
869 PCAP_LIBRARIES MATCHES "/Application/Xcode.app/.*"))
870 check_function_exists(pcap_open HAVE_PCAP_OPEN)
871 check_function_exists(pcap_findalldevs_ex HAVE_PCAP_FINDALLDEVS_EX)
872 endif()
873
874 #
875 # On Windows, check for pcap_wsockinit(); if we don't have it, check for
876 # wsockinit().
877 #
878 if(WIN32)
879 check_function_exists(pcap_wsockinit HAVE_PCAP_WSOCKINIT)
880 if(NOT HAVE_PCAP_WSOCKINIT)
881 check_function_exists(wsockinit HAVE_WSOCKINIT)
882 endif(NOT HAVE_PCAP_WSOCKINIT)
883 endif(WIN32)
884
885 #
886 # Check for special debugging functions
887 #
888 check_function_exists(pcap_set_parser_debug HAVE_PCAP_SET_PARSER_DEBUG)
889 if(NOT HAVE_PCAP_SET_PARSER_DEBUG)
890 # Check whether libpcap defines pcap_debug or yydebug
891 check_variable_exists(pcap_debug HAVE_PCAP_DEBUG)
892 if(NOT HAVE_PCAP_DEBUG)
893 check_variable_exists(yydebug HAVE_YYDEBUG)
894 endif(NOT HAVE_PCAP_DEBUG)
895 endif(NOT HAVE_PCAP_SET_PARSER_DEBUG)
896
897 check_function_exists(pcap_set_optimizer_debug HAVE_PCAP_SET_OPTIMIZER_DEBUG)
898
899 #
900 # bpf_dump() moved from tcpdump to libpcap in libpcap 0.6, but not all
901 # versions of libpcap didn't pick that change up; the OpenBSD libpcap
902 # picked up most of the new APIs from versions up to 1.0, but didn't
903 # pick up bpf_dump(), so we need to provide it if it's absent.
904 #
905 check_function_exists(bpf_dump HAVE_BPF_DUMP)
906
907 cmake_pop_check_state()
908 cmake_pop_check_state()
909
910 #
911 # We have libpcap.
912 #
913 include_directories(SYSTEM ${PCAP_INCLUDE_DIRS})
914 set(TCPDUMP_LINK_LIBRARIES ${PCAP_LIBRARIES} ${TCPDUMP_LINK_LIBRARIES})
915
916 #
917 # Optional libraries.
918 #
919
920 #
921 # libsmi.
922 #
923 if(WITH_SMI)
924 find_package(SMI)
925 if(SMI_FOUND)
926 include_directories(SYSTEM ${SMI_INCLUDE_DIRS})
927 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} ${SMI_LIBRARIES})
928 set(USE_LIBSMI ON)
929 endif(SMI_FOUND)
930 endif(WITH_SMI)
931
932 #
933 # OpenSSL/libressl libcrypto.
934 #
935 if(WITH_CRYPTO)
936 find_package(CRYPTO)
937 if(CRYPTO_FOUND)
938 #
939 # 1) do we have EVP_CIPHER_CTX_new?
940 # If so, we use it to allocate an EVP_CIPHER_CTX, as
941 # EVP_CIPHER_CTX may be opaque; otherwise, we allocate
942 # it ourselves.
943 #
944 cmake_push_check_state()
945 set(CMAKE_REQUIRED_LIBRARIES "${CRYPTO_LIBRARIES}")
946
947 check_function_exists(EVP_CIPHER_CTX_new HAVE_EVP_CIPHER_CTX_NEW)
948
949 #
950 # 2) do we have EVP_DecryptInit_ex()?
951 # If so, we use it, because we need to be able to make two
952 # "initialize the cipher" calls, one with the cipher and key,
953 # and one with the IV, and, as of OpenSSL 1.1, You Can't Do That
954 # with EVP_DecryptInit(), because a call to EVP_DecryptInit() will
955 # unconditionally clear the context, and if you don't supply a
956 # cipher, it'll clear the cipher, rendering the context unusable
957 # and causing a crash.
958 #
959 check_function_exists(EVP_DecryptInit_ex HAVE_EVP_DECRYPTINIT_EX)
960
961 cmake_pop_check_state()
962
963 #
964 # We have libcrypto.
965 #
966 include_directories(SYSTEM ${CRYPTO_INCLUDE_DIRS})
967 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} ${CRYPTO_LIBRARIES})
968 set(HAVE_LIBCRYPTO ON)
969 endif(CRYPTO_FOUND)
970 endif(WITH_CRYPTO)
971
972 #
973 # Capsicum sandboxing.
974 # Some of this is in the system library, some of it is in other libraries.
975 #
976 if(WITH_CAPSICUM)
977 check_include_files("sys/capsicum.h" HAVE_SYS_CAPSICUM_H)
978 if(HAVE_SYS_CAPSICUM_H)
979 check_function_exists(cap_enter HAVE_CAP_ENTER)
980 check_function_exists(cap_rights_limit HAVE_CAP_RIGHTS_LIMIT)
981 check_function_exists(cap_ioctls_limit HAVE_CAP_IOCTLS_LIMIT)
982 check_function_exists(openat HAVE_OPENAT)
983 if(HAVE_CAP_ENTER AND HAVE_CAP_RIGHTS_LIMIT AND
984 HAVE_CAP_IOCTLS_LIMIT AND HAVE_OPENAT)
985 #
986 # OK, we have the functions we need to support Capsicum.
987 #
988 set(HAVE_CAPSICUM TRUE)
989
990 #
991 # OK, can we use Casper?
992 #
993 check_library_exists(casper cap_init "" HAVE_CAP_INIT)
994 if(HAVE_CAP_INIT)
995 cmake_push_check_state()
996 set(CMAKE_REQUIRED_LIBRARIES casper)
997 check_library_exists(cap_dns cap_gethostbyaddr "" HAVE_CAP_GETHOSTBYADDR)
998 cmake_pop_check_state()
999 if(HAVE_CAP_GETHOSTBYADDR)
1000 set(HAVE_CASPER TRUE)
1001 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} casper cap_dns)
1002 endif(HAVE_CAP_GETHOSTBYADDR)
1003 endif(HAVE_CAP_INIT)
1004 endif(HAVE_CAP_ENTER AND HAVE_CAP_RIGHTS_LIMIT AND
1005 HAVE_CAP_IOCTLS_LIMIT AND HAVE_OPENAT)
1006 endif(HAVE_SYS_CAPSICUM_H)
1007 endif(WITH_CAPSICUM)
1008
1009 #
1010 # libcap-ng.
1011 #
1012 if(WITH_CAP_NG)
1013 check_include_file(cap-ng.h HAVE_CAP_NG_H)
1014 check_library_exists(cap-ng capng_change_id "" HAVE_LIBCAP_NG)
1015 if(HAVE_LIBCAP_NG)
1016 set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} cap-ng)
1017 endif(HAVE_LIBCAP_NG)
1018 endif(WITH_CAP_NG)
1019
1020 ###################################################################
1021 # Warning options
1022 ###################################################################
1023
1024 #
1025 # Check and add warning options if we have a .devel file.
1026 #
1027 if(EXISTS ${CMAKE_SOURCE_DIR}/.devel OR EXISTS ${CMAKE_BINARY_DIR}/.devel)
1028 #
1029 # Warning options.
1030 #
1031 if(MSVC AND NOT ${CMAKE_C_COMPILER} MATCHES "clang*")
1032 #
1033 # MSVC, with Microsoft's front end and code generator.
1034 # "MSVC" is also set for Microsoft's compiler with a Clang
1035 # front end and their code generator ("Clang/C2"), so we
1036 # check for clang.exe and treat that differently.
1037 #
1038 check_and_add_compiler_option(-Wall)
1039 #
1040 # Disable some pointless warnings that /Wall turns on.
1041 #
1042 # Unfortunately, MSVC does not appear to have an equivalent
1043 # to "__attribute__((unused))" to mark a particular function
1044 # parameter as being known to be unused, so that the compiler
1045 # won't warn about it (for example, the function might have
1046 # that parameter because a pointer to it is being used, and
1047 # the signature of that function includes that parameter).
1048 # C++ lets you give a parameter a type but no name, but C
1049 # doesn't have that.
1050 #
1051 check_and_add_compiler_option(-wd4100)
1052 #
1053 # In theory, we care whether somebody uses f() rather than
1054 # f(void) to declare a function with no arguments, but, in
1055 # practice, there are places in the Windows header files
1056 # that appear to do that, so we squelch that warning.
1057 #
1058 check_and_add_compiler_option(-wd4255)
1059 #
1060 # Windows FD_SET() generates this, so we suppress it.
1061 #
1062 check_and_add_compiler_option(-wd4548)
1063 #
1064 # Perhaps testing something #defined to be 0 with #ifdef is an
1065 # error, and it should be tested with #if, but perhaps it's
1066 # not, and Microsoft does that in its headers, so we squelch
1067 # that warning.
1068 #
1069 check_and_add_compiler_option(-wd4574)
1070 #
1071 # The Windows headers also test not-defined values in #if, so
1072 # we don't want warnings about that, either.
1073 #
1074 check_and_add_compiler_option(-wd4668)
1075 #
1076 # We do *not* care whether some function is, or isn't, going to be
1077 # expanded inline.
1078 #
1079 check_and_add_compiler_option(-wd4710)
1080 check_and_add_compiler_option(-wd4711)
1081 #
1082 # We do *not* care whether we're adding padding bytes after
1083 # structure members.
1084 #
1085 check_and_add_compiler_option(-wd4820)
1086 #
1087 # We do *not* care about every single place the compiler would
1088 # have inserted Spectre mitigation if only we had told it to
1089 # do so with /Qspectre. I guess the theory is that it's seeing
1090 # bounds checks that would prevent out-of-bounds loads and that
1091 # those out-of-bounds loads could be done speculatively and that
1092 # the Spectre attack could detect the value of the out-of-bounds
1093 # data *if* it's within our address space, but unless I'm
1094 # missing something I don't see that as being any form of
1095 # security hole.
1096 #
1097 # XXX - add /Qspectre if that is really worth doing.
1098 #
1099 check_and_add_compiler_option(-wd5045)
1100 #
1101 # We do *not* care whether a structure had padding added at
1102 # the end because of __declspec(align) - *we* don't use
1103 # __declspec(align), because the only structures whose layout
1104 # we precisely specify are those that get overlaid on packet
1105 # data, and in those every element is an array of octets so
1106 # that we have full control over the size and alignment, and,
1107 # apparently, jmp_buf has such a declaration on x86, meaning
1108 # that everything that includes netdissect.h, i.e. almost every
1109 # file in tcpdump, gets a warning.
1110 #
1111 check_and_add_compiler_option(-wd4324)
1112 else()
1113 #
1114 # Other compilers, including MSVC with a Clang front end and
1115 # Microsoft's code generator. We currently treat them as if
1116 # they might support GCC-style -W options.
1117 #
1118 check_and_add_compiler_option(-W)
1119 check_and_add_compiler_option(-Wall)
1120 check_and_add_compiler_option(-Wassign-enum)
1121 check_and_add_compiler_option(-Wcast-qual)
1122 check_and_add_compiler_option(-Wmissing-prototypes)
1123 check_and_add_compiler_option(-Wmissing-variable-declarations)
1124 check_and_add_compiler_option(-Wold-style-definition)
1125 if(NOT CMAKE_C_COMPILER_ID MATCHES "Sun")
1126 # In Sun C versions that implement GCC compatibility "-Wpedantic"
1127 # means the same as "-pedantic". The latter is mutually exclusive
1128 # with several other options. One of those is "-xc99", which has
1129 # already been set for Sun C above.
1130 check_and_add_compiler_option(-Wpedantic)
1131 endif()
1132 check_and_add_compiler_option(-Wpointer-arith)
1133 check_and_add_compiler_option(-Wpointer-sign)
1134 check_and_add_compiler_option(-Wshadow)
1135 check_and_add_compiler_option(-Wsign-compare)
1136 check_and_add_compiler_option(-Wstrict-prototypes)
1137 check_and_add_compiler_option(-Wundef)
1138 check_and_add_compiler_option(-Wunreachable-code-return)
1139 check_and_add_compiler_option(-Wused-but-marked-unused)
1140 check_and_add_compiler_option(-Wwrite-strings)
1141 endif()
1142 endif()
1143
1144 #
1145 # Extra compiler options for the build matrix scripts to request -Werror or
1146 # its equivalent if required. The CMake variable name cannot be CFLAGS
1147 # because that is already used for a different purpose in CMake. Example
1148 # usage: cmake -DEXTRA_CFLAGS='-Wall -Wextra -Werror' ...
1149 #
1150 if(NOT "${EXTRA_CFLAGS}" STREQUAL "")
1151 # The meaning of EXTRA_CFLAGS is "use the exact specified options, or the
1152 # build risks failing to fail", not "try every specified option, omit those
1153 # that do not work and use the rest". Thus use add_compile_options(), not
1154 # foreach()/check_and_add_compiler_option().
1155 string(REPLACE " " ";" _extra_cflags_list ${EXTRA_CFLAGS})
1156 add_compile_options(${_extra_cflags_list})
1157 message(STATUS "Added extra compile options (${EXTRA_CFLAGS})")
1158 endif()
1159
1160 ######################################
1161 # Input files
1162 ######################################
1163
1164 if(ENABLE_SMB)
1165 #
1166 # We allow the SMB dissector to be omitted.
1167 #
1168 set(LOCALSRC ${LOCALSRC}
1169 print-smb.c
1170 smbutil.c)
1171 endif(ENABLE_SMB)
1172
1173 set(NETDISSECT_SOURCE_LIST_C
1174 addrtoname.c
1175 addrtostr.c
1176 af.c
1177 ascii_strcasecmp.c
1178 checksum.c
1179 cpack.c
1180 gmpls.c
1181 in_cksum.c
1182 ipproto.c
1183 l2vpn.c
1184 netdissect.c
1185 netdissect-alloc.c
1186 nlpid.c
1187 ntp.c
1188 oui.c
1189 parsenfsfh.c
1190 print.c
1191 print-802_11.c
1192 print-802_15_4.c
1193 print-ah.c
1194 print-ahcp.c
1195 print-aodv.c
1196 print-aoe.c
1197 print-ap1394.c
1198 print-arcnet.c
1199 print-arista.c
1200 print-arp.c
1201 print-ascii.c
1202 print-atalk.c
1203 print-atm.c
1204 print-babel.c
1205 print-bcm-li.c
1206 print-beep.c
1207 print-bfd.c
1208 print-bgp.c
1209 print-bootp.c
1210 print-brcmtag.c
1211 print-bt.c
1212 print-calm-fast.c
1213 print-carp.c
1214 print-cdp.c
1215 print-cfm.c
1216 print-chdlc.c
1217 print-cip.c
1218 print-cnfp.c
1219 print-dccp.c
1220 print-decnet.c
1221 print-dhcp6.c
1222 print-domain.c
1223 print-dsa.c
1224 print-dtp.c
1225 print-dvmrp.c
1226 print-eap.c
1227 print-egp.c
1228 print-eigrp.c
1229 print-enc.c
1230 print-erspan.c
1231 print-esp.c
1232 print-ether.c
1233 print-fddi.c
1234 print-forces.c
1235 print-fr.c
1236 print-frag6.c
1237 print-ftp.c
1238 print-geneve.c
1239 print-geonet.c
1240 print-gre.c
1241 print-hncp.c
1242 print-hsrp.c
1243 print-http.c
1244 print-icmp.c
1245 print-icmp6.c
1246 print-igmp.c
1247 print-igrp.c
1248 print-ip-demux.c
1249 print-ip.c
1250 print-ip6.c
1251 print-ip6opts.c
1252 print-ipcomp.c
1253 print-ipfc.c
1254 print-ipnet.c
1255 print-ipoib.c
1256 print-ipx.c
1257 print-isakmp.c
1258 print-isoclns.c
1259 print-juniper.c
1260 print-krb.c
1261 print-l2tp.c
1262 print-lane.c
1263 print-ldp.c
1264 print-lisp.c
1265 print-llc.c
1266 print-lldp.c
1267 print-lmp.c
1268 print-loopback.c
1269 print-lspping.c
1270 print-lwapp.c
1271 print-lwres.c
1272 print-m3ua.c
1273 print-macsec.c
1274 print-mobile.c
1275 print-mobility.c
1276 print-mpcp.c
1277 print-mpls.c
1278 print-mptcp.c
1279 print-msdp.c
1280 print-msnlb.c
1281 print-nflog.c
1282 print-nfs.c
1283 print-nhrp.c
1284 print-nsh.c
1285 print-ntp.c
1286 print-null.c
1287 print-olsr.c
1288 print-openflow-1.0.c
1289 print-openflow-1.3.c
1290 print-openflow.c
1291 print-ospf.c
1292 print-ospf6.c
1293 print-otv.c
1294 print-pflog.c
1295 print-pgm.c
1296 print-pim.c
1297 print-pktap.c
1298 print-ppi.c
1299 print-ppp.c
1300 print-pppoe.c
1301 print-pptp.c
1302 print-ptp.c
1303 print-quic.c
1304 print-radius.c
1305 print-raw.c
1306 print-realtek.c
1307 print-resp.c
1308 print-rip.c
1309 print-ripng.c
1310 print-rpki-rtr.c
1311 print-rsvp.c
1312 print-rt6.c
1313 print-rtsp.c
1314 print-rx.c
1315 print-sctp.c
1316 print-sflow.c
1317 print-sip.c
1318 print-sl.c
1319 print-sll.c
1320 print-slow.c
1321 print-smtp.c
1322 print-snmp.c
1323 print-someip.c
1324 print-ssh.c
1325 print-stp.c
1326 print-sunatm.c
1327 print-sunrpc.c
1328 print-symantec.c
1329 print-syslog.c
1330 print-tcp.c
1331 print-telnet.c
1332 print-tftp.c
1333 print-timed.c
1334 print-tipc.c
1335 print-token.c
1336 print-udld.c
1337 print-udp.c
1338 print-unsupported.c
1339 print-usb.c
1340 print-vjc.c
1341 print-vqp.c
1342 print-vrrp.c
1343 print-vsock.c
1344 print-vtp.c
1345 print-vxlan-gpe.c
1346 print-vxlan.c
1347 print-wb.c
1348 print-whois.c
1349 print-zep.c
1350 print-zephyr.c
1351 print-zeromq.c
1352 ${LOCALSRC}
1353 signature.c
1354 strtoaddr.c
1355 util-print.c
1356 )
1357
1358 #
1359 # Replace missing functions
1360 #
1361 foreach(FUNC strlcat strlcpy strsep getservent getopt_long)
1362 string(TOUPPER ${FUNC} FUNC_UPPERCASE)
1363 set(HAVE_FUNC_UPPERCASE HAVE_${FUNC_UPPERCASE})
1364 if(NOT ${HAVE_FUNC_UPPERCASE})
1365 set(NETDISSECT_SOURCE_LIST_C ${NETDISSECT_SOURCE_LIST_C} missing/${FUNC}.c)
1366 endif()
1367 endforeach()
1368
1369 add_library(netdissect STATIC
1370 ${NETDISSECT_SOURCE_LIST_C}
1371 )
1372 if(NOT C_ADDITIONAL_FLAGS STREQUAL "")
1373 set_target_properties(netdissect PROPERTIES COMPILE_FLAGS ${C_ADDITIONAL_FLAGS})
1374 endif()
1375
1376 set(TCPDUMP_SOURCE_LIST_C fptype.c tcpdump.c)
1377
1378 if(NOT HAVE_BPF_DUMP)
1379 set(TCPDUMP_SOURCE_LIST_C ${TCPDUMP_SOURCE_LIST_C} bpf_dump.c)
1380 endif(NOT HAVE_BPF_DUMP)
1381
1382 set(PROJECT_SOURCE_LIST_C ${NETDISSECT_SOURCE_LIST_C} ${TCPDUMP_SOURCE_LIST_C})
1383
1384 file(GLOB PROJECT_SOURCE_LIST_H
1385 *.h
1386 )
1387
1388 #
1389 # Assume, by default, no support for shared libraries and V7/BSD
1390 # convention for man pages (devices in section 4, file formats in
1391 # section 5, miscellaneous info in section 7, administrative commands
1392 # and daemons in section 8). Individual cases can override this.
1393 # Individual cases can override this.
1394 #
1395 set(MAN_FILE_FORMATS 5)
1396 set(MAN_MISC_INFO 7)
1397 if(CMAKE_SYSTEM_NAME STREQUAL "HP-UX")
1398 #
1399 # Use System V conventions for man pages.
1400 #
1401 set(MAN_FILE_FORMATS 4)
1402 set(MAN_MISC_INFO 5)
1403 elseif(CMAKE_SYSTEM_NAME STREQUAL "SunOS" AND CMAKE_SYSTEM_VERSION MATCHES "5[.][0-9.]*")
1404 #
1405 # SunOS 5.x.
1406 #
1407 if(CMAKE_SYSTEM_VERSION STREQUAL "5.12")
1408 else()
1409 #
1410 # Use System V conventions for man pages.
1411 #
1412 set(MAN_FILE_FORMATS 4)
1413 set(MAN_MISC_INFO 5)
1414 endif()
1415 endif()
1416
1417 source_group("Source Files" FILES ${PROJECT_SOURCE_LIST_C})
1418 source_group("Header Files" FILES ${PROJECT_SOURCE_LIST_H})
1419
1420 ######################################
1421 # Register targets
1422 ######################################
1423
1424 add_executable(tcpdump ${TCPDUMP_SOURCE_LIST_C})
1425 if(NOT C_ADDITIONAL_FLAGS STREQUAL "")
1426 set_target_properties(tcpdump PROPERTIES COMPILE_FLAGS ${C_ADDITIONAL_FLAGS})
1427 endif()
1428 if(NOT "${PCAP_LINK_FLAGS}" STREQUAL "")
1429 set_target_properties(tcpdump PROPERTIES LINK_FLAGS ${PCAP_LINK_FLAGS})
1430 endif()
1431 target_link_libraries(tcpdump netdissect ${TCPDUMP_LINK_LIBRARIES})
1432
1433 ######################################
1434 # Write out the config.h file
1435 ######################################
1436
1437 configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmakeconfig.h.in ${CMAKE_CURRENT_BINARY_DIR}/config.h)
1438
1439 ######################################
1440 # Install tcpdump and man pages
1441 ######################################
1442
1443 #
1444 # "Define GNU standard installation directories", which actually
1445 # are also defined, to some degree, by autotools, and at least
1446 # some of which are general UN*X conventions.
1447 #
1448 include(GNUInstallDirs)
1449
1450 set(MAN1_EXPAND tcpdump.1.in)
1451
1452 if(WIN32)
1453 # XXX TODO where to install on Windows?
1454 else(WIN32)
1455 install(TARGETS tcpdump DESTINATION bin)
1456 endif(WIN32)
1457
1458 # On UN*X, and on Windows when not using MSVC, process man pages and
1459 # arrange that they be installed.
1460 if(NOT MSVC)
1461 #
1462 # Man pages.
1463 #
1464 # For each section of the manual for which we have man pages
1465 # that require macro expansion, do the expansion.
1466 #
1467 set(MAN1 "")
1468 foreach(TEMPLATE_MANPAGE ${MAN1_EXPAND})
1469 string(REPLACE ".in" "" MANPAGE ${TEMPLATE_MANPAGE})
1470 configure_file(${CMAKE_SOURCE_DIR}/${TEMPLATE_MANPAGE} ${CMAKE_CURRENT_BINARY_DIR}/${MANPAGE} @ONLY)
1471 set(MAN1 ${MAN1} ${CMAKE_CURRENT_BINARY_DIR}/${MANPAGE})
1472 endforeach(TEMPLATE_MANPAGE)
1473 install(FILES ${MAN1} DESTINATION ${CMAKE_INSTALL_MANDIR}/man1)
1474 endif(NOT MSVC)
1475
1476 # uninstall target
1477 configure_file(
1478 "${CMAKE_CURRENT_SOURCE_DIR}/cmake_uninstall.cmake.in"
1479 "${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake"
1480 IMMEDIATE @ONLY)
1481
1482 add_custom_target(uninstall
1483 COMMAND ${CMAKE_COMMAND} -P ${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake)
1484
1485 #
1486 # tcpdump tests
1487 # We try to find the Perl interpreter and, if we do, we have the check
1488 # rule run tests/TESTrun with it, because just trying to run the TESTrun
1489 # script as a command won't work on Windows.
1490 #
1491 find_program(PERL perl)
1492 if(PERL)
1493 message(STATUS "Found perl at ${PERL}")
1494 add_custom_target(check
1495 COMMAND ${PERL} ${CMAKE_SOURCE_DIR}/tests/TESTrun)
1496 else()
1497 message(STATUS "Didn't find perl")
1498 endif()