xref: /aosp_15_r20/external/webrtc/rtc_base/string_to_number.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright 2017 The WebRTC Project Authors. All rights reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/string_to_number.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include <ctype.h>
14*d9f75844SAndroid Build Coastguard Worker 
15*d9f75844SAndroid Build Coastguard Worker #include <cerrno>
16*d9f75844SAndroid Build Coastguard Worker #include <cstdlib>
17*d9f75844SAndroid Build Coastguard Worker 
18*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h"
19*d9f75844SAndroid Build Coastguard Worker 
20*d9f75844SAndroid Build Coastguard Worker namespace rtc {
21*d9f75844SAndroid Build Coastguard Worker namespace string_to_number_internal {
22*d9f75844SAndroid Build Coastguard Worker 
ParseSigned(absl::string_view str,int base)23*d9f75844SAndroid Build Coastguard Worker absl::optional<signed_type> ParseSigned(absl::string_view str, int base) {
24*d9f75844SAndroid Build Coastguard Worker   if (str.empty())
25*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
26*d9f75844SAndroid Build Coastguard Worker 
27*d9f75844SAndroid Build Coastguard Worker   if (isdigit(static_cast<unsigned char>(str[0])) || str[0] == '-') {
28*d9f75844SAndroid Build Coastguard Worker     std::string str_str(str);
29*d9f75844SAndroid Build Coastguard Worker     char* end = nullptr;
30*d9f75844SAndroid Build Coastguard Worker     errno = 0;
31*d9f75844SAndroid Build Coastguard Worker     const signed_type value = std::strtoll(str_str.c_str(), &end, base);
32*d9f75844SAndroid Build Coastguard Worker     // Check for errors and also make sure that there were no embedded nuls in
33*d9f75844SAndroid Build Coastguard Worker     // the input string.
34*d9f75844SAndroid Build Coastguard Worker     if (end == str_str.c_str() + str_str.size() && errno == 0) {
35*d9f75844SAndroid Build Coastguard Worker       return value;
36*d9f75844SAndroid Build Coastguard Worker     }
37*d9f75844SAndroid Build Coastguard Worker   }
38*d9f75844SAndroid Build Coastguard Worker   return absl::nullopt;
39*d9f75844SAndroid Build Coastguard Worker }
40*d9f75844SAndroid Build Coastguard Worker 
ParseUnsigned(absl::string_view str,int base)41*d9f75844SAndroid Build Coastguard Worker absl::optional<unsigned_type> ParseUnsigned(absl::string_view str, int base) {
42*d9f75844SAndroid Build Coastguard Worker   if (str.empty())
43*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
44*d9f75844SAndroid Build Coastguard Worker 
45*d9f75844SAndroid Build Coastguard Worker   if (isdigit(static_cast<unsigned char>(str[0])) || str[0] == '-') {
46*d9f75844SAndroid Build Coastguard Worker     std::string str_str(str);
47*d9f75844SAndroid Build Coastguard Worker     // Explicitly discard negative values. std::strtoull parsing causes unsigned
48*d9f75844SAndroid Build Coastguard Worker     // wraparound. We cannot just reject values that start with -, though, since
49*d9f75844SAndroid Build Coastguard Worker     // -0 is perfectly fine, as is -0000000000000000000000000000000.
50*d9f75844SAndroid Build Coastguard Worker     const bool is_negative = str[0] == '-';
51*d9f75844SAndroid Build Coastguard Worker     char* end = nullptr;
52*d9f75844SAndroid Build Coastguard Worker     errno = 0;
53*d9f75844SAndroid Build Coastguard Worker     const unsigned_type value = std::strtoull(str_str.c_str(), &end, base);
54*d9f75844SAndroid Build Coastguard Worker     // Check for errors and also make sure that there were no embedded nuls in
55*d9f75844SAndroid Build Coastguard Worker     // the input string.
56*d9f75844SAndroid Build Coastguard Worker     if (end == str_str.c_str() + str_str.size() && errno == 0 &&
57*d9f75844SAndroid Build Coastguard Worker         (value == 0 || !is_negative)) {
58*d9f75844SAndroid Build Coastguard Worker       return value;
59*d9f75844SAndroid Build Coastguard Worker     }
60*d9f75844SAndroid Build Coastguard Worker   }
61*d9f75844SAndroid Build Coastguard Worker   return absl::nullopt;
62*d9f75844SAndroid Build Coastguard Worker }
63*d9f75844SAndroid Build Coastguard Worker 
64*d9f75844SAndroid Build Coastguard Worker template <typename T>
65*d9f75844SAndroid Build Coastguard Worker T StrToT(const char* str, char** str_end);
66*d9f75844SAndroid Build Coastguard Worker 
67*d9f75844SAndroid Build Coastguard Worker template <>
StrToT(const char * str,char ** str_end)68*d9f75844SAndroid Build Coastguard Worker inline float StrToT(const char* str, char** str_end) {
69*d9f75844SAndroid Build Coastguard Worker   return std::strtof(str, str_end);
70*d9f75844SAndroid Build Coastguard Worker }
71*d9f75844SAndroid Build Coastguard Worker 
72*d9f75844SAndroid Build Coastguard Worker template <>
StrToT(const char * str,char ** str_end)73*d9f75844SAndroid Build Coastguard Worker inline double StrToT(const char* str, char** str_end) {
74*d9f75844SAndroid Build Coastguard Worker   return std::strtod(str, str_end);
75*d9f75844SAndroid Build Coastguard Worker }
76*d9f75844SAndroid Build Coastguard Worker 
77*d9f75844SAndroid Build Coastguard Worker template <>
StrToT(const char * str,char ** str_end)78*d9f75844SAndroid Build Coastguard Worker inline long double StrToT(const char* str, char** str_end) {
79*d9f75844SAndroid Build Coastguard Worker   return std::strtold(str, str_end);
80*d9f75844SAndroid Build Coastguard Worker }
81*d9f75844SAndroid Build Coastguard Worker 
82*d9f75844SAndroid Build Coastguard Worker template <typename T>
ParseFloatingPoint(absl::string_view str)83*d9f75844SAndroid Build Coastguard Worker absl::optional<T> ParseFloatingPoint(absl::string_view str) {
84*d9f75844SAndroid Build Coastguard Worker   if (str.empty())
85*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
86*d9f75844SAndroid Build Coastguard Worker 
87*d9f75844SAndroid Build Coastguard Worker   if (str[0] == '\0')
88*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
89*d9f75844SAndroid Build Coastguard Worker   std::string str_str(str);
90*d9f75844SAndroid Build Coastguard Worker   char* end = nullptr;
91*d9f75844SAndroid Build Coastguard Worker   errno = 0;
92*d9f75844SAndroid Build Coastguard Worker   const T value = StrToT<T>(str_str.c_str(), &end);
93*d9f75844SAndroid Build Coastguard Worker   if (end == str_str.c_str() + str_str.size() && errno == 0) {
94*d9f75844SAndroid Build Coastguard Worker     return value;
95*d9f75844SAndroid Build Coastguard Worker   }
96*d9f75844SAndroid Build Coastguard Worker   return absl::nullopt;
97*d9f75844SAndroid Build Coastguard Worker }
98*d9f75844SAndroid Build Coastguard Worker 
99*d9f75844SAndroid Build Coastguard Worker template absl::optional<float> ParseFloatingPoint(absl::string_view str);
100*d9f75844SAndroid Build Coastguard Worker template absl::optional<double> ParseFloatingPoint(absl::string_view str);
101*d9f75844SAndroid Build Coastguard Worker template absl::optional<long double> ParseFloatingPoint(absl::string_view str);
102*d9f75844SAndroid Build Coastguard Worker 
103*d9f75844SAndroid Build Coastguard Worker }  // namespace string_to_number_internal
104*d9f75844SAndroid Build Coastguard Worker }  // namespace rtc
105