xref: /freebsd/contrib/llvm-project/libcxx/include/locale (revision 0fca6ea1d4eea4c934cfff25ac9ee8ad6fe95583)
10b57cec5SDimitry Andric// -*- C++ -*-
2349cc55cSDimitry Andric//===----------------------------------------------------------------------===//
30b57cec5SDimitry Andric//
40b57cec5SDimitry Andric// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
50b57cec5SDimitry Andric// See https://llvm.org/LICENSE.txt for license information.
60b57cec5SDimitry Andric// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
70b57cec5SDimitry Andric//
80b57cec5SDimitry Andric//===----------------------------------------------------------------------===//
90b57cec5SDimitry Andric
100b57cec5SDimitry Andric#ifndef _LIBCPP_LOCALE
110b57cec5SDimitry Andric#define _LIBCPP_LOCALE
120b57cec5SDimitry Andric
130b57cec5SDimitry Andric/*
140b57cec5SDimitry Andric    locale synopsis
150b57cec5SDimitry Andric
160b57cec5SDimitry Andricnamespace std
170b57cec5SDimitry Andric{
180b57cec5SDimitry Andric
190b57cec5SDimitry Andricclass locale
200b57cec5SDimitry Andric{
210b57cec5SDimitry Andricpublic:
220b57cec5SDimitry Andric    // types:
230b57cec5SDimitry Andric    class facet;
240b57cec5SDimitry Andric    class id;
250b57cec5SDimitry Andric
260b57cec5SDimitry Andric    typedef int category;
270b57cec5SDimitry Andric    static const category // values assigned here are for exposition only
280b57cec5SDimitry Andric        none     = 0x000,
290b57cec5SDimitry Andric        collate  = 0x010,
300b57cec5SDimitry Andric        ctype    = 0x020,
310b57cec5SDimitry Andric        monetary = 0x040,
320b57cec5SDimitry Andric        numeric  = 0x080,
330b57cec5SDimitry Andric        time     = 0x100,
340b57cec5SDimitry Andric        messages = 0x200,
350b57cec5SDimitry Andric        all = collate | ctype | monetary | numeric | time | messages;
360b57cec5SDimitry Andric
370b57cec5SDimitry Andric    // construct/copy/destroy:
380b57cec5SDimitry Andric    locale() noexcept;
390b57cec5SDimitry Andric    locale(const locale& other) noexcept;
400b57cec5SDimitry Andric    explicit locale(const char* std_name);
410b57cec5SDimitry Andric    explicit locale(const string& std_name);
420b57cec5SDimitry Andric    locale(const locale& other, const char* std_name, category);
430b57cec5SDimitry Andric    locale(const locale& other, const string& std_name, category);
440b57cec5SDimitry Andric    template <class Facet> locale(const locale& other, Facet* f);
450b57cec5SDimitry Andric    locale(const locale& other, const locale& one, category);
460b57cec5SDimitry Andric
470b57cec5SDimitry Andric    ~locale(); // not virtual
480b57cec5SDimitry Andric
490b57cec5SDimitry Andric    const locale& operator=(const locale& other) noexcept;
500b57cec5SDimitry Andric
510b57cec5SDimitry Andric    template <class Facet> locale combine(const locale& other) const;
520b57cec5SDimitry Andric
530b57cec5SDimitry Andric    // locale operations:
540b57cec5SDimitry Andric    basic_string<char> name() const;
550b57cec5SDimitry Andric    bool operator==(const locale& other) const;
5606c3fb27SDimitry Andric    bool operator!=(const locale& other) const;                              // removed C++20
570b57cec5SDimitry Andric    template <class charT, class Traits, class Allocator>
580b57cec5SDimitry Andric      bool operator()(const basic_string<charT,Traits,Allocator>& s1,
590b57cec5SDimitry Andric                      const basic_string<charT,Traits,Allocator>& s2) const;
600b57cec5SDimitry Andric
610b57cec5SDimitry Andric    // global locale objects:
620b57cec5SDimitry Andric    static locale global(const locale&);
630b57cec5SDimitry Andric    static const locale& classic();
640b57cec5SDimitry Andric};
650b57cec5SDimitry Andric
660b57cec5SDimitry Andrictemplate <class Facet> const Facet& use_facet(const locale&);
670b57cec5SDimitry Andrictemplate <class Facet> bool has_facet(const locale&) noexcept;
680b57cec5SDimitry Andric
690b57cec5SDimitry Andric// 22.3.3, convenience interfaces:
700b57cec5SDimitry Andrictemplate <class charT> bool isspace (charT c, const locale& loc);
710b57cec5SDimitry Andrictemplate <class charT> bool isprint (charT c, const locale& loc);
720b57cec5SDimitry Andrictemplate <class charT> bool iscntrl (charT c, const locale& loc);
730b57cec5SDimitry Andrictemplate <class charT> bool isupper (charT c, const locale& loc);
740b57cec5SDimitry Andrictemplate <class charT> bool islower (charT c, const locale& loc);
750b57cec5SDimitry Andrictemplate <class charT> bool isalpha (charT c, const locale& loc);
760b57cec5SDimitry Andrictemplate <class charT> bool isdigit (charT c, const locale& loc);
770b57cec5SDimitry Andrictemplate <class charT> bool ispunct (charT c, const locale& loc);
780b57cec5SDimitry Andrictemplate <class charT> bool isxdigit(charT c, const locale& loc);
790b57cec5SDimitry Andrictemplate <class charT> bool isalnum (charT c, const locale& loc);
800b57cec5SDimitry Andrictemplate <class charT> bool isgraph (charT c, const locale& loc);
810b57cec5SDimitry Andrictemplate <class charT> charT toupper(charT c, const locale& loc);
820b57cec5SDimitry Andrictemplate <class charT> charT tolower(charT c, const locale& loc);
830b57cec5SDimitry Andric
840b57cec5SDimitry Andrictemplate<class Codecvt, class Elem = wchar_t,
850b57cec5SDimitry Andric         class Wide_alloc = allocator<Elem>,
860b57cec5SDimitry Andric         class Byte_alloc = allocator<char>>
87*0fca6ea1SDimitry Andricclass wstring_convert                                      // Removed in C++26
880b57cec5SDimitry Andric{
890b57cec5SDimitry Andricpublic:
900b57cec5SDimitry Andric    typedef basic_string<char, char_traits<char>, Byte_alloc> byte_string;
910b57cec5SDimitry Andric    typedef basic_string<Elem, char_traits<Elem>, Wide_alloc> wide_string;
920b57cec5SDimitry Andric    typedef typename Codecvt::state_type                      state_type;
930b57cec5SDimitry Andric    typedef typename wide_string::traits_type::int_type       int_type;
940b57cec5SDimitry Andric
95e8d8bef9SDimitry Andric    wstring_convert(Codecvt* pcvt = new Codecvt);          // before C++14
96e8d8bef9SDimitry Andric    explicit wstring_convert(Codecvt* pcvt = new Codecvt); // before C++20
97e8d8bef9SDimitry Andric    wstring_convert() : wstring_convert(new Codecvt) {}    // C++20
98e8d8bef9SDimitry Andric    explicit wstring_convert(Codecvt* pcvt);               // C++20
99e8d8bef9SDimitry Andric
1000b57cec5SDimitry Andric    wstring_convert(Codecvt* pcvt, state_type state);
1010b57cec5SDimitry Andric    explicit wstring_convert(const byte_string& byte_err,           // explicit in C++14
1020b57cec5SDimitry Andric                    const wide_string& wide_err = wide_string());
1030b57cec5SDimitry Andric    wstring_convert(const wstring_convert&) = delete;               // C++14
1040b57cec5SDimitry Andric    wstring_convert & operator=(const wstring_convert &) = delete;  // C++14
1050b57cec5SDimitry Andric    ~wstring_convert();
1060b57cec5SDimitry Andric
1070b57cec5SDimitry Andric    wide_string from_bytes(char byte);
1080b57cec5SDimitry Andric    wide_string from_bytes(const char* ptr);
1090b57cec5SDimitry Andric    wide_string from_bytes(const byte_string& str);
1100b57cec5SDimitry Andric    wide_string from_bytes(const char* first, const char* last);
1110b57cec5SDimitry Andric
1120b57cec5SDimitry Andric    byte_string to_bytes(Elem wchar);
1130b57cec5SDimitry Andric    byte_string to_bytes(const Elem* wptr);
1140b57cec5SDimitry Andric    byte_string to_bytes(const wide_string& wstr);
1150b57cec5SDimitry Andric    byte_string to_bytes(const Elem* first, const Elem* last);
1160b57cec5SDimitry Andric
1170b57cec5SDimitry Andric    size_t converted() const; // noexcept in C++14
1180b57cec5SDimitry Andric    state_type state() const;
1190b57cec5SDimitry Andric};
1200b57cec5SDimitry Andric
1210b57cec5SDimitry Andrictemplate <class Codecvt, class Elem = wchar_t, class Tr = char_traits<Elem>>
122*0fca6ea1SDimitry Andricclass wbuffer_convert                                               // Removed in C++26
1230b57cec5SDimitry Andric    : public basic_streambuf<Elem, Tr>
1240b57cec5SDimitry Andric{
1250b57cec5SDimitry Andricpublic:
1260b57cec5SDimitry Andric    typedef typename Tr::state_type state_type;
1270b57cec5SDimitry Andric
128e8d8bef9SDimitry Andric    wbuffer_convert(streambuf* bytebuf = 0, Codecvt* pcvt = new Codecvt,
129e8d8bef9SDimitry Andric                    state_type state = state_type());          // before C++14
130e8d8bef9SDimitry Andric    explicit wbuffer_convert(streambuf* bytebuf = nullptr, Codecvt* pcvt = new Codecvt,
131e8d8bef9SDimitry Andric                            state_type state = state_type()); // before C++20
132e8d8bef9SDimitry Andric    wbuffer_convert() : wbuffer_convert(nullptr) {} // C++20
133e8d8bef9SDimitry Andric    explicit wbuffer_convert(streambuf* bytebuf, Codecvt* pcvt = new Codecvt,
134e8d8bef9SDimitry Andric                            state_type state = state_type()); // C++20
135e8d8bef9SDimitry Andric
1360b57cec5SDimitry Andric    wbuffer_convert(const wbuffer_convert&) = delete;               // C++14
1370b57cec5SDimitry Andric    wbuffer_convert & operator=(const wbuffer_convert &) = delete;  // C++14
1380b57cec5SDimitry Andric    ~wbuffer_convert();                                             // C++14
1390b57cec5SDimitry Andric
1400b57cec5SDimitry Andric    streambuf* rdbuf() const;
1410b57cec5SDimitry Andric    streambuf* rdbuf(streambuf* bytebuf);
1420b57cec5SDimitry Andric
1430b57cec5SDimitry Andric    state_type state() const;
1440b57cec5SDimitry Andric};
1450b57cec5SDimitry Andric
1460b57cec5SDimitry Andric// 22.4.1 and 22.4.1.3, ctype:
1470b57cec5SDimitry Andricclass ctype_base;
1480b57cec5SDimitry Andrictemplate <class charT> class ctype;
1490b57cec5SDimitry Andrictemplate <> class ctype<char>; // specialization
1500b57cec5SDimitry Andrictemplate <class charT> class ctype_byname;
1510b57cec5SDimitry Andrictemplate <> class ctype_byname<char>; // specialization
1520b57cec5SDimitry Andric
1530b57cec5SDimitry Andricclass codecvt_base;
1540b57cec5SDimitry Andrictemplate <class internT, class externT, class stateT> class codecvt;
1550b57cec5SDimitry Andrictemplate <class internT, class externT, class stateT> class codecvt_byname;
1560b57cec5SDimitry Andric
1570b57cec5SDimitry Andric// 22.4.2 and 22.4.3, numeric:
1580b57cec5SDimitry Andrictemplate <class charT, class InputIterator> class num_get;
1590b57cec5SDimitry Andrictemplate <class charT, class OutputIterator> class num_put;
1600b57cec5SDimitry Andrictemplate <class charT> class numpunct;
1610b57cec5SDimitry Andrictemplate <class charT> class numpunct_byname;
1620b57cec5SDimitry Andric
1630b57cec5SDimitry Andric// 22.4.4, col lation:
1640b57cec5SDimitry Andrictemplate <class charT> class collate;
1650b57cec5SDimitry Andrictemplate <class charT> class collate_byname;
1660b57cec5SDimitry Andric
1670b57cec5SDimitry Andric// 22.4.5, date and time:
1680b57cec5SDimitry Andricclass time_base;
1690b57cec5SDimitry Andrictemplate <class charT, class InputIterator> class time_get;
1700b57cec5SDimitry Andrictemplate <class charT, class InputIterator> class time_get_byname;
1710b57cec5SDimitry Andrictemplate <class charT, class OutputIterator> class time_put;
1720b57cec5SDimitry Andrictemplate <class charT, class OutputIterator> class time_put_byname;
1730b57cec5SDimitry Andric
1740b57cec5SDimitry Andric// 22.4.6, money:
1750b57cec5SDimitry Andricclass money_base;
1760b57cec5SDimitry Andrictemplate <class charT, class InputIterator> class money_get;
1770b57cec5SDimitry Andrictemplate <class charT, class OutputIterator> class money_put;
1780b57cec5SDimitry Andrictemplate <class charT, bool Intl> class moneypunct;
1790b57cec5SDimitry Andrictemplate <class charT, bool Intl> class moneypunct_byname;
1800b57cec5SDimitry Andric
1810b57cec5SDimitry Andric// 22.4.7, message retrieval:
1820b57cec5SDimitry Andricclass messages_base;
1830b57cec5SDimitry Andrictemplate <class charT> class messages;
1840b57cec5SDimitry Andrictemplate <class charT> class messages_byname;
1850b57cec5SDimitry Andric
1860b57cec5SDimitry Andric}  // std
1870b57cec5SDimitry Andric
1880b57cec5SDimitry Andric*/
1890b57cec5SDimitry Andric
190*0fca6ea1SDimitry Andric#include <__config>
191*0fca6ea1SDimitry Andric
192*0fca6ea1SDimitry Andric#if !defined(_LIBCPP_HAS_NO_LOCALIZATION)
193*0fca6ea1SDimitry Andric
19481ad6265SDimitry Andric#  include <__algorithm/copy.h>
19581ad6265SDimitry Andric#  include <__algorithm/equal.h>
19681ad6265SDimitry Andric#  include <__algorithm/find.h>
19781ad6265SDimitry Andric#  include <__algorithm/max.h>
19881ad6265SDimitry Andric#  include <__algorithm/reverse.h>
19981ad6265SDimitry Andric#  include <__algorithm/unwrap_iter.h>
200*0fca6ea1SDimitry Andric#  include <__assert>
20181ad6265SDimitry Andric#  include <__iterator/access.h>
20281ad6265SDimitry Andric#  include <__iterator/back_insert_iterator.h>
20381ad6265SDimitry Andric#  include <__iterator/istreambuf_iterator.h>
20481ad6265SDimitry Andric#  include <__iterator/ostreambuf_iterator.h>
205fe6060f1SDimitry Andric#  include <__locale>
206bdd1243dSDimitry Andric#  include <__memory/unique_ptr.h>
20706c3fb27SDimitry Andric#  include <__type_traits/make_unsigned.h>
20806c3fb27SDimitry Andric#  include <cerrno>
209fe6060f1SDimitry Andric#  include <cstdio>
2100b57cec5SDimitry Andric#  include <cstdlib>
2110b57cec5SDimitry Andric#  include <ctime>
212fe6060f1SDimitry Andric#  include <ios>
213fe6060f1SDimitry Andric#  include <limits>
214bdd1243dSDimitry Andric#  include <new>
215fe6060f1SDimitry Andric#  include <streambuf>
216fe6060f1SDimitry Andric#  include <version>
217fe6060f1SDimitry Andric
218bdd1243dSDimitry Andric// TODO: Fix __bsd_locale_defaults.h
219bdd1243dSDimitry Andric// NOLINTBEGIN(libcpp-robust-against-adl)
22081ad6265SDimitry Andric
221fe6060f1SDimitry Andric#  if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
222fe6060f1SDimitry Andric// Most unix variants have catopen.  These are the specific ones that don't.
22381ad6265SDimitry Andric#    if !defined(__BIONIC__) && !defined(_NEWLIB_VERSION) && !defined(__EMSCRIPTEN__)
224fe6060f1SDimitry Andric#      define _LIBCPP_HAS_CATOPEN 1
2250b57cec5SDimitry Andric#      include <nl_types.h>
2260b57cec5SDimitry Andric#    endif
227fe6060f1SDimitry Andric#  endif
2280b57cec5SDimitry Andric
2290b57cec5SDimitry Andric#  ifdef _LIBCPP_LOCALE__L_EXTENSIONS
23006c3fb27SDimitry Andric#    include <__locale_dir/locale_base_api/bsd_locale_defaults.h>
2310b57cec5SDimitry Andric#  else
23206c3fb27SDimitry Andric#    include <__locale_dir/locale_base_api/bsd_locale_fallbacks.h>
2330b57cec5SDimitry Andric#  endif
2340b57cec5SDimitry Andric
2350b57cec5SDimitry Andric#  if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
2360b57cec5SDimitry Andric#    pragma GCC system_header
2370b57cec5SDimitry Andric#  endif
2380b57cec5SDimitry Andric
2390b57cec5SDimitry Andric_LIBCPP_PUSH_MACROS
2400b57cec5SDimitry Andric#  include <__undef_macros>
2410b57cec5SDimitry Andric
2420b57cec5SDimitry Andric_LIBCPP_BEGIN_NAMESPACE_STD
2430b57cec5SDimitry Andric
2440b57cec5SDimitry Andric#  if defined(__APPLE__) || defined(__FreeBSD__)
2450b57cec5SDimitry Andric#    define _LIBCPP_GET_C_LOCALE 0
246349cc55cSDimitry Andric#  elif defined(__NetBSD__)
2470b57cec5SDimitry Andric#    define _LIBCPP_GET_C_LOCALE LC_C_LOCALE
2480b57cec5SDimitry Andric#  else
2490b57cec5SDimitry Andric#    define _LIBCPP_GET_C_LOCALE __cloc()
2500b57cec5SDimitry Andric// Get the C locale object
25106c3fb27SDimitry Andric_LIBCPP_EXPORTED_FROM_ABI locale_t __cloc();
2520b57cec5SDimitry Andric#    define __cloc_defined
2530b57cec5SDimitry Andric#  endif
2540b57cec5SDimitry Andric
2550b57cec5SDimitry Andric// __scan_keyword
2560b57cec5SDimitry Andric// Scans [__b, __e) until a match is found in the basic_strings range
2570b57cec5SDimitry Andric//  [__kb, __ke) or until it can be shown that there is no match in [__kb, __ke).
2580b57cec5SDimitry Andric//  __b will be incremented (visibly), consuming CharT until a match is found
2590b57cec5SDimitry Andric//  or proved to not exist.  A keyword may be "", in which will match anything.
2600b57cec5SDimitry Andric//  If one keyword is a prefix of another, and the next CharT in the input
2610b57cec5SDimitry Andric//  might match another keyword, the algorithm will attempt to find the longest
2620b57cec5SDimitry Andric//  matching keyword.  If the longer matching keyword ends up not matching, then
2630b57cec5SDimitry Andric//  no keyword match is found.  If no keyword match is found, __ke is returned
2640b57cec5SDimitry Andric//  and failbit is set in __err.
2650b57cec5SDimitry Andric//  Else an iterator pointing to the matching keyword is found.  If more than
2660b57cec5SDimitry Andric//  one keyword matches, an iterator to the first matching keyword is returned.
2670b57cec5SDimitry Andric//  If on exit __b == __e, eofbit is set in __err.  If __case_sensitive is false,
2680b57cec5SDimitry Andric//  __ct is used to force to lower case before comparing characters.
2690b57cec5SDimitry Andric//  Examples:
2700b57cec5SDimitry Andric//  Keywords:  "a", "abb"
2710b57cec5SDimitry Andric//  If the input is "a", the first keyword matches and eofbit is set.
2720b57cec5SDimitry Andric//  If the input is "abc", no match is found and "ab" are consumed.
2730b57cec5SDimitry Andrictemplate <class _InputIterator, class _ForwardIterator, class _Ctype>
274cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI _ForwardIterator __scan_keyword(
275cb14a3feSDimitry Andric    _InputIterator& __b,
276cb14a3feSDimitry Andric    _InputIterator __e,
277cb14a3feSDimitry Andric    _ForwardIterator __kb,
278cb14a3feSDimitry Andric    _ForwardIterator __ke,
279cb14a3feSDimitry Andric    const _Ctype& __ct,
280cb14a3feSDimitry Andric    ios_base::iostate& __err,
281cb14a3feSDimitry Andric    bool __case_sensitive = true) {
2820b57cec5SDimitry Andric  typedef typename iterator_traits<_InputIterator>::value_type _CharT;
2835f757f3fSDimitry Andric  size_t __nkw                       = static_cast<size_t>(std::distance(__kb, __ke));
2840b57cec5SDimitry Andric  const unsigned char __doesnt_match = '\0';
2850b57cec5SDimitry Andric  const unsigned char __might_match  = '\1';
2860b57cec5SDimitry Andric  const unsigned char __does_match   = '\2';
2870b57cec5SDimitry Andric  unsigned char __statbuf[100];
2880b57cec5SDimitry Andric  unsigned char* __status = __statbuf;
289e8d8bef9SDimitry Andric  unique_ptr<unsigned char, void (*)(void*)> __stat_hold(nullptr, free);
290cb14a3feSDimitry Andric  if (__nkw > sizeof(__statbuf)) {
2910b57cec5SDimitry Andric    __status = (unsigned char*)malloc(__nkw);
292e8d8bef9SDimitry Andric    if (__status == nullptr)
2930b57cec5SDimitry Andric      __throw_bad_alloc();
2940b57cec5SDimitry Andric    __stat_hold.reset(__status);
2950b57cec5SDimitry Andric  }
2960b57cec5SDimitry Andric  size_t __n_might_match = __nkw; // At this point, any keyword might match
2970b57cec5SDimitry Andric  size_t __n_does_match  = 0;     // but none of them definitely do
2980b57cec5SDimitry Andric  // Initialize all statuses to __might_match, except for "" keywords are __does_match
2990b57cec5SDimitry Andric  unsigned char* __st = __status;
300cb14a3feSDimitry Andric  for (_ForwardIterator __ky = __kb; __ky != __ke; ++__ky, (void)++__st) {
3010b57cec5SDimitry Andric    if (!__ky->empty())
3020b57cec5SDimitry Andric      *__st = __might_match;
303cb14a3feSDimitry Andric    else {
3040b57cec5SDimitry Andric      *__st = __does_match;
3050b57cec5SDimitry Andric      --__n_might_match;
3060b57cec5SDimitry Andric      ++__n_does_match;
3070b57cec5SDimitry Andric    }
3080b57cec5SDimitry Andric  }
3090b57cec5SDimitry Andric  // While there might be a match, test keywords against the next CharT
310cb14a3feSDimitry Andric  for (size_t __indx = 0; __b != __e && __n_might_match > 0; ++__indx) {
3110b57cec5SDimitry Andric    // Peek at the next CharT but don't consume it
3120b57cec5SDimitry Andric    _CharT __c = *__b;
3130b57cec5SDimitry Andric    if (!__case_sensitive)
3140b57cec5SDimitry Andric      __c = __ct.toupper(__c);
3150b57cec5SDimitry Andric    bool __consume = false;
3160b57cec5SDimitry Andric    // For each keyword which might match, see if the __indx character is __c
3170b57cec5SDimitry Andric    // If a match if found, consume __c
3180b57cec5SDimitry Andric    // If a match is found, and that is the last character in the keyword,
3190b57cec5SDimitry Andric    //    then that keyword matches.
3200b57cec5SDimitry Andric    // If the keyword doesn't match this character, then change the keyword
3210b57cec5SDimitry Andric    //    to doesn't match
3220b57cec5SDimitry Andric    __st = __status;
323cb14a3feSDimitry Andric    for (_ForwardIterator __ky = __kb; __ky != __ke; ++__ky, (void)++__st) {
324cb14a3feSDimitry Andric      if (*__st == __might_match) {
3250b57cec5SDimitry Andric        _CharT __kc = (*__ky)[__indx];
3260b57cec5SDimitry Andric        if (!__case_sensitive)
3270b57cec5SDimitry Andric          __kc = __ct.toupper(__kc);
328cb14a3feSDimitry Andric        if (__c == __kc) {
3290b57cec5SDimitry Andric          __consume = true;
330cb14a3feSDimitry Andric          if (__ky->size() == __indx + 1) {
3310b57cec5SDimitry Andric            *__st = __does_match;
3320b57cec5SDimitry Andric            --__n_might_match;
3330b57cec5SDimitry Andric            ++__n_does_match;
3340b57cec5SDimitry Andric          }
335cb14a3feSDimitry Andric        } else {
3360b57cec5SDimitry Andric          *__st = __doesnt_match;
3370b57cec5SDimitry Andric          --__n_might_match;
3380b57cec5SDimitry Andric        }
3390b57cec5SDimitry Andric      }
3400b57cec5SDimitry Andric    }
3410b57cec5SDimitry Andric    // consume if we matched a character
342cb14a3feSDimitry Andric    if (__consume) {
3430b57cec5SDimitry Andric      ++__b;
3440b57cec5SDimitry Andric      // If we consumed a character and there might be a matched keyword that
3450b57cec5SDimitry Andric      //   was marked matched on a previous iteration, then such keywords
3460b57cec5SDimitry Andric      //   which are now marked as not matching.
347cb14a3feSDimitry Andric      if (__n_might_match + __n_does_match > 1) {
3480b57cec5SDimitry Andric        __st = __status;
349cb14a3feSDimitry Andric        for (_ForwardIterator __ky = __kb; __ky != __ke; ++__ky, (void)++__st) {
350cb14a3feSDimitry Andric          if (*__st == __does_match && __ky->size() != __indx + 1) {
3510b57cec5SDimitry Andric            *__st = __doesnt_match;
3520b57cec5SDimitry Andric            --__n_does_match;
3530b57cec5SDimitry Andric          }
3540b57cec5SDimitry Andric        }
3550b57cec5SDimitry Andric      }
3560b57cec5SDimitry Andric    }
3570b57cec5SDimitry Andric  }
3580b57cec5SDimitry Andric  // We've exited the loop because we hit eof and/or we have no more "might matches".
3590b57cec5SDimitry Andric  if (__b == __e)
3600b57cec5SDimitry Andric    __err |= ios_base::eofbit;
3610b57cec5SDimitry Andric  // Return the first matching result
3620b57cec5SDimitry Andric  for (__st = __status; __kb != __ke; ++__kb, (void)++__st)
3630b57cec5SDimitry Andric    if (*__st == __does_match)
3640b57cec5SDimitry Andric      break;
3650b57cec5SDimitry Andric  if (__kb == __ke)
3660b57cec5SDimitry Andric    __err |= ios_base::failbit;
3670b57cec5SDimitry Andric  return __kb;
3680b57cec5SDimitry Andric}
3690b57cec5SDimitry Andric
370cb14a3feSDimitry Andricstruct _LIBCPP_EXPORTED_FROM_ABI __num_get_base {
3710b57cec5SDimitry Andric  static const int __num_get_buf_sz = 40;
3720b57cec5SDimitry Andric
3730b57cec5SDimitry Andric  static int __get_base(ios_base&);
374*0fca6ea1SDimitry Andric  static const char __src[33]; // "0123456789abcdefABCDEFxX+-pPiInN"
375*0fca6ea1SDimitry Andric  // count of leading characters in __src used for parsing integers ("012..X+-")
376*0fca6ea1SDimitry Andric  static const size_t __int_chr_cnt = 26;
377*0fca6ea1SDimitry Andric  // count of leading characters in __src used for parsing floating-point values ("012..-pP")
378*0fca6ea1SDimitry Andric  static const size_t __fp_chr_cnt = 28;
3790b57cec5SDimitry Andric};
3800b57cec5SDimitry Andric
381cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI void
382cb14a3feSDimitry Andric__check_grouping(const string& __grouping, unsigned* __g, unsigned* __g_end, ios_base::iostate& __err);
3830b57cec5SDimitry Andric
3840b57cec5SDimitry Andrictemplate <class _CharT>
385cb14a3feSDimitry Andricstruct __num_get : protected __num_get_base {
386cb14a3feSDimitry Andric  static string __stage2_float_prep(ios_base& __iob, _CharT* __atoms, _CharT& __decimal_point, _CharT& __thousands_sep);
3870b57cec5SDimitry Andric
388cb14a3feSDimitry Andric  static int __stage2_float_loop(
389cb14a3feSDimitry Andric      _CharT __ct,
390cb14a3feSDimitry Andric      bool& __in_units,
391cb14a3feSDimitry Andric      char& __exp,
392cb14a3feSDimitry Andric      char* __a,
393cb14a3feSDimitry Andric      char*& __a_end,
394cb14a3feSDimitry Andric      _CharT __decimal_point,
395cb14a3feSDimitry Andric      _CharT __thousands_sep,
396cb14a3feSDimitry Andric      const string& __grouping,
397cb14a3feSDimitry Andric      unsigned* __g,
398cb14a3feSDimitry Andric      unsigned*& __g_end,
399cb14a3feSDimitry Andric      unsigned& __dc,
400cb14a3feSDimitry Andric      _CharT* __atoms);
4010b57cec5SDimitry Andric#  ifndef _LIBCPP_ABI_OPTIMIZED_LOCALE_NUM_GET
4020b57cec5SDimitry Andric  static string __stage2_int_prep(ios_base& __iob, _CharT* __atoms, _CharT& __thousands_sep);
403cb14a3feSDimitry Andric  static int __stage2_int_loop(
404cb14a3feSDimitry Andric      _CharT __ct,
405cb14a3feSDimitry Andric      int __base,
406cb14a3feSDimitry Andric      char* __a,
407cb14a3feSDimitry Andric      char*& __a_end,
408cb14a3feSDimitry Andric      unsigned& __dc,
409cb14a3feSDimitry Andric      _CharT __thousands_sep,
410cb14a3feSDimitry Andric      const string& __grouping,
411cb14a3feSDimitry Andric      unsigned* __g,
412cb14a3feSDimitry Andric      unsigned*& __g_end,
413cb14a3feSDimitry Andric      _CharT* __atoms);
4140b57cec5SDimitry Andric
4150b57cec5SDimitry Andric#  else
416cb14a3feSDimitry Andric  static string __stage2_int_prep(ios_base& __iob, _CharT& __thousands_sep) {
4170b57cec5SDimitry Andric    locale __loc                 = __iob.getloc();
4180b57cec5SDimitry Andric    const numpunct<_CharT>& __np = use_facet<numpunct<_CharT> >(__loc);
4190b57cec5SDimitry Andric    __thousands_sep              = __np.thousands_sep();
4200b57cec5SDimitry Andric    return __np.grouping();
4210b57cec5SDimitry Andric  }
4220b57cec5SDimitry Andric
423cb14a3feSDimitry Andric  const _CharT* __do_widen(ios_base& __iob, _CharT* __atoms) const { return __do_widen_p(__iob, __atoms); }
4240b57cec5SDimitry Andric
425cb14a3feSDimitry Andric  static int __stage2_int_loop(
426cb14a3feSDimitry Andric      _CharT __ct,
427cb14a3feSDimitry Andric      int __base,
428cb14a3feSDimitry Andric      char* __a,
429cb14a3feSDimitry Andric      char*& __a_end,
430cb14a3feSDimitry Andric      unsigned& __dc,
431cb14a3feSDimitry Andric      _CharT __thousands_sep,
432cb14a3feSDimitry Andric      const string& __grouping,
433cb14a3feSDimitry Andric      unsigned* __g,
434cb14a3feSDimitry Andric      unsigned*& __g_end,
435cb14a3feSDimitry Andric      const _CharT* __atoms);
4360b57cec5SDimitry Andric
4370b57cec5SDimitry Andricprivate:
43804eeddc0SDimitry Andric  template <typename _Tp>
439cb14a3feSDimitry Andric  const _Tp* __do_widen_p(ios_base& __iob, _Tp* __atoms) const {
4400b57cec5SDimitry Andric    locale __loc = __iob.getloc();
441*0fca6ea1SDimitry Andric    use_facet<ctype<_Tp> >(__loc).widen(__src, __src + __int_chr_cnt, __atoms);
4420b57cec5SDimitry Andric    return __atoms;
4430b57cec5SDimitry Andric  }
4440b57cec5SDimitry Andric
445cb14a3feSDimitry Andric  const char* __do_widen_p(ios_base& __iob, char* __atoms) const {
4460b57cec5SDimitry Andric    (void)__iob;
4470b57cec5SDimitry Andric    (void)__atoms;
4480b57cec5SDimitry Andric    return __src;
4490b57cec5SDimitry Andric  }
4500b57cec5SDimitry Andric#  endif
4510b57cec5SDimitry Andric};
4520b57cec5SDimitry Andric
4530b57cec5SDimitry Andric#  ifndef _LIBCPP_ABI_OPTIMIZED_LOCALE_NUM_GET
4540b57cec5SDimitry Andrictemplate <class _CharT>
455cb14a3feSDimitry Andricstring __num_get<_CharT>::__stage2_int_prep(ios_base& __iob, _CharT* __atoms, _CharT& __thousands_sep) {
4560b57cec5SDimitry Andric  locale __loc = __iob.getloc();
457*0fca6ea1SDimitry Andric  std::use_facet<ctype<_CharT> >(__loc).widen(__src, __src + __int_chr_cnt, __atoms);
458bdd1243dSDimitry Andric  const numpunct<_CharT>& __np = std::use_facet<numpunct<_CharT> >(__loc);
4590b57cec5SDimitry Andric  __thousands_sep              = __np.thousands_sep();
4600b57cec5SDimitry Andric  return __np.grouping();
4610b57cec5SDimitry Andric}
4620b57cec5SDimitry Andric#  endif
4630b57cec5SDimitry Andric
4640b57cec5SDimitry Andrictemplate <class _CharT>
465cb14a3feSDimitry Andricstring __num_get<_CharT>::__stage2_float_prep(
466cb14a3feSDimitry Andric    ios_base& __iob, _CharT* __atoms, _CharT& __decimal_point, _CharT& __thousands_sep) {
4670b57cec5SDimitry Andric  locale __loc = __iob.getloc();
468*0fca6ea1SDimitry Andric  std::use_facet<ctype<_CharT> >(__loc).widen(__src, __src + __fp_chr_cnt, __atoms);
469bdd1243dSDimitry Andric  const numpunct<_CharT>& __np = std::use_facet<numpunct<_CharT> >(__loc);
4700b57cec5SDimitry Andric  __decimal_point              = __np.decimal_point();
4710b57cec5SDimitry Andric  __thousands_sep              = __np.thousands_sep();
4720b57cec5SDimitry Andric  return __np.grouping();
4730b57cec5SDimitry Andric}
4740b57cec5SDimitry Andric
4750b57cec5SDimitry Andrictemplate <class _CharT>
4760b57cec5SDimitry Andricint
4770b57cec5SDimitry Andric#  ifndef _LIBCPP_ABI_OPTIMIZED_LOCALE_NUM_GET
4780b57cec5SDimitry Andric__num_get<_CharT>::__stage2_int_loop(_CharT __ct, int __base, char* __a, char*& __a_end,
4790b57cec5SDimitry Andric                  unsigned& __dc, _CharT __thousands_sep, const string& __grouping,
4800b57cec5SDimitry Andric                  unsigned* __g, unsigned*& __g_end, _CharT* __atoms)
4810b57cec5SDimitry Andric#  else
4820b57cec5SDimitry Andric__num_get<_CharT>::__stage2_int_loop(_CharT __ct, int __base, char* __a, char*& __a_end,
4830b57cec5SDimitry Andric                  unsigned& __dc, _CharT __thousands_sep, const string& __grouping,
4840b57cec5SDimitry Andric                  unsigned* __g, unsigned*& __g_end, const _CharT* __atoms)
4850b57cec5SDimitry Andric
4860b57cec5SDimitry Andric#  endif
4870b57cec5SDimitry Andric{
488cb14a3feSDimitry Andric  if (__a_end == __a && (__ct == __atoms[24] || __ct == __atoms[25])) {
4890b57cec5SDimitry Andric    *__a_end++ = __ct == __atoms[24] ? '+' : '-';
4900b57cec5SDimitry Andric    __dc       = 0;
4910b57cec5SDimitry Andric    return 0;
4920b57cec5SDimitry Andric  }
493cb14a3feSDimitry Andric  if (__grouping.size() != 0 && __ct == __thousands_sep) {
494cb14a3feSDimitry Andric    if (__g_end - __g < __num_get_buf_sz) {
4950b57cec5SDimitry Andric      *__g_end++ = __dc;
4960b57cec5SDimitry Andric      __dc       = 0;
4970b57cec5SDimitry Andric    }
4980b57cec5SDimitry Andric    return 0;
4990b57cec5SDimitry Andric  }
500*0fca6ea1SDimitry Andric  ptrdiff_t __f = std::find(__atoms, __atoms + __int_chr_cnt, __ct) - __atoms;
5010b57cec5SDimitry Andric  if (__f >= 24)
5020b57cec5SDimitry Andric    return -1;
503cb14a3feSDimitry Andric  switch (__base) {
5040b57cec5SDimitry Andric  case 8:
5050b57cec5SDimitry Andric  case 10:
5060b57cec5SDimitry Andric    if (__f >= __base)
5070b57cec5SDimitry Andric      return -1;
5080b57cec5SDimitry Andric    break;
5090b57cec5SDimitry Andric  case 16:
5100b57cec5SDimitry Andric    if (__f < 22)
5110b57cec5SDimitry Andric      break;
512cb14a3feSDimitry Andric    if (__a_end != __a && __a_end - __a <= 2 && __a_end[-1] == '0') {
5130b57cec5SDimitry Andric      __dc       = 0;
5140b57cec5SDimitry Andric      *__a_end++ = __src[__f];
5150b57cec5SDimitry Andric      return 0;
5160b57cec5SDimitry Andric    }
5170b57cec5SDimitry Andric    return -1;
5180b57cec5SDimitry Andric  }
5190b57cec5SDimitry Andric  *__a_end++ = __src[__f];
5200b57cec5SDimitry Andric  ++__dc;
5210b57cec5SDimitry Andric  return 0;
5220b57cec5SDimitry Andric}
5230b57cec5SDimitry Andric
5240b57cec5SDimitry Andrictemplate <class _CharT>
525cb14a3feSDimitry Andricint __num_get<_CharT>::__stage2_float_loop(
526cb14a3feSDimitry Andric    _CharT __ct,
527cb14a3feSDimitry Andric    bool& __in_units,
528cb14a3feSDimitry Andric    char& __exp,
529cb14a3feSDimitry Andric    char* __a,
530cb14a3feSDimitry Andric    char*& __a_end,
531cb14a3feSDimitry Andric    _CharT __decimal_point,
532cb14a3feSDimitry Andric    _CharT __thousands_sep,
533cb14a3feSDimitry Andric    const string& __grouping,
534cb14a3feSDimitry Andric    unsigned* __g,
535cb14a3feSDimitry Andric    unsigned*& __g_end,
536cb14a3feSDimitry Andric    unsigned& __dc,
537cb14a3feSDimitry Andric    _CharT* __atoms) {
538cb14a3feSDimitry Andric  if (__ct == __decimal_point) {
5390b57cec5SDimitry Andric    if (!__in_units)
5400b57cec5SDimitry Andric      return -1;
5410b57cec5SDimitry Andric    __in_units = false;
5420b57cec5SDimitry Andric    *__a_end++ = '.';
5430b57cec5SDimitry Andric    if (__grouping.size() != 0 && __g_end - __g < __num_get_buf_sz)
5440b57cec5SDimitry Andric      *__g_end++ = __dc;
5450b57cec5SDimitry Andric    return 0;
5460b57cec5SDimitry Andric  }
547cb14a3feSDimitry Andric  if (__ct == __thousands_sep && __grouping.size() != 0) {
5480b57cec5SDimitry Andric    if (!__in_units)
5490b57cec5SDimitry Andric      return -1;
550cb14a3feSDimitry Andric    if (__g_end - __g < __num_get_buf_sz) {
5510b57cec5SDimitry Andric      *__g_end++ = __dc;
5520b57cec5SDimitry Andric      __dc       = 0;
5530b57cec5SDimitry Andric    }
5540b57cec5SDimitry Andric    return 0;
5550b57cec5SDimitry Andric  }
556*0fca6ea1SDimitry Andric  ptrdiff_t __f = std::find(__atoms, __atoms + __num_get_base::__fp_chr_cnt, __ct) - __atoms;
557*0fca6ea1SDimitry Andric  if (__f >= static_cast<ptrdiff_t>(__num_get_base::__fp_chr_cnt))
5580b57cec5SDimitry Andric    return -1;
5590b57cec5SDimitry Andric  char __x = __src[__f];
560cb14a3feSDimitry Andric  if (__x == '-' || __x == '+') {
561cb14a3feSDimitry Andric    if (__a_end == __a || (std::toupper(__a_end[-1]) == std::toupper(__exp))) {
5620b57cec5SDimitry Andric      *__a_end++ = __x;
5630b57cec5SDimitry Andric      return 0;
5640b57cec5SDimitry Andric    }
5650b57cec5SDimitry Andric    return -1;
5660b57cec5SDimitry Andric  }
5670b57cec5SDimitry Andric  if (__x == 'x' || __x == 'X')
5680b57cec5SDimitry Andric    __exp = 'P';
569cb14a3feSDimitry Andric  else if (std::toupper(__x) == __exp) {
570bdd1243dSDimitry Andric    __exp = std::tolower(__exp);
571cb14a3feSDimitry Andric    if (__in_units) {
5720b57cec5SDimitry Andric      __in_units = false;
5730b57cec5SDimitry Andric      if (__grouping.size() != 0 && __g_end - __g < __num_get_buf_sz)
5740b57cec5SDimitry Andric        *__g_end++ = __dc;
5750b57cec5SDimitry Andric    }
5760b57cec5SDimitry Andric  }
5770b57cec5SDimitry Andric  *__a_end++ = __x;
5780b57cec5SDimitry Andric  if (__f >= 22)
5790b57cec5SDimitry Andric    return 0;
5800b57cec5SDimitry Andric  ++__dc;
5810b57cec5SDimitry Andric  return 0;
5820b57cec5SDimitry Andric}
5830b57cec5SDimitry Andric
58481ad6265SDimitry Andricextern template struct _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __num_get<char>;
585349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
58681ad6265SDimitry Andricextern template struct _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __num_get<wchar_t>;
587349cc55cSDimitry Andric#  endif
5880b57cec5SDimitry Andric
5890b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
590cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS num_get : public locale::facet, private __num_get<_CharT> {
5910b57cec5SDimitry Andricpublic:
5920b57cec5SDimitry Andric  typedef _CharT char_type;
5930b57cec5SDimitry Andric  typedef _InputIterator iter_type;
5940b57cec5SDimitry Andric
595cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit num_get(size_t __refs = 0) : locale::facet(__refs) {}
5960b57cec5SDimitry Andric
597cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
598cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, bool& __v) const {
5990b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6000b57cec5SDimitry Andric  }
6010b57cec5SDimitry Andric
602cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
603cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, long& __v) const {
6040b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6050b57cec5SDimitry Andric  }
6060b57cec5SDimitry Andric
607cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
608cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, long long& __v) const {
6090b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6100b57cec5SDimitry Andric  }
6110b57cec5SDimitry Andric
612cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
613cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned short& __v) const {
6140b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6150b57cec5SDimitry Andric  }
6160b57cec5SDimitry Andric
617cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
618cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned int& __v) const {
6190b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6200b57cec5SDimitry Andric  }
6210b57cec5SDimitry Andric
622cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
623cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned long& __v) const {
6240b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6250b57cec5SDimitry Andric  }
6260b57cec5SDimitry Andric
627cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
628cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned long long& __v) const {
6290b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6300b57cec5SDimitry Andric  }
6310b57cec5SDimitry Andric
632cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
633cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, float& __v) const {
6340b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6350b57cec5SDimitry Andric  }
6360b57cec5SDimitry Andric
637cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
638cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, double& __v) const {
6390b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6400b57cec5SDimitry Andric  }
6410b57cec5SDimitry Andric
642cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
643cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, long double& __v) const {
6440b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6450b57cec5SDimitry Andric  }
6460b57cec5SDimitry Andric
647cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
648cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, void*& __v) const {
6490b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __v);
6500b57cec5SDimitry Andric  }
6510b57cec5SDimitry Andric
6520b57cec5SDimitry Andric  static locale::id id;
6530b57cec5SDimitry Andric
6540b57cec5SDimitry Andricprotected:
655bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~num_get() override {}
6560b57cec5SDimitry Andric
6570b57cec5SDimitry Andric  template <class _Fp>
658cb14a3feSDimitry Andric  _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS iter_type
659cb14a3feSDimitry Andric  __do_get_floating_point(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, _Fp& __v) const;
6600b57cec5SDimitry Andric
6610b57cec5SDimitry Andric  template <class _Signed>
662cb14a3feSDimitry Andric  _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS iter_type
663cb14a3feSDimitry Andric  __do_get_signed(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, _Signed& __v) const;
6640b57cec5SDimitry Andric
6650b57cec5SDimitry Andric  template <class _Unsigned>
666cb14a3feSDimitry Andric  _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS iter_type
667cb14a3feSDimitry Andric  __do_get_unsigned(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, _Unsigned& __v) const;
6680b57cec5SDimitry Andric
669cb14a3feSDimitry Andric  virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, bool& __v) const;
6700b57cec5SDimitry Andric
671cb14a3feSDimitry Andric  virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, long& __v) const {
672cb14a3feSDimitry Andric    return this->__do_get_signed(__b, __e, __iob, __err, __v);
673cb14a3feSDimitry Andric  }
6740b57cec5SDimitry Andric
675cb14a3feSDimitry Andric  virtual iter_type
676cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, long long& __v) const {
677cb14a3feSDimitry Andric    return this->__do_get_signed(__b, __e, __iob, __err, __v);
678cb14a3feSDimitry Andric  }
6790b57cec5SDimitry Andric
680cb14a3feSDimitry Andric  virtual iter_type
681cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned short& __v) const {
682cb14a3feSDimitry Andric    return this->__do_get_unsigned(__b, __e, __iob, __err, __v);
683cb14a3feSDimitry Andric  }
6840b57cec5SDimitry Andric
685cb14a3feSDimitry Andric  virtual iter_type
686cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned int& __v) const {
687cb14a3feSDimitry Andric    return this->__do_get_unsigned(__b, __e, __iob, __err, __v);
688cb14a3feSDimitry Andric  }
6890b57cec5SDimitry Andric
690cb14a3feSDimitry Andric  virtual iter_type
691cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned long& __v) const {
692cb14a3feSDimitry Andric    return this->__do_get_unsigned(__b, __e, __iob, __err, __v);
693cb14a3feSDimitry Andric  }
6940b57cec5SDimitry Andric
695cb14a3feSDimitry Andric  virtual iter_type
696cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, unsigned long long& __v) const {
697cb14a3feSDimitry Andric    return this->__do_get_unsigned(__b, __e, __iob, __err, __v);
698cb14a3feSDimitry Andric  }
6990b57cec5SDimitry Andric
700cb14a3feSDimitry Andric  virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, float& __v) const {
701cb14a3feSDimitry Andric    return this->__do_get_floating_point(__b, __e, __iob, __err, __v);
702cb14a3feSDimitry Andric  }
7030b57cec5SDimitry Andric
704cb14a3feSDimitry Andric  virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, double& __v) const {
705cb14a3feSDimitry Andric    return this->__do_get_floating_point(__b, __e, __iob, __err, __v);
706cb14a3feSDimitry Andric  }
7070b57cec5SDimitry Andric
708cb14a3feSDimitry Andric  virtual iter_type
709cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, long double& __v) const {
710cb14a3feSDimitry Andric    return this->__do_get_floating_point(__b, __e, __iob, __err, __v);
711cb14a3feSDimitry Andric  }
7120b57cec5SDimitry Andric
713cb14a3feSDimitry Andric  virtual iter_type do_get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, void*& __v) const;
7140b57cec5SDimitry Andric};
7150b57cec5SDimitry Andric
7160b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
717cb14a3feSDimitry Andriclocale::id num_get<_CharT, _InputIterator>::id;
7180b57cec5SDimitry Andric
7190b57cec5SDimitry Andrictemplate <class _Tp>
720bdd1243dSDimitry Andric_LIBCPP_HIDE_FROM_ABI _Tp
721cb14a3feSDimitry Andric__num_get_signed_integral(const char* __a, const char* __a_end, ios_base::iostate& __err, int __base) {
722cb14a3feSDimitry Andric  if (__a != __a_end) {
723bdd1243dSDimitry Andric    __libcpp_remove_reference_t<decltype(errno)> __save_errno = errno;
7240b57cec5SDimitry Andric    errno                                                     = 0;
7250b57cec5SDimitry Andric    char* __p2;
7260b57cec5SDimitry Andric    long long __ll                                               = strtoll_l(__a, &__p2, __base, _LIBCPP_GET_C_LOCALE);
727bdd1243dSDimitry Andric    __libcpp_remove_reference_t<decltype(errno)> __current_errno = errno;
7280b57cec5SDimitry Andric    if (__current_errno == 0)
7290b57cec5SDimitry Andric      errno = __save_errno;
730cb14a3feSDimitry Andric    if (__p2 != __a_end) {
7310b57cec5SDimitry Andric      __err = ios_base::failbit;
7320b57cec5SDimitry Andric      return 0;
733cb14a3feSDimitry Andric    } else if (__current_errno == ERANGE || __ll < numeric_limits<_Tp>::min() || numeric_limits<_Tp>::max() < __ll) {
7340b57cec5SDimitry Andric      __err = ios_base::failbit;
7350b57cec5SDimitry Andric      if (__ll > 0)
7360b57cec5SDimitry Andric        return numeric_limits<_Tp>::max();
7370b57cec5SDimitry Andric      else
7380b57cec5SDimitry Andric        return numeric_limits<_Tp>::min();
7390b57cec5SDimitry Andric    }
7400b57cec5SDimitry Andric    return static_cast<_Tp>(__ll);
7410b57cec5SDimitry Andric  }
7420b57cec5SDimitry Andric  __err = ios_base::failbit;
7430b57cec5SDimitry Andric  return 0;
7440b57cec5SDimitry Andric}
7450b57cec5SDimitry Andric
7460b57cec5SDimitry Andrictemplate <class _Tp>
747bdd1243dSDimitry Andric_LIBCPP_HIDE_FROM_ABI _Tp
748cb14a3feSDimitry Andric__num_get_unsigned_integral(const char* __a, const char* __a_end, ios_base::iostate& __err, int __base) {
749cb14a3feSDimitry Andric  if (__a != __a_end) {
7500b57cec5SDimitry Andric    const bool __negate = *__a == '-';
7510b57cec5SDimitry Andric    if (__negate && ++__a == __a_end) {
7520b57cec5SDimitry Andric      __err = ios_base::failbit;
7530b57cec5SDimitry Andric      return 0;
7540b57cec5SDimitry Andric    }
755bdd1243dSDimitry Andric    __libcpp_remove_reference_t<decltype(errno)> __save_errno = errno;
7560b57cec5SDimitry Andric    errno                                                     = 0;
7570b57cec5SDimitry Andric    char* __p2;
7580b57cec5SDimitry Andric    unsigned long long __ll                                      = strtoull_l(__a, &__p2, __base, _LIBCPP_GET_C_LOCALE);
759bdd1243dSDimitry Andric    __libcpp_remove_reference_t<decltype(errno)> __current_errno = errno;
7600b57cec5SDimitry Andric    if (__current_errno == 0)
7610b57cec5SDimitry Andric      errno = __save_errno;
762cb14a3feSDimitry Andric    if (__p2 != __a_end) {
7630b57cec5SDimitry Andric      __err = ios_base::failbit;
7640b57cec5SDimitry Andric      return 0;
765cb14a3feSDimitry Andric    } else if (__current_errno == ERANGE || numeric_limits<_Tp>::max() < __ll) {
7660b57cec5SDimitry Andric      __err = ios_base::failbit;
7670b57cec5SDimitry Andric      return numeric_limits<_Tp>::max();
7680b57cec5SDimitry Andric    }
7690b57cec5SDimitry Andric    _Tp __res = static_cast<_Tp>(__ll);
770cb14a3feSDimitry Andric    if (__negate)
771cb14a3feSDimitry Andric      __res = -__res;
7720b57cec5SDimitry Andric    return __res;
7730b57cec5SDimitry Andric  }
7740b57cec5SDimitry Andric  __err = ios_base::failbit;
7750b57cec5SDimitry Andric  return 0;
7760b57cec5SDimitry Andric}
7770b57cec5SDimitry Andric
7780b57cec5SDimitry Andrictemplate <class _Tp>
779cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI _Tp __do_strtod(const char* __a, char** __p2);
7800b57cec5SDimitry Andric
7810b57cec5SDimitry Andrictemplate <>
782cb14a3feSDimitry Andricinline _LIBCPP_HIDE_FROM_ABI float __do_strtod<float>(const char* __a, char** __p2) {
7830b57cec5SDimitry Andric  return strtof_l(__a, __p2, _LIBCPP_GET_C_LOCALE);
7840b57cec5SDimitry Andric}
7850b57cec5SDimitry Andric
7860b57cec5SDimitry Andrictemplate <>
787cb14a3feSDimitry Andricinline _LIBCPP_HIDE_FROM_ABI double __do_strtod<double>(const char* __a, char** __p2) {
7880b57cec5SDimitry Andric  return strtod_l(__a, __p2, _LIBCPP_GET_C_LOCALE);
7890b57cec5SDimitry Andric}
7900b57cec5SDimitry Andric
7910b57cec5SDimitry Andrictemplate <>
792cb14a3feSDimitry Andricinline _LIBCPP_HIDE_FROM_ABI long double __do_strtod<long double>(const char* __a, char** __p2) {
7930b57cec5SDimitry Andric  return strtold_l(__a, __p2, _LIBCPP_GET_C_LOCALE);
7940b57cec5SDimitry Andric}
7950b57cec5SDimitry Andric
7960b57cec5SDimitry Andrictemplate <class _Tp>
797cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI _Tp __num_get_float(const char* __a, const char* __a_end, ios_base::iostate& __err) {
798cb14a3feSDimitry Andric  if (__a != __a_end) {
799bdd1243dSDimitry Andric    __libcpp_remove_reference_t<decltype(errno)> __save_errno = errno;
8000b57cec5SDimitry Andric    errno                                                     = 0;
8010b57cec5SDimitry Andric    char* __p2;
802bdd1243dSDimitry Andric    _Tp __ld                                                     = std::__do_strtod<_Tp>(__a, &__p2);
803bdd1243dSDimitry Andric    __libcpp_remove_reference_t<decltype(errno)> __current_errno = errno;
8040b57cec5SDimitry Andric    if (__current_errno == 0)
8050b57cec5SDimitry Andric      errno = __save_errno;
806cb14a3feSDimitry Andric    if (__p2 != __a_end) {
8070b57cec5SDimitry Andric      __err = ios_base::failbit;
8080b57cec5SDimitry Andric      return 0;
809cb14a3feSDimitry Andric    } else if (__current_errno == ERANGE)
8100b57cec5SDimitry Andric      __err = ios_base::failbit;
8110b57cec5SDimitry Andric    return __ld;
8120b57cec5SDimitry Andric  }
8130b57cec5SDimitry Andric  __err = ios_base::failbit;
8140b57cec5SDimitry Andric  return 0;
8150b57cec5SDimitry Andric}
8160b57cec5SDimitry Andric
8170b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
818cb14a3feSDimitry Andric_InputIterator num_get<_CharT, _InputIterator>::do_get(
819cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, bool& __v) const {
820cb14a3feSDimitry Andric  if ((__iob.flags() & ios_base::boolalpha) == 0) {
8210b57cec5SDimitry Andric    long __lv = -1;
8220b57cec5SDimitry Andric    __b       = do_get(__b, __e, __iob, __err, __lv);
823cb14a3feSDimitry Andric    switch (__lv) {
8240b57cec5SDimitry Andric    case 0:
8250b57cec5SDimitry Andric      __v = false;
8260b57cec5SDimitry Andric      break;
8270b57cec5SDimitry Andric    case 1:
8280b57cec5SDimitry Andric      __v = true;
8290b57cec5SDimitry Andric      break;
8300b57cec5SDimitry Andric    default:
8310b57cec5SDimitry Andric      __v   = true;
8320b57cec5SDimitry Andric      __err = ios_base::failbit;
8330b57cec5SDimitry Andric      break;
8340b57cec5SDimitry Andric    }
8350b57cec5SDimitry Andric    return __b;
8360b57cec5SDimitry Andric  }
837bdd1243dSDimitry Andric  const ctype<_CharT>& __ct    = std::use_facet<ctype<_CharT> >(__iob.getloc());
838bdd1243dSDimitry Andric  const numpunct<_CharT>& __np = std::use_facet<numpunct<_CharT> >(__iob.getloc());
8390b57cec5SDimitry Andric  typedef typename numpunct<_CharT>::string_type string_type;
8400b57cec5SDimitry Andric  const string_type __names[2] = {__np.truename(), __np.falsename()};
841cb14a3feSDimitry Andric  const string_type* __i       = std::__scan_keyword(__b, __e, __names, __names + 2, __ct, __err);
8420b57cec5SDimitry Andric  __v                          = __i == __names;
8430b57cec5SDimitry Andric  return __b;
8440b57cec5SDimitry Andric}
8450b57cec5SDimitry Andric
8460b57cec5SDimitry Andric// signed
8470b57cec5SDimitry Andric
8480b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
8490b57cec5SDimitry Andrictemplate <class _Signed>
850cb14a3feSDimitry Andric_InputIterator num_get<_CharT, _InputIterator>::__do_get_signed(
851cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, _Signed& __v) const {
8520b57cec5SDimitry Andric  // Stage 1
8530b57cec5SDimitry Andric  int __base = this->__get_base(__iob);
8540b57cec5SDimitry Andric  // Stage 2
8550b57cec5SDimitry Andric  char_type __thousands_sep;
856*0fca6ea1SDimitry Andric  const int __atoms_size = __num_get_base::__int_chr_cnt;
8570b57cec5SDimitry Andric#  ifdef _LIBCPP_ABI_OPTIMIZED_LOCALE_NUM_GET
8580b57cec5SDimitry Andric  char_type __atoms1[__atoms_size];
8590b57cec5SDimitry Andric  const char_type* __atoms = this->__do_widen(__iob, __atoms1);
8600b57cec5SDimitry Andric  string __grouping        = this->__stage2_int_prep(__iob, __thousands_sep);
8610b57cec5SDimitry Andric#  else
8620b57cec5SDimitry Andric  char_type __atoms[__atoms_size];
8630b57cec5SDimitry Andric  string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
8640b57cec5SDimitry Andric#  endif
8650b57cec5SDimitry Andric  string __buf;
8660b57cec5SDimitry Andric  __buf.resize(__buf.capacity());
8670b57cec5SDimitry Andric  char* __a     = &__buf[0];
8680b57cec5SDimitry Andric  char* __a_end = __a;
8690b57cec5SDimitry Andric  unsigned __g[__num_get_base::__num_get_buf_sz];
8700b57cec5SDimitry Andric  unsigned* __g_end = __g;
8710b57cec5SDimitry Andric  unsigned __dc     = 0;
872cb14a3feSDimitry Andric  for (; __b != __e; ++__b) {
873cb14a3feSDimitry Andric    if (__a_end == __a + __buf.size()) {
8740b57cec5SDimitry Andric      size_t __tmp = __buf.size();
8750b57cec5SDimitry Andric      __buf.resize(2 * __buf.size());
8760b57cec5SDimitry Andric      __buf.resize(__buf.capacity());
8770b57cec5SDimitry Andric      __a     = &__buf[0];
8780b57cec5SDimitry Andric      __a_end = __a + __tmp;
8790b57cec5SDimitry Andric    }
880cb14a3feSDimitry Andric    if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc, __thousands_sep, __grouping, __g, __g_end, __atoms))
8810b57cec5SDimitry Andric      break;
8820b57cec5SDimitry Andric  }
8830b57cec5SDimitry Andric  if (__grouping.size() != 0 && __g_end - __g < __num_get_base::__num_get_buf_sz)
8840b57cec5SDimitry Andric    *__g_end++ = __dc;
8850b57cec5SDimitry Andric  // Stage 3
886bdd1243dSDimitry Andric  __v = std::__num_get_signed_integral<_Signed>(__a, __a_end, __err, __base);
8870b57cec5SDimitry Andric  // Digit grouping checked
8880b57cec5SDimitry Andric  __check_grouping(__grouping, __g, __g_end, __err);
8890b57cec5SDimitry Andric  // EOF checked
8900b57cec5SDimitry Andric  if (__b == __e)
8910b57cec5SDimitry Andric    __err |= ios_base::eofbit;
8920b57cec5SDimitry Andric  return __b;
8930b57cec5SDimitry Andric}
8940b57cec5SDimitry Andric
8950b57cec5SDimitry Andric// unsigned
8960b57cec5SDimitry Andric
8970b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
8980b57cec5SDimitry Andrictemplate <class _Unsigned>
899cb14a3feSDimitry Andric_InputIterator num_get<_CharT, _InputIterator>::__do_get_unsigned(
900cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, _Unsigned& __v) const {
9010b57cec5SDimitry Andric  // Stage 1
9020b57cec5SDimitry Andric  int __base = this->__get_base(__iob);
9030b57cec5SDimitry Andric  // Stage 2
9040b57cec5SDimitry Andric  char_type __thousands_sep;
905*0fca6ea1SDimitry Andric  const int __atoms_size = __num_get_base::__int_chr_cnt;
9060b57cec5SDimitry Andric#  ifdef _LIBCPP_ABI_OPTIMIZED_LOCALE_NUM_GET
9070b57cec5SDimitry Andric  char_type __atoms1[__atoms_size];
9080b57cec5SDimitry Andric  const char_type* __atoms = this->__do_widen(__iob, __atoms1);
9090b57cec5SDimitry Andric  string __grouping        = this->__stage2_int_prep(__iob, __thousands_sep);
9100b57cec5SDimitry Andric#  else
9110b57cec5SDimitry Andric  char_type __atoms[__atoms_size];
9120b57cec5SDimitry Andric  string __grouping = this->__stage2_int_prep(__iob, __atoms, __thousands_sep);
9130b57cec5SDimitry Andric#  endif
9140b57cec5SDimitry Andric  string __buf;
9150b57cec5SDimitry Andric  __buf.resize(__buf.capacity());
9160b57cec5SDimitry Andric  char* __a     = &__buf[0];
9170b57cec5SDimitry Andric  char* __a_end = __a;
9180b57cec5SDimitry Andric  unsigned __g[__num_get_base::__num_get_buf_sz];
9190b57cec5SDimitry Andric  unsigned* __g_end = __g;
9200b57cec5SDimitry Andric  unsigned __dc     = 0;
921cb14a3feSDimitry Andric  for (; __b != __e; ++__b) {
922cb14a3feSDimitry Andric    if (__a_end == __a + __buf.size()) {
9230b57cec5SDimitry Andric      size_t __tmp = __buf.size();
9240b57cec5SDimitry Andric      __buf.resize(2 * __buf.size());
9250b57cec5SDimitry Andric      __buf.resize(__buf.capacity());
9260b57cec5SDimitry Andric      __a     = &__buf[0];
9270b57cec5SDimitry Andric      __a_end = __a + __tmp;
9280b57cec5SDimitry Andric    }
929cb14a3feSDimitry Andric    if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc, __thousands_sep, __grouping, __g, __g_end, __atoms))
9300b57cec5SDimitry Andric      break;
9310b57cec5SDimitry Andric  }
9320b57cec5SDimitry Andric  if (__grouping.size() != 0 && __g_end - __g < __num_get_base::__num_get_buf_sz)
9330b57cec5SDimitry Andric    *__g_end++ = __dc;
9340b57cec5SDimitry Andric  // Stage 3
935bdd1243dSDimitry Andric  __v = std::__num_get_unsigned_integral<_Unsigned>(__a, __a_end, __err, __base);
9360b57cec5SDimitry Andric  // Digit grouping checked
9370b57cec5SDimitry Andric  __check_grouping(__grouping, __g, __g_end, __err);
9380b57cec5SDimitry Andric  // EOF checked
9390b57cec5SDimitry Andric  if (__b == __e)
9400b57cec5SDimitry Andric    __err |= ios_base::eofbit;
9410b57cec5SDimitry Andric  return __b;
9420b57cec5SDimitry Andric}
9430b57cec5SDimitry Andric
9440b57cec5SDimitry Andric// floating point
9450b57cec5SDimitry Andric
9460b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
9470b57cec5SDimitry Andrictemplate <class _Fp>
948cb14a3feSDimitry Andric_InputIterator num_get<_CharT, _InputIterator>::__do_get_floating_point(
949cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, _Fp& __v) const {
9500b57cec5SDimitry Andric  // Stage 1, nothing to do
9510b57cec5SDimitry Andric  // Stage 2
952*0fca6ea1SDimitry Andric  char_type __atoms[__num_get_base::__fp_chr_cnt];
9530b57cec5SDimitry Andric  char_type __decimal_point;
9540b57cec5SDimitry Andric  char_type __thousands_sep;
955cb14a3feSDimitry Andric  string __grouping = this->__stage2_float_prep(__iob, __atoms, __decimal_point, __thousands_sep);
9560b57cec5SDimitry Andric  string __buf;
9570b57cec5SDimitry Andric  __buf.resize(__buf.capacity());
9580b57cec5SDimitry Andric  char* __a     = &__buf[0];
9590b57cec5SDimitry Andric  char* __a_end = __a;
9600b57cec5SDimitry Andric  unsigned __g[__num_get_base::__num_get_buf_sz];
9610b57cec5SDimitry Andric  unsigned* __g_end        = __g;
9620b57cec5SDimitry Andric  unsigned __dc            = 0;
9630b57cec5SDimitry Andric  bool __in_units          = true;
9640b57cec5SDimitry Andric  char __exp               = 'E';
965*0fca6ea1SDimitry Andric  bool __is_leading_parsed = false;
966cb14a3feSDimitry Andric  for (; __b != __e; ++__b) {
967cb14a3feSDimitry Andric    if (__a_end == __a + __buf.size()) {
9680b57cec5SDimitry Andric      size_t __tmp = __buf.size();
9690b57cec5SDimitry Andric      __buf.resize(2 * __buf.size());
9700b57cec5SDimitry Andric      __buf.resize(__buf.capacity());
9710b57cec5SDimitry Andric      __a     = &__buf[0];
9720b57cec5SDimitry Andric      __a_end = __a + __tmp;
9730b57cec5SDimitry Andric    }
974cb14a3feSDimitry Andric    if (this->__stage2_float_loop(
975cb14a3feSDimitry Andric            *__b,
976cb14a3feSDimitry Andric            __in_units,
977cb14a3feSDimitry Andric            __exp,
978cb14a3feSDimitry Andric            __a,
979cb14a3feSDimitry Andric            __a_end,
980cb14a3feSDimitry Andric            __decimal_point,
981cb14a3feSDimitry Andric            __thousands_sep,
982cb14a3feSDimitry Andric            __grouping,
983cb14a3feSDimitry Andric            __g,
984cb14a3feSDimitry Andric            __g_end,
985cb14a3feSDimitry Andric            __dc,
986cb14a3feSDimitry Andric            __atoms))
9870b57cec5SDimitry Andric      break;
988*0fca6ea1SDimitry Andric
989*0fca6ea1SDimitry Andric    // the leading character excluding the sign must be a decimal digit
990*0fca6ea1SDimitry Andric    if (!__is_leading_parsed) {
991*0fca6ea1SDimitry Andric      if (__a_end - __a >= 1 && __a[0] != '-' && __a[0] != '+') {
992*0fca6ea1SDimitry Andric        if (('0' <= __a[0] && __a[0] <= '9') || __a[0] == '.')
993*0fca6ea1SDimitry Andric          __is_leading_parsed = true;
994*0fca6ea1SDimitry Andric        else
995*0fca6ea1SDimitry Andric          break;
996*0fca6ea1SDimitry Andric      } else if (__a_end - __a >= 2 && (__a[0] == '-' || __a[0] == '+')) {
997*0fca6ea1SDimitry Andric        if (('0' <= __a[1] && __a[1] <= '9') || __a[1] == '.')
998*0fca6ea1SDimitry Andric          __is_leading_parsed = true;
999*0fca6ea1SDimitry Andric        else
1000*0fca6ea1SDimitry Andric          break;
1001*0fca6ea1SDimitry Andric      }
1002*0fca6ea1SDimitry Andric    }
10030b57cec5SDimitry Andric  }
10040b57cec5SDimitry Andric  if (__grouping.size() != 0 && __in_units && __g_end - __g < __num_get_base::__num_get_buf_sz)
10050b57cec5SDimitry Andric    *__g_end++ = __dc;
10060b57cec5SDimitry Andric  // Stage 3
1007bdd1243dSDimitry Andric  __v = std::__num_get_float<_Fp>(__a, __a_end, __err);
10080b57cec5SDimitry Andric  // Digit grouping checked
10090b57cec5SDimitry Andric  __check_grouping(__grouping, __g, __g_end, __err);
10100b57cec5SDimitry Andric  // EOF checked
10110b57cec5SDimitry Andric  if (__b == __e)
10120b57cec5SDimitry Andric    __err |= ios_base::eofbit;
10130b57cec5SDimitry Andric  return __b;
10140b57cec5SDimitry Andric}
10150b57cec5SDimitry Andric
10160b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1017cb14a3feSDimitry Andric_InputIterator num_get<_CharT, _InputIterator>::do_get(
1018cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, void*& __v) const {
10190b57cec5SDimitry Andric  // Stage 1
10200b57cec5SDimitry Andric  int __base = 16;
10210b57cec5SDimitry Andric  // Stage 2
1022*0fca6ea1SDimitry Andric  char_type __atoms[__num_get_base::__int_chr_cnt];
10235f757f3fSDimitry Andric  char_type __thousands_sep = char_type();
10240b57cec5SDimitry Andric  string __grouping;
1025*0fca6ea1SDimitry Andric  std::use_facet<ctype<_CharT> >(__iob.getloc())
1026*0fca6ea1SDimitry Andric      .widen(__num_get_base::__src, __num_get_base::__src + __num_get_base::__int_chr_cnt, __atoms);
10270b57cec5SDimitry Andric  string __buf;
10280b57cec5SDimitry Andric  __buf.resize(__buf.capacity());
10290b57cec5SDimitry Andric  char* __a     = &__buf[0];
10300b57cec5SDimitry Andric  char* __a_end = __a;
10310b57cec5SDimitry Andric  unsigned __g[__num_get_base::__num_get_buf_sz];
10320b57cec5SDimitry Andric  unsigned* __g_end = __g;
10330b57cec5SDimitry Andric  unsigned __dc     = 0;
1034cb14a3feSDimitry Andric  for (; __b != __e; ++__b) {
1035cb14a3feSDimitry Andric    if (__a_end == __a + __buf.size()) {
10360b57cec5SDimitry Andric      size_t __tmp = __buf.size();
10370b57cec5SDimitry Andric      __buf.resize(2 * __buf.size());
10380b57cec5SDimitry Andric      __buf.resize(__buf.capacity());
10390b57cec5SDimitry Andric      __a     = &__buf[0];
10400b57cec5SDimitry Andric      __a_end = __a + __tmp;
10410b57cec5SDimitry Andric    }
1042cb14a3feSDimitry Andric    if (this->__stage2_int_loop(*__b, __base, __a, __a_end, __dc, __thousands_sep, __grouping, __g, __g_end, __atoms))
10430b57cec5SDimitry Andric      break;
10440b57cec5SDimitry Andric  }
10450b57cec5SDimitry Andric  // Stage 3
10460b57cec5SDimitry Andric  __buf.resize(__a_end - __a);
10470b57cec5SDimitry Andric  if (__libcpp_sscanf_l(__buf.c_str(), _LIBCPP_GET_C_LOCALE, "%p", &__v) != 1)
10480b57cec5SDimitry Andric    __err = ios_base::failbit;
10490b57cec5SDimitry Andric  // EOF checked
10500b57cec5SDimitry Andric  if (__b == __e)
10510b57cec5SDimitry Andric    __err |= ios_base::eofbit;
10520b57cec5SDimitry Andric  return __b;
10530b57cec5SDimitry Andric}
10540b57cec5SDimitry Andric
105581ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS num_get<char>;
1056349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
105781ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS num_get<wchar_t>;
1058349cc55cSDimitry Andric#  endif
10590b57cec5SDimitry Andric
1060cb14a3feSDimitry Andricstruct _LIBCPP_EXPORTED_FROM_ABI __num_put_base {
10610b57cec5SDimitry Andricprotected:
1062cb14a3feSDimitry Andric  static void __format_int(char* __fmt, const char* __len, bool __signd, ios_base::fmtflags __flags);
1063cb14a3feSDimitry Andric  static bool __format_float(char* __fmt, const char* __len, ios_base::fmtflags __flags);
1064cb14a3feSDimitry Andric  static char* __identify_padding(char* __nb, char* __ne, const ios_base& __iob);
10650b57cec5SDimitry Andric};
10660b57cec5SDimitry Andric
10670b57cec5SDimitry Andrictemplate <class _CharT>
1068cb14a3feSDimitry Andricstruct __num_put : protected __num_put_base {
1069cb14a3feSDimitry Andric  static void __widen_and_group_int(
1070cb14a3feSDimitry Andric      char* __nb, char* __np, char* __ne, _CharT* __ob, _CharT*& __op, _CharT*& __oe, const locale& __loc);
1071cb14a3feSDimitry Andric  static void __widen_and_group_float(
1072cb14a3feSDimitry Andric      char* __nb, char* __np, char* __ne, _CharT* __ob, _CharT*& __op, _CharT*& __oe, const locale& __loc);
10730b57cec5SDimitry Andric};
10740b57cec5SDimitry Andric
10750b57cec5SDimitry Andrictemplate <class _CharT>
1076cb14a3feSDimitry Andricvoid __num_put<_CharT>::__widen_and_group_int(
1077cb14a3feSDimitry Andric    char* __nb, char* __np, char* __ne, _CharT* __ob, _CharT*& __op, _CharT*& __oe, const locale& __loc) {
1078bdd1243dSDimitry Andric  const ctype<_CharT>& __ct     = std::use_facet<ctype<_CharT> >(__loc);
1079bdd1243dSDimitry Andric  const numpunct<_CharT>& __npt = std::use_facet<numpunct<_CharT> >(__loc);
10800b57cec5SDimitry Andric  string __grouping             = __npt.grouping();
1081cb14a3feSDimitry Andric  if (__grouping.empty()) {
10820b57cec5SDimitry Andric    __ct.widen(__nb, __ne, __ob);
10830b57cec5SDimitry Andric    __oe = __ob + (__ne - __nb);
1084cb14a3feSDimitry Andric  } else {
10850b57cec5SDimitry Andric    __oe       = __ob;
10860b57cec5SDimitry Andric    char* __nf = __nb;
10870b57cec5SDimitry Andric    if (*__nf == '-' || *__nf == '+')
10880b57cec5SDimitry Andric      *__oe++ = __ct.widen(*__nf++);
1089cb14a3feSDimitry Andric    if (__ne - __nf >= 2 && __nf[0] == '0' && (__nf[1] == 'x' || __nf[1] == 'X')) {
10900b57cec5SDimitry Andric      *__oe++ = __ct.widen(*__nf++);
10910b57cec5SDimitry Andric      *__oe++ = __ct.widen(*__nf++);
10920b57cec5SDimitry Andric    }
1093bdd1243dSDimitry Andric    std::reverse(__nf, __ne);
10940b57cec5SDimitry Andric    _CharT __thousands_sep = __npt.thousands_sep();
10950b57cec5SDimitry Andric    unsigned __dc          = 0;
10960b57cec5SDimitry Andric    unsigned __dg          = 0;
1097cb14a3feSDimitry Andric    for (char* __p = __nf; __p < __ne; ++__p) {
1098cb14a3feSDimitry Andric      if (static_cast<unsigned>(__grouping[__dg]) > 0 && __dc == static_cast<unsigned>(__grouping[__dg])) {
10990b57cec5SDimitry Andric        *__oe++ = __thousands_sep;
11000b57cec5SDimitry Andric        __dc    = 0;
11010b57cec5SDimitry Andric        if (__dg < __grouping.size() - 1)
11020b57cec5SDimitry Andric          ++__dg;
11030b57cec5SDimitry Andric      }
11040b57cec5SDimitry Andric      *__oe++ = __ct.widen(*__p);
11050b57cec5SDimitry Andric      ++__dc;
11060b57cec5SDimitry Andric    }
1107bdd1243dSDimitry Andric    std::reverse(__ob + (__nf - __nb), __oe);
11080b57cec5SDimitry Andric  }
11090b57cec5SDimitry Andric  if (__np == __ne)
11100b57cec5SDimitry Andric    __op = __oe;
11110b57cec5SDimitry Andric  else
11120b57cec5SDimitry Andric    __op = __ob + (__np - __nb);
11130b57cec5SDimitry Andric}
11140b57cec5SDimitry Andric
11150b57cec5SDimitry Andrictemplate <class _CharT>
1116cb14a3feSDimitry Andricvoid __num_put<_CharT>::__widen_and_group_float(
1117cb14a3feSDimitry Andric    char* __nb, char* __np, char* __ne, _CharT* __ob, _CharT*& __op, _CharT*& __oe, const locale& __loc) {
1118bdd1243dSDimitry Andric  const ctype<_CharT>& __ct     = std::use_facet<ctype<_CharT> >(__loc);
1119bdd1243dSDimitry Andric  const numpunct<_CharT>& __npt = std::use_facet<numpunct<_CharT> >(__loc);
11200b57cec5SDimitry Andric  string __grouping             = __npt.grouping();
11210b57cec5SDimitry Andric  __oe                          = __ob;
11220b57cec5SDimitry Andric  char* __nf                    = __nb;
11230b57cec5SDimitry Andric  if (*__nf == '-' || *__nf == '+')
11240b57cec5SDimitry Andric    *__oe++ = __ct.widen(*__nf++);
11250b57cec5SDimitry Andric  char* __ns;
1126cb14a3feSDimitry Andric  if (__ne - __nf >= 2 && __nf[0] == '0' && (__nf[1] == 'x' || __nf[1] == 'X')) {
11270b57cec5SDimitry Andric    *__oe++ = __ct.widen(*__nf++);
11280b57cec5SDimitry Andric    *__oe++ = __ct.widen(*__nf++);
11290b57cec5SDimitry Andric    for (__ns = __nf; __ns < __ne; ++__ns)
11300b57cec5SDimitry Andric      if (!isxdigit_l(*__ns, _LIBCPP_GET_C_LOCALE))
11310b57cec5SDimitry Andric        break;
1132cb14a3feSDimitry Andric  } else {
11330b57cec5SDimitry Andric    for (__ns = __nf; __ns < __ne; ++__ns)
11340b57cec5SDimitry Andric      if (!isdigit_l(*__ns, _LIBCPP_GET_C_LOCALE))
11350b57cec5SDimitry Andric        break;
11360b57cec5SDimitry Andric  }
1137cb14a3feSDimitry Andric  if (__grouping.empty()) {
11380b57cec5SDimitry Andric    __ct.widen(__nf, __ns, __oe);
11390b57cec5SDimitry Andric    __oe += __ns - __nf;
1140cb14a3feSDimitry Andric  } else {
1141bdd1243dSDimitry Andric    std::reverse(__nf, __ns);
11420b57cec5SDimitry Andric    _CharT __thousands_sep = __npt.thousands_sep();
11430b57cec5SDimitry Andric    unsigned __dc          = 0;
11440b57cec5SDimitry Andric    unsigned __dg          = 0;
1145cb14a3feSDimitry Andric    for (char* __p = __nf; __p < __ns; ++__p) {
1146cb14a3feSDimitry Andric      if (__grouping[__dg] > 0 && __dc == static_cast<unsigned>(__grouping[__dg])) {
11470b57cec5SDimitry Andric        *__oe++ = __thousands_sep;
11480b57cec5SDimitry Andric        __dc    = 0;
11490b57cec5SDimitry Andric        if (__dg < __grouping.size() - 1)
11500b57cec5SDimitry Andric          ++__dg;
11510b57cec5SDimitry Andric      }
11520b57cec5SDimitry Andric      *__oe++ = __ct.widen(*__p);
11530b57cec5SDimitry Andric      ++__dc;
11540b57cec5SDimitry Andric    }
1155bdd1243dSDimitry Andric    std::reverse(__ob + (__nf - __nb), __oe);
11560b57cec5SDimitry Andric  }
1157cb14a3feSDimitry Andric  for (__nf = __ns; __nf < __ne; ++__nf) {
1158cb14a3feSDimitry Andric    if (*__nf == '.') {
11590b57cec5SDimitry Andric      *__oe++ = __npt.decimal_point();
11600b57cec5SDimitry Andric      ++__nf;
11610b57cec5SDimitry Andric      break;
1162cb14a3feSDimitry Andric    } else
11630b57cec5SDimitry Andric      *__oe++ = __ct.widen(*__nf);
11640b57cec5SDimitry Andric  }
11650b57cec5SDimitry Andric  __ct.widen(__nf, __ne, __oe);
11660b57cec5SDimitry Andric  __oe += __ne - __nf;
11670b57cec5SDimitry Andric  if (__np == __ne)
11680b57cec5SDimitry Andric    __op = __oe;
11690b57cec5SDimitry Andric  else
11700b57cec5SDimitry Andric    __op = __ob + (__np - __nb);
11710b57cec5SDimitry Andric}
11720b57cec5SDimitry Andric
117381ad6265SDimitry Andricextern template struct _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __num_put<char>;
1174349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
117581ad6265SDimitry Andricextern template struct _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __num_put<wchar_t>;
1176349cc55cSDimitry Andric#  endif
11770b57cec5SDimitry Andric
11780b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
1179cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS num_put : public locale::facet, private __num_put<_CharT> {
11800b57cec5SDimitry Andricpublic:
11810b57cec5SDimitry Andric  typedef _CharT char_type;
11820b57cec5SDimitry Andric  typedef _OutputIterator iter_type;
11830b57cec5SDimitry Andric
1184cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit num_put(size_t __refs = 0) : locale::facet(__refs) {}
11850b57cec5SDimitry Andric
1186cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, bool __v) const {
11870b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
11880b57cec5SDimitry Andric  }
11890b57cec5SDimitry Andric
1190cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, long __v) const {
11910b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
11920b57cec5SDimitry Andric  }
11930b57cec5SDimitry Andric
1194cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, long long __v) const {
11950b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
11960b57cec5SDimitry Andric  }
11970b57cec5SDimitry Andric
1198cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, unsigned long __v) const {
11990b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
12000b57cec5SDimitry Andric  }
12010b57cec5SDimitry Andric
1202cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, unsigned long long __v) const {
12030b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
12040b57cec5SDimitry Andric  }
12050b57cec5SDimitry Andric
1206cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, double __v) const {
12070b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
12080b57cec5SDimitry Andric  }
12090b57cec5SDimitry Andric
1210cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, long double __v) const {
12110b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
12120b57cec5SDimitry Andric  }
12130b57cec5SDimitry Andric
1214cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type put(iter_type __s, ios_base& __iob, char_type __fl, const void* __v) const {
12150b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __v);
12160b57cec5SDimitry Andric  }
12170b57cec5SDimitry Andric
12180b57cec5SDimitry Andric  static locale::id id;
12190b57cec5SDimitry Andric
12200b57cec5SDimitry Andricprotected:
1221bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~num_put() override {}
12220b57cec5SDimitry Andric
1223cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, bool __v) const;
1224cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, long __v) const;
1225cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, long long __v) const;
1226cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, unsigned long) const;
1227cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, unsigned long long) const;
1228cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, double __v) const;
1229cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, long double __v) const;
1230cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base& __iob, char_type __fl, const void* __v) const;
1231349cc55cSDimitry Andric
1232349cc55cSDimitry Andric  template <class _Integral>
1233cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI inline _OutputIterator
1234cb14a3feSDimitry Andric  __do_put_integral(iter_type __s, ios_base& __iob, char_type __fl, _Integral __v, char const* __len) const;
1235349cc55cSDimitry Andric
1236349cc55cSDimitry Andric  template <class _Float>
1237cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI inline _OutputIterator
1238cb14a3feSDimitry Andric  __do_put_floating_point(iter_type __s, ios_base& __iob, char_type __fl, _Float __v, char const* __len) const;
12390b57cec5SDimitry Andric};
12400b57cec5SDimitry Andric
12410b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
1242cb14a3feSDimitry Andriclocale::id num_put<_CharT, _OutputIterator>::id;
12430b57cec5SDimitry Andric
12440b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
1245cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI _OutputIterator __pad_and_output(
1246cb14a3feSDimitry Andric    _OutputIterator __s, const _CharT* __ob, const _CharT* __op, const _CharT* __oe, ios_base& __iob, _CharT __fl) {
12470b57cec5SDimitry Andric  streamsize __sz = __oe - __ob;
12480b57cec5SDimitry Andric  streamsize __ns = __iob.width();
12490b57cec5SDimitry Andric  if (__ns > __sz)
12500b57cec5SDimitry Andric    __ns -= __sz;
12510b57cec5SDimitry Andric  else
12520b57cec5SDimitry Andric    __ns = 0;
12530b57cec5SDimitry Andric  for (; __ob < __op; ++__ob, ++__s)
12540b57cec5SDimitry Andric    *__s = *__ob;
12550b57cec5SDimitry Andric  for (; __ns; --__ns, ++__s)
12560b57cec5SDimitry Andric    *__s = __fl;
12570b57cec5SDimitry Andric  for (; __ob < __oe; ++__ob, ++__s)
12580b57cec5SDimitry Andric    *__s = *__ob;
12590b57cec5SDimitry Andric  __iob.width(0);
12600b57cec5SDimitry Andric  return __s;
12610b57cec5SDimitry Andric}
12620b57cec5SDimitry Andric
12630b57cec5SDimitry Andrictemplate <class _CharT, class _Traits>
1264cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI ostreambuf_iterator<_CharT, _Traits> __pad_and_output(
1265cb14a3feSDimitry Andric    ostreambuf_iterator<_CharT, _Traits> __s,
1266cb14a3feSDimitry Andric    const _CharT* __ob,
1267cb14a3feSDimitry Andric    const _CharT* __op,
1268cb14a3feSDimitry Andric    const _CharT* __oe,
1269cb14a3feSDimitry Andric    ios_base& __iob,
1270cb14a3feSDimitry Andric    _CharT __fl) {
12710b57cec5SDimitry Andric  if (__s.__sbuf_ == nullptr)
12720b57cec5SDimitry Andric    return __s;
12730b57cec5SDimitry Andric  streamsize __sz = __oe - __ob;
12740b57cec5SDimitry Andric  streamsize __ns = __iob.width();
12750b57cec5SDimitry Andric  if (__ns > __sz)
12760b57cec5SDimitry Andric    __ns -= __sz;
12770b57cec5SDimitry Andric  else
12780b57cec5SDimitry Andric    __ns = 0;
12790b57cec5SDimitry Andric  streamsize __np = __op - __ob;
1280cb14a3feSDimitry Andric  if (__np > 0) {
1281cb14a3feSDimitry Andric    if (__s.__sbuf_->sputn(__ob, __np) != __np) {
12820b57cec5SDimitry Andric      __s.__sbuf_ = nullptr;
12830b57cec5SDimitry Andric      return __s;
12840b57cec5SDimitry Andric    }
12850b57cec5SDimitry Andric  }
1286cb14a3feSDimitry Andric  if (__ns > 0) {
12870b57cec5SDimitry Andric    basic_string<_CharT, _Traits> __sp(__ns, __fl);
1288cb14a3feSDimitry Andric    if (__s.__sbuf_->sputn(__sp.data(), __ns) != __ns) {
12890b57cec5SDimitry Andric      __s.__sbuf_ = nullptr;
12900b57cec5SDimitry Andric      return __s;
12910b57cec5SDimitry Andric    }
12920b57cec5SDimitry Andric  }
12930b57cec5SDimitry Andric  __np = __oe - __op;
1294cb14a3feSDimitry Andric  if (__np > 0) {
1295cb14a3feSDimitry Andric    if (__s.__sbuf_->sputn(__op, __np) != __np) {
12960b57cec5SDimitry Andric      __s.__sbuf_ = nullptr;
12970b57cec5SDimitry Andric      return __s;
12980b57cec5SDimitry Andric    }
12990b57cec5SDimitry Andric  }
13000b57cec5SDimitry Andric  __iob.width(0);
13010b57cec5SDimitry Andric  return __s;
13020b57cec5SDimitry Andric}
13030b57cec5SDimitry Andric
13040b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
13050b57cec5SDimitry Andric_OutputIterator
1306cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, bool __v) const {
13070b57cec5SDimitry Andric  if ((__iob.flags() & ios_base::boolalpha) == 0)
13080b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, (unsigned long)__v);
1309bdd1243dSDimitry Andric  const numpunct<char_type>& __np = std::use_facet<numpunct<char_type> >(__iob.getloc());
13100b57cec5SDimitry Andric  typedef typename numpunct<char_type>::string_type string_type;
13110b57cec5SDimitry Andric  string_type __nm = __v ? __np.truename() : __np.falsename();
13120b57cec5SDimitry Andric  for (typename string_type::iterator __i = __nm.begin(); __i != __nm.end(); ++__i, ++__s)
13130b57cec5SDimitry Andric    *__s = *__i;
13140b57cec5SDimitry Andric  return __s;
13150b57cec5SDimitry Andric}
13160b57cec5SDimitry Andric
13170b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
1318349cc55cSDimitry Andrictemplate <class _Integral>
1319cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI inline _OutputIterator num_put<_CharT, _OutputIterator>::__do_put_integral(
1320cb14a3feSDimitry Andric    iter_type __s, ios_base& __iob, char_type __fl, _Integral __v, char const* __len) const {
13210b57cec5SDimitry Andric  // Stage 1 - Get number in narrow char
1322349cc55cSDimitry Andric  char __fmt[8] = {'%', 0};
1323349cc55cSDimitry Andric  this->__format_int(__fmt + 1, __len, is_signed<_Integral>::value, __iob.flags());
1324fe6060f1SDimitry Andric  // Worst case is octal, with showbase enabled. Note that octal is always
1325fe6060f1SDimitry Andric  // printed as an unsigned value.
1326349cc55cSDimitry Andric  using _Unsigned = typename make_unsigned<_Integral>::type;
1327cb14a3feSDimitry Andric  _LIBCPP_CONSTEXPR const unsigned __nbuf =
1328cb14a3feSDimitry Andric      (numeric_limits<_Unsigned>::digits / 3)          // 1 char per 3 bits
1329349cc55cSDimitry Andric      + ((numeric_limits<_Unsigned>::digits % 3) != 0) // round up
1330fe6060f1SDimitry Andric      + 2;                                             // base prefix + terminating null character
13310b57cec5SDimitry Andric  char __nar[__nbuf];
133281ad6265SDimitry Andric  _LIBCPP_DIAGNOSTIC_PUSH
133381ad6265SDimitry Andric  _LIBCPP_CLANG_DIAGNOSTIC_IGNORED("-Wformat-nonliteral")
133481ad6265SDimitry Andric  _LIBCPP_GCC_DIAGNOSTIC_IGNORED("-Wformat-nonliteral")
13350b57cec5SDimitry Andric  int __nc = __libcpp_snprintf_l(__nar, sizeof(__nar), _LIBCPP_GET_C_LOCALE, __fmt, __v);
133681ad6265SDimitry Andric  _LIBCPP_DIAGNOSTIC_POP
13370b57cec5SDimitry Andric  char* __ne = __nar + __nc;
13380b57cec5SDimitry Andric  char* __np = this->__identify_padding(__nar, __ne, __iob);
13390b57cec5SDimitry Andric  // Stage 2 - Widen __nar while adding thousands separators
13400b57cec5SDimitry Andric  char_type __o[2 * (__nbuf - 1) - 1];
13410b57cec5SDimitry Andric  char_type* __op; // pad here
13420b57cec5SDimitry Andric  char_type* __oe; // end of output
13430b57cec5SDimitry Andric  this->__widen_and_group_int(__nar, __np, __ne, __o, __op, __oe, __iob.getloc());
13440b57cec5SDimitry Andric  // [__o, __oe) contains thousands_sep'd wide number
13450b57cec5SDimitry Andric  // Stage 3 & 4
1346bdd1243dSDimitry Andric  return std::__pad_and_output(__s, __o, __op, __oe, __iob, __fl);
13470b57cec5SDimitry Andric}
13480b57cec5SDimitry Andric
13490b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
13500b57cec5SDimitry Andric_OutputIterator
1351cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, long __v) const {
1352349cc55cSDimitry Andric  return this->__do_put_integral(__s, __iob, __fl, __v, "l");
1353349cc55cSDimitry Andric}
1354349cc55cSDimitry Andric
1355349cc55cSDimitry Andrictemplate <class _CharT, class _OutputIterator>
1356349cc55cSDimitry Andric_OutputIterator
1357cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, long long __v) const {
1358349cc55cSDimitry Andric  return this->__do_put_integral(__s, __iob, __fl, __v, "ll");
13590b57cec5SDimitry Andric}
13600b57cec5SDimitry Andric
13610b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
13620b57cec5SDimitry Andric_OutputIterator
1363cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, unsigned long __v) const {
1364349cc55cSDimitry Andric  return this->__do_put_integral(__s, __iob, __fl, __v, "l");
13650b57cec5SDimitry Andric}
13660b57cec5SDimitry Andric
13670b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
13680b57cec5SDimitry Andric_OutputIterator
1369cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, unsigned long long __v) const {
1370349cc55cSDimitry Andric  return this->__do_put_integral(__s, __iob, __fl, __v, "ll");
1371349cc55cSDimitry Andric}
1372349cc55cSDimitry Andric
1373349cc55cSDimitry Andrictemplate <class _CharT, class _OutputIterator>
1374349cc55cSDimitry Andrictemplate <class _Float>
1375cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI inline _OutputIterator num_put<_CharT, _OutputIterator>::__do_put_floating_point(
1376cb14a3feSDimitry Andric    iter_type __s, ios_base& __iob, char_type __fl, _Float __v, char const* __len) const {
13770b57cec5SDimitry Andric  // Stage 1 - Get number in narrow char
13780b57cec5SDimitry Andric  char __fmt[8]            = {'%', 0};
1379349cc55cSDimitry Andric  bool __specify_precision = this->__format_float(__fmt + 1, __len, __iob.flags());
1380349cc55cSDimitry Andric  const unsigned __nbuf    = 30;
13810b57cec5SDimitry Andric  char __nar[__nbuf];
1382349cc55cSDimitry Andric  char* __nb = __nar;
1383349cc55cSDimitry Andric  int __nc;
138481ad6265SDimitry Andric  _LIBCPP_DIAGNOSTIC_PUSH
138581ad6265SDimitry Andric  _LIBCPP_CLANG_DIAGNOSTIC_IGNORED("-Wformat-nonliteral")
138681ad6265SDimitry Andric  _LIBCPP_GCC_DIAGNOSTIC_IGNORED("-Wformat-nonliteral")
1387349cc55cSDimitry Andric  if (__specify_precision)
1388cb14a3feSDimitry Andric    __nc = __libcpp_snprintf_l(__nb, __nbuf, _LIBCPP_GET_C_LOCALE, __fmt, (int)__iob.precision(), __v);
1389349cc55cSDimitry Andric  else
1390349cc55cSDimitry Andric    __nc = __libcpp_snprintf_l(__nb, __nbuf, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1391349cc55cSDimitry Andric  unique_ptr<char, void (*)(void*)> __nbh(nullptr, free);
1392cb14a3feSDimitry Andric  if (__nc > static_cast<int>(__nbuf - 1)) {
1393349cc55cSDimitry Andric    if (__specify_precision)
1394349cc55cSDimitry Andric      __nc = __libcpp_asprintf_l(&__nb, _LIBCPP_GET_C_LOCALE, __fmt, (int)__iob.precision(), __v);
1395349cc55cSDimitry Andric    else
1396349cc55cSDimitry Andric      __nc = __libcpp_asprintf_l(&__nb, _LIBCPP_GET_C_LOCALE, __fmt, __v);
1397349cc55cSDimitry Andric    if (__nc == -1)
1398349cc55cSDimitry Andric      __throw_bad_alloc();
1399349cc55cSDimitry Andric    __nbh.reset(__nb);
1400349cc55cSDimitry Andric  }
140181ad6265SDimitry Andric  _LIBCPP_DIAGNOSTIC_POP
1402349cc55cSDimitry Andric  char* __ne = __nb + __nc;
1403349cc55cSDimitry Andric  char* __np = this->__identify_padding(__nb, __ne, __iob);
14040b57cec5SDimitry Andric  // Stage 2 - Widen __nar while adding thousands separators
14050b57cec5SDimitry Andric  char_type __o[2 * (__nbuf - 1) - 1];
1406349cc55cSDimitry Andric  char_type* __ob = __o;
1407349cc55cSDimitry Andric  unique_ptr<char_type, void (*)(void*)> __obh(0, free);
1408cb14a3feSDimitry Andric  if (__nb != __nar) {
1409349cc55cSDimitry Andric    __ob = (char_type*)malloc(2 * static_cast<size_t>(__nc) * sizeof(char_type));
1410349cc55cSDimitry Andric    if (__ob == 0)
1411349cc55cSDimitry Andric      __throw_bad_alloc();
1412349cc55cSDimitry Andric    __obh.reset(__ob);
1413349cc55cSDimitry Andric  }
14140b57cec5SDimitry Andric  char_type* __op; // pad here
14150b57cec5SDimitry Andric  char_type* __oe; // end of output
1416349cc55cSDimitry Andric  this->__widen_and_group_float(__nb, __np, __ne, __ob, __op, __oe, __iob.getloc());
14170b57cec5SDimitry Andric  // [__o, __oe) contains thousands_sep'd wide number
14180b57cec5SDimitry Andric  // Stage 3 & 4
1419bdd1243dSDimitry Andric  __s = std::__pad_and_output(__s, __ob, __op, __oe, __iob, __fl);
1420349cc55cSDimitry Andric  return __s;
14210b57cec5SDimitry Andric}
14220b57cec5SDimitry Andric
14230b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
14240b57cec5SDimitry Andric_OutputIterator
1425cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, double __v) const {
1426349cc55cSDimitry Andric  return this->__do_put_floating_point(__s, __iob, __fl, __v, "");
14270b57cec5SDimitry Andric}
14280b57cec5SDimitry Andric
14290b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
14300b57cec5SDimitry Andric_OutputIterator
1431cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, long double __v) const {
1432349cc55cSDimitry Andric  return this->__do_put_floating_point(__s, __iob, __fl, __v, "L");
14330b57cec5SDimitry Andric}
14340b57cec5SDimitry Andric
14350b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
14360b57cec5SDimitry Andric_OutputIterator
1437cb14a3feSDimitry Andricnum_put<_CharT, _OutputIterator>::do_put(iter_type __s, ios_base& __iob, char_type __fl, const void* __v) const {
14380b57cec5SDimitry Andric  // Stage 1 - Get pointer in narrow char
14390b57cec5SDimitry Andric  const unsigned __nbuf = 20;
14400b57cec5SDimitry Andric  char __nar[__nbuf];
1441349cc55cSDimitry Andric  int __nc   = __libcpp_snprintf_l(__nar, sizeof(__nar), _LIBCPP_GET_C_LOCALE, "%p", __v);
14420b57cec5SDimitry Andric  char* __ne = __nar + __nc;
14430b57cec5SDimitry Andric  char* __np = this->__identify_padding(__nar, __ne, __iob);
14440b57cec5SDimitry Andric  // Stage 2 - Widen __nar
14450b57cec5SDimitry Andric  char_type __o[2 * (__nbuf - 1) - 1];
14460b57cec5SDimitry Andric  char_type* __op; // pad here
14470b57cec5SDimitry Andric  char_type* __oe; // end of output
1448bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
14490b57cec5SDimitry Andric  __ct.widen(__nar, __ne, __o);
14500b57cec5SDimitry Andric  __oe = __o + (__ne - __nar);
14510b57cec5SDimitry Andric  if (__np == __ne)
14520b57cec5SDimitry Andric    __op = __oe;
14530b57cec5SDimitry Andric  else
14540b57cec5SDimitry Andric    __op = __o + (__np - __nar);
14550b57cec5SDimitry Andric  // [__o, __oe) contains wide number
14560b57cec5SDimitry Andric  // Stage 3 & 4
1457bdd1243dSDimitry Andric  return std::__pad_and_output(__s, __o, __op, __oe, __iob, __fl);
14580b57cec5SDimitry Andric}
14590b57cec5SDimitry Andric
146081ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS num_put<char>;
1461349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
146281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS num_put<wchar_t>;
1463349cc55cSDimitry Andric#  endif
14640b57cec5SDimitry Andric
14650b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1466cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI int __get_up_to_n_digits(
1467cb14a3feSDimitry Andric    _InputIterator& __b, _InputIterator __e, ios_base::iostate& __err, const ctype<_CharT>& __ct, int __n) {
14680b57cec5SDimitry Andric  // Precondition:  __n >= 1
1469cb14a3feSDimitry Andric  if (__b == __e) {
14700b57cec5SDimitry Andric    __err |= ios_base::eofbit | ios_base::failbit;
14710b57cec5SDimitry Andric    return 0;
14720b57cec5SDimitry Andric  }
14730b57cec5SDimitry Andric  // get first digit
14740b57cec5SDimitry Andric  _CharT __c = *__b;
1475cb14a3feSDimitry Andric  if (!__ct.is(ctype_base::digit, __c)) {
14760b57cec5SDimitry Andric    __err |= ios_base::failbit;
14770b57cec5SDimitry Andric    return 0;
14780b57cec5SDimitry Andric  }
14790b57cec5SDimitry Andric  int __r = __ct.narrow(__c, 0) - '0';
1480cb14a3feSDimitry Andric  for (++__b, (void)--__n; __b != __e && __n > 0; ++__b, (void)--__n) {
14810b57cec5SDimitry Andric    // get next digit
14820b57cec5SDimitry Andric    __c = *__b;
14830b57cec5SDimitry Andric    if (!__ct.is(ctype_base::digit, __c))
14840b57cec5SDimitry Andric      return __r;
14850b57cec5SDimitry Andric    __r = __r * 10 + __ct.narrow(__c, 0) - '0';
14860b57cec5SDimitry Andric  }
14870b57cec5SDimitry Andric  if (__b == __e)
14880b57cec5SDimitry Andric    __err |= ios_base::eofbit;
14890b57cec5SDimitry Andric  return __r;
14900b57cec5SDimitry Andric}
14910b57cec5SDimitry Andric
1492cb14a3feSDimitry Andricclass _LIBCPP_EXPORTED_FROM_ABI time_base {
14930b57cec5SDimitry Andricpublic:
14940b57cec5SDimitry Andric  enum dateorder { no_order, dmy, mdy, ymd, ydm };
14950b57cec5SDimitry Andric};
14960b57cec5SDimitry Andric
14970b57cec5SDimitry Andrictemplate <class _CharT>
1498cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS __time_get_c_storage {
14990b57cec5SDimitry Andricprotected:
15000b57cec5SDimitry Andric  typedef basic_string<_CharT> string_type;
15010b57cec5SDimitry Andric
15020b57cec5SDimitry Andric  virtual const string_type* __weeks() const;
15030b57cec5SDimitry Andric  virtual const string_type* __months() const;
15040b57cec5SDimitry Andric  virtual const string_type* __am_pm() const;
15050b57cec5SDimitry Andric  virtual const string_type& __c() const;
15060b57cec5SDimitry Andric  virtual const string_type& __r() const;
15070b57cec5SDimitry Andric  virtual const string_type& __x() const;
15080b57cec5SDimitry Andric  virtual const string_type& __X() const;
15090b57cec5SDimitry Andric
1510cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI ~__time_get_c_storage() {}
15110b57cec5SDimitry Andric};
15120b57cec5SDimitry Andric
1513cb14a3feSDimitry Andrictemplate <>
1514cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string* __time_get_c_storage<char>::__weeks() const;
1515cb14a3feSDimitry Andrictemplate <>
1516cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string* __time_get_c_storage<char>::__months() const;
1517cb14a3feSDimitry Andrictemplate <>
1518cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string* __time_get_c_storage<char>::__am_pm() const;
1519cb14a3feSDimitry Andrictemplate <>
1520cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string& __time_get_c_storage<char>::__c() const;
1521cb14a3feSDimitry Andrictemplate <>
1522cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string& __time_get_c_storage<char>::__r() const;
1523cb14a3feSDimitry Andrictemplate <>
1524cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string& __time_get_c_storage<char>::__x() const;
1525cb14a3feSDimitry Andrictemplate <>
1526cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const string& __time_get_c_storage<char>::__X() const;
15270b57cec5SDimitry Andric
1528349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
1529cb14a3feSDimitry Andrictemplate <>
1530cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring* __time_get_c_storage<wchar_t>::__weeks() const;
1531cb14a3feSDimitry Andrictemplate <>
1532cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring* __time_get_c_storage<wchar_t>::__months() const;
1533cb14a3feSDimitry Andrictemplate <>
1534cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring* __time_get_c_storage<wchar_t>::__am_pm() const;
1535cb14a3feSDimitry Andrictemplate <>
1536cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring& __time_get_c_storage<wchar_t>::__c() const;
1537cb14a3feSDimitry Andrictemplate <>
1538cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring& __time_get_c_storage<wchar_t>::__r() const;
1539cb14a3feSDimitry Andrictemplate <>
1540cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring& __time_get_c_storage<wchar_t>::__x() const;
1541cb14a3feSDimitry Andrictemplate <>
1542cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI const wstring& __time_get_c_storage<wchar_t>::__X() const;
1543349cc55cSDimitry Andric#  endif
15440b57cec5SDimitry Andric
15450b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
1546cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS time_get : public locale::facet, public time_base, private __time_get_c_storage<_CharT> {
15470b57cec5SDimitry Andricpublic:
15480b57cec5SDimitry Andric  typedef _CharT char_type;
15490b57cec5SDimitry Andric  typedef _InputIterator iter_type;
15500b57cec5SDimitry Andric  typedef time_base::dateorder dateorder;
15510b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
15520b57cec5SDimitry Andric
1553cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_get(size_t __refs = 0) : locale::facet(__refs) {}
15540b57cec5SDimitry Andric
1555cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI dateorder date_order() const { return this->do_date_order(); }
15560b57cec5SDimitry Andric
1557cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
1558cb14a3feSDimitry Andric  get_time(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
15590b57cec5SDimitry Andric    return do_get_time(__b, __e, __iob, __err, __tm);
15600b57cec5SDimitry Andric  }
15610b57cec5SDimitry Andric
1562cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
1563cb14a3feSDimitry Andric  get_date(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
15640b57cec5SDimitry Andric    return do_get_date(__b, __e, __iob, __err, __tm);
15650b57cec5SDimitry Andric  }
15660b57cec5SDimitry Andric
1567cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
1568cb14a3feSDimitry Andric  get_weekday(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
15690b57cec5SDimitry Andric    return do_get_weekday(__b, __e, __iob, __err, __tm);
15700b57cec5SDimitry Andric  }
15710b57cec5SDimitry Andric
1572cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
1573cb14a3feSDimitry Andric  get_monthname(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
15740b57cec5SDimitry Andric    return do_get_monthname(__b, __e, __iob, __err, __tm);
15750b57cec5SDimitry Andric  }
15760b57cec5SDimitry Andric
1577cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
1578cb14a3feSDimitry Andric  get_year(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
15790b57cec5SDimitry Andric    return do_get_year(__b, __e, __iob, __err, __tm);
15800b57cec5SDimitry Andric  }
15810b57cec5SDimitry Andric
1582cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
1583cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm, char __fmt, char __mod = 0)
1584cb14a3feSDimitry Andric      const {
15850b57cec5SDimitry Andric    return do_get(__b, __e, __iob, __err, __tm, __fmt, __mod);
15860b57cec5SDimitry Andric  }
15870b57cec5SDimitry Andric
1588cb14a3feSDimitry Andric  iter_type
1589cb14a3feSDimitry Andric  get(iter_type __b,
1590cb14a3feSDimitry Andric      iter_type __e,
1591cb14a3feSDimitry Andric      ios_base& __iob,
1592cb14a3feSDimitry Andric      ios_base::iostate& __err,
1593cb14a3feSDimitry Andric      tm* __tm,
1594cb14a3feSDimitry Andric      const char_type* __fmtb,
1595cb14a3feSDimitry Andric      const char_type* __fmte) const;
15960b57cec5SDimitry Andric
15970b57cec5SDimitry Andric  static locale::id id;
15980b57cec5SDimitry Andric
15990b57cec5SDimitry Andricprotected:
1600bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~time_get() override {}
16010b57cec5SDimitry Andric
16020b57cec5SDimitry Andric  virtual dateorder do_date_order() const;
1603cb14a3feSDimitry Andric  virtual iter_type
1604cb14a3feSDimitry Andric  do_get_time(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const;
1605cb14a3feSDimitry Andric  virtual iter_type
1606cb14a3feSDimitry Andric  do_get_date(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const;
1607cb14a3feSDimitry Andric  virtual iter_type
1608cb14a3feSDimitry Andric  do_get_weekday(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const;
1609cb14a3feSDimitry Andric  virtual iter_type
1610cb14a3feSDimitry Andric  do_get_monthname(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const;
1611cb14a3feSDimitry Andric  virtual iter_type
1612cb14a3feSDimitry Andric  do_get_year(iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const;
1613cb14a3feSDimitry Andric  virtual iter_type do_get(
1614cb14a3feSDimitry Andric      iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm, char __fmt, char __mod) const;
16150b57cec5SDimitry Andric
1616cb14a3feSDimitry Andricprivate:
1617cb14a3feSDimitry Andric  void __get_white_space(iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1618cb14a3feSDimitry Andric  void __get_percent(iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1619cb14a3feSDimitry Andric
1620cb14a3feSDimitry Andric  void __get_weekdayname(
1621cb14a3feSDimitry Andric      int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1622cb14a3feSDimitry Andric  void __get_monthname(
1623cb14a3feSDimitry Andric      int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1624cb14a3feSDimitry Andric  void __get_day(int& __d, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1625cb14a3feSDimitry Andric  void
1626cb14a3feSDimitry Andric  __get_month(int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1627cb14a3feSDimitry Andric  void
1628cb14a3feSDimitry Andric  __get_year(int& __y, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1629cb14a3feSDimitry Andric  void
1630cb14a3feSDimitry Andric  __get_year4(int& __y, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1631cb14a3feSDimitry Andric  void
1632cb14a3feSDimitry Andric  __get_hour(int& __d, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1633cb14a3feSDimitry Andric  void
1634cb14a3feSDimitry Andric  __get_12_hour(int& __h, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1635cb14a3feSDimitry Andric  void
1636cb14a3feSDimitry Andric  __get_am_pm(int& __h, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1637cb14a3feSDimitry Andric  void
1638cb14a3feSDimitry Andric  __get_minute(int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1639cb14a3feSDimitry Andric  void
1640cb14a3feSDimitry Andric  __get_second(int& __s, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1641cb14a3feSDimitry Andric  void
1642cb14a3feSDimitry Andric  __get_weekday(int& __w, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
1643cb14a3feSDimitry Andric  void __get_day_year_num(
1644cb14a3feSDimitry Andric      int& __w, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const;
16450b57cec5SDimitry Andric};
16460b57cec5SDimitry Andric
16470b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1648cb14a3feSDimitry Andriclocale::id time_get<_CharT, _InputIterator>::id;
16490b57cec5SDimitry Andric
16500b57cec5SDimitry Andric// time_get primitives
16510b57cec5SDimitry Andric
16520b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1653cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_weekdayname(
1654cb14a3feSDimitry Andric    int& __w, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
16550b57cec5SDimitry Andric  // Note:  ignoring case comes from the POSIX strptime spec
16560b57cec5SDimitry Andric  const string_type* __wk = this->__weeks();
16575f757f3fSDimitry Andric  ptrdiff_t __i           = std::__scan_keyword(__b, __e, __wk, __wk + 14, __ct, __err, false) - __wk;
16580b57cec5SDimitry Andric  if (__i < 14)
16590b57cec5SDimitry Andric    __w = __i % 7;
16600b57cec5SDimitry Andric}
16610b57cec5SDimitry Andric
16620b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1663cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_monthname(
1664cb14a3feSDimitry Andric    int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
16650b57cec5SDimitry Andric  // Note:  ignoring case comes from the POSIX strptime spec
16660b57cec5SDimitry Andric  const string_type* __month = this->__months();
16675f757f3fSDimitry Andric  ptrdiff_t __i              = std::__scan_keyword(__b, __e, __month, __month + 24, __ct, __err, false) - __month;
16680b57cec5SDimitry Andric  if (__i < 24)
16690b57cec5SDimitry Andric    __m = __i % 12;
16700b57cec5SDimitry Andric}
16710b57cec5SDimitry Andric
16720b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1673cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_day(
1674cb14a3feSDimitry Andric    int& __d, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
16755f757f3fSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 2);
16760b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && 1 <= __t && __t <= 31)
16770b57cec5SDimitry Andric    __d = __t;
16780b57cec5SDimitry Andric  else
16790b57cec5SDimitry Andric    __err |= ios_base::failbit;
16800b57cec5SDimitry Andric}
16810b57cec5SDimitry Andric
16820b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1683cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_month(
1684cb14a3feSDimitry Andric    int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1685bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 2) - 1;
168681ad6265SDimitry Andric  if (!(__err & ios_base::failbit) && 0 <= __t && __t <= 11)
16870b57cec5SDimitry Andric    __m = __t;
16880b57cec5SDimitry Andric  else
16890b57cec5SDimitry Andric    __err |= ios_base::failbit;
16900b57cec5SDimitry Andric}
16910b57cec5SDimitry Andric
16920b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1693cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_year(
1694cb14a3feSDimitry Andric    int& __y, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1695bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 4);
1696cb14a3feSDimitry Andric  if (!(__err & ios_base::failbit)) {
16970b57cec5SDimitry Andric    if (__t < 69)
16980b57cec5SDimitry Andric      __t += 2000;
16990b57cec5SDimitry Andric    else if (69 <= __t && __t <= 99)
17000b57cec5SDimitry Andric      __t += 1900;
17010b57cec5SDimitry Andric    __y = __t - 1900;
17020b57cec5SDimitry Andric  }
17030b57cec5SDimitry Andric}
17040b57cec5SDimitry Andric
17050b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1706cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_year4(
1707cb14a3feSDimitry Andric    int& __y, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1708bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 4);
17090b57cec5SDimitry Andric  if (!(__err & ios_base::failbit))
17100b57cec5SDimitry Andric    __y = __t - 1900;
17110b57cec5SDimitry Andric}
17120b57cec5SDimitry Andric
17130b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1714cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_hour(
1715cb14a3feSDimitry Andric    int& __h, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1716bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 2);
17170b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && __t <= 23)
17180b57cec5SDimitry Andric    __h = __t;
17190b57cec5SDimitry Andric  else
17200b57cec5SDimitry Andric    __err |= ios_base::failbit;
17210b57cec5SDimitry Andric}
17220b57cec5SDimitry Andric
17230b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1724cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_12_hour(
1725cb14a3feSDimitry Andric    int& __h, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1726bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 2);
17270b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && 1 <= __t && __t <= 12)
17280b57cec5SDimitry Andric    __h = __t;
17290b57cec5SDimitry Andric  else
17300b57cec5SDimitry Andric    __err |= ios_base::failbit;
17310b57cec5SDimitry Andric}
17320b57cec5SDimitry Andric
17330b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1734cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_minute(
1735cb14a3feSDimitry Andric    int& __m, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1736bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 2);
17370b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && __t <= 59)
17380b57cec5SDimitry Andric    __m = __t;
17390b57cec5SDimitry Andric  else
17400b57cec5SDimitry Andric    __err |= ios_base::failbit;
17410b57cec5SDimitry Andric}
17420b57cec5SDimitry Andric
17430b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1744cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_second(
1745cb14a3feSDimitry Andric    int& __s, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1746bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 2);
17470b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && __t <= 60)
17480b57cec5SDimitry Andric    __s = __t;
17490b57cec5SDimitry Andric  else
17500b57cec5SDimitry Andric    __err |= ios_base::failbit;
17510b57cec5SDimitry Andric}
17520b57cec5SDimitry Andric
17530b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1754cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_weekday(
1755cb14a3feSDimitry Andric    int& __w, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1756bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 1);
17570b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && __t <= 6)
17580b57cec5SDimitry Andric    __w = __t;
17590b57cec5SDimitry Andric  else
17600b57cec5SDimitry Andric    __err |= ios_base::failbit;
17610b57cec5SDimitry Andric}
17620b57cec5SDimitry Andric
17630b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1764cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_day_year_num(
1765cb14a3feSDimitry Andric    int& __d, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1766bdd1243dSDimitry Andric  int __t = std::__get_up_to_n_digits(__b, __e, __err, __ct, 3);
17670b57cec5SDimitry Andric  if (!(__err & ios_base::failbit) && __t <= 365)
17680b57cec5SDimitry Andric    __d = __t;
17690b57cec5SDimitry Andric  else
17700b57cec5SDimitry Andric    __err |= ios_base::failbit;
17710b57cec5SDimitry Andric}
17720b57cec5SDimitry Andric
17730b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1774cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_white_space(
1775cb14a3feSDimitry Andric    iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
17760b57cec5SDimitry Andric  for (; __b != __e && __ct.is(ctype_base::space, *__b); ++__b)
17770b57cec5SDimitry Andric    ;
17780b57cec5SDimitry Andric  if (__b == __e)
17790b57cec5SDimitry Andric    __err |= ios_base::eofbit;
17800b57cec5SDimitry Andric}
17810b57cec5SDimitry Andric
17820b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1783cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_am_pm(
1784cb14a3feSDimitry Andric    int& __h, iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
17850b57cec5SDimitry Andric  const string_type* __ap = this->__am_pm();
1786cb14a3feSDimitry Andric  if (__ap[0].size() + __ap[1].size() == 0) {
17870b57cec5SDimitry Andric    __err |= ios_base::failbit;
17880b57cec5SDimitry Andric    return;
17890b57cec5SDimitry Andric  }
17905f757f3fSDimitry Andric  ptrdiff_t __i = std::__scan_keyword(__b, __e, __ap, __ap + 2, __ct, __err, false) - __ap;
17910b57cec5SDimitry Andric  if (__i == 0 && __h == 12)
17920b57cec5SDimitry Andric    __h = 0;
17930b57cec5SDimitry Andric  else if (__i == 1 && __h < 12)
17940b57cec5SDimitry Andric    __h += 12;
17950b57cec5SDimitry Andric}
17960b57cec5SDimitry Andric
17970b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1798cb14a3feSDimitry Andricvoid time_get<_CharT, _InputIterator>::__get_percent(
1799cb14a3feSDimitry Andric    iter_type& __b, iter_type __e, ios_base::iostate& __err, const ctype<char_type>& __ct) const {
1800cb14a3feSDimitry Andric  if (__b == __e) {
18010b57cec5SDimitry Andric    __err |= ios_base::eofbit | ios_base::failbit;
18020b57cec5SDimitry Andric    return;
18030b57cec5SDimitry Andric  }
18040b57cec5SDimitry Andric  if (__ct.narrow(*__b, 0) != '%')
18050b57cec5SDimitry Andric    __err |= ios_base::failbit;
18060b57cec5SDimitry Andric  else if (++__b == __e)
18070b57cec5SDimitry Andric    __err |= ios_base::eofbit;
18080b57cec5SDimitry Andric}
18090b57cec5SDimitry Andric
18100b57cec5SDimitry Andric// time_get end primitives
18110b57cec5SDimitry Andric
18120b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1813cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::get(
1814cb14a3feSDimitry Andric    iter_type __b,
1815cb14a3feSDimitry Andric    iter_type __e,
18160b57cec5SDimitry Andric    ios_base& __iob,
1817cb14a3feSDimitry Andric    ios_base::iostate& __err,
1818cb14a3feSDimitry Andric    tm* __tm,
1819cb14a3feSDimitry Andric    const char_type* __fmtb,
1820cb14a3feSDimitry Andric    const char_type* __fmte) const {
1821bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
18220b57cec5SDimitry Andric  __err                        = ios_base::goodbit;
1823cb14a3feSDimitry Andric  while (__fmtb != __fmte && __err == ios_base::goodbit) {
1824cb14a3feSDimitry Andric    if (__b == __e) {
18250b57cec5SDimitry Andric      __err = ios_base::failbit;
18260b57cec5SDimitry Andric      break;
18270b57cec5SDimitry Andric    }
1828cb14a3feSDimitry Andric    if (__ct.narrow(*__fmtb, 0) == '%') {
1829cb14a3feSDimitry Andric      if (++__fmtb == __fmte) {
18300b57cec5SDimitry Andric        __err = ios_base::failbit;
18310b57cec5SDimitry Andric        break;
18320b57cec5SDimitry Andric      }
18330b57cec5SDimitry Andric      char __cmd = __ct.narrow(*__fmtb, 0);
18340b57cec5SDimitry Andric      char __opt = '\0';
1835cb14a3feSDimitry Andric      if (__cmd == 'E' || __cmd == '0') {
1836cb14a3feSDimitry Andric        if (++__fmtb == __fmte) {
18370b57cec5SDimitry Andric          __err = ios_base::failbit;
18380b57cec5SDimitry Andric          break;
18390b57cec5SDimitry Andric        }
18400b57cec5SDimitry Andric        __opt = __cmd;
18410b57cec5SDimitry Andric        __cmd = __ct.narrow(*__fmtb, 0);
18420b57cec5SDimitry Andric      }
18430b57cec5SDimitry Andric      __b = do_get(__b, __e, __iob, __err, __tm, __cmd, __opt);
18440b57cec5SDimitry Andric      ++__fmtb;
1845cb14a3feSDimitry Andric    } else if (__ct.is(ctype_base::space, *__fmtb)) {
18460b57cec5SDimitry Andric      for (++__fmtb; __fmtb != __fmte && __ct.is(ctype_base::space, *__fmtb); ++__fmtb)
18470b57cec5SDimitry Andric        ;
18480b57cec5SDimitry Andric      for (; __b != __e && __ct.is(ctype_base::space, *__b); ++__b)
18490b57cec5SDimitry Andric        ;
1850cb14a3feSDimitry Andric    } else if (__ct.toupper(*__b) == __ct.toupper(*__fmtb)) {
18510b57cec5SDimitry Andric      ++__b;
18520b57cec5SDimitry Andric      ++__fmtb;
1853cb14a3feSDimitry Andric    } else
18540b57cec5SDimitry Andric      __err = ios_base::failbit;
18550b57cec5SDimitry Andric  }
18560b57cec5SDimitry Andric  if (__b == __e)
18570b57cec5SDimitry Andric    __err |= ios_base::eofbit;
18580b57cec5SDimitry Andric  return __b;
18590b57cec5SDimitry Andric}
18600b57cec5SDimitry Andric
18610b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1862cb14a3feSDimitry Andrictypename time_get<_CharT, _InputIterator>::dateorder time_get<_CharT, _InputIterator>::do_date_order() const {
18630b57cec5SDimitry Andric  return mdy;
18640b57cec5SDimitry Andric}
18650b57cec5SDimitry Andric
18660b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1867cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::do_get_time(
1868cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
18690b57cec5SDimitry Andric  const char_type __fmt[] = {'%', 'H', ':', '%', 'M', ':', '%', 'S'};
18700b57cec5SDimitry Andric  return get(__b, __e, __iob, __err, __tm, __fmt, __fmt + sizeof(__fmt) / sizeof(__fmt[0]));
18710b57cec5SDimitry Andric}
18720b57cec5SDimitry Andric
18730b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1874cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::do_get_date(
1875cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
18760b57cec5SDimitry Andric  const string_type& __fmt = this->__x();
18770b57cec5SDimitry Andric  return get(__b, __e, __iob, __err, __tm, __fmt.data(), __fmt.data() + __fmt.size());
18780b57cec5SDimitry Andric}
18790b57cec5SDimitry Andric
18800b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1881cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::do_get_weekday(
1882cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
1883bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
18840b57cec5SDimitry Andric  __get_weekdayname(__tm->tm_wday, __b, __e, __err, __ct);
18850b57cec5SDimitry Andric  return __b;
18860b57cec5SDimitry Andric}
18870b57cec5SDimitry Andric
18880b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1889cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::do_get_monthname(
1890cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
1891bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
18920b57cec5SDimitry Andric  __get_monthname(__tm->tm_mon, __b, __e, __err, __ct);
18930b57cec5SDimitry Andric  return __b;
18940b57cec5SDimitry Andric}
18950b57cec5SDimitry Andric
18960b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1897cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::do_get_year(
1898cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm) const {
1899bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
19000b57cec5SDimitry Andric  __get_year(__tm->tm_year, __b, __e, __err, __ct);
19010b57cec5SDimitry Andric  return __b;
19020b57cec5SDimitry Andric}
19030b57cec5SDimitry Andric
19040b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
1905cb14a3feSDimitry Andric_InputIterator time_get<_CharT, _InputIterator>::do_get(
1906cb14a3feSDimitry Andric    iter_type __b, iter_type __e, ios_base& __iob, ios_base::iostate& __err, tm* __tm, char __fmt, char) const {
19070b57cec5SDimitry Andric  __err                        = ios_base::goodbit;
1908bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
1909cb14a3feSDimitry Andric  switch (__fmt) {
19100b57cec5SDimitry Andric  case 'a':
19110b57cec5SDimitry Andric  case 'A':
19120b57cec5SDimitry Andric    __get_weekdayname(__tm->tm_wday, __b, __e, __err, __ct);
19130b57cec5SDimitry Andric    break;
19140b57cec5SDimitry Andric  case 'b':
19150b57cec5SDimitry Andric  case 'B':
19160b57cec5SDimitry Andric  case 'h':
19170b57cec5SDimitry Andric    __get_monthname(__tm->tm_mon, __b, __e, __err, __ct);
19180b57cec5SDimitry Andric    break;
1919cb14a3feSDimitry Andric  case 'c': {
19200b57cec5SDimitry Andric    const string_type& __fm = this->__c();
19210b57cec5SDimitry Andric    __b                     = get(__b, __e, __iob, __err, __tm, __fm.data(), __fm.data() + __fm.size());
1922cb14a3feSDimitry Andric  } break;
19230b57cec5SDimitry Andric  case 'd':
19240b57cec5SDimitry Andric  case 'e':
19250b57cec5SDimitry Andric    __get_day(__tm->tm_mday, __b, __e, __err, __ct);
19260b57cec5SDimitry Andric    break;
1927cb14a3feSDimitry Andric  case 'D': {
19280b57cec5SDimitry Andric    const char_type __fm[] = {'%', 'm', '/', '%', 'd', '/', '%', 'y'};
19290b57cec5SDimitry Andric    __b                    = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm) / sizeof(__fm[0]));
1930cb14a3feSDimitry Andric  } break;
1931cb14a3feSDimitry Andric  case 'F': {
19320b57cec5SDimitry Andric    const char_type __fm[] = {'%', 'Y', '-', '%', 'm', '-', '%', 'd'};
19330b57cec5SDimitry Andric    __b                    = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm) / sizeof(__fm[0]));
1934cb14a3feSDimitry Andric  } break;
19350b57cec5SDimitry Andric  case 'H':
19360b57cec5SDimitry Andric    __get_hour(__tm->tm_hour, __b, __e, __err, __ct);
19370b57cec5SDimitry Andric    break;
19380b57cec5SDimitry Andric  case 'I':
19390b57cec5SDimitry Andric    __get_12_hour(__tm->tm_hour, __b, __e, __err, __ct);
19400b57cec5SDimitry Andric    break;
19410b57cec5SDimitry Andric  case 'j':
19420b57cec5SDimitry Andric    __get_day_year_num(__tm->tm_yday, __b, __e, __err, __ct);
19430b57cec5SDimitry Andric    break;
19440b57cec5SDimitry Andric  case 'm':
19450b57cec5SDimitry Andric    __get_month(__tm->tm_mon, __b, __e, __err, __ct);
19460b57cec5SDimitry Andric    break;
19470b57cec5SDimitry Andric  case 'M':
19480b57cec5SDimitry Andric    __get_minute(__tm->tm_min, __b, __e, __err, __ct);
19490b57cec5SDimitry Andric    break;
19500b57cec5SDimitry Andric  case 'n':
19510b57cec5SDimitry Andric  case 't':
19520b57cec5SDimitry Andric    __get_white_space(__b, __e, __err, __ct);
19530b57cec5SDimitry Andric    break;
19540b57cec5SDimitry Andric  case 'p':
19550b57cec5SDimitry Andric    __get_am_pm(__tm->tm_hour, __b, __e, __err, __ct);
19560b57cec5SDimitry Andric    break;
1957cb14a3feSDimitry Andric  case 'r': {
19580b57cec5SDimitry Andric    const char_type __fm[] = {'%', 'I', ':', '%', 'M', ':', '%', 'S', ' ', '%', 'p'};
19590b57cec5SDimitry Andric    __b                    = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm) / sizeof(__fm[0]));
1960cb14a3feSDimitry Andric  } break;
1961cb14a3feSDimitry Andric  case 'R': {
19620b57cec5SDimitry Andric    const char_type __fm[] = {'%', 'H', ':', '%', 'M'};
19630b57cec5SDimitry Andric    __b                    = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm) / sizeof(__fm[0]));
1964cb14a3feSDimitry Andric  } break;
19650b57cec5SDimitry Andric  case 'S':
19660b57cec5SDimitry Andric    __get_second(__tm->tm_sec, __b, __e, __err, __ct);
19670b57cec5SDimitry Andric    break;
1968cb14a3feSDimitry Andric  case 'T': {
19690b57cec5SDimitry Andric    const char_type __fm[] = {'%', 'H', ':', '%', 'M', ':', '%', 'S'};
19700b57cec5SDimitry Andric    __b                    = get(__b, __e, __iob, __err, __tm, __fm, __fm + sizeof(__fm) / sizeof(__fm[0]));
1971cb14a3feSDimitry Andric  } break;
19720b57cec5SDimitry Andric  case 'w':
19730b57cec5SDimitry Andric    __get_weekday(__tm->tm_wday, __b, __e, __err, __ct);
19740b57cec5SDimitry Andric    break;
19750b57cec5SDimitry Andric  case 'x':
19760b57cec5SDimitry Andric    return do_get_date(__b, __e, __iob, __err, __tm);
1977cb14a3feSDimitry Andric  case 'X': {
19780b57cec5SDimitry Andric    const string_type& __fm = this->__X();
19790b57cec5SDimitry Andric    __b                     = get(__b, __e, __iob, __err, __tm, __fm.data(), __fm.data() + __fm.size());
1980cb14a3feSDimitry Andric  } break;
19810b57cec5SDimitry Andric  case 'y':
19820b57cec5SDimitry Andric    __get_year(__tm->tm_year, __b, __e, __err, __ct);
19830b57cec5SDimitry Andric    break;
19840b57cec5SDimitry Andric  case 'Y':
19850b57cec5SDimitry Andric    __get_year4(__tm->tm_year, __b, __e, __err, __ct);
19860b57cec5SDimitry Andric    break;
19870b57cec5SDimitry Andric  case '%':
19880b57cec5SDimitry Andric    __get_percent(__b, __e, __err, __ct);
19890b57cec5SDimitry Andric    break;
19900b57cec5SDimitry Andric  default:
19910b57cec5SDimitry Andric    __err |= ios_base::failbit;
19920b57cec5SDimitry Andric  }
19930b57cec5SDimitry Andric  return __b;
19940b57cec5SDimitry Andric}
19950b57cec5SDimitry Andric
199681ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_get<char>;
1997349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
199881ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_get<wchar_t>;
1999349cc55cSDimitry Andric#  endif
20000b57cec5SDimitry Andric
2001cb14a3feSDimitry Andricclass _LIBCPP_EXPORTED_FROM_ABI __time_get {
20020b57cec5SDimitry Andricprotected:
20030b57cec5SDimitry Andric  locale_t __loc_;
20040b57cec5SDimitry Andric
20050b57cec5SDimitry Andric  __time_get(const char* __nm);
20060b57cec5SDimitry Andric  __time_get(const string& __nm);
20070b57cec5SDimitry Andric  ~__time_get();
20080b57cec5SDimitry Andric};
20090b57cec5SDimitry Andric
20100b57cec5SDimitry Andrictemplate <class _CharT>
2011cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS __time_get_storage : public __time_get {
20120b57cec5SDimitry Andricprotected:
20130b57cec5SDimitry Andric  typedef basic_string<_CharT> string_type;
20140b57cec5SDimitry Andric
20150b57cec5SDimitry Andric  string_type __weeks_[14];
20160b57cec5SDimitry Andric  string_type __months_[24];
20170b57cec5SDimitry Andric  string_type __am_pm_[2];
20180b57cec5SDimitry Andric  string_type __c_;
20190b57cec5SDimitry Andric  string_type __r_;
20200b57cec5SDimitry Andric  string_type __x_;
20210b57cec5SDimitry Andric  string_type __X_;
20220b57cec5SDimitry Andric
20230b57cec5SDimitry Andric  explicit __time_get_storage(const char* __nm);
20240b57cec5SDimitry Andric  explicit __time_get_storage(const string& __nm);
20250b57cec5SDimitry Andric
20265f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI ~__time_get_storage() {}
20270b57cec5SDimitry Andric
20280b57cec5SDimitry Andric  time_base::dateorder __do_date_order() const;
20290b57cec5SDimitry Andric
20300b57cec5SDimitry Andricprivate:
20310b57cec5SDimitry Andric  void init(const ctype<_CharT>&);
20320b57cec5SDimitry Andric  string_type __analyze(char __fmt, const ctype<_CharT>&);
20330b57cec5SDimitry Andric};
20340b57cec5SDimitry Andric
20350b57cec5SDimitry Andric#  define _LIBCPP_TIME_GET_STORAGE_EXPLICIT_INSTANTIATION(_CharT)                                                      \
2036cb14a3feSDimitry Andric    template <>                                                                                                        \
2037cb14a3feSDimitry Andric    _LIBCPP_EXPORTED_FROM_ABI time_base::dateorder __time_get_storage<_CharT>::__do_date_order() const;                \
2038cb14a3feSDimitry Andric    template <>                                                                                                        \
2039cb14a3feSDimitry Andric    _LIBCPP_EXPORTED_FROM_ABI __time_get_storage<_CharT>::__time_get_storage(const char*);                             \
2040cb14a3feSDimitry Andric    template <>                                                                                                        \
2041cb14a3feSDimitry Andric    _LIBCPP_EXPORTED_FROM_ABI __time_get_storage<_CharT>::__time_get_storage(const string&);                           \
2042cb14a3feSDimitry Andric    template <>                                                                                                        \
2043cb14a3feSDimitry Andric    _LIBCPP_EXPORTED_FROM_ABI void __time_get_storage<_CharT>::init(const ctype<_CharT>&);                             \
2044cb14a3feSDimitry Andric    template <>                                                                                                        \
2045cb14a3feSDimitry Andric    _LIBCPP_EXPORTED_FROM_ABI __time_get_storage<_CharT>::string_type __time_get_storage<_CharT>::__analyze(           \
2046cb14a3feSDimitry Andric        char, const ctype<_CharT>&);                                                                                   \
2047*0fca6ea1SDimitry Andric    extern template _LIBCPP_EXPORTED_FROM_ABI time_base::dateorder __time_get_storage<_CharT>::__do_date_order()       \
2048*0fca6ea1SDimitry Andric        const;                                                                                                         \
204906c3fb27SDimitry Andric    extern template _LIBCPP_EXPORTED_FROM_ABI __time_get_storage<_CharT>::__time_get_storage(const char*);             \
205006c3fb27SDimitry Andric    extern template _LIBCPP_EXPORTED_FROM_ABI __time_get_storage<_CharT>::__time_get_storage(const string&);           \
205106c3fb27SDimitry Andric    extern template _LIBCPP_EXPORTED_FROM_ABI void __time_get_storage<_CharT>::init(const ctype<_CharT>&);             \
2052cb14a3feSDimitry Andric    extern template _LIBCPP_EXPORTED_FROM_ABI __time_get_storage<_CharT>::string_type                                  \
2053cb14a3feSDimitry Andric    __time_get_storage<_CharT>::__analyze(char, const ctype<_CharT>&);                                                 \
20540b57cec5SDimitry Andric    /**/
20550b57cec5SDimitry Andric
20560b57cec5SDimitry Andric_LIBCPP_TIME_GET_STORAGE_EXPLICIT_INSTANTIATION(char)
2057bdd1243dSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
20580b57cec5SDimitry Andric_LIBCPP_TIME_GET_STORAGE_EXPLICIT_INSTANTIATION(wchar_t)
2059bdd1243dSDimitry Andric#  endif
20600b57cec5SDimitry Andric#  undef _LIBCPP_TIME_GET_STORAGE_EXPLICIT_INSTANTIATION
20610b57cec5SDimitry Andric
20620b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
20630b57cec5SDimitry Andricclass _LIBCPP_TEMPLATE_VIS time_get_byname
20640b57cec5SDimitry Andric    : public time_get<_CharT, _InputIterator>,
2065cb14a3feSDimitry Andric      private __time_get_storage<_CharT> {
20660b57cec5SDimitry Andricpublic:
20670b57cec5SDimitry Andric  typedef time_base::dateorder dateorder;
20680b57cec5SDimitry Andric  typedef _InputIterator iter_type;
20690b57cec5SDimitry Andric  typedef _CharT char_type;
20700b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
20710b57cec5SDimitry Andric
2072cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_get_byname(const char* __nm, size_t __refs = 0)
2073cb14a3feSDimitry Andric      : time_get<_CharT, _InputIterator>(__refs), __time_get_storage<_CharT>(__nm) {}
2074cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_get_byname(const string& __nm, size_t __refs = 0)
2075cb14a3feSDimitry Andric      : time_get<_CharT, _InputIterator>(__refs), __time_get_storage<_CharT>(__nm) {}
20760b57cec5SDimitry Andric
20770b57cec5SDimitry Andricprotected:
2078bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~time_get_byname() override {}
20790b57cec5SDimitry Andric
2080bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL dateorder do_date_order() const override { return this->__do_date_order(); }
2081cb14a3feSDimitry Andric
20820b57cec5SDimitry Andricprivate:
2083bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type* __weeks() const override { return this->__weeks_; }
2084bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type* __months() const override { return this->__months_; }
2085bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type* __am_pm() const override { return this->__am_pm_; }
2086bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type& __c() const override { return this->__c_; }
2087bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type& __r() const override { return this->__r_; }
2088bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type& __x() const override { return this->__x_; }
2089bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL const string_type& __X() const override { return this->__X_; }
20900b57cec5SDimitry Andric};
20910b57cec5SDimitry Andric
209281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_get_byname<char>;
2093349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
209481ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_get_byname<wchar_t>;
2095349cc55cSDimitry Andric#  endif
20960b57cec5SDimitry Andric
2097cb14a3feSDimitry Andricclass _LIBCPP_EXPORTED_FROM_ABI __time_put {
20980b57cec5SDimitry Andric  locale_t __loc_;
2099cb14a3feSDimitry Andric
21000b57cec5SDimitry Andricprotected:
21015f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI __time_put() : __loc_(_LIBCPP_GET_C_LOCALE) {}
21020b57cec5SDimitry Andric  __time_put(const char* __nm);
21030b57cec5SDimitry Andric  __time_put(const string& __nm);
21040b57cec5SDimitry Andric  ~__time_put();
2105cb14a3feSDimitry Andric  void __do_put(char* __nb, char*& __ne, const tm* __tm, char __fmt, char __mod) const;
2106bdd1243dSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
2107cb14a3feSDimitry Andric  void __do_put(wchar_t* __wb, wchar_t*& __we, const tm* __tm, char __fmt, char __mod) const;
2108bdd1243dSDimitry Andric#  endif
21090b57cec5SDimitry Andric};
21100b57cec5SDimitry Andric
21110b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
2112cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS time_put : public locale::facet, private __time_put {
21130b57cec5SDimitry Andricpublic:
21140b57cec5SDimitry Andric  typedef _CharT char_type;
21150b57cec5SDimitry Andric  typedef _OutputIterator iter_type;
21160b57cec5SDimitry Andric
2117cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_put(size_t __refs = 0) : locale::facet(__refs) {}
21180b57cec5SDimitry Andric
2119cb14a3feSDimitry Andric  iter_type
2120cb14a3feSDimitry Andric  put(iter_type __s, ios_base& __iob, char_type __fl, const tm* __tm, const char_type* __pb, const char_type* __pe)
2121cb14a3feSDimitry Andric      const;
21220b57cec5SDimitry Andric
2123cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
2124cb14a3feSDimitry Andric  put(iter_type __s, ios_base& __iob, char_type __fl, const tm* __tm, char __fmt, char __mod = 0) const {
21250b57cec5SDimitry Andric    return do_put(__s, __iob, __fl, __tm, __fmt, __mod);
21260b57cec5SDimitry Andric  }
21270b57cec5SDimitry Andric
21280b57cec5SDimitry Andric  static locale::id id;
21290b57cec5SDimitry Andric
21300b57cec5SDimitry Andricprotected:
2131bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~time_put() override {}
2132cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, ios_base&, char_type, const tm* __tm, char __fmt, char __mod) const;
21330b57cec5SDimitry Andric
2134cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_put(const char* __nm, size_t __refs) : locale::facet(__refs), __time_put(__nm) {}
2135cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_put(const string& __nm, size_t __refs)
2136cb14a3feSDimitry Andric      : locale::facet(__refs), __time_put(__nm) {}
21370b57cec5SDimitry Andric};
21380b57cec5SDimitry Andric
21390b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
2140cb14a3feSDimitry Andriclocale::id time_put<_CharT, _OutputIterator>::id;
21410b57cec5SDimitry Andric
21420b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
2143cb14a3feSDimitry Andric_OutputIterator time_put<_CharT, _OutputIterator>::put(
2144cb14a3feSDimitry Andric    iter_type __s, ios_base& __iob, char_type __fl, const tm* __tm, const char_type* __pb, const char_type* __pe)
2145cb14a3feSDimitry Andric    const {
2146bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__iob.getloc());
2147cb14a3feSDimitry Andric  for (; __pb != __pe; ++__pb) {
2148cb14a3feSDimitry Andric    if (__ct.narrow(*__pb, 0) == '%') {
2149cb14a3feSDimitry Andric      if (++__pb == __pe) {
21500b57cec5SDimitry Andric        *__s++ = __pb[-1];
21510b57cec5SDimitry Andric        break;
21520b57cec5SDimitry Andric      }
21530b57cec5SDimitry Andric      char __mod = 0;
21540b57cec5SDimitry Andric      char __fmt = __ct.narrow(*__pb, 0);
2155cb14a3feSDimitry Andric      if (__fmt == 'E' || __fmt == 'O') {
2156cb14a3feSDimitry Andric        if (++__pb == __pe) {
21570b57cec5SDimitry Andric          *__s++ = __pb[-2];
21580b57cec5SDimitry Andric          *__s++ = __pb[-1];
21590b57cec5SDimitry Andric          break;
21600b57cec5SDimitry Andric        }
21610b57cec5SDimitry Andric        __mod = __fmt;
21620b57cec5SDimitry Andric        __fmt = __ct.narrow(*__pb, 0);
21630b57cec5SDimitry Andric      }
21640b57cec5SDimitry Andric      __s = do_put(__s, __iob, __fl, __tm, __fmt, __mod);
2165cb14a3feSDimitry Andric    } else
21660b57cec5SDimitry Andric      *__s++ = *__pb;
21670b57cec5SDimitry Andric  }
21680b57cec5SDimitry Andric  return __s;
21690b57cec5SDimitry Andric}
21700b57cec5SDimitry Andric
21710b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
2172cb14a3feSDimitry Andric_OutputIterator time_put<_CharT, _OutputIterator>::do_put(
2173cb14a3feSDimitry Andric    iter_type __s, ios_base&, char_type, const tm* __tm, char __fmt, char __mod) const {
21740b57cec5SDimitry Andric  char_type __nar[100];
21750b57cec5SDimitry Andric  char_type* __nb = __nar;
21760b57cec5SDimitry Andric  char_type* __ne = __nb + 100;
21770b57cec5SDimitry Andric  __do_put(__nb, __ne, __tm, __fmt, __mod);
21785f757f3fSDimitry Andric  return std::copy(__nb, __ne, __s);
21790b57cec5SDimitry Andric}
21800b57cec5SDimitry Andric
218181ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_put<char>;
2182349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
218381ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_put<wchar_t>;
2184349cc55cSDimitry Andric#  endif
21850b57cec5SDimitry Andric
21860b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
2187cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS time_put_byname : public time_put<_CharT, _OutputIterator> {
21880b57cec5SDimitry Andricpublic:
2189cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_put_byname(const char* __nm, size_t __refs = 0)
21900b57cec5SDimitry Andric      : time_put<_CharT, _OutputIterator>(__nm, __refs) {}
21910b57cec5SDimitry Andric
2192cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit time_put_byname(const string& __nm, size_t __refs = 0)
21930b57cec5SDimitry Andric      : time_put<_CharT, _OutputIterator>(__nm, __refs) {}
21940b57cec5SDimitry Andric
21950b57cec5SDimitry Andricprotected:
2196bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~time_put_byname() override {}
21970b57cec5SDimitry Andric};
21980b57cec5SDimitry Andric
219981ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_put_byname<char>;
2200349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
220181ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS time_put_byname<wchar_t>;
2202349cc55cSDimitry Andric#  endif
22030b57cec5SDimitry Andric
22040b57cec5SDimitry Andric// money_base
22050b57cec5SDimitry Andric
2206cb14a3feSDimitry Andricclass _LIBCPP_EXPORTED_FROM_ABI money_base {
22070b57cec5SDimitry Andricpublic:
22080b57cec5SDimitry Andric  enum part { none, space, symbol, sign, value };
2209cb14a3feSDimitry Andric  struct pattern {
2210cb14a3feSDimitry Andric    char field[4];
2211cb14a3feSDimitry Andric  };
22120b57cec5SDimitry Andric
22135f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI money_base() {}
22140b57cec5SDimitry Andric};
22150b57cec5SDimitry Andric
22160b57cec5SDimitry Andric// moneypunct
22170b57cec5SDimitry Andric
22180b57cec5SDimitry Andrictemplate <class _CharT, bool _International = false>
2219cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS moneypunct : public locale::facet, public money_base {
22200b57cec5SDimitry Andricpublic:
22210b57cec5SDimitry Andric  typedef _CharT char_type;
22220b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
22230b57cec5SDimitry Andric
2224cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit moneypunct(size_t __refs = 0) : locale::facet(__refs) {}
22250b57cec5SDimitry Andric
22265f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI char_type decimal_point() const { return do_decimal_point(); }
22275f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI char_type thousands_sep() const { return do_thousands_sep(); }
22285f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI string grouping() const { return do_grouping(); }
22295f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI string_type curr_symbol() const { return do_curr_symbol(); }
22305f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI string_type positive_sign() const { return do_positive_sign(); }
22315f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI string_type negative_sign() const { return do_negative_sign(); }
22325f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI int frac_digits() const { return do_frac_digits(); }
22335f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI pattern pos_format() const { return do_pos_format(); }
22345f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI pattern neg_format() const { return do_neg_format(); }
22350b57cec5SDimitry Andric
22360b57cec5SDimitry Andric  static locale::id id;
22370b57cec5SDimitry Andric  static const bool intl = _International;
22380b57cec5SDimitry Andric
22390b57cec5SDimitry Andricprotected:
2240bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~moneypunct() override {}
22410b57cec5SDimitry Andric
22420b57cec5SDimitry Andric  virtual char_type do_decimal_point() const { return numeric_limits<char_type>::max(); }
22430b57cec5SDimitry Andric  virtual char_type do_thousands_sep() const { return numeric_limits<char_type>::max(); }
22440b57cec5SDimitry Andric  virtual string do_grouping() const { return string(); }
22450b57cec5SDimitry Andric  virtual string_type do_curr_symbol() const { return string_type(); }
22460b57cec5SDimitry Andric  virtual string_type do_positive_sign() const { return string_type(); }
22470b57cec5SDimitry Andric  virtual string_type do_negative_sign() const { return string_type(1, '-'); }
22480b57cec5SDimitry Andric  virtual int do_frac_digits() const { return 0; }
2249cb14a3feSDimitry Andric  virtual pattern do_pos_format() const {
2250cb14a3feSDimitry Andric    pattern __p = {{symbol, sign, none, value}};
2251cb14a3feSDimitry Andric    return __p;
2252cb14a3feSDimitry Andric  }
2253cb14a3feSDimitry Andric  virtual pattern do_neg_format() const {
2254cb14a3feSDimitry Andric    pattern __p = {{symbol, sign, none, value}};
2255cb14a3feSDimitry Andric    return __p;
2256cb14a3feSDimitry Andric  }
22570b57cec5SDimitry Andric};
22580b57cec5SDimitry Andric
22590b57cec5SDimitry Andrictemplate <class _CharT, bool _International>
2260cb14a3feSDimitry Andriclocale::id moneypunct<_CharT, _International>::id;
22610b57cec5SDimitry Andric
22620b57cec5SDimitry Andrictemplate <class _CharT, bool _International>
2263cb14a3feSDimitry Andricconst bool moneypunct<_CharT, _International>::intl;
22640b57cec5SDimitry Andric
226581ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct<char, false>;
226681ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct<char, true>;
2267349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
226881ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct<wchar_t, false>;
226981ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct<wchar_t, true>;
2270349cc55cSDimitry Andric#  endif
22710b57cec5SDimitry Andric
22720b57cec5SDimitry Andric// moneypunct_byname
22730b57cec5SDimitry Andric
22740b57cec5SDimitry Andrictemplate <class _CharT, bool _International = false>
2275cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS moneypunct_byname : public moneypunct<_CharT, _International> {
22760b57cec5SDimitry Andricpublic:
22770b57cec5SDimitry Andric  typedef money_base::pattern pattern;
22780b57cec5SDimitry Andric  typedef _CharT char_type;
22790b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
22800b57cec5SDimitry Andric
2281cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit moneypunct_byname(const char* __nm, size_t __refs = 0)
2282cb14a3feSDimitry Andric      : moneypunct<_CharT, _International>(__refs) {
2283cb14a3feSDimitry Andric    init(__nm);
2284cb14a3feSDimitry Andric  }
22850b57cec5SDimitry Andric
2286cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit moneypunct_byname(const string& __nm, size_t __refs = 0)
2287cb14a3feSDimitry Andric      : moneypunct<_CharT, _International>(__refs) {
2288cb14a3feSDimitry Andric    init(__nm.c_str());
2289cb14a3feSDimitry Andric  }
22900b57cec5SDimitry Andric
22910b57cec5SDimitry Andricprotected:
2292bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~moneypunct_byname() override {}
22930b57cec5SDimitry Andric
2294bdd1243dSDimitry Andric  char_type do_decimal_point() const override { return __decimal_point_; }
2295bdd1243dSDimitry Andric  char_type do_thousands_sep() const override { return __thousands_sep_; }
2296bdd1243dSDimitry Andric  string do_grouping() const override { return __grouping_; }
2297bdd1243dSDimitry Andric  string_type do_curr_symbol() const override { return __curr_symbol_; }
2298bdd1243dSDimitry Andric  string_type do_positive_sign() const override { return __positive_sign_; }
2299bdd1243dSDimitry Andric  string_type do_negative_sign() const override { return __negative_sign_; }
2300bdd1243dSDimitry Andric  int do_frac_digits() const override { return __frac_digits_; }
2301bdd1243dSDimitry Andric  pattern do_pos_format() const override { return __pos_format_; }
2302bdd1243dSDimitry Andric  pattern do_neg_format() const override { return __neg_format_; }
23030b57cec5SDimitry Andric
23040b57cec5SDimitry Andricprivate:
23050b57cec5SDimitry Andric  char_type __decimal_point_;
23060b57cec5SDimitry Andric  char_type __thousands_sep_;
23070b57cec5SDimitry Andric  string __grouping_;
23080b57cec5SDimitry Andric  string_type __curr_symbol_;
23090b57cec5SDimitry Andric  string_type __positive_sign_;
23100b57cec5SDimitry Andric  string_type __negative_sign_;
23110b57cec5SDimitry Andric  int __frac_digits_;
23120b57cec5SDimitry Andric  pattern __pos_format_;
23130b57cec5SDimitry Andric  pattern __neg_format_;
23140b57cec5SDimitry Andric
23150b57cec5SDimitry Andric  void init(const char*);
23160b57cec5SDimitry Andric};
23170b57cec5SDimitry Andric
2318cb14a3feSDimitry Andrictemplate <>
2319cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI void moneypunct_byname<char, false>::init(const char*);
2320cb14a3feSDimitry Andrictemplate <>
2321cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI void moneypunct_byname<char, true>::init(const char*);
232281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct_byname<char, false>;
232381ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct_byname<char, true>;
2324349cc55cSDimitry Andric
2325349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
2326cb14a3feSDimitry Andrictemplate <>
2327cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI void moneypunct_byname<wchar_t, false>::init(const char*);
2328cb14a3feSDimitry Andrictemplate <>
2329cb14a3feSDimitry Andric_LIBCPP_EXPORTED_FROM_ABI void moneypunct_byname<wchar_t, true>::init(const char*);
233081ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct_byname<wchar_t, false>;
233181ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS moneypunct_byname<wchar_t, true>;
2332349cc55cSDimitry Andric#  endif
23330b57cec5SDimitry Andric
23340b57cec5SDimitry Andric// money_get
23350b57cec5SDimitry Andric
23360b57cec5SDimitry Andrictemplate <class _CharT>
2337cb14a3feSDimitry Andricclass __money_get {
23380b57cec5SDimitry Andricprotected:
23390b57cec5SDimitry Andric  typedef _CharT char_type;
23400b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
23410b57cec5SDimitry Andric
23425f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI __money_get() {}
23430b57cec5SDimitry Andric
2344cb14a3feSDimitry Andric  static void __gather_info(
2345cb14a3feSDimitry Andric      bool __intl,
2346cb14a3feSDimitry Andric      const locale& __loc,
2347cb14a3feSDimitry Andric      money_base::pattern& __pat,
2348cb14a3feSDimitry Andric      char_type& __dp,
2349cb14a3feSDimitry Andric      char_type& __ts,
2350cb14a3feSDimitry Andric      string& __grp,
2351cb14a3feSDimitry Andric      string_type& __sym,
2352cb14a3feSDimitry Andric      string_type& __psn,
2353cb14a3feSDimitry Andric      string_type& __nsn,
2354cb14a3feSDimitry Andric      int& __fd);
23550b57cec5SDimitry Andric};
23560b57cec5SDimitry Andric
23570b57cec5SDimitry Andrictemplate <class _CharT>
2358cb14a3feSDimitry Andricvoid __money_get<_CharT>::__gather_info(
2359cb14a3feSDimitry Andric    bool __intl,
2360cb14a3feSDimitry Andric    const locale& __loc,
2361cb14a3feSDimitry Andric    money_base::pattern& __pat,
2362cb14a3feSDimitry Andric    char_type& __dp,
2363cb14a3feSDimitry Andric    char_type& __ts,
2364cb14a3feSDimitry Andric    string& __grp,
2365cb14a3feSDimitry Andric    string_type& __sym,
2366cb14a3feSDimitry Andric    string_type& __psn,
2367cb14a3feSDimitry Andric    string_type& __nsn,
2368cb14a3feSDimitry Andric    int& __fd) {
2369cb14a3feSDimitry Andric  if (__intl) {
2370cb14a3feSDimitry Andric    const moneypunct<char_type, true>& __mp = std::use_facet<moneypunct<char_type, true> >(__loc);
23710b57cec5SDimitry Andric    __pat                                   = __mp.neg_format();
23720b57cec5SDimitry Andric    __nsn                                   = __mp.negative_sign();
23730b57cec5SDimitry Andric    __psn                                   = __mp.positive_sign();
23740b57cec5SDimitry Andric    __dp                                    = __mp.decimal_point();
23750b57cec5SDimitry Andric    __ts                                    = __mp.thousands_sep();
23760b57cec5SDimitry Andric    __grp                                   = __mp.grouping();
23770b57cec5SDimitry Andric    __sym                                   = __mp.curr_symbol();
23780b57cec5SDimitry Andric    __fd                                    = __mp.frac_digits();
2379cb14a3feSDimitry Andric  } else {
2380cb14a3feSDimitry Andric    const moneypunct<char_type, false>& __mp = std::use_facet<moneypunct<char_type, false> >(__loc);
23810b57cec5SDimitry Andric    __pat                                    = __mp.neg_format();
23820b57cec5SDimitry Andric    __nsn                                    = __mp.negative_sign();
23830b57cec5SDimitry Andric    __psn                                    = __mp.positive_sign();
23840b57cec5SDimitry Andric    __dp                                     = __mp.decimal_point();
23850b57cec5SDimitry Andric    __ts                                     = __mp.thousands_sep();
23860b57cec5SDimitry Andric    __grp                                    = __mp.grouping();
23870b57cec5SDimitry Andric    __sym                                    = __mp.curr_symbol();
23880b57cec5SDimitry Andric    __fd                                     = __mp.frac_digits();
23890b57cec5SDimitry Andric  }
23900b57cec5SDimitry Andric}
23910b57cec5SDimitry Andric
239281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __money_get<char>;
2393349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
239481ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __money_get<wchar_t>;
2395349cc55cSDimitry Andric#  endif
23960b57cec5SDimitry Andric
23970b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator = istreambuf_iterator<_CharT> >
2398cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS money_get : public locale::facet, private __money_get<_CharT> {
23990b57cec5SDimitry Andricpublic:
24000b57cec5SDimitry Andric  typedef _CharT char_type;
24010b57cec5SDimitry Andric  typedef _InputIterator iter_type;
24020b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
24030b57cec5SDimitry Andric
2404cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit money_get(size_t __refs = 0) : locale::facet(__refs) {}
24050b57cec5SDimitry Andric
2406cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
2407cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, bool __intl, ios_base& __iob, ios_base::iostate& __err, long double& __v) const {
24080b57cec5SDimitry Andric    return do_get(__b, __e, __intl, __iob, __err, __v);
24090b57cec5SDimitry Andric  }
24100b57cec5SDimitry Andric
2411cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
2412cb14a3feSDimitry Andric  get(iter_type __b, iter_type __e, bool __intl, ios_base& __iob, ios_base::iostate& __err, string_type& __v) const {
24130b57cec5SDimitry Andric    return do_get(__b, __e, __intl, __iob, __err, __v);
24140b57cec5SDimitry Andric  }
24150b57cec5SDimitry Andric
24160b57cec5SDimitry Andric  static locale::id id;
24170b57cec5SDimitry Andric
24180b57cec5SDimitry Andricprotected:
2419bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~money_get() override {}
24200b57cec5SDimitry Andric
2421cb14a3feSDimitry Andric  virtual iter_type
2422cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, bool __intl, ios_base& __iob, ios_base::iostate& __err, long double& __v) const;
2423cb14a3feSDimitry Andric  virtual iter_type
2424cb14a3feSDimitry Andric  do_get(iter_type __b, iter_type __e, bool __intl, ios_base& __iob, ios_base::iostate& __err, string_type& __v) const;
24250b57cec5SDimitry Andric
24260b57cec5SDimitry Andricprivate:
2427cb14a3feSDimitry Andric  static bool __do_get(
2428cb14a3feSDimitry Andric      iter_type& __b,
2429cb14a3feSDimitry Andric      iter_type __e,
2430cb14a3feSDimitry Andric      bool __intl,
2431cb14a3feSDimitry Andric      const locale& __loc,
2432cb14a3feSDimitry Andric      ios_base::fmtflags __flags,
2433cb14a3feSDimitry Andric      ios_base::iostate& __err,
2434cb14a3feSDimitry Andric      bool& __neg,
2435cb14a3feSDimitry Andric      const ctype<char_type>& __ct,
24360b57cec5SDimitry Andric      unique_ptr<char_type, void (*)(void*)>& __wb,
2437cb14a3feSDimitry Andric      char_type*& __wn,
2438cb14a3feSDimitry Andric      char_type* __we);
24390b57cec5SDimitry Andric};
24400b57cec5SDimitry Andric
24410b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
2442cb14a3feSDimitry Andriclocale::id money_get<_CharT, _InputIterator>::id;
24430b57cec5SDimitry Andric
244406c3fb27SDimitry Andric_LIBCPP_EXPORTED_FROM_ABI void __do_nothing(void*);
24450b57cec5SDimitry Andric
24460b57cec5SDimitry Andrictemplate <class _Tp>
2447cb14a3feSDimitry Andric_LIBCPP_HIDE_FROM_ABI void __double_or_nothing(unique_ptr<_Tp, void (*)(void*)>& __b, _Tp*& __n, _Tp*& __e) {
24480b57cec5SDimitry Andric  bool __owns      = __b.get_deleter() != __do_nothing;
24490b57cec5SDimitry Andric  size_t __cur_cap = static_cast<size_t>(__e - __b.get()) * sizeof(_Tp);
2450cb14a3feSDimitry Andric  size_t __new_cap = __cur_cap < numeric_limits<size_t>::max() / 2 ? 2 * __cur_cap : numeric_limits<size_t>::max();
24510b57cec5SDimitry Andric  if (__new_cap == 0)
24520b57cec5SDimitry Andric    __new_cap = sizeof(_Tp);
24530b57cec5SDimitry Andric  size_t __n_off = static_cast<size_t>(__n - __b.get());
2454bdd1243dSDimitry Andric  _Tp* __t       = (_Tp*)std::realloc(__owns ? __b.get() : 0, __new_cap);
24550b57cec5SDimitry Andric  if (__t == 0)
24560b57cec5SDimitry Andric    __throw_bad_alloc();
24570b57cec5SDimitry Andric  if (__owns)
24580b57cec5SDimitry Andric    __b.release();
24590b57cec5SDimitry Andric  __b = unique_ptr<_Tp, void (*)(void*)>(__t, free);
24600b57cec5SDimitry Andric  __new_cap /= sizeof(_Tp);
24610b57cec5SDimitry Andric  __n = __b.get() + __n_off;
24620b57cec5SDimitry Andric  __e = __b.get() + __new_cap;
24630b57cec5SDimitry Andric}
24640b57cec5SDimitry Andric
24650b57cec5SDimitry Andric// true == success
24660b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
2467cb14a3feSDimitry Andricbool money_get<_CharT, _InputIterator>::__do_get(
2468cb14a3feSDimitry Andric    iter_type& __b,
2469cb14a3feSDimitry Andric    iter_type __e,
2470cb14a3feSDimitry Andric    bool __intl,
2471cb14a3feSDimitry Andric    const locale& __loc,
24720b57cec5SDimitry Andric    ios_base::fmtflags __flags,
24730b57cec5SDimitry Andric    ios_base::iostate& __err,
24740b57cec5SDimitry Andric    bool& __neg,
24750b57cec5SDimitry Andric    const ctype<char_type>& __ct,
24760b57cec5SDimitry Andric    unique_ptr<char_type, void (*)(void*)>& __wb,
2477cb14a3feSDimitry Andric    char_type*& __wn,
2478cb14a3feSDimitry Andric    char_type* __we) {
2479349cc55cSDimitry Andric  if (__b == __e) {
2480349cc55cSDimitry Andric    __err |= ios_base::failbit;
2481349cc55cSDimitry Andric    return false;
2482349cc55cSDimitry Andric  }
24830b57cec5SDimitry Andric  const unsigned __bz = 100;
24840b57cec5SDimitry Andric  unsigned __gbuf[__bz];
24850b57cec5SDimitry Andric  unique_ptr<unsigned, void (*)(void*)> __gb(__gbuf, __do_nothing);
24860b57cec5SDimitry Andric  unsigned* __gn = __gb.get();
24870b57cec5SDimitry Andric  unsigned* __ge = __gn + __bz;
24880b57cec5SDimitry Andric  money_base::pattern __pat;
24890b57cec5SDimitry Andric  char_type __dp;
24900b57cec5SDimitry Andric  char_type __ts;
24910b57cec5SDimitry Andric  string __grp;
24920b57cec5SDimitry Andric  string_type __sym;
24930b57cec5SDimitry Andric  string_type __psn;
24940b57cec5SDimitry Andric  string_type __nsn;
24950b57cec5SDimitry Andric  // Capture the spaces read into money_base::{space,none} so they
24960b57cec5SDimitry Andric  // can be compared to initial spaces in __sym.
24970b57cec5SDimitry Andric  string_type __spaces;
24980b57cec5SDimitry Andric  int __fd;
2499cb14a3feSDimitry Andric  __money_get<_CharT>::__gather_info(__intl, __loc, __pat, __dp, __ts, __grp, __sym, __psn, __nsn, __fd);
25000b57cec5SDimitry Andric  const string_type* __trailing_sign = 0;
25010b57cec5SDimitry Andric  __wn                               = __wb.get();
2502cb14a3feSDimitry Andric  for (unsigned __p = 0; __p < 4 && __b != __e; ++__p) {
2503cb14a3feSDimitry Andric    switch (__pat.field[__p]) {
25040b57cec5SDimitry Andric    case money_base::space:
2505cb14a3feSDimitry Andric      if (__p != 3) {
25060b57cec5SDimitry Andric        if (__ct.is(ctype_base::space, *__b))
25070b57cec5SDimitry Andric          __spaces.push_back(*__b++);
2508cb14a3feSDimitry Andric        else {
25090b57cec5SDimitry Andric          __err |= ios_base::failbit;
25100b57cec5SDimitry Andric          return false;
25110b57cec5SDimitry Andric        }
25120b57cec5SDimitry Andric      }
25130b57cec5SDimitry Andric      _LIBCPP_FALLTHROUGH();
25140b57cec5SDimitry Andric    case money_base::none:
2515cb14a3feSDimitry Andric      if (__p != 3) {
25160b57cec5SDimitry Andric        while (__b != __e && __ct.is(ctype_base::space, *__b))
25170b57cec5SDimitry Andric          __spaces.push_back(*__b++);
25180b57cec5SDimitry Andric      }
25190b57cec5SDimitry Andric      break;
25200b57cec5SDimitry Andric    case money_base::sign:
2521cb14a3feSDimitry Andric      if (__psn.size() > 0 && *__b == __psn[0]) {
25220b57cec5SDimitry Andric        ++__b;
2523349cc55cSDimitry Andric        __neg = false;
25240b57cec5SDimitry Andric        if (__psn.size() > 1)
25250b57cec5SDimitry Andric          __trailing_sign = &__psn;
2526349cc55cSDimitry Andric        break;
25270b57cec5SDimitry Andric      }
2528cb14a3feSDimitry Andric      if (__nsn.size() > 0 && *__b == __nsn[0]) {
25290b57cec5SDimitry Andric        ++__b;
25300b57cec5SDimitry Andric        __neg = true;
25310b57cec5SDimitry Andric        if (__nsn.size() > 1)
25320b57cec5SDimitry Andric          __trailing_sign = &__nsn;
2533349cc55cSDimitry Andric        break;
25340b57cec5SDimitry Andric      }
2535cb14a3feSDimitry Andric      if (__psn.size() > 0 && __nsn.size() > 0) { // sign is required
25360b57cec5SDimitry Andric        __err |= ios_base::failbit;
25370b57cec5SDimitry Andric        return false;
25380b57cec5SDimitry Andric      }
2539349cc55cSDimitry Andric      if (__psn.size() == 0 && __nsn.size() == 0)
2540349cc55cSDimitry Andric        // locale has no way of specifying a sign. Use the initial value of __neg as a default
2541349cc55cSDimitry Andric        break;
2542349cc55cSDimitry Andric      __neg = (__nsn.size() == 0);
25430b57cec5SDimitry Andric      break;
2544cb14a3feSDimitry Andric    case money_base::symbol: {
2545cb14a3feSDimitry Andric      bool __more_needed =
2546cb14a3feSDimitry Andric          __trailing_sign || (__p < 2) || (__p == 2 && __pat.field[3] != static_cast<char>(money_base::none));
25470b57cec5SDimitry Andric      bool __sb = (__flags & ios_base::showbase) != 0;
2548cb14a3feSDimitry Andric      if (__sb || __more_needed) {
25490b57cec5SDimitry Andric        typename string_type::const_iterator __sym_space_end = __sym.begin();
2550cb14a3feSDimitry Andric        if (__p > 0 && (__pat.field[__p - 1] == money_base::none || __pat.field[__p - 1] == money_base::space)) {
25510b57cec5SDimitry Andric          // Match spaces we've already read against spaces at
25520b57cec5SDimitry Andric          // the beginning of __sym.
2553cb14a3feSDimitry Andric          while (__sym_space_end != __sym.end() && __ct.is(ctype_base::space, *__sym_space_end))
25540b57cec5SDimitry Andric            ++__sym_space_end;
25550b57cec5SDimitry Andric          const size_t __num_spaces = __sym_space_end - __sym.begin();
25560b57cec5SDimitry Andric          if (__num_spaces > __spaces.size() ||
2557cb14a3feSDimitry Andric              !std::equal(__spaces.end() - __num_spaces, __spaces.end(), __sym.begin())) {
25580b57cec5SDimitry Andric            // No match. Put __sym_space_end back at the
25590b57cec5SDimitry Andric            // beginning of __sym, which will prevent a
25600b57cec5SDimitry Andric            // match in the next loop.
25610b57cec5SDimitry Andric            __sym_space_end = __sym.begin();
25620b57cec5SDimitry Andric          }
25630b57cec5SDimitry Andric        }
25640b57cec5SDimitry Andric        typename string_type::const_iterator __sym_curr_char = __sym_space_end;
2565cb14a3feSDimitry Andric        while (__sym_curr_char != __sym.end() && __b != __e && *__b == *__sym_curr_char) {
25660b57cec5SDimitry Andric          ++__b;
25670b57cec5SDimitry Andric          ++__sym_curr_char;
25680b57cec5SDimitry Andric        }
2569cb14a3feSDimitry Andric        if (__sb && __sym_curr_char != __sym.end()) {
25700b57cec5SDimitry Andric          __err |= ios_base::failbit;
25710b57cec5SDimitry Andric          return false;
25720b57cec5SDimitry Andric        }
25730b57cec5SDimitry Andric      }
2574cb14a3feSDimitry Andric    } break;
2575cb14a3feSDimitry Andric    case money_base::value: {
25760b57cec5SDimitry Andric      unsigned __ng = 0;
2577cb14a3feSDimitry Andric      for (; __b != __e; ++__b) {
25780b57cec5SDimitry Andric        char_type __c = *__b;
2579cb14a3feSDimitry Andric        if (__ct.is(ctype_base::digit, __c)) {
25800b57cec5SDimitry Andric          if (__wn == __we)
2581bdd1243dSDimitry Andric            std::__double_or_nothing(__wb, __wn, __we);
25820b57cec5SDimitry Andric          *__wn++ = __c;
25830b57cec5SDimitry Andric          ++__ng;
2584cb14a3feSDimitry Andric        } else if (__grp.size() > 0 && __ng > 0 && __c == __ts) {
25850b57cec5SDimitry Andric          if (__gn == __ge)
2586bdd1243dSDimitry Andric            std::__double_or_nothing(__gb, __gn, __ge);
25870b57cec5SDimitry Andric          *__gn++ = __ng;
25880b57cec5SDimitry Andric          __ng    = 0;
2589cb14a3feSDimitry Andric        } else
25900b57cec5SDimitry Andric          break;
25910b57cec5SDimitry Andric      }
2592cb14a3feSDimitry Andric      if (__gb.get() != __gn && __ng > 0) {
25930b57cec5SDimitry Andric        if (__gn == __ge)
2594bdd1243dSDimitry Andric          std::__double_or_nothing(__gb, __gn, __ge);
25950b57cec5SDimitry Andric        *__gn++ = __ng;
25960b57cec5SDimitry Andric      }
2597cb14a3feSDimitry Andric      if (__fd > 0) {
2598cb14a3feSDimitry Andric        if (__b == __e || *__b != __dp) {
25990b57cec5SDimitry Andric          __err |= ios_base::failbit;
26000b57cec5SDimitry Andric          return false;
26010b57cec5SDimitry Andric        }
2602cb14a3feSDimitry Andric        for (++__b; __fd > 0; --__fd, ++__b) {
2603cb14a3feSDimitry Andric          if (__b == __e || !__ct.is(ctype_base::digit, *__b)) {
26040b57cec5SDimitry Andric            __err |= ios_base::failbit;
26050b57cec5SDimitry Andric            return false;
26060b57cec5SDimitry Andric          }
26070b57cec5SDimitry Andric          if (__wn == __we)
2608bdd1243dSDimitry Andric            std::__double_or_nothing(__wb, __wn, __we);
26090b57cec5SDimitry Andric          *__wn++ = *__b;
26100b57cec5SDimitry Andric        }
26110b57cec5SDimitry Andric      }
2612cb14a3feSDimitry Andric      if (__wn == __wb.get()) {
26130b57cec5SDimitry Andric        __err |= ios_base::failbit;
26140b57cec5SDimitry Andric        return false;
26150b57cec5SDimitry Andric      }
2616cb14a3feSDimitry Andric    } break;
26170b57cec5SDimitry Andric    }
26180b57cec5SDimitry Andric  }
2619cb14a3feSDimitry Andric  if (__trailing_sign) {
2620cb14a3feSDimitry Andric    for (unsigned __i = 1; __i < __trailing_sign->size(); ++__i, ++__b) {
2621cb14a3feSDimitry Andric      if (__b == __e || *__b != (*__trailing_sign)[__i]) {
26220b57cec5SDimitry Andric        __err |= ios_base::failbit;
26230b57cec5SDimitry Andric        return false;
26240b57cec5SDimitry Andric      }
26250b57cec5SDimitry Andric    }
26260b57cec5SDimitry Andric  }
2627cb14a3feSDimitry Andric  if (__gb.get() != __gn) {
26280b57cec5SDimitry Andric    ios_base::iostate __et = ios_base::goodbit;
26290b57cec5SDimitry Andric    __check_grouping(__grp, __gb.get(), __gn, __et);
2630cb14a3feSDimitry Andric    if (__et) {
26310b57cec5SDimitry Andric      __err |= ios_base::failbit;
26320b57cec5SDimitry Andric      return false;
26330b57cec5SDimitry Andric    }
26340b57cec5SDimitry Andric  }
26350b57cec5SDimitry Andric  return true;
26360b57cec5SDimitry Andric}
26370b57cec5SDimitry Andric
26380b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
2639cb14a3feSDimitry Andric_InputIterator money_get<_CharT, _InputIterator>::do_get(
2640cb14a3feSDimitry Andric    iter_type __b, iter_type __e, bool __intl, ios_base& __iob, ios_base::iostate& __err, long double& __v) const {
26410b57cec5SDimitry Andric  const int __bz = 100;
26420b57cec5SDimitry Andric  char_type __wbuf[__bz];
26430b57cec5SDimitry Andric  unique_ptr<char_type, void (*)(void*)> __wb(__wbuf, __do_nothing);
26440b57cec5SDimitry Andric  char_type* __wn;
26450b57cec5SDimitry Andric  char_type* __we              = __wbuf + __bz;
26460b57cec5SDimitry Andric  locale __loc                 = __iob.getloc();
2647bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__loc);
26480b57cec5SDimitry Andric  bool __neg                   = false;
2649cb14a3feSDimitry Andric  if (__do_get(__b, __e, __intl, __loc, __iob.flags(), __err, __neg, __ct, __wb, __wn, __we)) {
26500b57cec5SDimitry Andric    const char __src[] = "0123456789";
26510b57cec5SDimitry Andric    char_type __atoms[sizeof(__src) - 1];
26520b57cec5SDimitry Andric    __ct.widen(__src, __src + (sizeof(__src) - 1), __atoms);
26530b57cec5SDimitry Andric    char __nbuf[__bz];
26540b57cec5SDimitry Andric    char* __nc = __nbuf;
2655e8d8bef9SDimitry Andric    unique_ptr<char, void (*)(void*)> __h(nullptr, free);
2656cb14a3feSDimitry Andric    if (__wn - __wb.get() > __bz - 2) {
26570b57cec5SDimitry Andric      __h.reset((char*)malloc(static_cast<size_t>(__wn - __wb.get() + 2)));
2658e8d8bef9SDimitry Andric      if (__h.get() == nullptr)
26590b57cec5SDimitry Andric        __throw_bad_alloc();
26600b57cec5SDimitry Andric      __nc = __h.get();
26610b57cec5SDimitry Andric    }
26620b57cec5SDimitry Andric    if (__neg)
26630b57cec5SDimitry Andric      *__nc++ = '-';
26640b57cec5SDimitry Andric    for (const char_type* __w = __wb.get(); __w < __wn; ++__w, ++__nc)
26655f757f3fSDimitry Andric      *__nc = __src[std::find(__atoms, std::end(__atoms), *__w) - __atoms];
26660b57cec5SDimitry Andric    *__nc = char();
26670b57cec5SDimitry Andric    if (sscanf(__nbuf, "%Lf", &__v) != 1)
26680b57cec5SDimitry Andric      __throw_runtime_error("money_get error");
26690b57cec5SDimitry Andric  }
26700b57cec5SDimitry Andric  if (__b == __e)
26710b57cec5SDimitry Andric    __err |= ios_base::eofbit;
26720b57cec5SDimitry Andric  return __b;
26730b57cec5SDimitry Andric}
26740b57cec5SDimitry Andric
26750b57cec5SDimitry Andrictemplate <class _CharT, class _InputIterator>
2676cb14a3feSDimitry Andric_InputIterator money_get<_CharT, _InputIterator>::do_get(
2677cb14a3feSDimitry Andric    iter_type __b, iter_type __e, bool __intl, ios_base& __iob, ios_base::iostate& __err, string_type& __v) const {
26780b57cec5SDimitry Andric  const int __bz = 100;
26790b57cec5SDimitry Andric  char_type __wbuf[__bz];
26800b57cec5SDimitry Andric  unique_ptr<char_type, void (*)(void*)> __wb(__wbuf, __do_nothing);
26810b57cec5SDimitry Andric  char_type* __wn;
26820b57cec5SDimitry Andric  char_type* __we              = __wbuf + __bz;
26830b57cec5SDimitry Andric  locale __loc                 = __iob.getloc();
2684bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__loc);
26850b57cec5SDimitry Andric  bool __neg                   = false;
2686cb14a3feSDimitry Andric  if (__do_get(__b, __e, __intl, __loc, __iob.flags(), __err, __neg, __ct, __wb, __wn, __we)) {
26870b57cec5SDimitry Andric    __v.clear();
26880b57cec5SDimitry Andric    if (__neg)
26890b57cec5SDimitry Andric      __v.push_back(__ct.widen('-'));
26900b57cec5SDimitry Andric    char_type __z = __ct.widen('0');
26910b57cec5SDimitry Andric    char_type* __w;
26920b57cec5SDimitry Andric    for (__w = __wb.get(); __w < __wn - 1; ++__w)
26930b57cec5SDimitry Andric      if (*__w != __z)
26940b57cec5SDimitry Andric        break;
26950b57cec5SDimitry Andric    __v.append(__w, __wn);
26960b57cec5SDimitry Andric  }
26970b57cec5SDimitry Andric  if (__b == __e)
26980b57cec5SDimitry Andric    __err |= ios_base::eofbit;
26990b57cec5SDimitry Andric  return __b;
27000b57cec5SDimitry Andric}
27010b57cec5SDimitry Andric
270281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS money_get<char>;
2703349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
270481ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS money_get<wchar_t>;
2705349cc55cSDimitry Andric#  endif
27060b57cec5SDimitry Andric
27070b57cec5SDimitry Andric// money_put
27080b57cec5SDimitry Andric
27090b57cec5SDimitry Andrictemplate <class _CharT>
2710cb14a3feSDimitry Andricclass __money_put {
27110b57cec5SDimitry Andricprotected:
27120b57cec5SDimitry Andric  typedef _CharT char_type;
27130b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
27140b57cec5SDimitry Andric
27155f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI __money_put() {}
27160b57cec5SDimitry Andric
2717cb14a3feSDimitry Andric  static void __gather_info(
2718cb14a3feSDimitry Andric      bool __intl,
2719cb14a3feSDimitry Andric      bool __neg,
2720cb14a3feSDimitry Andric      const locale& __loc,
2721cb14a3feSDimitry Andric      money_base::pattern& __pat,
2722cb14a3feSDimitry Andric      char_type& __dp,
2723cb14a3feSDimitry Andric      char_type& __ts,
2724cb14a3feSDimitry Andric      string& __grp,
2725cb14a3feSDimitry Andric      string_type& __sym,
2726cb14a3feSDimitry Andric      string_type& __sn,
27270b57cec5SDimitry Andric      int& __fd);
2728cb14a3feSDimitry Andric  static void __format(
2729cb14a3feSDimitry Andric      char_type* __mb,
2730cb14a3feSDimitry Andric      char_type*& __mi,
2731cb14a3feSDimitry Andric      char_type*& __me,
27320b57cec5SDimitry Andric      ios_base::fmtflags __flags,
2733cb14a3feSDimitry Andric      const char_type* __db,
2734cb14a3feSDimitry Andric      const char_type* __de,
2735cb14a3feSDimitry Andric      const ctype<char_type>& __ct,
2736cb14a3feSDimitry Andric      bool __neg,
2737cb14a3feSDimitry Andric      const money_base::pattern& __pat,
2738cb14a3feSDimitry Andric      char_type __dp,
2739cb14a3feSDimitry Andric      char_type __ts,
2740cb14a3feSDimitry Andric      const string& __grp,
2741cb14a3feSDimitry Andric      const string_type& __sym,
2742cb14a3feSDimitry Andric      const string_type& __sn,
27430b57cec5SDimitry Andric      int __fd);
27440b57cec5SDimitry Andric};
27450b57cec5SDimitry Andric
27460b57cec5SDimitry Andrictemplate <class _CharT>
2747cb14a3feSDimitry Andricvoid __money_put<_CharT>::__gather_info(
2748cb14a3feSDimitry Andric    bool __intl,
2749cb14a3feSDimitry Andric    bool __neg,
2750cb14a3feSDimitry Andric    const locale& __loc,
2751cb14a3feSDimitry Andric    money_base::pattern& __pat,
2752cb14a3feSDimitry Andric    char_type& __dp,
2753cb14a3feSDimitry Andric    char_type& __ts,
2754cb14a3feSDimitry Andric    string& __grp,
2755cb14a3feSDimitry Andric    string_type& __sym,
2756cb14a3feSDimitry Andric    string_type& __sn,
2757cb14a3feSDimitry Andric    int& __fd) {
2758cb14a3feSDimitry Andric  if (__intl) {
2759cb14a3feSDimitry Andric    const moneypunct<char_type, true>& __mp = std::use_facet<moneypunct<char_type, true> >(__loc);
2760cb14a3feSDimitry Andric    if (__neg) {
27610b57cec5SDimitry Andric      __pat = __mp.neg_format();
27620b57cec5SDimitry Andric      __sn  = __mp.negative_sign();
2763cb14a3feSDimitry Andric    } else {
27640b57cec5SDimitry Andric      __pat = __mp.pos_format();
27650b57cec5SDimitry Andric      __sn  = __mp.positive_sign();
27660b57cec5SDimitry Andric    }
27670b57cec5SDimitry Andric    __dp  = __mp.decimal_point();
27680b57cec5SDimitry Andric    __ts  = __mp.thousands_sep();
27690b57cec5SDimitry Andric    __grp = __mp.grouping();
27700b57cec5SDimitry Andric    __sym = __mp.curr_symbol();
27710b57cec5SDimitry Andric    __fd  = __mp.frac_digits();
2772cb14a3feSDimitry Andric  } else {
2773cb14a3feSDimitry Andric    const moneypunct<char_type, false>& __mp = std::use_facet<moneypunct<char_type, false> >(__loc);
2774cb14a3feSDimitry Andric    if (__neg) {
27750b57cec5SDimitry Andric      __pat = __mp.neg_format();
27760b57cec5SDimitry Andric      __sn  = __mp.negative_sign();
2777cb14a3feSDimitry Andric    } else {
27780b57cec5SDimitry Andric      __pat = __mp.pos_format();
27790b57cec5SDimitry Andric      __sn  = __mp.positive_sign();
27800b57cec5SDimitry Andric    }
27810b57cec5SDimitry Andric    __dp  = __mp.decimal_point();
27820b57cec5SDimitry Andric    __ts  = __mp.thousands_sep();
27830b57cec5SDimitry Andric    __grp = __mp.grouping();
27840b57cec5SDimitry Andric    __sym = __mp.curr_symbol();
27850b57cec5SDimitry Andric    __fd  = __mp.frac_digits();
27860b57cec5SDimitry Andric  }
27870b57cec5SDimitry Andric}
27880b57cec5SDimitry Andric
27890b57cec5SDimitry Andrictemplate <class _CharT>
2790cb14a3feSDimitry Andricvoid __money_put<_CharT>::__format(
2791cb14a3feSDimitry Andric    char_type* __mb,
2792cb14a3feSDimitry Andric    char_type*& __mi,
2793cb14a3feSDimitry Andric    char_type*& __me,
27940b57cec5SDimitry Andric    ios_base::fmtflags __flags,
2795cb14a3feSDimitry Andric    const char_type* __db,
2796cb14a3feSDimitry Andric    const char_type* __de,
2797cb14a3feSDimitry Andric    const ctype<char_type>& __ct,
2798cb14a3feSDimitry Andric    bool __neg,
2799cb14a3feSDimitry Andric    const money_base::pattern& __pat,
2800cb14a3feSDimitry Andric    char_type __dp,
2801cb14a3feSDimitry Andric    char_type __ts,
2802cb14a3feSDimitry Andric    const string& __grp,
2803cb14a3feSDimitry Andric    const string_type& __sym,
2804cb14a3feSDimitry Andric    const string_type& __sn,
2805cb14a3feSDimitry Andric    int __fd) {
28060b57cec5SDimitry Andric  __me = __mb;
2807cb14a3feSDimitry Andric  for (char __p : __pat.field) {
2808cb14a3feSDimitry Andric    switch (__p) {
28090b57cec5SDimitry Andric    case money_base::none:
28100b57cec5SDimitry Andric      __mi = __me;
28110b57cec5SDimitry Andric      break;
28120b57cec5SDimitry Andric    case money_base::space:
28130b57cec5SDimitry Andric      __mi    = __me;
28140b57cec5SDimitry Andric      *__me++ = __ct.widen(' ');
28150b57cec5SDimitry Andric      break;
28160b57cec5SDimitry Andric    case money_base::sign:
28170b57cec5SDimitry Andric      if (!__sn.empty())
28180b57cec5SDimitry Andric        *__me++ = __sn[0];
28190b57cec5SDimitry Andric      break;
28200b57cec5SDimitry Andric    case money_base::symbol:
28210b57cec5SDimitry Andric      if (!__sym.empty() && (__flags & ios_base::showbase))
28225f757f3fSDimitry Andric        __me = std::copy(__sym.begin(), __sym.end(), __me);
28230b57cec5SDimitry Andric      break;
2824cb14a3feSDimitry Andric    case money_base::value: {
28250b57cec5SDimitry Andric      // remember start of value so we can reverse it
28260b57cec5SDimitry Andric      char_type* __t = __me;
28270b57cec5SDimitry Andric      // find beginning of digits
28280b57cec5SDimitry Andric      if (__neg)
28290b57cec5SDimitry Andric        ++__db;
28300b57cec5SDimitry Andric      // find end of digits
28310b57cec5SDimitry Andric      const char_type* __d;
28320b57cec5SDimitry Andric      for (__d = __db; __d < __de; ++__d)
28330b57cec5SDimitry Andric        if (!__ct.is(ctype_base::digit, *__d))
28340b57cec5SDimitry Andric          break;
28350b57cec5SDimitry Andric      // print fractional part
2836cb14a3feSDimitry Andric      if (__fd > 0) {
28370b57cec5SDimitry Andric        int __f;
28380b57cec5SDimitry Andric        for (__f = __fd; __d > __db && __f > 0; --__f)
28390b57cec5SDimitry Andric          *__me++ = *--__d;
28400b57cec5SDimitry Andric        char_type __z = __f > 0 ? __ct.widen('0') : char_type();
28410b57cec5SDimitry Andric        for (; __f > 0; --__f)
28420b57cec5SDimitry Andric          *__me++ = __z;
28430b57cec5SDimitry Andric        *__me++ = __dp;
28440b57cec5SDimitry Andric      }
28450b57cec5SDimitry Andric      // print units part
2846cb14a3feSDimitry Andric      if (__d == __db) {
28470b57cec5SDimitry Andric        *__me++ = __ct.widen('0');
2848cb14a3feSDimitry Andric      } else {
28490b57cec5SDimitry Andric        unsigned __ng = 0;
28500b57cec5SDimitry Andric        unsigned __ig = 0;
2851cb14a3feSDimitry Andric        unsigned __gl = __grp.empty() ? numeric_limits<unsigned>::max() : static_cast<unsigned>(__grp[__ig]);
2852cb14a3feSDimitry Andric        while (__d != __db) {
2853cb14a3feSDimitry Andric          if (__ng == __gl) {
28540b57cec5SDimitry Andric            *__me++ = __ts;
28550b57cec5SDimitry Andric            __ng    = 0;
28560b57cec5SDimitry Andric            if (++__ig < __grp.size())
2857cb14a3feSDimitry Andric              __gl = __grp[__ig] == numeric_limits<char>::max()
2858cb14a3feSDimitry Andric                       ? numeric_limits<unsigned>::max()
2859cb14a3feSDimitry Andric                       : static_cast<unsigned>(__grp[__ig]);
28600b57cec5SDimitry Andric          }
28610b57cec5SDimitry Andric          *__me++ = *--__d;
28620b57cec5SDimitry Andric          ++__ng;
28630b57cec5SDimitry Andric        }
28640b57cec5SDimitry Andric      }
28650b57cec5SDimitry Andric      // reverse it
2866bdd1243dSDimitry Andric      std::reverse(__t, __me);
2867cb14a3feSDimitry Andric    } break;
28680b57cec5SDimitry Andric    }
28690b57cec5SDimitry Andric  }
28700b57cec5SDimitry Andric  // print rest of sign, if any
28710b57cec5SDimitry Andric  if (__sn.size() > 1)
28725f757f3fSDimitry Andric    __me = std::copy(__sn.begin() + 1, __sn.end(), __me);
28730b57cec5SDimitry Andric  // set alignment
28740b57cec5SDimitry Andric  if ((__flags & ios_base::adjustfield) == ios_base::left)
28750b57cec5SDimitry Andric    __mi = __me;
28760b57cec5SDimitry Andric  else if ((__flags & ios_base::adjustfield) != ios_base::internal)
28770b57cec5SDimitry Andric    __mi = __mb;
28780b57cec5SDimitry Andric}
28790b57cec5SDimitry Andric
288081ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __money_put<char>;
2881349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
288281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __money_put<wchar_t>;
2883349cc55cSDimitry Andric#  endif
28840b57cec5SDimitry Andric
28850b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator = ostreambuf_iterator<_CharT> >
2886cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS money_put : public locale::facet, private __money_put<_CharT> {
28870b57cec5SDimitry Andricpublic:
28880b57cec5SDimitry Andric  typedef _CharT char_type;
28890b57cec5SDimitry Andric  typedef _OutputIterator iter_type;
28900b57cec5SDimitry Andric  typedef basic_string<char_type> string_type;
28910b57cec5SDimitry Andric
2892cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit money_put(size_t __refs = 0) : locale::facet(__refs) {}
28930b57cec5SDimitry Andric
2894cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
2895cb14a3feSDimitry Andric  put(iter_type __s, bool __intl, ios_base& __iob, char_type __fl, long double __units) const {
28960b57cec5SDimitry Andric    return do_put(__s, __intl, __iob, __fl, __units);
28970b57cec5SDimitry Andric  }
28980b57cec5SDimitry Andric
2899cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI iter_type
2900cb14a3feSDimitry Andric  put(iter_type __s, bool __intl, ios_base& __iob, char_type __fl, const string_type& __digits) const {
29010b57cec5SDimitry Andric    return do_put(__s, __intl, __iob, __fl, __digits);
29020b57cec5SDimitry Andric  }
29030b57cec5SDimitry Andric
29040b57cec5SDimitry Andric  static locale::id id;
29050b57cec5SDimitry Andric
29060b57cec5SDimitry Andricprotected:
2907bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~money_put() override {}
29080b57cec5SDimitry Andric
2909cb14a3feSDimitry Andric  virtual iter_type do_put(iter_type __s, bool __intl, ios_base& __iob, char_type __fl, long double __units) const;
2910cb14a3feSDimitry Andric  virtual iter_type
2911cb14a3feSDimitry Andric  do_put(iter_type __s, bool __intl, ios_base& __iob, char_type __fl, const string_type& __digits) const;
29120b57cec5SDimitry Andric};
29130b57cec5SDimitry Andric
29140b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
2915cb14a3feSDimitry Andriclocale::id money_put<_CharT, _OutputIterator>::id;
29160b57cec5SDimitry Andric
29170b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
2918cb14a3feSDimitry Andric_OutputIterator money_put<_CharT, _OutputIterator>::do_put(
2919cb14a3feSDimitry Andric    iter_type __s, bool __intl, ios_base& __iob, char_type __fl, long double __units) const {
29200b57cec5SDimitry Andric  // convert to char
29210b57cec5SDimitry Andric  const size_t __bs = 100;
29220b57cec5SDimitry Andric  char __buf[__bs];
29230b57cec5SDimitry Andric  char* __bb = __buf;
29240b57cec5SDimitry Andric  char_type __digits[__bs];
29250b57cec5SDimitry Andric  char_type* __db = __digits;
2926e8d8bef9SDimitry Andric  int __n         = snprintf(__bb, __bs, "%.0Lf", __units);
2927e8d8bef9SDimitry Andric  unique_ptr<char, void (*)(void*)> __hn(nullptr, free);
29280b57cec5SDimitry Andric  unique_ptr<char_type, void (*)(void*)> __hd(0, free);
29290b57cec5SDimitry Andric  // secure memory for digit storage
2930cb14a3feSDimitry Andric  if (static_cast<size_t>(__n) > __bs - 1) {
2931e8d8bef9SDimitry Andric    __n = __libcpp_asprintf_l(&__bb, _LIBCPP_GET_C_LOCALE, "%.0Lf", __units);
2932e8d8bef9SDimitry Andric    if (__n == -1)
29330b57cec5SDimitry Andric      __throw_bad_alloc();
29340b57cec5SDimitry Andric    __hn.reset(__bb);
2935e8d8bef9SDimitry Andric    __hd.reset((char_type*)malloc(static_cast<size_t>(__n) * sizeof(char_type)));
29360b57cec5SDimitry Andric    if (__hd == nullptr)
29370b57cec5SDimitry Andric      __throw_bad_alloc();
29380b57cec5SDimitry Andric    __db = __hd.get();
29390b57cec5SDimitry Andric  }
29400b57cec5SDimitry Andric  // gather info
29410b57cec5SDimitry Andric  locale __loc                 = __iob.getloc();
2942bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__loc);
29430b57cec5SDimitry Andric  __ct.widen(__bb, __bb + __n, __db);
29440b57cec5SDimitry Andric  bool __neg = __n > 0 && __bb[0] == '-';
29450b57cec5SDimitry Andric  money_base::pattern __pat;
29460b57cec5SDimitry Andric  char_type __dp;
29470b57cec5SDimitry Andric  char_type __ts;
29480b57cec5SDimitry Andric  string __grp;
29490b57cec5SDimitry Andric  string_type __sym;
29500b57cec5SDimitry Andric  string_type __sn;
29510b57cec5SDimitry Andric  int __fd;
29520b57cec5SDimitry Andric  this->__gather_info(__intl, __neg, __loc, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
29530b57cec5SDimitry Andric  // secure memory for formatting
29540b57cec5SDimitry Andric  char_type __mbuf[__bs];
29550b57cec5SDimitry Andric  char_type* __mb = __mbuf;
29560b57cec5SDimitry Andric  unique_ptr<char_type, void (*)(void*)> __hw(0, free);
2957cb14a3feSDimitry Andric  size_t __exn = __n > __fd ? (static_cast<size_t>(__n) - static_cast<size_t>(__fd)) * 2 + __sn.size() + __sym.size() +
2958cb14a3feSDimitry Andric                                  static_cast<size_t>(__fd) + 1
29590b57cec5SDimitry Andric                            : __sn.size() + __sym.size() + static_cast<size_t>(__fd) + 2;
2960cb14a3feSDimitry Andric  if (__exn > __bs) {
29610b57cec5SDimitry Andric    __hw.reset((char_type*)malloc(__exn * sizeof(char_type)));
29620b57cec5SDimitry Andric    __mb = __hw.get();
29630b57cec5SDimitry Andric    if (__mb == 0)
29640b57cec5SDimitry Andric      __throw_bad_alloc();
29650b57cec5SDimitry Andric  }
29660b57cec5SDimitry Andric  // format
29670b57cec5SDimitry Andric  char_type* __mi;
29680b57cec5SDimitry Andric  char_type* __me;
2969cb14a3feSDimitry Andric  this->__format(
2970cb14a3feSDimitry Andric      __mb, __mi, __me, __iob.flags(), __db, __db + __n, __ct, __neg, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
2971bdd1243dSDimitry Andric  return std::__pad_and_output(__s, __mb, __mi, __me, __iob, __fl);
29720b57cec5SDimitry Andric}
29730b57cec5SDimitry Andric
29740b57cec5SDimitry Andrictemplate <class _CharT, class _OutputIterator>
2975cb14a3feSDimitry Andric_OutputIterator money_put<_CharT, _OutputIterator>::do_put(
2976cb14a3feSDimitry Andric    iter_type __s, bool __intl, ios_base& __iob, char_type __fl, const string_type& __digits) const {
29770b57cec5SDimitry Andric  // gather info
29780b57cec5SDimitry Andric  locale __loc                 = __iob.getloc();
2979bdd1243dSDimitry Andric  const ctype<char_type>& __ct = std::use_facet<ctype<char_type> >(__loc);
29800b57cec5SDimitry Andric  bool __neg                   = __digits.size() > 0 && __digits[0] == __ct.widen('-');
29810b57cec5SDimitry Andric  money_base::pattern __pat;
29820b57cec5SDimitry Andric  char_type __dp;
29830b57cec5SDimitry Andric  char_type __ts;
29840b57cec5SDimitry Andric  string __grp;
29850b57cec5SDimitry Andric  string_type __sym;
29860b57cec5SDimitry Andric  string_type __sn;
29870b57cec5SDimitry Andric  int __fd;
29880b57cec5SDimitry Andric  this->__gather_info(__intl, __neg, __loc, __pat, __dp, __ts, __grp, __sym, __sn, __fd);
29890b57cec5SDimitry Andric  // secure memory for formatting
29900b57cec5SDimitry Andric  char_type __mbuf[100];
29910b57cec5SDimitry Andric  char_type* __mb = __mbuf;
29920b57cec5SDimitry Andric  unique_ptr<char_type, void (*)(void*)> __h(0, free);
2993cb14a3feSDimitry Andric  size_t __exn =
2994cb14a3feSDimitry Andric      static_cast<int>(__digits.size()) > __fd
2995cb14a3feSDimitry Andric          ? (__digits.size() - static_cast<size_t>(__fd)) * 2 + __sn.size() + __sym.size() + static_cast<size_t>(__fd) +
2996cb14a3feSDimitry Andric                1
29970b57cec5SDimitry Andric          : __sn.size() + __sym.size() + static_cast<size_t>(__fd) + 2;
2998cb14a3feSDimitry Andric  if (__exn > 100) {
29990b57cec5SDimitry Andric    __h.reset((char_type*)malloc(__exn * sizeof(char_type)));
30000b57cec5SDimitry Andric    __mb = __h.get();
30010b57cec5SDimitry Andric    if (__mb == 0)
30020b57cec5SDimitry Andric      __throw_bad_alloc();
30030b57cec5SDimitry Andric  }
30040b57cec5SDimitry Andric  // format
30050b57cec5SDimitry Andric  char_type* __mi;
30060b57cec5SDimitry Andric  char_type* __me;
3007cb14a3feSDimitry Andric  this->__format(
3008cb14a3feSDimitry Andric      __mb,
3009cb14a3feSDimitry Andric      __mi,
3010cb14a3feSDimitry Andric      __me,
3011cb14a3feSDimitry Andric      __iob.flags(),
3012cb14a3feSDimitry Andric      __digits.data(),
3013cb14a3feSDimitry Andric      __digits.data() + __digits.size(),
3014cb14a3feSDimitry Andric      __ct,
3015cb14a3feSDimitry Andric      __neg,
3016cb14a3feSDimitry Andric      __pat,
3017cb14a3feSDimitry Andric      __dp,
3018cb14a3feSDimitry Andric      __ts,
3019cb14a3feSDimitry Andric      __grp,
3020cb14a3feSDimitry Andric      __sym,
3021cb14a3feSDimitry Andric      __sn,
3022cb14a3feSDimitry Andric      __fd);
3023bdd1243dSDimitry Andric  return std::__pad_and_output(__s, __mb, __mi, __me, __iob, __fl);
30240b57cec5SDimitry Andric}
30250b57cec5SDimitry Andric
302681ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS money_put<char>;
3027349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
302881ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS money_put<wchar_t>;
3029349cc55cSDimitry Andric#  endif
30300b57cec5SDimitry Andric
30310b57cec5SDimitry Andric// messages
30320b57cec5SDimitry Andric
3033cb14a3feSDimitry Andricclass _LIBCPP_EXPORTED_FROM_ABI messages_base {
30340b57cec5SDimitry Andricpublic:
30355f757f3fSDimitry Andric  typedef intptr_t catalog;
30360b57cec5SDimitry Andric
30375f757f3fSDimitry Andric  _LIBCPP_HIDE_FROM_ABI messages_base() {}
30380b57cec5SDimitry Andric};
30390b57cec5SDimitry Andric
30400b57cec5SDimitry Andrictemplate <class _CharT>
3041cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS messages : public locale::facet, public messages_base {
30420b57cec5SDimitry Andricpublic:
30430b57cec5SDimitry Andric  typedef _CharT char_type;
30440b57cec5SDimitry Andric  typedef basic_string<_CharT> string_type;
30450b57cec5SDimitry Andric
3046cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit messages(size_t __refs = 0) : locale::facet(__refs) {}
30470b57cec5SDimitry Andric
3048cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI catalog open(const basic_string<char>& __nm, const locale& __loc) const {
30490b57cec5SDimitry Andric    return do_open(__nm, __loc);
30500b57cec5SDimitry Andric  }
30510b57cec5SDimitry Andric
3052cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI string_type get(catalog __c, int __set, int __msgid, const string_type& __dflt) const {
30530b57cec5SDimitry Andric    return do_get(__c, __set, __msgid, __dflt);
30540b57cec5SDimitry Andric  }
30550b57cec5SDimitry Andric
3056cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI void close(catalog __c) const { do_close(__c); }
30570b57cec5SDimitry Andric
30580b57cec5SDimitry Andric  static locale::id id;
30590b57cec5SDimitry Andric
30600b57cec5SDimitry Andricprotected:
3061bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~messages() override {}
30620b57cec5SDimitry Andric
30630b57cec5SDimitry Andric  virtual catalog do_open(const basic_string<char>&, const locale&) const;
3064cb14a3feSDimitry Andric  virtual string_type do_get(catalog, int __set, int __msgid, const string_type& __dflt) const;
30650b57cec5SDimitry Andric  virtual void do_close(catalog) const;
30660b57cec5SDimitry Andric};
30670b57cec5SDimitry Andric
30680b57cec5SDimitry Andrictemplate <class _CharT>
3069cb14a3feSDimitry Andriclocale::id messages<_CharT>::id;
30700b57cec5SDimitry Andric
30710b57cec5SDimitry Andrictemplate <class _CharT>
3072cb14a3feSDimitry Andrictypename messages<_CharT>::catalog messages<_CharT>::do_open(const basic_string<char>& __nm, const locale&) const {
30730b57cec5SDimitry Andric#  ifdef _LIBCPP_HAS_CATOPEN
30745f757f3fSDimitry Andric  return (catalog)catopen(__nm.c_str(), NL_CAT_LOCALE);
30750b57cec5SDimitry Andric#  else  // !_LIBCPP_HAS_CATOPEN
3076fe6060f1SDimitry Andric  (void)__nm;
30770b57cec5SDimitry Andric  return -1;
30780b57cec5SDimitry Andric#  endif // _LIBCPP_HAS_CATOPEN
30790b57cec5SDimitry Andric}
30800b57cec5SDimitry Andric
30810b57cec5SDimitry Andrictemplate <class _CharT>
30820b57cec5SDimitry Andrictypename messages<_CharT>::string_type
3083cb14a3feSDimitry Andricmessages<_CharT>::do_get(catalog __c, int __set, int __msgid, const string_type& __dflt) const {
30840b57cec5SDimitry Andric#  ifdef _LIBCPP_HAS_CATOPEN
30850b57cec5SDimitry Andric  string __ndflt;
3086cb14a3feSDimitry Andric  __narrow_to_utf8<sizeof(char_type) * __CHAR_BIT__>()(
3087cb14a3feSDimitry Andric      std::back_inserter(__ndflt), __dflt.c_str(), __dflt.c_str() + __dflt.size());
30880b57cec5SDimitry Andric  nl_catd __cat = (nl_catd)__c;
30895f757f3fSDimitry Andric  static_assert(sizeof(catalog) >= sizeof(nl_catd), "Unexpected nl_catd type");
30900b57cec5SDimitry Andric  char* __n = catgets(__cat, __set, __msgid, __ndflt.c_str());
30910b57cec5SDimitry Andric  string_type __w;
3092cb14a3feSDimitry Andric  __widen_from_utf8<sizeof(char_type) * __CHAR_BIT__>()(std::back_inserter(__w), __n, __n + std::strlen(__n));
30930b57cec5SDimitry Andric  return __w;
30940b57cec5SDimitry Andric#  else  // !_LIBCPP_HAS_CATOPEN
3095fe6060f1SDimitry Andric  (void)__c;
3096fe6060f1SDimitry Andric  (void)__set;
3097fe6060f1SDimitry Andric  (void)__msgid;
30980b57cec5SDimitry Andric  return __dflt;
30990b57cec5SDimitry Andric#  endif // _LIBCPP_HAS_CATOPEN
31000b57cec5SDimitry Andric}
31010b57cec5SDimitry Andric
31020b57cec5SDimitry Andrictemplate <class _CharT>
3103cb14a3feSDimitry Andricvoid messages<_CharT>::do_close(catalog __c) const {
31040b57cec5SDimitry Andric#  ifdef _LIBCPP_HAS_CATOPEN
31055f757f3fSDimitry Andric  catclose((nl_catd)__c);
31060b57cec5SDimitry Andric#  else  // !_LIBCPP_HAS_CATOPEN
3107fe6060f1SDimitry Andric  (void)__c;
31080b57cec5SDimitry Andric#  endif // _LIBCPP_HAS_CATOPEN
31090b57cec5SDimitry Andric}
31100b57cec5SDimitry Andric
311181ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS messages<char>;
3112349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
311381ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS messages<wchar_t>;
3114349cc55cSDimitry Andric#  endif
31150b57cec5SDimitry Andric
31160b57cec5SDimitry Andrictemplate <class _CharT>
3117cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS messages_byname : public messages<_CharT> {
31180b57cec5SDimitry Andricpublic:
31190b57cec5SDimitry Andric  typedef messages_base::catalog catalog;
31200b57cec5SDimitry Andric  typedef basic_string<_CharT> string_type;
31210b57cec5SDimitry Andric
3122cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit messages_byname(const char*, size_t __refs = 0) : messages<_CharT>(__refs) {}
31230b57cec5SDimitry Andric
3124cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit messages_byname(const string&, size_t __refs = 0) : messages<_CharT>(__refs) {}
31250b57cec5SDimitry Andric
31260b57cec5SDimitry Andricprotected:
3127bdd1243dSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL ~messages_byname() override {}
31280b57cec5SDimitry Andric};
31290b57cec5SDimitry Andric
313081ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS messages_byname<char>;
3131349cc55cSDimitry Andric#  ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
313281ad6265SDimitry Andricextern template class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS messages_byname<wchar_t>;
3133349cc55cSDimitry Andric#  endif
31340b57cec5SDimitry Andric
3135*0fca6ea1SDimitry Andric#  if _LIBCPP_STD_VER < 26 || defined(_LIBCPP_ENABLE_CXX26_REMOVED_WSTRING_CONVERT)
3136*0fca6ea1SDimitry Andric
3137cb14a3feSDimitry Andrictemplate <class _Codecvt,
3138cb14a3feSDimitry Andric          class _Elem      = wchar_t,
313906c3fb27SDimitry Andric          class _WideAlloc = allocator<_Elem>,
314006c3fb27SDimitry Andric          class _ByteAlloc = allocator<char> >
3141cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS _LIBCPP_DEPRECATED_IN_CXX17 wstring_convert {
31420b57cec5SDimitry Andricpublic:
314306c3fb27SDimitry Andric  typedef basic_string<char, char_traits<char>, _ByteAlloc> byte_string;
314406c3fb27SDimitry Andric  typedef basic_string<_Elem, char_traits<_Elem>, _WideAlloc> wide_string;
31450b57cec5SDimitry Andric  typedef typename _Codecvt::state_type state_type;
31460b57cec5SDimitry Andric  typedef typename wide_string::traits_type::int_type int_type;
31470b57cec5SDimitry Andric
31480b57cec5SDimitry Andricprivate:
31490b57cec5SDimitry Andric  byte_string __byte_err_string_;
31500b57cec5SDimitry Andric  wide_string __wide_err_string_;
31510b57cec5SDimitry Andric  _Codecvt* __cvtptr_;
31520b57cec5SDimitry Andric  state_type __cvtstate_;
31530b57cec5SDimitry Andric  size_t __cvtcount_;
31540b57cec5SDimitry Andric
31550b57cec5SDimitry Andricpublic:
3156e8d8bef9SDimitry Andric#    ifndef _LIBCPP_CXX03_LANG
3157cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI wstring_convert() : wstring_convert(new _Codecvt) {}
3158cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI explicit wstring_convert(_Codecvt* __pcvt);
3159e8d8bef9SDimitry Andric#    else
3160cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI _LIBCPP_EXPLICIT_SINCE_CXX14 wstring_convert(_Codecvt* __pcvt = new _Codecvt);
3161e8d8bef9SDimitry Andric#    endif
3162e8d8bef9SDimitry Andric
3163cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI wstring_convert(_Codecvt* __pcvt, state_type __state);
3164cb14a3feSDimitry Andric  _LIBCPP_EXPLICIT_SINCE_CXX14 _LIBCPP_HIDE_FROM_ABI
3165cb14a3feSDimitry Andric  wstring_convert(const byte_string& __byte_err, const wide_string& __wide_err = wide_string());
31660b57cec5SDimitry Andric#    ifndef _LIBCPP_CXX03_LANG
3167cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI wstring_convert(wstring_convert&& __wc);
31680b57cec5SDimitry Andric#    endif
316906c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI ~wstring_convert();
31700b57cec5SDimitry Andric
3171*0fca6ea1SDimitry Andric  wstring_convert(const wstring_convert& __wc)            = delete;
3172*0fca6ea1SDimitry Andric  wstring_convert& operator=(const wstring_convert& __wc) = delete;
3173*0fca6ea1SDimitry Andric
3174cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI wide_string from_bytes(char __byte) { return from_bytes(&__byte, &__byte + 1); }
3175cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI wide_string from_bytes(const char* __ptr) {
3176cb14a3feSDimitry Andric    return from_bytes(__ptr, __ptr + char_traits<char>::length(__ptr));
3177cb14a3feSDimitry Andric  }
3178cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI wide_string from_bytes(const byte_string& __str) {
3179cb14a3feSDimitry Andric    return from_bytes(__str.data(), __str.data() + __str.size());
3180cb14a3feSDimitry Andric  }
318106c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI wide_string from_bytes(const char* __first, const char* __last);
31820b57cec5SDimitry Andric
3183cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI byte_string to_bytes(_Elem __wchar) { return to_bytes(&__wchar, &__wchar + 1); }
3184cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI byte_string to_bytes(const _Elem* __wptr) {
3185cb14a3feSDimitry Andric    return to_bytes(__wptr, __wptr + char_traits<_Elem>::length(__wptr));
3186cb14a3feSDimitry Andric  }
3187cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI byte_string to_bytes(const wide_string& __wstr) {
3188cb14a3feSDimitry Andric    return to_bytes(__wstr.data(), __wstr.data() + __wstr.size());
3189cb14a3feSDimitry Andric  }
319006c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI byte_string to_bytes(const _Elem* __first, const _Elem* __last);
31910b57cec5SDimitry Andric
3192cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI size_t converted() const _NOEXCEPT { return __cvtcount_; }
3193cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI state_type state() const { return __cvtstate_; }
31940b57cec5SDimitry Andric};
31950b57cec5SDimitry Andric
319681ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
319706c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
3198cb14a3feSDimitry Andricinline wstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::wstring_convert(_Codecvt* __pcvt)
3199cb14a3feSDimitry Andric    : __cvtptr_(__pcvt), __cvtstate_(), __cvtcount_(0) {}
320081ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_POP
32010b57cec5SDimitry Andric
320206c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
3203cb14a3feSDimitry Andricinline wstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::wstring_convert(_Codecvt* __pcvt, state_type __state)
3204cb14a3feSDimitry Andric    : __cvtptr_(__pcvt), __cvtstate_(__state), __cvtcount_(0) {}
32050b57cec5SDimitry Andric
320606c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
3207cb14a3feSDimitry Andricwstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::wstring_convert(
3208cb14a3feSDimitry Andric    const byte_string& __byte_err, const wide_string& __wide_err)
3209cb14a3feSDimitry Andric    : __byte_err_string_(__byte_err), __wide_err_string_(__wide_err), __cvtstate_(), __cvtcount_(0) {
32100b57cec5SDimitry Andric  __cvtptr_ = new _Codecvt;
32110b57cec5SDimitry Andric}
32120b57cec5SDimitry Andric
32130b57cec5SDimitry Andric#    ifndef _LIBCPP_CXX03_LANG
32140b57cec5SDimitry Andric
321506c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
3216cb14a3feSDimitry Andricinline wstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::wstring_convert(wstring_convert&& __wc)
32175f757f3fSDimitry Andric    : __byte_err_string_(std::move(__wc.__byte_err_string_)),
32185f757f3fSDimitry Andric      __wide_err_string_(std::move(__wc.__wide_err_string_)),
32190b57cec5SDimitry Andric      __cvtptr_(__wc.__cvtptr_),
3220cb14a3feSDimitry Andric      __cvtstate_(__wc.__cvtstate_),
3221cb14a3feSDimitry Andric      __cvtcount_(__wc.__cvtcount_) {
32220b57cec5SDimitry Andric  __wc.__cvtptr_ = nullptr;
32230b57cec5SDimitry Andric}
32240b57cec5SDimitry Andric
32250b57cec5SDimitry Andric#    endif // _LIBCPP_CXX03_LANG
32260b57cec5SDimitry Andric
322781ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
322806c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
3229cb14a3feSDimitry Andricwstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::~wstring_convert() {
32300b57cec5SDimitry Andric  delete __cvtptr_;
32310b57cec5SDimitry Andric}
32320b57cec5SDimitry Andric
323306c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
323406c3fb27SDimitry Andrictypename wstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::wide_string
3235cb14a3feSDimitry Andricwstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::from_bytes(const char* __frm, const char* __frm_end) {
323681ad6265SDimitry Andric  _LIBCPP_SUPPRESS_DEPRECATED_POP
32370b57cec5SDimitry Andric  __cvtcount_ = 0;
3238cb14a3feSDimitry Andric  if (__cvtptr_ != nullptr) {
32390b57cec5SDimitry Andric    wide_string __ws(2 * (__frm_end - __frm), _Elem());
32400b57cec5SDimitry Andric    if (__frm != __frm_end)
32410b57cec5SDimitry Andric      __ws.resize(__ws.capacity());
32420b57cec5SDimitry Andric    codecvt_base::result __r = codecvt_base::ok;
32430b57cec5SDimitry Andric    state_type __st          = __cvtstate_;
3244cb14a3feSDimitry Andric    if (__frm != __frm_end) {
32450b57cec5SDimitry Andric      _Elem* __to     = &__ws[0];
32460b57cec5SDimitry Andric      _Elem* __to_end = __to + __ws.size();
32470b57cec5SDimitry Andric      const char* __frm_nxt;
3248cb14a3feSDimitry Andric      do {
32490b57cec5SDimitry Andric        _Elem* __to_nxt;
3250cb14a3feSDimitry Andric        __r = __cvtptr_->in(__st, __frm, __frm_end, __frm_nxt, __to, __to_end, __to_nxt);
32510b57cec5SDimitry Andric        __cvtcount_ += __frm_nxt - __frm;
3252cb14a3feSDimitry Andric        if (__frm_nxt == __frm) {
32530b57cec5SDimitry Andric          __r = codecvt_base::error;
3254cb14a3feSDimitry Andric        } else if (__r == codecvt_base::noconv) {
32550b57cec5SDimitry Andric          __ws.resize(__to - &__ws[0]);
32560b57cec5SDimitry Andric          // This only gets executed if _Elem is char
32570b57cec5SDimitry Andric          __ws.append((const _Elem*)__frm, (const _Elem*)__frm_end);
32580b57cec5SDimitry Andric          __frm = __frm_nxt;
32590b57cec5SDimitry Andric          __r   = codecvt_base::ok;
3260cb14a3feSDimitry Andric        } else if (__r == codecvt_base::ok) {
32610b57cec5SDimitry Andric          __ws.resize(__to_nxt - &__ws[0]);
32620b57cec5SDimitry Andric          __frm = __frm_nxt;
3263cb14a3feSDimitry Andric        } else if (__r == codecvt_base::partial) {
32640b57cec5SDimitry Andric          ptrdiff_t __s = __to_nxt - &__ws[0];
32650b57cec5SDimitry Andric          __ws.resize(2 * __s);
32660b57cec5SDimitry Andric          __to     = &__ws[0] + __s;
32670b57cec5SDimitry Andric          __to_end = &__ws[0] + __ws.size();
32680b57cec5SDimitry Andric          __frm    = __frm_nxt;
32690b57cec5SDimitry Andric        }
32700b57cec5SDimitry Andric      } while (__r == codecvt_base::partial && __frm_nxt < __frm_end);
32710b57cec5SDimitry Andric    }
32720b57cec5SDimitry Andric    if (__r == codecvt_base::ok)
32730b57cec5SDimitry Andric      return __ws;
32740b57cec5SDimitry Andric  }
32750b57cec5SDimitry Andric
32760b57cec5SDimitry Andric  if (__wide_err_string_.empty())
32770b57cec5SDimitry Andric    __throw_range_error("wstring_convert: from_bytes error");
32780b57cec5SDimitry Andric
32790b57cec5SDimitry Andric  return __wide_err_string_;
32800b57cec5SDimitry Andric}
32810b57cec5SDimitry Andric
328206c3fb27SDimitry Andrictemplate <class _Codecvt, class _Elem, class _WideAlloc, class _ByteAlloc>
328306c3fb27SDimitry Andrictypename wstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::byte_string
3284cb14a3feSDimitry Andricwstring_convert<_Codecvt, _Elem, _WideAlloc, _ByteAlloc>::to_bytes(const _Elem* __frm, const _Elem* __frm_end) {
32850b57cec5SDimitry Andric  __cvtcount_ = 0;
3286cb14a3feSDimitry Andric  if (__cvtptr_ != nullptr) {
32870b57cec5SDimitry Andric    byte_string __bs(2 * (__frm_end - __frm), char());
32880b57cec5SDimitry Andric    if (__frm != __frm_end)
32890b57cec5SDimitry Andric      __bs.resize(__bs.capacity());
32900b57cec5SDimitry Andric    codecvt_base::result __r = codecvt_base::ok;
32910b57cec5SDimitry Andric    state_type __st          = __cvtstate_;
3292cb14a3feSDimitry Andric    if (__frm != __frm_end) {
32930b57cec5SDimitry Andric      char* __to     = &__bs[0];
32940b57cec5SDimitry Andric      char* __to_end = __to + __bs.size();
32950b57cec5SDimitry Andric      const _Elem* __frm_nxt;
3296cb14a3feSDimitry Andric      do {
32970b57cec5SDimitry Andric        char* __to_nxt;
3298cb14a3feSDimitry Andric        __r = __cvtptr_->out(__st, __frm, __frm_end, __frm_nxt, __to, __to_end, __to_nxt);
32990b57cec5SDimitry Andric        __cvtcount_ += __frm_nxt - __frm;
3300cb14a3feSDimitry Andric        if (__frm_nxt == __frm) {
33010b57cec5SDimitry Andric          __r = codecvt_base::error;
3302cb14a3feSDimitry Andric        } else if (__r == codecvt_base::noconv) {
33030b57cec5SDimitry Andric          __bs.resize(__to - &__bs[0]);
33040b57cec5SDimitry Andric          // This only gets executed if _Elem is char
33050b57cec5SDimitry Andric          __bs.append((const char*)__frm, (const char*)__frm_end);
33060b57cec5SDimitry Andric          __frm = __frm_nxt;
33070b57cec5SDimitry Andric          __r   = codecvt_base::ok;
3308cb14a3feSDimitry Andric        } else if (__r == codecvt_base::ok) {
33090b57cec5SDimitry Andric          __bs.resize(__to_nxt - &__bs[0]);
33100b57cec5SDimitry Andric          __frm = __frm_nxt;
3311cb14a3feSDimitry Andric        } else if (__r == codecvt_base::partial) {
33120b57cec5SDimitry Andric          ptrdiff_t __s = __to_nxt - &__bs[0];
33130b57cec5SDimitry Andric          __bs.resize(2 * __s);
33140b57cec5SDimitry Andric          __to     = &__bs[0] + __s;
33150b57cec5SDimitry Andric          __to_end = &__bs[0] + __bs.size();
33160b57cec5SDimitry Andric          __frm    = __frm_nxt;
33170b57cec5SDimitry Andric        }
33180b57cec5SDimitry Andric      } while (__r == codecvt_base::partial && __frm_nxt < __frm_end);
33190b57cec5SDimitry Andric    }
3320cb14a3feSDimitry Andric    if (__r == codecvt_base::ok) {
33210b57cec5SDimitry Andric      size_t __s = __bs.size();
33220b57cec5SDimitry Andric      __bs.resize(__bs.capacity());
33230b57cec5SDimitry Andric      char* __to     = &__bs[0] + __s;
33240b57cec5SDimitry Andric      char* __to_end = __to + __bs.size();
3325cb14a3feSDimitry Andric      do {
33260b57cec5SDimitry Andric        char* __to_nxt;
33270b57cec5SDimitry Andric        __r = __cvtptr_->unshift(__st, __to, __to_end, __to_nxt);
3328cb14a3feSDimitry Andric        if (__r == codecvt_base::noconv) {
33290b57cec5SDimitry Andric          __bs.resize(__to - &__bs[0]);
33300b57cec5SDimitry Andric          __r = codecvt_base::ok;
3331cb14a3feSDimitry Andric        } else if (__r == codecvt_base::ok) {
33320b57cec5SDimitry Andric          __bs.resize(__to_nxt - &__bs[0]);
3333cb14a3feSDimitry Andric        } else if (__r == codecvt_base::partial) {
33340b57cec5SDimitry Andric          ptrdiff_t __sp = __to_nxt - &__bs[0];
33350b57cec5SDimitry Andric          __bs.resize(2 * __sp);
33360b57cec5SDimitry Andric          __to     = &__bs[0] + __sp;
33370b57cec5SDimitry Andric          __to_end = &__bs[0] + __bs.size();
33380b57cec5SDimitry Andric        }
33390b57cec5SDimitry Andric      } while (__r == codecvt_base::partial);
33400b57cec5SDimitry Andric      if (__r == codecvt_base::ok)
33410b57cec5SDimitry Andric        return __bs;
33420b57cec5SDimitry Andric    }
33430b57cec5SDimitry Andric  }
33440b57cec5SDimitry Andric
33450b57cec5SDimitry Andric  if (__byte_err_string_.empty())
33460b57cec5SDimitry Andric    __throw_range_error("wstring_convert: to_bytes error");
33470b57cec5SDimitry Andric
33480b57cec5SDimitry Andric  return __byte_err_string_;
33490b57cec5SDimitry Andric}
33500b57cec5SDimitry Andric
33510b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem = wchar_t, class _Tr = char_traits<_Elem> >
3352cb14a3feSDimitry Andricclass _LIBCPP_TEMPLATE_VIS _LIBCPP_DEPRECATED_IN_CXX17 wbuffer_convert : public basic_streambuf<_Elem, _Tr> {
33530b57cec5SDimitry Andricpublic:
33540b57cec5SDimitry Andric  // types:
33550b57cec5SDimitry Andric  typedef _Elem char_type;
33560b57cec5SDimitry Andric  typedef _Tr traits_type;
33570b57cec5SDimitry Andric  typedef typename traits_type::int_type int_type;
33580b57cec5SDimitry Andric  typedef typename traits_type::pos_type pos_type;
33590b57cec5SDimitry Andric  typedef typename traits_type::off_type off_type;
33600b57cec5SDimitry Andric  typedef typename _Codecvt::state_type state_type;
33610b57cec5SDimitry Andric
33620b57cec5SDimitry Andricprivate:
33630b57cec5SDimitry Andric  char* __extbuf_;
33640b57cec5SDimitry Andric  const char* __extbufnext_;
33650b57cec5SDimitry Andric  const char* __extbufend_;
33660b57cec5SDimitry Andric  char __extbuf_min_[8];
33670b57cec5SDimitry Andric  size_t __ebs_;
33680b57cec5SDimitry Andric  char_type* __intbuf_;
33690b57cec5SDimitry Andric  size_t __ibs_;
33700b57cec5SDimitry Andric  streambuf* __bufptr_;
33710b57cec5SDimitry Andric  _Codecvt* __cv_;
33720b57cec5SDimitry Andric  state_type __st_;
33730b57cec5SDimitry Andric  ios_base::openmode __cm_;
33740b57cec5SDimitry Andric  bool __owns_eb_;
33750b57cec5SDimitry Andric  bool __owns_ib_;
33760b57cec5SDimitry Andric  bool __always_noconv_;
33770b57cec5SDimitry Andric
33780b57cec5SDimitry Andricpublic:
3379e8d8bef9SDimitry Andric#    ifndef _LIBCPP_CXX03_LANG
338006c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI wbuffer_convert() : wbuffer_convert(nullptr) {}
3381cb14a3feSDimitry Andric  explicit _LIBCPP_HIDE_FROM_ABI
3382cb14a3feSDimitry Andric  wbuffer_convert(streambuf* __bytebuf, _Codecvt* __pcvt = new _Codecvt, state_type __state = state_type());
3383e8d8bef9SDimitry Andric#    else
338406c3fb27SDimitry Andric  _LIBCPP_EXPLICIT_SINCE_CXX14 _LIBCPP_HIDE_FROM_ABI
3385cb14a3feSDimitry Andric  wbuffer_convert(streambuf* __bytebuf = nullptr, _Codecvt* __pcvt = new _Codecvt, state_type __state = state_type());
3386e8d8bef9SDimitry Andric#    endif
3387e8d8bef9SDimitry Andric
338806c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI ~wbuffer_convert();
33890b57cec5SDimitry Andric
3390cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI streambuf* rdbuf() const { return __bufptr_; }
3391cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI streambuf* rdbuf(streambuf* __bytebuf) {
33920b57cec5SDimitry Andric    streambuf* __r = __bufptr_;
33930b57cec5SDimitry Andric    __bufptr_      = __bytebuf;
33940b57cec5SDimitry Andric    return __r;
33950b57cec5SDimitry Andric  }
33960b57cec5SDimitry Andric
3397*0fca6ea1SDimitry Andric  wbuffer_convert(const wbuffer_convert&)            = delete;
3398*0fca6ea1SDimitry Andric  wbuffer_convert& operator=(const wbuffer_convert&) = delete;
3399*0fca6ea1SDimitry Andric
3400cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI state_type state() const { return __st_; }
34010b57cec5SDimitry Andric
34020b57cec5SDimitry Andricprotected:
340306c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual int_type underflow();
340406c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual int_type pbackfail(int_type __c = traits_type::eof());
340506c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual int_type overflow(int_type __c = traits_type::eof());
3406cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual basic_streambuf<char_type, traits_type>* setbuf(char_type* __s, streamsize __n);
3407cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual pos_type
3408cb14a3feSDimitry Andric  seekoff(off_type __off, ios_base::seekdir __way, ios_base::openmode __wch = ios_base::in | ios_base::out);
3409cb14a3feSDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual pos_type
3410cb14a3feSDimitry Andric  seekpos(pos_type __sp, ios_base::openmode __wch = ios_base::in | ios_base::out);
341106c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL virtual int sync();
34120b57cec5SDimitry Andric
34130b57cec5SDimitry Andricprivate:
341406c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL bool __read_mode();
341506c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL void __write_mode();
341606c3fb27SDimitry Andric  _LIBCPP_HIDE_FROM_ABI_VIRTUAL wbuffer_convert* __close();
34170b57cec5SDimitry Andric};
34180b57cec5SDimitry Andric
341981ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
34200b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3421cb14a3feSDimitry Andricwbuffer_convert<_Codecvt, _Elem, _Tr>::wbuffer_convert(streambuf* __bytebuf, _Codecvt* __pcvt, state_type __state)
3422e8d8bef9SDimitry Andric    : __extbuf_(nullptr),
3423e8d8bef9SDimitry Andric      __extbufnext_(nullptr),
3424e8d8bef9SDimitry Andric      __extbufend_(nullptr),
34250b57cec5SDimitry Andric      __ebs_(0),
34260b57cec5SDimitry Andric      __intbuf_(0),
34270b57cec5SDimitry Andric      __ibs_(0),
34280b57cec5SDimitry Andric      __bufptr_(__bytebuf),
34290b57cec5SDimitry Andric      __cv_(__pcvt),
34300b57cec5SDimitry Andric      __st_(__state),
34310b57cec5SDimitry Andric      __cm_(0),
34320b57cec5SDimitry Andric      __owns_eb_(false),
34330b57cec5SDimitry Andric      __owns_ib_(false),
3434cb14a3feSDimitry Andric      __always_noconv_(__cv_ ? __cv_->always_noconv() : false) {
34350b57cec5SDimitry Andric  setbuf(0, 4096);
34360b57cec5SDimitry Andric}
34370b57cec5SDimitry Andric
34380b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3439cb14a3feSDimitry Andricwbuffer_convert<_Codecvt, _Elem, _Tr>::~wbuffer_convert() {
34400b57cec5SDimitry Andric  __close();
34410b57cec5SDimitry Andric  delete __cv_;
34420b57cec5SDimitry Andric  if (__owns_eb_)
34430b57cec5SDimitry Andric    delete[] __extbuf_;
34440b57cec5SDimitry Andric  if (__owns_ib_)
34450b57cec5SDimitry Andric    delete[] __intbuf_;
34460b57cec5SDimitry Andric}
34470b57cec5SDimitry Andric
34480b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3449cb14a3feSDimitry Andrictypename wbuffer_convert<_Codecvt, _Elem, _Tr>::int_type wbuffer_convert<_Codecvt, _Elem, _Tr>::underflow() {
345081ad6265SDimitry Andric  _LIBCPP_SUPPRESS_DEPRECATED_POP
34517a6dacacSDimitry Andric  if (__cv_ == 0 || __bufptr_ == nullptr)
34520b57cec5SDimitry Andric    return traits_type::eof();
34530b57cec5SDimitry Andric  bool __initial = __read_mode();
34540b57cec5SDimitry Andric  char_type __1buf;
34550b57cec5SDimitry Andric  if (this->gptr() == 0)
34560b57cec5SDimitry Andric    this->setg(&__1buf, &__1buf + 1, &__1buf + 1);
3457bdd1243dSDimitry Andric  const size_t __unget_sz = __initial ? 0 : std::min<size_t>((this->egptr() - this->eback()) / 2, 4);
34580b57cec5SDimitry Andric  int_type __c            = traits_type::eof();
3459cb14a3feSDimitry Andric  if (this->gptr() == this->egptr()) {
34605f757f3fSDimitry Andric    std::memmove(this->eback(), this->egptr() - __unget_sz, __unget_sz * sizeof(char_type));
3461cb14a3feSDimitry Andric    if (__always_noconv_) {
34620b57cec5SDimitry Andric      streamsize __nmemb = static_cast<streamsize>(this->egptr() - this->eback() - __unget_sz);
34630b57cec5SDimitry Andric      __nmemb            = __bufptr_->sgetn((char*)this->eback() + __unget_sz, __nmemb);
3464cb14a3feSDimitry Andric      if (__nmemb != 0) {
3465cb14a3feSDimitry Andric        this->setg(this->eback(), this->eback() + __unget_sz, this->eback() + __unget_sz + __nmemb);
34660b57cec5SDimitry Andric        __c = *this->gptr();
34670b57cec5SDimitry Andric      }
3468cb14a3feSDimitry Andric    } else {
3469bdd1243dSDimitry Andric      if (__extbufend_ != __extbufnext_) {
34705f757f3fSDimitry Andric        _LIBCPP_ASSERT_NON_NULL(__extbufnext_ != nullptr, "underflow moving from nullptr");
34715f757f3fSDimitry Andric        _LIBCPP_ASSERT_NON_NULL(__extbuf_ != nullptr, "underflow moving into nullptr");
34725f757f3fSDimitry Andric        std::memmove(__extbuf_, __extbufnext_, __extbufend_ - __extbufnext_);
3473bdd1243dSDimitry Andric      }
34740b57cec5SDimitry Andric      __extbufnext_      = __extbuf_ + (__extbufend_ - __extbufnext_);
34750b57cec5SDimitry Andric      __extbufend_       = __extbuf_ + (__extbuf_ == __extbuf_min_ ? sizeof(__extbuf_min_) : __ebs_);
34765f757f3fSDimitry Andric      streamsize __nmemb = std::min(static_cast<streamsize>(this->egptr() - this->eback() - __unget_sz),
34770b57cec5SDimitry Andric                                    static_cast<streamsize>(__extbufend_ - __extbufnext_));
34780b57cec5SDimitry Andric      codecvt_base::result __r;
34790b57cec5SDimitry Andric      // FIXME: Do we ever need to restore the state here?
34800b57cec5SDimitry Andric      // state_type __svs = __st_;
34810b57cec5SDimitry Andric      streamsize __nr = __bufptr_->sgetn(const_cast<char*>(__extbufnext_), __nmemb);
3482cb14a3feSDimitry Andric      if (__nr != 0) {
34830b57cec5SDimitry Andric        __extbufend_ = __extbufnext_ + __nr;
34840b57cec5SDimitry Andric        char_type* __inext;
3485cb14a3feSDimitry Andric        __r = __cv_->in(
3486cb14a3feSDimitry Andric            __st_, __extbuf_, __extbufend_, __extbufnext_, this->eback() + __unget_sz, this->egptr(), __inext);
3487cb14a3feSDimitry Andric        if (__r == codecvt_base::noconv) {
3488cb14a3feSDimitry Andric          this->setg((char_type*)__extbuf_, (char_type*)__extbuf_, (char_type*)const_cast<char*>(__extbufend_));
34890b57cec5SDimitry Andric          __c = *this->gptr();
3490cb14a3feSDimitry Andric        } else if (__inext != this->eback() + __unget_sz) {
34910b57cec5SDimitry Andric          this->setg(this->eback(), this->eback() + __unget_sz, __inext);
34920b57cec5SDimitry Andric          __c = *this->gptr();
34930b57cec5SDimitry Andric        }
34940b57cec5SDimitry Andric      }
34950b57cec5SDimitry Andric    }
3496cb14a3feSDimitry Andric  } else
34970b57cec5SDimitry Andric    __c = *this->gptr();
34980b57cec5SDimitry Andric  if (this->eback() == &__1buf)
34990b57cec5SDimitry Andric    this->setg(0, 0, 0);
35000b57cec5SDimitry Andric  return __c;
35010b57cec5SDimitry Andric}
35020b57cec5SDimitry Andric
350381ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
35040b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
35050b57cec5SDimitry Andrictypename wbuffer_convert<_Codecvt, _Elem, _Tr>::int_type
3506cb14a3feSDimitry Andricwbuffer_convert<_Codecvt, _Elem, _Tr>::pbackfail(int_type __c) {
350781ad6265SDimitry Andric  _LIBCPP_SUPPRESS_DEPRECATED_POP
35087a6dacacSDimitry Andric  if (__cv_ != 0 && __bufptr_ && this->eback() < this->gptr()) {
3509cb14a3feSDimitry Andric    if (traits_type::eq_int_type(__c, traits_type::eof())) {
35100b57cec5SDimitry Andric      this->gbump(-1);
35110b57cec5SDimitry Andric      return traits_type::not_eof(__c);
35120b57cec5SDimitry Andric    }
3513cb14a3feSDimitry Andric    if (traits_type::eq(traits_type::to_char_type(__c), this->gptr()[-1])) {
35140b57cec5SDimitry Andric      this->gbump(-1);
35150b57cec5SDimitry Andric      *this->gptr() = traits_type::to_char_type(__c);
35160b57cec5SDimitry Andric      return __c;
35170b57cec5SDimitry Andric    }
35180b57cec5SDimitry Andric  }
35190b57cec5SDimitry Andric  return traits_type::eof();
35200b57cec5SDimitry Andric}
35210b57cec5SDimitry Andric
352281ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
35230b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3524cb14a3feSDimitry Andrictypename wbuffer_convert<_Codecvt, _Elem, _Tr>::int_type wbuffer_convert<_Codecvt, _Elem, _Tr>::overflow(int_type __c) {
352581ad6265SDimitry Andric  _LIBCPP_SUPPRESS_DEPRECATED_POP
35267a6dacacSDimitry Andric  if (__cv_ == 0 || !__bufptr_)
35270b57cec5SDimitry Andric    return traits_type::eof();
35280b57cec5SDimitry Andric  __write_mode();
35290b57cec5SDimitry Andric  char_type __1buf;
35300b57cec5SDimitry Andric  char_type* __pb_save  = this->pbase();
35310b57cec5SDimitry Andric  char_type* __epb_save = this->epptr();
3532cb14a3feSDimitry Andric  if (!traits_type::eq_int_type(__c, traits_type::eof())) {
35330b57cec5SDimitry Andric    if (this->pptr() == 0)
35340b57cec5SDimitry Andric      this->setp(&__1buf, &__1buf + 1);
35350b57cec5SDimitry Andric    *this->pptr() = traits_type::to_char_type(__c);
35360b57cec5SDimitry Andric    this->pbump(1);
35370b57cec5SDimitry Andric  }
3538cb14a3feSDimitry Andric  if (this->pptr() != this->pbase()) {
3539cb14a3feSDimitry Andric    if (__always_noconv_) {
35400b57cec5SDimitry Andric      streamsize __nmemb = static_cast<streamsize>(this->pptr() - this->pbase());
35410b57cec5SDimitry Andric      if (__bufptr_->sputn((const char*)this->pbase(), __nmemb) != __nmemb)
35420b57cec5SDimitry Andric        return traits_type::eof();
3543cb14a3feSDimitry Andric    } else {
35440b57cec5SDimitry Andric      char* __extbe = __extbuf_;
35450b57cec5SDimitry Andric      codecvt_base::result __r;
3546cb14a3feSDimitry Andric      do {
35470b57cec5SDimitry Andric        const char_type* __e;
3548cb14a3feSDimitry Andric        __r = __cv_->out(__st_, this->pbase(), this->pptr(), __e, __extbuf_, __extbuf_ + __ebs_, __extbe);
35490b57cec5SDimitry Andric        if (__e == this->pbase())
35500b57cec5SDimitry Andric          return traits_type::eof();
3551cb14a3feSDimitry Andric        if (__r == codecvt_base::noconv) {
35520b57cec5SDimitry Andric          streamsize __nmemb = static_cast<size_t>(this->pptr() - this->pbase());
35530b57cec5SDimitry Andric          if (__bufptr_->sputn((const char*)this->pbase(), __nmemb) != __nmemb)
35540b57cec5SDimitry Andric            return traits_type::eof();
3555cb14a3feSDimitry Andric        } else if (__r == codecvt_base::ok || __r == codecvt_base::partial) {
35560b57cec5SDimitry Andric          streamsize __nmemb = static_cast<size_t>(__extbe - __extbuf_);
35570b57cec5SDimitry Andric          if (__bufptr_->sputn(__extbuf_, __nmemb) != __nmemb)
35580b57cec5SDimitry Andric            return traits_type::eof();
3559cb14a3feSDimitry Andric          if (__r == codecvt_base::partial) {
35600b57cec5SDimitry Andric            this->setp(const_cast<char_type*>(__e), this->pptr());
35610b57cec5SDimitry Andric            this->__pbump(this->epptr() - this->pbase());
35620b57cec5SDimitry Andric          }
3563cb14a3feSDimitry Andric        } else
35640b57cec5SDimitry Andric          return traits_type::eof();
35650b57cec5SDimitry Andric      } while (__r == codecvt_base::partial);
35660b57cec5SDimitry Andric    }
35670b57cec5SDimitry Andric    this->setp(__pb_save, __epb_save);
35680b57cec5SDimitry Andric  }
35690b57cec5SDimitry Andric  return traits_type::not_eof(__c);
35700b57cec5SDimitry Andric}
35710b57cec5SDimitry Andric
357281ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
35730b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3574cb14a3feSDimitry Andricbasic_streambuf<_Elem, _Tr>* wbuffer_convert<_Codecvt, _Elem, _Tr>::setbuf(char_type* __s, streamsize __n) {
357581ad6265SDimitry Andric  _LIBCPP_SUPPRESS_DEPRECATED_POP
35760b57cec5SDimitry Andric  this->setg(0, 0, 0);
35770b57cec5SDimitry Andric  this->setp(0, 0);
35780b57cec5SDimitry Andric  if (__owns_eb_)
35790b57cec5SDimitry Andric    delete[] __extbuf_;
35800b57cec5SDimitry Andric  if (__owns_ib_)
35810b57cec5SDimitry Andric    delete[] __intbuf_;
35820b57cec5SDimitry Andric  __ebs_ = __n;
3583cb14a3feSDimitry Andric  if (__ebs_ > sizeof(__extbuf_min_)) {
3584cb14a3feSDimitry Andric    if (__always_noconv_ && __s) {
35850b57cec5SDimitry Andric      __extbuf_  = (char*)__s;
35860b57cec5SDimitry Andric      __owns_eb_ = false;
3587cb14a3feSDimitry Andric    } else {
35880b57cec5SDimitry Andric      __extbuf_  = new char[__ebs_];
35890b57cec5SDimitry Andric      __owns_eb_ = true;
35900b57cec5SDimitry Andric    }
3591cb14a3feSDimitry Andric  } else {
35920b57cec5SDimitry Andric    __extbuf_  = __extbuf_min_;
35930b57cec5SDimitry Andric    __ebs_     = sizeof(__extbuf_min_);
35940b57cec5SDimitry Andric    __owns_eb_ = false;
35950b57cec5SDimitry Andric  }
3596cb14a3feSDimitry Andric  if (!__always_noconv_) {
35970b57cec5SDimitry Andric    __ibs_ = max<streamsize>(__n, sizeof(__extbuf_min_));
3598cb14a3feSDimitry Andric    if (__s && __ibs_ >= sizeof(__extbuf_min_)) {
35990b57cec5SDimitry Andric      __intbuf_  = __s;
36000b57cec5SDimitry Andric      __owns_ib_ = false;
3601cb14a3feSDimitry Andric    } else {
36020b57cec5SDimitry Andric      __intbuf_  = new char_type[__ibs_];
36030b57cec5SDimitry Andric      __owns_ib_ = true;
36040b57cec5SDimitry Andric    }
3605cb14a3feSDimitry Andric  } else {
36060b57cec5SDimitry Andric    __ibs_     = 0;
36070b57cec5SDimitry Andric    __intbuf_  = 0;
36080b57cec5SDimitry Andric    __owns_ib_ = false;
36090b57cec5SDimitry Andric  }
36100b57cec5SDimitry Andric  return this;
36110b57cec5SDimitry Andric}
36120b57cec5SDimitry Andric
361381ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
36140b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
36150b57cec5SDimitry Andrictypename wbuffer_convert<_Codecvt, _Elem, _Tr>::pos_type
3616cb14a3feSDimitry Andricwbuffer_convert<_Codecvt, _Elem, _Tr>::seekoff(off_type __off, ios_base::seekdir __way, ios_base::openmode __om) {
36170b57cec5SDimitry Andric  int __width = __cv_->encoding();
36187a6dacacSDimitry Andric  if (__cv_ == 0 || !__bufptr_ || (__width <= 0 && __off != 0) || sync())
36190b57cec5SDimitry Andric    return pos_type(off_type(-1));
36200b57cec5SDimitry Andric  // __width > 0 || __off == 0, now check __way
36210b57cec5SDimitry Andric  if (__way != ios_base::beg && __way != ios_base::cur && __way != ios_base::end)
36220b57cec5SDimitry Andric    return pos_type(off_type(-1));
36230b57cec5SDimitry Andric  pos_type __r = __bufptr_->pubseekoff(__width * __off, __way, __om);
36240b57cec5SDimitry Andric  __r.state(__st_);
36250b57cec5SDimitry Andric  return __r;
36260b57cec5SDimitry Andric}
36270b57cec5SDimitry Andric
36280b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
36290b57cec5SDimitry Andrictypename wbuffer_convert<_Codecvt, _Elem, _Tr>::pos_type
3630cb14a3feSDimitry Andricwbuffer_convert<_Codecvt, _Elem, _Tr>::seekpos(pos_type __sp, ios_base::openmode __wch) {
36317a6dacacSDimitry Andric  if (__cv_ == 0 || !__bufptr_ || sync())
36320b57cec5SDimitry Andric    return pos_type(off_type(-1));
36330b57cec5SDimitry Andric  if (__bufptr_->pubseekpos(__sp, __wch) == pos_type(off_type(-1)))
36340b57cec5SDimitry Andric    return pos_type(off_type(-1));
36350b57cec5SDimitry Andric  return __sp;
36360b57cec5SDimitry Andric}
36370b57cec5SDimitry Andric
36380b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3639cb14a3feSDimitry Andricint wbuffer_convert<_Codecvt, _Elem, _Tr>::sync() {
364081ad6265SDimitry Andric  _LIBCPP_SUPPRESS_DEPRECATED_POP
36417a6dacacSDimitry Andric  if (__cv_ == 0 || !__bufptr_)
36420b57cec5SDimitry Andric    return 0;
3643cb14a3feSDimitry Andric  if (__cm_ & ios_base::out) {
36440b57cec5SDimitry Andric    if (this->pptr() != this->pbase())
36450b57cec5SDimitry Andric      if (overflow() == traits_type::eof())
36460b57cec5SDimitry Andric        return -1;
36470b57cec5SDimitry Andric    codecvt_base::result __r;
3648cb14a3feSDimitry Andric    do {
36490b57cec5SDimitry Andric      char* __extbe;
36500b57cec5SDimitry Andric      __r                = __cv_->unshift(__st_, __extbuf_, __extbuf_ + __ebs_, __extbe);
36510b57cec5SDimitry Andric      streamsize __nmemb = static_cast<streamsize>(__extbe - __extbuf_);
36520b57cec5SDimitry Andric      if (__bufptr_->sputn(__extbuf_, __nmemb) != __nmemb)
36530b57cec5SDimitry Andric        return -1;
36540b57cec5SDimitry Andric    } while (__r == codecvt_base::partial);
36550b57cec5SDimitry Andric    if (__r == codecvt_base::error)
36560b57cec5SDimitry Andric      return -1;
36570b57cec5SDimitry Andric    if (__bufptr_->pubsync())
36580b57cec5SDimitry Andric      return -1;
3659cb14a3feSDimitry Andric  } else if (__cm_ & ios_base::in) {
36600b57cec5SDimitry Andric    off_type __c;
36610b57cec5SDimitry Andric    if (__always_noconv_)
36620b57cec5SDimitry Andric      __c = this->egptr() - this->gptr();
3663cb14a3feSDimitry Andric    else {
36640b57cec5SDimitry Andric      int __width = __cv_->encoding();
36650b57cec5SDimitry Andric      __c         = __extbufend_ - __extbufnext_;
36660b57cec5SDimitry Andric      if (__width > 0)
36670b57cec5SDimitry Andric        __c += __width * (this->egptr() - this->gptr());
3668cb14a3feSDimitry Andric      else {
3669cb14a3feSDimitry Andric        if (this->gptr() != this->egptr()) {
3670bdd1243dSDimitry Andric          std::reverse(this->gptr(), this->egptr());
36710b57cec5SDimitry Andric          codecvt_base::result __r;
36720b57cec5SDimitry Andric          const char_type* __e = this->gptr();
36730b57cec5SDimitry Andric          char* __extbe;
3674cb14a3feSDimitry Andric          do {
3675cb14a3feSDimitry Andric            __r = __cv_->out(__st_, __e, this->egptr(), __e, __extbuf_, __extbuf_ + __ebs_, __extbe);
3676cb14a3feSDimitry Andric            switch (__r) {
36770b57cec5SDimitry Andric            case codecvt_base::noconv:
36780b57cec5SDimitry Andric              __c += this->egptr() - this->gptr();
36790b57cec5SDimitry Andric              break;
36800b57cec5SDimitry Andric            case codecvt_base::ok:
36810b57cec5SDimitry Andric            case codecvt_base::partial:
36820b57cec5SDimitry Andric              __c += __extbe - __extbuf_;
36830b57cec5SDimitry Andric              break;
36840b57cec5SDimitry Andric            default:
36850b57cec5SDimitry Andric              return -1;
36860b57cec5SDimitry Andric            }
36870b57cec5SDimitry Andric          } while (__r == codecvt_base::partial);
36880b57cec5SDimitry Andric        }
36890b57cec5SDimitry Andric      }
36900b57cec5SDimitry Andric    }
36910b57cec5SDimitry Andric    if (__bufptr_->pubseekoff(-__c, ios_base::cur, __cm_) == pos_type(off_type(-1)))
36920b57cec5SDimitry Andric      return -1;
36930b57cec5SDimitry Andric    this->setg(0, 0, 0);
36940b57cec5SDimitry Andric    __cm_ = 0;
36950b57cec5SDimitry Andric  }
36960b57cec5SDimitry Andric  return 0;
36970b57cec5SDimitry Andric}
36980b57cec5SDimitry Andric
369981ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_PUSH
37000b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3701cb14a3feSDimitry Andricbool wbuffer_convert<_Codecvt, _Elem, _Tr>::__read_mode() {
3702cb14a3feSDimitry Andric  if (!(__cm_ & ios_base::in)) {
37030b57cec5SDimitry Andric    this->setp(0, 0);
37040b57cec5SDimitry Andric    if (__always_noconv_)
3705cb14a3feSDimitry Andric      this->setg((char_type*)__extbuf_, (char_type*)__extbuf_ + __ebs_, (char_type*)__extbuf_ + __ebs_);
37060b57cec5SDimitry Andric    else
37070b57cec5SDimitry Andric      this->setg(__intbuf_, __intbuf_ + __ibs_, __intbuf_ + __ibs_);
37080b57cec5SDimitry Andric    __cm_ = ios_base::in;
37090b57cec5SDimitry Andric    return true;
37100b57cec5SDimitry Andric  }
37110b57cec5SDimitry Andric  return false;
37120b57cec5SDimitry Andric}
37130b57cec5SDimitry Andric
37140b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3715cb14a3feSDimitry Andricvoid wbuffer_convert<_Codecvt, _Elem, _Tr>::__write_mode() {
3716cb14a3feSDimitry Andric  if (!(__cm_ & ios_base::out)) {
37170b57cec5SDimitry Andric    this->setg(0, 0, 0);
3718cb14a3feSDimitry Andric    if (__ebs_ > sizeof(__extbuf_min_)) {
37190b57cec5SDimitry Andric      if (__always_noconv_)
3720cb14a3feSDimitry Andric        this->setp((char_type*)__extbuf_, (char_type*)__extbuf_ + (__ebs_ - 1));
37210b57cec5SDimitry Andric      else
37220b57cec5SDimitry Andric        this->setp(__intbuf_, __intbuf_ + (__ibs_ - 1));
3723cb14a3feSDimitry Andric    } else
37240b57cec5SDimitry Andric      this->setp(0, 0);
37250b57cec5SDimitry Andric    __cm_ = ios_base::out;
37260b57cec5SDimitry Andric  }
37270b57cec5SDimitry Andric}
37280b57cec5SDimitry Andric
37290b57cec5SDimitry Andrictemplate <class _Codecvt, class _Elem, class _Tr>
3730cb14a3feSDimitry Andricwbuffer_convert<_Codecvt, _Elem, _Tr>* wbuffer_convert<_Codecvt, _Elem, _Tr>::__close() {
3731e8d8bef9SDimitry Andric  wbuffer_convert* __rt = nullptr;
3732cb14a3feSDimitry Andric  if (__cv_ != nullptr && __bufptr_ != nullptr) {
37330b57cec5SDimitry Andric    __rt = this;
37340b57cec5SDimitry Andric    if ((__cm_ & ios_base::out) && sync())
3735e8d8bef9SDimitry Andric      __rt = nullptr;
37360b57cec5SDimitry Andric  }
37370b57cec5SDimitry Andric  return __rt;
37380b57cec5SDimitry Andric}
37390b57cec5SDimitry Andric
374081ad6265SDimitry Andric_LIBCPP_SUPPRESS_DEPRECATED_POP
374181ad6265SDimitry Andric
3742*0fca6ea1SDimitry Andric#  endif // _LIBCPP_STD_VER < 26 || defined(_LIBCPP_ENABLE_CXX26_REMOVED_WSTRING_CONVERT)
3743*0fca6ea1SDimitry Andric
37440b57cec5SDimitry Andric_LIBCPP_END_NAMESPACE_STD
37450b57cec5SDimitry Andric
37460b57cec5SDimitry Andric_LIBCPP_POP_MACROS
37470b57cec5SDimitry Andric
3748bdd1243dSDimitry Andric// NOLINTEND(libcpp-robust-against-adl)
3749bdd1243dSDimitry Andric
3750*0fca6ea1SDimitry Andric#endif // !defined(_LIBCPP_HAS_NO_LOCALIZATION)
3751*0fca6ea1SDimitry Andric
3752bdd1243dSDimitry Andric#if !defined(_LIBCPP_REMOVE_TRANSITIVE_INCLUDES) && _LIBCPP_STD_VER <= 20
3753bdd1243dSDimitry Andric#  include <atomic>
3754bdd1243dSDimitry Andric#  include <concepts>
3755bdd1243dSDimitry Andric#  include <cstdarg>
3756bdd1243dSDimitry Andric#  include <iterator>
37575f757f3fSDimitry Andric#  include <mutex>
3758bdd1243dSDimitry Andric#  include <stdexcept>
3759bdd1243dSDimitry Andric#  include <type_traits>
3760bdd1243dSDimitry Andric#  include <typeinfo>
3761bdd1243dSDimitry Andric#endif
3762bdd1243dSDimitry Andric
37630b57cec5SDimitry Andric#endif // _LIBCPP_LOCALE
3764