14fee23f9Smrg // String Conversions -*- C++ -*-
24fee23f9Smrg
3*b1e83836Smrg // Copyright (C) 2008-2022 Free Software Foundation, Inc.
44fee23f9Smrg //
54fee23f9Smrg // This file is part of the GNU ISO C++ Library. This library is free
64fee23f9Smrg // software; you can redistribute it and/or modify it under the
74fee23f9Smrg // terms of the GNU General Public License as published by the
84fee23f9Smrg // Free Software Foundation; either version 3, or (at your option)
94fee23f9Smrg // any later version.
104fee23f9Smrg
114fee23f9Smrg // This library is distributed in the hope that it will be useful,
124fee23f9Smrg // but WITHOUT ANY WARRANTY; without even the implied warranty of
134fee23f9Smrg // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
144fee23f9Smrg // GNU General Public License for more details.
154fee23f9Smrg
164fee23f9Smrg // Under Section 7 of GPL version 3, you are granted additional
174fee23f9Smrg // permissions described in the GCC Runtime Library Exception, version
184fee23f9Smrg // 3.1, as published by the Free Software Foundation.
194fee23f9Smrg
204fee23f9Smrg // You should have received a copy of the GNU General Public License and
214fee23f9Smrg // a copy of the GCC Runtime Library Exception along with this program;
224fee23f9Smrg // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
234fee23f9Smrg // <http://www.gnu.org/licenses/>.
244fee23f9Smrg
2548fb7bfaSmrg /** @file ext/string_conversions.h
2648fb7bfaSmrg * This file is a GNU extension to the Standard C++ Library.
2748fb7bfaSmrg */
2848fb7bfaSmrg
294fee23f9Smrg #ifndef _STRING_CONVERSIONS_H
304fee23f9Smrg #define _STRING_CONVERSIONS_H 1
314fee23f9Smrg
324fee23f9Smrg #pragma GCC system_header
334fee23f9Smrg
3448fb7bfaSmrg #if __cplusplus < 201103L
3548fb7bfaSmrg # include <bits/c++0x_warning.h>
3648fb7bfaSmrg #else
3748fb7bfaSmrg
3848fb7bfaSmrg #include <bits/c++config.h>
394fee23f9Smrg #include <ext/numeric_traits.h>
404fee23f9Smrg #include <bits/functexcept.h>
414fee23f9Smrg #include <cstdlib>
424fee23f9Smrg #include <cwchar>
434fee23f9Smrg #include <cstdio>
444fee23f9Smrg #include <cerrno>
454fee23f9Smrg
_GLIBCXX_VISIBILITY(default)4648fb7bfaSmrg namespace __gnu_cxx _GLIBCXX_VISIBILITY(default)
4748fb7bfaSmrg {
4848fb7bfaSmrg _GLIBCXX_BEGIN_NAMESPACE_VERSION
494fee23f9Smrg
504fee23f9Smrg // Helper for all the sto* functions.
514fee23f9Smrg template<typename _TRet, typename _Ret = _TRet, typename _CharT,
524fee23f9Smrg typename... _Base>
534fee23f9Smrg _Ret
544fee23f9Smrg __stoa(_TRet (*__convf) (const _CharT*, _CharT**, _Base...),
554fee23f9Smrg const char* __name, const _CharT* __str, std::size_t* __idx,
564fee23f9Smrg _Base... __base)
574fee23f9Smrg {
584fee23f9Smrg _Ret __ret;
594fee23f9Smrg
604fee23f9Smrg _CharT* __endptr;
61f9a78e0eSmrg
62f9a78e0eSmrg struct _Save_errno {
63f9a78e0eSmrg _Save_errno() : _M_errno(errno) { errno = 0; }
64f9a78e0eSmrg ~_Save_errno() { if (errno == 0) errno = _M_errno; }
65f9a78e0eSmrg int _M_errno;
66f9a78e0eSmrg } const __save_errno;
67f9a78e0eSmrg
68b17d1066Smrg struct _Range_chk {
69b17d1066Smrg static bool
70b17d1066Smrg _S_chk(_TRet, std::false_type) { return false; }
71b17d1066Smrg
72b17d1066Smrg static bool
73b17d1066Smrg _S_chk(_TRet __val, std::true_type) // only called when _Ret is int
74b17d1066Smrg {
75b17d1066Smrg return __val < _TRet(__numeric_traits<int>::__min)
76b17d1066Smrg || __val > _TRet(__numeric_traits<int>::__max);
77b17d1066Smrg }
78b17d1066Smrg };
79b17d1066Smrg
804fee23f9Smrg const _TRet __tmp = __convf(__str, &__endptr, __base...);
814fee23f9Smrg
824fee23f9Smrg if (__endptr == __str)
834fee23f9Smrg std::__throw_invalid_argument(__name);
844fee23f9Smrg else if (errno == ERANGE
85b17d1066Smrg || _Range_chk::_S_chk(__tmp, std::is_same<_Ret, int>{}))
864fee23f9Smrg std::__throw_out_of_range(__name);
874fee23f9Smrg else
884fee23f9Smrg __ret = __tmp;
894fee23f9Smrg
904fee23f9Smrg if (__idx)
914fee23f9Smrg *__idx = __endptr - __str;
924fee23f9Smrg
934fee23f9Smrg return __ret;
944fee23f9Smrg }
954fee23f9Smrg
964fee23f9Smrg // Helper for the to_string / to_wstring functions.
974fee23f9Smrg template<typename _String, typename _CharT = typename _String::value_type>
984fee23f9Smrg _String
994fee23f9Smrg __to_xstring(int (*__convf) (_CharT*, std::size_t, const _CharT*,
1004fee23f9Smrg __builtin_va_list), std::size_t __n,
1014fee23f9Smrg const _CharT* __fmt, ...)
1024fee23f9Smrg {
103b17d1066Smrg // XXX Eventually the result should be constructed in-place in
104b17d1066Smrg // the __cxx11 string, likely with the help of internal hooks.
1054fee23f9Smrg _CharT* __s = static_cast<_CharT*>(__builtin_alloca(sizeof(_CharT)
1064fee23f9Smrg * __n));
1074fee23f9Smrg
1084fee23f9Smrg __builtin_va_list __args;
1094fee23f9Smrg __builtin_va_start(__args, __fmt);
1104fee23f9Smrg
1114fee23f9Smrg const int __len = __convf(__s, __n, __fmt, __args);
1124fee23f9Smrg
1134fee23f9Smrg __builtin_va_end(__args);
1144fee23f9Smrg
1154fee23f9Smrg return _String(__s, __s + __len);
1164fee23f9Smrg }
1174fee23f9Smrg
11848fb7bfaSmrg _GLIBCXX_END_NAMESPACE_VERSION
11948fb7bfaSmrg } // namespace
12048fb7bfaSmrg
12148fb7bfaSmrg #endif // C++11
1224fee23f9Smrg
1234fee23f9Smrg #endif // _STRING_CONVERSIONS_H
124