xref: /freebsd/contrib/kyua/utils/process/systembuf.cpp (revision b0d29bc47dba79f6f38e67eabadfb4b32ffd9390)
1*b0d29bc4SBrooks Davis // Copyright 2010 The Kyua Authors.
2*b0d29bc4SBrooks Davis // All rights reserved.
3*b0d29bc4SBrooks Davis //
4*b0d29bc4SBrooks Davis // Redistribution and use in source and binary forms, with or without
5*b0d29bc4SBrooks Davis // modification, are permitted provided that the following conditions are
6*b0d29bc4SBrooks Davis // met:
7*b0d29bc4SBrooks Davis //
8*b0d29bc4SBrooks Davis // * Redistributions of source code must retain the above copyright
9*b0d29bc4SBrooks Davis //   notice, this list of conditions and the following disclaimer.
10*b0d29bc4SBrooks Davis // * Redistributions in binary form must reproduce the above copyright
11*b0d29bc4SBrooks Davis //   notice, this list of conditions and the following disclaimer in the
12*b0d29bc4SBrooks Davis //   documentation and/or other materials provided with the distribution.
13*b0d29bc4SBrooks Davis // * Neither the name of Google Inc. nor the names of its contributors
14*b0d29bc4SBrooks Davis //   may be used to endorse or promote products derived from this software
15*b0d29bc4SBrooks Davis //   without specific prior written permission.
16*b0d29bc4SBrooks Davis //
17*b0d29bc4SBrooks Davis // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*b0d29bc4SBrooks Davis // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*b0d29bc4SBrooks Davis // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20*b0d29bc4SBrooks Davis // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21*b0d29bc4SBrooks Davis // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22*b0d29bc4SBrooks Davis // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23*b0d29bc4SBrooks Davis // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*b0d29bc4SBrooks Davis // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*b0d29bc4SBrooks Davis // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*b0d29bc4SBrooks Davis // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27*b0d29bc4SBrooks Davis // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*b0d29bc4SBrooks Davis 
29*b0d29bc4SBrooks Davis #include "utils/process/systembuf.hpp"
30*b0d29bc4SBrooks Davis 
31*b0d29bc4SBrooks Davis extern "C" {
32*b0d29bc4SBrooks Davis #include <unistd.h>
33*b0d29bc4SBrooks Davis }
34*b0d29bc4SBrooks Davis 
35*b0d29bc4SBrooks Davis #include "utils/auto_array.ipp"
36*b0d29bc4SBrooks Davis #include "utils/noncopyable.hpp"
37*b0d29bc4SBrooks Davis #include "utils/sanity.hpp"
38*b0d29bc4SBrooks Davis 
39*b0d29bc4SBrooks Davis using utils::process::systembuf;
40*b0d29bc4SBrooks Davis 
41*b0d29bc4SBrooks Davis 
42*b0d29bc4SBrooks Davis /// Private implementation fields for systembuf.
43*b0d29bc4SBrooks Davis struct systembuf::impl : utils::noncopyable {
44*b0d29bc4SBrooks Davis     /// File descriptor attached to the systembuf.
45*b0d29bc4SBrooks Davis     int _fd;
46*b0d29bc4SBrooks Davis 
47*b0d29bc4SBrooks Davis     /// Size of the _read_buf and _write_buf buffers.
48*b0d29bc4SBrooks Davis     std::size_t _bufsize;
49*b0d29bc4SBrooks Davis 
50*b0d29bc4SBrooks Davis     /// In-memory buffer for read operations.
51*b0d29bc4SBrooks Davis     utils::auto_array< char > _read_buf;
52*b0d29bc4SBrooks Davis 
53*b0d29bc4SBrooks Davis     /// In-memory buffer for write operations.
54*b0d29bc4SBrooks Davis     utils::auto_array< char > _write_buf;
55*b0d29bc4SBrooks Davis 
56*b0d29bc4SBrooks Davis     /// Initializes private implementation data.
57*b0d29bc4SBrooks Davis     ///
58*b0d29bc4SBrooks Davis     /// \param fd The file descriptor.
59*b0d29bc4SBrooks Davis     /// \param bufsize The size of the created read and write buffers.
implsystembuf::impl60*b0d29bc4SBrooks Davis     impl(const int fd, const std::size_t bufsize) :
61*b0d29bc4SBrooks Davis         _fd(fd),
62*b0d29bc4SBrooks Davis         _bufsize(bufsize),
63*b0d29bc4SBrooks Davis         _read_buf(new char[bufsize]),
64*b0d29bc4SBrooks Davis         _write_buf(new char[bufsize])
65*b0d29bc4SBrooks Davis     {
66*b0d29bc4SBrooks Davis     }
67*b0d29bc4SBrooks Davis };
68*b0d29bc4SBrooks Davis 
69*b0d29bc4SBrooks Davis 
70*b0d29bc4SBrooks Davis /// Constructs a new systembuf based on an open file descriptor.
71*b0d29bc4SBrooks Davis ///
72*b0d29bc4SBrooks Davis /// This grabs ownership of the file descriptor.
73*b0d29bc4SBrooks Davis ///
74*b0d29bc4SBrooks Davis /// \param fd The file descriptor to wrap.  Must be open and valid.
75*b0d29bc4SBrooks Davis /// \param bufsize The size to use for the internal read/write buffers.
systembuf(const int fd,std::size_t bufsize)76*b0d29bc4SBrooks Davis systembuf::systembuf(const int fd, std::size_t bufsize) :
77*b0d29bc4SBrooks Davis     _pimpl(new impl(fd, bufsize))
78*b0d29bc4SBrooks Davis {
79*b0d29bc4SBrooks Davis     setp(_pimpl->_write_buf.get(), _pimpl->_write_buf.get() + _pimpl->_bufsize);
80*b0d29bc4SBrooks Davis }
81*b0d29bc4SBrooks Davis 
82*b0d29bc4SBrooks Davis 
83*b0d29bc4SBrooks Davis /// Destroys a systembuf object.
84*b0d29bc4SBrooks Davis ///
85*b0d29bc4SBrooks Davis /// \post The file descriptor attached to this systembuf is closed.
~systembuf(void)86*b0d29bc4SBrooks Davis systembuf::~systembuf(void)
87*b0d29bc4SBrooks Davis {
88*b0d29bc4SBrooks Davis     ::close(_pimpl->_fd);
89*b0d29bc4SBrooks Davis }
90*b0d29bc4SBrooks Davis 
91*b0d29bc4SBrooks Davis 
92*b0d29bc4SBrooks Davis /// Reads new data when the systembuf read buffer underflows.
93*b0d29bc4SBrooks Davis ///
94*b0d29bc4SBrooks Davis /// \return The new character to be read, or EOF if no more.
95*b0d29bc4SBrooks Davis systembuf::int_type
underflow(void)96*b0d29bc4SBrooks Davis systembuf::underflow(void)
97*b0d29bc4SBrooks Davis {
98*b0d29bc4SBrooks Davis     PRE(gptr() >= egptr());
99*b0d29bc4SBrooks Davis 
100*b0d29bc4SBrooks Davis     bool ok;
101*b0d29bc4SBrooks Davis     ssize_t cnt = ::read(_pimpl->_fd, _pimpl->_read_buf.get(),
102*b0d29bc4SBrooks Davis                          _pimpl->_bufsize);
103*b0d29bc4SBrooks Davis     ok = (cnt != -1 && cnt != 0);
104*b0d29bc4SBrooks Davis 
105*b0d29bc4SBrooks Davis     if (!ok)
106*b0d29bc4SBrooks Davis         return traits_type::eof();
107*b0d29bc4SBrooks Davis     else {
108*b0d29bc4SBrooks Davis         setg(_pimpl->_read_buf.get(), _pimpl->_read_buf.get(),
109*b0d29bc4SBrooks Davis              _pimpl->_read_buf.get() + cnt);
110*b0d29bc4SBrooks Davis         return traits_type::to_int_type(*gptr());
111*b0d29bc4SBrooks Davis     }
112*b0d29bc4SBrooks Davis }
113*b0d29bc4SBrooks Davis 
114*b0d29bc4SBrooks Davis 
115*b0d29bc4SBrooks Davis /// Writes data to the file descriptor when the write buffer overflows.
116*b0d29bc4SBrooks Davis ///
117*b0d29bc4SBrooks Davis /// \param c The character that causes the overflow.
118*b0d29bc4SBrooks Davis ///
119*b0d29bc4SBrooks Davis /// \return EOF if error, some other value for success.
120*b0d29bc4SBrooks Davis ///
121*b0d29bc4SBrooks Davis /// \throw something TODO(jmmv): According to the documentation, it is OK for
122*b0d29bc4SBrooks Davis ///     this method to throw in case of errors.  Revisit this code to see if we
123*b0d29bc4SBrooks Davis ///     can do better.
124*b0d29bc4SBrooks Davis systembuf::int_type
overflow(int c)125*b0d29bc4SBrooks Davis systembuf::overflow(int c)
126*b0d29bc4SBrooks Davis {
127*b0d29bc4SBrooks Davis     PRE(pptr() >= epptr());
128*b0d29bc4SBrooks Davis     if (sync() == -1)
129*b0d29bc4SBrooks Davis         return traits_type::eof();
130*b0d29bc4SBrooks Davis     if (!traits_type::eq_int_type(c, traits_type::eof())) {
131*b0d29bc4SBrooks Davis         traits_type::assign(*pptr(), c);
132*b0d29bc4SBrooks Davis         pbump(1);
133*b0d29bc4SBrooks Davis     }
134*b0d29bc4SBrooks Davis     return traits_type::not_eof(c);
135*b0d29bc4SBrooks Davis }
136*b0d29bc4SBrooks Davis 
137*b0d29bc4SBrooks Davis 
138*b0d29bc4SBrooks Davis /// Synchronizes the stream with the file descriptor.
139*b0d29bc4SBrooks Davis ///
140*b0d29bc4SBrooks Davis /// \return 0 on success, -1 on error.
141*b0d29bc4SBrooks Davis int
sync(void)142*b0d29bc4SBrooks Davis systembuf::sync(void)
143*b0d29bc4SBrooks Davis {
144*b0d29bc4SBrooks Davis     ssize_t cnt = pptr() - pbase();
145*b0d29bc4SBrooks Davis 
146*b0d29bc4SBrooks Davis     bool ok;
147*b0d29bc4SBrooks Davis     ok = ::write(_pimpl->_fd, pbase(), cnt) == cnt;
148*b0d29bc4SBrooks Davis 
149*b0d29bc4SBrooks Davis     if (ok)
150*b0d29bc4SBrooks Davis         pbump(-cnt);
151*b0d29bc4SBrooks Davis     return ok ? 0 : -1;
152*b0d29bc4SBrooks Davis }
153