]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - include/locale
Import new libc++ to vendor branch.
[FreeBSD/FreeBSD.git] / include / locale
1 // -*- C++ -*-
2 //===-------------------------- locale ------------------------------------===//
3 //
4 //                     The LLVM Compiler Infrastructure
5 //
6 // This file is dual licensed under the MIT and the University of Illinois Open
7 // Source Licenses. See LICENSE.TXT for details.
8 //
9 //===----------------------------------------------------------------------===//
10
11 #ifndef _LIBCPP_LOCALE
12 #define _LIBCPP_LOCALE
13
14 /*
15     locale synopsis
16
17 namespace std
18 {
19
20 class locale
21 {
22 public:
23     // types:
24     class facet;
25     class id;
26
27     typedef int category;
28     static const category // values assigned here are for exposition only
29         none     = 0x000,
30         collate  = 0x010,
31         ctype    = 0x020,
32         monetary = 0x040,
33         numeric  = 0x080,
34         time     = 0x100,
35         messages = 0x200,
36         all = collate | ctype | monetary | numeric | time | messages;
37
38     // construct/copy/destroy:
39     locale() noexcept;
40     locale(const locale& other) noexcept;
41     explicit locale(const char* std_name);
42     explicit locale(const string& std_name);
43     locale(const locale& other, const char* std_name, category);
44     locale(const locale& other, const string& std_name, category);
45     template <class Facet> locale(const locale& other, Facet* f);
46     locale(const locale& other, const locale& one, category);
47
48     ~locale(); // not virtual
49
50     const locale& operator=(const locale& other) noexcept;
51
52     template <class Facet> locale combine(const locale& other) const;
53
54     // locale operations:
55     basic_string<char> name() const;
56     bool operator==(const locale& other) const;
57     bool operator!=(const locale& other) const;
58     template <class charT, class Traits, class Allocator>
59       bool operator()(const basic_string<charT,Traits,Allocator>& s1,
60                       const basic_string<charT,Traits,Allocator>& s2) const;
61
62     // global locale objects:
63     static locale global(const locale&);
64     static const locale& classic();
65 };
66
67 template <class Facet> const Facet& use_facet(const locale&);
68 template <class Facet> bool has_facet(const locale&) noexcept;
69
70 // 22.3.3, convenience interfaces:
71 template <class charT> bool isspace (charT c, const locale& loc);
72 template <class charT> bool isprint (charT c, const locale& loc);
73 template <class charT> bool iscntrl (charT c, const locale& loc);
74 template <class charT> bool isupper (charT c, const locale& loc);
75 template <class charT> bool islower (charT c, const locale& loc);
76 template <class charT> bool isalpha (charT c, const locale& loc);
77 template <class charT> bool isdigit (charT c, const locale& loc);
78 template <class charT> bool ispunct (charT c, const locale& loc);
79 template <class charT> bool isxdigit(charT c, const locale& loc);
80 template <class charT> bool isalnum (charT c, const locale& loc);
81 template <class charT> bool isgraph (charT c, const locale& loc);
82 template <class charT> charT toupper(charT c, const locale& loc);
83 template <class charT> charT tolower(charT c, const locale& loc);
84
85 template<class Codecvt, class Elem = wchar_t,
86          class Wide_alloc = allocator<Elem>,
87          class Byte_alloc = allocator<char>>
88 class wstring_convert
89 {
90 public:
91     typedef basic_string<char, char_traits<char>, Byte_alloc> byte_string;
92     typedef basic_string<Elem, char_traits<Elem>, Wide_alloc> wide_string;
93     typedef typename Codecvt::state_type                      state_type;
94     typedef typename wide_string::traits_type::int_type       int_type;
95
96     wstring_convert(Codecvt* pcvt = new Codecvt);
97     wstring_convert(Codecvt* pcvt, state_type state);
98     wstring_convert(const byte_string& byte_err,
99                     const wide_string& wide_err = wide_string());
100     ~wstring_convert();
101
102     wide_string from_bytes(char byte);
103     wide_string from_bytes(const char* ptr);
104     wide_string from_bytes(const byte_string& str);
105     wide_string from_bytes(const char* first, const char* last);
106
107     byte_string to_bytes(Elem wchar);
108     byte_string to_bytes(const Elem* wptr);
109     byte_string to_bytes(const wide_string& wstr);
110     byte_string to_bytes(const Elem* first, const Elem* last);
111
112     size_t converted() const;
113     state_type state() const;
114 };
115
116 template <class Codecvt, class Elem = wchar_t, class Tr = char_traits<Elem>>
117 class wbuffer_convert
118     : public basic_streambuf<Elem, Tr>
119 {
120 public:
121     typedef typename Tr::state_type state_type;
122
123     wbuffer_convert(streambuf* bytebuf = 0, Codecvt* pcvt = new Codecvt,
124                     state_type state = state_type());
125
126     streambuf* rdbuf() const;
127     streambuf* rdbuf(streambuf* bytebuf);
128
129     state_type state() const;
130 };
131
132 // 22.4.1 and 22.4.1.3, ctype:
133 class ctype_base;
134 template <class charT> class ctype;
135 template <> class ctype<char>; // specialization
136 template <class charT> class ctype_byname;
137 template <> class ctype_byname<char>; // specialization
138
139 class codecvt_base;
140 template <class internT, class externT, class stateT> class codecvt;
141 template <class internT, class externT, class stateT> class codecvt_byname;
142
143 // 22.4.2 and 22.4.3, numeric:
144 template <class charT, class InputIterator> class num_get;
145 template <class charT, class OutputIterator> class num_put;
146 template <class charT> class numpunct;
147 template <class charT> class numpunct_byname;
148
149 // 22.4.4, col lation:
150 template <class charT> class collate;
151 template <class charT> class collate_byname;
152
153 // 22.4.5, date and time:
154 class time_base;
155 template <class charT, class InputIterator> class time_get;
156 template <class charT, class InputIterator> class time_get_byname;
157 template <class charT, class OutputIterator> class time_put;
158 template <class charT, class OutputIterator> class time_put_byname;
159
160 // 22.4.6, money:
161 class money_base;
162 template <class charT, class InputIterator> class money_get;
163 template <class charT, class OutputIterator> class money_put;
164 template <class charT, bool Intl> class moneypunct;
165 template <class charT, bool Intl> class moneypunct_byname;
166
167 // 22.4.7, message retrieval:
168 class messages_base;
169 template <class charT> class messages;
170 template <class charT> class messages_byname;
171
172 }  // std
173
174 */
175
176 #include <__config>
177 #include <__locale>
178 #include <algorithm>
179 #include <memory>
180 #include <ios>
181 #include <streambuf>
182 #include <iterator>
183 #include <limits>
184 #if !__APPLE__
185 #include <cstdarg>
186 #endif
187 #include <cstdlib>
188 #include <ctime>
189 #if _WIN32
190 #include <support/win32/locale_win32.h>
191 #else // _WIN32
192 #include <nl_types.h>
193 #endif  // !_WIN32
194
195 #if __APPLE__
196 #include <Availability.h>
197 #endif
198
199 #include <__undef_min_max>
200
201 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
202 #pragma GCC system_header
203 #endif
204
205 _LIBCPP_BEGIN_NAMESPACE_STD
206
207 #if __APPLE__ || __FreeBSD__
208 #  define _LIBCPP_GET_C_LOCALE 0
209 #else
210 #  define _LIBCPP_GET_C_LOCALE __cloc()
211    // Get the C locale object
212    locale_t __cloc();
213 #define __cloc_defined
214 #endif
215
216 typedef _VSTD::remove_pointer<locale_t>::type __locale_struct;
217 typedef _VSTD::unique_ptr<__locale_struct, decltype(&freelocale)> __locale_unique_ptr;
218 #ifndef _LIBCPP_LOCALE__L_EXTENSIONS
219 typedef _VSTD::unique_ptr<__locale_struct, decltype(&uselocale)> __locale_raii;
220 #endif
221
222 // OSX has nice foo_l() functions that let you turn off use of the global
223 // locale.  Linux, not so much.  The following functions avoid the locale when
224 // that's possible and otherwise do the wrong thing.  FIXME.
225 #ifdef __linux__
226
227 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
228 decltype(MB_CUR_MAX_L(_VSTD::declval<locale_t>()))
229 inline _LIBCPP_INLINE_VISIBILITY
230 __mb_cur_max_l(locale_t __l)
231 {
232   return MB_CUR_MAX_L(__l);
233 }
234 #else  // _LIBCPP_LOCALE__L_EXTENSIONS
235 _LIBCPP_ALWAYS_INLINE inline
236 decltype(MB_CUR_MAX) __mb_cur_max_l(locale_t __l)
237 {
238   __locale_raii __current(uselocale(__l), uselocale);
239   return MB_CUR_MAX;
240 }
241 #endif // _LIBCPP_LOCALE__L_EXTENSIONS
242
243 _LIBCPP_ALWAYS_INLINE inline
244 wint_t __btowc_l(int __c, locale_t __l)
245 {
246 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
247   return btowc_l(__c, __l);
248 #else
249   __locale_raii __current(uselocale(__l), uselocale);
250   return btowc(__c);
251 #endif
252 }
253
254 _LIBCPP_ALWAYS_INLINE inline
255 int __wctob_l(wint_t __c, locale_t __l)
256 {
257 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
258   return wctob_l(__c, __l);
259 #else
260   __locale_raii __current(uselocale(__l), uselocale);
261   return wctob(__c);
262 #endif
263 }
264
265 _LIBCPP_ALWAYS_INLINE inline
266 size_t __wcsnrtombs_l(char *__dest, const wchar_t **__src, size_t __nwc,
267                       size_t __len, mbstate_t *__ps, locale_t __l)
268 {
269 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
270   return wcsnrtombs_l(__dest, __src, __nwc, __len, __ps, __l);
271 #else
272   __locale_raii __current(uselocale(__l), uselocale);
273   return wcsnrtombs(__dest, __src, __nwc, __len, __ps);
274 #endif
275 }
276
277 _LIBCPP_ALWAYS_INLINE inline
278 size_t __wcrtomb_l(char *__s, wchar_t __wc, mbstate_t *__ps, locale_t __l)
279 {
280 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
281   return wcrtomb_l(__s, __wc, __ps, __l);
282 #else
283   __locale_raii __current(uselocale(__l), uselocale);
284   return wcrtomb(__s, __wc, __ps);
285 #endif
286 }
287
288 _LIBCPP_ALWAYS_INLINE inline
289 size_t __mbsnrtowcs_l(wchar_t * __dest, const char **__src, size_t __nms,
290                       size_t __len, mbstate_t *__ps, locale_t __l)
291 {
292 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
293   return mbsnrtowcs_l(__dest, __src, __nms, __len, __ps, __l);
294 #else
295   __locale_raii __current(uselocale(__l), uselocale);
296   return mbsnrtowcs(__dest, __src, __nms, __len, __ps);
297 #endif
298 }
299
300 _LIBCPP_ALWAYS_INLINE inline
301 size_t __mbrtowc_l(wchar_t *__pwc, const char *__s, size_t __n,
302                    mbstate_t *__ps, locale_t __l)
303 {
304 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
305   return mbrtowc_l(__pwc, __s, __n, __ps, __l);
306 #else
307   __locale_raii __current(uselocale(__l), uselocale);
308   return mbrtowc(__pwc, __s, __n, __ps);
309 #endif
310 }
311
312 _LIBCPP_ALWAYS_INLINE inline
313 int __mbtowc_l(wchar_t *__pwc, const char *__pmb, size_t __max, locale_t __l)
314 {
315 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
316   return mbtowc_l(__pwc, __pmb, __max, __l);
317 #else
318   __locale_raii __current(uselocale(__l), uselocale);
319   return mbtowc(__pwc, __pmb, __max);
320 #endif
321 }
322
323 _LIBCPP_ALWAYS_INLINE inline
324 size_t __mbrlen_l(const char *__s, size_t __n, mbstate_t *__ps, locale_t __l)
325 {
326 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
327   return mbrlen_l(__s, __n, __ps, __l);
328 #else
329   __locale_raii __current(uselocale(__l), uselocale);
330   return mbrlen(__s, __n, __ps);
331 #endif
332 }
333
334 _LIBCPP_ALWAYS_INLINE inline
335 lconv *__localeconv_l(locale_t __l)
336 {
337 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
338   return localeconv_l(__l);
339 #else
340   __locale_raii __current(uselocale(__l), uselocale);
341   return localeconv();
342 #endif
343 }
344
345 _LIBCPP_ALWAYS_INLINE inline
346 size_t __mbsrtowcs_l(wchar_t *__dest, const char **__src, size_t __len,
347                      mbstate_t *__ps, locale_t __l)
348 {
349 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
350   return mbsrtowcs_l(__dest, __src, __len, __ps, __l);
351 #else
352   __locale_raii __current(uselocale(__l), uselocale);
353   return mbsrtowcs(__dest, __src, __len, __ps);
354 #endif
355 }
356
357 inline
358 int __sprintf_l(char *__s, locale_t __l, const char *__format, ...) {
359   va_list __va;
360   va_start(__va, __format);
361 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
362   int __res = vsprintf_l(__s, __l, __format, __va);
363 #else
364   __locale_raii __current(uselocale(__l), uselocale);
365   int __res = vsprintf(__s, __format, __va);
366 #endif
367   va_end(__va);
368   return __res;
369 }
370
371 inline
372 int __snprintf_l(char *__s, size_t __n, locale_t __l, const char *__format, ...) {
373   va_list __va;
374   va_start(__va, __format);
375 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
376   int __res = vsnprintf_l(__s, __n, __l, __format, __va);
377 #else
378   __locale_raii __current(uselocale(__l), uselocale);
379   int __res = vsnprintf(__s, __n, __format, __va);
380 #endif
381   va_end(__va);
382   return __res;
383 }
384
385 inline
386 int __asprintf_l(char **__s, locale_t __l, const char *__format, ...) {
387   va_list __va;
388   va_start(__va, __format);
389 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
390   int __res = vasprintf_l(__s, __l, __format, __va);
391 #else
392   __locale_raii __current(uselocale(__l), uselocale);
393   int __res = vasprintf(__s, __format, __va);
394 #endif
395   va_end(__va);
396   return __res;
397 }
398
399 inline
400 int __sscanf_l(const char *__s, locale_t __l, const char *__format, ...) {
401   va_list __va;
402   va_start(__va, __format);
403 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
404   int __res = vsscanf_l(__s, __l, __format, __va);
405 #else
406   __locale_raii __current(uselocale(__l), uselocale);
407   int __res = vsscanf(__s, __format, __va);
408 #endif
409   va_end(__va);
410   return __res;
411 }
412
413 #endif  // __linux__
414
415 // __scan_keyword
416 // Scans [__b, __e) until a match is found in the basic_strings range
417 //  [__kb, __ke) or until it can be shown that there is no match in [__kb, __ke).
418 //  __b will be incremented (visibly), consuming CharT until a match is found
419 //  or proved to not exist.  A keyword may be "", in which will match anything.
420 //  If one keyword is a prefix of another, and the next CharT in the input
421 //  might match another keyword, the algorithm will attempt to find the longest
422 //  matching keyword.  If the longer matching keyword ends up not matching, then
423 //  no keyword match is found.  If no keyword match is found, __ke is returned
424 //  and failbit is set in __err.
425 //  Else an iterator pointing to the matching keyword is found.  If more than
426 //  one keyword matches, an iterator to the first matching keyword is returned.
427 //  If on exit __b == __e, eofbit is set in __err.  If __case_senstive is false,
428 //  __ct is used to force to lower case before comparing characters.
429 //  Examples:
430 //  Keywords:  "a", "abb"
431 //  If the input is "a", the first keyword matches and eofbit is set.
432 //  If the input is "abc", no match is found and "ab" are consumed.
433 template <class _InputIterator, class _ForwardIterator, class _Ctype>
434 _LIBCPP_HIDDEN
435 _ForwardIterator
436 __scan_keyword(_InputIterator& __b, _InputIterator __e,
437                _ForwardIterator __kb, _ForwardIterator __ke,
438                const _Ctype& __ct, ios_base::iostate& __err,
439                bool __case_sensitive = true)
440 {
441     typedef typename iterator_traits<_InputIterator>::value_type _CharT;
442     size_t __nkw = static_cast<size_t>(_VSTD::distance(__kb, __ke));
443     const unsigned char __doesnt_match = '\0';
444     const unsigned char __might_match = '\1';
445     const unsigned char __does_match = '\2';
446     unsigned char __statbuf[100];
447     unsigned char* __status = __statbuf;
448     unique_ptr<unsigned char, void(*)(void*)> __stat_hold(0, free);
449     if (__nkw > sizeof(__statbuf))
450     {
451         __status = (unsigned char*)malloc(__nkw);
452         if (__status == 0)
453             __throw_bad_alloc();
454         __stat_hold.reset(__status);
455     }
456     size_t __n_might_match = __nkw;  // At this point, any keyword might match
457     size_t __n_does_match = 0;       // but none of them definitely do
458     // Initialize all statuses to __might_match, except for "" keywords are __does_match
459     unsigned char* __st = __status;
460     for (_ForwardIterator __ky = __kb; __ky != __ke; ++__ky, ++__st)
461     {
462         if (!__ky->empty())
463             *__st = __might_match;
464         else
465         {
466             *__st = __does_match;
467             --__n_might_match;
468             ++__n_does_match;
469         }
470     }
471     // While there might be a match, test keywords against the next CharT
472     for (size_t __indx = 0; __b != __e && __n_might_match > 0; ++__indx)
473     {
474         // Peek at the next CharT but don't consume it
475         _CharT __c = *__b;
476         if (!__case_sensitive)
477             __c = __ct.toupper(__c);
478         bool __consume = false;
479         // For each keyword which might match, see if the __indx character is __c
480         // If a match if found, consume __c
481         // If a match is found, and that is the last character in the keyword,
482         //    then that keyword matches.
483         // If the keyword doesn't match this character, then change the keyword
484         //    to doesn't match
485         __st = __status;
486         for (_ForwardIterator __ky = __kb; __ky != __ke; ++__ky, ++__st)
487         {
488             if (*__st == __might_match)
489             {
490                 _CharT __kc = (*__ky)[__indx];
491                 if (!__case_sensitive)
492                     __kc = __ct.toupper(__kc);
493                 if (__c == __kc)
494                 {
495                     __consume = true;
496                     if (__ky->size() == __indx+1)
497                     {
498                         *__st = __does_match;
499                         --__n_might_match;
500                         ++__n_does_match;
501                     }
502                 }
503                 else
504                 {
505                     *__st = __doesnt_match;
506                     --__n_might_match;
507                 }
508             }
509         }
510         // consume if we matched a character
511         if (__consume)
512         {
513             ++__b;
514             // If we consumed a character and there might be a matched keyword that
515             //   was marked matched on a previous iteration, then such keywords
516             //   which are now marked as not matching.
517             if (__n_might_match + __n_does_match > 1)
518             {
519                 __st = __status;
520                 for (_ForwardIterator __ky = __kb; __ky != __ke; ++__ky, ++__st)
521                 {
522                     if (*__st == __does_match && __ky->size() != __indx+1)
523                     {
524                         *__st = __doesnt_match;
525                         --__n_does_match;
526                     }
527                 }
528             }
529         }
530     }
531     // We've exited the loop because we hit eof and/or we have no more "might matches".
532     if (__b == __e)
533         __err |= ios_base::eofbit;
534     // Return the first matching result
535     for (__st = __status; __kb != __ke; ++__kb, ++__st)
536         if (*__st == __does_match)
537             break;
538     if (__kb == __ke)
539         __err |= ios_base::failbit;
540     return __kb;
541 }
542
543 struct __num_get_base
544 {
545     static const int __num_get_buf_sz = 40;
546
547     static int __get_base(ios_base&);
548     static const char __src[33];
549 };
550
551 void __check_grouping(const string& __grouping, unsigned* __g, unsigned* __g_end,
552                       ios_base::iostate& __err);
553
554 template <class _CharT>
555 struct __num_get
556     : protected __num_get_base
557 {
558     static string __stage2_int_prep(ios_base& __iob, _CharT* __atoms, _CharT& __thousands_sep);
559     static string __stage2_float_prep(ios_base& __iob, _CharT* __atoms, _CharT& __decimal_point,
560                                       _CharT& __thousands_sep);
561     static int __stage2_int_loop(_CharT __ct, int __base, char* __a, char*& __a_end,
562                   unsigned& __dc, _CharT __thousands_sep, const string& __grouping,
563                   unsigned* __g, unsigned*& __g_end, _CharT* __atoms);
564     static int __stage2_float_loop(_CharT __ct, bool& __in_units, char& __exp,
565                                    char* __a, char*& __a_end,
566                                    _CharT __decimal_point, _CharT __thousands_sep,
567                                    const string& __grouping, unsigned* __g,
568                                    unsigned*& __g_end, unsigned& __dc, _CharT* __atoms);
569 };
570
571 template <class _CharT>
572 string
573 __num_get<_CharT>::__stage2_int_prep(ios_base& __iob, _CharT* __atoms, _CharT& __thousands_sep)
574 {
575     locale __loc = __iob.getloc();
576     use_facet<ctype<_CharT> >(__loc).widen(__src, __src + 26, __atoms);
577     const numpunct<_CharT>& __np = use_facet<numpunct<_CharT> >(__loc);
578     __thousands_sep = __np.thousands_sep();
579     return __np.grouping();
580 }
581
582 template <class _CharT>
583 string
584 __num_get<_CharT>::__stage2_float_prep(ios_base& __iob, _CharT* __atoms, _CharT& __decimal_point,
585                     _CharT& __thousands_sep)
586 {
587     locale __loc = __iob.getloc();
588     use_facet<ctype<_CharT> >(__loc).widen(__src, __src + 32, __atoms);
589     const numpunct<_CharT>& __np = use_facet<numpunct<_CharT> >(__loc);
590     __decimal_point = __np.decimal_point();
591     __thousands_sep = __np.thousands_sep();
592     return __np.grouping();
593 }
594
595 template <class _CharT>
596 int
597 __num_get<_CharT>::__stage2_int_loop(_CharT __ct, int __base, char* __a, char*& __a_end,
598                   unsigned& __dc, _CharT __thousands_sep, const string& __grouping,
599                   unsigned* __g, unsigned*& __g_end, _CharT* __atoms)
600 {
601     if (__a_end == __a && (__ct == __atoms[24] || __ct == __atoms[25]))
602     {
603         *__a_end++ = __ct == __atoms[24] ? '+' : '-';
604         __dc = 0;
605         return 0;
606     }
607     if (__grouping.size() != 0 && __ct == __thousands_sep)
608     {
609         if (__g_end-__g < __num_get_buf_sz)
610         {
611             *__g_end++ = __dc;
612             __dc = 0;
613         }
614         return 0;
615     }
616     ptrdiff_t __f = find(__atoms, __atoms + 26, __ct) - __atoms;
617     if (__f >= 24)
618         return -1;
619     switch (__base)
620     {
621     case 8:
622     case 10:
623         if (__f >= __base)
624             return -1;
625         break;
626     case 16:
627         if (__f < 22)
628             break;
629         if (__a_end != __a && __a_end - __a <= 2 && __a_end[-1] == '0')
630         {
631             __dc = 0;
632             *__a_end++ = __src[__f];
633             return 0;
634         }
635         return -1;
636     }
637     if (__a_end-__a < __num_get_buf_sz - 1)
638         *__a_end++ = __src[__f];
639     ++__dc;
640     return 0;
641 }
642
643 template <class _CharT>
644 int
645 __num_get<_CharT>::__stage2_float_loop(_CharT __ct, bool& __in_units, char& __exp, char* __a, char*& __a_end,
646                     _CharT __decimal_point, _CharT __thousands_sep, const string& __grouping,
647                     unsigned* __g, unsigned*& __g_end, unsigned& __dc, _CharT* __atoms)
648 {
649     if (__ct == __decimal_point)
650     {
651         if (!__in_units)
652             return -1;
653         __in_units = false;
654         *__a_end++ = '.';
655         if (__grouping.size() != 0 && __g_end-__g < __num_get_buf_sz)
656             *__g_end++ = __dc;
657         return 0;
658     }
659     if (__ct == __thousands_sep && __grouping.size() != 0)
660     {
661         if (!__in_units)
662             return -1;
663         if (__g_end-__g < __num_get_buf_sz)
664         {
665             *__g_end++ = __dc;
666             __dc = 0;
667         }
668         return 0;
669     }
670     ptrdiff_t __f = find(__atoms, __atoms + 32, __ct) - __atoms;
671     if (__f >= 32)
672         return -1;
673     char __x = __src[__f];
674     if (__x == '-' || __x == '+')
675     {
676         if (__a_end == __a || (__a_end[-1] & 0xDF) == __exp)
677         {
678             *__a_end++ = __x;
679             return 0;
680         }
681         return -1;
682     }
683     if (__a_end-__a < __num_get_buf_sz - 1)
684         *__a_end++ = __x;
685     if (__x == 'x' || __x == 'X')
686         __exp = 'P';
687     else if ((__x & 0xDF) == __exp)
688     {
689         __in_units = false;
690         if (__grouping.size() != 0 && __g_end-__g < __num_get_buf_sz)
691             *__g_end++ = __dc;
692     }
693     if (__f >= 22)
694         return 0;
695     ++__dc;
696     return 0;
697 }
698
699 _LIBCPP_EXTERN_TEMPLATE(struct __num_get<char>)
700 _LIBCPP_EXTERN_TEMPLATE(struct __num_get<wchar_t>)
701
702 template <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
703 class _LIBCPP_VISIBLE num_get
704     : public locale::facet,
705       private __num_get<_CharT>
706 {
707 public:
708     typedef _CharT char_type;
709     typedef _InputIterator iter_type;
710
711     _LIBCPP_ALWAYS_INLINE
712     explicit num_get(size_t __refs = 0)
713         : locale::facet(__refs) {}
714
715     _LIBCPP_ALWAYS_INLINE
716     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
717                   ios_base::iostate& __err, bool& __v) const
718     {
719         return do_get(__b, __e, __iob, __err, __v);
720     }
721
722     _LIBCPP_ALWAYS_INLINE
723     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
724                   ios_base::iostate& __err, long& __v) const
725     {
726         return do_get(__b, __e, __iob, __err, __v);
727     }
728
729     _LIBCPP_ALWAYS_INLINE
730     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
731                   ios_base::iostate& __err, long long& __v) const
732     {
733         return do_get(__b, __e, __iob, __err, __v);
734     }
735
736     _LIBCPP_ALWAYS_INLINE
737     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
738                   ios_base::iostate& __err, unsigned short& __v) const
739     {
740         return do_get(__b, __e, __iob, __err, __v);
741     }
742
743     _LIBCPP_ALWAYS_INLINE
744     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
745                   ios_base::iostate& __err, unsigned int& __v) const
746     {
747         return do_get(__b, __e, __iob, __err, __v);
748     }
749
750     _LIBCPP_ALWAYS_INLINE
751     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
752                   ios_base::iostate& __err, unsigned long& __v) const
753     {
754         return do_get(__b, __e, __iob, __err, __v);
755     }
756
757     _LIBCPP_ALWAYS_INLINE
758     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
759                   ios_base::iostate& __err, unsigned long long& __v) const
760     {
761         return do_get(__b, __e, __iob, __err, __v);
762     }
763
764     _LIBCPP_ALWAYS_INLINE
765     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
766                   ios_base::iostate& __err, float& __v) const
767     {
768         return do_get(__b, __e, __iob, __err, __v);
769     }
770
771     _LIBCPP_ALWAYS_INLINE
772     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
773                   ios_base::iostate& __err, double& __v) const
774     {
775         return do_get(__b, __e, __iob, __err, __v);
776     }
777
778     _LIBCPP_ALWAYS_INLINE
779     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
780                   ios_base::iostate& __err, long double& __v) const
781     {
782         return do_get(__b, __e, __iob, __err, __v);
783     }
784
785     _LIBCPP_ALWAYS_INLINE
786     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
787                   ios_base::iostate& __err, void*& __v) const
788     {
789         return do_get(__b, __e, __iob, __err, __v);
790     }
791
792     static locale::id id;
793
794 protected:
795     _LIBCPP_ALWAYS_INLINE
796     ~num_get() {}
797
798     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
799                              ios_base::iostate& __err, bool& __v) const;
800     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
801                              ios_base::iostate& __err, long& __v) const;
802     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
803                              ios_base::iostate& __err, long long& __v) const;
804     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
805                              ios_base::iostate& __err, unsigned short& __v) const;
806     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
807                              ios_base::iostate& __err, unsigned int& __v) const;
808     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
809                              ios_base::iostate& __err, unsigned long& __v) const;
810     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
811                              ios_base::iostate& __err, unsigned long long& __v) const;
812     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
813                              ios_base::iostate& __err, float& __v) const;
814     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
815                              ios_base::iostate& __err, double& __v) const;
816     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
817                              ios_base::iostate& __err, long double& __v) const;
818     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
819                              ios_base::iostate& __err, void*& __v) const;
820 };
821
822 template <class _CharT, class _InputIterator>
823 locale::id
824 num_get<_CharT, _InputIterator>::id;
825
826 template <class _Tp>
827 _Tp
828 __num_get_signed_integral(const char* __a, const char* __a_end,
829                           ios_base::iostate& __err, int __base)
830 {
831     if (__a != __a_end)
832     {
833         typename remove_reference<decltype(errno)>::type __save_errno = errno;
834         errno = 0;
835         char *__p2;
836         long long __ll = strtoll_l(__a, &__p2, __base, _LIBCPP_GET_C_LOCALE);
837         typename remove_reference<decltype(errno)>::type __current_errno = errno;
838         if (__current_errno == 0)
839             errno = __save_errno;
840         if (__p2 != __a_end)
841         {
842             __err = ios_base::failbit;
843             return 0;
844         }
845         else if (__current_errno == ERANGE         ||
846                  __ll < numeric_limits<_Tp>::min() ||
847                  numeric_limits<_Tp>::max() < __ll)
848         {
849             __err = ios_base::failbit;
850             if (__ll > 0)
851                 return numeric_limits<_Tp>::max();
852             else
853                 return numeric_limits<_Tp>::min();
854         }
855         return static_cast<_Tp>(__ll);
856     }
857     __err = ios_base::failbit;
858     return 0;
859 }
860
861 template <class _Tp>
862 _Tp
863 __num_get_unsigned_integral(const char* __a, const char* __a_end,
864                             ios_base::iostate& __err, int __base)
865 {
866     if (__a != __a_end)
867     {
868         if (*__a == '-')
869         {
870             __err = ios_base::failbit;
871             return 0;
872         }
873         typename remove_reference<decltype(errno)>::type __save_errno = errno;
874         errno = 0;
875         char *__p2;
876         unsigned long long __ll = strtoull_l(__a, &__p2, __base, _LIBCPP_GET_C_LOCALE);
877         typename remove_reference<decltype(errno)>::type __current_errno = errno;
878         if (__current_errno == 0)
879             errno = __save_errno;
880         if (__p2 != __a_end)
881         {
882             __err = ios_base::failbit;
883             return 0;
884         }
885         else if (__current_errno == ERANGE ||
886                  numeric_limits<_Tp>::max() < __ll)
887         {
888             __err = ios_base::failbit;
889             return numeric_limits<_Tp>::max();
890         }
891         return static_cast<_Tp>(__ll);
892     }
893     __err = ios_base::failbit;
894     return 0;
895 }
896
897 template <class _Tp>
898 _Tp
899 __num_get_float(const char* __a, const char* __a_end, ios_base::iostate& __err)
900 {
901     if (__a != __a_end)
902     {
903         char *__p2;
904         long double __ld = strtold_l(__a, &__p2, _LIBCPP_GET_C_LOCALE);
905         if (__p2 != __a_end)
906         {
907             __err = ios_base::failbit;
908             return 0;
909         }
910         return static_cast<_Tp>(__ld);
911     }
912     __err = ios_base::failbit;
913     return 0;
914 }
915
916 template <class _CharT, class _InputIterator>
917 _InputIterator
918 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
919                                         ios_base& __iob,
920                                         ios_base::iostate& __err,
921                                         bool& __v) const
922 {
923     if ((__iob.flags() & ios_base::boolalpha) == 0)
924     {
925         long __lv = -1;
926         __b = do_get(__b, __e, __iob, __err, __lv);
927         switch (__lv)
928         {
929         case 0:
930             __v = false;
931             break;
932         case 1:
933             __v = true;
934             break;
935         default:
936             __v = true;
937             __err = ios_base::failbit;
938             break;
939         }
940         return __b;
941     }
942     const ctype<_CharT>& __ct = use_facet<ctype<_CharT> >(__iob.getloc());
943     const numpunct<_CharT>& __np = use_facet<numpunct<_CharT> >(__iob.getloc());
944     typedef typename numpunct<_CharT>::string_type string_type;
945     const string_type __names[2] = {__np.truename(), __np.falsename()};
946     const string_type* __i = __scan_keyword(__b, __e, __names, __names+2,
947                                             __ct, __err);
948     __v = __i == __names;
949     return __b;
950 }
951
952 template <class _CharT, class _InputIterator>
953 _InputIterator
954 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
955                                         ios_base& __iob,
956                                         ios_base::iostate& __err,
957                                         long& __v) const
958 {
959     // Stage 1
960     int __base = this->__get_base(__iob);
961     // Stage 2
962     char_type __atoms[26];
963     char_type __thousands_sep;
964     string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
965     char __a[__num_get_base::__num_get_buf_sz] = {0};
966     char* __a_end = __a;
967     unsigned __g[__num_get_base::__num_get_buf_sz];
968     unsigned* __g_end = __g;
969     unsigned __dc = 0;
970     for (; __b != __e; ++__b)
971         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
972                                     __thousands_sep, __grouping, __g, __g_end,
973                                     __atoms))
974             break;
975     if (__grouping.size() != 0 && __g_end-__g < __num_get_base::__num_get_buf_sz)
976         *__g_end++ = __dc;
977     // Stage 3
978     __v = __num_get_signed_integral<long>(__a, __a_end, __err, __base);
979     // Digit grouping checked
980     __check_grouping(__grouping, __g, __g_end, __err);
981     // EOF checked
982     if (__b == __e)
983         __err |= ios_base::eofbit;
984     return __b;
985 }
986
987 template <class _CharT, class _InputIterator>
988 _InputIterator
989 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
990                                         ios_base& __iob,
991                                         ios_base::iostate& __err,
992                                         long long& __v) const
993 {
994     // Stage 1
995     int __base = this->__get_base(__iob);
996     // Stage 2
997     char_type __atoms[26];
998     char_type __thousands_sep;
999     string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
1000     char __a[__num_get_base::__num_get_buf_sz] = {0};
1001     char* __a_end = __a;
1002     unsigned __g[__num_get_base::__num_get_buf_sz];
1003     unsigned* __g_end = __g;
1004     unsigned __dc = 0;
1005     for (; __b != __e; ++__b)
1006         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
1007                                     __thousands_sep, __grouping, __g, __g_end,
1008                                     __atoms))
1009             break;
1010     if (__grouping.size() != 0 && __g_end-__g < __num_get_base::__num_get_buf_sz)
1011         *__g_end++ = __dc;
1012     // Stage 3
1013     __v = __num_get_signed_integral<long long>(__a, __a_end, __err, __base);
1014     // Digit grouping checked
1015     __check_grouping(__grouping, __g, __g_end, __err);
1016     // EOF checked
1017     if (__b == __e)
1018         __err |= ios_base::eofbit;
1019     return __b;
1020 }
1021
1022 template <class _CharT, class _InputIterator>
1023 _InputIterator
1024 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1025                                         ios_base& __iob,
1026                                         ios_base::iostate& __err,
1027                                         unsigned short& __v) const
1028 {
1029     // Stage 1
1030     int __base = this->__get_base(__iob);
1031     // Stage 2
1032     char_type __atoms[26];
1033     char_type __thousands_sep;
1034     string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
1035     char __a[__num_get_base::__num_get_buf_sz] = {0};
1036     char* __a_end = __a;
1037     unsigned __g[__num_get_base::__num_get_buf_sz];
1038     unsigned* __g_end = __g;
1039     unsigned __dc = 0;
1040     for (; __b != __e; ++__b)
1041         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
1042                                     __thousands_sep, __grouping, __g, __g_end,
1043                                     __atoms))
1044             break;
1045     if (__grouping.size() != 0 && __g_end-__g < __num_get_base::__num_get_buf_sz)
1046         *__g_end++ = __dc;
1047     // Stage 3
1048     __v = __num_get_unsigned_integral<unsigned short>(__a, __a_end, __err, __base);
1049     // Digit grouping checked
1050     __check_grouping(__grouping, __g, __g_end, __err);
1051     // EOF checked
1052     if (__b == __e)
1053         __err |= ios_base::eofbit;
1054     return __b;
1055 }
1056
1057 template <class _CharT, class _InputIterator>
1058 _InputIterator
1059 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1060                                         ios_base& __iob,
1061                                         ios_base::iostate& __err,
1062                                         unsigned int& __v) const
1063 {
1064     // Stage 1
1065     int __base = this->__get_base(__iob);
1066     // Stage 2
1067     char_type __atoms[26];
1068     char_type __thousands_sep;
1069     string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
1070     char __a[__num_get_base::__num_get_buf_sz] = {0};
1071     char* __a_end = __a;
1072     unsigned __g[__num_get_base::__num_get_buf_sz];
1073     unsigned* __g_end = __g;
1074     unsigned __dc = 0;
1075     for (; __b != __e; ++__b)
1076         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
1077                                     __thousands_sep, __grouping, __g, __g_end,
1078                                     __atoms))
1079             break;
1080     if (__grouping.size() != 0 && __g_end-__g < __num_get_base::__num_get_buf_sz)
1081         *__g_end++ = __dc;
1082     // Stage 3
1083     __v = __num_get_unsigned_integral<unsigned int>(__a, __a_end, __err, __base);
1084     // Digit grouping checked
1085     __check_grouping(__grouping, __g, __g_end, __err);
1086     // EOF checked
1087     if (__b == __e)
1088         __err |= ios_base::eofbit;
1089     return __b;
1090 }
1091
1092 template <class _CharT, class _InputIterator>
1093 _InputIterator
1094 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1095                                         ios_base& __iob,
1096                                         ios_base::iostate& __err,
1097                                         unsigned long& __v) const
1098 {
1099     // Stage 1
1100     int __base = this->__get_base(__iob);
1101     // Stage 2
1102     char_type __atoms[26];
1103     char_type __thousands_sep;
1104     string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
1105     char __a[__num_get_base::__num_get_buf_sz] = {0};
1106     char* __a_end = __a;
1107     unsigned __g[__num_get_base::__num_get_buf_sz];
1108     unsigned* __g_end = __g;
1109     unsigned __dc = 0;
1110     for (; __b != __e; ++__b)
1111         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
1112                                     __thousands_sep, __grouping, __g, __g_end,
1113                                     __atoms))
1114             break;
1115     if (__grouping.size() != 0 && __g_end-__g < __num_get_base::__num_get_buf_sz)
1116         *__g_end++ = __dc;
1117     // Stage 3
1118     __v = __num_get_unsigned_integral<unsigned long>(__a, __a_end, __err, __base);
1119     // Digit grouping checked
1120     __check_grouping(__grouping, __g, __g_end, __err);
1121     // EOF checked
1122     if (__b == __e)
1123         __err |= ios_base::eofbit;
1124     return __b;
1125 }
1126
1127 template <class _CharT, class _InputIterator>
1128 _InputIterator
1129 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1130                                         ios_base& __iob,
1131                                         ios_base::iostate& __err,
1132                                         unsigned long long& __v) const
1133 {
1134     // Stage 1
1135     int __base = this->__get_base(__iob);
1136     // Stage 2
1137     char_type __atoms[26];
1138     char_type __thousands_sep;
1139     string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
1140     char __a[__num_get_base::__num_get_buf_sz] = {0};
1141     char* __a_end = __a;
1142     unsigned __g[__num_get_base::__num_get_buf_sz];
1143     unsigned* __g_end = __g;
1144     unsigned __dc = 0;
1145     for (; __b != __e; ++__b)
1146         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
1147                                     __thousands_sep, __grouping, __g, __g_end,
1148                                     __atoms))
1149             break;
1150     if (__grouping.size() != 0 && __g_end-__g < __num_get_base::__num_get_buf_sz)
1151         *__g_end++ = __dc;
1152     // Stage 3
1153     __v = __num_get_unsigned_integral<unsigned long long>(__a, __a_end, __err, __base);
1154     // Digit grouping checked
1155     __check_grouping(__grouping, __g, __g_end, __err);
1156     // EOF checked
1157     if (__b == __e)
1158         __err |= ios_base::eofbit;
1159     return __b;
1160 }
1161
1162 template <class _CharT, class _InputIterator>
1163 _InputIterator
1164 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1165                                         ios_base& __iob,
1166                                         ios_base::iostate& __err,
1167                                         float& __v) const
1168 {
1169     // Stage 1, nothing to do
1170     // Stage 2
1171     char_type __atoms[32];
1172     char_type __decimal_point;
1173     char_type __thousands_sep;
1174     string __grouping = this->__stage2_float_prep(__iob, __atoms,
1175                                                   __decimal_point,
1176                                                   __thousands_sep);
1177     char __a[__num_get_base::__num_get_buf_sz] = {0};
1178     char* __a_end = __a;
1179     unsigned __g[__num_get_base::__num_get_buf_sz];
1180     unsigned* __g_end = __g;
1181     unsigned __dc = 0;
1182     bool __in_units = true;
1183     char __exp = 'E';
1184     for (; __b != __e; ++__b)
1185         if (this->__stage2_float_loop(*__b, __in_units, __exp, __a, __a_end,
1186                                       __decimal_point, __thousands_sep,
1187                                       __grouping, __g, __g_end,
1188                                       __dc, __atoms))
1189             break;
1190     if (__grouping.size() != 0 && __in_units && __g_end-__g < __num_get_base::__num_get_buf_sz)
1191         *__g_end++ = __dc;
1192     // Stage 3
1193     __v = __num_get_float<float>(__a, __a_end, __err);
1194     // Digit grouping checked
1195     __check_grouping(__grouping, __g, __g_end, __err);
1196     // EOF checked
1197     if (__b == __e)
1198         __err |= ios_base::eofbit;
1199     return __b;
1200 }
1201
1202 template <class _CharT, class _InputIterator>
1203 _InputIterator
1204 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1205                                         ios_base& __iob,
1206                                         ios_base::iostate& __err,
1207                                         double& __v) const
1208 {
1209     // Stage 1, nothing to do
1210     // Stage 2
1211     char_type __atoms[32];
1212     char_type __decimal_point;
1213     char_type __thousands_sep;
1214     string __grouping = this->__stage2_float_prep(__iob, __atoms,
1215                                                   __decimal_point,
1216                                                   __thousands_sep);
1217     char __a[__num_get_base::__num_get_buf_sz] = {0};
1218     char* __a_end = __a;
1219     unsigned __g[__num_get_base::__num_get_buf_sz];
1220     unsigned* __g_end = __g;
1221     unsigned __dc = 0;
1222     bool __in_units = true;
1223     char __exp = 'E';
1224     for (; __b != __e; ++__b)
1225         if (this->__stage2_float_loop(*__b, __in_units, __exp, __a, __a_end,
1226                                       __decimal_point, __thousands_sep,
1227                                       __grouping, __g, __g_end,
1228                                       __dc, __atoms))
1229             break;
1230     if (__grouping.size() != 0 && __in_units && __g_end-__g < __num_get_base::__num_get_buf_sz)
1231         *__g_end++ = __dc;
1232     // Stage 3
1233     __v = __num_get_float<double>(__a, __a_end, __err);
1234     // Digit grouping checked
1235     __check_grouping(__grouping, __g, __g_end, __err);
1236     // EOF checked
1237     if (__b == __e)
1238         __err |= ios_base::eofbit;
1239     return __b;
1240 }
1241
1242 template <class _CharT, class _InputIterator>
1243 _InputIterator
1244 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1245                                         ios_base& __iob,
1246                                         ios_base::iostate& __err,
1247                                         long double& __v) const
1248 {
1249     // Stage 1, nothing to do
1250     // Stage 2
1251     char_type __atoms[32];
1252     char_type __decimal_point;
1253     char_type __thousands_sep;
1254     string __grouping = this->__stage2_float_prep(__iob, __atoms,
1255                                                   __decimal_point,
1256                                                   __thousands_sep);
1257     char __a[__num_get_base::__num_get_buf_sz] = {0};
1258     char* __a_end = __a;
1259     unsigned __g[__num_get_base::__num_get_buf_sz];
1260     unsigned* __g_end = __g;
1261     unsigned __dc = 0;
1262     bool __in_units = true;
1263     char __exp = 'E';
1264     for (; __b != __e; ++__b)
1265         if (this->__stage2_float_loop(*__b, __in_units, __exp, __a, __a_end,
1266                                       __decimal_point, __thousands_sep,
1267                                       __grouping, __g, __g_end,
1268                                       __dc, __atoms))
1269             break;
1270     if (__grouping.size() != 0 && __in_units && __g_end-__g < __num_get_base::__num_get_buf_sz)
1271         *__g_end++ = __dc;
1272     // Stage 3
1273     __v = __num_get_float<long double>(__a, __a_end, __err);
1274     // Digit grouping checked
1275     __check_grouping(__grouping, __g, __g_end, __err);
1276     // EOF checked
1277     if (__b == __e)
1278         __err |= ios_base::eofbit;
1279     return __b;
1280 }
1281
1282 template <class _CharT, class _InputIterator>
1283 _InputIterator
1284 num_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
1285                                         ios_base& __iob,
1286                                         ios_base::iostate& __err,
1287                                         void*& __v) const
1288 {
1289     // Stage 1
1290     int __base = 16;
1291     // Stage 2
1292     char_type __atoms[26];
1293     char_type __thousands_sep = 0;
1294     string __grouping;
1295     use_facet<ctype<_CharT> >(__iob.getloc()).widen(__num_get_base::__src,
1296                                                     __num_get_base::__src + 26, __atoms);
1297     char __a[__num_get_base::__num_get_buf_sz] = {0};
1298     char* __a_end = __a;
1299     unsigned __g[__num_get_base::__num_get_buf_sz];
1300     unsigned* __g_end = __g;
1301     unsigned __dc = 0;
1302     for (; __b != __e; ++__b)
1303         if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc,
1304                                     __thousands_sep, __grouping,
1305                                     __g, __g_end, __atoms))
1306             break;
1307     // Stage 3
1308     __a[sizeof(__a)-1] = 0;
1309 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1310     if (sscanf_l(__a, _LIBCPP_GET_C_LOCALE, "%p", &__v) != 1)
1311 #else
1312     if (__sscanf_l(__a, __cloc(), "%p", &__v) != 1)
1313 #endif
1314         __err = ios_base::failbit;
1315     // EOF checked
1316     if (__b == __e)
1317         __err |= ios_base::eofbit;
1318     return __b;
1319 }
1320
1321 _LIBCPP_EXTERN_TEMPLATE(class num_get<char>)
1322 _LIBCPP_EXTERN_TEMPLATE(class num_get<wchar_t>)
1323
1324 struct __num_put_base
1325 {
1326 protected:
1327     static void __format_int(char* __fmt, const char* __len, bool __signd,
1328                              ios_base::fmtflags __flags);
1329     static bool __format_float(char* __fmt, const char* __len,
1330                                ios_base::fmtflags __flags);
1331     static char* __identify_padding(char* __nb, char* __ne,
1332                                     const ios_base& __iob);
1333 };
1334
1335 template <class _CharT>
1336 struct __num_put
1337     : protected __num_put_base
1338 {
1339     static void __widen_and_group_int(char* __nb, char* __np, char* __ne,
1340                                       _CharT* __ob, _CharT*& __op, _CharT*& __oe,
1341                                       const locale& __loc);
1342     static void __widen_and_group_float(char* __nb, char* __np, char* __ne,
1343                                         _CharT* __ob, _CharT*& __op, _CharT*& __oe,
1344                                         const locale& __loc);
1345 };
1346
1347 template <class _CharT>
1348 void
1349 __num_put<_CharT>::__widen_and_group_int(char* __nb, char* __np, char* __ne,
1350                                          _CharT* __ob, _CharT*& __op, _CharT*& __oe,
1351                                          const locale& __loc)
1352 {
1353     const ctype<_CharT>&    __ct = use_facet<ctype<_CharT> >   (__loc);
1354     const numpunct<_CharT>& __npt = use_facet<numpunct<_CharT> >(__loc);
1355     string __grouping = __npt.grouping();
1356     if (__grouping.empty())
1357     {
1358         __ct.widen(__nb, __ne, __ob);
1359         __oe = __ob + (__ne - __nb);
1360     }
1361     else
1362     {
1363         __oe = __ob;
1364         char* __nf = __nb;
1365         if (*__nf == '-' || *__nf == '+')
1366             *__oe++ = __ct.widen(*__nf++);
1367         if (__ne - __nf >= 2 && __nf[0] == '0' && (__nf[1] == 'x' ||
1368                                                    __nf[1] == 'X'))
1369         {
1370             *__oe++ = __ct.widen(*__nf++);
1371             *__oe++ = __ct.widen(*__nf++);
1372         }
1373         reverse(__nf, __ne);
1374         _CharT __thousands_sep = __npt.thousands_sep();
1375         unsigned __dc = 0;
1376         unsigned __dg = 0;
1377         for (char* __p = __nf; __p < __ne; ++__p)
1378         {
1379             if (static_cast<unsigned>(__grouping[__dg]) > 0 &&
1380                 __dc == static_cast<unsigned>(__grouping[__dg]))
1381             {
1382                 *__oe++ = __thousands_sep;
1383                 __dc = 0;
1384                 if (__dg < __grouping.size()-1)
1385                     ++__dg;
1386             }
1387             *__oe++ = __ct.widen(*__p);
1388             ++__dc;
1389         }
1390         reverse(__ob + (__nf - __nb), __oe);
1391     }
1392     if (__np == __ne)
1393         __op = __oe;
1394     else
1395         __op = __ob + (__np - __nb);
1396 }
1397
1398 template <class _CharT>
1399 void
1400 __num_put<_CharT>::__widen_and_group_float(char* __nb, char* __np, char* __ne,
1401                                            _CharT* __ob, _CharT*& __op, _CharT*& __oe,
1402                                            const locale& __loc)
1403 {
1404     const ctype<_CharT>&    __ct = use_facet<ctype<_CharT> >   (__loc);
1405     const numpunct<_CharT>& __npt = use_facet<numpunct<_CharT> >(__loc);
1406     string __grouping = __npt.grouping();
1407     __oe = __ob;
1408     char* __nf = __nb;
1409     if (*__nf == '-' || *__nf == '+')
1410         *__oe++ = __ct.widen(*__nf++);
1411     char* __ns;
1412     if (__ne - __nf >= 2 && __nf[0] == '0' && (__nf[1] == 'x' ||
1413                                                __nf[1] == 'X'))
1414     {
1415         *__oe++ = __ct.widen(*__nf++);
1416         *__oe++ = __ct.widen(*__nf++);
1417         for (__ns = __nf; __ns < __ne; ++__ns)
1418             if (!isxdigit_l(*__ns, _LIBCPP_GET_C_LOCALE))
1419                 break;
1420     }
1421     else
1422     {
1423         for (__ns = __nf; __ns < __ne; ++__ns)
1424             if (!isdigit_l(*__ns, _LIBCPP_GET_C_LOCALE))
1425                 break;
1426     }
1427     if (__grouping.empty())
1428     {
1429         __ct.widen(__nf, __ns, __oe);
1430         __oe += __ns - __nf;
1431     }
1432     else
1433     {
1434         reverse(__nf, __ns);
1435         _CharT __thousands_sep = __npt.thousands_sep();
1436         unsigned __dc = 0;
1437         unsigned __dg = 0;
1438         for (char* __p = __nf; __p < __ns; ++__p)
1439         {
1440             if (__grouping[__dg] > 0 && __dc == static_cast<unsigned>(__grouping[__dg]))
1441             {
1442                 *__oe++ = __thousands_sep;
1443                 __dc = 0;
1444                 if (__dg < __grouping.size()-1)
1445                     ++__dg;
1446             }
1447             *__oe++ = __ct.widen(*__p);
1448             ++__dc;
1449         }
1450         reverse(__ob + (__nf - __nb), __oe);
1451     }
1452     for (__nf = __ns; __nf < __ne; ++__nf)
1453     {
1454         if (*__nf == '.')
1455         {
1456             *__oe++ = __npt.decimal_point();
1457             ++__nf;
1458             break;
1459         }
1460         else
1461             *__oe++ = __ct.widen(*__nf);
1462     }
1463     __ct.widen(__nf, __ne, __oe);
1464     __oe += __ne - __nf;
1465     if (__np == __ne)
1466         __op = __oe;
1467     else
1468         __op = __ob + (__np - __nb);
1469 }
1470
1471 _LIBCPP_EXTERN_TEMPLATE(struct __num_put<char>)
1472 _LIBCPP_EXTERN_TEMPLATE(struct __num_put<wchar_t>)
1473
1474 template <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
1475 class _LIBCPP_VISIBLE num_put
1476     : public locale::facet,
1477       private __num_put<_CharT>
1478 {
1479 public:
1480     typedef _CharT char_type;
1481     typedef _OutputIterator iter_type;
1482
1483     _LIBCPP_ALWAYS_INLINE
1484     explicit num_put(size_t __refs = 0)
1485         : locale::facet(__refs) {}
1486
1487     _LIBCPP_ALWAYS_INLINE
1488     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1489                   bool __v) const
1490     {
1491         return do_put(__s, __iob, __fl, __v);
1492     }
1493
1494     _LIBCPP_ALWAYS_INLINE
1495     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1496                   long __v) const
1497     {
1498         return do_put(__s, __iob, __fl, __v);
1499     }
1500
1501     _LIBCPP_ALWAYS_INLINE
1502     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1503                   long long __v) const
1504     {
1505         return do_put(__s, __iob, __fl, __v);
1506     }
1507
1508     _LIBCPP_ALWAYS_INLINE
1509     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1510                   unsigned long __v) const
1511     {
1512         return do_put(__s, __iob, __fl, __v);
1513     }
1514
1515     _LIBCPP_ALWAYS_INLINE
1516     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1517                   unsigned long long __v) const
1518     {
1519         return do_put(__s, __iob, __fl, __v);
1520     }
1521
1522     _LIBCPP_ALWAYS_INLINE
1523     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1524                   double __v) const
1525     {
1526         return do_put(__s, __iob, __fl, __v);
1527     }
1528
1529     _LIBCPP_ALWAYS_INLINE
1530     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1531                   long double __v) const
1532     {
1533         return do_put(__s, __iob, __fl, __v);
1534     }
1535
1536     _LIBCPP_ALWAYS_INLINE
1537     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
1538                   const void* __v) const
1539     {
1540         return do_put(__s, __iob, __fl, __v);
1541     }
1542
1543     static locale::id id;
1544
1545 protected:
1546     _LIBCPP_ALWAYS_INLINE
1547     ~num_put() {}
1548
1549     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1550                              bool __v) const;
1551     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1552                              long __v) const;
1553     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1554                              long long __v) const;
1555     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1556                              unsigned long) const;
1557     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1558                              unsigned long long) const;
1559     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1560                              double __v) const;
1561     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1562                              long double __v) const;
1563     virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl,
1564                              const void* __v) const;
1565 };
1566
1567 template <class _CharT, class _OutputIterator>
1568 locale::id
1569 num_put<_CharT, _OutputIterator>::id;
1570
1571 template <class _CharT, class _OutputIterator>
1572 _LIBCPP_HIDDEN
1573 _OutputIterator
1574 __pad_and_output(_OutputIterator __s,
1575                  const _CharT* __ob, const _CharT* __op, const _CharT* __oe,
1576                  ios_base& __iob, _CharT __fl)
1577 {
1578     streamsize __sz = __oe - __ob;
1579     streamsize __ns = __iob.width();
1580     if (__ns > __sz)
1581         __ns -= __sz;
1582     else
1583         __ns = 0;
1584     for (;__ob < __op; ++__ob, ++__s)
1585         *__s = *__ob;
1586     for (; __ns; --__ns, ++__s)
1587         *__s = __fl;
1588     for (; __ob < __oe; ++__ob, ++__s)
1589         *__s = *__ob;
1590     __iob.width(0);
1591     return __s;
1592 }
1593
1594 #if !defined(__APPLE__) || \
1595     (defined(__MAC_OS_X_VERSION_MIN_REQUIRED) && __MAC_OS_X_VERSION_MIN_REQUIRED > __MAC_10_8) || \
1596     (defined(__IPHONE_OS_VERSION_MIN_REQUIRED) && __IPHONE_OS_VERSION_MIN_REQUIRED > __IPHONE_6_0)
1597
1598 template <class _CharT, class _Traits>
1599 _LIBCPP_HIDDEN
1600 ostreambuf_iterator<_CharT, _Traits>
1601 __pad_and_output(ostreambuf_iterator<_CharT, _Traits> __s,
1602                  const _CharT* __ob, const _CharT* __op, const _CharT* __oe,
1603                  ios_base& __iob, _CharT __fl)
1604 {
1605     if (__s.__sbuf_ == nullptr)
1606         return __s;
1607     streamsize __sz = __oe - __ob;
1608     streamsize __ns = __iob.width();
1609     if (__ns > __sz)
1610         __ns -= __sz;
1611     else
1612         __ns = 0;
1613     streamsize __np = __op - __ob;
1614     if (__np > 0)
1615     {
1616         if (__s.__sbuf_->sputn(__ob, __np) != __np)
1617         {
1618             __s.__sbuf_ = nullptr;
1619             return __s;
1620         }
1621     }
1622     if (__ns > 0)
1623     {
1624         basic_string<_CharT, _Traits> __sp(__ns, __fl);
1625         if (__s.__sbuf_->sputn(__sp.data(), __ns) != __ns)
1626         {
1627             __s.__sbuf_ = nullptr;
1628             return __s;
1629         }
1630     }
1631     __np = __oe - __op;
1632     if (__np > 0)
1633     {
1634         if (__s.__sbuf_->sputn(__op, __np) != __np)
1635         {
1636             __s.__sbuf_ = nullptr;
1637             return __s;
1638         }
1639     }
1640     __iob.width(0);
1641     return __s;
1642 }
1643
1644 #endif
1645
1646 template <class _CharT, class _OutputIterator>
1647 _OutputIterator
1648 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1649                                          char_type __fl, bool __v) const
1650 {
1651     if ((__iob.flags() & ios_base::boolalpha) == 0)
1652         return do_put(__s, __iob, __fl, (unsigned long)__v);
1653     const numpunct<char_type>& __np = use_facet<numpunct<char_type> >(__iob.getloc());
1654     typedef typename numpunct<char_type>::string_type string_type;
1655     string_type __nm = __v ? __np.truename() : __np.falsename();
1656     for (typename string_type::iterator __i = __nm.begin(); __i != __nm.end(); ++__i, ++__s)
1657         *__s = *__i;
1658     return __s;
1659 }
1660
1661 template <class _CharT, class _OutputIterator>
1662 _OutputIterator
1663 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1664                                          char_type __fl, long __v) const
1665 {
1666     // Stage 1 - Get number in narrow char
1667     char __fmt[6] = {'%', 0};
1668     const char* __len = "l";
1669     this->__format_int(__fmt+1, __len, true, __iob.flags());
1670     const unsigned __nbuf = (numeric_limits<long>::digits / 3)
1671                           + ((numeric_limits<long>::digits % 3) != 0)
1672                           + 1;
1673     char __nar[__nbuf];
1674 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1675     int __nc = sprintf_l(__nar, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1676 #else
1677     int __nc = __sprintf_l(__nar, __cloc(), __fmt, __v);
1678 #endif
1679     char* __ne = __nar + __nc;
1680     char* __np = this->__identify_padding(__nar, __ne, __iob);
1681     // Stage 2 - Widen __nar while adding thousands separators
1682     char_type __o[2*(__nbuf-1) - 1];
1683     char_type* __op;  // pad here
1684     char_type* __oe;  // end of output
1685     this->__widen_and_group_int(__nar, __np, __ne, __o, __op, __oe, __iob.getloc());
1686     // [__o, __oe) contains thousands_sep'd wide number
1687     // Stage 3 & 4
1688     return __pad_and_output(__s, __o, __op, __oe, __iob, __fl);
1689 }
1690
1691 template <class _CharT, class _OutputIterator>
1692 _OutputIterator
1693 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1694                                          char_type __fl, long long __v) const
1695 {
1696     // Stage 1 - Get number in narrow char
1697     char __fmt[8] = {'%', 0};
1698     const char* __len = "ll";
1699     this->__format_int(__fmt+1, __len, true, __iob.flags());
1700     const unsigned __nbuf = (numeric_limits<long long>::digits / 3)
1701                           + ((numeric_limits<long long>::digits % 3) != 0)
1702                           + 1;
1703     char __nar[__nbuf];
1704 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1705     int __nc = sprintf_l(__nar, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1706 #else
1707     int __nc = __sprintf_l(__nar, __cloc(), __fmt, __v);
1708 #endif
1709     char* __ne = __nar + __nc;
1710     char* __np = this->__identify_padding(__nar, __ne, __iob);
1711     // Stage 2 - Widen __nar while adding thousands separators
1712     char_type __o[2*(__nbuf-1) - 1];
1713     char_type* __op;  // pad here
1714     char_type* __oe;  // end of output
1715     this->__widen_and_group_int(__nar, __np, __ne, __o, __op, __oe, __iob.getloc());
1716     // [__o, __oe) contains thousands_sep'd wide number
1717     // Stage 3 & 4
1718     return __pad_and_output(__s, __o, __op, __oe, __iob, __fl);
1719 }
1720
1721 template <class _CharT, class _OutputIterator>
1722 _OutputIterator
1723 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1724                                          char_type __fl, unsigned long __v) const
1725 {
1726     // Stage 1 - Get number in narrow char
1727     char __fmt[6] = {'%', 0};
1728     const char* __len = "l";
1729     this->__format_int(__fmt+1, __len, false, __iob.flags());
1730     const unsigned __nbuf = (numeric_limits<unsigned long>::digits / 3)
1731                           + ((numeric_limits<unsigned long>::digits % 3) != 0)
1732                           + 1;
1733     char __nar[__nbuf];
1734 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1735     int __nc = sprintf_l(__nar, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1736 #else
1737     int __nc = __sprintf_l(__nar, __cloc(), __fmt, __v);
1738 #endif
1739     char* __ne = __nar + __nc;
1740     char* __np = this->__identify_padding(__nar, __ne, __iob);
1741     // Stage 2 - Widen __nar while adding thousands separators
1742     char_type __o[2*(__nbuf-1) - 1];
1743     char_type* __op;  // pad here
1744     char_type* __oe;  // end of output
1745     this->__widen_and_group_int(__nar, __np, __ne, __o, __op, __oe, __iob.getloc());
1746     // [__o, __oe) contains thousands_sep'd wide number
1747     // Stage 3 & 4
1748     return __pad_and_output(__s, __o, __op, __oe, __iob, __fl);
1749 }
1750
1751 template <class _CharT, class _OutputIterator>
1752 _OutputIterator
1753 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1754                                          char_type __fl, unsigned long long __v) const
1755 {
1756     // Stage 1 - Get number in narrow char
1757     char __fmt[8] = {'%', 0};
1758     const char* __len = "ll";
1759     this->__format_int(__fmt+1, __len, false, __iob.flags());
1760     const unsigned __nbuf = (numeric_limits<unsigned long long>::digits / 3)
1761                           + ((numeric_limits<unsigned long long>::digits % 3) != 0)
1762                           + 1;
1763     char __nar[__nbuf];
1764 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1765     int __nc = sprintf_l(__nar, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1766 #else
1767     int __nc = __sprintf_l(__nar, __cloc(), __fmt, __v);
1768 #endif
1769     char* __ne = __nar + __nc;
1770     char* __np = this->__identify_padding(__nar, __ne, __iob);
1771     // Stage 2 - Widen __nar while adding thousands separators
1772     char_type __o[2*(__nbuf-1) - 1];
1773     char_type* __op;  // pad here
1774     char_type* __oe;  // end of output
1775     this->__widen_and_group_int(__nar, __np, __ne, __o, __op, __oe, __iob.getloc());
1776     // [__o, __oe) contains thousands_sep'd wide number
1777     // Stage 3 & 4
1778     return __pad_and_output(__s, __o, __op, __oe, __iob, __fl);
1779 }
1780
1781 template <class _CharT, class _OutputIterator>
1782 _OutputIterator
1783 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1784                                          char_type __fl, double __v) const
1785 {
1786     // Stage 1 - Get number in narrow char
1787     char __fmt[8] = {'%', 0};
1788     const char* __len = "";
1789     bool __specify_precision = this->__format_float(__fmt+1, __len, __iob.flags());
1790     const unsigned __nbuf = 30;
1791     char __nar[__nbuf];
1792     char* __nb = __nar;
1793     int __nc;
1794     if (__specify_precision)
1795 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1796         __nc = snprintf_l(__nb, __nbuf, _LIBCPP_GET_C_LOCALE, __fmt,
1797                                    (int)__iob.precision(), __v);
1798 #else
1799         __nc = __snprintf_l(__nb, __nbuf, __cloc(), __fmt,
1800                             (int)__iob.precision(), __v);
1801 #endif
1802     else
1803 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1804         __nc = snprintf_l(__nb, __nbuf, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1805 #else
1806         __nc = __snprintf_l(__nb, __nbuf, __cloc(), __fmt, __v);
1807 #endif
1808     unique_ptr<char, void(*)(void*)> __nbh(0, free);
1809     if (__nc > static_cast<int>(__nbuf-1))
1810     {
1811         if (__specify_precision)
1812 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1813             __nc = asprintf_l(&__nb, _LIBCPP_GET_C_LOCALE, __fmt, (int)__iob.precision(), __v);
1814 #else
1815             __nc = __asprintf_l(&__nb, __cloc(), __fmt,
1816                               (int)__iob.precision(), __v);
1817 #endif
1818         else
1819 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1820             __nc = asprintf_l(&__nb, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1821 #else
1822             __nc = __asprintf_l(&__nb, __cloc(), __fmt, (int)__iob.precision(), __v);
1823 #endif
1824         if (__nb == 0)
1825             __throw_bad_alloc();
1826         __nbh.reset(__nb);
1827     }
1828     char* __ne = __nb + __nc;
1829     char* __np = this->__identify_padding(__nb, __ne, __iob);
1830     // Stage 2 - Widen __nar while adding thousands separators
1831     char_type __o[2*(__nbuf-1) - 1];
1832     char_type* __ob = __o;
1833     unique_ptr<char_type, void(*)(void*)> __obh(0, free);
1834     if (__nb != __nar)
1835     {
1836         __ob = (char_type*)malloc(2*static_cast<size_t>(__nc)*sizeof(char_type));
1837         if (__ob == 0)
1838             __throw_bad_alloc();
1839         __obh.reset(__ob);
1840     }
1841     char_type* __op;  // pad here
1842     char_type* __oe;  // end of output
1843     this->__widen_and_group_float(__nb, __np, __ne, __ob, __op, __oe, __iob.getloc());
1844     // [__o, __oe) contains thousands_sep'd wide number
1845     // Stage 3 & 4
1846     __s = __pad_and_output(__s, __ob, __op, __oe, __iob, __fl);
1847     return __s;
1848 }
1849
1850 template <class _CharT, class _OutputIterator>
1851 _OutputIterator
1852 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1853                                          char_type __fl, long double __v) const
1854 {
1855     // Stage 1 - Get number in narrow char
1856     char __fmt[8] = {'%', 0};
1857     const char* __len = "L";
1858     bool __specify_precision = this->__format_float(__fmt+1, __len, __iob.flags());
1859     const unsigned __nbuf = 30;
1860     char __nar[__nbuf];
1861     char* __nb = __nar;
1862     int __nc;
1863     if (__specify_precision)
1864 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1865         __nc = snprintf_l(__nb, __nbuf, _LIBCPP_GET_C_LOCALE, __fmt,
1866                                    (int)__iob.precision(), __v);
1867 #else
1868         __nc = __snprintf_l(__nb, __nbuf, __cloc(), __fmt,
1869                             (int)__iob.precision(), __v);
1870 #endif
1871     else
1872 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1873         __nc = snprintf_l(__nb, __nbuf, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1874 #else
1875         __nc = __snprintf_l(__nb, __nbuf, __cloc(), __fmt, __v);
1876 #endif
1877     unique_ptr<char, void(*)(void*)> __nbh(0, free);
1878     if (__nc > static_cast<int>(__nbuf-1))
1879     {
1880         if (__specify_precision)
1881 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1882             __nc = asprintf_l(&__nb, _LIBCPP_GET_C_LOCALE, __fmt, (int)__iob.precision(), __v);
1883 #else
1884             __nc = __asprintf_l(&__nb, __cloc(), __fmt,
1885                               (int)__iob.precision(), __v);
1886 #endif
1887         else
1888 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1889             __nc = asprintf_l(&__nb, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1890 #else
1891             __nc = __asprintf_l(&__nb, __cloc(), __fmt, __v);
1892 #endif
1893         if (__nb == 0)
1894             __throw_bad_alloc();
1895         __nbh.reset(__nb);
1896     }
1897     char* __ne = __nb + __nc;
1898     char* __np = this->__identify_padding(__nb, __ne, __iob);
1899     // Stage 2 - Widen __nar while adding thousands separators
1900     char_type __o[2*(__nbuf-1) - 1];
1901     char_type* __ob = __o;
1902     unique_ptr<char_type, void(*)(void*)> __obh(0, free);
1903     if (__nb != __nar)
1904     {
1905         __ob = (char_type*)malloc(2*static_cast<size_t>(__nc)*sizeof(char_type));
1906         if (__ob == 0)
1907             __throw_bad_alloc();
1908         __obh.reset(__ob);
1909     }
1910     char_type* __op;  // pad here
1911     char_type* __oe;  // end of output
1912     this->__widen_and_group_float(__nb, __np, __ne, __ob, __op, __oe, __iob.getloc());
1913     // [__o, __oe) contains thousands_sep'd wide number
1914     // Stage 3 & 4
1915     __s = __pad_and_output(__s, __ob, __op, __oe, __iob, __fl);
1916     return __s;
1917 }
1918
1919 template <class _CharT, class _OutputIterator>
1920 _OutputIterator
1921 num_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob,
1922                                          char_type __fl, const void* __v) const
1923 {
1924     // Stage 1 - Get pointer in narrow char
1925     char __fmt[6] = "%p";
1926     const unsigned __nbuf = 20;
1927     char __nar[__nbuf];
1928 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
1929     int __nc = sprintf_l(__nar, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1930 #else
1931     int __nc = __sprintf_l(__nar, __cloc(), __fmt, __v);
1932 #endif
1933     char* __ne = __nar + __nc;
1934     char* __np = this->__identify_padding(__nar, __ne, __iob);
1935     // Stage 2 - Widen __nar
1936     char_type __o[2*(__nbuf-1) - 1];
1937     char_type* __op;  // pad here
1938     char_type* __oe;  // end of output
1939     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
1940     __ct.widen(__nar, __ne, __o);
1941     __oe = __o + (__ne - __nar);
1942     if (__np == __ne)
1943         __op = __oe;
1944     else
1945         __op = __o + (__np - __nar);
1946     // [__o, __oe) contains wide number
1947     // Stage 3 & 4
1948     return __pad_and_output(__s, __o, __op, __oe, __iob, __fl);
1949 }
1950
1951 _LIBCPP_EXTERN_TEMPLATE(class num_put<char>)
1952 _LIBCPP_EXTERN_TEMPLATE(class num_put<wchar_t>)
1953
1954 template <class _CharT, class _InputIterator>
1955 _LIBCPP_HIDDEN
1956 int
1957 __get_up_to_n_digits(_InputIterator& __b, _InputIterator __e,
1958                      ios_base::iostate& __err, const ctype<_CharT>& __ct, int __n)
1959 {
1960     // Precondition:  __n >= 1
1961     if (__b == __e)
1962     {
1963         __err |= ios_base::eofbit | ios_base::failbit;
1964         return 0;
1965     }
1966     // get first digit
1967     _CharT __c = *__b;
1968     if (!__ct.is(ctype_base::digit, __c))
1969     {
1970         __err |= ios_base::failbit;
1971         return 0;
1972     }
1973     int __r = __ct.narrow(__c, 0) - '0';
1974     for (++__b, --__n; __b != __e && __n > 0; ++__b, --__n)
1975     {
1976         // get next digit
1977         __c = *__b;
1978         if (!__ct.is(ctype_base::digit, __c))
1979             return __r;
1980         __r = __r * 10 + __ct.narrow(__c, 0) - '0';
1981     }
1982     if (__b == __e)
1983         __err |= ios_base::eofbit;
1984     return __r;
1985 }
1986
1987 class _LIBCPP_VISIBLE time_base
1988 {
1989 public:
1990     enum dateorder {no_order, dmy, mdy, ymd, ydm};
1991 };
1992
1993 template <class _CharT>
1994 class __time_get_c_storage  // purposefully not decorated
1995 {
1996 protected:
1997     typedef basic_string<_CharT> string_type;
1998
1999     virtual const string_type* __weeks() const;
2000     virtual const string_type* __months() const;
2001     virtual const string_type* __am_pm() const;
2002     virtual const string_type& __c() const;
2003     virtual const string_type& __r() const;
2004     virtual const string_type& __x() const;
2005     virtual const string_type& __X() const;
2006 };
2007
2008 template <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
2009 class _LIBCPP_VISIBLE time_get
2010     : public locale::facet,
2011       public time_base,
2012       private __time_get_c_storage<_CharT>
2013 {
2014 public:
2015     typedef _CharT                  char_type;
2016     typedef _InputIterator          iter_type;
2017     typedef time_base::dateorder    dateorder;
2018     typedef basic_string<char_type> string_type;
2019
2020     _LIBCPP_ALWAYS_INLINE
2021     explicit time_get(size_t __refs = 0)
2022         : locale::facet(__refs) {}
2023
2024     _LIBCPP_ALWAYS_INLINE
2025     dateorder date_order() const
2026     {
2027         return this->do_date_order();
2028     }
2029
2030     _LIBCPP_ALWAYS_INLINE
2031     iter_type get_time(iter_type __b, iter_type __e, ios_base& __iob,
2032                        ios_base::iostate& __err, tm* __tm) const
2033     {
2034         return do_get_time(__b, __e, __iob, __err, __tm);
2035     }
2036
2037     _LIBCPP_ALWAYS_INLINE
2038     iter_type get_date(iter_type __b, iter_type __e, ios_base& __iob,
2039                        ios_base::iostate& __err, tm* __tm) const
2040     {
2041         return do_get_date(__b, __e, __iob, __err, __tm);
2042     }
2043
2044     _LIBCPP_ALWAYS_INLINE
2045     iter_type get_weekday(iter_type __b, iter_type __e, ios_base& __iob,
2046                           ios_base::iostate& __err, tm* __tm) const
2047     {
2048         return do_get_weekday(__b, __e, __iob, __err, __tm);
2049     }
2050
2051     _LIBCPP_ALWAYS_INLINE
2052     iter_type get_monthname(iter_type __b, iter_type __e, ios_base& __iob,
2053                             ios_base::iostate& __err, tm* __tm) const
2054     {
2055         return do_get_monthname(__b, __e, __iob, __err, __tm);
2056     }
2057
2058     _LIBCPP_ALWAYS_INLINE
2059     iter_type get_year(iter_type __b, iter_type __e, ios_base& __iob,
2060                        ios_base::iostate& __err, tm* __tm) const
2061     {
2062         return do_get_year(__b, __e, __iob, __err, __tm);
2063     }
2064
2065     _LIBCPP_ALWAYS_INLINE
2066     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
2067                   ios_base::iostate& __err, tm *__tm,
2068                   char __fmt, char __mod = 0) const
2069     {
2070         return do_get(__b, __e, __iob, __err, __tm, __fmt, __mod);
2071     }
2072
2073     iter_type get(iter_type __b, iter_type __e, ios_base& __iob,
2074                   ios_base::iostate& __err, tm* __tm,
2075                   const char_type* __fmtb, const char_type* __fmte) const;
2076
2077     static locale::id id;
2078
2079 protected:
2080     _LIBCPP_ALWAYS_INLINE
2081     ~time_get() {}
2082
2083     virtual dateorder do_date_order() const;
2084     virtual iter_type do_get_time(iter_type __b, iter_type __e, ios_base& __iob,
2085                                   ios_base::iostate& __err, tm* __tm) const;
2086     virtual iter_type do_get_date(iter_type __b, iter_type __e, ios_base& __iob,
2087                                   ios_base::iostate& __err, tm* __tm) const;
2088     virtual iter_type do_get_weekday(iter_type __b, iter_type __e, ios_base& __iob,
2089                                      ios_base::iostate& __err, tm* __tm) const;
2090     virtual iter_type do_get_monthname(iter_type __b, iter_type __e, ios_base& __iob,
2091                                        ios_base::iostate& __err, tm* __tm) const;
2092     virtual iter_type do_get_year(iter_type __b, iter_type __e, ios_base& __iob,
2093                                   ios_base::iostate& __err, tm* __tm) const;
2094     virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob,
2095                              ios_base::iostate& __err, tm* __tm,
2096                              char __fmt, char __mod) const;
2097 private:
2098     void __get_white_space(iter_type& __b, iter_type __e,
2099                            ios_base::iostate& __err, const ctype<char_type>& __ct) const;
2100     void __get_percent(iter_type& __b, iter_type __e, ios_base::iostate& __err,
2101                        const ctype<char_type>& __ct) const;
2102
2103     void __get_weekdayname(int& __m,
2104                            iter_type& __b, iter_type __e,
2105                            ios_base::iostate& __err,
2106                            const ctype<char_type>& __ct) const;
2107     void __get_monthname(int& __m,
2108                          iter_type& __b, iter_type __e,
2109                          ios_base::iostate& __err,
2110                          const ctype<char_type>& __ct) const;
2111     void __get_day(int& __d,
2112                    iter_type& __b, iter_type __e,
2113                    ios_base::iostate& __err,
2114                    const ctype<char_type>& __ct) const;
2115     void __get_month(int& __m,
2116                      iter_type& __b, iter_type __e,
2117                      ios_base::iostate& __err,
2118                      const ctype<char_type>& __ct) const;
2119     void __get_year(int& __y,
2120                    iter_type& __b, iter_type __e,
2121                    ios_base::iostate& __err,
2122                    const ctype<char_type>& __ct) const;
2123     void __get_year4(int& __y,
2124                     iter_type& __b, iter_type __e,
2125                     ios_base::iostate& __err,
2126                     const ctype<char_type>& __ct) const;
2127     void __get_hour(int& __d,
2128                     iter_type& __b, iter_type __e,
2129                     ios_base::iostate& __err,
2130                     const ctype<char_type>& __ct) const;
2131     void __get_12_hour(int& __h,
2132                        iter_type& __b, iter_type __e,
2133                        ios_base::iostate& __err,
2134                        const ctype<char_type>& __ct) const;
2135     void __get_am_pm(int& __h,
2136                      iter_type& __b, iter_type __e,
2137                      ios_base::iostate& __err,
2138                      const ctype<char_type>& __ct) const;
2139     void __get_minute(int& __m,
2140                       iter_type& __b, iter_type __e,
2141                       ios_base::iostate& __err,
2142                       const ctype<char_type>& __ct) const;
2143     void __get_second(int& __s,
2144                       iter_type& __b, iter_type __e,
2145                       ios_base::iostate& __err,
2146                       const ctype<char_type>& __ct) const;
2147     void __get_weekday(int& __w,
2148                        iter_type& __b, iter_type __e,
2149                        ios_base::iostate& __err,
2150                        const ctype<char_type>& __ct) const;
2151     void __get_day_year_num(int& __w,
2152                             iter_type& __b, iter_type __e,
2153                             ios_base::iostate& __err,
2154                             const ctype<char_type>& __ct) const;
2155 };
2156
2157 template <class _CharT, class _InputIterator>
2158 locale::id
2159 time_get<_CharT, _InputIterator>::id;
2160
2161 // time_get primatives
2162
2163 template <class _CharT, class _InputIterator>
2164 void
2165 time_get<_CharT, _InputIterator>::__get_weekdayname(int& __w,
2166                                                     iter_type& __b, iter_type __e,
2167                                                     ios_base::iostate& __err,
2168                                                     const ctype<char_type>& __ct) const
2169 {
2170     // Note:  ignoring case comes from the POSIX strptime spec
2171     const string_type* __wk = this->__weeks();
2172     ptrdiff_t __i = __scan_keyword(__b, __e, __wk, __wk+14, __ct, __err, false) - __wk;
2173     if (__i < 14)
2174         __w = __i % 7;
2175 }
2176
2177 template <class _CharT, class _InputIterator>
2178 void
2179 time_get<_CharT, _InputIterator>::__get_monthname(int& __m,
2180                                                   iter_type& __b, iter_type __e,
2181                                                   ios_base::iostate& __err,
2182                                                   const ctype<char_type>& __ct) const
2183 {
2184     // Note:  ignoring case comes from the POSIX strptime spec
2185     const string_type* __month = this->__months();
2186     ptrdiff_t __i = __scan_keyword(__b, __e, __month, __month+24, __ct, __err, false) - __month;
2187     if (__i < 24)
2188         __m = __i % 12;
2189 }
2190
2191 template <class _CharT, class _InputIterator>
2192 void
2193 time_get<_CharT, _InputIterator>::__get_day(int& __d,
2194                                             iter_type& __b, iter_type __e,
2195                                             ios_base::iostate& __err,
2196                                             const ctype<char_type>& __ct) const
2197 {
2198     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 2);
2199     if (!(__err & ios_base::failbit) && 1 <= __t && __t <= 31)
2200         __d = __t;
2201     else
2202         __err |= ios_base::failbit;
2203 }
2204
2205 template <class _CharT, class _InputIterator>
2206 void
2207 time_get<_CharT, _InputIterator>::__get_month(int& __m,
2208                                               iter_type& __b, iter_type __e,
2209                                               ios_base::iostate& __err,
2210                                               const ctype<char_type>& __ct) const
2211 {
2212     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 2) - 1;
2213     if (!(__err & ios_base::failbit) && __t <= 11)
2214         __m = __t;
2215     else
2216         __err |= ios_base::failbit;
2217 }
2218
2219 template <class _CharT, class _InputIterator>
2220 void
2221 time_get<_CharT, _InputIterator>::__get_year(int& __y,
2222                                              iter_type& __b, iter_type __e,
2223                                              ios_base::iostate& __err,
2224                                              const ctype<char_type>& __ct) const
2225 {
2226     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 4);
2227     if (!(__err & ios_base::failbit))
2228     {
2229         if (__t < 69)
2230             __t += 2000;
2231         else if (69 <= __t && __t <= 99)
2232             __t += 1900;
2233         __y = __t - 1900;
2234     }
2235 }
2236
2237 template <class _CharT, class _InputIterator>
2238 void
2239 time_get<_CharT, _InputIterator>::__get_year4(int& __y,
2240                                               iter_type& __b, iter_type __e,
2241                                               ios_base::iostate& __err,
2242                                               const ctype<char_type>& __ct) const
2243 {
2244     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 4);
2245     if (!(__err & ios_base::failbit))
2246         __y = __t - 1900;
2247 }
2248
2249 template <class _CharT, class _InputIterator>
2250 void
2251 time_get<_CharT, _InputIterator>::__get_hour(int& __h,
2252                                              iter_type& __b, iter_type __e,
2253                                              ios_base::iostate& __err,
2254                                              const ctype<char_type>& __ct) const
2255 {
2256     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 2);
2257     if (!(__err & ios_base::failbit) && __t <= 23)
2258         __h = __t;
2259     else
2260         __err |= ios_base::failbit;
2261 }
2262
2263 template <class _CharT, class _InputIterator>
2264 void
2265 time_get<_CharT, _InputIterator>::__get_12_hour(int& __h,
2266                                                 iter_type& __b, iter_type __e,
2267                                                 ios_base::iostate& __err,
2268                                                 const ctype<char_type>& __ct) const
2269 {
2270     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 2);
2271     if (!(__err & ios_base::failbit) && 1 <= __t && __t <= 12)
2272         __h = __t;
2273     else
2274         __err |= ios_base::failbit;
2275 }
2276
2277 template <class _CharT, class _InputIterator>
2278 void
2279 time_get<_CharT, _InputIterator>::__get_minute(int& __m,
2280                                                iter_type& __b, iter_type __e,
2281                                                ios_base::iostate& __err,
2282                                                const ctype<char_type>& __ct) const
2283 {
2284     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 2);
2285     if (!(__err & ios_base::failbit) && __t <= 59)
2286         __m = __t;
2287     else
2288         __err |= ios_base::failbit;
2289 }
2290
2291 template <class _CharT, class _InputIterator>
2292 void
2293 time_get<_CharT, _InputIterator>::__get_second(int& __s,
2294                                                iter_type& __b, iter_type __e,
2295                                                ios_base::iostate& __err,
2296                                                const ctype<char_type>& __ct) const
2297 {
2298     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 2);
2299     if (!(__err & ios_base::failbit) && __t <= 60)
2300         __s = __t;
2301     else
2302         __err |= ios_base::failbit;
2303 }
2304
2305 template <class _CharT, class _InputIterator>
2306 void
2307 time_get<_CharT, _InputIterator>::__get_weekday(int& __w,
2308                                                 iter_type& __b, iter_type __e,
2309                                                 ios_base::iostate& __err,
2310                                                 const ctype<char_type>& __ct) const
2311 {
2312     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 1);
2313     if (!(__err & ios_base::failbit) && __t <= 6)
2314         __w = __t;
2315     else
2316         __err |= ios_base::failbit;
2317 }
2318
2319 template <class _CharT, class _InputIterator>
2320 void
2321 time_get<_CharT, _InputIterator>::__get_day_year_num(int& __d,
2322                                                      iter_type& __b, iter_type __e,
2323                                                      ios_base::iostate& __err,
2324                                                      const ctype<char_type>& __ct) const
2325 {
2326     int __t = __get_up_to_n_digits(__b, __e, __err, __ct, 3);
2327     if (!(__err & ios_base::failbit) && __t <= 365)
2328         __d = __t;
2329     else
2330         __err |= ios_base::failbit;
2331 }
2332
2333 template <class _CharT, class _InputIterator>
2334 void
2335 time_get<_CharT, _InputIterator>::__get_white_space(iter_type& __b, iter_type __e,
2336                                                     ios_base::iostate& __err,
2337                                                     const ctype<char_type>& __ct) const
2338 {
2339     for (; __b != __e && __ct.is(ctype_base::space, *__b); ++__b)
2340         ;
2341     if (__b == __e)
2342         __err |= ios_base::eofbit;
2343 }
2344
2345 template <class _CharT, class _InputIterator>
2346 void
2347 time_get<_CharT, _InputIterator>::__get_am_pm(int& __h,
2348                                               iter_type& __b, iter_type __e,
2349                                               ios_base::iostate& __err,
2350                                               const ctype<char_type>& __ct) const
2351 {
2352     const string_type* __ap = this->__am_pm();
2353     if (__ap[0].size() + __ap[1].size() == 0)
2354     {
2355         __err |= ios_base::failbit;
2356         return;
2357     }
2358     ptrdiff_t __i = __scan_keyword(__b, __e, __ap, __ap+2, __ct, __err, false) - __ap;
2359     if (__i == 0 && __h == 12)
2360         __h = 0;
2361     else if (__i == 1 && __h < 12)
2362         __h += 12;
2363 }
2364
2365 template <class _CharT, class _InputIterator>
2366 void
2367 time_get<_CharT, _InputIterator>::__get_percent(iter_type& __b, iter_type __e,
2368                                                 ios_base::iostate& __err,
2369                                                 const ctype<char_type>& __ct) const
2370 {
2371     if (__b == __e)
2372     {
2373         __err |= ios_base::eofbit | ios_base::failbit;
2374         return;
2375     }
2376     if (__ct.narrow(*__b, 0) != '%')
2377         __err |= ios_base::failbit;
2378     else if(++__b == __e)
2379         __err |= ios_base::eofbit;
2380 }
2381
2382 // time_get end primatives
2383
2384 template <class _CharT, class _InputIterator>
2385 _InputIterator
2386 time_get<_CharT, _InputIterator>::get(iter_type __b, iter_type __e,
2387                                       ios_base& __iob,
2388                                       ios_base::iostate& __err, tm* __tm,
2389                                       const char_type* __fmtb, const char_type* __fmte) const
2390 {
2391     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
2392     __err = ios_base::goodbit;
2393     while (__fmtb != __fmte && __err == ios_base::goodbit)
2394     {
2395         if (__b == __e)
2396         {
2397             __err = ios_base::failbit;
2398             break;
2399         }
2400         if (__ct.narrow(*__fmtb, 0) == '%')
2401         {
2402             if (++__fmtb == __fmte)
2403             {
2404                 __err = ios_base::failbit;
2405                 break;
2406             }
2407             char __cmd = __ct.narrow(*__fmtb, 0);
2408             char __opt = '\0';
2409             if (__cmd == 'E' || __cmd == '0')
2410             {
2411                 if (++__fmtb == __fmte)
2412                 {
2413                     __err = ios_base::failbit;
2414                     break;
2415                 }
2416                 __opt = __cmd;
2417                 __cmd = __ct.narrow(*__fmtb, 0);
2418             }
2419             __b = do_get(__b, __e, __iob, __err, __tm, __cmd, __opt);
2420             ++__fmtb;
2421         }
2422         else if (__ct.is(ctype_base::space, *__fmtb))
2423         {
2424             for (++__fmtb; __fmtb != __fmte && __ct.is(ctype_base::space, *__fmtb); ++__fmtb)
2425                 ;
2426             for (        ;    __b != __e    && __ct.is(ctype_base::space, *__b);    ++__b)
2427                 ;
2428         }
2429         else if (__ct.toupper(*__b) == __ct.toupper(*__fmtb))
2430         {
2431             ++__b;
2432             ++__fmtb;
2433         }
2434         else
2435             __err = ios_base::failbit;
2436     }
2437     if (__b == __e)
2438         __err |= ios_base::eofbit;
2439     return __b;
2440 }
2441
2442 template <class _CharT, class _InputIterator>
2443 typename time_get<_CharT, _InputIterator>::dateorder
2444 time_get<_CharT, _InputIterator>::do_date_order() const
2445 {
2446     return mdy;
2447 }
2448
2449 template <class _CharT, class _InputIterator>
2450 _InputIterator
2451 time_get<_CharT, _InputIterator>::do_get_time(iter_type __b, iter_type __e,
2452                                               ios_base& __iob,
2453                                               ios_base::iostate& __err,
2454                                               tm* __tm) const
2455 {
2456     const char_type __fmt[] = {'%', 'H', ':', '%', 'M', ':', '%', 'S'};
2457     return get(__b, __e, __iob, __err, __tm, __fmt, __fmt + sizeof(__fmt)/sizeof(__fmt[0]));
2458 }
2459
2460 template <class _CharT, class _InputIterator>
2461 _InputIterator
2462 time_get<_CharT, _InputIterator>::do_get_date(iter_type __b, iter_type __e,
2463                                               ios_base& __iob,
2464                                               ios_base::iostate& __err,
2465                                               tm* __tm) const
2466 {
2467     const string_type& __fmt = this->__x();
2468     return get(__b, __e, __iob, __err, __tm, __fmt.data(), __fmt.data() + __fmt.size());
2469 }
2470
2471 template <class _CharT, class _InputIterator>
2472 _InputIterator
2473 time_get<_CharT, _InputIterator>::do_get_weekday(iter_type __b, iter_type __e,
2474                                                  ios_base& __iob,
2475                                                  ios_base::iostate& __err,
2476                                                  tm* __tm) const
2477 {
2478     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
2479     __get_weekdayname(__tm->tm_wday, __b, __e, __err, __ct);
2480     return __b;
2481 }
2482
2483 template <class _CharT, class _InputIterator>
2484 _InputIterator
2485 time_get<_CharT, _InputIterator>::do_get_monthname(iter_type __b, iter_type __e,
2486                                                    ios_base& __iob,
2487                                                    ios_base::iostate& __err,
2488                                                    tm* __tm) const
2489 {
2490     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
2491     __get_monthname(__tm->tm_mon, __b, __e, __err, __ct);
2492     return __b;
2493 }
2494
2495 template <class _CharT, class _InputIterator>
2496 _InputIterator
2497 time_get<_CharT, _InputIterator>::do_get_year(iter_type __b, iter_type __e,
2498                                               ios_base& __iob,
2499                                               ios_base::iostate& __err,
2500                                               tm* __tm) const
2501 {
2502     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
2503     __get_year(__tm->tm_year, __b, __e, __err, __ct);
2504     return __b;
2505 }
2506
2507 template <class _CharT, class _InputIterator>
2508 _InputIterator
2509 time_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
2510                                          ios_base& __iob,
2511                                          ios_base::iostate& __err, tm* __tm,
2512                                          char __fmt, char) const
2513 {
2514     __err = ios_base::goodbit;
2515     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
2516     switch (__fmt)
2517     {
2518     case 'a':
2519     case 'A':
2520         __get_weekdayname(__tm->tm_wday, __b, __e, __err, __ct);
2521         break;
2522     case 'b':
2523     case 'B':
2524     case 'h':
2525         __get_monthname(__tm->tm_mon, __b, __e, __err, __ct);
2526         break;
2527     case 'c':
2528         {
2529         const string_type& __fm = this->__c();
2530         __b = get(__b, __e, __iob, __err, __tm, __fm.data(), __fm.data() + __fm.size());
2531         }
2532         break;
2533     case 'd':
2534     case 'e':
2535         __get_day(__tm->tm_mday, __b, __e, __err, __ct);
2536         break;
2537     case 'D':
2538         {
2539         const char_type __fm[] = {'%', 'm', '/', '%', 'd', '/', '%', 'y'};
2540         __b = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm)/sizeof(__fm[0]));
2541         }
2542         break;
2543     case 'F':
2544         {
2545         const char_type __fm[] = {'%', 'Y', '-', '%', 'm', '-', '%', 'd'};
2546         __b = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm)/sizeof(__fm[0]));
2547         }
2548         break;
2549     case 'H':
2550         __get_hour(__tm->tm_hour, __b, __e, __err, __ct);
2551         break;
2552     case 'I':
2553         __get_12_hour(__tm->tm_hour, __b, __e, __err, __ct);
2554         break;
2555     case 'j':
2556         __get_day_year_num(__tm->tm_yday, __b, __e, __err, __ct);
2557         break;
2558     case 'm':
2559         __get_month(__tm->tm_mon, __b, __e, __err, __ct);
2560         break;
2561     case 'M':
2562         __get_minute(__tm->tm_min, __b, __e, __err, __ct);
2563         break;
2564     case 'n':
2565     case 't':
2566         __get_white_space(__b, __e, __err, __ct);
2567         break;
2568     case 'p':
2569         __get_am_pm(__tm->tm_hour, __b, __e, __err, __ct);
2570         break;
2571     case 'r':
2572         {
2573         const char_type __fm[] = {'%', 'I', ':', '%', 'M', ':', '%', 'S', ' ', '%', 'p'};
2574         __b = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm)/sizeof(__fm[0]));
2575         }
2576         break;
2577     case 'R':
2578         {
2579         const char_type __fm[] = {'%', 'H', ':', '%', 'M'};
2580         __b = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm)/sizeof(__fm[0]));
2581         }
2582         break;
2583     case 'S':
2584         __get_second(__tm->tm_sec, __b, __e, __err, __ct);
2585         break;
2586     case 'T':
2587         {
2588         const char_type __fm[] = {'%', 'H', ':', '%', 'M', ':', '%', 'S'};
2589         __b = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm)/sizeof(__fm[0]));
2590         }
2591         break;
2592     case 'w':
2593         __get_weekday(__tm->tm_wday, __b, __e, __err, __ct);
2594         break;
2595     case 'x':
2596         return do_get_date(__b, __e, __iob, __err, __tm);
2597     case 'X':
2598         {
2599         const string_type& __fm = this->__X();
2600         __b = get(__b, __e, __iob, __err, __tm, __fm.data(), __fm.data() + __fm.size());
2601         }
2602         break;
2603     case 'y':
2604         __get_year(__tm->tm_year, __b, __e, __err, __ct);
2605         break;
2606     case 'Y':
2607         __get_year4(__tm->tm_year, __b, __e, __err, __ct);
2608         break;
2609     case '%':
2610         __get_percent(__b, __e, __err, __ct);
2611         break;
2612     default:
2613         __err |= ios_base::failbit;
2614     }
2615     return __b;
2616 }
2617
2618 _LIBCPP_EXTERN_TEMPLATE(class time_get<char>)
2619 _LIBCPP_EXTERN_TEMPLATE(class time_get<wchar_t>)
2620
2621 class __time_get
2622 {
2623 protected:
2624     locale_t __loc_;
2625
2626     __time_get(const char* __nm);
2627     __time_get(const string& __nm);
2628     ~__time_get();
2629 };
2630
2631 template <class _CharT>
2632 class __time_get_storage
2633     : public __time_get
2634 {
2635 protected:
2636     typedef basic_string<_CharT> string_type;
2637
2638     string_type __weeks_[14];
2639     string_type __months_[24];
2640     string_type __am_pm_[2];
2641     string_type __c_;
2642     string_type __r_;
2643     string_type __x_;
2644     string_type __X_;
2645
2646     explicit __time_get_storage(const char* __nm);
2647     explicit __time_get_storage(const string& __nm);
2648
2649     _LIBCPP_ALWAYS_INLINE ~__time_get_storage() {}
2650
2651     time_base::dateorder __do_date_order() const;
2652
2653 private:
2654     void init(const ctype<_CharT>&);
2655     string_type __analyze(char __fmt, const ctype<_CharT>&);
2656 };
2657
2658 template <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
2659 class _LIBCPP_VISIBLE time_get_byname
2660     : public time_get<_CharT, _InputIterator>,
2661       private __time_get_storage<_CharT>
2662 {
2663 public:
2664     typedef time_base::dateorder    dateorder;
2665     typedef _InputIterator          iter_type;
2666     typedef _CharT                  char_type;
2667     typedef basic_string<char_type> string_type;
2668
2669     _LIBCPP_INLINE_VISIBILITY
2670     explicit time_get_byname(const char* __nm, size_t __refs = 0)
2671         : time_get<_CharT, _InputIterator>(__refs),
2672           __time_get_storage<_CharT>(__nm) {}
2673     _LIBCPP_INLINE_VISIBILITY
2674     explicit time_get_byname(const string& __nm, size_t __refs = 0)
2675         : time_get<_CharT, _InputIterator>(__refs),
2676           __time_get_storage<_CharT>(__nm) {}
2677
2678 protected:
2679     _LIBCPP_INLINE_VISIBILITY
2680     ~time_get_byname() {}
2681
2682     _LIBCPP_INLINE_VISIBILITY
2683     virtual dateorder do_date_order() const {return this->__do_date_order();}
2684 private:
2685     _LIBCPP_INLINE_VISIBILITY
2686     virtual const string_type* __weeks() const  {return this->__weeks_;}
2687     _LIBCPP_INLINE_VISIBILITY
2688     virtual const string_type* __months() const {return this->__months_;}
2689     _LIBCPP_INLINE_VISIBILITY
2690     virtual const string_type* __am_pm() const  {return this->__am_pm_;}
2691     _LIBCPP_INLINE_VISIBILITY
2692     virtual const string_type& __c() const      {return this->__c_;}
2693     _LIBCPP_INLINE_VISIBILITY
2694     virtual const string_type& __r() const      {return this->__r_;}
2695     _LIBCPP_INLINE_VISIBILITY
2696     virtual const string_type& __x() const      {return this->__x_;}
2697     _LIBCPP_INLINE_VISIBILITY
2698     virtual const string_type& __X() const      {return this->__X_;}
2699 };
2700
2701 _LIBCPP_EXTERN_TEMPLATE(class time_get_byname<char>)
2702 _LIBCPP_EXTERN_TEMPLATE(class time_get_byname<wchar_t>)
2703
2704 class __time_put
2705 {
2706     locale_t __loc_;
2707 protected:
2708     _LIBCPP_ALWAYS_INLINE __time_put() : __loc_(_LIBCPP_GET_C_LOCALE) {}
2709     __time_put(const char* __nm);
2710     __time_put(const string& __nm);
2711     ~__time_put();
2712     void __do_put(char* __nb, char*& __ne, const tm* __tm,
2713                   char __fmt, char __mod) const;
2714     void __do_put(wchar_t* __wb, wchar_t*& __we, const tm* __tm,
2715                   char __fmt, char __mod) const;
2716 };
2717
2718 template <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
2719 class _LIBCPP_VISIBLE time_put
2720     : public locale::facet,
2721       private __time_put
2722 {
2723 public:
2724     typedef _CharT char_type;
2725     typedef _OutputIterator iter_type;
2726
2727     _LIBCPP_ALWAYS_INLINE
2728     explicit time_put(size_t __refs = 0)
2729         : locale::facet(__refs) {}
2730
2731     iter_type put(iter_type __s, ios_base& __iob, char_type __fl, const tm* __tm,
2732                   const char_type* __pb, const char_type* __pe) const;
2733
2734     _LIBCPP_ALWAYS_INLINE
2735     iter_type put(iter_type __s, ios_base& __iob, char_type __fl,
2736                   const tm* __tm, char __fmt, char __mod = 0) const
2737     {
2738         return do_put(__s, __iob, __fl, __tm, __fmt, __mod);
2739     }
2740
2741     static locale::id id;
2742
2743 protected:
2744     _LIBCPP_ALWAYS_INLINE
2745     ~time_put() {}
2746     virtual iter_type do_put(iter_type __s, ios_base&, char_type, const tm* __tm,
2747                              char __fmt, char __mod) const;
2748
2749     _LIBCPP_ALWAYS_INLINE
2750     explicit time_put(const char* __nm, size_t __refs)
2751         : locale::facet(__refs),
2752           __time_put(__nm) {}
2753     _LIBCPP_ALWAYS_INLINE
2754     explicit time_put(const string& __nm, size_t __refs)
2755         : locale::facet(__refs),
2756           __time_put(__nm) {}
2757 };
2758
2759 template <class _CharT, class _OutputIterator>
2760 locale::id
2761 time_put<_CharT, _OutputIterator>::id;
2762
2763 template <class _CharT, class _OutputIterator>
2764 _OutputIterator
2765 time_put<_CharT, _OutputIterator>::put(iter_type __s, ios_base& __iob,
2766                                        char_type __fl, const tm* __tm,
2767                                        const char_type* __pb,
2768                                        const char_type* __pe) const
2769 {
2770     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__iob.getloc());
2771     for (; __pb != __pe; ++__pb)
2772     {
2773         if (__ct.narrow(*__pb, 0) == '%')
2774         {
2775             if (++__pb == __pe)
2776             {
2777                 *__s++ = __pb[-1];
2778                 break;
2779             }
2780             char __mod = 0;
2781             char __fmt = __ct.narrow(*__pb, 0);
2782             if (__fmt == 'E' || __fmt == 'O')
2783             {
2784                 if (++__pb == __pe)
2785                 {
2786                     *__s++ = __pb[-2];
2787                     *__s++ = __pb[-1];
2788                     break;
2789                 }
2790                 __mod = __fmt;
2791                 __fmt = __ct.narrow(*__pb, 0);
2792             }
2793             __s = do_put(__s, __iob, __fl, __tm, __fmt, __mod);
2794         }
2795         else
2796             *__s++ = *__pb;
2797     }
2798     return __s;
2799 }
2800
2801 template <class _CharT, class _OutputIterator>
2802 _OutputIterator
2803 time_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base&,
2804                                           char_type, const tm* __tm,
2805                                           char __fmt, char __mod) const
2806 {
2807     char_type __nar[100];
2808     char_type* __nb = __nar;
2809     char_type* __ne = __nb + 100;
2810     __do_put(__nb, __ne, __tm, __fmt, __mod);
2811     return _VSTD::copy(__nb, __ne, __s);
2812 }
2813
2814 _LIBCPP_EXTERN_TEMPLATE(class time_put<char>)
2815 _LIBCPP_EXTERN_TEMPLATE(class time_put<wchar_t>)
2816
2817 template <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
2818 class _LIBCPP_VISIBLE time_put_byname
2819     : public time_put<_CharT, _OutputIterator>
2820 {
2821 public:
2822     _LIBCPP_ALWAYS_INLINE
2823     explicit time_put_byname(const char* __nm, size_t __refs = 0)
2824         : time_put<_CharT, _OutputIterator>(__nm, __refs) {}
2825
2826     _LIBCPP_ALWAYS_INLINE
2827     explicit time_put_byname(const string& __nm, size_t __refs = 0)
2828         : time_put<_CharT, _OutputIterator>(__nm, __refs) {}
2829
2830 protected:
2831     _LIBCPP_ALWAYS_INLINE
2832     ~time_put_byname() {}
2833 };
2834
2835 _LIBCPP_EXTERN_TEMPLATE(class time_put_byname<char>)
2836 _LIBCPP_EXTERN_TEMPLATE(class time_put_byname<wchar_t>)
2837
2838 // money_base
2839
2840 class _LIBCPP_VISIBLE money_base
2841 {
2842 public:
2843     enum part {none, space, symbol, sign, value};
2844     struct pattern {char field[4];};
2845
2846     _LIBCPP_ALWAYS_INLINE money_base() {}
2847 };
2848
2849 // moneypunct
2850
2851 template <class _CharT, bool _International = false>
2852 class _LIBCPP_VISIBLE moneypunct
2853     : public locale::facet,
2854       public money_base
2855 {
2856 public:
2857     typedef _CharT                  char_type;
2858     typedef basic_string<char_type> string_type;
2859
2860     _LIBCPP_ALWAYS_INLINE
2861     explicit moneypunct(size_t __refs = 0)
2862         : locale::facet(__refs) {}
2863
2864     _LIBCPP_ALWAYS_INLINE char_type   decimal_point() const {return do_decimal_point();}
2865     _LIBCPP_ALWAYS_INLINE char_type   thousands_sep() const {return do_thousands_sep();}
2866     _LIBCPP_ALWAYS_INLINE string      grouping()      const {return do_grouping();}
2867     _LIBCPP_ALWAYS_INLINE string_type curr_symbol()   const {return do_curr_symbol();}
2868     _LIBCPP_ALWAYS_INLINE string_type positive_sign() const {return do_positive_sign();}
2869     _LIBCPP_ALWAYS_INLINE string_type negative_sign() const {return do_negative_sign();}
2870     _LIBCPP_ALWAYS_INLINE int         frac_digits()   const {return do_frac_digits();}
2871     _LIBCPP_ALWAYS_INLINE pattern     pos_format()    const {return do_pos_format();}
2872     _LIBCPP_ALWAYS_INLINE pattern     neg_format()    const {return do_neg_format();}
2873
2874     static locale::id id;
2875     static const bool intl = _International;
2876
2877 protected:
2878     _LIBCPP_ALWAYS_INLINE
2879     ~moneypunct() {}
2880
2881     virtual char_type   do_decimal_point() const {return numeric_limits<char_type>::max();}
2882     virtual char_type   do_thousands_sep() const {return numeric_limits<char_type>::max();}
2883     virtual string      do_grouping()      const {return string();}
2884     virtual string_type do_curr_symbol()   const {return string_type();}
2885     virtual string_type do_positive_sign() const {return string_type();}
2886     virtual string_type do_negative_sign() const {return string_type(1, '-');}
2887     virtual int         do_frac_digits()   const {return 0;}
2888     virtual pattern     do_pos_format()    const
2889         {pattern __p = {{symbol, sign, none, value}}; return __p;}
2890     virtual pattern     do_neg_format()    const
2891         {pattern __p = {{symbol, sign, none, value}}; return __p;}
2892 };
2893
2894 template <class _CharT, bool _International>
2895 locale::id
2896 moneypunct<_CharT, _International>::id;
2897
2898 template <class _CharT, bool _International>
2899 const bool
2900 moneypunct<_CharT, _International>::intl;
2901
2902 _LIBCPP_EXTERN_TEMPLATE(class moneypunct<char, false>)
2903 _LIBCPP_EXTERN_TEMPLATE(class moneypunct<char, true>)
2904 _LIBCPP_EXTERN_TEMPLATE(class moneypunct<wchar_t, false>)
2905 _LIBCPP_EXTERN_TEMPLATE(class moneypunct<wchar_t, true>)
2906
2907 // moneypunct_byname
2908
2909 template <class _CharT, bool _International = false>
2910 class _LIBCPP_VISIBLE moneypunct_byname
2911     : public moneypunct<_CharT, _International>
2912 {
2913 public:
2914     typedef money_base::pattern  pattern;
2915     typedef _CharT                  char_type;
2916     typedef basic_string<char_type> string_type;
2917
2918     _LIBCPP_ALWAYS_INLINE
2919     explicit moneypunct_byname(const char* __nm, size_t __refs = 0)
2920         : moneypunct<_CharT, _International>(__refs) {init(__nm);}
2921
2922     _LIBCPP_ALWAYS_INLINE
2923     explicit moneypunct_byname(const string& __nm, size_t __refs = 0)
2924         : moneypunct<_CharT, _International>(__refs) {init(__nm.c_str());}
2925
2926 protected:
2927     _LIBCPP_ALWAYS_INLINE
2928     ~moneypunct_byname() {}
2929
2930     virtual char_type   do_decimal_point() const {return __decimal_point_;}
2931     virtual char_type   do_thousands_sep() const {return __thousands_sep_;}
2932     virtual string      do_grouping()      const {return __grouping_;}
2933     virtual string_type do_curr_symbol()   const {return __curr_symbol_;}
2934     virtual string_type do_positive_sign() const {return __positive_sign_;}
2935     virtual string_type do_negative_sign() const {return __negative_sign_;}
2936     virtual int         do_frac_digits()   const {return __frac_digits_;}
2937     virtual pattern     do_pos_format()    const {return __pos_format_;}
2938     virtual pattern     do_neg_format()    const {return __neg_format_;}
2939
2940 private:
2941     char_type   __decimal_point_;
2942     char_type   __thousands_sep_;
2943     string      __grouping_;
2944     string_type __curr_symbol_;
2945     string_type __positive_sign_;
2946     string_type __negative_sign_;
2947     int         __frac_digits_;
2948     pattern     __pos_format_;
2949     pattern     __neg_format_;
2950
2951     void init(const char*);
2952 };
2953
2954 template<> void moneypunct_byname<char, false>::init(const char*);
2955 template<> void moneypunct_byname<char, true>::init(const char*);
2956 template<> void moneypunct_byname<wchar_t, false>::init(const char*);
2957 template<> void moneypunct_byname<wchar_t, true>::init(const char*);
2958
2959 _LIBCPP_EXTERN_TEMPLATE(class moneypunct_byname<char, false>)
2960 _LIBCPP_EXTERN_TEMPLATE(class moneypunct_byname<char, true>)
2961 _LIBCPP_EXTERN_TEMPLATE(class moneypunct_byname<wchar_t, false>)
2962 _LIBCPP_EXTERN_TEMPLATE(class moneypunct_byname<wchar_t, true>)
2963
2964 // money_get
2965
2966 template <class _CharT>
2967 class __money_get
2968 {
2969 protected:
2970     typedef _CharT                  char_type;
2971     typedef basic_string<char_type> string_type;
2972
2973     _LIBCPP_ALWAYS_INLINE __money_get() {}
2974
2975     static void __gather_info(bool __intl, const locale& __loc,
2976                               money_base::pattern& __pat, char_type& __dp,
2977                               char_type& __ts, string& __grp,
2978                               string_type& __sym, string_type& __psn,
2979                               string_type& __nsn, int& __fd);
2980 };
2981
2982 template <class _CharT>
2983 void
2984 __money_get<_CharT>::__gather_info(bool __intl, const locale& __loc,
2985                                    money_base::pattern& __pat, char_type& __dp,
2986                                    char_type& __ts, string& __grp,
2987                                    string_type& __sym, string_type& __psn,
2988                                    string_type& __nsn, int& __fd)
2989 {
2990     if (__intl)
2991     {
2992         const moneypunct<char_type, true>& __mp =
2993             use_facet<moneypunct<char_type, true> >(__loc);
2994         __pat = __mp.neg_format();
2995         __nsn = __mp.negative_sign();
2996         __psn = __mp.positive_sign();
2997         __dp = __mp.decimal_point();
2998         __ts = __mp.thousands_sep();
2999         __grp = __mp.grouping();
3000         __sym = __mp.curr_symbol();
3001         __fd = __mp.frac_digits();
3002     }
3003     else
3004     {
3005         const moneypunct<char_type, false>& __mp =
3006             use_facet<moneypunct<char_type, false> >(__loc);
3007         __pat = __mp.neg_format();
3008         __nsn = __mp.negative_sign();
3009         __psn = __mp.positive_sign();
3010         __dp = __mp.decimal_point();
3011         __ts = __mp.thousands_sep();
3012         __grp = __mp.grouping();
3013         __sym = __mp.curr_symbol();
3014         __fd = __mp.frac_digits();
3015     }
3016 }
3017
3018 _LIBCPP_EXTERN_TEMPLATE(class __money_get<char>)
3019 _LIBCPP_EXTERN_TEMPLATE(class __money_get<wchar_t>)
3020
3021 template <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
3022 class _LIBCPP_VISIBLE money_get
3023     : public locale::facet,
3024       private __money_get<_CharT>
3025 {
3026 public:
3027     typedef _CharT                  char_type;
3028     typedef _InputIterator          iter_type;
3029     typedef basic_string<char_type> string_type;
3030
3031     _LIBCPP_ALWAYS_INLINE
3032     explicit money_get(size_t __refs = 0)
3033         : locale::facet(__refs) {}
3034
3035     _LIBCPP_ALWAYS_INLINE
3036     iter_type get(iter_type __b, iter_type __e, bool __intl, ios_base& __iob,
3037                   ios_base::iostate& __err, long double& __v) const
3038     {
3039         return do_get(__b, __e, __intl, __iob, __err, __v);
3040     }
3041
3042     _LIBCPP_ALWAYS_INLINE
3043     iter_type get(iter_type __b, iter_type __e, bool __intl, ios_base& __iob,
3044                   ios_base::iostate& __err, string_type& __v) const
3045     {
3046         return do_get(__b, __e, __intl, __iob, __err, __v);
3047     }
3048
3049     static locale::id id;
3050
3051 protected:
3052
3053     _LIBCPP_ALWAYS_INLINE
3054     ~money_get() {}
3055
3056     virtual iter_type do_get(iter_type __b, iter_type __e, bool __intl,
3057                              ios_base& __iob, ios_base::iostate& __err,
3058                              long double& __v) const;
3059     virtual iter_type do_get(iter_type __b, iter_type __e, bool __intl,
3060                              ios_base& __iob, ios_base::iostate& __err,
3061                              string_type& __v) const;
3062
3063 private:
3064     static bool __do_get(iter_type& __b, iter_type __e,
3065                          bool __intl, const locale& __loc,
3066                          ios_base::fmtflags __flags, ios_base::iostate& __err,
3067                          bool& __neg, const ctype<char_type>& __ct,
3068                          unique_ptr<char_type, void(*)(void*)>& __wb,
3069                          char_type*& __wn, char_type* __we);
3070 };
3071
3072 template <class _CharT, class _InputIterator>
3073 locale::id
3074 money_get<_CharT, _InputIterator>::id;
3075
3076 void __do_nothing(void*);
3077
3078 template <class _Tp>
3079 _LIBCPP_HIDDEN
3080 void
3081 __double_or_nothing(unique_ptr<_Tp, void(*)(void*)>& __b, _Tp*& __n, _Tp*& __e)
3082 {
3083     bool __owns = __b.get_deleter() != __do_nothing;
3084     size_t __cur_cap = static_cast<size_t>(__e-__b.get()) * sizeof(_Tp);
3085     size_t __new_cap = __cur_cap < numeric_limits<size_t>::max() / 2 ?
3086                        2 * __cur_cap : numeric_limits<size_t>::max();
3087     size_t __n_off = static_cast<size_t>(__n - __b.get());
3088     _Tp* __t = (_Tp*)realloc(__owns ? __b.get() : 0, __new_cap);
3089     if (__t == 0)
3090         __throw_bad_alloc();
3091     if (__owns)
3092         __b.release();
3093     __b = unique_ptr<_Tp, void(*)(void*)>(__t, free);
3094     __new_cap /= sizeof(_Tp);
3095     __n = __b.get() + __n_off;
3096     __e = __b.get() + __new_cap;
3097 }
3098
3099 // true == success
3100 template <class _CharT, class _InputIterator>
3101 bool
3102 money_get<_CharT, _InputIterator>::__do_get(iter_type& __b, iter_type __e,
3103                                             bool __intl, const locale& __loc,
3104                                             ios_base::fmtflags __flags,
3105                                             ios_base::iostate& __err,
3106                                             bool& __neg,
3107                                             const ctype<char_type>& __ct,
3108                                             unique_ptr<char_type, void(*)(void*)>& __wb,
3109                                             char_type*& __wn, char_type* __we)
3110 {
3111     const unsigned __bz = 100;
3112     unsigned __gbuf[__bz];
3113     unique_ptr<unsigned, void(*)(void*)> __gb(__gbuf, __do_nothing);
3114     unsigned* __gn = __gb.get();
3115     unsigned* __ge = __gn + __bz;
3116     money_base::pattern __pat;
3117     char_type __dp;
3118     char_type __ts;
3119     string __grp;
3120     string_type __sym;
3121     string_type __psn;
3122     string_type __nsn;
3123     // Capture the spaces read into money_base::{space,none} so they
3124     // can be compared to initial spaces in __sym.
3125     string_type __spaces;
3126     int __fd;
3127     __money_get<_CharT>::__gather_info(__intl, __loc, __pat, __dp, __ts, __grp,
3128                                        __sym, __psn, __nsn, __fd);
3129     const string_type* __trailing_sign = 0;
3130     __wn = __wb.get();
3131     for (unsigned __p = 0; __p < 4 && __b != __e; ++__p)
3132     {
3133         switch (__pat.field[__p])
3134         {
3135         case money_base::space:
3136             if (__p != 3)
3137             {
3138                 if (__ct.is(ctype_base::space, *__b))
3139                     __spaces.push_back(*__b++);
3140                 else
3141                 {
3142                     __err |= ios_base::failbit;
3143                     return false;
3144                 }
3145             }
3146             // drop through
3147         case money_base::none:
3148             if (__p != 3)
3149             {
3150                 while (__b != __e && __ct.is(ctype_base::space, *__b))
3151                     __spaces.push_back(*__b++);
3152             }
3153             break;
3154         case money_base::sign:
3155             if (__psn.size() + __nsn.size() > 0)
3156             {
3157                 if (__psn.size() == 0 || __nsn.size() == 0)
3158                 {   // sign is optional
3159                     if (__psn.size() > 0)
3160                     {   // __nsn.size() == 0
3161                         if (*__b == __psn[0])
3162                         {
3163                             ++__b;
3164                             if (__psn.size() > 1)
3165                                 __trailing_sign = &__psn;
3166                         }
3167                         else
3168                             __neg = true;
3169                     }
3170                     else if (*__b == __nsn[0])  // __nsn.size() > 0 &&  __psn.size() == 0
3171                     {
3172                         ++__b;
3173                         __neg = true;
3174                         if (__nsn.size() > 1)
3175                             __trailing_sign = &__nsn;
3176                     }
3177                 }
3178                 else  // sign is required
3179                 {
3180                     if (*__b == __psn[0])
3181                     {
3182                         ++__b;
3183                         if (__psn.size() > 1)
3184                             __trailing_sign = &__psn;
3185                     }
3186                     else if (*__b == __nsn[0])
3187                     {
3188                         ++__b;
3189                         __neg = true;
3190                         if (__nsn.size() > 1)
3191                             __trailing_sign = &__nsn;
3192                     }
3193                     else
3194                     {
3195                         __err |= ios_base::failbit;
3196                         return false;
3197                     }
3198                 }
3199             }
3200             break;
3201         case money_base::symbol:
3202             {
3203             bool __more_needed = __trailing_sign ||
3204                                  (__p < 2)       ||
3205                                  (__p == 2 && __pat.field[3] != static_cast<char>(money_base::none));
3206             bool __sb = __flags & ios_base::showbase;
3207             if (__sb || __more_needed)
3208             {
3209                 typename string_type::const_iterator __sym_space_end = __sym.begin();
3210                 if (__p > 0 && (__pat.field[__p - 1] == money_base::none ||
3211                                 __pat.field[__p - 1] == money_base::space)) {
3212                     // Match spaces we've already read against spaces at
3213                     // the beginning of __sym.
3214                     while (__sym_space_end != __sym.end() &&
3215                            __ct.is(ctype_base::space, *__sym_space_end))
3216                         ++__sym_space_end;
3217                     const size_t __num_spaces = __sym_space_end - __sym.begin();
3218                     if (__num_spaces > __spaces.size() ||
3219                         !equal(__spaces.end() - __num_spaces, __spaces.end(),
3220                                __sym.begin())) {
3221                         // No match. Put __sym_space_end back at the
3222                         // beginning of __sym, which will prevent a
3223                         // match in the next loop.
3224                         __sym_space_end = __sym.begin();
3225                     }
3226                 }
3227                 typename string_type::const_iterator __sym_curr_char = __sym_space_end;
3228                 while (__sym_curr_char != __sym.end() && __b != __e &&
3229                        *__b == *__sym_curr_char) {
3230                     ++__b;
3231                     ++__sym_curr_char;
3232                 }
3233                 if (__sb && __sym_curr_char != __sym.end())
3234                 {
3235                     __err |= ios_base::failbit;
3236                     return false;
3237                 }
3238             }
3239             }
3240             break;
3241         case money_base::value:
3242             {
3243             unsigned __ng = 0;
3244             for (; __b != __e; ++__b)
3245             {
3246                 char_type __c = *__b;
3247                 if (__ct.is(ctype_base::digit, __c))
3248                 {
3249                     if (__wn == __we)
3250                         __double_or_nothing(__wb, __wn, __we);
3251                     *__wn++ = __c;
3252                     ++__ng;
3253                 }
3254                 else if (__grp.size() > 0 && __ng > 0 && __c == __ts)
3255                 {
3256                     if (__gn == __ge)
3257                         __double_or_nothing(__gb, __gn, __ge);
3258                     *__gn++ = __ng;
3259                     __ng = 0;
3260                 }
3261                 else
3262                     break;
3263             }
3264             if (__gb.get() != __gn && __ng > 0)
3265             {
3266                 if (__gn == __ge)
3267                     __double_or_nothing(__gb, __gn, __ge);
3268                 *__gn++ = __ng;
3269             }
3270             if (__fd > 0)
3271             {
3272                 if (__b == __e || *__b != __dp)
3273                 {
3274                     __err |= ios_base::failbit;
3275                     return false;
3276                 }
3277                 for (++__b; __fd > 0; --__fd, ++__b)
3278                 {
3279                     if (__b == __e || !__ct.is(ctype_base::digit, *__b))
3280                     {
3281                         __err |= ios_base::failbit;
3282                         return false;
3283                     }
3284                     if (__wn == __we)
3285                         __double_or_nothing(__wb, __wn, __we);
3286                     *__wn++ = *__b;
3287                 }
3288             }
3289             if (__wn == __wb.get())
3290             {
3291                 __err |= ios_base::failbit;
3292                 return false;
3293             }
3294             }
3295             break;
3296         }
3297     }
3298     if (__trailing_sign)
3299     {
3300         for (unsigned __i = 1; __i < __trailing_sign->size(); ++__i, ++__b)
3301         {
3302             if (__b == __e || *__b != (*__trailing_sign)[__i])
3303             {
3304                 __err |= ios_base::failbit;
3305                 return false;
3306             }
3307         }
3308     }
3309     if (__gb.get() != __gn)
3310     {
3311         ios_base::iostate __et = ios_base::goodbit;
3312         __check_grouping(__grp, __gb.get(), __gn, __et);
3313         if (__et)
3314         {
3315             __err |= ios_base::failbit;
3316             return false;
3317         }
3318     }
3319     return true;
3320 }
3321
3322 template <class _CharT, class _InputIterator>
3323 _InputIterator
3324 money_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
3325                                           bool __intl, ios_base& __iob,
3326                                           ios_base::iostate& __err,
3327                                           long double& __v) const
3328 {
3329     const int __bz = 100;
3330     char_type __wbuf[__bz];
3331     unique_ptr<char_type, void(*)(void*)> __wb(__wbuf, __do_nothing);
3332     char_type* __wn;
3333     char_type* __we = __wbuf + __bz;
3334     locale __loc = __iob.getloc();
3335     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__loc);
3336     bool __neg = false;
3337     if (__do_get(__b, __e, __intl, __loc, __iob.flags(), __err, __neg, __ct,
3338                  __wb, __wn, __we))
3339     {
3340         const char __src[] = "0123456789";
3341         char_type __atoms[sizeof(__src)-1];
3342         __ct.widen(__src, __src + (sizeof(__src)-1), __atoms);
3343         char __nbuf[__bz];
3344         char* __nc = __nbuf;
3345         unique_ptr<char, void(*)(void*)> __h(0, free);
3346         if (__wn - __wb.get() > __bz-2)
3347         {
3348             __h.reset((char*)malloc(static_cast<size_t>(__wn - __wb.get() + 2)));
3349             if (__h.get() == 0)
3350                 __throw_bad_alloc();
3351             __nc = __h.get();
3352         }
3353         if (__neg)
3354             *__nc++ = '-';
3355         for (const char_type* __w = __wb.get(); __w < __wn; ++__w, ++__nc)
3356             *__nc = __src[find(__atoms, __atoms+sizeof(__atoms), *__w) - __atoms];
3357         *__nc = char();
3358         if (sscanf(__nbuf, "%Lf", &__v) != 1)
3359             __throw_runtime_error("money_get error");
3360     }
3361     if (__b == __e)
3362         __err |= ios_base::eofbit;
3363     return __b;
3364 }
3365
3366 template <class _CharT, class _InputIterator>
3367 _InputIterator
3368 money_get<_CharT, _InputIterator>::do_get(iter_type __b, iter_type __e,
3369                                           bool __intl, ios_base& __iob,
3370                                           ios_base::iostate& __err,
3371                                           string_type& __v) const
3372 {
3373     const int __bz = 100;
3374     char_type __wbuf[__bz];
3375     unique_ptr<char_type, void(*)(void*)> __wb(__wbuf, __do_nothing);
3376     char_type* __wn;
3377     char_type* __we = __wbuf + __bz;
3378     locale __loc = __iob.getloc();
3379     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__loc);
3380     bool __neg = false;
3381     if (__do_get(__b, __e, __intl, __loc, __iob.flags(), __err, __neg, __ct,
3382                  __wb, __wn, __we))
3383     {
3384         __v.clear();
3385         if (__neg)
3386             __v.push_back(__ct.widen('-'));
3387         char_type __z = __ct.widen('0');
3388         char_type* __w;
3389         for (__w = __wb.get(); __w < __wn-1; ++__w)
3390             if (*__w != __z)
3391                 break;
3392         __v.append(__w, __wn);
3393     }
3394     if (__b == __e)
3395         __err |= ios_base::eofbit;
3396     return __b;
3397 }
3398
3399 _LIBCPP_EXTERN_TEMPLATE(class money_get<char>)
3400 _LIBCPP_EXTERN_TEMPLATE(class money_get<wchar_t>)
3401
3402 // money_put
3403
3404 template <class _CharT>
3405 class __money_put
3406 {
3407 protected:
3408     typedef _CharT                  char_type;
3409     typedef basic_string<char_type> string_type;
3410
3411     _LIBCPP_ALWAYS_INLINE __money_put() {}
3412
3413     static void __gather_info(bool __intl, bool __neg, const locale& __loc,
3414                               money_base::pattern& __pat, char_type& __dp,
3415                               char_type& __ts, string& __grp,
3416                               string_type& __sym, string_type& __sn,
3417                               int& __fd);
3418     static void __format(char_type* __mb, char_type*& __mi, char_type*& __me,
3419                          ios_base::fmtflags __flags,
3420                          const char_type* __db, const char_type* __de,
3421                          const ctype<char_type>& __ct, bool __neg,
3422                          const money_base::pattern& __pat, char_type __dp,
3423                          char_type __ts, const string& __grp,
3424                          const string_type& __sym, const string_type& __sn,
3425                          int __fd);
3426 };
3427
3428 template <class _CharT>
3429 void
3430 __money_put<_CharT>::__gather_info(bool __intl, bool __neg, const locale& __loc,
3431                                    money_base::pattern& __pat, char_type& __dp,
3432                                    char_type& __ts, string& __grp,
3433                                    string_type& __sym, string_type& __sn,
3434                                    int& __fd)
3435 {
3436     if (__intl)
3437     {
3438         const moneypunct<char_type, true>& __mp =
3439             use_facet<moneypunct<char_type, true> >(__loc);
3440         if (__neg)
3441         {
3442             __pat = __mp.neg_format();
3443             __sn = __mp.negative_sign();
3444         }
3445         else
3446         {
3447             __pat = __mp.pos_format();
3448             __sn = __mp.positive_sign();
3449         }
3450         __dp = __mp.decimal_point();
3451         __ts = __mp.thousands_sep();
3452         __grp = __mp.grouping();
3453         __sym = __mp.curr_symbol();
3454         __fd = __mp.frac_digits();
3455     }
3456     else
3457     {
3458         const moneypunct<char_type, false>& __mp =
3459             use_facet<moneypunct<char_type, false> >(__loc);
3460         if (__neg)
3461         {
3462             __pat = __mp.neg_format();
3463             __sn = __mp.negative_sign();
3464         }
3465         else
3466         {
3467             __pat = __mp.pos_format();
3468             __sn = __mp.positive_sign();
3469         }
3470         __dp = __mp.decimal_point();
3471         __ts = __mp.thousands_sep();
3472         __grp = __mp.grouping();
3473         __sym = __mp.curr_symbol();
3474         __fd = __mp.frac_digits();
3475     }
3476 }
3477
3478 template <class _CharT>
3479 void
3480 __money_put<_CharT>::__format(char_type* __mb, char_type*& __mi, char_type*& __me,
3481                               ios_base::fmtflags __flags,
3482                               const char_type* __db, const char_type* __de,
3483                               const ctype<char_type>& __ct, bool __neg,
3484                               const money_base::pattern& __pat, char_type __dp,
3485                               char_type __ts, const string& __grp,
3486                               const string_type& __sym, const string_type& __sn,
3487                               int __fd)
3488 {
3489     __me = __mb;
3490     for (unsigned __p = 0; __p < 4; ++__p)
3491     {
3492         switch (__pat.field[__p])
3493         {
3494         case money_base::none:
3495             __mi = __me;
3496             break;
3497         case money_base::space:
3498             __mi = __me;
3499             *__me++ = __ct.widen(' ');
3500             break;
3501         case money_base::sign:
3502             if (!__sn.empty())
3503                 *__me++ = __sn[0];
3504             break;
3505         case money_base::symbol:
3506             if (!__sym.empty() && (__flags & ios_base::showbase))
3507                 __me = _VSTD::copy(__sym.begin(), __sym.end(), __me);
3508             break;
3509         case money_base::value:
3510             {
3511             // remember start of value so we can reverse it
3512             char_type* __t = __me;
3513             // find beginning of digits
3514             if (__neg)
3515                 ++__db;
3516             // find end of digits
3517             const char_type* __d;
3518             for (__d = __db; __d < __de; ++__d)
3519                 if (!__ct.is(ctype_base::digit, *__d))
3520                     break;
3521             // print fractional part
3522             if (__fd > 0)
3523             {
3524                 int __f;
3525                 for (__f = __fd; __d > __db && __f > 0; --__f)
3526                     *__me++ = *--__d;
3527                 char_type __z = __f > 0 ? __ct.widen('0') : char_type();
3528                 for (; __f > 0; --__f)
3529                     *__me++ = __z;
3530                 *__me++ = __dp;
3531             }
3532             // print units part
3533             if (__d == __db)
3534             {
3535                 *__me++ = __ct.widen('0');
3536             }
3537             else
3538             {
3539                 unsigned __ng = 0;
3540                 unsigned __ig = 0;
3541                 unsigned __gl = __grp.empty() ? numeric_limits<unsigned>::max()
3542                                               : static_cast<unsigned>(__grp[__ig]);
3543                 while (__d != __db)
3544                 {
3545                     if (__ng == __gl)
3546                     {
3547                         *__me++ = __ts;
3548                         __ng = 0;
3549                         if (++__ig < __grp.size())
3550                             __gl = __grp[__ig] == numeric_limits<char>::max() ?
3551                                         numeric_limits<unsigned>::max() :
3552                                         static_cast<unsigned>(__grp[__ig]);
3553                     }
3554                     *__me++ = *--__d;
3555                     ++__ng;
3556                 }
3557             }
3558             // reverse it
3559             reverse(__t, __me);
3560             }
3561             break;
3562         }
3563     }
3564     // print rest of sign, if any
3565     if (__sn.size() > 1)
3566         __me = _VSTD::copy(__sn.begin()+1, __sn.end(), __me);
3567     // set alignment
3568     if ((__flags & ios_base::adjustfield) == ios_base::left)
3569         __mi = __me;
3570     else if ((__flags & ios_base::adjustfield) != ios_base::internal)
3571         __mi = __mb;
3572 }
3573
3574 _LIBCPP_EXTERN_TEMPLATE(class __money_put<char>)
3575 _LIBCPP_EXTERN_TEMPLATE(class __money_put<wchar_t>)
3576
3577 template <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
3578 class _LIBCPP_VISIBLE money_put
3579     : public locale::facet,
3580       private __money_put<_CharT>
3581 {
3582 public:
3583     typedef _CharT                  char_type;
3584     typedef _OutputIterator         iter_type;
3585     typedef basic_string<char_type> string_type;
3586
3587     _LIBCPP_ALWAYS_INLINE
3588     explicit money_put(size_t __refs = 0)
3589         : locale::facet(__refs) {}
3590
3591     _LIBCPP_ALWAYS_INLINE
3592     iter_type put(iter_type __s, bool __intl, ios_base& __iob, char_type __fl,
3593                   long double __units) const
3594     {
3595         return do_put(__s, __intl, __iob, __fl, __units);
3596     }
3597
3598     _LIBCPP_ALWAYS_INLINE
3599     iter_type put(iter_type __s, bool __intl, ios_base& __iob, char_type __fl,
3600                   const string_type& __digits) const
3601     {
3602         return do_put(__s, __intl, __iob, __fl, __digits);
3603     }
3604
3605     static locale::id id;
3606
3607 protected:
3608     _LIBCPP_ALWAYS_INLINE
3609     ~money_put() {}
3610
3611     virtual iter_type do_put(iter_type __s, bool __intl, ios_base& __iob,
3612                              char_type __fl, long double __units) const;
3613     virtual iter_type do_put(iter_type __s, bool __intl, ios_base& __iob,
3614                              char_type __fl, const string_type& __digits) const;
3615 };
3616
3617 template <class _CharT, class _OutputIterator>
3618 locale::id
3619 money_put<_CharT, _OutputIterator>::id;
3620
3621 template <class _CharT, class _OutputIterator>
3622 _OutputIterator
3623 money_put<_CharT, _OutputIterator>::do_put(iter_type __s, bool __intl,
3624                                            ios_base& __iob, char_type __fl,
3625                                            long double __units) const
3626 {
3627     // convert to char
3628     const size_t __bs = 100;
3629     char __buf[__bs];
3630     char* __bb = __buf;
3631     char_type __digits[__bs];
3632     char_type* __db = __digits;
3633     size_t __n = static_cast<size_t>(snprintf(__bb, __bs, "%.0Lf", __units));
3634     unique_ptr<char, void(*)(void*)> __hn(0, free);
3635     unique_ptr<char_type, void(*)(void*)> __hd(0, free);
3636     // secure memory for digit storage
3637     if (__n > __bs-1)
3638     {
3639 #ifdef _LIBCPP_LOCALE__L_EXTENSIONS
3640         __n = static_cast<size_t>(asprintf_l(&__bb, _LIBCPP_GET_C_LOCALE, "%.0Lf", __units));
3641 #else
3642         __n = __asprintf_l(&__bb, __cloc(), "%.0Lf", __units);
3643 #endif
3644         if (__bb == 0)
3645             __throw_bad_alloc();
3646         __hn.reset(__bb);
3647         __hd.reset((char_type*)malloc(__n * sizeof(char_type)));
3648         if (__hd == nullptr)
3649             __throw_bad_alloc();
3650         __db = __hd.get();
3651     }
3652     // gather info
3653     locale __loc = __iob.getloc();
3654     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__loc);
3655     __ct.widen(__bb, __bb + __n, __db);
3656     bool __neg = __n > 0 && __bb[0] == '-';
3657     money_base::pattern __pat;
3658     char_type __dp;
3659     char_type __ts;
3660     string __grp;
3661     string_type __sym;
3662     string_type __sn;
3663     int __fd;
3664     this->__gather_info(__intl, __neg, __loc, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
3665     // secure memory for formatting
3666     char_type __mbuf[__bs];
3667     char_type* __mb = __mbuf;
3668     unique_ptr<char_type, void(*)(void*)> __hw(0, free);
3669     size_t __exn = static_cast<int>(__n) > __fd ?
3670                    (__n - static_cast<size_t>(__fd)) * 2 + __sn.size() +
3671                     __sym.size() + static_cast<size_t>(__fd) + 1
3672                  : __sn.size() + __sym.size() + static_cast<size_t>(__fd) + 2;
3673     if (__exn > __bs)
3674     {
3675         __hw.reset((char_type*)malloc(__exn * sizeof(char_type)));
3676         __mb = __hw.get();
3677         if (__mb == 0)
3678             __throw_bad_alloc();
3679     }
3680     // format
3681     char_type* __mi;
3682     char_type* __me;
3683     this->__format(__mb, __mi, __me, __iob.flags(),
3684                    __db, __db + __n, __ct,
3685                    __neg, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
3686     return __pad_and_output(__s, __mb, __mi, __me, __iob, __fl);
3687 }
3688
3689 template <class _CharT, class _OutputIterator>
3690 _OutputIterator
3691 money_put<_CharT, _OutputIterator>::do_put(iter_type __s, bool __intl,
3692                                            ios_base& __iob, char_type __fl,
3693                                            const string_type& __digits) const
3694 {
3695     // gather info
3696     locale __loc = __iob.getloc();
3697     const ctype<char_type>& __ct = use_facet<ctype<char_type> >(__loc);
3698     bool __neg = __digits.size() > 0 && __digits[0] == __ct.widen('-');
3699     money_base::pattern __pat;
3700     char_type __dp;
3701     char_type __ts;
3702     string __grp;
3703     string_type __sym;
3704     string_type __sn;
3705     int __fd;
3706     this->__gather_info(__intl, __neg, __loc, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
3707     // secure memory for formatting
3708     char_type __mbuf[100];
3709     char_type* __mb = __mbuf;
3710     unique_ptr<char_type, void(*)(void*)> __h(0, free);
3711     size_t __exn = static_cast<int>(__digits.size()) > __fd ?
3712                    (__digits.size() - static_cast<size_t>(__fd)) * 2 +
3713                     __sn.size() + __sym.size() + static_cast<size_t>(__fd) + 1
3714                  : __sn.size() + __sym.size() + static_cast<size_t>(__fd) + 2;
3715     if (__exn > 100)
3716     {
3717         __h.reset((char_type*)malloc(__exn * sizeof(char_type)));
3718         __mb = __h.get();
3719         if (__mb == 0)
3720             __throw_bad_alloc();
3721     }
3722     // format
3723     char_type* __mi;
3724     char_type* __me;
3725     this->__format(__mb, __mi, __me, __iob.flags(),
3726                    __digits.data(), __digits.data() + __digits.size(), __ct,
3727                    __neg, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
3728     return __pad_and_output(__s, __mb, __mi, __me, __iob, __fl);
3729 }
3730
3731 _LIBCPP_EXTERN_TEMPLATE(class money_put<char>)
3732 _LIBCPP_EXTERN_TEMPLATE(class money_put<wchar_t>)
3733
3734 // messages
3735
3736 class _LIBCPP_VISIBLE messages_base
3737 {
3738 public:
3739     typedef ptrdiff_t catalog;
3740
3741     _LIBCPP_ALWAYS_INLINE messages_base() {}
3742 };
3743
3744 template <class _CharT>
3745 class _LIBCPP_VISIBLE messages
3746     : public locale::facet,
3747       public messages_base
3748 {
3749 public:
3750     typedef _CharT               char_type;
3751     typedef basic_string<_CharT> string_type;
3752
3753     _LIBCPP_ALWAYS_INLINE
3754     explicit messages(size_t __refs = 0)
3755         : locale::facet(__refs) {}
3756
3757     _LIBCPP_ALWAYS_INLINE
3758     catalog open(const basic_string<char>& __nm, const locale& __loc) const
3759     {
3760         return do_open(__nm, __loc);
3761     }
3762
3763     _LIBCPP_ALWAYS_INLINE
3764     string_type get(catalog __c, int __set, int __msgid,
3765                     const string_type& __dflt) const
3766     {
3767         return do_get(__c, __set, __msgid, __dflt);
3768     }
3769
3770     _LIBCPP_ALWAYS_INLINE
3771     void close(catalog __c) const
3772     {
3773         do_close(__c);
3774     }
3775
3776     static locale::id id;
3777
3778 protected:
3779     _LIBCPP_ALWAYS_INLINE
3780     ~messages() {}
3781
3782     virtual catalog do_open(const basic_string<char>&, const locale&) const;
3783     virtual string_type do_get(catalog, int __set, int __msgid,
3784                                const string_type& __dflt) const;
3785     virtual void do_close(catalog) const;
3786 };
3787
3788 template <class _CharT>
3789 locale::id
3790 messages<_CharT>::id;
3791
3792 template <class _CharT>
3793 typename messages<_CharT>::catalog
3794 messages<_CharT>::do_open(const basic_string<char>& __nm, const locale&) const
3795 {
3796 #if _WIN32
3797     return -1;
3798 #else // _WIN32
3799     catalog __cat = (catalog)catopen(__nm.c_str(), NL_CAT_LOCALE);
3800     if (__cat != -1)
3801         __cat = static_cast<catalog>((static_cast<size_t>(__cat) >> 1));
3802     return __cat;
3803 #endif // _WIN32
3804 }
3805
3806 template <class _CharT>
3807 typename messages<_CharT>::string_type
3808 messages<_CharT>::do_get(catalog __c, int __set, int __msgid,
3809                          const string_type& __dflt) const
3810 {
3811 #if _WIN32
3812     return __dflt;
3813 #else // _WIN32
3814     string __ndflt;
3815     __narrow_to_utf8<sizeof(char_type)*__CHAR_BIT__>()(back_inserter(__ndflt),
3816                                                        __dflt.c_str(),
3817                                                        __dflt.c_str() + __dflt.size());
3818     if (__c != -1)
3819         __c <<= 1;
3820     nl_catd __cat = (nl_catd)__c;
3821     char* __n = catgets(__cat, __set, __msgid, __ndflt.c_str());
3822     string_type __w;
3823     __widen_from_utf8<sizeof(char_type)*__CHAR_BIT__>()(back_inserter(__w),
3824                                                         __n, __n + strlen(__n));
3825     return __w;
3826 #endif // _WIN32
3827 }
3828
3829 template <class _CharT>
3830 void
3831 messages<_CharT>::do_close(catalog __c) const
3832 {
3833 #if !_WIN32
3834     if (__c != -1)
3835         __c <<= 1;
3836     nl_catd __cat = (nl_catd)__c;
3837     catclose(__cat);
3838 #endif // !_WIN32
3839 }
3840
3841 _LIBCPP_EXTERN_TEMPLATE(class messages<char>)
3842 _LIBCPP_EXTERN_TEMPLATE(class messages<wchar_t>)
3843
3844 template <class _CharT>
3845 class _LIBCPP_VISIBLE messages_byname
3846     : public messages<_CharT>
3847 {
3848 public:
3849     typedef messages_base::catalog catalog;
3850     typedef basic_string<_CharT> string_type;
3851
3852     _LIBCPP_ALWAYS_INLINE
3853     explicit messages_byname(const char*, size_t __refs = 0)
3854         : messages<_CharT>(__refs) {}
3855
3856     _LIBCPP_ALWAYS_INLINE
3857     explicit messages_byname(const string&, size_t __refs = 0)
3858         : messages<_CharT>(__refs) {}
3859
3860 protected:
3861     _LIBCPP_ALWAYS_INLINE
3862     ~messages_byname() {}
3863 };
3864
3865 _LIBCPP_EXTERN_TEMPLATE(class messages_byname<char>)
3866 _LIBCPP_EXTERN_TEMPLATE(class messages_byname<wchar_t>)
3867
3868 template<class _Codecvt, class _Elem = wchar_t,
3869          class _Wide_alloc = allocator<_Elem>,
3870          class _Byte_alloc = allocator<char> >
3871 class _LIBCPP_VISIBLE wstring_convert
3872 {
3873 public:
3874     typedef basic_string<char, char_traits<char>, _Byte_alloc>   byte_string;
3875     typedef basic_string<_Elem, char_traits<_Elem>, _Wide_alloc> wide_string;
3876     typedef typename _Codecvt::state_type                        state_type;
3877     typedef typename wide_string::traits_type::int_type          int_type;
3878
3879 private:
3880     byte_string __byte_err_string_;
3881     wide_string __wide_err_string_;
3882     _Codecvt* __cvtptr_;
3883     state_type __cvtstate_;
3884     size_t __cvtcount_;
3885
3886     wstring_convert(const wstring_convert& __wc);
3887     wstring_convert& operator=(const wstring_convert& __wc);
3888 public:
3889     wstring_convert(_Codecvt* __pcvt = new _Codecvt);
3890     wstring_convert(_Codecvt* __pcvt, state_type __state);
3891     wstring_convert(const byte_string& __byte_err,
3892                     const wide_string& __wide_err = wide_string());
3893 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
3894     wstring_convert(wstring_convert&& __wc);
3895 #endif
3896     ~wstring_convert();
3897
3898     _LIBCPP_ALWAYS_INLINE
3899     wide_string from_bytes(char __byte)
3900         {return from_bytes(&__byte, &__byte+1);}
3901     _LIBCPP_ALWAYS_INLINE
3902     wide_string from_bytes(const char* __ptr)
3903         {return from_bytes(__ptr, __ptr + char_traits<char>::length(__ptr));}
3904     _LIBCPP_ALWAYS_INLINE
3905     wide_string from_bytes(const byte_string& __str)
3906         {return from_bytes(__str.data(), __str.data() + __str.size());}
3907     wide_string from_bytes(const char* __first, const char* __last);
3908
3909     _LIBCPP_ALWAYS_INLINE
3910     byte_string to_bytes(_Elem __wchar)
3911         {return to_bytes(&__wchar, &__wchar+1);}
3912     _LIBCPP_ALWAYS_INLINE
3913     byte_string to_bytes(const _Elem* __wptr)
3914         {return to_bytes(__wptr, __wptr + char_traits<_Elem>::length(__wptr));}
3915     _LIBCPP_ALWAYS_INLINE
3916     byte_string to_bytes(const wide_string& __wstr)
3917         {return to_bytes(__wstr.data(), __wstr.data() + __wstr.size());}
3918     byte_string to_bytes(const _Elem* __first, const _Elem* __last);
3919
3920     _LIBCPP_ALWAYS_INLINE
3921     size_t converted() const {return __cvtcount_;}
3922     _LIBCPP_ALWAYS_INLINE
3923     state_type state() const {return __cvtstate_;}
3924 };
3925
3926 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
3927 inline _LIBCPP_ALWAYS_INLINE
3928 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::
3929     wstring_convert(_Codecvt* __pcvt)
3930         : __cvtptr_(__pcvt), __cvtstate_(), __cvtcount_(0)
3931 {
3932 }
3933
3934 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
3935 inline _LIBCPP_ALWAYS_INLINE
3936 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::
3937     wstring_convert(_Codecvt* __pcvt, state_type __state)
3938         : __cvtptr_(__pcvt), __cvtstate_(__state), __cvtcount_(0)
3939 {
3940 }
3941
3942 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
3943 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::
3944     wstring_convert(const byte_string& __byte_err, const wide_string& __wide_err)
3945         : __byte_err_string_(__byte_err), __wide_err_string_(__wide_err),
3946           __cvtstate_(), __cvtcount_(0)
3947 {
3948     __cvtptr_ = new _Codecvt;
3949 }
3950
3951 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
3952
3953 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
3954 inline _LIBCPP_ALWAYS_INLINE
3955 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::
3956     wstring_convert(wstring_convert&& __wc)
3957         : __byte_err_string_(_VSTD::move(__wc.__byte_err_string_)),
3958           __wide_err_string_(_VSTD::move(__wc.__wide_err_string_)),
3959           __cvtptr_(__wc.__cvtptr_),
3960           __cvtstate_(__wc.__cvtstate_), __cvtcount_(__wc.__cvtstate_)
3961 {
3962     __wc.__cvtptr_ = nullptr;
3963 }
3964
3965 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
3966
3967 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
3968 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::~wstring_convert()
3969 {
3970     delete __cvtptr_;
3971 }
3972
3973 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
3974 typename wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::wide_string
3975 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::
3976     from_bytes(const char* __frm, const char* __frm_end)
3977 {
3978     __cvtcount_ = 0;
3979     if (__cvtptr_ != nullptr)
3980     {
3981         wide_string __ws(2*(__frm_end - __frm), _Elem());
3982         if (__frm != __frm_end)
3983             __ws.resize(__ws.capacity());
3984         codecvt_base::result __r = codecvt_base::ok;
3985         state_type __st = __cvtstate_;
3986         if (__frm != __frm_end)
3987         {
3988             _Elem* __to = &__ws[0];
3989             _Elem* __to_end = __to + __ws.size();
3990             const char* __frm_nxt;
3991             do
3992             {
3993                 _Elem* __to_nxt;
3994                 __r = __cvtptr_->in(__st, __frm, __frm_end, __frm_nxt,
3995                                           __to, __to_end, __to_nxt);
3996                 __cvtcount_ += __frm_nxt - __frm;
3997                 if (__frm_nxt == __frm)
3998                 {
3999                     __r = codecvt_base::error;
4000                 }
4001                 else if (__r == codecvt_base::noconv)
4002                 {
4003                     __ws.resize(__to - &__ws[0]);
4004                     // This only gets executed if _Elem is char
4005                     __ws.append((const _Elem*)__frm, (const _Elem*)__frm_end);
4006                     __frm = __frm_nxt;
4007                     __r = codecvt_base::ok;
4008                 }
4009                 else if (__r == codecvt_base::ok)
4010                 {
4011                     __ws.resize(__to_nxt - &__ws[0]);
4012                     __frm = __frm_nxt;
4013                 }
4014                 else if (__r == codecvt_base::partial)
4015                 {
4016                     ptrdiff_t __s = __to_nxt - &__ws[0];
4017                     __ws.resize(2 * __s);
4018                     __to = &__ws[0] + __s;
4019                     __to_end = &__ws[0] + __ws.size();
4020                     __frm = __frm_nxt;
4021                 }
4022             } while (__r == codecvt_base::partial && __frm_nxt < __frm_end);
4023         }
4024         if (__r == codecvt_base::ok)
4025             return __ws;
4026     }
4027 #ifndef _LIBCPP_NO_EXCEPTIONS
4028     if (__wide_err_string_.empty())
4029         throw range_error("wstring_convert: from_bytes error");
4030 #endif  // _LIBCPP_NO_EXCEPTIONS
4031     return __wide_err_string_;
4032 }
4033
4034 template<class _Codecvt, class _Elem, class _Wide_alloc, class _Byte_alloc>
4035 typename wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::byte_string
4036 wstring_convert<_Codecvt, _Elem, _Wide_alloc, _Byte_alloc>::
4037     to_bytes(const _Elem* __frm, const _Elem* __frm_end)
4038 {
4039     __cvtcount_ = 0;
4040     if (__cvtptr_ != nullptr)
4041     {
4042         byte_string __bs(2*(__frm_end - __frm), char());
4043         if (__frm != __frm_end)
4044             __bs.resize(__bs.capacity());
4045         codecvt_base::result __r = codecvt_base::ok;
4046         state_type __st = __cvtstate_;
4047         if (__frm != __frm_end)
4048         {
4049             char* __to = &__bs[0];
4050             char* __to_end = __to + __bs.size();
4051             const _Elem* __frm_nxt;
4052             do
4053             {
4054                 char* __to_nxt;
4055                 __r = __cvtptr_->out(__st, __frm, __frm_end, __frm_nxt,
4056                                            __to, __to_end, __to_nxt);
4057                 __cvtcount_ += __frm_nxt - __frm;
4058                 if (__frm_nxt == __frm)
4059                 {
4060                     __r = codecvt_base::error;
4061                 }
4062                 else if (__r == codecvt_base::noconv)
4063                 {
4064                     __bs.resize(__to - &__bs[0]);
4065                     // This only gets executed if _Elem is char
4066                     __bs.append((const char*)__frm, (const char*)__frm_end);
4067                     __frm = __frm_nxt;
4068                     __r = codecvt_base::ok;
4069                 }
4070                 else if (__r == codecvt_base::ok)
4071                 {
4072                     __bs.resize(__to_nxt - &__bs[0]);
4073                     __frm = __frm_nxt;
4074                 }
4075                 else if (__r == codecvt_base::partial)
4076                 {
4077                     ptrdiff_t __s = __to_nxt - &__bs[0];
4078                     __bs.resize(2 * __s);
4079                     __to = &__bs[0] + __s;
4080                     __to_end = &__bs[0] + __bs.size();
4081                     __frm = __frm_nxt;
4082                 }
4083             } while (__r == codecvt_base::partial && __frm_nxt < __frm_end);
4084         }
4085         if (__r == codecvt_base::ok)
4086         {
4087             size_t __s = __bs.size();
4088             __bs.resize(__bs.capacity());
4089             char* __to = &__bs[0] + __s;
4090             char* __to_end = __to + __bs.size();
4091             do
4092             {
4093                 char* __to_nxt;
4094                 __r = __cvtptr_->unshift(__st, __to, __to_end, __to_nxt);
4095                 if (__r == codecvt_base::noconv)
4096                 {
4097                     __bs.resize(__to - &__bs[0]);
4098                     __r = codecvt_base::ok;
4099                 }
4100                 else if (__r == codecvt_base::ok)
4101                 {
4102                     __bs.resize(__to_nxt - &__bs[0]);
4103                 }
4104                 else if (__r == codecvt_base::partial)
4105                 {
4106                     ptrdiff_t __sp = __to_nxt - &__bs[0];
4107                     __bs.resize(2 * __sp);
4108                     __to = &__bs[0] + __sp;
4109                     __to_end = &__bs[0] + __bs.size();
4110                 }
4111             } while (__r == codecvt_base::partial);
4112             if (__r == codecvt_base::ok)
4113                 return __bs;
4114         }
4115     }
4116 #ifndef _LIBCPP_NO_EXCEPTIONS
4117     if (__byte_err_string_.empty())
4118         throw range_error("wstring_convert: to_bytes error");
4119 #endif  // _LIBCPP_NO_EXCEPTIONS
4120     return __byte_err_string_;
4121 }
4122
4123 template <class _Codecvt, class _Elem = wchar_t, class _Tr = char_traits<_Elem> >
4124 class _LIBCPP_VISIBLE wbuffer_convert
4125     : public basic_streambuf<_Elem, _Tr>
4126 {
4127 public:
4128     // types:
4129     typedef _Elem                          char_type;
4130     typedef _Tr                            traits_type;
4131     typedef typename traits_type::int_type int_type;
4132     typedef typename traits_type::pos_type pos_type;
4133     typedef typename traits_type::off_type off_type;
4134     typedef typename _Codecvt::state_type  state_type;
4135
4136 private:
4137     char*       __extbuf_;
4138     const char* __extbufnext_;
4139     const char* __extbufend_;
4140     char __extbuf_min_[8];
4141     size_t __ebs_;
4142     char_type* __intbuf_;
4143     size_t __ibs_;
4144     streambuf* __bufptr_;
4145     _Codecvt* __cv_;
4146     state_type __st_;
4147     ios_base::openmode __cm_;
4148     bool __owns_eb_;
4149     bool __owns_ib_;
4150     bool __always_noconv_;
4151
4152     wbuffer_convert(const wbuffer_convert&);
4153     wbuffer_convert& operator=(const wbuffer_convert&);
4154 public:
4155     wbuffer_convert(streambuf* __bytebuf = 0, _Codecvt* __pcvt = new _Codecvt,
4156                     state_type __state = state_type());
4157     ~wbuffer_convert();
4158
4159     _LIBCPP_INLINE_VISIBILITY
4160     streambuf* rdbuf() const {return __bufptr_;}
4161     _LIBCPP_INLINE_VISIBILITY
4162     streambuf* rdbuf(streambuf* __bytebuf)
4163     {
4164         streambuf* __r = __bufptr_;
4165         __bufptr_ = __bytebuf;
4166         return __r;
4167     }
4168
4169     _LIBCPP_INLINE_VISIBILITY
4170     state_type state() const {return __st_;}
4171
4172 protected:
4173     virtual int_type underflow();
4174     virtual int_type pbackfail(int_type __c = traits_type::eof());
4175     virtual int_type overflow (int_type __c = traits_type::eof());
4176     virtual basic_streambuf<char_type, traits_type>* setbuf(char_type* __s,
4177                                                             streamsize __n);
4178     virtual pos_type seekoff(off_type __off, ios_base::seekdir __way,
4179                              ios_base::openmode __wch = ios_base::in | ios_base::out);
4180     virtual pos_type seekpos(pos_type __sp,
4181                              ios_base::openmode __wch = ios_base::in | ios_base::out);
4182     virtual int sync();
4183
4184 private:
4185     bool __read_mode();
4186     void __write_mode();
4187     wbuffer_convert* __close();
4188 };
4189
4190 template <class _Codecvt, class _Elem, class _Tr>
4191 wbuffer_convert<_Codecvt, _Elem, _Tr>::
4192     wbuffer_convert(streambuf* __bytebuf, _Codecvt* __pcvt, state_type __state)
4193     : __extbuf_(0),
4194       __extbufnext_(0),
4195       __extbufend_(0),
4196       __ebs_(0),
4197       __intbuf_(0),
4198       __ibs_(0),
4199       __bufptr_(__bytebuf),
4200       __cv_(__pcvt),
4201       __st_(__state),
4202       __cm_(0),
4203       __owns_eb_(false),
4204       __owns_ib_(false),
4205       __always_noconv_(__cv_ ? __cv_->always_noconv() : false)
4206 {
4207     setbuf(0, 4096);
4208 }
4209
4210 template <class _Codecvt, class _Elem, class _Tr>
4211 wbuffer_convert<_Codecvt, _Elem, _Tr>::~wbuffer_convert()
4212 {
4213     __close();
4214     delete __cv_;
4215     if (__owns_eb_)
4216         delete [] __extbuf_;
4217     if (__owns_ib_)
4218         delete [] __intbuf_;
4219 }
4220
4221 template <class _Codecvt, class _Elem, class _Tr>
4222 typename wbuffer_convert<_Codecvt, _Elem, _Tr>::int_type
4223 wbuffer_convert<_Codecvt, _Elem, _Tr>::underflow()
4224 {
4225     if (__cv_ == 0 || __bufptr_ == 0)
4226         return traits_type::eof();
4227     bool __initial = __read_mode();
4228     char_type __1buf;
4229     if (this->gptr() == 0)
4230         this->setg(&__1buf, &__1buf+1, &__1buf+1);
4231     const size_t __unget_sz = __initial ? 0 : min<size_t>((this->egptr() - this->eback()) / 2, 4);
4232     int_type __c = traits_type::eof();
4233     if (this->gptr() == this->egptr())
4234     {
4235         memmove(this->eback(), this->egptr() - __unget_sz, __unget_sz * sizeof(char_type));
4236         if (__always_noconv_)
4237         {
4238             streamsize __nmemb = static_cast<streamsize>(this->egptr() - this->eback() - __unget_sz);
4239             __nmemb = __bufptr_->sgetn((char*)this->eback() + __unget_sz, __nmemb);
4240             if (__nmemb != 0)
4241             {
4242                 this->setg(this->eback(),
4243                            this->eback() + __unget_sz,
4244                            this->eback() + __unget_sz + __nmemb);
4245                 __c = *this->gptr();
4246             }
4247         }
4248         else
4249         {
4250             memmove(__extbuf_, __extbufnext_, __extbufend_ - __extbufnext_);
4251             __extbufnext_ = __extbuf_ + (__extbufend_ - __extbufnext_);
4252             __extbufend_ = __extbuf_ + (__extbuf_ == __extbuf_min_ ? sizeof(__extbuf_min_) : __ebs_);
4253             streamsize __nmemb = _VSTD::min(static_cast<streamsize>(this->egptr() - this->eback() - __unget_sz),
4254                                  static_cast<streamsize>(__extbufend_ - __extbufnext_));
4255             codecvt_base::result __r;
4256             state_type __svs = __st_;
4257             streamsize __nr = __bufptr_->sgetn(const_cast<char*>(__extbufnext_), __nmemb);
4258             if (__nr != 0)
4259             {
4260                 __extbufend_ = __extbufnext_ + __nr;
4261                 char_type*  __inext;
4262                 __r = __cv_->in(__st_, __extbuf_, __extbufend_, __extbufnext_,
4263                                        this->eback() + __unget_sz,
4264                                        this->egptr(), __inext);
4265                 if (__r == codecvt_base::noconv)
4266                 {
4267                     this->setg((char_type*)__extbuf_, (char_type*)__extbuf_, (char_type*)__extbufend_);
4268                     __c = *this->gptr();
4269                 }
4270                 else if (__inext != this->eback() + __unget_sz)
4271                 {
4272                     this->setg(this->eback(), this->eback() + __unget_sz, __inext);
4273                     __c = *this->gptr();
4274                 }
4275             }
4276         }
4277     }
4278     else
4279         __c = *this->gptr();
4280     if (this->eback() == &__1buf)
4281         this->setg(0, 0, 0);
4282     return __c;
4283 }
4284
4285 template <class _Codecvt, class _Elem, class _Tr>
4286 typename wbuffer_convert<_Codecvt, _Elem, _Tr>::int_type
4287 wbuffer_convert<_Codecvt, _Elem, _Tr>::pbackfail(int_type __c)
4288 {
4289     if (__cv_ != 0 && __bufptr_ != 0 && this->eback() < this->gptr())
4290     {
4291         if (traits_type::eq_int_type(__c, traits_type::eof()))
4292         {
4293             this->gbump(-1);
4294             return traits_type::not_eof(__c);
4295         }
4296         if (traits_type::eq(traits_type::to_char_type(__c), this->gptr()[-1]))
4297         {
4298             this->gbump(-1);
4299             *this->gptr() = traits_type::to_char_type(__c);
4300             return __c;
4301         }
4302     }
4303     return traits_type::eof();
4304 }
4305
4306 template <class _Codecvt, class _Elem, class _Tr>
4307 typename wbuffer_convert<_Codecvt, _Elem, _Tr>::int_type
4308 wbuffer_convert<_Codecvt, _Elem, _Tr>::overflow(int_type __c)
4309 {
4310     if (__cv_ == 0 || __bufptr_ == 0)
4311         return traits_type::eof();
4312     __write_mode();
4313     char_type __1buf;
4314     char_type* __pb_save = this->pbase();
4315     char_type* __epb_save = this->epptr();
4316     if (!traits_type::eq_int_type(__c, traits_type::eof()))
4317     {
4318         if (this->pptr() == 0)
4319             this->setp(&__1buf, &__1buf+1);
4320         *this->pptr() = traits_type::to_char_type(__c);
4321         this->pbump(1);
4322     }
4323     if (this->pptr() != this->pbase())
4324     {
4325         if (__always_noconv_)
4326         {
4327             streamsize __nmemb = static_cast<streamsize>(this->pptr() - this->pbase());
4328             if (__bufptr_->sputn((const char*)this->pbase(), __nmemb) != __nmemb)
4329                 return traits_type::eof();
4330         }
4331         else
4332         {
4333             char* __extbe = __extbuf_;
4334             codecvt_base::result __r;
4335             do
4336             {
4337                 const char_type* __e;
4338                 __r = __cv_->out(__st_, this->pbase(), this->pptr(), __e,
4339                                         __extbuf_, __extbuf_ + __ebs_, __extbe);
4340                 if (__e == this->pbase())
4341                     return traits_type::eof();
4342                 if (__r == codecvt_base::noconv)
4343                 {
4344                     streamsize __nmemb = static_cast<size_t>(this->pptr() - this->pbase());
4345                     if (__bufptr_->sputn((const char*)this->pbase(), __nmemb) != __nmemb)
4346                         return traits_type::eof();
4347                 }
4348                 else if (__r == codecvt_base::ok || __r == codecvt_base::partial)
4349                 {
4350                     streamsize __nmemb = static_cast<size_t>(__extbe - __extbuf_);
4351                     if (__bufptr_->sputn(__extbuf_, __nmemb) != __nmemb)
4352                         return traits_type::eof();
4353                     if (__r == codecvt_base::partial)
4354                     {
4355                         this->setp((char_type*)__e, this->pptr());
4356                         this->pbump(this->epptr() - this->pbase());
4357                     }
4358                 }
4359                 else
4360                     return traits_type::eof();
4361             } while (__r == codecvt_base::partial);
4362         }
4363         this->setp(__pb_save, __epb_save);
4364     }
4365     return traits_type::not_eof(__c);
4366 }
4367
4368 template <class _Codecvt, class _Elem, class _Tr>
4369 basic_streambuf<_Elem, _Tr>*
4370 wbuffer_convert<_Codecvt, _Elem, _Tr>::setbuf(char_type* __s, streamsize __n)
4371 {
4372     this->setg(0, 0, 0);
4373     this->setp(0, 0);
4374     if (__owns_eb_)
4375         delete [] __extbuf_;
4376     if (__owns_ib_)
4377         delete [] __intbuf_;
4378     __ebs_ = __n;
4379     if (__ebs_ > sizeof(__extbuf_min_))
4380     {
4381         if (__always_noconv_ && __s)
4382         {
4383             __extbuf_ = (char*)__s;
4384             __owns_eb_ = false;
4385         }
4386         else
4387         {
4388             __extbuf_ = new char[__ebs_];
4389             __owns_eb_ = true;
4390         }
4391     }
4392     else
4393     {
4394         __extbuf_ = __extbuf_min_;
4395         __ebs_ = sizeof(__extbuf_min_);
4396         __owns_eb_ = false;
4397     }
4398     if (!__always_noconv_)
4399     {
4400         __ibs_ = max<streamsize>(__n, sizeof(__extbuf_min_));
4401         if (__s && __ibs_ >= sizeof(__extbuf_min_))
4402         {
4403             __intbuf_ = __s;
4404             __owns_ib_ = false;
4405         }
4406         else
4407         {
4408             __intbuf_ = new char_type[__ibs_];
4409             __owns_ib_ = true;
4410         }
4411     }
4412     else
4413     {
4414         __ibs_ = 0;
4415         __intbuf_ = 0;
4416         __owns_ib_ = false;
4417     }
4418     return this;
4419 }
4420
4421 template <class _Codecvt, class _Elem, class _Tr>
4422 typename wbuffer_convert<_Codecvt, _Elem, _Tr>::pos_type
4423 wbuffer_convert<_Codecvt, _Elem, _Tr>::seekoff(off_type __off, ios_base::seekdir __way,
4424                                         ios_base::openmode __om)
4425 {
4426     int __width = __cv_->encoding();
4427     if (__cv_ == 0 || __bufptr_ == 0 || (__width <= 0 && __off != 0) || sync())
4428         return pos_type(off_type(-1));
4429     // __width > 0 || __off == 0
4430     switch (__way)
4431     {
4432     case ios_base::beg:
4433         break;
4434     case ios_base::cur:
4435         break;
4436     case ios_base::end:
4437         break;
4438     default:
4439         return pos_type(off_type(-1));
4440     }
4441     pos_type __r = __bufptr_->pubseekoff(__width * __off, __way, __om);
4442     __r.state(__st_);
4443     return __r;
4444 }
4445
4446 template <class _Codecvt, class _Elem, class _Tr>
4447 typename wbuffer_convert<_Codecvt, _Elem, _Tr>::pos_type
4448 wbuffer_convert<_Codecvt, _Elem, _Tr>::seekpos(pos_type __sp, ios_base::openmode __wch)
4449 {
4450     if (__cv_ == 0 || __bufptr_ == 0 || sync())
4451         return pos_type(off_type(-1));
4452     if (__bufptr_->pubseekpos(__sp, __wch) == pos_type(off_type(-1)))
4453         return pos_type(off_type(-1));
4454     return __sp;
4455 }
4456
4457 template <class _Codecvt, class _Elem, class _Tr>
4458 int
4459 wbuffer_convert<_Codecvt, _Elem, _Tr>::sync()
4460 {
4461     if (__cv_ == 0 || __bufptr_ == 0)
4462         return 0;
4463     if (__cm_ & ios_base::out)
4464     {
4465         if (this->pptr() != this->pbase())
4466             if (overflow() == traits_type::eof())
4467                 return -1;
4468         codecvt_base::result __r;
4469         do
4470         {
4471             char* __extbe;
4472             __r = __cv_->unshift(__st_, __extbuf_, __extbuf_ + __ebs_, __extbe);
4473             streamsize __nmemb = static_cast<streamsize>(__extbe - __extbuf_);
4474             if (__bufptr_->sputn(__extbuf_, __nmemb) != __nmemb)
4475                 return -1;
4476         } while (__r == codecvt_base::partial);
4477         if (__r == codecvt_base::error)
4478             return -1;
4479         if (__bufptr_->pubsync())
4480             return -1;
4481     }
4482     else if (__cm_ & ios_base::in)
4483     {
4484         off_type __c;
4485         if (__always_noconv_)
4486             __c = this->egptr() - this->gptr();
4487         else
4488         {
4489             int __width = __cv_->encoding();
4490             __c = __extbufend_ - __extbufnext_;
4491             if (__width > 0)
4492                 __c += __width * (this->egptr() - this->gptr());
4493             else
4494             {
4495                 if (this->gptr() != this->egptr())
4496                 {
4497                     reverse(this->gptr(), this->egptr());
4498                     codecvt_base::result __r;
4499                     const char_type* __e = this->gptr();
4500                     char* __extbe;
4501                     do
4502                     {
4503                         __r = __cv_->out(__st_, __e, this->egptr(), __e,
4504                                          __extbuf_, __extbuf_ + __ebs_, __extbe);
4505                         switch (__r)
4506                         {
4507                         case codecvt_base::noconv:
4508                             __c += this->egptr() - this->gptr();
4509                             break;
4510                         case codecvt_base::ok:
4511                         case codecvt_base::partial:
4512                             __c += __extbe - __extbuf_;
4513                             break;
4514                         default:
4515                             return -1;
4516                         }
4517                     } while (__r == codecvt_base::partial);
4518                 }
4519             }
4520         }
4521         if (__bufptr_->pubseekoff(-__c, ios_base::cur, __cm_) == pos_type(off_type(-1)))
4522             return -1;
4523         this->setg(0, 0, 0);
4524         __cm_ = 0;
4525     }
4526     return 0;
4527 }
4528
4529 template <class _Codecvt, class _Elem, class _Tr>
4530 bool
4531 wbuffer_convert<_Codecvt, _Elem, _Tr>::__read_mode()
4532 {
4533     if (!(__cm_ & ios_base::in))
4534     {
4535         this->setp(0, 0);
4536         if (__always_noconv_)
4537             this->setg((char_type*)__extbuf_,
4538                        (char_type*)__extbuf_ + __ebs_,
4539                        (char_type*)__extbuf_ + __ebs_);
4540         else
4541             this->setg(__intbuf_, __intbuf_ + __ibs_, __intbuf_ + __ibs_);
4542         __cm_ = ios_base::in;
4543         return true;
4544     }
4545     return false;
4546 }
4547
4548 template <class _Codecvt, class _Elem, class _Tr>
4549 void
4550 wbuffer_convert<_Codecvt, _Elem, _Tr>::__write_mode()
4551 {
4552     if (!(__cm_ & ios_base::out))
4553     {
4554         this->setg(0, 0, 0);
4555         if (__ebs_ > sizeof(__extbuf_min_))
4556         {
4557             if (__always_noconv_)
4558                 this->setp((char_type*)__extbuf_,
4559                            (char_type*)__extbuf_ + (__ebs_ - 1));
4560             else
4561                 this->setp(__intbuf_, __intbuf_ + (__ibs_ - 1));
4562         }
4563         else
4564             this->setp(0, 0);
4565         __cm_ = ios_base::out;
4566     }
4567 }
4568
4569 template <class _Codecvt, class _Elem, class _Tr>
4570 wbuffer_convert<_Codecvt, _Elem, _Tr>*
4571 wbuffer_convert<_Codecvt, _Elem, _Tr>::__close()
4572 {
4573     wbuffer_convert* __rt = 0;
4574     if (__cv_ != 0 && __bufptr_ != 0)
4575     {
4576         __rt = this;
4577         if ((__cm_ & ios_base::out) && sync())
4578             __rt = 0;
4579     }
4580     return __rt;
4581 }
4582
4583 _LIBCPP_END_NAMESPACE_STD
4584
4585 #endif  // _LIBCPP_LOCALE