1cd62a92dSRobert Mustacchi /*
2cd62a92dSRobert Mustacchi  * Copyright (c) 2016 Ingo Schwarze <schwarze@openbsd.org>
3cd62a92dSRobert Mustacchi  *
4cd62a92dSRobert Mustacchi  * Permission to use, copy, modify, and distribute this software for any
5cd62a92dSRobert Mustacchi  * purpose with or without fee is hereby granted, provided that the above
6cd62a92dSRobert Mustacchi  * copyright notice and this permission notice appear in all copies.
7cd62a92dSRobert Mustacchi  *
8cd62a92dSRobert Mustacchi  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9cd62a92dSRobert Mustacchi  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10cd62a92dSRobert Mustacchi  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11cd62a92dSRobert Mustacchi  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12cd62a92dSRobert Mustacchi  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13cd62a92dSRobert Mustacchi  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14cd62a92dSRobert Mustacchi  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15cd62a92dSRobert Mustacchi  */
16cd62a92dSRobert Mustacchi 
17cd62a92dSRobert Mustacchi #include <sys/types.h>
18cd62a92dSRobert Mustacchi #include <err.h>
19cd62a92dSRobert Mustacchi #include <errno.h>
20cd62a92dSRobert Mustacchi #include <locale.h>
21cd62a92dSRobert Mustacchi #include <stdlib.h>
22cd62a92dSRobert Mustacchi #include <string.h>
23cd62a92dSRobert Mustacchi #include <wchar.h>
24cd62a92dSRobert Mustacchi 
25cd62a92dSRobert Mustacchi static mbstate_t	 mbs;
26cd62a92dSRobert Mustacchi 
27cd62a92dSRobert Mustacchi void
onetest(const char * name,const char * in,size_t insz,int outerr,size_t outsz,wint_t out)28cd62a92dSRobert Mustacchi onetest(const char *name, const char *in, size_t insz,
29cd62a92dSRobert Mustacchi     int outerr, size_t outsz, wint_t out)
30cd62a92dSRobert Mustacchi {
31cd62a92dSRobert Mustacchi 	wchar_t		 wc;
32cd62a92dSRobert Mustacchi 	size_t		 sz;
33cd62a92dSRobert Mustacchi 
34cd62a92dSRobert Mustacchi 	sz = mbrtowc(&wc, in, insz, &mbs);
35cd62a92dSRobert Mustacchi 	if (errno != outerr)
36*6353250fSRobert Mustacchi 		err(1, "%u %s(%zd)", MB_CUR_MAX, name, insz);
37cd62a92dSRobert Mustacchi 	if (sz != outsz || (out != WEOF && wc != out))
38*6353250fSRobert Mustacchi 		errx(1, "%u %s(%zu) = (%zu, %" _PRIdWC ") != (%zu, %" _PRIdWC
39*6353250fSRobert Mustacchi 		    ")", MB_CUR_MAX, name, insz, sz, wc, outsz, out);
40cd62a92dSRobert Mustacchi 	if (mbsinit(&mbs) == (insz && outsz == (size_t)-2))
41*6353250fSRobert Mustacchi 		errx(1, "%u %s(%zd) mbsinit", MB_CUR_MAX, name, insz);
42cd62a92dSRobert Mustacchi 	if (errno == 0 && outerr == 0)
43cd62a92dSRobert Mustacchi 		return;
44cd62a92dSRobert Mustacchi 	errno = 0;
45cd62a92dSRobert Mustacchi 	memset(&mbs, 0, sizeof (mbs));
46cd62a92dSRobert Mustacchi }
47cd62a92dSRobert Mustacchi 
48cd62a92dSRobert Mustacchi int
main(void)49cd62a92dSRobert Mustacchi main(void)
50cd62a92dSRobert Mustacchi {
51cd62a92dSRobert Mustacchi 	onetest("NUL", "", 0, 0, -2, WEOF);
52cd62a92dSRobert Mustacchi 	onetest("NUL", "", 2, 0, 0, L'\0');
53cd62a92dSRobert Mustacchi 	onetest("BEL", "\a", 2, 0, 1, L'\a');
54cd62a92dSRobert Mustacchi 	onetest("A", "A", 2, 0, 1, L'A');
55cd62a92dSRobert Mustacchi 	onetest("DEL", "\177", 2, 0, 1, L'\177');
56cd62a92dSRobert Mustacchi 	onetest("CSI", "\233", 2, 0, 1, L'\233');
57cd62a92dSRobert Mustacchi 
58cd62a92dSRobert Mustacchi 	if (setlocale(LC_CTYPE, "en_US.UTF-8") == NULL)
59cd62a92dSRobert Mustacchi 		errx(1, "setlocale(UTF-8) failed");
60cd62a92dSRobert Mustacchi 
61cd62a92dSRobert Mustacchi 	onetest("NUL", "", 0, 0, -2, WEOF);
62cd62a92dSRobert Mustacchi 	onetest("NUL", "", 8, 0, 0, L'\0');
63cd62a92dSRobert Mustacchi 	onetest("BEL", "\a", 8, 0, 1, L'\a');
64cd62a92dSRobert Mustacchi 	onetest("A", "A", 8, 0, 1, L'A');
65cd62a92dSRobert Mustacchi 	onetest("DEL", "\177", 8, 0, 1, L'\177');
66cd62a92dSRobert Mustacchi 	onetest("0x80", "\200", 8, EILSEQ, -1, WEOF);
67cd62a92dSRobert Mustacchi 	onetest("0xc3", "\303", 1, 0, -2, WEOF);
68cd62a92dSRobert Mustacchi 	onetest("U+00E9", "\251", 8, 0, 1, 0xe9);
69cd62a92dSRobert Mustacchi 	onetest("0xec", "\354", 1, 0, -2, WEOF);
70cd62a92dSRobert Mustacchi 	onetest("0xecbf", "\277", 1, 0, -2, WEOF);
71cd62a92dSRobert Mustacchi 	onetest("U+CFFF", "\277", 8, 0, 1, 0xcfff);
72cd62a92dSRobert Mustacchi 
73cd62a92dSRobert Mustacchi 	if (setlocale(LC_CTYPE, "POSIX") == NULL)
74cd62a92dSRobert Mustacchi 		errx(1, "setlocale(POSIX) failed");
75cd62a92dSRobert Mustacchi 
76cd62a92dSRobert Mustacchi 	onetest("0xff", "\277", 2, 0, 1, L'\277');
77cd62a92dSRobert Mustacchi 
78cd62a92dSRobert Mustacchi 	if (setlocale(LC_CTYPE, "en_US.UTF-8") == NULL)
79cd62a92dSRobert Mustacchi 		errx(1, "second setlocale(UTF-8) failed");
80cd62a92dSRobert Mustacchi 
81cd62a92dSRobert Mustacchi 	onetest("U+13000", "\360\223\200\200", 8, 0, 4, 0x13000);
82cd62a92dSRobert Mustacchi 
83cd62a92dSRobert Mustacchi 	return (0);
84cd62a92dSRobert Mustacchi }
85