re PR libstdc++/9424 (i/ostream::operator>>/<<(streambuf*) drops characters)
2003-03-08 Paolo Carlini <pcarlini@unitus.it> Petur Runolfsson <peturr02@ru.is> PR libstdc++/9424 * include/bits/streambuf.tcc (__copy_streambufs): Use sgetn-sputn only when sputn cannot fail, otherwise fall back to safe snextc-sputc. * testsuite/27_io/streambuf_members.cc (test11, test12): Add. Co-Authored-By: Petur Runolfsson <peturr02@ru.is> From-SVN: r63974
This commit is contained in:
parent
ed97aa662b
commit
feecf61403
|
@ -1,3 +1,12 @@
|
||||||
|
2003-03-08 Paolo Carlini <pcarlini@unitus.it>
|
||||||
|
Petur Runolfsson <peturr02@ru.is>
|
||||||
|
|
||||||
|
PR libstdc++/9424
|
||||||
|
* include/bits/streambuf.tcc (__copy_streambufs): Use
|
||||||
|
sgetn-sputn only when sputn cannot fail, otherwise fall back
|
||||||
|
to safe snextc-sputc.
|
||||||
|
* testsuite/27_io/streambuf_members.cc (test11, test12): Add.
|
||||||
|
|
||||||
2003-03-08 Jerry Quinn <jlquinn@optonline.net>
|
2003-03-08 Jerry Quinn <jlquinn@optonline.net>
|
||||||
|
|
||||||
* include/bits/locale_facets.tcc (num_put::do_put(bool)): Use
|
* include/bits/locale_facets.tcc (num_put::do_put(bool)): Use
|
||||||
|
|
|
@ -193,6 +193,9 @@ namespace std
|
||||||
streamsize __ret = 0;
|
streamsize __ret = 0;
|
||||||
streamsize __bufsize = __sbin->in_avail();
|
streamsize __bufsize = __sbin->in_avail();
|
||||||
streamsize __xtrct;
|
streamsize __xtrct;
|
||||||
|
const typename _Traits::off_type __size_opt =
|
||||||
|
__sbin->_M_buf_size_opt > 0 ? __sbin->_M_buf_size_opt : 1;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
while (__bufsize != -1)
|
while (__bufsize != -1)
|
||||||
|
@ -208,12 +211,33 @@ namespace std
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
size_t __size =
|
streamsize __charsread;
|
||||||
__sbin->_M_buf_size_opt > 0 ? __sbin->_M_buf_size_opt : 1;
|
const streamsize __size =
|
||||||
|
std::min(__size_opt, __sbout->_M_out_buf_size());
|
||||||
|
if (__size > 1)
|
||||||
|
{
|
||||||
_CharT* __buf =
|
_CharT* __buf =
|
||||||
static_cast<_CharT*>(__builtin_alloca(sizeof(_CharT) * __size));
|
static_cast<_CharT*>(__builtin_alloca(sizeof(_CharT)
|
||||||
streamsize __charsread = __sbin->sgetn(__buf, __size);
|
* __size));
|
||||||
|
// Since the next sputn cannot fail sgetn can be
|
||||||
|
// safely used.
|
||||||
|
__charsread = __sbin->sgetn(__buf, __size);
|
||||||
__xtrct = __sbout->sputn(__buf, __charsread);
|
__xtrct = __sbout->sputn(__buf, __charsread);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
__xtrct = __charsread = 0;
|
||||||
|
int_type __c = __sbin->sgetc();
|
||||||
|
while (!_Traits::eq_int_type(__c, _Traits::eof()))
|
||||||
|
{
|
||||||
|
++__charsread;
|
||||||
|
if (_Traits::eq_int_type(__sbout->sputc(_Traits::to_char_type(__c)),
|
||||||
|
_Traits::eof()))
|
||||||
|
break;
|
||||||
|
++__xtrct;
|
||||||
|
__c = __sbin->snextc();
|
||||||
|
}
|
||||||
|
}
|
||||||
__ret += __xtrct;
|
__ret += __xtrct;
|
||||||
if (__xtrct != __charsread)
|
if (__xtrct != __charsread)
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -431,6 +431,91 @@ void test10()
|
||||||
VERIFY( buf.result() == "Bad Moon Rising" );
|
VERIFY( buf.result() == "Bad Moon Rising" );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// libstdc++/9424
|
||||||
|
class Outbuf_2 : public std::streambuf
|
||||||
|
{
|
||||||
|
char buf[1];
|
||||||
|
|
||||||
|
public:
|
||||||
|
Outbuf_2()
|
||||||
|
{
|
||||||
|
setp(buf, buf + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
int_type overflow(int_type c)
|
||||||
|
{
|
||||||
|
int_type eof = traits_type::eof();
|
||||||
|
|
||||||
|
if (pptr() < epptr())
|
||||||
|
{
|
||||||
|
if (traits_type::eq_int_type(c, eof))
|
||||||
|
return traits_type::not_eof(c);
|
||||||
|
|
||||||
|
*pptr() = traits_type::to_char_type(c);
|
||||||
|
pbump(1);
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
|
return eof;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class Inbuf_2 : public std::streambuf
|
||||||
|
{
|
||||||
|
static const char buf[];
|
||||||
|
const char* current;
|
||||||
|
int size;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Inbuf_2()
|
||||||
|
{
|
||||||
|
current = buf;
|
||||||
|
size = std::strlen(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
int_type underflow()
|
||||||
|
{
|
||||||
|
if (current < buf + size)
|
||||||
|
return traits_type::to_int_type(*current);
|
||||||
|
return traits_type::eof();
|
||||||
|
}
|
||||||
|
|
||||||
|
int_type uflow()
|
||||||
|
{
|
||||||
|
if (current < buf + size)
|
||||||
|
return traits_type::to_int_type(*current++);
|
||||||
|
return traits_type::eof();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const char Inbuf_2::buf[] = "Atteivlis";
|
||||||
|
|
||||||
|
// <1>
|
||||||
|
void test11()
|
||||||
|
{
|
||||||
|
bool test = true;
|
||||||
|
|
||||||
|
Inbuf_2 inbuf1;
|
||||||
|
std::istream is(&inbuf1);
|
||||||
|
Outbuf_2 outbuf1;
|
||||||
|
is >> &outbuf1;
|
||||||
|
VERIFY( inbuf1.sgetc() == 't' );
|
||||||
|
VERIFY( is.good() );
|
||||||
|
}
|
||||||
|
|
||||||
|
// <2>
|
||||||
|
void test12()
|
||||||
|
{
|
||||||
|
bool test = true;
|
||||||
|
|
||||||
|
Outbuf_2 outbuf2;
|
||||||
|
std::ostream os (&outbuf2);
|
||||||
|
Inbuf_2 inbuf2;
|
||||||
|
os << &inbuf2;
|
||||||
|
VERIFY( inbuf2.sgetc() == 't' );
|
||||||
|
VERIFY( os.good() );
|
||||||
|
}
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
test01();
|
test01();
|
||||||
|
@ -445,5 +530,7 @@ int main()
|
||||||
|
|
||||||
test09();
|
test09();
|
||||||
test10();
|
test10();
|
||||||
|
test11();
|
||||||
|
test12();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue