* MAINTAINERS: Move myself to reviewers (Fortran).
[official-gcc.git] / libstdc++-v3 / include / bits / locale_facets_nonio.tcc
blobc9f8dac1d1d6a14ad71ae37ce1323e52a42f19c0
1 // Locale support -*- C++ -*-
3 // Copyright (C) 2007-2014 Free Software Foundation, Inc.
4 //
5 // This file is part of the GNU ISO C++ Library.  This library is free
6 // software; you can redistribute it and/or modify it under the
7 // terms of the GNU General Public License as published by the
8 // Free Software Foundation; either version 3, or (at your option)
9 // any later version.
11 // This library is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 // GNU General Public License for more details.
16 // Under Section 7 of GPL version 3, you are granted additional
17 // permissions described in the GCC Runtime Library Exception, version
18 // 3.1, as published by the Free Software Foundation.
20 // You should have received a copy of the GNU General Public License and
21 // a copy of the GCC Runtime Library Exception along with this program;
22 // see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
23 // <http://www.gnu.org/licenses/>.
25 /** @file bits/locale_facets_nonio.tcc
26  *  This is an internal header file, included by other library headers.
27  *  Do not attempt to use it directly. @headername{locale}
28  */
30 #ifndef _LOCALE_FACETS_NONIO_TCC
31 #define _LOCALE_FACETS_NONIO_TCC 1
33 #pragma GCC system_header
35 namespace std _GLIBCXX_VISIBILITY(default)
37 _GLIBCXX_BEGIN_NAMESPACE_VERSION
39   template<typename _CharT, bool _Intl>
40     struct __use_cache<__moneypunct_cache<_CharT, _Intl> >
41     {
42       const __moneypunct_cache<_CharT, _Intl>*
43       operator() (const locale& __loc) const
44       {
45         const size_t __i = moneypunct<_CharT, _Intl>::id._M_id();
46         const locale::facet** __caches = __loc._M_impl->_M_caches;
47         if (!__caches[__i])
48           {
49             __moneypunct_cache<_CharT, _Intl>* __tmp = 0;
50             __try
51               {
52                 __tmp = new __moneypunct_cache<_CharT, _Intl>;
53                 __tmp->_M_cache(__loc);
54               }
55             __catch(...)
56               {
57                 delete __tmp;
58                 __throw_exception_again;
59               }
60             __loc._M_impl->_M_install_cache(__tmp, __i);
61           }
62         return static_cast<
63           const __moneypunct_cache<_CharT, _Intl>*>(__caches[__i]);
64       }
65     };
67   template<typename _CharT, bool _Intl>
68     void
69     __moneypunct_cache<_CharT, _Intl>::_M_cache(const locale& __loc)
70     {
71       _M_allocated = true;
73       const moneypunct<_CharT, _Intl>& __mp =
74         use_facet<moneypunct<_CharT, _Intl> >(__loc);
76       _M_decimal_point = __mp.decimal_point();
77       _M_thousands_sep = __mp.thousands_sep();
78       _M_frac_digits = __mp.frac_digits();
80       char* __grouping = 0;
81       _CharT* __curr_symbol = 0;
82       _CharT* __positive_sign = 0;
83       _CharT* __negative_sign = 0;     
84       __try
85         {
86           _M_grouping_size = __mp.grouping().size();
87           __grouping = new char[_M_grouping_size];
88           __mp.grouping().copy(__grouping, _M_grouping_size);
89           _M_grouping = __grouping;
90           _M_use_grouping = (_M_grouping_size
91                              && static_cast<signed char>(_M_grouping[0]) > 0
92                              && (_M_grouping[0]
93                                  != __gnu_cxx::__numeric_traits<char>::__max));
95           _M_curr_symbol_size = __mp.curr_symbol().size();
96           __curr_symbol = new _CharT[_M_curr_symbol_size];
97           __mp.curr_symbol().copy(__curr_symbol, _M_curr_symbol_size);
98           _M_curr_symbol = __curr_symbol;
100           _M_positive_sign_size = __mp.positive_sign().size();
101           __positive_sign = new _CharT[_M_positive_sign_size];
102           __mp.positive_sign().copy(__positive_sign, _M_positive_sign_size);
103           _M_positive_sign = __positive_sign;
105           _M_negative_sign_size = __mp.negative_sign().size();
106           __negative_sign = new _CharT[_M_negative_sign_size];
107           __mp.negative_sign().copy(__negative_sign, _M_negative_sign_size);
108           _M_negative_sign = __negative_sign;
110           _M_pos_format = __mp.pos_format();
111           _M_neg_format = __mp.neg_format();
113           const ctype<_CharT>& __ct = use_facet<ctype<_CharT> >(__loc);
114           __ct.widen(money_base::_S_atoms,
115                      money_base::_S_atoms + money_base::_S_end, _M_atoms);
116         }
117       __catch(...)
118         {
119           delete [] __grouping;
120           delete [] __curr_symbol;
121           delete [] __positive_sign;
122           delete [] __negative_sign;
123           __throw_exception_again;
124         }
125     }
127 _GLIBCXX_BEGIN_NAMESPACE_LDBL
129   template<typename _CharT, typename _InIter>
130     template<bool _Intl>
131       _InIter
132       money_get<_CharT, _InIter>::
133       _M_extract(iter_type __beg, iter_type __end, ios_base& __io,
134                  ios_base::iostate& __err, string& __units) const
135       {
136         typedef char_traits<_CharT>                       __traits_type;
137         typedef typename string_type::size_type           size_type;    
138         typedef money_base::part                          part;
139         typedef __moneypunct_cache<_CharT, _Intl>         __cache_type;
140         
141         const locale& __loc = __io._M_getloc();
142         const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
144         __use_cache<__cache_type> __uc;
145         const __cache_type* __lc = __uc(__loc);
146         const char_type* __lit = __lc->_M_atoms;
148         // Deduced sign.
149         bool __negative = false;
150         // Sign size.
151         size_type __sign_size = 0;
152         // True if sign is mandatory.
153         const bool __mandatory_sign = (__lc->_M_positive_sign_size
154                                        && __lc->_M_negative_sign_size);
155         // String of grouping info from thousands_sep plucked from __units.
156         string __grouping_tmp;
157         if (__lc->_M_use_grouping)
158           __grouping_tmp.reserve(32);
159         // Last position before the decimal point.
160         int __last_pos = 0;
161         // Separator positions, then, possibly, fractional digits.
162         int __n = 0;
163         // If input iterator is in a valid state.
164         bool __testvalid = true;
165         // Flag marking when a decimal point is found.
166         bool __testdecfound = false;
168         // The tentative returned string is stored here.
169         string __res;
170         __res.reserve(32);
172         const char_type* __lit_zero = __lit + money_base::_S_zero;
173         const money_base::pattern __p = __lc->_M_neg_format;
174         for (int __i = 0; __i < 4 && __testvalid; ++__i)
175           {
176             const part __which = static_cast<part>(__p.field[__i]);
177             switch (__which)
178               {
179               case money_base::symbol:
180                 // According to 22.2.6.1.2, p2, symbol is required
181                 // if (__io.flags() & ios_base::showbase), otherwise
182                 // is optional and consumed only if other characters
183                 // are needed to complete the format.
184                 if (__io.flags() & ios_base::showbase || __sign_size > 1
185                     || __i == 0
186                     || (__i == 1 && (__mandatory_sign
187                                      || (static_cast<part>(__p.field[0])
188                                          == money_base::sign)
189                                      || (static_cast<part>(__p.field[2])
190                                          == money_base::space)))
191                     || (__i == 2 && ((static_cast<part>(__p.field[3])
192                                       == money_base::value)
193                                      || (__mandatory_sign
194                                          && (static_cast<part>(__p.field[3])
195                                              == money_base::sign)))))
196                   {
197                     const size_type __len = __lc->_M_curr_symbol_size;
198                     size_type __j = 0;
199                     for (; __beg != __end && __j < __len
200                            && *__beg == __lc->_M_curr_symbol[__j];
201                          ++__beg, ++__j);
202                     if (__j != __len
203                         && (__j || __io.flags() & ios_base::showbase))
204                       __testvalid = false;
205                   }
206                 break;
207               case money_base::sign:
208                 // Sign might not exist, or be more than one character long.
209                 if (__lc->_M_positive_sign_size && __beg != __end
210                     && *__beg == __lc->_M_positive_sign[0])
211                   {
212                     __sign_size = __lc->_M_positive_sign_size;
213                     ++__beg;
214                   }
215                 else if (__lc->_M_negative_sign_size && __beg != __end
216                          && *__beg == __lc->_M_negative_sign[0])
217                   {
218                     __negative = true;
219                     __sign_size = __lc->_M_negative_sign_size;
220                     ++__beg;
221                   }
222                 else if (__lc->_M_positive_sign_size
223                          && !__lc->_M_negative_sign_size)
224                   // "... if no sign is detected, the result is given the sign
225                   // that corresponds to the source of the empty string"
226                   __negative = true;
227                 else if (__mandatory_sign)
228                   __testvalid = false;
229                 break;
230               case money_base::value:
231                 // Extract digits, remove and stash away the
232                 // grouping of found thousands separators.
233                 for (; __beg != __end; ++__beg)
234                   {
235                     const char_type __c = *__beg;
236                     const char_type* __q = __traits_type::find(__lit_zero, 
237                                                                10, __c);
238                     if (__q != 0)
239                       {
240                         __res += money_base::_S_atoms[__q - __lit];
241                         ++__n;
242                       }
243                     else if (__c == __lc->_M_decimal_point 
244                              && !__testdecfound)
245                       {
246                         if (__lc->_M_frac_digits <= 0)
247                           break;
249                         __last_pos = __n;
250                         __n = 0;
251                         __testdecfound = true;
252                       }
253                     else if (__lc->_M_use_grouping
254                              && __c == __lc->_M_thousands_sep
255                              && !__testdecfound)
256                       {
257                         if (__n)
258                           {
259                             // Mark position for later analysis.
260                             __grouping_tmp += static_cast<char>(__n);
261                             __n = 0;
262                           }
263                         else
264                           {
265                             __testvalid = false;
266                             break;
267                           }
268                       }
269                     else
270                       break;
271                   }
272                 if (__res.empty())
273                   __testvalid = false;
274                 break;
275               case money_base::space:
276                 // At least one space is required.
277                 if (__beg != __end && __ctype.is(ctype_base::space, *__beg))
278                   ++__beg;
279                 else
280                   __testvalid = false;
281               case money_base::none:
282                 // Only if not at the end of the pattern.
283                 if (__i != 3)
284                   for (; __beg != __end
285                          && __ctype.is(ctype_base::space, *__beg); ++__beg);
286                 break;
287               }
288           }
290         // Need to get the rest of the sign characters, if they exist.
291         if (__sign_size > 1 && __testvalid)
292           {
293             const char_type* __sign = __negative ? __lc->_M_negative_sign
294                                                  : __lc->_M_positive_sign;
295             size_type __i = 1;
296             for (; __beg != __end && __i < __sign_size
297                    && *__beg == __sign[__i]; ++__beg, ++__i);
298             
299             if (__i != __sign_size)
300               __testvalid = false;
301           }
303         if (__testvalid)
304           {
305             // Strip leading zeros.
306             if (__res.size() > 1)
307               {
308                 const size_type __first = __res.find_first_not_of('0');
309                 const bool __only_zeros = __first == string::npos;
310                 if (__first)
311                   __res.erase(0, __only_zeros ? __res.size() - 1 : __first);
312               }
314             // 22.2.6.1.2, p4
315             if (__negative && __res[0] != '0')
316               __res.insert(__res.begin(), '-');
317             
318             // Test for grouping fidelity.
319             if (__grouping_tmp.size())
320               {
321                 // Add the ending grouping.
322                 __grouping_tmp += static_cast<char>(__testdecfound ? __last_pos
323                                                                    : __n);
324                 if (!std::__verify_grouping(__lc->_M_grouping,
325                                             __lc->_M_grouping_size,
326                                             __grouping_tmp))
327                   __err |= ios_base::failbit;
328               }
329             
330             // Iff not enough digits were supplied after the decimal-point.
331             if (__testdecfound && __n != __lc->_M_frac_digits)
332               __testvalid = false;
333           }
335         // Iff valid sequence is not recognized.
336         if (!__testvalid)
337           __err |= ios_base::failbit;
338         else
339           __units.swap(__res);
340         
341         // Iff no more characters are available.
342         if (__beg == __end)
343           __err |= ios_base::eofbit;
344         return __beg;
345       }
347 #if defined _GLIBCXX_LONG_DOUBLE_COMPAT && defined __LONG_DOUBLE_128__
348   template<typename _CharT, typename _InIter>
349     _InIter
350     money_get<_CharT, _InIter>::
351     __do_get(iter_type __beg, iter_type __end, bool __intl, ios_base& __io,
352              ios_base::iostate& __err, double& __units) const
353     {
354       string __str;
355       __beg = __intl ? _M_extract<true>(__beg, __end, __io, __err, __str)
356                      : _M_extract<false>(__beg, __end, __io, __err, __str);
357       std::__convert_to_v(__str.c_str(), __units, __err, _S_get_c_locale());
358       return __beg;
359     }
360 #endif
362   template<typename _CharT, typename _InIter>
363     _InIter
364     money_get<_CharT, _InIter>::
365     do_get(iter_type __beg, iter_type __end, bool __intl, ios_base& __io,
366            ios_base::iostate& __err, long double& __units) const
367     {
368       string __str;
369       __beg = __intl ? _M_extract<true>(__beg, __end, __io, __err, __str)
370                      : _M_extract<false>(__beg, __end, __io, __err, __str);
371       std::__convert_to_v(__str.c_str(), __units, __err, _S_get_c_locale());
372       return __beg;
373     }
375   template<typename _CharT, typename _InIter>
376     _InIter
377     money_get<_CharT, _InIter>::
378     do_get(iter_type __beg, iter_type __end, bool __intl, ios_base& __io,
379            ios_base::iostate& __err, string_type& __digits) const
380     {
381       typedef typename string::size_type                  size_type;
383       const locale& __loc = __io._M_getloc();
384       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
386       string __str;
387       __beg = __intl ? _M_extract<true>(__beg, __end, __io, __err, __str)
388                      : _M_extract<false>(__beg, __end, __io, __err, __str);
389       const size_type __len = __str.size();
390       if (__len)
391         {
392           __digits.resize(__len);
393           __ctype.widen(__str.data(), __str.data() + __len, &__digits[0]);
394         }
395       return __beg;
396     }
398   template<typename _CharT, typename _OutIter>
399     template<bool _Intl>
400       _OutIter
401       money_put<_CharT, _OutIter>::
402       _M_insert(iter_type __s, ios_base& __io, char_type __fill,
403                 const string_type& __digits) const
404       {
405         typedef typename string_type::size_type           size_type;
406         typedef money_base::part                          part;
407         typedef __moneypunct_cache<_CharT, _Intl>         __cache_type;
408       
409         const locale& __loc = __io._M_getloc();
410         const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
412         __use_cache<__cache_type> __uc;
413         const __cache_type* __lc = __uc(__loc);
414         const char_type* __lit = __lc->_M_atoms;
416         // Determine if negative or positive formats are to be used, and
417         // discard leading negative_sign if it is present.
418         const char_type* __beg = __digits.data();
420         money_base::pattern __p;
421         const char_type* __sign;
422         size_type __sign_size;
423         if (!(*__beg == __lit[money_base::_S_minus]))
424           {
425             __p = __lc->_M_pos_format;
426             __sign = __lc->_M_positive_sign;
427             __sign_size = __lc->_M_positive_sign_size;
428           }
429         else
430           {
431             __p = __lc->_M_neg_format;
432             __sign = __lc->_M_negative_sign;
433             __sign_size = __lc->_M_negative_sign_size;
434             if (__digits.size())
435               ++__beg;
436           }
437        
438         // Look for valid numbers in the ctype facet within input digits.
439         size_type __len = __ctype.scan_not(ctype_base::digit, __beg,
440                                            __beg + __digits.size()) - __beg;
441         if (__len)
442           {
443             // Assume valid input, and attempt to format.
444             // Break down input numbers into base components, as follows:
445             //   final_value = grouped units + (decimal point) + (digits)
446             string_type __value;
447             __value.reserve(2 * __len);
449             // Add thousands separators to non-decimal digits, per
450             // grouping rules.
451             long __paddec = __len - __lc->_M_frac_digits;
452             if (__paddec > 0)
453               {
454                 if (__lc->_M_frac_digits < 0)
455                   __paddec = __len;
456                 if (__lc->_M_grouping_size)
457                   {
458                     __value.assign(2 * __paddec, char_type());
459                     _CharT* __vend = 
460                       std::__add_grouping(&__value[0], __lc->_M_thousands_sep,
461                                           __lc->_M_grouping,
462                                           __lc->_M_grouping_size,
463                                           __beg, __beg + __paddec);
464                     __value.erase(__vend - &__value[0]);
465                   }
466                 else
467                   __value.assign(__beg, __paddec);
468               }
470             // Deal with decimal point, decimal digits.
471             if (__lc->_M_frac_digits > 0)
472               {
473                 __value += __lc->_M_decimal_point;
474                 if (__paddec >= 0)
475                   __value.append(__beg + __paddec, __lc->_M_frac_digits);
476                 else
477                   {
478                     // Have to pad zeros in the decimal position.
479                     __value.append(-__paddec, __lit[money_base::_S_zero]);
480                     __value.append(__beg, __len);
481                   }
482               }
483   
484             // Calculate length of resulting string.
485             const ios_base::fmtflags __f = __io.flags() 
486                                            & ios_base::adjustfield;
487             __len = __value.size() + __sign_size;
488             __len += ((__io.flags() & ios_base::showbase)
489                       ? __lc->_M_curr_symbol_size : 0);
491             string_type __res;
492             __res.reserve(2 * __len);
493             
494             const size_type __width = static_cast<size_type>(__io.width());  
495             const bool __testipad = (__f == ios_base::internal
496                                      && __len < __width);
497             // Fit formatted digits into the required pattern.
498             for (int __i = 0; __i < 4; ++__i)
499               {
500                 const part __which = static_cast<part>(__p.field[__i]);
501                 switch (__which)
502                   {
503                   case money_base::symbol:
504                     if (__io.flags() & ios_base::showbase)
505                       __res.append(__lc->_M_curr_symbol,
506                                    __lc->_M_curr_symbol_size);
507                     break;
508                   case money_base::sign:
509                     // Sign might not exist, or be more than one
510                     // character long. In that case, add in the rest
511                     // below.
512                     if (__sign_size)
513                       __res += __sign[0];
514                     break;
515                   case money_base::value:
516                     __res += __value;
517                     break;
518                   case money_base::space:
519                     // At least one space is required, but if internal
520                     // formatting is required, an arbitrary number of
521                     // fill spaces will be necessary.
522                     if (__testipad)
523                       __res.append(__width - __len, __fill);
524                     else
525                       __res += __fill;
526                     break;
527                   case money_base::none:
528                     if (__testipad)
529                       __res.append(__width - __len, __fill);
530                     break;
531                   }
532               }
533             
534             // Special case of multi-part sign parts.
535             if (__sign_size > 1)
536               __res.append(__sign + 1, __sign_size - 1);
537             
538             // Pad, if still necessary.
539             __len = __res.size();
540             if (__width > __len)
541               {
542                 if (__f == ios_base::left)
543                   // After.
544                   __res.append(__width - __len, __fill);
545                 else
546                   // Before.
547                   __res.insert(0, __width - __len, __fill);
548                 __len = __width;
549               }
550             
551             // Write resulting, fully-formatted string to output iterator.
552             __s = std::__write(__s, __res.data(), __len);
553           }
554         __io.width(0);
555         return __s;    
556       }
558 #if defined _GLIBCXX_LONG_DOUBLE_COMPAT && defined __LONG_DOUBLE_128__
559   template<typename _CharT, typename _OutIter>
560     _OutIter
561     money_put<_CharT, _OutIter>::
562     __do_put(iter_type __s, bool __intl, ios_base& __io, char_type __fill,
563              double __units) const
564     { return this->do_put(__s, __intl, __io, __fill, (long double) __units); }
565 #endif
567   template<typename _CharT, typename _OutIter>
568     _OutIter
569     money_put<_CharT, _OutIter>::
570     do_put(iter_type __s, bool __intl, ios_base& __io, char_type __fill,
571            long double __units) const
572     {
573       const locale __loc = __io.getloc();
574       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
575 #ifdef _GLIBCXX_USE_C99
576       // First try a buffer perhaps big enough.
577       int __cs_size = 64;
578       char* __cs = static_cast<char*>(__builtin_alloca(__cs_size));
579       // _GLIBCXX_RESOLVE_LIB_DEFECTS
580       // 328. Bad sprintf format modifier in money_put<>::do_put()
581       int __len = std::__convert_from_v(_S_get_c_locale(), __cs, __cs_size,
582                                         "%.*Lf", 0, __units);
583       // If the buffer was not large enough, try again with the correct size.
584       if (__len >= __cs_size)
585         {
586           __cs_size = __len + 1;
587           __cs = static_cast<char*>(__builtin_alloca(__cs_size));
588           __len = std::__convert_from_v(_S_get_c_locale(), __cs, __cs_size,
589                                         "%.*Lf", 0, __units);
590         }
591 #else
592       // max_exponent10 + 1 for the integer part, + 2 for sign and '\0'.
593       const int __cs_size =
594         __gnu_cxx::__numeric_traits<long double>::__max_exponent10 + 3;
595       char* __cs = static_cast<char*>(__builtin_alloca(__cs_size));
596       int __len = std::__convert_from_v(_S_get_c_locale(), __cs, 0, "%.*Lf", 
597                                         0, __units);
598 #endif
599       string_type __digits(__len, char_type());
600       __ctype.widen(__cs, __cs + __len, &__digits[0]);
601       return __intl ? _M_insert<true>(__s, __io, __fill, __digits)
602                     : _M_insert<false>(__s, __io, __fill, __digits);
603     }
605   template<typename _CharT, typename _OutIter>
606     _OutIter
607     money_put<_CharT, _OutIter>::
608     do_put(iter_type __s, bool __intl, ios_base& __io, char_type __fill,
609            const string_type& __digits) const
610     { return __intl ? _M_insert<true>(__s, __io, __fill, __digits)
611                     : _M_insert<false>(__s, __io, __fill, __digits); }
613 _GLIBCXX_END_NAMESPACE_LDBL
615   // NB: Not especially useful. Without an ios_base object or some
616   // kind of locale reference, we are left clawing at the air where
617   // the side of the mountain used to be...
618   template<typename _CharT, typename _InIter>
619     time_base::dateorder
620     time_get<_CharT, _InIter>::do_date_order() const
621     { return time_base::no_order; }
623   // Expand a strftime format string and parse it.  E.g., do_get_date() may
624   // pass %m/%d/%Y => extracted characters.
625   template<typename _CharT, typename _InIter>
626     _InIter
627     time_get<_CharT, _InIter>::
628     _M_extract_via_format(iter_type __beg, iter_type __end, ios_base& __io,
629                           ios_base::iostate& __err, tm* __tm,
630                           const _CharT* __format) const
631     {
632       const locale& __loc = __io._M_getloc();
633       const __timepunct<_CharT>& __tp = use_facet<__timepunct<_CharT> >(__loc);
634       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
635       const size_t __len = char_traits<_CharT>::length(__format);
637       ios_base::iostate __tmperr = ios_base::goodbit;
638       size_t __i = 0;
639       for (; __beg != __end && __i < __len && !__tmperr; ++__i)
640         {
641           if (__ctype.narrow(__format[__i], 0) == '%')
642             {
643               // Verify valid formatting code, attempt to extract.
644               char __c = __ctype.narrow(__format[++__i], 0);
645               int __mem = 0;
646               if (__c == 'E' || __c == 'O')
647                 __c = __ctype.narrow(__format[++__i], 0);
648               switch (__c)
649                 {
650                   const char* __cs;
651                   _CharT __wcs[10];
652                 case 'a':
653                   // Abbreviated weekday name [tm_wday]
654                   const char_type*  __days1[7];
655                   __tp._M_days_abbreviated(__days1);
656                   __beg = _M_extract_name(__beg, __end, __tm->tm_wday, __days1,
657                                           7, __io, __tmperr);
658                   break;
659                 case 'A':
660                   // Weekday name [tm_wday].
661                   const char_type*  __days2[7];
662                   __tp._M_days(__days2);
663                   __beg = _M_extract_name(__beg, __end, __tm->tm_wday, __days2,
664                                           7, __io, __tmperr);
665                   break;
666                 case 'h':
667                 case 'b':
668                   // Abbreviated month name [tm_mon]
669                   const char_type*  __months1[12];
670                   __tp._M_months_abbreviated(__months1);
671                   __beg = _M_extract_name(__beg, __end, __tm->tm_mon, 
672                                           __months1, 12, __io, __tmperr);
673                   break;
674                 case 'B':
675                   // Month name [tm_mon].
676                   const char_type*  __months2[12];
677                   __tp._M_months(__months2);
678                   __beg = _M_extract_name(__beg, __end, __tm->tm_mon, 
679                                           __months2, 12, __io, __tmperr);
680                   break;
681                 case 'c':
682                   // Default time and date representation.
683                   const char_type*  __dt[2];
684                   __tp._M_date_time_formats(__dt);
685                   __beg = _M_extract_via_format(__beg, __end, __io, __tmperr, 
686                                                 __tm, __dt[0]);
687                   break;
688                 case 'd':
689                   // Day [01, 31]. [tm_mday]
690                   __beg = _M_extract_num(__beg, __end, __tm->tm_mday, 1, 31, 2,
691                                          __io, __tmperr);
692                   break;
693                 case 'e':
694                   // Day [1, 31], with single digits preceded by
695                   // space. [tm_mday]
696                   if (__ctype.is(ctype_base::space, *__beg))
697                     __beg = _M_extract_num(++__beg, __end, __tm->tm_mday, 1, 9,
698                                            1, __io, __tmperr);
699                   else
700                     __beg = _M_extract_num(__beg, __end, __tm->tm_mday, 10, 31,
701                                            2, __io, __tmperr);
702                   break;
703                 case 'D':
704                   // Equivalent to %m/%d/%y.[tm_mon, tm_mday, tm_year]
705                   __cs = "%m/%d/%y";
706                   __ctype.widen(__cs, __cs + 9, __wcs);
707                   __beg = _M_extract_via_format(__beg, __end, __io, __tmperr, 
708                                                 __tm, __wcs);
709                   break;
710                 case 'H':
711                   // Hour [00, 23]. [tm_hour]
712                   __beg = _M_extract_num(__beg, __end, __tm->tm_hour, 0, 23, 2,
713                                          __io, __tmperr);
714                   break;
715                 case 'I':
716                   // Hour [01, 12]. [tm_hour]
717                   __beg = _M_extract_num(__beg, __end, __tm->tm_hour, 1, 12, 2,
718                                          __io, __tmperr);
719                   break;
720                 case 'm':
721                   // Month [01, 12]. [tm_mon]
722                   __beg = _M_extract_num(__beg, __end, __mem, 1, 12, 2, 
723                                          __io, __tmperr);
724                   if (!__tmperr)
725                     __tm->tm_mon = __mem - 1;
726                   break;
727                 case 'M':
728                   // Minute [00, 59]. [tm_min]
729                   __beg = _M_extract_num(__beg, __end, __tm->tm_min, 0, 59, 2,
730                                          __io, __tmperr);
731                   break;
732                 case 'n':
733                   if (__ctype.narrow(*__beg, 0) == '\n')
734                     ++__beg;
735                   else
736                     __tmperr |= ios_base::failbit;
737                   break;
738                 case 'R':
739                   // Equivalent to (%H:%M).
740                   __cs = "%H:%M";
741                   __ctype.widen(__cs, __cs + 6, __wcs);
742                   __beg = _M_extract_via_format(__beg, __end, __io, __tmperr, 
743                                                 __tm, __wcs);
744                   break;
745                 case 'S':
746                   // Seconds. [tm_sec]
747                   // [00, 60] in C99 (one leap-second), [00, 61] in C89.
748 #ifdef _GLIBCXX_USE_C99
749                   __beg = _M_extract_num(__beg, __end, __tm->tm_sec, 0, 60, 2,
750 #else
751                   __beg = _M_extract_num(__beg, __end, __tm->tm_sec, 0, 61, 2,
752 #endif
753                                          __io, __tmperr);
754                   break;
755                 case 't':
756                   if (__ctype.narrow(*__beg, 0) == '\t')
757                     ++__beg;
758                   else
759                     __tmperr |= ios_base::failbit;
760                   break;
761                 case 'T':
762                   // Equivalent to (%H:%M:%S).
763                   __cs = "%H:%M:%S";
764                   __ctype.widen(__cs, __cs + 9, __wcs);
765                   __beg = _M_extract_via_format(__beg, __end, __io, __tmperr, 
766                                                 __tm, __wcs);
767                   break;
768                 case 'x':
769                   // Locale's date.
770                   const char_type*  __dates[2];
771                   __tp._M_date_formats(__dates);
772                   __beg = _M_extract_via_format(__beg, __end, __io, __tmperr, 
773                                                 __tm, __dates[0]);
774                   break;
775                 case 'X':
776                   // Locale's time.
777                   const char_type*  __times[2];
778                   __tp._M_time_formats(__times);
779                   __beg = _M_extract_via_format(__beg, __end, __io, __tmperr, 
780                                                 __tm, __times[0]);
781                   break;
782                 case 'y':
783                 case 'C': // C99
784                   // Two digit year.
785                 case 'Y':
786                   // Year [1900).
787                   // NB: We parse either two digits, implicitly years since
788                   // 1900, or 4 digits, full year.  In both cases we can 
789                   // reconstruct [tm_year].  See also libstdc++/26701.
790                   __beg = _M_extract_num(__beg, __end, __mem, 0, 9999, 4,
791                                          __io, __tmperr);
792                   if (!__tmperr)
793                     __tm->tm_year = __mem < 0 ? __mem + 100 : __mem - 1900;
794                   break;
795                 case 'Z':
796                   // Timezone info.
797                   if (__ctype.is(ctype_base::upper, *__beg))
798                     {
799                       int __tmp;
800                       __beg = _M_extract_name(__beg, __end, __tmp,
801                                        __timepunct_cache<_CharT>::_S_timezones,
802                                               14, __io, __tmperr);
804                       // GMT requires special effort.
805                       if (__beg != __end && !__tmperr && __tmp == 0
806                           && (*__beg == __ctype.widen('-')
807                               || *__beg == __ctype.widen('+')))
808                         {
809                           __beg = _M_extract_num(__beg, __end, __tmp, 0, 23, 2,
810                                                  __io, __tmperr);
811                           __beg = _M_extract_num(__beg, __end, __tmp, 0, 59, 2,
812                                                  __io, __tmperr);
813                         }
814                     }
815                   else
816                     __tmperr |= ios_base::failbit;
817                   break;
818                 default:
819                   // Not recognized.
820                   __tmperr |= ios_base::failbit;
821                 }
822             }
823           else
824             {
825               // Verify format and input match, extract and discard.
826               if (__format[__i] == *__beg)
827                 ++__beg;
828               else
829                 __tmperr |= ios_base::failbit;
830             }
831         }
833       if (__tmperr || __i != __len)
834         __err |= ios_base::failbit;
835   
836       return __beg;
837     }
839   template<typename _CharT, typename _InIter>
840     _InIter
841     time_get<_CharT, _InIter>::
842     _M_extract_num(iter_type __beg, iter_type __end, int& __member,
843                    int __min, int __max, size_t __len,
844                    ios_base& __io, ios_base::iostate& __err) const
845     {
846       const locale& __loc = __io._M_getloc();
847       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
849       // As-is works for __len = 1, 2, 4, the values actually used.
850       int __mult = __len == 2 ? 10 : (__len == 4 ? 1000 : 1);
852       ++__min;
853       size_t __i = 0;
854       int __value = 0;
855       for (; __beg != __end && __i < __len; ++__beg, ++__i)
856         {
857           const char __c = __ctype.narrow(*__beg, '*');
858           if (__c >= '0' && __c <= '9')
859             {
860               __value = __value * 10 + (__c - '0');
861               const int __valuec = __value * __mult;
862               if (__valuec > __max || __valuec + __mult < __min)
863                 break;
864               __mult /= 10;
865             }
866           else
867             break;
868         }
869       if (__i == __len)
870         __member = __value;
871       // Special encoding for do_get_year, 'y', and 'Y' above.
872       else if (__len == 4 && __i == 2)
873         __member = __value - 100;
874       else
875         __err |= ios_base::failbit;
877       return __beg;
878     }
880   // Assumptions:
881   // All elements in __names are unique.
882   template<typename _CharT, typename _InIter>
883     _InIter
884     time_get<_CharT, _InIter>::
885     _M_extract_name(iter_type __beg, iter_type __end, int& __member,
886                     const _CharT** __names, size_t __indexlen,
887                     ios_base& __io, ios_base::iostate& __err) const
888     {
889       typedef char_traits<_CharT>               __traits_type;
890       const locale& __loc = __io._M_getloc();
891       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
893       int* __matches = static_cast<int*>(__builtin_alloca(sizeof(int)
894                                                           * __indexlen));
895       size_t __nmatches = 0;
896       size_t __pos = 0;
897       bool __testvalid = true;
898       const char_type* __name;
900       // Look for initial matches.
901       // NB: Some of the locale data is in the form of all lowercase
902       // names, and some is in the form of initially-capitalized
903       // names. Look for both.
904       if (__beg != __end)
905         {
906           const char_type __c = *__beg;
907           for (size_t __i1 = 0; __i1 < __indexlen; ++__i1)
908             if (__c == __names[__i1][0]
909                 || __c == __ctype.toupper(__names[__i1][0]))
910               __matches[__nmatches++] = __i1;
911         }
913       while (__nmatches > 1)
914         {
915           // Find smallest matching string.
916           size_t __minlen = __traits_type::length(__names[__matches[0]]);
917           for (size_t __i2 = 1; __i2 < __nmatches; ++__i2)
918             __minlen = std::min(__minlen,
919                               __traits_type::length(__names[__matches[__i2]]));
920           ++__beg, ++__pos;
921           if (__pos < __minlen && __beg != __end)
922             for (size_t __i3 = 0; __i3 < __nmatches;)
923               {
924                 __name = __names[__matches[__i3]];
925                 if (!(__name[__pos] == *__beg))
926                   __matches[__i3] = __matches[--__nmatches];
927                 else
928                   ++__i3;
929               }
930           else
931             break;
932         }
934       if (__nmatches == 1)
935         {
936           // Make sure found name is completely extracted.
937           ++__beg, ++__pos;
938           __name = __names[__matches[0]];
939           const size_t __len = __traits_type::length(__name);
940           while (__pos < __len && __beg != __end && __name[__pos] == *__beg)
941             ++__beg, ++__pos;
943           if (__len == __pos)
944             __member = __matches[0];
945           else
946             __testvalid = false;
947         }
948       else
949         __testvalid = false;
950       if (!__testvalid)
951         __err |= ios_base::failbit;
953       return __beg;
954     }
956   template<typename _CharT, typename _InIter>
957     _InIter
958     time_get<_CharT, _InIter>::
959     _M_extract_wday_or_month(iter_type __beg, iter_type __end, int& __member,
960                              const _CharT** __names, size_t __indexlen,
961                              ios_base& __io, ios_base::iostate& __err) const
962     {
963       typedef char_traits<_CharT>               __traits_type;
964       const locale& __loc = __io._M_getloc();
965       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
967       int* __matches = static_cast<int*>(__builtin_alloca(2 * sizeof(int)
968                                                           * __indexlen));
969       size_t __nmatches = 0;
970       size_t* __matches_lengths = 0;
971       size_t __pos = 0;
973       if (__beg != __end)
974         {
975           const char_type __c = *__beg;
976           for (size_t __i = 0; __i < 2 * __indexlen; ++__i)
977             if (__c == __names[__i][0]
978                 || __c == __ctype.toupper(__names[__i][0]))
979               __matches[__nmatches++] = __i;
980         }
982       if (__nmatches)
983         {
984           ++__beg, ++__pos;
986           __matches_lengths
987             = static_cast<size_t*>(__builtin_alloca(sizeof(size_t)
988                                                     * __nmatches));
989           for (size_t __i = 0; __i < __nmatches; ++__i)
990             __matches_lengths[__i]
991               = __traits_type::length(__names[__matches[__i]]);
992         }
994       for (; __beg != __end; ++__beg, ++__pos)
995         {
996           size_t __nskipped = 0;
997           const char_type __c = *__beg;
998           for (size_t __i = 0; __i < __nmatches;)
999             {
1000               const char_type* __name = __names[__matches[__i]];
1001               if (__pos >= __matches_lengths[__i])
1002                 ++__nskipped, ++__i;
1003               else if (!(__name[__pos] == __c))
1004                 {
1005                   --__nmatches;
1006                   __matches[__i] = __matches[__nmatches];
1007                   __matches_lengths[__i] = __matches_lengths[__nmatches];
1008                 }
1009               else
1010                 ++__i;
1011             }
1012           if (__nskipped == __nmatches)
1013             break;
1014         }
1016       if ((__nmatches == 1 && __matches_lengths[0] == __pos)
1017           || (__nmatches == 2 && (__matches_lengths[0] == __pos
1018                                   || __matches_lengths[1] == __pos)))
1019         __member = (__matches[0] >= __indexlen
1020                     ? __matches[0] - __indexlen : __matches[0]);
1021       else
1022         __err |= ios_base::failbit;
1024       return __beg;
1025     }
1027   template<typename _CharT, typename _InIter>
1028     _InIter
1029     time_get<_CharT, _InIter>::
1030     do_get_time(iter_type __beg, iter_type __end, ios_base& __io,
1031                 ios_base::iostate& __err, tm* __tm) const
1032     {
1033       const locale& __loc = __io._M_getloc();
1034       const __timepunct<_CharT>& __tp = use_facet<__timepunct<_CharT> >(__loc);
1035       const char_type*  __times[2];
1036       __tp._M_time_formats(__times);
1037       __beg = _M_extract_via_format(__beg, __end, __io, __err, 
1038                                     __tm, __times[0]);
1039       if (__beg == __end)
1040         __err |= ios_base::eofbit;
1041       return __beg;
1042     }
1044   template<typename _CharT, typename _InIter>
1045     _InIter
1046     time_get<_CharT, _InIter>::
1047     do_get_date(iter_type __beg, iter_type __end, ios_base& __io,
1048                 ios_base::iostate& __err, tm* __tm) const
1049     {
1050       const locale& __loc = __io._M_getloc();
1051       const __timepunct<_CharT>& __tp = use_facet<__timepunct<_CharT> >(__loc);
1052       const char_type*  __dates[2];
1053       __tp._M_date_formats(__dates);
1054       __beg = _M_extract_via_format(__beg, __end, __io, __err, 
1055                                     __tm, __dates[0]);
1056       if (__beg == __end)
1057         __err |= ios_base::eofbit;
1058       return __beg;
1059     }
1061   template<typename _CharT, typename _InIter>
1062     _InIter
1063     time_get<_CharT, _InIter>::
1064     do_get_weekday(iter_type __beg, iter_type __end, ios_base& __io,
1065                    ios_base::iostate& __err, tm* __tm) const
1066     {
1067       const locale& __loc = __io._M_getloc();
1068       const __timepunct<_CharT>& __tp = use_facet<__timepunct<_CharT> >(__loc);
1069       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
1070       const char_type* __days[14];
1071       __tp._M_days_abbreviated(__days);
1072       __tp._M_days(__days + 7);
1073       int __tmpwday;
1074       ios_base::iostate __tmperr = ios_base::goodbit;
1076       __beg = _M_extract_wday_or_month(__beg, __end, __tmpwday, __days, 7,
1077                                        __io, __tmperr);
1078       if (!__tmperr)
1079         __tm->tm_wday = __tmpwday;
1080       else
1081         __err |= ios_base::failbit;
1083       if (__beg == __end)
1084         __err |= ios_base::eofbit;
1085       return __beg;
1086      }
1088   template<typename _CharT, typename _InIter>
1089     _InIter
1090     time_get<_CharT, _InIter>::
1091     do_get_monthname(iter_type __beg, iter_type __end,
1092                      ios_base& __io, ios_base::iostate& __err, tm* __tm) const
1093     {
1094       const locale& __loc = __io._M_getloc();
1095       const __timepunct<_CharT>& __tp = use_facet<__timepunct<_CharT> >(__loc);
1096       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
1097       const char_type*  __months[24];
1098       __tp._M_months_abbreviated(__months);
1099       __tp._M_months(__months + 12);
1100       int __tmpmon;
1101       ios_base::iostate __tmperr = ios_base::goodbit;
1103       __beg = _M_extract_wday_or_month(__beg, __end, __tmpmon, __months, 12,
1104                                        __io, __tmperr);
1105       if (!__tmperr)
1106         __tm->tm_mon = __tmpmon;
1107       else
1108         __err |= ios_base::failbit;
1110       if (__beg == __end)
1111         __err |= ios_base::eofbit;
1112       return __beg;
1113     }
1115   template<typename _CharT, typename _InIter>
1116     _InIter
1117     time_get<_CharT, _InIter>::
1118     do_get_year(iter_type __beg, iter_type __end, ios_base& __io,
1119                 ios_base::iostate& __err, tm* __tm) const
1120     {
1121       const locale& __loc = __io._M_getloc();
1122       const ctype<_CharT>& __ctype = use_facet<ctype<_CharT> >(__loc);
1123       int __tmpyear;
1124       ios_base::iostate __tmperr = ios_base::goodbit;
1126       __beg = _M_extract_num(__beg, __end, __tmpyear, 0, 9999, 4,
1127                              __io, __tmperr);
1128       if (!__tmperr)
1129         __tm->tm_year = __tmpyear < 0 ? __tmpyear + 100 : __tmpyear - 1900;
1130       else
1131         __err |= ios_base::failbit;
1133       if (__beg == __end)
1134         __err |= ios_base::eofbit;
1135       return __beg;
1136     }
1138   template<typename _CharT, typename _OutIter>
1139     _OutIter
1140     time_put<_CharT, _OutIter>::
1141     put(iter_type __s, ios_base& __io, char_type __fill, const tm* __tm,
1142         const _CharT* __beg, const _CharT* __end) const
1143     {
1144       const locale& __loc = __io._M_getloc();
1145       ctype<_CharT> const& __ctype = use_facet<ctype<_CharT> >(__loc);
1146       for (; __beg != __end; ++__beg)
1147         if (__ctype.narrow(*__beg, 0) != '%')
1148           {
1149             *__s = *__beg;
1150             ++__s;
1151           }
1152         else if (++__beg != __end)
1153           {
1154             char __format;
1155             char __mod = 0;
1156             const char __c = __ctype.narrow(*__beg, 0);
1157             if (__c != 'E' && __c != 'O')
1158               __format = __c;
1159             else if (++__beg != __end)
1160               {
1161                 __mod = __c;
1162                 __format = __ctype.narrow(*__beg, 0);
1163               }
1164             else
1165               break;
1166             __s = this->do_put(__s, __io, __fill, __tm, __format, __mod);
1167           }
1168         else
1169           break;
1170       return __s;
1171     }
1173   template<typename _CharT, typename _OutIter>
1174     _OutIter
1175     time_put<_CharT, _OutIter>::
1176     do_put(iter_type __s, ios_base& __io, char_type, const tm* __tm,
1177            char __format, char __mod) const
1178     {
1179       const locale& __loc = __io._M_getloc();
1180       ctype<_CharT> const& __ctype = use_facet<ctype<_CharT> >(__loc);
1181       __timepunct<_CharT> const& __tp = use_facet<__timepunct<_CharT> >(__loc);
1183       // NB: This size is arbitrary. Should this be a data member,
1184       // initialized at construction?
1185       const size_t __maxlen = 128;
1186       char_type __res[__maxlen];
1188       // NB: In IEE 1003.1-200x, and perhaps other locale models, it
1189       // is possible that the format character will be longer than one
1190       // character. Possibilities include 'E' or 'O' followed by a
1191       // format character: if __mod is not the default argument, assume
1192       // it's a valid modifier.
1193       char_type __fmt[4];
1194       __fmt[0] = __ctype.widen('%');
1195       if (!__mod)
1196         {
1197           __fmt[1] = __format;
1198           __fmt[2] = char_type();
1199         }
1200       else
1201         {
1202           __fmt[1] = __mod;
1203           __fmt[2] = __format;
1204           __fmt[3] = char_type();
1205         }
1207       __tp._M_put(__res, __maxlen, __fmt, __tm);
1209       // Write resulting, fully-formatted string to output iterator.
1210       return std::__write(__s, __res, char_traits<char_type>::length(__res));
1211     }
1214   // Inhibit implicit instantiations for required instantiations,
1215   // which are defined via explicit instantiations elsewhere.
1216 #if _GLIBCXX_EXTERN_TEMPLATE
1217   extern template class moneypunct<char, false>;
1218   extern template class moneypunct<char, true>;
1219   extern template class moneypunct_byname<char, false>;
1220   extern template class moneypunct_byname<char, true>;
1221   extern template class _GLIBCXX_NAMESPACE_LDBL money_get<char>;
1222   extern template class _GLIBCXX_NAMESPACE_LDBL money_put<char>;
1223   extern template class __timepunct<char>;
1224   extern template class time_put<char>;
1225   extern template class time_put_byname<char>;
1226   extern template class time_get<char>;
1227   extern template class time_get_byname<char>;
1228   extern template class messages<char>;
1229   extern template class messages_byname<char>;
1231   extern template
1232     const moneypunct<char, true>&
1233     use_facet<moneypunct<char, true> >(const locale&);
1235   extern template
1236     const moneypunct<char, false>&
1237     use_facet<moneypunct<char, false> >(const locale&);
1239   extern template
1240     const money_put<char>&
1241     use_facet<money_put<char> >(const locale&);
1243   extern template
1244     const money_get<char>&
1245     use_facet<money_get<char> >(const locale&);
1247   extern template
1248     const __timepunct<char>&
1249     use_facet<__timepunct<char> >(const locale&);
1251   extern template
1252     const time_put<char>&
1253     use_facet<time_put<char> >(const locale&);
1255   extern template
1256     const time_get<char>&
1257     use_facet<time_get<char> >(const locale&);
1259   extern template
1260     const messages<char>&
1261     use_facet<messages<char> >(const locale&);
1263   extern template
1264     bool
1265     has_facet<moneypunct<char> >(const locale&);
1267   extern template
1268     bool
1269     has_facet<money_put<char> >(const locale&);
1271   extern template
1272     bool
1273     has_facet<money_get<char> >(const locale&);
1275   extern template
1276     bool
1277     has_facet<__timepunct<char> >(const locale&);
1279   extern template
1280     bool
1281     has_facet<time_put<char> >(const locale&);
1283   extern template
1284     bool
1285     has_facet<time_get<char> >(const locale&);
1287   extern template
1288     bool
1289     has_facet<messages<char> >(const locale&);
1291 #ifdef _GLIBCXX_USE_WCHAR_T
1292   extern template class moneypunct<wchar_t, false>;
1293   extern template class moneypunct<wchar_t, true>;
1294   extern template class moneypunct_byname<wchar_t, false>;
1295   extern template class moneypunct_byname<wchar_t, true>;
1296   extern template class _GLIBCXX_NAMESPACE_LDBL money_get<wchar_t>;
1297   extern template class _GLIBCXX_NAMESPACE_LDBL money_put<wchar_t>;
1298   extern template class __timepunct<wchar_t>;
1299   extern template class time_put<wchar_t>;
1300   extern template class time_put_byname<wchar_t>;
1301   extern template class time_get<wchar_t>;
1302   extern template class time_get_byname<wchar_t>;
1303   extern template class messages<wchar_t>;
1304   extern template class messages_byname<wchar_t>;
1306   extern template
1307     const moneypunct<wchar_t, true>&
1308     use_facet<moneypunct<wchar_t, true> >(const locale&);
1310   extern template
1311     const moneypunct<wchar_t, false>&
1312     use_facet<moneypunct<wchar_t, false> >(const locale&);
1314   extern template
1315     const money_put<wchar_t>&
1316     use_facet<money_put<wchar_t> >(const locale&);
1318   extern template
1319     const money_get<wchar_t>&
1320     use_facet<money_get<wchar_t> >(const locale&);
1322   extern template
1323     const __timepunct<wchar_t>&
1324     use_facet<__timepunct<wchar_t> >(const locale&);
1326   extern template
1327     const time_put<wchar_t>&
1328     use_facet<time_put<wchar_t> >(const locale&);
1330   extern template
1331     const time_get<wchar_t>&
1332     use_facet<time_get<wchar_t> >(const locale&);
1334   extern template
1335     const messages<wchar_t>&
1336     use_facet<messages<wchar_t> >(const locale&);
1338   extern template
1339     bool
1340     has_facet<moneypunct<wchar_t> >(const locale&);
1342   extern template
1343     bool
1344     has_facet<money_put<wchar_t> >(const locale&);
1346   extern template
1347     bool
1348     has_facet<money_get<wchar_t> >(const locale&);
1350   extern template
1351     bool
1352     has_facet<__timepunct<wchar_t> >(const locale&);
1354   extern template
1355     bool
1356     has_facet<time_put<wchar_t> >(const locale&);
1358   extern template
1359     bool
1360     has_facet<time_get<wchar_t> >(const locale&);
1362   extern template
1363     bool
1364     has_facet<messages<wchar_t> >(const locale&);
1365 #endif
1366 #endif
1368 _GLIBCXX_END_NAMESPACE_VERSION
1369 } // namespace std
1371 #endif