#[[
Copyright (c) 2025-2026 The Johns Hopkins University Applied Physics
Laboratory LLC.

This file is part of the Bundle Protocol Security Library (BSL).

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

This work was performed for the Jet Propulsion Laboratory, California
Institute of Technology, sponsored by the United States Government under
the prime contract 80NM0018D0004 between the Caltech and NASA under
subcontract 1700763.
]]
cmake_minimum_required(VERSION 3.10)

option(BUILD_LIB "Build the library itself" ON)
option(BUILD_SHARED_LIBS "Build using shared libraries" ON)
option(BUILD_DOCS_API "Enable API documentation building" OFF)
option(BUILD_DOCS_MAN "Enable manpage building" OFF)
option(BUILD_MOCK_BPA "Enable building the Mock BPA executable" ON)
option(BUILD_UNITTEST "Enable building unit tests" ON)
option(TEST_MEMCHECK "Enable test runtime memory checking" ON)
option(BUILD_COVERAGE "Enable runtime coverage logging and reporting" OFF)
option(BUILD_FUZZING "Enable building fuzzing executables" OFF)

list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake")

set(CMAKE_EXPORT_COMPILE_COMMANDS ON)

if(NOT PROJECT_VERSION)
    # Ingest the git tag as project version name
    find_package(Git)
    execute_process(
        COMMAND ${GIT_EXECUTABLE} describe --tags --long --dirty
        RESULT_VARIABLE GIT_TAG_EXIT
        OUTPUT_VARIABLE GIT_TAG_REV
        OUTPUT_STRIP_TRAILING_WHITESPACE
        ERROR_QUIET
    )
    if(NOT GIT_TAG_EXIT EQUAL 0)
        execute_process(
            COMMAND ${GIT_EXECUTABLE} describe --always --dirty
            OUTPUT_VARIABLE GIT_COMMIT
            OUTPUT_STRIP_TRAILING_WHITESPACE
        )
        set(GIT_TAG_REV "v0.0.0-0-g${GIT_COMMIT}")
        message(WARNING "No git tag found, marking as ${GIT_TAG_REV}")
    endif()
    # Make version compatible with CMake and RPM needs
    STRING(REGEX REPLACE [[^v([0-9]+\.[0-9]+\.[0-9]+).*]] [[\1]] GIT_TAG_VERS ${GIT_TAG_REV})
    STRING(REGEX REPLACE [[^v[0-9\.]*-(.+)]] [[\1]] GIT_TAG_MOD ${GIT_TAG_REV})
    STRING(REPLACE "-" "." GIT_TAG_MOD ${GIT_TAG_MOD})
    message(STATUS "Using version marking ${GIT_TAG_VERS} - ${GIT_TAG_MOD}")
    set(PROJECT_VERSION ${GIT_TAG_VERS})
endif(NOT PROJECT_VERSION)

project(bsl
    LANGUAGES C CXX
    VERSION ${PROJECT_VERSION}
)

# Language options
set(CMAKE_C_STANDARD 99)
set(CMAKE_C_STANDARD_REQUIRED ON)
set(CMAKE_C_EXTENSIONS OFF)
# Force specific POSIX compatibility version
add_definitions(
  -D_XOPEN_SOURCE
  -D_POSIX_C_SOURCE=200809L
)
# Generic warn/error options
add_compile_options(
    -Wall
    $<$<COMPILE_LANGUAGE:C>:-Wextra>
    $<$<COMPILE_LANGUAGE:C>:-Wpedantic>
    $<$<COMPILE_LANGUAGE:C>:-Werror>
    $<$<COMPILE_LANGUAGE:C>:-Wformat>
    -Wshadow -Wpointer-arith -Wstrict-prototypes
    -Wmissing-prototypes -Wredundant-decls -Wcast-align
    -Wformat=2
    -fno-strict-aliasing -Werror=format-security -fno-common
    -Wstrict-aliasing=2
    # -Wconversion
    -ffunction-sections
    -fdata-sections
    -fno-omit-frame-pointer
    $<$<COMPILE_LANGUAGE:C>:-Wswitch-enum>
)
if(CMAKE_C_COMPILER_ID MATCHES "GNU")
    set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -O0")
    set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -O0")
    add_compile_options(
        -Wlogical-op -Wduplicated-branches
        $<$<COMPILE_LANGUAGE:CXX>:-fno-enforce-eh-specs>
        $<$<COMPILE_LANGUAGE:CXX>:-fnothrow-opt>
    )
elseif(CMAKE_C_COMPILER_ID MATCHES "Clang")
    set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -gdwarf-4")
endif()

