]> pd.if.org Git - pdclib.old/blob - strtox_main.c
1556ec4bab7a9a201740ae91de8ceb20aedb422a
[pdclib.old] / strtox_main.c
1 /* $Id$ */
2
3 /* _PDCLIB_strtox_main( const char * *, int, _PDCLIB_uintmax_t, _PDCLIB_uintmax_t, int )
4
5    This file is part of the Public Domain C Library (PDCLib).
6    Permission is granted to use, modify, and / or redistribute at will.
7 */
8
9 #include <_PDCLIB_int.h>
10 #include <ctype.h>
11 #include <errno.h>
12 #include <string.h>
13 #include <stdint.h>
14
15 _PDCLIB_uintmax_t _PDCLIB_strtox_main( const char ** p, unsigned int base, uintmax_t error, uintmax_t limval, int limdigit, char * sign )
16 {
17     _PDCLIB_uintmax_t rc = 0;
18     int digit = -1;
19     const char * x;
20     while ( ( x = memchr( _PDCLIB_digits, tolower(**p), base ) ) != NULL )
21     {
22         digit = x - _PDCLIB_digits;
23         if ( ( rc < limval ) || ( ( rc == limval ) && ( digit <= limdigit ) ) )
24         {
25             rc = rc * base + (unsigned)digit;
26             ++(*p);
27         }
28         else
29         {
30             errno = ERANGE;
31             /* TODO: Only if endptr != NULL - but do we really want *another* parameter? */
32             /* TODO: Earlier version was missing tolower() here but was not caught by tests */
33             while ( memchr( _PDCLIB_digits, tolower(**p), base ) != NULL ) ++(*p);
34             /* TODO: This is ugly, but keeps caller from negating the error value */
35             *sign = '+';
36             return error;
37         }
38     }
39     if ( digit == -1 )
40     {
41         *p = NULL;
42         return 0;
43     }
44     return rc;
45 }
46
47 #ifdef TEST
48 #include <_PDCLIB_test.h>
49 #include <errno.h>
50
51 int main( void )
52 {
53     const char * p;
54     char test[] = "123_";
55     char fail[] = "xxx";
56     char sign = '-';
57     /* basic functionality */
58     p = test;
59     errno = 0;
60     TESTCASE( _PDCLIB_strtox_main( &p, 10u, (uintmax_t)999, (uintmax_t)12, 3, &sign ) == 123 );
61     TESTCASE( errno == 0 );
62     TESTCASE( p == &test[3] );
63     /* proper functioning to smaller base */
64     p = test;
65     TESTCASE( _PDCLIB_strtox_main( &p, 8u, (uintmax_t)999, (uintmax_t)12, 3, &sign ) == 0123 );
66     TESTCASE( errno == 0 );
67     TESTCASE( p == &test[3] );
68     /* overflowing subject sequence must still return proper endptr */
69     p = test;
70     TESTCASE( _PDCLIB_strtox_main( &p, 4u, (uintmax_t)999, (uintmax_t)1, 2, &sign ) == 999 );
71     TESTCASE( errno == ERANGE );
72     TESTCASE( p == &test[3] );
73     TESTCASE( sign == '+' );
74     /* testing conversion failure */
75     errno = 0;
76     p = fail;
77     sign = '-';
78     TESTCASE( _PDCLIB_strtox_main( &p, 10u, (uintmax_t)999, (uintmax_t)99, 8, &sign ) == 0 );
79     TESTCASE( p == NULL );
80     return TEST_RESULTS;
81 }
82
83 #endif