xref: /aosp_15_r20/system/libbase/include/android-base/parseint.h (revision 8f0ba417480079999ba552f1087ae592091b9d02)
1*8f0ba417SAndroid Build Coastguard Worker /*
2*8f0ba417SAndroid Build Coastguard Worker  * Copyright (C) 2015 The Android Open Source Project
3*8f0ba417SAndroid Build Coastguard Worker  *
4*8f0ba417SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*8f0ba417SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*8f0ba417SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*8f0ba417SAndroid Build Coastguard Worker  *
8*8f0ba417SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*8f0ba417SAndroid Build Coastguard Worker  *
10*8f0ba417SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*8f0ba417SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*8f0ba417SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*8f0ba417SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*8f0ba417SAndroid Build Coastguard Worker  * limitations under the License.
15*8f0ba417SAndroid Build Coastguard Worker  */
16*8f0ba417SAndroid Build Coastguard Worker 
17*8f0ba417SAndroid Build Coastguard Worker #pragma once
18*8f0ba417SAndroid Build Coastguard Worker 
19*8f0ba417SAndroid Build Coastguard Worker #include <errno.h>
20*8f0ba417SAndroid Build Coastguard Worker #include <stdlib.h>
21*8f0ba417SAndroid Build Coastguard Worker #include <string.h>
22*8f0ba417SAndroid Build Coastguard Worker 
23*8f0ba417SAndroid Build Coastguard Worker #include <limits>
24*8f0ba417SAndroid Build Coastguard Worker #include <string>
25*8f0ba417SAndroid Build Coastguard Worker #include <type_traits>
26*8f0ba417SAndroid Build Coastguard Worker 
27*8f0ba417SAndroid Build Coastguard Worker #define LIBBASE_ALWAYS_INLINE __attribute__((__always_inline__))
28*8f0ba417SAndroid Build Coastguard Worker 
29*8f0ba417SAndroid Build Coastguard Worker namespace android {
30*8f0ba417SAndroid Build Coastguard Worker namespace base {
31*8f0ba417SAndroid Build Coastguard Worker 
32*8f0ba417SAndroid Build Coastguard Worker // Parses the unsigned decimal or hexadecimal integer in the string 's' and sets
33*8f0ba417SAndroid Build Coastguard Worker // 'out' to that value if it is specified. Optionally allows the caller to define
34*8f0ba417SAndroid Build Coastguard Worker // a 'max' beyond which otherwise valid values will be rejected. Returns boolean
35*8f0ba417SAndroid Build Coastguard Worker // success; 'out' is untouched if parsing fails.
36*8f0ba417SAndroid Build Coastguard Worker template <typename T>
37*8f0ba417SAndroid Build Coastguard Worker LIBBASE_ALWAYS_INLINE bool ParseUint(const char* s, T* out, T max = std::numeric_limits<T>::max(),
38*8f0ba417SAndroid Build Coastguard Worker                                      bool allow_suffixes = false) {
39*8f0ba417SAndroid Build Coastguard Worker   static_assert(std::is_unsigned<T>::value, "ParseUint can only be used with unsigned types");
40*8f0ba417SAndroid Build Coastguard Worker   while (isspace(*s)) {
41*8f0ba417SAndroid Build Coastguard Worker     s++;
42*8f0ba417SAndroid Build Coastguard Worker   }
43*8f0ba417SAndroid Build Coastguard Worker 
44*8f0ba417SAndroid Build Coastguard Worker   if (s[0] == '-') {
45*8f0ba417SAndroid Build Coastguard Worker     errno = EINVAL;
46*8f0ba417SAndroid Build Coastguard Worker     return false;
47*8f0ba417SAndroid Build Coastguard Worker   }
48*8f0ba417SAndroid Build Coastguard Worker 
49*8f0ba417SAndroid Build Coastguard Worker   // This is never out of bounds. If string is zero-sized, s[0] == '\0'
50*8f0ba417SAndroid Build Coastguard Worker   // so the second condition is not checked. If string is "0",
51*8f0ba417SAndroid Build Coastguard Worker   // s[1] will compare against the '\0'.
52*8f0ba417SAndroid Build Coastguard Worker   int base = (s[0] == '0' && (s[1] == 'x' || s[1] == 'X')) ? 16 : 10;
53*8f0ba417SAndroid Build Coastguard Worker   errno = 0;
54*8f0ba417SAndroid Build Coastguard Worker   char* end;
55*8f0ba417SAndroid Build Coastguard Worker   unsigned long long int result = strtoull(s, &end, base);
56*8f0ba417SAndroid Build Coastguard Worker   if (errno != 0) return false;
57*8f0ba417SAndroid Build Coastguard Worker   if (end == s) {
58*8f0ba417SAndroid Build Coastguard Worker     errno = EINVAL;
59*8f0ba417SAndroid Build Coastguard Worker     return false;
60*8f0ba417SAndroid Build Coastguard Worker   }
61*8f0ba417SAndroid Build Coastguard Worker   if (*end != '\0') {
62*8f0ba417SAndroid Build Coastguard Worker     const char* suffixes = "bkmgtpe";
63*8f0ba417SAndroid Build Coastguard Worker     const char* suffix;
64*8f0ba417SAndroid Build Coastguard Worker     if ((!allow_suffixes || (suffix = strchr(suffixes, tolower(*end))) == nullptr) ||
65*8f0ba417SAndroid Build Coastguard Worker         __builtin_mul_overflow(result, 1ULL << (10 * (suffix - suffixes)), &result)) {
66*8f0ba417SAndroid Build Coastguard Worker       errno = EINVAL;
67*8f0ba417SAndroid Build Coastguard Worker       return false;
68*8f0ba417SAndroid Build Coastguard Worker     }
69*8f0ba417SAndroid Build Coastguard Worker   }
70*8f0ba417SAndroid Build Coastguard Worker   if (max < result) {
71*8f0ba417SAndroid Build Coastguard Worker     errno = ERANGE;
72*8f0ba417SAndroid Build Coastguard Worker     return false;
73*8f0ba417SAndroid Build Coastguard Worker   }
74*8f0ba417SAndroid Build Coastguard Worker   if (out != nullptr) {
75*8f0ba417SAndroid Build Coastguard Worker     *out = static_cast<T>(result);
76*8f0ba417SAndroid Build Coastguard Worker   }
77*8f0ba417SAndroid Build Coastguard Worker   return true;
78*8f0ba417SAndroid Build Coastguard Worker }
79*8f0ba417SAndroid Build Coastguard Worker 
80*8f0ba417SAndroid Build Coastguard Worker // TODO: string_view
81*8f0ba417SAndroid Build Coastguard Worker template <typename T>
82*8f0ba417SAndroid Build Coastguard Worker LIBBASE_ALWAYS_INLINE bool ParseUint(const std::string& s, T* out,
83*8f0ba417SAndroid Build Coastguard Worker                                      T max = std::numeric_limits<T>::max(),
84*8f0ba417SAndroid Build Coastguard Worker                                      bool allow_suffixes = false) {
85*8f0ba417SAndroid Build Coastguard Worker   return ParseUint(s.c_str(), out, max, allow_suffixes);
86*8f0ba417SAndroid Build Coastguard Worker }
87*8f0ba417SAndroid Build Coastguard Worker 
88*8f0ba417SAndroid Build Coastguard Worker template <typename T>
89*8f0ba417SAndroid Build Coastguard Worker LIBBASE_ALWAYS_INLINE bool ParseByteCount(const char* s, T* out,
90*8f0ba417SAndroid Build Coastguard Worker                                           T max = std::numeric_limits<T>::max()) {
91*8f0ba417SAndroid Build Coastguard Worker   return ParseUint(s, out, max, true);
92*8f0ba417SAndroid Build Coastguard Worker }
93*8f0ba417SAndroid Build Coastguard Worker 
94*8f0ba417SAndroid Build Coastguard Worker // TODO: string_view
95*8f0ba417SAndroid Build Coastguard Worker template <typename T>
96*8f0ba417SAndroid Build Coastguard Worker LIBBASE_ALWAYS_INLINE bool ParseByteCount(const std::string& s, T* out,
97*8f0ba417SAndroid Build Coastguard Worker                                           T max = std::numeric_limits<T>::max()) {
98*8f0ba417SAndroid Build Coastguard Worker   return ParseByteCount(s.c_str(), out, max);
99*8f0ba417SAndroid Build Coastguard Worker }
100*8f0ba417SAndroid Build Coastguard Worker 
101*8f0ba417SAndroid Build Coastguard Worker // Parses the signed decimal or hexadecimal integer in the string 's' and sets
102*8f0ba417SAndroid Build Coastguard Worker // 'out' to that value if it is specified. Optionally allows the caller to define
103*8f0ba417SAndroid Build Coastguard Worker // a 'min' and 'max' beyond which otherwise valid values will be rejected. Returns
104*8f0ba417SAndroid Build Coastguard Worker // boolean success; 'out' is untouched if parsing fails.
105*8f0ba417SAndroid Build Coastguard Worker template <typename T>
106*8f0ba417SAndroid Build Coastguard Worker LIBBASE_ALWAYS_INLINE bool ParseInt(const char* s, T* out, T min = std::numeric_limits<T>::min(),
107*8f0ba417SAndroid Build Coastguard Worker                                     T max = std::numeric_limits<T>::max()) {
108*8f0ba417SAndroid Build Coastguard Worker   static_assert(std::is_signed<T>::value, "ParseInt can only be used with signed types");
109*8f0ba417SAndroid Build Coastguard Worker   while (isspace(*s)) {
110*8f0ba417SAndroid Build Coastguard Worker     s++;
111*8f0ba417SAndroid Build Coastguard Worker   }
112*8f0ba417SAndroid Build Coastguard Worker 
113*8f0ba417SAndroid Build Coastguard Worker   int base = (s[0] == '0' && (s[1] == 'x' || s[1] == 'X')) ? 16 : 10;
114*8f0ba417SAndroid Build Coastguard Worker   errno = 0;
115*8f0ba417SAndroid Build Coastguard Worker   char* end;
116*8f0ba417SAndroid Build Coastguard Worker   long long int result = strtoll(s, &end, base);
117*8f0ba417SAndroid Build Coastguard Worker   if (errno != 0) {
118*8f0ba417SAndroid Build Coastguard Worker     return false;
119*8f0ba417SAndroid Build Coastguard Worker   }
120*8f0ba417SAndroid Build Coastguard Worker   if (s == end || *end != '\0') {
121*8f0ba417SAndroid Build Coastguard Worker     errno = EINVAL;
122*8f0ba417SAndroid Build Coastguard Worker     return false;
123*8f0ba417SAndroid Build Coastguard Worker   }
124*8f0ba417SAndroid Build Coastguard Worker   if (result < min || max < result) {
125*8f0ba417SAndroid Build Coastguard Worker     errno = ERANGE;
126*8f0ba417SAndroid Build Coastguard Worker     return false;
127*8f0ba417SAndroid Build Coastguard Worker   }
128*8f0ba417SAndroid Build Coastguard Worker   if (out != nullptr) {
129*8f0ba417SAndroid Build Coastguard Worker     *out = static_cast<T>(result);
130*8f0ba417SAndroid Build Coastguard Worker   }
131*8f0ba417SAndroid Build Coastguard Worker   return true;
132*8f0ba417SAndroid Build Coastguard Worker }
133*8f0ba417SAndroid Build Coastguard Worker 
134*8f0ba417SAndroid Build Coastguard Worker // TODO: string_view
135*8f0ba417SAndroid Build Coastguard Worker template <typename T>
136*8f0ba417SAndroid Build Coastguard Worker LIBBASE_ALWAYS_INLINE bool ParseInt(const std::string& s, T* out,
137*8f0ba417SAndroid Build Coastguard Worker                                     T min = std::numeric_limits<T>::min(),
138*8f0ba417SAndroid Build Coastguard Worker                                     T max = std::numeric_limits<T>::max()) {
139*8f0ba417SAndroid Build Coastguard Worker   return ParseInt(s.c_str(), out, min, max);
140*8f0ba417SAndroid Build Coastguard Worker }
141*8f0ba417SAndroid Build Coastguard Worker 
142*8f0ba417SAndroid Build Coastguard Worker }  // namespace base
143*8f0ba417SAndroid Build Coastguard Worker }  // namespace android
144*8f0ba417SAndroid Build Coastguard Worker 
145*8f0ba417SAndroid Build Coastguard Worker #undef LIBBASE_ALWAYS_INLINE
146