| /* |
| * Copyright (c) 1999 |
| * Silicon Graphics Computer Systems, Inc. |
| * |
| * Copyright (c) 1999 |
| * Boris Fomitchev |
| * |
| * This material is provided "as is", with absolutely no warranty expressed |
| * or implied. Any use is at your own risk. |
| * |
| * Permission to use or copy this software for any purpose is hereby granted |
| * without fee, provided the above notices are retained on all copies. |
| * Permission to modify the code and to distribute modified code is granted, |
| * provided the above notices are retained, and a notice that the code was |
| * modified is included with the above copyright notice. |
| * |
| */ |
| |
| // Implementation of the classes in header <strstream>. |
| // WARNING: The classes defined in <strstream> are DEPRECATED. This |
| // header is defined in section D.7.1 of the C++ standard, and it |
| // MAY BE REMOVED in a future standard revision. You should use the |
| // header <sstream> instead. |
| |
| #include "stlport_prefix.h" |
| |
| #include <strstream> |
| #include <algorithm> |
| #include <limits> |
| |
| _STLP_BEGIN_NAMESPACE |
| |
| // strstreambuf constructor, destructor. |
| strstreambuf::strstreambuf(streamsize initial_capacity) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(true), _M_frozen(false), _M_constant(false) { |
| size_t n = (sizeof(streamsize) > sizeof(size_t)) ? __STATIC_CAST(size_t, (min)(__STATIC_CAST(streamsize, (numeric_limits<size_t>::max)()), |
| (max)(initial_capacity, streamsize(16)))) |
| : __STATIC_CAST(size_t, (max)(initial_capacity, streamsize(16))); |
| |
| char* buf = _M_alloc(n); |
| if (buf) { |
| setp(buf, buf + n); |
| setg(buf, buf, buf); |
| } |
| } |
| |
| strstreambuf::strstreambuf(__alloc_fn alloc_f, __free_fn free_f) |
| : _M_alloc_fun(alloc_f), _M_free_fun(free_f), |
| _M_dynamic(true), _M_frozen(false), _M_constant(false) { |
| size_t n = 16; |
| |
| char* buf = _M_alloc(n); |
| if (buf) { |
| setp(buf, buf + n); |
| setg(buf, buf, buf); |
| } |
| } |
| |
| strstreambuf::strstreambuf(char* get, streamsize n, char* put) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(false), _M_frozen(false), _M_constant(false) { |
| _M_setup(get, put, n); |
| } |
| |
| strstreambuf::strstreambuf(signed char* get, streamsize n, signed char* put) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(false), _M_frozen(false), _M_constant(false) { |
| _M_setup(__REINTERPRET_CAST(char*,get), __REINTERPRET_CAST(char*,put), n); |
| } |
| |
| strstreambuf::strstreambuf(unsigned char* get, streamsize n, |
| unsigned char* put) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(false), _M_frozen(false), _M_constant(false) { |
| _M_setup(__REINTERPRET_CAST(char*,get), __REINTERPRET_CAST(char*,put), n); |
| } |
| |
| strstreambuf::strstreambuf(const char* get, streamsize n) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(false), _M_frozen(false), _M_constant(true) { |
| _M_setup(__CONST_CAST(char*,get), 0, n); |
| } |
| |
| strstreambuf::strstreambuf(const signed char* get, streamsize n) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(false), _M_frozen(false), _M_constant(true) { |
| _M_setup(__REINTERPRET_CAST(char*, __CONST_CAST(signed char*,get)), 0, n); |
| } |
| |
| strstreambuf::strstreambuf(const unsigned char* get, streamsize n) |
| : _M_alloc_fun(0), _M_free_fun(0), |
| _M_dynamic(false), _M_frozen(false), _M_constant(true) { |
| _M_setup(__REINTERPRET_CAST(char*, __CONST_CAST(unsigned char*,get)), 0, n); |
| } |
| |
| strstreambuf::~strstreambuf() { |
| if (_M_dynamic && !_M_frozen) |
| _M_free(eback()); |
| } |
| |
| void strstreambuf::freeze(bool frozenflag) { |
| if (_M_dynamic) |
| _M_frozen = frozenflag; |
| } |
| |
| char* strstreambuf::str() { |
| freeze(true); |
| return eback(); |
| } |
| |
| int strstreambuf::pcount() const { |
| return int(pptr() ? pptr() - pbase() : 0); |
| } |
| |
| strstreambuf::int_type strstreambuf::overflow(int_type c) { |
| if (c == traits_type::eof()) |
| return traits_type::not_eof(c); |
| |
| // Try to expand the buffer. |
| if (pptr() == epptr() && _M_dynamic && !_M_frozen && !_M_constant) { |
| ptrdiff_t old_size = epptr() - pbase(); |
| ptrdiff_t new_size = (max)(2 * old_size, ptrdiff_t(1)); |
| |
| char* buf = _M_alloc(new_size); |
| if (buf) { |
| memcpy(buf, pbase(), old_size); |
| |
| char* old_buffer = pbase(); |
| bool reposition_get = false; |
| ptrdiff_t old_get_offset; |
| if (gptr() != 0) { |
| reposition_get = true; |
| old_get_offset = gptr() - eback(); |
| } |
| |
| setp(buf, buf + new_size); |
| pbump((int)old_size); |
| |
| if (reposition_get) |
| setg(buf, buf + old_get_offset, buf + (max)(old_get_offset, old_size)); |
| |
| _M_free(old_buffer); |
| } |
| } |
| |
| if (pptr() != epptr()) { |
| *pptr() = traits_type::to_char_type(c); |
| pbump(1); |
| return c; |
| } |
| else |
| return traits_type::eof(); |
| } |
| |
| strstreambuf::int_type strstreambuf::pbackfail(int_type c) { |
| if (gptr() != eback()) { |
| if (c == traits_type::eof()) { |
| gbump(-1); |
| return traits_type::not_eof(c); |
| } |
| else if (c == gptr()[-1]) { |
| gbump(-1); |
| return c; |
| } |
| else if (!_M_constant) { |
| gbump(-1); |
| *gptr() = traits_type::to_char_type(c); |
| return c; |
| } |
| } |
| |
| return traits_type::eof(); |
| } |
| |
| strstreambuf::int_type strstreambuf::underflow() { |
| if (gptr() == egptr() && pptr() && pptr() > egptr()) |
| setg(eback(), gptr(), pptr()); |
| |
| if (gptr() != egptr()) |
| return (unsigned char) *gptr(); |
| else |
| return _Traits::eof(); |
| } |
| |
| basic_streambuf<char, char_traits<char> >* |
| strstreambuf::setbuf(char*, streamsize) { |
| return this; |
| } |
| |
| strstreambuf::pos_type |
| strstreambuf::seekoff(off_type off, |
| ios_base::seekdir dir, ios_base::openmode mode) { |
| bool do_get = false; |
| bool do_put = false; |
| |
| if ((mode & (ios_base::in | ios_base::out)) == |
| (ios_base::in | ios_base::out) && |
| (dir == ios_base::beg || dir == ios_base::end)) |
| do_get = do_put = true; |
| else if (mode & ios_base::in) |
| do_get = true; |
| else if (mode & ios_base::out) |
| do_put = true; |
| |
| // !gptr() is here because, according to D.7.1 paragraph 4, the seekable |
| // area is undefined if there is no get area. |
| if ((!do_get && !do_put) || (do_put && !pptr()) || !gptr()) |
| return pos_type(off_type(-1)); |
| |
| char* seeklow = eback(); |
| char* seekhigh = epptr() ? epptr() : egptr(); |
| |
| off_type newoff; |
| switch(dir) { |
| case ios_base::beg: |
| newoff = 0; |
| break; |
| case ios_base::end: |
| newoff = seekhigh - seeklow; |
| break; |
| case ios_base::cur: |
| newoff = do_put ? pptr() - seeklow : gptr() - seeklow; |
| break; |
| default: |
| return pos_type(off_type(-1)); |
| } |
| |
| off += newoff; |
| if (off < 0 || off > seekhigh - seeklow) |
| return pos_type(off_type(-1)); |
| |
| if (do_put) { |
| if (seeklow + __STATIC_CAST(ptrdiff_t, off) < pbase()) { |
| setp(seeklow, epptr()); |
| pbump((int)off); |
| } |
| else { |
| setp(pbase(), epptr()); |
| pbump((int)(off - (pbase() - seeklow))); |
| } |
| } |
| if (do_get) { |
| if (off <= egptr() - seeklow) |
| setg(seeklow, seeklow + __STATIC_CAST(ptrdiff_t, off), egptr()); |
| else if (off <= pptr() - seeklow) |
| setg(seeklow, seeklow + __STATIC_CAST(ptrdiff_t, off), pptr()); |
| else |
| setg(seeklow, seeklow + __STATIC_CAST(ptrdiff_t, off), epptr()); |
| } |
| |
| return pos_type(newoff); |
| } |
| |
| strstreambuf::pos_type |
| strstreambuf::seekpos(pos_type pos, ios_base::openmode mode) { |
| return seekoff(pos - pos_type(off_type(0)), ios_base::beg, mode); |
| } |
| |
| |
| char* strstreambuf::_M_alloc(size_t n) { |
| if (_M_alloc_fun) |
| return __STATIC_CAST(char*,_M_alloc_fun(n)); |
| else |
| return new char[n]; |
| } |
| |
| void strstreambuf::_M_free(char* p) { |
| if (p) { |
| if (_M_free_fun) |
| _M_free_fun(p); |
| else |
| delete[] p; |
| } |
| } |
| |
| void strstreambuf::_M_setup(char* get, char* put, streamsize n) { |
| if (get) { |
| size_t N = n > 0 ? size_t(n) : n == 0 ? strlen(get) : size_t(INT_MAX); |
| |
| if (put) { |
| setg(get, get, get + N); |
| setp(put, put + N); |
| } |
| else { |
| setg(get, get, get + N); |
| } |
| } |
| } |
| |
| //---------------------------------------------------------------------- |
| // Class istrstream |
| |
| istrstream::istrstream(char* s) |
| : basic_istream<char, char_traits<char> >(0), _M_buf(s, 0) { |
| this->init(&_M_buf); |
| } |
| |
| istrstream::istrstream(const char* s) |
| : basic_istream<char, char_traits<char> >(0), _M_buf(s, 0) { |
| this->init(&_M_buf); |
| } |
| |
| istrstream::istrstream(char* s, streamsize n) |
| : basic_istream<char, char_traits<char> >(0), _M_buf(s, n) { |
| this->init(&_M_buf); |
| } |
| |
| istrstream::istrstream(const char* s, streamsize n) |
| : basic_istream<char, char_traits<char> >(0), _M_buf(s, n) { |
| this->init(&_M_buf); |
| } |
| |
| istrstream::~istrstream() {} |
| |
| strstreambuf* istrstream::rdbuf() const { |
| return __CONST_CAST(strstreambuf*,&_M_buf); |
| } |
| |
| char* istrstream::str() { return _M_buf.str(); } |
| |
| //---------------------------------------------------------------------- |
| // Class ostrstream |
| |
| ostrstream::ostrstream() |
| : basic_ostream<char, char_traits<char> >(0), _M_buf() { |
| basic_ios<char, char_traits<char> >::init(&_M_buf); |
| } |
| |
| ostrstream::ostrstream(char* s, int n, ios_base::openmode mode) |
| : basic_ostream<char, char_traits<char> >(0), |
| _M_buf(s, n, mode & ios_base::app ? s + strlen(s) : s) { |
| basic_ios<char, char_traits<char> >::init(&_M_buf); |
| } |
| |
| ostrstream::~ostrstream() {} |
| |
| strstreambuf* ostrstream::rdbuf() const { |
| return __CONST_CAST(strstreambuf*,&_M_buf); |
| } |
| |
| void ostrstream::freeze(bool freezeflag) { |
| _M_buf.freeze(freezeflag); |
| } |
| |
| char* ostrstream::str() { |
| return _M_buf.str(); |
| } |
| |
| int ostrstream::pcount() const { |
| return _M_buf.pcount(); |
| } |
| |
| |
| //---------------------------------------------------------------------- |
| // Class strstream |
| |
| strstream::strstream() |
| : basic_iostream<char, char_traits<char> >(0), _M_buf() { |
| basic_ios<char, char_traits<char> >::init(&_M_buf); |
| } |
| |
| strstream::strstream(char* s, int n, ios_base::openmode mode) |
| : basic_iostream<char, char_traits<char> >(0), |
| _M_buf(s, n, mode & ios_base::app ? s + strlen(s) : s) { |
| basic_ios<char, char_traits<char> >::init(&_M_buf); |
| } |
| |
| strstream::~strstream() {} |
| |
| strstreambuf* strstream::rdbuf() const { |
| return __CONST_CAST(strstreambuf*,&_M_buf); |
| } |
| |
| void strstream::freeze(bool freezeflag) { |
| _M_buf.freeze(freezeflag); |
| } |
| |
| int strstream::pcount() const { |
| return _M_buf.pcount(); |
| } |
| |
| char* strstream::str() { |
| return _M_buf.str(); |
| } |
| |
| _STLP_END_NAMESPACE |
| |
| // Local Variables: |
| // mode:C++ |
| // End: |