diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index b60ace57bc..964fc8ad9a 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -126,6 +126,20 @@ pmacc-compile-reduced-matrix:
job: pmacc-generate-reduced-matrix
strategy: depend
+picongpu-unittest-generate-reduced-matrix:
+ variables:
+ PIC_INPUTS: "unit"
+ TEST_TUPLE_NUM_ELEM: 1
+ extends: ".base_generate-reduced-matrix"
+
+picongpu-unittest-compile-reduced-matrix:
+ stage: test
+ trigger:
+ include:
+ - artifact: compile.yml
+ job: picongpu-unittest-generate-reduced-matrix
+ strategy: depend
+
pypicongpu-generate-full-matrix:
stage: generate
image: ubuntu:22.04
diff --git a/share/ci/backendFlags.sh b/share/ci/backendFlags.sh
new file mode 100755
index 0000000000..97c6f97435
--- /dev/null
+++ b/share/ci/backendFlags.sh
@@ -0,0 +1,57 @@
+#!/bin/bash
+
+###################################################
+# translate PIConGPU backend names into CMake Flags
+###################################################
+
+get_backend_flags()
+{
+ backend_cfg=(${1//:/ })
+ num_options="${#backend_cfg[@]}"
+ if [ $num_options -gt 2 ] ; then
+ echo "-b|--backend must be contain 'backend:arch' or 'backend'" >&2
+ exit 1
+ fi
+ if [ "${backend_cfg[0]}" == "cuda" ] ; then
+ result+=" -Dalpaka_ACC_GPU_CUDA_ENABLE=ON -Dalpaka_ACC_GPU_CUDA_ONLY_MODE=ON"
+ if [ $num_options -eq 2 ] ; then
+ result+=" -DCMAKE_CUDA_ARCHITECTURES=\"${backend_cfg[1]}\""
+ else
+ result+=" -DCMAKE_CUDA_ARCHITECTURES=52"
+ fi
+ elif [ "${backend_cfg[0]}" == "omp2b" ] ; then
+ result+=" -Dalpaka_ACC_CPU_B_OMP2_T_SEQ_ENABLE=ON"
+ if [ $num_options -eq 2 ] ; then
+ result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
+ fi
+ elif [ "${backend_cfg[0]}" == "serial" ] ; then
+ result+=" -Dalpaka_ACC_CPU_B_SEQ_T_SEQ_ENABLE=ON"
+ if [ $num_options -eq 2 ] ; then
+ result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
+ fi
+ elif [ "${backend_cfg[0]}" == "tbb" ] ; then
+ result+=" -Dalpaka_ACC_CPU_B_TBB_T_SEQ_ENABLE=ON"
+ if [ $num_options -eq 2 ] ; then
+ result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
+ fi
+ elif [ "${backend_cfg[0]}" == "threads" ] ; then
+ result+=" -Dalpaka_ACC_CPU_B_SEQ_T_THREADS_ENABLE=ON"
+ if [ $num_options -eq 2 ] ; then
+ result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
+ fi
+ elif [ "${backend_cfg[0]}" == "hip" ] ; then
+ result+=" -Dalpaka_ACC_GPU_HIP_ENABLE=ON -Dalpaka_ACC_GPU_HIP_ONLY_MODE=ON"
+ if [ $num_options -eq 2 ] ; then
+ result+=" -DGPU_TARGETS=\"${backend_cfg[1]}\""
+ else
+ # If no architecture is given build for Radeon VII or MI50/60.
+ result+=" -DGPU_TARGETS=gfx906"
+ fi
+ else
+ echo "unsupported backend given '$1'" >&2
+ exit 1
+ fi
+
+ echo "$result"
+ exit 0
+}
diff --git a/share/ci/generate_reduced_matrix.sh b/share/ci/generate_reduced_matrix.sh
index 4b29e54587..f45324ec79 100755
--- a/share/ci/generate_reduced_matrix.sh
+++ b/share/ci/generate_reduced_matrix.sh
@@ -33,6 +33,9 @@ folders=()
if [ "$PIC_INPUTS" == "pmacc" ] ; then
# create test cases for PMacc
echo "pmacc" | tr " " "\n" | n_wise_generator.py $@ --limit_boost_version
+elif [ "$PIC_INPUTS" == "unit" ] ; then
+ # create test cases for PMacc
+ echo "unit" | tr " " "\n" | n_wise_generator.py $@ --limit_boost_version
else
# create test cases for PIConGPU
for CASE in ${PIC_INPUTS}; do
diff --git a/share/ci/n_wise_generator.py b/share/ci/n_wise_generator.py
index 37c8511b7d..d1075d2ae0 100755
--- a/share/ci/n_wise_generator.py
+++ b/share/ci/n_wise_generator.py
@@ -296,7 +296,8 @@ def is_valid_combination(row):
v_cuda_hip_str = "" if v_cuda_hip == 0 else str(v_cuda_hip)
os_name = pairs[2][0]
os_version = get_version(pairs[2])
- image_prefix = "_run" if folder == "pmacc" else "_compile"
+ image_prefix = "_run" if folder == "pmacc" or folder == "unit"\
+ else "_compile"
job_name = compiler + "_" + backend + v_cuda_hip_str + \
"_boost" + boost_version + "_" + folder.replace("/", ".")
print(job_name + ":")
diff --git a/share/ci/run_picongpu_tests.sh b/share/ci/run_picongpu_tests.sh
index 04a2f60a3c..b74c224caf 100755
--- a/share/ci/run_picongpu_tests.sh
+++ b/share/ci/run_picongpu_tests.sh
@@ -4,7 +4,6 @@ set -e
set -o pipefail
# the default build type is Release
-# if neccesary, you can rerun the pipeline with another build type-> https://docs.gitlab.com/ee/ci/pipelines.html#manually-executing-pipelines
# to change the build type, you must set the environment variable PIC_BUILD_TYPE
if [[ ! -v PIC_BUILD_TYPE ]] ; then
PIC_BUILD_TYPE=Release ;
diff --git a/share/ci/run_picongpu_unit_tests.sh b/share/ci/run_picongpu_unit_tests.sh
new file mode 100755
index 0000000000..b06c33c467
--- /dev/null
+++ b/share/ci/run_picongpu_unit_tests.sh
@@ -0,0 +1,98 @@
+#!/bin/bash
+# Execute PIConGPU's unit tests
+
+set -e
+set -o pipefail
+
+export code_DIR=$CI_PROJECT_DIR
+source $code_DIR/share/ci/backendFlags.sh
+
+# the default build type is Release
+# to change the build type, you must set the environment variable PIC_BUILD_TYPE
+if [[ ! -v PIC_BUILD_TYPE ]] ; then
+ PIC_BUILD_TYPE=Release ;
+fi
+
+if [[ "$CI_RUNNER_TAGS" =~ .*cpuonly.* ]] ; then
+ # In cases where the compile-only job is executed on a GPU runner but with different kinds of accelerators
+ # we need to reset the variables to avoid compiling for the wrong architecture and accelerator.
+ unset CI_GPUS
+ unset CI_GPU_ARCH
+fi
+
+if [ -n "$CI_GPUS" ] ; then
+ # select randomly a device if multiple exists
+ # CI_GPUS is provided by the gitlab CI runner
+ SELECTED_DEVICE_ID=$((RANDOM%CI_GPUS))
+ export HIP_VISIBLE_DEVICES=$SELECTED_DEVICE_ID
+ export CUDA_VISIBLE_DEVICES=$SELECTED_DEVICE_ID
+ echo "selected device '$SELECTED_DEVICE_ID' of '$CI_GPUS'"
+else
+ echo "No GPU device selected because environment variable CI_GPUS is not set."
+fi
+
+if [[ "$PIC_BACKEND" =~ hip.* ]] || [[ "$PIC_BACKEND" =~ cuda.* ]] ; then
+ if [ -n "$CI_GPU_ARCH" ] ; then
+ export PIC_BACKEND="${PIC_BACKEND}:${CI_GPU_ARCH}"
+ fi
+fi
+
+###################################################
+# cmake config builder
+###################################################
+
+PIC_CONST_ARGS=""
+PIC_CONST_ARGS="${PIC_CONST_ARGS} -DCMAKE_BUILD_TYPE=${PIC_BUILD_TYPE}"
+CMAKE_ARGS="${PIC_CONST_ARGS} ${PIC_CMAKE_ARGS} -DCMAKE_CXX_COMPILER=${CXX_VERSION} -DBOOST_ROOT=/opt/boost/${BOOST_VERSION}"
+CMAKE_ARGS="$CMAKE_ARGS -DUSE_MPI_AS_ROOT_USER=ON"
+
+# check and activate if clang should be used as CUDA device compiler
+if [ -n "$CI_CLANG_AS_CUDA_COMPILER" ] ; then
+ export PATH="$(agc-manager -b cuda)/bin:$PATH"
+ CMAKE_ARGS="$CMAKE_ARGS -DCMAKE_CUDA_COMPILER=${CXX_VERSION}"
+fi
+
+alpaka_backend=$(get_backend_flags ${PIC_BACKEND})
+CMAKE_ARGS="$CMAKE_ARGS $alpaka_backend"
+
+###################################################
+# build and run unit tests
+###################################################
+
+# adjust number of parallel builds to avoid out of memory errors
+# PIC_BUILD_REQUIRED_MEM_BYTES is a configured variable in the CI web interface
+PIC_PARALLEL_BUILDS=$(($CI_RAM_BYTES_TOTAL/$PIC_BUILD_REQUIRED_MEM_BYTES))
+
+# limit to number of available cores
+if [ $PIC_PARALLEL_BUILDS -gt $CI_CPUS ] ; then
+ PIC_PARALLEL_BUILDS=$CI_CPUS
+fi
+
+# CI_MAX_PARALLELISM is a configured variable in the CI web interface
+if [ $PIC_PARALLEL_BUILDS -gt $CI_MAX_PARALLELISM ] ; then
+ PIC_PARALLEL_BUILDS=$CI_MAX_PARALLELISM
+fi
+
+## run unit tests
+export unitTest_folder=$HOME/buildPICUnitTest
+mkdir -p $unitTest_folder
+cd $unitTest_folder
+
+echo -e "\033[0;32m///////////////////////////////////////////////////"
+echo "PIC_BUILD_REQUIRED_MEM_BYTES-> ${PIC_BUILD_REQUIRED_MEM_BYTES}"
+echo "CI_RAM_BYTES_TOTAL -> ${CI_RAM_BYTES_TOTAL}"
+echo "CI_CPUS -> ${CI_CPUS}"
+echo "CI_MAX_PARALLELISM -> ${CI_MAX_PARALLELISM}"
+echo "number of processor threads -> $(nproc)"
+echo "number of parallel builds -> $PIC_PARALLEL_BUILDS"
+echo "cmake version -> $(cmake --version | head -n 1)"
+echo "build directory -> $(pwd)"
+echo "CMAKE_ARGS -> ${CMAKE_ARGS}"
+echo "accelerator -> ${PIC_BACKEND}"
+echo "input set -> ${PIC_TEST_CASE_FOLDER}"
+echo -e "/////////////////////////////////////////////////// \033[0m \n\n"
+
+cmake $CMAKE_ARGS $code_DIR/share/picongpu/unit
+make -j $PIC_PARALLEL_BUILDS
+# execute on one device
+ctest -V
diff --git a/share/ci/run_pmacc_tests.sh b/share/ci/run_pmacc_tests.sh
index 86488453cc..b446848a7a 100755
--- a/share/ci/run_pmacc_tests.sh
+++ b/share/ci/run_pmacc_tests.sh
@@ -3,8 +3,9 @@
set -e
set -o pipefail
+source $CI_PROJECT_DIR/share/ci/backendFlags.sh
+
# the default build type is Release
-# if neccesary, you can rerun the pipeline with another build type-> https://docs.gitlab.com/ee/ci/pipelines.html#manually-executing-pipelines
# to change the build type, you must set the environment variable PMACC_BUILD_TYPE
if [[ ! -v PMACC_BUILD_TYPE ]] ; then
PMACC_BUILD_TYPE=Release;
@@ -28,62 +29,6 @@ if [ -n "$CI_CLANG_AS_CUDA_COMPILER" ] ; then
CMAKE_ARGS="$CMAKE_ARGS -DCMAKE_CUDA_COMPILER=${CXX_VERSION}"
fi
-###################################################
-# translate PIConGPU backend names into CMake Flags
-###################################################
-
-get_backend_flags()
-{
- backend_cfg=(${1//:/ })
- num_options="${#backend_cfg[@]}"
- if [ $num_options -gt 2 ] ; then
- echo "-b|--backend must be contain 'backend:arch' or 'backend'" >&2
- exit 1
- fi
- if [ "${backend_cfg[0]}" == "cuda" ] ; then
- result+=" -Dalpaka_ACC_GPU_CUDA_ENABLE=ON -Dalpaka_ACC_GPU_CUDA_ONLY_MODE=ON"
- if [ $num_options -eq 2 ] ; then
- result+=" -DCMAKE_CUDA_ARCHITECTURES=\"${backend_cfg[1]}\""
- else
- result+=" -DCMAKE_CUDA_ARCHITECTURES=52"
- fi
- elif [ "${backend_cfg[0]}" == "omp2b" ] ; then
- result+=" -Dalpaka_ACC_CPU_B_OMP2_T_SEQ_ENABLE=ON"
- if [ $num_options -eq 2 ] ; then
- result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
- fi
- elif [ "${backend_cfg[0]}" == "serial" ] ; then
- result+=" -Dalpaka_ACC_CPU_B_SEQ_T_SEQ_ENABLE=ON"
- if [ $num_options -eq 2 ] ; then
- result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
- fi
- elif [ "${backend_cfg[0]}" == "tbb" ] ; then
- result+=" -Dalpaka_ACC_CPU_B_TBB_T_SEQ_ENABLE=ON"
- if [ $num_options -eq 2 ] ; then
- result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
- fi
- elif [ "${backend_cfg[0]}" == "threads" ] ; then
- result+=" -Dalpaka_ACC_CPU_B_SEQ_T_THREADS_ENABLE=ON"
- if [ $num_options -eq 2 ] ; then
- result+=" -DPMACC_CPU_ARCH=\"${backend_cfg[1]}\""
- fi
- elif [ "${backend_cfg[0]}" == "hip" ] ; then
- result+=" -Dalpaka_ACC_GPU_HIP_ENABLE=ON -Dalpaka_ACC_GPU_HIP_ONLY_MODE=ON"
- if [ $num_options -eq 2 ] ; then
- result+=" -DGPU_TARGETS=\"${backend_cfg[1]}\""
- else
- # If no architecture is given build for Radeon VII or MI50/60.
- result+=" -DGPU_TARGETS=gfx906"
- fi
- else
- echo "unsupported backend given '$1'" >&2
- exit 1
- fi
-
- echo "$result"
- exit 0
-}
-
###################################################
# build an run tests
###################################################
diff --git a/share/ci/run_tests.sh b/share/ci/run_tests.sh
index 1d9c6f515e..74691378f3 100755
--- a/share/ci/run_tests.sh
+++ b/share/ci/run_tests.sh
@@ -5,6 +5,8 @@ set -o pipefail
if [ "$1" == "pmacc" ] ; then
$CI_PROJECT_DIR/share/ci/run_pmacc_tests.sh
+elif [ "$1" == "unit" ] ; then
+ $CI_PROJECT_DIR/share/ci/run_picongpu_unit_tests.sh
else
$CI_PROJECT_DIR/share/ci/run_picongpu_tests.sh
fi
diff --git a/share/picongpu/unit/CMakeLists.txt b/share/picongpu/unit/CMakeLists.txt
new file mode 100644
index 0000000000..4bf48d8d18
--- /dev/null
+++ b/share/picongpu/unit/CMakeLists.txt
@@ -0,0 +1,93 @@
+# Copyright 2023 Rene Widera
+#
+# This file is part of PIConGPU.
+#
+# PIConGPU is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# PIConGPU is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with PIConGPU.
+# If not, see .
+#
+
+cmake_minimum_required(VERSION 3.22.0)
+project("UnitTest")
+
+###############################################################################
+# Language Flags
+###############################################################################
+
+# enforce C++17
+set(CMAKE_CXX_STANDARD_REQUIRED ON)
+set(CMAKE_CXX_EXTENSIONS OFF)
+set(CMAKE_CXX_STANDARD 17)
+
+
+################################################################################
+# CMake policies
+#
+# Search in _ROOT:
+# https://cmake.org/cmake/help/v3.12/policy/CMP0074.html
+################################################################################
+if(POLICY CMP0074)
+ cmake_policy(SET CMP0074 NEW)
+endif()
+
+################################################################################
+# PMacc
+################################################################################
+find_package(PMacc REQUIRED CONFIG PATHS "${CMAKE_CURRENT_SOURCE_DIR}/../../../include/pmacc")
+
+###############################################################################
+# Catch2
+###############################################################################
+
+add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/../../../thirdParty/catch2 ${CMAKE_BINARY_DIR}/catch2)
+
+################################################################################
+# MPI
+################################################################################
+
+# MPI is provided by pmacc but to execute the binaries via root additional flags must be given to the execution command
+option(USE_MPI_AS_ROOT_USER "add --allow-run-as-root mpiexec used by ctest" OFF)
+
+if(USE_MPI_AS_ROOT_USER)
+ set(MPI_RUNTIME_FLAGS "--allow-run-as-root")
+endif()
+
+# PIConGPU
+
+include_directories(BEFORE "${CMAKE_CURRENT_SOURCE_DIR}/../../../include")
+
+###############################################################################
+# Targets
+###############################################################################
+
+include_directories(${CMAKE_CURRENT_SOURCE_DIR}/test)
+
+# CTest
+enable_testing()
+
+# Test cases
+# Each *UT.cpp file is an independent executable with one or more test cases
+file(GLOB_RECURSE TESTS *.cpp)
+foreach(dim 2 3)
+ foreach(testCaseFilepath ${TESTS})
+ get_filename_component(testCaseFilename ${testCaseFilepath} NAME)
+ string(REPLACE "UT.cpp" "" testCase ${testCaseFilename})
+ set(testExe "${PROJECT_NAME}-${testCase}-${dim}D")
+ cupla_add_executable(${testExe} ${testCaseFilepath})
+ target_compile_definitions(${testExe} PRIVATE TEST_DIM=${dim})
+ target_link_libraries(${testExe} PUBLIC Catch2 Catch2WithMain)
+ target_link_libraries(${testExe} PRIVATE pmacc::pmacc)
+ add_test(NAME "${testCase}-${dim}D" COMMAND mpiexec ${MPI_RUNTIME_FLAGS} -n 1 ./${testExe})
+ endforeach()
+ string(REPLACE "-DTEST_DIM=${dim}" "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}")
+endforeach()
diff --git a/share/picongpu/unit/README.rst b/share/picongpu/unit/README.rst
new file mode 100755
index 0000000000..3dd3b7cc3b
--- /dev/null
+++ b/share/picongpu/unit/README.rst
@@ -0,0 +1,12 @@
+PIConGPU unit test
+==================
+
+Test components in an as best as possible isolated environment.
+
+Example how to compile and execute the tests.compile
+
+.. code-block:: bash
+ # compile for NVIDIA GPUs
+ cmake /share/picongpu/unit/ -Dalpaka_ACC_GPU_CUDA_ENABLE=ON
+ make -j
+ ctest
\ No newline at end of file
diff --git a/share/picongpu/unit/shape.cpp b/share/picongpu/unit/shape.cpp
new file mode 100644
index 0000000000..c5599f3ebf
--- /dev/null
+++ b/share/picongpu/unit/shape.cpp
@@ -0,0 +1,206 @@
+/* Copyright 2023 Rene Widera
+ *
+ * This file is part of PIConGPU.
+ *
+ * PIConGPU is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * PIConGPU is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with PIConGPU.
+ * If not, see .
+ */
+
+#include
+
+#include
+
+// STL
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include
+#include
+#include
+
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+
+//! Helper to setup the PMacc environment
+using TestFixture = pmacc::test::PMaccFixture;
+static TestFixture fixture;
+
+using namespace picongpu;
+using namespace pmacc;
+
+constexpr uint32_t numValues = 1024;
+constexpr uint32_t elemPerBlock = 256;
+
+/** check if floating point result is equal
+ *
+ * Allows an error of one epsilon.
+ * @return true if equal, else false
+ */
+template
+static bool isApproxEqual(T const& a, T const& b)
+{
+ return a == Catch::Approx(b).margin(std::numeric_limits::epsilon());
+}
+
+/** Do not shift the in cell position. */
+struct NoPositionShift
+{
+ template
+ HDINLINE float_X shift(float_X pos)
+ {
+ return pos;
+ }
+};
+
+/** Shift the in cell position.
+ *
+ * Shifting the in cell position before querying the on support shape is required to fulfill the pre conditions of the
+ * shape function.
+ */
+struct PositionShift
+{
+ template
+ HDINLINE float_X shift(float_X pos)
+ {
+ const float_X v_pos = pos - 0.5_X;
+ int intShift;
+ if constexpr(isEven)
+ {
+ // pos range [-1.0;0.5)
+ intShift = v_pos >= float_X{-0.5} ? 0 : -1;
+ }
+ else
+ {
+ // pos range [-1.0;0.5)
+ intShift = v_pos >= float_X{0.0} ? 1 : 0;
+ }
+ return v_pos - float_X(intShift) + float_X{0.5};
+ }
+};
+
+
+/** Test a shape
+ *
+ * Evaluate the assignment shape at all grid points based on a random particle position.
+ * The sum of shape values must be 1.0.
+ *
+ * @tparam T_Shape assignment shape type, supports shape::ChargeAssignment and shape::ChargeAssignmentOnSupport
+ */
+template
+struct TestShape
+{
+ /** Validates the shape
+ *
+ * @param inCellPositionBuffer Buffer with positions, each value must be in rage [0.0;1.0).
+ * @param posShiftFunctor Functor which shifts the position into a valid range to be passed to the shape.
+ */
+ template
+ void operator()(T_PosBuffer& inCellPositionBuffer, T_ShiftFunctor posShiftFunctor)
+ {
+ std::cout << "Test Shape" << typeid(T_Shape).name() << std::endl;
+ ::pmacc::DeviceBuffer deviceInCellPositionBuffer(numValues);
+ ::pmacc::HostBuffer resultHost(numValues);
+ ::pmacc::DeviceBuffer resultDevice(numValues);
+
+ deviceInCellPositionBuffer.copyFrom(inCellPositionBuffer);
+ resultDevice.setValue(0.0_X);
+
+ auto shapeTestKernel
+ = [this] DEVICEONLY(auto const& worker, auto positionShift, auto const& positions, auto result)
+ {
+ auto blockIdx = cupla::blockIdx(worker.getAcc()).x;
+
+ auto forEach = lockstep::makeForEach(worker);
+
+ forEach(
+ [&](uint32_t const idx)
+ {
+ auto valueIdx = blockIdx * elemPerBlock + idx;
+
+ if(valueIdx < numValues)
+ {
+ using Shape = T_Shape;
+ auto shape = Shape{};
+
+ for(int g = Shape::begin; g <= Shape::end; ++g)
+ {
+ // shift the particle position into a valid range to be used to querry the shape
+ auto p = positionShift.template shift(positions[valueIdx]);
+ result[valueIdx] += shape(g - p);
+ }
+ }
+ });
+ };
+
+ auto workerCfg = lockstep::makeWorkerCfg();
+ auto numBlocks = (numValues + elemPerBlock - 1) / elemPerBlock;
+ PMACC_LOCKSTEP_KERNEL(shapeTestKernel, workerCfg)
+ (numBlocks)(posShiftFunctor, deviceInCellPositionBuffer.getDataBox(), resultDevice.getDataBox());
+
+ resultHost.copyFrom(resultDevice);
+
+ auto res = resultHost.getDataBox();
+ for(uint32_t i = 0u; i < numValues; ++i)
+ {
+ auto isCorrect = isApproxEqual(res[i], 1.0_X);
+ if(!isCorrect)
+ std::cerr << "pos=" << inCellPositionBuffer.getDataBox()[i] << " sum=" << res[i] << std::endl;
+ REQUIRE(isCorrect);
+ }
+ }
+};
+
+TEST_CASE("unit::shape", "[shape test]")
+{
+ ::pmacc::HostBuffer inCellPositionBuffer(numValues);
+
+ std::mt19937 mt(42.0);
+ std::uniform_real_distribution<> dist(0.0, 1.0);
+
+ auto posBox = inCellPositionBuffer.getDataBox();
+ // provide random in cell positions
+ for(uint32_t i = 0u; i < numValues; ++i)
+ posBox[i] = dist(mt);
+
+ // check on support assignment shape
+ using OnSupportShapes = pmacc::MakeSeq_t<
+ particles::shapes::NGP::ChargeAssignmentOnSupport,
+ particles::shapes::CIC::ChargeAssignmentOnSupport,
+ particles::shapes::TSC::ChargeAssignmentOnSupport,
+ particles::shapes::PQS::ChargeAssignmentOnSupport,
+ particles::shapes::PCS::ChargeAssignmentOnSupport>;
+
+ meta::ForEach>{}(inCellPositionBuffer, PositionShift{});
+
+ // check assignment shape outside of the uspport
+ using NotOnSupportShapes = pmacc::MakeSeq_t<
+ particles::shapes::NGP::ChargeAssignment,
+ particles::shapes::CIC::ChargeAssignment,
+ particles::shapes::TSC::ChargeAssignment,
+ particles::shapes::PQS::ChargeAssignment,
+ particles::shapes::PCS::ChargeAssignment>;
+
+ meta::ForEach>{}(inCellPositionBuffer, NoPositionShift{});
+}