// Copyright (c) 2006, 2007 Julio M. Merino Vidal // Copyright (c) 2008 Ilya Sokolov, Boris Schaeling // Copyright (c) 2009 Boris Schaeling // Copyright (c) 2010 Felipe Tanus, Boris Schaeling // Copyright (c) 2011, 2012 Jeff Flinn, Boris Schaeling // // 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_DETAIL_POSIX_WAIT_FOR_EXIT_HPP #define BOOST_PROCESS_DETAIL_POSIX_WAIT_FOR_EXIT_HPP #include #include #include #include #include #include namespace boost { namespace process { namespace detail { namespace posix { inline void wait(const child_handle &p, int & exit_code) { pid_t ret; int status; do { ret = ::waitpid(p.pid, &status, 0); } while (((ret == -1) && (errno == EINTR)) || (ret != -1 && !WIFEXITED(status) && !WIFSIGNALED(status))); if (ret == -1) boost::process::detail::throw_last_error("waitpid(2) failed"); if (WIFSIGNALED(status)) throw process_error(std::error_code(), "process terminated due to receipt of a signal"); exit_code = status; } inline void wait(const child_handle &p, int & exit_code, std::error_code &ec) noexcept { pid_t ret; int status; do { ret = ::waitpid(p.pid, &status, 0); } while (((ret == -1) && (errno == EINTR)) || (ret != -1 && !WIFEXITED(status) && !WIFSIGNALED(status))); if (ret == -1) ec = boost::process::detail::get_last_error(); else { ec.clear(); exit_code = status; } } template< class Rep, class Period > inline bool wait_for( const child_handle &p, int & exit_code, const std::chrono::duration& rel_time) { pid_t ret; int status; auto start = std::chrono::system_clock::now(); auto time_out = start + rel_time; bool timed_out; do { ret = ::waitpid(p.pid, &status, WNOHANG); if (ret == 0) { timed_out = std::chrono::system_clock::now() >= time_out; if (timed_out) return false; } } while ((ret == 0) || ((ret == -1) && errno == EINTR) || ((ret != -1) && !WIFEXITED(status) && !WIFSIGNALED(status))); if (ret == -1) boost::process::detail::throw_last_error("waitpid(2) failed"); exit_code = status; return true; } template< class Rep, class Period > inline bool wait_for( const child_handle &p, int & exit_code, const std::chrono::duration& rel_time, std::error_code & ec) noexcept { pid_t ret; int status; auto start = std::chrono::system_clock::now(); auto time_out = start + rel_time; bool timed_out; do { ret = ::waitpid(p.pid, &status, WNOHANG); if (ret == 0) { timed_out = std::chrono::system_clock::now() >= time_out; if (timed_out) return false; } } while ((ret == 0) || (((ret == -1) && errno == EINTR) || ((ret != -1) && !WIFEXITED(status) && !WIFSIGNALED(status)))); if (timed_out && (ret == -1)) return false; if (ret == -1) ec = boost::process::detail::get_last_error(); else { ec.clear(); exit_code = status; } return true; } template< class Clock, class Duration > inline bool wait_until( const child_handle &p, int & exit_code, const std::chrono::time_point& time_out) { pid_t ret; int status; bool timed_out; do { ret = ::waitpid(p.pid, &status, WNOHANG); if (ret == 0) { timed_out = std::chrono::system_clock::now() >= time_out; if (timed_out) return false; } } while ((ret == 0) || (((ret == -1) && errno == EINTR) || ((ret != -1) && !WIFEXITED(status) && !WIFSIGNALED(status)))); if (timed_out && !WIFEXITED(status)) return false; if (ret == -1) boost::process::detail::throw_last_error("waitpid(2) failed"); exit_code = status; return true; } template< class Clock, class Duration > inline bool wait_until( const child_handle &p, int & exit_code, const std::chrono::time_point& time_out, std::error_code & ec) noexcept { pid_t ret; int status; bool timed_out; do { ret = ::waitpid(p.pid, &status, WNOHANG); if (ret == 0) { timed_out = std::chrono::system_clock::now() >= time_out; if (timed_out) return false; } } while ((ret == 0) || (((ret == -1) && errno == EINTR) || ((ret != -1) && !WIFEXITED(status) && !WIFSIGNALED(status)))); if (timed_out && !WIFEXITED(status)) return false; if (ret == -1) ec = boost::process::detail::get_last_error(); else { ec.clear(); exit_code = status; } return true; } }}}} #endif