00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031 #ifndef MORE_PIPESTREAM_H
00032 #define MORE_PIPESTREAM_H
00033
00034 #include <stdexcept>
00035 #include <ios>
00036 #include <iostream>
00037 #include <string>
00038
00039 #include <unistd.h>
00040
00041
00042 namespace more {
00043
00044
00045 class pipe_streambuf : public std::streambuf {
00046 typedef std::char_traits<char> traits;
00047 typedef traits::int_type int_type;
00048 typedef traits::char_type char_type;
00049 int fd[2];
00050
00051 public:
00052 pipe_streambuf() {
00053 if(pipe(fd) < 0)
00054 throw std::runtime_error("Could not open pipe.");
00055 }
00056 ~pipe_streambuf() {
00057 if(fd[0] != -1) close(fd[0]);
00058 if(fd[1] != -1) close(fd[1]);
00059 }
00060 void close_input() { close(fd[0]); fd[0] = -1; }
00061 void close_output() { close(fd[1]); fd[1] = -1; }
00062
00063 protected:
00064
00065 int_type sync() {
00066 std::streamsize n = pptr() - pbase();
00067 if(n && write(fd[1], pbase(), n*sizeof(char_type)) < 0)
00068 return EOF;
00069 return 0;
00070 }
00071
00072 int_type underflow() {
00073 char_type buf[2];
00074 if(read(fd[0], buf, sizeof(char_type)) != 1) return traits::eof();
00075 else return traits::to_int_type(buf[0]);
00076 }
00077
00078 int_type uflow() { return underflow(); }
00079
00080
00081 int_type overflow(int_type c = traits::eof()) {
00082 return write(fd[1], &c, sizeof(char_type)) == 1? 1 : traits::eof();
00083 }
00084 std::streamsize xsputn(const char_type* s, std::streamsize n) {
00085 return write(fd[1], s, n*sizeof(char_type));
00086
00087
00088 }
00089 };
00090
00091
00092 class pipe_stream : virtual public pipe_streambuf, public std::iostream {
00093 public:
00094 pipe_stream() : std::iostream(this) {}
00095 };
00096
00097 }
00098
00099 #endif