1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #pragma once
18 
19 #include <limits>
20 #include <sstream>
21 #include <string>
22 #include <vector>
23 #include <stdint.h>
24 #include <cmath>
25 
26 namespace android
27 {
28 
29 namespace utilities
30 {
31 
32 /**
33  * Convert a given source type to a given destination type.
34  *
35  * String conversion to T reads the value of the type T in the given string.
36  * The function does not allow to have white spaces around the value to parse
37  * and tries to parse the whole string, which means that if some bytes were not
38  * read in the string, the function fails.
39  * Hexadecimal representation (ie. numbers starting with 0x) is supported only
40  * for integral types conversions.
41  *
42  * Numeric conversion to string formats the source value to decimal space.
43  *
44  * Vector to vector conversion calls convertTo on each element.
45  *
46  * @tparam srcType source type, default value is string type
47  * @tparam dstType destination type
48  * @param[in] input The source to convert from.
49  * @param[out] result Converted value if success, undefined on failure.
50  *
51  * @return true if conversion was successful, false otherwise.
52  */
53 template <typename srcType, typename dstType>
54 static inline bool convertTo(const srcType &input, dstType &result);
55 
56 /* details namespace is here to hide implementation details to header end user. It
57  * is NOT intended to be used outside. */
58 namespace details
59 {
60 
61 /** Helper class to limit instantiation of templates */
62 template <typename T>
63 struct ConversionFromStringAllowed;
64 template <typename T>
65 struct ConversionToStringAllowed;
66 
67 /* List of allowed types for conversion */
68 template <>
69 struct ConversionFromStringAllowed<bool> {};
70 template <>
71 struct ConversionFromStringAllowed<uint64_t> {};
72 template <>
73 struct ConversionFromStringAllowed<int64_t> {};
74 template <>
75 struct ConversionFromStringAllowed<uint32_t> {};
76 template <>
77 struct ConversionFromStringAllowed<int32_t> {};
78 template <>
79 struct ConversionFromStringAllowed<uint16_t> {};
80 template <>
81 struct ConversionFromStringAllowed<int16_t> {};
82 template <>
83 struct ConversionFromStringAllowed<float> {};
84 template <>
85 struct ConversionFromStringAllowed<double> {};
86 
87 template <>
88 struct ConversionToStringAllowed<int64_t> {};
89 template <>
90 struct ConversionToStringAllowed<uint64_t> {};
91 template <>
92 struct ConversionToStringAllowed<uint32_t> {};
93 template <>
94 struct ConversionToStringAllowed<int32_t> {};
95 template <>
96 struct ConversionToStringAllowed<double> {};
97 template <>
98 struct ConversionToStringAllowed<float> {};
99 
100 /**
101  * Set the decimal precision to 10 digits.
102  * Note that this setting is aligned with Android Audio Parameter
103  * policy concerning float storage into string.
104  */
105 static const uint32_t gFloatPrecision = 10;
106 
107 template <typename T>
108 static inline bool fromString(const std::string &str, T &result)
109 {
110     /* Check that conversion to that type is allowed.
111      * If this fails, this means that this template was not intended to be used
112      * with this type, thus that the result is undefined. */
113     ConversionFromStringAllowed<T>();
114 
115     if (str.find_first_of(std::string("\r\n\t\v ")) != std::string::npos) {
116         return false;
117     }
118 
119     /* Check for a '-' in string. If type is unsigned and a - is found, the
120      * parsing fails. This is made necessary because "-1" is read as 65535 for
121      * uint16_t, for example */
122     if (str.find('-') != std::string::npos
123         && !std::numeric_limits<T>::is_signed) {
124         return false;
125     }
126 
127     std::stringstream ss(str);
128 
129     /* Sadly, the stream conversion does not handle hexadecimal format, thus
130      * check is done manually */
131     if (str.substr(0, 2) == "0x") {
132         if (std::numeric_limits<T>::is_integer) {
133             ss >> std::hex >> result;
134         } else {
135             /* Conversion undefined for non integers */
136             return false;
137         }
138     } else {
139         ss >> result;
140     }
141 
142     return ss.eof() && !ss.fail() && !ss.bad();
143 }
144 
145 template <typename T>
146 static inline bool toString(const T &value, std::string &str)
147 {
148     /* Check that conversion from that type is allowed.
149      * If this fails, this means that this template was not intended to be used
150      * with this type, thus that the result is undefined. */
151     ConversionToStringAllowed<T>();
152 
153     std::stringstream oss;
154     oss.precision(gFloatPrecision);
155     oss << value;
156     str = oss.str();
157     return !oss.fail() && !oss.bad();
158 }
159 
160 template <typename srcType, typename dstType>
161 class Converter;
162 
163 template <typename dstType>
164 class Converter<std::string, dstType>
165 {
166 public:
167     static inline bool run(const std::string &str, dstType &result)
168     {
169         return fromString<dstType>(str, result);
170     }
171 };
172 
173 template <typename srcType>
174 class Converter<srcType, std::string>
175 {
176 public:
177     static inline bool run(const srcType &str, std::string &result)
178     {
179         return toString<srcType>(str, result);
180     }
181 };
182 
183 /** Convert a vector by applying convertTo on each element.
184  *
185  * @tparam SrcElem Type of the src elements.
186  * @tparam DstElem Type of the destination elements.
187  */
188 template <typename SrcElem, typename DstElem>
189 class Converter<std::vector<SrcElem>, std::vector<DstElem> >
190 {
191 public:
192     typedef const std::vector<SrcElem> Src;
193     typedef std::vector<DstElem> Dst;
194 
195     static inline bool run(Src &src, Dst &dst)
196     {
197         typedef typename Src::const_iterator SrcIt;
198         dst.clear();
199         dst.reserve(src.size());
200         for (SrcIt it = src.begin(); it != src.end(); ++it) {
201             DstElem dstElem;
202             if (not convertTo(*it, dstElem)) {
203                 return false;
204             }
205             dst.push_back(dstElem);
206         }
207         return true;
208     }
209 };
210 
211 } // namespace details
212 
213 template <typename srcType, typename dstType>
214 static inline bool convertTo(const srcType &input, dstType &result)
215 {
216     return details::Converter<srcType, dstType>::run(input, result);
217 }
218 
219 /**
220  * Specialization for int16_t of convertTo template function.
221  *
222  * This function follows the same paradigm than it's generic version.
223  *
224  * The specific implementation is made necessary because the stlport version of
225  * string streams is bugged and does not fail when giving overflowed values.
226  * This specialisation can be safely removed when stlport behaviour is fixed.
227  *
228  * @param[in]  str    the string to parse.
229  * @param[out] result reference to object where to store the result.
230  *
231  * @return true if conversion was successful, false otherwise.
232  */
233 template <>
234 inline bool convertTo<std::string, int16_t>(const std::string &str, int16_t &result)
235 {
236     int64_t res;
237 
238     if (!convertTo<std::string, int64_t>(str, res)) {
239         return false;
240     }
241 
242     if (res > std::numeric_limits<int16_t>::max() || res < std::numeric_limits<int16_t>::min()) {
243         return false;
244     }
245 
246     result = static_cast<int16_t>(res);
247     return true;
248 }
249 
250 /**
251  * Specialization for float of convertTo template function.
252  *
253  * This function follows the same paradigm than it's generic version and is
254  * based on it but makes furthers checks on the returned value.
255  *
256  * The specific implementation is made necessary because the stlport conversion
257  * from string to float behaves differently than GNU STL: overflow produce
258  * +/-Infinity rather than an error.
259  *
260  * @param[in]  str    the string to parse.
261  * @param[out] result reference to object where to store the result.
262  *
263  * @return true if conversion was successful, false otherwise.
264  */
265 template <>
266 inline bool convertTo<std::string, float>(const std::string &str, float &result)
267 {
268     if (!details::Converter<std::string, float>::run(str, result)) {
269         return false;
270     }
271 
272     if (std::abs(result) == std::numeric_limits<float>::infinity() ||
273         result == std::numeric_limits<float>::quiet_NaN()) {
274         return false;
275     }
276 
277     return true;
278 }
279 
280 /**
281  * Specialization for double of convertTo template function.
282  *
283  * This function follows the same paradigm than it's generic version and is
284  * based on it but makes furthers checks on the returned value.
285  *
286  * The specific implementation is made necessary because the stlport conversion
287  * from string to double behaves differently than GNU STL: overflow produce
288  * +/-Infinity rather than an error.
289  *
290  * @param[in]  str    the string to parse.
291  * @param[out] result reference to object where to store the result.
292  *
293  * @return true if conversion was successful, false otherwise.
294  */
295 template <>
296 inline bool convertTo<std::string, double>(const std::string &str, double &result)
297 {
298     if (!details::Converter<std::string, double>::run(str, result)) {
299         return false;
300     }
301 
302     if (std::abs(result) == std::numeric_limits<double>::infinity() ||
303         result == std::numeric_limits<double>::quiet_NaN()) {
304         return false;
305     }
306 
307     return true;
308 }
309 
310 /**
311  * Specialization for boolean of convertTo template function.
312  *
313  * This function follows the same paradigm than it's generic version.
314  * This function accepts to parse boolean as "0/1" or "false/true" or
315  * "FALSE/TRUE".
316  * The specific implementation is made necessary because the behaviour of
317  * string streams when parsing boolean values is not sufficient to fit our
318  * requirements. Indeed, parsing "true" will correctly parse the value, but the
319  * end of stream is not reached which makes the ss.eof() fails in the generic
320  * implementation.
321  *
322  * @param[in]  str    the string to parse.
323  * @param[out] result reference to object where to store the result.
324  *
325  * @return true if conversion was successful, false otherwise.
326  */
327 template <>
328 inline bool convertTo<std::string, bool>(const std::string &str, bool &result)
329 {
330     if (str == "0" || str == "FALSE" || str == "false") {
331         result = false;
332         return true;
333     }
334 
335     if (str == "1" || str == "TRUE" || str == "true") {
336         result = true;
337         return true;
338     }
339 
340     return false;
341 }
342 
343 /**
344  * Specialization for boolean to string of convertTo template function.
345  *
346  * This function follows the same paradigm than it's generic version.
347  * This function arbitrarily decides to return "false/true".
348  * It is compatible with the specialization from string to boolean.
349  *
350  * @param[in]  isSet  boolean to convert to a string.
351  * @param[out] result reference to object where to store the result.
352  *
353  * @return true if conversion was successful, false otherwise.
354  */
355 template <>
356 inline bool convertTo<bool, std::string>(const bool &isSet, std::string &result)
357 {
358     result = isSet ? "true" : "false";
359     return true;
360 }
361 
362 /**
363  * Specialization for string to string of convertTo template function.
364  *
365  * This function is a dummy conversion from string to string.
366  * In case of clients using template as well, this implementation avoids adding extra
367  * specialization to bypass the conversion from string to string.
368  *
369  * @param[in]  str    the string to parse.
370  * @param[out] result reference to object where to store the result.
371  *
372  * @return true if conversion was successful, false otherwise.
373  */
374 template <>
375 inline bool convertTo<std::string, std::string>(const std::string &str, std::string &result)
376 {
377     result = str;
378     return true;
379 }
380 
381 } // namespace utilities
382 
383 } // namespace android
384