123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230 |
- // 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_DETAIL_WINDOWS_PIPE_HPP
- #define BOOST_PROCESS_DETAIL_WINDOWS_PIPE_HPP
- #include <boost/winapi/basic_types.hpp>
- #include <boost/winapi/error_codes.hpp>
- #include <boost/winapi/pipes.hpp>
- #include <boost/winapi/handles.hpp>
- #include <boost/winapi/file_management.hpp>
- #include <boost/winapi/get_last_error.hpp>
- #include <boost/winapi/access_rights.hpp>
- #include <boost/winapi/process.hpp>
- #include <boost/process/detail/windows/compare_handles.hpp>
- #include <system_error>
- #include <string>
- namespace boost { namespace process { namespace detail { namespace windows {
- template<class CharT, class Traits = std::char_traits<CharT>>
- class basic_pipe
- {
- ::boost::winapi::HANDLE_ _source = ::boost::winapi::INVALID_HANDLE_VALUE_;
- ::boost::winapi::HANDLE_ _sink = ::boost::winapi::INVALID_HANDLE_VALUE_;
- public:
- typedef CharT char_type ;
- typedef Traits traits_type;
- typedef typename Traits::int_type int_type ;
- typedef typename Traits::pos_type pos_type ;
- typedef typename Traits::off_type off_type ;
- typedef ::boost::winapi::HANDLE_ native_handle_type;
- explicit basic_pipe(::boost::winapi::HANDLE_ source, ::boost::winapi::HANDLE_ sink)
- : _source(source), _sink(sink) {}
- inline explicit basic_pipe(const std::string & name);
- inline basic_pipe(const basic_pipe& p);
- basic_pipe(basic_pipe&& lhs) : _source(lhs._source), _sink(lhs._sink)
- {
- lhs._source = ::boost::winapi::INVALID_HANDLE_VALUE_;
- lhs._sink = ::boost::winapi::INVALID_HANDLE_VALUE_;
- }
- inline basic_pipe& operator=(const basic_pipe& p);
- inline basic_pipe& operator=(basic_pipe&& lhs);
- ~basic_pipe()
- {
- if (_sink != ::boost::winapi::INVALID_HANDLE_VALUE_)
- ::boost::winapi::CloseHandle(_sink);
- if (_source != ::boost::winapi::INVALID_HANDLE_VALUE_)
- ::boost::winapi::CloseHandle(_source);
- }
- native_handle_type native_source() const {return _source;}
- native_handle_type native_sink () const {return _sink;}
- void assign_source(native_handle_type h) { _source = h;}
- void assign_sink (native_handle_type h) { _sink = h;}
- basic_pipe()
- {
- if (!::boost::winapi::CreatePipe(&_source, &_sink, nullptr, 0))
- throw_last_error("CreatePipe() failed");
- }
- int_type write(const char_type * data, int_type count)
- {
- ::boost::winapi::DWORD_ write_len;
- if (!::boost::winapi::WriteFile(
- _sink, data, count * sizeof(char_type), &write_len, nullptr
- ))
- {
- auto ec = ::boost::process::detail::get_last_error();
- if ((ec.value() == ::boost::winapi::ERROR_BROKEN_PIPE_) ||
- (ec.value() == ::boost::winapi::ERROR_NO_DATA_))
- return 0;
- else
- throw process_error(ec, "WriteFile failed");
- }
- return static_cast<int_type>(write_len);
- }
- int_type read(char_type * data, int_type count)
- {
- ::boost::winapi::DWORD_ read_len;
- if (!::boost::winapi::ReadFile(
- _source, data, count * sizeof(char_type), &read_len, nullptr
- ))
- {
- auto ec = ::boost::process::detail::get_last_error();
- if ((ec.value() == ::boost::winapi::ERROR_BROKEN_PIPE_) ||
- (ec.value() == ::boost::winapi::ERROR_NO_DATA_))
- return 0;
- else
- throw process_error(ec, "ReadFile failed");
- }
- return static_cast<int_type>(read_len);
- }
- bool is_open() const
- {
- return (_source != ::boost::winapi::INVALID_HANDLE_VALUE_) ||
- (_sink != ::boost::winapi::INVALID_HANDLE_VALUE_);
- }
- void close()
- {
- ::boost::winapi::CloseHandle(_source);
- ::boost::winapi::CloseHandle(_sink);
- _source = ::boost::winapi::INVALID_HANDLE_VALUE_;
- _sink = ::boost::winapi::INVALID_HANDLE_VALUE_;
- }
- };
- template<class Char, class Traits>
- basic_pipe<Char, Traits>::basic_pipe(const basic_pipe & p)
- {
- auto proc = ::boost::winapi::GetCurrentProcess();
- if (p._source == ::boost::winapi::INVALID_HANDLE_VALUE_)
- _source = ::boost::winapi::INVALID_HANDLE_VALUE_;
- else if (!::boost::winapi::DuplicateHandle(
- proc, p._source, proc, &_source, 0,
- static_cast<::boost::winapi::BOOL_>(true),
- ::boost::winapi::DUPLICATE_SAME_ACCESS_))
- throw_last_error("Duplicate Pipe Failed");
- if (p._sink == ::boost::winapi::INVALID_HANDLE_VALUE_)
- _sink = ::boost::winapi::INVALID_HANDLE_VALUE_;
- else if (!::boost::winapi::DuplicateHandle(
- proc, p._sink, proc, &_sink, 0,
- static_cast<::boost::winapi::BOOL_>(true),
- ::boost::winapi::DUPLICATE_SAME_ACCESS_))
- throw_last_error("Duplicate Pipe Failed");
- }
- template<class Char, class Traits>
- basic_pipe<Char, Traits>::basic_pipe(const std::string & name)
- {
- static constexpr int OPEN_EXISTING_ = 3; //temporary.
- static constexpr int FILE_FLAG_OVERLAPPED_ = 0x40000000; //temporary
- //static constexpr int FILE_ATTRIBUTE_NORMAL_ = 0x00000080; //temporary
- #if BOOST_NO_ANSI_APIS
- std::wstring name_ = boost::process::detail::convert(name);
- #else
- auto &name_ = name;
- #endif
- ::boost::winapi::HANDLE_ source = ::boost::winapi::create_named_pipe(
- name_.c_str(),
- ::boost::winapi::PIPE_ACCESS_INBOUND_
- | FILE_FLAG_OVERLAPPED_, //write flag
- 0, ::boost::winapi::PIPE_UNLIMITED_INSTANCES_, 8192, 8192, 0, nullptr);
- if (source == boost::winapi::INVALID_HANDLE_VALUE_)
- ::boost::process::detail::throw_last_error("create_named_pipe() failed");
- ::boost::winapi::HANDLE_ sink = boost::winapi::create_file(
- name.c_str(),
- ::boost::winapi::GENERIC_WRITE_, 0, nullptr,
- OPEN_EXISTING_,
- FILE_FLAG_OVERLAPPED_, //to allow read
- nullptr);
- if (sink == ::boost::winapi::INVALID_HANDLE_VALUE_)
- ::boost::process::detail::throw_last_error("create_file() failed");
- _source = source;
- _sink = sink;
- }
- template<class Char, class Traits>
- basic_pipe<Char, Traits>& basic_pipe<Char, Traits>::operator=(const basic_pipe & p)
- {
- auto proc = ::boost::winapi::GetCurrentProcess();
- if (p._source == ::boost::winapi::INVALID_HANDLE_VALUE_)
- _source = ::boost::winapi::INVALID_HANDLE_VALUE_;
- else if (!::boost::winapi::DuplicateHandle(
- proc, p._source, proc, &_source, 0,
- static_cast<::boost::winapi::BOOL_>(true),
- ::boost::winapi::DUPLICATE_SAME_ACCESS_))
- throw_last_error("Duplicate Pipe Failed");
- if (p._sink == ::boost::winapi::INVALID_HANDLE_VALUE_)
- _sink = ::boost::winapi::INVALID_HANDLE_VALUE_;
- else if (!::boost::winapi::DuplicateHandle(
- proc, p._sink, proc, &_sink, 0,
- static_cast<::boost::winapi::BOOL_>(true),
- ::boost::winapi::DUPLICATE_SAME_ACCESS_))
- throw_last_error("Duplicate Pipe Failed");
- return *this;
- }
- template<class Char, class Traits>
- basic_pipe<Char, Traits>& basic_pipe<Char, Traits>::operator=(basic_pipe && lhs)
- {
- if (_source != ::boost::winapi::INVALID_HANDLE_VALUE_)
- ::boost::winapi::CloseHandle(_source);
- if (_sink != ::boost::winapi::INVALID_HANDLE_VALUE_)
- ::boost::winapi::CloseHandle(_sink);
- _source = lhs._source;
- _sink = lhs._sink;
- lhs._source = ::boost::winapi::INVALID_HANDLE_VALUE_;
- lhs._sink = ::boost::winapi::INVALID_HANDLE_VALUE_;
- return *this;
- }
- template<class Char, class Traits>
- inline bool operator==(const basic_pipe<Char, Traits> & lhs, const basic_pipe<Char, Traits> & rhs)
- {
- return compare_handles(lhs.native_source(), rhs.native_source()) &&
- compare_handles(lhs.native_sink(), rhs.native_sink());
- }
- template<class Char, class Traits>
- inline bool operator!=(const basic_pipe<Char, Traits> & lhs, const basic_pipe<Char, Traits> & rhs)
- {
- return !compare_handles(lhs.native_source(), rhs.native_source()) ||
- !compare_handles(lhs.native_sink(), rhs.native_sink());
- }
- }}}}
- #endif
|