diff --git a/.gitignore b/.gitignore
index d6efb9cf7..eb4610723 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,6 +1,14 @@
+# PyCharm and CLion
+/.idea/*
+!/.idea/runConfigurations
+!/.idea/cmake.xml
+!/.idea/codeStyles
+
+# Eclipse
.cproject
.project
.settings
.metadata
/build*
+/cmake-build*
diff --git a/.idea/codeStyles/Project.xml b/.idea/codeStyles/Project.xml
new file mode 100644
index 000000000..0f3b1a4bf
--- /dev/null
+++ b/.idea/codeStyles/Project.xml
@@ -0,0 +1,14 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/codeStyles/codeStyleConfig.xml b/.idea/codeStyles/codeStyleConfig.xml
new file mode 100644
index 000000000..79ee123c2
--- /dev/null
+++ b/.idea/codeStyles/codeStyleConfig.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 335c0f7bb..f83c5078f 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -8,6 +8,170 @@ and this project adheres to [Semantic Versioning](http://semver.org/).
# [unreleased]
+# [v5.0.0] 25.07.2022
+
+## Changes
+
+- Renamed auto-formatting script to `auto-formatter.sh` and made it more robust.
+ If `cmake-format` is installed, it will also auto-format the `CMakeLists.txt` files now.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/625
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/626
+- Bump C++ required version to C++17. Every project which uses the FSFW and every modern
+ compiler supports it
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/622
+- New CMake option `FSFW_HAL_LINUX_ADD_LIBGPIOD` to specifically exclude `gpiod` code.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/572
+- HAL Devicehandlers: Periodic printout is run-time configurable now
+- `oneShotAction` flag in the `TestTask` class is not static anymore
+- `SimpleRingBuffer::writeData` now checks if the amount is larger than the total size of the
+ Buffer and rejects such writeData calls with `HasReturnvaluesIF::RETURN_FAILED`
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/586
+- Major update for version handling, using `git describe` to fetch version information with git.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/601
+ - Add helper functions provided by [`cmake-modules`](https://github.com/bilke/cmake-modules)
+ manually now. Those should not change too often and only a small subset is needed
+ - Separate folder for easier update and for distinction
+ - LICENSE file included
+ - use `int` for version numbers to allow unset or uninitialized version
+ - Initialize Version object with numbers set to -1
+ - Instead of hardcoding the git hash, it is now retrieved from git
+ - `Version` now allows specifying additional version information like the git SHA1 hash and the
+ versions since the last tag
+ - Additional information is set to the last part of the git describe output for `FSFW_VERSION` now.
+ - Version still need to be hand-updated if the FSFW is not included as a submodule for now.
+- IPC Message Queue Handling: Allow passing an optional `MqArgs` argument into the MessageQueue
+ creation call. It allows passing context information and an arbitrary user argument into
+ the message queue. Also streamlined and simplified `MessageQueue` implementation for all OSALs
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/583
+- Internal API change: Moved the `fsfw_hal` to the `src` folder and integration and internal
+ tests part of `fsfw_tests` to `src`. Unittests are now in a dedicated folder called `unittests`
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/653
+
+### Task Module Refactoring
+
+PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/636
+
+**Refactoring general task code**
+
+- There was a lot of duplicate/boilerplate code inside the individual task IF OSAL implementations.
+ Remove it by introducing base classes `PeriodicTaskBase` and `FixedTimeslotTaskBase`.
+
+**Refactor PeriodicTaskIF**
+
+- Convert `virtual ReturnValue_t addComponent(object_id_t object)` to
+ `virtual ReturnValue_t addComponent(object_id_t object, uint8_t opCode = 0)`, allowing to pass
+ the operation code passed to `performOperation`. Updated API taking
+ an `ExecutableObjectIF` accordingly
+
+**Refactor FixedTimeslotTaskIF**
+
+- Add additional `addSlot` function which takes an `ExecutableObjectIF` pointer and its Object ID
+
+**Refactor FixedSequenceSlot**
+
+- Introduce typedef `CustomCheckFunc` for `ReturnValue_t (*customCheckFunction)(const SlotList&)`.
+- Convert `ReturnValue_t (*customCheckFunction)(const SlotList&)` to
+ `ReturnValue_t (*customCheckFunction)(const SlotList&, void*)`, allowing arbitrary user arguments
+ for the custom checker
+
+**Linux Task Module**
+
+- Use composition instead of inheritance for the `PeriodicPosixTask` and make the `PosixTask` a
+ member of the class
+
+### HAL
+
+- HAL Linux Uart: Baudrate and bits per word are enums now, avoiding misconfigurations
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/585
+- HAL Linux SPI: Set the Clock Default State when setting new SPI speed
+ and mode
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/573
+- GPIO HAL: `Direction`, `GpioOperation` and `Levels` are enum classes now, which prevents
+ name clashes with Windows defines.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/572
+- HAL Linux Uart: Baudrate and bits per word are enums now, avoiding misconfigurations
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/585
+
+### Time
+
+PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/584 and
+https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/593
+
+- `timeval` to `TimeOfDay_t`
+- Added Mutex for gmtime calls: (compare http://www.opengate.at/blog/2020/01/timeless/)
+- Moved the statics used by Clock in ClockCommon.cpp to this file
+- Better check for leap seconds
+- Added Unittests for Clock (only getter)
+
+### Power
+
+- `PowerSwitchIF`: Remove `const` specifier from `sendSwitchCommand` and `sendFuseOnCommand` and
+ also specify a `ReturnValue_t` return type
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/590
+- Extend `PowerSwitcher` module to optionally check current state when calling `turnOn` or
+ `turnOff`. Tis can be helpful to avoid commanding switches which do not need commanding
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/590
+
+## Removed
+
+- Removed the `HkSwitchHelper`. This module should not be needed anymore, now that the local
+ datapools have been implemented.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/557
+
+## Additions
+
+- New constructor for PoolEntry which allows to simply specify the length of the pool entry.
+ This is also the new default constructor for scalar value with 0 as an initial value
+- Added options for CI/CD builds: `FSFW_CICD_BUILD`. This allows the source code to know
+ whether it is running in CI/CD
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/623
+- Basic `clion` support: Update `.gitignore` and add some basic run configurations
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/625
+- LTO support: Allow using LTO/IPO by setting `FSFW_ENABLE_LTO=1`. CMake is able to detect whether
+ the user compiler supports IPO/LPO. LTO is on by default now. Most modern compilers support it,
+ can make good use of it and it usually makes the code faster and/or smaller.
+ After some more research:
+ Enabling LTO will actually cause the compiler to only produce thin LTO by adding
+ `-flto -fno-fat-lto-objects` to the compiler options. I am not sure this is an ideal choice
+ because if an application linking against the FSFW does not use LTO, there can be compile
+ issues (e.g. observed when compiling the FSFW tests without LTO). This is a known issue as
+ can be seen in the multiple CMake issues for it:
+ - https://gitlab.kitware.com/cmake/cmake/-/issues/22913,
+ - https://gitlab.kitware.com/cmake/cmake/-/issues/16808,
+ - https://gitlab.kitware.com/cmake/cmake/-/issues/21696
+ Easiest solution for now: Keep this option OFF by default.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/616
+- Linux HAL: Add wiretapping option for I2C. Enabled with `FSFW_HAL_I2C_WIRETAPPING` defined to 1
+- Dedicated Version class and constant `fsfw::FSFW_VERSION` containing version information
+ inside `fsfw/version.h`
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/559
+- Added generic PUS TC Scheduler Service 11. It depends on the new added Emebeded Template Library
+ (ETL) dependency.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/594
+- Added ETL dependency and improved library dependency management
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/592
+- Add a `DummyPowerSwitcher` module which can be useful for test setups when no PCDU is available
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/590
+- New typedef for switcher type
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/590
+- `Subsystem`: New API to add table and sequence entries
+
+## Fixed
+
+- TCP TMTC Server: `MutexGuard` was not created properly in
+ `TcpTmTcServer::handleTmSending(socket_t connSocket, bool& tmSent)` call.
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/618
+- Fix infinite recursion in `prepareHealthSetReply` of PUS Health Service 201.
+ Is not currently used right now but might be used in the future
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/617
+- Move some CMake directives further up top so they are not ignored
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/621
+- Small bugfix in STM32 HAL for SPI
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/599
+- HAL GPIO: Improved error checking in `LinuxLibgpioIF::configureGpios(...)`. If a GPIO
+ configuration fails, the function will exit prematurely with a dedicated error code
+ PR: https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/602
+
# [v4.0.0]
## Additions
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 79258db2d..cfae2acb9 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -1,23 +1,115 @@
cmake_minimum_required(VERSION 3.13)
-set(FSFW_VERSION 4)
-set(FSFW_SUBVERSION 0)
-set(FSFW_REVISION 0)
+set(MSG_PREFIX "fsfw |")
# Add the cmake folder so the FindSphinx module is found
-set(CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake" ${CMAKE_MODULE_PATH})
+list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake")
+list(APPEND CMAKE_MODULE_PATH
+ "${CMAKE_CURRENT_SOURCE_DIR}/cmake/cmake-modules/bilke")
+list(APPEND CMAKE_MODULE_PATH
+ "${CMAKE_CURRENT_SOURCE_DIR}/cmake/cmake-modules/rpavlik")
-option(FSFW_GENERATE_SECTIONS
- "Generate function and data sections. Required to remove unused code" ON
-)
-if(FSFW_GENERATE_SECTIONS)
- option(FSFW_REMOVE_UNUSED_CODE "Remove unused code" ON)
+# ##############################################################################
+# Version file handling #
+# ##############################################################################
+
+set(FSFW_VERSION_IF_GIT_FAILS 5)
+set(FSFW_SUBVERSION_IF_GIT_FAILS 0)
+set(FSFW_REVISION_IF_GIT_FAILS 0)
+
+set(FSFW_GIT_VER_HANDLING_OK FALSE)
+# Version handling
+if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/.git)
+ message(STATUS "${MSG_PREFIX} Determining version information with git")
+ include(FsfwHelpers)
+ determine_version_with_git("--exclude" "docker_*")
+ if(GIT_INFO)
+ set(FSFW_GIT_INFO
+ ${GIT_INFO}
+ CACHE STRING "Version information retrieved with git describe")
+ list(GET FSFW_GIT_INFO 1 FSFW_VERSION)
+ list(GET FSFW_GIT_INFO 2 FSFW_SUBVERSION)
+ list(GET FSFW_GIT_INFO 3 FSFW_REVISION)
+ list(GET FSFW_GIT_INFO 4 FSFW_VCS_INFO)
+ if(NOT FSFW_VERSION)
+ set(FSFW_VERSION ${FSFW_VERSION_IF_GIT_FAILS})
+ endif()
+ if(NOT FSFW_SUBVERSION)
+ set(FSFW_SUBVERSION ${FSFW_SUBVERSION_IF_GIT_FAILS})
+ endif()
+ if(NOT FSFW_REVISION)
+ set(FSFW_REVISION ${FSFW_REVISION_IF_GIT_FAILS})
+ endif()
+ set(FSFW_GIT_VER_HANDLING_OK TRUE)
+ else()
+ set(FSFW_GIT_VER_HANDLING_OK FALSE)
+ endif()
+endif()
+if(NOT FSFW_GIT_VER_HANDLING_OK)
+ set(FSFW_VERSION ${FSFW_VERSION_IF_GIT_FAILS})
+ set(FSFW_SUBVERSION ${FSFW_SUBVERSION_IF_GIT_FAILS})
+ set(FSFW_REVISION ${FSFW_REVISION_IF_GIT_FAILS})
endif()
-option(FSFW_BUILD_UNITTESTS "Build unittest binary in addition to static library" OFF)
+set(LIB_FSFW_NAME fsfw)
+project(${LIB_FSFW_NAME}
+ VERSION ${FSFW_VERSION}.${FSFW_SUBVERSION}.${FSFW_REVISION})
+
+if(NOT CMAKE_CXX_STANDARD)
+ set(CMAKE_CXX_STANDARD 17)
+ set(CMAKE_CXX_STANDARD_REQUIRED True)
+elseif(${CMAKE_CXX_STANDARD} LESS 17)
+ message(
+ FATAL_ERROR
+ "${MSG_PREFIX} Compiling the FSFW requires a minimum of C++17 support")
+endif()
+
+set(FSFW_SOURCES_DIR "${CMAKE_SOURCE_DIR}/src/fsfw")
+
+set(FSFW_ETL_LIB_NAME etl)
+set(FSFW_ETL_LINK_TARGET etl::etl)
+set(FSFW_ETL_LIB_MAJOR_VERSION
+ 20
+ CACHE STRING "ETL library major version requirement")
+set(FSFW_ETL_LIB_VERSION
+ ${FSFW_ETL_LIB_MAJOR_VERSION}.28.0
+ CACHE STRING "ETL library exact version requirement")
+set(FSFW_ETL_LINK_TARGET etl::etl)
+
+set(FSFW_CATCH2_LIB_MAJOR_VERSION
+ 3
+ CACHE STRING "Catch2 library major version requirement")
+set(FSFW_CATCH2_LIB_VERSION
+ v${FSFW_CATCH2_LIB_MAJOR_VERSION}.0.0-preview5
+ CACHE STRING "Catch2 library exact version requirement")
+
+# Keep this off by default for now. See PR:
+# https://egit.irs.uni-stuttgart.de/fsfw/fsfw/pulls/616 for information which
+# keeping this on by default is problematic
+option(
+ FSFW_ENABLE_IPO
+ "Enable interprocedural optimization or link-time optimization if available"
+ OFF)
+if(FSFW_ENABLE_IPO)
+ include(CheckIPOSupported)
+ check_ipo_supported(RESULT IPO_SUPPORTED OUTPUT IPO_ERROR)
+ if(NOT IPO_SUPPORTED)
+ message(STATUS "FSFW | IPO/LTO not supported: ${IPO_ERROR}")
+ endif()
+endif()
+
+option(FSFW_GENERATE_SECTIONS
+ "Generate function and data sections. Required to remove unused code" ON)
+if(FSFW_GENERATE_SECTIONS)
+ option(FSFW_REMOVE_UNUSED_CODE "Remove unused code" ON)
+endif()
+
+option(FSFW_BUILD_TESTS "Build unittest binary in addition to static library"
+ OFF)
+option(FSFW_CICD_BUILD "Build for CI/CD. This can disable problematic test" OFF)
option(FSFW_BUILD_DOCS "Build documentation with Sphinx and Doxygen" OFF)
-if(FSFW_BUILD_UNITTESTS)
- option(FSFW_TESTS_GEN_COV "Generate coverage data for unittests" ON)
+if(FSFW_BUILD_TESTS)
+ option(FSFW_TESTS_GEN_COV "Generate coverage data for unittests" ON)
endif()
option(FSFW_WARNING_SHADOW_LOCAL_GCC "Enable -Wshadow=local warning in GCC" ON)
@@ -38,54 +130,92 @@ option(FSFW_ADD_TMSTORAGE "Compile with tm storage components" OFF)
# Contrib sources
option(FSFW_ADD_SGP4_PROPAGATOR "Add SGP4 propagator code" OFF)
-set(LIB_FSFW_NAME fsfw)
set(FSFW_TEST_TGT fsfw-tests)
set(FSFW_DUMMY_TGT fsfw-dummy)
-project(${LIB_FSFW_NAME})
add_library(${LIB_FSFW_NAME})
-if(FSFW_BUILD_UNITTESTS)
- message(STATUS "Building the FSFW unittests in addition to the static library")
- # Check whether the user has already installed Catch2 first
- find_package(Catch2 3)
- # Not installed, so use FetchContent to download and provide Catch2
- if(NOT Catch2_FOUND)
- include(FetchContent)
+if(IPO_SUPPORTED AND FSFW_ENABLE_IPO)
+ set_property(TARGET ${LIB_FSFW_NAME} PROPERTY INTERPROCEDURAL_OPTIMIZATION
+ TRUE)
+endif()
- FetchContent_Declare(
- Catch2
- GIT_REPOSITORY https://github.com/catchorg/Catch2.git
- GIT_TAG v3.0.0-preview4
- )
+if(FSFW_BUILD_TESTS)
+ message(
+ STATUS
+ "${MSG_PREFIX} Building the FSFW unittests in addition to the static library"
+ )
+ # Check whether the user has already installed Catch2 first
+ find_package(Catch2 ${FSFW_CATCH2_LIB_MAJOR_VERSION})
+ # Not installed, so use FetchContent to download and provide Catch2
+ if(NOT Catch2_FOUND)
+ message(
+ STATUS
+ "${MSG_PREFIX} Catch2 installation not found. Downloading Catch2 library with FetchContent"
+ )
+ include(FetchContent)
- FetchContent_MakeAvailable(Catch2)
- #fixes regression -preview4, to be confirmed in later releases
- set_target_properties(Catch2 PROPERTIES DEBUG_POSTFIX "")
- endif()
+ FetchContent_Declare(
+ Catch2
+ GIT_REPOSITORY https://github.com/catchorg/Catch2.git
+ GIT_TAG ${FSFW_CATCH2_LIB_VERSION})
- set(FSFW_CONFIG_PATH tests/src/fsfw_tests/unit/testcfg)
- configure_file(tests/src/fsfw_tests/unit/testcfg/FSFWConfig.h.in FSFWConfig.h)
- configure_file(tests/src/fsfw_tests/unit/testcfg/TestsConfig.h.in tests/TestsConfig.h)
+ list(APPEND FSFW_FETCH_CONTENT_TARGETS Catch2)
+ endif()
- project(${FSFW_TEST_TGT} CXX C)
- add_executable(${FSFW_TEST_TGT})
+ set(FSFW_CONFIG_PATH unittests/testcfg)
+ configure_file(unittests/testcfg/FSFWConfig.h.in FSFWConfig.h)
+ configure_file(unittests/testcfg/TestsConfig.h.in tests/TestsConfig.h)
- if(FSFW_TESTS_GEN_COV)
- message(STATUS "Generating coverage data for the library")
- message(STATUS "Targets linking against ${LIB_FSFW_NAME} "
- "will be compiled with coverage data as well"
- )
- include(FetchContent)
- FetchContent_Declare(
- cmake-modules
- GIT_REPOSITORY https://github.com/bilke/cmake-modules.git
- )
- FetchContent_MakeAvailable(cmake-modules)
- set(CMAKE_BUILD_TYPE "Debug")
- list(APPEND CMAKE_MODULE_PATH ${cmake-modules_SOURCE_DIR})
- include(CodeCoverage)
- endif()
+ project(${FSFW_TEST_TGT} CXX C)
+ add_executable(${FSFW_TEST_TGT})
+ if(IPO_SUPPORTED AND FSFW_ENABLE_IPO)
+ set_property(TARGET ${FSFW_TEST_TGT} PROPERTY INTERPROCEDURAL_OPTIMIZATION
+ TRUE)
+ endif()
+
+ if(FSFW_TESTS_GEN_COV)
+ message(STATUS "${MSG_PREFIX} Generating coverage data for the library")
+ message(STATUS "${MSG_PREFIX} Targets linking against ${LIB_FSFW_NAME} "
+ "will be compiled with coverage data as well")
+ set(CMAKE_BUILD_TYPE "Debug")
+ include(CodeCoverage)
+ endif()
+endif()
+
+message(STATUS "${MSG_PREFIX} Finding and/or providing ETL library")
+
+# Check whether the user has already installed ETL first
+find_package(${FSFW_ETL_LIB_NAME} ${FSFW_ETL_LIB_MAJOR_VERSION} QUIET)
+# Not installed, so use FetchContent to download and provide etl
+if(NOT ${FSFW_ETL_LIB_NAME}_FOUND)
+ message(
+ STATUS
+ "No ETL installation was found with find_package. Installing and providing "
+ "etl with FindPackage")
+ include(FetchContent)
+
+ FetchContent_Declare(
+ ${FSFW_ETL_LIB_NAME}
+ GIT_REPOSITORY https://github.com/ETLCPP/etl
+ GIT_TAG ${FSFW_ETL_LIB_VERSION})
+
+ list(APPEND FSFW_FETCH_CONTENT_TARGETS ${FSFW_ETL_LIB_NAME})
+endif()
+
+# The documentation for FetchContent recommends declaring all the dependencies
+# before making them available. We make all declared dependency available here
+# after their declaration
+if(FSFW_FETCH_CONTENT_TARGETS)
+ FetchContent_MakeAvailable(${FSFW_FETCH_CONTENT_TARGETS})
+ if(TARGET ${FSFW_ETL_LIB_NAME})
+ add_library(${FSFW_ETL_LINK_TARGET} ALIAS ${FSFW_ETL_LIB_NAME})
+ endif()
+ if(TARGET Catch2)
+ # Fixes regression -preview4, to be confirmed in later releases Related
+ # GitHub issue: https://github.com/catchorg/Catch2/issues/2417
+ set_target_properties(Catch2 PROPERTIES DEBUG_POSTFIX "")
+ endif()
endif()
set(FSFW_CORE_INC_PATH "inc")
@@ -93,256 +223,241 @@ set(FSFW_CORE_INC_PATH "inc")
set_property(CACHE FSFW_OSAL PROPERTY STRINGS host linux rtems freertos)
# For configure files
-target_include_directories(${LIB_FSFW_NAME} PRIVATE
- ${CMAKE_CURRENT_BINARY_DIR}
-)
-target_include_directories(${LIB_FSFW_NAME} INTERFACE
- ${CMAKE_CURRENT_BINARY_DIR}
-)
-
-if(NOT CMAKE_CXX_STANDARD)
- set(CMAKE_CXX_STANDARD 11)
- set(CMAKE_CXX_STANDARD_REQUIRED True)
-elseif(${CMAKE_CXX_STANDARD} LESS 11)
- message(FATAL_ERROR "Compiling the FSFW requires a minimum of C++11 support")
-endif()
+target_include_directories(${LIB_FSFW_NAME} PRIVATE ${CMAKE_CURRENT_BINARY_DIR})
+target_include_directories(${LIB_FSFW_NAME}
+ INTERFACE ${CMAKE_CURRENT_BINARY_DIR})
# Backwards comptability
if(OS_FSFW AND NOT FSFW_OSAL)
- message(WARNING "Please pass the FSFW OSAL as FSFW_OSAL instead of OS_FSFW")
- set(FSFW_OSAL OS_FSFW)
+ message(
+ WARNING
+ "${MSG_PREFIX} Please pass the FSFW OSAL as FSFW_OSAL instead of OS_FSFW")
+ set(FSFW_OSAL OS_FSFW)
endif()
if(NOT FSFW_OSAL)
- message(STATUS "No OS for FSFW via FSFW_OSAL set. Assuming host OS")
- # Assume host OS and autodetermine from OS_FSFW
- if(UNIX)
- set(FSFW_OSAL "linux"
- CACHE STRING
- "OS abstraction layer used in the FSFW"
- )
- elseif(WIN32)
- set(FSFW_OSAL "host"
- CACHE STRING "OS abstraction layer used in the FSFW"
- )
- endif()
-
+ message(STATUS "No OS for FSFW via FSFW_OSAL set. Assuming host OS")
+ # Assume host OS and autodetermine from OS_FSFW
+ if(UNIX)
+ set(FSFW_OSAL
+ "linux"
+ CACHE STRING "OS abstraction layer used in the FSFW")
+ elseif(WIN32)
+ set(FSFW_OSAL
+ "host"
+ CACHE STRING "OS abstraction layer used in the FSFW")
+ endif()
endif()
set(FSFW_OSAL_DEFINITION FSFW_OSAL_HOST)
if(FSFW_OSAL MATCHES host)
- set(FSFW_OS_NAME "Host")
- set(FSFW_OSAL_HOST ON)
+ set(FSFW_OS_NAME "Host")
+ set(FSFW_OSAL_HOST ON)
elseif(FSFW_OSAL MATCHES linux)
- set(FSFW_OS_NAME "Linux")
- set(FSFW_OSAL_LINUX ON)
+ set(FSFW_OS_NAME "Linux")
+ set(FSFW_OSAL_LINUX ON)
elseif(FSFW_OSAL MATCHES freertos)
- set(FSFW_OS_NAME "FreeRTOS")
- set(FSFW_OSAL_FREERTOS ON)
- target_link_libraries(${LIB_FSFW_NAME} PRIVATE
- ${LIB_OS_NAME}
- )
+ set(FSFW_OS_NAME "FreeRTOS")
+ set(FSFW_OSAL_FREERTOS ON)
+ target_link_libraries(${LIB_FSFW_NAME} PRIVATE ${LIB_OS_NAME})
elseif(FSFW_OSAL STREQUAL rtems)
- set(FSFW_OS_NAME "RTEMS")
- set(FSFW_OSAL_RTEMS ON)
+ set(FSFW_OS_NAME "RTEMS")
+ set(FSFW_OSAL_RTEMS ON)
else()
- message(WARNING
- "Invalid operating system for FSFW specified! Setting to host.."
- )
- set(FSFW_OS_NAME "Host")
- set(OS_FSFW "host")
+ message(
+ WARNING
+ "${MSG_PREFIX} Invalid operating system for FSFW specified! Setting to host.."
+ )
+ set(FSFW_OS_NAME "Host")
+ set(OS_FSFW "host")
endif()
configure_file(src/fsfw/FSFW.h.in fsfw/FSFW.h)
configure_file(src/fsfw/FSFWVersion.h.in fsfw/FSFWVersion.h)
-message(STATUS "Compiling FSFW for the ${FSFW_OS_NAME} operating system.")
+message(
+ STATUS "${MSG_PREFIX} Compiling FSFW for the ${FSFW_OS_NAME} operating system"
+)
add_subdirectory(src)
-add_subdirectory(tests)
-if(FSFW_ADD_HAL)
- add_subdirectory(hal)
-endif()
add_subdirectory(contrib)
+if(FSFW_BUILD_TESTS)
+ add_subdirectory(unittests)
+endif()
if(FSFW_BUILD_DOCS)
- add_subdirectory(docs)
+ add_subdirectory(docs)
endif()
-if(FSFW_BUILD_UNITTESTS)
- if(FSFW_TESTS_GEN_COV)
- if(CMAKE_COMPILER_IS_GNUCXX)
- include(CodeCoverage)
+if(FSFW_BUILD_TESTS)
+ if(FSFW_TESTS_GEN_COV)
+ if(CMAKE_COMPILER_IS_GNUCXX)
+ include(CodeCoverage)
- # Remove quotes.
- separate_arguments(COVERAGE_COMPILER_FLAGS
- NATIVE_COMMAND "${COVERAGE_COMPILER_FLAGS}"
- )
+ # Remove quotes.
+ separate_arguments(COVERAGE_COMPILER_FLAGS NATIVE_COMMAND
+ "${COVERAGE_COMPILER_FLAGS}")
- # Add compile options manually, we don't want coverage for Catch2
- target_compile_options(${FSFW_TEST_TGT} PRIVATE
- "${COVERAGE_COMPILER_FLAGS}"
- )
- target_compile_options(${LIB_FSFW_NAME} PRIVATE
- "${COVERAGE_COMPILER_FLAGS}"
- )
+ # Add compile options manually, we don't want coverage for Catch2
+ target_compile_options(${FSFW_TEST_TGT}
+ PRIVATE "${COVERAGE_COMPILER_FLAGS}")
+ target_compile_options(${LIB_FSFW_NAME}
+ PRIVATE "${COVERAGE_COMPILER_FLAGS}")
- # Exclude directories here
- if(WIN32)
- set(GCOVR_ADDITIONAL_ARGS
- "--exclude-throw-branches"
- "--exclude-unreachable-branches"
- )
- set(COVERAGE_EXCLUDES
- "/c/msys64/mingw64/*" "*/fsfw_hal/*"
- )
- elseif(UNIX)
- set(COVERAGE_EXCLUDES
- "/usr/include/*" "/usr/bin/*" "Catch2/*"
- "/usr/local/include/*" "*/fsfw_tests/*"
- "*/catch2-src/*" "*/fsfw_hal/*"
- )
- endif()
+ # Exclude directories here
+ if(WIN32)
+ set(GCOVR_ADDITIONAL_ARGS "--exclude-throw-branches"
+ "--exclude-unreachable-branches")
+ set(COVERAGE_EXCLUDES "/c/msys64/mingw64/*" "*/fsfw_hal/*")
+ elseif(UNIX)
+ set(COVERAGE_EXCLUDES
+ "/usr/include/*"
+ "/usr/bin/*"
+ "Catch2/*"
+ "/usr/local/include/*"
+ "*/fsfw_tests/*"
+ "*/catch2-src/*"
+ "*/fsfw_hal/*")
+ endif()
- target_link_options(${FSFW_TEST_TGT} PRIVATE
- -fprofile-arcs
- -ftest-coverage
- )
- target_link_options(${LIB_FSFW_NAME} PRIVATE
- -fprofile-arcs
- -ftest-coverage
- )
- # Need to specify this as an interface, otherwise there will the compile issues
- target_link_options(${LIB_FSFW_NAME} INTERFACE
- -fprofile-arcs
- -ftest-coverage
- )
+ target_link_options(${FSFW_TEST_TGT} PRIVATE -fprofile-arcs
+ -ftest-coverage)
+ target_link_options(${LIB_FSFW_NAME} PRIVATE -fprofile-arcs
+ -ftest-coverage)
+ # Need to specify this as an interface, otherwise there will the compile
+ # issues
+ target_link_options(${LIB_FSFW_NAME} INTERFACE -fprofile-arcs
+ -ftest-coverage)
- if(WIN32)
- setup_target_for_coverage_gcovr_html(
- NAME ${FSFW_TEST_TGT}_coverage
- EXECUTABLE ${FSFW_TEST_TGT}
- DEPENDENCIES ${FSFW_TEST_TGT}
- )
- else()
- setup_target_for_coverage_lcov(
- NAME ${FSFW_TEST_TGT}_coverage
- EXECUTABLE ${FSFW_TEST_TGT}
- DEPENDENCIES ${FSFW_TEST_TGT}
- )
- endif()
- endif()
+ if(WIN32)
+ setup_target_for_coverage_gcovr_html(
+ NAME ${FSFW_TEST_TGT}_coverage EXECUTABLE ${FSFW_TEST_TGT}
+ DEPENDENCIES ${FSFW_TEST_TGT})
+ else()
+ setup_target_for_coverage_lcov(
+ NAME ${FSFW_TEST_TGT}_coverage EXECUTABLE ${FSFW_TEST_TGT}
+ DEPENDENCIES ${FSFW_TEST_TGT})
+ endif()
endif()
- target_link_libraries(${FSFW_TEST_TGT} PRIVATE Catch2::Catch2 ${LIB_FSFW_NAME})
+ endif()
+ target_link_libraries(${FSFW_TEST_TGT} PRIVATE Catch2::Catch2
+ ${LIB_FSFW_NAME})
endif()
-# The project CMakeLists file has to set the FSFW_CONFIG_PATH and add it.
-# If this is not given, we include the default configuration and emit a warning.
+# The project CMakeLists file has to set the FSFW_CONFIG_PATH and add it. If
+# this is not given, we include the default configuration and emit a warning.
if(NOT FSFW_CONFIG_PATH)
- set(DEF_CONF_PATH misc/defaultcfg/fsfwconfig)
- if(NOT FSFW_BUILD_DOCS)
- message(WARNING "Flight Software Framework configuration path not set!")
- message(WARNING "Setting default configuration from ${DEF_CONF_PATH} ..")
- endif()
- add_subdirectory(${DEF_CONF_PATH})
- set(FSFW_CONFIG_PATH ${DEF_CONF_PATH})
+ set(DEF_CONF_PATH misc/defaultcfg/fsfwconfig)
+ if(NOT FSFW_BUILD_DOCS)
+ message(
+ WARNING
+ "${MSG_PREFIX} Flight Software Framework configuration path not set")
+ message(
+ WARNING
+ "${MSG_PREFIX} Setting default configuration from ${DEF_CONF_PATH} ..")
+ endif()
+ add_subdirectory(${DEF_CONF_PATH})
+ set(FSFW_CONFIG_PATH ${DEF_CONF_PATH})
endif()
-# FSFW might be part of a possibly complicated folder structure, so we
-# extract the absolute path of the fsfwconfig folder.
+# FSFW might be part of a possibly complicated folder structure, so we extract
+# the absolute path of the fsfwconfig folder.
if(IS_ABSOLUTE ${FSFW_CONFIG_PATH})
- set(FSFW_CONFIG_PATH_ABSOLUTE ${FSFW_CONFIG_PATH})
+ set(FSFW_CONFIG_PATH_ABSOLUTE ${FSFW_CONFIG_PATH})
else()
- get_filename_component(FSFW_CONFIG_PATH_ABSOLUTE
- ${FSFW_CONFIG_PATH} REALPATH BASE_DIR ${CMAKE_SOURCE_DIR}
- )
+ get_filename_component(FSFW_CONFIG_PATH_ABSOLUTE ${FSFW_CONFIG_PATH} REALPATH
+ BASE_DIR ${CMAKE_SOURCE_DIR})
endif()
foreach(INCLUDE_PATH ${FSFW_ADDITIONAL_INC_PATHS})
- if(IS_ABSOLUTE ${INCLUDE_PATH})
- set(CURR_ABS_INC_PATH "${INCLUDE_PATH}")
- else()
- get_filename_component(CURR_ABS_INC_PATH
- ${INCLUDE_PATH} REALPATH BASE_DIR ${CMAKE_SOURCE_DIR})
- endif()
+ if(IS_ABSOLUTE ${INCLUDE_PATH})
+ set(CURR_ABS_INC_PATH "${INCLUDE_PATH}")
+ else()
+ get_filename_component(CURR_ABS_INC_PATH ${INCLUDE_PATH} REALPATH BASE_DIR
+ ${CMAKE_SOURCE_DIR})
+ endif()
- if(CMAKE_VERBOSE)
- message(STATUS "FSFW include path: ${CURR_ABS_INC_PATH}")
- endif()
+ if(CMAKE_VERBOSE)
+ message(STATUS "FSFW include path: ${CURR_ABS_INC_PATH}")
+ endif()
- list(APPEND FSFW_ADD_INC_PATHS_ABS ${CURR_ABS_INC_PATH})
+ list(APPEND FSFW_ADD_INC_PATHS_ABS ${CURR_ABS_INC_PATH})
endforeach()
if(CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
- if(NOT DEFINED FSFW_WARNING_FLAGS)
- set(FSFW_WARNING_FLAGS
- -Wall
- -Wextra
- -Wimplicit-fallthrough=1
- -Wno-unused-parameter
- -Wno-psabi
- )
- endif()
+ if(NOT DEFINED FSFW_WARNING_FLAGS)
+ set(FSFW_WARNING_FLAGS
+ -Wall
+ -Wextra
+ -Wimplicit-fallthrough=1
+ -Wno-unused-parameter
+ -Wno-psabi
+ -Wduplicated-cond # check for duplicate conditions
+ -Wduplicated-branches # check for duplicate branches
+ -Wlogical-op # Search for bitwise operations instead of logical
+ -Wnull-dereference # Search for NULL dereference
+ -Wundef # Warn if undefind marcos are used
+ -Wformat=2 # Format string problem detection
+ -Wformat-overflow=2 # Formatting issues in printf
+ -Wformat-truncation=2 # Formatting issues in printf
+ -Wformat-security # Search for dangerous printf operations
+ -Wstrict-overflow=3 # Warn if integer overflows might happen
+ -Warray-bounds=2 # Some array bounds violations will be found
+ -Wshift-overflow=2 # Search for bit left shift overflows (
+mkdir build && cd build
+cmake ..
+cmake --install .
+```
+
+It is recommended to install `20.27.2` or newer for the package version handling of
+ETL to work.
+
## Adding the library
The following steps show how to add and use FSFW components. It is still recommended to
@@ -71,9 +99,9 @@ add and link against the FSFW library in general.
4. Link against the FSFW library
- ```cmake
- target_link_libraries( PRIVATE fsfw)
- ```
+ ```sh
+ target_link_libraries(${YourProjectName} PRIVATE fsfw)
+ ```
5. It should now be possible use the FSFW as a static library from the user code.
@@ -83,6 +111,19 @@ The FSFW also has unittests which use the [Catch2 library](https://github.com/ca
These are built by setting the CMake option `FSFW_BUILD_UNITTESTS` to `ON` or `TRUE`
from your project `CMakeLists.txt` file or from the command line.
+You can install the Catch2 library, which prevents the build system to avoid re-downloading
+the dependency if the unit tests are completely rebuilt. The current recommended version
+can be found inside the fsfw `CMakeLists.txt` file or by using `ccmake` and looking up
+the `FSFW_CATCH2_LIB_VERSION` variable.
+
+```sh
+git clone https://github.com/catchorg/Catch2.git
+cd Catch2
+git checkout
+cmake -Bbuild -H. -DBUILD_TESTING=OFF
+sudo cmake --build build/ --target install
+```
+
The fsfw-tests binary will be built as part of the static library and dropped alongside it.
If the unittests are built, the library and the tests will be built with coverage information by
default. This can be disabled by setting the `FSFW_TESTS_COV_GEN` option to `OFF` or `FALSE`.
@@ -90,7 +131,7 @@ default. This can be disabled by setting the `FSFW_TESTS_COV_GEN` option to `OFF
You can use the following commands inside the `fsfw` folder to set up the build system
```sh
-mkdir build-Unittest && cd build-Unittest
+mkdir build-tests && cd build-tests
cmake -DFSFW_BUILD_UNITTESTS=ON -DFSFW_OSAL=host -DCMAKE_BUILD_TYPE=Debug ..
```
@@ -98,7 +139,7 @@ You can also use `-DFSFW_OSAL=linux` on Linux systems.
Coverage data in HTML format can be generated using the `CodeCoverage`
[CMake module](https://github.com/bilke/cmake-modules/tree/master).
-To build the unittests, run them and then generare the coverage data in this format,
+To build the unittests, run them and then generate the coverage data in this format,
the following command can be used inside the build directory after the build system was set up
```sh
@@ -147,7 +188,10 @@ and open the documentation conveniently. Try `helper.py -h for more information.
The formatting is done by the `clang-format` tool. The configuration is contained within the
`.clang-format` file in the repository root. As long as `clang-format` is installed, you
-can run the `apply-clang-format.sh` helper script to format all source files consistently.
+can run the `auto-format.sh` helper script to format all source files consistently. Furthermore cmake-format is required to format CMake files which can be installed with:
+````sh
+sudo pip install cmakelang
+````
## Index
diff --git a/automation/Dockerfile b/automation/Dockerfile
index 9df67fc82..2ed2a7d9a 100644
--- a/automation/Dockerfile
+++ b/automation/Dockerfile
@@ -6,3 +6,15 @@ RUN apt-get --yes upgrade
#tzdata is a dependency, won't install otherwise
ARG DEBIAN_FRONTEND=noninteractive
RUN apt-get --yes install gcc g++ cmake make lcov git valgrind nano iputils-ping
+
+RUN git clone https://github.com/catchorg/Catch2.git && \
+ cd Catch2 && \
+ git checkout v3.0.0-preview5 && \
+ cmake -Bbuild -H. -DBUILD_TESTING=OFF && \
+ cmake --build build/ --target install
+
+RUN git clone https://github.com/ETLCPP/etl.git && \
+ cd etl && \
+ git checkout 20.28.0 && \
+ cmake -B build . && \
+ cmake --install build/
diff --git a/automation/Jenkinsfile b/automation/Jenkinsfile
index 3424f986c..c5dcbe029 100644
--- a/automation/Jenkinsfile
+++ b/automation/Jenkinsfile
@@ -3,7 +3,7 @@ pipeline {
BUILDDIR = 'build-tests'
}
agent {
- docker { image 'fsfw-ci:d1'}
+ docker { image 'fsfw-ci:d3'}
}
stages {
stage('Clean') {
@@ -14,21 +14,21 @@ pipeline {
stage('Configure') {
steps {
dir(BUILDDIR) {
- sh 'cmake -DFSFW_OSAL=host -DFSFW_BUILD_UNITTESTS=ON ..'
+ sh 'cmake -DFSFW_OSAL=host -DFSFW_BUILD_TESTS=ON -DFSFW_CICD_BUILD=ON ..'
}
}
}
stage('Build') {
steps {
dir(BUILDDIR) {
- sh 'cmake --build . -j'
+ sh 'cmake --build . -j4'
}
}
}
stage('Unittests') {
steps {
dir(BUILDDIR) {
- sh 'cmake --build . -- fsfw-tests_coverage -j'
+ sh 'cmake --build . -- fsfw-tests_coverage -j4'
}
}
}
diff --git a/cmake/FsfwHelpers.cmake b/cmake/FsfwHelpers.cmake
new file mode 100644
index 000000000..3f22ebde1
--- /dev/null
+++ b/cmake/FsfwHelpers.cmake
@@ -0,0 +1,28 @@
+# Determines the git version with git describe and returns it by setting
+# the GIT_INFO list in the parent scope. The list has the following entries
+# 1. Full version string
+# 2. Major version
+# 3. Minor version
+# 4. Revision
+# 5. git SHA hash and commits since tag
+function(determine_version_with_git)
+ include(GetGitRevisionDescription)
+ git_describe(VERSION ${ARGN})
+ string(FIND ${VERSION} "." VALID_VERSION)
+ if(VALID_VERSION EQUAL -1)
+ message(WARNING "Version string ${VERSION} retrieved with git describe is invalid")
+ return()
+ endif()
+ # Parse the version information into pieces.
+ string(REGEX REPLACE "^v([0-9]+)\\..*" "\\1" _VERSION_MAJOR "${VERSION}")
+ string(REGEX REPLACE "^v[0-9]+\\.([0-9]+).*" "\\1" _VERSION_MINOR "${VERSION}")
+ string(REGEX REPLACE "^v[0-9]+\\.[0-9]+\\.([0-9]+).*" "\\1" _VERSION_PATCH "${VERSION}")
+ string(REGEX REPLACE "^v[0-9]+\\.[0-9]+\\.[0-9]+-(.*)" "\\1" VERSION_SHA1 "${VERSION}")
+ set(GIT_INFO ${VERSION})
+ list(APPEND GIT_INFO ${_VERSION_MAJOR})
+ list(APPEND GIT_INFO ${_VERSION_MINOR})
+ list(APPEND GIT_INFO ${_VERSION_PATCH})
+ list(APPEND GIT_INFO ${VERSION_SHA1})
+ set(GIT_INFO ${GIT_INFO} PARENT_SCOPE)
+ message(STATUS "${MSG_PREFIX} Set git version info into GIT_INFO from the git tag ${VERSION}")
+endfunction()
diff --git a/cmake/cmake-modules/README.md b/cmake/cmake-modules/README.md
new file mode 100644
index 000000000..b6355c3fc
--- /dev/null
+++ b/cmake/cmake-modules/README.md
@@ -0,0 +1,7 @@
+The files in the `bilke` folder were manually copy and pasted from the
+[cmake-modules repository](https://github.com/bilke/cmake-modules). It was decided to do
+this because only a small subset of its provided functions are needed.
+
+The files in the `rpavlik` folder were manually copy and pasted from the
+[cmake-modules repository](https://github.com/rpavlik/cmake-modules). It was decided to do
+this because only a small subset of its provided functions are needed.
diff --git a/cmake/cmake-modules/bilke/CodeCoverage.cmake b/cmake/cmake-modules/bilke/CodeCoverage.cmake
new file mode 100644
index 000000000..aef3d9436
--- /dev/null
+++ b/cmake/cmake-modules/bilke/CodeCoverage.cmake
@@ -0,0 +1,719 @@
+# Copyright (c) 2012 - 2017, Lars Bilke
+# All rights reserved.
+#
+# Redistribution and use in source and binary forms, with or without modification,
+# are permitted provided that the following conditions are met:
+#
+# 1. Redistributions of source code must retain the above copyright notice, this
+# list of conditions and the following disclaimer.
+#
+# 2. Redistributions in binary form must reproduce the above copyright notice,
+# this list of conditions and the following disclaimer in the documentation
+# and/or other materials provided with the distribution.
+#
+# 3. Neither the name of the copyright holder nor the names of its contributors
+# may be used to endorse or promote products derived from this software without
+# specific prior written permission.
+#
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
+# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
+# ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+#
+# CHANGES:
+#
+# 2012-01-31, Lars Bilke
+# - Enable Code Coverage
+#
+# 2013-09-17, Joakim Söderberg
+# - Added support for Clang.
+# - Some additional usage instructions.
+#
+# 2016-02-03, Lars Bilke
+# - Refactored functions to use named parameters
+#
+# 2017-06-02, Lars Bilke
+# - Merged with modified version from github.com/ufz/ogs
+#
+# 2019-05-06, Anatolii Kurotych
+# - Remove unnecessary --coverage flag
+#
+# 2019-12-13, FeRD (Frank Dana)
+# - Deprecate COVERAGE_LCOVR_EXCLUDES and COVERAGE_GCOVR_EXCLUDES lists in favor
+# of tool-agnostic COVERAGE_EXCLUDES variable, or EXCLUDE setup arguments.
+# - CMake 3.4+: All excludes can be specified relative to BASE_DIRECTORY
+# - All setup functions: accept BASE_DIRECTORY, EXCLUDE list
+# - Set lcov basedir with -b argument
+# - Add automatic --demangle-cpp in lcovr, if 'c++filt' is available (can be
+# overridden with NO_DEMANGLE option in setup_target_for_coverage_lcovr().)
+# - Delete output dir, .info file on 'make clean'
+# - Remove Python detection, since version mismatches will break gcovr
+# - Minor cleanup (lowercase function names, update examples...)
+#
+# 2019-12-19, FeRD (Frank Dana)
+# - Rename Lcov outputs, make filtered file canonical, fix cleanup for targets
+#
+# 2020-01-19, Bob Apthorpe
+# - Added gfortran support
+#
+# 2020-02-17, FeRD (Frank Dana)
+# - Make all add_custom_target()s VERBATIM to auto-escape wildcard characters
+# in EXCLUDEs, and remove manual escaping from gcovr targets
+#
+# 2021-01-19, Robin Mueller
+# - Add CODE_COVERAGE_VERBOSE option which will allow to print out commands which are run
+# - Added the option for users to set the GCOVR_ADDITIONAL_ARGS variable to supply additional
+# flags to the gcovr command
+#
+# 2020-05-04, Mihchael Davis
+# - Add -fprofile-abs-path to make gcno files contain absolute paths
+# - Fix BASE_DIRECTORY not working when defined
+# - Change BYPRODUCT from folder to index.html to stop ninja from complaining about double defines
+#
+# 2021-05-10, Martin Stump
+# - Check if the generator is multi-config before warning about non-Debug builds
+#
+# 2022-02-22, Marko Wehle
+# - Change gcovr output from -o for --xml and --html output respectively.
+# This will allow for Multiple Output Formats at the same time by making use of GCOVR_ADDITIONAL_ARGS, e.g. GCOVR_ADDITIONAL_ARGS "--txt".
+#
+# USAGE:
+#
+# 1. Copy this file into your cmake modules path.
+#
+# 2. Add the following line to your CMakeLists.txt (best inside an if-condition
+# using a CMake option() to enable it just optionally):
+# include(CodeCoverage)
+#
+# 3. Append necessary compiler flags for all supported source files:
+# append_coverage_compiler_flags()
+# Or for specific target:
+# append_coverage_compiler_flags_to_target(YOUR_TARGET_NAME)
+#
+# 3.a (OPTIONAL) Set appropriate optimization flags, e.g. -O0, -O1 or -Og
+#
+# 4. If you need to exclude additional directories from the report, specify them
+# using full paths in the COVERAGE_EXCLUDES variable before calling
+# setup_target_for_coverage_*().
+# Example:
+# set(COVERAGE_EXCLUDES
+# '${PROJECT_SOURCE_DIR}/src/dir1/*'
+# '/path/to/my/src/dir2/*')
+# Or, use the EXCLUDE argument to setup_target_for_coverage_*().
+# Example:
+# setup_target_for_coverage_lcov(
+# NAME coverage
+# EXECUTABLE testrunner
+# EXCLUDE "${PROJECT_SOURCE_DIR}/src/dir1/*" "/path/to/my/src/dir2/*")
+#
+# 4.a NOTE: With CMake 3.4+, COVERAGE_EXCLUDES or EXCLUDE can also be set
+# relative to the BASE_DIRECTORY (default: PROJECT_SOURCE_DIR)
+# Example:
+# set(COVERAGE_EXCLUDES "dir1/*")
+# setup_target_for_coverage_gcovr_html(
+# NAME coverage
+# EXECUTABLE testrunner
+# BASE_DIRECTORY "${PROJECT_SOURCE_DIR}/src"
+# EXCLUDE "dir2/*")
+#
+# 5. Use the functions described below to create a custom make target which
+# runs your test executable and produces a code coverage report.
+#
+# 6. Build a Debug build:
+# cmake -DCMAKE_BUILD_TYPE=Debug ..
+# make
+# make my_coverage_target
+#
+
+include(CMakeParseArguments)
+
+option(CODE_COVERAGE_VERBOSE "Verbose information" FALSE)
+
+# Check prereqs
+find_program( GCOV_PATH gcov )
+find_program( LCOV_PATH NAMES lcov lcov.bat lcov.exe lcov.perl)
+find_program( FASTCOV_PATH NAMES fastcov fastcov.py )
+find_program( GENHTML_PATH NAMES genhtml genhtml.perl genhtml.bat )
+find_program( GCOVR_PATH gcovr PATHS ${CMAKE_SOURCE_DIR}/scripts/test)
+find_program( CPPFILT_PATH NAMES c++filt )
+
+if(NOT GCOV_PATH)
+ message(FATAL_ERROR "gcov not found! Aborting...")
+endif() # NOT GCOV_PATH
+
+get_property(LANGUAGES GLOBAL PROPERTY ENABLED_LANGUAGES)
+list(GET LANGUAGES 0 LANG)
+
+if("${CMAKE_${LANG}_COMPILER_ID}" MATCHES "(Apple)?[Cc]lang")
+ if("${CMAKE_${LANG}_COMPILER_VERSION}" VERSION_LESS 3)
+ message(FATAL_ERROR "Clang version must be 3.0.0 or greater! Aborting...")
+ endif()
+elseif(NOT CMAKE_COMPILER_IS_GNUCXX)
+ if("${CMAKE_Fortran_COMPILER_ID}" MATCHES "[Ff]lang")
+ # Do nothing; exit conditional without error if true
+ elseif("${CMAKE_Fortran_COMPILER_ID}" MATCHES "GNU")
+ # Do nothing; exit conditional without error if true
+ else()
+ message(FATAL_ERROR "Compiler is not GNU gcc! Aborting...")
+ endif()
+endif()
+
+set(COVERAGE_COMPILER_FLAGS "-g -fprofile-arcs -ftest-coverage"
+ CACHE INTERNAL "")
+if(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Clang)")
+ include(CheckCXXCompilerFlag)
+ check_cxx_compiler_flag(-fprofile-abs-path HAVE_fprofile_abs_path)
+ if(HAVE_fprofile_abs_path)
+ set(COVERAGE_COMPILER_FLAGS "${COVERAGE_COMPILER_FLAGS} -fprofile-abs-path")
+ endif()
+endif()
+
+set(CMAKE_Fortran_FLAGS_COVERAGE
+ ${COVERAGE_COMPILER_FLAGS}
+ CACHE STRING "Flags used by the Fortran compiler during coverage builds."
+ FORCE )
+set(CMAKE_CXX_FLAGS_COVERAGE
+ ${COVERAGE_COMPILER_FLAGS}
+ CACHE STRING "Flags used by the C++ compiler during coverage builds."
+ FORCE )
+set(CMAKE_C_FLAGS_COVERAGE
+ ${COVERAGE_COMPILER_FLAGS}
+ CACHE STRING "Flags used by the C compiler during coverage builds."
+ FORCE )
+set(CMAKE_EXE_LINKER_FLAGS_COVERAGE
+ ""
+ CACHE STRING "Flags used for linking binaries during coverage builds."
+ FORCE )
+set(CMAKE_SHARED_LINKER_FLAGS_COVERAGE
+ ""
+ CACHE STRING "Flags used by the shared libraries linker during coverage builds."
+ FORCE )
+mark_as_advanced(
+ CMAKE_Fortran_FLAGS_COVERAGE
+ CMAKE_CXX_FLAGS_COVERAGE
+ CMAKE_C_FLAGS_COVERAGE
+ CMAKE_EXE_LINKER_FLAGS_COVERAGE
+ CMAKE_SHARED_LINKER_FLAGS_COVERAGE )
+
+get_property(GENERATOR_IS_MULTI_CONFIG GLOBAL PROPERTY GENERATOR_IS_MULTI_CONFIG)
+if(NOT (CMAKE_BUILD_TYPE STREQUAL "Debug" OR GENERATOR_IS_MULTI_CONFIG))
+ message(WARNING "Code coverage results with an optimised (non-Debug) build may be misleading")
+endif() # NOT (CMAKE_BUILD_TYPE STREQUAL "Debug" OR GENERATOR_IS_MULTI_CONFIG)
+
+if(CMAKE_C_COMPILER_ID STREQUAL "GNU" OR CMAKE_Fortran_COMPILER_ID STREQUAL "GNU")
+ link_libraries(gcov)
+endif()
+
+# Defines a target for running and collection code coverage information
+# Builds dependencies, runs the given executable and outputs reports.
+# NOTE! The executable should always have a ZERO as exit code otherwise
+# the coverage generation will not complete.
+#
+# setup_target_for_coverage_lcov(
+# NAME testrunner_coverage # New target name
+# EXECUTABLE testrunner -j ${PROCESSOR_COUNT} # Executable in PROJECT_BINARY_DIR
+# DEPENDENCIES testrunner # Dependencies to build first
+# BASE_DIRECTORY "../" # Base directory for report
+# # (defaults to PROJECT_SOURCE_DIR)
+# EXCLUDE "src/dir1/*" "src/dir2/*" # Patterns to exclude (can be relative
+# # to BASE_DIRECTORY, with CMake 3.4+)
+# NO_DEMANGLE # Don't demangle C++ symbols
+# # even if c++filt is found
+# )
+function(setup_target_for_coverage_lcov)
+
+ set(options NO_DEMANGLE)
+ set(oneValueArgs BASE_DIRECTORY NAME)
+ set(multiValueArgs EXCLUDE EXECUTABLE EXECUTABLE_ARGS DEPENDENCIES LCOV_ARGS GENHTML_ARGS)
+ cmake_parse_arguments(Coverage "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
+
+ if(NOT LCOV_PATH)
+ message(FATAL_ERROR "lcov not found! Aborting...")
+ endif() # NOT LCOV_PATH
+
+ if(NOT GENHTML_PATH)
+ message(FATAL_ERROR "genhtml not found! Aborting...")
+ endif() # NOT GENHTML_PATH
+
+ # Set base directory (as absolute path), or default to PROJECT_SOURCE_DIR
+ if(DEFINED Coverage_BASE_DIRECTORY)
+ get_filename_component(BASEDIR ${Coverage_BASE_DIRECTORY} ABSOLUTE)
+ else()
+ set(BASEDIR ${PROJECT_SOURCE_DIR})
+ endif()
+
+ # Collect excludes (CMake 3.4+: Also compute absolute paths)
+ set(LCOV_EXCLUDES "")
+ foreach(EXCLUDE ${Coverage_EXCLUDE} ${COVERAGE_EXCLUDES} ${COVERAGE_LCOV_EXCLUDES})
+ if(CMAKE_VERSION VERSION_GREATER 3.4)
+ get_filename_component(EXCLUDE ${EXCLUDE} ABSOLUTE BASE_DIR ${BASEDIR})
+ endif()
+ list(APPEND LCOV_EXCLUDES "${EXCLUDE}")
+ endforeach()
+ list(REMOVE_DUPLICATES LCOV_EXCLUDES)
+
+ # Conditional arguments
+ if(CPPFILT_PATH AND NOT ${Coverage_NO_DEMANGLE})
+ set(GENHTML_EXTRA_ARGS "--demangle-cpp")
+ endif()
+
+ # Setting up commands which will be run to generate coverage data.
+ # Cleanup lcov
+ set(LCOV_CLEAN_CMD
+ ${LCOV_PATH} ${Coverage_LCOV_ARGS} --gcov-tool ${GCOV_PATH} -directory .
+ -b ${BASEDIR} --zerocounters
+ )
+ # Create baseline to make sure untouched files show up in the report
+ set(LCOV_BASELINE_CMD
+ ${LCOV_PATH} ${Coverage_LCOV_ARGS} --gcov-tool ${GCOV_PATH} -c -i -d . -b
+ ${BASEDIR} -o ${Coverage_NAME}.base
+ )
+ # Run tests
+ set(LCOV_EXEC_TESTS_CMD
+ ${Coverage_EXECUTABLE} ${Coverage_EXECUTABLE_ARGS}
+ )
+ # Capturing lcov counters and generating report
+ set(LCOV_CAPTURE_CMD
+ ${LCOV_PATH} ${Coverage_LCOV_ARGS} --gcov-tool ${GCOV_PATH} --directory . -b
+ ${BASEDIR} --capture --output-file ${Coverage_NAME}.capture
+ )
+ # add baseline counters
+ set(LCOV_BASELINE_COUNT_CMD
+ ${LCOV_PATH} ${Coverage_LCOV_ARGS} --gcov-tool ${GCOV_PATH} -a ${Coverage_NAME}.base
+ -a ${Coverage_NAME}.capture --output-file ${Coverage_NAME}.total
+ )
+ # filter collected data to final coverage report
+ set(LCOV_FILTER_CMD
+ ${LCOV_PATH} ${Coverage_LCOV_ARGS} --gcov-tool ${GCOV_PATH} --remove
+ ${Coverage_NAME}.total ${LCOV_EXCLUDES} --output-file ${Coverage_NAME}.info
+ )
+ # Generate HTML output
+ set(LCOV_GEN_HTML_CMD
+ ${GENHTML_PATH} ${GENHTML_EXTRA_ARGS} ${Coverage_GENHTML_ARGS} -o
+ ${Coverage_NAME} ${Coverage_NAME}.info
+ )
+
+
+ if(CODE_COVERAGE_VERBOSE)
+ message(STATUS "Executed command report")
+ message(STATUS "Command to clean up lcov: ")
+ string(REPLACE ";" " " LCOV_CLEAN_CMD_SPACED "${LCOV_CLEAN_CMD}")
+ message(STATUS "${LCOV_CLEAN_CMD_SPACED}")
+
+ message(STATUS "Command to create baseline: ")
+ string(REPLACE ";" " " LCOV_BASELINE_CMD_SPACED "${LCOV_BASELINE_CMD}")
+ message(STATUS "${LCOV_BASELINE_CMD_SPACED}")
+
+ message(STATUS "Command to run the tests: ")
+ string(REPLACE ";" " " LCOV_EXEC_TESTS_CMD_SPACED "${LCOV_EXEC_TESTS_CMD}")
+ message(STATUS "${LCOV_EXEC_TESTS_CMD_SPACED}")
+
+ message(STATUS "Command to capture counters and generate report: ")
+ string(REPLACE ";" " " LCOV_CAPTURE_CMD_SPACED "${LCOV_CAPTURE_CMD}")
+ message(STATUS "${LCOV_CAPTURE_CMD_SPACED}")
+
+ message(STATUS "Command to add baseline counters: ")
+ string(REPLACE ";" " " LCOV_BASELINE_COUNT_CMD_SPACED "${LCOV_BASELINE_COUNT_CMD}")
+ message(STATUS "${LCOV_BASELINE_COUNT_CMD_SPACED}")
+
+ message(STATUS "Command to filter collected data: ")
+ string(REPLACE ";" " " LCOV_FILTER_CMD_SPACED "${LCOV_FILTER_CMD}")
+ message(STATUS "${LCOV_FILTER_CMD_SPACED}")
+
+ message(STATUS "Command to generate lcov HTML output: ")
+ string(REPLACE ";" " " LCOV_GEN_HTML_CMD_SPACED "${LCOV_GEN_HTML_CMD}")
+ message(STATUS "${LCOV_GEN_HTML_CMD_SPACED}")
+ endif()
+
+ # Setup target
+ add_custom_target(${Coverage_NAME}
+ COMMAND ${LCOV_CLEAN_CMD}
+ COMMAND ${LCOV_BASELINE_CMD}
+ COMMAND ${LCOV_EXEC_TESTS_CMD}
+ COMMAND ${LCOV_CAPTURE_CMD}
+ COMMAND ${LCOV_BASELINE_COUNT_CMD}
+ COMMAND ${LCOV_FILTER_CMD}
+ COMMAND ${LCOV_GEN_HTML_CMD}
+
+ # Set output files as GENERATED (will be removed on 'make clean')
+ BYPRODUCTS
+ ${Coverage_NAME}.base
+ ${Coverage_NAME}.capture
+ ${Coverage_NAME}.total
+ ${Coverage_NAME}.info
+ ${Coverage_NAME}/index.html
+ WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
+ DEPENDS ${Coverage_DEPENDENCIES}
+ VERBATIM # Protect arguments to commands
+ COMMENT "Resetting code coverage counters to zero.\nProcessing code coverage counters and generating report."
+ )
+
+ # Show where to find the lcov info report
+ add_custom_command(TARGET ${Coverage_NAME} POST_BUILD
+ COMMAND ;
+ COMMENT "Lcov code coverage info report saved in ${Coverage_NAME}.info."
+ )
+
+ # Show info where to find the report
+ add_custom_command(TARGET ${Coverage_NAME} POST_BUILD
+ COMMAND ;
+ COMMENT "Open ./${Coverage_NAME}/index.html in your browser to view the coverage report."
+ )
+
+endfunction() # setup_target_for_coverage_lcov
+
+# Defines a target for running and collection code coverage information
+# Builds dependencies, runs the given executable and outputs reports.
+# NOTE! The executable should always have a ZERO as exit code otherwise
+# the coverage generation will not complete.
+#
+# setup_target_for_coverage_gcovr_xml(
+# NAME ctest_coverage # New target name
+# EXECUTABLE ctest -j ${PROCESSOR_COUNT} # Executable in PROJECT_BINARY_DIR
+# DEPENDENCIES executable_target # Dependencies to build first
+# BASE_DIRECTORY "../" # Base directory for report
+# # (defaults to PROJECT_SOURCE_DIR)
+# EXCLUDE "src/dir1/*" "src/dir2/*" # Patterns to exclude (can be relative
+# # to BASE_DIRECTORY, with CMake 3.4+)
+# )
+# The user can set the variable GCOVR_ADDITIONAL_ARGS to supply additional flags to the
+# GCVOR command.
+function(setup_target_for_coverage_gcovr_xml)
+
+ set(options NONE)
+ set(oneValueArgs BASE_DIRECTORY NAME)
+ set(multiValueArgs EXCLUDE EXECUTABLE EXECUTABLE_ARGS DEPENDENCIES)
+ cmake_parse_arguments(Coverage "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
+
+ if(NOT GCOVR_PATH)
+ message(FATAL_ERROR "gcovr not found! Aborting...")
+ endif() # NOT GCOVR_PATH
+
+ # Set base directory (as absolute path), or default to PROJECT_SOURCE_DIR
+ if(DEFINED Coverage_BASE_DIRECTORY)
+ get_filename_component(BASEDIR ${Coverage_BASE_DIRECTORY} ABSOLUTE)
+ else()
+ set(BASEDIR ${PROJECT_SOURCE_DIR})
+ endif()
+
+ # Collect excludes (CMake 3.4+: Also compute absolute paths)
+ set(GCOVR_EXCLUDES "")
+ foreach(EXCLUDE ${Coverage_EXCLUDE} ${COVERAGE_EXCLUDES} ${COVERAGE_GCOVR_EXCLUDES})
+ if(CMAKE_VERSION VERSION_GREATER 3.4)
+ get_filename_component(EXCLUDE ${EXCLUDE} ABSOLUTE BASE_DIR ${BASEDIR})
+ endif()
+ list(APPEND GCOVR_EXCLUDES "${EXCLUDE}")
+ endforeach()
+ list(REMOVE_DUPLICATES GCOVR_EXCLUDES)
+
+ # Combine excludes to several -e arguments
+ set(GCOVR_EXCLUDE_ARGS "")
+ foreach(EXCLUDE ${GCOVR_EXCLUDES})
+ list(APPEND GCOVR_EXCLUDE_ARGS "-e")
+ list(APPEND GCOVR_EXCLUDE_ARGS "${EXCLUDE}")
+ endforeach()
+
+ # Set up commands which will be run to generate coverage data
+ # Run tests
+ set(GCOVR_XML_EXEC_TESTS_CMD
+ ${Coverage_EXECUTABLE} ${Coverage_EXECUTABLE_ARGS}
+ )
+ # Running gcovr
+ set(GCOVR_XML_CMD
+ ${GCOVR_PATH} --xml ${Coverage_NAME}.xml -r ${BASEDIR} ${GCOVR_ADDITIONAL_ARGS}
+ ${GCOVR_EXCLUDE_ARGS} --object-directory=${PROJECT_BINARY_DIR}
+ )
+
+ if(CODE_COVERAGE_VERBOSE)
+ message(STATUS "Executed command report")
+
+ message(STATUS "Command to run tests: ")
+ string(REPLACE ";" " " GCOVR_XML_EXEC_TESTS_CMD_SPACED "${GCOVR_XML_EXEC_TESTS_CMD}")
+ message(STATUS "${GCOVR_XML_EXEC_TESTS_CMD_SPACED}")
+
+ message(STATUS "Command to generate gcovr XML coverage data: ")
+ string(REPLACE ";" " " GCOVR_XML_CMD_SPACED "${GCOVR_XML_CMD}")
+ message(STATUS "${GCOVR_XML_CMD_SPACED}")
+ endif()
+
+ add_custom_target(${Coverage_NAME}
+ COMMAND ${GCOVR_XML_EXEC_TESTS_CMD}
+ COMMAND ${GCOVR_XML_CMD}
+
+ BYPRODUCTS ${Coverage_NAME}.xml
+ WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
+ DEPENDS ${Coverage_DEPENDENCIES}
+ VERBATIM # Protect arguments to commands
+ COMMENT "Running gcovr to produce Cobertura code coverage report."
+ )
+
+ # Show info where to find the report
+ add_custom_command(TARGET ${Coverage_NAME} POST_BUILD
+ COMMAND ;
+ COMMENT "Cobertura code coverage report saved in ${Coverage_NAME}.xml."
+ )
+endfunction() # setup_target_for_coverage_gcovr_xml
+
+# Defines a target for running and collection code coverage information
+# Builds dependencies, runs the given executable and outputs reports.
+# NOTE! The executable should always have a ZERO as exit code otherwise
+# the coverage generation will not complete.
+#
+# setup_target_for_coverage_gcovr_html(
+# NAME ctest_coverage # New target name
+# EXECUTABLE ctest -j ${PROCESSOR_COUNT} # Executable in PROJECT_BINARY_DIR
+# DEPENDENCIES executable_target # Dependencies to build first
+# BASE_DIRECTORY "../" # Base directory for report
+# # (defaults to PROJECT_SOURCE_DIR)
+# EXCLUDE "src/dir1/*" "src/dir2/*" # Patterns to exclude (can be relative
+# # to BASE_DIRECTORY, with CMake 3.4+)
+# )
+# The user can set the variable GCOVR_ADDITIONAL_ARGS to supply additional flags to the
+# GCVOR command.
+function(setup_target_for_coverage_gcovr_html)
+
+ set(options NONE)
+ set(oneValueArgs BASE_DIRECTORY NAME)
+ set(multiValueArgs EXCLUDE EXECUTABLE EXECUTABLE_ARGS DEPENDENCIES)
+ cmake_parse_arguments(Coverage "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
+
+ if(NOT GCOVR_PATH)
+ message(FATAL_ERROR "gcovr not found! Aborting...")
+ endif() # NOT GCOVR_PATH
+
+ # Set base directory (as absolute path), or default to PROJECT_SOURCE_DIR
+ if(DEFINED Coverage_BASE_DIRECTORY)
+ get_filename_component(BASEDIR ${Coverage_BASE_DIRECTORY} ABSOLUTE)
+ else()
+ set(BASEDIR ${PROJECT_SOURCE_DIR})
+ endif()
+
+ # Collect excludes (CMake 3.4+: Also compute absolute paths)
+ set(GCOVR_EXCLUDES "")
+ foreach(EXCLUDE ${Coverage_EXCLUDE} ${COVERAGE_EXCLUDES} ${COVERAGE_GCOVR_EXCLUDES})
+ if(CMAKE_VERSION VERSION_GREATER 3.4)
+ get_filename_component(EXCLUDE ${EXCLUDE} ABSOLUTE BASE_DIR ${BASEDIR})
+ endif()
+ list(APPEND GCOVR_EXCLUDES "${EXCLUDE}")
+ endforeach()
+ list(REMOVE_DUPLICATES GCOVR_EXCLUDES)
+
+ # Combine excludes to several -e arguments
+ set(GCOVR_EXCLUDE_ARGS "")
+ foreach(EXCLUDE ${GCOVR_EXCLUDES})
+ list(APPEND GCOVR_EXCLUDE_ARGS "-e")
+ list(APPEND GCOVR_EXCLUDE_ARGS "${EXCLUDE}")
+ endforeach()
+
+ # Set up commands which will be run to generate coverage data
+ # Run tests
+ set(GCOVR_HTML_EXEC_TESTS_CMD
+ ${Coverage_EXECUTABLE} ${Coverage_EXECUTABLE_ARGS}
+ )
+ # Create folder
+ set(GCOVR_HTML_FOLDER_CMD
+ ${CMAKE_COMMAND} -E make_directory ${PROJECT_BINARY_DIR}/${Coverage_NAME}
+ )
+ # Running gcovr
+ set(GCOVR_HTML_CMD
+ ${GCOVR_PATH} --html ${Coverage_NAME}/index.html --html-details -r ${BASEDIR} ${GCOVR_ADDITIONAL_ARGS}
+ ${GCOVR_EXCLUDE_ARGS} --object-directory=${PROJECT_BINARY_DIR}
+ )
+
+ if(CODE_COVERAGE_VERBOSE)
+ message(STATUS "Executed command report")
+
+ message(STATUS "Command to run tests: ")
+ string(REPLACE ";" " " GCOVR_HTML_EXEC_TESTS_CMD_SPACED "${GCOVR_HTML_EXEC_TESTS_CMD}")
+ message(STATUS "${GCOVR_HTML_EXEC_TESTS_CMD_SPACED}")
+
+ message(STATUS "Command to create a folder: ")
+ string(REPLACE ";" " " GCOVR_HTML_FOLDER_CMD_SPACED "${GCOVR_HTML_FOLDER_CMD}")
+ message(STATUS "${GCOVR_HTML_FOLDER_CMD_SPACED}")
+
+ message(STATUS "Command to generate gcovr HTML coverage data: ")
+ string(REPLACE ";" " " GCOVR_HTML_CMD_SPACED "${GCOVR_HTML_CMD}")
+ message(STATUS "${GCOVR_HTML_CMD_SPACED}")
+ endif()
+
+ add_custom_target(${Coverage_NAME}
+ COMMAND ${GCOVR_HTML_EXEC_TESTS_CMD}
+ COMMAND ${GCOVR_HTML_FOLDER_CMD}
+ COMMAND ${GCOVR_HTML_CMD}
+
+ BYPRODUCTS ${PROJECT_BINARY_DIR}/${Coverage_NAME}/index.html # report directory
+ WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
+ DEPENDS ${Coverage_DEPENDENCIES}
+ VERBATIM # Protect arguments to commands
+ COMMENT "Running gcovr to produce HTML code coverage report."
+ )
+
+ # Show info where to find the report
+ add_custom_command(TARGET ${Coverage_NAME} POST_BUILD
+ COMMAND ;
+ COMMENT "Open ./${Coverage_NAME}/index.html in your browser to view the coverage report."
+ )
+
+endfunction() # setup_target_for_coverage_gcovr_html
+
+# Defines a target for running and collection code coverage information
+# Builds dependencies, runs the given executable and outputs reports.
+# NOTE! The executable should always have a ZERO as exit code otherwise
+# the coverage generation will not complete.
+#
+# setup_target_for_coverage_fastcov(
+# NAME testrunner_coverage # New target name
+# EXECUTABLE testrunner -j ${PROCESSOR_COUNT} # Executable in PROJECT_BINARY_DIR
+# DEPENDENCIES testrunner # Dependencies to build first
+# BASE_DIRECTORY "../" # Base directory for report
+# # (defaults to PROJECT_SOURCE_DIR)
+# EXCLUDE "src/dir1/" "src/dir2/" # Patterns to exclude.
+# NO_DEMANGLE # Don't demangle C++ symbols
+# # even if c++filt is found
+# SKIP_HTML # Don't create html report
+# POST_CMD perl -i -pe s!${PROJECT_SOURCE_DIR}/!!g ctest_coverage.json # E.g. for stripping source dir from file paths
+# )
+function(setup_target_for_coverage_fastcov)
+
+ set(options NO_DEMANGLE SKIP_HTML)
+ set(oneValueArgs BASE_DIRECTORY NAME)
+ set(multiValueArgs EXCLUDE EXECUTABLE EXECUTABLE_ARGS DEPENDENCIES FASTCOV_ARGS GENHTML_ARGS POST_CMD)
+ cmake_parse_arguments(Coverage "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
+
+ if(NOT FASTCOV_PATH)
+ message(FATAL_ERROR "fastcov not found! Aborting...")
+ endif()
+
+ if(NOT Coverage_SKIP_HTML AND NOT GENHTML_PATH)
+ message(FATAL_ERROR "genhtml not found! Aborting...")
+ endif()
+
+ # Set base directory (as absolute path), or default to PROJECT_SOURCE_DIR
+ if(Coverage_BASE_DIRECTORY)
+ get_filename_component(BASEDIR ${Coverage_BASE_DIRECTORY} ABSOLUTE)
+ else()
+ set(BASEDIR ${PROJECT_SOURCE_DIR})
+ endif()
+
+ # Collect excludes (Patterns, not paths, for fastcov)
+ set(FASTCOV_EXCLUDES "")
+ foreach(EXCLUDE ${Coverage_EXCLUDE} ${COVERAGE_EXCLUDES} ${COVERAGE_FASTCOV_EXCLUDES})
+ list(APPEND FASTCOV_EXCLUDES "${EXCLUDE}")
+ endforeach()
+ list(REMOVE_DUPLICATES FASTCOV_EXCLUDES)
+
+ # Conditional arguments
+ if(CPPFILT_PATH AND NOT ${Coverage_NO_DEMANGLE})
+ set(GENHTML_EXTRA_ARGS "--demangle-cpp")
+ endif()
+
+ # Set up commands which will be run to generate coverage data
+ set(FASTCOV_EXEC_TESTS_CMD ${Coverage_EXECUTABLE} ${Coverage_EXECUTABLE_ARGS})
+
+ set(FASTCOV_CAPTURE_CMD ${FASTCOV_PATH} ${Coverage_FASTCOV_ARGS} --gcov ${GCOV_PATH}
+ --search-directory ${BASEDIR}
+ --process-gcno
+ --output ${Coverage_NAME}.json
+ --exclude ${FASTCOV_EXCLUDES}
+ --exclude ${FASTCOV_EXCLUDES}
+ )
+
+ set(FASTCOV_CONVERT_CMD ${FASTCOV_PATH}
+ -C ${Coverage_NAME}.json --lcov --output ${Coverage_NAME}.info
+ )
+
+ if(Coverage_SKIP_HTML)
+ set(FASTCOV_HTML_CMD ";")
+ else()
+ set(FASTCOV_HTML_CMD ${GENHTML_PATH} ${GENHTML_EXTRA_ARGS} ${Coverage_GENHTML_ARGS}
+ -o ${Coverage_NAME} ${Coverage_NAME}.info
+ )
+ endif()
+
+ set(FASTCOV_POST_CMD ";")
+ if(Coverage_POST_CMD)
+ set(FASTCOV_POST_CMD ${Coverage_POST_CMD})
+ endif()
+
+ if(CODE_COVERAGE_VERBOSE)
+ message(STATUS "Code coverage commands for target ${Coverage_NAME} (fastcov):")
+
+ message(" Running tests:")
+ string(REPLACE ";" " " FASTCOV_EXEC_TESTS_CMD_SPACED "${FASTCOV_EXEC_TESTS_CMD}")
+ message(" ${FASTCOV_EXEC_TESTS_CMD_SPACED}")
+
+ message(" Capturing fastcov counters and generating report:")
+ string(REPLACE ";" " " FASTCOV_CAPTURE_CMD_SPACED "${FASTCOV_CAPTURE_CMD}")
+ message(" ${FASTCOV_CAPTURE_CMD_SPACED}")
+
+ message(" Converting fastcov .json to lcov .info:")
+ string(REPLACE ";" " " FASTCOV_CONVERT_CMD_SPACED "${FASTCOV_CONVERT_CMD}")
+ message(" ${FASTCOV_CONVERT_CMD_SPACED}")
+
+ if(NOT Coverage_SKIP_HTML)
+ message(" Generating HTML report: ")
+ string(REPLACE ";" " " FASTCOV_HTML_CMD_SPACED "${FASTCOV_HTML_CMD}")
+ message(" ${FASTCOV_HTML_CMD_SPACED}")
+ endif()
+ if(Coverage_POST_CMD)
+ message(" Running post command: ")
+ string(REPLACE ";" " " FASTCOV_POST_CMD_SPACED "${FASTCOV_POST_CMD}")
+ message(" ${FASTCOV_POST_CMD_SPACED}")
+ endif()
+ endif()
+
+ # Setup target
+ add_custom_target(${Coverage_NAME}
+
+ # Cleanup fastcov
+ COMMAND ${FASTCOV_PATH} ${Coverage_FASTCOV_ARGS} --gcov ${GCOV_PATH}
+ --search-directory ${BASEDIR}
+ --zerocounters
+
+ COMMAND ${FASTCOV_EXEC_TESTS_CMD}
+ COMMAND ${FASTCOV_CAPTURE_CMD}
+ COMMAND ${FASTCOV_CONVERT_CMD}
+ COMMAND ${FASTCOV_HTML_CMD}
+ COMMAND ${FASTCOV_POST_CMD}
+
+ # Set output files as GENERATED (will be removed on 'make clean')
+ BYPRODUCTS
+ ${Coverage_NAME}.info
+ ${Coverage_NAME}.json
+ ${Coverage_NAME}/index.html # report directory
+
+ WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
+ DEPENDS ${Coverage_DEPENDENCIES}
+ VERBATIM # Protect arguments to commands
+ COMMENT "Resetting code coverage counters to zero. Processing code coverage counters and generating report."
+ )
+
+ set(INFO_MSG "fastcov code coverage info report saved in ${Coverage_NAME}.info and ${Coverage_NAME}.json.")
+ if(NOT Coverage_SKIP_HTML)
+ string(APPEND INFO_MSG " Open ${PROJECT_BINARY_DIR}/${Coverage_NAME}/index.html in your browser to view the coverage report.")
+ endif()
+ # Show where to find the fastcov info report
+ add_custom_command(TARGET ${Coverage_NAME} POST_BUILD
+ COMMAND ${CMAKE_COMMAND} -E echo ${INFO_MSG}
+ )
+
+endfunction() # setup_target_for_coverage_fastcov
+
+function(append_coverage_compiler_flags)
+ set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${COVERAGE_COMPILER_FLAGS}" PARENT_SCOPE)
+ set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${COVERAGE_COMPILER_FLAGS}" PARENT_SCOPE)
+ set(CMAKE_Fortran_FLAGS "${CMAKE_Fortran_FLAGS} ${COVERAGE_COMPILER_FLAGS}" PARENT_SCOPE)
+ message(STATUS "Appending code coverage compiler flags: ${COVERAGE_COMPILER_FLAGS}")
+endfunction() # append_coverage_compiler_flags
+
+# Setup coverage for specific library
+function(append_coverage_compiler_flags_to_target name)
+ target_compile_options(${name}
+ PRIVATE ${COVERAGE_COMPILER_FLAGS})
+endfunction()
diff --git a/cmake/cmake-modules/bilke/LICENSE_1_0.txt b/cmake/cmake-modules/bilke/LICENSE_1_0.txt
new file mode 100644
index 000000000..36b7cd93c
--- /dev/null
+++ b/cmake/cmake-modules/bilke/LICENSE_1_0.txt
@@ -0,0 +1,23 @@
+Boost Software License - Version 1.0 - August 17th, 2003
+
+Permission is hereby granted, free of charge, to any person or organization
+obtaining a copy of the software and accompanying documentation covered by
+this license (the "Software") to use, reproduce, display, distribute,
+execute, and transmit the Software, and to prepare derivative works of the
+Software, and to permit third-parties to whom the Software is furnished to
+do so, all subject to the following:
+
+The copyright notices in the Software and this entire statement, including
+the above license grant, this restriction and the following disclaimer,
+must be included in all copies of the Software, in whole or in part, and
+all derivative works of the Software, unless such copies or derivative
+works are solely in the form of machine-executable object code generated by
+a source language processor.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
+SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
+FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
+ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
diff --git a/cmake/cmake-modules/rpavlik/GetGitRevisionDescription.cmake b/cmake/cmake-modules/rpavlik/GetGitRevisionDescription.cmake
new file mode 100644
index 000000000..69ef78b28
--- /dev/null
+++ b/cmake/cmake-modules/rpavlik/GetGitRevisionDescription.cmake
@@ -0,0 +1,284 @@
+# - Returns a version string from Git
+#
+# These functions force a re-configure on each git commit so that you can
+# trust the values of the variables in your build system.
+#
+# get_git_head_revision( [ALLOW_LOOKING_ABOVE_CMAKE_SOURCE_DIR])
+#
+# Returns the refspec and sha hash of the current head revision
+#
+# git_describe( [ ...])
+#
+# Returns the results of git describe on the source tree, and adjusting
+# the output so that it tests false if an error occurs.
+#
+# git_describe_working_tree( [ ...])
+#
+# Returns the results of git describe on the working tree (--dirty option),
+# and adjusting the output so that it tests false if an error occurs.
+#
+# git_get_exact_tag( [ ...])
+#
+# Returns the results of git describe --exact-match on the source tree,
+# and adjusting the output so that it tests false if there was no exact
+# matching tag.
+#
+# git_local_changes()
+#
+# Returns either "CLEAN" or "DIRTY" with respect to uncommitted changes.
+# Uses the return code of "git diff-index --quiet HEAD --".
+# Does not regard untracked files.
+#
+# Requires CMake 2.6 or newer (uses the 'function' command)
+#
+# Original Author:
+# 2009-2020 Ryan Pavlik
+# http://academic.cleardefinition.com
+#
+# Copyright 2009-2013, Iowa State University.
+# Copyright 2013-2020, Ryan Pavlik
+# Copyright 2013-2020, Contributors
+# SPDX-License-Identifier: BSL-1.0
+# Distributed under the Boost Software License, Version 1.0.
+# (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+if(__get_git_revision_description)
+ return()
+endif()
+set(__get_git_revision_description YES)
+
+# We must run the following at "include" time, not at function call time,
+# to find the path to this module rather than the path to a calling list file
+get_filename_component(_gitdescmoddir ${CMAKE_CURRENT_LIST_FILE} PATH)
+
+# Function _git_find_closest_git_dir finds the next closest .git directory
+# that is part of any directory in the path defined by _start_dir.
+# The result is returned in the parent scope variable whose name is passed
+# as variable _git_dir_var. If no .git directory can be found, the
+# function returns an empty string via _git_dir_var.
+#
+# Example: Given a path C:/bla/foo/bar and assuming C:/bla/.git exists and
+# neither foo nor bar contain a file/directory .git. This wil return
+# C:/bla/.git
+#
+function(_git_find_closest_git_dir _start_dir _git_dir_var)
+ set(cur_dir "${_start_dir}")
+ set(git_dir "${_start_dir}/.git")
+ while(NOT EXISTS "${git_dir}")
+ # .git dir not found, search parent directories
+ set(git_previous_parent "${cur_dir}")
+ get_filename_component(cur_dir "${cur_dir}" DIRECTORY)
+ if(cur_dir STREQUAL git_previous_parent)
+ # We have reached the root directory, we are not in git
+ set(${_git_dir_var}
+ ""
+ PARENT_SCOPE)
+ return()
+ endif()
+ set(git_dir "${cur_dir}/.git")
+ endwhile()
+ set(${_git_dir_var}
+ "${git_dir}"
+ PARENT_SCOPE)
+endfunction()
+
+function(get_git_head_revision _refspecvar _hashvar)
+ _git_find_closest_git_dir("${CMAKE_CURRENT_SOURCE_DIR}" GIT_DIR)
+
+ if("${ARGN}" STREQUAL "ALLOW_LOOKING_ABOVE_CMAKE_SOURCE_DIR")
+ set(ALLOW_LOOKING_ABOVE_CMAKE_SOURCE_DIR TRUE)
+ else()
+ set(ALLOW_LOOKING_ABOVE_CMAKE_SOURCE_DIR FALSE)
+ endif()
+ if(NOT "${GIT_DIR}" STREQUAL "")
+ file(RELATIVE_PATH _relative_to_source_dir "${CMAKE_SOURCE_DIR}"
+ "${GIT_DIR}")
+ if("${_relative_to_source_dir}" MATCHES "[.][.]" AND NOT ALLOW_LOOKING_ABOVE_CMAKE_SOURCE_DIR)
+ # We've gone above the CMake root dir.
+ set(GIT_DIR "")
+ endif()
+ endif()
+ if("${GIT_DIR}" STREQUAL "")
+ set(${_refspecvar}
+ "GITDIR-NOTFOUND"
+ PARENT_SCOPE)
+ set(${_hashvar}
+ "GITDIR-NOTFOUND"
+ PARENT_SCOPE)
+ return()
+ endif()
+
+ # Check if the current source dir is a git submodule or a worktree.
+ # In both cases .git is a file instead of a directory.
+ #
+ if(NOT IS_DIRECTORY ${GIT_DIR})
+ # The following git command will return a non empty string that
+ # points to the super project working tree if the current
+ # source dir is inside a git submodule.
+ # Otherwise the command will return an empty string.
+ #
+ execute_process(
+ COMMAND "${GIT_EXECUTABLE}" rev-parse
+ --show-superproject-working-tree
+ WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}"
+ OUTPUT_VARIABLE out
+ ERROR_QUIET OUTPUT_STRIP_TRAILING_WHITESPACE)
+ if(NOT "${out}" STREQUAL "")
+ # If out is empty, GIT_DIR/CMAKE_CURRENT_SOURCE_DIR is in a submodule
+ file(READ ${GIT_DIR} submodule)
+ string(REGEX REPLACE "gitdir: (.*)$" "\\1" GIT_DIR_RELATIVE
+ ${submodule})
+ string(STRIP ${GIT_DIR_RELATIVE} GIT_DIR_RELATIVE)
+ get_filename_component(SUBMODULE_DIR ${GIT_DIR} PATH)
+ get_filename_component(GIT_DIR ${SUBMODULE_DIR}/${GIT_DIR_RELATIVE}
+ ABSOLUTE)
+ set(HEAD_SOURCE_FILE "${GIT_DIR}/HEAD")
+ else()
+ # GIT_DIR/CMAKE_CURRENT_SOURCE_DIR is in a worktree
+ file(READ ${GIT_DIR} worktree_ref)
+ # The .git directory contains a path to the worktree information directory
+ # inside the parent git repo of the worktree.
+ #
+ string(REGEX REPLACE "gitdir: (.*)$" "\\1" git_worktree_dir
+ ${worktree_ref})
+ string(STRIP ${git_worktree_dir} git_worktree_dir)
+ _git_find_closest_git_dir("${git_worktree_dir}" GIT_DIR)
+ set(HEAD_SOURCE_FILE "${git_worktree_dir}/HEAD")
+ endif()
+ else()
+ set(HEAD_SOURCE_FILE "${GIT_DIR}/HEAD")
+ endif()
+ set(GIT_DATA "${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/git-data")
+ if(NOT EXISTS "${GIT_DATA}")
+ file(MAKE_DIRECTORY "${GIT_DATA}")
+ endif()
+
+ if(NOT EXISTS "${HEAD_SOURCE_FILE}")
+ return()
+ endif()
+ set(HEAD_FILE "${GIT_DATA}/HEAD")
+ configure_file("${HEAD_SOURCE_FILE}" "${HEAD_FILE}" COPYONLY)
+
+ configure_file("${_gitdescmoddir}/GetGitRevisionDescription.cmake.in"
+ "${GIT_DATA}/grabRef.cmake" @ONLY)
+ include("${GIT_DATA}/grabRef.cmake")
+
+ set(${_refspecvar}
+ "${HEAD_REF}"
+ PARENT_SCOPE)
+ set(${_hashvar}
+ "${HEAD_HASH}"
+ PARENT_SCOPE)
+endfunction()
+
+function(git_describe _var)
+ if(NOT GIT_FOUND)
+ find_package(Git QUIET)
+ endif()
+ get_git_head_revision(refspec hash)
+ if(NOT GIT_FOUND)
+ set(${_var}
+ "GIT-NOTFOUND"
+ PARENT_SCOPE)
+ return()
+ endif()
+ if(NOT hash)
+ set(${_var}
+ "HEAD-HASH-NOTFOUND"
+ PARENT_SCOPE)
+ return()
+ endif()
+
+ # TODO sanitize
+ #if((${ARGN}" MATCHES "&&") OR
+ # (ARGN MATCHES "||") OR
+ # (ARGN MATCHES "\\;"))
+ # message("Please report the following error to the project!")
+ # message(FATAL_ERROR "Looks like someone's doing something nefarious with git_describe! Passed arguments ${ARGN}")
+ #endif()
+
+ #message(STATUS "Arguments to execute_process: ${ARGN}")
+
+ execute_process(
+ COMMAND "${GIT_EXECUTABLE}" describe --tags --always ${hash} ${ARGN}
+ WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}"
+ RESULT_VARIABLE res
+ OUTPUT_VARIABLE out
+ ERROR_QUIET OUTPUT_STRIP_TRAILING_WHITESPACE)
+ if(NOT res EQUAL 0)
+ set(out "${out}-${res}-NOTFOUND")
+ endif()
+
+ set(${_var}
+ "${out}"
+ PARENT_SCOPE)
+endfunction()
+
+function(git_describe_working_tree _var)
+ if(NOT GIT_FOUND)
+ find_package(Git QUIET)
+ endif()
+ if(NOT GIT_FOUND)
+ set(${_var}
+ "GIT-NOTFOUND"
+ PARENT_SCOPE)
+ return()
+ endif()
+
+ execute_process(
+ COMMAND "${GIT_EXECUTABLE}" describe --dirty ${ARGN}
+ WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}"
+ RESULT_VARIABLE res
+ OUTPUT_VARIABLE out
+ ERROR_QUIET OUTPUT_STRIP_TRAILING_WHITESPACE)
+ if(NOT res EQUAL 0)
+ set(out "${out}-${res}-NOTFOUND")
+ endif()
+
+ set(${_var}
+ "${out}"
+ PARENT_SCOPE)
+endfunction()
+
+function(git_get_exact_tag _var)
+ git_describe(out --exact-match ${ARGN})
+ set(${_var}
+ "${out}"
+ PARENT_SCOPE)
+endfunction()
+
+function(git_local_changes _var)
+ if(NOT GIT_FOUND)
+ find_package(Git QUIET)
+ endif()
+ get_git_head_revision(refspec hash)
+ if(NOT GIT_FOUND)
+ set(${_var}
+ "GIT-NOTFOUND"
+ PARENT_SCOPE)
+ return()
+ endif()
+ if(NOT hash)
+ set(${_var}
+ "HEAD-HASH-NOTFOUND"
+ PARENT_SCOPE)
+ return()
+ endif()
+
+ execute_process(
+ COMMAND "${GIT_EXECUTABLE}" diff-index --quiet HEAD --
+ WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}"
+ RESULT_VARIABLE res
+ OUTPUT_VARIABLE out
+ ERROR_QUIET OUTPUT_STRIP_TRAILING_WHITESPACE)
+ if(res EQUAL 0)
+ set(${_var}
+ "CLEAN"
+ PARENT_SCOPE)
+ else()
+ set(${_var}
+ "DIRTY"
+ PARENT_SCOPE)
+ endif()
+endfunction()
diff --git a/cmake/cmake-modules/rpavlik/GetGitRevisionDescription.cmake.in b/cmake/cmake-modules/rpavlik/GetGitRevisionDescription.cmake.in
new file mode 100644
index 000000000..66eee6377
--- /dev/null
+++ b/cmake/cmake-modules/rpavlik/GetGitRevisionDescription.cmake.in
@@ -0,0 +1,43 @@
+#
+# Internal file for GetGitRevisionDescription.cmake
+#
+# Requires CMake 2.6 or newer (uses the 'function' command)
+#
+# Original Author:
+# 2009-2010 Ryan Pavlik
+# http://academic.cleardefinition.com
+# Iowa State University HCI Graduate Program/VRAC
+#
+# Copyright 2009-2012, Iowa State University
+# Copyright 2011-2015, Contributors
+# Distributed under the Boost Software License, Version 1.0.
+# (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+# SPDX-License-Identifier: BSL-1.0
+
+set(HEAD_HASH)
+
+file(READ "@HEAD_FILE@" HEAD_CONTENTS LIMIT 1024)
+
+string(STRIP "${HEAD_CONTENTS}" HEAD_CONTENTS)
+if(HEAD_CONTENTS MATCHES "ref")
+ # named branch
+ string(REPLACE "ref: " "" HEAD_REF "${HEAD_CONTENTS}")
+ if(EXISTS "@GIT_DIR@/${HEAD_REF}")
+ configure_file("@GIT_DIR@/${HEAD_REF}" "@GIT_DATA@/head-ref" COPYONLY)
+ else()
+ configure_file("@GIT_DIR@/packed-refs" "@GIT_DATA@/packed-refs" COPYONLY)
+ file(READ "@GIT_DATA@/packed-refs" PACKED_REFS)
+ if(${PACKED_REFS} MATCHES "([0-9a-z]*) ${HEAD_REF}")
+ set(HEAD_HASH "${CMAKE_MATCH_1}")
+ endif()
+ endif()
+else()
+ # detached HEAD
+ configure_file("@GIT_DIR@/HEAD" "@GIT_DATA@/head-ref" COPYONLY)
+endif()
+
+if(NOT HEAD_HASH)
+ file(READ "@GIT_DATA@/head-ref" HEAD_HASH LIMIT 1024)
+ string(STRIP "${HEAD_HASH}" HEAD_HASH)
+endif()
diff --git a/cmake/cmake-modules/rpavlik/LICENSES/BSD-3-Clause.txt b/cmake/cmake-modules/rpavlik/LICENSES/BSD-3-Clause.txt
new file mode 100644
index 000000000..0741db789
--- /dev/null
+++ b/cmake/cmake-modules/rpavlik/LICENSES/BSD-3-Clause.txt
@@ -0,0 +1,26 @@
+Copyright (c) . All rights reserved.
+
+Redistribution and use in source and binary forms, with or without modification,
+are permitted provided that the following conditions are met:
+
+1. Redistributions of source code must retain the above copyright notice,
+this list of conditions and the following disclaimer.
+
+2. Redistributions in binary form must reproduce the above copyright notice,
+this list of conditions and the following disclaimer in the documentation
+and/or other materials provided with the distribution.
+
+3. Neither the name of the copyright holder nor the names of its contributors
+may be used to endorse or promote products derived from this software without
+specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
+USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
diff --git a/cmake/cmake-modules/rpavlik/LICENSES/BSL-1.0.txt b/cmake/cmake-modules/rpavlik/LICENSES/BSL-1.0.txt
new file mode 100644
index 000000000..cff35365a
--- /dev/null
+++ b/cmake/cmake-modules/rpavlik/LICENSES/BSL-1.0.txt
@@ -0,0 +1,23 @@
+Boost Software License - Version 1.0 - August 17th, 2003
+
+Permission is hereby granted, free of charge, to any person or organization
+obtaining a copy of the software and accompanying documentation covered by
+this license (the "Software") to use, reproduce, display, distribute, execute,
+and transmit the Software, and to prepare derivative works of the Software,
+and to permit third-parties to whom the Software is furnished to do so, all
+subject to the following:
+
+The copyright notices in the Software and this entire statement, including
+the above license grant, this restriction and the following disclaimer, must
+be included in all copies of the Software, in whole or in part, and all derivative
+works of the Software, unless such copies or derivative works are solely in
+the form of machine-executable object code generated by a source language
+processor.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
+FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
+COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE FOR ANY DAMAGES
+OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
diff --git a/cmake/cmake-modules/rpavlik/LICENSE_1_0.txt b/cmake/cmake-modules/rpavlik/LICENSE_1_0.txt
new file mode 100644
index 000000000..36b7cd93c
--- /dev/null
+++ b/cmake/cmake-modules/rpavlik/LICENSE_1_0.txt
@@ -0,0 +1,23 @@
+Boost Software License - Version 1.0 - August 17th, 2003
+
+Permission is hereby granted, free of charge, to any person or organization
+obtaining a copy of the software and accompanying documentation covered by
+this license (the "Software") to use, reproduce, display, distribute,
+execute, and transmit the Software, and to prepare derivative works of the
+Software, and to permit third-parties to whom the Software is furnished to
+do so, all subject to the following:
+
+The copyright notices in the Software and this entire statement, including
+the above license grant, this restriction and the following disclaimer,
+must be included in all copies of the Software, in whole or in part, and
+all derivative works of the Software, unless such copies or derivative
+works are solely in the form of machine-executable object code generated by
+a source language processor.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
+SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
+FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
+ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
diff --git a/docs/getting_started.rst b/docs/getting_started.rst
index 34547211e..01724b3a4 100644
--- a/docs/getting_started.rst
+++ b/docs/getting_started.rst
@@ -19,6 +19,29 @@ A template configuration folder was provided and can be copied into the project
a starting point. The [configuration section](docs/README-config.md#top) provides more specific
information about the possible options.
+Prerequisites
+-------------------
+
+The Embedded Template Library (etl) is a dependency of the FSFW which is automatically
+installed and provided by the build system unless the correction version was installed.
+The current recommended version can be found inside the fsfw ``CMakeLists.txt`` file or by using
+``ccmake`` and looking up the ``FSFW_ETL_LIB_MAJOR_VERSION`` variable.
+
+You can install the ETL library like this. On Linux, it might be necessary to add ``sudo`` before
+the install call:
+
+.. code-block:: console
+
+ git clone https://github.com/ETLCPP/etl
+ cd etl
+ git checkout
+ mkdir build && cd build
+ cmake ..
+ cmake --install .
+
+It is recommended to install ``20.27.2`` or newer for the package version handling of
+ETL to work.
+
Adding the library
-------------------
@@ -60,6 +83,20 @@ The FSFW also has unittests which use the `Catch2 library`_.
These are built by setting the CMake option ``FSFW_BUILD_UNITTESTS`` to ``ON`` or `TRUE`
from your project `CMakeLists.txt` file or from the command line.
+You can install the Catch2 library, which prevents the build system to avoid re-downloading
+the dependency if the unit tests are completely rebuilt. The current recommended version
+can be found inside the fsfw ``CMakeLists.txt`` file or by using ``ccmake`` and looking up
+the ``FSFW_CATCH2_LIB_VERSION`` variable.
+
+.. code-block:: console
+
+ git clone https://github.com/catchorg/Catch2.git
+ cd Catch2
+ git checkout
+ cmake -Bbuild -H. -DBUILD_TESTING=OFF
+ sudo cmake --build build/ --target install
+
+
The fsfw-tests binary will be built as part of the static library and dropped alongside it.
If the unittests are built, the library and the tests will be built with coverage information by
default. This can be disabled by setting the `FSFW_TESTS_COV_GEN` option to `OFF` or `FALSE`.
diff --git a/hal/CMakeLists.txt b/hal/CMakeLists.txt
deleted file mode 100644
index 424a012d5..000000000
--- a/hal/CMakeLists.txt
+++ /dev/null
@@ -1,47 +0,0 @@
-cmake_minimum_required(VERSION 3.13)
-
-# Can also be changed by upper CMakeLists.txt file
-find_library(LIB_FSFW_NAME fsfw REQUIRED)
-
-option(FSFW_HAL_ADD_LINUX "Add the Linux HAL to the sources. Requires gpiod library" OFF)
-# On by default for now because I did not have an issue including and compiling those files
-# and libraries on a Desktop Linux system and the primary target of the FSFW is still embedded
-# Linux. The only exception from this is the gpiod library which requires a dedicated installation,
-# but CMake is able to determine whether this library is installed with find_library.
-option(FSFW_HAL_LINUX_ADD_PERIPHERAL_DRIVERS "Add peripheral drivers for embedded Linux" ON)
-
-option(FSFW_HAL_ADD_RASPBERRY_PI "Add Raspberry Pi specific code to the sources" OFF)
-option(FSFW_HAL_ADD_STM32H7 "Add the STM32H7 HAL to the sources" OFF)
-option(FSFW_HAL_WARNING_SHADOW_LOCAL_GCC "Enable -Wshadow=local warning in GCC" ON)
-
-set(LINUX_HAL_PATH_NAME linux)
-set(STM32H7_PATH_NAME stm32h7)
-
-add_subdirectory(src)
-
-foreach(INCLUDE_PATH ${FSFW_HAL_ADDITIONAL_INC_PATHS})
- if(IS_ABSOLUTE ${INCLUDE_PATH})
- set(CURR_ABS_INC_PATH "${INCLUDE_PATH}")
- else()
- get_filename_component(CURR_ABS_INC_PATH
- ${INCLUDE_PATH} REALPATH BASE_DIR ${CMAKE_SOURCE_DIR})
- endif()
-
- if(CMAKE_VERBOSE)
- message(STATUS "FSFW include path: ${CURR_ABS_INC_PATH}")
- endif()
-
- list(APPEND FSFW_HAL_ADD_INC_PATHS_ABS ${CURR_ABS_INC_PATH})
-endforeach()
-
-target_include_directories(${LIB_FSFW_NAME} PRIVATE
- ${FSFW_HAL_ADD_INC_PATHS_ABS}
-)
-
-target_compile_definitions(${LIB_FSFW_NAME} PRIVATE
- ${FSFW_HAL_DEFINES}
-)
-
-target_link_libraries(${LIB_FSFW_NAME} PRIVATE
- ${FSFW_HAL_LINK_LIBS}
-)
diff --git a/hal/src/CMakeLists.txt b/hal/src/CMakeLists.txt
deleted file mode 100644
index 76ee45c6d..000000000
--- a/hal/src/CMakeLists.txt
+++ /dev/null
@@ -1,9 +0,0 @@
-target_include_directories(${LIB_FSFW_NAME} PRIVATE
- ${CMAKE_CURRENT_SOURCE_DIR}
-)
-
-target_include_directories(${LIB_FSFW_NAME} INTERFACE
- ${CMAKE_CURRENT_SOURCE_DIR}
-)
-
-add_subdirectory(fsfw_hal)
diff --git a/hal/src/fsfw_hal/common/gpio/CMakeLists.txt b/hal/src/fsfw_hal/common/gpio/CMakeLists.txt
deleted file mode 100644
index 098c05fa1..000000000
--- a/hal/src/fsfw_hal/common/gpio/CMakeLists.txt
+++ /dev/null
@@ -1,3 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- GpioCookie.cpp
-)
\ No newline at end of file
diff --git a/hal/src/fsfw_hal/devicehandlers/CMakeLists.txt b/hal/src/fsfw_hal/devicehandlers/CMakeLists.txt
deleted file mode 100644
index 94e67c722..000000000
--- a/hal/src/fsfw_hal/devicehandlers/CMakeLists.txt
+++ /dev/null
@@ -1,5 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- GyroL3GD20Handler.cpp
- MgmRM3100Handler.cpp
- MgmLIS3MDLHandler.cpp
-)
diff --git a/hal/src/fsfw_hal/linux/CMakeLists.txt b/hal/src/fsfw_hal/linux/CMakeLists.txt
deleted file mode 100644
index 0fb2d385c..000000000
--- a/hal/src/fsfw_hal/linux/CMakeLists.txt
+++ /dev/null
@@ -1,18 +0,0 @@
-if(FSFW_HAL_ADD_RASPBERRY_PI)
- add_subdirectory(rpi)
-endif()
-
-target_sources(${LIB_FSFW_NAME} PRIVATE
- UnixFileGuard.cpp
- CommandExecutor.cpp
- utility.cpp
-)
-
-if(FSFW_HAL_LINUX_ADD_PERIPHERAL_DRIVERS)
- add_subdirectory(gpio)
- add_subdirectory(spi)
- add_subdirectory(i2c)
- add_subdirectory(uart)
-endif()
-
-add_subdirectory(uio)
diff --git a/hal/src/fsfw_hal/linux/gpio/CMakeLists.txt b/hal/src/fsfw_hal/linux/gpio/CMakeLists.txt
deleted file mode 100644
index b16098504..000000000
--- a/hal/src/fsfw_hal/linux/gpio/CMakeLists.txt
+++ /dev/null
@@ -1,16 +0,0 @@
-# This abstraction layer requires the gpiod library. You can install this library
-# with "sudo apt-get install -y libgpiod-dev". If you are cross-compiling, you need
-# to install the package before syncing the sysroot to your host computer.
-find_library(LIB_GPIO gpiod)
-
-if(${LIB_GPIO} MATCHES LIB_GPIO-NOTFOUND)
- message(STATUS "gpiod library not found, not linking against it")
-else()
- target_sources(${LIB_FSFW_NAME} PRIVATE
- LinuxLibgpioIF.cpp
- )
- target_link_libraries(${LIB_FSFW_NAME} PRIVATE
- ${LIB_GPIO}
- )
-endif()
-
diff --git a/hal/src/fsfw_hal/linux/i2c/CMakeLists.txt b/hal/src/fsfw_hal/linux/i2c/CMakeLists.txt
deleted file mode 100644
index 3eb0882cc..000000000
--- a/hal/src/fsfw_hal/linux/i2c/CMakeLists.txt
+++ /dev/null
@@ -1,8 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PUBLIC
- I2cComIF.cpp
- I2cCookie.cpp
-)
-
-
-
-
diff --git a/hal/src/fsfw_hal/linux/rpi/CMakeLists.txt b/hal/src/fsfw_hal/linux/rpi/CMakeLists.txt
deleted file mode 100644
index 47be218c0..000000000
--- a/hal/src/fsfw_hal/linux/rpi/CMakeLists.txt
+++ /dev/null
@@ -1,3 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- GpioRPi.cpp
-)
\ No newline at end of file
diff --git a/hal/src/fsfw_hal/linux/spi/CMakeLists.txt b/hal/src/fsfw_hal/linux/spi/CMakeLists.txt
deleted file mode 100644
index 404e1f477..000000000
--- a/hal/src/fsfw_hal/linux/spi/CMakeLists.txt
+++ /dev/null
@@ -1,8 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PUBLIC
- SpiComIF.cpp
- SpiCookie.cpp
-)
-
-
-
-
diff --git a/hal/src/fsfw_hal/linux/uart/CMakeLists.txt b/hal/src/fsfw_hal/linux/uart/CMakeLists.txt
deleted file mode 100644
index 21ed0278f..000000000
--- a/hal/src/fsfw_hal/linux/uart/CMakeLists.txt
+++ /dev/null
@@ -1,4 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PUBLIC
- UartComIF.cpp
- UartCookie.cpp
-)
diff --git a/hal/src/fsfw_hal/linux/uio/CMakeLists.txt b/hal/src/fsfw_hal/linux/uio/CMakeLists.txt
deleted file mode 100644
index e98a08657..000000000
--- a/hal/src/fsfw_hal/linux/uio/CMakeLists.txt
+++ /dev/null
@@ -1,3 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PUBLIC
- UioMapper.cpp
-)
diff --git a/hal/src/fsfw_hal/stm32h7/devicetest/CMakeLists.txt b/hal/src/fsfw_hal/stm32h7/devicetest/CMakeLists.txt
deleted file mode 100644
index 7bd4c3a96..000000000
--- a/hal/src/fsfw_hal/stm32h7/devicetest/CMakeLists.txt
+++ /dev/null
@@ -1,3 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- GyroL3GD20H.cpp
-)
\ No newline at end of file
diff --git a/hal/src/fsfw_hal/stm32h7/gpio/CMakeLists.txt b/hal/src/fsfw_hal/stm32h7/gpio/CMakeLists.txt
deleted file mode 100644
index 35245b253..000000000
--- a/hal/src/fsfw_hal/stm32h7/gpio/CMakeLists.txt
+++ /dev/null
@@ -1,3 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- gpio.cpp
-)
diff --git a/hal/src/fsfw_hal/stm32h7/i2c/CMakeLists.txt b/hal/src/fsfw_hal/stm32h7/i2c/CMakeLists.txt
deleted file mode 100644
index 5ecb09909..000000000
--- a/hal/src/fsfw_hal/stm32h7/i2c/CMakeLists.txt
+++ /dev/null
@@ -1,2 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
-)
diff --git a/hal/src/fsfw_hal/stm32h7/spi/CMakeLists.txt b/hal/src/fsfw_hal/stm32h7/spi/CMakeLists.txt
deleted file mode 100644
index aa5541bc1..000000000
--- a/hal/src/fsfw_hal/stm32h7/spi/CMakeLists.txt
+++ /dev/null
@@ -1,9 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- spiCore.cpp
- spiDefinitions.cpp
- spiInterrupts.cpp
- mspInit.cpp
- SpiCookie.cpp
- SpiComIF.cpp
- stm32h743zi.cpp
-)
diff --git a/hal/src/fsfw_hal/stm32h7/uart/CMakeLists.txt b/hal/src/fsfw_hal/stm32h7/uart/CMakeLists.txt
deleted file mode 100644
index 5ecb09909..000000000
--- a/hal/src/fsfw_hal/stm32h7/uart/CMakeLists.txt
+++ /dev/null
@@ -1,2 +0,0 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
-)
diff --git a/scripts/apply-clang-format.sh b/scripts/apply-clang-format.sh
deleted file mode 100755
index 272023240..000000000
--- a/scripts/apply-clang-format.sh
+++ /dev/null
@@ -1,8 +0,0 @@
-#!/bin/bash
-if [[ ! -f README.md ]]; then
- cd ..
-fi
-
-find ./src -iname *.h -o -iname *.cpp -o -iname *.c | xargs clang-format --style=file -i
-find ./hal -iname *.h -o -iname *.cpp -o -iname *.c | xargs clang-format --style=file -i
-find ./tests -iname *.h -o -iname *.cpp -o -iname *.c | xargs clang-format --style=file -i
diff --git a/scripts/auto-formatter.sh b/scripts/auto-formatter.sh
new file mode 100755
index 000000000..c0ae7099c
--- /dev/null
+++ b/scripts/auto-formatter.sh
@@ -0,0 +1,22 @@
+#!/bin/bash
+if [[ ! -f README.md ]]; then
+ cd ..
+fi
+
+cmake_fmt="cmake-format"
+file_selectors="-iname CMakeLists.txt"
+if command -v ${cmake_fmt} &> /dev/null; then
+ ${cmake_fmt} -i CMakeLists.txt
+ find ./src ${file_selectors} | xargs ${cmake_fmt} -i
+else
+ echo "No ${cmake_fmt} tool found, not formatting CMake files"
+fi
+
+cpp_format="clang-format"
+file_selectors="-iname *.h -o -iname *.cpp -o -iname *.c -o -iname *.tpp"
+if command -v ${cpp_format} &> /dev/null; then
+ find ./src ${file_selectors} | xargs ${cpp_format} --style=file -i
+ find ./unittests ${file_selectors} | xargs ${cpp_format} --style=file -i
+else
+ echo "No ${cpp_format} tool found, not formatting C++/C files"
+fi
diff --git a/scripts/helper.py b/scripts/helper.py
index 68693c40f..0ac616b61 100755
--- a/scripts/helper.py
+++ b/scripts/helper.py
@@ -48,6 +48,20 @@ def main():
action="store_true",
help="Run valgrind on generated test binary",
)
+ parser.add_argument(
+ "-g",
+ "--generators",
+ default = "Ninja",
+ action="store",
+ help="CMake generators",
+ )
+ parser.add_argument(
+ "-w",
+ "--windows",
+ default=False,
+ action="store_true",
+ help="Run on windows",
+ )
args = parser.parse_args()
if args.all:
@@ -97,11 +111,11 @@ def handle_docs_type(args, build_dir_list: list):
build_directory = determine_build_dir(build_dir_list)
os.chdir(build_directory)
if args.build:
- os.system("cmake --build . -j")
+ cmd_runner("cmake --build . -j")
if args.open:
if not os.path.isfile("docs/sphinx/index.html"):
# try again..
- os.system("cmake --build . -j")
+ cmd_runner("cmake --build . -j")
if not os.path.isfile("docs/sphinx/index.html"):
print(
"No Sphinx documentation file detected. "
@@ -115,14 +129,14 @@ def handle_tests_type(args, build_dir_list: list):
if args.create:
if os.path.exists(UNITTEST_FOLDER_NAME):
shutil.rmtree(UNITTEST_FOLDER_NAME)
- create_tests_build_cfg()
+ create_tests_build_cfg(args)
build_directory = UNITTEST_FOLDER_NAME
elif len(build_dir_list) == 0:
print(
"No valid CMake tests build directory found. "
"Trying to set up test build system"
)
- create_tests_build_cfg()
+ create_tests_build_cfg(args)
build_directory = UNITTEST_FOLDER_NAME
elif len(build_dir_list) == 1:
build_directory = build_dir_list[0]
@@ -143,25 +157,26 @@ def handle_tests_type(args, build_dir_list: list):
if which("valgrind") is None:
print("Please install valgrind first")
sys.exit(1)
- if os.path.split(os.getcwd())[1] != UNITTEST_FOLDER_NAME:
- # If we are in a different directory we try to switch into it but
- # this might fail
- os.chdir(UNITTEST_FOLDER_NAME)
- os.system("valgrind --leak-check=full ./fsfw-tests")
+ cmd_runner("valgrind --leak-check=full ./fsfw-tests")
os.chdir("..")
-def create_tests_build_cfg():
+def create_tests_build_cfg(args):
os.mkdir(UNITTEST_FOLDER_NAME)
os.chdir(UNITTEST_FOLDER_NAME)
- os.system("cmake -DFSFW_OSAL=host -DFSFW_BUILD_UNITTESTS=ON ..")
+ if args.windows:
+ cmake_cmd = 'cmake -G "' + args.generators + '" -DFSFW_OSAL=host -DFSFW_BUILD_TESTS=ON \
+ -DGCOVR_PATH="py -m gcovr" ..'
+ else:
+ cmake_cmd = 'cmake -G "' + args.generators + '" -DFSFW_OSAL=host -DFSFW_BUILD_TESTS=ON ..'
+ cmd_runner(cmake_cmd)
os.chdir("..")
def create_docs_build_cfg():
os.mkdir(DOCS_FOLDER_NAME)
os.chdir(DOCS_FOLDER_NAME)
- os.system("cmake -DFSFW_OSAL=host -DFSFW_BUILD_DOCS=ON ..")
+ cmd_runner("cmake -DFSFW_OSAL=host -DFSFW_BUILD_DOCS=ON ..")
os.chdir("..")
@@ -184,7 +199,7 @@ def check_for_cmake_build_dir(build_dir_list: list) -> list:
def perform_lcov_operation(directory: str, chdir: bool):
if chdir:
os.chdir(directory)
- os.system("cmake --build . -- fsfw-tests_coverage -j")
+ cmd_runner("cmake --build . -- fsfw-tests_coverage -j")
def determine_build_dir(build_dir_list: List[str]):
@@ -206,5 +221,10 @@ def determine_build_dir(build_dir_list: List[str]):
return build_directory
+def cmd_runner(cmd: str):
+ print(f"Executing command: {cmd}")
+ os.system(cmd)
+
+
if __name__ == "__main__":
main()
diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt
index ed2f2522e..57b24bd55 100644
--- a/src/CMakeLists.txt
+++ b/src/CMakeLists.txt
@@ -1,9 +1,11 @@
-target_include_directories(${LIB_FSFW_NAME} PRIVATE
- ${CMAKE_CURRENT_SOURCE_DIR}
-)
+target_include_directories(${LIB_FSFW_NAME} PRIVATE ${CMAKE_CURRENT_SOURCE_DIR})
-target_include_directories(${LIB_FSFW_NAME} INTERFACE
- ${CMAKE_CURRENT_SOURCE_DIR}
-)
+target_include_directories(${LIB_FSFW_NAME}
+ INTERFACE ${CMAKE_CURRENT_SOURCE_DIR})
add_subdirectory(fsfw)
+if(FSFW_ADD_HAL)
+ add_subdirectory(fsfw_hal)
+endif()
+
+add_subdirectory(fsfw_tests)
diff --git a/src/fsfw/CMakeLists.txt b/src/fsfw/CMakeLists.txt
index 12c673ed6..1daad7142 100644
--- a/src/fsfw/CMakeLists.txt
+++ b/src/fsfw/CMakeLists.txt
@@ -1,3 +1,5 @@
+target_sources(${LIB_FSFW_NAME} PRIVATE version.cpp)
+
# Core
add_subdirectory(action)
@@ -33,22 +35,22 @@ add_subdirectory(tmtcservices)
# Optional
if(FSFW_ADD_MONITORING)
-add_subdirectory(monitoring)
+ add_subdirectory(monitoring)
endif()
if(FSFW_ADD_PUS)
- add_subdirectory(pus)
+ add_subdirectory(pus)
endif()
if(FSFW_ADD_TMSTORAGE)
- add_subdirectory(tmstorage)
+ add_subdirectory(tmstorage)
endif()
if(FSFW_ADD_COORDINATES)
- add_subdirectory(coordinates)
+ add_subdirectory(coordinates)
endif()
if(FSFW_ADD_RMAP)
- add_subdirectory(rmap)
+ add_subdirectory(rmap)
endif()
if(FSFW_ADD_DATALINKLAYER)
- add_subdirectory(datalinklayer)
+ add_subdirectory(datalinklayer)
endif()
# OSAL
diff --git a/src/fsfw/FSFW.h.in b/src/fsfw/FSFW.h.in
index 88ad10cf3..0eeb00eb0 100644
--- a/src/fsfw/FSFW.h.in
+++ b/src/fsfw/FSFW.h.in
@@ -30,6 +30,10 @@
#define FSFW_VERBOSE_LEVEL 1
#endif /* FSFW_VERBOSE_LEVEL */
+#ifndef FSFW_DISABLE_PRINTOUT
+#define FSFW_DISABLE_PRINTOUT 0
+#endif
+
#ifndef FSFW_USE_REALTIME_FOR_LINUX
#define FSFW_USE_REALTIME_FOR_LINUX 0
#endif /* FSFW_USE_REALTIME_FOR_LINUX */
@@ -57,16 +61,9 @@
#define FSFW_HAL_SPI_WIRETAPPING 0
#endif
-#ifndef FSFW_HAL_L3GD20_GYRO_DEBUG
-#define FSFW_HAL_L3GD20_GYRO_DEBUG 0
-#endif /* FSFW_HAL_L3GD20_GYRO_DEBUG */
-
-#ifndef FSFW_HAL_RM3100_MGM_DEBUG
-#define FSFW_HAL_RM3100_MGM_DEBUG 0
-#endif /* FSFW_HAL_RM3100_MGM_DEBUG */
-
-#ifndef FSFW_HAL_LIS3MDL_MGM_DEBUG
-#define FSFW_HAL_LIS3MDL_MGM_DEBUG 0
-#endif /* FSFW_HAL_LIS3MDL_MGM_DEBUG */
+// Can be used for low-level debugging of the I2C bus
+#ifndef FSFW_HAL_I2C_WIRETAPPING
+#define FSFW_HAL_I2C_WIRETAPPING 0
+#endif
#endif /* FSFW_FSFW_H_ */
diff --git a/src/fsfw/FSFWVersion.h.in b/src/fsfw/FSFWVersion.h.in
index 7935b2f61..caff1efbd 100644
--- a/src/fsfw/FSFWVersion.h.in
+++ b/src/fsfw/FSFWVersion.h.in
@@ -1,9 +1,11 @@
#ifndef FSFW_VERSION_H_
#define FSFW_VERSION_H_
-// Versioning is kept in project CMakeLists.txt file
-#define FSFW_VERSION @FSFW_VERSION@
-#define FSFW_SUBVERSION @FSFW_SUBVERSION@
-#define FSFW_REVISION @FSFW_REVISION@
+// Versioning is managed in project CMakeLists.txt file
+static constexpr int FSFW_VERSION_MAJOR = @FSFW_VERSION@;
+static constexpr int FSFW_VERSION_MINOR = @FSFW_SUBVERSION@;
+static constexpr int FSFW_VERSION_REVISION = @FSFW_REVISION@;
+// Also contains CST (Commits since tag) information
+static const char FSFW_VCS_INFO[] = "@FSFW_VCS_INFO@";
#endif /* FSFW_VERSION_H_ */
diff --git a/src/fsfw/action/ActionHelper.cpp b/src/fsfw/action/ActionHelper.cpp
index d41c78b4b..bee68b404 100644
--- a/src/fsfw/action/ActionHelper.cpp
+++ b/src/fsfw/action/ActionHelper.cpp
@@ -6,7 +6,7 @@
ActionHelper::ActionHelper(HasActionsIF* setOwner, MessageQueueIF* useThisQueue)
: owner(setOwner), queueToUse(useThisQueue) {}
-ActionHelper::~ActionHelper() {}
+ActionHelper::~ActionHelper() = default;
ReturnValue_t ActionHelper::handleActionMessage(CommandMessage* command) {
if (command->getCommand() == ActionMessage::EXECUTE_ACTION) {
@@ -59,7 +59,7 @@ void ActionHelper::setQueueToUse(MessageQueueIF* queue) { queueToUse = queue; }
void ActionHelper::prepareExecution(MessageQueueId_t commandedBy, ActionId_t actionId,
store_address_t dataAddress) {
- const uint8_t* dataPtr = NULL;
+ const uint8_t* dataPtr = nullptr;
size_t size = 0;
ReturnValue_t result = ipcStore->getData(dataAddress, &dataPtr, &size);
if (result != HasReturnvaluesIF::RETURN_OK) {
diff --git a/src/fsfw/action/ActionHelper.h b/src/fsfw/action/ActionHelper.h
index d86e87d23..a9910b05f 100644
--- a/src/fsfw/action/ActionHelper.h
+++ b/src/fsfw/action/ActionHelper.h
@@ -1,9 +1,9 @@
#ifndef FSFW_ACTION_ACTIONHELPER_H_
#define FSFW_ACTION_ACTIONHELPER_H_
-#include "../ipc/MessageQueueIF.h"
-#include "../serialize/SerializeIF.h"
#include "ActionMessage.h"
+#include "fsfw/ipc/MessageQueueIF.h"
+#include "fsfw/serialize/SerializeIF.h"
/**
* @brief Action Helper is a helper class which handles action messages
*
diff --git a/src/fsfw/action/ActionMessage.cpp b/src/fsfw/action/ActionMessage.cpp
index 40a516b17..7fc685586 100644
--- a/src/fsfw/action/ActionMessage.cpp
+++ b/src/fsfw/action/ActionMessage.cpp
@@ -2,9 +2,9 @@
#include "fsfw/objectmanager/ObjectManager.h"
#include "fsfw/storagemanager/StorageManagerIF.h"
-ActionMessage::ActionMessage() {}
+ActionMessage::ActionMessage() = default;
-ActionMessage::~ActionMessage() {}
+ActionMessage::~ActionMessage() = default;
void ActionMessage::setCommand(CommandMessage* message, ActionId_t fid,
store_address_t parameters) {
@@ -64,9 +64,8 @@ void ActionMessage::clear(CommandMessage* message) {
switch (message->getCommand()) {
case EXECUTE_ACTION:
case DATA_REPLY: {
- StorageManagerIF* ipcStore =
- ObjectManager::instance()->get(objects::IPC_STORE);
- if (ipcStore != NULL) {
+ auto* ipcStore = ObjectManager::instance()->get(objects::IPC_STORE);
+ if (ipcStore != nullptr) {
ipcStore->deleteData(getStoreId(message));
}
break;
diff --git a/src/fsfw/action/CMakeLists.txt b/src/fsfw/action/CMakeLists.txt
index f9ac451df..7fb397af3 100644
--- a/src/fsfw/action/CMakeLists.txt
+++ b/src/fsfw/action/CMakeLists.txt
@@ -1,7 +1,3 @@
-target_sources(${LIB_FSFW_NAME}
- PRIVATE
- ActionHelper.cpp
- ActionMessage.cpp
- CommandActionHelper.cpp
- SimpleActionHelper.cpp
-)
\ No newline at end of file
+target_sources(
+ ${LIB_FSFW_NAME} PRIVATE ActionHelper.cpp ActionMessage.cpp
+ CommandActionHelper.cpp SimpleActionHelper.cpp)
diff --git a/src/fsfw/action/CommandActionHelper.cpp b/src/fsfw/action/CommandActionHelper.cpp
index 19d8e9b89..a06bc44c9 100644
--- a/src/fsfw/action/CommandActionHelper.cpp
+++ b/src/fsfw/action/CommandActionHelper.cpp
@@ -2,14 +2,14 @@
#include "fsfw/objectmanager/ObjectManager.h"
CommandActionHelper::CommandActionHelper(CommandsActionsIF *setOwner)
- : owner(setOwner), queueToUse(NULL), ipcStore(NULL), commandCount(0), lastTarget(0) {}
+ : owner(setOwner), queueToUse(nullptr), ipcStore(nullptr), commandCount(0), lastTarget(0) {}
-CommandActionHelper::~CommandActionHelper() {}
+CommandActionHelper::~CommandActionHelper() = default;
ReturnValue_t CommandActionHelper::commandAction(object_id_t commandTo, ActionId_t actionId,
SerializeIF *data) {
- HasActionsIF *receiver = ObjectManager::instance()->get(commandTo);
- if (receiver == NULL) {
+ auto *receiver = ObjectManager::instance()->get(commandTo);
+ if (receiver == nullptr) {
return CommandsActionsIF::OBJECT_HAS_NO_FUNCTIONS;
}
store_address_t storeId;
@@ -29,11 +29,8 @@ ReturnValue_t CommandActionHelper::commandAction(object_id_t commandTo, ActionId
ReturnValue_t CommandActionHelper::commandAction(object_id_t commandTo, ActionId_t actionId,
const uint8_t *data, uint32_t size) {
- // if (commandCount != 0) {
- // return CommandsFunctionsIF::ALREADY_COMMANDING;
- // }
- HasActionsIF *receiver = ObjectManager::instance()->get(commandTo);
- if (receiver == NULL) {
+ auto *receiver = ObjectManager::instance()->get(commandTo);
+ if (receiver == nullptr) {
return CommandsActionsIF::OBJECT_HAS_NO_FUNCTIONS;
}
store_address_t storeId;
@@ -59,12 +56,12 @@ ReturnValue_t CommandActionHelper::sendCommand(MessageQueueId_t queueId, ActionI
ReturnValue_t CommandActionHelper::initialize() {
ipcStore = ObjectManager::instance()->get(objects::IPC_STORE);
- if (ipcStore == NULL) {
+ if (ipcStore == nullptr) {
return HasReturnvaluesIF::RETURN_FAILED;
}
queueToUse = owner->getCommandQueuePtr();
- if (queueToUse == NULL) {
+ if (queueToUse == nullptr) {
return HasReturnvaluesIF::RETURN_FAILED;
}
return HasReturnvaluesIF::RETURN_OK;
@@ -104,7 +101,7 @@ ReturnValue_t CommandActionHelper::handleReply(CommandMessage *reply) {
uint8_t CommandActionHelper::getCommandCount() const { return commandCount; }
void CommandActionHelper::extractDataForOwner(ActionId_t actionId, store_address_t storeId) {
- const uint8_t *data = NULL;
+ const uint8_t *data = nullptr;
size_t size = 0;
ReturnValue_t result = ipcStore->getData(storeId, &data, &size);
if (result != HasReturnvaluesIF::RETURN_OK) {
diff --git a/src/fsfw/action/CommandActionHelper.h b/src/fsfw/action/CommandActionHelper.h
index dd8ad7f1c..a6ec0ca72 100644
--- a/src/fsfw/action/CommandActionHelper.h
+++ b/src/fsfw/action/CommandActionHelper.h
@@ -14,14 +14,14 @@ class CommandActionHelper {
friend class CommandsActionsIF;
public:
- CommandActionHelper(CommandsActionsIF* owner);
+ explicit CommandActionHelper(CommandsActionsIF* owner);
virtual ~CommandActionHelper();
ReturnValue_t commandAction(object_id_t commandTo, ActionId_t actionId, const uint8_t* data,
uint32_t size);
ReturnValue_t commandAction(object_id_t commandTo, ActionId_t actionId, SerializeIF* data);
ReturnValue_t initialize();
ReturnValue_t handleReply(CommandMessage* reply);
- uint8_t getCommandCount() const;
+ [[nodiscard]] uint8_t getCommandCount() const;
private:
CommandsActionsIF* owner;
diff --git a/src/fsfw/action/CommandsActionsIF.h b/src/fsfw/action/CommandsActionsIF.h
index 5870a9f2a..94d9a7c4e 100644
--- a/src/fsfw/action/CommandsActionsIF.h
+++ b/src/fsfw/action/CommandsActionsIF.h
@@ -1,9 +1,9 @@
#ifndef FSFW_ACTION_COMMANDSACTIONSIF_H_
#define FSFW_ACTION_COMMANDSACTIONSIF_H_
-#include "../ipc/MessageQueueIF.h"
-#include "../returnvalues/HasReturnvaluesIF.h"
#include "CommandActionHelper.h"
+#include "fsfw/ipc/MessageQueueIF.h"
+#include "fsfw/returnvalues/HasReturnvaluesIF.h"
/**
* Interface to separate commanding actions of other objects.
@@ -21,7 +21,7 @@ class CommandsActionsIF {
static const uint8_t INTERFACE_ID = CLASS_ID::COMMANDS_ACTIONS_IF;
static const ReturnValue_t OBJECT_HAS_NO_FUNCTIONS = MAKE_RETURN_CODE(1);
static const ReturnValue_t ALREADY_COMMANDING = MAKE_RETURN_CODE(2);
- virtual ~CommandsActionsIF() {}
+ virtual ~CommandsActionsIF() = default;
virtual MessageQueueIF* getCommandQueuePtr() = 0;
protected:
diff --git a/src/fsfw/action/HasActionsIF.h b/src/fsfw/action/HasActionsIF.h
index acc502d71..89e58adde 100644
--- a/src/fsfw/action/HasActionsIF.h
+++ b/src/fsfw/action/HasActionsIF.h
@@ -1,11 +1,11 @@
#ifndef FSFW_ACTION_HASACTIONSIF_H_
#define FSFW_ACTION_HASACTIONSIF_H_
-#include "../ipc/MessageQueueIF.h"
-#include "../returnvalues/HasReturnvaluesIF.h"
#include "ActionHelper.h"
#include "ActionMessage.h"
#include "SimpleActionHelper.h"
+#include "fsfw/ipc/MessageQueueIF.h"
+#include "fsfw/returnvalues/HasReturnvaluesIF.h"
/**
* @brief
@@ -40,12 +40,12 @@ class HasActionsIF {
static const ReturnValue_t INVALID_PARAMETERS = MAKE_RETURN_CODE(2);
static const ReturnValue_t EXECUTION_FINISHED = MAKE_RETURN_CODE(3);
static const ReturnValue_t INVALID_ACTION_ID = MAKE_RETURN_CODE(4);
- virtual ~HasActionsIF() {}
+ virtual ~HasActionsIF() = default;
/**
* Function to get the MessageQueueId_t of the implementing object
* @return MessageQueueId_t of the object
*/
- virtual MessageQueueId_t getCommandQueue() const = 0;
+ [[nodiscard]] virtual MessageQueueId_t getCommandQueue() const = 0;
/**
* Execute or initialize the execution of a certain function.
* The ActionHelpers will execute this function and behave differently
diff --git a/src/fsfw/action/SimpleActionHelper.cpp b/src/fsfw/action/SimpleActionHelper.cpp
index 894f0df6f..fc7e064e8 100644
--- a/src/fsfw/action/SimpleActionHelper.cpp
+++ b/src/fsfw/action/SimpleActionHelper.cpp
@@ -3,7 +3,7 @@
SimpleActionHelper::SimpleActionHelper(HasActionsIF* setOwner, MessageQueueIF* useThisQueue)
: ActionHelper(setOwner, useThisQueue), isExecuting(false) {}
-SimpleActionHelper::~SimpleActionHelper() {}
+SimpleActionHelper::~SimpleActionHelper() = default;
void SimpleActionHelper::step(ReturnValue_t result) {
// STEP_OFFESET is subtracted to compensate for adding offset in base
@@ -38,7 +38,7 @@ void SimpleActionHelper::prepareExecution(MessageQueueId_t commandedBy, ActionId
ActionMessage::setStepReply(&reply, actionId, 0, HasActionsIF::IS_BUSY);
queueToUse->sendMessage(commandedBy, &reply);
}
- const uint8_t* dataPtr = NULL;
+ const uint8_t* dataPtr = nullptr;
size_t size = 0;
ReturnValue_t result = ipcStore->getData(dataAddress, &dataPtr, &size);
if (result != HasReturnvaluesIF::RETURN_OK) {
diff --git a/src/fsfw/action/SimpleActionHelper.h b/src/fsfw/action/SimpleActionHelper.h
index 5cb85fbd7..973c7cf2a 100644
--- a/src/fsfw/action/SimpleActionHelper.h
+++ b/src/fsfw/action/SimpleActionHelper.h
@@ -11,15 +11,15 @@
class SimpleActionHelper : public ActionHelper {
public:
SimpleActionHelper(HasActionsIF* setOwner, MessageQueueIF* useThisQueue);
- virtual ~SimpleActionHelper();
+ ~SimpleActionHelper() override;
void step(ReturnValue_t result = HasReturnvaluesIF::RETURN_OK);
void finish(ReturnValue_t result = HasReturnvaluesIF::RETURN_OK);
ReturnValue_t reportData(SerializeIF* data);
protected:
void prepareExecution(MessageQueueId_t commandedBy, ActionId_t actionId,
- store_address_t dataAddress);
- virtual void resetHelper();
+ store_address_t dataAddress) override;
+ void resetHelper() override;
private:
bool isExecuting;
@@ -28,4 +28,4 @@ class SimpleActionHelper : public ActionHelper {
uint8_t stepCount = 0;
};
-#endif /* SIMPLEACTIONHELPER_H_ */
+#endif /* FSFW_ACTION_SIMPLEACTIONHELPER_H_ */
diff --git a/src/fsfw/cfdp/CMakeLists.txt b/src/fsfw/cfdp/CMakeLists.txt
index 908dc32a8..0b926a9a2 100644
--- a/src/fsfw/cfdp/CMakeLists.txt
+++ b/src/fsfw/cfdp/CMakeLists.txt
@@ -1,7 +1,4 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- CFDPHandler.cpp
- CFDPMessage.cpp
-)
+target_sources(${LIB_FSFW_NAME} PRIVATE CFDPHandler.cpp CFDPMessage.cpp)
add_subdirectory(pdu)
add_subdirectory(tlv)
diff --git a/src/fsfw/cfdp/pdu/CMakeLists.txt b/src/fsfw/cfdp/pdu/CMakeLists.txt
index 931db3066..4f345bdc0 100644
--- a/src/fsfw/cfdp/pdu/CMakeLists.txt
+++ b/src/fsfw/cfdp/pdu/CMakeLists.txt
@@ -1,32 +1,30 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- PduConfig.cpp
- VarLenField.cpp
- HeaderSerializer.cpp
- HeaderDeserializer.cpp
- FileDirectiveDeserializer.cpp
- FileDirectiveSerializer.cpp
-
- AckInfo.cpp
- AckPduSerializer.cpp
- AckPduDeserializer.cpp
- EofInfo.cpp
- EofPduSerializer.cpp
- EofPduDeserializer.cpp
- NakInfo.cpp
- NakPduSerializer.cpp
- NakPduDeserializer.cpp
- FinishedInfo.cpp
- FinishedPduSerializer.cpp
- FinishedPduDeserializer.cpp
- MetadataInfo.cpp
- MetadataPduSerializer.cpp
- MetadataPduDeserializer.cpp
- KeepAlivePduSerializer.cpp
- KeepAlivePduDeserializer.cpp
- PromptPduSerializer.cpp
- PromptPduDeserializer.cpp
-
- FileDataSerializer.cpp
- FileDataDeserializer.cpp
- FileDataInfo.cpp
-)
\ No newline at end of file
+target_sources(
+ ${LIB_FSFW_NAME}
+ PRIVATE PduConfig.cpp
+ VarLenField.cpp
+ HeaderSerializer.cpp
+ HeaderDeserializer.cpp
+ FileDirectiveDeserializer.cpp
+ FileDirectiveSerializer.cpp
+ AckInfo.cpp
+ AckPduSerializer.cpp
+ AckPduDeserializer.cpp
+ EofInfo.cpp
+ EofPduSerializer.cpp
+ EofPduDeserializer.cpp
+ NakInfo.cpp
+ NakPduSerializer.cpp
+ NakPduDeserializer.cpp
+ FinishedInfo.cpp
+ FinishedPduSerializer.cpp
+ FinishedPduDeserializer.cpp
+ MetadataInfo.cpp
+ MetadataPduSerializer.cpp
+ MetadataPduDeserializer.cpp
+ KeepAlivePduSerializer.cpp
+ KeepAlivePduDeserializer.cpp
+ PromptPduSerializer.cpp
+ PromptPduDeserializer.cpp
+ FileDataSerializer.cpp
+ FileDataDeserializer.cpp
+ FileDataInfo.cpp)
diff --git a/src/fsfw/cfdp/pdu/HeaderSerializer.h b/src/fsfw/cfdp/pdu/HeaderSerializer.h
index 8f2fc3fdd..1de97d63d 100644
--- a/src/fsfw/cfdp/pdu/HeaderSerializer.h
+++ b/src/fsfw/cfdp/pdu/HeaderSerializer.h
@@ -44,7 +44,7 @@ class HeaderSerializer : public SerializeIF, public PduHeaderIF {
cfdp::WidthInBytes getLenEntityIds() const override;
cfdp::WidthInBytes getLenSeqNum() const override;
cfdp::SegmentMetadataFlag getSegmentMetadataFlag() const override;
- bool hasSegmentMetadataFlag() const;
+ bool hasSegmentMetadataFlag() const override;
void setSegmentationControl(cfdp::SegmentationControl);
void getSourceId(cfdp::EntityId& sourceId) const override;
diff --git a/src/fsfw/cfdp/tlv/CMakeLists.txt b/src/fsfw/cfdp/tlv/CMakeLists.txt
index 24459cf83..cdf7b44a0 100644
--- a/src/fsfw/cfdp/tlv/CMakeLists.txt
+++ b/src/fsfw/cfdp/tlv/CMakeLists.txt
@@ -1,10 +1,10 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- EntityIdTlv.cpp
- FilestoreRequestTlv.cpp
- FilestoreResponseTlv.cpp
- Lv.cpp
- Tlv.cpp
- FlowLabelTlv.cpp
- MessageToUserTlv.cpp
- FaultHandlerOverrideTlv.cpp
-)
\ No newline at end of file
+target_sources(
+ ${LIB_FSFW_NAME}
+ PRIVATE EntityIdTlv.cpp
+ FilestoreRequestTlv.cpp
+ FilestoreResponseTlv.cpp
+ Lv.cpp
+ Tlv.cpp
+ FlowLabelTlv.cpp
+ MessageToUserTlv.cpp
+ FaultHandlerOverrideTlv.cpp)
diff --git a/src/fsfw/cfdp/tlv/FilestoreTlvBase.h b/src/fsfw/cfdp/tlv/FilestoreTlvBase.h
index bb9f10bba..04012cda2 100644
--- a/src/fsfw/cfdp/tlv/FilestoreTlvBase.h
+++ b/src/fsfw/cfdp/tlv/FilestoreTlvBase.h
@@ -6,10 +6,13 @@
#include
#include
#include
+#include
#include
#include
+#include "fsfw/FSFW.h"
+
namespace cfdp {
enum FilestoreActionCode {
diff --git a/src/fsfw/container/CMakeLists.txt b/src/fsfw/container/CMakeLists.txt
index 13eced1d0..52087ff00 100644
--- a/src/fsfw/container/CMakeLists.txt
+++ b/src/fsfw/container/CMakeLists.txt
@@ -1,5 +1,2 @@
-target_sources(${LIB_FSFW_NAME}
- PRIVATE
- SharedRingBuffer.cpp
- SimpleRingBuffer.cpp
-)
\ No newline at end of file
+target_sources(${LIB_FSFW_NAME} PRIVATE SharedRingBuffer.cpp
+ SimpleRingBuffer.cpp)
diff --git a/src/fsfw/container/FIFOBase.tpp b/src/fsfw/container/FIFOBase.tpp
index 2e6a3829c..91804b6c8 100644
--- a/src/fsfw/container/FIFOBase.tpp
+++ b/src/fsfw/container/FIFOBase.tpp
@@ -5,89 +5,88 @@
#error Include FIFOBase.h before FIFOBase.tpp!
#endif
-template
-inline FIFOBase::FIFOBase(T* values, const size_t maxCapacity):
- maxCapacity(maxCapacity), values(values){};
+template
+inline FIFOBase::FIFOBase(T* values, const size_t maxCapacity)
+ : maxCapacity(maxCapacity), values(values){};
-template
+template
inline ReturnValue_t FIFOBase::insert(T value) {
- if (full()) {
- return FULL;
- } else {
- values[writeIndex] = value;
- writeIndex = next(writeIndex);
- ++currentSize;
- return HasReturnvaluesIF::RETURN_OK;
- }
+ if (full()) {
+ return FULL;
+ } else {
+ values[writeIndex] = value;
+ writeIndex = next(writeIndex);
+ ++currentSize;
+ return HasReturnvaluesIF::RETURN_OK;
+ }
};
-template
+template
inline ReturnValue_t FIFOBase::retrieve(T* value) {
- if (empty()) {
- return EMPTY;
- } else {
- if (value == nullptr){
- return HasReturnvaluesIF::RETURN_FAILED;
- }
- *value = values[readIndex];
- readIndex = next(readIndex);
- --currentSize;
- return HasReturnvaluesIF::RETURN_OK;
+ if (empty()) {
+ return EMPTY;
+ } else {
+ if (value == nullptr) {
+ return HasReturnvaluesIF::RETURN_FAILED;
}
+ *value = values[readIndex];
+ readIndex = next(readIndex);
+ --currentSize;
+ return HasReturnvaluesIF::RETURN_OK;
+ }
};
-template
+template
inline ReturnValue_t FIFOBase::peek(T* value) {
- if(empty()) {
- return EMPTY;
- } else {
- if (value == nullptr){
- return HasReturnvaluesIF::RETURN_FAILED;
- }
- *value = values[readIndex];
- return HasReturnvaluesIF::RETURN_OK;
+ if (empty()) {
+ return EMPTY;
+ } else {
+ if (value == nullptr) {
+ return HasReturnvaluesIF::RETURN_FAILED;
}
+ *value = values[readIndex];
+ return HasReturnvaluesIF::RETURN_OK;
+ }
};
-template
+template
inline ReturnValue_t FIFOBase::pop() {
- T value;
- return this->retrieve(&value);
+ T value;
+ return this->retrieve(&value);
};
-template
+template
inline bool FIFOBase::empty() {
- return (currentSize == 0);
+ return (currentSize == 0);
};
-template
+template
inline bool FIFOBase::full() {
- return (currentSize == maxCapacity);
+ return (currentSize == maxCapacity);
}
-template
+template
inline size_t FIFOBase::size() {
- return currentSize;
+ return currentSize;
}
-template
+template
inline size_t FIFOBase::next(size_t current) {
- ++current;
- if (current == maxCapacity) {
- current = 0;
- }
- return current;
+ ++current;
+ if (current == maxCapacity) {
+ current = 0;
+ }
+ return current;
}
-template
+template
inline size_t FIFOBase::getMaxCapacity() const {
- return maxCapacity;
+ return maxCapacity;
}
-
-template
-inline void FIFOBase::setContainer(T *data) {
- this->values = data;
+template
+inline void FIFOBase::setContainer(T* data) {
+ this->values = data;
}
#endif
diff --git a/src/fsfw/container/FixedArrayList.h b/src/fsfw/container/FixedArrayList.h
index c09a421eb..fc8be393a 100644
--- a/src/fsfw/container/FixedArrayList.h
+++ b/src/fsfw/container/FixedArrayList.h
@@ -2,6 +2,7 @@
#define FIXEDARRAYLIST_H_
#include
+#include
#include "ArrayList.h"
/**
@@ -9,10 +10,9 @@
*/
template
class FixedArrayList : public ArrayList {
-#if !defined(_MSC_VER)
- static_assert(MAX_SIZE <= (std::pow(2, sizeof(count_t) * 8) - 1),
+ static_assert(MAX_SIZE <= std::numeric_limits::max(),
"count_t is not large enough to hold MAX_SIZE");
-#endif
+
private:
T data[MAX_SIZE];
diff --git a/src/fsfw/container/FixedOrderedMultimap.tpp b/src/fsfw/container/FixedOrderedMultimap.tpp
index 294a161fe..fd58bc44a 100644
--- a/src/fsfw/container/FixedOrderedMultimap.tpp
+++ b/src/fsfw/container/FixedOrderedMultimap.tpp
@@ -1,109 +1,109 @@
#ifndef FRAMEWORK_CONTAINER_FIXEDORDEREDMULTIMAP_TPP_
#define FRAMEWORK_CONTAINER_FIXEDORDEREDMULTIMAP_TPP_
-
-template
-inline ReturnValue_t FixedOrderedMultimap::insert(key_t key, T value, Iterator *storedValue) {
- if (_size == theMap.maxSize()) {
- return MAP_FULL;
- }
- size_t position = findNicePlace(key);
- memmove(static_cast(&theMap[position + 1]),static_cast(&theMap[position]),
- (_size - position) * sizeof(std::pair));
- theMap[position].first = key;
- theMap[position].second = value;
- ++_size;
- if (storedValue != nullptr) {
- *storedValue = Iterator(&theMap[position]);
- }
- return HasReturnvaluesIF::RETURN_OK;
+template
+inline ReturnValue_t FixedOrderedMultimap::insert(key_t key, T value,
+ Iterator *storedValue) {
+ if (_size == theMap.maxSize()) {
+ return MAP_FULL;
+ }
+ size_t position = findNicePlace(key);
+ memmove(static_cast(&theMap[position + 1]), static_cast(&theMap[position]),
+ (_size - position) * sizeof(std::pair));
+ theMap[position].first = key;
+ theMap[position].second = value;
+ ++_size;
+ if (storedValue != nullptr) {
+ *storedValue = Iterator(&theMap[position]);
+ }
+ return HasReturnvaluesIF::RETURN_OK;
}
-template
+template
inline ReturnValue_t FixedOrderedMultimap::insert(std::pair pair) {
- return insert(pair.first, pair.second);
+ return insert(pair.first, pair.second);
}
-template
+template
inline ReturnValue_t FixedOrderedMultimap::exists(key_t key) const {
- ReturnValue_t result = KEY_DOES_NOT_EXIST;
- if (findFirstIndex(key) < _size) {
- result = HasReturnvaluesIF::RETURN_OK;
- }
- return result;
+ ReturnValue_t result = KEY_DOES_NOT_EXIST;
+ if (findFirstIndex(key) < _size) {
+ result = HasReturnvaluesIF::RETURN_OK;
+ }
+ return result;
}
-template
+template
inline ReturnValue_t FixedOrderedMultimap::erase(Iterator *iter) {
- size_t i;
- if ((i = findFirstIndex((*iter).value->first)) >= _size) {
- return KEY_DOES_NOT_EXIST;
- }
- removeFromPosition(i);
- if (*iter != begin()) {
- (*iter)--;
- } else {
- *iter = begin();
- }
- return HasReturnvaluesIF::RETURN_OK;
+ size_t i;
+ if ((i = findFirstIndex((*iter).value->first)) >= _size) {
+ return KEY_DOES_NOT_EXIST;
+ }
+ removeFromPosition(i);
+ if (*iter != begin()) {
+ (*iter)--;
+ } else {
+ *iter = begin();
+ }
+ return HasReturnvaluesIF::RETURN_OK;
}
-template
+template
inline ReturnValue_t FixedOrderedMultimap::erase(key_t key) {
- size_t i;
- if ((i = findFirstIndex(key)) >= _size) {
- return KEY_DOES_NOT_EXIST;
- }
- do {
- removeFromPosition(i);
- i = findFirstIndex(key, i);
- } while (i < _size);
- return HasReturnvaluesIF::RETURN_OK;
+ size_t i;
+ if ((i = findFirstIndex(key)) >= _size) {
+ return KEY_DOES_NOT_EXIST;
+ }
+ do {
+ removeFromPosition(i);
+ i = findFirstIndex(key, i);
+ } while (i < _size);
+ return HasReturnvaluesIF::RETURN_OK;
}
-template
+template
inline ReturnValue_t FixedOrderedMultimap::find(key_t key, T **value) const {
- ReturnValue_t result = exists(key);
- if (result != HasReturnvaluesIF::RETURN_OK) {
- return result;
- }
- *value = &theMap[findFirstIndex(key)].second;
- return HasReturnvaluesIF::RETURN_OK;
+ ReturnValue_t result = exists(key);
+ if (result != HasReturnvaluesIF::RETURN_OK) {
+ return result;
+ }
+ *value = &theMap[findFirstIndex(key)].second;
+ return HasReturnvaluesIF::RETURN_OK;
}
-template
-inline size_t FixedOrderedMultimap::findFirstIndex(key_t key, size_t startAt) const {
- if (startAt >= _size) {
- return startAt + 1;
+template
+inline size_t FixedOrderedMultimap::findFirstIndex(key_t key,
+ size_t startAt) const {
+ if (startAt >= _size) {
+ return startAt + 1;
+ }
+ size_t i = startAt;
+ for (i = startAt; i < _size; ++i) {
+ if (theMap[i].first == key) {
+ return i;
}
- size_t i = startAt;
- for (i = startAt; i < _size; ++i) {
- if (theMap[i].first == key) {
- return i;
- }
- }
- return i;
+ }
+ return i;
}
-template
+template
inline size_t FixedOrderedMultimap::findNicePlace(key_t key) const {
- size_t i = 0;
- for (i = 0; i < _size; ++i) {
- if (myComp(key, theMap[i].first)) {
- return i;
- }
+ size_t i = 0;
+ for (i = 0; i < _size; ++i) {
+ if (myComp(key, theMap[i].first)) {
+ return i;
}
- return i;
+ }
+ return i;
}
-template
+template
inline void FixedOrderedMultimap::removeFromPosition(size_t position) {
- if (_size <= position) {
- return;
- }
- memmove(static_cast(&theMap[position]), static_cast(&theMap[position + 1]),
- (_size - position - 1) * sizeof(std::pair));
- --_size;
+ if (_size <= position) {
+ return;
+ }
+ memmove(static_cast(&theMap[position]), static_cast(&theMap[position + 1]),
+ (_size - position - 1) * sizeof(std::pair));
+ --_size;
}
-
#endif /* FRAMEWORK_CONTAINER_FIXEDORDEREDMULTIMAP_TPP_ */
diff --git a/src/fsfw/container/HybridIterator.h b/src/fsfw/container/HybridIterator.h
index e8b24a3d6..ad000ec25 100644
--- a/src/fsfw/container/HybridIterator.h
+++ b/src/fsfw/container/HybridIterator.h
@@ -10,16 +10,23 @@ class HybridIterator : public LinkedElement::Iterator, public ArrayList::Iterator *iter)
- : LinkedElement::Iterator(*iter), value(iter->value), linked(true) {}
+ : LinkedElement::Iterator(*iter), linked(true) {
+ if (iter != nullptr) {
+ value = iter->value;
+ }
+ }
- HybridIterator(LinkedElement *start)
- : LinkedElement::Iterator(start), value(start->value), linked(true) {}
+ HybridIterator(LinkedElement *start) : LinkedElement::Iterator(start), linked(true) {
+ if (start != nullptr) {
+ value = start->value;
+ }
+ }
HybridIterator(typename ArrayList::Iterator start,
typename ArrayList::Iterator end)
: ArrayList::Iterator(start), value(start.value), linked(false), end(end.value) {
if (value == this->end) {
- value = NULL;
+ value = nullptr;
}
}
diff --git a/src/fsfw/container/SimpleRingBuffer.cpp b/src/fsfw/container/SimpleRingBuffer.cpp
index bcf3cf20d..c104ea971 100644
--- a/src/fsfw/container/SimpleRingBuffer.cpp
+++ b/src/fsfw/container/SimpleRingBuffer.cpp
@@ -2,6 +2,9 @@
#include
+#include "fsfw/FSFW.h"
+#include "fsfw/serviceinterface.h"
+
SimpleRingBuffer::SimpleRingBuffer(const size_t size, bool overwriteOld, size_t maxExcessBytes)
: RingBufferBase<>(0, size, overwriteOld), maxExcessBytes(maxExcessBytes) {
if (maxExcessBytes > size) {
@@ -48,6 +51,19 @@ void SimpleRingBuffer::confirmBytesWritten(size_t amount) {
}
ReturnValue_t SimpleRingBuffer::writeData(const uint8_t* data, size_t amount) {
+ if (data == nullptr) {
+ return HasReturnvaluesIF::RETURN_FAILED;
+ }
+ if (amount > getMaxSize()) {
+#if FSFW_VERBOSE_LEVEL >= 1
+#if FSFW_CPP_OSTREAM_ENABLED == 1
+ sif::error << "SimpleRingBuffer::writeData: Amount of data too large" << std::endl;
+#else
+ sif::printError("SimpleRingBuffer::writeData: Amount of data too large\n");
+#endif
+#endif
+ return HasReturnvaluesIF::RETURN_FAILED;
+ }
if (availableWriteSpace() >= amount or overwriteOld) {
size_t amountTillWrap = writeTillWrap();
if (amountTillWrap >= amount) {
diff --git a/src/fsfw/controller/CMakeLists.txt b/src/fsfw/controller/CMakeLists.txt
index 550acfcdd..c8c000d81 100644
--- a/src/fsfw/controller/CMakeLists.txt
+++ b/src/fsfw/controller/CMakeLists.txt
@@ -1,4 +1,2 @@
-target_sources(${LIB_FSFW_NAME} PRIVATE
- ControllerBase.cpp
- ExtendedControllerBase.cpp
-)
\ No newline at end of file
+target_sources(${LIB_FSFW_NAME} PRIVATE ControllerBase.cpp
+ ExtendedControllerBase.cpp)
diff --git a/src/fsfw/controller/ControllerBase.cpp b/src/fsfw/controller/ControllerBase.cpp
index 953dacb46..7a8b6bc41 100644
--- a/src/fsfw/controller/ControllerBase.cpp
+++ b/src/fsfw/controller/ControllerBase.cpp
@@ -26,7 +26,7 @@ ReturnValue_t ControllerBase::initialize() {
MessageQueueId_t parentQueue = 0;
if (parentId != objects::NO_OBJECT) {
- SubsystemBase* parent = ObjectManager::instance()->get(parentId);
+ auto* parent = ObjectManager::instance()->get(parentId);
if (parent == nullptr) {
return RETURN_FAILED;
}
@@ -52,7 +52,7 @@ MessageQueueId_t ControllerBase::getCommandQueue() const { return commandQueue->
void ControllerBase::handleQueue() {
CommandMessage command;
- ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
+ ReturnValue_t result;
for (result = commandQueue->receiveMessage(&command); result == RETURN_OK;
result = commandQueue->receiveMessage(&command)) {
result = modeHelper.handleModeCommand(&command);
@@ -73,20 +73,20 @@ void ControllerBase::handleQueue() {
}
}
-void ControllerBase::startTransition(Mode_t mode, Submode_t submode) {
+void ControllerBase::startTransition(Mode_t mode_, Submode_t submode_) {
changeHK(this->mode, this->submode, false);
triggerEvent(CHANGING_MODE, mode, submode);
- this->mode = mode;
- this->submode = submode;
+ mode = mode_;
+ submode = submode_;
modeHelper.modeChanged(mode, submode);
modeChanged(mode, submode);
announceMode(false);
changeHK(this->mode, this->submode, true);
}
-void ControllerBase::getMode(Mode_t* mode, Submode_t* submode) {
- *mode = this->mode;
- *submode = this->submode;
+void ControllerBase::getMode(Mode_t* mode_, Submode_t* submode_) {
+ *mode_ = this->mode;
+ *submode_ = this->submode;
}
void ControllerBase::setToExternalControl() { healthHelper.setHealth(EXTERNAL_CONTROL); }
@@ -99,7 +99,7 @@ ReturnValue_t ControllerBase::performOperation(uint8_t opCode) {
return RETURN_OK;
}
-void ControllerBase::modeChanged(Mode_t mode, Submode_t submode) { return; }
+void ControllerBase::modeChanged(Mode_t mode_, Submode_t submode_) {}
ReturnValue_t ControllerBase::setHealth(HealthState health) {
switch (health) {
@@ -115,6 +115,6 @@ ReturnValue_t ControllerBase::setHealth(HealthState health) {
HasHealthIF::HealthState ControllerBase::getHealth() { return healthHelper.getHealth(); }
void ControllerBase::setTaskIF(PeriodicTaskIF* task_) { executingTask = task_; }
-void ControllerBase::changeHK(Mode_t mode, Submode_t submode, bool enable) {}
+void ControllerBase::changeHK(Mode_t mode_, Submode_t submode_, bool enable) {}
ReturnValue_t ControllerBase::initializeAfterTaskCreation() { return HasReturnvaluesIF::RETURN_OK; }
diff --git a/src/fsfw/controller/ControllerBase.h b/src/fsfw/controller/ControllerBase.h
index db75982c8..550659b82 100644
--- a/src/fsfw/controller/ControllerBase.h
+++ b/src/fsfw/controller/ControllerBase.h
@@ -1,7 +1,6 @@
#ifndef FSFW_CONTROLLER_CONTROLLERBASE_H_
#define FSFW_CONTROLLER_CONTROLLERBASE_H_
-#include "fsfw/datapool/HkSwitchHelper.h"
#include "fsfw/health/HasHealthIF.h"
#include "fsfw/health/HealthHelper.h"
#include "fsfw/modes/HasModesIF.h"
@@ -25,21 +24,21 @@ class ControllerBase : public HasModesIF,
static const Mode_t MODE_NORMAL = 2;
ControllerBase(object_id_t setObjectId, object_id_t parentId, size_t commandQueueDepth = 3);
- virtual ~ControllerBase();
+ ~ControllerBase() override;
/** SystemObject override */
- virtual ReturnValue_t initialize() override;
+ ReturnValue_t initialize() override;
- virtual MessageQueueId_t getCommandQueue() const override;
+ [[nodiscard]] MessageQueueId_t getCommandQueue() const override;
/** HasHealthIF overrides */
- virtual ReturnValue_t setHealth(HealthState health) override;
- virtual HasHealthIF::HealthState getHealth() override;
+ ReturnValue_t setHealth(HealthState health) override;
+ HasHealthIF::HealthState getHealth() override;
/** ExecutableObjectIF overrides */
- virtual ReturnValue_t performOperation(uint8_t opCode) override;
- virtual void setTaskIF(PeriodicTaskIF *task) override;
- virtual ReturnValue_t initializeAfterTaskCreation() override;
+ ReturnValue_t performOperation(uint8_t opCode) override;
+ void setTaskIF(PeriodicTaskIF *task) override;
+ ReturnValue_t initializeAfterTaskCreation() override;
protected:
/**
@@ -55,8 +54,8 @@ class ControllerBase : public HasModesIF,
*/
virtual void performControlOperation() = 0;
- virtual ReturnValue_t checkModeCommand(Mode_t mode, Submode_t submode,
- uint32_t *msToReachTheMode) = 0;
+ ReturnValue_t checkModeCommand(Mode_t mode, Submode_t submode,
+ uint32_t *msToReachTheMode) override = 0;
const object_id_t parentId;
@@ -81,10 +80,10 @@ class ControllerBase : public HasModesIF,
/** Mode helpers */
virtual void modeChanged(Mode_t mode, Submode_t submode);
- virtual void startTransition(Mode_t mode, Submode_t submode);
- virtual void getMode(Mode_t *mode, Submode_t *submode);
- virtual void setToExternalControl();
- virtual void announceMode(bool recursive);
+ void startTransition(Mode_t mode, Submode_t submode) override;
+ void getMode(Mode_t *mode, Submode_t *submode) override;
+ void setToExternalControl() override;
+ void announceMode(bool recursive) override;
/** HK helpers */
virtual void changeHK(Mode_t mode, Submode_t submode, bool enable);
};
diff --git a/src/fsfw/controller/ExtendedControllerBase.cpp b/src/fsfw/controller/ExtendedControllerBase.cpp
index 0dfd9dc7c..64b39a315 100644
--- a/src/fsfw/controller/ExtendedControllerBase.cpp
+++ b/src/fsfw/controller/ExtendedControllerBase.cpp
@@ -6,7 +6,7 @@ ExtendedControllerBase::ExtendedControllerBase(object_id_t objectId, object_id_t
poolManager(this, commandQueue),
actionHelper(this, commandQueue) {}
-ExtendedControllerBase::~ExtendedControllerBase() {}
+ExtendedControllerBase::~ExtendedControllerBase() = default;
ReturnValue_t ExtendedControllerBase::executeAction(ActionId_t actionId,
MessageQueueId_t commandedBy,
@@ -31,7 +31,7 @@ ReturnValue_t ExtendedControllerBase::handleCommandMessage(CommandMessage *messa
void ExtendedControllerBase::handleQueue() {
CommandMessage command;
- ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
+ ReturnValue_t result;
for (result = commandQueue->receiveMessage(&command); result == RETURN_OK;
result = commandQueue->receiveMessage(&command)) {
result = actionHelper.handleActionMessage(&command);
diff --git a/src/fsfw/controller/ExtendedControllerBase.h b/src/fsfw/controller/ExtendedControllerBase.h
index 0c64f5b9a..b5583a880 100644
--- a/src/fsfw/controller/ExtendedControllerBase.h
+++ b/src/fsfw/controller/ExtendedControllerBase.h
@@ -18,16 +18,16 @@ class ExtendedControllerBase : public ControllerBase,
public HasLocalDataPoolIF {
public:
ExtendedControllerBase(object_id_t objectId, object_id_t parentId, size_t commandQueueDepth = 3);
- virtual ~ExtendedControllerBase();
+ ~ExtendedControllerBase() override;
/* SystemObjectIF overrides */
- virtual ReturnValue_t initialize() override;
+ ReturnValue_t initialize() override;
- virtual MessageQueueId_t getCommandQueue() const override;
+ [[nodiscard]] MessageQueueId_t getCommandQueue() const override;
/* ExecutableObjectIF overrides */
- virtual ReturnValue_t performOperation(uint8_t opCode) override;
- virtual ReturnValue_t initializeAfterTaskCreation() override;
+ ReturnValue_t performOperation(uint8_t opCode) override;
+ ReturnValue_t initializeAfterTaskCreation() override;
protected:
LocalDataPoolManager poolManager;
@@ -39,32 +39,32 @@ class ExtendedControllerBase : public ControllerBase,
* @param message
* @return
*/
- virtual ReturnValue_t handleCommandMessage(CommandMessage* message) = 0;
+ ReturnValue_t handleCommandMessage(CommandMessage* message) override = 0;
/**
* Periodic helper from ControllerBase, implemented by child class.
*/
- virtual void performControlOperation() = 0;
+ void performControlOperation() override = 0;
/* Handle the four messages mentioned above */
void handleQueue() override;
/* HasActionsIF overrides */
- virtual ReturnValue_t executeAction(ActionId_t actionId, MessageQueueId_t commandedBy,
- const uint8_t* data, size_t size) override;
+ ReturnValue_t executeAction(ActionId_t actionId, MessageQueueId_t commandedBy,
+ const uint8_t* data, size_t size) override;
/* HasLocalDatapoolIF overrides */
- virtual LocalDataPoolManager* getHkManagerHandle() override;
- virtual object_id_t getObjectId() const override;
- virtual uint32_t getPeriodicOperationFrequency() const override;
+ LocalDataPoolManager* getHkManagerHandle() override;
+ [[nodiscard]] object_id_t getObjectId() const override;
+ [[nodiscard]] uint32_t getPeriodicOperationFrequency() const override;
- virtual ReturnValue_t initializeLocalDataPool(localpool::DataPool& localDataPoolMap,
- LocalDataPoolManager& poolManager) override = 0;
- virtual LocalPoolDataSetBase* getDataSetHandle(sid_t sid) override = 0;
+ ReturnValue_t initializeLocalDataPool(localpool::DataPool& localDataPoolMap,
+ LocalDataPoolManager& poolManager) override = 0;
+ LocalPoolDataSetBase* getDataSetHandle(sid_t sid) override = 0;
// Mode abstract functions
- virtual ReturnValue_t checkModeCommand(Mode_t mode, Submode_t submode,
- uint32_t* msToReachTheMode) override = 0;
+ ReturnValue_t checkModeCommand(Mode_t mode, Submode_t submode,
+ uint32_t* msToReachTheMode) override = 0;
};
#endif /* FSFW_CONTROLLER_EXTENDEDCONTROLLERBASE_H_ */
diff --git a/src/fsfw/coordinates/CMakeLists.txt b/src/fsfw/coordinates/CMakeLists.txt
index a1fa1e52b..15452b1ca 100644
--- a/src/fsfw/coordinates/CMakeLists.txt
+++ b/src/fsfw/coordinates/CMakeLists.txt
@@ -1,5 +1,2 @@
-target_sources(${LIB_FSFW_NAME}
- PRIVATE
- CoordinateTransformations.cpp
- Sgp4Propagator.cpp
-)
\ No newline at end of file
+target_sources(${LIB_FSFW_NAME} PRIVATE CoordinateTransformations.cpp
+ Sgp4Propagator.cpp)
diff --git a/src/fsfw/datalinklayer/CMakeLists.txt b/src/fsfw/datalinklayer/CMakeLists.txt
index 148e7c5de..cc18088f8 100644
--- a/src/fsfw/datalinklayer/CMakeLists.txt
+++ b/src/fsfw/datalinklayer/CMakeLists.txt
@@ -1,12 +1,11 @@
-target_sources(${LIB_FSFW_NAME}
- PRIVATE
- Clcw.cpp
- DataLinkLayer.cpp
- Farm1StateLockout.cpp
- Farm1StateOpen.cpp
- Farm1StateWait.cpp
- MapPacketExtraction.cpp
- TcTransferFrame.cpp
- TcTransferFrameLocal.cpp
- VirtualChannelReception.cpp
-)
\ No newline at end of file
+target_sources(
+ ${LIB_FSFW_NAME}
+ PRIVATE Clcw.cpp
+ DataLinkLayer.cpp
+ Farm1StateLockout.cpp
+ Farm1StateOpen.cpp
+ Farm1StateWait.cpp
+ MapPacketExtraction.cpp
+ TcTransferFrame.cpp
+ TcTransferFrameLocal.cpp
+ VirtualChannelReception.cpp)
diff --git a/src/fsfw/datapool/CMakeLists.txt b/src/fsfw/datapool/CMakeLists.txt
index 0d53e1ba1..b2ac592c4 100644
--- a/src/fsfw/datapool/CMakeLists.txt
+++ b/src/fsfw/datapool/CMakeLists.txt
@@ -1,6 +1 @@
-target_sources(${LIB_FSFW_NAME}
- PRIVATE
- HkSwitchHelper.cpp
- PoolDataSetBase.cpp
- PoolEntry.cpp
-)
\ No newline at end of file
+target_sources(${LIB_FSFW_NAME} PRIVATE PoolDataSetBase.cpp PoolEntry.cpp)
diff --git a/src/fsfw/datapool/HkSwitchHelper.cpp b/src/fsfw/datapool/HkSwitchHelper.cpp
deleted file mode 100644
index 7f6ffd17f..000000000
--- a/src/fsfw/datapool/HkSwitchHelper.cpp
+++ /dev/null
@@ -1,67 +0,0 @@
-#include "fsfw/datapool/HkSwitchHelper.h"
-
-#include "fsfw/ipc/QueueFactory.h"
-
-HkSwitchHelper::HkSwitchHelper(EventReportingProxyIF* eventProxy)
- : commandActionHelper(this), eventProxy(eventProxy) {
- actionQueue = QueueFactory::instance()->createMessageQueue();
-}
-
-HkSwitchHelper::~HkSwitchHelper() { QueueFactory::instance()->deleteMessageQueue(actionQueue); }
-
-ReturnValue_t HkSwitchHelper::initialize() {
- ReturnValue_t result = commandActionHelper.initialize();
-
- if (result != HasReturnvaluesIF::RETURN_OK) {
- return result;
- }
-
- return result;
-}
-
-ReturnValue_t HkSwitchHelper::performOperation(uint8_t operationCode) {
- CommandMessage command;
- while (actionQueue->receiveMessage(&command) == HasReturnvaluesIF::RETURN_OK) {
- ReturnValue_t result = commandActionHelper.handleReply(&command);
- if (result == HasReturnvaluesIF::RETURN_OK) {
- continue;
- }
- command.setToUnknownCommand();
- actionQueue->reply(&command);
- }
-
- return HasReturnvaluesIF::RETURN_OK;
-}
-
-void HkSwitchHelper::stepSuccessfulReceived(ActionId_t actionId, uint8_t step) {}
-
-void HkSwitchHelper::stepFailedReceived(ActionId_t actionId, uint8_t step,
- ReturnValue_t returnCode) {
- eventProxy->forwardEvent(SWITCHING_TM_FAILED, returnCode, actionId);
-}
-
-void HkSwitchHelper::dataReceived(ActionId_t actionId, const uint8_t* data, uint32_t size) {}
-
-void HkSwitchHelper::completionSuccessfulReceived(ActionId_t actionId) {}
-
-void HkSwitchHelper::completionFailedReceived(ActionId_t actionId, ReturnValue_t returnCode) {
- eventProxy->forwardEvent(SWITCHING_TM_FAILED, returnCode, actionId);
-}
-
-ReturnValue_t HkSwitchHelper::switchHK(SerializeIF* sids, bool enable) {
- // ActionId_t action = HKService::DISABLE_HK;
- // if (enable) {
- // action = HKService::ENABLE_HK;
- // }
- //
- // ReturnValue_t result = commandActionHelper.commandAction(
- // objects::PUS_HK_SERVICE, action, sids);
- //
- // if (result != HasReturnvaluesIF::RETURN_OK) {
- // eventProxy->forwardEvent(SWITCHING_TM_FAILED, result);
- // }
- // return result;
- return HasReturnvaluesIF::RETURN_OK;
-}
-
-MessageQueueIF* HkSwitchHelper::getCommandQueuePtr() { return actionQueue; }
diff --git a/src/fsfw/datapool/HkSwitchHelper.h b/src/fsfw/datapool/HkSwitchHelper.h
deleted file mode 100644
index a0becd81b..000000000
--- a/src/fsfw/datapool/HkSwitchHelper.h
+++ /dev/null
@@ -1,44 +0,0 @@
-#ifndef FRAMEWORK_DATAPOOL_HKSWITCHHELPER_H_
-#define FRAMEWORK_DATAPOOL_HKSWITCHHELPER_H_
-
-#include "fsfw/action/CommandsActionsIF.h"
-#include "fsfw/events/EventReportingProxyIF.h"
-#include "fsfw/tasks/ExecutableObjectIF.h"
-
-// TODO this class violations separation between mission and framework
-// but it is only a transitional solution until the Datapool is
-// implemented decentrally
-
-class HkSwitchHelper : public ExecutableObjectIF, public CommandsActionsIF {
- public:
- static const uint8_t SUBSYSTEM_ID = SUBSYSTEM_ID::HK;
- static const Event SWITCHING_TM_FAILED =
- MAKE_EVENT(1, severity::LOW); //!< Commanding the HK Service failed, p1: error code, p2
- //!< action: 0 disable / 1 enable
-
- HkSwitchHelper(EventReportingProxyIF* eventProxy);
- virtual ~HkSwitchHelper();
-
- ReturnValue_t initialize();
-
- virtual ReturnValue_t performOperation(uint8_t operationCode = 0);
-
- ReturnValue_t switchHK(SerializeIF* sids, bool enable);
-
- virtual void setTaskIF(PeriodicTaskIF* task_){};
-
- protected:
- virtual void stepSuccessfulReceived(ActionId_t actionId, uint8_t step);
- virtual void stepFailedReceived(ActionId_t actionId, uint8_t step, ReturnValue_t returnCode);
- virtual void dataReceived(ActionId_t actionId, const uint8_t* data, uint32_t size);
- virtual void completionSuccessfulReceived(ActionId_t actionId);
- virtual void completionFailedReceived(ActionId_t actionId, ReturnValue_t returnCode);
- virtual MessageQueueIF* getCommandQueuePtr();
-
- private:
- CommandActionHelper commandActionHelper;
- MessageQueueIF* actionQueue;
- EventReportingProxyIF* eventProxy;
-};
-
-#endif /* FRAMEWORK_DATAPOOL_HKSWITCHHELPER_H_ */
diff --git a/src/fsfw/datapool/PoolDataSetBase.h b/src/fsfw/datapool/PoolDataSetBase.h
index 1b4bacdad..dc6ec135a 100644
--- a/src/fsfw/datapool/PoolDataSetBase.h
+++ b/src/fsfw/datapool/PoolDataSetBase.h
@@ -109,7 +109,7 @@ class PoolDataSetBase : public PoolDataSetIF, public SerializeIF, public HasRetu
*/
virtual ReturnValue_t unlockDataPool() override;
- virtual uint16_t getFillCount() const;
+ virtual uint16_t getFillCount() const override;
/* SerializeIF implementations */
virtual ReturnValue_t serialize(uint8_t** buffer, size_t* size, const size_t maxSize,
diff --git a/src/fsfw/datapool/PoolEntry.cpp b/src/fsfw/datapool/PoolEntry.cpp
index fd110e6c7..9138a7059 100644
--- a/src/fsfw/datapool/PoolEntry.cpp
+++ b/src/fsfw/datapool/PoolEntry.cpp
@@ -7,24 +7,26 @@
#include "fsfw/serviceinterface/ServiceInterface.h"
template
-PoolEntry::PoolEntry(std::initializer_list initValue, bool setValid)
- : length(static_cast(initValue.size())), valid(setValid) {
- this->address = new T[this->length];
- if (initValue.size() == 0) {
- std::memset(this->address, 0, this->getByteSize());
- } else {
- std::copy(initValue.begin(), initValue.end(), this->address);
+PoolEntry::PoolEntry(uint8_t len, bool setValid) : length(len), valid(setValid) {
+ this->address = new T[this->length]();
+ std::memset(this->address, 0, this->getByteSize());
+}
+
+template
+PoolEntry::PoolEntry(std::initializer_list initValues, bool setValid)
+ : length(static_cast(initValues.size())), valid(setValid) {
+ this->address = new T[this->length]();
+ if (initValues.size() > 0) {
+ std::copy(initValues.begin(), initValues.end(), this->address);
}
}
template
-PoolEntry::PoolEntry(T* initValue, uint8_t setLength, bool setValid)
+PoolEntry::PoolEntry(const T* initValue, uint8_t setLength, bool setValid)
: length(setLength), valid(setValid) {
- this->address = new T[this->length];
+ this->address = new T[this->length]();
if (initValue != nullptr) {
std::memcpy(this->address, initValue, this->getByteSize());
- } else {
- std::memset(this->address, 0, this->getByteSize());
}
}
diff --git a/src/fsfw/datapool/PoolEntry.h b/src/fsfw/datapool/PoolEntry.h
index d3d80f093..4010f78d2 100644
--- a/src/fsfw/datapool/PoolEntry.h
+++ b/src/fsfw/datapool/PoolEntry.h
@@ -33,6 +33,9 @@ class PoolEntry : public PoolEntryIF {
"instead! The ECSS standard defines a boolean as a one bit "
"field. Therefore it is preferred to store a boolean as an "
"uint8_t");
+
+ PoolEntry(uint8_t len = 1, bool setValid = false);
+
/**
* @brief In the classe's constructor, space is allocated on the heap and
* potential initialization values are copied to that space.
@@ -49,7 +52,7 @@ class PoolEntry : public PoolEntryIF {
* @param setValid
* Sets the initialization flag. It is invalid by default.
*/
- PoolEntry(std::initializer_list initValue = {0}, bool setValid = false);
+ PoolEntry(std::initializer_list initValue, bool setValid = false);
/**
* @brief In the classe's constructor, space is allocated on the heap and
@@ -62,7 +65,7 @@ class PoolEntry : public PoolEntryIF {
* @param setValid
* Sets the initialization flag. It is invalid by default.
*/
- PoolEntry(T* initValue, uint8_t setLength = 1, bool setValid = false);
+ PoolEntry(const T* initValue, uint8_t setLength = 1, bool setValid = false);
//! Explicitely deleted copy ctor, copying is not allowed.
PoolEntry(const PoolEntry&) = delete;
diff --git a/src/fsfw/datapoollocal/CMakeLists.txt b/src/fsfw/datapoollocal/CMakeLists.txt
index e2db39eb0..749ef688b 100644
--- a/src/fsfw/datapoollocal/CMakeLists.txt
+++ b/src/fsfw/datapoollocal/CMakeLists.txt
@@ -1,10 +1,6 @@
-target_sources(${LIB_FSFW_NAME}
- PRIVATE
- LocalDataPoolManager.cpp
- LocalDataSet.cpp
- LocalPoolDataSetBase.cpp
- LocalPoolObjectBase.cpp
- SharedLocalDataSet.cpp
-)
+target_sources(
+ ${LIB_FSFW_NAME}
+ PRIVATE LocalDataPoolManager.cpp LocalDataSet.cpp LocalPoolDataSetBase.cpp
+ LocalPoolObjectBase.cpp SharedLocalDataSet.cpp)
-add_subdirectory(internal)
\ No newline at end of file
+add_subdirectory(internal)
diff --git a/src/fsfw/datapoollocal/LocalDataPoolManager.cpp b/src/fsfw/datapoollocal/LocalDataPoolManager.cpp
index 9057de31b..9b7f800f1 100644
--- a/src/fsfw/datapoollocal/LocalDataPoolManager.cpp
+++ b/src/fsfw/datapoollocal/LocalDataPoolManager.cpp
@@ -84,8 +84,8 @@ ReturnValue_t LocalDataPoolManager::initializeHousekeepingPoolEntriesOnce() {
return result;
}
- printWarningOrError(sif::OutputTypes::OUT_WARNING, "initialize", HasReturnvaluesIF::RETURN_FAILED,
- "The map should only be initialized once");
+ printWarningOrError(sif::OutputTypes::OUT_WARNING, "initializeHousekeepingPoolEntriesOnce",
+ HasReturnvaluesIF::RETURN_FAILED, "The map should only be initialized once");
return HasReturnvaluesIF::RETURN_OK;
}
@@ -696,9 +696,10 @@ void LocalDataPoolManager::performPeriodicHkGeneration(HkReceiver& receiver) {
if (result != HasReturnvaluesIF::RETURN_OK) {
/* Configuration error */
#if FSFW_CPP_OSTREAM_ENABLED == 1
- sif::warning << "LocalDataPoolManager::performHkOperation: HK generation failed." << std::endl;
+ sif::warning << "LocalDataPoolManager::performPeriodicHkOperation: HK generation failed."
+ << std::endl;
#else
- sif::printWarning("LocalDataPoolManager::performHkOperation: HK generation failed.\n");
+ sif::printWarning("LocalDataPoolManager::performPeriodicHkOperation: HK generation failed.\n");
#endif
}
}
@@ -787,6 +788,10 @@ ReturnValue_t LocalDataPoolManager::generateSetStructurePacket(sid_t sid, bool i
// Serialize set packet into store.
size_t size = 0;
result = setPacket.serialize(&storePtr, &size, expectedSize, SerializeIF::Endianness::BIG);
+ if (result != HasReturnvaluesIF::RETURN_OK) {
+ ipcStore->deleteData(storeId);
+ return result;
+ }
if (expectedSize != size) {
printWarningOrError(sif::OutputTypes::OUT_WARNING, "generateSetStructurePacket",
HasReturnvaluesIF::RETURN_FAILED,
@@ -801,7 +806,10 @@ ReturnValue_t LocalDataPoolManager::generateSetStructurePacket(sid_t sid, bool i
HousekeepingMessage::setHkStuctureReportReply(&reply, sid, storeId);
}
- hkQueue->reply(&reply);
+ result = hkQueue->reply(&reply);
+ if (result != HasReturnvaluesIF::RETURN_OK) {
+ ipcStore->deleteData(storeId);
+ }
return result;
}
diff --git a/src/fsfw/datapoollocal/LocalPoolDataSetBase.cpp b/src/fsfw/datapoollocal/LocalPoolDataSetBase.cpp
index 4a0762126..62fdb184f 100644
--- a/src/fsfw/datapoollocal/LocalPoolDataSetBase.cpp
+++ b/src/fsfw/datapoollocal/LocalPoolDataSetBase.cpp
@@ -94,13 +94,14 @@ ReturnValue_t LocalPoolDataSetBase::serializeWithValidityBuffer(
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
const uint8_t validityMaskSize = std::ceil(static_cast(fillCount) / 8.0);
uint8_t *validityPtr = nullptr;
-#ifdef _MSC_VER
- /* Use a std::vector here because MSVC will (rightly) not create a fixed size array
- with a non constant size specifier */
- std::vector validityMask(validityMaskSize);
+#if defined(_MSC_VER) || defined(__clang__)
+ // Use a std::vector here because MSVC will (rightly) not create a fixed size array
+ // with a non constant size specifier. The Apple compiler (LLVM) will not accept
+ // the initialization of a variable sized array
+ std::vector validityMask(validityMaskSize, 0);
validityPtr = validityMask.data();
#else
- uint8_t validityMask[validityMaskSize] = {0};
+ uint8_t validityMask[validityMaskSize] = {};
validityPtr = validityMask;
#endif
uint8_t validBufferIndex = 0;
diff --git a/src/fsfw/datapoollocal/LocalPoolObjectBase.cpp b/src/fsfw/datapoollocal/LocalPoolObjectBase.cpp
index c974601cc..82aefc18c 100644
--- a/src/fsfw/datapoollocal/LocalPoolObjectBase.cpp
+++ b/src/fsfw/datapoollocal/LocalPoolObjectBase.cpp
@@ -47,13 +47,14 @@ LocalPoolObjectBase::LocalPoolObjectBase(object_id_t poolOwner, lp_id_t poolId,
HasLocalDataPoolIF* hkOwner = ObjectManager::instance()->get(poolOwner);
if (hkOwner == nullptr) {
#if FSFW_CPP_OSTREAM_ENABLED == 1
- sif::error << "LocalPoolVariable: The supplied pool owner did not implement the correct "
- "interface HasLocalDataPoolIF!"
- << std::endl;
+ sif::error << "LocalPoolVariable: The supplied pool owner 0x" << std::hex << poolOwner
+ << std::dec << " did not implement the correct interface "
+ << "HasLocalDataPoolIF" << std::endl;
#else
sif::printError(
- "LocalPoolVariable: The supplied pool owner did not implement the correct "
- "interface HasLocalDataPoolIF!\n");
+ "LocalPoolVariable: The supplied pool owner 0x%08x did not implement the correct "
+ "interface HasLocalDataPoolIF\n",
+ poolOwner);
#endif
return;
}
diff --git a/src/fsfw/datapoollocal/LocalPoolObjectBase.h b/src/fsfw/datapoollocal/LocalPoolObjectBase.h
index 56e190df1..b2ffa4c18 100644
--- a/src/fsfw/datapoollocal/LocalPoolObjectBase.h
+++ b/src/fsfw/datapoollocal/LocalPoolObjectBase.h
@@ -23,8 +23,8 @@ class LocalPoolObjectBase : public PoolVariableIF, public HasReturnvaluesIF, pub
LocalPoolObjectBase(object_id_t poolOwner, lp_id_t poolId, DataSetIF* dataSet = nullptr,
pool_rwm_t setReadWriteMode = pool_rwm_t::VAR_READ_WRITE);
- void setReadWriteMode(pool_rwm_t newReadWriteMode);
- pool_rwm_t getReadWriteMode() const;
+ void setReadWriteMode(pool_rwm_t newReadWriteMode) override;
+ pool_rwm_t getReadWriteMode() const override;
bool isValid() const override;
void setValid(bool valid) override;
diff --git a/src/fsfw/datapoollocal/LocalPoolVariable.tpp b/src/fsfw/datapoollocal/LocalPoolVariable.tpp
index 9bb30611b..f800dfd3e 100644
--- a/src/fsfw/datapoollocal/LocalPoolVariable.tpp
+++ b/src/fsfw/datapoollocal/LocalPoolVariable.tpp
@@ -5,205 +5,189 @@
#error Include LocalPoolVariable.h before LocalPoolVariable.tpp!
#endif
-template
-inline LocalPoolVariable::LocalPoolVariable(HasLocalDataPoolIF* hkOwner,
- lp_id_t poolId, DataSetIF* dataSet, pool_rwm_t setReadWriteMode):
- LocalPoolObjectBase(poolId, hkOwner, dataSet, setReadWriteMode) {}
+template
+inline LocalPoolVariable::LocalPoolVariable(HasLocalDataPoolIF* hkOwner, lp_id_t poolId,
+ DataSetIF* dataSet, pool_rwm_t setReadWriteMode)
+ : LocalPoolObjectBase(poolId, hkOwner, dataSet, setReadWriteMode) {}
-template
-inline LocalPoolVariable::LocalPoolVariable(object_id_t poolOwner,
- lp_id_t poolId, DataSetIF *dataSet, pool_rwm_t setReadWriteMode):
- LocalPoolObjectBase(poolOwner, poolId, dataSet, setReadWriteMode) {}
+template
+inline LocalPoolVariable::LocalPoolVariable(object_id_t poolOwner, lp_id_t poolId,
+ DataSetIF* dataSet, pool_rwm_t setReadWriteMode)
+ : LocalPoolObjectBase(poolOwner, poolId, dataSet, setReadWriteMode) {}
+template
+inline LocalPoolVariable::LocalPoolVariable(gp_id_t globalPoolId, DataSetIF* dataSet,
+ pool_rwm_t setReadWriteMode)
+ : LocalPoolObjectBase(globalPoolId.objectId, globalPoolId.localPoolId, dataSet,
+ setReadWriteMode) {}
-template
-inline LocalPoolVariable::LocalPoolVariable(gp_id_t globalPoolId,
- DataSetIF *dataSet, pool_rwm_t setReadWriteMode):
- LocalPoolObjectBase(globalPoolId.objectId, globalPoolId.localPoolId,
- dataSet, setReadWriteMode){}
-
-
-template
-inline ReturnValue_t LocalPoolVariable::read(
- MutexIF::TimeoutType timeoutType, uint32_t timeoutMs) {
- if(hkManager == nullptr) {
- return readWithoutLock();
- }
- MutexIF* mutex = LocalDpManagerAttorney::getMutexHandle(*hkManager);
- ReturnValue_t result = mutex->lockMutex(timeoutType, timeoutMs);
- if(result != HasReturnvaluesIF::RETURN_OK) {
- return result;
- }
- result = readWithoutLock();
- mutex->unlockMutex();
+template
+inline ReturnValue_t LocalPoolVariable::read(MutexIF::TimeoutType timeoutType,
+ uint32_t timeoutMs) {
+ if (hkManager == nullptr) {
+ return readWithoutLock();
+ }
+ MutexIF* mutex = LocalDpManagerAttorney::getMutexHandle(*hkManager);
+ ReturnValue_t result = mutex->lockMutex(timeoutType, timeoutMs);
+ if (result != HasReturnvaluesIF::RETURN_OK) {
return result;
+ }
+ result = readWithoutLock();
+ mutex->unlockMutex();
+ return result;
}
-template
+template
inline ReturnValue_t LocalPoolVariable::readWithoutLock() {
- if(readWriteMode == pool_rwm_t::VAR_WRITE) {
- object_id_t targetObjectId = hkManager->getCreatorObjectId();
- reportReadCommitError("LocalPoolVector",
- PoolVariableIF::INVALID_READ_WRITE_MODE, true, targetObjectId,
- localPoolId);
- return PoolVariableIF::INVALID_READ_WRITE_MODE;
- }
+ if (readWriteMode == pool_rwm_t::VAR_WRITE) {
+ object_id_t targetObjectId = hkManager->getCreatorObjectId();
+ reportReadCommitError("LocalPoolVector", PoolVariableIF::INVALID_READ_WRITE_MODE, true,
+ targetObjectId, localPoolId);
+ return PoolVariableIF::INVALID_READ_WRITE_MODE;
+ }
- PoolEntry* poolEntry = nullptr;
- ReturnValue_t result = LocalDpManagerAttorney::fetchPoolEntry(*hkManager, localPoolId,
- &poolEntry);
- if(result != RETURN_OK) {
- object_id_t ownerObjectId = hkManager->getCreatorObjectId();
- reportReadCommitError("LocalPoolVariable", result,
- false, ownerObjectId, localPoolId);
- return result;
- }
-
- this->value = *(poolEntry->getDataPtr());
- this->valid = poolEntry->getValid();
- return RETURN_OK;
-}
-
-template
-inline ReturnValue_t LocalPoolVariable::commit(bool setValid,
- MutexIF::TimeoutType timeoutType, uint32_t timeoutMs) {
- this->setValid(setValid);
- return commit(timeoutType, timeoutMs);
-}
-
-template
-inline ReturnValue_t LocalPoolVariable::commit(
- MutexIF::TimeoutType timeoutType, uint32_t timeoutMs) {
- if(hkManager == nullptr) {
- return commitWithoutLock();
- }
- MutexIF* mutex = LocalDpManagerAttorney::getMutexHandle(*hkManager);
- ReturnValue_t result = mutex->lockMutex(timeoutType, timeoutMs);
- if(result != HasReturnvaluesIF::RETURN_OK) {
- return result;
- }
- result = commitWithoutLock();
- mutex->unlockMutex();
+ PoolEntry* poolEntry = nullptr;
+ ReturnValue_t result =
+ LocalDpManagerAttorney::fetchPoolEntry(*hkManager, localPoolId, &poolEntry);
+ if (result != RETURN_OK) {
+ object_id_t ownerObjectId = hkManager->getCreatorObjectId();
+ reportReadCommitError("LocalPoolVariable", result, false, ownerObjectId, localPoolId);
return result;
+ }
+
+ this->value = *(poolEntry->getDataPtr());
+ this->valid = poolEntry->getValid();
+ return RETURN_OK;
}
-template
+template
+inline ReturnValue_t LocalPoolVariable::commit(bool setValid, MutexIF::TimeoutType timeoutType,
+ uint32_t timeoutMs) {
+ this->setValid(setValid);
+ return commit(timeoutType, timeoutMs);
+}
+
+template
+inline ReturnValue_t LocalPoolVariable::commit(MutexIF::TimeoutType timeoutType,
+ uint32_t timeoutMs) {
+ if (hkManager == nullptr) {
+ return commitWithoutLock();
+ }
+ MutexIF* mutex = LocalDpManagerAttorney::getMutexHandle(*hkManager);
+ ReturnValue_t result = mutex->lockMutex(timeoutType, timeoutMs);
+ if (result != HasReturnvaluesIF::RETURN_OK) {
+ return result;
+ }
+ result = commitWithoutLock();
+ mutex->unlockMutex();
+ return result;
+}
+
+template
inline ReturnValue_t LocalPoolVariable::commitWithoutLock() {
- if(readWriteMode == pool_rwm_t::VAR_READ) {
- object_id_t targetObjectId = hkManager->getCreatorObjectId();
- reportReadCommitError("LocalPoolVector",
- PoolVariableIF::INVALID_READ_WRITE_MODE, false, targetObjectId,
- localPoolId);
- return PoolVariableIF::INVALID_READ_WRITE_MODE;
- }
+ if (readWriteMode == pool_rwm_t::VAR_READ) {
+ object_id_t targetObjectId = hkManager->getCreatorObjectId();
+ reportReadCommitError("LocalPoolVector", PoolVariableIF::INVALID_READ_WRITE_MODE, false,
+ targetObjectId, localPoolId);
+ return PoolVariableIF::INVALID_READ_WRITE_MODE;
+ }
- PoolEntry* poolEntry = nullptr;
- ReturnValue_t result = LocalDpManagerAttorney::fetchPoolEntry(*hkManager, localPoolId,
- &poolEntry);
- if(result != RETURN_OK) {
- object_id_t ownerObjectId = hkManager->getCreatorObjectId();
- reportReadCommitError("LocalPoolVariable", result,
- false, ownerObjectId, localPoolId);
- return result;
- }
+ PoolEntry* poolEntry = nullptr;
+ ReturnValue_t result =
+ LocalDpManagerAttorney::fetchPoolEntry(*hkManager, localPoolId, &poolEntry);
+ if (result != RETURN_OK) {
+ object_id_t ownerObjectId = hkManager->getCreatorObjectId();
+ reportReadCommitError("LocalPoolVariable", result, false, ownerObjectId, localPoolId);
+ return result;
+ }
- *(poolEntry->getDataPtr()) = this->value;
- poolEntry->setValid(this->valid);
- return RETURN_OK;
+ *(poolEntry->getDataPtr()) = this->value;
+ poolEntry->setValid(this->valid);
+ return RETURN_OK;
}
-template
-inline ReturnValue_t LocalPoolVariable::serialize(uint8_t** buffer,
- size_t* size, const size_t max_size,
- SerializeIF::Endianness streamEndianness) const {
- return SerializeAdapter::serialize(&value,
- buffer, size ,max_size, streamEndianness);
+template
+inline ReturnValue_t LocalPoolVariable::serialize(
+ uint8_t** buffer, size_t* size, const size_t max_size,
+ SerializeIF::Endianness streamEndianness) const {
+ return SerializeAdapter::serialize(&value, buffer, size, max_size, streamEndianness);
}
-template
+template
inline size_t LocalPoolVariable::getSerializedSize() const {
- return SerializeAdapter::getSerializedSize(&value);
+ return SerializeAdapter::getSerializedSize(&value);
}
-template
-inline ReturnValue_t LocalPoolVariable::deSerialize(const uint8_t** buffer,
- size_t* size, SerializeIF::Endianness streamEndianness) {
- return SerializeAdapter::deSerialize(&value, buffer, size, streamEndianness);
+template
+inline ReturnValue_t LocalPoolVariable::deSerialize(const uint8_t** buffer, size_t* size,
+ SerializeIF::Endianness streamEndianness) {
+ return SerializeAdapter::deSerialize(&value, buffer, size, streamEndianness);
}
#if FSFW_CPP_OSTREAM_ENABLED == 1
-template
-inline std::ostream& operator<< (std::ostream &out,
- const LocalPoolVariable &var) {
- out << var.value;
- return out;
+template
+inline std::ostream& operator<<(std::ostream& out, const LocalPoolVariable& var) {
+ out << var.value;
+ return out;
}
#endif
-template
+template
inline LocalPoolVariable::operator T() const {
- return value;
+ return value;
}
-template
-inline LocalPoolVariable & LocalPoolVariable::operator=(
- const T& newValue) {
- value = newValue;
- return *this;
+template
+inline LocalPoolVariable& LocalPoolVariable::operator=(const T& newValue) {
+ value = newValue;
+ return *this;
}
-template
-inline LocalPoolVariable& LocalPoolVariable::operator =(
- const LocalPoolVariable& newPoolVariable) {
- value = newPoolVariable.value;
- return *this;
+template
+inline LocalPoolVariable& LocalPoolVariable::operator=(
+ const LocalPoolVariable& newPoolVariable) {
+ value = newPoolVariable.value;
+ return *this;
}
-template
-inline bool LocalPoolVariable::operator ==(
- const LocalPoolVariable &other) const {
- return this->value == other.value;
+template
+inline bool LocalPoolVariable::operator==(const LocalPoolVariable& other) const {
+ return this->value == other.value;
}
-template
-inline bool LocalPoolVariable::operator ==(const T &other) const {
- return this->value == other;
+template
+inline bool LocalPoolVariable::operator==(const T& other) const {
+ return this->value == other;
}
-
-template
-inline bool LocalPoolVariable::operator !=(
- const LocalPoolVariable &other) const {
- return not (*this == other);
+template
+inline bool LocalPoolVariable::operator!=(const LocalPoolVariable& other) const {
+ return not(*this == other);
}
-template
-inline bool LocalPoolVariable::operator !=(const T &other) const {
- return not (*this == other);
+template
+inline bool LocalPoolVariable::operator!=(const T& other) const {
+ return not(*this == other);
}
-
-template
-inline bool LocalPoolVariable::operator <(
- const LocalPoolVariable &other) const {
- return this->value < other.value;
+template
+inline bool LocalPoolVariable::operator<(const LocalPoolVariable& other) const {
+ return this->value < other.value;
}
-template
-inline bool LocalPoolVariable::operator <(const T &other) const {
- return this->value < other;
+template
+inline bool LocalPoolVariable::operator<(const T& other) const {
+ return this->value < other;
}
-
-template
-inline bool LocalPoolVariable::operator >(
- const LocalPoolVariable &other) const {
- return not (*this < other);
+template
+inline bool LocalPoolVariable::operator>(const LocalPoolVariable& other) const {
+ return not(*this < other);
}
-template
-inline bool LocalPoolVariable::operator >(const T &other) const {
- return not (*this < other);
+template