12 #ifndef ROOT_StringConv
13 #define ROOT_StringConv
18 #include "RConfigure.h"
37 template <
typename value_type>
38 void ToHumanReadableSize(value_type bytes,
44 static const char *
const suffix[][2] =
53 value_type unit = si ? 1000 : 1024;
61 }
else if (bytes > 0) {
62 exp = std::min( (
int) (std::log(bytes) / std::log(unit)),
63 (
int) (
sizeof(suffix) /
sizeof(suffix[0]) - 1));
65 *coeff = bytes / std::pow(unit, exp);
66 *units = suffix[exp][!si];
69 enum class EFromHumanReadableSize {
86 EFromHumanReadableSize FromHumanReadableSize(std::string_view str, T &value)
89 size_t cur, size = str.size();
91 const double coeff = stod(std::string(str.data(), str.size()), &cur);
94 while (cur<size && isspace(str[cur])) ++cur;
97 int exp = 0, unit = 1000;
99 auto result = [coeff,&exp,&unit,&value]() {
100 double v = exp ? coeff * std::pow(unit, exp / 3) : coeff;
101 if (v < (
double) std::numeric_limits<T>::max()) {
103 return EFromHumanReadableSize::kSuccess;
105 return EFromHumanReadableSize::kOverflow;
108 if (cur==size)
return result();
110 switch (toupper(str[cur])) {
111 case 'B': exp = 0;
break;
112 case 'K': exp = 3;
break;
113 case 'M': exp = 6;
break;
114 case 'G': exp = 9;
break;
115 case 'T': exp = 12;
break;
116 case 'E': exp = 15;
break;
117 case 'Z': exp = 18;
break;
118 case 'Y': exp = 21;
break;
120 default:
return EFromHumanReadableSize::kParseFail;
125 if (cur<size && toupper(str[cur]) ==
'I') {
130 if (cur==size)
return result();
133 switch (toupper(str[cur])) {
136 case '\t': ++cur;
break;
138 case '\0':
return result();
140 default:
return EFromHumanReadableSize::kParseFail;
152 return EFromHumanReadableSize::kParseFail;
157 template <
typename T>
158 EFromHumanReadableSize FromHumanReadableSize(ROOT::Internal::TStringView str, T &value)
160 return FromHumanReadableSize(std::string_view(str),value);
165 #endif // ROOT_StringConv