Со времен простейшей формы подъёма. Токен асинхронного завершения операции Asio является функцией обратного вызова, мы можем применить ту же тактику для Asio, что и для нашего гипотетического.AsyncAPIАсинхронные операции.
К счастью, нам это не нужно. Повышаю. Asio использует механизм, с помощью которого абонент может настроить поведение уведомления любой операции асинхронизации. Поэтому мы можем построитьтокен завершения, который при переходе наувеличивается. Asioасинхронизирует работу, запрашивает блокировку вызывающего волокна.
Типичная асинхронная функция Асио может выглядеть примерно так:
template < ..., class CompletionToken >
deduced_return_type
async_something( ... , CompletionToken&& token)
{
handler_type<CompletionToken, ...>::type handler(token);
async_result<decltype(handler)> result(handler);
return result.get();
}
Мы будем задействовать этот механизм, который основан на специализации Asio’shandler_type<>шаблон дляТокен завершениятипа и подпись конкретного обратного вызова. Остальная часть этого обсуждения будет ссылаться наasync_something()в качестве рассматриваемой функции Asio async.
В реализации, описанной ниже, используются средства более низкого уровня, чемобещаниеибудущее, потому чтообещаниемеханизм плохо взаимодействует сio_service::стоп. Он даетнарушенное обещаниеисключения.
бустер::волокна::асио::выходвыходявляется завершением токена такого рода.доходностьявляется примеромдоходность:
class yield_t {
public:
yield_t() = default;
yield_t operator[]( boost::system::error_code & ec) const {
yield_t tmp;
tmp.ec_ = & ec;
return tmp;
}
boost::system::error_code * ec_{ nullptr };
};
на самом деле только заполнитель, способ вызвать повышение. Азио кастомизация. Он может связыватьбустер::система::error_codeдля использования фактическим обработчиком.
доходностьобъявляется как:
thread_local yield_t yield{};
Настройка Asio осуществляется посредством специализацииboost::asio::handler_type<>дляyield_t:
template< typename ReturnType >
struct handler_type< fibers::asio::yield_t, ReturnType( boost::system::error_code) >
{ typedef fibers::asio::detail::yield_handler< void > type; };
(На самом деле вDetail/yield.hppесть четыре различные специализации, по одной для каждой из четырех подписей Asio async callback, которые мы ожидаем.)
Вышеуказанное предписывает Asio использоватьyield_handlerв качестве фактического обработчика для операции асинхронизации, к которойдоходностьпередается. Там’s дженерикyield_handler<T>реализация иyield_handler<пустота>специализация. Начнем с<void>специализация:
template<>
class yield_handler< void >: public yield_handler_base {
public:
explicit yield_handler( yield_t const& y) :
yield_handler_base{ y } {
}
void operator()() {
( * this)( boost::system::error_code() );
}
using yield_handler_base::operator();
};
async_something, проконсультировавшись сhandler_type<>специализацией признаков, представляет собойyield_handler<void>, который будет передан в качестве фактического обратного вызова для операции асинхронизации.yield_handler& #8217;s конструктор принимаетyield_tэкземплярyieldобъект передан функции асинхронизации] и передает его вyield_handler_base:
class yield_handler_base {
public:
yield_handler_base( yield_t const& y) :
ctx_{ boost::fibers::context::active() },
yt_( y ) {
}
void operator()( boost::system::error_code const& ec) {
BOOST_ASSERT_MSG( ycomp_,
"Must inject yield_completion* "
"before calling yield_handler_base::operator()()");
BOOST_ASSERT_MSG( yt_.ec_,
"Must inject boost::system::error_code* "
"before calling yield_handler_base::operator()()");
yield_completion::lock_t lk{ ycomp_->mtx_ };
ycomp_->completed_ = true;
* yt_.ec_ = ec;
if ( fibers::context::active() != ctx_ ) {
fibers::context::active()->set_ready( ctx_);
}
}
boost::fibers::context * ctx_;
yield_t yt_;
yield_completion * ycomp_{ nullptr };
};
yield_handler_baseхранит копиюyield_tэкземпляра — который, как показано выше, содержит толькоerror_code*. Он также захватывает контекст* для текущего волокна, вызывая контекст::active().
Вы заметите, чтоyield_handler_baseимеет еще один элемент данныхycomp_, который инициализируется доnullptrего конструктором и #8212; хотя егооператор[]метод опирается наycomp_, являющийся ненулевым. Подробнее об этом в один момент.
Сконструировавyield_handler<void>,async_something[]async_resulthandler_typeтипа: в этом случаеasync_result<250]void. Он передаетyield_handler<void>экземпляр новомуasync_resultэкземпляру.
template<>
class async_result< boost::fibers::asio::detail::yield_handler< void > > :
public boost::fibers::asio::detail::async_result_base {
public:
typedef void type;
explicit async_result( boost::fibers::asio::detail::yield_handler< void > & h):
boost::fibers::asio::detail::async_result_base{ h } {
}
};
Естественно, это приводит нас прямо кasync_result_base:
class async_result_base {
public:
explicit async_result_base( yield_handler_base & h) {
h.ycomp_ = & this->ycomp_;
if ( ! h.yt_.ec_) {
h.yt_.ec_ = & ec_;
}
}
void get() {
ycomp_.wait();
if ( ec_) {
throw_exception( boost::system::system_error{ ec_ } );
}
}
private:
boost::system::error_code ec_{};
yield_completion ycomp_{};
};
Вот какyield_handler_base::ycomp_становится ненулевым:async_result_base& #8217; конструктор вводит указатель обратно в свойyield_completionчлен.
Напомним, что каноническийвыход_tэкземплярвыходинициализирует свойошибку_код*членec_донульприт. Если этот экземпляр передаетсяasync_Something[]по-прежнемуnullptr, копия, хранящаяся вyield_handler_base, также будет иметь нульec_.async_result_base& #8217;s yield_handler_base& #8217;sec_ошибка_codeчлен.
Сейчас сцена установлена.async_something[]инициирует фактическую операцию асинхронизации, организуя ееyield_handler<void>экземпляр по завершении. Let’s говорят, ради аргумента, что фактическая операция асинхронизации’s callback имеет подписьvoiderror_code.
Но так как это’s асинхронная операция, управление возвращается сразу кasync_Something.async_Somethingвызываетasync_result<yield_handler<void>>::получает(]и вернет свое возвращаемое значение.
async_result<yield_handler<void>>::getнаследуетasync_result_base::get.
async_result_base::получить()немедленно вызываетвыход_завершение::ждать().
struct yield_completion {
typedef fibers::detail::spinlock mutex_t;
typedef std::unique_lock< mutex_t > lock_t;
mutex_t mtx_{};
bool completed_{ false };
void wait() {
lock_t lk{ mtx_ };
if ( ! completed_) {
fibers::context::active()->suspend( lk);
}
}
};
Предполагая, что ожидающая операция асинхронизации еще не завершена,yield_completion::completed_по-прежнему будетложным, аждать[]вызоветконтекст::suspend()на текущем волокне.
Другие волокна теперь будут иметь возможность работать.
Спустя некоторое время асинхронная операция завершается. Он называетyield_handler<void>::оператор()error_codeconst&]с ошибкой_code, указывающей либо на успех, либо на неудачу. Мы и #8217 рассмотрим оба случая.
yield_handler<void>эксплицитно наследуетоператор()error_codeconst&]отyield_handler_base.
yield_handler_base::оператор()error_codeconst&]первые наборыyield_completion::завершено_истинно. Таким образом, еслиasync_что-то()& #8217; операция асинхронизации завершается немедленно & #8212; еслиyield_handler_base::оператор()называется еще до того, какasync_result_base::[]& #8212; вызывающее волокнонеприостанавливается.
Фактическийerror_code, создаваемый операцией асинхронизации, затем хранится через сохраненныйуказатель yield_t::ec_. Еслиasync_Something[]& #8217;s caller используется (например)выходmy_ecдля связывания локальногоerror_codeэкземпляра, фактическое значениеerror_codeсохраняется в переменной caller’. В противном случае он хранится вasync_result_base::ec_.
Если сохраненный контекст волокнаyield_handler_base::ctx_уже не работает, он помечается как готовый к запуску, передавая его в контекст::set_ready(). Затем управление возвращается отyield_handler_base::оператор[]: Обратный звонок закончен.
Со временем это волокно возобновляется. Контроль возвращается из контекста::suspend()вyield_completion::Wait(), который возвращается вasync_result_base::get().
- Если первоначальный абонент передал
выходmy_ecasync_что-тодля связывания локальногоerror_code, товыход_handler_base::оператор[]сохранил свойerror_codeдля абонента’smy_ec, оставивasync_result_base::ec_инициализированным к успеху. - Если первоначальный абонент передал
, тоasync_Something[]без привязки к локальномуerror_codeпеременная, затемyield_handler_base::оператор[]сохранил свойerror_codeвasync_result_base::ec_. Еслиошибка_кодявляется успехом, то все хорошо. - В противном случае — первоначальный абонент не связывал локальный
error_codeиyield_handler_base::оператор[]был вызванerror_codeс указанием ошибки —async_result_base::получить[]бросаетsystem_errorс этимerror_code.
Случай, в которомasync_something[]& #8217; завершение обратного вызова имеет сигнатуруvoid.yield_handler<void>::оператор()вызывает механизм выше с& #8220;успех& #8221;ошибка_code.
Завершение обратного вызова с подписьюvoiderror_code,T(то есть: помимоerror_code, обратный вызов получает некоторый элемент данных) обрабатывается несколько иначе. Для этого вида подписиhandler_type<>::typeуказываетyield_handler<T>(дляT, кромеvoid.
Ayield_handler<T>резервируетзначение_указатель на значение типаT:
template< typename T >
class yield_handler: public yield_handler_base {
public:
explicit yield_handler( yield_t const& y) :
yield_handler_base{ y } {
}
void operator()( T t) {
(*this)( boost::system::error_code(), std::move(t) );
}
void operator()( boost::system::error_code const& ec, T t) {
BOOST_ASSERT_MSG( value_,
"Must inject value ptr "
"before caling yield_handler<T>::operator()()");
* value_ = std::move( t);
yield_handler_base::operator()( ec);
}
T * value_{ nullptr };
};
Этот указатель инициализируется доnullptr.
Когдаasync_somethingмгновенноasync_reult<yield_handler<T>>:
template< typename T >
class async_result< boost::fibers::asio::detail::yield_handler< T > > :
public boost::fibers::asio::detail::async_result_base {
public:
typedef T type;
explicit async_result( boost::fibers::asio::detail::yield_handler< T > & h) :
boost::fibers::asio::detail::async_result_base{ h } {
h.value_ = & value_;
}
type get() {
boost::fibers::asio::detail::async_result_base::get();
return std::move( value_);
}
private:
type value_{};
};
Этоaync_result<>специализация резервирует член типаTдля получения переданного элемента данных и устанавливаетвыход_handler<T>::значение_для указания на его собственный член данных.
async_result<yield_handler<T>>переопределенияполучают. Оверрайд-звонкиasync_result_base::получают(), поэтому вызывающее волокно приостанавливается, как описано выше.
yield_handler<>::оператор[] []error_code,Tсохраняет свое пройденноеTзначение вasync_result<T>>]::значение_.
Затем он передает управлениеyield_handler_base::оператор()error_codeдля борьбы с пробуждением исходного волокна, как описано выше.
Когдаaync_result<yield_handler<T>>::[]возобновляется, он возвращает сохраненноезначение_async_somethingи, в конечном счете,async_something[8217].
Случай обратного вызова подписиvoidобрабатывается, имеяyield_handler<T>:оператор[]Tзадействуютvoidошибку_codeмашины, пройдя& #8220;успех& #8221;ошибку_code.
Приведенный выше исходный код содержится вyield.hppиdetail/yield.hpp.