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