From aae22ffc3e48e8b9e88f092baf46950374b4f56c Mon Sep 17 00:00:00 2001 From: Lisa Julia Nebel Date: Mon, 8 Jul 2024 16:56:19 +0200 Subject: [PATCH] Add test for failing tasklets --- CMakeLists.txt | 2 + tests/test_tasklets_failure.cc | 114 +++++++++++++++++++++++++++++++++ 2 files changed, 116 insertions(+) create mode 100644 tests/test_tasklets_failure.cc diff --git a/CMakeLists.txt b/CMakeLists.txt index 6a89ccf2b..8cce181a4 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -323,6 +323,8 @@ opm_add_test(tutorial1 opm_add_test(test_tasklets DRIVER_ARGS --plain) +opm_add_test(test_tasklets_failure + DRIVER_ARGS --plain) opm_add_test(test_mpiutil PROCESSORS 4 diff --git a/tests/test_tasklets_failure.cc b/tests/test_tasklets_failure.cc new file mode 100644 index 000000000..3babf68d5 --- /dev/null +++ b/tests/test_tasklets_failure.cc @@ -0,0 +1,114 @@ +// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- +// vi: set et ts=4 sw=4 sts=4: +/* + This file is part of the Open Porous Media project (OPM). + + OPM is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 2 of the License, or + (at your option) any later version. + + OPM 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 OPM. If not, see . + + Consult the COPYING file in the top-level source directory of this + module for the precise wording of the license and the list of + copyright holders. +*/ +/*! + * \file + * + * \brief This file serves as an example of how to use the tasklet mechanism for + * asynchronous work. + */ +#include "config.h" + +#include + +#include +#include +#include + +std::mutex outputMutex; + +Opm::TaskletRunner *runner; + +class SleepTasklet : public Opm::TaskletInterface +{ +public: + SleepTasklet(int mseconds, int id) + : mseconds_(mseconds), + id_(id) + {} + + void run() override + { + assert(0 <= runner->workerThreadIndex() && runner->workerThreadIndex() < runner->numWorkerThreads()); + std::cout << "Sleep tasklet " << id_ << " of " << mseconds_ << " ms starting sleep on worker thread " << runner->workerThreadIndex() << std::endl; + std::this_thread::sleep_for(std::chrono::milliseconds(mseconds_)); + outputMutex.lock(); + std::cout << "Sleep tasklet " << id_ << " of " << mseconds_ << " ms completed by worker thread " << runner->workerThreadIndex() << std::endl; + outputMutex.unlock(); + } + +private: + int mseconds_; + int id_; +}; + +class FailingSleepTasklet : public Opm::TaskletInterface +{ +public: + FailingSleepTasklet(int mseconds) + : mseconds_(mseconds) + {} + void run() override + { + std::this_thread::sleep_for(std::chrono::milliseconds(mseconds_)); + outputMutex.lock(); + std::cout << "Failing sleep tasklet of " << mseconds_ << " ms failing now, on work thread " << runner->workerThreadIndex() << std::endl; + outputMutex.unlock(); + throw std::logic_error("Intentional failure for testing"); + } + +private: + int mseconds_; +}; + +int main() +{ + int numWorkers = 2; + runner = new Opm::TaskletRunner(numWorkers); + + // the master thread is not a worker thread + assert(runner->workerThreadIndex() < 0); + assert(runner->numWorkerThreads() == numWorkers); + + // Dispatch some successful tasklets + for (int i = 0; i < 5; ++i) { + auto st = std::make_shared(10,i); + runner->dispatch(st); + } + + // Dispatch a failing tasklet + auto failingSleepTasklet = std::make_shared(100); + runner->dispatch(failingSleepTasklet); + + // Dispatch more successful tasklets + for (int i = 5; i < 10; ++i) { + auto st = std::make_shared(10,i); + runner->dispatch(st); + } + + std::cout << "before barrier" << std::endl; + runner->barrier(); + + // The program should never reach here, because the failingSleepTasklet will end the program before. + assert(false); + return 1; +}