// file : butl/fdstream -*- C++ -*- // copyright : Copyright (c) 2014-2015 Code Synthesis Ltd // license : MIT; see accompanying LICENSE file #ifndef BUTL_FDSTREAM #define BUTL_FDSTREAM #include #include namespace butl { // An iostream that is initialized with a file descriptor rather than // a file name. // // Notes and limitations: // // - char only // - input or output but not both // - no support for put back // - throws std::system_error in case of a read()/write() error // - not movable, though can be easily supported // class fdbuf: public std::basic_streambuf { public: virtual ~fdbuf (); fdbuf () = default; fdbuf (int fd) {open (fd);} fdbuf (const fdbuf&) = delete; fdbuf& operator= (const fdbuf&) = delete; void close (); void open (int fd); bool is_open () const {return fd_ != -1;} public: using int_type = std::basic_streambuf::int_type; using traits_type = std::basic_streambuf::traits_type; // basic_streambuf input interface. // public: virtual std::streamsize showmanyc (); virtual int_type underflow (); private: bool load (); // basic_streambuf output interface. // public: virtual int_type overflow (int_type); virtual int sync (); private: bool save (); private: int fd_ = -1; char buf_[2048]; }; class fdstream_base { protected: fdstream_base () = default; fdstream_base (int fd): buf_ (fd) {} protected: fdbuf buf_; }; class ifdstream: fdstream_base, public std::istream { public: ifdstream (): std::istream (&buf_) {} ifdstream (int fd): fdstream_base (fd), std::istream (&buf_) {} void close () {buf_.close ();} void open (int fd) {buf_.open (fd);} bool is_open () const {return buf_.is_open ();} }; class ofdstream: fdstream_base, public std::ostream { public: ofdstream (): std::ostream (&buf_) {} ofdstream (int fd): fdstream_base (fd), std::ostream (&buf_) {} void close () {buf_.close ();} void open (int fd) {buf_.open (fd);} bool is_open () const {return buf_.is_open ();} }; } #endif // BUTL_FDSTREAM