| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317 | //// impl/executor.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_IMPL_EXECUTOR_HPP#define ASIO_IMPL_EXECUTOR_HPP#if defined(_MSC_VER) && (_MSC_VER >= 1200)# pragma once#endif // defined(_MSC_VER) && (_MSC_VER >= 1200)#include "asio/detail/config.hpp"#if !defined(ASIO_NO_TS_EXECUTORS)#include <new>#include "asio/detail/atomic_count.hpp"#include "asio/detail/global.hpp"#include "asio/detail/memory.hpp"#include "asio/executor.hpp"#include "asio/system_executor.hpp"#include "asio/detail/push_options.hpp"namespace asio {#if !defined(GENERATING_DOCUMENTATION)// Default polymorphic executor implementation.template <typename Executor, typename Allocator>class executor::impl  : public executor::impl_base{public:  typedef ASIO_REBIND_ALLOC(Allocator, impl) allocator_type;  static impl_base* create(const Executor& e, Allocator a = Allocator())  {    raw_mem mem(a);    impl* p = new (mem.ptr_) impl(e, a);    mem.ptr_ = 0;    return p;  }  static impl_base* create(std::nothrow_t, const Executor& e) noexcept  {    return new (std::nothrow) impl(e, std::allocator<void>());  }  impl(const Executor& e, const Allocator& a) noexcept    : impl_base(false),      ref_count_(1),      executor_(e),      allocator_(a)  {  }  impl_base* clone() const noexcept  {    detail::ref_count_up(ref_count_);    return const_cast<impl_base*>(static_cast<const impl_base*>(this));  }  void destroy() noexcept  {    if (detail::ref_count_down(ref_count_))    {      allocator_type alloc(allocator_);      impl* p = this;      p->~impl();      alloc.deallocate(p, 1);    }  }  void on_work_started() noexcept  {    executor_.on_work_started();  }  void on_work_finished() noexcept  {    executor_.on_work_finished();  }  execution_context& context() noexcept  {    return executor_.context();  }  void dispatch(function&& f)  {    executor_.dispatch(static_cast<function&&>(f), allocator_);  }  void post(function&& f)  {    executor_.post(static_cast<function&&>(f), allocator_);  }  void defer(function&& f)  {    executor_.defer(static_cast<function&&>(f), allocator_);  }  type_id_result_type target_type() const noexcept  {    return type_id<Executor>();  }  void* target() noexcept  {    return &executor_;  }  const void* target() const noexcept  {    return &executor_;  }  bool equals(const impl_base* e) const noexcept  {    if (this == e)      return true;    if (target_type() != e->target_type())      return false;    return executor_ == *static_cast<const Executor*>(e->target());  }private:  mutable detail::atomic_count ref_count_;  Executor executor_;  Allocator allocator_;  struct raw_mem  {    allocator_type allocator_;    impl* ptr_;    explicit raw_mem(const Allocator& a)      : allocator_(a),        ptr_(allocator_.allocate(1))    {    }    ~raw_mem()    {      if (ptr_)        allocator_.deallocate(ptr_, 1);    }  private:    // Disallow copying and assignment.    raw_mem(const raw_mem&);    raw_mem operator=(const raw_mem&);  };};// Polymorphic executor specialisation for system_executor.template <typename Allocator>class executor::impl<system_executor, Allocator>  : public executor::impl_base{public:  static impl_base* create(const system_executor&,      const Allocator& = Allocator())  {    return &detail::global<impl<system_executor, std::allocator<void>> >();  }  static impl_base* create(std::nothrow_t, const system_executor&) noexcept  {    return &detail::global<impl<system_executor, std::allocator<void>> >();  }  impl()    : impl_base(true)  {  }  impl_base* clone() const noexcept  {    return const_cast<impl_base*>(static_cast<const impl_base*>(this));  }  void destroy() noexcept  {  }  void on_work_started() noexcept  {    executor_.on_work_started();  }  void on_work_finished() noexcept  {    executor_.on_work_finished();  }  execution_context& context() noexcept  {    return executor_.context();  }  void dispatch(function&& f)  {    executor_.dispatch(static_cast<function&&>(f),        std::allocator<void>());  }  void post(function&& f)  {    executor_.post(static_cast<function&&>(f),        std::allocator<void>());  }  void defer(function&& f)  {    executor_.defer(static_cast<function&&>(f),        std::allocator<void>());  }  type_id_result_type target_type() const noexcept  {    return type_id<system_executor>();  }  void* target() noexcept  {    return &executor_;  }  const void* target() const noexcept  {    return &executor_;  }  bool equals(const impl_base* e) const noexcept  {    return this == e;  }private:  system_executor executor_;};template <typename Executor>executor::executor(Executor e)  : impl_(impl<Executor, std::allocator<void>>::create(e)){}template <typename Executor>executor::executor(std::nothrow_t, Executor e) noexcept  : impl_(impl<Executor, std::allocator<void>>::create(std::nothrow, e)){}template <typename Executor, typename Allocator>executor::executor(allocator_arg_t, const Allocator& a, Executor e)  : impl_(impl<Executor, Allocator>::create(e, a)){}template <typename Function, typename Allocator>void executor::dispatch(Function&& f,    const Allocator& a) const{  impl_base* i = get_impl();  if (i->fast_dispatch_)    system_executor().dispatch(static_cast<Function&&>(f), a);  else    i->dispatch(function(static_cast<Function&&>(f), a));}template <typename Function, typename Allocator>void executor::post(Function&& f,    const Allocator& a) const{  get_impl()->post(function(static_cast<Function&&>(f), a));}template <typename Function, typename Allocator>void executor::defer(Function&& f,    const Allocator& a) const{  get_impl()->defer(function(static_cast<Function&&>(f), a));}template <typename Executor>Executor* executor::target() noexcept{  return impl_ && impl_->target_type() == type_id<Executor>()    ? static_cast<Executor*>(impl_->target()) : 0;}template <typename Executor>const Executor* executor::target() const noexcept{  return impl_ && impl_->target_type() == type_id<Executor>()    ? static_cast<Executor*>(impl_->target()) : 0;}#endif // !defined(GENERATING_DOCUMENTATION)} // namespace asio#include "asio/detail/pop_options.hpp"#endif // !defined(ASIO_NO_TS_EXECUTORS)#endif // ASIO_IMPL_EXECUTOR_HPP
 |