Files
unison/buildsupport/macros_and_definitions.cmake

1044 lines
37 KiB
CMake
Raw Normal View History

2021-11-10 22:28:44 -03:00
# Copyright (c) 2017-2021 Universidade de Brasília
#
# This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public
# License version 2 as published by the Free Software Foundation;
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along with this program; if not, write to the Free
# Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
# Author: Gabriel Ferreira <gabrielcarvfer@gmail.com>
# Export compile time variable setting the directory to the NS3 root folder
add_definitions(-DPROJECT_SOURCE_PATH="${PROJECT_SOURCE_DIR}")
# Cache options for INT64X64
set(INT64X64 "INT128" CACHE STRING "Int64x64 implementation")
set(INT64X64 "CAIRO" CACHE STRING "Int64x64 implementation")
set(INT64X64 "DOUBLE" CACHE STRING "Int64x64 implementation")
set_property(CACHE INT64X64 PROPERTY STRINGS INT128 CAIRO DOUBLE)
# WSLv1 doesn't support tap features
if(EXISTS "/proc/version")
file(READ "/proc/version" CMAKE_LINUX_DISTRO)
string(FIND ${CMAKE_LINUX_DISTRO} "Microsoft" res)
if(res EQUAL -1)
set(WSLv1 False)
else()
set(WSLv1 True)
endif()
endif()
# Set Linux flag if on Linux
if(UNIX AND NOT APPLE)
set(LINUX TRUE)
add_definitions(-D__LINUX__)
endif()
if(APPLE)
add_definitions(-D__APPLE__)
endif()
set(cat_command cat)
if(CMAKE_XCODE_BUILD_SYSTEM)
set(XCODE True)
else()
set(XCODE False)
endif()
# Check the number of threads
include(ProcessorCount)
ProcessorCount(NumThreads)
# Output folders
set(CMAKE_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/build)
set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_OUTPUT_DIRECTORY}/lib)
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_OUTPUT_DIRECTORY}/lib)
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_OUTPUT_DIRECTORY})
set(CMAKE_HEADER_OUTPUT_DIRECTORY ${CMAKE_OUTPUT_DIRECTORY}/include/ns3)
set(THIRD_PARTY_DIRECTORY ${PROJECT_SOURCE_DIR}/3rd-party)
# Get installation folder default values for each platform and include package configuration macro
include(GNUInstallDirs)
include(buildsupport/custom_modules/ns3_cmake_package.cmake)
if(${XCODE})
# Is that so hard not to break people's CI, AAPL? Why would you output the targets to a Debug/Release subfolder? Why?
foreach(OUTPUTCONFIG ${CMAKE_CONFIGURATION_TYPES})
string(TOUPPER ${OUTPUTCONFIG} OUTPUTCONFIG)
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_${OUTPUTCONFIG} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY_${OUTPUTCONFIG} ${CMAKE_LIBRARY_OUTPUT_DIRECTORY})
set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY_${OUTPUTCONFIG} ${CMAKE_ARCHIVE_OUTPUT_DIRECTORY})
endforeach()
endif()
# fPIC (position-independent code) and fPIE (position-independent executable)
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
# Identify compiler
set(CLANG FALSE)
if("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang")
set(CLANG TRUE)
endif()
set(GCC FALSE)
if("${CMAKE_CXX_COMPILER_ID}" MATCHES "GNU")
set(GCC TRUE)
endif()
# Set compiler options and get command to force unused function linkage (useful for libraries)
set(CXX_UNSUPPORTED_STANDARDS 98 11 14)
set(CMAKE_CXX_STANDARD_MINIMUM 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(LIB_AS_NEEDED_PRE)
set(LIB_AS_NEEDED_POST)
if(${GCC} AND NOT APPLE)
# using GCC
set(LIB_AS_NEEDED_PRE -Wl,--no-as-needed)
set(LIB_AS_NEEDED_POST -Wl,--as-needed)
set(LIB_AS_NEEDED_PRE_STATIC -Wl,--whole-archive,-Bstatic)
set(LIB_AS_NEEDED_POST_STATIC -Wl,--no-whole-archive)
set(LIB_AS_NEEDED_POST_STATIC_DYN -Wl,-Bdynamic,--no-whole-archive)
endif()
if(${CLANG} AND APPLE)
# using Clang set(LIB_AS_NEEDED_PRE -all_load)
set(LIB_AS_NEEDED_POST)
endif()
macro(SUBDIRLIST result curdir)
file(GLOB children RELATIVE ${curdir} ${curdir}/*)
set(dirlist "")
foreach(child ${children})
if(IS_DIRECTORY ${curdir}/${child})
list(APPEND dirlist ${child})
endif()
endforeach()
set(${result} ${dirlist})
endmacro()
macro(library_target_name libname targetname)
set(${targetname} lib${libname})
endmacro()
macro(clear_global_cached_variables)
# clear cache variables
unset(build_profile CACHE)
unset(build_profile_suffix CACHE)
unset(lib-ns3-static-objs CACHE)
unset(ns3-contrib-libs CACHE)
unset(ns3-example-folders CACHE)
unset(ns3-execs CACHE)
unset(ns3-external-libs CACHE)
unset(ns3-libs CACHE)
unset(ns3-libs-tests CACHE)
unset(ns3-python-bindings-modules CACHE)
endmacro()
# process all options passed in main cmakeLists
macro(process_options)
clear_global_cached_variables()
# process debug switch Used in build-profile-test-suite
string(TOLOWER ${CMAKE_BUILD_TYPE} cmakeBuildType)
set(build_profile "${cmakeBuildType}" CACHE INTERNAL "")
if(${cmakeBuildType} STREQUAL "debug")
add_definitions(-DNS3_BUILD_PROFILE_DEBUG)
elseif(${cmakeBuildType} STREQUAL "relwithdebinfo")
add_definitions(-DNS3_BUILD_PROFILE_RELEASE)
elseif(${cmakeBuildType} STREQUAL "release")
add_definitions(-DNS3_BUILD_PROFILE_OPTIMIZED)
if(${NS3_NATIVE_OPTIMIZATIONS})
set(build_profile "optimized" CACHE INTERNAL "")
endif()
else()
add_definitions(-DNS3_BUILD_PROFILE_OPTIMIZED)
endif()
set(profiles_without_suffixes release)
set(build_profile_suffix "" CACHE INTERNAL "")
if(NOT (${build_profile} IN_LIST profiles_without_suffixes))
set(build_profile_suffix -${build_profile} CACHE INTERNAL "")
endif()
# Set warning level and warning as errors
if (${NS3_WARNINGS})
if (MSVC)
add_compile_options(/W3) # /W4 = -Wall + -Wextra
if (${NS3_WARNINGS_AS_ERRORS})
add_compile_options(/WX)
endif()
else()
add_compile_options(-Wall) # -Wextra
if (${NS3_WARNINGS_AS_ERRORS})
add_compile_options(-Werror)
endif()
endif()
endif()
if(${NS3_ENABLE_BUILD_VERSION})
include(buildsupport/custom_modules/ns3_versioning.cmake)
add_definitions(-DENABLE_BUILD_VERSION=1)
find_program(GIT git)
if(NOT GIT)
message(FATAL_ERROR "Baking build version into libraries require Git.")
endif()
check_git_repo_has_ns3_tags(HAS_NS3_TAGS NS3_VERSION_TAG)
if(NOT ${HAS_NS3_TAGS})
message(FATAL_ERROR "This repository doesn't contain ns-3 git tags to bake into the libraries.")
endif()
check_ns3_closest_tags(
NS3_VERSION_CLOSEST_TAG NS3_VERSION_TAG_DISTANCE NS3_VERSION_COMMIT_HASH NS3_VERSION_DIRTY_FLAG
)
# Split commit tag (ns-3.<minor>[.patch][-RC<digit>]) into (ns;3.<minor>[.patch];[-RC<digit>]):
string(REPLACE "-" ";" NS3_VER_LIST ${NS3_VERSION_TAG})
list(LENGTH NS3_VER_LIST NS3_VER_LIST_LEN)
# Get last version tag fragment (RC<digit>)
set(NS3_VERSION_RELEASE_CANDIDATE)
if(${NS3_VER_LIST_LEN} GREATER 2)
list(GET NS3_VER_LIST 2 RELEASE_CANDIDATE)
set(NS3_VERSION_RELEASE_CANDIDATE "${RELEASE_CANDIDATE}")
endif()
# Get 3.<minor>[.patch]
list(GET NS3_VER_LIST 1 VERSION_STRING)
# Split into a list 3;<minor>[;patch]
string(REPLACE "." ";" VERSION_LIST ${VERSION_STRING})
list(LENGTH VERSION_LIST VER_LIST_LEN)
list(GET VERSION_LIST 0 NS3_VERSION_MAJOR)
if(${VER_LIST_LEN} GREATER 1)
list(GET VERSION_LIST 1 NS3_VERSION_MINOR)
if(${VER_LIST_LEN} GREATER 2)
list(GET VERSION_LIST 2 NS3_VERSION_PATCH)
endif()
endif()
# Transform list with 1 entry into strings
set(NS3_VERSION_MAJOR "${NS3_VERSION_MAJOR}")
set(NS3_VERSION_MINOR "${NS3_VERSION_MINOR}")
set(NS3_VERSION_PATCH "${NS3_VERSION_PATCH}")
set(NS3_VERSION_TAG "${NS3_VERSION_TAG}")
# Set
set(NS3_VERSION_BUILD_PROFILE ${cmakeBuildType})
configure_file(buildsupport/version-defines-template.h ${CMAKE_HEADER_OUTPUT_DIRECTORY}/version-defines.h)
endif()
if(${NS3_CLANG_FORMAT})
find_program(CLANG_FORMAT clang-format)
if(CLANG_FORMAT)
file(
GLOB_RECURSE
ALL_CXX_SOURCE_FILES
src/*.cc
src/*.h
examples/*.cc
examples/*.h
utils/*.cc
utils/*.h
scratch/*.cc
scratch/*.h
)
add_custom_target(clang-format COMMAND clang-format -style=file -i ${ALL_CXX_SOURCE_FILES})
unset(ALL_CXX_SOURCE_FILES)
else()
message(STATUS "Proceeding without clang-format target")
endif()
endif()
if(${NS3_CLANG_TIDY})
find_program(CLANG_TIDY clang-tidy)
if(CLANG_TIDY)
set(CMAKE_CXX_CLANG_TIDY "clang-tidy")
else()
message(STATUS "Proceeding without clang-tidy static analysis")
endif()
endif()
if(${NS3_CLANG_TIMETRACE})
if(${CLANG})
# Fetch and build clang build analyzer
include(ExternalProject)
ExternalProject_Add(
ClangBuildAnalyzer
GIT_REPOSITORY "https://github.com/aras-p/ClangBuildAnalyzer.git"
GIT_TAG "5d40542c5b2080637206e9d51e2957d0017ba8b3"
INSTALL_COMMAND cmake -E copy_if_different ClangBuildAnalyzer ${PROJECT_BINARY_DIR}
)
# Add compiler flag and create target to do everything automatically
add_definitions(-ftime-trace)
add_custom_target(
timeTraceReport
COMMAND ${PROJECT_BINARY_DIR}/ClangBuildAnalyzer --all ${PROJECT_BINARY_DIR}
${PROJECT_BINARY_DIR}/clangBuildAnalyzerReport.bin
COMMAND ${PROJECT_BINARY_DIR}/ClangBuildAnalyzer --analyze ${PROJECT_BINARY_DIR}/clangBuildAnalyzerReport.bin >
${PROJECT_SOURCE_DIR}/ClangBuildAnalyzerReport.txt
DEPENDS ClangBuildAnalyzer
)
else()
message(FATAL_ERROR "TimeTrace is a Clang feature, but you're using a different compiler.")
endif()
endif()
find_program(CMAKE_FORMAT_PROGRAM cmake-format HINTS ~/.local/bin)
if(CMAKE_FORMAT_PROGRAM)
file(GLOB_RECURSE ALL_CMAKE_FILES CMakeLists.txt buildsupport/*.cmake)
add_custom_target(
cmake-format COMMAND ${CMAKE_FORMAT_PROGRAM} -c ${PROJECT_SOURCE_DIR}/buildsupport/.cmake-format.txt -i
${ALL_CMAKE_FILES}
)
unset(ALL_CMAKE_FILES)
endif()
# If the user has not set a CXX standard version, assume the minimum
if(NOT "${CMAKE_CXX_STANDARD}")
set(CMAKE_CXX_STANDARD ${CMAKE_CXX_STANDARD_MINIMUM})
endif()
# Search standard provided by the user in the unsupported standards list
list(FIND CXX_UNSUPPORTED_STANDARDS ${CMAKE_CXX_STANDARD} unsupported)
if(${unsupported} GREATER -1)
message(FATAL_ERROR "You're trying to use the unsupported C++ ${CMAKE_CXX_STANDARD}.\n"
"Try -DCMAKE_CXX_STANDARD=${CMAKE_CXX_STANDARD_MINIMUM}."
)
endif()
if(${NS3_DES_METRICS})
add_definitions(-DENABLE_DES_METRICS)
endif()
if(${NS3_SANITIZE} AND ${NS3_SANITIZE_MEMORY})
message(FATAL_ERROR "The memory sanitizer can't be used with other sanitizers. Disable one of them.")
endif()
if(${NS3_SANITIZE})
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address,leak,undefined")
endif()
if(${NS3_SANITIZE_MEMORY})
if(${CLANG})
set(blacklistfile ${PROJECT_SOURCE_DIR}/memory-sanitizer-blacklist.txt)
if(EXISTS ${blacklistfile})
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=memory -fsanitize-blacklist=${blacklistfile}")
else()
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=memory")
endif()
if(NOT ($ENV{MSAN_OPTIONS} MATCHES "strict_memcmp=0"))
message(
WARNING "Please export MSAN_OPTIONS=strict_memcmp=0 "
"and call the generated buildsystem directly to proceed.\n"
"Trying to build with cmake or IDEs will probably fail since"
"CMake can't export environment variables to its parent process."
)
endif()
unset(blacklistfile)
else()
message(FATAL_ERROR "The memory sanitizer is only supported by Clang")
endif()
endif()
if(${NS3_NATIVE_OPTIMIZATIONS} AND ${GCC})
add_compile_options(-march=native -mtune=native)
endif()
if(${NS3_LINK_TIME_OPTIMIZATION})
# Link-time optimization (LTO) if available
include(CheckIPOSupported)
check_ipo_supported(RESULT LTO_AVAILABLE OUTPUT output)
if(LTO_AVAILABLE)
set(CMAKE_INTERPROCEDURAL_OPTIMIZATION TRUE)
message(STATUS "Link-tiem optimization (LTO) is supported.")
else()
message(STATUS "Link-time optimization (LTO) is not supported: ${output}.")
endif()
endif()
if(${NS3_LINK_WHAT_YOU_USE})
set(CMAKE_LINK_WHAT_YOU_USE TRUE)
else()
set(CMAKE_LINK_WHAT_YOU_USE FALSE)
endif()
if(${NS3_INCLUDE_WHAT_YOU_USE})
# Before using, install iwyu, run "strings /usr/bin/iwyu | grep LLVM" to find the appropriate clang version and
# install it. If you don't do that, it will fail. Worse than that: it will fail silently. We use the wrapper to get
# a iwyu.log file in the ns-3-dev folder.
find_program(INCLUDE_WHAT_YOU_USE_PROG iwyu)
if(NOT INCLUDE_WHAT_YOU_USE_PROG)
message(FATAL_ERROR "iwyu (include-what-you-use) was not found.")
endif()
message(STATUS "iwyu is enabled")
set(CMAKE_CXX_INCLUDE_WHAT_YOU_USE ${PROJECT_SOURCE_DIR}/buildsupport/iwyu_wrapper.sh;${PROJECT_SOURCE_DIR})
else()
unset(CMAKE_CXX_INCLUDE_WHAT_YOU_USE)
endif()
if(${XCODE})
if(${NS3_STATIC} OR ${NS3_MONOLIB})
message(
FATAL_ERROR "Xcode doesn't play nicely with CMake object libraries,"
"and those are used for NS3_STATIC and NS3_MONOLIB.\n" "Disable them or try a different generator"
)
endif()
set(CMAKE_XCODE_GENERATE_TOP_LEVEL_PROJECT_ONLY ON)
set(CMAKE_XCODE_ATTRIBUTE_CODE_SIGNING_REQUIRED NO)
set(CMAKE_XCODE_ATTRIBUTE_CODE_SIGN_IDENTITY "")
else()
unset(CMAKE_XCODE_GENERATE_TOP_LEVEL_PROJECT_ONLY)
unset(CMAKE_XCODE_ATTRIBUTE_CODE_SIGNING_REQUIRED)
unset(CMAKE_XCODE_ATTRIBUTE_CODE_SIGN_IDENTITY)
endif()
# Set common include folder (./build/include, where we find ns3/core-module.h)
include_directories(${CMAKE_OUTPUT_DIRECTORY}/include)
# find required dependencies
list(APPEND CMAKE_MODULE_PATH "${PROJECT_SOURCE_DIR}/buildsupport/custom_modules")
list(APPEND CMAKE_MODULE_PATH "${PROJECT_SOURCE_DIR}/buildsupport/3rd_party")
# GTK3 Don't search for it if you don't have it installed, as it take an insane amount of time
if(${NS3_GTK3})
find_package(HarfBuzz QUIET)
if(NOT ${HarfBuzz_FOUND})
message(STATUS "Harfbuzz is required by GTK3 and was not found.")
else()
set(CMAKE_SUPPRESS_DEVELOPER_WARNINGS 1 CACHE BOOL "")
find_package(GTK3 QUIET)
unset(CMAKE_SUPPRESS_DEVELOPER_WARNINGS CACHE)
if(NOT ${GTK3_FOUND})
message(STATUS "GTK3 was not found. Continuing without it.")
else()
message(STATUS "GTK3 was found.")
endif()
endif()
endif()
if(${NS3_STATIC})
# Warn users that they may be using shared libraries, which won't produce a standalone static library
set(ENABLE_REALTIME FALSE)
set(HAVE_RT FALSE)
message(STATUS "Statically linking 3rd party libraries have not been tested.\n"
"Disable Brite, Click, Gtk, GSL, Mpi, Openflow and SQLite"
" if you want a standalone static ns-3 library."
)
else()
find_package(LibXml2 QUIET)
if(NOT ${LIBXML2_FOUND})
message(STATUS "LibXML2 was not found. Continuing without it.")
else()
message(STATUS "LibXML2 was found.")
add_definitions(-DHAVE_LIBXML2)
endif()
# LibRT
if(${NS3_REALTIME})
if(APPLE)
message(STATUS "Lib RT is not supported on Mac OS X. Continuing without it.")
else()
find_library(LIBRT rt QUIET)
if(NOT ${LIBRT_FOUND})
message(FATAL_ERROR "LibRT was not found.")
set(ENABLE_REALTIME FALSE)
else()
message(STATUS "LibRT was found.")
set(ENABLE_REALTIME TRUE)
set(HAVE_RT TRUE) # for core-config.h
endif()
endif()
endif()
endif()
set(THREADS_ENABLED FALSE)
set(PTHREADS_FOUND FALSE)
if(${NS3_PTHREAD})
set(THREADS_PREFER_PTHREAD_FLAG)
find_package(Threads QUIET)
if(${CMAKE_USE_PTHREADS_INIT})
include_directories(${THREADS_PTHREADS_INCLUDE_DIR})
set(PTHREADS_FOUND TRUE)
set(HAVE_PTHREAD_H TRUE) # for core-config.h
set(THREADS_ENABLED TRUE)
endif()
endif()
find_package(Python3 COMPONENTS Interpreter Development)
if(${NS3_PYTHON_BINDINGS})
if(NOT ${Python3_FOUND})
message(FATAL_ERROR "NS3_PYTHON_BINDINGS requires Python3")
endif()
link_directories(${Python3_LIBRARY_DIRS})
include_directories(${Python3_INCLUDE_DIRS})
set(PYTHONDIR ${Python3_SITELIB})
set(PYTHONARCHDIR ${Python3_SITEARCH})
set(HAVE_PYEMBED TRUE)
set(HAVE_PYEXT TRUE)
set(HAVE_PYTHON_H TRUE)
set(destination_dir ${CMAKE_OUTPUT_DIRECTORY}/bindings/python/ns)
file(COPY bindings/python/ns__init__.py DESTINATION ${destination_dir})
file(RENAME ${destination_dir}/ns__init__.py ${destination_dir}/__init__.py)
endif()
if(${NS3_SCAN_PYTHON_BINDINGS})
# empty scan target that will depend on other module scan targets to scan all of them
add_custom_target(apiscan-all)
endif()
if(${NS3_COVERAGE} AND (NOT ${NS3_TESTS} OR NOT ${NS3_EXAMPLES}))
message(FATAL_ERROR "Code coverage requires examples and tests.\n"
"Try reconfiguring CMake with -DNS3_TESTS=ON -DNS3_EXAMPLES=ON"
)
endif()
if(${NS3_TESTS})
add_custom_target(all-test-targets)
# Create a custom target to run test.py --nowaf Target is also used to produce code coverage output
add_custom_target(
run_test_py
COMMAND ${Python3_EXECUTABLE} test.py --nowaf
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
DEPENDS all-test-targets
)
if(${NS3_EXAMPLES})
include(buildsupport/custom_modules/ns3_coverage.cmake)
endif()
endif()
# Process config-store-config
configure_file(buildsupport/config-store-config-template.h ${CMAKE_HEADER_OUTPUT_DIRECTORY}/config-store-config.h)
set(ENABLE_MPI FALSE)
if(${NS3_MPI})
find_package(MPI QUIET)
if(NOT ${MPI_FOUND})
message(STATUS "MPI was not found. Continuing without it.")
else()
message(STATUS "MPI was found.")
add_definitions(-DNS3_MPI)
set(ENABLE_MPI TRUE)
endif()
endif()
set(ENABLE_VISUALIZER FALSE)
if(${NS3_VISUALIZER})
if((NOT ${NS3_PYTHON_BINDINGS}) OR (NOT ${Python3_FOUND}))
message(FATAL_ERROR "Visualizer requires NS3_PYTHON_BINDINGS and Python3")
endif()
set(ENABLE_VISUALIZER TRUE)
endif()
find_package(Boost QUIET)
if(NOT ${BOOST_FOUND})
message(STATUS "Boost was not found. Continuing without it.")
else()
message(STATUS "Boost was found.")
include_directories(${Boost_INCLUDE_DIRS})
set(CMAKE_REQUIRED_INCLUDES ${Boost_INCLUDE_DIRS})
endif()
if(${NS3_GSL})
find_package(GSL QUIET)
if(NOT ${GSL_FOUND})
message(STATUS "GSL was not found. Continuing without it.")
else()
message(STATUS "GSL was found.")
endif()
endif()
if(${NS3_GNUPLOT})
find_package(Gnuplot-ios) # Not sure what package would contain the correct header/library
if(NOT ${GNUPLOT_FOUND})
message(STATUS "GNUPLOT was not found. Continuing without it.")
else()
message(STATUS "GNUPLOT was found.")
include_directories(${GNUPLOT_INCLUDE_DIRS})
link_directories(${GNUPLOT_LIBRARY})
endif()
endif()
if(${NS3_DOCS})
find_package(Doxygen REQUIRED)
find_program(DOT dot)
find_program(DIA dia)
if((NOT DOT) OR (NOT DIA))
message(FATAL_ERROR "Dot and Dia are required by Doxygen docs."
"They're shipped within the graphviz and dia packages on Ubuntu"
)
endif()
# Get introspected doxygen
add_custom_target(
run-print-introspected-doxygen
COMMAND ${CMAKE_COMMAND} -E env ${CMAKE_OUTPUT_DIRECTORY}/utils/print-introspected-doxygen >
${PROJECT_SOURCE_DIR}/doc/introspected-doxygen.h DEPENDS print-introspected-doxygen
)
add_custom_target(
run-introspected-command-line
COMMAND ${CMAKE_COMMAND} -E env NS_COMMANDLINE_INTROSPECTION=.. ${Python3_EXECUTABLE} ./test.py --nowaf
--constrain=example
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
DEPENDS all-test-targets # all-test-targets only exists if NS3_TESTS is set to ON
)
file(
WRITE ${PROJECT_SOURCE_DIR}/doc/introspected-command-line.h
"/* This file is automatically generated by
CommandLine::PrintDoxygenUsage() from the CommandLine configuration
in various example programs. Do not edit this file! Edit the
CommandLine configuration in those files instead.
*/
\n"
)
add_custom_target(
assemble-introspected-command-line
# works on CMake 3.18 or newer > COMMAND ${CMAKE_COMMAND} -E cat
# ${PROJECT_SOURCE_DIR}/testpy-output/*.command-line >
# ${PROJECT_SOURCE_DIR}/doc/introspected-command-line.h
COMMAND ${cat_command} ${PROJECT_SOURCE_DIR}/testpy-output/*.command-line >
${PROJECT_SOURCE_DIR}/doc/introspected-command-line.h 2> NULL
DEPENDS run-introspected-command-line
)
add_custom_target(
doxygen
COMMAND Doxygen::doxygen ${PROJECT_SOURCE_DIR}/doc/doxygen.conf
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
DEPENDS run-print-introspected-doxygen assemble-introspected-command-line
)
find_package(Sphinx REQUIRED)
find_program(EPSTOPDF epstopdf)
find_program(PDFLATEX pdflatex)
find_program(LATEXMK latexmk)
find_program(CONVERT convert)
find_program(DVIPNG dvipng)
if((NOT EPSTOPDF)
OR (NOT PDFLATEX)
OR (NOT LATEXMK)
OR (NOT CONVERT)
OR (NOT DVIPNG)
)
message(
FATAL_ERROR
"Convert, Dvipng, Epstopdf, Pdflatex, Latexmk and fncychap.sty are required by Sphinx docs."
" They're shipped within the dvipng, imagemagick, texlive, texlive-font-utils and latexmk packages on Ubuntu."
" Imagemagick may fail due to a security policy issue."
" https://gitlab.com/nsnam/ns-3-dev/-/blob/master/utils/tests/gitlab-ci-doc.yml#L10-11"
)
endif()
add_custom_target(sphinx COMMENT "Building sphinx documents")
function(sphinx_target targetname)
add_custom_target(
sphinx_${targetname} COMMAND make SPHINXOPTS=-N -k html singlehtml latexpdf
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}/doc/${targetname}
)
add_dependencies(sphinx sphinx_${targetname})
endfunction()
sphinx_target(manual)
sphinx_target(models)
sphinx_target(tutorial)
endif()
# Process core-config
if(${INT64X64} MATCHES "INT128")
include(buildsupport/3rd_party/FindInt128.cmake)
find_int128_types()
if(UINT128_FOUND)
set(HAVE___UINT128_T TRUE)
set(INT64X64_USE_128 TRUE)
else()
message(STATUS "Int128 was not found. Falling back to Cairo.")
set(INT64X64 "CAIRO")
endif()
endif()
if(${INT64X64} MATCHES "CAIRO")
set(INT64X64_USE_CAIRO TRUE)
endif()
if(${INT64X64} MATCHES "DOUBLE")
# WSLv1 has a long double issue that will result in a few tests failing https://github.com/microsoft/WSL/issues/830
include(CheckTypeSize)
check_type_size("double" SIZEOF_DOUBLE)
check_type_size("long double" SIZEOF_LONG_DOUBLE)
if(${SIZEOF_LONG_DOUBLE} EQUAL ${SIZEOF_DOUBLE})
message(
STATUS "Long double has the wrong size: LD ${SIZEOF_LONG_DOUBLE} vs D ${SIZEOF_DOUBLE}. Falling back to CAIRO."
)
set(INT64X64 "CAIRO")
else()
set(INT64X64_USE_DOUBLE TRUE)
endif()
endif()
include(CheckIncludeFileCXX)
include(CheckFunctionExists)
# Check for required headers and functions, set flags if they're found or warn if they're not found
check_include_file_cxx("stdint.h" "HAVE_STDINT_H")
check_include_file_cxx("inttypes.h" "HAVE_INTTYPES_H")
check_include_file_cxx("sys/types.h" "HAVE_SYS_TYPES_H")
check_include_file_cxx("stat.h" "HAVE_SYS_STAT_H")
check_include_file_cxx("dirent.h" "HAVE_DIRENT_H")
check_include_file_cxx("stdlib.h" "HAVE_STDLIB_H")
check_include_file_cxx("signal.h" "HAVE_SIGNAL_H")
check_include_file_cxx("netpacket/packet.h" "HAVE_PACKETH")
check_function_exists("getenv" "HAVE_GETENV")
configure_file(buildsupport/core-config-template.h ${CMAKE_HEADER_OUTPUT_DIRECTORY}/core-config.h)
# Force enable ns-3 logging in debug builds and if requested for other build types
if(${NS3_LOG} OR (${build_profile} STREQUAL "debug"))
add_definitions(-DNS3_LOG_ENABLE)
endif()
# Force enable ns-3 asserts in debug builds and if requested for other build types
if(${NS3_ASSERT} OR (${build_profile} STREQUAL "debug"))
add_definitions(-DNS3_ASSERT_ENABLE)
endif()
# Enable examples as tests suites
if(${NS3_EXAMPLES})
set(NS3_ENABLE_EXAMPLES "1")
add_definitions(-DNS3_ENABLE_EXAMPLES -DCMAKE_EXAMPLE_AS_TEST)
endif()
set(PLATFORM_UNSUPPORTED_PRE "Platform doesn't support")
set(PLATFORM_UNSUPPORTED_POST "features. Continuing without them.")
# Remove from libs_to_build all incompatible libraries or the ones that dependencies couldn't be installed
if(APPLE OR WSLv1)
set(NS3_TAP OFF)
set(NS3_EMU OFF)
list(REMOVE_ITEM libs_to_build fd-net-device)
message(STATUS "${PLATFORM_UNSUPPORTED_PRE} TAP and EMU ${PLATFORM_UNSUPPORTED_POST}")
endif()
if(NOT ${ENABLE_MPI})
list(REMOVE_ITEM libs_to_build mpi)
endif()
if(NOT ${ENABLE_VISUALIZER})
list(REMOVE_ITEM libs_to_build visualizer)
endif()
if(NOT ${NS3_TAP})
list(REMOVE_ITEM libs_to_build tap-bridge)
endif()
# Create library names to solve dependency problems with macros that will be called at each lib subdirectory
set(ns3-libs)
set(ns3-all-modules)
set(ns3-libs-tests)
set(ns3-contrib-libs)
set(lib-ns3-static-objs)
set(ns3-external-libs)
set(ns3-python-bindings ns${NS3_VER}-pybindings${build_profile_suffix})
set(ns3-python-bindings-modules)
foreach(libname ${scanned_modules})
# Create libname of output library of module
library_target_name(${libname} targetname)
set(lib${libname} ${targetname} CACHE INTERNAL "")
set(lib${libname}-obj ${targetname}-obj CACHE INTERNAL "")
endforeach()
unset(optional_visualizer_lib)
if(${ENABLE_VISUALIZER})
set(optional_visualizer_lib ${libvisualizer})
endif()
if(${NS3_PRECOMPILE_HEADERS})
set(precompiled_header_libraries
<iostream>
<stdint.h>
<stdlib.h>
<map>
<unordered_map>
<vector>
<list>
<algorithm>
<string>
<set>
<sstream>
<fstream>
<cstdlib>
<exception>
<cstring>
<limits>
<math.h>
)
add_library(stdlib_pch OBJECT ${PROJECT_SOURCE_DIR}/buildsupport/empty.cc)
target_precompile_headers(stdlib_pch PUBLIC "${precompiled_header_libraries}")
add_executable(stdlib_pch_exec ${PROJECT_SOURCE_DIR}/buildsupport/empty_main.cc)
target_precompile_headers(stdlib_pch_exec PUBLIC "${precompiled_header_libraries}")
set_runtime_outputdirectory(stdlib_pch_exec ${CMAKE_BINARY_DIR})
endif()
# Create new lib for NS3 static builds
set(lib-ns3-static ns${NS3_VER}-static${build_profile_suffix})
# Create a new lib for NS3 monolibrary shared builds
set(lib-ns3-monolib ns${NS3_VER}-monolib${build_profile_suffix})
# All contrib libraries can be linked afterwards linking with ${ns3-contrib-libs}
process_contribution("${contrib_libs_to_build}")
# Netanim depends on ns-3 core, so we built it later
if(${NS3_NETANIM})
include(FetchContent)
FetchContent_Declare(netanim GIT_REPOSITORY https://gitlab.com/nsnam/netanim.git GIT_TAG netanim-3.108)
FetchContent_Populate(netanim)
file(COPY buildsupport/3rd_party/netanim_cmakelists.cmake DESTINATION ${netanim_SOURCE_DIR})
file(RENAME ${netanim_SOURCE_DIR}/netanim_cmakelists.cmake ${netanim_SOURCE_DIR}/CMakeLists.txt)
add_subdirectory(${netanim_SOURCE_DIR})
endif()
endmacro()
function(set_runtime_outputdirectory target_name output_directory)
set(ns3-exec-outputname ns${NS3_VER}-${target_name}${build_profile_suffix})
set(ns3-execs "${output_directory}${ns3-exec-outputname};${ns3-execs}" CACHE INTERNAL "list of c++ executables")
set_target_properties(
${target_name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${output_directory} RUNTIME_OUTPUT_NAME ${ns3-exec-outputname}
)
if(${XCODE})
# Is that so hard not to break people's CI, AAPL?? Why would you output the targets to a Debug/Release subfolder?
# Why?
foreach(OUTPUTCONFIG ${CMAKE_CONFIGURATION_TYPES})
string(TOUPPER ${OUTPUTCONFIG} OUTPUTCONFIG)
set_target_properties(
${target_name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY_${OUTPUTCONFIG} ${output_directory}
RUNTIME_OUTPUT_NAME_${OUTPUTCONFIG} ${ns3-exec-outputname}
)
endforeach(OUTPUTCONFIG CMAKE_CONFIGURATION_TYPES)
endif()
if(${NS3_TESTS})
add_dependencies(all-test-targets ${target_name})
endif()
if(${NS3_CLANG_TIMETRACE})
add_dependencies(timeTraceReport ${target_name})
endif()
endfunction(set_runtime_outputdirectory)
add_custom_target(copy_all_headers)
function(copy_headers_before_building_lib libname outputdir headers visibility)
set(copy_headers_target copy_headers_${libname}_${visibility})
add_custom_target(${copy_headers_target})
foreach(header ${headers})
get_filename_component(header_name ${header} NAME)
add_custom_command(
TARGET ${copy_headers_target} PRE_BUILD COMMAND ${CMAKE_COMMAND} -E copy_if_different
${CMAKE_CURRENT_SOURCE_DIR}/${header} ${outputdir}/${header_name}
)
endforeach()
# Create a target to copy all headers
add_dependencies(copy_all_headers ${copy_headers_target})
# And make sure module headers are copied before compiling them
if(${XCODE})
add_dependencies(${lib${libname}} copy_all_headers) # Xcode doesn't play nicely with object libraries
else()
add_dependencies(${lib${libname}-obj} copy_all_headers)
endif()
endfunction(copy_headers_before_building_lib)
# Import macros used for modules and define specialized versions for src modules
include(buildsupport/custom_modules/ns3_module_macros.cmake)
macro(build_lib libname source_files header_files libraries_to_link test_sources)
build_lib_impl("src" "${libname}" "${source_files}" "${header_files}" "${libraries_to_link}" "${test_sources}")
endmacro()
macro(build_lib_example name source_files header_files libraries_to_link)
build_lib_example_impl("src/${libname}" "${name}" "${source_files}" "${header_files}" "${libraries_to_link}")
endmacro()
# Contrib modules counterparts of macros above
include(buildsupport/custom_modules/ns3_contributions.cmake)
# Macro to build examples in ns-3-dev/examples/
macro(build_example name source_files header_files libraries_to_link)
set(missing_dependencies FALSE)
foreach(lib ${libraries_to_link})
string(REPLACE "lib" "" lib ${lib})
if(NOT (${lib} IN_LIST ns3-all-modules))
set(missing_dependencies TRUE)
endif()
endforeach()
if(NOT ${missing_dependencies})
# Create shared library with sources and headers
add_executable(${name} "${source_files}" "${header_files}")
if(${NS3_STATIC})
target_link_libraries(${name} ${LIB_AS_NEEDED_PRE_STATIC} ${lib-ns3-static})
elseif(${NS3_MONOLIB})
target_link_libraries(${name} ${LIB_AS_NEEDED_PRE} ${lib-ns3-monolib} ${LIB_AS_NEEDED_POST})
else()
# Link the shared library with the libraries passed
target_link_libraries(
${name} ${LIB_AS_NEEDED_PRE} ${libraries_to_link} ${optional_visualizer_lib} ${LIB_AS_NEEDED_POST}
)
endif()
if(${NS3_PRECOMPILE_HEADERS})
target_precompile_headers(${name} REUSE_FROM stdlib_pch_exec)
endif()
set_runtime_outputdirectory(${name} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY}/examples/${examplefolder}/)
endif()
endmacro()
function(filter_modules modules_to_filter all_modules_list filter_in)
set(new_modules_to_build)
# We are receiveing variable names as arguments, so we have to "dereference" them first That is why we have the
# duplicated ${${}}
foreach(module ${${all_modules_list}})
if(${filter_in} (${module} IN_LIST ${modules_to_filter}))
list(APPEND new_modules_to_build ${module})
endif()
endforeach()
set(${all_modules_list} ${new_modules_to_build} PARENT_SCOPE)
endfunction()
function(resolve_dependencies module_name dependencies contrib_dependencies)
# Create cache variables to hold dependencies list and visited
set(dependency_visited "" CACHE INTERNAL "")
set(contrib_dependency_visited "" CACHE INTERNAL "")
recursive_dependency(${module_name})
if(${module_name} IN_LIST dependency_visited)
set(temp ${dependency_visited})
list(REMOVE_AT temp 0)
set(${dependencies} ${temp} PARENT_SCOPE)
set(${contrib_dependencies} ${contrib_dependency_visited} PARENT_SCOPE)
else()
set(temp ${contrib_dependency_visited})
list(REMOVE_AT temp 0)
set(${dependencies} ${dependency_visited} PARENT_SCOPE)
set(${contrib_dependencies} ${temp} PARENT_SCOPE)
endif()
unset(dependency_visited CACHE)
unset(contrib_dependency_visited CACHE)
endfunction()
function(filter_libraries cmakelists_contents libraries)
string(REGEX MATCHALL "{lib[^}]*[^obj]}" matches "${cmakelists_content}")
list(REMOVE_ITEM matches "{libraries_to_link}")
string(REPLACE "{lib" "" matches "${matches}")
string(REPLACE "}" "" matches "${matches}")
set(${libraries} ${matches} PARENT_SCOPE)
endfunction()
function(recursive_dependency module_name)
# Read module CMakeLists.txt and search for ns-3 modules
set(src_cmakelist ${PROJECT_SOURCE_DIR}/src/${module_name}/CMakeLists.txt)
set(contrib_cmakelist ${PROJECT_SOURCE_DIR}/contrib/${module_name}/CMakeLists.txt)
set(contrib FALSE)
if(EXISTS ${src_cmakelist})
file(READ ${src_cmakelist} cmakelists_content)
elseif(EXISTS ${contrib_cmakelist})
file(READ ${contrib_cmakelist} cmakelists_content)
set(contrib TRUE)
else()
message(STATUS "The CMakeLists.txt file for module ${module_name} was not found.")
endif()
filter_libraries(${cmakelists_content} matches)
# Add this visited module dependencies to the dependencies list
if(contrib)
set(contrib_dependency_visited "${contrib_dependency_visited};${module_name}" CACHE INTERNAL "")
set(examples_cmakelist ${contrib_cmakelist})
else()
set(dependency_visited "${dependency_visited};${module_name}" CACHE INTERNAL "")
set(examples_cmakelist ${src_cmakelist})
endif()
# cmake-format: off
# Scan dependencies required by this module examples
#if(${NS3_EXAMPLES})
# string(REPLACE "${module_name}" "${module_name}/examples" examples_cmakelist ${examples_cmakelist})
# if(EXISTS ${examples_cmakelist})
# file(READ ${examples_cmakelist} cmakelists_content)
# filter_libraries(${cmakelists_content} example_matches)
# endif()
#endif()
# cmake-format: on
# For each dependency, call this same function
set(matches "${matches};${example_matches}")
foreach(match ${matches})
if(NOT ((${match} IN_LIST dependency_visited) OR (${match} IN_LIST contrib_dependency_visited)))
recursive_dependency(${match})
endif()
endforeach()
endfunction()
macro(filter_enabled_and_disabled_modules libs_to_build contrib_libs_to_build NS3_ENABLED_MODULES NS3_DISABLED_MODULES)
# Before filtering, we set a variable with all scanned moduled in the src directory
set(scanned_modules ${${libs_to_build}})
# Now that scanning modules finished, we can remove the disabled modules or replace the modules list with the ones in
# the enabled list
if(NS3_ENABLED_MODULES)
# Filter enabled modules
filter_modules(NS3_ENABLED_MODULES libs_to_build "")
filter_modules(NS3_ENABLED_MODULES contrib_libs_to_build "")
# Recursively build a list of module dependendencies
foreach(lib ${${contrib_libs_to_build}})
message(WARNING "contrib ${lib}")
resolve_dependencies(${lib} dependencies contrib_dependencies)
list(APPEND ${contrib_libs_to_build} "${contrib_dependencies}")
list(APPEND ${libs_to_build} "${dependencies}")
unset(dependencies)
unset(contrib_dependencies)
endforeach()
foreach(lib ${${libs_to_build}})
resolve_dependencies(${lib} dependencies contrib_dependencies)
list(APPEND ${libs_to_build} "${dependencies}")
unset(dependencies)
unset(contrib_dependencies)
endforeach()
if(${NS3_TESTS})
list(APPEND ${libs_to_build} test) # force enable test module if NS3_TESTS is enabled
endif()
endif()
if(NS3_DISABLED_MODULES)
filter_modules(NS3_DISABLED_MODULES libs_to_build "NOT")
filter_modules(NS3_DISABLED_MODULES contrib_libs_to_build "NOT")
endif()
# Filter out any eventual duplicates
list(REMOVE_DUPLICATES ${libs_to_build})
list(REMOVE_DUPLICATES ${contrib_libs_to_build})
# Export list with all enabled modules (used to separate ns libraries from non-ns libraries in ns3_module_macros)
set(ns3-all-modules "${${libs_to_build}};${${contrib_libs_to_build}}" CACHE INTERNAL "list with all enabled modules")
endmacro()
# Waf workaround scripts
include(buildsupport/custom_modules/waf_workaround_c4cache.cmake)
include(buildsupport/custom_modules/waf_workaround_buildstatus.cmake)
include(buildsupport/custom_modules/waf_workaround_lock.cmake)
include(buildsupport/custom_modules/waf_workaround_fakeconfig.cmake)