add_custom_target(clang-tidy
  COMMAND clang-tidy -quiet --config-file=.clang-tidy src/*.h src/backend/*.h src/security_context/*.h src/policy_provider/*.h -- -I src -I deps/mlib -I deps/QCBOR/inc 
          && clang-tidy -quiet --config-file=.clang-tidy src/backend/*.c src/security_context/*.c src/policy_provider/*.c -- -I src -I deps/mlib -I deps/QCBOR/inc 
  WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
  COMMENT "Running clang-tidy..."
)

if(BUILD_LIB)
    find_package(MLIB REQUIRED)
    find_package(QCBOR REQUIRED)
    find_package(OpenSSL REQUIRED)
    find_package(Threads REQUIRED)
    find_package(Jansson REQUIRED)
endif(BUILD_LIB)

include(CheckSymbolExists)
set(CMAKE_REQUIRED_DEFINITIONS "-D_POSIX_C_SOURCE=200809L")
check_symbol_exists(timespec_get "time.h" HAVE_TIMESPEC_GET)
check_symbol_exists(clock_gettime "time.h" HAVE_CLOCK_GETTIME)

include(CheckTypeSize)
check_type_size("int" SIZE_INT LANGUAGE C)
check_type_size("size_t" SIZE_SIZET LANGUAGE C)

if(BUILD_UNITTEST)
  if(TEST_MEMCHECK)
    find_package(valgrind REQUIRED)

    find_program(MEMCHECK_CMD valgrind REQUIRED)
    message(STATUS "Using valgrind memcheck for tests: ${MEMCHECK_CMD}")
    set(MEMCHECK_OPTIONS
      "--tool=memcheck"
      "--trace-children=yes"
      "--track-origins=yes"
      "--leak-check=full" "--show-leak-kinds=all"
      "--suppressions=${CMAKE_CURRENT_SOURCE_DIR}/resources/memcheck.supp"
      "--error-exitcode=2"
    )
    # Arguments as list into global scope for Findunitytools.cmake
    set(TEST_EXEC_PREFIX "${MEMCHECK_CMD}" ${MEMCHECK_OPTIONS})
  endif(TEST_MEMCHECK)
endif(BUILD_UNITTEST)
if(BUILD_FUZZING)
  find_package(libfuzzer)
endif(BUILD_FUZZING)
if(BUILD_COVERAGE)
  include(CodeCoverage)
  append_coverage_compiler_flags()

  set(COVERAGE_EXCLUDES
    "${CMAKE_CURRENT_SOURCE_DIR}/deps/*"
    "${CMAKE_CURRENT_SOURCE_DIR}/testroot/*"
    "${CMAKE_CURRENT_BINARY_DIR}/test/*"
  )
  set(GCOVR_ADDITIONAL_ARGS
    --gcov-ignore-parse-errors=negative_hits.warn_once_per_file
    --gcov-keep
  )
  setup_target_for_coverage_gcovr_xml(
    NAME coverage-xml
    BASE_DIRECTORY "${PROJECT_SOURCE_DIR}"
  )
  setup_target_for_coverage_gcovr_xml(
    NAME coverage-sonarqube
    BASE_DIRECTORY "${PROJECT_SOURCE_DIR}"
    SONARQUBE
  )
  setup_target_for_coverage_gcovr_html(
    NAME coverage-html
    BASE_DIRECTORY "${PROJECT_SOURCE_DIR}"
  )
endif(BUILD_COVERAGE)

include(CTest)
set(CMAKE_CTEST_ARGUMENTS
  --output-junit testresults.xml
  --output-on-failure
)

# Install config used by tall targets
include(GNUInstallDirs)

if(BUILD_LIB)
    add_subdirectory(src)
endif(BUILD_LIB)

set(TEST_INSTALL_PREFIX "${CMAKE_INSTALL_LIBEXECDIR}/${PROJECT_NAME}")
add_subdirectory(test)
add_subdirectory(docs)


add_custom_command(
    DEPENDS CMakeGraphVizOptions.cmake
    OUTPUT dep_chart.dot
    COMMAND ${CMAKE_COMMAND} "--graphviz=dep_chart.dot" .
)
find_program(CMD_DOT NAMES dot)
message(STATUS "Found dot at ${CMD_DOT}")
if(CMD_DOT)
    add_custom_command(
        DEPENDS dep_chart.dot
        OUTPUT dep_chart.svg
        COMMAND ${CMD_DOT} -Tsvg dep_chart.dot -o dep_chart.svg
    )
    add_custom_command(
        DEPENDS dep_chart.dot
        OUTPUT dep_chart.png
        COMMAND ${CMD_DOT} -Tpng dep_chart.dot -o dep_chart.png
    )
    add_custom_target(graphviz
        DEPENDS dep_chart.svg dep_chart.png
    )
endif(CMD_DOT)
