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