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 | //===----------------------------------------------------------------------===// |
8 | |
9 | #include <__assert> |
10 | #include <cerrno> |
11 | #include <charconv> |
12 | #include <cstdlib> |
13 | #include <limits> |
14 | #include <stdexcept> |
15 | #include <string> |
16 | |
17 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
18 | # include <cwchar> |
19 | #endif |
20 | |
21 | _LIBCPP_BEGIN_NAMESPACE_STD |
22 | |
23 | #ifndef _LIBCPP_ABI_DO_NOT_EXPORT_BASIC_STRING_COMMON |
24 | |
25 | template <bool> |
26 | struct __basic_string_common; |
27 | |
28 | // The struct isn't declared anymore in the headers. It's only here for ABI compatibility. |
29 | template <> |
30 | struct __basic_string_common<true> { |
31 | _LIBCPP_NORETURN _LIBCPP_EXPORTED_FROM_ABI void __throw_length_error() const; |
32 | _LIBCPP_NORETURN _LIBCPP_EXPORTED_FROM_ABI void __throw_out_of_range() const; |
33 | }; |
34 | |
35 | void __basic_string_common<true>::__throw_length_error() const { std::__throw_length_error(msg: "basic_string" ); } |
36 | void __basic_string_common<true>::__throw_out_of_range() const { std::__throw_out_of_range(msg: "basic_string" ); } |
37 | |
38 | #endif // _LIBCPP_ABI_DO_NOT_EXPORT_BASIC_STRING_COMMON |
39 | |
40 | #define _LIBCPP_EXTERN_TEMPLATE_DEFINE(...) template __VA_ARGS__; |
41 | #ifdef _LIBCPP_ABI_STRING_OPTIMIZED_EXTERNAL_INSTANTIATION |
42 | _LIBCPP_STRING_UNSTABLE_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE_DEFINE, char) |
43 | # ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
44 | _LIBCPP_STRING_UNSTABLE_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE_DEFINE, wchar_t) |
45 | # endif |
46 | #else |
47 | _LIBCPP_STRING_V1_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE_DEFINE, char) |
48 | # ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
49 | _LIBCPP_STRING_V1_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE_DEFINE, wchar_t) |
50 | # endif |
51 | #endif |
52 | #undef _LIBCPP_EXTERN_TEMPLATE_DEFINE |
53 | |
54 | template string operator+ <char, char_traits<char>, allocator<char>>(char const*, string const&); |
55 | |
56 | namespace { |
57 | |
58 | inline void throw_from_string_out_of_range(const string& func) { |
59 | std::__throw_out_of_range(msg: (func + ": out of range" ).c_str()); |
60 | } |
61 | |
62 | inline void throw_from_string_invalid_arg(const string& func) { |
63 | std::__throw_invalid_argument(msg: (func + ": no conversion" ).c_str()); |
64 | } |
65 | |
66 | // as_integer |
67 | |
68 | template <typename V, typename S, typename F> |
69 | inline V as_integer_helper(const string& func, const S& str, size_t* idx, int base, F f) { |
70 | typename S::value_type* ptr = nullptr; |
71 | const typename S::value_type* const p = str.c_str(); |
72 | __libcpp_remove_reference_t<decltype(errno)> errno_save = errno; |
73 | errno = 0; |
74 | V r = f(p, &ptr, base); |
75 | swap(errno, y&: errno_save); |
76 | if (errno_save == ERANGE) |
77 | throw_from_string_out_of_range(func); |
78 | if (ptr == p) |
79 | throw_from_string_invalid_arg(func); |
80 | if (idx) |
81 | *idx = static_cast<size_t>(ptr - p); |
82 | return r; |
83 | } |
84 | |
85 | template <typename V, typename S> |
86 | inline V as_integer(const string& func, const S& s, size_t* idx, int base); |
87 | |
88 | // string |
89 | template <> |
90 | inline int as_integer(const string& func, const string& s, size_t* idx, int base) { |
91 | // Use long as no Standard string to integer exists. |
92 | long r = as_integer_helper<long>(func, str: s, idx, base, f: strtol); |
93 | if (r < numeric_limits<int>::min() || numeric_limits<int>::max() < r) |
94 | throw_from_string_out_of_range(func); |
95 | return static_cast<int>(r); |
96 | } |
97 | |
98 | template <> |
99 | inline long as_integer(const string& func, const string& s, size_t* idx, int base) { |
100 | return as_integer_helper<long>(func, str: s, idx, base, f: strtol); |
101 | } |
102 | |
103 | template <> |
104 | inline unsigned long as_integer(const string& func, const string& s, size_t* idx, int base) { |
105 | return as_integer_helper<unsigned long>(func, str: s, idx, base, f: strtoul); |
106 | } |
107 | |
108 | template <> |
109 | inline long long as_integer(const string& func, const string& s, size_t* idx, int base) { |
110 | return as_integer_helper<long long>(func, str: s, idx, base, f: strtoll); |
111 | } |
112 | |
113 | template <> |
114 | inline unsigned long long as_integer(const string& func, const string& s, size_t* idx, int base) { |
115 | return as_integer_helper<unsigned long long>(func, str: s, idx, base, f: strtoull); |
116 | } |
117 | |
118 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
119 | // wstring |
120 | template <> |
121 | inline int as_integer(const string& func, const wstring& s, size_t* idx, int base) { |
122 | // Use long as no Stantard string to integer exists. |
123 | long r = as_integer_helper<long>(func, str: s, idx, base, f: wcstol); |
124 | if (r < numeric_limits<int>::min() || numeric_limits<int>::max() < r) |
125 | throw_from_string_out_of_range(func); |
126 | return static_cast<int>(r); |
127 | } |
128 | |
129 | template <> |
130 | inline long as_integer(const string& func, const wstring& s, size_t* idx, int base) { |
131 | return as_integer_helper<long>(func, str: s, idx, base, f: wcstol); |
132 | } |
133 | |
134 | template <> |
135 | inline unsigned long as_integer(const string& func, const wstring& s, size_t* idx, int base) { |
136 | return as_integer_helper<unsigned long>(func, str: s, idx, base, f: wcstoul); |
137 | } |
138 | |
139 | template <> |
140 | inline long long as_integer(const string& func, const wstring& s, size_t* idx, int base) { |
141 | return as_integer_helper<long long>(func, str: s, idx, base, f: wcstoll); |
142 | } |
143 | |
144 | template <> |
145 | inline unsigned long long as_integer(const string& func, const wstring& s, size_t* idx, int base) { |
146 | return as_integer_helper<unsigned long long>(func, str: s, idx, base, f: wcstoull); |
147 | } |
148 | #endif // _LIBCPP_HAS_NO_WIDE_CHARACTERS |
149 | |
150 | // as_float |
151 | |
152 | template <typename V, typename S, typename F> |
153 | inline V as_float_helper(const string& func, const S& str, size_t* idx, F f) { |
154 | typename S::value_type* ptr = nullptr; |
155 | const typename S::value_type* const p = str.c_str(); |
156 | __libcpp_remove_reference_t<decltype(errno)> errno_save = errno; |
157 | errno = 0; |
158 | V r = f(p, &ptr); |
159 | swap(errno, y&: errno_save); |
160 | if (errno_save == ERANGE) |
161 | throw_from_string_out_of_range(func); |
162 | if (ptr == p) |
163 | throw_from_string_invalid_arg(func); |
164 | if (idx) |
165 | *idx = static_cast<size_t>(ptr - p); |
166 | return r; |
167 | } |
168 | |
169 | template <typename V, typename S> |
170 | inline V as_float(const string& func, const S& s, size_t* idx = nullptr); |
171 | |
172 | template <> |
173 | inline float as_float(const string& func, const string& s, size_t* idx) { |
174 | return as_float_helper<float>(func, str: s, idx, f: strtof); |
175 | } |
176 | |
177 | template <> |
178 | inline double as_float(const string& func, const string& s, size_t* idx) { |
179 | return as_float_helper<double>(func, str: s, idx, f: strtod); |
180 | } |
181 | |
182 | template <> |
183 | inline long double as_float(const string& func, const string& s, size_t* idx) { |
184 | return as_float_helper<long double>(func, str: s, idx, f: strtold); |
185 | } |
186 | |
187 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
188 | template <> |
189 | inline float as_float(const string& func, const wstring& s, size_t* idx) { |
190 | return as_float_helper<float>(func, str: s, idx, f: wcstof); |
191 | } |
192 | |
193 | template <> |
194 | inline double as_float(const string& func, const wstring& s, size_t* idx) { |
195 | return as_float_helper<double>(func, str: s, idx, f: wcstod); |
196 | } |
197 | |
198 | template <> |
199 | inline long double as_float(const string& func, const wstring& s, size_t* idx) { |
200 | return as_float_helper<long double>(func, str: s, idx, f: wcstold); |
201 | } |
202 | #endif // _LIBCPP_HAS_NO_WIDE_CHARACTERS |
203 | |
204 | } // unnamed namespace |
205 | |
206 | int stoi(const string& str, size_t* idx, int base) { return as_integer<int>(func: "stoi" , s: str, idx, base); } |
207 | |
208 | long stol(const string& str, size_t* idx, int base) { return as_integer<long>(func: "stol" , s: str, idx, base); } |
209 | |
210 | unsigned long stoul(const string& str, size_t* idx, int base) { |
211 | return as_integer<unsigned long>(func: "stoul" , s: str, idx, base); |
212 | } |
213 | |
214 | long long stoll(const string& str, size_t* idx, int base) { return as_integer<long long>(func: "stoll" , s: str, idx, base); } |
215 | |
216 | unsigned long long stoull(const string& str, size_t* idx, int base) { |
217 | return as_integer<unsigned long long>(func: "stoull" , s: str, idx, base); |
218 | } |
219 | |
220 | float stof(const string& str, size_t* idx) { return as_float<float>(func: "stof" , s: str, idx); } |
221 | |
222 | double stod(const string& str, size_t* idx) { return as_float<double>(func: "stod" , s: str, idx); } |
223 | |
224 | long double stold(const string& str, size_t* idx) { return as_float<long double>(func: "stold" , s: str, idx); } |
225 | |
226 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
227 | int stoi(const wstring& str, size_t* idx, int base) { return as_integer<int>(func: "stoi" , s: str, idx, base); } |
228 | |
229 | long stol(const wstring& str, size_t* idx, int base) { return as_integer<long>(func: "stol" , s: str, idx, base); } |
230 | |
231 | unsigned long stoul(const wstring& str, size_t* idx, int base) { |
232 | return as_integer<unsigned long>(func: "stoul" , s: str, idx, base); |
233 | } |
234 | |
235 | long long stoll(const wstring& str, size_t* idx, int base) { return as_integer<long long>(func: "stoll" , s: str, idx, base); } |
236 | |
237 | unsigned long long stoull(const wstring& str, size_t* idx, int base) { |
238 | return as_integer<unsigned long long>(func: "stoull" , s: str, idx, base); |
239 | } |
240 | |
241 | float stof(const wstring& str, size_t* idx) { return as_float<float>(func: "stof" , s: str, idx); } |
242 | |
243 | double stod(const wstring& str, size_t* idx) { return as_float<double>(func: "stod" , s: str, idx); } |
244 | |
245 | long double stold(const wstring& str, size_t* idx) { return as_float<long double>(func: "stold" , s: str, idx); } |
246 | #endif // !_LIBCPP_HAS_NO_WIDE_CHARACTERS |
247 | |
248 | // to_string |
249 | |
250 | namespace { |
251 | |
252 | // as_string |
253 | |
254 | template <typename S, typename P, typename V > |
255 | inline S as_string(P sprintf_like, S s, const typename S::value_type* fmt, V a) { |
256 | typedef typename S::size_type size_type; |
257 | size_type available = s.size(); |
258 | while (true) { |
259 | int status = sprintf_like(&s[0], available + 1, fmt, a); |
260 | if (status >= 0) { |
261 | size_type used = static_cast<size_type>(status); |
262 | if (used <= available) { |
263 | s.resize(used); |
264 | break; |
265 | } |
266 | available = used; // Assume this is advice of how much space we need. |
267 | } else |
268 | available = available * 2 + 1; |
269 | s.resize(available); |
270 | } |
271 | return s; |
272 | } |
273 | |
274 | template <class S> |
275 | struct initial_string; |
276 | |
277 | template <> |
278 | struct initial_string<string> { |
279 | string operator()() const { |
280 | string s; |
281 | s.resize(n: s.capacity()); |
282 | return s; |
283 | } |
284 | }; |
285 | |
286 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
287 | template <> |
288 | struct initial_string<wstring> { |
289 | wstring operator()() const { |
290 | wstring s(20, wchar_t()); |
291 | s.resize(n: s.capacity()); |
292 | return s; |
293 | } |
294 | }; |
295 | |
296 | typedef int (*wide_printf)(wchar_t* __restrict, size_t, const wchar_t* __restrict, ...); |
297 | |
298 | inline wide_printf get_swprintf() { |
299 | # ifndef _LIBCPP_MSVCRT |
300 | return swprintf; |
301 | # else |
302 | return static_cast<int(__cdecl*)(wchar_t* __restrict, size_t, const wchar_t* __restrict, ...)>(_snwprintf); |
303 | # endif |
304 | } |
305 | #endif // _LIBCPP_HAS_NO_WIDE_CHARACTERS |
306 | |
307 | template <typename S, typename V> |
308 | S i_to_string(V v) { |
309 | // numeric_limits::digits10 returns value less on 1 than desired for unsigned numbers. |
310 | // For example, for 1-byte unsigned value digits10 is 2 (999 can not be represented), |
311 | // so we need +1 here. |
312 | constexpr size_t bufsize = numeric_limits<V>::digits10 + 2; // +1 for minus, +1 for digits10 |
313 | char buf[bufsize]; |
314 | const auto res = to_chars(buf, buf + bufsize, v); |
315 | _LIBCPP_ASSERT_INTERNAL(res.ec == errc(), "bufsize must be large enough to accomodate the value" ); |
316 | return S(buf, res.ptr); |
317 | } |
318 | |
319 | } // unnamed namespace |
320 | |
321 | string to_string(int val) { return i_to_string< string>(v: val); } |
322 | string to_string(long val) { return i_to_string< string>(v: val); } |
323 | string to_string(long long val) { return i_to_string< string>(v: val); } |
324 | string to_string(unsigned val) { return i_to_string< string>(v: val); } |
325 | string to_string(unsigned long val) { return i_to_string< string>(v: val); } |
326 | string to_string(unsigned long long val) { return i_to_string< string>(v: val); } |
327 | |
328 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
329 | wstring to_wstring(int val) { return i_to_string<wstring>(v: val); } |
330 | wstring to_wstring(long val) { return i_to_string<wstring>(v: val); } |
331 | wstring to_wstring(long long val) { return i_to_string<wstring>(v: val); } |
332 | wstring to_wstring(unsigned val) { return i_to_string<wstring>(v: val); } |
333 | wstring to_wstring(unsigned long val) { return i_to_string<wstring>(v: val); } |
334 | wstring to_wstring(unsigned long long val) { return i_to_string<wstring>(v: val); } |
335 | #endif |
336 | |
337 | string to_string(float val) { return as_string(sprintf_like: snprintf, s: initial_string< string>()(), fmt: "%f" , a: val); } |
338 | string to_string(double val) { return as_string(sprintf_like: snprintf, s: initial_string< string>()(), fmt: "%f" , a: val); } |
339 | string to_string(long double val) { return as_string(sprintf_like: snprintf, s: initial_string< string>()(), fmt: "%Lf" , a: val); } |
340 | |
341 | #ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS |
342 | wstring to_wstring(float val) { return as_string(sprintf_like: get_swprintf(), s: initial_string<wstring>()(), fmt: L"%f" , a: val); } |
343 | wstring to_wstring(double val) { return as_string(sprintf_like: get_swprintf(), s: initial_string<wstring>()(), fmt: L"%f" , a: val); } |
344 | wstring to_wstring(long double val) { return as_string(sprintf_like: get_swprintf(), s: initial_string<wstring>()(), fmt: L"%Lf" , a: val); } |
345 | #endif |
346 | |
347 | _LIBCPP_END_NAMESPACE_STD |
348 | |