xref: /dflybsd-src/lib/i18n_module/ISO2022/citrus_iso2022.c (revision d4548c479b343e1c680e0ac11a26753138a8ae59)
145203ddbSJohn Marino /* $FreeBSD: head/lib/libiconv_modules/ISO2022/citrus_iso2022.c 281550 2015-04-15 09:09:20Z tijl $ */
245203ddbSJohn Marino /*	$NetBSD: citrus_iso2022.c,v 1.20 2010/12/07 22:01:45 joerg Exp $	*/
30d5acd74SJohn Marino 
40d5acd74SJohn Marino /*-
50d5acd74SJohn Marino  * Copyright (c)1999, 2002 Citrus Project,
60d5acd74SJohn Marino  * All rights reserved.
70d5acd74SJohn Marino  *
80d5acd74SJohn Marino  * Redistribution and use in source and binary forms, with or without
90d5acd74SJohn Marino  * modification, are permitted provided that the following conditions
100d5acd74SJohn Marino  * are met:
110d5acd74SJohn Marino  * 1. Redistributions of source code must retain the above copyright
120d5acd74SJohn Marino  *    notice, this list of conditions and the following disclaimer.
130d5acd74SJohn Marino  * 2. Redistributions in binary form must reproduce the above copyright
140d5acd74SJohn Marino  *    notice, this list of conditions and the following disclaimer in the
150d5acd74SJohn Marino  *    documentation and/or other materials provided with the distribution.
160d5acd74SJohn Marino  *
170d5acd74SJohn Marino  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
180d5acd74SJohn Marino  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
190d5acd74SJohn Marino  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
200d5acd74SJohn Marino  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
210d5acd74SJohn Marino  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
220d5acd74SJohn Marino  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
230d5acd74SJohn Marino  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
240d5acd74SJohn Marino  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
250d5acd74SJohn Marino  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
260d5acd74SJohn Marino  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
270d5acd74SJohn Marino  * SUCH DAMAGE.
280d5acd74SJohn Marino  *
290d5acd74SJohn Marino  *	$Citrus: xpg4dl/FreeBSD/lib/libc/locale/iso2022.c,v 1.23 2001/06/21 01:51:44 yamt Exp $
300d5acd74SJohn Marino  */
310d5acd74SJohn Marino 
320d5acd74SJohn Marino #include <sys/cdefs.h>
330d5acd74SJohn Marino #include <sys/types.h>
340d5acd74SJohn Marino 
350d5acd74SJohn Marino #include <assert.h>
360d5acd74SJohn Marino #include <errno.h>
370d5acd74SJohn Marino #include <limits.h>
380d5acd74SJohn Marino #include <stdbool.h>
390d5acd74SJohn Marino #include <stddef.h>
400d5acd74SJohn Marino #include <stdio.h>
410d5acd74SJohn Marino #include <stdlib.h>
420d5acd74SJohn Marino #include <string.h>
430d5acd74SJohn Marino #include <wchar.h>
440d5acd74SJohn Marino 
450d5acd74SJohn Marino #include "citrus_namespace.h"
460d5acd74SJohn Marino #include "citrus_types.h"
470d5acd74SJohn Marino #include "citrus_module.h"
480d5acd74SJohn Marino #include "citrus_stdenc.h"
490d5acd74SJohn Marino #include "citrus_iso2022.h"
500d5acd74SJohn Marino 
510d5acd74SJohn Marino 
520d5acd74SJohn Marino /* ----------------------------------------------------------------------
530d5acd74SJohn Marino  * private stuffs used by templates
540d5acd74SJohn Marino  */
550d5acd74SJohn Marino 
560d5acd74SJohn Marino 
570d5acd74SJohn Marino /*
580d5acd74SJohn Marino  * wchar_t mappings:
590d5acd74SJohn Marino  * ASCII (ESC ( B)		00000000 00000000 00000000 0xxxxxxx
600d5acd74SJohn Marino  * iso-8859-1 (ESC , A)		00000000 00000000 00000000 1xxxxxxx
610d5acd74SJohn Marino  * 94 charset (ESC ( F)		0fffffff 00000000 00000000 0xxxxxxx
620d5acd74SJohn Marino  * 94 charset (ESC ( M F)	0fffffff 1mmmmmmm 00000000 0xxxxxxx
630d5acd74SJohn Marino  * 96 charset (ESC , F)		0fffffff 00000000 00000000 1xxxxxxx
640d5acd74SJohn Marino  * 96 charset (ESC , M F)	0fffffff 1mmmmmmm 00000000 1xxxxxxx
650d5acd74SJohn Marino  * 94x94 charset (ESC $ ( F)	0fffffff 00000000 0xxxxxxx 0xxxxxxx
660d5acd74SJohn Marino  * 96x96 charset (ESC $ , F)	0fffffff 00000000 0xxxxxxx 1xxxxxxx
670d5acd74SJohn Marino  * 94x94 charset (ESC & V ESC $ ( F)
680d5acd74SJohn Marino  *				0fffffff 1vvvvvvv 0xxxxxxx 0xxxxxxx
690d5acd74SJohn Marino  * 94x94x94 charset (ESC $ ( F)	0fffffff 0xxxxxxx 0xxxxxxx 0xxxxxxx
700d5acd74SJohn Marino  * 96x96x96 charset (ESC $ , F)	0fffffff 0xxxxxxx 0xxxxxxx 1xxxxxxx
710d5acd74SJohn Marino  * reserved for UCS4 co-existence (UCS4 is 31bit encoding thanks to mohta bit)
720d5acd74SJohn Marino  *				1xxxxxxx xxxxxxxx xxxxxxxx xxxxxxxx
730d5acd74SJohn Marino  */
740d5acd74SJohn Marino 
750d5acd74SJohn Marino #define CS94		(0U)
760d5acd74SJohn Marino #define CS96		(1U)
770d5acd74SJohn Marino #define CS94MULTI	(2U)
780d5acd74SJohn Marino #define CS96MULTI	(3U)
790d5acd74SJohn Marino 
800d5acd74SJohn Marino typedef struct {
810d5acd74SJohn Marino 	unsigned char	 type;
8245203ddbSJohn Marino 	unsigned char	 final;
8345203ddbSJohn Marino 	unsigned char	 interm;
840d5acd74SJohn Marino 	unsigned char	 vers;
850d5acd74SJohn Marino } _ISO2022Charset;
860d5acd74SJohn Marino 
870d5acd74SJohn Marino static const _ISO2022Charset ascii    = { CS94, 'B', '\0', '\0' };
880d5acd74SJohn Marino static const _ISO2022Charset iso88591 = { CS96, 'A', '\0', '\0' };
890d5acd74SJohn Marino 
900d5acd74SJohn Marino typedef struct {
910d5acd74SJohn Marino 	_ISO2022Charset	g[4];
920d5acd74SJohn Marino 	/* need 3 bits to hold -1, 0, ..., 3 */
930d5acd74SJohn Marino 	int	gl:3,
940d5acd74SJohn Marino 		gr:3,
950d5acd74SJohn Marino 		singlegl:3,
960d5acd74SJohn Marino 		singlegr:3;
970d5acd74SJohn Marino 	char ch[7];	/* longest escape sequence (ESC & V ESC $ ( F) */
980d5acd74SJohn Marino 	size_t chlen;
990d5acd74SJohn Marino 	int flags;
1000d5acd74SJohn Marino #define _ISO2022STATE_FLAG_INITIALIZED	1
1010d5acd74SJohn Marino } _ISO2022State;
1020d5acd74SJohn Marino 
1030d5acd74SJohn Marino typedef struct {
1040d5acd74SJohn Marino 	_ISO2022Charset	*recommend[4];
1050d5acd74SJohn Marino 	size_t	recommendsize[4];
1060d5acd74SJohn Marino 	_ISO2022Charset	initg[4];
1070d5acd74SJohn Marino 	int	maxcharset;
1080d5acd74SJohn Marino 	int	flags;
1090d5acd74SJohn Marino #define	F_8BIT	0x0001
1100d5acd74SJohn Marino #define	F_NOOLD	0x0002
1110d5acd74SJohn Marino #define	F_SI	0x0010	/*0F*/
1120d5acd74SJohn Marino #define	F_SO	0x0020	/*0E*/
1130d5acd74SJohn Marino #define	F_LS0	0x0010	/*0F*/
1140d5acd74SJohn Marino #define	F_LS1	0x0020	/*0E*/
1150d5acd74SJohn Marino #define	F_LS2	0x0040	/*ESC n*/
1160d5acd74SJohn Marino #define	F_LS3	0x0080	/*ESC o*/
1170d5acd74SJohn Marino #define	F_LS1R	0x0100	/*ESC ~*/
1180d5acd74SJohn Marino #define	F_LS2R	0x0200	/*ESC }*/
1190d5acd74SJohn Marino #define	F_LS3R	0x0400	/*ESC |*/
1200d5acd74SJohn Marino #define	F_SS2	0x0800	/*ESC N*/
1210d5acd74SJohn Marino #define	F_SS3	0x1000	/*ESC O*/
1220d5acd74SJohn Marino #define	F_SS2R	0x2000	/*8E*/
1230d5acd74SJohn Marino #define	F_SS3R	0x4000	/*8F*/
1240d5acd74SJohn Marino } _ISO2022EncodingInfo;
1250d5acd74SJohn Marino 
1260d5acd74SJohn Marino #define _CEI_TO_EI(_cei_)		(&(_cei_)->ei)
1270d5acd74SJohn Marino #define _CEI_TO_STATE(_cei_, _func_)	(_cei_)->states.s_##_func_
1280d5acd74SJohn Marino 
1290d5acd74SJohn Marino #define _FUNCNAME(m)			_citrus_ISO2022_##m
1300d5acd74SJohn Marino #define _ENCODING_INFO			_ISO2022EncodingInfo
1310d5acd74SJohn Marino #define _ENCODING_STATE			_ISO2022State
1320d5acd74SJohn Marino #define _ENCODING_MB_CUR_MAX(_ei_)	MB_LEN_MAX
1330d5acd74SJohn Marino #define _ENCODING_IS_STATE_DEPENDENT	1
1340d5acd74SJohn Marino #define _STATE_NEEDS_EXPLICIT_INIT(_ps_)	\
1350d5acd74SJohn Marino     (!((_ps_)->flags & _ISO2022STATE_FLAG_INITIALIZED))
1360d5acd74SJohn Marino 
1370d5acd74SJohn Marino 
1380d5acd74SJohn Marino #define _ISO2022INVALID (wchar_t)-1
1390d5acd74SJohn Marino 
isc0(__uint8_t x)1400d5acd74SJohn Marino static __inline bool isc0(__uint8_t x)
1410d5acd74SJohn Marino {
1420d5acd74SJohn Marino 
1430d5acd74SJohn Marino 	return ((x & 0x1f) == x);
1440d5acd74SJohn Marino }
1450d5acd74SJohn Marino 
isc1(__uint8_t x)1460d5acd74SJohn Marino static __inline bool isc1(__uint8_t x)
1470d5acd74SJohn Marino {
1480d5acd74SJohn Marino 
1490d5acd74SJohn Marino 	return (0x80 <= x && x <= 0x9f);
1500d5acd74SJohn Marino }
1510d5acd74SJohn Marino 
iscntl(__uint8_t x)1520d5acd74SJohn Marino static __inline bool iscntl(__uint8_t x)
1530d5acd74SJohn Marino {
1540d5acd74SJohn Marino 
1550d5acd74SJohn Marino 	return (isc0(x) || isc1(x) || x == 0x7f);
1560d5acd74SJohn Marino }
1570d5acd74SJohn Marino 
is94(__uint8_t x)1580d5acd74SJohn Marino static __inline bool is94(__uint8_t x)
1590d5acd74SJohn Marino {
1600d5acd74SJohn Marino 
1610d5acd74SJohn Marino 	return (0x21 <= x && x <= 0x7e);
1620d5acd74SJohn Marino }
1630d5acd74SJohn Marino 
is96(__uint8_t x)1640d5acd74SJohn Marino static __inline bool is96(__uint8_t x)
1650d5acd74SJohn Marino {
1660d5acd74SJohn Marino 
1670d5acd74SJohn Marino 	return (0x20 <= x && x <= 0x7f);
1680d5acd74SJohn Marino }
1690d5acd74SJohn Marino 
isecma(__uint8_t x)1700d5acd74SJohn Marino static __inline bool isecma(__uint8_t x)
1710d5acd74SJohn Marino {
1720d5acd74SJohn Marino 
1730d5acd74SJohn Marino 	return (0x30 <= x && x <= 0x7f);
1740d5acd74SJohn Marino }
1750d5acd74SJohn Marino 
isinterm(__uint8_t x)1760d5acd74SJohn Marino static __inline bool isinterm(__uint8_t x)
1770d5acd74SJohn Marino {
1780d5acd74SJohn Marino 
1790d5acd74SJohn Marino 	return (0x20 <= x && x <= 0x2f);
1800d5acd74SJohn Marino }
1810d5acd74SJohn Marino 
isthree(__uint8_t x)1820d5acd74SJohn Marino static __inline bool isthree(__uint8_t x)
1830d5acd74SJohn Marino {
1840d5acd74SJohn Marino 
1850d5acd74SJohn Marino 	return (0x60 <= x && x <= 0x6f);
1860d5acd74SJohn Marino }
1870d5acd74SJohn Marino 
1880d5acd74SJohn Marino static __inline int
getcs(const char * __restrict p,_ISO2022Charset * __restrict cs)1890d5acd74SJohn Marino getcs(const char * __restrict p, _ISO2022Charset * __restrict cs)
1900d5acd74SJohn Marino {
1910d5acd74SJohn Marino 
1920d5acd74SJohn Marino 	if (!strncmp(p, "94$", 3) && p[3] && !p[4]) {
1930d5acd74SJohn Marino 		cs->final = (unsigned char)(p[3] & 0xff);
1940d5acd74SJohn Marino 		cs->interm = '\0';
1950d5acd74SJohn Marino 		cs->vers = '\0';
1960d5acd74SJohn Marino 		cs->type = CS94MULTI;
1970d5acd74SJohn Marino 	} else if (!strncmp(p, "96$", 3) && p[3] && !p[4]) {
1980d5acd74SJohn Marino 		cs->final = (unsigned char)(p[3] & 0xff);
1990d5acd74SJohn Marino 		cs->interm = '\0';
2000d5acd74SJohn Marino 		cs->vers = '\0';
2010d5acd74SJohn Marino 		cs->type = CS96MULTI;
2020d5acd74SJohn Marino 	} else if (!strncmp(p, "94", 2) && p[2] && !p[3]) {
2030d5acd74SJohn Marino 		cs->final = (unsigned char)(p[2] & 0xff);
2040d5acd74SJohn Marino 		cs->interm = '\0';
2050d5acd74SJohn Marino 		cs->vers = '\0';
2060d5acd74SJohn Marino 		cs->type = CS94;
2070d5acd74SJohn Marino 	} else if (!strncmp(p, "96", 2) && p[2] && !p[3]) {
2080d5acd74SJohn Marino 		cs->final = (unsigned char )(p[2] & 0xff);
2090d5acd74SJohn Marino 		cs->interm = '\0';
2100d5acd74SJohn Marino 		cs->vers = '\0';
2110d5acd74SJohn Marino 		cs->type = CS96;
2120d5acd74SJohn Marino 	} else
2130d5acd74SJohn Marino 		return (1);
2140d5acd74SJohn Marino 
2150d5acd74SJohn Marino 	return (0);
2160d5acd74SJohn Marino }
2170d5acd74SJohn Marino 
2180d5acd74SJohn Marino 
2190d5acd74SJohn Marino #define _NOTMATCH	0
2200d5acd74SJohn Marino #define _MATCH		1
2210d5acd74SJohn Marino #define _PARSEFAIL	2
2220d5acd74SJohn Marino 
2230d5acd74SJohn Marino static __inline int
get_recommend(_ISO2022EncodingInfo * __restrict ei,const char * __restrict token)2240d5acd74SJohn Marino get_recommend(_ISO2022EncodingInfo * __restrict ei,
2250d5acd74SJohn Marino     const char * __restrict token)
2260d5acd74SJohn Marino {
2270d5acd74SJohn Marino 	_ISO2022Charset cs, *p;
2280d5acd74SJohn Marino 	int i;
2290d5acd74SJohn Marino 
2300d5acd74SJohn Marino 	if (!strchr("0123", token[0]) || token[1] != '=')
2310d5acd74SJohn Marino 		return (_NOTMATCH);
2320d5acd74SJohn Marino 
2330d5acd74SJohn Marino 	if (getcs(&token[2], &cs) == 0)
2340d5acd74SJohn Marino 		;
2350d5acd74SJohn Marino 	else if (!strcmp(&token[2], "94")) {
2360d5acd74SJohn Marino 		cs.final = (unsigned char)(token[4]);
2370d5acd74SJohn Marino 		cs.interm = '\0';
2380d5acd74SJohn Marino 		cs.vers = '\0';
2390d5acd74SJohn Marino 		cs.type = CS94;
2400d5acd74SJohn Marino 	} else if (!strcmp(&token[2], "96")) {
2410d5acd74SJohn Marino 		cs.final = (unsigned char)(token[4]);
2420d5acd74SJohn Marino 		cs.interm = '\0';
2430d5acd74SJohn Marino 		cs.vers = '\0';
2440d5acd74SJohn Marino 		cs.type = CS96;
2450d5acd74SJohn Marino 	} else if (!strcmp(&token[2], "94$")) {
2460d5acd74SJohn Marino 		cs.final = (unsigned char)(token[5]);
2470d5acd74SJohn Marino 		cs.interm = '\0';
2480d5acd74SJohn Marino 		cs.vers = '\0';
2490d5acd74SJohn Marino 		cs.type = CS94MULTI;
2500d5acd74SJohn Marino 	} else if (!strcmp(&token[2], "96$")) {
2510d5acd74SJohn Marino 		cs.final = (unsigned char)(token[5]);
2520d5acd74SJohn Marino 		cs.interm = '\0';
2530d5acd74SJohn Marino 		cs.vers = '\0';
2540d5acd74SJohn Marino 		cs.type = CS96MULTI;
2550d5acd74SJohn Marino 	} else
2560d5acd74SJohn Marino 		return (_PARSEFAIL);
2570d5acd74SJohn Marino 
2580d5acd74SJohn Marino 	i = token[0] - '0';
2590d5acd74SJohn Marino 	if (!ei->recommend[i])
2600d5acd74SJohn Marino 		ei->recommend[i] = malloc(sizeof(_ISO2022Charset));
2610d5acd74SJohn Marino 	else {
2620d5acd74SJohn Marino 		p = realloc(ei->recommend[i],
2630d5acd74SJohn Marino 		    sizeof(_ISO2022Charset) * (ei->recommendsize[i] + 1));
2640d5acd74SJohn Marino 		if (!p)
2650d5acd74SJohn Marino 			return (_PARSEFAIL);
2660d5acd74SJohn Marino 		ei->recommend[i] = p;
2670d5acd74SJohn Marino 	}
2680d5acd74SJohn Marino 	if (!ei->recommend[i])
2690d5acd74SJohn Marino 		return (_PARSEFAIL);
2700d5acd74SJohn Marino 	ei->recommendsize[i]++;
2710d5acd74SJohn Marino 
2720d5acd74SJohn Marino 	(ei->recommend[i] + (ei->recommendsize[i] - 1))->final = cs.final;
2730d5acd74SJohn Marino 	(ei->recommend[i] + (ei->recommendsize[i] - 1))->interm = cs.interm;
2740d5acd74SJohn Marino 	(ei->recommend[i] + (ei->recommendsize[i] - 1))->vers = cs.vers;
2750d5acd74SJohn Marino 	(ei->recommend[i] + (ei->recommendsize[i] - 1))->type = cs.type;
2760d5acd74SJohn Marino 
2770d5acd74SJohn Marino 	return (_MATCH);
2780d5acd74SJohn Marino }
2790d5acd74SJohn Marino 
2800d5acd74SJohn Marino static __inline int
get_initg(_ISO2022EncodingInfo * __restrict ei,const char * __restrict token)2810d5acd74SJohn Marino get_initg(_ISO2022EncodingInfo * __restrict ei,
2820d5acd74SJohn Marino     const char * __restrict token)
2830d5acd74SJohn Marino {
2840d5acd74SJohn Marino 	_ISO2022Charset cs;
2850d5acd74SJohn Marino 
2860d5acd74SJohn Marino 	if (strncmp("INIT", &token[0], 4) ||
2870d5acd74SJohn Marino 	    !strchr("0123", token[4]) ||
2880d5acd74SJohn Marino 	    token[5] != '=')
2890d5acd74SJohn Marino 		return (_NOTMATCH);
2900d5acd74SJohn Marino 
2910d5acd74SJohn Marino 	if (getcs(&token[6], &cs) != 0)
2920d5acd74SJohn Marino 		return (_PARSEFAIL);
2930d5acd74SJohn Marino 
2940d5acd74SJohn Marino 	ei->initg[token[4] - '0'].type = cs.type;
2950d5acd74SJohn Marino 	ei->initg[token[4] - '0'].final = cs.final;
2960d5acd74SJohn Marino 	ei->initg[token[4] - '0'].interm = cs.interm;
2970d5acd74SJohn Marino 	ei->initg[token[4] - '0'].vers = cs.vers;
2980d5acd74SJohn Marino 
2990d5acd74SJohn Marino 	return (_MATCH);
3000d5acd74SJohn Marino }
3010d5acd74SJohn Marino 
3020d5acd74SJohn Marino static __inline int
get_max(_ISO2022EncodingInfo * __restrict ei,const char * __restrict token)3030d5acd74SJohn Marino get_max(_ISO2022EncodingInfo * __restrict ei,
3040d5acd74SJohn Marino     const char * __restrict token)
3050d5acd74SJohn Marino {
3060d5acd74SJohn Marino 	if (!strcmp(token, "MAX1"))
3070d5acd74SJohn Marino 		ei->maxcharset = 1;
3080d5acd74SJohn Marino 	else if (!strcmp(token, "MAX2"))
3090d5acd74SJohn Marino 		ei->maxcharset = 2;
3100d5acd74SJohn Marino 	else if (!strcmp(token, "MAX3"))
3110d5acd74SJohn Marino 		ei->maxcharset = 3;
3120d5acd74SJohn Marino 	else
3130d5acd74SJohn Marino 		return (_NOTMATCH);
3140d5acd74SJohn Marino 
3150d5acd74SJohn Marino 	return (_MATCH);
3160d5acd74SJohn Marino }
3170d5acd74SJohn Marino 
3180d5acd74SJohn Marino 
3190d5acd74SJohn Marino static __inline int
get_flags(_ISO2022EncodingInfo * __restrict ei,const char * __restrict token)3200d5acd74SJohn Marino get_flags(_ISO2022EncodingInfo * __restrict ei,
3210d5acd74SJohn Marino     const char * __restrict token)
3220d5acd74SJohn Marino {
3230d5acd74SJohn Marino 	static struct {
3240d5acd74SJohn Marino 		const char	*tag;
3250d5acd74SJohn Marino 		int		flag;
3260d5acd74SJohn Marino 	} const tags[] = {
3270d5acd74SJohn Marino 		{ "DUMMY",	0	},
3280d5acd74SJohn Marino 		{ "8BIT",	F_8BIT	},
3290d5acd74SJohn Marino 		{ "NOOLD",	F_NOOLD	},
3300d5acd74SJohn Marino 		{ "SI",		F_SI	},
3310d5acd74SJohn Marino 		{ "SO",		F_SO	},
3320d5acd74SJohn Marino 		{ "LS0",	F_LS0	},
3330d5acd74SJohn Marino 		{ "LS1",	F_LS1	},
3340d5acd74SJohn Marino 		{ "LS2",	F_LS2	},
3350d5acd74SJohn Marino 		{ "LS3",	F_LS3	},
3360d5acd74SJohn Marino 		{ "LS1R",	F_LS1R	},
3370d5acd74SJohn Marino 		{ "LS2R",	F_LS2R	},
3380d5acd74SJohn Marino 		{ "LS3R",	F_LS3R	},
3390d5acd74SJohn Marino 		{ "SS2",	F_SS2	},
3400d5acd74SJohn Marino 		{ "SS3",	F_SS3	},
3410d5acd74SJohn Marino 		{ "SS2R",	F_SS2R	},
3420d5acd74SJohn Marino 		{ "SS3R",	F_SS3R	},
3430d5acd74SJohn Marino 		{ NULL,		0 }
3440d5acd74SJohn Marino 	};
3450d5acd74SJohn Marino 	int i;
3460d5acd74SJohn Marino 
3470d5acd74SJohn Marino 	for (i = 0; tags[i].tag; i++)
3480d5acd74SJohn Marino 		if (!strcmp(token, tags[i].tag)) {
3490d5acd74SJohn Marino 			ei->flags |= tags[i].flag;
3500d5acd74SJohn Marino 			return (_MATCH);
3510d5acd74SJohn Marino 		}
3520d5acd74SJohn Marino 
3530d5acd74SJohn Marino 	return (_NOTMATCH);
3540d5acd74SJohn Marino }
3550d5acd74SJohn Marino 
3560d5acd74SJohn Marino 
3570d5acd74SJohn Marino static __inline int
_citrus_ISO2022_parse_variable(_ISO2022EncodingInfo * __restrict ei,const void * __restrict var,size_t lenvar __unused)3580d5acd74SJohn Marino _citrus_ISO2022_parse_variable(_ISO2022EncodingInfo * __restrict ei,
3590d5acd74SJohn Marino     const void * __restrict var, size_t lenvar __unused)
3600d5acd74SJohn Marino {
3610d5acd74SJohn Marino 	char const *e, *v;
3620d5acd74SJohn Marino 	char buf[20];
3630d5acd74SJohn Marino 	size_t len;
3640d5acd74SJohn Marino 	int i, ret;
3650d5acd74SJohn Marino 
3660d5acd74SJohn Marino 	/*
3670d5acd74SJohn Marino 	 * parse VARIABLE section.
3680d5acd74SJohn Marino 	 */
3690d5acd74SJohn Marino 
3700d5acd74SJohn Marino 	if (!var)
3710d5acd74SJohn Marino 		return (EFTYPE);
3720d5acd74SJohn Marino 
3730d5acd74SJohn Marino 	v = (const char *) var;
3740d5acd74SJohn Marino 
3750d5acd74SJohn Marino 	/* initialize structure */
3760d5acd74SJohn Marino 	ei->maxcharset = 0;
3770d5acd74SJohn Marino 	for (i = 0; i < 4; i++) {
3780d5acd74SJohn Marino 		ei->recommend[i] = NULL;
3790d5acd74SJohn Marino 		ei->recommendsize[i] = 0;
3800d5acd74SJohn Marino 	}
3810d5acd74SJohn Marino 	ei->flags = 0;
3820d5acd74SJohn Marino 
3830d5acd74SJohn Marino 	while (*v) {
3840d5acd74SJohn Marino 		while (*v == ' ' || *v == '\t')
3850d5acd74SJohn Marino 			++v;
3860d5acd74SJohn Marino 
3870d5acd74SJohn Marino 		/* find the token */
3880d5acd74SJohn Marino 		e = v;
3890d5acd74SJohn Marino 		while (*e && *e != ' ' && *e != '\t')
3900d5acd74SJohn Marino 			++e;
3910d5acd74SJohn Marino 
3920d5acd74SJohn Marino 		len = e - v;
3930d5acd74SJohn Marino 		if (len == 0)
3940d5acd74SJohn Marino 			break;
3950d5acd74SJohn Marino 		if (len >= sizeof(buf))
3960d5acd74SJohn Marino 			goto parsefail;
3970d5acd74SJohn Marino 		snprintf(buf, sizeof(buf), "%.*s", (int)len, v);
3980d5acd74SJohn Marino 
3990d5acd74SJohn Marino 		if ((ret = get_recommend(ei, buf)) != _NOTMATCH)
4000d5acd74SJohn Marino 			;
4010d5acd74SJohn Marino 		else if ((ret = get_initg(ei, buf)) != _NOTMATCH)
4020d5acd74SJohn Marino 			;
4030d5acd74SJohn Marino 		else if ((ret = get_max(ei, buf)) != _NOTMATCH)
4040d5acd74SJohn Marino 			;
4050d5acd74SJohn Marino 		else if ((ret = get_flags(ei, buf)) != _NOTMATCH)
4060d5acd74SJohn Marino 			;
4070d5acd74SJohn Marino 		else
4080d5acd74SJohn Marino 			ret = _PARSEFAIL;
4090d5acd74SJohn Marino 		if (ret == _PARSEFAIL)
4100d5acd74SJohn Marino 			goto parsefail;
4110d5acd74SJohn Marino 		v = e;
4120d5acd74SJohn Marino 
4130d5acd74SJohn Marino 	}
4140d5acd74SJohn Marino 
4150d5acd74SJohn Marino 	return (0);
4160d5acd74SJohn Marino 
4170d5acd74SJohn Marino parsefail:
4180d5acd74SJohn Marino 	free(ei->recommend[0]);
4190d5acd74SJohn Marino 	free(ei->recommend[1]);
4200d5acd74SJohn Marino 	free(ei->recommend[2]);
4210d5acd74SJohn Marino 	free(ei->recommend[3]);
4220d5acd74SJohn Marino 
4230d5acd74SJohn Marino 	return (EFTYPE);
4240d5acd74SJohn Marino }
4250d5acd74SJohn Marino 
4260d5acd74SJohn Marino static __inline void
4270d5acd74SJohn Marino /*ARGSUSED*/
_citrus_ISO2022_init_state(_ISO2022EncodingInfo * __restrict ei,_ISO2022State * __restrict s)4280d5acd74SJohn Marino _citrus_ISO2022_init_state(_ISO2022EncodingInfo * __restrict ei,
4290d5acd74SJohn Marino     _ISO2022State * __restrict s)
4300d5acd74SJohn Marino {
4310d5acd74SJohn Marino 	int i;
4320d5acd74SJohn Marino 
4330d5acd74SJohn Marino 	memset(s, 0, sizeof(*s));
4340d5acd74SJohn Marino 	s->gl = 0;
4350d5acd74SJohn Marino 	s->gr = (ei->flags & F_8BIT) ? 1 : -1;
4360d5acd74SJohn Marino 
4370d5acd74SJohn Marino 	for (i = 0; i < 4; i++)
4380d5acd74SJohn Marino 		if (ei->initg[i].final) {
4390d5acd74SJohn Marino 			s->g[i].type = ei->initg[i].type;
4400d5acd74SJohn Marino 			s->g[i].final = ei->initg[i].final;
4410d5acd74SJohn Marino 			s->g[i].interm = ei->initg[i].interm;
4420d5acd74SJohn Marino 		}
4430d5acd74SJohn Marino 	s->singlegl = s->singlegr = -1;
4440d5acd74SJohn Marino 	s->flags |= _ISO2022STATE_FLAG_INITIALIZED;
4450d5acd74SJohn Marino }
4460d5acd74SJohn Marino 
44771ea2de5SJohn Marino #if 0
4480d5acd74SJohn Marino static __inline void
4490d5acd74SJohn Marino /*ARGSUSED*/
4500d5acd74SJohn Marino _citrus_ISO2022_pack_state(_ISO2022EncodingInfo * __restrict ei __unused,
4510d5acd74SJohn Marino     void * __restrict pspriv, const _ISO2022State * __restrict s)
4520d5acd74SJohn Marino {
4530d5acd74SJohn Marino 
4540d5acd74SJohn Marino 	memcpy(pspriv, (const void *)s, sizeof(*s));
4550d5acd74SJohn Marino }
4560d5acd74SJohn Marino 
4570d5acd74SJohn Marino static __inline void
4580d5acd74SJohn Marino /*ARGSUSED*/
4590d5acd74SJohn Marino _citrus_ISO2022_unpack_state(_ISO2022EncodingInfo * __restrict ei __unused,
4600d5acd74SJohn Marino     _ISO2022State * __restrict s, const void * __restrict pspriv)
4610d5acd74SJohn Marino {
4620d5acd74SJohn Marino 
4630d5acd74SJohn Marino 	memcpy((void *)s, pspriv, sizeof(*s));
4640d5acd74SJohn Marino }
46571ea2de5SJohn Marino #endif
4660d5acd74SJohn Marino 
4670d5acd74SJohn Marino static int
4680d5acd74SJohn Marino /*ARGSUSED*/
_citrus_ISO2022_encoding_module_init(_ISO2022EncodingInfo * __restrict ei,const void * __restrict var,size_t lenvar)4690d5acd74SJohn Marino _citrus_ISO2022_encoding_module_init(_ISO2022EncodingInfo * __restrict ei,
4700d5acd74SJohn Marino     const void * __restrict var, size_t lenvar)
4710d5acd74SJohn Marino {
4720d5acd74SJohn Marino 
4730d5acd74SJohn Marino 	return (_citrus_ISO2022_parse_variable(ei, var, lenvar));
4740d5acd74SJohn Marino }
4750d5acd74SJohn Marino 
4760d5acd74SJohn Marino static void
4770d5acd74SJohn Marino /*ARGSUSED*/
_citrus_ISO2022_encoding_module_uninit(_ISO2022EncodingInfo * ei __unused)4780d5acd74SJohn Marino _citrus_ISO2022_encoding_module_uninit(_ISO2022EncodingInfo *ei __unused)
4790d5acd74SJohn Marino {
4800d5acd74SJohn Marino 
4810d5acd74SJohn Marino }
4820d5acd74SJohn Marino 
4830d5acd74SJohn Marino #define	ESC	'\033'
4840d5acd74SJohn Marino #define	ECMA	-1
4850d5acd74SJohn Marino #define	INTERM	-2
4860d5acd74SJohn Marino #define	OECMA	-3
4870d5acd74SJohn Marino static const struct seqtable {
4880d5acd74SJohn Marino 	int type;
4890d5acd74SJohn Marino 	int csoff;
4900d5acd74SJohn Marino 	int finaloff;
4910d5acd74SJohn Marino 	int intermoff;
4920d5acd74SJohn Marino 	int versoff;
4930d5acd74SJohn Marino 	int len;
4940d5acd74SJohn Marino 	int chars[10];
4950d5acd74SJohn Marino } seqtable[] = {
4960d5acd74SJohn Marino 	/* G0 94MULTI special */
4970d5acd74SJohn Marino 	{ CS94MULTI, -1, 2, -1, -1,	3, { ESC, '$', OECMA }, },
4980d5acd74SJohn Marino 	/* G0 94MULTI special with version identification */
4990d5acd74SJohn Marino 	{ CS94MULTI, -1, 5, -1, 2,	6, { ESC, '&', ECMA, ESC, '$', OECMA }, },
5000d5acd74SJohn Marino 	/* G? 94 */
5010d5acd74SJohn Marino 	{ CS94, 1, 2, -1, -1,		3, { ESC, CS94, ECMA, }, },
5020d5acd74SJohn Marino 	/* G? 94 with 2nd intermediate char */
5030d5acd74SJohn Marino 	{ CS94, 1, 3, 2, -1,		4, { ESC, CS94, INTERM, ECMA, }, },
5040d5acd74SJohn Marino 	/* G? 96 */
5050d5acd74SJohn Marino 	{ CS96, 1, 2, -1, -1,		3, { ESC, CS96, ECMA, }, },
5060d5acd74SJohn Marino 	/* G? 96 with 2nd intermediate char */
5070d5acd74SJohn Marino 	{ CS96, 1, 3, 2, -1,		4, { ESC, CS96, INTERM, ECMA, }, },
5080d5acd74SJohn Marino 	/* G? 94MULTI */
5090d5acd74SJohn Marino 	{ CS94MULTI, 2, 3, -1, -1,	4, { ESC, '$', CS94, ECMA, }, },
5100d5acd74SJohn Marino 	/* G? 96MULTI */
5110d5acd74SJohn Marino 	{ CS96MULTI, 2, 3, -1, -1,	4, { ESC, '$', CS96, ECMA, }, },
5120d5acd74SJohn Marino 	/* G? 94MULTI with version specification */
5130d5acd74SJohn Marino 	{ CS94MULTI, 5, 6, -1, 2,	7, { ESC, '&', ECMA, ESC, '$', CS94, ECMA, }, },
5140d5acd74SJohn Marino 	/* LS2/3 */
5150d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, 'n', }, },
5160d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, 'o', }, },
5170d5acd74SJohn Marino 	/* LS1/2/3R */
5180d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, '~', }, },
5190d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, /*{*/ '}', }, },
5200d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, '|', }, },
5210d5acd74SJohn Marino 	/* SS2/3 */
5220d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, 'N', }, },
5230d5acd74SJohn Marino 	{ -1, -1, -1, -1, -1,		2, { ESC, 'O', }, },
5240d5acd74SJohn Marino 	/* end of records */
5250d5acd74SJohn Marino //	{ 0, }
5260d5acd74SJohn Marino 	{  0,  0,  0,  0,  0,		0, { ESC,  0, }, }
5270d5acd74SJohn Marino };
5280d5acd74SJohn Marino 
5290d5acd74SJohn Marino static int
seqmatch(const char * __restrict s,size_t n,const struct seqtable * __restrict sp)5300d5acd74SJohn Marino seqmatch(const char * __restrict s, size_t n,
5310d5acd74SJohn Marino     const struct seqtable * __restrict sp)
5320d5acd74SJohn Marino {
5330d5acd74SJohn Marino 	const int *p;
5340d5acd74SJohn Marino 
5350d5acd74SJohn Marino 	p = sp->chars;
5360d5acd74SJohn Marino 	while ((size_t)(p - sp->chars) < n && p - sp->chars < sp->len) {
5370d5acd74SJohn Marino 		switch (*p) {
5380d5acd74SJohn Marino 		case ECMA:
5390d5acd74SJohn Marino 			if (!isecma(*s))
5400d5acd74SJohn Marino 				goto terminate;
5410d5acd74SJohn Marino 			break;
5420d5acd74SJohn Marino 		case OECMA:
5430d5acd74SJohn Marino 			if (*s && strchr("@AB", *s))
5440d5acd74SJohn Marino 				break;
5450d5acd74SJohn Marino 			else
5460d5acd74SJohn Marino 				goto terminate;
5470d5acd74SJohn Marino 		case INTERM:
5480d5acd74SJohn Marino 			if (!isinterm(*s))
5490d5acd74SJohn Marino 				goto terminate;
5500d5acd74SJohn Marino 			break;
5510d5acd74SJohn Marino 		case CS94:
5520d5acd74SJohn Marino 			if (*s && strchr("()*+", *s))
5530d5acd74SJohn Marino 				break;
5540d5acd74SJohn Marino 			else
5550d5acd74SJohn Marino 				goto terminate;
5560d5acd74SJohn Marino 		case CS96:
5570d5acd74SJohn Marino 			if (*s && strchr(",-./", *s))
5580d5acd74SJohn Marino 				break;
5590d5acd74SJohn Marino 			else
5600d5acd74SJohn Marino 				goto terminate;
5610d5acd74SJohn Marino 		default:
5620d5acd74SJohn Marino 			if (*s != *p)
5630d5acd74SJohn Marino 				goto terminate;
5640d5acd74SJohn Marino 			break;
5650d5acd74SJohn Marino 		}
5660d5acd74SJohn Marino 
5670d5acd74SJohn Marino 		p++;
5680d5acd74SJohn Marino 		s++;
5690d5acd74SJohn Marino 	}
5700d5acd74SJohn Marino 
5710d5acd74SJohn Marino terminate:
5720d5acd74SJohn Marino 	return (p - sp->chars);
5730d5acd74SJohn Marino }
5740d5acd74SJohn Marino 
5750d5acd74SJohn Marino static wchar_t
_ISO2022_sgetwchar(_ISO2022EncodingInfo * __restrict ei __unused,char * __restrict string,size_t n,char ** __restrict result,_ISO2022State * __restrict psenc)5760d5acd74SJohn Marino _ISO2022_sgetwchar(_ISO2022EncodingInfo * __restrict ei __unused,
5770db70a6aSJohn Marino     char * __restrict string, size_t n, char ** __restrict result,
5780d5acd74SJohn Marino     _ISO2022State * __restrict psenc)
5790d5acd74SJohn Marino {
5800d5acd74SJohn Marino 	const struct seqtable *sp;
5810d5acd74SJohn Marino 	wchar_t wchar = 0;
5820d5acd74SJohn Marino 	int i, cur, nmatch;
5830d5acd74SJohn Marino 
5840d5acd74SJohn Marino 	while (1) {
5850d5acd74SJohn Marino 		/* SI/SO */
5860d5acd74SJohn Marino 		if (1 <= n && string[0] == '\017') {
5870d5acd74SJohn Marino 			psenc->gl = 0;
5880d5acd74SJohn Marino 			string++;
5890d5acd74SJohn Marino 			n--;
5900d5acd74SJohn Marino 			continue;
5910d5acd74SJohn Marino 		}
5920d5acd74SJohn Marino 		if (1 <= n && string[0] == '\016') {
5930d5acd74SJohn Marino 			psenc->gl = 1;
5940d5acd74SJohn Marino 			string++;
5950d5acd74SJohn Marino 			n--;
5960d5acd74SJohn Marino 			continue;
5970d5acd74SJohn Marino 		}
5980d5acd74SJohn Marino 
5990d5acd74SJohn Marino 		/* SS2/3R */
6000d5acd74SJohn Marino 		if (1 <= n && string[0] && strchr("\217\216", string[0])) {
6010d5acd74SJohn Marino 			psenc->singlegl = psenc->singlegr =
6020d5acd74SJohn Marino 			    (string[0] - '\216') + 2;
6030d5acd74SJohn Marino 			string++;
6040d5acd74SJohn Marino 			n--;
6050d5acd74SJohn Marino 			continue;
6060d5acd74SJohn Marino 		}
6070d5acd74SJohn Marino 
6080d5acd74SJohn Marino 		/* eat the letter if this is not ESC */
6090d5acd74SJohn Marino 		if (1 <= n && string[0] != '\033')
6100d5acd74SJohn Marino 			break;
6110d5acd74SJohn Marino 
6120d5acd74SJohn Marino 		/* look for a perfect match from escape sequences */
6130d5acd74SJohn Marino 		for (sp = &seqtable[0]; sp->len; sp++) {
6140d5acd74SJohn Marino 			nmatch = seqmatch(string, n, sp);
6150d5acd74SJohn Marino 			if (sp->len == nmatch && n >= (size_t)(sp->len))
6160d5acd74SJohn Marino 				break;
6170d5acd74SJohn Marino 		}
6180d5acd74SJohn Marino 
6190d5acd74SJohn Marino 		if (!sp->len)
6200d5acd74SJohn Marino 			goto notseq;
6210d5acd74SJohn Marino 
6220d5acd74SJohn Marino 		if (sp->type != -1) {
6230d5acd74SJohn Marino 			if (sp->csoff == -1)
6240d5acd74SJohn Marino 				i = 0;
6250d5acd74SJohn Marino 			else {
6260d5acd74SJohn Marino 				switch (sp->type) {
6270d5acd74SJohn Marino 				case CS94:
6280d5acd74SJohn Marino 				case CS94MULTI:
6290d5acd74SJohn Marino 					i = string[sp->csoff] - '(';
6300d5acd74SJohn Marino 					break;
6310d5acd74SJohn Marino 				case CS96:
6320d5acd74SJohn Marino 				case CS96MULTI:
6330d5acd74SJohn Marino 					i = string[sp->csoff] - ',';
6340d5acd74SJohn Marino 					break;
6350d5acd74SJohn Marino 				default:
6360d5acd74SJohn Marino 					return (_ISO2022INVALID);
6370d5acd74SJohn Marino 				}
6380d5acd74SJohn Marino 			}
6390d5acd74SJohn Marino 			psenc->g[i].type = sp->type;
6400d5acd74SJohn Marino 			psenc->g[i].final = '\0';
6410d5acd74SJohn Marino 			psenc->g[i].interm = '\0';
6420d5acd74SJohn Marino 			psenc->g[i].vers = '\0';
6430d5acd74SJohn Marino 			/* sp->finaloff must not be -1 */
6440d5acd74SJohn Marino 			if (sp->finaloff != -1)
6450d5acd74SJohn Marino 				psenc->g[i].final = string[sp->finaloff];
6460d5acd74SJohn Marino 			if (sp->intermoff != -1)
6470d5acd74SJohn Marino 				psenc->g[i].interm = string[sp->intermoff];
6480d5acd74SJohn Marino 			if (sp->versoff != -1)
6490d5acd74SJohn Marino 				psenc->g[i].vers = string[sp->versoff];
6500d5acd74SJohn Marino 
6510d5acd74SJohn Marino 			string += sp->len;
6520d5acd74SJohn Marino 			n -= sp->len;
6530d5acd74SJohn Marino 			continue;
6540d5acd74SJohn Marino 		}
6550d5acd74SJohn Marino 
6560d5acd74SJohn Marino 		/* LS2/3 */
6570d5acd74SJohn Marino 		if (2 <= n && string[0] == '\033' &&
6580d5acd74SJohn Marino 		    string[1] && strchr("no", string[1])) {
6590d5acd74SJohn Marino 			psenc->gl = string[1] - 'n' + 2;
6600d5acd74SJohn Marino 			string += 2;
6610d5acd74SJohn Marino 			n -= 2;
6620d5acd74SJohn Marino 			continue;
6630d5acd74SJohn Marino 		}
6640d5acd74SJohn Marino 
6650d5acd74SJohn Marino 		/* LS1/2/3R */
6660d5acd74SJohn Marino 			/* XXX: { for vi showmatch */
6670d5acd74SJohn Marino 		if (2 <= n && string[0] == '\033' &&
6680d5acd74SJohn Marino 		    string[1] && strchr("~}|", string[1])) {
6690d5acd74SJohn Marino 			psenc->gr = 3 - (string[1] - '|');
6700d5acd74SJohn Marino 			string += 2;
6710d5acd74SJohn Marino 			n -= 2;
6720d5acd74SJohn Marino 			continue;
6730d5acd74SJohn Marino 		}
6740d5acd74SJohn Marino 
6750d5acd74SJohn Marino 		/* SS2/3 */
6760d5acd74SJohn Marino 		if (2 <= n && string[0] == '\033' && string[1] &&
6770d5acd74SJohn Marino 		    strchr("NO", string[1])) {
6780d5acd74SJohn Marino 			psenc->singlegl = (string[1] - 'N') + 2;
6790d5acd74SJohn Marino 			string += 2;
6800d5acd74SJohn Marino 			n -= 2;
6810d5acd74SJohn Marino 			continue;
6820d5acd74SJohn Marino 		}
6830d5acd74SJohn Marino 
6840d5acd74SJohn Marino 	notseq:
6850d5acd74SJohn Marino 		/*
6860d5acd74SJohn Marino 		 * if we've got an unknown escape sequence, eat the ESC at the
6870d5acd74SJohn Marino 		 * head.  otherwise, wait till full escape sequence comes.
6880d5acd74SJohn Marino 		 */
6890d5acd74SJohn Marino 		for (sp = &seqtable[0]; sp->len; sp++) {
6900d5acd74SJohn Marino 			nmatch = seqmatch(string, n, sp);
6910d5acd74SJohn Marino 			if (!nmatch)
6920d5acd74SJohn Marino 				continue;
6930d5acd74SJohn Marino 
6940d5acd74SJohn Marino 			/*
6950d5acd74SJohn Marino 			 * if we are in the middle of escape sequence,
6960d5acd74SJohn Marino 			 * we still need to wait for more characters to come
6970d5acd74SJohn Marino 			 */
6980d5acd74SJohn Marino 			if (n < (size_t)(sp->len)) {
6990d5acd74SJohn Marino 				if ((size_t)(nmatch) == n) {
7000d5acd74SJohn Marino 					if (result)
7010d5acd74SJohn Marino 						*result = string;
7020d5acd74SJohn Marino 					return (_ISO2022INVALID);
7030d5acd74SJohn Marino 				}
7040d5acd74SJohn Marino 			} else {
7050d5acd74SJohn Marino 				if (nmatch == sp->len) {
7060d5acd74SJohn Marino 					/* this case should not happen */
7070d5acd74SJohn Marino 					goto eat;
7080d5acd74SJohn Marino 				}
7090d5acd74SJohn Marino 			}
7100d5acd74SJohn Marino 		}
7110d5acd74SJohn Marino 
7120d5acd74SJohn Marino 		break;
7130d5acd74SJohn Marino 	}
7140d5acd74SJohn Marino 
7150d5acd74SJohn Marino eat:
7160d5acd74SJohn Marino 	/* no letter to eat */
7170d5acd74SJohn Marino 	if (n < 1) {
7180d5acd74SJohn Marino 		if (result)
7190d5acd74SJohn Marino 			*result = string;
7200d5acd74SJohn Marino 		return (_ISO2022INVALID);
7210d5acd74SJohn Marino 	}
7220d5acd74SJohn Marino 
7230d5acd74SJohn Marino 	/* normal chars.  always eat C0/C1 as is. */
7240d5acd74SJohn Marino 	if (iscntl(*string & 0xff))
7250d5acd74SJohn Marino 		cur = -1;
7260d5acd74SJohn Marino 	else if (*string & 0x80)
7270d5acd74SJohn Marino 		cur = (psenc->singlegr == -1) ? psenc->gr : psenc->singlegr;
7280d5acd74SJohn Marino 	else
7290d5acd74SJohn Marino 		cur = (psenc->singlegl == -1) ? psenc->gl : psenc->singlegl;
7300d5acd74SJohn Marino 
7310d5acd74SJohn Marino 	if (cur == -1) {
7320d5acd74SJohn Marino asis:
7330d5acd74SJohn Marino 		wchar = *string++ & 0xff;
7340d5acd74SJohn Marino 		if (result)
7350d5acd74SJohn Marino 			*result = string;
7360d5acd74SJohn Marino 		/* reset single shift state */
7370d5acd74SJohn Marino 		psenc->singlegr = psenc->singlegl = -1;
7380d5acd74SJohn Marino 		return (wchar);
7390d5acd74SJohn Marino 	}
7400d5acd74SJohn Marino 
7410d5acd74SJohn Marino 	/* length error check */
7420d5acd74SJohn Marino 	switch (psenc->g[cur].type) {
7430d5acd74SJohn Marino 	case CS94MULTI:
7440d5acd74SJohn Marino 	case CS96MULTI:
7450d5acd74SJohn Marino 		if (!isthree(psenc->g[cur].final)) {
7460d5acd74SJohn Marino 			if (2 <= n &&
7470d5acd74SJohn Marino 			    (string[0] & 0x80) == (string[1] & 0x80))
7480d5acd74SJohn Marino 				break;
7490d5acd74SJohn Marino 		} else {
7500d5acd74SJohn Marino 			if (3 <= n &&
7510d5acd74SJohn Marino 			    (string[0] & 0x80) == (string[1] & 0x80) &&
7520d5acd74SJohn Marino 			    (string[0] & 0x80) == (string[2] & 0x80))
7530d5acd74SJohn Marino 				break;
7540d5acd74SJohn Marino 		}
7550d5acd74SJohn Marino 
7560d5acd74SJohn Marino 		/* we still need to wait for more characters to come */
7570d5acd74SJohn Marino 		if (result)
7580d5acd74SJohn Marino 			*result = string;
7590d5acd74SJohn Marino 		return (_ISO2022INVALID);
7600d5acd74SJohn Marino 
7610d5acd74SJohn Marino 	case CS94:
7620d5acd74SJohn Marino 	case CS96:
7630d5acd74SJohn Marino 		if (1 <= n)
7640d5acd74SJohn Marino 			break;
7650d5acd74SJohn Marino 
7660d5acd74SJohn Marino 		/* we still need to wait for more characters to come */
7670d5acd74SJohn Marino 		if (result)
7680d5acd74SJohn Marino 			*result = string;
7690d5acd74SJohn Marino 		return (_ISO2022INVALID);
7700d5acd74SJohn Marino 	}
7710d5acd74SJohn Marino 
7720d5acd74SJohn Marino 	/* range check */
7730d5acd74SJohn Marino 	switch (psenc->g[cur].type) {
7740d5acd74SJohn Marino 	case CS94:
7750d5acd74SJohn Marino 		if (!(is94(string[0] & 0x7f)))
7760d5acd74SJohn Marino 			goto asis;
777*d4548c47SSascha Wildner 		break;
7780d5acd74SJohn Marino 	case CS96:
7790d5acd74SJohn Marino 		if (!(is96(string[0] & 0x7f)))
7800d5acd74SJohn Marino 			goto asis;
7810d5acd74SJohn Marino 		break;
7820d5acd74SJohn Marino 	case CS94MULTI:
7830d5acd74SJohn Marino 		if (!(is94(string[0] & 0x7f) && is94(string[1] & 0x7f)))
7840d5acd74SJohn Marino 			goto asis;
7850d5acd74SJohn Marino 		break;
7860d5acd74SJohn Marino 	case CS96MULTI:
7870d5acd74SJohn Marino 		if (!(is96(string[0] & 0x7f) && is96(string[1] & 0x7f)))
7880d5acd74SJohn Marino 			goto asis;
7890d5acd74SJohn Marino 		break;
7900d5acd74SJohn Marino 	}
7910d5acd74SJohn Marino 
7920d5acd74SJohn Marino 	/* extract the character. */
7930d5acd74SJohn Marino 	switch (psenc->g[cur].type) {
7940d5acd74SJohn Marino 	case CS94:
7950d5acd74SJohn Marino 		/* special case for ASCII. */
7960d5acd74SJohn Marino 		if (psenc->g[cur].final == 'B' && !psenc->g[cur].interm) {
7970d5acd74SJohn Marino 			wchar = *string++;
7980d5acd74SJohn Marino 			wchar &= 0x7f;
7990d5acd74SJohn Marino 			break;
8000d5acd74SJohn Marino 		}
8010d5acd74SJohn Marino 		wchar = psenc->g[cur].final;
8020d5acd74SJohn Marino 		wchar = (wchar << 8);
8030d5acd74SJohn Marino 		wchar |= (psenc->g[cur].interm ? (0x80 | psenc->g[cur].interm) : 0);
8040d5acd74SJohn Marino 		wchar = (wchar << 8);
8050d5acd74SJohn Marino 		wchar = (wchar << 8) | (*string++ & 0x7f);
8060d5acd74SJohn Marino 		break;
8070d5acd74SJohn Marino 	case CS96:
8080d5acd74SJohn Marino 		/* special case for ISO-8859-1. */
8090d5acd74SJohn Marino 		if (psenc->g[cur].final == 'A' && !psenc->g[cur].interm) {
8100d5acd74SJohn Marino 			wchar = *string++;
8110d5acd74SJohn Marino 			wchar &= 0x7f;
8120d5acd74SJohn Marino 			wchar |= 0x80;
8130d5acd74SJohn Marino 			break;
8140d5acd74SJohn Marino 		}
8150d5acd74SJohn Marino 		wchar = psenc->g[cur].final;
8160d5acd74SJohn Marino 		wchar = (wchar << 8);
8170d5acd74SJohn Marino 		wchar |= (psenc->g[cur].interm ? (0x80 | psenc->g[cur].interm) : 0);
8180d5acd74SJohn Marino 		wchar = (wchar << 8);
8190d5acd74SJohn Marino 		wchar = (wchar << 8) | (*string++ & 0x7f);
8200d5acd74SJohn Marino 		wchar |= 0x80;
8210d5acd74SJohn Marino 		break;
8220d5acd74SJohn Marino 	case CS94MULTI:
8230d5acd74SJohn Marino 	case CS96MULTI:
8240d5acd74SJohn Marino 		wchar = psenc->g[cur].final;
8250d5acd74SJohn Marino 		wchar = (wchar << 8);
8260d5acd74SJohn Marino 		if (isthree(psenc->g[cur].final))
8270d5acd74SJohn Marino 			wchar |= (*string++ & 0x7f);
8280d5acd74SJohn Marino 		wchar = (wchar << 8) | (*string++ & 0x7f);
8290d5acd74SJohn Marino 		wchar = (wchar << 8) | (*string++ & 0x7f);
8300d5acd74SJohn Marino 		if (psenc->g[cur].type == CS96MULTI)
8310d5acd74SJohn Marino 			wchar |= 0x80;
8320d5acd74SJohn Marino 		break;
8330d5acd74SJohn Marino 	}
8340d5acd74SJohn Marino 
8350d5acd74SJohn Marino 	if (result)
8360d5acd74SJohn Marino 		*result = string;
8370d5acd74SJohn Marino 	/* reset single shift state */
8380d5acd74SJohn Marino 	psenc->singlegr = psenc->singlegl = -1;
8390d5acd74SJohn Marino 	return (wchar);
8400d5acd74SJohn Marino }
8410d5acd74SJohn Marino 
8420d5acd74SJohn Marino 
8430d5acd74SJohn Marino 
8440d5acd74SJohn Marino static int
_citrus_ISO2022_mbrtowc_priv(_ISO2022EncodingInfo * __restrict ei,wchar_t * __restrict pwc,char ** __restrict s,size_t n,_ISO2022State * __restrict psenc,size_t * __restrict nresult)8450d5acd74SJohn Marino _citrus_ISO2022_mbrtowc_priv(_ISO2022EncodingInfo * __restrict ei,
8460db70a6aSJohn Marino     wchar_t * __restrict pwc, char ** __restrict s,
8470d5acd74SJohn Marino     size_t n, _ISO2022State * __restrict psenc, size_t * __restrict nresult)
8480d5acd74SJohn Marino {
8490db70a6aSJohn Marino 	char *p, *result, *s0;
8500d5acd74SJohn Marino 	wchar_t wchar;
8510d5acd74SJohn Marino 	int c, chlenbak;
8520d5acd74SJohn Marino 
8530d5acd74SJohn Marino 	if (*s == NULL) {
8540d5acd74SJohn Marino 		_citrus_ISO2022_init_state(ei, psenc);
8550d5acd74SJohn Marino 		*nresult = _ENCODING_IS_STATE_DEPENDENT;
8560d5acd74SJohn Marino 		return (0);
8570d5acd74SJohn Marino 	}
8580d5acd74SJohn Marino 	s0 = *s;
8590d5acd74SJohn Marino 	c = 0;
8600d5acd74SJohn Marino 	chlenbak = psenc->chlen;
8610d5acd74SJohn Marino 
8620d5acd74SJohn Marino 	/*
8630d5acd74SJohn Marino 	 * if we have something in buffer, use that.
8640d5acd74SJohn Marino 	 * otherwise, skip here
8650d5acd74SJohn Marino 	 */
8660d5acd74SJohn Marino 	if (psenc->chlen > sizeof(psenc->ch)) {
8670d5acd74SJohn Marino 		/* illgeal state */
8680d5acd74SJohn Marino 		_citrus_ISO2022_init_state(ei, psenc);
8690d5acd74SJohn Marino 		goto encoding_error;
8700d5acd74SJohn Marino 	}
8710d5acd74SJohn Marino 	if (psenc->chlen == 0)
8720d5acd74SJohn Marino 		goto emptybuf;
8730d5acd74SJohn Marino 
8740d5acd74SJohn Marino 	/* buffer is not empty */
8750d5acd74SJohn Marino 	p = psenc->ch;
8760d5acd74SJohn Marino 	while (psenc->chlen < sizeof(psenc->ch)) {
8770d5acd74SJohn Marino 		if (n > 0) {
8780d5acd74SJohn Marino 			psenc->ch[psenc->chlen++] = *s0++;
8790d5acd74SJohn Marino 			n--;
8800d5acd74SJohn Marino 		}
8810d5acd74SJohn Marino 
8820d5acd74SJohn Marino 		wchar = _ISO2022_sgetwchar(ei, p, psenc->chlen - (p-psenc->ch),
8830d5acd74SJohn Marino 		    &result, psenc);
8840d5acd74SJohn Marino 		c += result - p;
8850d5acd74SJohn Marino 		if (wchar != _ISO2022INVALID) {
8860d5acd74SJohn Marino 			if (psenc->chlen > (size_t)c)
8870d5acd74SJohn Marino 				memmove(psenc->ch, result, psenc->chlen - c);
8880d5acd74SJohn Marino 			if (psenc->chlen < (size_t)c)
8890d5acd74SJohn Marino 				psenc->chlen = 0;
8900d5acd74SJohn Marino 			else
8910d5acd74SJohn Marino 				psenc->chlen -= c;
8920d5acd74SJohn Marino 			goto output;
8930d5acd74SJohn Marino 		}
8940d5acd74SJohn Marino 
8950d5acd74SJohn Marino 		if (n == 0) {
8960d5acd74SJohn Marino 			if ((size_t)(result - p) == psenc->chlen)
8970d5acd74SJohn Marino 				/* complete shift sequence. */
8980d5acd74SJohn Marino 				psenc->chlen = 0;
8990d5acd74SJohn Marino 			goto restart;
9000d5acd74SJohn Marino 		}
9010d5acd74SJohn Marino 
9020d5acd74SJohn Marino 		p = result;
9030d5acd74SJohn Marino 	}
9040d5acd74SJohn Marino 
9050d5acd74SJohn Marino 	/* escape sequence too long? */
9060d5acd74SJohn Marino 	goto encoding_error;
9070d5acd74SJohn Marino 
9080d5acd74SJohn Marino emptybuf:
9090d5acd74SJohn Marino 	wchar = _ISO2022_sgetwchar(ei, s0, n, &result, psenc);
9100d5acd74SJohn Marino 	if (wchar != _ISO2022INVALID) {
9110d5acd74SJohn Marino 		c += result - s0;
9120d5acd74SJohn Marino 		psenc->chlen = 0;
9130d5acd74SJohn Marino 		s0 = result;
9140d5acd74SJohn Marino 		goto output;
9150d5acd74SJohn Marino 	}
9160d5acd74SJohn Marino 	if (result > s0) {
9170d5acd74SJohn Marino 		c += (result - s0);
9180d5acd74SJohn Marino 		n -= (result - s0);
9190d5acd74SJohn Marino 		s0 = result;
9200d5acd74SJohn Marino 		if (n > 0)
9210d5acd74SJohn Marino 			goto emptybuf;
9220d5acd74SJohn Marino 		/* complete shift sequence. */
9230d5acd74SJohn Marino 		goto restart;
9240d5acd74SJohn Marino 	}
9250d5acd74SJohn Marino 	n += c;
9260d5acd74SJohn Marino 	if (n < sizeof(psenc->ch)) {
9270d5acd74SJohn Marino 		memcpy(psenc->ch, s0 - c, n);
9280d5acd74SJohn Marino 		psenc->chlen = n;
9290d5acd74SJohn Marino 		s0 = result;
9300d5acd74SJohn Marino 		goto restart;
9310d5acd74SJohn Marino 	}
9320d5acd74SJohn Marino 
9330d5acd74SJohn Marino 	/* escape sequence too long? */
9340d5acd74SJohn Marino 
9350d5acd74SJohn Marino encoding_error:
9360d5acd74SJohn Marino 	psenc->chlen = 0;
9370d5acd74SJohn Marino 	*nresult = (size_t)-1;
9380d5acd74SJohn Marino 	return (EILSEQ);
9390d5acd74SJohn Marino 
9400d5acd74SJohn Marino output:
9410d5acd74SJohn Marino 	*s = s0;
9420d5acd74SJohn Marino 	if (pwc)
9430d5acd74SJohn Marino 		*pwc = wchar;
9440d5acd74SJohn Marino 	*nresult = wchar ? c - chlenbak : 0;
9450d5acd74SJohn Marino 	return (0);
9460d5acd74SJohn Marino 
9470d5acd74SJohn Marino restart:
9480d5acd74SJohn Marino 	*s = s0;
9490d5acd74SJohn Marino 	*nresult = (size_t)-2;
9500d5acd74SJohn Marino 
9510d5acd74SJohn Marino 	return (0);
9520d5acd74SJohn Marino }
9530d5acd74SJohn Marino 
9540d5acd74SJohn Marino static int
recommendation(_ISO2022EncodingInfo * __restrict ei,_ISO2022Charset * __restrict cs)9550d5acd74SJohn Marino recommendation(_ISO2022EncodingInfo * __restrict ei,
9560d5acd74SJohn Marino     _ISO2022Charset * __restrict cs)
9570d5acd74SJohn Marino {
9580d5acd74SJohn Marino 	_ISO2022Charset *recommend;
9590d5acd74SJohn Marino 	size_t j;
9600d5acd74SJohn Marino 	int i;
9610d5acd74SJohn Marino 
9620d5acd74SJohn Marino 	/* first, try a exact match. */
9630d5acd74SJohn Marino 	for (i = 0; i < 4; i++) {
9640d5acd74SJohn Marino 		recommend = ei->recommend[i];
9650d5acd74SJohn Marino 		for (j = 0; j < ei->recommendsize[i]; j++) {
9660d5acd74SJohn Marino 			if (cs->type != recommend[j].type)
9670d5acd74SJohn Marino 				continue;
9680d5acd74SJohn Marino 			if (cs->final != recommend[j].final)
9690d5acd74SJohn Marino 				continue;
9700d5acd74SJohn Marino 			if (cs->interm != recommend[j].interm)
9710d5acd74SJohn Marino 				continue;
9720d5acd74SJohn Marino 
9730d5acd74SJohn Marino 			return (i);
9740d5acd74SJohn Marino 		}
9750d5acd74SJohn Marino 	}
9760d5acd74SJohn Marino 
9770d5acd74SJohn Marino 	/* then, try a wildcard match over final char. */
9780d5acd74SJohn Marino 	for (i = 0; i < 4; i++) {
9790d5acd74SJohn Marino 		recommend = ei->recommend[i];
9800d5acd74SJohn Marino 		for (j = 0; j < ei->recommendsize[i]; j++) {
9810d5acd74SJohn Marino 			if (cs->type != recommend[j].type)
9820d5acd74SJohn Marino 				continue;
9830d5acd74SJohn Marino 			if (cs->final && (cs->final != recommend[j].final))
9840d5acd74SJohn Marino 				continue;
9850d5acd74SJohn Marino 			if (cs->interm && (cs->interm != recommend[j].interm))
9860d5acd74SJohn Marino 				continue;
9870d5acd74SJohn Marino 
9880d5acd74SJohn Marino 			return (i);
9890d5acd74SJohn Marino 		}
9900d5acd74SJohn Marino 	}
9910d5acd74SJohn Marino 
9920d5acd74SJohn Marino 	/* there's no recommendation. make a guess. */
9930d5acd74SJohn Marino 	if (ei->maxcharset == 0) {
9940d5acd74SJohn Marino 		return (0);
9950d5acd74SJohn Marino 	} else {
9960d5acd74SJohn Marino 		switch (cs->type) {
9970d5acd74SJohn Marino 		case CS94:
9980d5acd74SJohn Marino 		case CS94MULTI:
9990d5acd74SJohn Marino 			return (0);
10000d5acd74SJohn Marino 		case CS96:
10010d5acd74SJohn Marino 		case CS96MULTI:
10020d5acd74SJohn Marino 			return (1);
10030d5acd74SJohn Marino 		}
10040d5acd74SJohn Marino 	}
10050d5acd74SJohn Marino 	return (0);
10060d5acd74SJohn Marino }
10070d5acd74SJohn Marino 
10080d5acd74SJohn Marino static int
_ISO2022_sputwchar(_ISO2022EncodingInfo * __restrict ei,wchar_t wc,char * __restrict string,size_t n,char ** __restrict result,_ISO2022State * __restrict psenc,size_t * __restrict nresult)10090d5acd74SJohn Marino _ISO2022_sputwchar(_ISO2022EncodingInfo * __restrict ei, wchar_t wc,
10100d5acd74SJohn Marino     char * __restrict string, size_t n, char ** __restrict result,
10110d5acd74SJohn Marino     _ISO2022State * __restrict psenc, size_t * __restrict nresult)
10120d5acd74SJohn Marino {
10130d5acd74SJohn Marino 	_ISO2022Charset cs;
10140d5acd74SJohn Marino 	char *p;
10150d5acd74SJohn Marino 	char tmp[MB_LEN_MAX];
10160d5acd74SJohn Marino 	size_t len;
10170d5acd74SJohn Marino 	int bit8, i = 0, target;
10180d5acd74SJohn Marino 	unsigned char mask;
10190d5acd74SJohn Marino 
10200d5acd74SJohn Marino 	if (isc0(wc & 0xff)) {
10210d5acd74SJohn Marino 		/* go back to INIT0 or ASCII on control chars */
10220d5acd74SJohn Marino 		cs = ei->initg[0].final ? ei->initg[0] : ascii;
10230d5acd74SJohn Marino 	} else if (isc1(wc & 0xff)) {
10240d5acd74SJohn Marino 		/* go back to INIT1 or ISO-8859-1 on control chars */
10250d5acd74SJohn Marino 		cs = ei->initg[1].final ? ei->initg[1] : iso88591;
10260d5acd74SJohn Marino 	} else if (!(wc & ~0xff)) {
10270d5acd74SJohn Marino 		if (wc & 0x80) {
10280d5acd74SJohn Marino 			/* special treatment for ISO-8859-1 */
10290d5acd74SJohn Marino 			cs = iso88591;
10300d5acd74SJohn Marino 		} else {
10310d5acd74SJohn Marino 			/* special treatment for ASCII */
10320d5acd74SJohn Marino 			cs = ascii;
10330d5acd74SJohn Marino 		}
10340d5acd74SJohn Marino 	} else {
10350d5acd74SJohn Marino 		cs.final = (wc >> 24) & 0x7f;
10360d5acd74SJohn Marino 		if ((wc >> 16) & 0x80)
10370d5acd74SJohn Marino 			cs.interm = (wc >> 16) & 0x7f;
10380d5acd74SJohn Marino 		else
10390d5acd74SJohn Marino 			cs.interm = '\0';
10400d5acd74SJohn Marino 		if (wc & 0x80)
10410d5acd74SJohn Marino 			cs.type = (wc & 0x00007f00) ? CS96MULTI : CS96;
10420d5acd74SJohn Marino 		else
10430d5acd74SJohn Marino 			cs.type = (wc & 0x00007f00) ? CS94MULTI : CS94;
10440d5acd74SJohn Marino 	}
10450d5acd74SJohn Marino 	target = recommendation(ei, &cs);
10460d5acd74SJohn Marino 	p = tmp;
10470d5acd74SJohn Marino 	bit8 = ei->flags & F_8BIT;
10480d5acd74SJohn Marino 
10490d5acd74SJohn Marino 	/* designate the charset onto the target plane(G0/1/2/3). */
10500d5acd74SJohn Marino 	if (psenc->g[target].type == cs.type &&
10510d5acd74SJohn Marino 	    psenc->g[target].final == cs.final &&
10520d5acd74SJohn Marino 	    psenc->g[target].interm == cs.interm)
10530d5acd74SJohn Marino 		goto planeok;
10540d5acd74SJohn Marino 
10550d5acd74SJohn Marino 	*p++ = '\033';
10560d5acd74SJohn Marino 	if (cs.type == CS94MULTI || cs.type == CS96MULTI)
10570d5acd74SJohn Marino 		*p++ = '$';
10580d5acd74SJohn Marino 	if (target == 0 && cs.type == CS94MULTI && strchr("@AB", cs.final) &&
10590d5acd74SJohn Marino 	    !cs.interm && !(ei->flags & F_NOOLD))
10600d5acd74SJohn Marino 		;
10610d5acd74SJohn Marino 	else if (cs.type == CS94 || cs.type == CS94MULTI)
10620d5acd74SJohn Marino 		*p++ = "()*+"[target];
10630d5acd74SJohn Marino 	else
10640d5acd74SJohn Marino 		*p++ = ",-./"[target];
10650d5acd74SJohn Marino 	if (cs.interm)
10660d5acd74SJohn Marino 		*p++ = cs.interm;
10670d5acd74SJohn Marino 	*p++ = cs.final;
10680d5acd74SJohn Marino 
10690d5acd74SJohn Marino 	psenc->g[target].type = cs.type;
10700d5acd74SJohn Marino 	psenc->g[target].final = cs.final;
10710d5acd74SJohn Marino 	psenc->g[target].interm = cs.interm;
10720d5acd74SJohn Marino 
10730d5acd74SJohn Marino planeok:
10740d5acd74SJohn Marino 	/* invoke the plane onto GL or GR. */
10750d5acd74SJohn Marino 	if (psenc->gl == target)
10760d5acd74SJohn Marino 		goto sideok;
10770d5acd74SJohn Marino 	if (bit8 && psenc->gr == target)
10780d5acd74SJohn Marino 		goto sideok;
10790d5acd74SJohn Marino 
10800d5acd74SJohn Marino 	if (target == 0 && (ei->flags & F_LS0)) {
10810d5acd74SJohn Marino 		*p++ = '\017';
10820d5acd74SJohn Marino 		psenc->gl = 0;
10830d5acd74SJohn Marino 	} else if (target == 1 && (ei->flags & F_LS1)) {
10840d5acd74SJohn Marino 		*p++ = '\016';
10850d5acd74SJohn Marino 		psenc->gl = 1;
10860d5acd74SJohn Marino 	} else if (target == 2 && (ei->flags & F_LS2)) {
10870d5acd74SJohn Marino 		*p++ = '\033';
10880d5acd74SJohn Marino 		*p++ = 'n';
10890d5acd74SJohn Marino 		psenc->gl = 2;
10900d5acd74SJohn Marino 	} else if (target == 3 && (ei->flags & F_LS3)) {
10910d5acd74SJohn Marino 		*p++ = '\033';
10920d5acd74SJohn Marino 		*p++ = 'o';
10930d5acd74SJohn Marino 		psenc->gl = 3;
10940d5acd74SJohn Marino 	} else if (bit8 && target == 1 && (ei->flags & F_LS1R)) {
10950d5acd74SJohn Marino 		*p++ = '\033';
10960d5acd74SJohn Marino 		*p++ = '~';
10970d5acd74SJohn Marino 		psenc->gr = 1;
10980d5acd74SJohn Marino 	} else if (bit8 && target == 2 && (ei->flags & F_LS2R)) {
10990d5acd74SJohn Marino 		*p++ = '\033';
11000d5acd74SJohn Marino 		/*{*/
11010d5acd74SJohn Marino 		*p++ = '}';
11020d5acd74SJohn Marino 		psenc->gr = 2;
11030d5acd74SJohn Marino 	} else if (bit8 && target == 3 && (ei->flags & F_LS3R)) {
11040d5acd74SJohn Marino 		*p++ = '\033';
11050d5acd74SJohn Marino 		*p++ = '|';
11060d5acd74SJohn Marino 		psenc->gr = 3;
11070d5acd74SJohn Marino 	} else if (target == 2 && (ei->flags & F_SS2)) {
11080d5acd74SJohn Marino 		*p++ = '\033';
11090d5acd74SJohn Marino 		*p++ = 'N';
11100d5acd74SJohn Marino 		psenc->singlegl = 2;
11110d5acd74SJohn Marino 	} else if (target == 3 && (ei->flags & F_SS3)) {
11120d5acd74SJohn Marino 		*p++ = '\033';
11130d5acd74SJohn Marino 		*p++ = 'O';
11140d5acd74SJohn Marino 		psenc->singlegl = 3;
11150d5acd74SJohn Marino 	} else if (bit8 && target == 2 && (ei->flags & F_SS2R)) {
11160d5acd74SJohn Marino 		*p++ = '\216';
11170d5acd74SJohn Marino 		*p++ = 'N';
11180d5acd74SJohn Marino 		psenc->singlegl = psenc->singlegr = 2;
11190d5acd74SJohn Marino 	} else if (bit8 && target == 3 && (ei->flags & F_SS3R)) {
11200d5acd74SJohn Marino 		*p++ = '\217';
11210d5acd74SJohn Marino 		*p++ = 'O';
11220d5acd74SJohn Marino 		psenc->singlegl = psenc->singlegr = 3;
11230d5acd74SJohn Marino 	} else
11240d5acd74SJohn Marino 		goto ilseq;
11250d5acd74SJohn Marino 
11260d5acd74SJohn Marino sideok:
11270d5acd74SJohn Marino 	if (psenc->singlegl == target)
11280d5acd74SJohn Marino 		mask = 0x00;
11290d5acd74SJohn Marino 	else if (psenc->singlegr == target)
11300d5acd74SJohn Marino 		mask = 0x80;
11310d5acd74SJohn Marino 	else if (psenc->gl == target)
11320d5acd74SJohn Marino 		mask = 0x00;
11330d5acd74SJohn Marino 	else if ((ei->flags & F_8BIT) && psenc->gr == target)
11340d5acd74SJohn Marino 		mask = 0x80;
11350d5acd74SJohn Marino 	else
11360d5acd74SJohn Marino 		goto ilseq;
11370d5acd74SJohn Marino 
11380d5acd74SJohn Marino 	switch (cs.type) {
11390d5acd74SJohn Marino 	case CS94:
11400d5acd74SJohn Marino 	case CS96:
11410d5acd74SJohn Marino 		i = 1;
11420d5acd74SJohn Marino 		break;
11430d5acd74SJohn Marino 	case CS94MULTI:
11440d5acd74SJohn Marino 	case CS96MULTI:
11450d5acd74SJohn Marino 		i = !iscntl(wc & 0xff) ?
11460d5acd74SJohn Marino 		    (isthree(cs.final) ? 3 : 2) : 1;
11470d5acd74SJohn Marino 		break;
11480d5acd74SJohn Marino 	}
11490d5acd74SJohn Marino 	while (i-- > 0)
11500d5acd74SJohn Marino 		*p++ = ((wc >> (i << 3)) & 0x7f) | mask;
11510d5acd74SJohn Marino 
11520d5acd74SJohn Marino 	/* reset single shift state */
11530d5acd74SJohn Marino 	psenc->singlegl = psenc->singlegr = -1;
11540d5acd74SJohn Marino 
11550d5acd74SJohn Marino 	len = (size_t)(p - tmp);
11560d5acd74SJohn Marino 	if (n < len) {
11570d5acd74SJohn Marino 		if (result)
11580d5acd74SJohn Marino 			*result = (char *)0;
11590d5acd74SJohn Marino 		*nresult = (size_t)-1;
11600d5acd74SJohn Marino 		return (E2BIG);
11610d5acd74SJohn Marino 	}
11620d5acd74SJohn Marino 	if (result)
11630d5acd74SJohn Marino 		*result = string + len;
11640d5acd74SJohn Marino 	memcpy(string, tmp, len);
11650d5acd74SJohn Marino 	*nresult = len;
11660d5acd74SJohn Marino 
11670d5acd74SJohn Marino 	return (0);
11680d5acd74SJohn Marino 
11690d5acd74SJohn Marino ilseq:
11700d5acd74SJohn Marino 	*nresult = (size_t)-1;
11710d5acd74SJohn Marino 	return (EILSEQ);
11720d5acd74SJohn Marino }
11730d5acd74SJohn Marino 
11740d5acd74SJohn Marino static int
_citrus_ISO2022_put_state_reset(_ISO2022EncodingInfo * __restrict ei,char * __restrict s,size_t n,_ISO2022State * __restrict psenc,size_t * __restrict nresult)11750d5acd74SJohn Marino _citrus_ISO2022_put_state_reset(_ISO2022EncodingInfo * __restrict ei,
11760d5acd74SJohn Marino     char * __restrict s, size_t n, _ISO2022State * __restrict psenc,
11770d5acd74SJohn Marino     size_t * __restrict nresult)
11780d5acd74SJohn Marino {
11790d5acd74SJohn Marino 	char *result;
11800d5acd74SJohn Marino 	char buf[MB_LEN_MAX];
11810d5acd74SJohn Marino 	size_t len;
11820d5acd74SJohn Marino 	int ret;
11830d5acd74SJohn Marino 
11840d5acd74SJohn Marino 	/* XXX state will be modified after this operation... */
11850d5acd74SJohn Marino 	ret = _ISO2022_sputwchar(ei, L'\0', buf, sizeof(buf), &result, psenc,
11860d5acd74SJohn Marino 	    &len);
11870d5acd74SJohn Marino 	if (ret) {
11880d5acd74SJohn Marino 		*nresult = len;
11890d5acd74SJohn Marino 		return (ret);
11900d5acd74SJohn Marino 	}
11910d5acd74SJohn Marino 
11920d5acd74SJohn Marino 	if (sizeof(buf) < len || n < len-1) {
11930d5acd74SJohn Marino 		/* XXX should recover state? */
11940d5acd74SJohn Marino 		*nresult = (size_t)-1;
11950d5acd74SJohn Marino 		return (E2BIG);
11960d5acd74SJohn Marino 	}
11970d5acd74SJohn Marino 
11980d5acd74SJohn Marino 	memcpy(s, buf, len - 1);
11990d5acd74SJohn Marino 	*nresult = len - 1;
12000d5acd74SJohn Marino 	return (0);
12010d5acd74SJohn Marino }
12020d5acd74SJohn Marino 
12030d5acd74SJohn Marino static int
_citrus_ISO2022_wcrtomb_priv(_ISO2022EncodingInfo * __restrict ei,char * __restrict s,size_t n,wchar_t wc,_ISO2022State * __restrict psenc,size_t * __restrict nresult)12040d5acd74SJohn Marino _citrus_ISO2022_wcrtomb_priv(_ISO2022EncodingInfo * __restrict ei,
12050d5acd74SJohn Marino     char * __restrict s, size_t n, wchar_t wc,
12060d5acd74SJohn Marino     _ISO2022State * __restrict psenc, size_t * __restrict nresult)
12070d5acd74SJohn Marino {
12080d5acd74SJohn Marino 	char *result;
12090d5acd74SJohn Marino 	char buf[MB_LEN_MAX];
12100d5acd74SJohn Marino 	size_t len;
12110d5acd74SJohn Marino 	int ret;
12120d5acd74SJohn Marino 
12130d5acd74SJohn Marino 	/* XXX state will be modified after this operation... */
12140d5acd74SJohn Marino 	ret = _ISO2022_sputwchar(ei, wc, buf, sizeof(buf), &result, psenc,
12150d5acd74SJohn Marino 	    &len);
12160d5acd74SJohn Marino 	if (ret) {
12170d5acd74SJohn Marino 		*nresult = len;
12180d5acd74SJohn Marino 		return (ret);
12190d5acd74SJohn Marino 	}
12200d5acd74SJohn Marino 
12210d5acd74SJohn Marino 	if (sizeof(buf) < len || n < len) {
12220d5acd74SJohn Marino 		/* XXX should recover state? */
12230d5acd74SJohn Marino 		*nresult = (size_t)-1;
12240d5acd74SJohn Marino 		return (E2BIG);
12250d5acd74SJohn Marino 	}
12260d5acd74SJohn Marino 
12270d5acd74SJohn Marino 	memcpy(s, buf, len);
12280d5acd74SJohn Marino 	*nresult = len;
12290d5acd74SJohn Marino 	return (0);
12300d5acd74SJohn Marino }
12310d5acd74SJohn Marino 
12320d5acd74SJohn Marino static __inline int
12330d5acd74SJohn Marino /*ARGSUSED*/
_citrus_ISO2022_stdenc_wctocs(_ISO2022EncodingInfo * __restrict ei __unused,_csid_t * __restrict csid,_index_t * __restrict idx,wchar_t wc)12340d5acd74SJohn Marino _citrus_ISO2022_stdenc_wctocs(_ISO2022EncodingInfo * __restrict ei __unused,
12350d5acd74SJohn Marino     _csid_t * __restrict csid, _index_t * __restrict idx, wchar_t wc)
12360d5acd74SJohn Marino {
12370d5acd74SJohn Marino 	wchar_t m, nm;
12380d5acd74SJohn Marino 
12390d5acd74SJohn Marino 	m = wc & 0x7FFF8080;
12400d5acd74SJohn Marino 	nm = wc & 0x007F7F7F;
12410d5acd74SJohn Marino 	if (m & 0x00800000)
12420d5acd74SJohn Marino 		nm &= 0x00007F7F;
12430d5acd74SJohn Marino 	else
12440d5acd74SJohn Marino 		m &= 0x7F008080;
12450d5acd74SJohn Marino 	if (nm & 0x007F0000) {
12460d5acd74SJohn Marino 		/* ^3 mark */
12470d5acd74SJohn Marino 		m |= 0x007F0000;
12480d5acd74SJohn Marino 	} else if (nm & 0x00007F00) {
12490d5acd74SJohn Marino 		/* ^2 mark */
12500d5acd74SJohn Marino 		m |= 0x00007F00;
12510d5acd74SJohn Marino 	}
12520d5acd74SJohn Marino 	*csid = (_csid_t)m;
12530d5acd74SJohn Marino 	*idx  = (_index_t)nm;
12540d5acd74SJohn Marino 
12550d5acd74SJohn Marino 	return (0);
12560d5acd74SJohn Marino }
12570d5acd74SJohn Marino 
12580d5acd74SJohn Marino static __inline int
12590d5acd74SJohn Marino /*ARGSUSED*/
_citrus_ISO2022_stdenc_cstowc(_ISO2022EncodingInfo * __restrict ei __unused,wchar_t * __restrict wc,_csid_t csid,_index_t idx)12600d5acd74SJohn Marino _citrus_ISO2022_stdenc_cstowc(_ISO2022EncodingInfo * __restrict ei __unused,
12610d5acd74SJohn Marino     wchar_t * __restrict wc, _csid_t csid, _index_t idx)
12620d5acd74SJohn Marino {
12630d5acd74SJohn Marino 
12640d5acd74SJohn Marino 	*wc = (wchar_t)(csid & 0x7F808080) | (wchar_t)idx;
12650d5acd74SJohn Marino 
12660d5acd74SJohn Marino 	return (0);
12670d5acd74SJohn Marino }
12680d5acd74SJohn Marino 
12690d5acd74SJohn Marino static __inline int
12700d5acd74SJohn Marino /*ARGSUSED*/
_citrus_ISO2022_stdenc_get_state_desc_generic(_ISO2022EncodingInfo * __restrict ei __unused,_ISO2022State * __restrict psenc,int * __restrict rstate)12710d5acd74SJohn Marino _citrus_ISO2022_stdenc_get_state_desc_generic(_ISO2022EncodingInfo * __restrict ei __unused,
12720d5acd74SJohn Marino     _ISO2022State * __restrict psenc, int * __restrict rstate)
12730d5acd74SJohn Marino {
12740d5acd74SJohn Marino 
12750d5acd74SJohn Marino 	if (psenc->chlen == 0) {
12760d5acd74SJohn Marino 		/* XXX: it should distinguish initial and stable. */
12770d5acd74SJohn Marino 		*rstate = _STDENC_SDGEN_STABLE;
12780d5acd74SJohn Marino 	} else
12790d5acd74SJohn Marino 		*rstate = (psenc->ch[0] == '\033') ?
12800d5acd74SJohn Marino 		    _STDENC_SDGEN_INCOMPLETE_SHIFT :
12810d5acd74SJohn Marino 		    _STDENC_SDGEN_INCOMPLETE_CHAR;
12820d5acd74SJohn Marino 	return (0);
12830d5acd74SJohn Marino }
12840d5acd74SJohn Marino 
12850d5acd74SJohn Marino /* ----------------------------------------------------------------------
12860d5acd74SJohn Marino  * public interface for stdenc
12870d5acd74SJohn Marino  */
12880d5acd74SJohn Marino 
12890d5acd74SJohn Marino _CITRUS_STDENC_DECLS(ISO2022);
12900d5acd74SJohn Marino _CITRUS_STDENC_DEF_OPS(ISO2022);
12910d5acd74SJohn Marino 
12920d5acd74SJohn Marino #include "citrus_stdenc_template.h"
1293