summaryrefslogtreecommitdiff
path: root/libc/sysdeps/linux/common/bits/uClibc_ctype.h
blob: 87507045220694d44a90557ce663b8db3ad9dc76 (plain)
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
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
/*  Copyright (C) 2002     Manuel Novoa III
 *
 *  This library is free software; you can redistribute it and/or
 *  modify it under the terms of the GNU Library General Public
 *  License as published by the Free Software Foundation; either
 *  version 2 of the License, or (at your option) any later version.
 *
 *  This library is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *  Library General Public License for more details.
 *
 *  You should have received a copy of the GNU Library General Public
 *  License along with this library; if not, write to the Free
 *  Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 */

/*  ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION!
 *
 *  Besides uClibc, I'm using this code in my libc for elks, which is
 *  a 16-bit environment with a fairly limited compiler.  It would make
 *  things much easier for me if this file isn't modified unnecessarily.
 *  In particular, please put any new or replacement functions somewhere
 *  else, and modify the makefile to use your version instead.
 *  Thanks.  Manuel
 *
 *  ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION! */

#if !defined(_CTYPE_H) && !defined(_WCTYPE_H)
#error Always include <{w}ctype.h> rather than <bits/uClibc_ctype.h>
#endif

#ifndef _BITS_CTYPE_H
#define _BITS_CTYPE_H

/* Taking advantage of the C99 mutual-exclusion guarantees for the various
 * (w)ctype classes, including the descriptions of printing and control
 * (w)chars, we can place each in one of the following mutually-exlusive
 * subsets.  Since there are less than 16, we can store the data for
 * each (w)chars in a nibble. In contrast, glibc uses an unsigned int
 * per (w)char, with one bit flag for each is* type.  While this allows
 * a simple '&' operation to determine the type vs. a range test and a
 * little special handling for the "blank" and "xdigit" types in my
 * approach, it also uses 8 times the space for the tables on the typical
 * 32-bit archs we supported.*/
enum {
	__CTYPE_unclassified = 0,
	__CTYPE_alpha_nonupper_nonlower,
	__CTYPE_alpha_lower,
	__CTYPE_alpha_upper_lower,
	__CTYPE_alpha_upper,
	__CTYPE_digit,
	__CTYPE_punct,
	__CTYPE_graph,
	__CTYPE_print_space_nonblank,
	__CTYPE_print_space_blank,
	__CTYPE_space_nonblank_noncntrl,
	__CTYPE_space_blank_noncntrl,
	__CTYPE_cntrl_space_nonblank,
	__CTYPE_cntrl_space_blank,
	__CTYPE_cntrl_nonspace,
};

/* Some macros that test for various (w)ctype classes when passed one of the
 * designator values enumerated above. */
#define __CTYPE_isalnum(D)		((unsigned int)(D-1) <= (__CTYPE_digit-1))
#define __CTYPE_isalpha(D)		((unsigned int)(D-1) <= (__CTYPE_alpha_upper-1))
#define __CTYPE_isblank(D) \
	((((unsigned int)(D - __CTYPE_print_space_nonblank)) <= 5) && (D & 1))
#define __CTYPE_iscntrl(D)		(((unsigned int)(D - __CTYPE_cntrl_space_nonblank)) <= 2)
#define __CTYPE_isdigit(D)		(D == __CTYPE_digit)
#define __CTYPE_isgraph(D)		((unsigned int)(D-1) <= (__CTYPE_graph-1))
#define __CTYPE_islower(D)		(((unsigned int)(D - __CTYPE_alpha_lower)) <= 1)
#define __CTYPE_isprint(D)		((unsigned int)(D-1) <= (__CTYPE_print_space_blank-1))
#define __CTYPE_ispunct(D)		(D == __CTYPE_punct)
#define __CTYPE_isspace(D)		(((unsigned int)(D - __CTYPE_print_space_nonblank)) <= 5)
#define __CTYPE_isupper(D)		(((unsigned int)(D - __CTYPE_alpha_upper_lower)) <= 1)
/*  #define __CTYPE_isxdigit(D) -- isxdigit is untestable this way. 
 *  But that's ok as isxdigit() (and isdigit() too) are locale-invariant. */

/* The values for wctype_t. */
enum {
	_CTYPE_unclassified = 0,
	_CTYPE_isalnum,
	_CTYPE_isalpha,
	_CTYPE_isblank,
	_CTYPE_iscntrl,
	_CTYPE_isdigit,
	_CTYPE_isgraph,
	_CTYPE_islower,
	_CTYPE_isprint,
	_CTYPE_ispunct,
	_CTYPE_isspace,
	_CTYPE_isupper,
	_CTYPE_isxdigit				/* _MUST_ be last of the standard classes! */
};


