PStreams
pstream.h
Go to the documentation of this file.
1// PStreams - POSIX Process I/O for C++
2
3// Copyright (C) 2001 - 2024 Jonathan Wakely
4// Distributed under the Boost Software License, Version 1.0.
5// (See accompanying file LICENSE_1_0.txt or copy at
6// http://www.boost.org/LICENSE_1_0.txt)
7//
8// SPDX-License-Identifier: BSL-1.0
9
19#ifndef REDI_PSTREAM_H_SEEN
20#define REDI_PSTREAM_H_SEEN
21
22#include <ios>
23#include <streambuf>
24#include <istream>
25#include <ostream>
26#include <string>
27#include <vector>
28#include <algorithm> // for min()
29#include <cerrno> // for errno
30#include <cstddef> // for size_t, NULL
31#include <cstdlib> // for exit()
32#include <sys/types.h> // for pid_t
33#include <sys/wait.h> // for waitpid()
34#include <sys/ioctl.h> // for ioctl() and FIONREAD
35#if defined(__sun)
36# include <sys/filio.h> // for FIONREAD on Solaris 2.5
37#endif
38#include <unistd.h> // for pipe() fork() exec() and filedes functions
39#include <signal.h> // for kill()
40#include <fcntl.h> // for fcntl()
41#if REDI_EVISCERATE_PSTREAMS
42# include <stdio.h> // for FILE, fdopen()
43#endif
44
45
47#define PSTREAMS_VERSION 0x0104 // 1.0.4
48
62namespace redi
63{
65 struct pstreams
66 {
68 typedef std::ios_base::openmode pmode;
69
71 typedef std::vector<std::string> argv_type;
72
74 typedef int fd_type;
75
76 static const pmode pstdin = std::ios_base::out;
77 static const pmode pstdout = std::ios_base::in;
78 static const pmode pstderr = std::ios_base::app;
79
81 static const pmode newpg = std::ios_base::trunc;
82
83 protected:
84 enum {
85 bufsz = 32,
86 pbsz = 2
87 };
88
89#if __cplusplus >= 201103L
90 template<typename T>
91 using stringable = decltype((void)std::string(std::declval<const T&>()));
92#endif
93 };
94
96 template <typename CharT, typename Traits = std::char_traits<CharT> >
98 : public std::basic_streambuf<CharT, Traits>
99 , public pstreams
100 {
101 public:
102 // Type definitions for dependent types
103 typedef CharT char_type;
104 typedef Traits traits_type;
105 typedef typename traits_type::int_type int_type;
106 typedef typename traits_type::off_type off_type;
107 typedef typename traits_type::pos_type pos_type;
109 typedef fd_type fd_t;
110
113
115 basic_pstreambuf(const std::string& cmd, pmode mode);
116
118 basic_pstreambuf( const std::string& file,
119 const argv_type& argv,
120 pmode mode );
121
122#if __cplusplus >= 201103L
124 basic_pstreambuf& operator=(basic_pstreambuf&&) noexcept;
125 void swap(basic_pstreambuf&) noexcept;
126#endif
127
130
133 open(const std::string& cmd, pmode mode);
134
137 open(const std::string& file, const argv_type& argv, pmode mode);
138
142
145 kill(int signal = SIGTERM);
146
149 killpg(int signal = SIGTERM);
150
152 void
154
156 bool
157 read_err(bool readerr = true);
158
160 bool
161 is_open() const;
162
164 bool
166
167#if REDI_EVISCERATE_PSTREAMS
169 std::size_t
170 fopen(FILE*& in, FILE*& out, FILE*& err);
171#endif
172
174 int
175 status() const;
176
178 int
179 error() const;
180
181 protected:
183 int_type
184 overflow(int_type c);
185
187 int_type
189
191 int_type
192 pbackfail(int_type c = traits_type::eof());
193
195 int
197
199 std::streamsize
200 xsputn(const char_type* s, std::streamsize n);
201
203 std::streamsize
204 write(const char_type* s, std::streamsize n);
205
207 std::streamsize
208 read(char_type* s, std::streamsize n);
209
211 std::streamsize
213
214 protected:
216 enum buf_read_src { rsrc_out = 0, rsrc_err = 1 };
217
219 pid_t
220 fork(pmode mode);
221
223 int
224 wait(bool nohang = false);
225
227 fd_type&
229
231 fd_type&
233
235 fd_type&
237
238 void
239 create_buffers(pmode mode);
240
241 void
242 destroy_buffers(pmode mode);
243
245 bool
247
248 bool
249 fill_buffer(bool non_blocking = false);
250
252 char_type*
254
256 switch_read_buffer(buf_read_src);
257
258 private:
259#if __cplusplus >= 201103L
260 using basic_streambuf = std::basic_streambuf<char_type, traits_type>;
261#else
263 basic_pstreambuf& operator=(const basic_pstreambuf&);
264#endif
265
266 void
267 init_rbuffers();
268
269 pid_t ppid_; // pid of process
270 fd_type wpipe_; // pipe used to write to process' stdin
271 fd_type rpipe_[2]; // two pipes to read from, stdout and stderr
272 char_type* wbuffer_;
273 char_type* rbuffer_[2];
274 char_type* rbufstate_[3];
276 buf_read_src rsrc_;
277 int status_; // hold exit status of child process
278 int error_; // hold errno if fork() or exec() fails
279 };
280
282 template <typename CharT, typename Traits = std::char_traits<CharT> >
284 : virtual public std::basic_ios<CharT, Traits>
285 , virtual public pstreams
286 {
287 protected:
289 typedef std::basic_ios<CharT, Traits> ios_type;
290
291 typedef pstreams::pmode pmode;
292 typedef pstreams::argv_type argv_type;
293
296
298 pstream_common(const std::string& cmd, pmode mode);
299
301 pstream_common(const std::string& file, const argv_type& argv, pmode mode);
302
304 virtual
306
307#if __cplusplus >= 201103L
308 pstream_common(pstream_common&& rhs) noexcept
309 : command_(std::move(rhs.command_))
310 , buf_(std::move(rhs.buf_))
311 {
312 /* derived class is responsible for ios_type::move(rhs) happening */
313 }
314
316 operator=(pstream_common&& rhs) noexcept
317 {
318 command_ = std::move(rhs.command_);
319 buf_ = std::move(rhs.buf_);
320 return *this;
321 }
322
323 void
324 swap(pstream_common& rhs) noexcept
325 {
326 /* derived class is responsible for ios_type::swap(rhs) happening */
327 command_.swap(rhs.command_);
328 buf_.swap(rhs.buf_);
329 }
330#endif // C++11
331
333 void
334 do_open(const std::string& cmd, pmode mode);
335
337 void
338 do_open(const std::string& file, const argv_type& argv, pmode mode);
339
340 public:
343 int
345
347 bool
348 is_open() const;
349
351 const std::string&
352 command() const;
353
356 rdbuf() const;
357
358#if REDI_EVISCERATE_PSTREAMS
360 std::size_t
361 fopen(FILE*& in, FILE*& out, FILE*& err);
362#endif
363
364 protected:
365 std::string command_;
367 };
368
369
380 template <typename CharT, typename Traits = std::char_traits<CharT> >
382 : public std::basic_istream<CharT, Traits>
383 , public pstream_common<CharT, Traits>
384 , virtual public pstreams
385 {
386 typedef std::basic_istream<CharT, Traits> istream_type;
388
389 using pbase_type::buf_; // declare name in this scope
390
391 // Ensure a basic_ipstream will read from at least one pipe
392 pmode readable(pmode mode)
393 {
394 if (!(mode & (pstdout|pstderr)))
395 mode |= pstdout;
396 return mode;
397 }
398
399 public:
401 typedef typename pbase_type::pmode pmode;
402
404 typedef typename pbase_type::argv_type argv_type;
405
408 : istream_type(NULL), pbase_type()
409 { }
410
421 explicit
422 basic_ipstream(const std::string& cmd, pmode mode = pstdout)
423 : istream_type(NULL), pbase_type(cmd, readable(mode))
424 { }
425
437 basic_ipstream( const std::string& file,
438 const argv_type& argv,
439 pmode mode = pstdout )
440 : istream_type(NULL), pbase_type(file, argv, readable(mode))
441 { }
442
453 explicit
455 : istream_type(NULL), pbase_type(argv.at(0), argv, readable(mode))
456 { }
457
458#if __cplusplus >= 201103L
459 template<typename T, typename = stringable<T>>
460 explicit
461 basic_ipstream(std::initializer_list<T> args, pmode mode = pstdout)
462 : basic_ipstream(argv_type(args.begin(), args.end()), mode)
463 { }
464
466 : istream_type(std::move(rhs))
467 , pbase_type(std::move(rhs))
468 { istream_type::set_rdbuf(std::addressof(pbase_type::buf_)); }
469
471 operator=(basic_ipstream&& rhs)
472 {
473 istream_type::operator=(std::move(rhs));
474 pbase_type::operator=(std::move(rhs));
475 return *this;
476 }
477
478 void
479 swap(basic_ipstream& rhs)
480 {
481 istream_type::swap(rhs);
482 pbase_type::swap(rhs);
483 }
484#endif // C++11
485
492 { }
493
503 void
504 open(const std::string& cmd, pmode mode = pstdout)
505 {
506 this->do_open(cmd, readable(mode));
507 }
508
519 void
520 open( const std::string& file,
521 const argv_type& argv,
522 pmode mode = pstdout )
523 {
524 this->do_open(file, argv, readable(mode));
525 }
526
533 {
534 this->buf_.read_err(false);
535 return *this;
536 }
537
544 {
545 this->buf_.read_err(true);
546 return *this;
547 }
548 };
549
550
560 template <typename CharT, typename Traits = std::char_traits<CharT> >
562 : public std::basic_ostream<CharT, Traits>
563 , public pstream_common<CharT, Traits>
564 , virtual public pstreams
565 {
566 typedef std::basic_ostream<CharT, Traits> ostream_type;
568
569 using pbase_type::buf_; // declare name in this scope
570
571 public:
573 typedef typename pbase_type::pmode pmode;
574
576 typedef typename pbase_type::argv_type argv_type;
577
580 : ostream_type(NULL), pbase_type()
581 { }
582
593 explicit
594 basic_opstream(const std::string& cmd, pmode mode = pstdin)
595 : ostream_type(NULL), pbase_type(cmd, mode|pstdin)
596 { }
597
609 basic_opstream( const std::string& file,
610 const argv_type& argv,
611 pmode mode = pstdin )
612 : ostream_type(NULL), pbase_type(file, argv, mode|pstdin)
613 { }
614
625 explicit
627 : ostream_type(NULL), pbase_type(argv.at(0), argv, mode|pstdin)
628 { }
629
630#if __cplusplus >= 201103L
638 template<typename T, typename = stringable<T>>
639 explicit
640 basic_opstream(std::initializer_list<T> args, pmode mode = pstdin)
641 : basic_opstream(argv_type(args.begin(), args.end()), mode)
642 { }
643
645 : ostream_type(std::move(rhs))
646 , pbase_type(std::move(rhs))
647 { ostream_type::set_rdbuf(std::addressof(pbase_type::buf_)); }
648
650 operator=(basic_opstream&& rhs)
651 {
652 ostream_type::operator=(std::move(rhs));
653 pbase_type::operator=(std::move(rhs));
654 return *this;
655 }
656
657 void
658 swap(basic_opstream& rhs)
659 {
660 ostream_type::swap(rhs);
661 pbase_type::swap(rhs);
662 }
663#endif // C++11
664
671
681 void
682 open(const std::string& cmd, pmode mode = pstdin)
683 {
684 this->do_open(cmd, mode|pstdin);
685 }
686
697 void
698 open( const std::string& file,
699 const argv_type& argv,
700 pmode mode = pstdin)
701 {
702 this->do_open(file, argv, mode|pstdin);
703 }
704 };
705
706
720 template <typename CharT, typename Traits = std::char_traits<CharT> >
722 : public std::basic_iostream<CharT, Traits>
723 , public pstream_common<CharT, Traits>
724 , virtual public pstreams
725 {
726 typedef std::basic_iostream<CharT, Traits> iostream_type;
728
729 using pbase_type::buf_; // declare name in this scope
730
731 public:
733 typedef typename pbase_type::pmode pmode;
734
736 typedef typename pbase_type::argv_type argv_type;
737
740 : iostream_type(NULL), pbase_type()
741 { }
742
753 explicit
754 basic_pstream(const std::string& cmd, pmode mode = pstdout|pstdin)
755 : iostream_type(NULL), pbase_type(cmd, mode)
756 { }
757
769 basic_pstream( const std::string& file,
770 const argv_type& argv,
771 pmode mode = pstdout|pstdin )
772 : iostream_type(NULL), pbase_type(file, argv, mode)
773 { }
774
785 explicit
787 : iostream_type(NULL), pbase_type(argv.at(0), argv, mode)
788 { }
789
790#if __cplusplus >= 201103L
798 template<typename T, typename = stringable<T>>
799 explicit
800 basic_pstream(std::initializer_list<T> l, pmode mode = pstdout|pstdin)
801 : basic_pstream(argv_type(l.begin(), l.end()), mode)
802 { }
803
805 : iostream_type(std::move(rhs))
806 , pbase_type(std::move(rhs))
807 { iostream_type::set_rdbuf(std::addressof(pbase_type::buf_)); }
808
810 operator=(basic_pstream&& rhs)
811 {
812 iostream_type::operator=(std::move(rhs));
813 pbase_type::operator=(std::move(rhs));
814 return *this;
815 }
816
817 void
818 swap(basic_pstream& rhs)
819 {
820 iostream_type::swap(rhs);
821 pbase_type::swap(rhs);
822 }
823#endif // C++11
824
831
841 void
842 open(const std::string& cmd, pmode mode = pstdout|pstdin)
843 {
844 this->do_open(cmd, mode);
845 }
846
857 void
858 open( const std::string& file,
859 const argv_type& argv,
860 pmode mode = pstdout|pstdin )
861 {
862 this->do_open(file, argv, mode);
863 }
864
871 {
872 this->buf_.read_err(false);
873 return *this;
874 }
875
882 {
883 this->buf_.read_err(true);
884 return *this;
885 }
886 };
887
888
910 template <typename CharT, typename Traits = std::char_traits<CharT> >
912 : public std::basic_ostream<CharT, Traits>
913 , private std::basic_istream<CharT, Traits>
914 , private pstream_common<CharT, Traits>
915 , virtual public pstreams
916 {
917 typedef std::basic_ostream<CharT, Traits> ostream_type;
918 typedef std::basic_istream<CharT, Traits> istream_type;
920
921 using pbase_type::buf_; // declare name in this scope
922
923 public:
925 typedef typename pbase_type::pmode pmode;
926
928 typedef typename pbase_type::argv_type argv_type;
929
932 : ostream_type(NULL), istream_type(NULL), pbase_type()
933 { }
934
945 explicit
946 basic_rpstream(const std::string& cmd, pmode mode = pstdout|pstdin)
947 : ostream_type(NULL) , istream_type(NULL) , pbase_type(cmd, mode)
948 { }
949
961 basic_rpstream( const std::string& file,
962 const argv_type& argv,
963 pmode mode = pstdout|pstdin )
964 : ostream_type(NULL), istream_type(NULL), pbase_type(file, argv, mode)
965 { }
966
977 explicit
979 : ostream_type(NULL), istream_type(NULL)
980 , pbase_type(argv.at(0), argv, mode)
981 { }
982
983#if __cplusplus >= 201103L
991 template<typename T, typename = stringable<T>>
992 explicit
993 basic_rpstream(std::initializer_list<T> l, pmode mode = pstdout|pstdin)
994 : basic_rpstream(argv_type(l.begin(), l.end()), mode)
995 { }
996
998 : ostream_type(NULL), istream_type(std::move(rhs))
999 , pbase_type(std::move(rhs))
1000 { istream_type::set_rdbuf(std::addressof(pbase_type::buf_)); }
1001
1003 operator=(basic_rpstream&& rhs)
1004 {
1005 swap(rhs);
1006 return *this;
1007 }
1008
1009 void
1010 swap(basic_rpstream& rhs)
1011 {
1012 istream_type::swap(rhs);
1013 pbase_type::swap(rhs);
1014 }
1015#endif // C++11
1016
1019
1029 void
1030 open(const std::string& cmd, pmode mode = pstdout|pstdin)
1031 {
1032 this->do_open(cmd, mode);
1033 }
1034
1045 void
1046 open( const std::string& file,
1047 const argv_type& argv,
1048 pmode mode = pstdout|pstdin )
1049 {
1050 this->do_open(file, argv, mode);
1051 }
1052
1058 istream_type&
1060 {
1061 this->buf_.read_err(false);
1062 return *this;
1063 }
1064
1070 istream_type&
1072 {
1073 this->buf_.read_err(true);
1074 return *this;
1075 }
1076 };
1077
1078
1089
1090
1103 template <typename C, typename T>
1104 inline std::basic_ostream<C,T>&
1105 peof(std::basic_ostream<C,T>& s)
1106 {
1107 typedef basic_pstreambuf<C,T> pstreambuf_type;
1108 if (pstreambuf_type* p = dynamic_cast<pstreambuf_type*>(s.rdbuf()))
1109 p->peof();
1110 return s;
1111 }
1112
1113
1114 /*
1115 * member definitions for pstreambuf
1116 */
1117
1118
1125 template <typename C, typename T>
1126 inline
1128 : ppid_(-1) // initialise to -1 to indicate no process run yet.
1129 , wpipe_(-1)
1130 , wbuffer_()
1131 , rbuffer_()
1132 , rbufstate_()
1133 , rsrc_(rsrc_out)
1134 , status_(-1)
1135 , error_(0)
1136 {
1137 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1138 }
1139
1148 template <typename C, typename T>
1149 inline
1151 : ppid_(-1) // initialise to -1 to indicate no process run yet.
1152 , wpipe_(-1)
1153 , wbuffer_()
1154 , rbuffer_()
1155 , rbufstate_()
1156 , rsrc_(rsrc_out)
1157 , status_(-1)
1158 , error_(0)
1159 {
1160 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1161 open(cmd, mode);
1162 }
1163
1173 template <typename C, typename T>
1174 inline
1176 const argv_type& argv,
1177 pmode mode )
1178 : ppid_(-1) // initialise to -1 to indicate no process run yet.
1179 , wpipe_(-1)
1180 , wbuffer_()
1181 , rbuffer_()
1182 , rbufstate_()
1183 , rsrc_(rsrc_out)
1184 , status_(-1)
1185 , error_(0)
1186 {
1187 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1188 open(file, argv, mode);
1189 }
1190
1195 template <typename C, typename T>
1196 inline
1198 {
1199 close();
1200 }
1201
1202#if __cplusplus >= 201103L
1206 template <typename C, typename T>
1207 inline
1209 : basic_streambuf(static_cast<const basic_streambuf&>(rhs))
1210 , ppid_(rhs.ppid_)
1211 , wpipe_(rhs.wpipe_)
1212 , rpipe_{rhs.rpipe_[0], rhs.rpipe_[1]}
1213 , wbuffer_(rhs.wbuffer_)
1214 , rbuffer_{rhs.rbuffer_[0], rhs.rbuffer_[1]}
1215 , rbufstate_{rhs.rbufstate_[0], rhs.rbufstate_[1], rhs.rbufstate_[2]}
1216 , rsrc_(rhs.rsrc_)
1217 , status_(rhs.status_)
1218 , error_(rhs.error_)
1219 {
1220 rhs.ppid_ = -1;
1221 rhs.wpipe_ = -1;
1222 rhs.rpipe_[0] = rhs.rpipe_[1] = -1;
1223 rhs.wbuffer_ = nullptr;
1224 rhs.rbuffer_[0] = rhs.rbuffer_[1] = nullptr;
1225 rhs.rbufstate_[0] = rhs.rbufstate_[1] = rhs.rbufstate_[2] = nullptr;
1226 rhs.rsrc_ = rsrc_out;
1227 rhs.status_ = -1;
1228 rhs.error_ = 0;
1229 rhs.setg(nullptr, nullptr, nullptr);
1230 rhs.setp(nullptr, nullptr);
1231 }
1232
1233 template <typename C, typename T>
1234 inline basic_pstreambuf<C,T>&
1235 basic_pstreambuf<C,T>::operator=( basic_pstreambuf&& rhs ) noexcept
1236 {
1237 close();
1238 basic_streambuf::operator=(static_cast<const basic_streambuf&>(rhs));
1239 swap(rhs);
1240 return *this;
1241 }
1242
1243 template <typename C, typename T>
1244 inline void
1245 basic_pstreambuf<C,T>::swap( basic_pstreambuf& rhs ) noexcept
1246 {
1247 basic_streambuf::swap(static_cast<basic_streambuf&>(rhs));
1248 std::swap(ppid_, rhs.ppid_);
1249 std::swap(wpipe_, rhs.wpipe_);
1250 std::swap(rpipe_, rhs.rpipe_);
1251 std::swap(wbuffer_, rhs.wbuffer_);
1252 std::swap(rbuffer_, rhs.rbuffer_);
1253 std::swap(rbufstate_, rhs.rbufstate_);
1254 std::swap(rsrc_, rhs.rsrc_);
1255 std::swap(status_, rhs.status_);
1256 std::swap(error_, rhs.error_);
1257 }
1258#endif // C++11
1259
1287 template <typename C, typename T>
1288 basic_pstreambuf<C,T>*
1289 basic_pstreambuf<C,T>::open(const std::string& command, pmode mode)
1290 {
1291 const char * shell_path = "/bin/sh";
1292#if 0
1293 const std::string argv[] = { "sh", "-c", command };
1294 return this->open(shell_path, argv_type(argv, argv+3), mode);
1295#else
1296 basic_pstreambuf<C,T>* ret = NULL;
1297
1298 if (!is_open())
1299 {
1300 switch(fork(mode))
1301 {
1302 case 0 :
1303 // this is the new process, exec command
1304 ::execl(shell_path, "sh", "-c", command.c_str(), (char*)NULL);
1305
1306 // can only reach this point if exec() failed
1307
1308 // parent can get exit code from waitpid()
1309 ::_exit(errno);
1310 // using std::exit() would make static dtors run twice
1311
1312 case -1 :
1313 // couldn't fork, error already handled in pstreambuf::fork()
1314 break;
1315
1316 default :
1317 // this is the parent process
1318 // activate buffers
1319 create_buffers(mode);
1320 ret = this;
1321 }
1322 }
1323 return ret;
1324#endif
1325 }
1326
1335 inline void
1337 {
1338 if (fd >= 0 && ::close(fd) == 0)
1339 fd = -1;
1340 }
1341
1352 template <int N>
1353 inline void
1355 {
1356 for (std::size_t i = 0; i < N; ++i)
1357 close_fd(fds[i]);
1358 }
1359
1389 template <typename C, typename T>
1391 basic_pstreambuf<C,T>::open( const std::string& file,
1392 const argv_type& argv,
1393 pmode mode )
1394 {
1395 basic_pstreambuf<C,T>* ret = NULL;
1396
1397 if (!is_open())
1398 {
1399 // constants for read/write ends of pipe
1400 enum { RD, WR };
1401
1402 // open another pipe and set close-on-exec
1403 fd_type ck_exec[] = { -1, -1 };
1404 if (-1 == ::pipe(ck_exec)
1405 || -1 == ::fcntl(ck_exec[RD], F_SETFD, FD_CLOEXEC)
1406 || -1 == ::fcntl(ck_exec[WR], F_SETFD, FD_CLOEXEC))
1407 {
1408 error_ = errno;
1409 close_fd_array(ck_exec);
1410 }
1411 else
1412 {
1413 switch(fork(mode))
1414 {
1415 case 0 :
1416 // this is the new process, exec command
1417 {
1418 char** arg_v = new char*[argv.size()+1];
1419 for (std::size_t i = 0; i < argv.size(); ++i)
1420 {
1421 const std::string& src = argv[i];
1422 char*& dest = arg_v[i];
1423 dest = new char[src.size()+1];
1424 dest[ src.copy(dest, src.size()) ] = '\0';
1425 }
1426 arg_v[argv.size()] = NULL;
1427
1428 ::execvp(file.c_str(), arg_v);
1429
1430 // can only reach this point if exec() failed
1431
1432 // parent can get error code from ck_exec pipe
1433 error_ = errno;
1434
1435 while (::write(ck_exec[WR], &error_, sizeof(error_)) == -1
1436 && errno == EINTR)
1437 { }
1438
1439 ::close(ck_exec[WR]);
1440 ::close(ck_exec[RD]);
1441
1442 ::_exit(error_);
1443 // using std::exit() would make static dtors run twice
1444 }
1445
1446 case -1 :
1447 // couldn't fork, error already handled in pstreambuf::fork()
1448 close_fd_array(ck_exec);
1449 break;
1450
1451 default :
1452 // this is the parent process
1453
1454 // check child called exec() successfully
1455 ::close(ck_exec[WR]);
1456 switch (::read(ck_exec[RD], &error_, sizeof(error_)))
1457 {
1458 case 0:
1459 // activate buffers
1460 create_buffers(mode);
1461 ret = this;
1462 break;
1463 case -1:
1464 error_ = errno;
1465 break;
1466 default:
1467 // error_ contains error code from child
1468 // call wait() to clean up and set ppid_ to 0
1469 this->wait();
1470 break;
1471 }
1472 ::close(ck_exec[RD]);
1473 }
1474 }
1475 }
1476 return ret;
1477 }
1478
1495 template <typename C, typename T>
1496 pid_t
1498 {
1499 pid_t pid = -1;
1500
1501 // Three pairs of file descriptors, for pipes connected to the
1502 // process' stdin, stdout and stderr
1503 // (stored in a single array so close_fd_array() can close all at once)
1504 fd_type fd[] = { -1, -1, -1, -1, -1, -1 };
1505 fd_type* const pin = fd;
1506 fd_type* const pout = fd+2;
1507 fd_type* const perr = fd+4;
1508
1509 // constants for read/write ends of pipe
1510 enum { RD, WR };
1511
1512 // N.B.
1513 // For the pstreambuf pin is an output stream and
1514 // pout and perr are input streams.
1515
1516 if (!error_ && mode&pstdin && ::pipe(pin))
1517 error_ = errno;
1518
1519 if (!error_ && mode&pstdout && ::pipe(pout))
1520 error_ = errno;
1521
1522 if (!error_ && mode&pstderr && ::pipe(perr))
1523 error_ = errno;
1524
1525 if (!error_)
1526 {
1527 pid = ::fork();
1528 switch (pid)
1529 {
1530 case 0 :
1531 {
1532 // this is the new process
1533
1534 // for each open pipe close one end and redirect the
1535 // respective standard stream to the other end
1536
1537 if (*pin >= 0)
1538 {
1539 ::close(pin[WR]);
1540 ::dup2(pin[RD], STDIN_FILENO);
1541 ::close(pin[RD]);
1542 }
1543 if (*pout >= 0)
1544 {
1545 ::close(pout[RD]);
1546 ::dup2(pout[WR], STDOUT_FILENO);
1547 ::close(pout[WR]);
1548 }
1549 if (*perr >= 0)
1550 {
1551 ::close(perr[RD]);
1552 ::dup2(perr[WR], STDERR_FILENO);
1553 ::close(perr[WR]);
1554 }
1555
1556#ifdef _POSIX_JOB_CONTROL
1557 if (mode&newpg)
1558 ::setpgid(0, 0); // Change to a new process group
1559#endif
1560
1561 break;
1562 }
1563 case -1 :
1564 {
1565 // couldn't fork for some reason
1566 error_ = errno;
1567 // close any open pipes
1568 close_fd_array(fd);
1569 break;
1570 }
1571 default :
1572 {
1573 // this is the parent process, store process' pid
1574 ppid_ = pid;
1575
1576 // store one end of open pipes and close other end
1577 if (*pin >= 0)
1578 {
1579 wpipe_ = pin[WR];
1580 ::close(pin[RD]);
1581 }
1582 if (*pout >= 0)
1583 {
1584 rpipe_[rsrc_out] = pout[RD];
1585 ::close(pout[WR]);
1586 }
1587 if (*perr >= 0)
1588 {
1589 rpipe_[rsrc_err] = perr[RD];
1590 ::close(perr[WR]);
1591 }
1592 }
1593 }
1594 }
1595 else
1596 {
1597 // close any pipes we opened before failure
1598 close_fd_array(fd);
1599 }
1600 return pid;
1601 }
1602
1612 template <typename C, typename T>
1615 {
1616 const bool running = is_open();
1617
1618 basic_pstreambuf::sync(); // might call wait() and reap the child process
1619
1620 // rather than trying to work out whether or not we need to clean up
1621 // just do it anyway, all cleanup functions are safe to call twice.
1622
1623 destroy_buffers(pstdin|pstdout|pstderr);
1624
1625 // close pipes before wait() so child gets EOF/SIGPIPE
1626 close_fd(wpipe_);
1627 close_fd_array(rpipe_);
1628
1629 do
1630 {
1631 error_ = 0;
1632 } while (wait() == -1 && error() == EINTR);
1633
1634 return running ? this : NULL;
1635 }
1636
1641 template <typename C, typename T>
1642#if __cplusplus >= 201402L && __has_cpp_attribute(deprecated)
1643 [[deprecated]]
1644#elif __GNUC__
1645 __attribute__((deprecated))
1646#endif
1647 inline void
1649 {
1650 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1651 rbuffer_[rsrc_out] = rbuffer_[rsrc_err] = NULL;
1652 rbufstate_[0] = rbufstate_[1] = rbufstate_[2] = NULL;
1653 }
1654
1655 template <typename C, typename T>
1656 void
1657 basic_pstreambuf<C,T>::create_buffers(pmode mode)
1658 {
1659 if (mode & pstdin)
1660 {
1661 delete[] wbuffer_;
1662 wbuffer_ = new char_type[bufsz];
1663 this->setp(wbuffer_, wbuffer_ + bufsz);
1664 }
1665 if (mode & pstdout)
1666 {
1667 delete[] rbuffer_[rsrc_out];
1668 rbuffer_[rsrc_out] = new char_type[bufsz];
1669 rsrc_ = rsrc_out;
1670 this->setg(rbuffer_[rsrc_out] + pbsz, rbuffer_[rsrc_out] + pbsz,
1671 rbuffer_[rsrc_out] + pbsz);
1672 }
1673 if (mode & pstderr)
1674 {
1675 delete[] rbuffer_[rsrc_err];
1676 rbuffer_[rsrc_err] = new char_type[bufsz];
1677 if (!(mode & pstdout))
1678 {
1679 rsrc_ = rsrc_err;
1680 this->setg(rbuffer_[rsrc_err] + pbsz, rbuffer_[rsrc_err] + pbsz,
1681 rbuffer_[rsrc_err] + pbsz);
1682 }
1683 }
1684 }
1685
1686 template <typename C, typename T>
1687 void
1688 basic_pstreambuf<C,T>::destroy_buffers(pmode mode)
1689 {
1690 if (mode & pstdin)
1691 {
1692 this->setp(NULL, NULL);
1693 delete[] wbuffer_;
1694 wbuffer_ = NULL;
1695 }
1696 if (mode & pstdout)
1697 {
1698 if (rsrc_ == rsrc_out)
1699 this->setg(NULL, NULL, NULL);
1700 delete[] rbuffer_[rsrc_out];
1701 rbuffer_[rsrc_out] = NULL;
1702 }
1703 if (mode & pstderr)
1704 {
1705 if (rsrc_ == rsrc_err)
1706 this->setg(NULL, NULL, NULL);
1707 delete[] rbuffer_[rsrc_err];
1708 rbuffer_[rsrc_err] = NULL;
1709 }
1710 }
1711
1712 template <typename C, typename T>
1714 basic_pstreambuf<C,T>::switch_read_buffer(buf_read_src src)
1715 {
1716 if (rsrc_ != src)
1717 {
1718 char_type* tmpbufstate[] = {this->eback(), this->gptr(), this->egptr()};
1719 this->setg(rbufstate_[0], rbufstate_[1], rbufstate_[2]);
1720 for (std::size_t i = 0; i < 3; ++i)
1721 rbufstate_[i] = tmpbufstate[i];
1722 rsrc_ = src;
1723 }
1724 return rsrc_;
1725 }
1726
1743 template <typename C, typename T>
1744 int
1746 {
1747 int child_exited = -1;
1748 if (is_open())
1749 {
1750 int exit_status;
1751 switch(::waitpid(ppid_, &exit_status, nohang ? WNOHANG : 0))
1752 {
1753 case 0 :
1754 // nohang was true and process has not exited
1755 child_exited = 0;
1756 break;
1757 case -1 :
1758 error_ = errno;
1759 break;
1760 default :
1761 // process has exited
1762 ppid_ = 0;
1763 status_ = exit_status;
1764 child_exited = 1;
1765 // Close wpipe, would get SIGPIPE if we used it.
1766 destroy_buffers(pstdin);
1767 close_fd(wpipe_);
1768 // Must free read buffers and pipes on destruction
1769 // or next call to open()/close()
1770 break;
1771 }
1772 }
1773 return child_exited;
1774 }
1775
1786 template <typename C, typename T>
1787 inline basic_pstreambuf<C,T>*
1789 {
1790 basic_pstreambuf<C,T>* ret = NULL;
1791 if (is_open())
1792 {
1793 if (::kill(ppid_, signal))
1794 error_ = errno;
1795 else
1796 {
1797#if 0
1798 // TODO call exited() to check for exit and clean up? leave to user?
1799 if (signal==SIGTERM || signal==SIGKILL)
1800 this->exited();
1801#endif
1802 ret = this;
1803 }
1804 }
1805 return ret;
1806 }
1807
1821 template <typename C, typename T>
1822 inline basic_pstreambuf<C,T>*
1824 {
1825 basic_pstreambuf<C,T>* ret = NULL;
1826#ifdef _POSIX_JOB_CONTROL
1827 if (is_open())
1828 {
1829 pid_t pgid = ::getpgid(ppid_);
1830 if (pgid == -1)
1831 error_ = errno;
1832 else if (pgid == ::getpgrp())
1833 error_ = EPERM; // Don't commit suicide
1834 else if (::killpg(pgid, signal))
1835 error_ = errno;
1836 else
1837 ret = this;
1838 }
1839#else
1840 error_ = ENOTSUP;
1841#endif
1842 return ret;
1843 }
1844
1852 template <typename C, typename T>
1853 inline bool
1855 {
1856 return ppid_ == 0 || wait(true) == 1;
1857 }
1858
1859
1865 template <typename C, typename T>
1866 inline int
1868 {
1869 return status_;
1870 }
1871
1875 template <typename C, typename T>
1876 inline int
1878 {
1879 return error_;
1880 }
1881
1886 template <typename C, typename T>
1887 inline void
1889 {
1890 sync();
1891 destroy_buffers(pstdin);
1892 close_fd(wpipe_);
1893 }
1894
1905 template <typename C, typename T>
1906 inline bool
1908 {
1909 return ppid_ > 0;
1910 }
1911
1920 template <typename C, typename T>
1921 inline bool
1923 {
1924 buf_read_src src = readerr ? rsrc_err : rsrc_out;
1925 if (rpipe_[src]>=0)
1926 {
1927 switch_read_buffer(src);
1928 return true;
1929 }
1930 return false;
1931 }
1932
1943 template <typename C, typename T>
1944 typename basic_pstreambuf<C,T>::int_type
1946 {
1947 if (!empty_buffer())
1948 return traits_type::eof();
1949 else if (!traits_type::eq_int_type(c, traits_type::eof()))
1950 return this->sputc(c);
1951 else
1952 return traits_type::not_eof(c);
1953 }
1954
1955
1956 template <typename C, typename T>
1957 int
1959 {
1960 return !exited() && empty_buffer() ? 0 : -1;
1961 }
1962
1968 template <typename C, typename T>
1969 std::streamsize
1970 basic_pstreambuf<C,T>::xsputn(const char_type* s, std::streamsize n)
1971 {
1972 std::streamsize done = 0;
1973 while (done < n)
1974 {
1975 if (std::streamsize nbuf = this->epptr() - this->pptr())
1976 {
1977 nbuf = std::min(nbuf, n - done);
1978 traits_type::copy(this->pptr(), s + done, nbuf);
1979 this->pbump(nbuf);
1980 done += nbuf;
1981 }
1982 else if (!empty_buffer())
1983 break;
1984 }
1985 return done;
1986 }
1987
1991 template <typename C, typename T>
1992 bool
1994 {
1995 const std::streamsize count = this->pptr() - this->pbase();
1996 if (count > 0)
1997 {
1998 const std::streamsize written = this->write(this->wbuffer_, count);
1999 if (written > 0)
2000 {
2001 if (const std::streamsize unwritten = count - written)
2002 traits_type::move(this->pbase(), this->pbase()+written, unwritten);
2003 this->pbump(-written);
2004 return true;
2005 }
2006 }
2007 return false;
2008 }
2009
2017 template <typename C, typename T>
2018 typename basic_pstreambuf<C,T>::int_type
2020 {
2021 if (this->gptr() < this->egptr() || fill_buffer())
2022 return traits_type::to_int_type(*this->gptr());
2023 else
2024 return traits_type::eof();
2025 }
2026
2035 template <typename C, typename T>
2036 typename basic_pstreambuf<C,T>::int_type
2038 {
2039 if (this->gptr() != this->eback())
2040 {
2041 this->gbump(-1);
2042 if (!traits_type::eq_int_type(c, traits_type::eof()))
2043 *this->gptr() = traits_type::to_char_type(c);
2044 return traits_type::not_eof(c);
2045 }
2046 else
2047 return traits_type::eof();
2048 }
2049
2050 template <typename C, typename T>
2051 std::streamsize
2053 {
2054 int avail = 0;
2055 if (sizeof(char_type) == 1)
2056 avail = fill_buffer(true) ? this->egptr() - this->gptr() : -1;
2057#ifdef FIONREAD
2058 else
2059 {
2060 if (::ioctl(rpipe(), FIONREAD, &avail) == -1)
2061 avail = -1;
2062 else if (avail)
2063 avail /= sizeof(char_type);
2064 }
2065#endif
2066 return std::streamsize(avail);
2067 }
2068
2072 template <typename C, typename T>
2073 bool
2075 {
2076 const std::streamsize pb1 = this->gptr() - this->eback();
2077 const std::streamsize pb2 = pbsz;
2078 const std::streamsize npb = std::min(pb1, pb2);
2079
2080 char_type* const rbuf = rbuffer();
2081
2082 if (npb)
2083 traits_type::move(rbuf + pbsz - npb, this->gptr() - npb, npb);
2084
2085 std::streamsize rc = -1;
2086
2087 if (non_blocking)
2088 {
2089 const int flags = ::fcntl(rpipe(), F_GETFL);
2090 if (flags != -1)
2091 {
2092 const bool blocking = !(flags & O_NONBLOCK);
2093 if (blocking)
2094 ::fcntl(rpipe(), F_SETFL, flags | O_NONBLOCK); // set non-blocking
2095
2096 error_ = 0;
2097 rc = read(rbuf + pbsz, bufsz - pbsz);
2098
2099 if (rc == -1 && error_ == EAGAIN) // nothing available
2100 rc = 0;
2101 else if (rc == 0) // EOF
2102 rc = -1;
2103
2104 if (blocking)
2105 ::fcntl(rpipe(), F_SETFL, flags); // restore
2106 }
2107 }
2108 else
2109 rc = read(rbuf + pbsz, bufsz - pbsz);
2110
2111 if (rc > 0 || (rc == 0 && non_blocking))
2112 {
2113 this->setg( rbuf + pbsz - npb,
2114 rbuf + pbsz,
2115 rbuf + pbsz + rc );
2116 return true;
2117 }
2118 else
2119 {
2120 this->setg(NULL, NULL, NULL);
2121 return false;
2122 }
2123 }
2124
2132 template <typename C, typename T>
2133 inline std::streamsize
2134 basic_pstreambuf<C,T>::write(const char_type* s, std::streamsize n)
2135 {
2136 std::streamsize nwritten = 0;
2137 if (wpipe() >= 0)
2138 {
2139 nwritten = ::write(wpipe(), s, n * sizeof(char_type));
2140 if (nwritten == -1)
2141 error_ = errno;
2142 else
2143 nwritten /= sizeof(char_type);
2144 }
2145 return nwritten;
2146 }
2147
2155 template <typename C, typename T>
2156 inline std::streamsize
2157 basic_pstreambuf<C,T>::read(char_type* s, std::streamsize n)
2158 {
2159 std::streamsize nread = 0;
2160 if (rpipe() >= 0)
2161 {
2162 nread = ::read(rpipe(), s, n * sizeof(char_type));
2163 if (nread == -1)
2164 error_ = errno;
2165 else
2166 nread /= sizeof(char_type);
2167 }
2168 return nread;
2169 }
2170
2172 template <typename C, typename T>
2173 inline pstreams::fd_type&
2175 {
2176 return wpipe_;
2177 }
2178
2180 template <typename C, typename T>
2181 inline pstreams::fd_type&
2183 {
2184 return rpipe_[rsrc_];
2185 }
2186
2188 template <typename C, typename T>
2189 inline pstreams::fd_type&
2191 {
2192 return rpipe_[which];
2193 }
2194
2196 template <typename C, typename T>
2197 inline typename basic_pstreambuf<C,T>::char_type*
2199 {
2200 return rbuffer_[rsrc_];
2201 }
2202
2203
2204 /*
2205 * member definitions for pstream_common
2206 */
2207
2217 template <typename C, typename T>
2218 inline
2220 : std::basic_ios<C,T>(NULL)
2221 , command_()
2222 , buf_()
2223 {
2224 this->std::basic_ios<C,T>::rdbuf(&buf_);
2225 }
2226
2235 template <typename C, typename T>
2236 inline
2237 pstream_common<C,T>::pstream_common(const std::string& cmd, pmode mode)
2238 : std::basic_ios<C,T>(NULL)
2239 , command_(cmd)
2240 , buf_()
2241 {
2242 this->std::basic_ios<C,T>::rdbuf(&buf_);
2243 do_open(cmd, mode);
2244 }
2245
2255 template <typename C, typename T>
2256 inline
2257 pstream_common<C,T>::pstream_common( const std::string& file,
2258 const argv_type& argv,
2259 pmode mode )
2260 : std::basic_ios<C,T>(NULL)
2261 , command_(file)
2262 , buf_()
2263 {
2264 this->std::basic_ios<C,T>::rdbuf(&buf_);
2265 do_open(file, argv, mode);
2266 }
2267
2277 template <typename C, typename T>
2278 inline
2280 {
2281 }
2282
2291 template <typename C, typename T>
2292 inline void
2293 pstream_common<C,T>::do_open(const std::string& cmd, pmode mode)
2294 {
2295 if (!buf_.open((command_=cmd), mode))
2296 this->setstate(std::ios_base::failbit);
2297 }
2298
2308 template <typename C, typename T>
2309 inline void
2310 pstream_common<C,T>::do_open( const std::string& file,
2311 const argv_type& argv,
2312 pmode mode )
2313 {
2314 if (!buf_.open((command_=file), argv, mode))
2315 this->setstate(std::ios_base::failbit);
2316 }
2317
2320 template <typename C, typename T>
2321 inline int
2323 {
2324 if (!buf_.close())
2325 this->setstate(std::ios_base::failbit);
2326 return buf_.status();
2327 }
2328
2333 template <typename C, typename T>
2334 inline bool
2336 {
2337 return buf_.is_open();
2338 }
2339
2341 template <typename C, typename T>
2342 inline const std::string&
2344 {
2345 return command_;
2346 }
2347
2349 // TODO document behaviour if buffer replaced.
2350 template <typename C, typename T>
2353 {
2354 return const_cast<streambuf_type*>(&buf_);
2355 }
2356
2357
2358#if REDI_EVISCERATE_PSTREAMS
2391 template <typename C, typename T>
2392 std::size_t
2393 basic_pstreambuf<C,T>::fopen(FILE*& in, FILE*& out, FILE*& err)
2394 {
2395 in = out = err = NULL;
2396 std::size_t open_files = 0;
2397 if (wpipe() > -1)
2398 {
2399 if ((in = ::fdopen(wpipe(), "w")))
2400 {
2401 open_files |= pstdin;
2402 }
2403 }
2404 if (rpipe(rsrc_out) > -1)
2405 {
2406 if ((out = ::fdopen(rpipe(rsrc_out), "r")))
2407 {
2408 open_files |= pstdout;
2409 }
2410 }
2411 if (rpipe(rsrc_err) > -1)
2412 {
2413 if ((err = ::fdopen(rpipe(rsrc_err), "r")))
2414 {
2415 open_files |= pstderr;
2416 }
2417 }
2418 return open_files;
2419 }
2420
2431 template <typename C, typename T>
2432 inline std::size_t
2433 pstream_common<C,T>::fopen(FILE*& fin, FILE*& fout, FILE*& ferr)
2434 {
2435 return buf_.fopen(fin, fout, ferr);
2436 }
2437
2438#endif // REDI_EVISCERATE_PSTREAMS
2439
2440
2441} // namespace redi
2442
2448#endif // REDI_PSTREAM_H_SEEN
2449
2450// vim: ts=2 sw=2 expandtab
2451
Class template for Input PStreams.
Definition pstream.h:385
basic_ipstream(const std::string &file, const argv_type &argv, pmode mode=pstdout)
Constructor that initialises the stream by starting a process.
Definition pstream.h:437
basic_ipstream(const argv_type &argv, pmode mode=pstdout)
Constructor that initialises the stream by starting a process.
Definition pstream.h:454
basic_ipstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:407
~basic_ipstream()
Destructor.
Definition pstream.h:491
void open(const std::string &file, const argv_type &argv, pmode mode=pstdout)
Start a process.
Definition pstream.h:520
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:404
basic_ipstream & err()
Set streambuf to read from process' stderr.
Definition pstream.h:543
void open(const std::string &cmd, pmode mode=pstdout)
Start a process.
Definition pstream.h:504
basic_ipstream & out()
Set streambuf to read from process' stdout.
Definition pstream.h:532
basic_ipstream(const std::string &cmd, pmode mode=pstdout)
Constructor that initialises the stream by starting a process.
Definition pstream.h:422
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:401
Class template for Output PStreams.
Definition pstream.h:565
void open(const std::string &file, const argv_type &argv, pmode mode=pstdin)
Start a process.
Definition pstream.h:698
basic_opstream(const argv_type &argv, pmode mode=pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:626
basic_opstream(const std::string &file, const argv_type &argv, pmode mode=pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:609
basic_opstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:579
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:573
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:576
~basic_opstream()
Destructor.
Definition pstream.h:670
basic_opstream(const std::string &cmd, pmode mode=pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:594
void open(const std::string &cmd, pmode mode=pstdin)
Start a process.
Definition pstream.h:682
std::basic_ostream< C, T > & peof(std::basic_ostream< C, T > &s)
Manipulator to close the pipe connected to the process' stdin.
Definition pstream.h:1105
Class template for Bidirectional PStreams.
Definition pstream.h:725
basic_pstream(const std::string &cmd, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:754
void open(const std::string &cmd, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:842
basic_pstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:739
~basic_pstream()
Destructor.
Definition pstream.h:830
basic_pstream(const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:786
void open(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:858
basic_pstream & out()
Set streambuf to read from process' stdout.
Definition pstream.h:870
basic_pstream(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:769
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:736
basic_pstream & err()
Set streambuf to read from process' stderr.
Definition pstream.h:881
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:733
Class template for stream buffer.
Definition pstream.h:100
fd_type & rpipe(buf_read_src which)
Return the file descriptor for the specified input pipe.
Definition pstream.h:2190
basic_pstreambuf * killpg(int signal=SIGTERM)
Send a signal to the process' process group.
Definition pstream.h:1823
void close_fd(pstreams::fd_type &fd)
Helper function to close a file descriptor.
Definition pstream.h:1336
int status() const
Return the exit status of the process.
Definition pstream.h:1867
basic_pstreambuf * open(const std::string &file, const argv_type &argv, pmode mode)
Initialise the stream buffer with file and argv.
Definition pstream.h:1391
std::streamsize xsputn(const char_type *s, std::streamsize n)
Insert multiple characters into the pipe.
Definition pstream.h:1970
std::streamsize read(char_type *s, std::streamsize n)
Extract a sequence of characters from the pipe.
Definition pstream.h:2157
std::streamsize write(const char_type *s, std::streamsize n)
Insert a sequence of characters into the pipe.
Definition pstream.h:2134
~basic_pstreambuf()
Destructor.
Definition pstream.h:1197
int_type overflow(int_type c)
Transfer characters to the pipe when character buffer overflows.
Definition pstream.h:1945
int sync()
Write any buffered characters to the stream.
Definition pstream.h:1958
basic_pstreambuf * kill(int signal=SIGTERM)
Send a signal to the process.
Definition pstream.h:1788
void peof()
Close the pipe connected to the process' stdin.
Definition pstream.h:1888
pid_t fork(pmode mode)
Initialise pipes and fork process.
Definition pstream.h:1497
basic_pstreambuf * close()
Close the stream buffer and wait for the process to exit.
Definition pstream.h:1614
basic_pstreambuf(const std::string &file, const argv_type &argv, pmode mode)
Constructor that initialises the buffer with file and argv.
Definition pstream.h:1175
int error() const
Return the error number (errno) for the most recent failed operation.
Definition pstream.h:1877
buf_read_src
Enumerated type to indicate whether stdout or stderr is to be read.
Definition pstream.h:216
void close_fd_array(pstreams::fd_type(&fds)[N])
Helper function to close an array of file descriptors.
Definition pstream.h:1354
fd_type & wpipe()
Return the file descriptor for the output pipe.
Definition pstream.h:2174
bool exited()
Report whether the process has exited.
Definition pstream.h:1854
bool empty_buffer()
Writes buffered characters to the process' stdin pipe.
Definition pstream.h:1993
fd_type fd_t
Definition pstream.h:109
char_type * rbuffer()
Return the active input buffer.
Definition pstream.h:2198
std::streamsize showmanyc()
Report how many characters can be read from active input without blocking.
Definition pstream.h:2052
bool read_err(bool readerr=true)
Change active input source.
Definition pstream.h:1922
bool fill_buffer(bool non_blocking=false)
Definition pstream.h:2074
basic_pstreambuf * open(const std::string &cmd, pmode mode)
Initialise the stream buffer with cmd.
Definition pstream.h:1289
int_type pbackfail(int_type c=traits_type::eof())
Make a character available to be returned by the next extraction.
Definition pstream.h:2037
fd_type & rpipe()
Return the file descriptor for the active input pipe.
Definition pstream.h:2182
basic_pstreambuf()
Default constructor.
Definition pstream.h:1127
bool is_open() const
Report whether the stream buffer has been initialised.
Definition pstream.h:1907
int_type underflow()
Transfer characters from the pipe when the character buffer is empty.
Definition pstream.h:2019
int wait(bool nohang=false)
Wait for the child process to exit.
Definition pstream.h:1745
basic_pstreambuf(const std::string &cmd, pmode mode)
Constructor that initialises the buffer with cmd.
Definition pstream.h:1150
Class template for Restricted PStreams.
Definition pstream.h:916
istream_type & err()
Obtain a reference to the istream that reads the process' stderr.
Definition pstream.h:1071
~basic_rpstream()
Destructor.
Definition pstream.h:1018
basic_rpstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:931
basic_rpstream(const std::string &cmd, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:946
void open(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:1046
void open(const std::string &cmd, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:1030
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:928
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:925
basic_rpstream(const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:978
basic_rpstream(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:961
istream_type & out()
Obtain a reference to the istream that reads the process' stdout.
Definition pstream.h:1059
Class template for common base class.
Definition pstream.h:286
pstream_common()
Default constructor.
Definition pstream.h:2219
void do_open(const std::string &cmd, pmode mode)
Start a process.
Definition pstream.h:2293
void do_open(const std::string &file, const argv_type &argv, pmode mode)
Start a process.
Definition pstream.h:2310
pstream_common(const std::string &file, const argv_type &argv, pmode mode)
Constructor that initialises the stream by starting a process.
Definition pstream.h:2257
pstream_common(const std::string &cmd, pmode mode)
Constructor that initialises the stream by starting a process.
Definition pstream.h:2237
streambuf_type * rdbuf() const
Return a pointer to the stream buffer.
Definition pstream.h:2352
streambuf_type buf_
The stream buffer.
Definition pstream.h:366
int close()
Definition pstream.h:2322
virtual ~pstream_common()=0
Pure virtual destructor.
Definition pstream.h:2279
bool is_open() const
Report whether the stream's buffer has been initialised.
Definition pstream.h:2335
std::string command_
The command used to start the process.
Definition pstream.h:365
const std::string & command() const
Return the command used to initialise the stream.
Definition pstream.h:2343
All PStreams classes are declared in namespace redi.
basic_pstreambuf< char > pstreambuf
Type definition for common template specialisation.
Definition pstream.h:1080
basic_ipstream< char > ipstream
Type definition for common template specialisation.
Definition pstream.h:1082
basic_rpstream< char > rpstream
Type definition for common template specialisation.
Definition pstream.h:1088
basic_opstream< char > opstream
Type definition for common template specialisation.
Definition pstream.h:1084
basic_pstream< char > pstream
Type definition for common template specialisation.
Definition pstream.h:1086
Common base class providing constants and typenames.
Definition pstream.h:66
std::ios_base::openmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:68
static const pmode pstderr
Read from stderr.
Definition pstream.h:78
static const pmode pstdin
Write to stdin.
Definition pstream.h:76
static const pmode newpg
Create a new process group for the child process.
Definition pstream.h:81
@ bufsz
Size of pstreambuf buffers.
Definition pstream.h:85
@ pbsz
Number of putback characters kept.
Definition pstream.h:86
static const pmode pstdout
Read from stdout.
Definition pstream.h:77
int fd_type
Type used for file descriptors.
Definition pstream.h:74
std::vector< std::string > argv_type
Type used to hold the arguments for a command.
Definition pstream.h:71