1 /*
2 * Copyright (c) 1999
3 * Silicon Graphics Computer Systems, Inc.
4 *
5 * Copyright (c) 1999
6 * Boris Fomitchev
7 *
8 * This material is provided "as is", with absolutely no warranty expressed
9 * or implied. Any use is at your own risk.
10 *
11 * Permission to use or copy this software for any purpose is hereby granted
12 * without fee, provided the above notices are retained on all copies.
13 * Permission to modify the code and to distribute modified code is granted,
14 * provided the above notices are retained, and a notice that the code was
15 * modified is included with the above copyright notice.
16 *
17 */
18 #ifndef _STLP_ISTREAM_C
19 #define _STLP_ISTREAM_C
20
21 #ifndef _STLP_INTERNAL_ISTREAM
22 # include <stl/_istream.h>
23 #endif
24
25 #ifndef _STLP_INTERNAL_LIMITS
26 # include <stl/_limits.h>
27 #endif
28
29 #ifndef _STLP_INTERNAL_NUM_GET_H
30 # include <stl/_num_get.h>
31 #endif
32
33 #if defined ( _STLP_NESTED_TYPE_PARAM_BUG )
34 // no wchar_t is supported for this mode
35 # define __BIS_int_type__ int
36 # define __BIS_pos_type__ streampos
37 # define __BIS_off_type__ streamoff
38 #else
39 # define __BIS_int_type__ _STLP_TYPENAME_ON_RETURN_TYPE basic_istream<_CharT, _Traits>::int_type
40 # define __BIS_pos_type__ _STLP_TYPENAME_ON_RETURN_TYPE basic_istream<_CharT, _Traits>::pos_type
41 # define __BIS_off_type__ _STLP_TYPENAME_ON_RETURN_TYPE basic_istream<_CharT, _Traits>::off_type
42 #endif
43
44 _STLP_BEGIN_NAMESPACE
45
46 //----------------------------------------------------------------------
47 // Function object structs used by some member functions.
48
49 _STLP_MOVE_TO_PRIV_NAMESPACE
50
51 template <class _Traits>
52 struct _Is_not_wspace {
53 typedef typename _Traits::char_type argument_type;
54 typedef bool result_type;
55
56 const ctype<argument_type>* _M_ctype;
57
_Is_not_wspace_Is_not_wspace58 _Is_not_wspace(const ctype<argument_type>* __c_type) : _M_ctype(__c_type) {}
operator_Is_not_wspace59 bool operator()(argument_type __c) const
60 { return !_M_ctype->is(ctype_base::space, __c); }
61 };
62
63 template <class _Traits>
64 struct _Is_wspace_null {
65 typedef typename _Traits::char_type argument_type;
66 typedef bool result_type;
67
68 const ctype<argument_type>* _M_ctype;
69
_Is_wspace_null_Is_wspace_null70 _Is_wspace_null(const ctype<argument_type>* __c_type) : _M_ctype(__c_type) {}
operator_Is_wspace_null71 bool operator()(argument_type __c) const {
72 return _Traits::eq(__c, argument_type()) ||
73 _M_ctype->is(ctype_base::space, __c);
74 }
75 };
76
77 template <class _Traits>
78 struct _Scan_for_wspace {
79 typedef typename _Traits::char_type char_type;
80 typedef char_type* first_argument_type;
81 typedef char_type* second_argument_type;
82 typedef char_type* result_type;
83
84 const ctype<char_type>* _M_ctype;
85
_Scan_for_wspace_Scan_for_wspace86 _Scan_for_wspace(const ctype<char_type>* __ctype) : _M_ctype(__ctype) {}
87 const char_type*
operator_Scan_for_wspace88 operator()(const char_type* __first, const char_type* __last) const {
89 return _M_ctype->scan_is(ctype_base::space, __first, __last);
90 }
91 };
92
93 template <class _Traits>
94 struct _Scan_wspace_null {
95 typedef typename _Traits::char_type char_type;
96 typedef char_type* first_argument_type;
97 typedef char_type* second_argument_type;
98 typedef char_type* result_type;
99
100 const ctype<char_type>* _M_ctype;
101
_Scan_wspace_null_Scan_wspace_null102 _Scan_wspace_null(const ctype<char_type>* __c_type) : _M_ctype(__c_type) {}
103 const char_type*
operator_Scan_wspace_null104 operator()(const char_type* __first, const char_type* __last) const {
105 __last = find_if(__first, __last,
106 _Eq_char_bound<_Traits>(char_type()));
107 return _M_ctype->scan_is(ctype_base::space, __first, __last);
108 }
109 };
110
111 template <class _Traits>
112 struct _Scan_for_not_wspace {
113 typedef typename _Traits::char_type char_type;
114 typedef char_type* first_argument_type;
115 typedef char_type* second_argument_type;
116 typedef char_type* result_type;
117
118 const ctype<char_type>* _M_ctype;
119
_Scan_for_not_wspace_Scan_for_not_wspace120 _Scan_for_not_wspace(const ctype<char_type>* __c_type) : _M_ctype(__c_type) {}
121 const char_type*
operator_Scan_for_not_wspace122 operator()(const char_type* __first, const char_type* __last) const {
123 return _M_ctype->scan_not(ctype_base::space, __first, __last);
124 }
125 };
126
127 template <class _Traits>
128 struct _Scan_for_char_val {
129 typedef typename _Traits::char_type char_type;
130 typedef char_type* first_argument_type;
131 typedef char_type* second_argument_type;
132 typedef char_type* result_type;
133
134 char_type _M_val;
135
_Scan_for_char_val_Scan_for_char_val136 _Scan_for_char_val(char_type __val) : _M_val(__val) {}
137
138 const char_type*
operator_Scan_for_char_val139 operator()(const char_type* __first, const char_type* __last) const {
140 return find_if(__first, __last, _Eq_char_bound<_Traits>(_M_val));
141 }
142 };
143
144 template <class _Traits>
145 struct _Scan_for_int_val {
146 typedef typename _Traits::char_type char_type;
147 typedef typename _Traits::int_type int_type;
148 typedef char_type* first_argument_type;
149 typedef char_type* second_argument_type;
150 typedef char_type* result_type;
151
152 int_type _M_val;
153
_Scan_for_int_val_Scan_for_int_val154 _Scan_for_int_val(int_type __val) : _M_val(__val) {}
155
156 const char_type*
operator_Scan_for_int_val157 operator()(const char_type* __first, const char_type* __last) const {
158 return find_if(__first, __last,
159 _Eq_int_bound<_Traits>(_M_val));
160 }
161 };
162
163 // Helper function: try to push back a character to a streambuf,
164 // return true if the pushback succeeded. Does not throw.
165
166 template <class _CharT, class _Traits>
167 bool _STLP_CALL
__pushback(basic_streambuf<_CharT,_Traits> * __buf,_CharT __c)168 __pushback(basic_streambuf<_CharT, _Traits>* __buf, _CharT __c) {
169 bool ret;
170 _STLP_TRY {
171 const typename _Traits::int_type __eof = _Traits::eof();
172 ret = !_Traits::eq_int_type(__buf->sputbackc(__c), __eof);
173 }
174 _STLP_CATCH_ALL {
175 ret = false;
176 }
177 return ret;
178 }
179
180 //----------------------------------------------------------------------
181 // Definitions of basic_istream<>'s noninline member functions.
182
183 // Helper function for formatted input of numbers.
184 template <class _CharT, class _Traits, class _Number>
185 ios_base::iostate _STLP_CALL
__get_num(basic_istream<_CharT,_Traits> & __that,_Number & __val)186 __get_num(basic_istream<_CharT, _Traits>& __that, _Number& __val) {
187 typedef typename basic_istream<_CharT, _Traits>::sentry _Sentry;
188 ios_base::iostate __err = 0;
189 _Sentry __sentry( __that ); // Skip whitespace.
190 if (__sentry) {
191 typedef num_get<_CharT, istreambuf_iterator<_CharT, _Traits> > _Num_get;
192 _STLP_TRY {
193 // Do not remove additional parenthesis around use_facet instanciation, some compilers (VC6)
194 // require it when building the library.
195 (use_facet<_Num_get>(__that.getloc())).get(istreambuf_iterator<_CharT, _Traits>(__that.rdbuf()),
196 0, __that, __err, __val);
197 }
198 _STLP_CATCH_ALL {
199 __that._M_handle_exception(ios_base::badbit);
200 }
201 if (__err) __that.setstate(__err);
202 }
203 return __err;
204 }
205
206 _STLP_MOVE_TO_STD_NAMESPACE
207
208 template <class _CharT, class _Traits>
209 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (short& __val) {
210 long __lval;
211 _STLP_PRIV __get_num(*this, __lval);
212 if ( this->fail() ) {
213 return *this;
214 }
215 short __tmp = __STATIC_CAST(short, __lval);
216 unsigned short __uval = __STATIC_CAST(unsigned short, __lval);
217 // check if we lose digits
218 // if ((__val != __lval) && ((unsigned short)__val != __lval))
219 if ((__tmp != __lval) && ((long)__uval != __lval))
220 this->setstate(ios_base::failbit);
221 else
222 __val = __tmp;
223 return *this;
224 }
225
226 template <class _CharT, class _Traits>
227 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (int& __val) {
228 long __lval;
229 _STLP_PRIV __get_num(*this, __lval);
230 if ( this->fail() ) {
231 return *this;
232 }
233 int __tmp = __lval;
234 unsigned int __uval = __lval;
235 // check if we lose digits
236 // if ((__val != __lval) && ((unsigned int)__val != __lval))
237 if ((__tmp != __lval) && ((long)__uval != __lval))
238 this->setstate(ios_base::failbit);
239 else
240 __val = __tmp;
241 return *this;
242 }
243
244 template <class _CharT, class _Traits>
245 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (unsigned short& __val) {
246 _STLP_PRIV __get_num(*this, __val);
247 return *this;
248 }
249
250 template <class _CharT, class _Traits>
251 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (unsigned int& __val) {
252 _STLP_PRIV __get_num(*this, __val);
253 return *this;
254 }
255
256 template <class _CharT, class _Traits>
257 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (long& __val) {
258 _STLP_PRIV __get_num(*this, __val);
259 return *this;
260 }
261
262 template <class _CharT, class _Traits>
263 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (unsigned long& __val) {
264 _STLP_PRIV __get_num(*this, __val);
265 return *this;
266 }
267
268 #if defined (_STLP_LONG_LONG)
269 template <class _CharT, class _Traits>
270 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (_STLP_LONG_LONG& __val) {
271 _STLP_PRIV __get_num(*this, __val);
272 return *this;
273 }
274
275 template <class _CharT, class _Traits>
276 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (unsigned _STLP_LONG_LONG& __val) {
277 _STLP_PRIV __get_num(*this, __val);
278 return *this;
279 }
280 #endif
281 template <class _CharT, class _Traits>
282 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (float& __val) {
283 _STLP_PRIV __get_num(*this, __val);
284 return *this;
285 }
286 template <class _CharT, class _Traits>
287 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (double& __val) {
288 _STLP_PRIV __get_num(*this, __val);
289 return *this;
290 }
291 #if !defined (_STLP_NO_LONG_DOUBLE)
292 template <class _CharT, class _Traits>
293 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (long double& __val) {
294 _STLP_PRIV __get_num(*this, __val);
295 return *this;
296 }
297 #endif
298 #if !defined (_STLP_NO_BOOL)
299 template <class _CharT, class _Traits>
300 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (bool& __val) {
301 _STLP_PRIV __get_num(*this, __val);
302 return *this;
303 }
304 #endif
305
306 template <class _CharT, class _Traits>
307 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::operator>> (void*& __val) {
308 _STLP_PRIV __get_num(*this, __val);
309 return *this;
310 }
311
312 // Unformatted input
313
314 template <class _CharT, class _Traits>
315 __BIS_int_type__
peek()316 basic_istream<_CharT, _Traits>::peek() {
317 typename _Traits::int_type __tmp = _Traits::eof();
318
319 this->_M_gcount = 0;
320 sentry __sentry(*this, _No_Skip_WS());
321
322 if (__sentry) {
323 _STLP_TRY {
324 __tmp = this->rdbuf()->sgetc();
325 }
326 _STLP_CATCH_ALL {
327 this->_M_handle_exception(ios_base::badbit);
328 }
329 if (this->_S_eof(__tmp))
330 this->setstate(ios_base::eofbit);
331 }
332
333 return __tmp;
334 }
335
336
337 template <class _CharT, class _Traits>
338 __BIS_int_type__
get()339 basic_istream<_CharT, _Traits>::get() {
340 typename _Traits::int_type __tmp = _Traits::eof();
341 sentry __sentry(*this, _No_Skip_WS());
342 this->_M_gcount = 0;
343
344 if (__sentry) {
345 _STLP_TRY {
346 __tmp = this->rdbuf()->sbumpc();
347 }
348 _STLP_CATCH_ALL {
349 this->_M_handle_exception(ios_base::badbit);
350 }
351
352 if (!this->_S_eof(__tmp))
353 this->_M_gcount = 1;
354 }
355
356 if (_M_gcount == 0)
357 this->setstate(ios_base::eofbit | ios_base::failbit);
358
359 return __tmp;
360 }
361
362 template <class _CharT, class _Traits>
363 basic_istream<_CharT, _Traits>&
get(_CharT & __c)364 basic_istream<_CharT, _Traits>::get(_CharT& __c) {
365 sentry __sentry(*this, _No_Skip_WS());
366 this->_M_gcount = 0;
367
368 if (__sentry) {
369 typename _Traits::int_type __tmp = _Traits::eof();
370 _STLP_TRY {
371 __tmp = this->rdbuf()->sbumpc();
372 }
373 _STLP_CATCH_ALL {
374 this->_M_handle_exception(ios_base::badbit);
375 }
376
377 if (!this->_S_eof(__tmp)) {
378 this->_M_gcount = 1;
379 __c = _Traits::to_char_type(__tmp);
380 }
381 }
382
383 if (this->_M_gcount == 0)
384 this->setstate(ios_base::eofbit | ios_base::failbit);
385
386 return *this;
387 }
388
389
390 // Read characters and discard them. The standard specifies a single
391 // function with two arguments, each with a default. We instead use
392 // three overloded functions, because it's possible to implement the
393 // first two more efficiently than the fully general third version.
394 template <class _CharT, class _Traits>
ignore()395 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::ignore() {
396 sentry __sentry(*this, _No_Skip_WS());
397 this->_M_gcount = 0;
398
399 if (__sentry) {
400 int_type __c;
401 _STLP_TRY {
402 __c = this->rdbuf()->sbumpc();
403 }
404 _STLP_CATCH_ALL {
405 this->_M_handle_exception(ios_base::badbit);
406 return *this;
407 }
408
409 if (!this->_S_eof(__c))
410 this->_M_gcount = 1;
411 else
412 this->setstate(ios_base::eofbit);
413 }
414
415 return *this;
416 }
417
418 // Putback
419
420 template <class _CharT, class _Traits>
421 basic_istream<_CharT, _Traits>&
putback(_CharT __c)422 basic_istream<_CharT, _Traits>::putback(_CharT __c) {
423 this->_M_gcount = 0;
424 sentry __sentry(*this, _No_Skip_WS());
425
426 if (__sentry) {
427 typename _Traits::int_type __tmp = _Traits::eof();
428 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
429 // if (!__buf || this->_S_eof(__buf->sputbackc(__c)))
430 if (__buf) {
431 _STLP_TRY {
432 __tmp = __buf->sputbackc(__c);
433 }
434 _STLP_CATCH_ALL {
435 this->_M_handle_exception(ios_base::badbit);
436 }
437 }
438 if (this->_S_eof(__tmp))
439 this->setstate(ios_base::badbit);
440 }
441 else
442 this->setstate(ios_base::failbit);
443
444 return *this;
445 }
446
447 template <class _CharT, class _Traits>
unget()448 basic_istream<_CharT, _Traits>& basic_istream<_CharT, _Traits>::unget() {
449 this->_M_gcount = 0;
450
451 sentry __sentry(*this, _No_Skip_WS());
452
453 if (__sentry) {
454 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
455 // if (!__buf || _Traits::eq_int_type(__buf->sungetc(), _Traits::eof()))
456 if (__buf) {
457 _STLP_TRY {
458 if (this->_S_eof(__buf->sungetc()))
459 this->setstate(ios_base::badbit);
460 }
461 _STLP_CATCH_ALL {
462 this->_M_handle_exception(ios_base::badbit);
463 }
464 } else
465 this->setstate(ios_base::badbit);
466 }
467 else
468 this->setstate(ios_base::failbit);
469
470 return *this;
471 }
472
473 // Positioning and buffer control.
474
475 template <class _CharT, class _Traits>
sync()476 int basic_istream<_CharT, _Traits>::sync() {
477 sentry __sentry(*this, _No_Skip_WS());
478
479 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
480 if (__buf) {
481 if (__buf->pubsync() == -1) {
482 this->setstate(ios_base::badbit);
483 return -1;
484 }
485 else
486 return 0;
487 }
488 else
489 return -1;
490 }
491
492 template <class _CharT, class _Traits>
493 __BIS_pos_type__
tellg()494 basic_istream<_CharT, _Traits>::tellg() {
495 sentry __sentry(*this, _No_Skip_WS());
496
497 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
498 return (__buf && !this->fail()) ? __buf->pubseekoff(0, ios_base::cur, ios_base::in)
499 : pos_type(-1);
500 }
501
502 template <class _CharT, class _Traits>
503 basic_istream<_CharT, _Traits>&
seekg(pos_type __pos)504 basic_istream<_CharT, _Traits>::seekg(pos_type __pos) {
505 sentry __sentry(*this, _No_Skip_WS());
506
507 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
508 if (!this->fail() && __buf) {
509 if (__buf->pubseekpos(__pos, ios_base::in) == pos_type(-1)) {
510 this->setstate(ios_base::failbit);
511 }
512 }
513 return *this;
514 }
515
516 template <class _CharT, class _Traits>
517 basic_istream<_CharT, _Traits>&
seekg(off_type __off,ios_base::seekdir __dir)518 basic_istream<_CharT, _Traits>::seekg(off_type __off, ios_base::seekdir __dir) {
519 sentry __sentry(*this, _No_Skip_WS());
520
521 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
522 if (!this->fail() && __buf)
523 __buf->pubseekoff(__off, __dir, ios_base::in);
524 return *this;
525 }
526
527 // Formatted input of characters and character arrays.
528
529 template <class _CharT, class _Traits>
_M_formatted_get(_CharT & __c)530 void basic_istream<_CharT, _Traits>::_M_formatted_get(_CharT& __c) {
531 // typename _Traits::int_type __tmp = _Traits::eof();
532
533 sentry __sentry(*this); // Skip whitespace.
534
535 if (__sentry) {
536 typename _Traits::int_type __tmp;// = _Traits::eof();
537
538 _STLP_TRY {
539 __tmp = this->rdbuf()->sbumpc();
540 }
541 _STLP_CATCH_ALL {
542 this->_M_handle_exception(ios_base::badbit);
543 return;
544 }
545
546 if (!this->_S_eof(__tmp))
547 __c = _Traits::to_char_type(__tmp);
548 else
549 this->setstate(ios_base::eofbit | ios_base::failbit);
550 }
551 }
552
553
554 //---------------------------------------------------------------------------
555 // istream's helper functions.
556
557 // A generic function for unbuffered input. We stop when we reach EOF,
558 // or when we have extracted _Num characters, or when the function object
559 // __is_delim return true. In the last case, it extracts the character
560 // for which __is_delim is true, if and only if __extract_delim is true.
561 // It appends a null character to the end of the string; this means that
562 // it may store up to _Num + 1 characters.
563 //
564 // __is_getline governs two corner cases: reading _Num characters without
565 // encountering delim or eof (in which case failbit is set if __is_getline
566 // is true); and reading _Num characters where the _Num+1'st character is
567 // eof (in which case eofbit is set if __is_getline is true).
568 //
569 // It is assumed that __is_delim never throws.
570 //
571 // Return value is the number of characters extracted, including the
572 // delimiter if it is extracted. Note that the number of characaters
573 // extracted isn't necessarily the same as the number stored.
574
575 _STLP_MOVE_TO_PRIV_NAMESPACE
576
577 template < class _CharT, class _Traits, class _Is_Delim>
578 streamsize _STLP_CALL
__read_unbuffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __buf,streamsize _Num,_CharT * __s,_Is_Delim __is_delim,bool __extract_delim,bool __append_null,bool __is_getline)579 __read_unbuffered(basic_istream<_CharT, _Traits>* __that, basic_streambuf<_CharT, _Traits>* __buf,
580 streamsize _Num, _CharT* __s,
581 _Is_Delim __is_delim,
582 bool __extract_delim, bool __append_null,
583 bool __is_getline)
584 {
585 streamsize __n = 0;
586 ios_base::iostate __status = 0;
587
588 typedef typename basic_istream<_CharT, _Traits>::int_type int_type;
589 // The operations that can potentially throw are sbumpc, snextc, and sgetc.
590 _STLP_TRY {
591 for (;;) {
592 if (__n == _Num) {
593 if (__is_getline) // didn't find delimiter as one of the _Num chars
594 __status |= ios_base::failbit;
595 break;
596 }
597 int_type __c = __buf->sbumpc(); // sschwarz
598
599 if (__that->_S_eof(__c)) {
600 if (__n < _Num || __is_getline)
601 __status |= ios_base::eofbit;
602 break;
603 } else if (__is_delim(_Traits::to_char_type(__c))) {
604 if (__extract_delim) { // Extract and discard current character.
605 ++__n;
606 } else if ( !__pushback(__buf, _Traits::to_char_type(__c)) ) { // leave delimiter
607 __status |= ios_base::failbit;
608 }
609 break;
610 }
611 // regular character
612 *__s++ = _Traits::to_char_type(__c);
613 ++__n;
614 }
615 }
616 _STLP_CATCH_ALL {
617 __that->_M_handle_exception(ios_base::badbit);
618 *__s = _STLP_DEFAULT_CONSTRUCTED(_CharT);
619 return __n;
620 }
621
622 if (__append_null)
623 *__s = _STLP_DEFAULT_CONSTRUCTED(_CharT);
624 if (__status)
625 __that->setstate(__status); // This might throw.
626 return __n;
627 }
628
629 // Much like __read_unbuffered, but with one additional function object:
630 // __scan_delim(first, last) returns the first pointer p in [first, last)
631 // such that __is_delim(p) is true.
632
633 template < class _CharT, class _Traits, class _Is_Delim, class _Scan_Delim>
634 streamsize _STLP_CALL
__read_buffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __buf,streamsize _Num,_CharT * __s,_Is_Delim __is_delim,_Scan_Delim __scan_delim,bool __extract_delim,bool __append_null,bool __is_getline)635 __read_buffered(basic_istream<_CharT, _Traits>* __that, basic_streambuf<_CharT, _Traits>* __buf,
636 streamsize _Num, _CharT* __s,
637 _Is_Delim __is_delim, _Scan_Delim __scan_delim,
638 bool __extract_delim, bool __append_null,
639 bool __is_getline) {
640 streamsize __n = 0;
641 ios_base::iostate __status = 0;
642 bool __done = false;
643
644 _STLP_TRY {
645 while (__buf->_M_egptr() != __buf->_M_gptr() && !__done) {
646 const _CharT* __first = __buf->_M_gptr();
647 const _CharT* __last = __buf->_M_egptr();
648 //casting numeric_limits<ptrdiff_t>::max to streamsize only works is ptrdiff_t is signed or streamsize representation
649 //is larger than ptrdiff_t one.
650 _STLP_STATIC_ASSERT((sizeof(streamsize) > sizeof(ptrdiff_t)) ||
651 ((sizeof(streamsize) == sizeof(ptrdiff_t)) && numeric_limits<ptrdiff_t>::is_signed))
652 ptrdiff_t __request = __STATIC_CAST(ptrdiff_t, (min) (__STATIC_CAST(streamsize, (numeric_limits<ptrdiff_t>::max)()), _Num - __n));
653
654 const _CharT* __p = __scan_delim(__first, __last);
655 ptrdiff_t __chunk = (min) (ptrdiff_t(__p - __first), __request);
656 _Traits::copy(__s, __first, __chunk);
657 __s += __chunk;
658 __n += __chunk;
659 __buf->_M_gbump((int)__chunk);
660
661 // We terminated by finding delim.
662 if (__p != __last && __p - __first <= __request) {
663 if (__extract_delim) {
664 __n += 1;
665 __buf->_M_gbump(1);
666 }
667 __done = true;
668 }
669
670 // We terminated by reading all the characters we were asked for.
671 else if (__n == _Num) {
672
673 // Find out if we have reached eof. This matters for getline.
674 if (__is_getline) {
675 if (__chunk == __last - __first) {
676 if (__that->_S_eof(__buf->sgetc()))
677 __status |= ios_base::eofbit;
678 }
679 else
680 __status |= ios_base::failbit;
681 }
682 __done = true;
683 }
684
685 // The buffer contained fewer than _Num - __n characters. Either we're
686 // at eof, or we should refill the buffer and try again.
687 else {
688 if (__that->_S_eof(__buf->sgetc())) {
689 __status |= ios_base::eofbit;
690 __done = true;
691 }
692 }
693 } // Close the while loop.
694 }
695 _STLP_CATCH_ALL {
696 __that->_M_handle_exception(ios_base::badbit);
697 __done = true;
698 }
699
700 if (__done) {
701 if (__append_null)
702 *__s = _STLP_DEFAULT_CONSTRUCTED(_CharT);
703 if (__status != 0)
704 __that->setstate(__status); // This might throw.
705 return __n;
706 }
707
708 // If execution has reached this point, then we have an empty buffer but
709 // we have not reached eof. What that means is that the streambuf has
710 // decided to switch from buffered to unbuffered input. We switch to
711 // to __read_unbuffered.
712
713 return __n + __read_unbuffered(__that, __buf, _Num - __n, __s, __is_delim,
714 __extract_delim,__append_null,__is_getline);
715 }
716
717 _STLP_MOVE_TO_STD_NAMESPACE
718
719 template <class _CharT, class _Traits>
720 basic_istream<_CharT, _Traits>&
get(_CharT * __s,streamsize __n,_CharT __delim)721 basic_istream<_CharT, _Traits>::get(_CharT* __s, streamsize __n,
722 _CharT __delim) {
723 sentry __sentry(*this, _No_Skip_WS());
724 this->_M_gcount = 0;
725
726 if (__sentry) {
727 if (__n > 0) {
728 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
729
730 if (__buf->egptr() != __buf->gptr())
731 this->_M_gcount =
732 _STLP_PRIV __read_buffered(this, __buf, __n - 1, __s,
733 _STLP_PRIV _Eq_char_bound<_Traits>(__delim),
734 _STLP_PRIV _Scan_for_char_val<_Traits>(__delim),
735 false, true, false);
736 else
737 this->_M_gcount =
738 _STLP_PRIV __read_unbuffered(this, __buf, __n - 1, __s,
739 _STLP_PRIV _Eq_char_bound<_Traits>(__delim),
740 false, true, false);
741 }
742 }
743
744 if (this->_M_gcount == 0)
745 this->setstate(ios_base::failbit);
746
747 return *this;
748 }
749
750 // Getline is essentially identical to get, except that it extracts
751 // the delimiter.
752 template <class _CharT, class _Traits>
753 basic_istream<_CharT, _Traits>&
getline(_CharT * __s,streamsize __n,_CharT __delim)754 basic_istream<_CharT, _Traits>::getline(_CharT* __s, streamsize __n,
755 _CharT __delim) {
756 sentry __sentry(*this, _No_Skip_WS());
757 this->_M_gcount = 0;
758
759 if (__sentry) {
760 if (__n > 0) {
761 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
762 this->_M_gcount = __buf->egptr() != __buf->gptr()
763 ? _STLP_PRIV __read_buffered(this, __buf, __n - 1, __s,
764 _STLP_PRIV _Eq_char_bound<_Traits>(__delim),
765 _STLP_PRIV _Scan_for_char_val<_Traits>(__delim),
766 true, true, true)
767 : _STLP_PRIV __read_unbuffered(this, __buf, __n - 1, __s,
768 _STLP_PRIV _Eq_char_bound<_Traits>(__delim),
769 true, true, true);
770 }
771 }
772
773 if (this->_M_gcount == 0)
774 this->setstate(ios_base::failbit);
775
776 return *this;
777 }
778
779 // Read n characters. We don't look for any delimiter, and we don't
780 // put in a terminating null character.
781 template <class _CharT, class _Traits>
782 basic_istream<_CharT, _Traits>&
read(char_type * __s,streamsize __n)783 basic_istream<_CharT, _Traits>::read(char_type* __s, streamsize __n) {
784 sentry __sentry(*this, _No_Skip_WS());
785 this->_M_gcount = 0;
786
787 if (__sentry && !this->eof()) {
788 basic_streambuf<_CharT, _Traits>*__buf = this->rdbuf();
789 if (__buf->gptr() != __buf->egptr())
790 _M_gcount
791 = _STLP_PRIV __read_buffered(this, __buf, __n, __s,
792 _STLP_PRIV _Constant_unary_fun<bool, int_type>(false),
793 _STLP_PRIV _Project2nd<const _CharT*, const _CharT*>(),
794 false, false, false);
795 else
796 _M_gcount
797 = _STLP_PRIV __read_unbuffered(this, __buf, __n, __s,
798 _STLP_PRIV _Constant_unary_fun<bool, int_type>(false),
799 false, false, false);
800 }
801 else
802 this->setstate(ios_base::failbit);
803
804 if (this->eof())
805 this->setstate(ios_base::eofbit | ios_base::failbit);
806
807 return *this;
808 }
809
810
811 // Read n or fewer characters. We don't look for any delimiter, and
812 // we don't put in a terminating null character.
813 template <class _CharT, class _Traits>
814 streamsize
readsome(char_type * __s,streamsize __nmax)815 basic_istream<_CharT, _Traits>::readsome(char_type* __s, streamsize __nmax) {
816 sentry __sentry(*this, _No_Skip_WS());
817 this->_M_gcount = 0;
818
819 if (__sentry && !this->eof() && __nmax >= 0) {
820
821 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
822 streamsize __avail = __buf->in_avail();
823
824 // fbp : isn't full-blown setstate required here ?
825 if (__avail == -1)
826 this->_M_setstate_nothrow(ios_base::eofbit);
827
828 else if (__avail != 0) {
829
830 if (__buf->gptr() != __buf->egptr())
831 _M_gcount
832 = _STLP_PRIV __read_buffered(this, __buf, (min) (__avail, __nmax), __s,
833 _STLP_PRIV _Constant_unary_fun<bool, int_type>(false),
834 _STLP_PRIV _Project2nd<const _CharT*, const _CharT*>(),
835 false, false, false);
836 else
837 _M_gcount
838 = _STLP_PRIV __read_unbuffered(this, __buf, (min) (__avail, __nmax), __s,
839 _STLP_PRIV _Constant_unary_fun<bool, int_type>(false),
840 false, false, false);
841 }
842 }
843 else {
844 // fbp : changed so that failbit is set only there, to pass Dietmar's test
845 if (this->eof())
846 this->setstate(ios_base::eofbit | ios_base::failbit);
847 else
848 this->setstate(ios_base::failbit);
849 }
850
851 // if (this->eof())
852 // this->setstate(ios_base::eofbit | ios_base::failbit);
853
854 return _M_gcount;
855 }
856
857 template <class _CharT, class _Traits>
_M_formatted_get(_CharT * __s)858 void basic_istream<_CharT, _Traits>::_M_formatted_get(_CharT* __s) {
859 sentry __sentry(*this); // Skip whitespace.
860
861 if (__sentry) {
862 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
863 streamsize __nmax = this->width() > 0
864 ? this->width() - 1
865 : ((numeric_limits<streamsize>::max)() / sizeof(_CharT)) - 1;
866
867 streamsize __n = __buf->gptr() != __buf->egptr()
868 ? _STLP_PRIV __read_buffered(this, __buf, __nmax, __s,
869 _STLP_PRIV _Is_wspace_null<_Traits>(this->_M_ctype_facet()),
870 _STLP_PRIV _Scan_wspace_null<_Traits>(this->_M_ctype_facet()),
871 false, true, false)
872 : _STLP_PRIV __read_unbuffered(this, __buf, __nmax, __s,
873 _STLP_PRIV _Is_wspace_null<_Traits>(this->_M_ctype_facet()),
874 false, true, false);
875 if (__n == 0)
876 this->setstate(ios_base::failbit);
877 }
878 this->width(0);
879 }
880
881 // A generic unbuffered function for ignoring characters. We stop
882 // when we reach EOF, or when the function object __is_delim returns
883 // true. In the last case, it extracts the character for which
884 // __is_delim is true, if and only if __extract_delim is true.
885
886 template < class _CharT, class _Traits, class _Is_Delim>
887 void _STLP_CALL
_M_ignore_unbuffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __buf,_Is_Delim __is_delim,bool __extract_delim,bool __set_failbit)888 _M_ignore_unbuffered(basic_istream<_CharT, _Traits>* __that,
889 basic_streambuf<_CharT, _Traits>* __buf,
890 _Is_Delim __is_delim,
891 bool __extract_delim, bool __set_failbit) {
892 bool __done = false;
893 ios_base::iostate __status = 0;
894 typedef typename basic_istream<_CharT, _Traits>::int_type int_type;
895
896 _STLP_TRY {
897 while (!__done) {
898 int_type __c = __buf->sbumpc();
899
900 if (__that->_S_eof(__c)) {
901 __done = true;
902 __status |= __set_failbit ? ios_base::eofbit | ios_base::failbit
903 : ios_base::eofbit;
904 }
905
906 else if (__is_delim(_Traits::to_char_type(__c))) {
907 __done = true;
908 if (!__extract_delim)
909 if (__that->_S_eof(__buf->sputbackc(_Traits::to_char_type(__c))))
910 __status |= ios_base::failbit;
911 }
912 }
913 }
914 _STLP_CATCH_ALL {
915 __that->_M_handle_exception(ios_base::badbit);
916 }
917
918 __that->setstate(__status);
919 }
920
921 // A generic buffered function for ignoring characters. Much like
922 // _M_ignore_unbuffered, but with one additional function object:
923 // __scan_delim(first, last) returns the first pointer p in [first,
924 // last) such that __is_delim(p) is true.
925
926 template < class _CharT, class _Traits, class _Is_Delim, class _Scan_Delim>
927 void _STLP_CALL
_M_ignore_buffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __buf,_Is_Delim __is_delim,_Scan_Delim __scan_delim,bool __extract_delim,bool __set_failbit)928 _M_ignore_buffered(basic_istream<_CharT, _Traits>* __that,
929 basic_streambuf<_CharT, _Traits>* __buf,
930 _Is_Delim __is_delim, _Scan_Delim __scan_delim,
931 bool __extract_delim, bool __set_failbit) {
932 bool __at_eof = false;
933 bool __found_delim = false;
934
935 _STLP_TRY {
936 while (__buf->_M_egptr() != __buf->_M_gptr() && !__at_eof && !__found_delim) {
937 const _CharT* __p = __scan_delim(__buf->_M_gptr(), __buf->_M_egptr());
938 __buf->_M_gbump((int)(__p - __buf->_M_gptr()));
939
940 if (__p != __buf->_M_egptr()) { // We found delim, so we're done.
941 if (__extract_delim)
942 __buf->_M_gbump(1);
943 __found_delim = true;
944 }
945
946 else // No delim. Try to refil the buffer.
947 __at_eof = __that->_S_eof(__buf->sgetc());
948 } // Close the while loop.
949 }
950 _STLP_CATCH_ALL {
951 __that->_M_handle_exception(ios_base::badbit);
952 return;
953 }
954
955 if (__at_eof) {
956 __that->setstate(__set_failbit ? ios_base::eofbit | ios_base::failbit
957 : ios_base::eofbit);
958 return;
959 }
960 if (__found_delim)
961 return;
962
963 // If execution has reached this point, then we have an empty buffer but
964 // we have not reached eof. What that means is that the streambuf has
965 // decided to switch from a buffered to an unbuffered mode. We switch
966 // to _M_ignore_unbuffered.
967 _M_ignore_unbuffered(__that, __buf, __is_delim, __extract_delim, __set_failbit);
968 }
969
970 // Overloaded versions of _M_ignore_unbuffered and _M_ignore_unbuffered
971 // with an explicit count _Num. Return value is the number of
972 // characters extracted.
973 //
974 // The function object __max_chars takes two arguments, _Num and __n
975 // (the latter being the number of characters we have already read),
976 // and returns the maximum number of characters to read from the buffer.
977 // We parameterize _M_ignore_buffered so that we can use it for both
978 // bounded and unbounded input; for the former the function object should
979 // be minus<>, and for the latter it should return a constant maximum value.
980
981 template < class _CharT, class _Traits, class _Max_Chars, class _Is_Delim>
982 streamsize _STLP_CALL
_M_ignore_unbuffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __buf,streamsize _Num,_Max_Chars __max_chars,_Is_Delim __is_delim,bool __extract_delim,bool __set_failbit)983 _M_ignore_unbuffered(basic_istream<_CharT, _Traits>* __that,
984 basic_streambuf<_CharT, _Traits>* __buf,
985 streamsize _Num, _Max_Chars __max_chars,
986 _Is_Delim __is_delim,
987 bool __extract_delim, bool __set_failbit) {
988 streamsize __n = 0;
989 ios_base::iostate __status = 0;
990 typedef typename basic_istream<_CharT, _Traits>::int_type int_type;
991
992 _STLP_TRY {
993 while (__max_chars(_Num, __n) > 0) {
994 int_type __c = __buf->sbumpc();
995
996 if (__that->_S_eof(__c)) {
997 __status |= __set_failbit ? ios_base::eofbit | ios_base::failbit
998 : ios_base::eofbit;
999 break;
1000 }
1001
1002 else if (__is_delim(_Traits::to_char_type(__c))) {
1003 if (__extract_delim)
1004 ++__n;
1005 else if (__that->_S_eof(__buf->sputbackc(_Traits::to_char_type(__c))))
1006 __status |= ios_base::failbit;
1007
1008 break;
1009 }
1010 // fbp : added counter increment to pass Dietmar's test
1011 ++__n;
1012 }
1013 }
1014 _STLP_CATCH_ALL {
1015 __that->_M_handle_exception(ios_base::badbit);
1016 }
1017
1018 if (__status)
1019 __that->setstate(__status); // This might throw.
1020 return __n;
1021 }
1022
1023 template < class _CharT, class _Traits, class _Max_Chars, class _Is_Delim, class _Scan_Delim>
1024 streamsize _STLP_CALL
_M_ignore_buffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __buf,streamsize _Num,_Max_Chars __max_chars,_Is_Delim __is_delim,_Scan_Delim __scan_delim,bool __extract_delim,bool __set_failbit)1025 _M_ignore_buffered(basic_istream<_CharT, _Traits>* __that,
1026 basic_streambuf<_CharT, _Traits>* __buf,
1027 streamsize _Num,
1028 _Max_Chars __max_chars,
1029 _Is_Delim __is_delim, _Scan_Delim __scan_delim,
1030 bool __extract_delim, bool __set_failbit) {
1031 streamsize __n = 0;
1032 bool __at_eof = false;
1033 bool __done = false;
1034
1035 _STLP_TRY {
1036 while (__buf->_M_egptr() != __buf->_M_gptr() && !__done) {
1037 ptrdiff_t __avail = __buf->_M_egptr() - __buf->_M_gptr();
1038 streamsize __m = __max_chars(_Num, __n);
1039
1040 if (__avail >= __m) { // We have more characters than we need.
1041 const _CharT* __last = __buf->_M_gptr() + __STATIC_CAST(ptrdiff_t, __m);
1042 const _CharT* __p = __scan_delim(__buf->_M_gptr(), __last);
1043 ptrdiff_t __chunk = __p - __buf->_M_gptr();
1044 __n += __chunk;
1045 __buf->_M_gbump((int)__chunk);
1046
1047 if (__extract_delim && __p != __last) {
1048 __n += 1;
1049 __buf->_M_gbump(1);
1050 }
1051
1052 __done = true;
1053 }
1054
1055 else {
1056 const _CharT* __p = __scan_delim(__buf->_M_gptr(), __buf->_M_egptr());
1057 ptrdiff_t __chunk = __p - __buf->_M_gptr();
1058 __n += __chunk;
1059 __buf->_M_gbump((int)__chunk);
1060
1061 if (__p != __buf->_M_egptr()) { // We found delim.
1062 if (__extract_delim) {
1063 __n += 1;
1064 __buf->_M_gbump(1);
1065 }
1066
1067 __done = true;
1068 }
1069
1070 // We didn't find delim. Try to refill the buffer.
1071 else if (__that->_S_eof(__buf->sgetc())) {
1072 __done = true;
1073 __at_eof = true;
1074 }
1075 }
1076 } // Close the while loop.
1077 }
1078 _STLP_CATCH_ALL {
1079 __that->_M_handle_exception(ios_base::badbit);
1080 return __n;
1081 }
1082
1083 if (__at_eof)
1084 __that->setstate(__set_failbit ? ios_base::eofbit | ios_base::failbit
1085 : ios_base::eofbit);
1086
1087 if (__done)
1088 return __n;
1089
1090 // If execution has reached this point, then we have an empty buffer but
1091 // we have not reached eof. What that means is that the streambuf has
1092 // decided to switch from buffered to unbuffered input. We switch to
1093 // to _M_ignore_unbuffered.
1094
1095 return __n + _M_ignore_unbuffered(__that, __buf, _Num, __max_chars,
1096 __is_delim, __extract_delim, __set_failbit);
1097 }
1098
1099
1100 template <class _CharT, class _Traits>
1101 basic_istream<_CharT, _Traits>&
ignore(streamsize __n)1102 basic_istream<_CharT, _Traits>::ignore(streamsize __n) {
1103 sentry __sentry(*this, _No_Skip_WS());
1104 this->_M_gcount = 0;
1105
1106 if (__sentry) {
1107 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
1108 typedef _STLP_PRIV _Constant_unary_fun<bool, int_type> _Const_bool;
1109 typedef _STLP_PRIV _Constant_binary_fun<streamsize, streamsize, streamsize> _Const_streamsize;
1110 const streamsize __maxss = (numeric_limits<streamsize>::max)();
1111
1112 if (__n == (numeric_limits<int>::max)()) {
1113 if (__buf->gptr() != __buf->egptr())
1114 _M_gcount = _M_ignore_buffered(this, __buf,
1115 __maxss, _Const_streamsize(__maxss),
1116 _Const_bool(false),
1117 _STLP_PRIV _Project2nd<const _CharT*, const _CharT*>(),
1118 false, false);
1119 else
1120 _M_gcount = _M_ignore_unbuffered(this, __buf,
1121 __maxss, _Const_streamsize(__maxss),
1122 _Const_bool(false), false, false);
1123 }
1124 else {
1125 if (__buf->gptr() != __buf->egptr())
1126 _M_gcount = _M_ignore_buffered(this, __buf,
1127 __n, minus<streamsize>(),
1128 _Const_bool(false),
1129 _STLP_PRIV _Project2nd<const _CharT*, const _CharT*>(),
1130 false, false);
1131 else
1132 _M_gcount = _M_ignore_unbuffered(this, __buf, __n, minus<streamsize>(),
1133 _Const_bool(false), false, false);
1134 }
1135 }
1136
1137 return *this;
1138 }
1139
1140 template <class _CharT, class _Traits>
1141 basic_istream<_CharT, _Traits>&
ignore(streamsize __n,int_type __delim)1142 basic_istream<_CharT, _Traits>::ignore(streamsize __n, int_type __delim) {
1143 sentry __sentry(*this, _No_Skip_WS());
1144 this->_M_gcount = 0;
1145
1146 if (__sentry) {
1147 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
1148 typedef _STLP_PRIV _Constant_unary_fun<bool, int_type> _Const_bool;
1149 typedef _STLP_PRIV _Constant_binary_fun<streamsize, streamsize, streamsize>
1150 _Const_streamsize;
1151 const streamsize __maxss = (numeric_limits<streamsize>::max)();
1152
1153 if (__n == (numeric_limits<int>::max)()) {
1154 if (__buf->gptr() != __buf->egptr())
1155 _M_gcount = _M_ignore_buffered(this, __buf,
1156 __maxss, _Const_streamsize(__maxss),
1157 _STLP_PRIV _Eq_int_bound<_Traits>(__delim),
1158 _STLP_PRIV _Scan_for_int_val<_Traits>(__delim),
1159 true, false);
1160 else
1161 _M_gcount = _M_ignore_unbuffered(this, __buf,
1162 __maxss, _Const_streamsize(__maxss),
1163 _STLP_PRIV _Eq_int_bound<_Traits>(__delim),
1164 true, false);
1165 }
1166 else {
1167 if (__buf->gptr() != __buf->egptr())
1168 _M_gcount = _M_ignore_buffered(this, __buf,
1169 __n, minus<streamsize>(),
1170 _STLP_PRIV _Eq_int_bound<_Traits>(__delim),
1171 _STLP_PRIV _Scan_for_int_val<_Traits>(__delim),
1172 true, false);
1173 else
1174 _M_gcount = _M_ignore_unbuffered(this, __buf, __n, minus<streamsize>(),
1175 _STLP_PRIV _Eq_int_bound<_Traits>(__delim),
1176 true, false);
1177 }
1178 }
1179
1180 return *this;
1181 }
1182
1183 // This member function does not construct a sentry object, because
1184 // it is called from sentry's constructor.
1185 template <class _CharT, class _Traits>
_M_skip_whitespace(bool __set_failbit)1186 void basic_istream<_CharT, _Traits>::_M_skip_whitespace(bool __set_failbit) {
1187 basic_streambuf<_CharT, _Traits>* __buf = this->rdbuf();
1188 if (!__buf)
1189 this->setstate(ios_base::badbit);
1190 else if (__buf->gptr() != __buf->egptr())
1191 _M_ignore_buffered(this, __buf,
1192 _STLP_PRIV _Is_not_wspace<_Traits>(this->_M_ctype_facet()),
1193 _STLP_PRIV _Scan_for_not_wspace<_Traits>(this->_M_ctype_facet()),
1194 false, __set_failbit);
1195 else
1196 _M_ignore_unbuffered(this, __buf,
1197 _STLP_PRIV _Is_not_wspace<_Traits>(this->_M_ctype_facet()),
1198 false, __set_failbit);
1199 }
1200
1201
1202 // This is a very simple loop that reads characters from __src and puts
1203 // them into __dest. It looks complicated because of the (standard-
1204 // mandated) exception handling policy.
1205 //
1206 // We stop when we get an exception, when we fail to insert into the
1207 // output streambuf, or when __is_delim is true.
1208
1209 _STLP_MOVE_TO_PRIV_NAMESPACE
1210
1211 template < class _CharT, class _Traits, class _Is_Delim>
1212 streamsize _STLP_CALL
__copy_unbuffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __src,basic_streambuf<_CharT,_Traits> * __dest,_Is_Delim __is_delim,bool __extract_delim,bool __rethrow)1213 __copy_unbuffered(basic_istream<_CharT, _Traits>* __that, basic_streambuf<_CharT, _Traits>* __src,
1214 basic_streambuf<_CharT, _Traits>* __dest,
1215 _Is_Delim __is_delim,
1216 bool __extract_delim, bool __rethrow) {
1217 streamsize __extracted = 0;
1218 ios_base::iostate __status = 0;
1219 typedef typename basic_istream<_CharT, _Traits>::int_type int_type;
1220 int_type __c;
1221
1222 _STLP_TRY {
1223 for (;;) {
1224 // Get a character. If there's an exception, catch and (maybe) rethrow it.
1225 __c = __src->sbumpc();
1226
1227 // If we failed to get a character, then quit.
1228 if (__that->_S_eof(__c)) {
1229 __status |= ios_base::eofbit;
1230 break;
1231 }
1232 // If it's the delimiter, then quit.
1233 else if (__is_delim(_Traits::to_char_type(__c))) {
1234 if (!__extract_delim && !__pushback(__src, _Traits::to_char_type(__c)))
1235 __status |= ios_base::failbit;
1236 break;
1237 }
1238 else {
1239 // Try to put the character in the output streambuf.
1240 bool __failed = false;
1241 _STLP_TRY {
1242 if (!__that->_S_eof(__dest->sputc(_Traits::to_char_type(__c))))
1243 ++__extracted;
1244 else
1245 __failed = true;
1246 }
1247 _STLP_CATCH_ALL {
1248 __failed = true;
1249 }
1250
1251 // If we failed to put the character in the output streambuf, then
1252 // try to push it back to the input streambuf.
1253 if (__failed && !__pushback(__src, _Traits::to_char_type(__c)))
1254 __status |= ios_base::failbit;
1255
1256 // fbp : avoiding infinite loop in io-27-6-1-2-3.exp
1257 if (__failed)
1258 break;
1259 }
1260
1261 } /* for (;;) */
1262
1263 }
1264 // fbp : this try/catch moved here in reasonable assumption
1265 // __is_delim never throw (__pushback is guaranteed not to)
1266 _STLP_CATCH_ALL {
1267 // See 27.6.1.2.3, paragraph 13.
1268 if (__rethrow && __extracted == 0)
1269 __that->_M_handle_exception(ios_base::failbit);
1270 }
1271 __that->setstate(__status);
1272 return __extracted;
1273 }
1274
1275 // Buffered copying from one streambuf to another. We copy the characters
1276 // in chunks, rather than one at a time. We still have to worry about all
1277 // of the error conditions we checked in __copy_unbuffered, plus one more:
1278 // the streambuf might decide to switch from a buffered to an unbuffered mode.
1279
1280 template < class _CharT, class _Traits, class _Is_Delim, class _Scan_Delim>
1281 streamsize _STLP_CALL
__copy_buffered(basic_istream<_CharT,_Traits> * __that,basic_streambuf<_CharT,_Traits> * __src,basic_streambuf<_CharT,_Traits> * __dest,_Scan_Delim __scan_delim,_Is_Delim __is_delim,bool __extract_delim,bool __rethrow)1282 __copy_buffered(basic_istream<_CharT, _Traits>* __that, basic_streambuf<_CharT, _Traits>* __src,
1283 basic_streambuf<_CharT, _Traits>* __dest,
1284 _Scan_Delim __scan_delim, _Is_Delim __is_delim,
1285 bool __extract_delim, bool __rethrow) {
1286 streamsize __extracted = 0;
1287 ios_base::iostate __status = 0;
1288 typedef typename basic_istream<_CharT, _Traits>::int_type int_type;
1289 //Borland compiler generates a warning if assignment because value is never used:
1290 int_type __c /*= _Traits::eof()*/;
1291 _CharT* __first = __src->_M_gptr();
1292 ptrdiff_t __avail = __src->_M_egptr() - __first;
1293 // fbp : introduced to move catch/try blocks out of the loop
1294 bool __do_handle_exceptions = false;
1295
1296 _STLP_TRY {
1297 for (;;) {
1298 const _CharT* __last = __scan_delim(__first, __src->_M_egptr());
1299
1300 // Try to copy the entire input buffer to the output buffer.
1301 streamsize __n = __dest->sputn(__first, __extract_delim && __last != __src->_M_egptr()
1302 ? (__last - __first) + 1
1303 : (__last - __first));
1304 __src->_M_gbump((int)__n);
1305 __extracted += __n;
1306
1307 // from this on, catch() will call _M_handle_exceptions()
1308 __do_handle_exceptions = true;
1309
1310 if (__n < __avail) // We found the delimiter, or else failed to
1311 break; // copy some characters.
1312
1313 __c = __src->sgetc();
1314
1315 // Three possibilities: we succeeded in refilling the buffer, or
1316 // we got EOF, or the streambuf has switched to unbuffered mode.
1317 __first = __src->_M_gptr();
1318 __avail = __src->_M_egptr() - __first;
1319
1320 if (__avail > 0)
1321 {} // dwa 1/16/00 -- suppress a Metrowerks warning
1322 else if (__that->_S_eof(__c)) {
1323 __status |= ios_base::eofbit;
1324 break;
1325 }
1326 else {
1327 return __extracted + __copy_unbuffered(__that, __src, __dest, __is_delim,
1328 __extract_delim, __rethrow);
1329 }
1330
1331 __do_handle_exceptions = false;
1332 }
1333 }
1334
1335 _STLP_CATCH_ALL {
1336 // See 27.6.1.2.3, paragraph 13.
1337 if (__rethrow && __do_handle_exceptions && __extracted == 0)
1338 __that->_M_handle_exception(ios_base::failbit);
1339 }
1340
1341 if (__status)
1342 __that->setstate(__status); // This might throw.
1343 return __extracted;
1344 }
1345
1346 _STLP_MOVE_TO_STD_NAMESPACE
1347
1348 template <class _CharT, class _Traits>
1349 basic_istream<_CharT, _Traits>&
1350 basic_istream<_CharT, _Traits>
get(basic_streambuf<_CharT,_Traits> & __dest,_CharT __delim)1351 ::get(basic_streambuf<_CharT, _Traits>& __dest, _CharT __delim) {
1352 sentry __sentry(*this, _No_Skip_WS());
1353 this->_M_gcount = 0;
1354
1355 if (__sentry) {
1356 basic_streambuf<_CharT, _Traits>* __src = this->rdbuf();
1357
1358 if (__src)
1359 this->_M_gcount = __src->egptr() != __src->gptr()
1360 ? _STLP_PRIV __copy_buffered(this, __src, &__dest,
1361 _STLP_PRIV _Scan_for_char_val<_Traits>(__delim),
1362 _STLP_PRIV _Eq_char_bound<_Traits>(__delim),
1363 false, false)
1364 : _STLP_PRIV __copy_unbuffered(this, __src, &__dest,
1365 _STLP_PRIV _Eq_char_bound<_Traits>(__delim),
1366 false, false);
1367 }
1368
1369 if (this->_M_gcount == 0)
1370 this->setstate(ios_base::failbit);
1371
1372 return *this;
1373 }
1374
1375 // Copying characters into a streambuf.
1376 template <class _CharT, class _Traits>
1377 basic_istream<_CharT, _Traits>&
1378 basic_istream<_CharT, _Traits>
1379 ::operator>>(basic_streambuf<_CharT, _Traits>* __dest) {
1380 streamsize __n = 0;
1381 typedef typename basic_istream<_CharT, _Traits>::sentry _Sentry;
1382 _Sentry __sentry(*this);
1383 if (__sentry) {
1384 basic_streambuf<_CharT, _Traits>* __src = this->rdbuf();
1385 if (__src && __dest)
1386 __n = __src->egptr() != __src->gptr()
1387 ? _STLP_PRIV __copy_buffered(this, __src, __dest,
1388 _STLP_PRIV _Project2nd<const _CharT*, const _CharT*>(),
1389 _STLP_PRIV _Constant_unary_fun<bool, int_type>(false),
1390 false, true)
1391 : _STLP_PRIV __copy_unbuffered(this, __src, __dest,
1392 _STLP_PRIV _Constant_unary_fun<bool, int_type>(false),
1393 false, true);
1394 }
1395
1396 if (__n == 0)
1397 this->setstate(ios_base::failbit);
1398
1399 return *this;
1400 }
1401
1402 // ----------------------------------------------------------------
1403 // basic_iostream<> class
1404 // ----------------------------------------------------------------
1405
1406 template <class _CharT, class _Traits>
1407 basic_iostream<_CharT, _Traits>
basic_iostream(basic_streambuf<_CharT,_Traits> * __buf)1408 ::basic_iostream(basic_streambuf<_CharT, _Traits>* __buf)
1409 : basic_ios<_CharT, _Traits>(),
1410 basic_istream<_CharT, _Traits>(__buf),
1411 basic_ostream<_CharT, _Traits>(__buf) {
1412 this->init(__buf);
1413 }
1414
1415 template <class _CharT, class _Traits>
~basic_iostream()1416 basic_iostream<_CharT, _Traits>::~basic_iostream()
1417 {}
1418
1419 _STLP_END_NAMESPACE
1420
1421 #undef __BIS_int_type__
1422 #undef __BIS_pos_type__
1423 #undef __BIS_off_type__
1424
1425 #endif /* _STLP_ISTREAM_C */
1426
1427 // Local Variables:
1428 // mode:C++
1429 // End:
1430