Line data Source code
1 : #define _GNU_SOURCE
2 : #include "fd_cstr.h"
3 :
4 : /* FIXME: WEAN THIS OFF STDLIB FOR NON-HOSTED TARGETS */
5 : #include <stdio.h>
6 : #include <stdlib.h>
7 : #include <stdarg.h>
8 : #include <strings.h>
9 : #include <ctype.h>
10 :
11 2 : char const * fd_cstr_to_cstr ( char const * cstr ) { return cstr; }
12 0 : char fd_cstr_to_char ( char const * cstr ) { return cstr[0]; }
13 0 : schar fd_cstr_to_schar ( char const * cstr ) { return (schar) strtol ( cstr, NULL, 0 ); }
14 0 : short fd_cstr_to_short ( char const * cstr ) { return (short) strtol ( cstr, NULL, 0 ); }
15 0 : int fd_cstr_to_int ( char const * cstr ) { return (int) strtol ( cstr, NULL, 0 ); }
16 0 : long fd_cstr_to_long ( char const * cstr ) { return (long) strtol ( cstr, NULL, 0 ); }
17 0 : uchar fd_cstr_to_uchar ( char const * cstr ) { return (uchar) strtoul( cstr, NULL, 0 ); }
18 0 : ushort fd_cstr_to_ushort( char const * cstr ) { return (ushort)strtoul( cstr, NULL, 0 ); }
19 0 : uint fd_cstr_to_uint ( char const * cstr ) { return (uint) strtoul( cstr, NULL, 0 ); }
20 2 : ulong fd_cstr_to_ulong ( char const * cstr ) { return (ulong) strtoul( cstr, NULL, 0 ); }
21 0 : float fd_cstr_to_float ( char const * cstr ) { return strtof ( cstr, NULL ); }
22 : #if FD_HAS_DOUBLE
23 0 : double fd_cstr_to_double( char const * cstr ) { return strtod ( cstr, NULL ); }
24 : #endif
25 :
26 0 : ulong fd_cstr_to_ulong_octal( char const * cstr ) { return (ulong)strtoul( cstr, NULL, 8 ); }
27 :
28 : ulong
29 : fd_cstr_to_ulong_seq( char const * cstr,
30 : ulong * seq,
31 0 : ulong seq_max ) {
32 0 : ulong seq_cnt = 0UL;
33 :
34 0 : if( !cstr ) return seq_cnt;
35 :
36 0 : char const * p = cstr;
37 0 : for(;;) {
38 :
39 0 : char c;
40 0 : char * q;
41 :
42 0 : c = *p; while( fd_isspace( (int)c ) ) c = *(++p); /* Move and peek at next non-white-space character */
43 0 : if( c=='\0' ) break; /* end of sequence */
44 :
45 0 : ulong seq_ele_0 = strtoul( p, &q, 0 );
46 0 : if( FD_UNLIKELY( p==(char const *)q ) ) return 0UL; /* Malformed sequence, seq_ele_0 is not a ulong */
47 0 : p = (char const *)q;
48 :
49 0 : ulong seq_ele_1 = seq_ele_0;
50 0 : ulong seq_stride = 1UL;
51 :
52 0 : c = *p; while( fd_isspace( (int)c ) ) c = *(++p); /* Move and peek at next non-white-space character */
53 0 : if( c=='-' ) {
54 0 : p++;
55 :
56 0 : seq_ele_1 = strtoul( p, &q, 0 );
57 0 : if( FD_UNLIKELY( p==(char const *)q ) ) return 0UL; /* Malformed sequence, seq_ele_1 is not a ulong */
58 0 : p = (char const *)q;
59 :
60 0 : c = *p; while( fd_isspace( (int)c ) ) c = *(++p); /* Move and peek at next non-white-space character */
61 0 : if( c=='/' || c==':' ) {
62 0 : p++;
63 :
64 0 : seq_stride = strtoul( p, &q, 0 );
65 0 : if( FD_UNLIKELY( p==(char const *)q ) ) return 0UL; /* Malformed sequence, seq_stride is not a ulong */
66 0 : p = (char const *)q;
67 0 : }
68 0 : }
69 :
70 0 : c = *p; while( fd_isspace( (int)c ) ) c = *(++p); /* Move and peek at next non-white-space character */
71 0 : if( !(c==',' || c=='\0' ) ) return 0UL; /* Malformed sequence, delimiter */
72 0 : if( c==',' ) p++;
73 :
74 : /* Append the range to sequence. Written this slightly funny way to
75 : be robust against overflow with seq_ele_1 and/or seq_stride being
76 : near or equal to ULONG_MAX */
77 :
78 0 : if( FD_UNLIKELY( (seq_ele_1<seq_ele_0) | (!seq_stride) )) return 0UL; /* Malformed sequence, bad range */
79 :
80 0 : ulong seq_ele = seq_ele_0;
81 0 : while( ((seq_ele_0<=seq_ele) & (seq_ele<seq_ele_1)) ) {
82 0 : if( FD_LIKELY( seq_cnt<seq_max ) ) seq[ seq_cnt ] = seq_ele;
83 0 : seq_cnt++;
84 0 : seq_ele += seq_stride;
85 0 : }
86 0 : if( seq_ele==seq_ele_1 ) {
87 0 : if( FD_LIKELY( seq_cnt<seq_max ) ) seq[ seq_cnt ] = seq_ele;
88 0 : seq_cnt++;
89 0 : }
90 0 : }
91 :
92 0 : return seq_cnt;
93 0 : }
94 :
95 : int
96 : fd_cstr_casecmp( char const * a,
97 1 : char const * b ) {
98 1 : return strcasecmp( a, b );
99 1 : }
100 :
101 : ulong
102 : fd_cstr_nlen( char const * s,
103 0 : ulong m ) {
104 0 : return strnlen( s, m );
105 0 : }
106 :
107 : char *
108 : fd_cstr_ncpy( char * d,
109 : char const * s,
110 0 : ulong m ) {
111 0 : if( FD_LIKELY( m ) ){
112 0 : ulong i = 0UL;
113 0 : if( FD_LIKELY( s ) ) {
114 0 : for( ; i<m-1UL; i++ ) {
115 0 : char c = s[i];
116 0 : if( !c ) break;
117 0 : d[i] = c;
118 0 : }
119 0 : }
120 0 : memset( d+i, 0, m-i );
121 0 : }
122 0 : return d;
123 0 : }
124 :
125 : char *
126 : fd_cstr_printf( char * buf,
127 : ulong sz,
128 : ulong * opt_len,
129 67 : char const * fmt, ... ) {
130 67 : if( FD_UNLIKELY( (!buf) | (!sz) ) ) {
131 0 : if( opt_len ) *opt_len = 0UL;
132 0 : return buf;
133 0 : }
134 67 : va_list ap;
135 67 : va_start( ap, fmt );
136 67 : int ret = vsnprintf( buf, sz, fmt, ap );
137 67 : ulong len = fd_ulong_if( ret<0, 0UL, fd_ulong_min( (ulong)ret, sz-1UL ) );
138 67 : buf[ len ] = '\0';
139 67 : va_end( ap );
140 67 : if( opt_len ) *opt_len = len;
141 67 : return buf;
142 67 : }
143 :
144 : int
145 : fd_cstr_printf_check( char * buf,
146 : ulong sz,
147 : ulong * opt_len,
148 12 : char const * fmt, ... ) {
149 12 : if( FD_UNLIKELY( (!buf) | (!sz) ) ) {
150 0 : if( opt_len ) *opt_len = 0UL;
151 0 : return 0;
152 0 : }
153 12 : va_list ap;
154 12 : va_start( ap, fmt );
155 12 : int ret = vsnprintf( buf, sz, fmt, ap );
156 12 : ulong len = fd_ulong_if( ret<0, 0UL, fd_ulong_min( (ulong)ret, sz-1UL ) );
157 12 : buf[ len ] = '\0';
158 12 : va_end( ap );
159 12 : if( opt_len ) *opt_len = len;
160 12 : return len==(ulong)ret;
161 12 : }
162 :
163 : char *
164 : fd_cstr_append_printf( char * buf,
165 0 : char const * fmt, ... ) {
166 0 : if( FD_UNLIKELY( !buf ) ) return NULL;
167 0 : va_list ap;
168 0 : va_start( ap, fmt );
169 0 : int ret = vsprintf( buf, fmt, ap );
170 0 : va_end( ap );
171 0 : return buf + fd_ulong_if( ret<0, 0UL, (ulong)ret );
172 0 : }
173 :
174 : ulong
175 : fd_cstr_tokenize( char ** tok,
176 : ulong tok_max,
177 : char * p,
178 0 : char delim ) {
179 0 : if( FD_UNLIKELY( !p ) ) return 0UL;
180 :
181 0 : ulong tok_cnt = 0UL;
182 0 : for(;;) {
183 :
184 : /* Find token start and record it (if possible) */
185 0 : while( fd_isspace( (int)p[0] ) ) p++;
186 0 : if( p[0]=='\0' ) break;
187 0 : if( tok_cnt<tok_max ) tok[ tok_cnt ] = p;
188 0 : tok_cnt++;
189 :
190 : /* Find the token end and terminate it */
191 0 : while( ((p[0]!=delim) & (p[0]!='\0')) ) p++;
192 0 : if( p[0]=='\0' ) break;
193 0 : p[0] = '\0';
194 0 : p++;
195 0 : }
196 :
197 0 : return tok_cnt;
198 0 : }
|