/* The following is used to implement wctype(), but it is defined
 * here because the ordering must agree with that of the enumeration
 * above (ignoring unclassified). */
#define __CTYPE_TYPESTRING \
	"\6alnum\0\6alpha\0\6blank\0\6cntrl\0\6digit\0\6graph\0\6lower\0" \
	"\6print\0\6punct\0\6space\0\6upper\0\7xdigit\0\0"

/* Used in implementing iswctype(), but defined here as it must agree
 * in ordering with the string above. */
#define __CTYPE_RANGES \
	0, -1,								/* unclassified */ \
	1, __CTYPE_digit - 1,				/* alnum */ \
	1, __CTYPE_alpha_upper - 1,			/* alpha */ \
	__CTYPE_print_space_blank, 5,		/* blank -- also must be odd! */ \
	__CTYPE_cntrl_space_nonblank, 2,	/* cntrl */ \
	__CTYPE_digit, 0,					/* digit */ \
	1, __CTYPE_graph - 1,				/* graph */ \
	__CTYPE_alpha_lower, 1,				/* lower */ \
	1, __CTYPE_print_space_blank - 1,	/* print */ \
	__CTYPE_punct, 0,					/* punct */ \
	__CTYPE_print_space_nonblank, 5, 	/* space */ \
	__CTYPE_alpha_upper_lower, 1,		/* upper */ \
	/* No entry for xdigit as it is handled specially. */

#define _CTYPE_iswalnum		_CTYPE_isalnum
#define _CTYPE_iswalpha		_CTYPE_isalpha
#define _CTYPE_iswblank		_CTYPE_isblank
#define _CTYPE_iswcntrl		_CTYPE_iscntrl
#define _CTYPE_iswdigit		_CTYPE_isdigit
#define _CTYPE_iswgraph		_CTYPE_isgraph
#define _CTYPE_iswlower		_CTYPE_islower
#define _CTYPE_iswprint		_CTYPE_isprint
#define _CTYPE_iswpunct		_CTYPE_ispunct
#define _CTYPE_iswspace		_CTYPE_isspace
#define _CTYPE_iswupper		_CTYPE_isupper
#define _CTYPE_iswxdigit	_CTYPE_isxdigit

/* The following is used to implement wctrans(). */

enum {
	_CTYPE_tolower = 1,
	_CTYPE_toupper,
	_CTYPE_totitle
};

#define __CTYPE_TRANSTRING	"\10tolower\0\10toupper\0\10totitle\0\0"

/* Now define some ctype macros valid for the C/POSIX locale. */

/* ASCII ords of \t, \f, \n, \r, and \v are 9, 12, 10, 13, 11 respectively. */
#define __C_isspace(c) \
	((sizeof(c) == sizeof(char)) \
	 ? ((((c) == ' ') || (((unsigned char)((c) - 9)) <= (13 - 9)))) \
	 : ((((c) == ' ') || (((unsigned int)((c) - 9)) <= (13 - 9)))))
#define __C_isblank(c) (((c) == ' ') || ((c) == '\t'))
#define __C_isdigit(c) \
	((sizeof(c) == sizeof(char)) \
	 ? (((unsigned char)((c) - '0')) < 10) \
	 : (((unsigned int)((c) - '0')) < 10))
#define __C_isxdigit(c) \
	(__C_isdigit(c) \
	 || ((sizeof(c) == sizeof(char)) \
		 ? (((unsigned char)((((c)) | 0x20) - 'a')) < 6) \
		 : (((unsigned int)((((c)) | 0x20) - 'a')) < 6)))
#define __C_iscntrl(c) \
	((sizeof(c) == sizeof(char)) \
	 ? ((((unsigned char)(c)) < 0x20) || ((c) == 0x7f)) \
	 : ((((unsigned int)(c)) < 0x20) || ((c) == 0x7f)))
#define __C_isalpha(c) \
	((sizeof(c) == sizeof(char)) \
	 ? (((unsigned char)(((c) | 0x20) - 'a')) < 26) \
	 : (((unsigned int)(((c) | 0x20) - 'a')) < 26))
#define __C_isalnum(c) (__C_isalpha(c) || __C_isdigit(c))
#define __C_isprint(c) \
	((sizeof(c) == sizeof(char)) \
	 ? (((unsigned char)((c) - 0x20)) <= (0x7e - 0x20)) \
	 : (((unsigned int)((c) - 0x20)) <= (0x7e - 0x20)))
#define __C_islower(c) \
	((sizeof(c) == sizeof(char)) \
	 ? (((unsigned char)((c) - 'a')) < 26) \
	 : (((unsigned int)((c) - 'a')) < 26))
#define __C_isupper(c) \
	((sizeof(c) == sizeof(char)) \
	 ? (((unsigned char)((c) - 'A')) < 26) \
	 : (((unsigned int)((c) - 'A')) < 26))
#define __C_ispunct(c) \
	((!__C_isalnum(c)) \
	 && ((sizeof(c) == sizeof(char)) \
		 ? (((unsigned char)((c) - 0x21)) <= (0x7e - 0x21)) \
		 : (((unsigned int)((c) - 0x21)) <= (0x7e - 0x21))))
#define __C_isgraph(c) \
	((sizeof(c) == sizeof(char)) \
	 ? (((unsigned int)((c) - 0x21)) <= (0x7e - 0x21)) \
	 : (((unsigned int)((c) - 0x21)) <= (0x7e - 0x21)))

#define __C_tolower(c) (__C_isupper(c) ? ((c) | 0x20) : (c))
#define __C_toupper(c) (__C_islower(c) ? ((c) ^ 0x20) : (c))

#define __C_isxlower(c) \
	(__C_isdigit(c) \
	 || ((sizeof(c) == sizeof(char)) \
		 ? (((unsigned char)(((c)) - 'a')) < 6) \
		 : (((unsigned int)(((c)) - 'a')) < 6)))
#define __C_isxupper(c) \
	(__C_isdigit(c) \
	 || ((sizeof(c) == sizeof(char)) \
		 ? (((unsigned char)(((c)) - 'A')) < 6) \
		 : (((unsigned int)(((c)) - 'A')) < 6)))

/* TODO: Replace the above with expressions like the following? */
/*  #define __C_isdigit(c) ((sizeof(c) == sizeof(char)) \ */
/*  						? (((unsigned char)((c) - '0')) < 10) \ */
/*  						: (((unsigned int)((c) - '0')) < 10)) */

/* Similarly, define some wctype macros valid for the C/POSIX locale. */

/* First, we need some way to make sure the arg is in range. */
#define __C_classed(c) \
	((sizeof(c) <= sizeof(int)) || (c == ((unsigned char)c)))

#define __C_iswspace(c)		(__C_classed(c) && __C_isspace(c))
#define __C_iswblank(c)		(__C_classed(c) && __C_isblank(c))
#define __C_iswdigit(c)		(__C_classed(c) && __C_isdigit(c))
#define __C_iswxdigit(c)	(__C_classed(c) && __C_isxdigit(c))
#define __C_iswcntrl(c)		(__C_classed(c) && __C_iscntrl(c))
#define __C_iswalpha(c)		(__C_classed(c) && __C_isalpha(c))
#define __C_iswalnum(c)		(__C_classed(c) && __C_isalnum(c))
#define __C_iswprint(c)		(__C_classed(c) && __C_isprint(c))
#define __C_iswlower(c)		(__C_classed(c) && __C_islower(c))
#define __C_iswupper(c)		(__C_classed(c) && __C_isupper(c))
#define __C_iswpunct(c)		(__C_classed(c) && __C_ispunct(c))
#define __C_iswgraph(c)		(__C_classed(c) && __C_isgraph(c))
#define __C_towlower(c) \
	((__C_classed(c) && __C_isupper(c)) ? ((c) | 0x20) : (c))
#define __C_towupper(c) \
	((__C_classed(c) && __C_islower(c)) ? ((c) ^ 0x20) : (c))

/* Now define some macros to aviod the extra wrapper-function call. */
#define __iswalnum(c)		iswctype(c, _CTYPE_iswalnum)
#define __iswalpha(c)		iswctype(c, _CTYPE_iswalpha)
#define __iswblank(c)		iswctype(c, _CTYPE_iswblank)
#define __iswcntrl(c)		iswctype(c, _CTYPE_iswcntrl)
#define __iswgraph(c)		iswctype(c, _CTYPE_iswgraph)
#define __iswlower(c)		iswctype(c, _CTYPE_iswlower)
#define __iswprint(c)		iswctype(c, _CTYPE_iswprint)
#define __iswpunct(c)		iswctype(c, _CTYPE_iswpunct)
#define __iswspace(c)		iswctype(c, _CTYPE_iswspace)
#define __iswupper(c)		iswctype(c, _CTYPE_iswupper)
#define __iswdigit(c)		__C_iswdigit(c)
#define __iswxdigit(c)		__C_iswxdigit(c)

#endif /* _BITS_CTYPE_H */