diff options
author | Clyne Sullivan <tullivan99@gmail.com> | 2016-11-11 15:02:17 -0500 |
---|---|---|
committer | Clyne Sullivan <tullivan99@gmail.com> | 2016-11-11 15:02:17 -0500 |
commit | 7772ea4579a45bcf63ebd5e68be66ba1a9c72dfa (patch) | |
tree | 9e1ce52ea97102d3513e519a77d999eac228820b /include/distortos/internal/synchronization/CallOnceControlBlock.hpp | |
parent | 02b3ff42cccf32617c88c0ca65436b8c9d4f61eb (diff) |
chibios!
Diffstat (limited to 'include/distortos/internal/synchronization/CallOnceControlBlock.hpp')
-rw-r--r-- | include/distortos/internal/synchronization/CallOnceControlBlock.hpp | 172 |
1 files changed, 0 insertions, 172 deletions
diff --git a/include/distortos/internal/synchronization/CallOnceControlBlock.hpp b/include/distortos/internal/synchronization/CallOnceControlBlock.hpp deleted file mode 100644 index 1684faf..0000000 --- a/include/distortos/internal/synchronization/CallOnceControlBlock.hpp +++ /dev/null @@ -1,172 +0,0 @@ -/** - * \file - * \brief CallOnceControlBlock class header - * - * \author Copyright (C) 2015 Kamil Szczygiel http://www.distortec.com http://www.freddiechopin.info - * - * \par License - * This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not - * distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/. - */ - -#ifndef INCLUDE_DISTORTOS_INTERNAL_SYNCHRONIZATION_CALLONCECONTROLBLOCK_HPP_ -#define INCLUDE_DISTORTOS_INTERNAL_SYNCHRONIZATION_CALLONCECONTROLBLOCK_HPP_ - -#include "estd/invoke.hpp" -#include "estd/TypeErasedFunctor.hpp" - -#include <sys/features.h> - -namespace distortos -{ - -/// GCC 4.9 is needed for CallOnceControlBlock::operator()() function - and thus for OnceFlag and callOnce() - earlier -/// versions don't support parameter pack expansion in lambdas -#define DISTORTOS_CALLONCE_SUPPORTED __GNUC_PREREQ(4, 9) - -#if DISTORTOS_CALLONCE_SUPPORTED == 1 || DOXYGEN == 1 - -namespace internal -{ - -class ThreadList; - -/// CallOnceControlBlock class implements functionality of OnceFlag class and callOnce() -/// \note This class requires GCC 4.9. -class CallOnceControlBlock -{ -public: - - /** - * \brief CallOnceControlBlock's constructor - */ - - constexpr CallOnceControlBlock() : - blockedList_{}, - done_{} - { - - } - - /** - * \brief CallOnceControlBlock's function call operator - * - * Does nothing if any function was already called for this object. In other case provided function and arguments - * are wrapped in a type-erased functor and passed to callOnceImplementation(). - * - * \tparam Function is the function object that will be executed - * \tparam Args are the arguments for \a Function - * - * \param [in] function is the function object that will be executed - * \param [in] args are arguments for \a function - */ - - template<typename Function, typename... Args> - void operator()(Function&& function, Args&&... args); - -private: - - /// Functor is a type-erased interface for functors which execute bounded function with bounded arguments - class Functor : public estd::TypeErasedFunctor<void()> - { - - }; - - /** - * \brief BoundedFunctor is a type-erased Functor which calls its bounded functor - * - * \tparam F is the type of bounded functor - */ - - template<typename F> - class BoundedFunctor : public Functor - { - public: - - /** - * \brief BoundedFunctor's constructor - * - * \param [in] boundedFunctor is a rvalue reference to bounded functor which will be used to move-construct - * internal bounded functor - */ - - constexpr explicit BoundedFunctor(F&& boundedFunctor) : - boundedFunctor_{std::move(boundedFunctor)} - { - - } - - /** - * \brief BoundedFunctor's function call operator - * - * Calls the bounded functor. - */ - - void operator()() const override - { - boundedFunctor_(); - } - - private: - - /// bounded functor - F boundedFunctor_; - }; - - /** - * \brief Helper factory function to make BoundedFunctor object with deduced template arguments - * - * \tparam F is the type of bounded functor - * - * \param [in] boundedFunctor is a rvalue reference to bounded functor which will be used to move-construct internal - * bounded functor - * - * \return BoundedFunctor object with deduced template arguments - */ - - template<typename F> - constexpr static BoundedFunctor<F> makeBoundedFunctor(F&& boundedFunctor) - { - return BoundedFunctor<F>{std::move(boundedFunctor)}; - } - - /** - * \brief Implements callOnce() using type-erased functor. - * - * Does nothing if any function was already called for this object. If the function is currently being executed, but - * not yet done, then the calling thread is blocked. In other case the function is executed and - after it is done - - * all blocked threads are unblocked. - * - * \param [in] functor is a reference to functor which will execute bounded function with bounded arguments - */ - - void callOnceImplementation(const Functor& functor); - - /// pointer to stack-allocated list of ThreadControlBlock objects blocked on associated OnceFlag - ThreadList* blockedList_; - - /// tells whether any function was already called for this object (true) or not (false) - bool done_; -}; - -template<typename Function, typename... Args> -void CallOnceControlBlock::operator()(Function&& function, Args&&... args) -{ - if (done_ == true) // function already executed? - return; - - const auto functor = makeBoundedFunctor( - [&function, &args...]() - { - estd::invoke(std::forward<Function>(function), std::forward<Args>(args)...); - }); - callOnceImplementation(functor); -} - -} // namespace internal - -#endif // DISTORTOS_CALLONCE_SUPPORTED == 1 || DOXYGEN == 1 - -} // namespace distortos - -#endif // INCLUDE_DISTORTOS_INTERNAL_SYNCHRONIZATION_CALLONCECONTROLBLOCK_HPP_ |