fix aliasing violations in mbtowc and mbrtowc
[musl] / src / multibyte / mbtowc.c
1 /* 
2  * This code was written by Rich Felker in 2010; no copyright is claimed.
3  * This code is in the public domain. Attribution is appreciated but
4  * unnecessary.
5  */
6
7 #include <wchar.h>
8 #include <errno.h>
9 #include "internal.h"
10
11 int mbtowc(wchar_t *restrict wc, const char *restrict src, size_t n)
12 {
13         unsigned c;
14         const unsigned char *s = (const void *)src;
15         wchar_t dummy;
16
17         if (!s) return 0;
18         if (!n) goto ilseq;
19         if (!wc) wc = &dummy;
20
21         if (*s < 0x80) return !!(*wc = *s);
22         if (*s-SA > SB-SA) goto ilseq;
23         c = bittab[*s++-SA];
24
25         /* Avoid excessive checks against n: If shifting the state n-1
26          * times does not clear the high bit, then the value of n is
27          * insufficient to read a character */
28         if (n<4 && ((c<<(6*n-6)) & (1U<<31))) goto ilseq;
29
30         if (OOB(c,*s)) goto ilseq;
31         c = c<<6 | *s++-0x80;
32         if (!(c&(1U<<31))) {
33                 *wc = c;
34                 return 2;
35         }
36
37         if (*s-0x80u >= 0x40) goto ilseq;
38         c = c<<6 | *s++-0x80;
39         if (!(c&(1U<<31))) {
40                 *wc = c;
41                 return 3;
42         }
43
44         if (*s-0x80u >= 0x40) goto ilseq;
45         *wc = c<<6 | *s++-0x80;
46         return 4;
47
48 ilseq:
49         errno = EILSEQ;
50         return -1;
51 }