153 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
	
		
		
			
		
	
	
			153 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
	
|  | //
 | ||
|  | // execution/detail/as_invocable.hpp
 | ||
|  | // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
|  | //
 | ||
|  | // Copyright (c) 2003-2023 Christopher M. Kohlhoff (chris at kohlhoff dot com)
 | ||
|  | //
 | ||
|  | // 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 ASIO_EXECUTION_DETAIL_AS_INVOCABLE_HPP
 | ||
|  | #define ASIO_EXECUTION_DETAIL_AS_INVOCABLE_HPP
 | ||
|  | 
 | ||
|  | #if defined(_MSC_VER) && (_MSC_VER >= 1200)
 | ||
|  | # pragma once
 | ||
|  | #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
 | ||
|  | 
 | ||
|  | #include "asio/detail/config.hpp"
 | ||
|  | #include "asio/detail/atomic_count.hpp"
 | ||
|  | #include "asio/detail/memory.hpp"
 | ||
|  | #include "asio/detail/type_traits.hpp"
 | ||
|  | #include "asio/execution/receiver_invocation_error.hpp"
 | ||
|  | #include "asio/execution/set_done.hpp"
 | ||
|  | #include "asio/execution/set_error.hpp"
 | ||
|  | #include "asio/execution/set_value.hpp"
 | ||
|  | 
 | ||
|  | #include "asio/detail/push_options.hpp"
 | ||
|  | 
 | ||
|  | namespace asio { | ||
|  | namespace execution { | ||
|  | namespace detail { | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_MOVE)
 | ||
|  | 
 | ||
|  | template <typename Receiver, typename> | ||
|  | struct as_invocable | ||
|  | { | ||
|  |   Receiver* receiver_; | ||
|  | 
 | ||
|  |   explicit as_invocable(Receiver& r) ASIO_NOEXCEPT | ||
|  |     : receiver_(asio::detail::addressof(r)) | ||
|  |   { | ||
|  |   } | ||
|  | 
 | ||
|  |   as_invocable(as_invocable&& other) ASIO_NOEXCEPT | ||
|  |     : receiver_(other.receiver_) | ||
|  |   { | ||
|  |     other.receiver_ = 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   ~as_invocable() | ||
|  |   { | ||
|  |     if (receiver_) | ||
|  |       execution::set_done(ASIO_MOVE_OR_LVALUE(Receiver)(*receiver_)); | ||
|  |   } | ||
|  | 
 | ||
|  |   void operator()() ASIO_LVALUE_REF_QUAL ASIO_NOEXCEPT | ||
|  |   { | ||
|  | #if !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |     try | ||
|  |     { | ||
|  | #endif // !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |       execution::set_value(ASIO_MOVE_CAST(Receiver)(*receiver_)); | ||
|  |       receiver_ = 0; | ||
|  | #if !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |     } | ||
|  |     catch (...) | ||
|  |     { | ||
|  | #if defined(ASIO_HAS_STD_EXCEPTION_PTR)
 | ||
|  |       execution::set_error(ASIO_MOVE_CAST(Receiver)(*receiver_), | ||
|  |           std::make_exception_ptr(receiver_invocation_error())); | ||
|  |       receiver_ = 0; | ||
|  | #else // defined(ASIO_HAS_STD_EXCEPTION_PTR)
 | ||
|  |       std::terminate(); | ||
|  | #endif // defined(ASIO_HAS_STD_EXCEPTION_PTR)
 | ||
|  |     } | ||
|  | #endif // !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | #else // defined(ASIO_HAS_MOVE)
 | ||
|  | 
 | ||
|  | template <typename Receiver, typename> | ||
|  | struct as_invocable | ||
|  | { | ||
|  |   Receiver* receiver_; | ||
|  |   asio::detail::shared_ptr<asio::detail::atomic_count> ref_count_; | ||
|  | 
 | ||
|  |   explicit as_invocable(Receiver& r, | ||
|  |       const asio::detail::shared_ptr< | ||
|  |         asio::detail::atomic_count>& c) ASIO_NOEXCEPT | ||
|  |     : receiver_(asio::detail::addressof(r)), | ||
|  |       ref_count_(c) | ||
|  |   { | ||
|  |   } | ||
|  | 
 | ||
|  |   as_invocable(const as_invocable& other) ASIO_NOEXCEPT | ||
|  |     : receiver_(other.receiver_), | ||
|  |       ref_count_(other.ref_count_) | ||
|  |   { | ||
|  |     ++(*ref_count_); | ||
|  |   } | ||
|  | 
 | ||
|  |   ~as_invocable() | ||
|  |   { | ||
|  |     if (--(*ref_count_) == 0) | ||
|  |       execution::set_done(*receiver_); | ||
|  |   } | ||
|  | 
 | ||
|  |   void operator()() ASIO_LVALUE_REF_QUAL ASIO_NOEXCEPT | ||
|  |   { | ||
|  | #if !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |     try | ||
|  |     { | ||
|  | #endif // !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |       execution::set_value(*receiver_); | ||
|  |       ++(*ref_count_); | ||
|  |     } | ||
|  | #if !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |     catch (...) | ||
|  |     { | ||
|  | #if defined(ASIO_HAS_STD_EXCEPTION_PTR)
 | ||
|  |       execution::set_error(*receiver_, | ||
|  |           std::make_exception_ptr(receiver_invocation_error())); | ||
|  |       ++(*ref_count_); | ||
|  | #else // defined(ASIO_HAS_STD_EXCEPTION_PTR)
 | ||
|  |       std::terminate(); | ||
|  | #endif // defined(ASIO_HAS_STD_EXCEPTION_PTR)
 | ||
|  |     } | ||
|  | #endif // !defined(ASIO_NO_EXCEPTIONS)
 | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_MOVE)
 | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | struct is_as_invocable : false_type | ||
|  | { | ||
|  | }; | ||
|  | 
 | ||
|  | template <typename Function, typename T> | ||
|  | struct is_as_invocable<as_invocable<Function, T> > : true_type | ||
|  | { | ||
|  | }; | ||
|  | 
 | ||
|  | } // namespace detail
 | ||
|  | } // namespace execution
 | ||
|  | } // namespace asio
 | ||
|  | 
 | ||
|  | #include "asio/detail/pop_options.hpp"
 | ||
|  | 
 | ||
|  | #endif // ASIO_EXECUTION_DETAIL_AS_INVOCABLE_HPP
 |