133 lines
3.6 KiB
C++
133 lines
3.6 KiB
C++
|
// Copyright (c) 2016 Klemens D. Morgenstern
|
||
|
//
|
||
|
// 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)
|
||
|
|
||
|
/** \file boost/process/async.hpp
|
||
|
|
||
|
The header which provides the basic asynchrounous features.
|
||
|
It provides the on_exit property, which allows callbacks when the process exits.
|
||
|
It also implements the necessary traits for passing an boost::asio::io_context,
|
||
|
which is needed for asynchronous communication.
|
||
|
|
||
|
It also pulls the [boost::asio::buffer](http://www.boost.org/doc/libs/release/doc/html/boost_asio/reference/buffer.html)
|
||
|
into the boost::process namespace for convenience.
|
||
|
|
||
|
\xmlonly
|
||
|
<programlisting>
|
||
|
namespace boost {
|
||
|
namespace process {
|
||
|
<emphasis>unspecified</emphasis> <ulink url="http://www.boost.org/doc/libs/1_61_0/doc/html/boost_asio/reference/buffer.html">buffer</ulink>;
|
||
|
<emphasis>unspecified</emphasis> <globalname alt="boost::process::on_exit">on_exit</globalname>;
|
||
|
}
|
||
|
}
|
||
|
</programlisting>
|
||
|
|
||
|
\endxmlonly
|
||
|
*/
|
||
|
|
||
|
#ifndef BOOST_PROCESS_ASYNC_HPP_
|
||
|
#define BOOST_PROCESS_ASYNC_HPP_
|
||
|
|
||
|
#include <boost/process/detail/traits.hpp>
|
||
|
#include <boost/process/detail/on_exit.hpp>
|
||
|
|
||
|
#include <boost/asio/io_context.hpp>
|
||
|
#include <boost/asio/streambuf.hpp>
|
||
|
#include <boost/asio/buffer.hpp>
|
||
|
#include <type_traits>
|
||
|
#include <boost/fusion/iterator/deref.hpp>
|
||
|
|
||
|
#if defined(BOOST_POSIX_API)
|
||
|
#include <boost/process/detail/posix/io_context_ref.hpp>
|
||
|
#include <boost/process/detail/posix/async_in.hpp>
|
||
|
#include <boost/process/detail/posix/async_out.hpp>
|
||
|
#include <boost/process/detail/posix/on_exit.hpp>
|
||
|
|
||
|
#elif defined(BOOST_WINDOWS_API)
|
||
|
#include <boost/process/detail/windows/io_context_ref.hpp>
|
||
|
#include <boost/process/detail/windows/async_in.hpp>
|
||
|
#include <boost/process/detail/windows/async_out.hpp>
|
||
|
#include <boost/process/detail/windows/on_exit.hpp>
|
||
|
#endif
|
||
|
|
||
|
namespace boost { namespace process { namespace detail {
|
||
|
|
||
|
struct async_tag;
|
||
|
|
||
|
template<typename T>
|
||
|
struct is_io_context : std::false_type {};
|
||
|
template<>
|
||
|
struct is_io_context<api::io_context_ref> : std::true_type {};
|
||
|
|
||
|
template<typename Tuple>
|
||
|
inline asio::io_context& get_io_context(const Tuple & tup)
|
||
|
{
|
||
|
auto& ref = *boost::fusion::find_if<is_io_context<boost::mpl::_>>(tup);
|
||
|
return ref.get();
|
||
|
}
|
||
|
|
||
|
struct async_builder
|
||
|
{
|
||
|
boost::asio::io_context * ios;
|
||
|
|
||
|
void operator()(boost::asio::io_context & ios_) {this->ios = &ios_;};
|
||
|
|
||
|
typedef api::io_context_ref result_type;
|
||
|
api::io_context_ref get_initializer() {return api::io_context_ref (*ios);};
|
||
|
};
|
||
|
|
||
|
|
||
|
template<>
|
||
|
struct initializer_builder<async_tag>
|
||
|
{
|
||
|
typedef async_builder type;
|
||
|
};
|
||
|
|
||
|
}
|
||
|
|
||
|
using ::boost::asio::buffer;
|
||
|
|
||
|
|
||
|
#if defined(BOOST_PROCESS_DOXYGEN)
|
||
|
/** When an io_context is passed, the on_exit property can be used, to be notified
|
||
|
when the child process exits.
|
||
|
|
||
|
|
||
|
The following syntax is valid
|
||
|
|
||
|
\code{.cpp}
|
||
|
on_exit=function;
|
||
|
on_exit(function);
|
||
|
\endcode
|
||
|
|
||
|
with `function` being a callable object with the signature `(int, const std::error_code&)` or an
|
||
|
`std::future<int>`.
|
||
|
|
||
|
\par Example
|
||
|
|
||
|
\code{.cpp}
|
||
|
io_context ios;
|
||
|
|
||
|
child c("ls", ios, on_exit=[](int exit, const std::error_code& ec_in){});
|
||
|
|
||
|
std::future<int> exit_code;
|
||
|
chlid c2("ls", ios, on_exit=exit_code);
|
||
|
|
||
|
\endcode
|
||
|
|
||
|
\note The handler is not invoked when the launch fails.
|
||
|
\warning When used \ref ignore_error it might get invoked on error.
|
||
|
\warning `on_exit` uses `boost::asio::signal_set` to listen for `SIGCHLD` on posix, and so has the
|
||
|
same restrictions as that class (do not register a handler for `SIGCHLD` except by using
|
||
|
`boost::asio::signal_set`).
|
||
|
*/
|
||
|
constexpr static ::boost::process::detail::on_exit_ on_exit{};
|
||
|
#endif
|
||
|
|
||
|
}}
|
||
|
|
||
|
|
||
|
|
||
|
#endif /* INCLUDE_BOOST_PROCESS_DETAIL_ASYNC_HPP_ */
|