# Copyright (C) 2024 KeePassXC Team # Copyright (C) 2010 Felix Geyer # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 2 or (at your option) # version 3 of the License. # # 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, see . cmake_minimum_required(VERSION 3.16.0) project(KeePassXC) set(APP_ID "org.keepassxc.${PROJECT_NAME}") # Version Number set(KEEPASSXC_VERSION_MAJOR "2") set(KEEPASSXC_VERSION_MINOR "8") set(KEEPASSXC_VERSION_PATCH "0") set(KEEPASSXC_VERSION "${KEEPASSXC_VERSION_MAJOR}.${KEEPASSXC_VERSION_MINOR}.${KEEPASSXC_VERSION_PATCH}") string(TOLOWER "${CMAKE_BUILD_TYPE}" CMAKE_BUILD_TYPE_LOWER) # CMake Modules set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} ${CMAKE_CURRENT_SOURCE_DIR}/cmake) include(CMakeToolsHelpers OPTIONAL) # Support Visual Studio Code include(FeatureSummary) include(KPXCMacDeployHelpers) include(CLangFormat) include(CompilerFlags) include(CheckCCompilerFlag) include(CheckCXXCompilerFlag) include(CheckCXXSourceCompiles) # Build Scope option(WITH_TESTS "Enable building of unit tests" ON) option(WITH_GUI_TESTS "Enable building of GUI tests" OFF) option(WITH_WARN_DEPRECATED "Development only: warn about deprecated methods, including Qt." OFF) option(WITH_ASAN "Enable address sanitizer checks (Linux / macOS only)" OFF) option(WITH_COVERAGE "Use to build with coverage tests (GCC only)." OFF) option(WITH_APP_BUNDLE "Enable Application Bundle for macOS" ON) option(WITH_CCACHE "Use ccache for build" OFF) option(WITH_X11 "Enable building with X11 dependencies" ON) # Advanced Features Control option(KPXC_MINIMAL "Build KeePassXC with the minimal feature set required for basic usage" OFF) option(KPXC_FEATURE_BROWSER "Browser integration and passkeys support" ON) option(KPXC_FEATURE_SSHAGENT "SSH Agent integration" ON) option(KPXC_FEATURE_FDOSECRETS "freedesktop.org Secret Service integration; replace system keyring" ON) if(KPXC_MINIMAL) # Disable advanced features in minimal mode set(KPXC_FEATURE_BROWSER OFF) set(KPXC_FEATURE_SSHAGENT OFF) set(KPXC_FEATURE_FDOSECRETS OFF) endif() # Minor Feature Flags option(KPXC_FEATURE_NETWORK "Include code that reaches out to external networks (e.g. downloading icons)" ON) option(KPXC_FEATURE_UPDATES "Include automatic update checks; disable for managed distributions" ON) option(KPXC_FEATURE_DOCS "Build offline documentation; requires asciidoctor tool" ON) # Reconcile update feature with overall network feature if(NOT KPXC_FEATURE_NETWORK AND KPXC_FEATURE_UPDATES) message(STATUS "Disabling KPXC_FEATURE_UPDATES because KPXC_FEATURE_NETWORK is disabled") set(KPXC_FEATURE_UPDATES OFF) endif() # FDO Secrets is only available on Linux if(NOT UNIX OR APPLE OR HAIKU) set(KPXC_FEATURE_FDOSECRETS OFF) endif() # Define feature summaries add_feature_info("Browser" KPXC_FEATURE_BROWSER "Browser integration and passkeys support") add_feature_info("SSH Agent" KPXC_FEATURE_SSHAGENT "SSH Agent integration") if(UNIX AND NOT APPLE) add_feature_info("Secret Service" KPXC_FEATURE_FDOSECRETS "Replace system keyring with freedesktop.org Secret Service integration") endif() add_feature_info("Networking" KPXC_FEATURE_NETWORK "Code that can reach out to external networks is included (e.g. downloading icons)") add_feature_info("Update Checks" KPXC_FEATURE_UPDATES "Periodic update checks can be performed") add_feature_info("Documentation" KPXC_FEATURE_DOCS "Offline documentation") # Retrieve git HEAD revision hash set(GIT_HEAD_OVERRIDE "" CACHE STRING "Manually set the Git HEAD hash when missing (eg, when no .git folder exists)") execute_process(COMMAND git rev-parse --short=7 HEAD WORKING_DIRECTORY ${CMAKE_SOURCE_DIR} OUTPUT_VARIABLE GIT_HEAD ERROR_QUIET) string(STRIP "${GIT_HEAD}" GIT_HEAD) if(GIT_HEAD STREQUAL "" AND NOT GIT_HEAD_OVERRIDE STREQUAL "") string(SUBSTRING "${GIT_HEAD_OVERRIDE}" 0 7 GIT_HEAD) elseif(EXISTS ${CMAKE_SOURCE_DIR}/.gitrev) file(READ ${CMAKE_SOURCE_DIR}/.gitrev GIT_HEAD) endif() message(STATUS "Found Git HEAD Revision: ${GIT_HEAD}\n") # KeePassXC Versioning and Build Type set(OVERRIDE_VERSION "" CACHE STRING "Override the KeePassXC Version for Snapshot builds") set(KEEPASSXC_BUILD_TYPE "Snapshot" CACHE STRING "Set KeePassXC build type to distinguish between stable releases and snapshots") set_property(CACHE KEEPASSXC_BUILD_TYPE PROPERTY STRINGS Snapshot Release) # Check if on a tag or has .version file, if so build as a release execute_process(COMMAND git tag --points-at HEAD WORKING_DIRECTORY ${CMAKE_SOURCE_DIR} OUTPUT_VARIABLE GIT_TAG ERROR_QUIET) string(REGEX REPLACE "latest" "" GIT_TAG "${GIT_TAG}") if(GIT_TAG MATCHES "[0-9]+\.[0-9]+\.[0-9]+") string(STRIP "${GIT_TAG}" GIT_TAG) set(OVERRIDE_VERSION ${GIT_TAG}) elseif(EXISTS ${CMAKE_SOURCE_DIR}/.version) file(READ ${CMAKE_SOURCE_DIR}/.version OVERRIDE_VERSION) endif() string(REGEX REPLACE "(\r?\n)+" "" OVERRIDE_VERSION "${OVERRIDE_VERSION}") if(OVERRIDE_VERSION) if(OVERRIDE_VERSION MATCHES "^[\\.0-9]+$") set(KEEPASSXC_BUILD_TYPE "Release") set(KEEPASSXC_VERSION ${OVERRIDE_VERSION}) else() set(KEEPASSXC_BUILD_TYPE "Snapshot") endif() endif() if(KEEPASSXC_BUILD_TYPE STREQUAL "Release") set(KEEPASSXC_BUILD_TYPE_RELEASE ON) else() set(KEEPASSXC_BUILD_TYPE_SNAPSHOT ON) set(KEEPASSXC_VERSION "${KEEPASSXC_VERSION}-snapshot") endif() message(STATUS "Setting up build for KeePassXC v${KEEPASSXC_VERSION}\n") # Distribution info set(KEEPASSXC_DIST_TYPE "Native" CACHE STRING "KeePassXC Distribution Type") set_property(CACHE KEEPASSXC_DIST_TYPE PROPERTY STRINGS Snap AppImage Flatpak Native) if(KEEPASSXC_DIST_TYPE STREQUAL "Snap") set(KEEPASSXC_DIST_SNAP ON) elseif(KEEPASSXC_DIST_TYPE STREQUAL "AppImage") set(KEEPASSXC_DIST_APPIMAGE ON) elseif(KEEPASSXC_DIST_TYPE STREQUAL "Flatpak") set(KEEPASSXC_DIST_FLATPAK ON) endif() # Standards set(CMAKE_C_STANDARD 99) set(CMAKE_CXX_STANDARD 20) set(CMAKE_CXX_STANDARD_REQUIRED ON) # Compiler Features if(APPLE) # Perform the platform checks before applying the stricter compiler flags. # Otherwise the kSecAccessControlTouchIDCurrentSet deprecation warning will result in an error. try_compile(XC_APPLE_COMPILER_SUPPORT_BIOMETRY ${CMAKE_BINARY_DIR}/macos-trycompile/ ${CMAKE_SOURCE_DIR}/cmake/compiler-checks/control_biometry_support.mm) message(STATUS "Biometry compiler support: ${XC_APPLE_COMPILER_SUPPORT_BIOMETRY}") try_compile(XC_APPLE_COMPILER_SUPPORT_TOUCH_ID ${CMAKE_BINARY_DIR}/macos-trycompile/ ${CMAKE_SOURCE_DIR}/cmake/compiler-checks/control_touch_id_support.mm) message(STATUS "Touch ID compiler support: ${XC_APPLE_COMPILER_SUPPORT_TOUCH_ID}") try_compile(XC_APPLE_COMPILER_SUPPORT_WATCH ${CMAKE_BINARY_DIR}/macos-trycompile/ ${CMAKE_SOURCE_DIR}/cmake/compiler-checks/control_watch_support.mm) message(STATUS "Apple watch compiler support: ${XC_APPLE_COMPILER_SUPPORT_WATCH}") try_compile(HAVE_PT_DENY_ATTACH ${CMAKE_BINARY_DIR}/macos-trycompile/ ${CMAKE_SOURCE_DIR}/cmake/compiler-checks/ptrace_deny_attach.cpp) endif() if(UNIX) check_cxx_source_compiles("#include int main() { prctl(PR_SET_DUMPABLE, 0); return 0; }" HAVE_PR_SET_DUMPABLE) check_cxx_source_compiles("#include int main() { return 0; }" HAVE_MALLOC_H) check_cxx_source_compiles("#include int main() { malloc_usable_size(NULL); return 0; }" HAVE_MALLOC_USABLE_SIZE) check_cxx_source_compiles("#include int main() { struct rlimit limit; limit.rlim_cur = 0; limit.rlim_max = 0; setrlimit(RLIMIT_CORE, &limit); return 0; }" HAVE_RLIMIT_CORE) endif() # ccache support if(WITH_CCACHE) find_program(CCACHE_FOUND ccache) if(NOT CCACHE_FOUND) message(FATAL_ERROR "ccache requested but cannot be found.") endif() set_property(GLOBAL PROPERTY RULE_LAUNCH_COMPILE ${CCACHE_FOUND}) endif() # Create position independent code for shared libraries and executables set(CMAKE_POSITION_INDEPENDENT_CODE ON) if(${CMAKE_VERSION} VERSION_GREATER_EQUAL "3.14.0") cmake_policy(SET CMP0083 NEW) include(CheckPIESupported) check_pie_supported() endif() if("${CMAKE_SIZEOF_VOID_P}" EQUAL "4") set(IS_32BIT TRUE) endif() set(CLANG_COMPILER_ID_REGEX "^(Apple)?[Cc]lang$") if("${CMAKE_C_COMPILER}" MATCHES "clang$" OR "${CMAKE_EXTRA_GENERATOR_C_SYSTEM_DEFINED_MACROS}" MATCHES "__clang__" OR "${CMAKE_C_COMPILER_ID}" MATCHES ${CLANG_COMPILER_ID_REGEX}) set(CMAKE_COMPILER_IS_CLANG 1) endif() if("${CMAKE_CXX_COMPILER}" MATCHES "clang(\\+\\+)?$" OR "${CMAKE_EXTRA_GENERATOR_CXX_SYSTEM_DEFINED_MACROS}" MATCHES "__clang__" OR "${CMAKE_CXX_COMPILER_ID}" MATCHES ${CLANG_COMPILER_ID_REGEX}) set(CMAKE_COMPILER_IS_CLANGXX 1) endif() # Compiler Flags add_gcc_compiler_flags("-fno-common") find_package(OpenMP) if(OpenMP_FOUND) add_gcc_compiler_cflags(${OpenMP_C_FLAGS}) add_gcc_compiler_cxxflags(${OpenMP_CXX_FLAGS}) endif() add_gcc_compiler_flags("-Wall -Wextra -Wundef -Wpointer-arith -Wno-long-long") add_gcc_compiler_flags("-Wformat=2 -Wmissing-format-attribute") add_gcc_compiler_flags("-fvisibility=hidden") add_gcc_compiler_cxxflags("-fvisibility-inlines-hidden") if(CMAKE_BUILD_TYPE_LOWER STREQUAL "debug") check_add_gcc_compiler_flag("-Wshadow-compatible-local") check_add_gcc_compiler_flag("-Wshadow-local") add_gcc_compiler_flags("-Werror") # C++20 marks enum arithmetic as deprecated, but we use it in Botan and Qt5 add_gcc_compiler_cxxflags("-Wno-deprecated-enum-enum-conversion -Wno-error=deprecated ") endif() if (NOT HAIKU) if((CMAKE_COMPILER_IS_GNUCXX AND CMAKE_CXX_COMPILER_VERSION VERSION_GREATER 4.8.999) OR CMAKE_COMPILER_IS_CLANGXX) add_gcc_compiler_flags("-fstack-protector-strong") else() add_gcc_compiler_flags("-fstack-protector --param=ssp-buffer-size=4") endif() endif() add_gcc_compiler_cxxflags("-Wnon-virtual-dtor -Wold-style-cast -Woverloaded-virtual") add_gcc_compiler_cflags("-Wchar-subscripts -Wwrite-strings") if(WITH_ASAN) if(NOT (CMAKE_SYSTEM_NAME STREQUAL "Linux" OR APPLE)) message(FATAL_ERROR "WITH_ASAN is only supported on Linux / macOS at the moment.") endif() add_gcc_compiler_flags("-fsanitize=address -DWITH_ASAN") if(CMAKE_SYSTEM_NAME STREQUAL "Linux") if(NOT (CMAKE_COMPILER_IS_GNUCXX AND CMAKE_CXX_COMPILER_VERSION VERSION_LESS 4.9)) add_gcc_compiler_flags("-fsanitize=leak -DWITH_LSAN") endif() endif() endif() if(CMAKE_BUILD_TYPE_LOWER MATCHES "(release|relwithdebinfo|minsizerel)") add_gcc_compiler_flags("-U_FORTIFY_SOURCE -D_FORTIFY_SOURCE=2") endif() check_add_gcc_compiler_flag("-Werror=format-security") check_add_gcc_compiler_flag("-Werror=implicit-function-declaration" C) check_add_gcc_compiler_flag("-Wcast-align") if(UNIX AND NOT APPLE) check_add_gcc_compiler_flag("-Qunused-arguments") set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,--as-needed -Wl,--no-undefined") set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,-z,relro,-z,now -pie") set(CMAKE_MODULE_LINKER_FLAGS "${CMAKE_MODULE_LINKER_FLAGS} -Wl,--as-needed") set(CMAKE_MODULE_LINKER_FLAGS "${CMAKE_MODULE_LINKER_FLAGS} -Wl,-z,relro,-z,now") endif() check_cxx_compiler_flag("-fsized-deallocation" CXX_HAS_fsized_deallocation) if(CXX_HAS_fsized_deallocation) # Do additional check: the deallocation functions must be there too. set(CMAKE_REQUIRED_FLAGS "-fsized-deallocation") check_cxx_source_compiles("#include int main() { void * ptr = nullptr; std::size_t size = 1; ::operator delete(ptr, size); }" HAVE_DEALLOCATION_FUNCTIONS) if(HAVE_DEALLOCATION_FUNCTIONS) check_add_gcc_compiler_flag("-fsized-deallocation" CXX) endif() unset(CMAKE_REQUIRED_FLAGS) endif() if(APPLE AND CMAKE_COMPILER_IS_CLANGXX) add_gcc_compiler_cxxflags("-stdlib=libc++") endif() if(WITH_WARN_DEPRECATED) add_definitions(-DQT_DEPRECATED_WARNINGS) else() add_definitions(-DQT_NO_DEPRECATED_WARNINGS) add_gcc_compiler_cxxflags("-Wno-deprecated-declarations") endif() # MSVC specific options if (MSVC) if(MSVC_TOOLSET_VERSION LESS 142) message(FATAL_ERROR "Only Microsoft Visual Studio 2019 and newer are supported!") endif() # Turn on multi-processor support and faster PDB generation (/Zf) add_compile_options(/permissive- /utf-8 /MP /Zf) # Enable built-in ASAN add_compile_definitions(/fsanitize=address) # Enable high entropy ASLR on release builds if(CMAKE_BUILD_TYPE_LOWER STREQUAL "release") add_compile_options(/guard:cf) add_link_options(/DYNAMICBASE /HIGHENTROPYVA /GUARD:CF) endif() elseif(MINGW) # Enable high entropy ASLR on release builds if(CMAKE_BUILD_TYPE_LOWER STREQUAL "release") set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,--nxcompat -Wl,--dynamicbase -Wl,--high-entropy-va") set(CMAKE_MODULE_LINKER_FLAGS "${CMAKE_MODULE_LINKER_FLAGS} -Wl,--nxcompat -Wl,--dynamicbase -Wl,--high-entropy-va") endif() endif() if(WIN32) set(PROGNAME KeePassXC) set(CLI_INSTALL_DIR ".") set(PROXY_INSTALL_DIR ".") set(BIN_INSTALL_DIR ".") set(PLUGIN_INSTALL_DIR ".") set(DATA_INSTALL_DIR "share") elseif(APPLE AND WITH_APP_BUNDLE) set(PROGNAME KeePassXC) set(BUNDLE_INSTALL_DIR "${PROGNAME}.app/Contents") set(CMAKE_INSTALL_MANDIR "${BUNDLE_INSTALL_DIR}/Resources/man") set(CLI_INSTALL_DIR "${BUNDLE_INSTALL_DIR}/MacOS") set(PROXY_INSTALL_DIR "${BUNDLE_INSTALL_DIR}/MacOS") set(BIN_INSTALL_DIR "${BUNDLE_INSTALL_DIR}/MacOS") set(PLUGIN_INSTALL_DIR "${BUNDLE_INSTALL_DIR}/PlugIns") set(DATA_INSTALL_DIR "${BUNDLE_INSTALL_DIR}/Resources") add_definitions(-DWITH_APP_BUNDLE) else() include(GNUInstallDirs) set(PROGNAME keepassxc) set(CLI_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}") set(PROXY_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}") set(BIN_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}") set(PLUGIN_INSTALL_DIR "${CMAKE_INSTALL_LIBDIR}/keepassxc") set(DATA_INSTALL_DIR "${CMAKE_INSTALL_DATADIR}/keepassxc") endif() if(WITH_COVERAGE) # Include code coverage, use with -DCMAKE_BUILD_TYPE=Debug include(CodeCoverage) append_coverage_compiler_flags() set(COVERAGE_EXCLUDES "'^(.+/)?thirdparty/.*'" "'^(.+/)?main\\.cpp$$'" "'^(.+/)?cli/keepassxc-cli\\.cpp$$'" "'^(.+/)?proxy/keepassxc-proxy\\.cpp$$'") if(WITH_COVERAGE AND CMAKE_COMPILER_IS_CLANGXX) set(MAIN_BINARIES "$" "$" "$") setup_target_for_coverage_llvm( NAME coverage BINARY ${MAIN_BINARIES} SOURCES_ROOT ${CMAKE_SOURCE_DIR}/src ) else() setup_target_for_coverage_gcovr( NAME coverage SOURCES_ROOT ${CMAKE_SOURCE_DIR}/src ) endif() endif() # Find Qt set(QT_COMPONENTS Core Network Concurrent Gui Svg Widgets Test LinguistTools) if(UNIX AND NOT APPLE) if(WITH_X11) list(APPEND QT_COMPONENTS X11Extras) endif() find_package(Qt5 COMPONENTS ${QT_COMPONENTS} DBus REQUIRED) elseif(APPLE) find_package(Qt5 COMPONENTS ${QT_COMPONENTS} REQUIRED HINTS /usr/local/opt/qt@5/lib/cmake /usr/local/Cellar/qt@5/*/lib/cmake /opt/homebrew/opt/qt@5/lib/cmake ENV PATH) find_package(Qt5 COMPONENTS MacExtras HINTS /usr/local/opt/qt@5/lib/cmake /usr/local/Cellar/qt@5/*/lib/cmake /opt/homebrew/opt/qt@5/lib/cmake ENV PATH) else() find_package(Qt5 COMPONENTS ${QT_COMPONENTS} REQUIRED) endif() # Minimum Qt version check if(Qt5Core_VERSION VERSION_LESS "5.12.0") message(FATAL_ERROR "Qt version 5.12.0 or higher is required") endif() # C++20 is not supported before Qt 5.15.0 if(Qt5Core_VERSION VERSION_LESS "5.15.0") set(CMAKE_CXX_STANDARD 17) endif() # Process moc automatically set(CMAKE_AUTOMOC ON) # Process .ui files automatically set(CMAKE_AUTOUIC ON) # Process .qrc files automatically set(CMAKE_AUTORCC ON) add_definitions(-DQT_NO_EXCEPTIONS -DQT_STRICT_ITERATORS -DQT_NO_CAST_TO_ASCII) if(NOT CMAKE_BUILD_TYPE_LOWER STREQUAL "debug") add_definitions(-DQT_NO_DEBUG_OUTPUT) endif() get_filename_component(Qt5_PREFIX ${Qt5_DIR}/../../.. REALPATH) if(APPLE) # Add includes under Qt5 Prefix in case Qt6 is also installed include_directories(SYSTEM ${Qt5_PREFIX}/include) set(CMAKE_MACOSX_RPATH TRUE) find_program(MACDEPLOYQT_EXE macdeployqt HINTS ${Qt5_PREFIX}/bin ${Qt5_PREFIX}/tools/qt5/bin ENV PATH) if(NOT MACDEPLOYQT_EXE) message(FATAL_ERROR "macdeployqt is required to build on macOS") endif() message(STATUS "Using macdeployqt: ${MACDEPLOYQT_EXE}") set(MACDEPLOYQT_EXTRA_BINARIES "") elseif(WIN32) find_program(WINDEPLOYQT_EXE windeployqt HINTS ${Qt5_PREFIX}/bin ${Qt5_PREFIX}/tools/qt5/bin ENV PATH) if(NOT WINDEPLOYQT_EXE) message(FATAL_ERROR "windeployqt is required to build on Windows") endif() message(STATUS "Using windeployqt: ${WINDEPLOYQT_EXE}") endif() # Find Botan # TODO: Increase minimum to 2.19.1 and drop Argon2 package find_package(Botan REQUIRED) if(BOTAN_VERSION VERSION_GREATER_EQUAL "3.0.0") set(WITH_BOTAN3 TRUE) elseif(BOTAN_VERSION VERSION_LESS "2.12.0") # Check for minimum Botan version message(FATAL_ERROR "Botan 2.12.0 or higher is required") endif() include_directories(SYSTEM ${BOTAN_INCLUDE_DIR}) # Find Argon2 -- Botan 2.18 and below does not support threaded Argon2 find_library(ARGON2_LIBRARIES NAMES argon2) find_path(ARGON2_INCLUDE_DIR NAMES argon2.h PATH_SUFFIXES local/include) include_directories(SYSTEM ${ARGON2_INCLUDE_DIR}) # Find zlib find_package(ZLIB REQUIRED) if(ZLIB_VERSION_STRING VERSION_LESS "1.2.0") message(FATAL_ERROR "zlib 1.2.0 or higher is required to use the gzip format") endif() include_directories(SYSTEM ${ZLIB_INCLUDE_DIR}) # Find Minizip find_package(Minizip REQUIRED) # Find PCSC and LibUSB for hardware key support find_package(PCSC REQUIRED) include_directories(SYSTEM ${PCSC_INCLUDE_DIRS}) if(UNIX AND NOT APPLE) find_library(LIBUSB_LIBRARIES NAMES usb-1.0 REQUIRED) find_path(LIBUSB_INCLUDE_DIR NAMES libusb.h PATH_SUFFIXES "libusb-1.0" "libusb" REQUIRED) include_directories(SYSTEM ${LIBUSB_INCLUDE_DIR}) # For PolKit QuickUnlock find_library(KEYUTILS_LIBRARIES NAMES keyutils REQUIRED) endif() # Find zxcvbn or use the bundled version find_library(ZXCVBN_LIBRARIES zxcvbn) if(NOT ZXCVBN_LIBRARIES) add_subdirectory(src/thirdparty/zxcvbn) set(ZXCVBN_LIBRARIES zxcvbn) endif() # Add KeePassXC sources and tests add_subdirectory(src) add_subdirectory(share) if(WITH_TESTS) enable_testing() add_subdirectory(tests) endif() if(KPXC_FEATURE_DOCS) add_subdirectory(docs) endif() # Print out summary information message(STATUS "") feature_summary(QUIET_ON_EMPTY WHAT ENABLED_FEATURES DESCRIPTION "Enabled features:") feature_summary(QUIET_ON_EMPTY WHAT DISABLED_FEATURES DESCRIPTION "Disabled features:")