This is the FORTRAN library for Computational Environmental Systems of the
Department Computational Hydrosystems
Helmholtz Centre for Environmental Research - UFZ
Permoserstr. 15
04318 Leipzig, Germany
It is a lightweight fork of the jams_fortran library maintained by Matthias Cuntz et al: https://github.com/mcuntz/jams_fortran
The jams_fortran library was formerly developed at the CHS department at the UFZ and is now released under the MIT license.
[TOC]
Let's assume, you want to write a fortran program using forces, like this example test.f90:
program test
use mo_message, only : message
implicit none
call message("This is working!")
end program testYou should create a minimal CMakeLists.txt file next to the test.f90 file like this:
cmake_minimum_required(VERSION 3.18 FATAL_ERROR)
# get CPM (package manager)
set(CPM_DOWNLOAD_LOCATION "${CMAKE_BINARY_DIR}/cmake/CPM_0.35.0.cmake")
file(DOWNLOAD https://github.com/cpm-cmake/CPM.cmake/releases/download/v0.35.0/CPM.cmake ${CPM_DOWNLOAD_LOCATION})
include(${CPM_DOWNLOAD_LOCATION})
# create project
project(MyProject LANGUAGES Fortran)
# add executable
add_executable(test test.f90)
# add FORCES dependency
CPMAddPackage("https://git.ufz.de/chs/forces.git@0.3.1")
# link dependencies
target_link_libraries(test forces)There, CPM (the cmake package manager) is downloaded on the fly and used to get FORCES to be able to link against it.
Afterwards you only need to do the following to configure, compile and execute your program:
cmake -B build
cmake --build build --parallel
./build/testAnd it will happily write:
This is working!
If you have the FORCES sources downloaded and you want to link a local executable against it, you can specify a path to CMake with FORCES_EXE:
cmake -B build -DFORCES_EXE=test.f90
cmake --build build --parallel
./build/mainYou can use this with the examples provided in the examples/ directory, e.g. FORCES_EXE=examples/01_grids/01_regridding.f90.
The executable will be always called main.
FORCES uses standard CMake inputs such as CMAKE_BUILD_TYPE and a small set of project-specific cache options:
FORCES_BUILD_TESTING: build the FORCES pfUnit tests. Defaults toOFF.FORCES_WITH_COVERAGE: enable GNU coverage instrumentation for FORCES test builds.FORCES_WITH_OpenMP: enable OpenMP support.FORCES_WITH_MPI: enable MPI support.FORCES_WITH_NETCDF: enable NetCDF support.FORCES_WITH_OPTIMIZATION: include optimization routines.FORCES_ENABLE_NATIVE: enable host-native tuning forReleaseandRelWithDebInfobuilds.FORCES_EXE: build a local executable linked againstforces.
Typical configure commands:
# test build
cmake -B build -DCMAKE_BUILD_TYPE=Debug -DFORCES_BUILD_TESTING=ON
# OpenMP build
cmake -B build -DCMAKE_BUILD_TYPE=Release -DFORCES_BUILD_TESTING=ON -DFORCES_WITH_OpenMP=ON
# GNU coverage build
cmake -B build -DCMAKE_BUILD_TYPE=Debug -DFORCES_BUILD_TESTING=ON -DFORCES_WITH_COVERAGE=ON
# local executable build
cmake -B build -DCMAKE_BUILD_TYPE=Debug -DFORCES_EXE=test.f90FORCES_WITH_COVERAGE is only meaningful for test builds and requires GNU Fortran.
- Fortran compiler: We support gfortran, nagfor and ifort
- Build system: We support make and ninja
- cmake: Software for build automation
- NetCDF-C: NetCDF I/O library
- (optional) fypp: Fortran pre-processor written in Python
It is recommended to have a clean installation at a custom location for a C compiler, a Fortran compiler and the NetCDF C library with consistent compilers.
We recommend to use a conda environment by using Miniconda to get all dependencies easily:
conda create -y --prefix ./forces_env
conda activate ./forces_env
conda config --add channels conda-forge
conda config --set channel_priority strict
conda install -y cmake make fortran-compiler libnetcdf fyppWith this you could now proceed with the example given above.
FORCES uses pFUnit for its unit tests. When FORCES_BUILD_TESTING=ON, CMake will look for PFUNIT. If pFUnit is not available, configuration still succeeds, but no FORCES tests are added.
You need to provide pFUnit to run the tests. It is unfortunately not available on conda yet. To compile it by hand in the current conda environment, you can do the following:
conda install -y m4
git clone https://github.com/Goddard-Fortran-Ecosystem/pFUnit.git
cmake -DCMAKE_BUILD_TYPE=Release -B pFUnit/build -S pFUnit
cmake --build pFUnit/build --parallel
cmake --install pFUnit/build --prefix $CONDA_PREFIX
rm -rf pFUnitThen you can compile and run the tests and/or calculate coverage in the same environment:
cmake -B build -DCMAKE_BUILD_TYPE=Debug -DFORCES_BUILD_TESTING=ON -DFORCES_WITH_COVERAGE=ON
cmake --build build --parallel
cmake --build build --target test
cmake --build build --target coverageRun all configured tests with:
ctest --test-dir build --output-on-failureRun a single pFUnit suite with ctest -R, for example:
ctest --test-dir build --output-on-failure -R test_mo_messageIf you only want to configure a subset of the pFUnit files, use FORCES_TEST_GLOB:
cmake -B build -DCMAKE_BUILD_TYPE=Debug -DFORCES_BUILD_TESTING=ON -DFORCES_TEST_GLOB='test_mo_message.pf'This is useful when iterating on one module or when you want to avoid configuring optional suites such as NetCDF- or optimization-related tests.
The coverage target additionally requires lcov to be available on PATH.
For a more complex project, prepared for unit-tests, documentation and modules, have a look at the Fortran Template.
LGPLv3 (c) 2005-2026 CHS-Developers
