xref: /netbsd-src/external/gpl3/gcc/dist/libstdc++-v3/include/ext/string_conversions.h (revision b1e838363e3c6fc78a55519254d99869742dd33c)
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