[MemProf] Templatize CallStackRadixTreeBuilder (NFC) (#117014)
[llvm-project.git] / libcxx / src / locale.cpp
bloba1e10401f0b299eccbd569c06928b0467ce96aa2
1 //===----------------------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
9 #include <__utility/no_destroy.h>
10 #include <algorithm>
11 #include <clocale>
12 #include <codecvt>
13 #include <cstddef>
14 #include <cstdio>
15 #include <cstdlib>
16 #include <cstring>
17 #include <locale>
18 #include <new>
19 #include <string>
20 #include <type_traits>
21 #include <typeinfo>
22 #include <utility>
23 #include <vector>
25 #if _LIBCPP_HAS_WIDE_CHARACTERS
26 # include <cwctype>
27 #endif
29 #if defined(_AIX)
30 # include <sys/localedef.h> // for __lc_ctype_ptr
31 #endif
33 #if defined(_LIBCPP_MSVCRT)
34 # define _CTYPE_DISABLE_MACROS
35 #endif
37 #if __has_include("<langinfo.h>")
38 # include <langinfo.h>
39 #endif
41 #include "include/atomic_support.h"
42 #include "include/sso_allocator.h"
44 // On Linux, wint_t and wchar_t have different signed-ness, and this causes
45 // lots of noise in the build log, but no bugs that I know of.
46 _LIBCPP_CLANG_DIAGNOSTIC_IGNORED("-Wsign-conversion")
48 _LIBCPP_PUSH_MACROS
49 #include <__undef_macros>
51 _LIBCPP_BEGIN_NAMESPACE_STD
53 struct __libcpp_unique_locale {
54 __libcpp_unique_locale(const char* nm) : __loc_(__locale::__newlocale(LC_ALL_MASK, nm, 0)) {}
56 ~__libcpp_unique_locale() {
57 if (__loc_)
58 __locale::__freelocale(__loc_);
61 explicit operator bool() const { return __loc_; }
63 __locale::__locale_t& get() { return __loc_; }
65 __locale::__locale_t __loc_;
67 private:
68 __libcpp_unique_locale(__libcpp_unique_locale const&);
69 __libcpp_unique_locale& operator=(__libcpp_unique_locale const&);
72 #ifdef __cloc_defined
73 __locale::__locale_t __cloc() {
74 // In theory this could create a race condition. In practice
75 // the race condition is non-fatal since it will just create
76 // a little resource leak. Better approach would be appreciated.
77 static __locale::__locale_t result = __locale::__newlocale(LC_ALL_MASK, "C", 0);
78 return result;
80 #endif // __cloc_defined
82 namespace {
84 struct releaser {
85 void operator()(locale::facet* p) { p->__release_shared(); }
88 template <class T, class... Args>
89 T& make(Args... args) {
90 alignas(T) static std::byte buf[sizeof(T)];
91 auto* obj = ::new (&buf) T(args...);
92 return *obj;
95 template <typename T, size_t N>
96 inline constexpr size_t countof(const T (&)[N]) {
97 return N;
100 template <typename T>
101 inline constexpr size_t countof(const T* const begin, const T* const end) {
102 return static_cast<size_t>(end - begin);
105 string build_name(const string& other, const string& one, locale::category c) {
106 if (other == "*" || one == "*")
107 return "*";
108 if (c == locale::none || other == one)
109 return other;
111 // FIXME: Handle the more complicated cases, such as when the locale has
112 // different names for different categories.
113 return "*";
116 } // namespace
118 const locale::category locale::none;
119 const locale::category locale::collate;
120 const locale::category locale::ctype;
121 const locale::category locale::monetary;
122 const locale::category locale::numeric;
123 const locale::category locale::time;
124 const locale::category locale::messages;
125 const locale::category locale::all;
127 class _LIBCPP_HIDDEN locale::__imp : public facet {
128 enum { N = 30 };
129 vector<facet*, __sso_allocator<facet*, N> > facets_;
130 string name_;
132 public:
133 explicit __imp(size_t refs = 0);
134 explicit __imp(const string& name, size_t refs = 0);
135 __imp(const __imp&);
136 __imp(const __imp&, const string&, locale::category c);
137 __imp(const __imp& other, const __imp& one, locale::category c);
138 __imp(const __imp&, facet* f, long id);
139 ~__imp();
141 const string& name() const { return name_; }
142 bool has_facet(long id) const { return static_cast<size_t>(id) < facets_.size() && facets_[static_cast<size_t>(id)]; }
143 const locale::facet* use_facet(long id) const;
145 void acquire();
146 void release();
147 static __no_destroy<__imp> classic_locale_imp_;
149 private:
150 void install(facet* f, long id);
151 template <class F>
152 void install(F* f) {
153 install(f, f->id.__get());
155 template <class F>
156 void install_from(const __imp& other);
159 locale::__imp::__imp(size_t refs) : facet(refs), facets_(N), name_("C") {
160 facets_.clear();
161 install(&make<std::collate<char> >(1u));
162 #if _LIBCPP_HAS_WIDE_CHARACTERS
163 install(&make<std::collate<wchar_t> >(1u));
164 #endif
165 install(&make<std::ctype<char> >(nullptr, false, 1u));
166 #if _LIBCPP_HAS_WIDE_CHARACTERS
167 install(&make<std::ctype<wchar_t> >(1u));
168 #endif
169 install(&make<codecvt<char, char, mbstate_t> >(1u));
170 #if _LIBCPP_HAS_WIDE_CHARACTERS
171 install(&make<codecvt<wchar_t, char, mbstate_t> >(1u));
172 #endif
173 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
174 install(&make<codecvt<char16_t, char, mbstate_t> >(1u));
175 install(&make<codecvt<char32_t, char, mbstate_t> >(1u));
176 _LIBCPP_SUPPRESS_DEPRECATED_POP
177 #if _LIBCPP_HAS_CHAR8_T
178 install(&make<codecvt<char16_t, char8_t, mbstate_t> >(1u));
179 install(&make<codecvt<char32_t, char8_t, mbstate_t> >(1u));
180 #endif
181 install(&make<numpunct<char> >(1u));
182 #if _LIBCPP_HAS_WIDE_CHARACTERS
183 install(&make<numpunct<wchar_t> >(1u));
184 #endif
185 install(&make<num_get<char> >(1u));
186 #if _LIBCPP_HAS_WIDE_CHARACTERS
187 install(&make<num_get<wchar_t> >(1u));
188 #endif
189 install(&make<num_put<char> >(1u));
190 #if _LIBCPP_HAS_WIDE_CHARACTERS
191 install(&make<num_put<wchar_t> >(1u));
192 #endif
193 install(&make<moneypunct<char, false> >(1u));
194 install(&make<moneypunct<char, true> >(1u));
195 #if _LIBCPP_HAS_WIDE_CHARACTERS
196 install(&make<moneypunct<wchar_t, false> >(1u));
197 install(&make<moneypunct<wchar_t, true> >(1u));
198 #endif
199 install(&make<money_get<char> >(1u));
200 #if _LIBCPP_HAS_WIDE_CHARACTERS
201 install(&make<money_get<wchar_t> >(1u));
202 #endif
203 install(&make<money_put<char> >(1u));
204 #if _LIBCPP_HAS_WIDE_CHARACTERS
205 install(&make<money_put<wchar_t> >(1u));
206 #endif
207 install(&make<time_get<char> >(1u));
208 #if _LIBCPP_HAS_WIDE_CHARACTERS
209 install(&make<time_get<wchar_t> >(1u));
210 #endif
211 install(&make<time_put<char> >(1u));
212 #if _LIBCPP_HAS_WIDE_CHARACTERS
213 install(&make<time_put<wchar_t> >(1u));
214 #endif
215 install(&make<std::messages<char> >(1u));
216 #if _LIBCPP_HAS_WIDE_CHARACTERS
217 install(&make<std::messages<wchar_t> >(1u));
218 #endif
221 locale::__imp::__imp(const string& name, size_t refs) : facet(refs), facets_(N), name_(name) {
222 #if _LIBCPP_HAS_EXCEPTIONS
223 try {
224 #endif // _LIBCPP_HAS_EXCEPTIONS
225 facets_ = locale::classic().__locale_->facets_;
226 for (unsigned i = 0; i < facets_.size(); ++i)
227 if (facets_[i])
228 facets_[i]->__add_shared();
229 install(new collate_byname<char>(name_));
230 #if _LIBCPP_HAS_WIDE_CHARACTERS
231 install(new collate_byname<wchar_t>(name_));
232 #endif
233 install(new ctype_byname<char>(name_));
234 #if _LIBCPP_HAS_WIDE_CHARACTERS
235 install(new ctype_byname<wchar_t>(name_));
236 #endif
237 install(new codecvt_byname<char, char, mbstate_t>(name_));
238 #if _LIBCPP_HAS_WIDE_CHARACTERS
239 install(new codecvt_byname<wchar_t, char, mbstate_t>(name_));
240 #endif
241 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
242 install(new codecvt_byname<char16_t, char, mbstate_t>(name_));
243 install(new codecvt_byname<char32_t, char, mbstate_t>(name_));
244 _LIBCPP_SUPPRESS_DEPRECATED_POP
245 #if _LIBCPP_HAS_CHAR8_T
246 install(new codecvt_byname<char16_t, char8_t, mbstate_t>(name_));
247 install(new codecvt_byname<char32_t, char8_t, mbstate_t>(name_));
248 #endif
249 install(new numpunct_byname<char>(name_));
250 #if _LIBCPP_HAS_WIDE_CHARACTERS
251 install(new numpunct_byname<wchar_t>(name_));
252 #endif
253 install(new moneypunct_byname<char, false>(name_));
254 install(new moneypunct_byname<char, true>(name_));
255 #if _LIBCPP_HAS_WIDE_CHARACTERS
256 install(new moneypunct_byname<wchar_t, false>(name_));
257 install(new moneypunct_byname<wchar_t, true>(name_));
258 #endif
259 install(new time_get_byname<char>(name_));
260 #if _LIBCPP_HAS_WIDE_CHARACTERS
261 install(new time_get_byname<wchar_t>(name_));
262 #endif
263 install(new time_put_byname<char>(name_));
264 #if _LIBCPP_HAS_WIDE_CHARACTERS
265 install(new time_put_byname<wchar_t>(name_));
266 #endif
267 install(new messages_byname<char>(name_));
268 #if _LIBCPP_HAS_WIDE_CHARACTERS
269 install(new messages_byname<wchar_t>(name_));
270 #endif
271 #if _LIBCPP_HAS_EXCEPTIONS
272 } catch (...) {
273 for (unsigned i = 0; i < facets_.size(); ++i)
274 if (facets_[i])
275 facets_[i]->__release_shared();
276 throw;
278 #endif // _LIBCPP_HAS_EXCEPTIONS
281 locale::__imp::__imp(const __imp& other) : facets_(max<size_t>(N, other.facets_.size())), name_(other.name_) {
282 facets_ = other.facets_;
283 for (unsigned i = 0; i < facets_.size(); ++i)
284 if (facets_[i])
285 facets_[i]->__add_shared();
288 locale::__imp::__imp(const __imp& other, const string& name, locale::category c)
289 : facets_(N), name_(build_name(other.name_, name, c)) {
290 facets_ = other.facets_;
291 for (unsigned i = 0; i < facets_.size(); ++i)
292 if (facets_[i])
293 facets_[i]->__add_shared();
294 #if _LIBCPP_HAS_EXCEPTIONS
295 try {
296 #endif // _LIBCPP_HAS_EXCEPTIONS
297 if (c & locale::collate) {
298 install(new collate_byname<char>(name));
299 #if _LIBCPP_HAS_WIDE_CHARACTERS
300 install(new collate_byname<wchar_t>(name));
301 #endif
303 if (c & locale::ctype) {
304 install(new ctype_byname<char>(name));
305 #if _LIBCPP_HAS_WIDE_CHARACTERS
306 install(new ctype_byname<wchar_t>(name));
307 #endif
308 install(new codecvt_byname<char, char, mbstate_t>(name));
309 #if _LIBCPP_HAS_WIDE_CHARACTERS
310 install(new codecvt_byname<wchar_t, char, mbstate_t>(name));
311 #endif
312 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
313 install(new codecvt_byname<char16_t, char, mbstate_t>(name));
314 install(new codecvt_byname<char32_t, char, mbstate_t>(name));
315 _LIBCPP_SUPPRESS_DEPRECATED_POP
316 #if _LIBCPP_HAS_CHAR8_T
317 install(new codecvt_byname<char16_t, char8_t, mbstate_t>(name));
318 install(new codecvt_byname<char32_t, char8_t, mbstate_t>(name));
319 #endif
321 if (c & locale::monetary) {
322 install(new moneypunct_byname<char, false>(name));
323 install(new moneypunct_byname<char, true>(name));
324 #if _LIBCPP_HAS_WIDE_CHARACTERS
325 install(new moneypunct_byname<wchar_t, false>(name));
326 install(new moneypunct_byname<wchar_t, true>(name));
327 #endif
329 if (c & locale::numeric) {
330 install(new numpunct_byname<char>(name));
331 #if _LIBCPP_HAS_WIDE_CHARACTERS
332 install(new numpunct_byname<wchar_t>(name));
333 #endif
335 if (c & locale::time) {
336 install(new time_get_byname<char>(name));
337 #if _LIBCPP_HAS_WIDE_CHARACTERS
338 install(new time_get_byname<wchar_t>(name));
339 #endif
340 install(new time_put_byname<char>(name));
341 #if _LIBCPP_HAS_WIDE_CHARACTERS
342 install(new time_put_byname<wchar_t>(name));
343 #endif
345 if (c & locale::messages) {
346 install(new messages_byname<char>(name));
347 #if _LIBCPP_HAS_WIDE_CHARACTERS
348 install(new messages_byname<wchar_t>(name));
349 #endif
351 #if _LIBCPP_HAS_EXCEPTIONS
352 } catch (...) {
353 for (unsigned i = 0; i < facets_.size(); ++i)
354 if (facets_[i])
355 facets_[i]->__release_shared();
356 throw;
358 #endif // _LIBCPP_HAS_EXCEPTIONS
361 template <class F>
362 inline void locale::__imp::install_from(const locale::__imp& one) {
363 long id = F::id.__get();
364 install(const_cast<F*>(static_cast<const F*>(one.use_facet(id))), id);
367 locale::__imp::__imp(const __imp& other, const __imp& one, locale::category c)
368 : facets_(N), name_(build_name(other.name_, one.name_, c)) {
369 facets_ = other.facets_;
370 for (unsigned i = 0; i < facets_.size(); ++i)
371 if (facets_[i])
372 facets_[i]->__add_shared();
373 #if _LIBCPP_HAS_EXCEPTIONS
374 try {
375 #endif // _LIBCPP_HAS_EXCEPTIONS
376 if (c & locale::collate) {
377 install_from<std::collate<char> >(one);
378 #if _LIBCPP_HAS_WIDE_CHARACTERS
379 install_from<std::collate<wchar_t> >(one);
380 #endif
382 if (c & locale::ctype) {
383 install_from<std::ctype<char> >(one);
384 #if _LIBCPP_HAS_WIDE_CHARACTERS
385 install_from<std::ctype<wchar_t> >(one);
386 #endif
387 install_from<std::codecvt<char, char, mbstate_t> >(one);
388 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
389 install_from<std::codecvt<char16_t, char, mbstate_t> >(one);
390 install_from<std::codecvt<char32_t, char, mbstate_t> >(one);
391 _LIBCPP_SUPPRESS_DEPRECATED_POP
392 #if _LIBCPP_HAS_CHAR8_T
393 install_from<std::codecvt<char16_t, char8_t, mbstate_t> >(one);
394 install_from<std::codecvt<char32_t, char8_t, mbstate_t> >(one);
395 #endif
396 #if _LIBCPP_HAS_WIDE_CHARACTERS
397 install_from<std::codecvt<wchar_t, char, mbstate_t> >(one);
398 #endif
400 if (c & locale::monetary) {
401 install_from<moneypunct<char, false> >(one);
402 install_from<moneypunct<char, true> >(one);
403 #if _LIBCPP_HAS_WIDE_CHARACTERS
404 install_from<moneypunct<wchar_t, false> >(one);
405 install_from<moneypunct<wchar_t, true> >(one);
406 #endif
407 install_from<money_get<char> >(one);
408 #if _LIBCPP_HAS_WIDE_CHARACTERS
409 install_from<money_get<wchar_t> >(one);
410 #endif
411 install_from<money_put<char> >(one);
412 #if _LIBCPP_HAS_WIDE_CHARACTERS
413 install_from<money_put<wchar_t> >(one);
414 #endif
416 if (c & locale::numeric) {
417 install_from<numpunct<char> >(one);
418 #if _LIBCPP_HAS_WIDE_CHARACTERS
419 install_from<numpunct<wchar_t> >(one);
420 #endif
421 install_from<num_get<char> >(one);
422 #if _LIBCPP_HAS_WIDE_CHARACTERS
423 install_from<num_get<wchar_t> >(one);
424 #endif
425 install_from<num_put<char> >(one);
426 #if _LIBCPP_HAS_WIDE_CHARACTERS
427 install_from<num_put<wchar_t> >(one);
428 #endif
430 if (c & locale::time) {
431 install_from<time_get<char> >(one);
432 #if _LIBCPP_HAS_WIDE_CHARACTERS
433 install_from<time_get<wchar_t> >(one);
434 #endif
435 install_from<time_put<char> >(one);
436 #if _LIBCPP_HAS_WIDE_CHARACTERS
437 install_from<time_put<wchar_t> >(one);
438 #endif
440 if (c & locale::messages) {
441 install_from<std::messages<char> >(one);
442 #if _LIBCPP_HAS_WIDE_CHARACTERS
443 install_from<std::messages<wchar_t> >(one);
444 #endif
446 #if _LIBCPP_HAS_EXCEPTIONS
447 } catch (...) {
448 for (unsigned i = 0; i < facets_.size(); ++i)
449 if (facets_[i])
450 facets_[i]->__release_shared();
451 throw;
453 #endif // _LIBCPP_HAS_EXCEPTIONS
456 locale::__imp::__imp(const __imp& other, facet* f, long id)
457 : facets_(max<size_t>(N, other.facets_.size() + 1)), name_("*") {
458 f->__add_shared();
459 unique_ptr<facet, releaser> hold(f);
460 facets_ = other.facets_;
461 for (unsigned i = 0; i < other.facets_.size(); ++i)
462 if (facets_[i])
463 facets_[i]->__add_shared();
464 install(hold.get(), id);
467 locale::__imp::~__imp() {
468 for (unsigned i = 0; i < facets_.size(); ++i)
469 if (facets_[i])
470 facets_[i]->__release_shared();
473 void locale::__imp::install(facet* f, long id) {
474 f->__add_shared();
475 unique_ptr<facet, releaser> hold(f);
476 if (static_cast<size_t>(id) >= facets_.size())
477 facets_.resize(static_cast<size_t>(id + 1));
478 if (facets_[static_cast<size_t>(id)])
479 facets_[static_cast<size_t>(id)]->__release_shared();
480 facets_[static_cast<size_t>(id)] = hold.release();
483 const locale::facet* locale::__imp::use_facet(long id) const {
484 if (!has_facet(id))
485 __throw_bad_cast();
486 return facets_[static_cast<size_t>(id)];
489 // locale
491 // We don't do reference counting on the classic locale.
492 // It's never destroyed anyway, but atomic reference counting may be very
493 // expensive in parallel applications. The classic locale is used by default
494 // in all streams. Note: if a new global locale is installed, then we lose
495 // the benefit of no reference counting.
496 constinit __no_destroy<locale::__imp>
497 locale::__imp::classic_locale_imp_(__uninitialized_tag{}); // initialized below in classic()
499 const locale& locale::classic() {
500 static const __no_destroy<locale> classic_locale(__private_constructor_tag{}, [] {
501 // executed exactly once on first initialization of `classic_locale`
502 locale::__imp::classic_locale_imp_.__emplace(1u);
503 return &locale::__imp::classic_locale_imp_.__get();
504 }());
505 return classic_locale.__get();
508 locale& locale::__global() {
509 static __no_destroy<locale> g(locale::classic());
510 return g.__get();
513 void locale::__imp::acquire() {
514 if (this != &locale::__imp::classic_locale_imp_.__get())
515 __add_shared();
518 void locale::__imp::release() {
519 if (this != &locale::__imp::classic_locale_imp_.__get())
520 __release_shared();
523 locale::locale() noexcept : __locale_(__global().__locale_) { __locale_->acquire(); }
525 locale::locale(const locale& l) noexcept : __locale_(l.__locale_) { __locale_->acquire(); }
527 locale::~locale() { __locale_->release(); }
529 const locale& locale::operator=(const locale& other) noexcept {
530 other.__locale_->acquire();
531 __locale_->release();
532 __locale_ = other.__locale_;
533 return *this;
536 locale::locale(const char* name)
537 : __locale_(name ? new __imp(name) : (__throw_runtime_error("locale constructed with null"), nullptr)) {
538 __locale_->acquire();
541 locale::locale(const string& name) : __locale_(new __imp(name)) { __locale_->acquire(); }
543 locale::locale(const locale& other, const char* name, category c)
544 : __locale_(name ? new __imp(*other.__locale_, name, c)
545 : (__throw_runtime_error("locale constructed with null"), nullptr)) {
546 __locale_->acquire();
549 locale::locale(const locale& other, const string& name, category c) : __locale_(new __imp(*other.__locale_, name, c)) {
550 __locale_->acquire();
553 locale::locale(const locale& other, const locale& one, category c)
554 : __locale_(new __imp(*other.__locale_, *one.__locale_, c)) {
555 __locale_->acquire();
558 string locale::name() const { return __locale_->name(); }
560 void locale::__install_ctor(const locale& other, facet* f, long facet_id) {
561 if (f)
562 __locale_ = new __imp(*other.__locale_, f, facet_id);
563 else
564 __locale_ = other.__locale_;
565 __locale_->acquire();
568 locale locale::global(const locale& loc) {
569 locale& g = __global();
570 locale r = g;
571 g = loc;
572 if (g.name() != "*")
573 setlocale(LC_ALL, g.name().c_str());
574 return r;
577 bool locale::has_facet(id& x) const { return __locale_->has_facet(x.__get()); }
579 const locale::facet* locale::use_facet(id& x) const { return __locale_->use_facet(x.__get()); }
581 bool locale::operator==(const locale& y) const {
582 return (__locale_ == y.__locale_) || (__locale_->name() != "*" && __locale_->name() == y.__locale_->name());
585 // locale::facet
587 locale::facet::~facet() {}
589 void locale::facet::__on_zero_shared() noexcept { delete this; }
591 // locale::id
593 constinit int32_t locale::id::__next_id = 0;
595 long locale::id::__get() {
596 call_once(__flag_, [&] { __id_ = __libcpp_atomic_add(&__next_id, 1); });
597 return __id_ - 1;
600 // template <> class collate_byname<char>
602 collate_byname<char>::collate_byname(const char* n, size_t refs)
603 : collate<char>(refs), __l_(__locale::__newlocale(LC_ALL_MASK, n, 0)) {
604 if (__l_ == 0)
605 __throw_runtime_error(
606 ("collate_byname<char>::collate_byname"
607 " failed to construct for " +
608 string(n))
609 .c_str());
612 collate_byname<char>::collate_byname(const string& name, size_t refs)
613 : collate<char>(refs), __l_(__locale::__newlocale(LC_ALL_MASK, name.c_str(), 0)) {
614 if (__l_ == 0)
615 __throw_runtime_error(
616 ("collate_byname<char>::collate_byname"
617 " failed to construct for " +
618 name)
619 .c_str());
622 collate_byname<char>::~collate_byname() { __locale::__freelocale(__l_); }
624 int collate_byname<char>::do_compare(
625 const char_type* __lo1, const char_type* __hi1, const char_type* __lo2, const char_type* __hi2) const {
626 string_type lhs(__lo1, __hi1);
627 string_type rhs(__lo2, __hi2);
628 int r = __locale::__strcoll(lhs.c_str(), rhs.c_str(), __l_);
629 if (r < 0)
630 return -1;
631 if (r > 0)
632 return 1;
633 return r;
636 collate_byname<char>::string_type collate_byname<char>::do_transform(const char_type* lo, const char_type* hi) const {
637 const string_type in(lo, hi);
638 string_type out(__locale::__strxfrm(0, in.c_str(), 0, __l_), char());
639 __locale::__strxfrm(const_cast<char*>(out.c_str()), in.c_str(), out.size() + 1, __l_);
640 return out;
643 // template <> class collate_byname<wchar_t>
645 #if _LIBCPP_HAS_WIDE_CHARACTERS
646 collate_byname<wchar_t>::collate_byname(const char* n, size_t refs)
647 : collate<wchar_t>(refs), __l_(__locale::__newlocale(LC_ALL_MASK, n, 0)) {
648 if (__l_ == 0)
649 __throw_runtime_error(
650 ("collate_byname<wchar_t>::collate_byname(size_t refs)"
651 " failed to construct for " +
652 string(n))
653 .c_str());
656 collate_byname<wchar_t>::collate_byname(const string& name, size_t refs)
657 : collate<wchar_t>(refs), __l_(__locale::__newlocale(LC_ALL_MASK, name.c_str(), 0)) {
658 if (__l_ == 0)
659 __throw_runtime_error(
660 ("collate_byname<wchar_t>::collate_byname(size_t refs)"
661 " failed to construct for " +
662 name)
663 .c_str());
666 collate_byname<wchar_t>::~collate_byname() { __locale::__freelocale(__l_); }
668 int collate_byname<wchar_t>::do_compare(
669 const char_type* __lo1, const char_type* __hi1, const char_type* __lo2, const char_type* __hi2) const {
670 string_type lhs(__lo1, __hi1);
671 string_type rhs(__lo2, __hi2);
672 int r = __locale::__wcscoll(lhs.c_str(), rhs.c_str(), __l_);
673 if (r < 0)
674 return -1;
675 if (r > 0)
676 return 1;
677 return r;
680 collate_byname<wchar_t>::string_type
681 collate_byname<wchar_t>::do_transform(const char_type* lo, const char_type* hi) const {
682 const string_type in(lo, hi);
683 string_type out(__locale::__wcsxfrm(0, in.c_str(), 0, __l_), wchar_t());
684 __locale::__wcsxfrm(const_cast<wchar_t*>(out.c_str()), in.c_str(), out.size() + 1, __l_);
685 return out;
687 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
689 const ctype_base::mask ctype_base::space;
690 const ctype_base::mask ctype_base::print;
691 const ctype_base::mask ctype_base::cntrl;
692 const ctype_base::mask ctype_base::upper;
693 const ctype_base::mask ctype_base::lower;
694 const ctype_base::mask ctype_base::alpha;
695 const ctype_base::mask ctype_base::digit;
696 const ctype_base::mask ctype_base::punct;
697 const ctype_base::mask ctype_base::xdigit;
698 const ctype_base::mask ctype_base::blank;
699 const ctype_base::mask ctype_base::alnum;
700 const ctype_base::mask ctype_base::graph;
702 // template <> class ctype<wchar_t>;
704 #if _LIBCPP_HAS_WIDE_CHARACTERS
705 constinit locale::id ctype<wchar_t>::id;
707 ctype<wchar_t>::~ctype() {}
709 bool ctype<wchar_t>::do_is(mask m, char_type c) const {
710 return isascii(c) ? (ctype<char>::classic_table()[c] & m) != 0 : false;
713 const wchar_t* ctype<wchar_t>::do_is(const char_type* low, const char_type* high, mask* vec) const {
714 for (; low != high; ++low, ++vec)
715 *vec = static_cast<mask>(isascii(*low) ? ctype<char>::classic_table()[*low] : 0);
716 return low;
719 const wchar_t* ctype<wchar_t>::do_scan_is(mask m, const char_type* low, const char_type* high) const {
720 for (; low != high; ++low)
721 if (isascii(*low) && (ctype<char>::classic_table()[*low] & m))
722 break;
723 return low;
726 const wchar_t* ctype<wchar_t>::do_scan_not(mask m, const char_type* low, const char_type* high) const {
727 for (; low != high; ++low)
728 if (!(isascii(*low) && (ctype<char>::classic_table()[*low] & m)))
729 break;
730 return low;
733 wchar_t ctype<wchar_t>::do_toupper(char_type c) const {
734 # ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
735 return isascii(c) ? _DefaultRuneLocale.__mapupper[c] : c;
736 # elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__NetBSD__) || defined(__MVS__)
737 return isascii(c) ? ctype<char>::__classic_upper_table()[c] : c;
738 # else
739 return (isascii(c) && __locale::__iswlower(c, _LIBCPP_GET_C_LOCALE)) ? c - L'a' + L'A' : c;
740 # endif
743 const wchar_t* ctype<wchar_t>::do_toupper(char_type* low, const char_type* high) const {
744 for (; low != high; ++low)
745 # ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
746 *low = isascii(*low) ? _DefaultRuneLocale.__mapupper[*low] : *low;
747 # elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__NetBSD__) || defined(__MVS__)
748 *low = isascii(*low) ? ctype<char>::__classic_upper_table()[*low] : *low;
749 # else
750 *low = (isascii(*low) && __locale::__islower(*low, _LIBCPP_GET_C_LOCALE)) ? (*low - L'a' + L'A') : *low;
751 # endif
752 return low;
755 wchar_t ctype<wchar_t>::do_tolower(char_type c) const {
756 # ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
757 return isascii(c) ? _DefaultRuneLocale.__maplower[c] : c;
758 # elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__NetBSD__) || defined(__MVS__)
759 return isascii(c) ? ctype<char>::__classic_lower_table()[c] : c;
760 # else
761 return (isascii(c) && __locale::__isupper(c, _LIBCPP_GET_C_LOCALE)) ? c - L'A' + 'a' : c;
762 # endif
765 const wchar_t* ctype<wchar_t>::do_tolower(char_type* low, const char_type* high) const {
766 for (; low != high; ++low)
767 # ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
768 *low = isascii(*low) ? _DefaultRuneLocale.__maplower[*low] : *low;
769 # elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__NetBSD__) || defined(__MVS__)
770 *low = isascii(*low) ? ctype<char>::__classic_lower_table()[*low] : *low;
771 # else
772 *low = (isascii(*low) && __locale::__isupper(*low, _LIBCPP_GET_C_LOCALE)) ? *low - L'A' + L'a' : *low;
773 # endif
774 return low;
777 wchar_t ctype<wchar_t>::do_widen(char c) const { return c; }
779 const char* ctype<wchar_t>::do_widen(const char* low, const char* high, char_type* dest) const {
780 for (; low != high; ++low, ++dest)
781 *dest = *low;
782 return low;
785 char ctype<wchar_t>::do_narrow(char_type c, char dfault) const {
786 if (isascii(c))
787 return static_cast<char>(c);
788 return dfault;
791 const wchar_t* ctype<wchar_t>::do_narrow(const char_type* low, const char_type* high, char dfault, char* dest) const {
792 for (; low != high; ++low, ++dest)
793 if (isascii(*low))
794 *dest = static_cast<char>(*low);
795 else
796 *dest = dfault;
797 return low;
799 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
801 // template <> class ctype<char>;
803 constinit locale::id ctype<char>::id;
805 const size_t ctype<char>::table_size;
807 ctype<char>::ctype(const mask* tab, bool del, size_t refs) : locale::facet(refs), __tab_(tab), __del_(del) {
808 if (__tab_ == 0)
809 __tab_ = classic_table();
812 ctype<char>::~ctype() {
813 if (__tab_ && __del_)
814 delete[] __tab_;
817 char ctype<char>::do_toupper(char_type c) const {
818 #ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
819 return isascii(c) ? static_cast<char>(_DefaultRuneLocale.__mapupper[static_cast<ptrdiff_t>(c)]) : c;
820 #elif defined(__NetBSD__)
821 return static_cast<char>(__classic_upper_table()[static_cast<unsigned char>(c)]);
822 #elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__MVS__)
823 return isascii(c) ? static_cast<char>(__classic_upper_table()[static_cast<unsigned char>(c)]) : c;
824 #else
825 return (isascii(c) && __locale::__islower(c, _LIBCPP_GET_C_LOCALE)) ? c - 'a' + 'A' : c;
826 #endif
829 const char* ctype<char>::do_toupper(char_type* low, const char_type* high) const {
830 for (; low != high; ++low)
831 #ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
832 *low = isascii(*low) ? static_cast<char>(_DefaultRuneLocale.__mapupper[static_cast<ptrdiff_t>(*low)]) : *low;
833 #elif defined(__NetBSD__)
834 *low = static_cast<char>(__classic_upper_table()[static_cast<unsigned char>(*low)]);
835 #elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__MVS__)
836 *low = isascii(*low) ? static_cast<char>(__classic_upper_table()[static_cast<size_t>(*low)]) : *low;
837 #else
838 *low = (isascii(*low) && __locale::__islower(*low, _LIBCPP_GET_C_LOCALE)) ? *low - 'a' + 'A' : *low;
839 #endif
840 return low;
843 char ctype<char>::do_tolower(char_type c) const {
844 #ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
845 return isascii(c) ? static_cast<char>(_DefaultRuneLocale.__maplower[static_cast<ptrdiff_t>(c)]) : c;
846 #elif defined(__NetBSD__)
847 return static_cast<char>(__classic_lower_table()[static_cast<unsigned char>(c)]);
848 #elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__MVS__)
849 return isascii(c) ? static_cast<char>(__classic_lower_table()[static_cast<size_t>(c)]) : c;
850 #else
851 return (isascii(c) && __locale::__isupper(c, _LIBCPP_GET_C_LOCALE)) ? c - 'A' + 'a' : c;
852 #endif
855 const char* ctype<char>::do_tolower(char_type* low, const char_type* high) const {
856 for (; low != high; ++low)
857 #ifdef _LIBCPP_HAS_DEFAULTRUNELOCALE
858 *low = isascii(*low) ? static_cast<char>(_DefaultRuneLocale.__maplower[static_cast<ptrdiff_t>(*low)]) : *low;
859 #elif defined(__NetBSD__)
860 *low = static_cast<char>(__classic_lower_table()[static_cast<unsigned char>(*low)]);
861 #elif defined(__GLIBC__) || defined(__EMSCRIPTEN__) || defined(__MVS__)
862 *low = isascii(*low) ? static_cast<char>(__classic_lower_table()[static_cast<size_t>(*low)]) : *low;
863 #else
864 *low = (isascii(*low) && __locale::__isupper(*low, _LIBCPP_GET_C_LOCALE)) ? *low - 'A' + 'a' : *low;
865 #endif
866 return low;
869 char ctype<char>::do_widen(char c) const { return c; }
871 const char* ctype<char>::do_widen(const char* low, const char* high, char_type* dest) const {
872 for (; low != high; ++low, ++dest)
873 *dest = *low;
874 return low;
877 char ctype<char>::do_narrow(char_type c, char dfault) const {
878 if (isascii(c))
879 return static_cast<char>(c);
880 return dfault;
883 const char* ctype<char>::do_narrow(const char_type* low, const char_type* high, char dfault, char* dest) const {
884 for (; low != high; ++low, ++dest)
885 if (isascii(*low))
886 *dest = *low;
887 else
888 *dest = dfault;
889 return low;
892 #if defined(__EMSCRIPTEN__)
893 extern "C" const unsigned short** __ctype_b_loc();
894 extern "C" const int** __ctype_tolower_loc();
895 extern "C" const int** __ctype_toupper_loc();
896 #endif
898 #ifdef _LIBCPP_PROVIDES_DEFAULT_RUNE_TABLE
899 const ctype<char>::mask* ctype<char>::classic_table() noexcept {
900 // clang-format off
901 static constexpr const ctype<char>::mask builtin_table[table_size] = {
902 cntrl, cntrl,
903 cntrl, cntrl,
904 cntrl, cntrl,
905 cntrl, cntrl,
906 cntrl, cntrl | space | blank,
907 cntrl | space, cntrl | space,
908 cntrl | space, cntrl | space,
909 cntrl, cntrl,
910 cntrl, cntrl,
911 cntrl, cntrl,
912 cntrl, cntrl,
913 cntrl, cntrl,
914 cntrl, cntrl,
915 cntrl, cntrl,
916 cntrl, cntrl,
917 cntrl, cntrl,
918 space | blank | print, punct | print,
919 punct | print, punct | print,
920 punct | print, punct | print,
921 punct | print, punct | print,
922 punct | print, punct | print,
923 punct | print, punct | print,
924 punct | print, punct | print,
925 punct | print, punct | print,
926 digit | print | xdigit, digit | print | xdigit,
927 digit | print | xdigit, digit | print | xdigit,
928 digit | print | xdigit, digit | print | xdigit,
929 digit | print | xdigit, digit | print | xdigit,
930 digit | print | xdigit, digit | print | xdigit,
931 punct | print, punct | print,
932 punct | print, punct | print,
933 punct | print, punct | print,
934 punct | print, upper | xdigit | print | alpha,
935 upper | xdigit | print | alpha, upper | xdigit | print | alpha,
936 upper | xdigit | print | alpha, upper | xdigit | print | alpha,
937 upper | xdigit | print | alpha, upper | print | alpha,
938 upper | print | alpha, upper | print | alpha,
939 upper | print | alpha, upper | print | alpha,
940 upper | print | alpha, upper | print | alpha,
941 upper | print | alpha, upper | print | alpha,
942 upper | print | alpha, upper | print | alpha,
943 upper | print | alpha, upper | print | alpha,
944 upper | print | alpha, upper | print | alpha,
945 upper | print | alpha, upper | print | alpha,
946 upper | print | alpha, upper | print | alpha,
947 upper | print | alpha, punct | print,
948 punct | print, punct | print,
949 punct | print, punct | print,
950 punct | print, lower | xdigit | print | alpha,
951 lower | xdigit | print | alpha, lower | xdigit | print | alpha,
952 lower | xdigit | print | alpha, lower | xdigit | print | alpha,
953 lower | xdigit | print | alpha, lower | print | alpha,
954 lower | print | alpha, lower | print | alpha,
955 lower | print | alpha, lower | print | alpha,
956 lower | print | alpha, lower | print | alpha,
957 lower | print | alpha, lower | print | alpha,
958 lower | print | alpha, lower | print | alpha,
959 lower | print | alpha, lower | print | alpha,
960 lower | print | alpha, lower | print | alpha,
961 lower | print | alpha, lower | print | alpha,
962 lower | print | alpha, lower | print | alpha,
963 lower | print | alpha, punct | print,
964 punct | print, punct | print,
965 punct | print, cntrl,
966 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
967 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
968 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
969 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
970 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
971 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
972 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
973 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
975 // clang-format on
976 return builtin_table;
978 #else
979 const ctype<char>::mask* ctype<char>::classic_table() noexcept {
980 # if defined(__APPLE__) || defined(__FreeBSD__)
981 return _DefaultRuneLocale.__runetype;
982 # elif defined(__NetBSD__)
983 return _C_ctype_tab_ + 1;
984 # elif defined(__GLIBC__)
985 return _LIBCPP_GET_C_LOCALE->__ctype_b;
986 # elif defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
987 return __pctype_func();
988 # elif defined(__EMSCRIPTEN__)
989 return *__ctype_b_loc();
990 # elif defined(_NEWLIB_VERSION)
991 // Newlib has a 257-entry table in ctype_.c, where (char)0 starts at [1].
992 return _ctype_ + 1;
993 # elif defined(_AIX)
994 return (const unsigned int*)__lc_ctype_ptr->obj->mask;
995 # elif defined(__MVS__)
996 # if defined(__NATIVE_ASCII_F)
997 return const_cast<const ctype<char>::mask*>(__OBJ_DATA(__lc_ctype_a)->mask);
998 # else
999 return const_cast<const ctype<char>::mask*>(__ctypec);
1000 # endif
1001 # else
1002 // Platform not supported: abort so the person doing the port knows what to
1003 // fix
1004 # warning ctype<char>::classic_table() is not implemented
1005 printf("ctype<char>::classic_table() is not implemented\n");
1006 abort();
1007 return nullptr;
1008 # endif
1010 #endif
1012 #if defined(__GLIBC__)
1013 const int* ctype<char>::__classic_lower_table() noexcept { return _LIBCPP_GET_C_LOCALE->__ctype_tolower; }
1015 const int* ctype<char>::__classic_upper_table() noexcept { return _LIBCPP_GET_C_LOCALE->__ctype_toupper; }
1016 #elif defined(__NetBSD__)
1017 const short* ctype<char>::__classic_lower_table() noexcept { return _C_tolower_tab_ + 1; }
1019 const short* ctype<char>::__classic_upper_table() noexcept { return _C_toupper_tab_ + 1; }
1021 #elif defined(__EMSCRIPTEN__)
1022 const int* ctype<char>::__classic_lower_table() noexcept { return *__ctype_tolower_loc(); }
1024 const int* ctype<char>::__classic_upper_table() noexcept { return *__ctype_toupper_loc(); }
1025 #elif defined(__MVS__)
1026 const unsigned short* ctype<char>::__classic_lower_table() _NOEXCEPT {
1027 # if defined(__NATIVE_ASCII_F)
1028 return const_cast<const unsigned short*>(__OBJ_DATA(__lc_ctype_a)->lower);
1029 # else
1030 return const_cast<const unsigned short*>(__ctype + __TOLOWER_INDEX);
1031 # endif
1033 const unsigned short* ctype<char>::__classic_upper_table() _NOEXCEPT {
1034 # if defined(__NATIVE_ASCII_F)
1035 return const_cast<const unsigned short*>(__OBJ_DATA(__lc_ctype_a)->upper);
1036 # else
1037 return const_cast<const unsigned short*>(__ctype + __TOUPPER_INDEX);
1038 # endif
1040 #endif // __GLIBC__ || __NETBSD__ || __EMSCRIPTEN__ || __MVS__
1042 // template <> class ctype_byname<char>
1044 ctype_byname<char>::ctype_byname(const char* name, size_t refs)
1045 : ctype<char>(0, false, refs), __l_(__locale::__newlocale(LC_ALL_MASK, name, 0)) {
1046 if (__l_ == 0)
1047 __throw_runtime_error(
1048 ("ctype_byname<char>::ctype_byname"
1049 " failed to construct for " +
1050 string(name))
1051 .c_str());
1054 ctype_byname<char>::ctype_byname(const string& name, size_t refs)
1055 : ctype<char>(0, false, refs), __l_(__locale::__newlocale(LC_ALL_MASK, name.c_str(), 0)) {
1056 if (__l_ == 0)
1057 __throw_runtime_error(
1058 ("ctype_byname<char>::ctype_byname"
1059 " failed to construct for " +
1060 name)
1061 .c_str());
1064 ctype_byname<char>::~ctype_byname() { __locale::__freelocale(__l_); }
1066 char ctype_byname<char>::do_toupper(char_type c) const {
1067 return static_cast<char>(__locale::__toupper(static_cast<unsigned char>(c), __l_));
1070 const char* ctype_byname<char>::do_toupper(char_type* low, const char_type* high) const {
1071 for (; low != high; ++low)
1072 *low = static_cast<char>(__locale::__toupper(static_cast<unsigned char>(*low), __l_));
1073 return low;
1076 char ctype_byname<char>::do_tolower(char_type c) const {
1077 return static_cast<char>(__locale::__tolower(static_cast<unsigned char>(c), __l_));
1080 const char* ctype_byname<char>::do_tolower(char_type* low, const char_type* high) const {
1081 for (; low != high; ++low)
1082 *low = static_cast<char>(__locale::__tolower(static_cast<unsigned char>(*low), __l_));
1083 return low;
1086 // template <> class ctype_byname<wchar_t>
1088 #if _LIBCPP_HAS_WIDE_CHARACTERS
1089 ctype_byname<wchar_t>::ctype_byname(const char* name, size_t refs)
1090 : ctype<wchar_t>(refs), __l_(__locale::__newlocale(LC_ALL_MASK, name, 0)) {
1091 if (__l_ == 0)
1092 __throw_runtime_error(
1093 ("ctype_byname<wchar_t>::ctype_byname"
1094 " failed to construct for " +
1095 string(name))
1096 .c_str());
1099 ctype_byname<wchar_t>::ctype_byname(const string& name, size_t refs)
1100 : ctype<wchar_t>(refs), __l_(__locale::__newlocale(LC_ALL_MASK, name.c_str(), 0)) {
1101 if (__l_ == 0)
1102 __throw_runtime_error(
1103 ("ctype_byname<wchar_t>::ctype_byname"
1104 " failed to construct for " +
1105 name)
1106 .c_str());
1109 ctype_byname<wchar_t>::~ctype_byname() { __locale::__freelocale(__l_); }
1111 bool ctype_byname<wchar_t>::do_is(mask m, char_type c) const {
1112 # ifdef _LIBCPP_WCTYPE_IS_MASK
1113 return static_cast<bool>(iswctype_l(c, m, __l_));
1114 # else
1115 bool result = false;
1116 wint_t ch = static_cast<wint_t>(c);
1117 if ((m & space) == space)
1118 result |= (__locale::__iswspace(ch, __l_) != 0);
1119 if ((m & print) == print)
1120 result |= (__locale::__iswprint(ch, __l_) != 0);
1121 if ((m & cntrl) == cntrl)
1122 result |= (__locale::__iswcntrl(ch, __l_) != 0);
1123 if ((m & upper) == upper)
1124 result |= (__locale::__iswupper(ch, __l_) != 0);
1125 if ((m & lower) == lower)
1126 result |= (__locale::__iswlower(ch, __l_) != 0);
1127 if ((m & alpha) == alpha)
1128 result |= (__locale::__iswalpha(ch, __l_) != 0);
1129 if ((m & digit) == digit)
1130 result |= (__locale::__iswdigit(ch, __l_) != 0);
1131 if ((m & punct) == punct)
1132 result |= (__locale::__iswpunct(ch, __l_) != 0);
1133 if ((m & xdigit) == xdigit)
1134 result |= (__locale::__iswxdigit(ch, __l_) != 0);
1135 if ((m & blank) == blank)
1136 result |= (__locale::__iswblank(ch, __l_) != 0);
1137 return result;
1138 # endif
1141 const wchar_t* ctype_byname<wchar_t>::do_is(const char_type* low, const char_type* high, mask* vec) const {
1142 for (; low != high; ++low, ++vec) {
1143 if (isascii(*low))
1144 *vec = static_cast<mask>(ctype<char>::classic_table()[*low]);
1145 else {
1146 *vec = 0;
1147 wint_t ch = static_cast<wint_t>(*low);
1148 if (__locale::__iswspace(ch, __l_))
1149 *vec |= space;
1150 # ifndef _LIBCPP_CTYPE_MASK_IS_COMPOSITE_PRINT
1151 if (__locale::__iswprint(ch, __l_))
1152 *vec |= print;
1153 # endif
1154 if (__locale::__iswcntrl(ch, __l_))
1155 *vec |= cntrl;
1156 if (__locale::__iswupper(ch, __l_))
1157 *vec |= upper;
1158 if (__locale::__iswlower(ch, __l_))
1159 *vec |= lower;
1160 # ifndef _LIBCPP_CTYPE_MASK_IS_COMPOSITE_ALPHA
1161 if (__locale::__iswalpha(ch, __l_))
1162 *vec |= alpha;
1163 # endif
1164 if (__locale::__iswdigit(ch, __l_))
1165 *vec |= digit;
1166 if (__locale::__iswpunct(ch, __l_))
1167 *vec |= punct;
1168 # ifndef _LIBCPP_CTYPE_MASK_IS_COMPOSITE_XDIGIT
1169 if (__locale::__iswxdigit(ch, __l_))
1170 *vec |= xdigit;
1171 # endif
1172 if (__locale::__iswblank(ch, __l_))
1173 *vec |= blank;
1176 return low;
1179 const wchar_t* ctype_byname<wchar_t>::do_scan_is(mask m, const char_type* low, const char_type* high) const {
1180 for (; low != high; ++low) {
1181 # ifdef _LIBCPP_WCTYPE_IS_MASK
1182 if (iswctype_l(*low, m, __l_))
1183 break;
1184 # else
1185 wint_t ch = static_cast<wint_t>(*low);
1186 if ((m & space) == space && __locale::__iswspace(ch, __l_))
1187 break;
1188 if ((m & print) == print && __locale::__iswprint(ch, __l_))
1189 break;
1190 if ((m & cntrl) == cntrl && __locale::__iswcntrl(ch, __l_))
1191 break;
1192 if ((m & upper) == upper && __locale::__iswupper(ch, __l_))
1193 break;
1194 if ((m & lower) == lower && __locale::__iswlower(ch, __l_))
1195 break;
1196 if ((m & alpha) == alpha && __locale::__iswalpha(ch, __l_))
1197 break;
1198 if ((m & digit) == digit && __locale::__iswdigit(ch, __l_))
1199 break;
1200 if ((m & punct) == punct && __locale::__iswpunct(ch, __l_))
1201 break;
1202 if ((m & xdigit) == xdigit && __locale::__iswxdigit(ch, __l_))
1203 break;
1204 if ((m & blank) == blank && __locale::__iswblank(ch, __l_))
1205 break;
1206 # endif
1208 return low;
1211 const wchar_t* ctype_byname<wchar_t>::do_scan_not(mask m, const char_type* low, const char_type* high) const {
1212 for (; low != high; ++low) {
1213 # ifdef _LIBCPP_WCTYPE_IS_MASK
1214 if (!iswctype_l(*low, m, __l_))
1215 break;
1216 # else
1217 wint_t ch = static_cast<wint_t>(*low);
1218 if ((m & space) == space && __locale::__iswspace(ch, __l_))
1219 continue;
1220 if ((m & print) == print && __locale::__iswprint(ch, __l_))
1221 continue;
1222 if ((m & cntrl) == cntrl && __locale::__iswcntrl(ch, __l_))
1223 continue;
1224 if ((m & upper) == upper && __locale::__iswupper(ch, __l_))
1225 continue;
1226 if ((m & lower) == lower && __locale::__iswlower(ch, __l_))
1227 continue;
1228 if ((m & alpha) == alpha && __locale::__iswalpha(ch, __l_))
1229 continue;
1230 if ((m & digit) == digit && __locale::__iswdigit(ch, __l_))
1231 continue;
1232 if ((m & punct) == punct && __locale::__iswpunct(ch, __l_))
1233 continue;
1234 if ((m & xdigit) == xdigit && __locale::__iswxdigit(ch, __l_))
1235 continue;
1236 if ((m & blank) == blank && __locale::__iswblank(ch, __l_))
1237 continue;
1238 break;
1239 # endif
1241 return low;
1244 wchar_t ctype_byname<wchar_t>::do_toupper(char_type c) const { return __locale::__towupper(c, __l_); }
1246 const wchar_t* ctype_byname<wchar_t>::do_toupper(char_type* low, const char_type* high) const {
1247 for (; low != high; ++low)
1248 *low = __locale::__towupper(*low, __l_);
1249 return low;
1252 wchar_t ctype_byname<wchar_t>::do_tolower(char_type c) const { return __locale::__towlower(c, __l_); }
1254 const wchar_t* ctype_byname<wchar_t>::do_tolower(char_type* low, const char_type* high) const {
1255 for (; low != high; ++low)
1256 *low = __locale::__towlower(*low, __l_);
1257 return low;
1260 wchar_t ctype_byname<wchar_t>::do_widen(char c) const { return __locale::__btowc(c, __l_); }
1262 const char* ctype_byname<wchar_t>::do_widen(const char* low, const char* high, char_type* dest) const {
1263 for (; low != high; ++low, ++dest)
1264 *dest = __locale::__btowc(*low, __l_);
1265 return low;
1268 char ctype_byname<wchar_t>::do_narrow(char_type c, char dfault) const {
1269 int r = __locale::__wctob(c, __l_);
1270 return (r != EOF) ? static_cast<char>(r) : dfault;
1273 const wchar_t*
1274 ctype_byname<wchar_t>::do_narrow(const char_type* low, const char_type* high, char dfault, char* dest) const {
1275 for (; low != high; ++low, ++dest) {
1276 int r = __locale::__wctob(*low, __l_);
1277 *dest = (r != EOF) ? static_cast<char>(r) : dfault;
1279 return low;
1281 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
1283 // template <> class codecvt<char, char, mbstate_t>
1285 constinit locale::id codecvt<char, char, mbstate_t>::id;
1287 codecvt<char, char, mbstate_t>::~codecvt() {}
1289 codecvt<char, char, mbstate_t>::result codecvt<char, char, mbstate_t>::do_out(
1290 state_type&,
1291 const intern_type* frm,
1292 const intern_type*,
1293 const intern_type*& frm_nxt,
1294 extern_type* to,
1295 extern_type*,
1296 extern_type*& to_nxt) const {
1297 frm_nxt = frm;
1298 to_nxt = to;
1299 return noconv;
1302 codecvt<char, char, mbstate_t>::result codecvt<char, char, mbstate_t>::do_in(
1303 state_type&,
1304 const extern_type* frm,
1305 const extern_type*,
1306 const extern_type*& frm_nxt,
1307 intern_type* to,
1308 intern_type*,
1309 intern_type*& to_nxt) const {
1310 frm_nxt = frm;
1311 to_nxt = to;
1312 return noconv;
1315 codecvt<char, char, mbstate_t>::result
1316 codecvt<char, char, mbstate_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
1317 to_nxt = to;
1318 return noconv;
1321 int codecvt<char, char, mbstate_t>::do_encoding() const noexcept { return 1; }
1323 bool codecvt<char, char, mbstate_t>::do_always_noconv() const noexcept { return true; }
1325 int codecvt<char, char, mbstate_t>::do_length(
1326 state_type&, const extern_type* frm, const extern_type* end, size_t mx) const {
1327 return static_cast<int>(min<size_t>(mx, static_cast<size_t>(end - frm)));
1330 int codecvt<char, char, mbstate_t>::do_max_length() const noexcept { return 1; }
1332 // template <> class codecvt<wchar_t, char, mbstate_t>
1334 #if _LIBCPP_HAS_WIDE_CHARACTERS
1335 constinit locale::id codecvt<wchar_t, char, mbstate_t>::id;
1337 codecvt<wchar_t, char, mbstate_t>::codecvt(size_t refs) : locale::facet(refs), __l_(_LIBCPP_GET_C_LOCALE) {}
1339 codecvt<wchar_t, char, mbstate_t>::codecvt(const char* nm, size_t refs)
1340 : locale::facet(refs), __l_(__locale::__newlocale(LC_ALL_MASK, nm, 0)) {
1341 if (__l_ == 0)
1342 __throw_runtime_error(
1343 ("codecvt_byname<wchar_t, char, mbstate_t>::codecvt_byname"
1344 " failed to construct for " +
1345 string(nm))
1346 .c_str());
1349 codecvt<wchar_t, char, mbstate_t>::~codecvt() {
1350 if (__l_ != _LIBCPP_GET_C_LOCALE)
1351 __locale::__freelocale(__l_);
1354 codecvt<wchar_t, char, mbstate_t>::result codecvt<wchar_t, char, mbstate_t>::do_out(
1355 state_type& st,
1356 const intern_type* frm,
1357 const intern_type* frm_end,
1358 const intern_type*& frm_nxt,
1359 extern_type* to,
1360 extern_type* to_end,
1361 extern_type*& to_nxt) const {
1362 // look for first internal null in frm
1363 const intern_type* fend = frm;
1364 for (; fend != frm_end; ++fend)
1365 if (*fend == 0)
1366 break;
1367 // loop over all null-terminated sequences in frm
1368 to_nxt = to;
1369 for (frm_nxt = frm; frm != frm_end && to != to_end; frm = frm_nxt, to = to_nxt) {
1370 // save state in case it is needed to recover to_nxt on error
1371 mbstate_t save_state = st;
1372 size_t n = __locale::__wcsnrtombs(
1373 to, &frm_nxt, static_cast<size_t>(fend - frm), static_cast<size_t>(to_end - to), &st, __l_);
1374 if (n == size_t(-1)) {
1375 // need to recover to_nxt
1376 for (to_nxt = to; frm != frm_nxt; ++frm) {
1377 n = __locale::__wcrtomb(to_nxt, *frm, &save_state, __l_);
1378 if (n == size_t(-1))
1379 break;
1380 to_nxt += n;
1382 frm_nxt = frm;
1383 return error;
1385 if (n == 0)
1386 return partial;
1387 to_nxt += n;
1388 if (to_nxt == to_end)
1389 break;
1390 if (fend != frm_end) // set up next null terminated sequence
1392 // Try to write the terminating null
1393 extern_type tmp[MB_LEN_MAX];
1394 n = __locale::__wcrtomb(tmp, intern_type(), &st, __l_);
1395 if (n == size_t(-1)) // on error
1396 return error;
1397 if (n > static_cast<size_t>(to_end - to_nxt)) // is there room?
1398 return partial;
1399 for (extern_type* p = tmp; n; --n) // write it
1400 *to_nxt++ = *p++;
1401 ++frm_nxt;
1402 // look for next null in frm
1403 for (fend = frm_nxt; fend != frm_end; ++fend)
1404 if (*fend == 0)
1405 break;
1408 return frm_nxt == frm_end ? ok : partial;
1411 codecvt<wchar_t, char, mbstate_t>::result codecvt<wchar_t, char, mbstate_t>::do_in(
1412 state_type& st,
1413 const extern_type* frm,
1414 const extern_type* frm_end,
1415 const extern_type*& frm_nxt,
1416 intern_type* to,
1417 intern_type* to_end,
1418 intern_type*& to_nxt) const {
1419 // look for first internal null in frm
1420 const extern_type* fend = frm;
1421 for (; fend != frm_end; ++fend)
1422 if (*fend == 0)
1423 break;
1424 // loop over all null-terminated sequences in frm
1425 to_nxt = to;
1426 for (frm_nxt = frm; frm != frm_end && to != to_end; frm = frm_nxt, to = to_nxt) {
1427 // save state in case it is needed to recover to_nxt on error
1428 mbstate_t save_state = st;
1429 size_t n = __locale::__mbsnrtowcs(
1430 to, &frm_nxt, static_cast<size_t>(fend - frm), static_cast<size_t>(to_end - to), &st, __l_);
1431 if (n == size_t(-1)) {
1432 // need to recover to_nxt
1433 for (to_nxt = to; frm != frm_nxt; ++to_nxt) {
1434 n = __locale::__mbrtowc(to_nxt, frm, static_cast<size_t>(fend - frm), &save_state, __l_);
1435 switch (n) {
1436 case 0:
1437 ++frm;
1438 break;
1439 case size_t(-1):
1440 frm_nxt = frm;
1441 return error;
1442 case size_t(-2):
1443 frm_nxt = frm;
1444 return partial;
1445 default:
1446 frm += n;
1447 break;
1450 frm_nxt = frm;
1451 return frm_nxt == frm_end ? ok : partial;
1453 if (n == size_t(-1))
1454 return error;
1455 to_nxt += n;
1456 if (to_nxt == to_end)
1457 break;
1458 if (fend != frm_end) // set up next null terminated sequence
1460 // Try to write the terminating null
1461 n = __locale::__mbrtowc(to_nxt, frm_nxt, 1, &st, __l_);
1462 if (n != 0) // on error
1463 return error;
1464 ++to_nxt;
1465 ++frm_nxt;
1466 // look for next null in frm
1467 for (fend = frm_nxt; fend != frm_end; ++fend)
1468 if (*fend == 0)
1469 break;
1472 return frm_nxt == frm_end ? ok : partial;
1475 codecvt<wchar_t, char, mbstate_t>::result codecvt<wchar_t, char, mbstate_t>::do_unshift(
1476 state_type& st, extern_type* to, extern_type* to_end, extern_type*& to_nxt) const {
1477 to_nxt = to;
1478 extern_type tmp[MB_LEN_MAX];
1479 size_t n = __locale::__wcrtomb(tmp, intern_type(), &st, __l_);
1480 if (n == size_t(-1) || n == 0) // on error
1481 return error;
1482 --n;
1483 if (n > static_cast<size_t>(to_end - to_nxt)) // is there room?
1484 return partial;
1485 for (extern_type* p = tmp; n; --n) // write it
1486 *to_nxt++ = *p++;
1487 return ok;
1490 int codecvt<wchar_t, char, mbstate_t>::do_encoding() const noexcept {
1491 if (__locale::__mbtowc(nullptr, nullptr, MB_LEN_MAX, __l_) != 0)
1492 return -1;
1494 // stateless encoding
1495 if (__l_ == 0 || __locale::__mb_len_max(__l_) == 1) // there are no known constant length encodings
1496 return 1; // which take more than 1 char to form a wchar_t
1497 return 0;
1500 bool codecvt<wchar_t, char, mbstate_t>::do_always_noconv() const noexcept { return false; }
1502 int codecvt<wchar_t, char, mbstate_t>::do_length(
1503 state_type& st, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
1504 int nbytes = 0;
1505 for (size_t nwchar_t = 0; nwchar_t < mx && frm != frm_end; ++nwchar_t) {
1506 size_t n = __locale::__mbrlen(frm, static_cast<size_t>(frm_end - frm), &st, __l_);
1507 switch (n) {
1508 case 0:
1509 ++nbytes;
1510 ++frm;
1511 break;
1512 case size_t(-1):
1513 case size_t(-2):
1514 return nbytes;
1515 default:
1516 nbytes += n;
1517 frm += n;
1518 break;
1521 return nbytes;
1524 int codecvt<wchar_t, char, mbstate_t>::do_max_length() const noexcept {
1525 return __l_ == 0 ? 1 : static_cast<int>(__locale::__mb_len_max(__l_));
1527 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
1529 // Valid UTF ranges
1530 // UTF-32 UTF-16 UTF-8 # of code points
1531 // first second first second third fourth
1532 // 000000 - 00007F 0000 - 007F 00 - 7F 127
1533 // 000080 - 0007FF 0080 - 07FF C2 - DF, 80 - BF 1920
1534 // 000800 - 000FFF 0800 - 0FFF E0 - E0, A0 - BF, 80 - BF 2048
1535 // 001000 - 00CFFF 1000 - CFFF E1 - EC, 80 - BF, 80 - BF 49152
1536 // 00D000 - 00D7FF D000 - D7FF ED - ED, 80 - 9F, 80 - BF 2048
1537 // 00D800 - 00DFFF invalid
1538 // 00E000 - 00FFFF E000 - FFFF EE - EF, 80 - BF, 80 - BF 8192
1539 // 010000 - 03FFFF D800 - D8BF, DC00 - DFFF F0 - F0, 90 - BF, 80 - BF, 80 - BF 196608
1540 // 040000 - 0FFFFF D8C0 - DBBF, DC00 - DFFF F1 - F3, 80 - BF, 80 - BF, 80 - BF 786432
1541 // 100000 - 10FFFF DBC0 - DBFF, DC00 - DFFF F4 - F4, 80 - 8F, 80 - BF, 80 - BF 65536
1543 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
1544 static codecvt_base::result utf16_to_utf8(
1545 const uint16_t* frm,
1546 const uint16_t* frm_end,
1547 const uint16_t*& frm_nxt,
1548 uint8_t* to,
1549 uint8_t* to_end,
1550 uint8_t*& to_nxt,
1551 unsigned long Maxcode = 0x10FFFF,
1552 codecvt_mode mode = codecvt_mode(0)) {
1553 frm_nxt = frm;
1554 to_nxt = to;
1555 if (mode & generate_header) {
1556 if (to_end - to_nxt < 3)
1557 return codecvt_base::partial;
1558 *to_nxt++ = static_cast<uint8_t>(0xEF);
1559 *to_nxt++ = static_cast<uint8_t>(0xBB);
1560 *to_nxt++ = static_cast<uint8_t>(0xBF);
1562 for (; frm_nxt < frm_end; ++frm_nxt) {
1563 uint16_t wc1 = *frm_nxt;
1564 if (wc1 > Maxcode)
1565 return codecvt_base::error;
1566 if (wc1 < 0x0080) {
1567 if (to_end - to_nxt < 1)
1568 return codecvt_base::partial;
1569 *to_nxt++ = static_cast<uint8_t>(wc1);
1570 } else if (wc1 < 0x0800) {
1571 if (to_end - to_nxt < 2)
1572 return codecvt_base::partial;
1573 *to_nxt++ = static_cast<uint8_t>(0xC0 | (wc1 >> 6));
1574 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc1 & 0x03F));
1575 } else if (wc1 < 0xD800) {
1576 if (to_end - to_nxt < 3)
1577 return codecvt_base::partial;
1578 *to_nxt++ = static_cast<uint8_t>(0xE0 | (wc1 >> 12));
1579 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc1 & 0x0FC0) >> 6));
1580 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc1 & 0x003F));
1581 } else if (wc1 < 0xDC00) {
1582 if (frm_end - frm_nxt < 2)
1583 return codecvt_base::partial;
1584 uint16_t wc2 = frm_nxt[1];
1585 if ((wc2 & 0xFC00) != 0xDC00)
1586 return codecvt_base::error;
1587 if (to_end - to_nxt < 4)
1588 return codecvt_base::partial;
1589 if (((((wc1 & 0x03C0UL) >> 6) + 1) << 16) + ((wc1 & 0x003FUL) << 10) + (wc2 & 0x03FF) > Maxcode)
1590 return codecvt_base::error;
1591 ++frm_nxt;
1592 uint8_t z = ((wc1 & 0x03C0) >> 6) + 1;
1593 *to_nxt++ = static_cast<uint8_t>(0xF0 | (z >> 2));
1594 *to_nxt++ = static_cast<uint8_t>(0x80 | ((z & 0x03) << 4) | ((wc1 & 0x003C) >> 2));
1595 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc1 & 0x0003) << 4) | ((wc2 & 0x03C0) >> 6));
1596 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc2 & 0x003F));
1597 } else if (wc1 < 0xE000) {
1598 return codecvt_base::error;
1599 } else {
1600 if (to_end - to_nxt < 3)
1601 return codecvt_base::partial;
1602 *to_nxt++ = static_cast<uint8_t>(0xE0 | (wc1 >> 12));
1603 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc1 & 0x0FC0) >> 6));
1604 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc1 & 0x003F));
1607 return codecvt_base::ok;
1610 static codecvt_base::result utf16_to_utf8(
1611 const uint32_t* frm,
1612 const uint32_t* frm_end,
1613 const uint32_t*& frm_nxt,
1614 uint8_t* to,
1615 uint8_t* to_end,
1616 uint8_t*& to_nxt,
1617 unsigned long Maxcode = 0x10FFFF,
1618 codecvt_mode mode = codecvt_mode(0)) {
1619 frm_nxt = frm;
1620 to_nxt = to;
1621 if (mode & generate_header) {
1622 if (to_end - to_nxt < 3)
1623 return codecvt_base::partial;
1624 *to_nxt++ = static_cast<uint8_t>(0xEF);
1625 *to_nxt++ = static_cast<uint8_t>(0xBB);
1626 *to_nxt++ = static_cast<uint8_t>(0xBF);
1628 for (; frm_nxt < frm_end; ++frm_nxt) {
1629 uint16_t wc1 = static_cast<uint16_t>(*frm_nxt);
1630 if (wc1 > Maxcode)
1631 return codecvt_base::error;
1632 if (wc1 < 0x0080) {
1633 if (to_end - to_nxt < 1)
1634 return codecvt_base::partial;
1635 *to_nxt++ = static_cast<uint8_t>(wc1);
1636 } else if (wc1 < 0x0800) {
1637 if (to_end - to_nxt < 2)
1638 return codecvt_base::partial;
1639 *to_nxt++ = static_cast<uint8_t>(0xC0 | (wc1 >> 6));
1640 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc1 & 0x03F));
1641 } else if (wc1 < 0xD800) {
1642 if (to_end - to_nxt < 3)
1643 return codecvt_base::partial;
1644 *to_nxt++ = static_cast<uint8_t>(0xE0 | (wc1 >> 12));
1645 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc1 & 0x0FC0) >> 6));
1646 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc1 & 0x003F));
1647 } else if (wc1 < 0xDC00) {
1648 if (frm_end - frm_nxt < 2)
1649 return codecvt_base::partial;
1650 uint16_t wc2 = static_cast<uint16_t>(frm_nxt[1]);
1651 if ((wc2 & 0xFC00) != 0xDC00)
1652 return codecvt_base::error;
1653 if (to_end - to_nxt < 4)
1654 return codecvt_base::partial;
1655 if (((((wc1 & 0x03C0UL) >> 6) + 1) << 16) + ((wc1 & 0x003FUL) << 10) + (wc2 & 0x03FF) > Maxcode)
1656 return codecvt_base::error;
1657 ++frm_nxt;
1658 uint8_t z = ((wc1 & 0x03C0) >> 6) + 1;
1659 *to_nxt++ = static_cast<uint8_t>(0xF0 | (z >> 2));
1660 *to_nxt++ = static_cast<uint8_t>(0x80 | ((z & 0x03) << 4) | ((wc1 & 0x003C) >> 2));
1661 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc1 & 0x0003) << 4) | ((wc2 & 0x03C0) >> 6));
1662 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc2 & 0x003F));
1663 } else if (wc1 < 0xE000) {
1664 return codecvt_base::error;
1665 } else {
1666 if (to_end - to_nxt < 3)
1667 return codecvt_base::partial;
1668 *to_nxt++ = static_cast<uint8_t>(0xE0 | (wc1 >> 12));
1669 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc1 & 0x0FC0) >> 6));
1670 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc1 & 0x003F));
1673 return codecvt_base::ok;
1676 static codecvt_base::result utf8_to_utf16(
1677 const uint8_t* frm,
1678 const uint8_t* frm_end,
1679 const uint8_t*& frm_nxt,
1680 uint16_t* to,
1681 uint16_t* to_end,
1682 uint16_t*& to_nxt,
1683 unsigned long Maxcode = 0x10FFFF,
1684 codecvt_mode mode = codecvt_mode(0)) {
1685 frm_nxt = frm;
1686 to_nxt = to;
1687 if (mode & consume_header) {
1688 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
1689 frm_nxt += 3;
1691 for (; frm_nxt < frm_end && to_nxt < to_end; ++to_nxt) {
1692 uint8_t c1 = *frm_nxt;
1693 if (c1 > Maxcode)
1694 return codecvt_base::error;
1695 if (c1 < 0x80) {
1696 *to_nxt = static_cast<uint16_t>(c1);
1697 ++frm_nxt;
1698 } else if (c1 < 0xC2) {
1699 return codecvt_base::error;
1700 } else if (c1 < 0xE0) {
1701 if (frm_end - frm_nxt < 2)
1702 return codecvt_base::partial;
1703 uint8_t c2 = frm_nxt[1];
1704 if ((c2 & 0xC0) != 0x80)
1705 return codecvt_base::error;
1706 uint16_t t = static_cast<uint16_t>(((c1 & 0x1F) << 6) | (c2 & 0x3F));
1707 if (t > Maxcode)
1708 return codecvt_base::error;
1709 *to_nxt = t;
1710 frm_nxt += 2;
1711 } else if (c1 < 0xF0) {
1712 if (frm_end - frm_nxt < 2)
1713 return codecvt_base::partial;
1714 uint8_t c2 = frm_nxt[1];
1715 switch (c1) {
1716 case 0xE0:
1717 if ((c2 & 0xE0) != 0xA0)
1718 return codecvt_base::error;
1719 break;
1720 case 0xED:
1721 if ((c2 & 0xE0) != 0x80)
1722 return codecvt_base::error;
1723 break;
1724 default:
1725 if ((c2 & 0xC0) != 0x80)
1726 return codecvt_base::error;
1727 break;
1729 if (frm_end - frm_nxt < 3)
1730 return codecvt_base::partial;
1731 uint8_t c3 = frm_nxt[2];
1732 if ((c3 & 0xC0) != 0x80)
1733 return codecvt_base::error;
1734 uint16_t t = static_cast<uint16_t>(((c1 & 0x0F) << 12) | ((c2 & 0x3F) << 6) | (c3 & 0x3F));
1735 if (t > Maxcode)
1736 return codecvt_base::error;
1737 *to_nxt = t;
1738 frm_nxt += 3;
1739 } else if (c1 < 0xF5) {
1740 if (frm_end - frm_nxt < 2)
1741 return codecvt_base::partial;
1742 uint8_t c2 = frm_nxt[1];
1743 switch (c1) {
1744 case 0xF0:
1745 if (!(0x90 <= c2 && c2 <= 0xBF))
1746 return codecvt_base::error;
1747 break;
1748 case 0xF4:
1749 if ((c2 & 0xF0) != 0x80)
1750 return codecvt_base::error;
1751 break;
1752 default:
1753 if ((c2 & 0xC0) != 0x80)
1754 return codecvt_base::error;
1755 break;
1757 if (frm_end - frm_nxt < 3)
1758 return codecvt_base::partial;
1759 uint8_t c3 = frm_nxt[2];
1760 if ((c3 & 0xC0) != 0x80)
1761 return codecvt_base::error;
1762 if (frm_end - frm_nxt < 4)
1763 return codecvt_base::partial;
1764 uint8_t c4 = frm_nxt[3];
1765 if ((c4 & 0xC0) != 0x80)
1766 return codecvt_base::error;
1767 if (to_end - to_nxt < 2)
1768 return codecvt_base::partial;
1769 if ((((c1 & 7UL) << 18) + ((c2 & 0x3FUL) << 12) + ((c3 & 0x3FUL) << 6) + (c4 & 0x3F)) > Maxcode)
1770 return codecvt_base::error;
1771 *to_nxt = static_cast<uint16_t>(
1772 0xD800 | (((((c1 & 0x07) << 2) | ((c2 & 0x30) >> 4)) - 1) << 6) | ((c2 & 0x0F) << 2) | ((c3 & 0x30) >> 4));
1773 *++to_nxt = static_cast<uint16_t>(0xDC00 | ((c3 & 0x0F) << 6) | (c4 & 0x3F));
1774 frm_nxt += 4;
1775 } else {
1776 return codecvt_base::error;
1779 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
1782 static codecvt_base::result utf8_to_utf16(
1783 const uint8_t* frm,
1784 const uint8_t* frm_end,
1785 const uint8_t*& frm_nxt,
1786 uint32_t* to,
1787 uint32_t* to_end,
1788 uint32_t*& to_nxt,
1789 unsigned long Maxcode = 0x10FFFF,
1790 codecvt_mode mode = codecvt_mode(0)) {
1791 frm_nxt = frm;
1792 to_nxt = to;
1793 if (mode & consume_header) {
1794 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
1795 frm_nxt += 3;
1797 for (; frm_nxt < frm_end && to_nxt < to_end; ++to_nxt) {
1798 uint8_t c1 = *frm_nxt;
1799 if (c1 > Maxcode)
1800 return codecvt_base::error;
1801 if (c1 < 0x80) {
1802 *to_nxt = static_cast<uint32_t>(c1);
1803 ++frm_nxt;
1804 } else if (c1 < 0xC2) {
1805 return codecvt_base::error;
1806 } else if (c1 < 0xE0) {
1807 if (frm_end - frm_nxt < 2)
1808 return codecvt_base::partial;
1809 uint8_t c2 = frm_nxt[1];
1810 if ((c2 & 0xC0) != 0x80)
1811 return codecvt_base::error;
1812 uint16_t t = static_cast<uint16_t>(((c1 & 0x1F) << 6) | (c2 & 0x3F));
1813 if (t > Maxcode)
1814 return codecvt_base::error;
1815 *to_nxt = static_cast<uint32_t>(t);
1816 frm_nxt += 2;
1817 } else if (c1 < 0xF0) {
1818 if (frm_end - frm_nxt < 2)
1819 return codecvt_base::partial;
1820 uint8_t c2 = frm_nxt[1];
1821 switch (c1) {
1822 case 0xE0:
1823 if ((c2 & 0xE0) != 0xA0)
1824 return codecvt_base::error;
1825 break;
1826 case 0xED:
1827 if ((c2 & 0xE0) != 0x80)
1828 return codecvt_base::error;
1829 break;
1830 default:
1831 if ((c2 & 0xC0) != 0x80)
1832 return codecvt_base::error;
1833 break;
1835 if (frm_end - frm_nxt < 3)
1836 return codecvt_base::partial;
1837 uint8_t c3 = frm_nxt[2];
1838 if ((c3 & 0xC0) != 0x80)
1839 return codecvt_base::error;
1840 uint16_t t = static_cast<uint16_t>(((c1 & 0x0F) << 12) | ((c2 & 0x3F) << 6) | (c3 & 0x3F));
1841 if (t > Maxcode)
1842 return codecvt_base::error;
1843 *to_nxt = static_cast<uint32_t>(t);
1844 frm_nxt += 3;
1845 } else if (c1 < 0xF5) {
1846 if (frm_end - frm_nxt < 2)
1847 return codecvt_base::partial;
1848 uint8_t c2 = frm_nxt[1];
1849 switch (c1) {
1850 case 0xF0:
1851 if (!(0x90 <= c2 && c2 <= 0xBF))
1852 return codecvt_base::error;
1853 break;
1854 case 0xF4:
1855 if ((c2 & 0xF0) != 0x80)
1856 return codecvt_base::error;
1857 break;
1858 default:
1859 if ((c2 & 0xC0) != 0x80)
1860 return codecvt_base::error;
1861 break;
1863 if (frm_end - frm_nxt < 3)
1864 return codecvt_base::partial;
1865 uint8_t c3 = frm_nxt[2];
1866 if ((c3 & 0xC0) != 0x80)
1867 return codecvt_base::error;
1868 if (frm_end - frm_nxt < 4)
1869 return codecvt_base::partial;
1870 uint8_t c4 = frm_nxt[3];
1871 if ((c4 & 0xC0) != 0x80)
1872 return codecvt_base::error;
1873 if (to_end - to_nxt < 2)
1874 return codecvt_base::partial;
1875 if ((((c1 & 7UL) << 18) + ((c2 & 0x3FUL) << 12) + ((c3 & 0x3FUL) << 6) + (c4 & 0x3F)) > Maxcode)
1876 return codecvt_base::error;
1877 *to_nxt = static_cast<uint32_t>(
1878 0xD800 | (((((c1 & 0x07) << 2) | ((c2 & 0x30) >> 4)) - 1) << 6) | ((c2 & 0x0F) << 2) | ((c3 & 0x30) >> 4));
1879 *++to_nxt = static_cast<uint32_t>(0xDC00 | ((c3 & 0x0F) << 6) | (c4 & 0x3F));
1880 frm_nxt += 4;
1881 } else {
1882 return codecvt_base::error;
1885 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
1888 static int utf8_to_utf16_length(
1889 const uint8_t* frm,
1890 const uint8_t* frm_end,
1891 size_t mx,
1892 unsigned long Maxcode = 0x10FFFF,
1893 codecvt_mode mode = codecvt_mode(0)) {
1894 const uint8_t* frm_nxt = frm;
1895 if (mode & consume_header) {
1896 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
1897 frm_nxt += 3;
1899 for (size_t nchar16_t = 0; frm_nxt < frm_end && nchar16_t < mx; ++nchar16_t) {
1900 uint8_t c1 = *frm_nxt;
1901 if (c1 > Maxcode)
1902 break;
1903 if (c1 < 0x80) {
1904 ++frm_nxt;
1905 } else if (c1 < 0xC2) {
1906 break;
1907 } else if (c1 < 0xE0) {
1908 if ((frm_end - frm_nxt < 2) || (frm_nxt[1] & 0xC0) != 0x80)
1909 break;
1910 uint16_t t = static_cast<uint16_t>(((c1 & 0x1F) << 6) | (frm_nxt[1] & 0x3F));
1911 if (t > Maxcode)
1912 break;
1913 frm_nxt += 2;
1914 } else if (c1 < 0xF0) {
1915 if (frm_end - frm_nxt < 3)
1916 break;
1917 uint8_t c2 = frm_nxt[1];
1918 uint8_t c3 = frm_nxt[2];
1919 switch (c1) {
1920 case 0xE0:
1921 if ((c2 & 0xE0) != 0xA0)
1922 return static_cast<int>(frm_nxt - frm);
1923 break;
1924 case 0xED:
1925 if ((c2 & 0xE0) != 0x80)
1926 return static_cast<int>(frm_nxt - frm);
1927 break;
1928 default:
1929 if ((c2 & 0xC0) != 0x80)
1930 return static_cast<int>(frm_nxt - frm);
1931 break;
1933 if ((c3 & 0xC0) != 0x80)
1934 break;
1935 if ((((c1 & 0x0Fu) << 12) | ((c2 & 0x3Fu) << 6) | (c3 & 0x3Fu)) > Maxcode)
1936 break;
1937 frm_nxt += 3;
1938 } else if (c1 < 0xF5) {
1939 if (frm_end - frm_nxt < 4 || mx - nchar16_t < 2)
1940 break;
1941 uint8_t c2 = frm_nxt[1];
1942 uint8_t c3 = frm_nxt[2];
1943 uint8_t c4 = frm_nxt[3];
1944 switch (c1) {
1945 case 0xF0:
1946 if (!(0x90 <= c2 && c2 <= 0xBF))
1947 return static_cast<int>(frm_nxt - frm);
1948 break;
1949 case 0xF4:
1950 if ((c2 & 0xF0) != 0x80)
1951 return static_cast<int>(frm_nxt - frm);
1952 break;
1953 default:
1954 if ((c2 & 0xC0) != 0x80)
1955 return static_cast<int>(frm_nxt - frm);
1956 break;
1958 if ((c3 & 0xC0) != 0x80 || (c4 & 0xC0) != 0x80)
1959 break;
1960 if ((((c1 & 7UL) << 18) + ((c2 & 0x3FUL) << 12) + ((c3 & 0x3FUL) << 6) + (c4 & 0x3F)) > Maxcode)
1961 break;
1962 ++nchar16_t;
1963 frm_nxt += 4;
1964 } else {
1965 break;
1968 return static_cast<int>(frm_nxt - frm);
1971 static codecvt_base::result ucs4_to_utf8(
1972 const uint32_t* frm,
1973 const uint32_t* frm_end,
1974 const uint32_t*& frm_nxt,
1975 uint8_t* to,
1976 uint8_t* to_end,
1977 uint8_t*& to_nxt,
1978 unsigned long Maxcode = 0x10FFFF,
1979 codecvt_mode mode = codecvt_mode(0)) {
1980 frm_nxt = frm;
1981 to_nxt = to;
1982 if (mode & generate_header) {
1983 if (to_end - to_nxt < 3)
1984 return codecvt_base::partial;
1985 *to_nxt++ = static_cast<uint8_t>(0xEF);
1986 *to_nxt++ = static_cast<uint8_t>(0xBB);
1987 *to_nxt++ = static_cast<uint8_t>(0xBF);
1989 for (; frm_nxt < frm_end; ++frm_nxt) {
1990 uint32_t wc = *frm_nxt;
1991 if ((wc & 0xFFFFF800) == 0x00D800 || wc > Maxcode)
1992 return codecvt_base::error;
1993 if (wc < 0x000080) {
1994 if (to_end - to_nxt < 1)
1995 return codecvt_base::partial;
1996 *to_nxt++ = static_cast<uint8_t>(wc);
1997 } else if (wc < 0x000800) {
1998 if (to_end - to_nxt < 2)
1999 return codecvt_base::partial;
2000 *to_nxt++ = static_cast<uint8_t>(0xC0 | (wc >> 6));
2001 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc & 0x03F));
2002 } else if (wc < 0x010000) {
2003 if (to_end - to_nxt < 3)
2004 return codecvt_base::partial;
2005 *to_nxt++ = static_cast<uint8_t>(0xE0 | (wc >> 12));
2006 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc & 0x0FC0) >> 6));
2007 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc & 0x003F));
2008 } else // if (wc < 0x110000)
2010 if (to_end - to_nxt < 4)
2011 return codecvt_base::partial;
2012 *to_nxt++ = static_cast<uint8_t>(0xF0 | (wc >> 18));
2013 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc & 0x03F000) >> 12));
2014 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc & 0x000FC0) >> 6));
2015 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc & 0x00003F));
2018 return codecvt_base::ok;
2021 static codecvt_base::result utf8_to_ucs4(
2022 const uint8_t* frm,
2023 const uint8_t* frm_end,
2024 const uint8_t*& frm_nxt,
2025 uint32_t* to,
2026 uint32_t* to_end,
2027 uint32_t*& to_nxt,
2028 unsigned long Maxcode = 0x10FFFF,
2029 codecvt_mode mode = codecvt_mode(0)) {
2030 frm_nxt = frm;
2031 to_nxt = to;
2032 if (mode & consume_header) {
2033 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
2034 frm_nxt += 3;
2036 for (; frm_nxt < frm_end && to_nxt < to_end; ++to_nxt) {
2037 uint8_t c1 = static_cast<uint8_t>(*frm_nxt);
2038 if (c1 < 0x80) {
2039 if (c1 > Maxcode)
2040 return codecvt_base::error;
2041 *to_nxt = static_cast<uint32_t>(c1);
2042 ++frm_nxt;
2043 } else if (c1 < 0xC2) {
2044 return codecvt_base::error;
2045 } else if (c1 < 0xE0) {
2046 if (frm_end - frm_nxt < 2)
2047 return codecvt_base::partial;
2048 uint8_t c2 = frm_nxt[1];
2049 if ((c2 & 0xC0) != 0x80)
2050 return codecvt_base::error;
2051 uint32_t t = static_cast<uint32_t>(((c1 & 0x1F) << 6) | (c2 & 0x3F));
2052 if (t > Maxcode)
2053 return codecvt_base::error;
2054 *to_nxt = t;
2055 frm_nxt += 2;
2056 } else if (c1 < 0xF0) {
2057 if (frm_end - frm_nxt < 2)
2058 return codecvt_base::partial;
2059 uint8_t c2 = frm_nxt[1];
2060 switch (c1) {
2061 case 0xE0:
2062 if ((c2 & 0xE0) != 0xA0)
2063 return codecvt_base::error;
2064 break;
2065 case 0xED:
2066 if ((c2 & 0xE0) != 0x80)
2067 return codecvt_base::error;
2068 break;
2069 default:
2070 if ((c2 & 0xC0) != 0x80)
2071 return codecvt_base::error;
2072 break;
2074 if (frm_end - frm_nxt < 3)
2075 return codecvt_base::partial;
2076 uint8_t c3 = frm_nxt[2];
2077 if ((c3 & 0xC0) != 0x80)
2078 return codecvt_base::error;
2079 uint32_t t = static_cast<uint32_t>(((c1 & 0x0F) << 12) | ((c2 & 0x3F) << 6) | (c3 & 0x3F));
2080 if (t > Maxcode)
2081 return codecvt_base::error;
2082 *to_nxt = t;
2083 frm_nxt += 3;
2084 } else if (c1 < 0xF5) {
2085 if (frm_end - frm_nxt < 2)
2086 return codecvt_base::partial;
2087 uint8_t c2 = frm_nxt[1];
2088 switch (c1) {
2089 case 0xF0:
2090 if (!(0x90 <= c2 && c2 <= 0xBF))
2091 return codecvt_base::error;
2092 break;
2093 case 0xF4:
2094 if ((c2 & 0xF0) != 0x80)
2095 return codecvt_base::error;
2096 break;
2097 default:
2098 if ((c2 & 0xC0) != 0x80)
2099 return codecvt_base::error;
2100 break;
2102 if (frm_end - frm_nxt < 3)
2103 return codecvt_base::partial;
2104 uint8_t c3 = frm_nxt[2];
2105 if ((c3 & 0xC0) != 0x80)
2106 return codecvt_base::error;
2107 if (frm_end - frm_nxt < 4)
2108 return codecvt_base::partial;
2109 uint8_t c4 = frm_nxt[3];
2110 if ((c4 & 0xC0) != 0x80)
2111 return codecvt_base::error;
2112 uint32_t t = static_cast<uint32_t>(((c1 & 0x07) << 18) | ((c2 & 0x3F) << 12) | ((c3 & 0x3F) << 6) | (c4 & 0x3F));
2113 if (t > Maxcode)
2114 return codecvt_base::error;
2115 *to_nxt = t;
2116 frm_nxt += 4;
2117 } else {
2118 return codecvt_base::error;
2121 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
2124 static int utf8_to_ucs4_length(
2125 const uint8_t* frm,
2126 const uint8_t* frm_end,
2127 size_t mx,
2128 unsigned long Maxcode = 0x10FFFF,
2129 codecvt_mode mode = codecvt_mode(0)) {
2130 const uint8_t* frm_nxt = frm;
2131 if (mode & consume_header) {
2132 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
2133 frm_nxt += 3;
2135 for (size_t nchar32_t = 0; frm_nxt < frm_end && nchar32_t < mx; ++nchar32_t) {
2136 uint8_t c1 = static_cast<uint8_t>(*frm_nxt);
2137 if (c1 < 0x80) {
2138 if (c1 > Maxcode)
2139 break;
2140 ++frm_nxt;
2141 } else if (c1 < 0xC2) {
2142 break;
2143 } else if (c1 < 0xE0) {
2144 if ((frm_end - frm_nxt < 2) || ((frm_nxt[1] & 0xC0) != 0x80))
2145 break;
2146 if ((((c1 & 0x1Fu) << 6) | (frm_nxt[1] & 0x3Fu)) > Maxcode)
2147 break;
2148 frm_nxt += 2;
2149 } else if (c1 < 0xF0) {
2150 if (frm_end - frm_nxt < 3)
2151 break;
2152 uint8_t c2 = frm_nxt[1];
2153 uint8_t c3 = frm_nxt[2];
2154 switch (c1) {
2155 case 0xE0:
2156 if ((c2 & 0xE0) != 0xA0)
2157 return static_cast<int>(frm_nxt - frm);
2158 break;
2159 case 0xED:
2160 if ((c2 & 0xE0) != 0x80)
2161 return static_cast<int>(frm_nxt - frm);
2162 break;
2163 default:
2164 if ((c2 & 0xC0) != 0x80)
2165 return static_cast<int>(frm_nxt - frm);
2166 break;
2168 if ((c3 & 0xC0) != 0x80)
2169 break;
2170 if ((((c1 & 0x0Fu) << 12) | ((c2 & 0x3Fu) << 6) | (c3 & 0x3Fu)) > Maxcode)
2171 break;
2172 frm_nxt += 3;
2173 } else if (c1 < 0xF5) {
2174 if (frm_end - frm_nxt < 4)
2175 break;
2176 uint8_t c2 = frm_nxt[1];
2177 uint8_t c3 = frm_nxt[2];
2178 uint8_t c4 = frm_nxt[3];
2179 switch (c1) {
2180 case 0xF0:
2181 if (!(0x90 <= c2 && c2 <= 0xBF))
2182 return static_cast<int>(frm_nxt - frm);
2183 break;
2184 case 0xF4:
2185 if ((c2 & 0xF0) != 0x80)
2186 return static_cast<int>(frm_nxt - frm);
2187 break;
2188 default:
2189 if ((c2 & 0xC0) != 0x80)
2190 return static_cast<int>(frm_nxt - frm);
2191 break;
2193 if ((c3 & 0xC0) != 0x80 || (c4 & 0xC0) != 0x80)
2194 break;
2195 if ((((c1 & 0x07u) << 18) | ((c2 & 0x3Fu) << 12) | ((c3 & 0x3Fu) << 6) | (c4 & 0x3Fu)) > Maxcode)
2196 break;
2197 frm_nxt += 4;
2198 } else {
2199 break;
2202 return static_cast<int>(frm_nxt - frm);
2205 static codecvt_base::result ucs2_to_utf8(
2206 const uint16_t* frm,
2207 const uint16_t* frm_end,
2208 const uint16_t*& frm_nxt,
2209 uint8_t* to,
2210 uint8_t* to_end,
2211 uint8_t*& to_nxt,
2212 unsigned long Maxcode = 0x10FFFF,
2213 codecvt_mode mode = codecvt_mode(0)) {
2214 frm_nxt = frm;
2215 to_nxt = to;
2216 if (mode & generate_header) {
2217 if (to_end - to_nxt < 3)
2218 return codecvt_base::partial;
2219 *to_nxt++ = static_cast<uint8_t>(0xEF);
2220 *to_nxt++ = static_cast<uint8_t>(0xBB);
2221 *to_nxt++ = static_cast<uint8_t>(0xBF);
2223 for (; frm_nxt < frm_end; ++frm_nxt) {
2224 uint16_t wc = *frm_nxt;
2225 if ((wc & 0xF800) == 0xD800 || wc > Maxcode)
2226 return codecvt_base::error;
2227 if (wc < 0x0080) {
2228 if (to_end - to_nxt < 1)
2229 return codecvt_base::partial;
2230 *to_nxt++ = static_cast<uint8_t>(wc);
2231 } else if (wc < 0x0800) {
2232 if (to_end - to_nxt < 2)
2233 return codecvt_base::partial;
2234 *to_nxt++ = static_cast<uint8_t>(0xC0 | (wc >> 6));
2235 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc & 0x03F));
2236 } else // if (wc <= 0xFFFF)
2238 if (to_end - to_nxt < 3)
2239 return codecvt_base::partial;
2240 *to_nxt++ = static_cast<uint8_t>(0xE0 | (wc >> 12));
2241 *to_nxt++ = static_cast<uint8_t>(0x80 | ((wc & 0x0FC0) >> 6));
2242 *to_nxt++ = static_cast<uint8_t>(0x80 | (wc & 0x003F));
2245 return codecvt_base::ok;
2248 static codecvt_base::result utf8_to_ucs2(
2249 const uint8_t* frm,
2250 const uint8_t* frm_end,
2251 const uint8_t*& frm_nxt,
2252 uint16_t* to,
2253 uint16_t* to_end,
2254 uint16_t*& to_nxt,
2255 unsigned long Maxcode = 0x10FFFF,
2256 codecvt_mode mode = codecvt_mode(0)) {
2257 frm_nxt = frm;
2258 to_nxt = to;
2259 if (mode & consume_header) {
2260 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
2261 frm_nxt += 3;
2263 for (; frm_nxt < frm_end && to_nxt < to_end; ++to_nxt) {
2264 uint8_t c1 = static_cast<uint8_t>(*frm_nxt);
2265 if (c1 < 0x80) {
2266 if (c1 > Maxcode)
2267 return codecvt_base::error;
2268 *to_nxt = static_cast<uint16_t>(c1);
2269 ++frm_nxt;
2270 } else if (c1 < 0xC2) {
2271 return codecvt_base::error;
2272 } else if (c1 < 0xE0) {
2273 if (frm_end - frm_nxt < 2)
2274 return codecvt_base::partial;
2275 uint8_t c2 = frm_nxt[1];
2276 if ((c2 & 0xC0) != 0x80)
2277 return codecvt_base::error;
2278 uint16_t t = static_cast<uint16_t>(((c1 & 0x1F) << 6) | (c2 & 0x3F));
2279 if (t > Maxcode)
2280 return codecvt_base::error;
2281 *to_nxt = t;
2282 frm_nxt += 2;
2283 } else if (c1 < 0xF0) {
2284 if (frm_end - frm_nxt < 2)
2285 return codecvt_base::partial;
2286 uint8_t c2 = frm_nxt[1];
2287 switch (c1) {
2288 case 0xE0:
2289 if ((c2 & 0xE0) != 0xA0)
2290 return codecvt_base::error;
2291 break;
2292 case 0xED:
2293 if ((c2 & 0xE0) != 0x80)
2294 return codecvt_base::error;
2295 break;
2296 default:
2297 if ((c2 & 0xC0) != 0x80)
2298 return codecvt_base::error;
2299 break;
2301 if (frm_end - frm_nxt < 3)
2302 return codecvt_base::partial;
2303 uint8_t c3 = frm_nxt[2];
2304 if ((c3 & 0xC0) != 0x80)
2305 return codecvt_base::error;
2306 uint16_t t = static_cast<uint16_t>(((c1 & 0x0F) << 12) | ((c2 & 0x3F) << 6) | (c3 & 0x3F));
2307 if (t > Maxcode)
2308 return codecvt_base::error;
2309 *to_nxt = t;
2310 frm_nxt += 3;
2311 } else {
2312 return codecvt_base::error;
2315 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
2318 static int utf8_to_ucs2_length(
2319 const uint8_t* frm,
2320 const uint8_t* frm_end,
2321 size_t mx,
2322 unsigned long Maxcode = 0x10FFFF,
2323 codecvt_mode mode = codecvt_mode(0)) {
2324 const uint8_t* frm_nxt = frm;
2325 if (mode & consume_header) {
2326 if (frm_end - frm_nxt >= 3 && frm_nxt[0] == 0xEF && frm_nxt[1] == 0xBB && frm_nxt[2] == 0xBF)
2327 frm_nxt += 3;
2329 for (size_t nchar32_t = 0; frm_nxt < frm_end && nchar32_t < mx; ++nchar32_t) {
2330 uint8_t c1 = static_cast<uint8_t>(*frm_nxt);
2331 if (c1 < 0x80) {
2332 if (c1 > Maxcode)
2333 break;
2334 ++frm_nxt;
2335 } else if (c1 < 0xC2) {
2336 break;
2337 } else if (c1 < 0xE0) {
2338 if ((frm_end - frm_nxt < 2) || ((frm_nxt[1] & 0xC0) != 0x80))
2339 break;
2340 if ((((c1 & 0x1Fu) << 6) | (frm_nxt[1] & 0x3Fu)) > Maxcode)
2341 break;
2342 frm_nxt += 2;
2343 } else if (c1 < 0xF0) {
2344 if (frm_end - frm_nxt < 3)
2345 break;
2346 uint8_t c2 = frm_nxt[1];
2347 uint8_t c3 = frm_nxt[2];
2348 switch (c1) {
2349 case 0xE0:
2350 if ((c2 & 0xE0) != 0xA0)
2351 return static_cast<int>(frm_nxt - frm);
2352 break;
2353 case 0xED:
2354 if ((c2 & 0xE0) != 0x80)
2355 return static_cast<int>(frm_nxt - frm);
2356 break;
2357 default:
2358 if ((c2 & 0xC0) != 0x80)
2359 return static_cast<int>(frm_nxt - frm);
2360 break;
2362 if ((c3 & 0xC0) != 0x80)
2363 break;
2364 if ((((c1 & 0x0Fu) << 12) | ((c2 & 0x3Fu) << 6) | (c3 & 0x3Fu)) > Maxcode)
2365 break;
2366 frm_nxt += 3;
2367 } else {
2368 break;
2371 return static_cast<int>(frm_nxt - frm);
2374 static codecvt_base::result ucs4_to_utf16be(
2375 const uint32_t* frm,
2376 const uint32_t* frm_end,
2377 const uint32_t*& frm_nxt,
2378 uint8_t* to,
2379 uint8_t* to_end,
2380 uint8_t*& to_nxt,
2381 unsigned long Maxcode = 0x10FFFF,
2382 codecvt_mode mode = codecvt_mode(0)) {
2383 frm_nxt = frm;
2384 to_nxt = to;
2385 if (mode & generate_header) {
2386 if (to_end - to_nxt < 2)
2387 return codecvt_base::partial;
2388 *to_nxt++ = static_cast<uint8_t>(0xFE);
2389 *to_nxt++ = static_cast<uint8_t>(0xFF);
2391 for (; frm_nxt < frm_end; ++frm_nxt) {
2392 uint32_t wc = *frm_nxt;
2393 if ((wc & 0xFFFFF800) == 0x00D800 || wc > Maxcode)
2394 return codecvt_base::error;
2395 if (wc < 0x010000) {
2396 if (to_end - to_nxt < 2)
2397 return codecvt_base::partial;
2398 *to_nxt++ = static_cast<uint8_t>(wc >> 8);
2399 *to_nxt++ = static_cast<uint8_t>(wc);
2400 } else {
2401 if (to_end - to_nxt < 4)
2402 return codecvt_base::partial;
2403 uint16_t t = static_cast<uint16_t>(0xD800 | ((((wc & 0x1F0000) >> 16) - 1) << 6) | ((wc & 0x00FC00) >> 10));
2404 *to_nxt++ = static_cast<uint8_t>(t >> 8);
2405 *to_nxt++ = static_cast<uint8_t>(t);
2406 t = static_cast<uint16_t>(0xDC00 | (wc & 0x03FF));
2407 *to_nxt++ = static_cast<uint8_t>(t >> 8);
2408 *to_nxt++ = static_cast<uint8_t>(t);
2411 return codecvt_base::ok;
2414 static codecvt_base::result utf16be_to_ucs4(
2415 const uint8_t* frm,
2416 const uint8_t* frm_end,
2417 const uint8_t*& frm_nxt,
2418 uint32_t* to,
2419 uint32_t* to_end,
2420 uint32_t*& to_nxt,
2421 unsigned long Maxcode = 0x10FFFF,
2422 codecvt_mode mode = codecvt_mode(0)) {
2423 frm_nxt = frm;
2424 to_nxt = to;
2425 if (mode & consume_header) {
2426 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFE && frm_nxt[1] == 0xFF)
2427 frm_nxt += 2;
2429 for (; frm_nxt < frm_end - 1 && to_nxt < to_end; ++to_nxt) {
2430 uint16_t c1 = static_cast<uint16_t>(frm_nxt[0] << 8 | frm_nxt[1]);
2431 if ((c1 & 0xFC00) == 0xDC00)
2432 return codecvt_base::error;
2433 if ((c1 & 0xFC00) != 0xD800) {
2434 if (c1 > Maxcode)
2435 return codecvt_base::error;
2436 *to_nxt = static_cast<uint32_t>(c1);
2437 frm_nxt += 2;
2438 } else {
2439 if (frm_end - frm_nxt < 4)
2440 return codecvt_base::partial;
2441 uint16_t c2 = static_cast<uint16_t>(frm_nxt[2] << 8 | frm_nxt[3]);
2442 if ((c2 & 0xFC00) != 0xDC00)
2443 return codecvt_base::error;
2444 uint32_t t = static_cast<uint32_t>(((((c1 & 0x03C0) >> 6) + 1) << 16) | ((c1 & 0x003F) << 10) | (c2 & 0x03FF));
2445 if (t > Maxcode)
2446 return codecvt_base::error;
2447 *to_nxt = t;
2448 frm_nxt += 4;
2451 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
2454 static int utf16be_to_ucs4_length(
2455 const uint8_t* frm,
2456 const uint8_t* frm_end,
2457 size_t mx,
2458 unsigned long Maxcode = 0x10FFFF,
2459 codecvt_mode mode = codecvt_mode(0)) {
2460 const uint8_t* frm_nxt = frm;
2461 if (mode & consume_header) {
2462 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFE && frm_nxt[1] == 0xFF)
2463 frm_nxt += 2;
2465 for (size_t nchar32_t = 0; frm_nxt < frm_end - 1 && nchar32_t < mx; ++nchar32_t) {
2466 uint16_t c1 = static_cast<uint16_t>(frm_nxt[0] << 8 | frm_nxt[1]);
2467 if ((c1 & 0xFC00) == 0xDC00)
2468 break;
2469 if ((c1 & 0xFC00) != 0xD800) {
2470 if (c1 > Maxcode)
2471 break;
2472 frm_nxt += 2;
2473 } else {
2474 if (frm_end - frm_nxt < 4)
2475 break;
2476 uint16_t c2 = static_cast<uint16_t>(frm_nxt[2] << 8 | frm_nxt[3]);
2477 if ((c2 & 0xFC00) != 0xDC00)
2478 break;
2479 uint32_t t = static_cast<uint32_t>(((((c1 & 0x03C0) >> 6) + 1) << 16) | ((c1 & 0x003F) << 10) | (c2 & 0x03FF));
2480 if (t > Maxcode)
2481 break;
2482 frm_nxt += 4;
2485 return static_cast<int>(frm_nxt - frm);
2488 static codecvt_base::result ucs4_to_utf16le(
2489 const uint32_t* frm,
2490 const uint32_t* frm_end,
2491 const uint32_t*& frm_nxt,
2492 uint8_t* to,
2493 uint8_t* to_end,
2494 uint8_t*& to_nxt,
2495 unsigned long Maxcode = 0x10FFFF,
2496 codecvt_mode mode = codecvt_mode(0)) {
2497 frm_nxt = frm;
2498 to_nxt = to;
2499 if (mode & generate_header) {
2500 if (to_end - to_nxt < 2)
2501 return codecvt_base::partial;
2502 *to_nxt++ = static_cast<uint8_t>(0xFF);
2503 *to_nxt++ = static_cast<uint8_t>(0xFE);
2505 for (; frm_nxt < frm_end; ++frm_nxt) {
2506 uint32_t wc = *frm_nxt;
2507 if ((wc & 0xFFFFF800) == 0x00D800 || wc > Maxcode)
2508 return codecvt_base::error;
2509 if (wc < 0x010000) {
2510 if (to_end - to_nxt < 2)
2511 return codecvt_base::partial;
2512 *to_nxt++ = static_cast<uint8_t>(wc);
2513 *to_nxt++ = static_cast<uint8_t>(wc >> 8);
2514 } else {
2515 if (to_end - to_nxt < 4)
2516 return codecvt_base::partial;
2517 uint16_t t = static_cast<uint16_t>(0xD800 | ((((wc & 0x1F0000) >> 16) - 1) << 6) | ((wc & 0x00FC00) >> 10));
2518 *to_nxt++ = static_cast<uint8_t>(t);
2519 *to_nxt++ = static_cast<uint8_t>(t >> 8);
2520 t = static_cast<uint16_t>(0xDC00 | (wc & 0x03FF));
2521 *to_nxt++ = static_cast<uint8_t>(t);
2522 *to_nxt++ = static_cast<uint8_t>(t >> 8);
2525 return codecvt_base::ok;
2528 static codecvt_base::result utf16le_to_ucs4(
2529 const uint8_t* frm,
2530 const uint8_t* frm_end,
2531 const uint8_t*& frm_nxt,
2532 uint32_t* to,
2533 uint32_t* to_end,
2534 uint32_t*& to_nxt,
2535 unsigned long Maxcode = 0x10FFFF,
2536 codecvt_mode mode = codecvt_mode(0)) {
2537 frm_nxt = frm;
2538 to_nxt = to;
2539 if (mode & consume_header) {
2540 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFF && frm_nxt[1] == 0xFE)
2541 frm_nxt += 2;
2543 for (; frm_nxt < frm_end - 1 && to_nxt < to_end; ++to_nxt) {
2544 uint16_t c1 = static_cast<uint16_t>(frm_nxt[1] << 8 | frm_nxt[0]);
2545 if ((c1 & 0xFC00) == 0xDC00)
2546 return codecvt_base::error;
2547 if ((c1 & 0xFC00) != 0xD800) {
2548 if (c1 > Maxcode)
2549 return codecvt_base::error;
2550 *to_nxt = static_cast<uint32_t>(c1);
2551 frm_nxt += 2;
2552 } else {
2553 if (frm_end - frm_nxt < 4)
2554 return codecvt_base::partial;
2555 uint16_t c2 = static_cast<uint16_t>(frm_nxt[3] << 8 | frm_nxt[2]);
2556 if ((c2 & 0xFC00) != 0xDC00)
2557 return codecvt_base::error;
2558 uint32_t t = static_cast<uint32_t>(((((c1 & 0x03C0) >> 6) + 1) << 16) | ((c1 & 0x003F) << 10) | (c2 & 0x03FF));
2559 if (t > Maxcode)
2560 return codecvt_base::error;
2561 *to_nxt = t;
2562 frm_nxt += 4;
2565 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
2568 static int utf16le_to_ucs4_length(
2569 const uint8_t* frm,
2570 const uint8_t* frm_end,
2571 size_t mx,
2572 unsigned long Maxcode = 0x10FFFF,
2573 codecvt_mode mode = codecvt_mode(0)) {
2574 const uint8_t* frm_nxt = frm;
2575 if (mode & consume_header) {
2576 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFF && frm_nxt[1] == 0xFE)
2577 frm_nxt += 2;
2579 for (size_t nchar32_t = 0; frm_nxt < frm_end - 1 && nchar32_t < mx; ++nchar32_t) {
2580 uint16_t c1 = static_cast<uint16_t>(frm_nxt[1] << 8 | frm_nxt[0]);
2581 if ((c1 & 0xFC00) == 0xDC00)
2582 break;
2583 if ((c1 & 0xFC00) != 0xD800) {
2584 if (c1 > Maxcode)
2585 break;
2586 frm_nxt += 2;
2587 } else {
2588 if (frm_end - frm_nxt < 4)
2589 break;
2590 uint16_t c2 = static_cast<uint16_t>(frm_nxt[3] << 8 | frm_nxt[2]);
2591 if ((c2 & 0xFC00) != 0xDC00)
2592 break;
2593 uint32_t t = static_cast<uint32_t>(((((c1 & 0x03C0) >> 6) + 1) << 16) | ((c1 & 0x003F) << 10) | (c2 & 0x03FF));
2594 if (t > Maxcode)
2595 break;
2596 frm_nxt += 4;
2599 return static_cast<int>(frm_nxt - frm);
2602 static codecvt_base::result ucs2_to_utf16be(
2603 const uint16_t* frm,
2604 const uint16_t* frm_end,
2605 const uint16_t*& frm_nxt,
2606 uint8_t* to,
2607 uint8_t* to_end,
2608 uint8_t*& to_nxt,
2609 unsigned long Maxcode = 0x10FFFF,
2610 codecvt_mode mode = codecvt_mode(0)) {
2611 frm_nxt = frm;
2612 to_nxt = to;
2613 if (mode & generate_header) {
2614 if (to_end - to_nxt < 2)
2615 return codecvt_base::partial;
2616 *to_nxt++ = static_cast<uint8_t>(0xFE);
2617 *to_nxt++ = static_cast<uint8_t>(0xFF);
2619 for (; frm_nxt < frm_end; ++frm_nxt) {
2620 uint16_t wc = *frm_nxt;
2621 if ((wc & 0xF800) == 0xD800 || wc > Maxcode)
2622 return codecvt_base::error;
2623 if (to_end - to_nxt < 2)
2624 return codecvt_base::partial;
2625 *to_nxt++ = static_cast<uint8_t>(wc >> 8);
2626 *to_nxt++ = static_cast<uint8_t>(wc);
2628 return codecvt_base::ok;
2631 static codecvt_base::result utf16be_to_ucs2(
2632 const uint8_t* frm,
2633 const uint8_t* frm_end,
2634 const uint8_t*& frm_nxt,
2635 uint16_t* to,
2636 uint16_t* to_end,
2637 uint16_t*& to_nxt,
2638 unsigned long Maxcode = 0x10FFFF,
2639 codecvt_mode mode = codecvt_mode(0)) {
2640 frm_nxt = frm;
2641 to_nxt = to;
2642 if (mode & consume_header) {
2643 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFE && frm_nxt[1] == 0xFF)
2644 frm_nxt += 2;
2646 for (; frm_nxt < frm_end - 1 && to_nxt < to_end; ++to_nxt) {
2647 uint16_t c1 = static_cast<uint16_t>(frm_nxt[0] << 8 | frm_nxt[1]);
2648 if ((c1 & 0xF800) == 0xD800 || c1 > Maxcode)
2649 return codecvt_base::error;
2650 *to_nxt = c1;
2651 frm_nxt += 2;
2653 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
2656 static int utf16be_to_ucs2_length(
2657 const uint8_t* frm,
2658 const uint8_t* frm_end,
2659 size_t mx,
2660 unsigned long Maxcode = 0x10FFFF,
2661 codecvt_mode mode = codecvt_mode(0)) {
2662 const uint8_t* frm_nxt = frm;
2663 if (mode & consume_header) {
2664 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFE && frm_nxt[1] == 0xFF)
2665 frm_nxt += 2;
2667 for (size_t nchar16_t = 0; frm_nxt < frm_end - 1 && nchar16_t < mx; ++nchar16_t) {
2668 uint16_t c1 = static_cast<uint16_t>(frm_nxt[0] << 8 | frm_nxt[1]);
2669 if ((c1 & 0xF800) == 0xD800 || c1 > Maxcode)
2670 break;
2671 frm_nxt += 2;
2673 return static_cast<int>(frm_nxt - frm);
2676 static codecvt_base::result ucs2_to_utf16le(
2677 const uint16_t* frm,
2678 const uint16_t* frm_end,
2679 const uint16_t*& frm_nxt,
2680 uint8_t* to,
2681 uint8_t* to_end,
2682 uint8_t*& to_nxt,
2683 unsigned long Maxcode = 0x10FFFF,
2684 codecvt_mode mode = codecvt_mode(0)) {
2685 frm_nxt = frm;
2686 to_nxt = to;
2687 if (mode & generate_header) {
2688 if (to_end - to_nxt < 2)
2689 return codecvt_base::partial;
2690 *to_nxt++ = static_cast<uint8_t>(0xFF);
2691 *to_nxt++ = static_cast<uint8_t>(0xFE);
2693 for (; frm_nxt < frm_end; ++frm_nxt) {
2694 uint16_t wc = *frm_nxt;
2695 if ((wc & 0xF800) == 0xD800 || wc > Maxcode)
2696 return codecvt_base::error;
2697 if (to_end - to_nxt < 2)
2698 return codecvt_base::partial;
2699 *to_nxt++ = static_cast<uint8_t>(wc);
2700 *to_nxt++ = static_cast<uint8_t>(wc >> 8);
2702 return codecvt_base::ok;
2705 static codecvt_base::result utf16le_to_ucs2(
2706 const uint8_t* frm,
2707 const uint8_t* frm_end,
2708 const uint8_t*& frm_nxt,
2709 uint16_t* to,
2710 uint16_t* to_end,
2711 uint16_t*& to_nxt,
2712 unsigned long Maxcode = 0x10FFFF,
2713 codecvt_mode mode = codecvt_mode(0)) {
2714 frm_nxt = frm;
2715 to_nxt = to;
2716 if (mode & consume_header) {
2717 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFF && frm_nxt[1] == 0xFE)
2718 frm_nxt += 2;
2720 for (; frm_nxt < frm_end - 1 && to_nxt < to_end; ++to_nxt) {
2721 uint16_t c1 = static_cast<uint16_t>(frm_nxt[1] << 8 | frm_nxt[0]);
2722 if ((c1 & 0xF800) == 0xD800 || c1 > Maxcode)
2723 return codecvt_base::error;
2724 *to_nxt = c1;
2725 frm_nxt += 2;
2727 return frm_nxt < frm_end ? codecvt_base::partial : codecvt_base::ok;
2730 static int utf16le_to_ucs2_length(
2731 const uint8_t* frm,
2732 const uint8_t* frm_end,
2733 size_t mx,
2734 unsigned long Maxcode = 0x10FFFF,
2735 codecvt_mode mode = codecvt_mode(0)) {
2736 const uint8_t* frm_nxt = frm;
2737 frm_nxt = frm;
2738 if (mode & consume_header) {
2739 if (frm_end - frm_nxt >= 2 && frm_nxt[0] == 0xFF && frm_nxt[1] == 0xFE)
2740 frm_nxt += 2;
2742 for (size_t nchar16_t = 0; frm_nxt < frm_end - 1 && nchar16_t < mx; ++nchar16_t) {
2743 uint16_t c1 = static_cast<uint16_t>(frm_nxt[1] << 8 | frm_nxt[0]);
2744 if ((c1 & 0xF800) == 0xD800 || c1 > Maxcode)
2745 break;
2746 frm_nxt += 2;
2748 return static_cast<int>(frm_nxt - frm);
2751 _LIBCPP_SUPPRESS_DEPRECATED_POP
2753 // template <> class codecvt<char16_t, char, mbstate_t>
2755 constinit locale::id codecvt<char16_t, char, mbstate_t>::id;
2757 codecvt<char16_t, char, mbstate_t>::~codecvt() {}
2759 codecvt<char16_t, char, mbstate_t>::result codecvt<char16_t, char, mbstate_t>::do_out(
2760 state_type&,
2761 const intern_type* frm,
2762 const intern_type* frm_end,
2763 const intern_type*& frm_nxt,
2764 extern_type* to,
2765 extern_type* to_end,
2766 extern_type*& to_nxt) const {
2767 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
2768 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
2769 const uint16_t* _frm_nxt = _frm;
2770 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
2771 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
2772 uint8_t* _to_nxt = _to;
2773 result r = utf16_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2774 frm_nxt = frm + (_frm_nxt - _frm);
2775 to_nxt = to + (_to_nxt - _to);
2776 return r;
2779 codecvt<char16_t, char, mbstate_t>::result codecvt<char16_t, char, mbstate_t>::do_in(
2780 state_type&,
2781 const extern_type* frm,
2782 const extern_type* frm_end,
2783 const extern_type*& frm_nxt,
2784 intern_type* to,
2785 intern_type* to_end,
2786 intern_type*& to_nxt) const {
2787 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2788 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2789 const uint8_t* _frm_nxt = _frm;
2790 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
2791 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
2792 uint16_t* _to_nxt = _to;
2793 result r = utf8_to_utf16(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2794 frm_nxt = frm + (_frm_nxt - _frm);
2795 to_nxt = to + (_to_nxt - _to);
2796 return r;
2799 codecvt<char16_t, char, mbstate_t>::result
2800 codecvt<char16_t, char, mbstate_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
2801 to_nxt = to;
2802 return noconv;
2805 int codecvt<char16_t, char, mbstate_t>::do_encoding() const noexcept { return 0; }
2807 bool codecvt<char16_t, char, mbstate_t>::do_always_noconv() const noexcept { return false; }
2809 int codecvt<char16_t, char, mbstate_t>::do_length(
2810 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
2811 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2812 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2813 return utf8_to_utf16_length(_frm, _frm_end, mx);
2816 int codecvt<char16_t, char, mbstate_t>::do_max_length() const noexcept { return 4; }
2818 #if _LIBCPP_HAS_CHAR8_T
2820 // template <> class codecvt<char16_t, char8_t, mbstate_t>
2822 constinit locale::id codecvt<char16_t, char8_t, mbstate_t>::id;
2824 codecvt<char16_t, char8_t, mbstate_t>::~codecvt() {}
2826 codecvt<char16_t, char8_t, mbstate_t>::result codecvt<char16_t, char8_t, mbstate_t>::do_out(
2827 state_type&,
2828 const intern_type* frm,
2829 const intern_type* frm_end,
2830 const intern_type*& frm_nxt,
2831 extern_type* to,
2832 extern_type* to_end,
2833 extern_type*& to_nxt) const {
2834 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
2835 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
2836 const uint16_t* _frm_nxt = _frm;
2837 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
2838 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
2839 uint8_t* _to_nxt = _to;
2840 result r = utf16_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2841 frm_nxt = frm + (_frm_nxt - _frm);
2842 to_nxt = to + (_to_nxt - _to);
2843 return r;
2846 codecvt<char16_t, char8_t, mbstate_t>::result codecvt<char16_t, char8_t, mbstate_t>::do_in(
2847 state_type&,
2848 const extern_type* frm,
2849 const extern_type* frm_end,
2850 const extern_type*& frm_nxt,
2851 intern_type* to,
2852 intern_type* to_end,
2853 intern_type*& to_nxt) const {
2854 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2855 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2856 const uint8_t* _frm_nxt = _frm;
2857 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
2858 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
2859 uint16_t* _to_nxt = _to;
2860 result r = utf8_to_utf16(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2861 frm_nxt = frm + (_frm_nxt - _frm);
2862 to_nxt = to + (_to_nxt - _to);
2863 return r;
2866 codecvt<char16_t, char8_t, mbstate_t>::result codecvt<char16_t, char8_t, mbstate_t>::do_unshift(
2867 state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
2868 to_nxt = to;
2869 return noconv;
2872 int codecvt<char16_t, char8_t, mbstate_t>::do_encoding() const noexcept { return 0; }
2874 bool codecvt<char16_t, char8_t, mbstate_t>::do_always_noconv() const noexcept { return false; }
2876 int codecvt<char16_t, char8_t, mbstate_t>::do_length(
2877 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
2878 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2879 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2880 return utf8_to_utf16_length(_frm, _frm_end, mx);
2883 int codecvt<char16_t, char8_t, mbstate_t>::do_max_length() const noexcept { return 4; }
2885 #endif
2887 // template <> class codecvt<char32_t, char, mbstate_t>
2889 constinit locale::id codecvt<char32_t, char, mbstate_t>::id;
2891 codecvt<char32_t, char, mbstate_t>::~codecvt() {}
2893 codecvt<char32_t, char, mbstate_t>::result codecvt<char32_t, char, mbstate_t>::do_out(
2894 state_type&,
2895 const intern_type* frm,
2896 const intern_type* frm_end,
2897 const intern_type*& frm_nxt,
2898 extern_type* to,
2899 extern_type* to_end,
2900 extern_type*& to_nxt) const {
2901 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
2902 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
2903 const uint32_t* _frm_nxt = _frm;
2904 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
2905 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
2906 uint8_t* _to_nxt = _to;
2907 result r = ucs4_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2908 frm_nxt = frm + (_frm_nxt - _frm);
2909 to_nxt = to + (_to_nxt - _to);
2910 return r;
2913 codecvt<char32_t, char, mbstate_t>::result codecvt<char32_t, char, mbstate_t>::do_in(
2914 state_type&,
2915 const extern_type* frm,
2916 const extern_type* frm_end,
2917 const extern_type*& frm_nxt,
2918 intern_type* to,
2919 intern_type* to_end,
2920 intern_type*& to_nxt) const {
2921 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2922 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2923 const uint8_t* _frm_nxt = _frm;
2924 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
2925 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
2926 uint32_t* _to_nxt = _to;
2927 result r = utf8_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2928 frm_nxt = frm + (_frm_nxt - _frm);
2929 to_nxt = to + (_to_nxt - _to);
2930 return r;
2933 codecvt<char32_t, char, mbstate_t>::result
2934 codecvt<char32_t, char, mbstate_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
2935 to_nxt = to;
2936 return noconv;
2939 int codecvt<char32_t, char, mbstate_t>::do_encoding() const noexcept { return 0; }
2941 bool codecvt<char32_t, char, mbstate_t>::do_always_noconv() const noexcept { return false; }
2943 int codecvt<char32_t, char, mbstate_t>::do_length(
2944 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
2945 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2946 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2947 return utf8_to_ucs4_length(_frm, _frm_end, mx);
2950 int codecvt<char32_t, char, mbstate_t>::do_max_length() const noexcept { return 4; }
2952 #if _LIBCPP_HAS_CHAR8_T
2954 // template <> class codecvt<char32_t, char8_t, mbstate_t>
2956 constinit locale::id codecvt<char32_t, char8_t, mbstate_t>::id;
2958 codecvt<char32_t, char8_t, mbstate_t>::~codecvt() {}
2960 codecvt<char32_t, char8_t, mbstate_t>::result codecvt<char32_t, char8_t, mbstate_t>::do_out(
2961 state_type&,
2962 const intern_type* frm,
2963 const intern_type* frm_end,
2964 const intern_type*& frm_nxt,
2965 extern_type* to,
2966 extern_type* to_end,
2967 extern_type*& to_nxt) const {
2968 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
2969 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
2970 const uint32_t* _frm_nxt = _frm;
2971 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
2972 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
2973 uint8_t* _to_nxt = _to;
2974 result r = ucs4_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2975 frm_nxt = frm + (_frm_nxt - _frm);
2976 to_nxt = to + (_to_nxt - _to);
2977 return r;
2980 codecvt<char32_t, char8_t, mbstate_t>::result codecvt<char32_t, char8_t, mbstate_t>::do_in(
2981 state_type&,
2982 const extern_type* frm,
2983 const extern_type* frm_end,
2984 const extern_type*& frm_nxt,
2985 intern_type* to,
2986 intern_type* to_end,
2987 intern_type*& to_nxt) const {
2988 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
2989 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
2990 const uint8_t* _frm_nxt = _frm;
2991 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
2992 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
2993 uint32_t* _to_nxt = _to;
2994 result r = utf8_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt);
2995 frm_nxt = frm + (_frm_nxt - _frm);
2996 to_nxt = to + (_to_nxt - _to);
2997 return r;
3000 codecvt<char32_t, char8_t, mbstate_t>::result codecvt<char32_t, char8_t, mbstate_t>::do_unshift(
3001 state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3002 to_nxt = to;
3003 return noconv;
3006 int codecvt<char32_t, char8_t, mbstate_t>::do_encoding() const noexcept { return 0; }
3008 bool codecvt<char32_t, char8_t, mbstate_t>::do_always_noconv() const noexcept { return false; }
3010 int codecvt<char32_t, char8_t, mbstate_t>::do_length(
3011 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3012 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3013 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3014 return utf8_to_ucs4_length(_frm, _frm_end, mx);
3017 int codecvt<char32_t, char8_t, mbstate_t>::do_max_length() const noexcept { return 4; }
3019 #endif
3021 // __codecvt_utf8<wchar_t>
3023 #if _LIBCPP_HAS_WIDE_CHARACTERS
3024 __codecvt_utf8<wchar_t>::result __codecvt_utf8<wchar_t>::do_out(
3025 state_type&,
3026 const intern_type* frm,
3027 const intern_type* frm_end,
3028 const intern_type*& frm_nxt,
3029 extern_type* to,
3030 extern_type* to_end,
3031 extern_type*& to_nxt) const {
3032 # if defined(_LIBCPP_SHORT_WCHAR)
3033 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3034 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3035 const uint16_t* _frm_nxt = _frm;
3036 # else
3037 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3038 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3039 const uint32_t* _frm_nxt = _frm;
3040 # endif
3041 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3042 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3043 uint8_t* _to_nxt = _to;
3044 # if defined(_LIBCPP_SHORT_WCHAR)
3045 result r = ucs2_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3046 # else
3047 result r = ucs4_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3048 # endif
3049 frm_nxt = frm + (_frm_nxt - _frm);
3050 to_nxt = to + (_to_nxt - _to);
3051 return r;
3054 __codecvt_utf8<wchar_t>::result __codecvt_utf8<wchar_t>::do_in(
3055 state_type&,
3056 const extern_type* frm,
3057 const extern_type* frm_end,
3058 const extern_type*& frm_nxt,
3059 intern_type* to,
3060 intern_type* to_end,
3061 intern_type*& to_nxt) const {
3062 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3063 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3064 const uint8_t* _frm_nxt = _frm;
3065 # if defined(_LIBCPP_SHORT_WCHAR)
3066 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3067 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3068 uint16_t* _to_nxt = _to;
3069 result r = utf8_to_ucs2(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3070 # else
3071 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3072 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3073 uint32_t* _to_nxt = _to;
3074 result r = utf8_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3075 # endif
3076 frm_nxt = frm + (_frm_nxt - _frm);
3077 to_nxt = to + (_to_nxt - _to);
3078 return r;
3081 __codecvt_utf8<wchar_t>::result
3082 __codecvt_utf8<wchar_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3083 to_nxt = to;
3084 return noconv;
3087 int __codecvt_utf8<wchar_t>::do_encoding() const noexcept { return 0; }
3089 bool __codecvt_utf8<wchar_t>::do_always_noconv() const noexcept { return false; }
3091 int __codecvt_utf8<wchar_t>::do_length(
3092 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3093 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3094 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3095 # if defined(_LIBCPP_SHORT_WCHAR)
3096 return utf8_to_ucs2_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3097 # else
3098 return utf8_to_ucs4_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3099 # endif
3102 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3103 int __codecvt_utf8<wchar_t>::do_max_length() const noexcept {
3104 # if defined(_LIBCPP_SHORT_WCHAR)
3105 if (__mode_ & consume_header)
3106 return 6;
3107 return 3;
3108 # else
3109 if (__mode_ & consume_header)
3110 return 7;
3111 return 4;
3112 # endif
3114 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
3116 // __codecvt_utf8<char16_t>
3118 __codecvt_utf8<char16_t>::result __codecvt_utf8<char16_t>::do_out(
3119 state_type&,
3120 const intern_type* frm,
3121 const intern_type* frm_end,
3122 const intern_type*& frm_nxt,
3123 extern_type* to,
3124 extern_type* to_end,
3125 extern_type*& to_nxt) const {
3126 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3127 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3128 const uint16_t* _frm_nxt = _frm;
3129 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3130 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3131 uint8_t* _to_nxt = _to;
3132 result r = ucs2_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3133 frm_nxt = frm + (_frm_nxt - _frm);
3134 to_nxt = to + (_to_nxt - _to);
3135 return r;
3138 __codecvt_utf8<char16_t>::result __codecvt_utf8<char16_t>::do_in(
3139 state_type&,
3140 const extern_type* frm,
3141 const extern_type* frm_end,
3142 const extern_type*& frm_nxt,
3143 intern_type* to,
3144 intern_type* to_end,
3145 intern_type*& to_nxt) const {
3146 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3147 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3148 const uint8_t* _frm_nxt = _frm;
3149 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3150 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3151 uint16_t* _to_nxt = _to;
3152 result r = utf8_to_ucs2(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3153 frm_nxt = frm + (_frm_nxt - _frm);
3154 to_nxt = to + (_to_nxt - _to);
3155 return r;
3158 __codecvt_utf8<char16_t>::result
3159 __codecvt_utf8<char16_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3160 to_nxt = to;
3161 return noconv;
3164 int __codecvt_utf8<char16_t>::do_encoding() const noexcept { return 0; }
3166 bool __codecvt_utf8<char16_t>::do_always_noconv() const noexcept { return false; }
3168 int __codecvt_utf8<char16_t>::do_length(
3169 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3170 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3171 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3172 return utf8_to_ucs2_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3175 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3176 int __codecvt_utf8<char16_t>::do_max_length() const noexcept {
3177 if (__mode_ & consume_header)
3178 return 6;
3179 return 3;
3181 _LIBCPP_SUPPRESS_DEPRECATED_POP
3183 // __codecvt_utf8<char32_t>
3185 __codecvt_utf8<char32_t>::result __codecvt_utf8<char32_t>::do_out(
3186 state_type&,
3187 const intern_type* frm,
3188 const intern_type* frm_end,
3189 const intern_type*& frm_nxt,
3190 extern_type* to,
3191 extern_type* to_end,
3192 extern_type*& to_nxt) const {
3193 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3194 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3195 const uint32_t* _frm_nxt = _frm;
3196 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3197 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3198 uint8_t* _to_nxt = _to;
3199 result r = ucs4_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3200 frm_nxt = frm + (_frm_nxt - _frm);
3201 to_nxt = to + (_to_nxt - _to);
3202 return r;
3205 __codecvt_utf8<char32_t>::result __codecvt_utf8<char32_t>::do_in(
3206 state_type&,
3207 const extern_type* frm,
3208 const extern_type* frm_end,
3209 const extern_type*& frm_nxt,
3210 intern_type* to,
3211 intern_type* to_end,
3212 intern_type*& to_nxt) const {
3213 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3214 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3215 const uint8_t* _frm_nxt = _frm;
3216 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3217 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3218 uint32_t* _to_nxt = _to;
3219 result r = utf8_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3220 frm_nxt = frm + (_frm_nxt - _frm);
3221 to_nxt = to + (_to_nxt - _to);
3222 return r;
3225 __codecvt_utf8<char32_t>::result
3226 __codecvt_utf8<char32_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3227 to_nxt = to;
3228 return noconv;
3231 int __codecvt_utf8<char32_t>::do_encoding() const noexcept { return 0; }
3233 bool __codecvt_utf8<char32_t>::do_always_noconv() const noexcept { return false; }
3235 int __codecvt_utf8<char32_t>::do_length(
3236 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3237 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3238 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3239 return utf8_to_ucs4_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3242 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3243 int __codecvt_utf8<char32_t>::do_max_length() const noexcept {
3244 if (__mode_ & consume_header)
3245 return 7;
3246 return 4;
3248 _LIBCPP_SUPPRESS_DEPRECATED_POP
3250 // __codecvt_utf16<wchar_t, false>
3252 #if _LIBCPP_HAS_WIDE_CHARACTERS
3253 __codecvt_utf16<wchar_t, false>::result __codecvt_utf16<wchar_t, false>::do_out(
3254 state_type&,
3255 const intern_type* frm,
3256 const intern_type* frm_end,
3257 const intern_type*& frm_nxt,
3258 extern_type* to,
3259 extern_type* to_end,
3260 extern_type*& to_nxt) const {
3261 # if defined(_LIBCPP_SHORT_WCHAR)
3262 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3263 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3264 const uint16_t* _frm_nxt = _frm;
3265 # else
3266 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3267 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3268 const uint32_t* _frm_nxt = _frm;
3269 # endif
3270 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3271 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3272 uint8_t* _to_nxt = _to;
3273 # if defined(_LIBCPP_SHORT_WCHAR)
3274 result r = ucs2_to_utf16be(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3275 # else
3276 result r = ucs4_to_utf16be(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3277 # endif
3278 frm_nxt = frm + (_frm_nxt - _frm);
3279 to_nxt = to + (_to_nxt - _to);
3280 return r;
3283 __codecvt_utf16<wchar_t, false>::result __codecvt_utf16<wchar_t, false>::do_in(
3284 state_type&,
3285 const extern_type* frm,
3286 const extern_type* frm_end,
3287 const extern_type*& frm_nxt,
3288 intern_type* to,
3289 intern_type* to_end,
3290 intern_type*& to_nxt) const {
3291 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3292 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3293 const uint8_t* _frm_nxt = _frm;
3294 # if defined(_LIBCPP_SHORT_WCHAR)
3295 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3296 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3297 uint16_t* _to_nxt = _to;
3298 result r = utf16be_to_ucs2(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3299 # else
3300 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3301 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3302 uint32_t* _to_nxt = _to;
3303 result r = utf16be_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3304 # endif
3305 frm_nxt = frm + (_frm_nxt - _frm);
3306 to_nxt = to + (_to_nxt - _to);
3307 return r;
3310 __codecvt_utf16<wchar_t, false>::result
3311 __codecvt_utf16<wchar_t, false>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3312 to_nxt = to;
3313 return noconv;
3316 int __codecvt_utf16<wchar_t, false>::do_encoding() const noexcept { return 0; }
3318 bool __codecvt_utf16<wchar_t, false>::do_always_noconv() const noexcept { return false; }
3320 int __codecvt_utf16<wchar_t, false>::do_length(
3321 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3322 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3323 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3324 # if defined(_LIBCPP_SHORT_WCHAR)
3325 return utf16be_to_ucs2_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3326 # else
3327 return utf16be_to_ucs4_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3328 # endif
3331 int __codecvt_utf16<wchar_t, false>::do_max_length() const noexcept {
3332 # if defined(_LIBCPP_SHORT_WCHAR)
3333 if (__mode_ & consume_header)
3334 return 4;
3335 return 2;
3336 # else
3337 if (__mode_ & consume_header)
3338 return 6;
3339 return 4;
3340 # endif
3343 // __codecvt_utf16<wchar_t, true>
3345 __codecvt_utf16<wchar_t, true>::result __codecvt_utf16<wchar_t, true>::do_out(
3346 state_type&,
3347 const intern_type* frm,
3348 const intern_type* frm_end,
3349 const intern_type*& frm_nxt,
3350 extern_type* to,
3351 extern_type* to_end,
3352 extern_type*& to_nxt) const {
3353 # if defined(_LIBCPP_SHORT_WCHAR)
3354 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3355 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3356 const uint16_t* _frm_nxt = _frm;
3357 # else
3358 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3359 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3360 const uint32_t* _frm_nxt = _frm;
3361 # endif
3362 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3363 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3364 uint8_t* _to_nxt = _to;
3365 # if defined(_LIBCPP_SHORT_WCHAR)
3366 result r = ucs2_to_utf16le(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3367 # else
3368 result r = ucs4_to_utf16le(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3369 # endif
3370 frm_nxt = frm + (_frm_nxt - _frm);
3371 to_nxt = to + (_to_nxt - _to);
3372 return r;
3375 __codecvt_utf16<wchar_t, true>::result __codecvt_utf16<wchar_t, true>::do_in(
3376 state_type&,
3377 const extern_type* frm,
3378 const extern_type* frm_end,
3379 const extern_type*& frm_nxt,
3380 intern_type* to,
3381 intern_type* to_end,
3382 intern_type*& to_nxt) const {
3383 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3384 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3385 const uint8_t* _frm_nxt = _frm;
3386 # if defined(_LIBCPP_SHORT_WCHAR)
3387 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3388 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3389 uint16_t* _to_nxt = _to;
3390 result r = utf16le_to_ucs2(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3391 # else
3392 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3393 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3394 uint32_t* _to_nxt = _to;
3395 result r = utf16le_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3396 # endif
3397 frm_nxt = frm + (_frm_nxt - _frm);
3398 to_nxt = to + (_to_nxt - _to);
3399 return r;
3402 __codecvt_utf16<wchar_t, true>::result
3403 __codecvt_utf16<wchar_t, true>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3404 to_nxt = to;
3405 return noconv;
3408 int __codecvt_utf16<wchar_t, true>::do_encoding() const noexcept { return 0; }
3410 bool __codecvt_utf16<wchar_t, true>::do_always_noconv() const noexcept { return false; }
3412 int __codecvt_utf16<wchar_t, true>::do_length(
3413 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3414 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3415 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3416 # if defined(_LIBCPP_SHORT_WCHAR)
3417 return utf16le_to_ucs2_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3418 # else
3419 return utf16le_to_ucs4_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3420 # endif
3423 int __codecvt_utf16<wchar_t, true>::do_max_length() const noexcept {
3424 # if defined(_LIBCPP_SHORT_WCHAR)
3425 if (__mode_ & consume_header)
3426 return 4;
3427 return 2;
3428 # else
3429 if (__mode_ & consume_header)
3430 return 6;
3431 return 4;
3432 # endif
3434 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
3436 // __codecvt_utf16<char16_t, false>
3438 __codecvt_utf16<char16_t, false>::result __codecvt_utf16<char16_t, false>::do_out(
3439 state_type&,
3440 const intern_type* frm,
3441 const intern_type* frm_end,
3442 const intern_type*& frm_nxt,
3443 extern_type* to,
3444 extern_type* to_end,
3445 extern_type*& to_nxt) const {
3446 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3447 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3448 const uint16_t* _frm_nxt = _frm;
3449 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3450 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3451 uint8_t* _to_nxt = _to;
3452 result r = ucs2_to_utf16be(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3453 frm_nxt = frm + (_frm_nxt - _frm);
3454 to_nxt = to + (_to_nxt - _to);
3455 return r;
3458 __codecvt_utf16<char16_t, false>::result __codecvt_utf16<char16_t, false>::do_in(
3459 state_type&,
3460 const extern_type* frm,
3461 const extern_type* frm_end,
3462 const extern_type*& frm_nxt,
3463 intern_type* to,
3464 intern_type* to_end,
3465 intern_type*& to_nxt) const {
3466 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3467 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3468 const uint8_t* _frm_nxt = _frm;
3469 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3470 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3471 uint16_t* _to_nxt = _to;
3472 result r = utf16be_to_ucs2(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3473 frm_nxt = frm + (_frm_nxt - _frm);
3474 to_nxt = to + (_to_nxt - _to);
3475 return r;
3478 __codecvt_utf16<char16_t, false>::result
3479 __codecvt_utf16<char16_t, false>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3480 to_nxt = to;
3481 return noconv;
3484 int __codecvt_utf16<char16_t, false>::do_encoding() const noexcept { return 0; }
3486 bool __codecvt_utf16<char16_t, false>::do_always_noconv() const noexcept { return false; }
3488 int __codecvt_utf16<char16_t, false>::do_length(
3489 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3490 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3491 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3492 return utf16be_to_ucs2_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3495 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3496 int __codecvt_utf16<char16_t, false>::do_max_length() const noexcept {
3497 if (__mode_ & consume_header)
3498 return 4;
3499 return 2;
3501 _LIBCPP_SUPPRESS_DEPRECATED_POP
3503 // __codecvt_utf16<char16_t, true>
3505 __codecvt_utf16<char16_t, true>::result __codecvt_utf16<char16_t, true>::do_out(
3506 state_type&,
3507 const intern_type* frm,
3508 const intern_type* frm_end,
3509 const intern_type*& frm_nxt,
3510 extern_type* to,
3511 extern_type* to_end,
3512 extern_type*& to_nxt) const {
3513 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3514 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3515 const uint16_t* _frm_nxt = _frm;
3516 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3517 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3518 uint8_t* _to_nxt = _to;
3519 result r = ucs2_to_utf16le(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3520 frm_nxt = frm + (_frm_nxt - _frm);
3521 to_nxt = to + (_to_nxt - _to);
3522 return r;
3525 __codecvt_utf16<char16_t, true>::result __codecvt_utf16<char16_t, true>::do_in(
3526 state_type&,
3527 const extern_type* frm,
3528 const extern_type* frm_end,
3529 const extern_type*& frm_nxt,
3530 intern_type* to,
3531 intern_type* to_end,
3532 intern_type*& to_nxt) const {
3533 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3534 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3535 const uint8_t* _frm_nxt = _frm;
3536 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3537 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3538 uint16_t* _to_nxt = _to;
3539 result r = utf16le_to_ucs2(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3540 frm_nxt = frm + (_frm_nxt - _frm);
3541 to_nxt = to + (_to_nxt - _to);
3542 return r;
3545 __codecvt_utf16<char16_t, true>::result
3546 __codecvt_utf16<char16_t, true>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3547 to_nxt = to;
3548 return noconv;
3551 int __codecvt_utf16<char16_t, true>::do_encoding() const noexcept { return 0; }
3553 bool __codecvt_utf16<char16_t, true>::do_always_noconv() const noexcept { return false; }
3555 int __codecvt_utf16<char16_t, true>::do_length(
3556 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3557 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3558 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3559 return utf16le_to_ucs2_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3562 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3563 int __codecvt_utf16<char16_t, true>::do_max_length() const noexcept {
3564 if (__mode_ & consume_header)
3565 return 4;
3566 return 2;
3568 _LIBCPP_SUPPRESS_DEPRECATED_POP
3570 // __codecvt_utf16<char32_t, false>
3572 __codecvt_utf16<char32_t, false>::result __codecvt_utf16<char32_t, false>::do_out(
3573 state_type&,
3574 const intern_type* frm,
3575 const intern_type* frm_end,
3576 const intern_type*& frm_nxt,
3577 extern_type* to,
3578 extern_type* to_end,
3579 extern_type*& to_nxt) const {
3580 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3581 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3582 const uint32_t* _frm_nxt = _frm;
3583 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3584 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3585 uint8_t* _to_nxt = _to;
3586 result r = ucs4_to_utf16be(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3587 frm_nxt = frm + (_frm_nxt - _frm);
3588 to_nxt = to + (_to_nxt - _to);
3589 return r;
3592 __codecvt_utf16<char32_t, false>::result __codecvt_utf16<char32_t, false>::do_in(
3593 state_type&,
3594 const extern_type* frm,
3595 const extern_type* frm_end,
3596 const extern_type*& frm_nxt,
3597 intern_type* to,
3598 intern_type* to_end,
3599 intern_type*& to_nxt) const {
3600 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3601 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3602 const uint8_t* _frm_nxt = _frm;
3603 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3604 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3605 uint32_t* _to_nxt = _to;
3606 result r = utf16be_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3607 frm_nxt = frm + (_frm_nxt - _frm);
3608 to_nxt = to + (_to_nxt - _to);
3609 return r;
3612 __codecvt_utf16<char32_t, false>::result
3613 __codecvt_utf16<char32_t, false>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3614 to_nxt = to;
3615 return noconv;
3618 int __codecvt_utf16<char32_t, false>::do_encoding() const noexcept { return 0; }
3620 bool __codecvt_utf16<char32_t, false>::do_always_noconv() const noexcept { return false; }
3622 int __codecvt_utf16<char32_t, false>::do_length(
3623 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3624 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3625 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3626 return utf16be_to_ucs4_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3629 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3630 int __codecvt_utf16<char32_t, false>::do_max_length() const noexcept {
3631 if (__mode_ & consume_header)
3632 return 6;
3633 return 4;
3635 _LIBCPP_SUPPRESS_DEPRECATED_POP
3637 // __codecvt_utf16<char32_t, true>
3639 __codecvt_utf16<char32_t, true>::result __codecvt_utf16<char32_t, true>::do_out(
3640 state_type&,
3641 const intern_type* frm,
3642 const intern_type* frm_end,
3643 const intern_type*& frm_nxt,
3644 extern_type* to,
3645 extern_type* to_end,
3646 extern_type*& to_nxt) const {
3647 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3648 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3649 const uint32_t* _frm_nxt = _frm;
3650 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3651 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3652 uint8_t* _to_nxt = _to;
3653 result r = ucs4_to_utf16le(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3654 frm_nxt = frm + (_frm_nxt - _frm);
3655 to_nxt = to + (_to_nxt - _to);
3656 return r;
3659 __codecvt_utf16<char32_t, true>::result __codecvt_utf16<char32_t, true>::do_in(
3660 state_type&,
3661 const extern_type* frm,
3662 const extern_type* frm_end,
3663 const extern_type*& frm_nxt,
3664 intern_type* to,
3665 intern_type* to_end,
3666 intern_type*& to_nxt) const {
3667 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3668 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3669 const uint8_t* _frm_nxt = _frm;
3670 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3671 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3672 uint32_t* _to_nxt = _to;
3673 result r = utf16le_to_ucs4(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3674 frm_nxt = frm + (_frm_nxt - _frm);
3675 to_nxt = to + (_to_nxt - _to);
3676 return r;
3679 __codecvt_utf16<char32_t, true>::result
3680 __codecvt_utf16<char32_t, true>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3681 to_nxt = to;
3682 return noconv;
3685 int __codecvt_utf16<char32_t, true>::do_encoding() const noexcept { return 0; }
3687 bool __codecvt_utf16<char32_t, true>::do_always_noconv() const noexcept { return false; }
3689 int __codecvt_utf16<char32_t, true>::do_length(
3690 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3691 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3692 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3693 return utf16le_to_ucs4_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3696 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3697 int __codecvt_utf16<char32_t, true>::do_max_length() const noexcept {
3698 if (__mode_ & consume_header)
3699 return 6;
3700 return 4;
3702 _LIBCPP_SUPPRESS_DEPRECATED_POP
3704 // __codecvt_utf8_utf16<wchar_t>
3706 #if _LIBCPP_HAS_WIDE_CHARACTERS
3707 __codecvt_utf8_utf16<wchar_t>::result __codecvt_utf8_utf16<wchar_t>::do_out(
3708 state_type&,
3709 const intern_type* frm,
3710 const intern_type* frm_end,
3711 const intern_type*& frm_nxt,
3712 extern_type* to,
3713 extern_type* to_end,
3714 extern_type*& to_nxt) const {
3715 # if defined(_LIBCPP_SHORT_WCHAR)
3716 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3717 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3718 const uint16_t* _frm_nxt = _frm;
3719 # else
3720 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3721 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3722 const uint32_t* _frm_nxt = _frm;
3723 # endif
3724 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3725 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3726 uint8_t* _to_nxt = _to;
3727 result r = utf16_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3728 frm_nxt = frm + (_frm_nxt - _frm);
3729 to_nxt = to + (_to_nxt - _to);
3730 return r;
3733 __codecvt_utf8_utf16<wchar_t>::result __codecvt_utf8_utf16<wchar_t>::do_in(
3734 state_type&,
3735 const extern_type* frm,
3736 const extern_type* frm_end,
3737 const extern_type*& frm_nxt,
3738 intern_type* to,
3739 intern_type* to_end,
3740 intern_type*& to_nxt) const {
3741 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3742 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3743 const uint8_t* _frm_nxt = _frm;
3744 # if defined(_LIBCPP_SHORT_WCHAR)
3745 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3746 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3747 uint16_t* _to_nxt = _to;
3748 # else
3749 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3750 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3751 uint32_t* _to_nxt = _to;
3752 # endif
3753 result r = utf8_to_utf16(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3754 frm_nxt = frm + (_frm_nxt - _frm);
3755 to_nxt = to + (_to_nxt - _to);
3756 return r;
3759 __codecvt_utf8_utf16<wchar_t>::result
3760 __codecvt_utf8_utf16<wchar_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3761 to_nxt = to;
3762 return noconv;
3765 int __codecvt_utf8_utf16<wchar_t>::do_encoding() const noexcept { return 0; }
3767 bool __codecvt_utf8_utf16<wchar_t>::do_always_noconv() const noexcept { return false; }
3769 int __codecvt_utf8_utf16<wchar_t>::do_length(
3770 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3771 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3772 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3773 return utf8_to_utf16_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3776 int __codecvt_utf8_utf16<wchar_t>::do_max_length() const noexcept {
3777 if (__mode_ & consume_header)
3778 return 7;
3779 return 4;
3781 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
3783 // __codecvt_utf8_utf16<char16_t>
3785 __codecvt_utf8_utf16<char16_t>::result __codecvt_utf8_utf16<char16_t>::do_out(
3786 state_type&,
3787 const intern_type* frm,
3788 const intern_type* frm_end,
3789 const intern_type*& frm_nxt,
3790 extern_type* to,
3791 extern_type* to_end,
3792 extern_type*& to_nxt) const {
3793 const uint16_t* _frm = reinterpret_cast<const uint16_t*>(frm);
3794 const uint16_t* _frm_end = reinterpret_cast<const uint16_t*>(frm_end);
3795 const uint16_t* _frm_nxt = _frm;
3796 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3797 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3798 uint8_t* _to_nxt = _to;
3799 result r = utf16_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3800 frm_nxt = frm + (_frm_nxt - _frm);
3801 to_nxt = to + (_to_nxt - _to);
3802 return r;
3805 __codecvt_utf8_utf16<char16_t>::result __codecvt_utf8_utf16<char16_t>::do_in(
3806 state_type&,
3807 const extern_type* frm,
3808 const extern_type* frm_end,
3809 const extern_type*& frm_nxt,
3810 intern_type* to,
3811 intern_type* to_end,
3812 intern_type*& to_nxt) const {
3813 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3814 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3815 const uint8_t* _frm_nxt = _frm;
3816 uint16_t* _to = reinterpret_cast<uint16_t*>(to);
3817 uint16_t* _to_end = reinterpret_cast<uint16_t*>(to_end);
3818 uint16_t* _to_nxt = _to;
3819 result r = utf8_to_utf16(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3820 frm_nxt = frm + (_frm_nxt - _frm);
3821 to_nxt = to + (_to_nxt - _to);
3822 return r;
3825 __codecvt_utf8_utf16<char16_t>::result
3826 __codecvt_utf8_utf16<char16_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3827 to_nxt = to;
3828 return noconv;
3831 int __codecvt_utf8_utf16<char16_t>::do_encoding() const noexcept { return 0; }
3833 bool __codecvt_utf8_utf16<char16_t>::do_always_noconv() const noexcept { return false; }
3835 int __codecvt_utf8_utf16<char16_t>::do_length(
3836 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3837 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3838 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3839 return utf8_to_utf16_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3842 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3843 int __codecvt_utf8_utf16<char16_t>::do_max_length() const noexcept {
3844 if (__mode_ & consume_header)
3845 return 7;
3846 return 4;
3848 _LIBCPP_SUPPRESS_DEPRECATED_POP
3850 // __codecvt_utf8_utf16<char32_t>
3852 __codecvt_utf8_utf16<char32_t>::result __codecvt_utf8_utf16<char32_t>::do_out(
3853 state_type&,
3854 const intern_type* frm,
3855 const intern_type* frm_end,
3856 const intern_type*& frm_nxt,
3857 extern_type* to,
3858 extern_type* to_end,
3859 extern_type*& to_nxt) const {
3860 const uint32_t* _frm = reinterpret_cast<const uint32_t*>(frm);
3861 const uint32_t* _frm_end = reinterpret_cast<const uint32_t*>(frm_end);
3862 const uint32_t* _frm_nxt = _frm;
3863 uint8_t* _to = reinterpret_cast<uint8_t*>(to);
3864 uint8_t* _to_end = reinterpret_cast<uint8_t*>(to_end);
3865 uint8_t* _to_nxt = _to;
3866 result r = utf16_to_utf8(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3867 frm_nxt = frm + (_frm_nxt - _frm);
3868 to_nxt = to + (_to_nxt - _to);
3869 return r;
3872 __codecvt_utf8_utf16<char32_t>::result __codecvt_utf8_utf16<char32_t>::do_in(
3873 state_type&,
3874 const extern_type* frm,
3875 const extern_type* frm_end,
3876 const extern_type*& frm_nxt,
3877 intern_type* to,
3878 intern_type* to_end,
3879 intern_type*& to_nxt) const {
3880 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3881 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3882 const uint8_t* _frm_nxt = _frm;
3883 uint32_t* _to = reinterpret_cast<uint32_t*>(to);
3884 uint32_t* _to_end = reinterpret_cast<uint32_t*>(to_end);
3885 uint32_t* _to_nxt = _to;
3886 result r = utf8_to_utf16(_frm, _frm_end, _frm_nxt, _to, _to_end, _to_nxt, __maxcode_, __mode_);
3887 frm_nxt = frm + (_frm_nxt - _frm);
3888 to_nxt = to + (_to_nxt - _to);
3889 return r;
3892 __codecvt_utf8_utf16<char32_t>::result
3893 __codecvt_utf8_utf16<char32_t>::do_unshift(state_type&, extern_type* to, extern_type*, extern_type*& to_nxt) const {
3894 to_nxt = to;
3895 return noconv;
3898 int __codecvt_utf8_utf16<char32_t>::do_encoding() const noexcept { return 0; }
3900 bool __codecvt_utf8_utf16<char32_t>::do_always_noconv() const noexcept { return false; }
3902 int __codecvt_utf8_utf16<char32_t>::do_length(
3903 state_type&, const extern_type* frm, const extern_type* frm_end, size_t mx) const {
3904 const uint8_t* _frm = reinterpret_cast<const uint8_t*>(frm);
3905 const uint8_t* _frm_end = reinterpret_cast<const uint8_t*>(frm_end);
3906 return utf8_to_utf16_length(_frm, _frm_end, mx, __maxcode_, __mode_);
3909 _LIBCPP_SUPPRESS_DEPRECATED_PUSH
3910 int __codecvt_utf8_utf16<char32_t>::do_max_length() const noexcept {
3911 if (__mode_ & consume_header)
3912 return 7;
3913 return 4;
3915 _LIBCPP_SUPPRESS_DEPRECATED_POP
3917 // __narrow_to_utf8<16>
3919 __narrow_to_utf8<16>::~__narrow_to_utf8() {}
3921 // __narrow_to_utf8<32>
3923 __narrow_to_utf8<32>::~__narrow_to_utf8() {}
3925 // __widen_from_utf8<16>
3927 __widen_from_utf8<16>::~__widen_from_utf8() {}
3929 // __widen_from_utf8<32>
3931 __widen_from_utf8<32>::~__widen_from_utf8() {}
3933 #if _LIBCPP_HAS_WIDE_CHARACTERS
3934 static bool checked_string_to_wchar_convert(wchar_t& dest, const char* ptr, __locale::__locale_t loc) {
3935 if (*ptr == '\0')
3936 return false;
3937 mbstate_t mb = {};
3938 wchar_t out;
3939 size_t ret = __locale::__mbrtowc(&out, ptr, strlen(ptr), &mb, loc);
3940 if (ret == static_cast<size_t>(-1) || ret == static_cast<size_t>(-2)) {
3941 return false;
3943 dest = out;
3944 return true;
3946 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
3948 #if !_LIBCPP_HAS_WIDE_CHARACTERS
3949 static bool is_narrow_non_breaking_space(const char* ptr) {
3950 // https://www.fileformat.info/info/unicode/char/202f/index.htm
3951 return ptr[0] == '\xe2' && ptr[1] == '\x80' && ptr[2] == '\xaf';
3954 static bool is_non_breaking_space(const char* ptr) {
3955 // https://www.fileformat.info/info/unicode/char/0a/index.htm
3956 return ptr[0] == '\xc2' && ptr[1] == '\xa0';
3958 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
3960 static bool checked_string_to_char_convert(char& dest, const char* ptr, __locale::__locale_t __loc) {
3961 if (*ptr == '\0')
3962 return false;
3963 if (!ptr[1]) {
3964 dest = *ptr;
3965 return true;
3968 #if _LIBCPP_HAS_WIDE_CHARACTERS
3969 // First convert the MBS into a wide char then attempt to narrow it using
3970 // wctob_l.
3971 wchar_t wout;
3972 if (!checked_string_to_wchar_convert(wout, ptr, __loc))
3973 return false;
3974 int res;
3975 if ((res = __locale::__wctob(wout, __loc)) != char_traits<char>::eof()) {
3976 dest = res;
3977 return true;
3979 // FIXME: Work around specific multibyte sequences that we can reasonably
3980 // translate into a different single byte.
3981 switch (wout) {
3982 case L'\u202F': // narrow non-breaking space
3983 case L'\u00A0': // non-breaking space
3984 dest = ' ';
3985 return true;
3986 default:
3987 return false;
3989 #else // _LIBCPP_HAS_WIDE_CHARACTERS
3990 // FIXME: Work around specific multibyte sequences that we can reasonably
3991 // translate into a different single byte.
3992 if (is_narrow_non_breaking_space(ptr) || is_non_breaking_space(ptr)) {
3993 dest = ' ';
3994 return true;
3997 return false;
3998 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
3999 __libcpp_unreachable();
4002 // numpunct<char> && numpunct<wchar_t>
4004 constinit locale::id numpunct<char>::id;
4005 #if _LIBCPP_HAS_WIDE_CHARACTERS
4006 constinit locale::id numpunct<wchar_t>::id;
4007 #endif
4009 numpunct<char>::numpunct(size_t refs) : locale::facet(refs), __decimal_point_('.'), __thousands_sep_(',') {}
4011 #if _LIBCPP_HAS_WIDE_CHARACTERS
4012 numpunct<wchar_t>::numpunct(size_t refs) : locale::facet(refs), __decimal_point_(L'.'), __thousands_sep_(L',') {}
4013 #endif
4015 numpunct<char>::~numpunct() {}
4017 #if _LIBCPP_HAS_WIDE_CHARACTERS
4018 numpunct<wchar_t>::~numpunct() {}
4019 #endif
4021 char numpunct< char >::do_decimal_point() const { return __decimal_point_; }
4022 #if _LIBCPP_HAS_WIDE_CHARACTERS
4023 wchar_t numpunct<wchar_t>::do_decimal_point() const { return __decimal_point_; }
4024 #endif
4026 char numpunct< char >::do_thousands_sep() const { return __thousands_sep_; }
4027 #if _LIBCPP_HAS_WIDE_CHARACTERS
4028 wchar_t numpunct<wchar_t>::do_thousands_sep() const { return __thousands_sep_; }
4029 #endif
4031 string numpunct< char >::do_grouping() const { return __grouping_; }
4032 #if _LIBCPP_HAS_WIDE_CHARACTERS
4033 string numpunct<wchar_t>::do_grouping() const { return __grouping_; }
4034 #endif
4036 string numpunct< char >::do_truename() const { return "true"; }
4037 #if _LIBCPP_HAS_WIDE_CHARACTERS
4038 wstring numpunct<wchar_t>::do_truename() const { return L"true"; }
4039 #endif
4041 string numpunct< char >::do_falsename() const { return "false"; }
4042 #if _LIBCPP_HAS_WIDE_CHARACTERS
4043 wstring numpunct<wchar_t>::do_falsename() const { return L"false"; }
4044 #endif
4046 // numpunct_byname<char>
4048 numpunct_byname<char>::numpunct_byname(const char* nm, size_t refs) : numpunct<char>(refs) { __init(nm); }
4050 numpunct_byname<char>::numpunct_byname(const string& nm, size_t refs) : numpunct<char>(refs) { __init(nm.c_str()); }
4052 numpunct_byname<char>::~numpunct_byname() {}
4054 void numpunct_byname<char>::__init(const char* nm) {
4055 typedef numpunct<char> base;
4056 if (strcmp(nm, "C") != 0) {
4057 __libcpp_unique_locale loc(nm);
4058 if (!loc)
4059 __throw_runtime_error(
4060 ("numpunct_byname<char>::numpunct_byname"
4061 " failed to construct for " +
4062 string(nm))
4063 .c_str());
4065 lconv* lc = __locale::__localeconv(loc.get());
4066 if (!checked_string_to_char_convert(__decimal_point_, lc->decimal_point, loc.get()))
4067 __decimal_point_ = base::do_decimal_point();
4068 if (!checked_string_to_char_convert(__thousands_sep_, lc->thousands_sep, loc.get()))
4069 __thousands_sep_ = base::do_thousands_sep();
4070 __grouping_ = lc->grouping;
4071 // localization for truename and falsename is not available
4075 // numpunct_byname<wchar_t>
4077 #if _LIBCPP_HAS_WIDE_CHARACTERS
4078 numpunct_byname<wchar_t>::numpunct_byname(const char* nm, size_t refs) : numpunct<wchar_t>(refs) { __init(nm); }
4080 numpunct_byname<wchar_t>::numpunct_byname(const string& nm, size_t refs) : numpunct<wchar_t>(refs) {
4081 __init(nm.c_str());
4084 numpunct_byname<wchar_t>::~numpunct_byname() {}
4086 void numpunct_byname<wchar_t>::__init(const char* nm) {
4087 if (strcmp(nm, "C") != 0) {
4088 __libcpp_unique_locale loc(nm);
4089 if (!loc)
4090 __throw_runtime_error(
4091 ("numpunct_byname<wchar_t>::numpunct_byname"
4092 " failed to construct for " +
4093 string(nm))
4094 .c_str());
4096 lconv* lc = __locale::__localeconv(loc.get());
4097 checked_string_to_wchar_convert(__decimal_point_, lc->decimal_point, loc.get());
4098 checked_string_to_wchar_convert(__thousands_sep_, lc->thousands_sep, loc.get());
4099 __grouping_ = lc->grouping;
4100 // localization for truename and falsename is not available
4103 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
4105 // num_get helpers
4107 int __num_get_base::__get_base(ios_base& iob) {
4108 ios_base::fmtflags __basefield = iob.flags() & ios_base::basefield;
4109 if (__basefield == ios_base::oct)
4110 return 8;
4111 else if (__basefield == ios_base::hex)
4112 return 16;
4113 else if (__basefield == 0)
4114 return 0;
4115 return 10;
4118 const char __num_get_base::__src[33] = "0123456789abcdefABCDEFxX+-pPiInN";
4120 void __check_grouping(const string& __grouping, unsigned* __g, unsigned* __g_end, ios_base::iostate& __err) {
4121 // if the grouping pattern is empty _or_ there are no grouping bits, then do nothing
4122 // we always have at least a single entry in [__g, __g_end); the end of the input sequence
4123 if (__grouping.size() != 0 && __g_end - __g > 1) {
4124 reverse(__g, __g_end);
4125 const char* __ig = __grouping.data();
4126 const char* __eg = __ig + __grouping.size();
4127 for (unsigned* __r = __g; __r < __g_end - 1; ++__r) {
4128 if (0 < *__ig && *__ig < numeric_limits<char>::max()) {
4129 if (static_cast<unsigned>(*__ig) != *__r) {
4130 __err = ios_base::failbit;
4131 return;
4134 if (__eg - __ig > 1)
4135 ++__ig;
4137 if (0 < *__ig && *__ig < numeric_limits<char>::max()) {
4138 if (static_cast<unsigned>(*__ig) < __g_end[-1] || __g_end[-1] == 0)
4139 __err = ios_base::failbit;
4144 void __num_put_base::__format_int(char* __fmtp, const char* __len, bool __signd, ios_base::fmtflags __flags) {
4145 if ((__flags & ios_base::showpos) && (__flags & ios_base::basefield) != ios_base::oct &&
4146 (__flags & ios_base::basefield) != ios_base::hex && __signd)
4147 *__fmtp++ = '+';
4148 if (__flags & ios_base::showbase)
4149 *__fmtp++ = '#';
4150 while (*__len)
4151 *__fmtp++ = *__len++;
4152 if ((__flags & ios_base::basefield) == ios_base::oct)
4153 *__fmtp = 'o';
4154 else if ((__flags & ios_base::basefield) == ios_base::hex) {
4155 if (__flags & ios_base::uppercase)
4156 *__fmtp = 'X';
4157 else
4158 *__fmtp = 'x';
4159 } else if (__signd)
4160 *__fmtp = 'd';
4161 else
4162 *__fmtp = 'u';
4165 bool __num_put_base::__format_float(char* __fmtp, const char* __len, ios_base::fmtflags __flags) {
4166 bool specify_precision = true;
4167 if (__flags & ios_base::showpos)
4168 *__fmtp++ = '+';
4169 if (__flags & ios_base::showpoint)
4170 *__fmtp++ = '#';
4171 ios_base::fmtflags floatfield = __flags & ios_base::floatfield;
4172 bool uppercase = (__flags & ios_base::uppercase) != 0;
4173 if (floatfield == (ios_base::fixed | ios_base::scientific))
4174 specify_precision = false;
4175 else {
4176 *__fmtp++ = '.';
4177 *__fmtp++ = '*';
4179 while (*__len)
4180 *__fmtp++ = *__len++;
4181 if (floatfield == ios_base::fixed) {
4182 if (uppercase)
4183 *__fmtp = 'F';
4184 else
4185 *__fmtp = 'f';
4186 } else if (floatfield == ios_base::scientific) {
4187 if (uppercase)
4188 *__fmtp = 'E';
4189 else
4190 *__fmtp = 'e';
4191 } else if (floatfield == (ios_base::fixed | ios_base::scientific)) {
4192 if (uppercase)
4193 *__fmtp = 'A';
4194 else
4195 *__fmtp = 'a';
4196 } else {
4197 if (uppercase)
4198 *__fmtp = 'G';
4199 else
4200 *__fmtp = 'g';
4202 return specify_precision;
4205 char* __num_put_base::__identify_padding(char* __nb, char* __ne, const ios_base& __iob) {
4206 switch (__iob.flags() & ios_base::adjustfield) {
4207 case ios_base::internal:
4208 if (__nb[0] == '-' || __nb[0] == '+')
4209 return __nb + 1;
4210 if (__ne - __nb >= 2 && __nb[0] == '0' && (__nb[1] == 'x' || __nb[1] == 'X'))
4211 return __nb + 2;
4212 break;
4213 case ios_base::left:
4214 return __ne;
4215 case ios_base::right:
4216 default:
4217 break;
4219 return __nb;
4222 // time_get
4224 static string* init_weeks() {
4225 static string weeks[14];
4226 weeks[0] = "Sunday";
4227 weeks[1] = "Monday";
4228 weeks[2] = "Tuesday";
4229 weeks[3] = "Wednesday";
4230 weeks[4] = "Thursday";
4231 weeks[5] = "Friday";
4232 weeks[6] = "Saturday";
4233 weeks[7] = "Sun";
4234 weeks[8] = "Mon";
4235 weeks[9] = "Tue";
4236 weeks[10] = "Wed";
4237 weeks[11] = "Thu";
4238 weeks[12] = "Fri";
4239 weeks[13] = "Sat";
4240 return weeks;
4243 #if _LIBCPP_HAS_WIDE_CHARACTERS
4244 static wstring* init_wweeks() {
4245 static wstring weeks[14];
4246 weeks[0] = L"Sunday";
4247 weeks[1] = L"Monday";
4248 weeks[2] = L"Tuesday";
4249 weeks[3] = L"Wednesday";
4250 weeks[4] = L"Thursday";
4251 weeks[5] = L"Friday";
4252 weeks[6] = L"Saturday";
4253 weeks[7] = L"Sun";
4254 weeks[8] = L"Mon";
4255 weeks[9] = L"Tue";
4256 weeks[10] = L"Wed";
4257 weeks[11] = L"Thu";
4258 weeks[12] = L"Fri";
4259 weeks[13] = L"Sat";
4260 return weeks;
4262 #endif
4264 template <>
4265 const string* __time_get_c_storage<char>::__weeks() const {
4266 static const string* weeks = init_weeks();
4267 return weeks;
4270 #if _LIBCPP_HAS_WIDE_CHARACTERS
4271 template <>
4272 const wstring* __time_get_c_storage<wchar_t>::__weeks() const {
4273 static const wstring* weeks = init_wweeks();
4274 return weeks;
4276 #endif
4278 static string* init_months() {
4279 static string months[24];
4280 months[0] = "January";
4281 months[1] = "February";
4282 months[2] = "March";
4283 months[3] = "April";
4284 months[4] = "May";
4285 months[5] = "June";
4286 months[6] = "July";
4287 months[7] = "August";
4288 months[8] = "September";
4289 months[9] = "October";
4290 months[10] = "November";
4291 months[11] = "December";
4292 months[12] = "Jan";
4293 months[13] = "Feb";
4294 months[14] = "Mar";
4295 months[15] = "Apr";
4296 months[16] = "May";
4297 months[17] = "Jun";
4298 months[18] = "Jul";
4299 months[19] = "Aug";
4300 months[20] = "Sep";
4301 months[21] = "Oct";
4302 months[22] = "Nov";
4303 months[23] = "Dec";
4304 return months;
4307 #if _LIBCPP_HAS_WIDE_CHARACTERS
4308 static wstring* init_wmonths() {
4309 static wstring months[24];
4310 months[0] = L"January";
4311 months[1] = L"February";
4312 months[2] = L"March";
4313 months[3] = L"April";
4314 months[4] = L"May";
4315 months[5] = L"June";
4316 months[6] = L"July";
4317 months[7] = L"August";
4318 months[8] = L"September";
4319 months[9] = L"October";
4320 months[10] = L"November";
4321 months[11] = L"December";
4322 months[12] = L"Jan";
4323 months[13] = L"Feb";
4324 months[14] = L"Mar";
4325 months[15] = L"Apr";
4326 months[16] = L"May";
4327 months[17] = L"Jun";
4328 months[18] = L"Jul";
4329 months[19] = L"Aug";
4330 months[20] = L"Sep";
4331 months[21] = L"Oct";
4332 months[22] = L"Nov";
4333 months[23] = L"Dec";
4334 return months;
4336 #endif
4338 template <>
4339 const string* __time_get_c_storage<char>::__months() const {
4340 static const string* months = init_months();
4341 return months;
4344 #if _LIBCPP_HAS_WIDE_CHARACTERS
4345 template <>
4346 const wstring* __time_get_c_storage<wchar_t>::__months() const {
4347 static const wstring* months = init_wmonths();
4348 return months;
4350 #endif
4352 static string* init_am_pm() {
4353 static string am_pm[2];
4354 am_pm[0] = "AM";
4355 am_pm[1] = "PM";
4356 return am_pm;
4359 #if _LIBCPP_HAS_WIDE_CHARACTERS
4360 static wstring* init_wam_pm() {
4361 static wstring am_pm[2];
4362 am_pm[0] = L"AM";
4363 am_pm[1] = L"PM";
4364 return am_pm;
4366 #endif
4368 template <>
4369 const string* __time_get_c_storage<char>::__am_pm() const {
4370 static const string* am_pm = init_am_pm();
4371 return am_pm;
4374 #if _LIBCPP_HAS_WIDE_CHARACTERS
4375 template <>
4376 const wstring* __time_get_c_storage<wchar_t>::__am_pm() const {
4377 static const wstring* am_pm = init_wam_pm();
4378 return am_pm;
4380 #endif
4382 template <>
4383 const string& __time_get_c_storage<char>::__x() const {
4384 static string s("%m/%d/%y");
4385 return s;
4388 #if _LIBCPP_HAS_WIDE_CHARACTERS
4389 template <>
4390 const wstring& __time_get_c_storage<wchar_t>::__x() const {
4391 static wstring s(L"%m/%d/%y");
4392 return s;
4394 #endif
4396 template <>
4397 const string& __time_get_c_storage<char>::__X() const {
4398 static string s("%H:%M:%S");
4399 return s;
4402 #if _LIBCPP_HAS_WIDE_CHARACTERS
4403 template <>
4404 const wstring& __time_get_c_storage<wchar_t>::__X() const {
4405 static wstring s(L"%H:%M:%S");
4406 return s;
4408 #endif
4410 template <>
4411 const string& __time_get_c_storage<char>::__c() const {
4412 static string s("%a %b %d %H:%M:%S %Y");
4413 return s;
4416 #if _LIBCPP_HAS_WIDE_CHARACTERS
4417 template <>
4418 const wstring& __time_get_c_storage<wchar_t>::__c() const {
4419 static wstring s(L"%a %b %d %H:%M:%S %Y");
4420 return s;
4422 #endif
4424 template <>
4425 const string& __time_get_c_storage<char>::__r() const {
4426 static string s("%I:%M:%S %p");
4427 return s;
4430 #if _LIBCPP_HAS_WIDE_CHARACTERS
4431 template <>
4432 const wstring& __time_get_c_storage<wchar_t>::__r() const {
4433 static wstring s(L"%I:%M:%S %p");
4434 return s;
4436 #endif
4438 // time_get_byname
4440 __time_get::__time_get(const char* nm) : __loc_(__locale::__newlocale(LC_ALL_MASK, nm, 0)) {
4441 if (__loc_ == 0)
4442 __throw_runtime_error(("time_get_byname failed to construct for " + string(nm)).c_str());
4445 __time_get::__time_get(const string& nm) : __loc_(__locale::__newlocale(LC_ALL_MASK, nm.c_str(), 0)) {
4446 if (__loc_ == 0)
4447 __throw_runtime_error(("time_get_byname failed to construct for " + nm).c_str());
4450 __time_get::~__time_get() { __locale::__freelocale(__loc_); }
4452 _LIBCPP_CLANG_DIAGNOSTIC_IGNORED("-Wmissing-field-initializers")
4454 template <>
4455 string __time_get_storage<char>::__analyze(char fmt, const ctype<char>& ct) {
4456 tm t = {0};
4457 t.tm_sec = 59;
4458 t.tm_min = 55;
4459 t.tm_hour = 23;
4460 t.tm_mday = 31;
4461 t.tm_mon = 11;
4462 t.tm_year = 161;
4463 t.tm_wday = 6;
4464 t.tm_yday = 364;
4465 t.tm_isdst = -1;
4466 char buf[100];
4467 char f[3] = {0};
4468 f[0] = '%';
4469 f[1] = fmt;
4470 size_t n = __locale::__strftime(buf, countof(buf), f, &t, __loc_);
4471 char* bb = buf;
4472 char* be = buf + n;
4473 string result;
4474 while (bb != be) {
4475 if (ct.is(ctype_base::space, *bb)) {
4476 result.push_back(' ');
4477 for (++bb; bb != be && ct.is(ctype_base::space, *bb); ++bb)
4479 continue;
4481 char* w = bb;
4482 ios_base::iostate err = ios_base::goodbit;
4483 ptrdiff_t i = __scan_keyword(w, be, this->__weeks_, this->__weeks_ + 14, ct, err, false) - this->__weeks_;
4484 if (i < 14) {
4485 result.push_back('%');
4486 if (i < 7)
4487 result.push_back('A');
4488 else
4489 result.push_back('a');
4490 bb = w;
4491 continue;
4493 w = bb;
4494 i = __scan_keyword(w, be, this->__months_, this->__months_ + 24, ct, err, false) - this->__months_;
4495 if (i < 24) {
4496 result.push_back('%');
4497 if (i < 12)
4498 result.push_back('B');
4499 else
4500 result.push_back('b');
4501 if (fmt == 'x' && ct.is(ctype_base::digit, this->__months_[i][0]))
4502 result.back() = 'm';
4503 bb = w;
4504 continue;
4506 if (this->__am_pm_[0].size() + this->__am_pm_[1].size() > 0) {
4507 w = bb;
4508 i = __scan_keyword(w, be, this->__am_pm_, this->__am_pm_ + 2, ct, err, false) - this->__am_pm_;
4509 if (i < 2) {
4510 result.push_back('%');
4511 result.push_back('p');
4512 bb = w;
4513 continue;
4516 w = bb;
4517 if (ct.is(ctype_base::digit, *bb)) {
4518 switch (__get_up_to_n_digits(bb, be, err, ct, 4)) {
4519 case 6:
4520 result.push_back('%');
4521 result.push_back('w');
4522 break;
4523 case 7:
4524 result.push_back('%');
4525 result.push_back('u');
4526 break;
4527 case 11:
4528 result.push_back('%');
4529 result.push_back('I');
4530 break;
4531 case 12:
4532 result.push_back('%');
4533 result.push_back('m');
4534 break;
4535 case 23:
4536 result.push_back('%');
4537 result.push_back('H');
4538 break;
4539 case 31:
4540 result.push_back('%');
4541 result.push_back('d');
4542 break;
4543 case 55:
4544 result.push_back('%');
4545 result.push_back('M');
4546 break;
4547 case 59:
4548 result.push_back('%');
4549 result.push_back('S');
4550 break;
4551 case 61:
4552 result.push_back('%');
4553 result.push_back('y');
4554 break;
4555 case 364:
4556 result.push_back('%');
4557 result.push_back('j');
4558 break;
4559 case 2061:
4560 result.push_back('%');
4561 result.push_back('Y');
4562 break;
4563 default:
4564 for (; w != bb; ++w)
4565 result.push_back(*w);
4566 break;
4568 continue;
4570 if (*bb == '%') {
4571 result.push_back('%');
4572 result.push_back('%');
4573 ++bb;
4574 continue;
4576 result.push_back(*bb);
4577 ++bb;
4579 return result;
4582 _LIBCPP_CLANG_DIAGNOSTIC_IGNORED("-Wmissing-braces")
4584 #if _LIBCPP_HAS_WIDE_CHARACTERS
4585 template <>
4586 wstring __time_get_storage<wchar_t>::__analyze(char fmt, const ctype<wchar_t>& ct) {
4587 tm t = {0};
4588 t.tm_sec = 59;
4589 t.tm_min = 55;
4590 t.tm_hour = 23;
4591 t.tm_mday = 31;
4592 t.tm_mon = 11;
4593 t.tm_year = 161;
4594 t.tm_wday = 6;
4595 t.tm_yday = 364;
4596 t.tm_isdst = -1;
4597 char buf[100];
4598 char f[3] = {0};
4599 f[0] = '%';
4600 f[1] = fmt;
4601 __locale::__strftime(buf, countof(buf), f, &t, __loc_);
4602 wchar_t wbuf[100];
4603 wchar_t* wbb = wbuf;
4604 mbstate_t mb = {0};
4605 const char* bb = buf;
4606 size_t j = __locale::__mbsrtowcs(wbb, &bb, countof(wbuf), &mb, __loc_);
4607 if (j == size_t(-1))
4608 __throw_runtime_error("locale not supported");
4609 wchar_t* wbe = wbb + j;
4610 wstring result;
4611 while (wbb != wbe) {
4612 if (ct.is(ctype_base::space, *wbb)) {
4613 result.push_back(L' ');
4614 for (++wbb; wbb != wbe && ct.is(ctype_base::space, *wbb); ++wbb)
4616 continue;
4618 wchar_t* w = wbb;
4619 ios_base::iostate err = ios_base::goodbit;
4620 ptrdiff_t i = __scan_keyword(w, wbe, this->__weeks_, this->__weeks_ + 14, ct, err, false) - this->__weeks_;
4621 if (i < 14) {
4622 result.push_back(L'%');
4623 if (i < 7)
4624 result.push_back(L'A');
4625 else
4626 result.push_back(L'a');
4627 wbb = w;
4628 continue;
4630 w = wbb;
4631 i = __scan_keyword(w, wbe, this->__months_, this->__months_ + 24, ct, err, false) - this->__months_;
4632 if (i < 24) {
4633 result.push_back(L'%');
4634 if (i < 12)
4635 result.push_back(L'B');
4636 else
4637 result.push_back(L'b');
4638 if (fmt == 'x' && ct.is(ctype_base::digit, this->__months_[i][0]))
4639 result.back() = L'm';
4640 wbb = w;
4641 continue;
4643 if (this->__am_pm_[0].size() + this->__am_pm_[1].size() > 0) {
4644 w = wbb;
4645 i = __scan_keyword(w, wbe, this->__am_pm_, this->__am_pm_ + 2, ct, err, false) - this->__am_pm_;
4646 if (i < 2) {
4647 result.push_back(L'%');
4648 result.push_back(L'p');
4649 wbb = w;
4650 continue;
4653 w = wbb;
4654 if (ct.is(ctype_base::digit, *wbb)) {
4655 switch (__get_up_to_n_digits(wbb, wbe, err, ct, 4)) {
4656 case 6:
4657 result.push_back(L'%');
4658 result.push_back(L'w');
4659 break;
4660 case 7:
4661 result.push_back(L'%');
4662 result.push_back(L'u');
4663 break;
4664 case 11:
4665 result.push_back(L'%');
4666 result.push_back(L'I');
4667 break;
4668 case 12:
4669 result.push_back(L'%');
4670 result.push_back(L'm');
4671 break;
4672 case 23:
4673 result.push_back(L'%');
4674 result.push_back(L'H');
4675 break;
4676 case 31:
4677 result.push_back(L'%');
4678 result.push_back(L'd');
4679 break;
4680 case 55:
4681 result.push_back(L'%');
4682 result.push_back(L'M');
4683 break;
4684 case 59:
4685 result.push_back(L'%');
4686 result.push_back(L'S');
4687 break;
4688 case 61:
4689 result.push_back(L'%');
4690 result.push_back(L'y');
4691 break;
4692 case 364:
4693 result.push_back(L'%');
4694 result.push_back(L'j');
4695 break;
4696 case 2061:
4697 result.push_back(L'%');
4698 result.push_back(L'Y');
4699 break;
4700 default:
4701 for (; w != wbb; ++w)
4702 result.push_back(*w);
4703 break;
4705 continue;
4707 if (ct.narrow(*wbb, 0) == '%') {
4708 result.push_back(L'%');
4709 result.push_back(L'%');
4710 ++wbb;
4711 continue;
4713 result.push_back(*wbb);
4714 ++wbb;
4716 return result;
4718 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
4720 template <>
4721 void __time_get_storage<char>::init(const ctype<char>& ct) {
4722 tm t = {0};
4723 char buf[100];
4724 // __weeks_
4725 for (int i = 0; i < 7; ++i) {
4726 t.tm_wday = i;
4727 __locale::__strftime(buf, countof(buf), "%A", &t, __loc_);
4728 __weeks_[i] = buf;
4729 __locale::__strftime(buf, countof(buf), "%a", &t, __loc_);
4730 __weeks_[i + 7] = buf;
4732 // __months_
4733 for (int i = 0; i < 12; ++i) {
4734 t.tm_mon = i;
4735 __locale::__strftime(buf, countof(buf), "%B", &t, __loc_);
4736 __months_[i] = buf;
4737 __locale::__strftime(buf, countof(buf), "%b", &t, __loc_);
4738 __months_[i + 12] = buf;
4740 // __am_pm_
4741 t.tm_hour = 1;
4742 __locale::__strftime(buf, countof(buf), "%p", &t, __loc_);
4743 __am_pm_[0] = buf;
4744 t.tm_hour = 13;
4745 __locale::__strftime(buf, countof(buf), "%p", &t, __loc_);
4746 __am_pm_[1] = buf;
4747 __c_ = __analyze('c', ct);
4748 __r_ = __analyze('r', ct);
4749 __x_ = __analyze('x', ct);
4750 __X_ = __analyze('X', ct);
4753 #if _LIBCPP_HAS_WIDE_CHARACTERS
4754 template <>
4755 void __time_get_storage<wchar_t>::init(const ctype<wchar_t>& ct) {
4756 tm t = {0};
4757 char buf[100];
4758 wchar_t wbuf[100];
4759 wchar_t* wbe;
4760 mbstate_t mb = {0};
4761 // __weeks_
4762 for (int i = 0; i < 7; ++i) {
4763 t.tm_wday = i;
4764 __locale::__strftime(buf, countof(buf), "%A", &t, __loc_);
4765 mb = mbstate_t();
4766 const char* bb = buf;
4767 size_t j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, __loc_);
4768 if (j == size_t(-1) || j == 0)
4769 __throw_runtime_error("locale not supported");
4770 wbe = wbuf + j;
4771 __weeks_[i].assign(wbuf, wbe);
4772 __locale::__strftime(buf, countof(buf), "%a", &t, __loc_);
4773 mb = mbstate_t();
4774 bb = buf;
4775 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, __loc_);
4776 if (j == size_t(-1) || j == 0)
4777 __throw_runtime_error("locale not supported");
4778 wbe = wbuf + j;
4779 __weeks_[i + 7].assign(wbuf, wbe);
4781 // __months_
4782 for (int i = 0; i < 12; ++i) {
4783 t.tm_mon = i;
4784 __locale::__strftime(buf, countof(buf), "%B", &t, __loc_);
4785 mb = mbstate_t();
4786 const char* bb = buf;
4787 size_t j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, __loc_);
4788 if (j == size_t(-1) || j == 0)
4789 __throw_runtime_error("locale not supported");
4790 wbe = wbuf + j;
4791 __months_[i].assign(wbuf, wbe);
4792 __locale::__strftime(buf, countof(buf), "%b", &t, __loc_);
4793 mb = mbstate_t();
4794 bb = buf;
4795 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, __loc_);
4796 if (j == size_t(-1) || j == 0)
4797 __throw_runtime_error("locale not supported");
4798 wbe = wbuf + j;
4799 __months_[i + 12].assign(wbuf, wbe);
4801 // __am_pm_
4802 t.tm_hour = 1;
4803 __locale::__strftime(buf, countof(buf), "%p", &t, __loc_);
4804 mb = mbstate_t();
4805 const char* bb = buf;
4806 size_t j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, __loc_);
4807 if (j == size_t(-1))
4808 __throw_runtime_error("locale not supported");
4809 wbe = wbuf + j;
4810 __am_pm_[0].assign(wbuf, wbe);
4811 t.tm_hour = 13;
4812 __locale::__strftime(buf, countof(buf), "%p", &t, __loc_);
4813 mb = mbstate_t();
4814 bb = buf;
4815 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, __loc_);
4816 if (j == size_t(-1))
4817 __throw_runtime_error("locale not supported");
4818 wbe = wbuf + j;
4819 __am_pm_[1].assign(wbuf, wbe);
4820 __c_ = __analyze('c', ct);
4821 __r_ = __analyze('r', ct);
4822 __x_ = __analyze('x', ct);
4823 __X_ = __analyze('X', ct);
4825 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
4827 template <class CharT>
4828 struct _LIBCPP_HIDDEN __time_get_temp : public ctype_byname<CharT> {
4829 explicit __time_get_temp(const char* nm) : ctype_byname<CharT>(nm, 1) {}
4830 explicit __time_get_temp(const string& nm) : ctype_byname<CharT>(nm, 1) {}
4833 template <>
4834 __time_get_storage<char>::__time_get_storage(const char* __nm) : __time_get(__nm) {
4835 const __time_get_temp<char> ct(__nm);
4836 init(ct);
4839 template <>
4840 __time_get_storage<char>::__time_get_storage(const string& __nm) : __time_get(__nm) {
4841 const __time_get_temp<char> ct(__nm);
4842 init(ct);
4845 #if _LIBCPP_HAS_WIDE_CHARACTERS
4846 template <>
4847 __time_get_storage<wchar_t>::__time_get_storage(const char* __nm) : __time_get(__nm) {
4848 const __time_get_temp<wchar_t> ct(__nm);
4849 init(ct);
4852 template <>
4853 __time_get_storage<wchar_t>::__time_get_storage(const string& __nm) : __time_get(__nm) {
4854 const __time_get_temp<wchar_t> ct(__nm);
4855 init(ct);
4857 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
4859 template <>
4860 time_base::dateorder __time_get_storage<char>::__do_date_order() const {
4861 unsigned i;
4862 for (i = 0; i < __x_.size(); ++i)
4863 if (__x_[i] == '%')
4864 break;
4865 ++i;
4866 switch (__x_[i]) {
4867 case 'y':
4868 case 'Y':
4869 for (++i; i < __x_.size(); ++i)
4870 if (__x_[i] == '%')
4871 break;
4872 if (i == __x_.size())
4873 break;
4874 ++i;
4875 switch (__x_[i]) {
4876 case 'm':
4877 for (++i; i < __x_.size(); ++i)
4878 if (__x_[i] == '%')
4879 break;
4880 if (i == __x_.size())
4881 break;
4882 ++i;
4883 if (__x_[i] == 'd')
4884 return time_base::ymd;
4885 break;
4886 case 'd':
4887 for (++i; i < __x_.size(); ++i)
4888 if (__x_[i] == '%')
4889 break;
4890 if (i == __x_.size())
4891 break;
4892 ++i;
4893 if (__x_[i] == 'm')
4894 return time_base::ydm;
4895 break;
4897 break;
4898 case 'm':
4899 for (++i; i < __x_.size(); ++i)
4900 if (__x_[i] == '%')
4901 break;
4902 if (i == __x_.size())
4903 break;
4904 ++i;
4905 if (__x_[i] == 'd') {
4906 for (++i; i < __x_.size(); ++i)
4907 if (__x_[i] == '%')
4908 break;
4909 if (i == __x_.size())
4910 break;
4911 ++i;
4912 if (__x_[i] == 'y' || __x_[i] == 'Y')
4913 return time_base::mdy;
4914 break;
4916 break;
4917 case 'd':
4918 for (++i; i < __x_.size(); ++i)
4919 if (__x_[i] == '%')
4920 break;
4921 if (i == __x_.size())
4922 break;
4923 ++i;
4924 if (__x_[i] == 'm') {
4925 for (++i; i < __x_.size(); ++i)
4926 if (__x_[i] == '%')
4927 break;
4928 if (i == __x_.size())
4929 break;
4930 ++i;
4931 if (__x_[i] == 'y' || __x_[i] == 'Y')
4932 return time_base::dmy;
4933 break;
4935 break;
4937 return time_base::no_order;
4940 #if _LIBCPP_HAS_WIDE_CHARACTERS
4941 template <>
4942 time_base::dateorder __time_get_storage<wchar_t>::__do_date_order() const {
4943 unsigned i;
4944 for (i = 0; i < __x_.size(); ++i)
4945 if (__x_[i] == L'%')
4946 break;
4947 ++i;
4948 switch (__x_[i]) {
4949 case L'y':
4950 case L'Y':
4951 for (++i; i < __x_.size(); ++i)
4952 if (__x_[i] == L'%')
4953 break;
4954 if (i == __x_.size())
4955 break;
4956 ++i;
4957 switch (__x_[i]) {
4958 case L'm':
4959 for (++i; i < __x_.size(); ++i)
4960 if (__x_[i] == L'%')
4961 break;
4962 if (i == __x_.size())
4963 break;
4964 ++i;
4965 if (__x_[i] == L'd')
4966 return time_base::ymd;
4967 break;
4968 case L'd':
4969 for (++i; i < __x_.size(); ++i)
4970 if (__x_[i] == L'%')
4971 break;
4972 if (i == __x_.size())
4973 break;
4974 ++i;
4975 if (__x_[i] == L'm')
4976 return time_base::ydm;
4977 break;
4979 break;
4980 case L'm':
4981 for (++i; i < __x_.size(); ++i)
4982 if (__x_[i] == L'%')
4983 break;
4984 if (i == __x_.size())
4985 break;
4986 ++i;
4987 if (__x_[i] == L'd') {
4988 for (++i; i < __x_.size(); ++i)
4989 if (__x_[i] == L'%')
4990 break;
4991 if (i == __x_.size())
4992 break;
4993 ++i;
4994 if (__x_[i] == L'y' || __x_[i] == L'Y')
4995 return time_base::mdy;
4996 break;
4998 break;
4999 case L'd':
5000 for (++i; i < __x_.size(); ++i)
5001 if (__x_[i] == L'%')
5002 break;
5003 if (i == __x_.size())
5004 break;
5005 ++i;
5006 if (__x_[i] == L'm') {
5007 for (++i; i < __x_.size(); ++i)
5008 if (__x_[i] == L'%')
5009 break;
5010 if (i == __x_.size())
5011 break;
5012 ++i;
5013 if (__x_[i] == L'y' || __x_[i] == L'Y')
5014 return time_base::dmy;
5015 break;
5017 break;
5019 return time_base::no_order;
5021 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
5023 // time_put
5025 __time_put::__time_put(const char* nm) : __loc_(__locale::__newlocale(LC_ALL_MASK, nm, 0)) {
5026 if (__loc_ == 0)
5027 __throw_runtime_error(("time_put_byname failed to construct for " + string(nm)).c_str());
5030 __time_put::__time_put(const string& nm) : __loc_(__locale::__newlocale(LC_ALL_MASK, nm.c_str(), 0)) {
5031 if (__loc_ == 0)
5032 __throw_runtime_error(("time_put_byname failed to construct for " + nm).c_str());
5035 __time_put::~__time_put() {
5036 if (__loc_ != _LIBCPP_GET_C_LOCALE)
5037 __locale::__freelocale(__loc_);
5040 void __time_put::__do_put(char* __nb, char*& __ne, const tm* __tm, char __fmt, char __mod) const {
5041 char fmt[] = {'%', __fmt, __mod, 0};
5042 if (__mod != 0)
5043 swap(fmt[1], fmt[2]);
5044 size_t n = __locale::__strftime(__nb, countof(__nb, __ne), fmt, __tm, __loc_);
5045 __ne = __nb + n;
5048 #if _LIBCPP_HAS_WIDE_CHARACTERS
5049 void __time_put::__do_put(wchar_t* __wb, wchar_t*& __we, const tm* __tm, char __fmt, char __mod) const {
5050 char __nar[100];
5051 char* __ne = __nar + 100;
5052 __do_put(__nar, __ne, __tm, __fmt, __mod);
5053 mbstate_t mb = {0};
5054 const char* __nb = __nar;
5055 size_t j = __locale::__mbsrtowcs(__wb, &__nb, countof(__wb, __we), &mb, __loc_);
5056 if (j == size_t(-1))
5057 __throw_runtime_error("locale not supported");
5058 __we = __wb + j;
5060 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
5062 // moneypunct_byname
5064 template <class charT>
5065 static void __init_pat(
5066 money_base::pattern& pat,
5067 basic_string<charT>& __curr_symbol_,
5068 bool intl,
5069 char cs_precedes,
5070 char sep_by_space,
5071 char sign_posn,
5072 charT space_char) {
5073 const char sign = static_cast<char>(money_base::sign);
5074 const char space = static_cast<char>(money_base::space);
5075 const char none = static_cast<char>(money_base::none);
5076 const char symbol = static_cast<char>(money_base::symbol);
5077 const char value = static_cast<char>(money_base::value);
5078 const bool symbol_contains_sep = intl && __curr_symbol_.size() == 4;
5080 // Comments on case branches reflect 'C11 7.11.2.1 The localeconv
5081 // function'. "Space between sign and symbol or value" means that
5082 // if the sign is adjacent to the symbol, there's a space between
5083 // them, and otherwise there's a space between the sign and value.
5085 // C11's localeconv specifies that the fourth character of an
5086 // international curr_symbol is used to separate the sign and
5087 // value when sep_by_space says to do so. C++ can't represent
5088 // that, so we just use a space. When sep_by_space says to
5089 // separate the symbol and value-or-sign with a space, we rearrange the
5090 // curr_symbol to put its spacing character on the correct side of
5091 // the symbol.
5093 // We also need to avoid adding an extra space between the sign
5094 // and value when the currency symbol is suppressed (by not
5095 // setting showbase). We match glibc's strfmon by interpreting
5096 // sep_by_space==1 as "omit the space when the currency symbol is
5097 // absent".
5099 // Users who want to get this right should use ICU instead.
5101 switch (cs_precedes) {
5102 case 0: // value before curr_symbol
5103 if (symbol_contains_sep) {
5104 // Move the separator to before the symbol, to place it
5105 // between the value and symbol.
5106 rotate(__curr_symbol_.begin(), __curr_symbol_.begin() + 3, __curr_symbol_.end());
5108 switch (sign_posn) {
5109 case 0: // Parentheses surround the quantity and currency symbol.
5110 pat.field[0] = sign;
5111 pat.field[1] = value;
5112 pat.field[2] = none; // Any space appears in the symbol.
5113 pat.field[3] = symbol;
5114 switch (sep_by_space) {
5115 case 0: // No space separates the currency symbol and value.
5116 // This case may have changed between C99 and C11;
5117 // assume the currency symbol matches the intention.
5118 case 2: // Space between sign and currency or value.
5119 // The "sign" is two parentheses, so no space here either.
5120 return;
5121 case 1: // Space between currency-and-sign or currency and value.
5122 if (!symbol_contains_sep) {
5123 // We insert the space into the symbol instead of
5124 // setting pat.field[2]=space so that when
5125 // showbase is not set, the space goes away too.
5126 __curr_symbol_.insert(0, 1, space_char);
5128 return;
5129 default:
5130 break;
5132 break;
5133 case 1: // The sign string precedes the quantity and currency symbol.
5134 pat.field[0] = sign;
5135 pat.field[3] = symbol;
5136 switch (sep_by_space) {
5137 case 0: // No space separates the currency symbol and value.
5138 pat.field[1] = value;
5139 pat.field[2] = none;
5140 return;
5141 case 1: // Space between currency-and-sign or currency and value.
5142 pat.field[1] = value;
5143 pat.field[2] = none;
5144 if (!symbol_contains_sep) {
5145 // We insert the space into the symbol instead of
5146 // setting pat.field[2]=space so that when
5147 // showbase is not set, the space goes away too.
5148 __curr_symbol_.insert(0, 1, space_char);
5150 return;
5151 case 2: // Space between sign and currency or value.
5152 pat.field[1] = space;
5153 pat.field[2] = value;
5154 if (symbol_contains_sep) {
5155 // Remove the separator from the symbol, since it
5156 // has already appeared after the sign.
5157 __curr_symbol_.erase(__curr_symbol_.begin());
5159 return;
5160 default:
5161 break;
5163 break;
5164 case 2: // The sign string succeeds the quantity and currency symbol.
5165 pat.field[0] = value;
5166 pat.field[3] = sign;
5167 switch (sep_by_space) {
5168 case 0: // No space separates the currency symbol and value.
5169 pat.field[1] = none;
5170 pat.field[2] = symbol;
5171 return;
5172 case 1: // Space between currency-and-sign or currency and value.
5173 if (!symbol_contains_sep) {
5174 // We insert the space into the symbol instead of
5175 // setting pat.field[1]=space so that when
5176 // showbase is not set, the space goes away too.
5177 __curr_symbol_.insert(0, 1, space_char);
5179 pat.field[1] = none;
5180 pat.field[2] = symbol;
5181 return;
5182 case 2: // Space between sign and currency or value.
5183 pat.field[1] = symbol;
5184 pat.field[2] = space;
5185 if (symbol_contains_sep) {
5186 // Remove the separator from the symbol, since it
5187 // should not be removed if showbase is absent.
5188 __curr_symbol_.erase(__curr_symbol_.begin());
5190 return;
5191 default:
5192 break;
5194 break;
5195 case 3: // The sign string immediately precedes the currency symbol.
5196 pat.field[0] = value;
5197 pat.field[3] = symbol;
5198 switch (sep_by_space) {
5199 case 0: // No space separates the currency symbol and value.
5200 pat.field[1] = none;
5201 pat.field[2] = sign;
5202 return;
5203 case 1: // Space between currency-and-sign or currency and value.
5204 pat.field[1] = space;
5205 pat.field[2] = sign;
5206 if (symbol_contains_sep) {
5207 // Remove the separator from the symbol, since it
5208 // has already appeared before the sign.
5209 __curr_symbol_.erase(__curr_symbol_.begin());
5211 return;
5212 case 2: // Space between sign and currency or value.
5213 pat.field[1] = sign;
5214 pat.field[2] = none;
5215 if (!symbol_contains_sep) {
5216 // We insert the space into the symbol instead of
5217 // setting pat.field[2]=space so that when
5218 // showbase is not set, the space goes away too.
5219 __curr_symbol_.insert(0, 1, space_char);
5221 return;
5222 default:
5223 break;
5225 break;
5226 case 4: // The sign string immediately succeeds the currency symbol.
5227 pat.field[0] = value;
5228 pat.field[3] = sign;
5229 switch (sep_by_space) {
5230 case 0: // No space separates the currency symbol and value.
5231 pat.field[1] = none;
5232 pat.field[2] = symbol;
5233 return;
5234 case 1: // Space between currency-and-sign or currency and value.
5235 pat.field[1] = none;
5236 pat.field[2] = symbol;
5237 if (!symbol_contains_sep) {
5238 // We insert the space into the symbol instead of
5239 // setting pat.field[1]=space so that when
5240 // showbase is not set, the space goes away too.
5241 __curr_symbol_.insert(0, 1, space_char);
5243 return;
5244 case 2: // Space between sign and currency or value.
5245 pat.field[1] = symbol;
5246 pat.field[2] = space;
5247 if (symbol_contains_sep) {
5248 // Remove the separator from the symbol, since it
5249 // should not disappear when showbase is absent.
5250 __curr_symbol_.erase(__curr_symbol_.begin());
5252 return;
5253 default:
5254 break;
5256 break;
5257 default:
5258 break;
5260 break;
5261 case 1: // curr_symbol before value
5262 switch (sign_posn) {
5263 case 0: // Parentheses surround the quantity and currency symbol.
5264 pat.field[0] = sign;
5265 pat.field[1] = symbol;
5266 pat.field[2] = none; // Any space appears in the symbol.
5267 pat.field[3] = value;
5268 switch (sep_by_space) {
5269 case 0: // No space separates the currency symbol and value.
5270 // This case may have changed between C99 and C11;
5271 // assume the currency symbol matches the intention.
5272 case 2: // Space between sign and currency or value.
5273 // The "sign" is two parentheses, so no space here either.
5274 return;
5275 case 1: // Space between currency-and-sign or currency and value.
5276 if (!symbol_contains_sep) {
5277 // We insert the space into the symbol instead of
5278 // setting pat.field[2]=space so that when
5279 // showbase is not set, the space goes away too.
5280 __curr_symbol_.insert(0, 1, space_char);
5282 return;
5283 default:
5284 break;
5286 break;
5287 case 1: // The sign string precedes the quantity and currency symbol.
5288 pat.field[0] = sign;
5289 pat.field[3] = value;
5290 switch (sep_by_space) {
5291 case 0: // No space separates the currency symbol and value.
5292 pat.field[1] = symbol;
5293 pat.field[2] = none;
5294 return;
5295 case 1: // Space between currency-and-sign or currency and value.
5296 pat.field[1] = symbol;
5297 pat.field[2] = none;
5298 if (!symbol_contains_sep) {
5299 // We insert the space into the symbol instead of
5300 // setting pat.field[2]=space so that when
5301 // showbase is not set, the space goes away too.
5302 __curr_symbol_.push_back(space_char);
5304 return;
5305 case 2: // Space between sign and currency or value.
5306 pat.field[1] = space;
5307 pat.field[2] = symbol;
5308 if (symbol_contains_sep) {
5309 // Remove the separator from the symbol, since it
5310 // has already appeared after the sign.
5311 __curr_symbol_.pop_back();
5313 return;
5314 default:
5315 break;
5317 break;
5318 case 2: // The sign string succeeds the quantity and currency symbol.
5319 pat.field[0] = symbol;
5320 pat.field[3] = sign;
5321 switch (sep_by_space) {
5322 case 0: // No space separates the currency symbol and value.
5323 pat.field[1] = none;
5324 pat.field[2] = value;
5325 return;
5326 case 1: // Space between currency-and-sign or currency and value.
5327 pat.field[1] = none;
5328 pat.field[2] = value;
5329 if (!symbol_contains_sep) {
5330 // We insert the space into the symbol instead of
5331 // setting pat.field[1]=space so that when
5332 // showbase is not set, the space goes away too.
5333 __curr_symbol_.push_back(space_char);
5335 return;
5336 case 2: // Space between sign and currency or value.
5337 pat.field[1] = value;
5338 pat.field[2] = space;
5339 if (symbol_contains_sep) {
5340 // Remove the separator from the symbol, since it
5341 // will appear before the sign.
5342 __curr_symbol_.pop_back();
5344 return;
5345 default:
5346 break;
5348 break;
5349 case 3: // The sign string immediately precedes the currency symbol.
5350 pat.field[0] = sign;
5351 pat.field[3] = value;
5352 switch (sep_by_space) {
5353 case 0: // No space separates the currency symbol and value.
5354 pat.field[1] = symbol;
5355 pat.field[2] = none;
5356 return;
5357 case 1: // Space between currency-and-sign or currency and value.
5358 pat.field[1] = symbol;
5359 pat.field[2] = none;
5360 if (!symbol_contains_sep) {
5361 // We insert the space into the symbol instead of
5362 // setting pat.field[2]=space so that when
5363 // showbase is not set, the space goes away too.
5364 __curr_symbol_.push_back(space_char);
5366 return;
5367 case 2: // Space between sign and currency or value.
5368 pat.field[1] = space;
5369 pat.field[2] = symbol;
5370 if (symbol_contains_sep) {
5371 // Remove the separator from the symbol, since it
5372 // has already appeared after the sign.
5373 __curr_symbol_.pop_back();
5375 return;
5376 default:
5377 break;
5379 break;
5380 case 4: // The sign string immediately succeeds the currency symbol.
5381 pat.field[0] = symbol;
5382 pat.field[3] = value;
5383 switch (sep_by_space) {
5384 case 0: // No space separates the currency symbol and value.
5385 pat.field[1] = sign;
5386 pat.field[2] = none;
5387 return;
5388 case 1: // Space between currency-and-sign or currency and value.
5389 pat.field[1] = sign;
5390 pat.field[2] = space;
5391 if (symbol_contains_sep) {
5392 // Remove the separator from the symbol, since it
5393 // should not disappear when showbase is absent.
5394 __curr_symbol_.pop_back();
5396 return;
5397 case 2: // Space between sign and currency or value.
5398 pat.field[1] = none;
5399 pat.field[2] = sign;
5400 if (!symbol_contains_sep) {
5401 // We insert the space into the symbol instead of
5402 // setting pat.field[1]=space so that when
5403 // showbase is not set, the space goes away too.
5404 __curr_symbol_.push_back(space_char);
5406 return;
5407 default:
5408 break;
5410 break;
5411 default:
5412 break;
5414 break;
5415 default:
5416 break;
5418 pat.field[0] = symbol;
5419 pat.field[1] = sign;
5420 pat.field[2] = none;
5421 pat.field[3] = value;
5424 template <>
5425 void moneypunct_byname<char, false>::init(const char* nm) {
5426 typedef moneypunct<char, false> base;
5427 __libcpp_unique_locale loc(nm);
5428 if (!loc)
5429 __throw_runtime_error(("moneypunct_byname failed to construct for " + string(nm)).c_str());
5431 lconv* lc = __locale::__localeconv(loc.get());
5432 if (!checked_string_to_char_convert(__decimal_point_, lc->mon_decimal_point, loc.get()))
5433 __decimal_point_ = base::do_decimal_point();
5434 if (!checked_string_to_char_convert(__thousands_sep_, lc->mon_thousands_sep, loc.get()))
5435 __thousands_sep_ = base::do_thousands_sep();
5437 __grouping_ = lc->mon_grouping;
5438 __curr_symbol_ = lc->currency_symbol;
5439 if (lc->frac_digits != CHAR_MAX)
5440 __frac_digits_ = lc->frac_digits;
5441 else
5442 __frac_digits_ = base::do_frac_digits();
5443 if (lc->p_sign_posn == 0)
5444 __positive_sign_ = "()";
5445 else
5446 __positive_sign_ = lc->positive_sign;
5447 if (lc->n_sign_posn == 0)
5448 __negative_sign_ = "()";
5449 else
5450 __negative_sign_ = lc->negative_sign;
5451 // Assume the positive and negative formats will want spaces in
5452 // the same places in curr_symbol since there's no way to
5453 // represent anything else.
5454 string_type __dummy_curr_symbol = __curr_symbol_;
5455 __init_pat(__pos_format_, __dummy_curr_symbol, false, lc->p_cs_precedes, lc->p_sep_by_space, lc->p_sign_posn, ' ');
5456 __init_pat(__neg_format_, __curr_symbol_, false, lc->n_cs_precedes, lc->n_sep_by_space, lc->n_sign_posn, ' ');
5459 template <>
5460 void moneypunct_byname<char, true>::init(const char* nm) {
5461 typedef moneypunct<char, true> base;
5462 __libcpp_unique_locale loc(nm);
5463 if (!loc)
5464 __throw_runtime_error(("moneypunct_byname failed to construct for " + string(nm)).c_str());
5466 lconv* lc = __locale::__localeconv(loc.get());
5467 if (!checked_string_to_char_convert(__decimal_point_, lc->mon_decimal_point, loc.get()))
5468 __decimal_point_ = base::do_decimal_point();
5469 if (!checked_string_to_char_convert(__thousands_sep_, lc->mon_thousands_sep, loc.get()))
5470 __thousands_sep_ = base::do_thousands_sep();
5471 __grouping_ = lc->mon_grouping;
5472 __curr_symbol_ = lc->int_curr_symbol;
5473 if (lc->int_frac_digits != CHAR_MAX)
5474 __frac_digits_ = lc->int_frac_digits;
5475 else
5476 __frac_digits_ = base::do_frac_digits();
5477 #if defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
5478 if (lc->p_sign_posn == 0)
5479 #else // _LIBCPP_MSVCRT
5480 if (lc->int_p_sign_posn == 0)
5481 #endif // !_LIBCPP_MSVCRT
5482 __positive_sign_ = "()";
5483 else
5484 __positive_sign_ = lc->positive_sign;
5485 #if defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
5486 if (lc->n_sign_posn == 0)
5487 #else // _LIBCPP_MSVCRT
5488 if (lc->int_n_sign_posn == 0)
5489 #endif // !_LIBCPP_MSVCRT
5490 __negative_sign_ = "()";
5491 else
5492 __negative_sign_ = lc->negative_sign;
5493 // Assume the positive and negative formats will want spaces in
5494 // the same places in curr_symbol since there's no way to
5495 // represent anything else.
5496 string_type __dummy_curr_symbol = __curr_symbol_;
5497 #if defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
5498 __init_pat(__pos_format_, __dummy_curr_symbol, true, lc->p_cs_precedes, lc->p_sep_by_space, lc->p_sign_posn, ' ');
5499 __init_pat(__neg_format_, __curr_symbol_, true, lc->n_cs_precedes, lc->n_sep_by_space, lc->n_sign_posn, ' ');
5500 #else // _LIBCPP_MSVCRT
5501 __init_pat(
5502 __pos_format_,
5503 __dummy_curr_symbol,
5504 true,
5505 lc->int_p_cs_precedes,
5506 lc->int_p_sep_by_space,
5507 lc->int_p_sign_posn,
5508 ' ');
5509 __init_pat(
5510 __neg_format_, __curr_symbol_, true, lc->int_n_cs_precedes, lc->int_n_sep_by_space, lc->int_n_sign_posn, ' ');
5511 #endif // !_LIBCPP_MSVCRT
5514 #if _LIBCPP_HAS_WIDE_CHARACTERS
5515 template <>
5516 void moneypunct_byname<wchar_t, false>::init(const char* nm) {
5517 typedef moneypunct<wchar_t, false> base;
5518 __libcpp_unique_locale loc(nm);
5519 if (!loc)
5520 __throw_runtime_error(("moneypunct_byname failed to construct for " + string(nm)).c_str());
5521 lconv* lc = __locale::__localeconv(loc.get());
5522 if (!checked_string_to_wchar_convert(__decimal_point_, lc->mon_decimal_point, loc.get()))
5523 __decimal_point_ = base::do_decimal_point();
5524 if (!checked_string_to_wchar_convert(__thousands_sep_, lc->mon_thousands_sep, loc.get()))
5525 __thousands_sep_ = base::do_thousands_sep();
5526 __grouping_ = lc->mon_grouping;
5527 wchar_t wbuf[100];
5528 mbstate_t mb = {0};
5529 const char* bb = lc->currency_symbol;
5530 size_t j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, loc.get());
5531 if (j == size_t(-1))
5532 __throw_runtime_error("locale not supported");
5533 wchar_t* wbe = wbuf + j;
5534 __curr_symbol_.assign(wbuf, wbe);
5535 if (lc->frac_digits != CHAR_MAX)
5536 __frac_digits_ = lc->frac_digits;
5537 else
5538 __frac_digits_ = base::do_frac_digits();
5539 if (lc->p_sign_posn == 0)
5540 __positive_sign_ = L"()";
5541 else {
5542 mb = mbstate_t();
5543 bb = lc->positive_sign;
5544 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, loc.get());
5545 if (j == size_t(-1))
5546 __throw_runtime_error("locale not supported");
5547 wbe = wbuf + j;
5548 __positive_sign_.assign(wbuf, wbe);
5550 if (lc->n_sign_posn == 0)
5551 __negative_sign_ = L"()";
5552 else {
5553 mb = mbstate_t();
5554 bb = lc->negative_sign;
5555 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, loc.get());
5556 if (j == size_t(-1))
5557 __throw_runtime_error("locale not supported");
5558 wbe = wbuf + j;
5559 __negative_sign_.assign(wbuf, wbe);
5561 // Assume the positive and negative formats will want spaces in
5562 // the same places in curr_symbol since there's no way to
5563 // represent anything else.
5564 string_type __dummy_curr_symbol = __curr_symbol_;
5565 __init_pat(__pos_format_, __dummy_curr_symbol, false, lc->p_cs_precedes, lc->p_sep_by_space, lc->p_sign_posn, L' ');
5566 __init_pat(__neg_format_, __curr_symbol_, false, lc->n_cs_precedes, lc->n_sep_by_space, lc->n_sign_posn, L' ');
5569 template <>
5570 void moneypunct_byname<wchar_t, true>::init(const char* nm) {
5571 typedef moneypunct<wchar_t, true> base;
5572 __libcpp_unique_locale loc(nm);
5573 if (!loc)
5574 __throw_runtime_error(("moneypunct_byname failed to construct for " + string(nm)).c_str());
5576 lconv* lc = __locale::__localeconv(loc.get());
5577 if (!checked_string_to_wchar_convert(__decimal_point_, lc->mon_decimal_point, loc.get()))
5578 __decimal_point_ = base::do_decimal_point();
5579 if (!checked_string_to_wchar_convert(__thousands_sep_, lc->mon_thousands_sep, loc.get()))
5580 __thousands_sep_ = base::do_thousands_sep();
5581 __grouping_ = lc->mon_grouping;
5582 wchar_t wbuf[100];
5583 mbstate_t mb = {0};
5584 const char* bb = lc->int_curr_symbol;
5585 size_t j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, loc.get());
5586 if (j == size_t(-1))
5587 __throw_runtime_error("locale not supported");
5588 wchar_t* wbe = wbuf + j;
5589 __curr_symbol_.assign(wbuf, wbe);
5590 if (lc->int_frac_digits != CHAR_MAX)
5591 __frac_digits_ = lc->int_frac_digits;
5592 else
5593 __frac_digits_ = base::do_frac_digits();
5594 # if defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
5595 if (lc->p_sign_posn == 0)
5596 # else // _LIBCPP_MSVCRT
5597 if (lc->int_p_sign_posn == 0)
5598 # endif // !_LIBCPP_MSVCRT
5599 __positive_sign_ = L"()";
5600 else {
5601 mb = mbstate_t();
5602 bb = lc->positive_sign;
5603 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, loc.get());
5604 if (j == size_t(-1))
5605 __throw_runtime_error("locale not supported");
5606 wbe = wbuf + j;
5607 __positive_sign_.assign(wbuf, wbe);
5609 # if defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
5610 if (lc->n_sign_posn == 0)
5611 # else // _LIBCPP_MSVCRT
5612 if (lc->int_n_sign_posn == 0)
5613 # endif // !_LIBCPP_MSVCRT
5614 __negative_sign_ = L"()";
5615 else {
5616 mb = mbstate_t();
5617 bb = lc->negative_sign;
5618 j = __locale::__mbsrtowcs(wbuf, &bb, countof(wbuf), &mb, loc.get());
5619 if (j == size_t(-1))
5620 __throw_runtime_error("locale not supported");
5621 wbe = wbuf + j;
5622 __negative_sign_.assign(wbuf, wbe);
5624 // Assume the positive and negative formats will want spaces in
5625 // the same places in curr_symbol since there's no way to
5626 // represent anything else.
5627 string_type __dummy_curr_symbol = __curr_symbol_;
5628 # if defined(_LIBCPP_MSVCRT) || defined(__MINGW32__)
5629 __init_pat(__pos_format_, __dummy_curr_symbol, true, lc->p_cs_precedes, lc->p_sep_by_space, lc->p_sign_posn, L' ');
5630 __init_pat(__neg_format_, __curr_symbol_, true, lc->n_cs_precedes, lc->n_sep_by_space, lc->n_sign_posn, L' ');
5631 # else // _LIBCPP_MSVCRT
5632 __init_pat(
5633 __pos_format_,
5634 __dummy_curr_symbol,
5635 true,
5636 lc->int_p_cs_precedes,
5637 lc->int_p_sep_by_space,
5638 lc->int_p_sign_posn,
5639 L' ');
5640 __init_pat(
5641 __neg_format_, __curr_symbol_, true, lc->int_n_cs_precedes, lc->int_n_sep_by_space, lc->int_n_sign_posn, L' ');
5642 # endif // !_LIBCPP_MSVCRT
5644 #endif // _LIBCPP_HAS_WIDE_CHARACTERS
5646 void __do_nothing(void*) {}
5648 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS collate<char>;
5649 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS collate<wchar_t>;)
5651 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS num_get<char>;
5652 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS num_get<wchar_t>;)
5654 template struct _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __num_get<char>;
5655 _LIBCPP_IF_WIDE_CHARACTERS(template struct _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __num_get<wchar_t>;)
5657 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS num_put<char>;
5658 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS num_put<wchar_t>;)
5660 template struct _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __num_put<char>;
5661 _LIBCPP_IF_WIDE_CHARACTERS(template struct _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __num_put<wchar_t>;)
5663 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_get<char>;
5664 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_get<wchar_t>;)
5666 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_get_byname<char>;
5667 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_get_byname<wchar_t>;)
5669 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_put<char>;
5670 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_put<wchar_t>;)
5672 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_put_byname<char>;
5673 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS time_put_byname<wchar_t>;)
5675 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct<char, false>;
5676 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct<char, true>;
5677 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct<wchar_t, false>;)
5678 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct<wchar_t, true>;)
5680 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct_byname<char, false>;
5681 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct_byname<char, true>;
5682 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct_byname<wchar_t, false>;)
5683 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS moneypunct_byname<wchar_t, true>;)
5685 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS money_get<char>;
5686 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS money_get<wchar_t>;)
5688 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __money_get<char>;
5689 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __money_get<wchar_t>;)
5691 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS money_put<char>;
5692 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS money_put<wchar_t>;)
5694 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __money_put<char>;
5695 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS __money_put<wchar_t>;)
5697 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS messages<char>;
5698 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS messages<wchar_t>;)
5700 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS messages_byname<char>;
5701 _LIBCPP_IF_WIDE_CHARACTERS(template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS messages_byname<wchar_t>;)
5703 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS codecvt_byname<char, char, mbstate_t>;
5704 _LIBCPP_IF_WIDE_CHARACTERS(
5705 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS codecvt_byname<wchar_t, char, mbstate_t>;)
5706 template class _LIBCPP_DEPRECATED_IN_CXX20 _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS
5707 codecvt_byname<char16_t, char, mbstate_t>;
5708 template class _LIBCPP_DEPRECATED_IN_CXX20 _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS
5709 codecvt_byname<char32_t, char, mbstate_t>;
5710 #if _LIBCPP_HAS_CHAR8_T
5711 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS codecvt_byname<char16_t, char8_t, mbstate_t>;
5712 template class _LIBCPP_CLASS_TEMPLATE_INSTANTIATION_VIS codecvt_byname<char32_t, char8_t, mbstate_t>;
5713 #endif
5715 _LIBCPP_END_NAMESPACE_STD
5717 _LIBCPP_POP_MACROS