123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172 |
- // 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)
- #ifndef BOOST_PROCESS_WINDOWS_IO_CONTEXT_REF_HPP_
- #define BOOST_PROCESS_WINDOWS_IO_CONTEXT_REF_HPP_
- #include <boost/process/v1/detail/handler_base.hpp>
- #include <boost/process/v1/detail/windows/async_handler.hpp>
- #include <boost/process/v1/detail/windows/is_running.hpp>
- #include <boost/asio/io_context.hpp>
- #include <boost/asio/post.hpp>
- #include <boost/asio/windows/object_handle.hpp>
- #include <boost/winapi/process.hpp>
- #include <boost/winapi/handles.hpp>
- #include <boost/fusion/algorithm/iteration/for_each.hpp>
- #include <boost/fusion/algorithm/transformation/filter_if.hpp>
- #include <boost/fusion/algorithm/transformation/transform.hpp>
- #include <boost/fusion/view/transform_view.hpp>
- #include <boost/fusion/container/vector/convert.hpp>
- #include <functional>
- #include <type_traits>
- #include <memory>
- #include <atomic>
- #include <vector>
- #include <boost/type_index.hpp>
- namespace boost { namespace process { BOOST_PROCESS_V1_INLINE namespace v1 { namespace detail { namespace windows {
- template<typename Executor>
- struct on_exit_handler_transformer
- {
- Executor & exec;
- on_exit_handler_transformer(Executor & exec) : exec(exec) {}
- template<typename Sig>
- struct result;
- template<typename T>
- struct result<on_exit_handler_transformer<Executor>(T&)>
- {
- typedef typename T::on_exit_handler_t type;
- };
- template<typename T>
- auto operator()(T& t) const -> typename T::on_exit_handler_t
- {
- return t.on_exit_handler(exec);
- }
- };
- template<typename Executor>
- struct async_handler_collector
- {
- Executor & exec;
- std::vector<std::function<void(int, const std::error_code & ec)>> &handlers;
- async_handler_collector(Executor & exec,
- std::vector<std::function<void(int, const std::error_code & ec)>> &handlers)
- : exec(exec), handlers(handlers) {}
- template<typename T>
- void operator()(T & t) const
- {
- handlers.push_back(t.on_exit_handler(exec));
- }
- };
- //Also set's up waiting for the exit, so it can close async stuff.
- struct io_context_ref : boost::process::v1::detail::handler_base
- {
- io_context_ref(boost::asio::io_context & ios)
- : ios(ios)
- {
- }
- boost::asio::io_context &get() {return ios;};
- template <class Executor>
- void on_success(Executor& exec) const
- {
- auto asyncs = boost::fusion::filter_if<
- is_async_handler<
- typename std::remove_reference< boost::mpl::_ > ::type
- >>(exec.seq);
- //ok, check if there are actually any.
- if (boost::fusion::empty(asyncs))
- {
- return;
- }
- ::boost::winapi::PROCESS_INFORMATION_ & proc = exec.proc_info;
- auto this_proc = ::boost::winapi::GetCurrentProcess();
- auto proc_in = proc.hProcess;;
- ::boost::winapi::HANDLE_ process_handle;
- if (!::boost::winapi::DuplicateHandle(
- this_proc, proc_in, this_proc, &process_handle, 0,
- static_cast<::boost::winapi::BOOL_>(true),
- ::boost::winapi::DUPLICATE_SAME_ACCESS_))
- exec.set_error(::boost::process::v1::detail::get_last_error(),
- "Duplicate Pipe Failed");
- std::vector<std::function<void(int, const std::error_code & ec)>> funcs;
- funcs.reserve(boost::fusion::size(asyncs));
- boost::fusion::for_each(asyncs, async_handler_collector<Executor>(exec, funcs));
- wait_handler wh(std::move(funcs), ios, process_handle, exec.exit_status);
- ::boost::winapi::DWORD_ code;
- if(::boost::winapi::GetExitCodeProcess(process_handle, &code)
- && code != still_active)
- {
- ::boost::asio::post(wh.handle->get_executor(), std::move(wh));
- return;
- }
- auto handle_p = wh.handle.get();
- handle_p->async_wait(std::move(wh));
- }
- struct wait_handler
- {
- std::vector<std::function<void(int, const std::error_code & ec)>> funcs;
- std::unique_ptr<boost::asio::windows::object_handle> handle;
- std::shared_ptr<std::atomic<int>> exit_status;
- wait_handler(const wait_handler & ) = delete;
- wait_handler(wait_handler && ) = default;
- wait_handler(std::vector<std::function<void(int, const std::error_code & ec)>> && funcs,
- boost::asio::io_context & ios, void * handle,
- const std::shared_ptr<std::atomic<int>> &exit_status)
- : funcs(std::move(funcs)),
- handle(new boost::asio::windows::object_handle(
- asio::prefer(ios.get_executor(), asio::execution::outstanding_work.tracked), handle)),
- exit_status(exit_status)
- {
- }
- void operator()(const boost::system::error_code & ec_in = {})
- {
- std::error_code ec;
- if (ec_in)
- ec = std::error_code(ec_in.value(), std::system_category());
- ::boost::winapi::DWORD_ code;
- ::boost::winapi::GetExitCodeProcess(handle->native_handle(), &code);
- exit_status->store(code);
- for (auto & func : funcs)
- func(code, ec);
- }
- };
- private:
- boost::asio::io_context &ios;
- };
- }}}}}
- #endif /* BOOST_PROCESS_WINDOWS_IO_CONTEXT_REF_HPP_ */
|