// Copyright Oliver Kowalke 2013. // 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) #ifndef BOOST_FIBERS_DETAIL_TASK_OBJECT_H #define BOOST_FIBERS_DETAIL_TASK_OBJECT_H #include #include #include #include #include #if defined(BOOST_NO_CXX17_STD_APPLY) #include #endif #include #include #ifdef BOOST_HAS_ABI_HEADERS # include BOOST_ABI_PREFIX #endif namespace boost { namespace fibers { namespace detail { template< typename Fn, typename Allocator, typename R, typename ... Args > class task_object : public task_base< R, Args ... > { private: typedef task_base< R, Args ... > base_type; public: typedef typename std::allocator_traits< Allocator >::template rebind_alloc< task_object > allocator_type; task_object( allocator_type const& alloc, Fn const& fn) : base_type{}, fn_{ fn }, alloc_{ alloc } { } task_object( allocator_type const& alloc, Fn && fn) : base_type{}, fn_{ std::move( fn) }, alloc_{ alloc } { } void run( Args && ... args) override final { try { this->set_value( #if defined(BOOST_NO_CXX17_STD_APPLY) boost::context::detail::apply( fn_, std::make_tuple( std::forward< Args >( args) ... ) ) #else std::apply( fn_, std::make_tuple( std::forward< Args >( args) ... ) ) #endif ); } catch (...) { this->set_exception( std::current_exception() ); } } typename base_type::ptr_type reset() override final { typedef std::allocator_traits< allocator_type > traity_type; typename traity_type::pointer ptr{ traity_type::allocate( alloc_, 1) }; try { traity_type::construct( alloc_, ptr, alloc_, std::move( fn_) ); } catch (...) { traity_type::deallocate( alloc_, ptr, 1); throw; } return { convert( ptr) }; } protected: void deallocate_future() noexcept override final { destroy_( alloc_, this); } private: Fn fn_; allocator_type alloc_; static void destroy_( allocator_type const& alloc, task_object * p) noexcept { allocator_type a{ alloc }; a.destroy( p); a.deallocate( p, 1); } }; template< typename Fn, typename Allocator, typename ... Args > class task_object< Fn, Allocator, void, Args ... > : public task_base< void, Args ... > { private: typedef task_base< void, Args ... > base_type; public: typedef typename Allocator::template rebind< task_object< Fn, Allocator, void, Args ... > >::other allocator_type; task_object( allocator_type const& alloc, Fn const& fn) : base_type{}, fn_{ fn }, alloc_{ alloc } { } task_object( allocator_type const& alloc, Fn && fn) : base_type{}, fn_{ std::move( fn) }, alloc_{ alloc } { } void run( Args && ... args) override final { try { #if defined(BOOST_NO_CXX17_STD_APPLY) boost::context::detail::apply( fn_, std::make_tuple( std::forward< Args >( args) ... ) ); #else std::apply( fn_, std::make_tuple( std::forward< Args >( args) ... ) ); #endif this->set_value(); } catch (...) { this->set_exception( std::current_exception() ); } } typename base_type::ptr_type reset() override final { typedef std::allocator_traits< allocator_type > traity_type; typename traity_type::pointer ptr{ traity_type::allocate( alloc_, 1) }; try { traity_type::construct( alloc_, ptr, alloc_, std::move( fn_) ); } catch (...) { traity_type::deallocate( alloc_, ptr, 1); throw; } return { convert( ptr) }; } protected: void deallocate_future() noexcept override final { destroy_( alloc_, this); } private: Fn fn_; allocator_type alloc_; static void destroy_( allocator_type const& alloc, task_object * p) noexcept { allocator_type a{ alloc }; a.destroy( p); a.deallocate( p, 1); } }; }}} #ifdef BOOST_HAS_ABI_HEADERS # include BOOST_ABI_SUFFIX #endif #endif // BOOST_FIBERS_DETAIL_TASK_OBJECT_H