VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/string/strtonum.cpp@ 74978

Last change on this file since 74978 was 74160, checked in by vboxsync, 6 years ago

IPRT/strtonum: Fixed problem handling INT64_MIN in RTStrToInt64Ex and friends.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 29.9 KB
Line 
1/* $Id: strtonum.cpp 74160 2018-09-09 13:53:25Z vboxsync $ */
2/** @file
3 * IPRT - String To Number Conversion.
4 */
5
6/*
7 * Copyright (C) 2006-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/string.h>
32#include "internal/iprt.h"
33
34#include <iprt/assert.h>
35#include <iprt/ctype.h> /* needed for RT_C_IS_DIGIT */
36#include <iprt/err.h>
37
38
39/*********************************************************************************************************************************
40* Global Variables *
41*********************************************************************************************************************************/
42/** 8-bit char -> digit. */
43static const unsigned char g_auchDigits[256] =
44{
45 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,
46 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,255,255,255,255,255,255,
47 255, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35,255,255,255,255,255,
48 255, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35,255,255,255,255,255,
49 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,
50 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,
51 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,
52 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255
53};
54/** Approximated overflow shift checks. */
55static const char g_auchShift[36] =
56{
57 /* 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 */
58 64, 64, 63, 63, 62, 62, 62, 62, 61, 61, 61, 61, 61, 61, 61, 61, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 60, 59, 59, 59, 59
59};
60
61/*
62#include <stdio.h>
63int main()
64{
65 int i;
66 printf("static const unsigned char g_auchDigits[256] =\n"
67 "{");
68 for (i = 0; i < 256; i++)
69 {
70 int ch = 255;
71 if (i >= '0' && i <= '9')
72 ch = i - '0';
73 else if (i >= 'a' && i <= 'z')
74 ch = i - 'a' + 10;
75 else if (i >= 'A' && i <= 'Z')
76 ch = i - 'A' + 10;
77 if (i == 0)
78 printf("\n %3d", ch);
79 else if ((i % 32) == 0)
80 printf(",\n %3d", ch);
81 else
82 printf(",%3d", ch);
83 }
84 printf("\n"
85 "};\n");
86 return 0;
87}
88*/
89
90
91/**
92 * Converts a string representation of a number to a 64-bit unsigned number.
93 *
94 * @returns iprt status code.
95 * Warnings are used to indicate conversion problems.
96 * @retval VWRN_NUMBER_TOO_BIG
97 * @retval VWRN_NEGATIVE_UNSIGNED
98 * @retval VWRN_TRAILING_CHARS
99 * @retval VWRN_TRAILING_SPACES
100 * @retval VINF_SUCCESS
101 * @retval VERR_NO_DIGITS
102 *
103 * @param pszValue Pointer to the string value.
104 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
105 * @param uBase The base of the representation used.
106 * If the function will look for known prefixes before defaulting to 10.
107 * @param pu64 Where to store the converted number. (optional)
108 */
109RTDECL(int) RTStrToUInt64Ex(const char *pszValue, char **ppszNext, unsigned uBase, uint64_t *pu64)
110{
111 const char *psz = pszValue;
112 int iShift;
113 int rc;
114 uint64_t u64;
115 unsigned char uch;
116
117 /*
118 * Positive/Negative stuff.
119 */
120 bool fPositive = true;
121 for (;; psz++)
122 {
123 if (*psz == '+')
124 fPositive = true;
125 else if (*psz == '-')
126 fPositive = !fPositive;
127 else
128 break;
129 }
130
131 /*
132 * Check for hex prefix.
133 */
134 if (!uBase)
135 {
136 if ( psz[0] == '0'
137 && (psz[1] == 'x' || psz[1] == 'X')
138 && g_auchDigits[(unsigned char)psz[2]] < 16)
139 {
140 uBase = 16;
141 psz += 2;
142 }
143 else if ( psz[0] == '0'
144 && g_auchDigits[(unsigned char)psz[1]] < 8)
145 {
146 uBase = 8;
147 psz++;
148 }
149 else
150 uBase = 10;
151 }
152 else if ( uBase == 16
153 && psz[0] == '0'
154 && (psz[1] == 'x' || psz[1] == 'X')
155 && g_auchDigits[(unsigned char)psz[2]] < 16)
156 psz += 2;
157
158 /*
159 * Interpret the value.
160 * Note: We only support ascii digits at this time... :-)
161 */
162 iShift = g_auchShift[uBase];
163 pszValue = psz; /* (Prefix and sign doesn't count in the digit counting.) */
164 rc = VINF_SUCCESS;
165 u64 = 0;
166 while ((uch = (unsigned char)*psz) != 0)
167 {
168 unsigned char chDigit = g_auchDigits[uch];
169 uint64_t u64Prev;
170
171 if (chDigit >= uBase)
172 break;
173
174 u64Prev = u64;
175 u64 *= uBase;
176 u64 += chDigit;
177 if (u64Prev > u64 || (u64Prev >> iShift))
178 rc = VWRN_NUMBER_TOO_BIG;
179 psz++;
180 }
181
182 if (!fPositive)
183 {
184 if (rc == VINF_SUCCESS)
185 rc = VWRN_NEGATIVE_UNSIGNED;
186 u64 = -(int64_t)u64;
187 }
188
189 if (pu64)
190 *pu64 = u64;
191
192 if (psz == pszValue)
193 rc = VERR_NO_DIGITS;
194
195 if (ppszNext)
196 *ppszNext = (char *)psz;
197
198 /*
199 * Warn about trailing chars/spaces.
200 */
201 if ( rc == VINF_SUCCESS
202 && *psz)
203 {
204 while (*psz == ' ' || *psz == '\t')
205 psz++;
206 rc = *psz ? VWRN_TRAILING_CHARS : VWRN_TRAILING_SPACES;
207 }
208
209 return rc;
210}
211RT_EXPORT_SYMBOL(RTStrToUInt64Ex);
212
213
214/**
215 * Converts a string representation of a number to a 64-bit unsigned number,
216 * making sure the full string is converted.
217 *
218 * @returns iprt status code.
219 * Warnings are used to indicate conversion problems.
220 * @retval VWRN_NUMBER_TOO_BIG
221 * @retval VWRN_NEGATIVE_UNSIGNED
222 * @retval VINF_SUCCESS
223 * @retval VERR_NO_DIGITS
224 * @retval VERR_TRAILING_SPACES
225 * @retval VERR_TRAILING_CHARS
226 *
227 * @param pszValue Pointer to the string value.
228 * @param uBase The base of the representation used.
229 * If the function will look for known prefixes before defaulting to 10.
230 * @param pu64 Where to store the converted number. (optional)
231 */
232RTDECL(int) RTStrToUInt64Full(const char *pszValue, unsigned uBase, uint64_t *pu64)
233{
234 char *psz;
235 int rc = RTStrToUInt64Ex(pszValue, &psz, uBase, pu64);
236 if (RT_SUCCESS(rc) && *psz)
237 {
238 if (rc == VWRN_TRAILING_CHARS || rc == VWRN_TRAILING_SPACES)
239 rc = -rc;
240 else
241 {
242 while (*psz == ' ' || *psz == '\t')
243 psz++;
244 rc = *psz ? VERR_TRAILING_CHARS : VERR_TRAILING_SPACES;
245 }
246 }
247 return rc;
248}
249RT_EXPORT_SYMBOL(RTStrToUInt64Full);
250
251
252/**
253 * Converts a string representation of a number to a 64-bit unsigned number.
254 * The base is guessed.
255 *
256 * @returns 64-bit unsigned number on success.
257 * @returns 0 on failure.
258 * @param pszValue Pointer to the string value.
259 */
260RTDECL(uint64_t) RTStrToUInt64(const char *pszValue)
261{
262 uint64_t u64;
263 int rc = RTStrToUInt64Ex(pszValue, NULL, 0, &u64);
264 if (RT_SUCCESS(rc))
265 return u64;
266 return 0;
267}
268RT_EXPORT_SYMBOL(RTStrToUInt64);
269
270
271/**
272 * Converts a string representation of a number to a 32-bit unsigned number.
273 *
274 * @returns iprt status code.
275 * Warnings are used to indicate conversion problems.
276 * @retval VWRN_NUMBER_TOO_BIG
277 * @retval VWRN_NEGATIVE_UNSIGNED
278 * @retval VWRN_TRAILING_CHARS
279 * @retval VWRN_TRAILING_SPACES
280 * @retval VINF_SUCCESS
281 * @retval VERR_NO_DIGITS
282 *
283 * @param pszValue Pointer to the string value.
284 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
285 * @param uBase The base of the representation used.
286 * If the function will look for known prefixes before defaulting to 10.
287 * @param pu32 Where to store the converted number. (optional)
288 */
289RTDECL(int) RTStrToUInt32Ex(const char *pszValue, char **ppszNext, unsigned uBase, uint32_t *pu32)
290{
291 uint64_t u64;
292 int rc = RTStrToUInt64Ex(pszValue, ppszNext, uBase, &u64);
293 if (RT_SUCCESS(rc))
294 {
295 if (u64 & ~0xffffffffULL)
296 rc = VWRN_NUMBER_TOO_BIG;
297 }
298 if (pu32)
299 *pu32 = (uint32_t)u64;
300 return rc;
301}
302RT_EXPORT_SYMBOL(RTStrToUInt32Ex);
303
304
305/**
306 * Converts a string representation of a number to a 32-bit unsigned number,
307 * making sure the full string is converted.
308 *
309 * @returns iprt status code.
310 * Warnings are used to indicate conversion problems.
311 * @retval VWRN_NUMBER_TOO_BIG
312 * @retval VWRN_NEGATIVE_UNSIGNED
313 * @retval VINF_SUCCESS
314 * @retval VERR_NO_DIGITS
315 * @retval VERR_TRAILING_SPACES
316 * @retval VERR_TRAILING_CHARS
317 *
318 * @param pszValue Pointer to the string value.
319 * @param uBase The base of the representation used.
320 * If the function will look for known prefixes before defaulting to 10.
321 * @param pu32 Where to store the converted number. (optional)
322 */
323RTDECL(int) RTStrToUInt32Full(const char *pszValue, unsigned uBase, uint32_t *pu32)
324{
325 uint64_t u64;
326 int rc = RTStrToUInt64Full(pszValue, uBase, &u64);
327 if (RT_SUCCESS(rc))
328 {
329 if (u64 & ~0xffffffffULL)
330 rc = VWRN_NUMBER_TOO_BIG;
331 }
332 if (pu32)
333 *pu32 = (uint32_t)u64;
334 return rc;
335}
336RT_EXPORT_SYMBOL(RTStrToUInt32Full);
337
338
339/**
340 * Converts a string representation of a number to a 64-bit unsigned number.
341 * The base is guessed.
342 *
343 * @returns 32-bit unsigned number on success.
344 * @returns 0 on failure.
345 * @param pszValue Pointer to the string value.
346 */
347RTDECL(uint32_t) RTStrToUInt32(const char *pszValue)
348{
349 uint32_t u32;
350 int rc = RTStrToUInt32Ex(pszValue, NULL, 0, &u32);
351 if (RT_SUCCESS(rc))
352 return u32;
353 return 0;
354}
355RT_EXPORT_SYMBOL(RTStrToUInt32);
356
357
358/**
359 * Converts a string representation of a number to a 16-bit unsigned number.
360 *
361 * @returns iprt status code.
362 * Warnings are used to indicate conversion problems.
363 * @retval VWRN_NUMBER_TOO_BIG
364 * @retval VWRN_NEGATIVE_UNSIGNED
365 * @retval VWRN_TRAILING_CHARS
366 * @retval VWRN_TRAILING_SPACES
367 * @retval VINF_SUCCESS
368 * @retval VERR_NO_DIGITS
369 *
370 * @param pszValue Pointer to the string value.
371 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
372 * @param uBase The base of the representation used.
373 * If the function will look for known prefixes before defaulting to 10.
374 * @param pu16 Where to store the converted number. (optional)
375 */
376RTDECL(int) RTStrToUInt16Ex(const char *pszValue, char **ppszNext, unsigned uBase, uint16_t *pu16)
377{
378 uint64_t u64;
379 int rc = RTStrToUInt64Ex(pszValue, ppszNext, uBase, &u64);
380 if (RT_SUCCESS(rc))
381 {
382 if (u64 & ~0xffffULL)
383 rc = VWRN_NUMBER_TOO_BIG;
384 }
385 if (pu16)
386 *pu16 = (uint16_t)u64;
387 return rc;
388}
389RT_EXPORT_SYMBOL(RTStrToUInt16Ex);
390
391
392/**
393 * Converts a string representation of a number to a 16-bit unsigned number,
394 * making sure the full string is converted.
395 *
396 * @returns iprt status code.
397 * Warnings are used to indicate conversion problems.
398 * @retval VWRN_NUMBER_TOO_BIG
399 * @retval VWRN_NEGATIVE_UNSIGNED
400 * @retval VINF_SUCCESS
401 * @retval VERR_NO_DIGITS
402 * @retval VERR_TRAILING_SPACES
403 * @retval VERR_TRAILING_CHARS
404 *
405 * @param pszValue Pointer to the string value.
406 * @param uBase The base of the representation used.
407 * If the function will look for known prefixes before defaulting to 10.
408 * @param pu16 Where to store the converted number. (optional)
409 */
410RTDECL(int) RTStrToUInt16Full(const char *pszValue, unsigned uBase, uint16_t *pu16)
411{
412 uint64_t u64;
413 int rc = RTStrToUInt64Full(pszValue, uBase, &u64);
414 if (RT_SUCCESS(rc))
415 {
416 if (u64 & ~0xffffULL)
417 rc = VWRN_NUMBER_TOO_BIG;
418 }
419 if (pu16)
420 *pu16 = (uint16_t)u64;
421 return rc;
422}
423RT_EXPORT_SYMBOL(RTStrToUInt16Full);
424
425
426/**
427 * Converts a string representation of a number to a 16-bit unsigned number.
428 * The base is guessed.
429 *
430 * @returns 16-bit unsigned number on success.
431 * @returns 0 on failure.
432 * @param pszValue Pointer to the string value.
433 */
434RTDECL(uint16_t) RTStrToUInt16(const char *pszValue)
435{
436 uint16_t u16;
437 int rc = RTStrToUInt16Ex(pszValue, NULL, 0, &u16);
438 if (RT_SUCCESS(rc))
439 return u16;
440 return 0;
441}
442RT_EXPORT_SYMBOL(RTStrToUInt16);
443
444
445/**
446 * Converts a string representation of a number to a 8-bit unsigned number.
447 *
448 * @returns iprt status code.
449 * Warnings are used to indicate conversion problems.
450 * @retval VWRN_NUMBER_TOO_BIG
451 * @retval VWRN_NEGATIVE_UNSIGNED
452 * @retval VWRN_TRAILING_CHARS
453 * @retval VWRN_TRAILING_SPACES
454 * @retval VINF_SUCCESS
455 * @retval VERR_NO_DIGITS
456 *
457 * @param pszValue Pointer to the string value.
458 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
459 * @param uBase The base of the representation used.
460 * If the function will look for known prefixes before defaulting to 10.
461 * @param pu8 Where to store the converted number. (optional)
462 */
463RTDECL(int) RTStrToUInt8Ex(const char *pszValue, char **ppszNext, unsigned uBase, uint8_t *pu8)
464{
465 uint64_t u64;
466 int rc = RTStrToUInt64Ex(pszValue, ppszNext, uBase, &u64);
467 if (RT_SUCCESS(rc))
468 {
469 if (u64 & ~0xffULL)
470 rc = VWRN_NUMBER_TOO_BIG;
471 }
472 if (pu8)
473 *pu8 = (uint8_t)u64;
474 return rc;
475}
476RT_EXPORT_SYMBOL(RTStrToUInt8Ex);
477
478
479/**
480 * Converts a string representation of a number to a 8-bit unsigned number,
481 * making sure the full string is converted.
482 *
483 * @returns iprt status code.
484 * Warnings are used to indicate conversion problems.
485 * @retval VWRN_NUMBER_TOO_BIG
486 * @retval VWRN_NEGATIVE_UNSIGNED
487 * @retval VINF_SUCCESS
488 * @retval VERR_NO_DIGITS
489 * @retval VERR_TRAILING_SPACES
490 * @retval VERR_TRAILING_CHARS
491 *
492 * @param pszValue Pointer to the string value.
493 * @param uBase The base of the representation used.
494 * If the function will look for known prefixes before defaulting to 10.
495 * @param pu8 Where to store the converted number. (optional)
496 */
497RTDECL(int) RTStrToUInt8Full(const char *pszValue, unsigned uBase, uint8_t *pu8)
498{
499 uint64_t u64;
500 int rc = RTStrToUInt64Full(pszValue, uBase, &u64);
501 if (RT_SUCCESS(rc))
502 {
503 if (u64 & ~0xffULL)
504 rc = VWRN_NUMBER_TOO_BIG;
505 }
506 if (pu8)
507 *pu8 = (uint8_t)u64;
508 return rc;
509}
510RT_EXPORT_SYMBOL(RTStrToUInt8Full);
511
512
513/**
514 * Converts a string representation of a number to a 8-bit unsigned number.
515 * The base is guessed.
516 *
517 * @returns 8-bit unsigned number on success.
518 * @returns 0 on failure.
519 * @param pszValue Pointer to the string value.
520 */
521RTDECL(uint8_t) RTStrToUInt8(const char *pszValue)
522{
523 uint8_t u8;
524 int rc = RTStrToUInt8Ex(pszValue, NULL, 0, &u8);
525 if (RT_SUCCESS(rc))
526 return u8;
527 return 0;
528}
529RT_EXPORT_SYMBOL(RTStrToUInt8);
530
531
532
533
534
535
536
537/**
538 * Converts a string representation of a number to a 64-bit signed number.
539 *
540 * @returns iprt status code.
541 * Warnings are used to indicate conversion problems.
542 * @retval VWRN_NUMBER_TOO_BIG
543 * @retval VWRN_TRAILING_CHARS
544 * @retval VWRN_TRAILING_SPACES
545 * @retval VINF_SUCCESS
546 * @retval VERR_NO_DIGITS
547 *
548 * @param pszValue Pointer to the string value.
549 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
550 * @param uBase The base of the representation used.
551 * If the function will look for known prefixes before defaulting to 10.
552 * @param pi64 Where to store the converted number. (optional)
553 */
554RTDECL(int) RTStrToInt64Ex(const char *pszValue, char **ppszNext, unsigned uBase, int64_t *pi64)
555{
556 const char *psz = pszValue;
557 int iShift;
558 int rc;
559 uint64_t u64;
560 unsigned char uch;
561
562 /*
563 * Positive/Negative stuff.
564 */
565 bool fPositive = true;
566 for (;; psz++)
567 {
568 if (*psz == '+')
569 fPositive = true;
570 else if (*psz == '-')
571 fPositive = !fPositive;
572 else
573 break;
574 }
575
576 /*
577 * Check for hex prefix.
578 */
579 if (!uBase)
580 {
581 if ( *psz == '0'
582 && (psz[1] == 'x' || psz[1] == 'X')
583 && g_auchDigits[(unsigned char)psz[2]] < 16)
584 {
585 uBase = 16;
586 psz += 2;
587 }
588 else if ( *psz == '0'
589 && g_auchDigits[(unsigned char)psz[1]] < 8)
590 {
591 uBase = 8;
592 psz++;
593 }
594 else
595 uBase = 10;
596 }
597 else if ( uBase == 16
598 && *psz == '0'
599 && (psz[1] == 'x' || psz[1] == 'X')
600 && g_auchDigits[(unsigned char)psz[2]] < 16)
601 psz += 2;
602
603 /*
604 * Interpret the value.
605 * Note: We only support ascii digits at this time... :-)
606 */
607 iShift = g_auchShift[uBase];
608 pszValue = psz; /* (Prefix and sign doesn't count in the digit counting.) */
609 rc = VINF_SUCCESS;
610 u64 = 0;
611 while ((uch = (unsigned char)*psz) != 0)
612 {
613 unsigned char chDigit = g_auchDigits[uch];
614 uint64_t u64Prev;
615
616 if (chDigit >= uBase)
617 break;
618
619 u64Prev = u64;
620 u64 *= uBase;
621 u64 += chDigit;
622 if (u64Prev > u64 || (u64Prev >> iShift))
623 rc = VWRN_NUMBER_TOO_BIG;
624 psz++;
625 }
626
627 if ( !(u64 & RT_BIT_64(63))
628 || (!fPositive && u64 == RT_BIT_64(63)) )
629 { /* likely */ }
630 else
631 rc = VWRN_NUMBER_TOO_BIG;
632
633 if (pi64)
634 *pi64 = fPositive ? u64 : -(int64_t)u64;
635
636 if (psz == pszValue)
637 rc = VERR_NO_DIGITS;
638
639 if (ppszNext)
640 *ppszNext = (char *)psz;
641
642 /*
643 * Warn about trailing chars/spaces.
644 */
645 if ( rc == VINF_SUCCESS
646 && *psz)
647 {
648 while (*psz == ' ' || *psz == '\t')
649 psz++;
650 rc = *psz ? VWRN_TRAILING_CHARS : VWRN_TRAILING_SPACES;
651 }
652
653 return rc;
654}
655RT_EXPORT_SYMBOL(RTStrToInt64Ex);
656
657
658/**
659 * Converts a string representation of a number to a 64-bit signed number,
660 * making sure the full string is converted.
661 *
662 * @returns iprt status code.
663 * Warnings are used to indicate conversion problems.
664 * @retval VWRN_NUMBER_TOO_BIG
665 * @retval VINF_SUCCESS
666 * @retval VERR_TRAILING_CHARS
667 * @retval VERR_TRAILING_SPACES
668 * @retval VERR_NO_DIGITS
669 *
670 * @param pszValue Pointer to the string value.
671 * @param uBase The base of the representation used.
672 * If the function will look for known prefixes before defaulting to 10.
673 * @param pi64 Where to store the converted number. (optional)
674 */
675RTDECL(int) RTStrToInt64Full(const char *pszValue, unsigned uBase, int64_t *pi64)
676{
677 char *psz;
678 int rc = RTStrToInt64Ex(pszValue, &psz, uBase, pi64);
679 if (RT_SUCCESS(rc) && *psz)
680 {
681 if (rc == VWRN_TRAILING_CHARS || rc == VWRN_TRAILING_SPACES)
682 rc = -rc;
683 else
684 {
685 while (*psz == ' ' || *psz == '\t')
686 psz++;
687 rc = *psz ? VERR_TRAILING_CHARS : VERR_TRAILING_SPACES;
688 }
689 }
690 return rc;
691}
692RT_EXPORT_SYMBOL(RTStrToInt64Full);
693
694
695/**
696 * Converts a string representation of a number to a 64-bit signed number.
697 * The base is guessed.
698 *
699 * @returns 64-bit signed number on success.
700 * @returns 0 on failure.
701 * @param pszValue Pointer to the string value.
702 */
703RTDECL(int64_t) RTStrToInt64(const char *pszValue)
704{
705 int64_t i64;
706 int rc = RTStrToInt64Ex(pszValue, NULL, 0, &i64);
707 if (RT_SUCCESS(rc))
708 return i64;
709 return 0;
710}
711RT_EXPORT_SYMBOL(RTStrToInt64);
712
713
714/**
715 * Converts a string representation of a number to a 32-bit signed number.
716 *
717 * @returns iprt status code.
718 * Warnings are used to indicate conversion problems.
719 * @retval VWRN_NUMBER_TOO_BIG
720 * @retval VWRN_TRAILING_CHARS
721 * @retval VWRN_TRAILING_SPACES
722 * @retval VINF_SUCCESS
723 * @retval VERR_NO_DIGITS
724 *
725 * @param pszValue Pointer to the string value.
726 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
727 * @param uBase The base of the representation used.
728 * If the function will look for known prefixes before defaulting to 10.
729 * @param pi32 Where to store the converted number. (optional)
730 */
731RTDECL(int) RTStrToInt32Ex(const char *pszValue, char **ppszNext, unsigned uBase, int32_t *pi32)
732{
733 int64_t i64;
734 int rc = RTStrToInt64Ex(pszValue, ppszNext, uBase, &i64);
735 if (RT_SUCCESS(rc))
736 {
737 int32_t i32 = (int32_t)i64;
738 if (i64 != (int64_t)i32)
739 rc = VWRN_NUMBER_TOO_BIG;
740 }
741 if (pi32)
742 *pi32 = (int32_t)i64;
743 return rc;
744}
745RT_EXPORT_SYMBOL(RTStrToInt32Ex);
746
747
748/**
749 * Converts a string representation of a number to a 32-bit signed number,
750 * making sure the full string is converted.
751 *
752 * @returns iprt status code.
753 * Warnings are used to indicate conversion problems.
754 * @retval VWRN_NUMBER_TOO_BIG
755 * @retval VINF_SUCCESS
756 * @retval VERR_TRAILING_CHARS
757 * @retval VERR_TRAILING_SPACES
758 * @retval VERR_NO_DIGITS
759 *
760 * @param pszValue Pointer to the string value.
761 * @param uBase The base of the representation used.
762 * If the function will look for known prefixes before defaulting to 10.
763 * @param pi32 Where to store the converted number. (optional)
764 */
765RTDECL(int) RTStrToInt32Full(const char *pszValue, unsigned uBase, int32_t *pi32)
766{
767 int64_t i64;
768 int rc = RTStrToInt64Full(pszValue, uBase, &i64);
769 if (RT_SUCCESS(rc))
770 {
771 int32_t i32 = (int32_t)i64;
772 if (i64 != (int64_t)i32)
773 rc = VWRN_NUMBER_TOO_BIG;
774 }
775 if (pi32)
776 *pi32 = (int32_t)i64;
777 return rc;
778}
779RT_EXPORT_SYMBOL(RTStrToInt32Full);
780
781
782/**
783 * Converts a string representation of a number to a 32-bit signed number.
784 * The base is guessed.
785 *
786 * @returns 32-bit signed number on success.
787 * @returns 0 on failure.
788 * @param pszValue Pointer to the string value.
789 */
790RTDECL(int32_t) RTStrToInt32(const char *pszValue)
791{
792 int32_t i32;
793 int rc = RTStrToInt32Ex(pszValue, NULL, 0, &i32);
794 if (RT_SUCCESS(rc))
795 return i32;
796 return 0;
797}
798RT_EXPORT_SYMBOL(RTStrToInt32);
799
800
801/**
802 * Converts a string representation of a number to a 16-bit signed number.
803 *
804 * @returns iprt status code.
805 * Warnings are used to indicate conversion problems.
806 * @retval VWRN_NUMBER_TOO_BIG
807 * @retval VWRN_TRAILING_CHARS
808 * @retval VWRN_TRAILING_SPACES
809 * @retval VINF_SUCCESS
810 * @retval VERR_NO_DIGITS
811 *
812 * @param pszValue Pointer to the string value.
813 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
814 * @param uBase The base of the representation used.
815 * If the function will look for known prefixes before defaulting to 10.
816 * @param pi16 Where to store the converted number. (optional)
817 */
818RTDECL(int) RTStrToInt16Ex(const char *pszValue, char **ppszNext, unsigned uBase, int16_t *pi16)
819{
820 int64_t i64;
821 int rc = RTStrToInt64Ex(pszValue, ppszNext, uBase, &i64);
822 if (RT_SUCCESS(rc))
823 {
824 int16_t i16 = (int16_t)i64;
825 if (i64 != (int64_t)i16)
826 rc = VWRN_NUMBER_TOO_BIG;
827 }
828 if (pi16)
829 *pi16 = (int16_t)i64;
830 return rc;
831}
832RT_EXPORT_SYMBOL(RTStrToInt16Ex);
833
834
835/**
836 * Converts a string representation of a number to a 16-bit signed number,
837 * making sure the full string is converted.
838 *
839 * @returns iprt status code.
840 * Warnings are used to indicate conversion problems.
841 * @retval VWRN_NUMBER_TOO_BIG
842 * @retval VINF_SUCCESS
843 * @retval VERR_TRAILING_CHARS
844 * @retval VERR_TRAILING_SPACES
845 * @retval VERR_NO_DIGITS
846 *
847 * @param pszValue Pointer to the string value.
848 * @param uBase The base of the representation used.
849 * If the function will look for known prefixes before defaulting to 10.
850 * @param pi16 Where to store the converted number. (optional)
851 */
852RTDECL(int) RTStrToInt16Full(const char *pszValue, unsigned uBase, int16_t *pi16)
853{
854 int64_t i64;
855 int rc = RTStrToInt64Full(pszValue, uBase, &i64);
856 if (RT_SUCCESS(rc))
857 {
858 int16_t i16 = (int16_t)i64;
859 if (i64 != (int64_t)i16)
860 rc = VWRN_NUMBER_TOO_BIG;
861 }
862 if (pi16)
863 *pi16 = (int16_t)i64;
864 return rc;
865}
866RT_EXPORT_SYMBOL(RTStrToInt16Full);
867
868
869/**
870 * Converts a string representation of a number to a 16-bit signed number.
871 * The base is guessed.
872 *
873 * @returns 16-bit signed number on success.
874 * @returns 0 on failure.
875 * @param pszValue Pointer to the string value.
876 */
877RTDECL(int16_t) RTStrToInt16(const char *pszValue)
878{
879 int16_t i16;
880 int rc = RTStrToInt16Ex(pszValue, NULL, 0, &i16);
881 if (RT_SUCCESS(rc))
882 return i16;
883 return 0;
884}
885RT_EXPORT_SYMBOL(RTStrToInt16);
886
887
888/**
889 * Converts a string representation of a number to a 8-bit signed number.
890 *
891 * @returns iprt status code.
892 * Warnings are used to indicate conversion problems.
893 * @retval VWRN_NUMBER_TOO_BIG
894 * @retval VWRN_TRAILING_CHARS
895 * @retval VWRN_TRAILING_SPACES
896 * @retval VINF_SUCCESS
897 * @retval VERR_NO_DIGITS
898 *
899 * @param pszValue Pointer to the string value.
900 * @param ppszNext Where to store the pointer to the first char following the number. (Optional)
901 * @param uBase The base of the representation used.
902 * If the function will look for known prefixes before defaulting to 10.
903 * @param pi8 Where to store the converted number. (optional)
904 */
905RTDECL(int) RTStrToInt8Ex(const char *pszValue, char **ppszNext, unsigned uBase, int8_t *pi8)
906{
907 int64_t i64;
908 int rc = RTStrToInt64Ex(pszValue, ppszNext, uBase, &i64);
909 if (RT_SUCCESS(rc))
910 {
911 int8_t i8 = (int8_t)i64;
912 if (i64 != (int64_t)i8)
913 rc = VWRN_NUMBER_TOO_BIG;
914 }
915 if (pi8)
916 *pi8 = (int8_t)i64;
917 return rc;
918}
919RT_EXPORT_SYMBOL(RTStrToInt8Ex);
920
921
922/**
923 * Converts a string representation of a number to a 8-bit signed number,
924 * making sure the full string is converted.
925 *
926 * @returns iprt status code.
927 * Warnings are used to indicate conversion problems.
928 * @retval VWRN_NUMBER_TOO_BIG
929 * @retval VINF_SUCCESS
930 * @retval VERR_TRAILING_CHARS
931 * @retval VERR_TRAILING_SPACES
932 * @retval VERR_NO_DIGITS
933 *
934 * @param pszValue Pointer to the string value.
935 * @param uBase The base of the representation used.
936 * If the function will look for known prefixes before defaulting to 10.
937 * @param pi8 Where to store the converted number. (optional)
938 */
939RTDECL(int) RTStrToInt8Full(const char *pszValue, unsigned uBase, int8_t *pi8)
940{
941 int64_t i64;
942 int rc = RTStrToInt64Full(pszValue, uBase, &i64);
943 if (RT_SUCCESS(rc))
944 {
945 int8_t i8 = (int8_t)i64;
946 if (i64 != (int64_t)i8)
947 rc = VWRN_NUMBER_TOO_BIG;
948 }
949 if (pi8)
950 *pi8 = (int8_t)i64;
951 return rc;
952}
953RT_EXPORT_SYMBOL(RTStrToInt8Full);
954
955
956/**
957 * Converts a string representation of a number to a 8-bit signed number.
958 * The base is guessed.
959 *
960 * @returns 8-bit signed number on success.
961 * @returns 0 on failure.
962 * @param pszValue Pointer to the string value.
963 */
964RTDECL(int8_t) RTStrToInt8(const char *pszValue)
965{
966 int8_t i8;
967 int rc = RTStrToInt8Ex(pszValue, NULL, 0, &i8);
968 if (RT_SUCCESS(rc))
969 return i8;
970 return 0;
971}
972RT_EXPORT_SYMBOL(RTStrToInt8);
973
974
975RTDECL(int) RTStrConvertHexBytes(char const *pszHex, void *pv, size_t cb, uint32_t fFlags)
976{
977 size_t cbDst;
978 uint8_t *pbDst;
979 const char *pszSrc;
980
981 AssertPtrReturn(pszHex, VERR_INVALID_POINTER);
982 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
983
984 cbDst = cb;
985 pbDst = (uint8_t *)pv;
986 pszSrc = pszHex;
987 for (;;)
988 {
989 /* Pick the next two digit from the string. */
990 char ch = *pszSrc++;
991 unsigned char uchDigit1 = g_auchDigits[(unsigned char)ch];
992 unsigned char uchDigit2;
993 if (uchDigit1 >= 16)
994 {
995 if (!ch)
996 return cbDst == 0 ? VINF_SUCCESS : VERR_BUFFER_UNDERFLOW;
997
998 while (ch == ' ' || ch == '\t')
999 ch = *pszSrc++;
1000 return ch ? VWRN_TRAILING_CHARS : VWRN_TRAILING_SPACES;
1001 }
1002
1003 ch = *pszSrc++;
1004 uchDigit2 = g_auchDigits[(unsigned char)ch];
1005 if (uchDigit2 >= 16)
1006 return VERR_UNEVEN_INPUT;
1007
1008 /* Add the byte to the output buffer. */
1009 if (!cbDst)
1010 return VERR_BUFFER_OVERFLOW;
1011 cbDst--;
1012 *pbDst++ = (uchDigit1 << 4) | uchDigit2;
1013 }
1014}
1015RT_EXPORT_SYMBOL(RTStrConvertHexBytes);
1016
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette