use _GL_ATTRIBUTE_CONST and _GL_ATTRIBUTE_PURE
[gnulib.git] / lib / c-ctype.c
1 /* Character handling in C locale.
2
3    Copyright 2000-2003, 2006, 2009-2011 Free Software Foundation, Inc.
4
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
9
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software Foundation,
17 Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  */
18
19 #include <config.h>
20
21 /* Specification.  */
22 #define NO_C_CTYPE_MACROS
23 #include "c-ctype.h"
24
25 /* The attribute __const__ was added in gcc 2.95.  */
26 #undef _GL_ATTRIBUTE_CONST
27 #if __GNUC__ > 2 || (__GNUC__ == 2 && __GNUC_MINOR__ >= 95)
28 # define _GL_ATTRIBUTE_CONST __attribute__ ((__const__))
29 #else
30 # define _GL_ATTRIBUTE_CONST /* empty */
31 #endif
32
33 /* The function isascii is not locale dependent. Its use in EBCDIC is
34    questionable. */
35 bool _GL_ATTRIBUTE_CONST
36 c_isascii (int c)
37 {
38   return (c >= 0x00 && c <= 0x7f);
39 }
40
41 bool
42 c_isalnum (int c)
43 {
44 #if C_CTYPE_CONSECUTIVE_DIGITS \
45     && C_CTYPE_CONSECUTIVE_UPPERCASE && C_CTYPE_CONSECUTIVE_LOWERCASE
46 #if C_CTYPE_ASCII
47   return ((c >= '0' && c <= '9')
48           || ((c & ~0x20) >= 'A' && (c & ~0x20) <= 'Z'));
49 #else
50   return ((c >= '0' && c <= '9')
51           || (c >= 'A' && c <= 'Z')
52           || (c >= 'a' && c <= 'z'));
53 #endif
54 #else
55   switch (c)
56     {
57     case '0': case '1': case '2': case '3': case '4': case '5':
58     case '6': case '7': case '8': case '9':
59     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
60     case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
61     case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
62     case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
63     case 'Y': case 'Z':
64     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
65     case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
66     case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
67     case 's': case 't': case 'u': case 'v': case 'w': case 'x':
68     case 'y': case 'z':
69       return 1;
70     default:
71       return 0;
72     }
73 #endif
74 }
75
76 bool
77 c_isalpha (int c)
78 {
79 #if C_CTYPE_CONSECUTIVE_UPPERCASE && C_CTYPE_CONSECUTIVE_LOWERCASE
80 #if C_CTYPE_ASCII
81   return ((c & ~0x20) >= 'A' && (c & ~0x20) <= 'Z');
82 #else
83   return ((c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z'));
84 #endif
85 #else
86   switch (c)
87     {
88     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
89     case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
90     case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
91     case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
92     case 'Y': case 'Z':
93     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
94     case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
95     case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
96     case 's': case 't': case 'u': case 'v': case 'w': case 'x':
97     case 'y': case 'z':
98       return 1;
99     default:
100       return 0;
101     }
102 #endif
103 }
104
105 bool
106 c_isblank (int c)
107 {
108   return (c == ' ' || c == '\t');
109 }
110
111 bool
112 c_iscntrl (int c)
113 {
114 #if C_CTYPE_ASCII
115   return ((c & ~0x1f) == 0 || c == 0x7f);
116 #else
117   switch (c)
118     {
119     case ' ': case '!': case '"': case '#': case '$': case '%':
120     case '&': case '\'': case '(': case ')': case '*': case '+':
121     case ',': case '-': case '.': case '/':
122     case '0': case '1': case '2': case '3': case '4': case '5':
123     case '6': case '7': case '8': case '9':
124     case ':': case ';': case '<': case '=': case '>': case '?':
125     case '@':
126     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
127     case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
128     case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
129     case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
130     case 'Y': case 'Z':
131     case '[': case '\\': case ']': case '^': case '_': case '`':
132     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
133     case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
134     case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
135     case 's': case 't': case 'u': case 'v': case 'w': case 'x':
136     case 'y': case 'z':
137     case '{': case '|': case '}': case '~':
138       return 0;
139     default:
140       return 1;
141     }
142 #endif
143 }
144
145 bool
146 c_isdigit (int c)
147 {
148 #if C_CTYPE_CONSECUTIVE_DIGITS
149   return (c >= '0' && c <= '9');
150 #else
151   switch (c)
152     {
153     case '0': case '1': case '2': case '3': case '4': case '5':
154     case '6': case '7': case '8': case '9':
155       return 1;
156     default:
157       return 0;
158     }
159 #endif
160 }
161
162 bool
163 c_islower (int c)
164 {
165 #if C_CTYPE_CONSECUTIVE_LOWERCASE
166   return (c >= 'a' && c <= 'z');
167 #else
168   switch (c)
169     {
170     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
171     case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
172     case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
173     case 's': case 't': case 'u': case 'v': case 'w': case 'x':
174     case 'y': case 'z':
175       return 1;
176     default:
177       return 0;
178     }
179 #endif
180 }
181
182 bool
183 c_isgraph (int c)
184 {
185 #if C_CTYPE_ASCII
186   return (c >= '!' && c <= '~');
187 #else
188   switch (c)
189     {
190     case '!': case '"': case '#': case '$': case '%': case '&':
191     case '\'': case '(': case ')': case '*': case '+': case ',':
192     case '-': case '.': case '/':
193     case '0': case '1': case '2': case '3': case '4': case '5':
194     case '6': case '7': case '8': case '9':
195     case ':': case ';': case '<': case '=': case '>': case '?':
196     case '@':
197     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
198     case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
199     case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
200     case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
201     case 'Y': case 'Z':
202     case '[': case '\\': case ']': case '^': case '_': case '`':
203     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
204     case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
205     case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
206     case 's': case 't': case 'u': case 'v': case 'w': case 'x':
207     case 'y': case 'z':
208     case '{': case '|': case '}': case '~':
209       return 1;
210     default:
211       return 0;
212     }
213 #endif
214 }
215
216 bool
217 c_isprint (int c)
218 {
219 #if C_CTYPE_ASCII
220   return (c >= ' ' && c <= '~');
221 #else
222   switch (c)
223     {
224     case ' ': case '!': case '"': case '#': case '$': case '%':
225     case '&': case '\'': case '(': case ')': case '*': case '+':
226     case ',': case '-': case '.': case '/':
227     case '0': case '1': case '2': case '3': case '4': case '5':
228     case '6': case '7': case '8': case '9':
229     case ':': case ';': case '<': case '=': case '>': case '?':
230     case '@':
231     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
232     case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
233     case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
234     case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
235     case 'Y': case 'Z':
236     case '[': case '\\': case ']': case '^': case '_': case '`':
237     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
238     case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
239     case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
240     case 's': case 't': case 'u': case 'v': case 'w': case 'x':
241     case 'y': case 'z':
242     case '{': case '|': case '}': case '~':
243       return 1;
244     default:
245       return 0;
246     }
247 #endif
248 }
249
250 bool
251 c_ispunct (int c)
252 {
253 #if C_CTYPE_ASCII
254   return ((c >= '!' && c <= '~')
255           && !((c >= '0' && c <= '9')
256                || ((c & ~0x20) >= 'A' && (c & ~0x20) <= 'Z')));
257 #else
258   switch (c)
259     {
260     case '!': case '"': case '#': case '$': case '%': case '&':
261     case '\'': case '(': case ')': case '*': case '+': case ',':
262     case '-': case '.': case '/':
263     case ':': case ';': case '<': case '=': case '>': case '?':
264     case '@':
265     case '[': case '\\': case ']': case '^': case '_': case '`':
266     case '{': case '|': case '}': case '~':
267       return 1;
268     default:
269       return 0;
270     }
271 #endif
272 }
273
274 bool
275 c_isspace (int c)
276 {
277   return (c == ' ' || c == '\t'
278           || c == '\n' || c == '\v' || c == '\f' || c == '\r');
279 }
280
281 bool
282 c_isupper (int c)
283 {
284 #if C_CTYPE_CONSECUTIVE_UPPERCASE
285   return (c >= 'A' && c <= 'Z');
286 #else
287   switch (c)
288     {
289     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
290     case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
291     case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
292     case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
293     case 'Y': case 'Z':
294       return 1;
295     default:
296       return 0;
297     }
298 #endif
299 }
300
301 bool
302 c_isxdigit (int c)
303 {
304 #if C_CTYPE_CONSECUTIVE_DIGITS \
305     && C_CTYPE_CONSECUTIVE_UPPERCASE && C_CTYPE_CONSECUTIVE_LOWERCASE
306 #if C_CTYPE_ASCII
307   return ((c >= '0' && c <= '9')
308           || ((c & ~0x20) >= 'A' && (c & ~0x20) <= 'F'));
309 #else
310   return ((c >= '0' && c <= '9')
311           || (c >= 'A' && c <= 'F')
312           || (c >= 'a' && c <= 'f'));
313 #endif
314 #else
315   switch (c)
316     {
317     case '0': case '1': case '2': case '3': case '4': case '5':
318     case '6': case '7': case '8': case '9':
319     case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
320     case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
321       return 1;
322     default:
323       return 0;
324     }
325 #endif
326 }
327
328 int
329 c_tolower (int c)
330 {
331 #if C_CTYPE_CONSECUTIVE_UPPERCASE && C_CTYPE_CONSECUTIVE_LOWERCASE
332   return (c >= 'A' && c <= 'Z' ? c - 'A' + 'a' : c);
333 #else
334   switch (c)
335     {
336     case 'A': return 'a';
337     case 'B': return 'b';
338     case 'C': return 'c';
339     case 'D': return 'd';
340     case 'E': return 'e';
341     case 'F': return 'f';
342     case 'G': return 'g';
343     case 'H': return 'h';
344     case 'I': return 'i';
345     case 'J': return 'j';
346     case 'K': return 'k';
347     case 'L': return 'l';
348     case 'M': return 'm';
349     case 'N': return 'n';
350     case 'O': return 'o';
351     case 'P': return 'p';
352     case 'Q': return 'q';
353     case 'R': return 'r';
354     case 'S': return 's';
355     case 'T': return 't';
356     case 'U': return 'u';
357     case 'V': return 'v';
358     case 'W': return 'w';
359     case 'X': return 'x';
360     case 'Y': return 'y';
361     case 'Z': return 'z';
362     default: return c;
363     }
364 #endif
365 }
366
367 int
368 c_toupper (int c)
369 {
370 #if C_CTYPE_CONSECUTIVE_UPPERCASE && C_CTYPE_CONSECUTIVE_LOWERCASE
371   return (c >= 'a' && c <= 'z' ? c - 'a' + 'A' : c);
372 #else
373   switch (c)
374     {
375     case 'a': return 'A';
376     case 'b': return 'B';
377     case 'c': return 'C';
378     case 'd': return 'D';
379     case 'e': return 'E';
380     case 'f': return 'F';
381     case 'g': return 'G';
382     case 'h': return 'H';
383     case 'i': return 'I';
384     case 'j': return 'J';
385     case 'k': return 'K';
386     case 'l': return 'L';
387     case 'm': return 'M';
388     case 'n': return 'N';
389     case 'o': return 'O';
390     case 'p': return 'P';
391     case 'q': return 'Q';
392     case 'r': return 'R';
393     case 's': return 'S';
394     case 't': return 'T';
395     case 'u': return 'U';
396     case 'v': return 'V';
397     case 'w': return 'W';
398     case 'x': return 'X';
399     case 'y': return 'Y';
400     case 'z': return 'Z';
401     default: return c;
402     }
403 #endif
404 }