mirror of
https://github.com/the-tcpdump-group/tcpdump.git
synced 2024-11-23 18:14:29 +08:00
89fc9f2161
See https://npcap.com/guide/npcap-devguide.html#npcap-feature-native-dll-implicitly for details on what's being done. Fix #1226.
1492 lines
49 KiB
CMake
1492 lines
49 KiB
CMake
if(WIN32)
|
|
#
|
|
# We need 3.12 or later, so that we can set policy CMP0074; see
|
|
# below.
|
|
#
|
|
cmake_minimum_required(VERSION 3.12)
|
|
else(WIN32)
|
|
#
|
|
# For now:
|
|
#
|
|
# if this is a version of CMake less than 3.5, require only
|
|
# 2.8.12, just in case somebody is configuring with CMake
|
|
# on a "long-term support" version # of some OS and that
|
|
# version supplies an older version of CMake;
|
|
#
|
|
# otherwise, require 3.5, so we don't get messages warning
|
|
# that support for versions of CMake lower than 3.5 is
|
|
# deprecated.
|
|
#
|
|
if(CMAKE_VERSION VERSION_LESS "3.5")
|
|
cmake_minimum_required(VERSION 2.8.12)
|
|
else()
|
|
cmake_minimum_required(VERSION 3.5)
|
|
endif()
|
|
endif(WIN32)
|
|
|
|
#
|
|
# We want find_path() and find_library() to honor {packagename}_ROOT,
|
|
# as that appears to be the standard way to say "hey, look here for
|
|
# this package" from the command line.
|
|
#
|
|
if(POLICY CMP0074)
|
|
cmake_policy(SET CMP0074 NEW)
|
|
endif()
|
|
|
|
#
|
|
# OK, this is a pain.
|
|
#
|
|
# When building on NetBSD, with a libpcap installed from pkgsrc,
|
|
# a -Wl,-rpath,/usr/pkg/lib option is added to the options when
|
|
# linking tcpdump. This puts /usr/pkg/lib into the run-time path.
|
|
#
|
|
# However, by default, CMake adds a rule to the install CMake script
|
|
# a CMake command (using an undocumented subcommand of file()) that
|
|
# strips /usr/pkg/lib *out* of the run-time path; the message in the
|
|
# output for the "install" target is
|
|
#
|
|
# -- Set runtime path of "{target-directory}/tcpdump" to ""
|
|
#
|
|
# I am not certain what the rationale is for doing this, but a
|
|
# *consequence* of this is that, when you run the installed tcpdump,
|
|
# it fails to find libpcap.so:
|
|
#
|
|
# $ {target-directory}/tcpdump -h
|
|
# {target-directory}/tcpdump: Shared object "libpcap.so.0" not found
|
|
#
|
|
# It also appears to be the case that, on Ubuntu 22.04, FreeBSD 12,
|
|
# DragonFly BSD 5.8, OpenBSD 6.6, and Solaris 11.4,
|
|
#
|
|
# On Ubuntu and Solaris, even if you have a libpcap in /usr/local, you
|
|
# have to provide not only -I/usr/local/include and -L/usr/local/lib,
|
|
# you also must provide -Wl,-rpath,/usr/local/lib in order to have
|
|
# the run-time linker look in /usr/local/lib for libpcap. If it's not
|
|
# specified, then, if the shared library major version number of the
|
|
# libpcap in /usr/lib is the same as the shared major version number
|
|
# of the libpcap in /usr/local/lib, the run-time linker will find the
|
|
# libpcap in /usr/lib; if the versions are different, the run-time
|
|
# linker will fail to find the libpcap in /usr/lib, so the program will
|
|
# fail to run.
|
|
#
|
|
# We suppress this by setting CMAKE_INSTALL_RPATH_USE_LINK_PATH to TRUE;
|
|
# as the documentation for that variable says:
|
|
#
|
|
# Add paths to linker search and installed rpath.
|
|
#
|
|
# CMAKE_INSTALL_RPATH_USE_LINK_PATH is a boolean that if set to True
|
|
# will append to the runtime search path (rpath) of installed
|
|
# binaries any directories outside the project that are in the linker
|
|
# search path or contain linked library files. The directories are
|
|
# appended after the value of the INSTALL_RPATH target property.
|
|
#
|
|
# If, for whatever reason, directories in which we search for external
|
|
# libraries, other than the standard system library directories, are
|
|
# added to the executable's rpath in the build process, we most
|
|
# definitely want them in the installed image's rpath if they are
|
|
# necessary in order to find the libraries at run time.
|
|
#
|
|
set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE)
|
|
|
|
set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake/Modules)
|
|
|
|
#
|
|
# We explicitly indicate what languages are used in tcpdump to avoid
|
|
# checking for a C++ compiler.
|
|
#
|
|
# One reason to avoid that check is that there's no need to waste
|
|
# configuration time performing it.
|
|
#
|
|
# Another reason is that:
|
|
#
|
|
# CMake will try to determine the sizes of some data types, including
|
|
# void *, early in the process of configuration; apparently, it's done
|
|
# as part of processing the project() command.
|
|
#
|
|
# At least as of CMake 2.8.6, it does so by checking the size of
|
|
# "void *" in C, setting CMAKE_C_SIZEOF_DATA_PTR based on that,
|
|
# setting CMAKE_SIZEOF_VOID_P to that, and then checking the size
|
|
# of "void *" in C++, setting CMAKE_CXX_SIZEOF_DATA_PTR based on
|
|
# that, and then setting CMAKE_SIZEOF_VOID_P to *that*.
|
|
#
|
|
# The compile tests include whatever C flags may have been provided
|
|
# to CMake in the CFLAGS and CXXFLAGS environment variables.
|
|
#
|
|
# If you set an architecture flag such as -m32 or -m64 in CFLAGS
|
|
# but *not* in CXXFLAGS, the size for C++ will win, and hilarity
|
|
# will ensue.
|
|
#
|
|
# Or if, at least on Solaris, you have a newer version of GCC
|
|
# installed, but *not* a newer version of G++, and you have Oracle
|
|
# Studio installed, it will find GCC, which will default to building
|
|
# 64-bit, and Oracle Studio's C++ compiler, which will default to
|
|
# building 32-bit, the size for C++ will win, and, again, hilarity
|
|
# will ensue.
|
|
#
|
|
project(tcpdump C)
|
|
|
|
#
|
|
# Export the size of void * as SIZEOF_VOID_P so that it can be
|
|
# tested with #if.
|
|
#
|
|
set(SIZEOF_VOID_P "${CMAKE_SIZEOF_VOID_P}")
|
|
|
|
#
|
|
# Show the bit width for which we're compiling.
|
|
# This can help debug problems if you're dealing with a compiler that
|
|
# defaults to generating 32-bit code even when running on a 64-bit
|
|
# platform, and where that platform may provide only 64-bit versions of
|
|
# libraries that we might use (looking at *you*, Oracle Studio!).
|
|
#
|
|
if(CMAKE_SIZEOF_VOID_P EQUAL 4)
|
|
message(STATUS "Building 32-bit")
|
|
elseif(CMAKE_SIZEOF_VOID_P EQUAL 8)
|
|
message(STATUS "Building 64-bit")
|
|
endif()
|
|
|
|
#
|
|
# Solaris pkg-config is annoying. For at least one package (D-Bus, I'm
|
|
# looking at *you*!), there are separate include files for 32-bit and
|
|
# 64-bit builds (I guess using "unsigned long long" as a 64-bit integer
|
|
# type on a 64-bit build is like crossing the beams or something), and
|
|
# there are two separate .pc files, so if we're doing a 32-bit build we
|
|
# should make sure we look in /usr/lib/pkgconfig for .pc files and if
|
|
# we're doing a 64-bit build we should make sure we look in
|
|
# /usr/lib/amd64/pkgconfig for .pc files.
|
|
#
|
|
if(CMAKE_SYSTEM_NAME STREQUAL "SunOS" AND CMAKE_SYSTEM_VERSION MATCHES "5[.][0-9.]*")
|
|
#
|
|
# Note: string(REPLACE) does not appear to support using ENV{...}
|
|
# as an argument, so we set a variable and then use set() to set
|
|
# the environment variable.
|
|
#
|
|
if(CMAKE_SIZEOF_VOID_P EQUAL 8)
|
|
#
|
|
# 64-bit build. If /usr/lib/pkgconfig appears in the path,
|
|
# prepend /usr/lib/amd64/pkgconfig to it; otherwise,
|
|
# put /usr/lib/amd64 at the end.
|
|
#
|
|
if((NOT DEFINED ENV{PKG_CONFIG_PATH}) OR "$ENV{PKG_CONFIG_PATH}" EQUAL "")
|
|
#
|
|
# Not set, or empty. Set it to /usr/lib/amd64/pkgconfig.
|
|
#
|
|
set(fixed_path "/usr/lib/amd64/pkgconfig")
|
|
elseif("$ENV{PKG_CONFIG_PATH}" MATCHES "/usr/lib/pkgconfig")
|
|
#
|
|
# It contains /usr/lib/pkgconfig. Prepend
|
|
# /usr/lib/amd64/pkgconfig to /usr/lib/pkgconfig.
|
|
#
|
|
string(REPLACE "/usr/lib/pkgconfig"
|
|
"/usr/lib/amd64/pkgconfig:/usr/lib/pkgconfig"
|
|
fixed_path "$ENV{PKG_CONFIG_PATH}")
|
|
else()
|
|
#
|
|
# Not empty, but doesn't contain /usr/lib/pkgconfig.
|
|
# Append /usr/lib/amd64/pkgconfig to it.
|
|
#
|
|
set(fixed_path "$ENV{PKG_CONFIG_PATH}:/usr/lib/amd64/pkgconfig")
|
|
endif()
|
|
set(ENV{PKG_CONFIG_PATH} "${fixed_path}")
|
|
elseif(CMAKE_SIZEOF_VOID_P EQUAL 4)
|
|
#
|
|
# 32-bit build. If /usr/amd64/lib/pkgconfig appears in the path,
|
|
# prepend /usr/lib/pkgconfig to it.
|
|
#
|
|
if("$ENV{PKG_CONFIG_PATH}" MATCHES "/usr/lib/amd64/pkgconfig")
|
|
#
|
|
# It contains /usr/lib/amd64/pkgconfig. Prepend
|
|
# /usr/lib/pkgconfig to /usr/lib/amd64/pkgconfig.
|
|
#
|
|
string(REPLACE "/usr/lib/amd64/pkgconfig"
|
|
"/usr/lib/pkgconfig:/usr/lib/amd64/pkgconfig"
|
|
fixed_path "$ENV{PKG_CONFIG_PATH}")
|
|
set(ENV{PKG_CONFIG_PATH} "${fixed_path}")
|
|
endif()
|
|
endif()
|
|
endif()
|
|
|
|
#
|
|
# For checking if a compiler flag works and adding it if it does.
|
|
#
|
|
include(CheckCCompilerFlag)
|
|
macro(check_and_add_compiler_option _option)
|
|
message(STATUS "Checking C compiler flag ${_option}")
|
|
string(REPLACE "=" "-" _temp_option_variable ${_option})
|
|
string(REGEX REPLACE "^-" "" _option_variable ${_temp_option_variable})
|
|
check_c_compiler_flag("${_option}" ${_option_variable})
|
|
if(${${_option_variable}})
|
|
set(C_ADDITIONAL_FLAGS "${C_ADDITIONAL_FLAGS} ${_option}")
|
|
endif()
|
|
endmacro()
|
|
|
|
#
|
|
# If we're building with Visual Studio, we require Visual Studio 2015,
|
|
# in order to get sufficient C99 compatibility. Check for that.
|
|
#
|
|
# If not, try the appropriate flag for the compiler to enable C99
|
|
# features.
|
|
#
|
|
set(C_ADDITIONAL_FLAGS "")
|
|
if(MSVC)
|
|
if(MSVC_VERSION LESS 1900)
|
|
message(FATAL_ERROR "Visual Studio 2015 or later is required")
|
|
endif()
|
|
|
|
#
|
|
# Treat source files as being in UTF-8 with MSVC if it's not using
|
|
# the Clang front end.
|
|
# We assume that UTF-8 source is OK with other compilers and with
|
|
# MSVC if it's using the Clang front end.
|
|
#
|
|
if(NOT ${CMAKE_C_COMPILER} MATCHES "clang*")
|
|
set(C_ADDITIONAL_FLAGS "${C_ADDITIONAL_FLAGS} /utf-8")
|
|
endif(NOT ${CMAKE_C_COMPILER} MATCHES "clang*")
|
|
else(MSVC)
|
|
#
|
|
# Try to enable as many C99 features as we can.
|
|
# At minimum, we want C++/C99-style // comments.
|
|
#
|
|
# Newer versions of compilers might default to supporting C99, but
|
|
# older versions may require a special flag.
|
|
#
|
|
# Prior to CMake 3.1, setting CMAKE_C_STANDARD will not have any effect,
|
|
# so, unless and until we require CMake 3.1 or later, we have to do it
|
|
# ourselves on pre-3.1 CMake, so we just do it ourselves on all versions
|
|
# of CMake.
|
|
#
|
|
# Note: with CMake 3.1 through 3.5, the only compilers for which CMake
|
|
# handles CMAKE_C_STANDARD are GCC and Clang. 3.6 adds support only
|
|
# for Intel C; 3.9 adds support for PGI C, Sun C, and IBM XL C, and
|
|
# 3.10 adds support for Cray C and IAR C, but no version of CMake has
|
|
# support for HP C. Therefore, even if we use CMAKE_C_STANDARD with
|
|
# compilers for which CMake supports it, we may still have to do it
|
|
# ourselves on other compilers.
|
|
#
|
|
# See the CMake documentation for the CMAKE_<LANG>_COMPILER_ID variables
|
|
# for a list of compiler IDs.
|
|
#
|
|
# XXX - this just tests whether the option works and adds it if it does.
|
|
# We don't test whether it's necessary in order to get the C99 features
|
|
# that we use; if we ever have a user who tries to compile with a compiler
|
|
# that can't be made to support those features, we can add a test to make
|
|
# sure we actually *have* C99 support.
|
|
#
|
|
if(CMAKE_C_COMPILER_ID MATCHES "GNU" OR
|
|
CMAKE_C_COMPILER_ID MATCHES "Clang")
|
|
check_and_add_compiler_option("-std=gnu99")
|
|
elseif(CMAKE_C_COMPILER_ID MATCHES "XL")
|
|
#
|
|
# We want support for extensions picked up for GNU C compatibility,
|
|
# so we use -qlanglvl=extc99.
|
|
#
|
|
check_and_add_compiler_option("-qlanglvl=extc99")
|
|
elseif(CMAKE_C_COMPILER_ID MATCHES "HP")
|
|
check_and_add_compiler_option("-AC99")
|
|
elseif(CMAKE_C_COMPILER_ID MATCHES "Sun")
|
|
check_and_add_compiler_option("-xc99")
|
|
elseif(CMAKE_C_COMPILER_ID MATCHES "Intel")
|
|
check_and_add_compiler_option("-c99")
|
|
endif()
|
|
endif(MSVC)
|
|
|
|
set(LIBRARY_NAME netdissect)
|
|
|
|
###################################################################
|
|
# Parameters
|
|
###################################################################
|
|
|
|
option(WITH_SMI "Build with libsmi, if available" ON)
|
|
option(WITH_CRYPTO "Build with OpenSSL/libressl libcrypto, if available" ON)
|
|
option(WITH_CAPSICUM "Build with Capsicum security functions, if available" ON)
|
|
option(WITH_CAP_NG "Use libcap-ng, if available" ON)
|
|
option(ENABLE_SMB "Build with the SMB dissector" OFF)
|
|
|
|
#
|
|
# String parameters. Neither of them are set, initially; only if the
|
|
# user explicitly configures them are they set.
|
|
#
|
|
# WITH_CHROOT is STRING, not PATH, as the directory need not exist
|
|
# when CMake is run.
|
|
#
|
|
set(WITH_CHROOT CACHE STRING
|
|
"Directory to which to chroot when dropping privileges")
|
|
set(WITH_USER CACHE STRING
|
|
"User to whom to set the UID when dropping privileges")
|
|
|
|
#
|
|
# By default, build universal with the appropriate set of architectures
|
|
# for the OS on which we're doing the build.
|
|
#
|
|
if(APPLE AND "${CMAKE_OSX_ARCHITECTURES}" STREQUAL "")
|
|
#
|
|
# Get the major version of Darwin.
|
|
#
|
|
string(REGEX MATCH "^([0-9]+)" SYSTEM_VERSION_MAJOR "${CMAKE_SYSTEM_VERSION}")
|
|
|
|
if(SYSTEM_VERSION_MAJOR EQUAL 9)
|
|
#
|
|
# Leopard. Build for x86 and 32-bit PowerPC, with
|
|
# x86 first. (That's what Apple does.)
|
|
#
|
|
set(CMAKE_OSX_ARCHITECTURES "i386;ppc")
|
|
elseif(SYSTEM_VERSION_MAJOR EQUAL 10)
|
|
#
|
|
# Snow Leopard. Build for x86-64 and x86, with
|
|
# x86-64 first. (That's what Apple does.)
|
|
#
|
|
set(CMAKE_OSX_ARCHITECTURES "x86_64;i386")
|
|
endif()
|
|
endif()
|
|
|
|
###################################################################
|
|
# Versioning
|
|
###################################################################
|
|
|
|
# Get, parse, format and set tcpdump's version string from
|
|
# [tcpdump_root]/VERSION for later use.
|
|
|
|
# Get MAJOR, MINOR, PATCH & SUFFIX
|
|
file(STRINGS ${tcpdump_SOURCE_DIR}/VERSION
|
|
PACKAGE_VERSION
|
|
LIMIT_COUNT 1 # Read only the first line
|
|
)
|
|
|
|
######################################
|
|
# Project settings
|
|
######################################
|
|
|
|
include_directories(
|
|
${CMAKE_CURRENT_BINARY_DIR}
|
|
${tcpdump_SOURCE_DIR}
|
|
)
|
|
|
|
if(MSVC)
|
|
add_definitions(-D__STDC__)
|
|
add_definitions(-D_CRT_SECURE_NO_WARNINGS)
|
|
endif(MSVC)
|
|
|
|
if(MSVC)
|
|
if (USE_STATIC_RT)
|
|
MESSAGE(STATUS "Use STATIC runtime")
|
|
set(NAME_RT MT)
|
|
set (CMAKE_CXX_FLAGS_MINSIZEREL "${CMAKE_CXX_FLAGS_MINSIZEREL} /MT")
|
|
set (CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} /MT")
|
|
set (CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MT")
|
|
set (CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd")
|
|
|
|
set (CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL} /MT")
|
|
set (CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO} /MT")
|
|
set (CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /MT")
|
|
set (CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /MTd")
|
|
else (USE_STATIC_RT)
|
|
MESSAGE(STATUS "Use DYNAMIC runtime")
|
|
set(NAME_RT MD)
|
|
set (CMAKE_CXX_FLAGS_MINSIZEREL "${CMAKE_CXX_FLAGS_MINSIZEREL} /MD")
|
|
set (CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} /MD")
|
|
set (CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MD")
|
|
set (CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MDd")
|
|
|
|
set (CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL} /MD")
|
|
set (CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO} /MD")
|
|
set (CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /MD")
|
|
set (CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /MDd")
|
|
endif (USE_STATIC_RT)
|
|
endif(MSVC)
|
|
|
|
#
|
|
# CMake's definition of "cross-compiling" appears to be "compiling
|
|
# for an *operating system* other than the one on which the build
|
|
# is being done*.
|
|
#
|
|
# This is an inadequate definition, as people build for the same
|
|
# operating system but a different instruction set, e.g. building
|
|
# on an IA-32 or x86-64 Linux box for an Arm embedded Linux box,
|
|
# or building Arm code on an IA-32 or x86-64 Windows box.
|
|
#
|
|
# At least for the Windows case, people may do those builds by
|
|
# setting the target with th -A flag to CMake; that causes
|
|
# CMAKE_GENERATOR_PLATFORM to be set to the target. If
|
|
# CMAKE_GENERATOR_PLATFORM is set, compare it with
|
|
# CMAKE_HOST_SYSTEM_PROCESSOR and, if they're not equal, set
|
|
# CMAKE_CROSSCOMPILING to TRUE.
|
|
#
|
|
if (CMAKE_GENERATOR_PLATFORM AND
|
|
NOT CMAKE_GENERATOR_PLATFORM STREQUAL CMAKE_HOST_SYSTEM_PROCESSOR)
|
|
set(CMAKE_CROSSCOMPILING TRUE)
|
|
endif()
|
|
|
|
###################################################################
|
|
# Detect available platform features
|
|
###################################################################
|
|
|
|
include(CMakePushCheckState)
|
|
include(CheckIncludeFile)
|
|
include(CheckIncludeFiles)
|
|
include(CheckFunctionExists)
|
|
include(CheckLibraryExists)
|
|
include(CheckSymbolExists)
|
|
include(CheckStructHasMember)
|
|
include(CheckVariableExists)
|
|
include(CheckTypeSize)
|
|
|
|
#
|
|
# Get the size of a time_t, to know whether it's 32-bit or 64-bit.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_EXTRA_INCLUDE_FILES time.h)
|
|
check_type_size("time_t" SIZEOF_TIME_T)
|
|
cmake_pop_check_state()
|
|
|
|
#
|
|
# Header files.
|
|
#
|
|
check_include_file(rpc/rpc.h HAVE_RPC_RPC_H)
|
|
if(HAVE_RPC_RPC_H)
|
|
check_include_files("rpc/rpc.h;rpc/rpcent.h" HAVE_RPC_RPCENT_H)
|
|
endif(HAVE_RPC_RPC_H)
|
|
|
|
#
|
|
# Functions.
|
|
#
|
|
check_function_exists(strlcat HAVE_STRLCAT)
|
|
check_function_exists(strlcpy HAVE_STRLCPY)
|
|
check_function_exists(strsep HAVE_STRSEP)
|
|
|
|
#
|
|
# Find library needed for gethostbyaddr.
|
|
# NOTE: if you hand check_library_exists as its last argument a variable
|
|
# that's been set, it skips the test, so we need different variables.
|
|
#
|
|
set(TCPDUMP_LINK_LIBRARIES "")
|
|
if(WIN32)
|
|
#
|
|
# We need winsock2.h and ws2tcpip.h.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES ws2_32)
|
|
check_symbol_exists(gethostbyaddr "winsock2.h;ws2tcpip.h" LIBWS2_32_HAS_GETHOSTBYADDR)
|
|
cmake_pop_check_state()
|
|
if(LIBWS2_32_HAS_GETHOSTBYADDR)
|
|
set(TCPDUMP_LINK_LIBRARIES ws2_32 ${TCPDUMP_LINK_LIBRARIES})
|
|
else(LIBWS2_32_HAS_GETHOSTBYADDR)
|
|
message(FATAL_ERROR "gethostbyaddr is required, but wasn't found")
|
|
endif(LIBWS2_32_HAS_GETHOSTBYADDR)
|
|
else(WIN32)
|
|
check_function_exists(gethostbyaddr STDLIBS_HAVE_GETHOSTBYADDR)
|
|
if(NOT STDLIBS_HAVE_GETHOSTBYADDR)
|
|
check_library_exists(socket gethostbyaddr "" LIBSOCKET_HAS_GETHOSTBYADDR)
|
|
if(LIBSOCKET_HAS_GETHOSTBYADDR)
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} socket)
|
|
else(LIBSOCKET_HAS_GETHOSTBYADDR)
|
|
check_library_exists(nsl gethostbyaddr "" LIBNSL_HAS_GETHOSTBYADDR)
|
|
if(LIBNSL_HAS_GETHOSTBYADDR)
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} nsl)
|
|
else(LIBNSL_HAS_GETHOSTBYADDR)
|
|
check_library_exists(network gethostbyaddr "" LIBNETWORK_HAS_GETHOSTBYADDR)
|
|
if(LIBNETWORK_HAS_GETHOSTBYADDR)
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} network)
|
|
else(LIBNETWORK_HAS_GETHOSTBYADDR)
|
|
message(FATAL_ERROR "gethostbyaddr is required, but wasn't found")
|
|
endif(LIBNETWORK_HAS_GETHOSTBYADDR)
|
|
endif(LIBNSL_HAS_GETHOSTBYADDR)
|
|
endif(LIBSOCKET_HAS_GETHOSTBYADDR)
|
|
endif(NOT STDLIBS_HAVE_GETHOSTBYADDR)
|
|
endif(WIN32)
|
|
|
|
#
|
|
# This may require additional libraries.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES})
|
|
check_function_exists(getservent STDLIBS_HAVE_GETSERVENT)
|
|
if(STDLIBS_HAVE_GETSERVENT)
|
|
set(HAVE_GETSERVENT TRUE)
|
|
else(STDLIBS_HAVE_GETSERVENT)
|
|
#
|
|
# Some platforms may need -lsocket for getservent.
|
|
#
|
|
set(CMAKE_REQUIRED_LIBRARIES socket ${TCPDUMP_LINK_LIBRARIES})
|
|
check_function_exists(getservent LIBSOCKET_HAS_GETSERVENT)
|
|
if(LIBSOCKET_HAS_GETSERVENT)
|
|
set(HAVE_GETSERVENT TRUE)
|
|
set(TCPDUMP_LINK_LIBRARIES socket ${TCPDUMP_LINK_LIBRARIES})
|
|
endif(LIBSOCKET_HAS_GETSERVENT)
|
|
endif(STDLIBS_HAVE_GETSERVENT)
|
|
cmake_pop_check_state()
|
|
|
|
if (NOT CMAKE_CROSSCOMPILING)
|
|
#
|
|
# Require a proof of suitable snprintf(3), same as in Autoconf.
|
|
#
|
|
include(CheckCSourceRuns)
|
|
check_c_source_runs("
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <inttypes.h>
|
|
#include <sys/types.h>
|
|
|
|
int main()
|
|
{
|
|
char buf[100];
|
|
uint64_t t = (uint64_t)1 << 32;
|
|
|
|
snprintf(buf, sizeof(buf), \"%zu\", sizeof(buf));
|
|
if (strncmp(buf, \"100\", sizeof(buf)))
|
|
return 1;
|
|
|
|
snprintf(buf, sizeof(buf), \"%zd\", -sizeof(buf));
|
|
if (strncmp(buf, \"-100\", sizeof(buf)))
|
|
return 2;
|
|
|
|
snprintf(buf, sizeof(buf), \"%\" PRId64, -t);
|
|
if (strncmp(buf, \"-4294967296\", sizeof(buf)))
|
|
return 3;
|
|
|
|
snprintf(buf, sizeof(buf), \"0o%\" PRIo64, t);
|
|
if (strncmp(buf, \"0o40000000000\", sizeof(buf)))
|
|
return 4;
|
|
|
|
snprintf(buf, sizeof(buf), \"0x%\" PRIx64, t);
|
|
if (strncmp(buf, \"0x100000000\", sizeof(buf)))
|
|
return 5;
|
|
|
|
snprintf(buf, sizeof(buf), \"%\" PRIu64, t);
|
|
if (strncmp(buf, \"4294967296\", sizeof(buf)))
|
|
return 6;
|
|
|
|
return 0;
|
|
}
|
|
|
|
"
|
|
SUITABLE_SNPRINTF
|
|
)
|
|
if(NOT SUITABLE_SNPRINTF)
|
|
message(FATAL_ERROR
|
|
"The snprintf(3) implementation in this libc is not suitable,
|
|
tcpdump would not work correctly even if it managed to compile."
|
|
)
|
|
endif()
|
|
else()
|
|
message(STATUS "Skipped SUITABLE_SNPRINTF because cross-compiling.")
|
|
endif()
|
|
|
|
check_function_exists(getopt_long HAVE_GETOPT_LONG)
|
|
#
|
|
# For Windows, don't need to waste time checking for fork() or vfork().
|
|
#
|
|
if(NOT WIN32)
|
|
check_function_exists(fork HAVE_FORK)
|
|
check_function_exists(vfork HAVE_VFORK)
|
|
endif(NOT WIN32)
|
|
|
|
#
|
|
# Some platforms may need -lnsl for getrpcbynumber.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES})
|
|
check_function_exists(getrpcbynumber STDLIBS_HAVE_GETRPCBYNUMBER)
|
|
if(STDLIBS_HAVE_GETRPCBYNUMBER)
|
|
set(HAVE_GETRPCBYNUMBER TRUE)
|
|
else(STDLIBS_HAVE_GETRPCBYNUMBER)
|
|
set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} nsl)
|
|
check_function_exists(getrpcbynumber LIBNSL_HAS_GETRPCBYNUMBER)
|
|
if(LIBNSL_HAS_GETRPCBYNUMBER)
|
|
set(HAVE_GETRPCBYNUMBER TRUE)
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} nsl)
|
|
endif(LIBNSL_HAS_GETRPCBYNUMBER)
|
|
endif(STDLIBS_HAVE_GETRPCBYNUMBER)
|
|
cmake_pop_check_state()
|
|
|
|
#
|
|
# This requires the libraries we require, as ether_ntohost might be
|
|
# in one of those libraries. That means we have to do this after
|
|
# we check for those libraries.
|
|
#
|
|
# You are in a twisty little maze of UN*Xes, all different.
|
|
# Some might not have ether_ntohost().
|
|
# Some might have it and declare it in <net/ethernet.h>.
|
|
# Some might have it and declare it in <netinet/ether.h>
|
|
# Some might have it and declare it in <sys/ethernet.h>.
|
|
# Some might have it and declare it in <arpa/inet.h>.
|
|
# Some might have it and declare it in <netinet/if_ether.h>.
|
|
# Some might have it and not declare it in any header file.
|
|
#
|
|
# Before you is a C compiler.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES ${TCPDUMP_LINK_LIBRARIES})
|
|
check_function_exists(ether_ntohost HAVE_ETHER_NTOHOST)
|
|
if(HAVE_ETHER_NTOHOST)
|
|
#
|
|
# OK, we have ether_ntohost(). We don't check whether it's buggy,
|
|
# as we assume any system that has CMake is likely to be new enough
|
|
# that, if it has ether_ntohost(), whatever bug is checked for in
|
|
# autotools is fixed; we just decide to use it.
|
|
#
|
|
set(USE_ETHER_NTOHOST TRUE)
|
|
|
|
#
|
|
# Is it declared in <net/ethernet.h>?
|
|
#
|
|
# This test fails if we don't have <net/ethernet.h> or if we do
|
|
# but it doesn't declare ether_ntohost().
|
|
#
|
|
check_symbol_exists(ether_ntohost net/ethernet.h NET_ETHERNET_H_DECLARES_ETHER_NTOHOST)
|
|
if(NET_ETHERNET_H_DECLARES_ETHER_NTOHOST)
|
|
#
|
|
# Yes - we have it declared.
|
|
#
|
|
set(HAVE_DECL_ETHER_NTOHOST TRUE)
|
|
endif()
|
|
#
|
|
# Did that succeed?
|
|
#
|
|
if(NOT HAVE_DECL_ETHER_NTOHOST)
|
|
#
|
|
# No - how about <netinet/ether.h>, as on Linux?
|
|
#
|
|
# This test fails if we don't have <netinet/ether.h>
|
|
# or if we do but it doesn't declare ether_ntohost().
|
|
#
|
|
check_symbol_exists(ether_ntohost netinet/ether.h NETINET_ETHER_H_DECLARES_ETHER_NTOHOST)
|
|
if(NETINET_ETHER_H_DECLARES_ETHER_NTOHOST)
|
|
#
|
|
# Yes - we have it declared.
|
|
#
|
|
set(HAVE_DECL_ETHER_NTOHOST TRUE)
|
|
endif()
|
|
endif()
|
|
#
|
|
# Did that succeed?
|
|
#
|
|
if(NOT HAVE_DECL_ETHER_NTOHOST)
|
|
#
|
|
# No - how about <sys/ethernet.h>, as on Solaris 10 and later?
|
|
#
|
|
# This test fails if we don't have <sys/ethernet.h>
|
|
# or if we do but it doesn't declare ether_ntohost().
|
|
#
|
|
check_symbol_exists(ether_ntohost sys/ethernet.h SYS_ETHERNET_H_DECLARES_ETHER_NTOHOST)
|
|
if(SYS_ETHERNET_H_DECLARES_ETHER_NTOHOST)
|
|
#
|
|
# Yes - we have it declared.
|
|
#
|
|
set(HAVE_DECL_ETHER_NTOHOST TRUE)
|
|
endif()
|
|
endif()
|
|
#
|
|
# Did that succeed?
|
|
#
|
|
if(NOT HAVE_DECL_ETHER_NTOHOST)
|
|
#
|
|
# No, how about <arpa/inet.h>, as on AIX?
|
|
#
|
|
# This test fails if we don't have <arpa/inet.h>
|
|
# or if we do but it doesn't declare ether_ntohost().
|
|
#
|
|
check_symbol_exists(ether_ntohost arpa/inet.h ARPA_INET_H_DECLARES_ETHER_NTOHOST)
|
|
if(ARPA_INET_H_DECLARES_ETHER_NTOHOST)
|
|
#
|
|
# Yes - we have it declared.
|
|
#
|
|
set(HAVE_DECL_ETHER_NTOHOST TRUE)
|
|
endif()
|
|
endif()
|
|
#
|
|
# Did that succeed?
|
|
#
|
|
if(NOT HAVE_DECL_ETHER_NTOHOST)
|
|
#
|
|
# No, how about <netinet/if_ether.h>?
|
|
# On some platforms, it requires <net/if.h> and
|
|
# <netinet/in.h>, and we always include it with
|
|
# both of them, so test it with both of them.
|
|
#
|
|
# This test fails if we don't have <netinet/if_ether.h>
|
|
# and the headers we include before it, or if we do but
|
|
# <netinet/if_ether.h> doesn't declare ether_ntohost().
|
|
#
|
|
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)
|
|
if(NETINET_IF_ETHER_H_DECLARES_ETHER_NTOHOST)
|
|
#
|
|
# Yes - we have it declared.
|
|
#
|
|
set(HAVE_DECL_ETHER_NTOHOST TRUE)
|
|
endif()
|
|
endif()
|
|
#
|
|
# After all that, is ether_ntohost() declared?
|
|
#
|
|
if(NOT HAVE_DECL_ETHER_NTOHOST)
|
|
#
|
|
# No, we'll have to declare it ourselves.
|
|
# Do we have "struct ether_addr" if we include<netinet/if_ether.h>?
|
|
#
|
|
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)
|
|
endif()
|
|
endif()
|
|
cmake_pop_check_state()
|
|
|
|
#
|
|
# Data types.
|
|
#
|
|
# XXX - there's no check_struct() macro that's like check_struct_has_member()
|
|
# except that it only checks for the existence of the structure type,
|
|
# so we use check_struct_has_member() and look for ss_family.
|
|
#
|
|
|
|
######################################
|
|
# External dependencies
|
|
######################################
|
|
|
|
#
|
|
# libpcap/WinPcap/Npcap.
|
|
# First, find it.
|
|
#
|
|
find_package(PCAP REQUIRED)
|
|
include_directories(${PCAP_INCLUDE_DIRS})
|
|
|
|
cmake_push_check_state()
|
|
|
|
#
|
|
# Now check headers.
|
|
#
|
|
set(CMAKE_REQUIRED_INCLUDES ${PCAP_INCLUDE_DIRS})
|
|
|
|
#
|
|
# Check whether we have pcap/pcap-inttypes.h.
|
|
# If we do, we use that to get the C99 types defined.
|
|
#
|
|
check_include_file(pcap/pcap-inttypes.h HAVE_PCAP_PCAP_INTTYPES_H)
|
|
|
|
#
|
|
# Check for various functions in libpcap/WinPcap/Npcap.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES ${PCAP_LIBRARIES})
|
|
|
|
#
|
|
# Do we have the new open API? Check for pcap_create() and for
|
|
# pcap_statustostr(), and assume that, if we have both of them,
|
|
# we also have pcap_activate() and the other new routines
|
|
# introduced in libpcap 1.0.
|
|
#
|
|
# We require those routines, so fail if we don't find it.
|
|
#
|
|
check_function_exists(pcap_create HAVE_PCAP_CREATE)
|
|
if(NOT HAVE_PCAP_CREATE)
|
|
if(WIN32)
|
|
MESSAGE(FATAL_ERROR "libpcap is too old; 1.0 or later is required")
|
|
else(WIN32)
|
|
MESSAGE(FATAL_ERROR "libpcap is too old; 1.0 or later is required - if you're using WinPcap, try Npcap")
|
|
endif(WIN32)
|
|
endif(NOT HAVE_PCAP_CREATE)
|
|
|
|
if(WIN32)
|
|
#
|
|
# We check for pcap_statustostr() as well, because WinPcap 4.1.3
|
|
# screwed up and exported pcap_create() but not other routines
|
|
# such as pcap_statustostr(), even though it defined them and
|
|
# even though you really want pcap_statustostr() to get strings
|
|
# corresponding to some of the status returns from the new routines.)
|
|
#
|
|
check_function_exists(pcap_statustostr HAVE_PCAP_STATUSTOSTR)
|
|
if(NOT HAVE_PCAP_STATUSTOSTR)
|
|
MESSAGE(FATAL_ERROR "pcap_create is available, but pcap_statustostr isn't - if you're using WinPcap, try Npcap ")
|
|
endif(NOT HAVE_PCAP_STATUSTOSTR)
|
|
endif(WIN32)
|
|
|
|
#
|
|
# OK, do we have pcap_set_tstamp_type? If so, assume we have
|
|
# pcap_list_tstamp_types and pcap_free_tstamp_types as well.
|
|
#
|
|
check_function_exists(pcap_set_tstamp_type HAVE_PCAP_SET_TSTAMP_TYPE)
|
|
|
|
#
|
|
# And do we have pcap_set_tstamp_precision? If so, we assume
|
|
# we also have pcap_open_offline_with_tstamp_precision.
|
|
#
|
|
check_function_exists(pcap_set_tstamp_precision HAVE_PCAP_SET_TSTAMP_PRECISION)
|
|
|
|
#
|
|
# Check for a miscellaneous collection of functions which we use
|
|
# if we have them.
|
|
#
|
|
check_function_exists(pcap_set_immediate_mode HAVE_PCAP_SET_IMMEDIATE_MODE)
|
|
check_function_exists(pcap_dump_ftell64 HAVE_PCAP_DUMP_FTELL64)
|
|
#
|
|
# macOS Sonoma's libpcap includes stub versions of the remote-
|
|
# capture APIs. They are exported as "weakly linked symbols".
|
|
#
|
|
# Xcode 15 offers only a macOS Sonoma SDK, which has a .tbd
|
|
# file for libpcap that claims it includes those APIs. (Newer
|
|
# versions of macOS don't provide the system shared libraries,
|
|
# they only provide the dyld shared cache containing those
|
|
# libraries, so the OS provides SDKs that include a .tbd file
|
|
# to use when linking.)
|
|
#
|
|
# This means that check_function_exists() will think that
|
|
# the remote-capture APIs are present, including pcap_open()
|
|
# and pcap_findalldevs_ex().
|
|
#
|
|
# However, they are *not* present in macOS Ventura and earlier,
|
|
# which means that building on Ventura with Xcode 15 produces
|
|
# executables that fail to start because one of those APIs
|
|
# isn't found in the system libpcap.
|
|
#
|
|
# Protecting calls to those APIs with __builtin_available()
|
|
# does not prevent this, because the libpcap header files
|
|
# in the Sonoma SDK mark them as being first available
|
|
# in macOS 10.13, just like all the other routines introduced
|
|
# in libpcap 1.9, even though they're only available if libpcap
|
|
# is built with remote capture enabled or stub routines are
|
|
# provided. (A fix to enable this has been checked into the
|
|
# libpcap repository, and may end up in a later version of
|
|
# the SDK.)
|
|
#
|
|
# Given all that, and given that the versions of the
|
|
# remote-capture APIs in Sonoma are stubs that always fail,
|
|
# there doesn't seem to be any point in checking for pcap_open()
|
|
# and pcap_findalldevs_ex() if we're linking against the Apple libpcap.
|
|
#
|
|
# However, if we're *not* linking against the Apple libpcap,
|
|
# we should check for it, so that we can use it if it's present.
|
|
#
|
|
# So we check for pcap_open() and pcap_findalldevs_ex() if 1) this isn't
|
|
# macOS or 2) the the libpcap we found is not a system library, meaning
|
|
# that its path begins neither with /usr/lib (meaning it's a system
|
|
# dylib) nor /Application/Xcode.app (meaning it's a file in
|
|
# the Xcode SDK).
|
|
#
|
|
if(NOT APPLE OR NOT
|
|
(PCAP_LIBRARIES MATCHES "/usr/lib/.*" OR
|
|
PCAP_LIBRARIES MATCHES "/Application/Xcode.app/.*"))
|
|
check_function_exists(pcap_open HAVE_PCAP_OPEN)
|
|
check_function_exists(pcap_findalldevs_ex HAVE_PCAP_FINDALLDEVS_EX)
|
|
endif()
|
|
|
|
#
|
|
# On Windows, check for pcap_wsockinit(); if we don't have it, check for
|
|
# wsockinit().
|
|
#
|
|
if(WIN32)
|
|
check_function_exists(pcap_wsockinit HAVE_PCAP_WSOCKINIT)
|
|
if(NOT HAVE_PCAP_WSOCKINIT)
|
|
check_function_exists(wsockinit HAVE_WSOCKINIT)
|
|
endif(NOT HAVE_PCAP_WSOCKINIT)
|
|
endif(WIN32)
|
|
|
|
#
|
|
# Check for special debugging functions
|
|
#
|
|
check_function_exists(pcap_set_parser_debug HAVE_PCAP_SET_PARSER_DEBUG)
|
|
if(NOT HAVE_PCAP_SET_PARSER_DEBUG)
|
|
# Check whether libpcap defines pcap_debug or yydebug
|
|
check_variable_exists(pcap_debug HAVE_PCAP_DEBUG)
|
|
if(NOT HAVE_PCAP_DEBUG)
|
|
check_variable_exists(yydebug HAVE_YYDEBUG)
|
|
endif(NOT HAVE_PCAP_DEBUG)
|
|
endif(NOT HAVE_PCAP_SET_PARSER_DEBUG)
|
|
|
|
check_function_exists(pcap_set_optimizer_debug HAVE_PCAP_SET_OPTIMIZER_DEBUG)
|
|
|
|
#
|
|
# bpf_dump() moved from tcpdump to libpcap in libpcap 0.6, but not all
|
|
# versions of libpcap didn't pick that change up; the OpenBSD libpcap
|
|
# picked up most of the new APIs from versions up to 1.0, but didn't
|
|
# pick up bpf_dump(), so we need to provide it if it's absent.
|
|
#
|
|
check_function_exists(bpf_dump HAVE_BPF_DUMP)
|
|
|
|
cmake_pop_check_state()
|
|
cmake_pop_check_state()
|
|
|
|
#
|
|
# We have libpcap.
|
|
#
|
|
include_directories(SYSTEM ${PCAP_INCLUDE_DIRS})
|
|
set(TCPDUMP_LINK_LIBRARIES ${PCAP_LIBRARIES} ${TCPDUMP_LINK_LIBRARIES})
|
|
|
|
#
|
|
# Optional libraries.
|
|
#
|
|
|
|
#
|
|
# libsmi.
|
|
#
|
|
if(WITH_SMI)
|
|
find_package(SMI)
|
|
if(SMI_FOUND)
|
|
include_directories(SYSTEM ${SMI_INCLUDE_DIRS})
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} ${SMI_LIBRARIES})
|
|
set(USE_LIBSMI ON)
|
|
endif(SMI_FOUND)
|
|
endif(WITH_SMI)
|
|
|
|
#
|
|
# OpenSSL/libressl libcrypto.
|
|
#
|
|
if(WITH_CRYPTO)
|
|
find_package(CRYPTO)
|
|
if(CRYPTO_FOUND)
|
|
#
|
|
# 1) do we have EVP_CIPHER_CTX_new?
|
|
# If so, we use it to allocate an EVP_CIPHER_CTX, as
|
|
# EVP_CIPHER_CTX may be opaque; otherwise, we allocate
|
|
# it ourselves.
|
|
#
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES "${CRYPTO_LIBRARIES}")
|
|
|
|
check_function_exists(EVP_CIPHER_CTX_new HAVE_EVP_CIPHER_CTX_NEW)
|
|
|
|
#
|
|
# 2) do we have EVP_DecryptInit_ex()?
|
|
# If so, we use it, because we need to be able to make two
|
|
# "initialize the cipher" calls, one with the cipher and key,
|
|
# and one with the IV, and, as of OpenSSL 1.1, You Can't Do That
|
|
# with EVP_DecryptInit(), because a call to EVP_DecryptInit() will
|
|
# unconditionally clear the context, and if you don't supply a
|
|
# cipher, it'll clear the cipher, rendering the context unusable
|
|
# and causing a crash.
|
|
#
|
|
check_function_exists(EVP_DecryptInit_ex HAVE_EVP_DECRYPTINIT_EX)
|
|
|
|
cmake_pop_check_state()
|
|
|
|
#
|
|
# We have libcrypto.
|
|
#
|
|
include_directories(SYSTEM ${CRYPTO_INCLUDE_DIRS})
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} ${CRYPTO_LIBRARIES})
|
|
set(HAVE_LIBCRYPTO ON)
|
|
endif(CRYPTO_FOUND)
|
|
endif(WITH_CRYPTO)
|
|
|
|
#
|
|
# Capsicum sandboxing.
|
|
# Some of this is in the system library, some of it is in other libraries.
|
|
#
|
|
if(WITH_CAPSICUM)
|
|
check_include_files("sys/capsicum.h" HAVE_SYS_CAPSICUM_H)
|
|
if(HAVE_SYS_CAPSICUM_H)
|
|
check_function_exists(cap_enter HAVE_CAP_ENTER)
|
|
check_function_exists(cap_rights_limit HAVE_CAP_RIGHTS_LIMIT)
|
|
check_function_exists(cap_ioctls_limit HAVE_CAP_IOCTLS_LIMIT)
|
|
check_function_exists(openat HAVE_OPENAT)
|
|
if(HAVE_CAP_ENTER AND HAVE_CAP_RIGHTS_LIMIT AND
|
|
HAVE_CAP_IOCTLS_LIMIT AND HAVE_OPENAT)
|
|
#
|
|
# OK, we have the functions we need to support Capsicum.
|
|
#
|
|
set(HAVE_CAPSICUM TRUE)
|
|
|
|
#
|
|
# OK, can we use Casper?
|
|
#
|
|
check_library_exists(casper cap_init "" HAVE_CAP_INIT)
|
|
if(HAVE_CAP_INIT)
|
|
cmake_push_check_state()
|
|
set(CMAKE_REQUIRED_LIBRARIES casper)
|
|
check_library_exists(cap_dns cap_gethostbyaddr "" HAVE_CAP_GETHOSTBYADDR)
|
|
cmake_pop_check_state()
|
|
if(HAVE_CAP_GETHOSTBYADDR)
|
|
set(HAVE_CASPER TRUE)
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} casper cap_dns)
|
|
endif(HAVE_CAP_GETHOSTBYADDR)
|
|
endif(HAVE_CAP_INIT)
|
|
endif(HAVE_CAP_ENTER AND HAVE_CAP_RIGHTS_LIMIT AND
|
|
HAVE_CAP_IOCTLS_LIMIT AND HAVE_OPENAT)
|
|
endif(HAVE_SYS_CAPSICUM_H)
|
|
endif(WITH_CAPSICUM)
|
|
|
|
#
|
|
# libcap-ng.
|
|
#
|
|
if(WITH_CAP_NG)
|
|
check_include_file(cap-ng.h HAVE_CAP_NG_H)
|
|
check_library_exists(cap-ng capng_change_id "" HAVE_LIBCAP_NG)
|
|
if(HAVE_LIBCAP_NG)
|
|
set(TCPDUMP_LINK_LIBRARIES ${TCPDUMP_LINK_LIBRARIES} cap-ng)
|
|
endif(HAVE_LIBCAP_NG)
|
|
endif(WITH_CAP_NG)
|
|
|
|
###################################################################
|
|
# Warning options
|
|
###################################################################
|
|
|
|
#
|
|
# Check and add warning options if we have a .devel file.
|
|
#
|
|
if(EXISTS ${CMAKE_SOURCE_DIR}/.devel OR EXISTS ${CMAKE_BINARY_DIR}/.devel)
|
|
#
|
|
# Warning options.
|
|
#
|
|
if(MSVC AND NOT ${CMAKE_C_COMPILER} MATCHES "clang*")
|
|
#
|
|
# MSVC, with Microsoft's front end and code generator.
|
|
# "MSVC" is also set for Microsoft's compiler with a Clang
|
|
# front end and their code generator ("Clang/C2"), so we
|
|
# check for clang.exe and treat that differently.
|
|
#
|
|
check_and_add_compiler_option(-Wall)
|
|
#
|
|
# Disable some pointless warnings that /Wall turns on.
|
|
#
|
|
# Unfortunately, MSVC does not appear to have an equivalent
|
|
# to "__attribute__((unused))" to mark a particular function
|
|
# parameter as being known to be unused, so that the compiler
|
|
# won't warn about it (for example, the function might have
|
|
# that parameter because a pointer to it is being used, and
|
|
# the signature of that function includes that parameter).
|
|
# C++ lets you give a parameter a type but no name, but C
|
|
# doesn't have that.
|
|
#
|
|
check_and_add_compiler_option(-wd4100)
|
|
#
|
|
# In theory, we care whether somebody uses f() rather than
|
|
# f(void) to declare a function with no arguments, but, in
|
|
# practice, there are places in the Windows header files
|
|
# that appear to do that, so we squelch that warning.
|
|
#
|
|
check_and_add_compiler_option(-wd4255)
|
|
#
|
|
# Windows FD_SET() generates this, so we suppress it.
|
|
#
|
|
check_and_add_compiler_option(-wd4548)
|
|
#
|
|
# Perhaps testing something #defined to be 0 with #ifdef is an
|
|
# error, and it should be tested with #if, but perhaps it's
|
|
# not, and Microsoft does that in its headers, so we squelch
|
|
# that warning.
|
|
#
|
|
check_and_add_compiler_option(-wd4574)
|
|
#
|
|
# The Windows headers also test not-defined values in #if, so
|
|
# we don't want warnings about that, either.
|
|
#
|
|
check_and_add_compiler_option(-wd4668)
|
|
#
|
|
# We do *not* care whether some function is, or isn't, going to be
|
|
# expanded inline.
|
|
#
|
|
check_and_add_compiler_option(-wd4710)
|
|
check_and_add_compiler_option(-wd4711)
|
|
#
|
|
# We do *not* care whether we're adding padding bytes after
|
|
# structure members.
|
|
#
|
|
check_and_add_compiler_option(-wd4820)
|
|
#
|
|
# We do *not* care about every single place the compiler would
|
|
# have inserted Spectre mitigation if only we had told it to
|
|
# do so with /Qspectre. I guess the theory is that it's seeing
|
|
# bounds checks that would prevent out-of-bounds loads and that
|
|
# those out-of-bounds loads could be done speculatively and that
|
|
# the Spectre attack could detect the value of the out-of-bounds
|
|
# data *if* it's within our address space, but unless I'm
|
|
# missing something I don't see that as being any form of
|
|
# security hole.
|
|
#
|
|
# XXX - add /Qspectre if that is really worth doing.
|
|
#
|
|
check_and_add_compiler_option(-wd5045)
|
|
#
|
|
# We do *not* care whether a structure had padding added at
|
|
# the end because of __declspec(align) - *we* don't use
|
|
# __declspec(align), because the only structures whose layout
|
|
# we precisely specify are those that get overlaid on packet
|
|
# data, and in those every element is an array of octets so
|
|
# that we have full control over the size and alignment, and,
|
|
# apparently, jmp_buf has such a declaration on x86, meaning
|
|
# that everything that includes netdissect.h, i.e. almost every
|
|
# file in tcpdump, gets a warning.
|
|
#
|
|
check_and_add_compiler_option(-wd4324)
|
|
else()
|
|
#
|
|
# Other compilers, including MSVC with a Clang front end and
|
|
# Microsoft's code generator. We currently treat them as if
|
|
# they might support GCC-style -W options.
|
|
#
|
|
check_and_add_compiler_option(-W)
|
|
check_and_add_compiler_option(-Wall)
|
|
check_and_add_compiler_option(-Wassign-enum)
|
|
check_and_add_compiler_option(-Wcast-qual)
|
|
check_and_add_compiler_option(-Wmissing-prototypes)
|
|
check_and_add_compiler_option(-Wmissing-variable-declarations)
|
|
check_and_add_compiler_option(-Wold-style-definition)
|
|
if(NOT CMAKE_C_COMPILER_ID MATCHES "Sun")
|
|
# In Sun C versions that implement GCC compatibility "-Wpedantic"
|
|
# means the same as "-pedantic". The latter is mutually exclusive
|
|
# with several other options. One of those is "-xc99", which has
|
|
# already been set for Sun C above.
|
|
check_and_add_compiler_option(-Wpedantic)
|
|
endif()
|
|
check_and_add_compiler_option(-Wpointer-arith)
|
|
check_and_add_compiler_option(-Wpointer-sign)
|
|
check_and_add_compiler_option(-Wshadow)
|
|
check_and_add_compiler_option(-Wsign-compare)
|
|
check_and_add_compiler_option(-Wstrict-prototypes)
|
|
check_and_add_compiler_option(-Wundef)
|
|
check_and_add_compiler_option(-Wunreachable-code-return)
|
|
check_and_add_compiler_option(-Wused-but-marked-unused)
|
|
check_and_add_compiler_option(-Wwrite-strings)
|
|
endif()
|
|
endif()
|
|
|
|
#
|
|
# Extra compiler options for the build matrix scripts to request -Werror or
|
|
# its equivalent if required. The CMake variable name cannot be CFLAGS
|
|
# because that is already used for a different purpose in CMake. Example
|
|
# usage: cmake -DEXTRA_CFLAGS='-Wall -Wextra -Werror' ...
|
|
#
|
|
if(NOT "${EXTRA_CFLAGS}" STREQUAL "")
|
|
# The meaning of EXTRA_CFLAGS is "use the exact specified options, or the
|
|
# build risks failing to fail", not "try every specified option, omit those
|
|
# that do not work and use the rest". Thus use add_compile_options(), not
|
|
# foreach()/check_and_add_compiler_option().
|
|
string(REPLACE " " ";" _extra_cflags_list ${EXTRA_CFLAGS})
|
|
add_compile_options(${_extra_cflags_list})
|
|
message(STATUS "Added extra compile options (${EXTRA_CFLAGS})")
|
|
endif()
|
|
|
|
######################################
|
|
# Input files
|
|
######################################
|
|
|
|
if(ENABLE_SMB)
|
|
#
|
|
# We allow the SMB dissector to be omitted.
|
|
#
|
|
set(LOCALSRC ${LOCALSRC}
|
|
print-smb.c
|
|
smbutil.c)
|
|
endif(ENABLE_SMB)
|
|
|
|
set(NETDISSECT_SOURCE_LIST_C
|
|
addrtoname.c
|
|
addrtostr.c
|
|
af.c
|
|
ascii_strcasecmp.c
|
|
checksum.c
|
|
cpack.c
|
|
gmpls.c
|
|
in_cksum.c
|
|
ipproto.c
|
|
l2vpn.c
|
|
netdissect.c
|
|
netdissect-alloc.c
|
|
nlpid.c
|
|
ntp.c
|
|
oui.c
|
|
parsenfsfh.c
|
|
print.c
|
|
print-802_11.c
|
|
print-802_15_4.c
|
|
print-ah.c
|
|
print-ahcp.c
|
|
print-aodv.c
|
|
print-aoe.c
|
|
print-ap1394.c
|
|
print-arcnet.c
|
|
print-arista.c
|
|
print-arp.c
|
|
print-ascii.c
|
|
print-atalk.c
|
|
print-atm.c
|
|
print-babel.c
|
|
print-bcm-li.c
|
|
print-beep.c
|
|
print-bfd.c
|
|
print-bgp.c
|
|
print-bootp.c
|
|
print-brcmtag.c
|
|
print-bt.c
|
|
print-calm-fast.c
|
|
print-carp.c
|
|
print-cdp.c
|
|
print-cfm.c
|
|
print-chdlc.c
|
|
print-cip.c
|
|
print-cnfp.c
|
|
print-dccp.c
|
|
print-decnet.c
|
|
print-dhcp6.c
|
|
print-domain.c
|
|
print-dsa.c
|
|
print-dtp.c
|
|
print-dvmrp.c
|
|
print-eap.c
|
|
print-egp.c
|
|
print-eigrp.c
|
|
print-enc.c
|
|
print-erspan.c
|
|
print-esp.c
|
|
print-ether.c
|
|
print-fddi.c
|
|
print-forces.c
|
|
print-fr.c
|
|
print-frag6.c
|
|
print-ftp.c
|
|
print-geneve.c
|
|
print-geonet.c
|
|
print-gre.c
|
|
print-hncp.c
|
|
print-hsrp.c
|
|
print-http.c
|
|
print-icmp.c
|
|
print-icmp6.c
|
|
print-igmp.c
|
|
print-igrp.c
|
|
print-ip-demux.c
|
|
print-ip.c
|
|
print-ip6.c
|
|
print-ip6opts.c
|
|
print-ipcomp.c
|
|
print-ipfc.c
|
|
print-ipnet.c
|
|
print-ipoib.c
|
|
print-ipx.c
|
|
print-isakmp.c
|
|
print-isoclns.c
|
|
print-juniper.c
|
|
print-krb.c
|
|
print-l2tp.c
|
|
print-lane.c
|
|
print-ldp.c
|
|
print-lisp.c
|
|
print-llc.c
|
|
print-lldp.c
|
|
print-lmp.c
|
|
print-loopback.c
|
|
print-lspping.c
|
|
print-lwapp.c
|
|
print-lwres.c
|
|
print-m3ua.c
|
|
print-macsec.c
|
|
print-mobile.c
|
|
print-mobility.c
|
|
print-mpcp.c
|
|
print-mpls.c
|
|
print-mptcp.c
|
|
print-msdp.c
|
|
print-msnlb.c
|
|
print-nflog.c
|
|
print-nfs.c
|
|
print-nhrp.c
|
|
print-nsh.c
|
|
print-ntp.c
|
|
print-null.c
|
|
print-olsr.c
|
|
print-openflow-1.0.c
|
|
print-openflow-1.3.c
|
|
print-openflow.c
|
|
print-ospf.c
|
|
print-ospf6.c
|
|
print-otv.c
|
|
print-pflog.c
|
|
print-pgm.c
|
|
print-pim.c
|
|
print-pktap.c
|
|
print-ppi.c
|
|
print-ppp.c
|
|
print-pppoe.c
|
|
print-pptp.c
|
|
print-ptp.c
|
|
print-quic.c
|
|
print-radius.c
|
|
print-raw.c
|
|
print-realtek.c
|
|
print-resp.c
|
|
print-rip.c
|
|
print-ripng.c
|
|
print-rpki-rtr.c
|
|
print-rsvp.c
|
|
print-rt6.c
|
|
print-rtsp.c
|
|
print-rx.c
|
|
print-sctp.c
|
|
print-sflow.c
|
|
print-sip.c
|
|
print-sl.c
|
|
print-sll.c
|
|
print-slow.c
|
|
print-smtp.c
|
|
print-snmp.c
|
|
print-someip.c
|
|
print-ssh.c
|
|
print-stp.c
|
|
print-sunatm.c
|
|
print-sunrpc.c
|
|
print-symantec.c
|
|
print-syslog.c
|
|
print-tcp.c
|
|
print-telnet.c
|
|
print-tftp.c
|
|
print-timed.c
|
|
print-tipc.c
|
|
print-token.c
|
|
print-udld.c
|
|
print-udp.c
|
|
print-unsupported.c
|
|
print-usb.c
|
|
print-vjc.c
|
|
print-vqp.c
|
|
print-vrrp.c
|
|
print-vsock.c
|
|
print-vtp.c
|
|
print-vxlan-gpe.c
|
|
print-vxlan.c
|
|
print-wb.c
|
|
print-whois.c
|
|
print-zep.c
|
|
print-zephyr.c
|
|
print-zeromq.c
|
|
${LOCALSRC}
|
|
signature.c
|
|
strtoaddr.c
|
|
util-print.c
|
|
)
|
|
|
|
#
|
|
# Replace missing functions
|
|
#
|
|
foreach(FUNC strlcat strlcpy strsep getservent getopt_long)
|
|
string(TOUPPER ${FUNC} FUNC_UPPERCASE)
|
|
set(HAVE_FUNC_UPPERCASE HAVE_${FUNC_UPPERCASE})
|
|
if(NOT ${HAVE_FUNC_UPPERCASE})
|
|
set(NETDISSECT_SOURCE_LIST_C ${NETDISSECT_SOURCE_LIST_C} missing/${FUNC}.c)
|
|
endif()
|
|
endforeach()
|
|
|
|
add_library(netdissect STATIC
|
|
${NETDISSECT_SOURCE_LIST_C}
|
|
)
|
|
if(NOT C_ADDITIONAL_FLAGS STREQUAL "")
|
|
set_target_properties(netdissect PROPERTIES COMPILE_FLAGS ${C_ADDITIONAL_FLAGS})
|
|
endif()
|
|
|
|
set(TCPDUMP_SOURCE_LIST_C fptype.c tcpdump.c)
|
|
|
|
if(NOT HAVE_BPF_DUMP)
|
|
set(TCPDUMP_SOURCE_LIST_C ${TCPDUMP_SOURCE_LIST_C} bpf_dump.c)
|
|
endif(NOT HAVE_BPF_DUMP)
|
|
|
|
set(PROJECT_SOURCE_LIST_C ${NETDISSECT_SOURCE_LIST_C} ${TCPDUMP_SOURCE_LIST_C})
|
|
|
|
file(GLOB PROJECT_SOURCE_LIST_H
|
|
*.h
|
|
)
|
|
|
|
#
|
|
# Assume, by default, no support for shared libraries and V7/BSD
|
|
# convention for man pages (devices in section 4, file formats in
|
|
# section 5, miscellaneous info in section 7, administrative commands
|
|
# and daemons in section 8). Individual cases can override this.
|
|
# Individual cases can override this.
|
|
#
|
|
set(MAN_FILE_FORMATS 5)
|
|
set(MAN_MISC_INFO 7)
|
|
if(CMAKE_SYSTEM_NAME STREQUAL "HP-UX")
|
|
#
|
|
# Use System V conventions for man pages.
|
|
#
|
|
set(MAN_FILE_FORMATS 4)
|
|
set(MAN_MISC_INFO 5)
|
|
elseif(CMAKE_SYSTEM_NAME STREQUAL "SunOS" AND CMAKE_SYSTEM_VERSION MATCHES "5[.][0-9.]*")
|
|
#
|
|
# SunOS 5.x.
|
|
#
|
|
if(CMAKE_SYSTEM_VERSION STREQUAL "5.12")
|
|
else()
|
|
#
|
|
# Use System V conventions for man pages.
|
|
#
|
|
set(MAN_FILE_FORMATS 4)
|
|
set(MAN_MISC_INFO 5)
|
|
endif()
|
|
endif()
|
|
|
|
source_group("Source Files" FILES ${PROJECT_SOURCE_LIST_C})
|
|
source_group("Header Files" FILES ${PROJECT_SOURCE_LIST_H})
|
|
|
|
######################################
|
|
# Register targets
|
|
######################################
|
|
|
|
add_executable(tcpdump ${TCPDUMP_SOURCE_LIST_C})
|
|
if(NOT C_ADDITIONAL_FLAGS STREQUAL "")
|
|
set_target_properties(tcpdump PROPERTIES COMPILE_FLAGS ${C_ADDITIONAL_FLAGS})
|
|
endif()
|
|
if(NOT "${PCAP_LINK_FLAGS}" STREQUAL "")
|
|
set_target_properties(tcpdump PROPERTIES LINK_FLAGS ${PCAP_LINK_FLAGS})
|
|
endif()
|
|
target_link_libraries(tcpdump netdissect ${TCPDUMP_LINK_LIBRARIES})
|
|
|
|
######################################
|
|
# Write out the config.h file
|
|
######################################
|
|
|
|
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmakeconfig.h.in ${CMAKE_CURRENT_BINARY_DIR}/config.h)
|
|
|
|
######################################
|
|
# Install tcpdump and man pages
|
|
######################################
|
|
|
|
#
|
|
# "Define GNU standard installation directories", which actually
|
|
# are also defined, to some degree, by autotools, and at least
|
|
# some of which are general UN*X conventions.
|
|
#
|
|
include(GNUInstallDirs)
|
|
|
|
set(MAN1_EXPAND tcpdump.1.in)
|
|
|
|
if(WIN32)
|
|
# XXX TODO where to install on Windows?
|
|
else(WIN32)
|
|
install(TARGETS tcpdump DESTINATION bin)
|
|
endif(WIN32)
|
|
|
|
# On UN*X, and on Windows when not using MSVC, process man pages and
|
|
# arrange that they be installed.
|
|
if(NOT MSVC)
|
|
#
|
|
# Man pages.
|
|
#
|
|
# For each section of the manual for which we have man pages
|
|
# that require macro expansion, do the expansion.
|
|
#
|
|
set(MAN1 "")
|
|
foreach(TEMPLATE_MANPAGE ${MAN1_EXPAND})
|
|
string(REPLACE ".in" "" MANPAGE ${TEMPLATE_MANPAGE})
|
|
configure_file(${CMAKE_SOURCE_DIR}/${TEMPLATE_MANPAGE} ${CMAKE_CURRENT_BINARY_DIR}/${MANPAGE} @ONLY)
|
|
set(MAN1 ${MAN1} ${CMAKE_CURRENT_BINARY_DIR}/${MANPAGE})
|
|
endforeach(TEMPLATE_MANPAGE)
|
|
install(FILES ${MAN1} DESTINATION ${CMAKE_INSTALL_MANDIR}/man1)
|
|
endif(NOT MSVC)
|
|
|
|
# uninstall target
|
|
configure_file(
|
|
"${CMAKE_CURRENT_SOURCE_DIR}/cmake_uninstall.cmake.in"
|
|
"${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake"
|
|
IMMEDIATE @ONLY)
|
|
|
|
add_custom_target(uninstall
|
|
COMMAND ${CMAKE_COMMAND} -P ${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake)
|
|
|
|
#
|
|
# tcpdump tests
|
|
# We try to find the Perl interpreter and, if we do, we have the check
|
|
# rule run tests/TESTrun with it, because just trying to run the TESTrun
|
|
# script as a command won't work on Windows.
|
|
#
|
|
find_program(PERL perl)
|
|
if(PERL)
|
|
message(STATUS "Found perl at ${PERL}")
|
|
add_custom_target(check
|
|
COMMAND ${PERL} ${CMAKE_SOURCE_DIR}/tests/TESTrun)
|
|
else()
|
|
message(STATUS "Didn't find perl")
|
|
endif()
|