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