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