Blame src/w32-iconv.c

Packit fc043f
/* w32-iconv.c - iconv implementation for Windows.
Packit fc043f
 * Copyright (C) 2016 g10 Code GmbH
Packit fc043f
 *
Packit fc043f
 * This file is part of libgpg-error.
Packit fc043f
 *
Packit fc043f
 * libgpg-error is free software; you can redistribute it and/or
Packit fc043f
 * modify it under the terms of the GNU Lesser General Public License
Packit fc043f
 * as published by the Free Software Foundation; either version 2.1 of
Packit fc043f
 * the License, or (at your option) any later version.
Packit fc043f
 *
Packit fc043f
 * libgpg-error is distributed in the hope that it will be useful, but
Packit fc043f
 * WITHOUT ANY WARRANTY; without even the implied warranty of
Packit fc043f
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit fc043f
 * Lesser General Public License for more details.
Packit fc043f
 *
Packit fc043f
 * You should have received a copy of the GNU Lesser General Public
Packit fc043f
 * License along with this program; if not, see <https://www.gnu.org/licenses/>.
Packit fc043f
 *
Packit fc043f
 **************************************************************
Packit fc043f
 * This code code is based on the file win_iconv.c as found
Packit fc043f
 * at https://github.com/win-iconv/win-iconv with the commit id
Packit fc043f
 * 8c23784e35327c9d85d22810b9e4a2cbd06ffe90, dated 2016-03-18.
Packit fc043f
 * Yukihiro Nakadaira <yukihiro.nakadaira@gmail.com> is the
Packit fc043f
 * original author.  The file win_iconv.c carried this notice:
Packit fc043f
 *--------------------------------------------------------------
Packit fc043f
 * iconv implementation using Win32 API to convert.
Packit fc043f
 *
Packit fc043f
 * This file is placed in the public domain.
Packit fc043f
 *---------------------------------------------------
Packit fc043f
 */
Packit fc043f
Packit fc043f
#if HAVE_CONFIG_H
Packit fc043f
#include <config.h>
Packit fc043f
#endif
Packit fc043f
#if !defined (_WIN32) && !defined (__CYGWIN32__)
Packit fc043f
#  error This module may only be build for Windows or Cygwin32
Packit fc043f
#endif
Packit fc043f
Packit fc043f
/* for WC_NO_BEST_FIT_CHARS */
Packit fc043f
#ifndef WINVER
Packit fc043f
# define WINVER 0x0500
Packit fc043f
#endif
Packit fc043f
Packit fc043f
#include <windows.h>
Packit fc043f
#include <errno.h>
Packit fc043f
#include <string.h>
Packit fc043f
#include <stdlib.h>
Packit fc043f
Packit fc043f
#include "gpgrt-int.h"
Packit fc043f
Packit fc043f
#undef USE_MLANG_DLL
Packit fc043f
Packit fc043f
#define MB_CHAR_MAX 16
Packit fc043f
Packit fc043f
#define UNICODE_MODE_BOM_DONE   1
Packit fc043f
#define UNICODE_MODE_SWAPPED    2
Packit fc043f
Packit fc043f
#define FLAG_USE_BOM            1
Packit fc043f
#define FLAG_TRANSLIT           2 /* //TRANSLIT */
Packit fc043f
#define FLAG_IGNORE             4 /* //IGNORE */
Packit fc043f
Packit fc043f
typedef unsigned char uchar;
Packit fc043f
typedef unsigned short ushort;
Packit fc043f
typedef unsigned int uint;
Packit fc043f
Packit fc043f
typedef struct compat_t compat_t;
Packit fc043f
typedef struct csconv_t csconv_t;
Packit fc043f
Packit fc043f
typedef int* (*f_errno)(void);
Packit fc043f
typedef int (*f_mbtowc)(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize);
Packit fc043f
typedef int (*f_wctomb)(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize);
Packit fc043f
typedef int (*f_mblen)(csconv_t *cv, const uchar *buf, int bufsize);
Packit fc043f
typedef int (*f_flush)(csconv_t *cv, uchar *buf, int bufsize);
Packit fc043f
Packit fc043f
#define COMPAT_IN   1
Packit fc043f
#define COMPAT_OUT  2
Packit fc043f
Packit fc043f
/* unicode mapping for compatibility with other conversion table. */
Packit fc043f
struct compat_t {
Packit fc043f
    uint in;
Packit fc043f
    uint out;
Packit fc043f
    uint flag;
Packit fc043f
};
Packit fc043f
Packit fc043f
struct csconv_t {
Packit fc043f
    int codepage;
Packit fc043f
    int flags;
Packit fc043f
    f_mbtowc mbtowc;
Packit fc043f
    f_wctomb wctomb;
Packit fc043f
    f_mblen mblen;
Packit fc043f
    f_flush flush;
Packit fc043f
    DWORD mode;
Packit fc043f
    compat_t *compat;
Packit fc043f
};
Packit fc043f
Packit fc043f
struct _gpgrt_w32_iconv_s {
Packit fc043f
    f_errno _errno;
Packit fc043f
    csconv_t from;
Packit fc043f
    csconv_t to;
Packit fc043f
};
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int load_mlang(void);
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
static int make_csconv(const char *name, csconv_t *cv);
Packit fc043f
static int name_to_codepage(const char *name);
Packit fc043f
static uint utf16_to_ucs4(const ushort *wbuf);
Packit fc043f
static void ucs4_to_utf16(uint wc, ushort *wbuf, int *wbufsize);
Packit fc043f
static int mbtowc_flags(int codepage);
Packit fc043f
static int must_use_null_useddefaultchar(int codepage);
Packit fc043f
static char *strrstr(const char *str, const char *token);
Packit fc043f
static char *xstrndup(const char *s, size_t n);
Packit fc043f
static int seterror(int err);
Packit fc043f
Packit fc043f
static int sbcs_mblen(csconv_t *cv, const uchar *buf, int bufsize);
Packit fc043f
static int dbcs_mblen(csconv_t *cv, const uchar *buf, int bufsize);
Packit fc043f
static int mbcs_mblen(csconv_t *cv, const uchar *buf, int bufsize);
Packit fc043f
static int utf8_mblen(csconv_t *cv, const uchar *buf, int bufsize);
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int eucjp_mblen(csconv_t *cv, const uchar *buf, int bufsize);
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
static int kernel_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize);
Packit fc043f
static int kernel_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize);
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int mlang_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize);
Packit fc043f
static int mlang_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize);
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
static int utf16_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize);
Packit fc043f
static int utf16_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize);
Packit fc043f
static int utf32_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize);
Packit fc043f
static int utf32_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize);
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int iso2022jp_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize);
Packit fc043f
static int iso2022jp_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize);
Packit fc043f
static int iso2022jp_flush(csconv_t *cv, uchar *buf, int bufsize);
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
static struct {
Packit fc043f
    int codepage;
Packit fc043f
    const char *name;
Packit fc043f
} codepage_alias[] = {
Packit fc043f
    {65001, "CP65001"},
Packit fc043f
    {65001, "UTF8"},
Packit fc043f
    {65001, "UTF-8"},
Packit fc043f
Packit fc043f
    {1200, "CP1200"},
Packit fc043f
    {1200, "UTF16LE"},
Packit fc043f
    {1200, "UTF-16LE"},
Packit fc043f
    {1200, "UCS2LE"},
Packit fc043f
    {1200, "UCS-2LE"},
Packit fc043f
    {1200, "UCS-2-INTERNAL"},
Packit fc043f
Packit fc043f
    {1201, "CP1201"},
Packit fc043f
    {1201, "UTF16BE"},
Packit fc043f
    {1201, "UTF-16BE"},
Packit fc043f
    {1201, "UCS2BE"},
Packit fc043f
    {1201, "UCS-2BE"},
Packit fc043f
    {1201, "unicodeFFFE"},
Packit fc043f
Packit fc043f
    {12000, "CP12000"},
Packit fc043f
    {12000, "UTF32LE"},
Packit fc043f
    {12000, "UTF-32LE"},
Packit fc043f
    {12000, "UCS4LE"},
Packit fc043f
    {12000, "UCS-4LE"},
Packit fc043f
Packit fc043f
    {12001, "CP12001"},
Packit fc043f
    {12001, "UTF32BE"},
Packit fc043f
    {12001, "UTF-32BE"},
Packit fc043f
    {12001, "UCS4BE"},
Packit fc043f
    {12001, "UCS-4BE"},
Packit fc043f
Packit fc043f
    /*
Packit fc043f
     * Default is big endian.
Packit fc043f
     * See rfc2781 4.3 Interpreting text labelled as UTF-16.
Packit fc043f
     */
Packit fc043f
    {1201, "UTF16"},
Packit fc043f
    {1201, "UTF-16"},
Packit fc043f
    {1201, "UCS2"},
Packit fc043f
    {1201, "UCS-2"},
Packit fc043f
    {12001, "UTF32"},
Packit fc043f
    {12001, "UTF-32"},
Packit fc043f
    {12001, "UCS-4"},
Packit fc043f
    {12001, "UCS4"},
Packit fc043f
Packit fc043f
    /* copy from libiconv `iconv -l` */
Packit fc043f
    /* !IsValidCodePage(367) */
Packit fc043f
    {20127, "ANSI_X3.4-1968"},
Packit fc043f
    {20127, "ANSI_X3.4-1986"},
Packit fc043f
    {20127, "ASCII"},
Packit fc043f
    {20127, "CP367"},
Packit fc043f
    {20127, "IBM367"},
Packit fc043f
    {20127, "ISO-IR-6"},
Packit fc043f
    {20127, "ISO646-US"},
Packit fc043f
    {20127, "ISO_646.IRV:1991"},
Packit fc043f
    {20127, "US"},
Packit fc043f
    {20127, "US-ASCII"},
Packit fc043f
    {20127, "CSASCII"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(819) */
Packit fc043f
    {1252, "CP819"},
Packit fc043f
    {1252, "IBM819"},
Packit fc043f
    {28591, "ISO-8859-1"},
Packit fc043f
    {28591, "ISO-IR-100"},
Packit fc043f
    {28591, "ISO8859-1"},
Packit fc043f
    {28591, "ISO_8859-1"},
Packit fc043f
    {28591, "ISO_8859-1:1987"},
Packit fc043f
    {28591, "L1"},
Packit fc043f
    {28591, "LATIN1"},
Packit fc043f
    {28591, "CSISOLATIN1"},
Packit fc043f
Packit fc043f
    {1250, "CP1250"},
Packit fc043f
    {1250, "MS-EE"},
Packit fc043f
    {1250, "WINDOWS-1250"},
Packit fc043f
Packit fc043f
    {1251, "CP1251"},
Packit fc043f
    {1251, "MS-CYRL"},
Packit fc043f
    {1251, "WINDOWS-1251"},
Packit fc043f
Packit fc043f
    {1252, "CP1252"},
Packit fc043f
    {1252, "MS-ANSI"},
Packit fc043f
    {1252, "WINDOWS-1252"},
Packit fc043f
Packit fc043f
    {1253, "CP1253"},
Packit fc043f
    {1253, "MS-GREEK"},
Packit fc043f
    {1253, "WINDOWS-1253"},
Packit fc043f
Packit fc043f
    {1254, "CP1254"},
Packit fc043f
    {1254, "MS-TURK"},
Packit fc043f
    {1254, "WINDOWS-1254"},
Packit fc043f
Packit fc043f
    {1255, "CP1255"},
Packit fc043f
    {1255, "MS-HEBR"},
Packit fc043f
    {1255, "WINDOWS-1255"},
Packit fc043f
Packit fc043f
    {1256, "CP1256"},
Packit fc043f
    {1256, "MS-ARAB"},
Packit fc043f
    {1256, "WINDOWS-1256"},
Packit fc043f
Packit fc043f
    {1257, "CP1257"},
Packit fc043f
    {1257, "WINBALTRIM"},
Packit fc043f
    {1257, "WINDOWS-1257"},
Packit fc043f
Packit fc043f
    {1258, "CP1258"},
Packit fc043f
    {1258, "WINDOWS-1258"},
Packit fc043f
Packit fc043f
    {850, "850"},
Packit fc043f
    {850, "CP850"},
Packit fc043f
    {850, "IBM850"},
Packit fc043f
    {850, "CSPC850MULTILINGUAL"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(862) */
Packit fc043f
    {862, "862"},
Packit fc043f
    {862, "CP862"},
Packit fc043f
    {862, "IBM862"},
Packit fc043f
    {862, "CSPC862LATINHEBREW"},
Packit fc043f
Packit fc043f
    {866, "866"},
Packit fc043f
    {866, "CP866"},
Packit fc043f
    {866, "IBM866"},
Packit fc043f
    {866, "CSIBM866"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(154) */
Packit fc043f
    {154, "CP154"},
Packit fc043f
    {154, "CYRILLIC-ASIAN"},
Packit fc043f
    {154, "PT154"},
Packit fc043f
    {154, "PTCP154"},
Packit fc043f
    {154, "CSPTCP154"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(1133) */
Packit fc043f
    {1133, "CP1133"},
Packit fc043f
    {1133, "IBM-CP1133"},
Packit fc043f
Packit fc043f
    {874, "CP874"},
Packit fc043f
    {874, "WINDOWS-874"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(51932) */
Packit fc043f
    {51932, "CP51932"},
Packit fc043f
    {51932, "MS51932"},
Packit fc043f
    {51932, "WINDOWS-51932"},
Packit fc043f
    {51932, "EUC-JP"},
Packit fc043f
Packit fc043f
    {932, "CP932"},
Packit fc043f
    {932, "MS932"},
Packit fc043f
    {932, "SHIFFT_JIS"},
Packit fc043f
    {932, "SHIFFT_JIS-MS"},
Packit fc043f
    {932, "SJIS"},
Packit fc043f
    {932, "SJIS-MS"},
Packit fc043f
    {932, "SJIS-OPEN"},
Packit fc043f
    {932, "SJIS-WIN"},
Packit fc043f
    {932, "WINDOWS-31J"},
Packit fc043f
    {932, "WINDOWS-932"},
Packit fc043f
    {932, "CSWINDOWS31J"},
Packit fc043f
Packit fc043f
    {50221, "CP50221"},
Packit fc043f
    {50221, "ISO-2022-JP"},
Packit fc043f
    {50221, "ISO-2022-JP-MS"},
Packit fc043f
    {50221, "ISO2022-JP"},
Packit fc043f
    {50221, "ISO2022-JP-MS"},
Packit fc043f
    {50221, "MS50221"},
Packit fc043f
    {50221, "WINDOWS-50221"},
Packit fc043f
Packit fc043f
    {936, "CP936"},
Packit fc043f
    {936, "GBK"},
Packit fc043f
    {936, "MS936"},
Packit fc043f
    {936, "WINDOWS-936"},
Packit fc043f
Packit fc043f
    {950, "CP950"},
Packit fc043f
    {950, "BIG5"},
Packit fc043f
    {950, "BIG5HKSCS"},
Packit fc043f
    {950, "BIG5-HKSCS"},
Packit fc043f
Packit fc043f
    {949, "CP949"},
Packit fc043f
    {949, "UHC"},
Packit fc043f
    {949, "EUC-KR"},
Packit fc043f
Packit fc043f
    {1361, "CP1361"},
Packit fc043f
    {1361, "JOHAB"},
Packit fc043f
Packit fc043f
    {437, "437"},
Packit fc043f
    {437, "CP437"},
Packit fc043f
    {437, "IBM437"},
Packit fc043f
    {437, "CSPC8CODEPAGE437"},
Packit fc043f
Packit fc043f
    {737, "CP737"},
Packit fc043f
Packit fc043f
    {775, "CP775"},
Packit fc043f
    {775, "IBM775"},
Packit fc043f
    {775, "CSPC775BALTIC"},
Packit fc043f
Packit fc043f
    {852, "852"},
Packit fc043f
    {852, "CP852"},
Packit fc043f
    {852, "IBM852"},
Packit fc043f
    {852, "CSPCP852"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(853) */
Packit fc043f
    {853, "CP853"},
Packit fc043f
Packit fc043f
    {855, "855"},
Packit fc043f
    {855, "CP855"},
Packit fc043f
    {855, "IBM855"},
Packit fc043f
    {855, "CSIBM855"},
Packit fc043f
Packit fc043f
    {857, "857"},
Packit fc043f
    {857, "CP857"},
Packit fc043f
    {857, "IBM857"},
Packit fc043f
    {857, "CSIBM857"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(858) */
Packit fc043f
    {858, "CP858"},
Packit fc043f
Packit fc043f
    {860, "860"},
Packit fc043f
    {860, "CP860"},
Packit fc043f
    {860, "IBM860"},
Packit fc043f
    {860, "CSIBM860"},
Packit fc043f
Packit fc043f
    {861, "861"},
Packit fc043f
    {861, "CP-IS"},
Packit fc043f
    {861, "CP861"},
Packit fc043f
    {861, "IBM861"},
Packit fc043f
    {861, "CSIBM861"},
Packit fc043f
Packit fc043f
    {863, "863"},
Packit fc043f
    {863, "CP863"},
Packit fc043f
    {863, "IBM863"},
Packit fc043f
    {863, "CSIBM863"},
Packit fc043f
Packit fc043f
    {864, "CP864"},
Packit fc043f
    {864, "IBM864"},
Packit fc043f
    {864, "CSIBM864"},
Packit fc043f
Packit fc043f
    {865, "865"},
Packit fc043f
    {865, "CP865"},
Packit fc043f
    {865, "IBM865"},
Packit fc043f
    {865, "CSIBM865"},
Packit fc043f
Packit fc043f
    {869, "869"},
Packit fc043f
    {869, "CP-GR"},
Packit fc043f
    {869, "CP869"},
Packit fc043f
    {869, "IBM869"},
Packit fc043f
    {869, "CSIBM869"},
Packit fc043f
Packit fc043f
    /* !IsValidCodePage(1152) */
Packit fc043f
    {1125, "CP1125"},
Packit fc043f
Packit fc043f
    /*
Packit fc043f
     * Code Page Identifiers
Packit fc043f
     * https://msdn.microsoft.com/en-us/library/dd317756.aspx
Packit fc043f
     */
Packit fc043f
    {37, "IBM037"}, /* IBM EBCDIC US-Canada */
Packit fc043f
    {437, "IBM437"}, /* OEM United States */
Packit fc043f
    {500, "IBM500"}, /* IBM EBCDIC International */
Packit fc043f
    {708, "ASMO-708"}, /* Arabic (ASMO 708) */
Packit fc043f
    /* 709 		Arabic (ASMO-449+, BCON V4) */
Packit fc043f
    /* 710 		Arabic - Transparent Arabic */
Packit fc043f
    {720, "DOS-720"}, /* Arabic (Transparent ASMO); Arabic (DOS) */
Packit fc043f
    {737, "ibm737"}, /* OEM Greek (formerly 437G); Greek (DOS) */
Packit fc043f
    {775, "ibm775"}, /* OEM Baltic; Baltic (DOS) */
Packit fc043f
    {850, "ibm850"}, /* OEM Multilingual Latin 1; Western European (DOS) */
Packit fc043f
    {852, "ibm852"}, /* OEM Latin 2; Central European (DOS) */
Packit fc043f
    {855, "IBM855"}, /* OEM Cyrillic (primarily Russian) */
Packit fc043f
    {857, "ibm857"}, /* OEM Turkish; Turkish (DOS) */
Packit fc043f
    {858, "IBM00858"}, /* OEM Multilingual Latin 1 + Euro symbol */
Packit fc043f
    {860, "IBM860"}, /* OEM Portuguese; Portuguese (DOS) */
Packit fc043f
    {861, "ibm861"}, /* OEM Icelandic; Icelandic (DOS) */
Packit fc043f
    {862, "DOS-862"}, /* OEM Hebrew; Hebrew (DOS) */
Packit fc043f
    {863, "IBM863"}, /* OEM French Canadian; French Canadian (DOS) */
Packit fc043f
    {864, "IBM864"}, /* OEM Arabic; Arabic (864) */
Packit fc043f
    {865, "IBM865"}, /* OEM Nordic; Nordic (DOS) */
Packit fc043f
    {866, "cp866"}, /* OEM Russian; Cyrillic (DOS) */
Packit fc043f
    {869, "ibm869"}, /* OEM Modern Greek; Greek, Modern (DOS) */
Packit fc043f
    {870, "IBM870"}, /* IBM EBCDIC Multilingual/ROECE (Latin 2); IBM EBCDIC Multilingual Latin 2 */
Packit fc043f
    {874, "windows-874"}, /* ANSI/OEM Thai (same as 28605, ISO 8859-15); Thai (Windows) */
Packit fc043f
    {875, "cp875"}, /* IBM EBCDIC Greek Modern */
Packit fc043f
    {932, "shift_jis"}, /* ANSI/OEM Japanese; Japanese (Shift-JIS) */
Packit fc043f
    {932, "shift-jis"}, /* alternative name for it */
Packit fc043f
    {936, "gb2312"}, /* ANSI/OEM Simplified Chinese (PRC, Singapore); Chinese Simplified (GB2312) */
Packit fc043f
    {949, "ks_c_5601-1987"}, /* ANSI/OEM Korean (Unified Hangul Code) */
Packit fc043f
    {950, "big5"}, /* ANSI/OEM Traditional Chinese (Taiwan; Hong Kong SAR, PRC); Chinese Traditional (Big5) */
Packit fc043f
    {950, "big5hkscs"}, /* ANSI/OEM Traditional Chinese (Hong Kong SAR); Chinese Traditional (Big5-HKSCS) */
Packit fc043f
    {950, "big5-hkscs"}, /* alternative name for it */
Packit fc043f
    {1026, "IBM1026"}, /* IBM EBCDIC Turkish (Latin 5) */
Packit fc043f
    {1047, "IBM01047"}, /* IBM EBCDIC Latin 1/Open System */
Packit fc043f
    {1140, "IBM01140"}, /* IBM EBCDIC US-Canada (037 + Euro symbol); IBM EBCDIC (US-Canada-Euro) */
Packit fc043f
    {1141, "IBM01141"}, /* IBM EBCDIC Germany (20273 + Euro symbol); IBM EBCDIC (Germany-Euro) */
Packit fc043f
    {1142, "IBM01142"}, /* IBM EBCDIC Denmark-Norway (20277 + Euro symbol); IBM EBCDIC (Denmark-Norway-Euro) */
Packit fc043f
    {1143, "IBM01143"}, /* IBM EBCDIC Finland-Sweden (20278 + Euro symbol); IBM EBCDIC (Finland-Sweden-Euro) */
Packit fc043f
    {1144, "IBM01144"}, /* IBM EBCDIC Italy (20280 + Euro symbol); IBM EBCDIC (Italy-Euro) */
Packit fc043f
    {1145, "IBM01145"}, /* IBM EBCDIC Latin America-Spain (20284 + Euro symbol); IBM EBCDIC (Spain-Euro) */
Packit fc043f
    {1146, "IBM01146"}, /* IBM EBCDIC United Kingdom (20285 + Euro symbol); IBM EBCDIC (UK-Euro) */
Packit fc043f
    {1147, "IBM01147"}, /* IBM EBCDIC France (20297 + Euro symbol); IBM EBCDIC (France-Euro) */
Packit fc043f
    {1148, "IBM01148"}, /* IBM EBCDIC International (500 + Euro symbol); IBM EBCDIC (International-Euro) */
Packit fc043f
    {1149, "IBM01149"}, /* IBM EBCDIC Icelandic (20871 + Euro symbol); IBM EBCDIC (Icelandic-Euro) */
Packit fc043f
    {1250, "windows-1250"}, /* ANSI Central European; Central European (Windows) */
Packit fc043f
    {1251, "windows-1251"}, /* ANSI Cyrillic; Cyrillic (Windows) */
Packit fc043f
    {1252, "windows-1252"}, /* ANSI Latin 1; Western European (Windows) */
Packit fc043f
    {1253, "windows-1253"}, /* ANSI Greek; Greek (Windows) */
Packit fc043f
    {1254, "windows-1254"}, /* ANSI Turkish; Turkish (Windows) */
Packit fc043f
    {1255, "windows-1255"}, /* ANSI Hebrew; Hebrew (Windows) */
Packit fc043f
    {1256, "windows-1256"}, /* ANSI Arabic; Arabic (Windows) */
Packit fc043f
    {1257, "windows-1257"}, /* ANSI Baltic; Baltic (Windows) */
Packit fc043f
    {1258, "windows-1258"}, /* ANSI/OEM Vietnamese; Vietnamese (Windows) */
Packit fc043f
    {1361, "Johab"}, /* Korean (Johab) */
Packit fc043f
    {10000, "macintosh"}, /* MAC Roman; Western European (Mac) */
Packit fc043f
    {10001, "x-mac-japanese"}, /* Japanese (Mac) */
Packit fc043f
    {10002, "x-mac-chinesetrad"}, /* MAC Traditional Chinese (Big5); Chinese Traditional (Mac) */
Packit fc043f
    {10003, "x-mac-korean"}, /* Korean (Mac) */
Packit fc043f
    {10004, "x-mac-arabic"}, /* Arabic (Mac) */
Packit fc043f
    {10005, "x-mac-hebrew"}, /* Hebrew (Mac) */
Packit fc043f
    {10006, "x-mac-greek"}, /* Greek (Mac) */
Packit fc043f
    {10007, "x-mac-cyrillic"}, /* Cyrillic (Mac) */
Packit fc043f
    {10008, "x-mac-chinesesimp"}, /* MAC Simplified Chinese (GB 2312); Chinese Simplified (Mac) */
Packit fc043f
    {10010, "x-mac-romanian"}, /* Romanian (Mac) */
Packit fc043f
    {10017, "x-mac-ukrainian"}, /* Ukrainian (Mac) */
Packit fc043f
    {10021, "x-mac-thai"}, /* Thai (Mac) */
Packit fc043f
    {10029, "x-mac-ce"}, /* MAC Latin 2; Central European (Mac) */
Packit fc043f
    {10079, "x-mac-icelandic"}, /* Icelandic (Mac) */
Packit fc043f
    {10081, "x-mac-turkish"}, /* Turkish (Mac) */
Packit fc043f
    {10082, "x-mac-croatian"}, /* Croatian (Mac) */
Packit fc043f
    {20000, "x-Chinese_CNS"}, /* CNS Taiwan; Chinese Traditional (CNS) */
Packit fc043f
    {20001, "x-cp20001"}, /* TCA Taiwan */
Packit fc043f
    {20002, "x_Chinese-Eten"}, /* Eten Taiwan; Chinese Traditional (Eten) */
Packit fc043f
    {20003, "x-cp20003"}, /* IBM5550 Taiwan */
Packit fc043f
    {20004, "x-cp20004"}, /* TeleText Taiwan */
Packit fc043f
    {20005, "x-cp20005"}, /* Wang Taiwan */
Packit fc043f
    {20105, "x-IA5"}, /* IA5 (IRV International Alphabet No. 5, 7-bit); Western European (IA5) */
Packit fc043f
    {20106, "x-IA5-German"}, /* IA5 German (7-bit) */
Packit fc043f
    {20107, "x-IA5-Swedish"}, /* IA5 Swedish (7-bit) */
Packit fc043f
    {20108, "x-IA5-Norwegian"}, /* IA5 Norwegian (7-bit) */
Packit fc043f
    {20127, "us-ascii"}, /* US-ASCII (7-bit) */
Packit fc043f
    {20261, "x-cp20261"}, /* T.61 */
Packit fc043f
    {20269, "x-cp20269"}, /* ISO 6937 Non-Spacing Accent */
Packit fc043f
    {20273, "IBM273"}, /* IBM EBCDIC Germany */
Packit fc043f
    {20277, "IBM277"}, /* IBM EBCDIC Denmark-Norway */
Packit fc043f
    {20278, "IBM278"}, /* IBM EBCDIC Finland-Sweden */
Packit fc043f
    {20280, "IBM280"}, /* IBM EBCDIC Italy */
Packit fc043f
    {20284, "IBM284"}, /* IBM EBCDIC Latin America-Spain */
Packit fc043f
    {20285, "IBM285"}, /* IBM EBCDIC United Kingdom */
Packit fc043f
    {20290, "IBM290"}, /* IBM EBCDIC Japanese Katakana Extended */
Packit fc043f
    {20297, "IBM297"}, /* IBM EBCDIC France */
Packit fc043f
    {20420, "IBM420"}, /* IBM EBCDIC Arabic */
Packit fc043f
    {20423, "IBM423"}, /* IBM EBCDIC Greek */
Packit fc043f
    {20424, "IBM424"}, /* IBM EBCDIC Hebrew */
Packit fc043f
    {20833, "x-EBCDIC-KoreanExtended"}, /* IBM EBCDIC Korean Extended */
Packit fc043f
    {20838, "IBM-Thai"}, /* IBM EBCDIC Thai */
Packit fc043f
    {20866, "koi8-r"}, /* Russian (KOI8-R); Cyrillic (KOI8-R) */
Packit fc043f
    {20871, "IBM871"}, /* IBM EBCDIC Icelandic */
Packit fc043f
    {20880, "IBM880"}, /* IBM EBCDIC Cyrillic Russian */
Packit fc043f
    {20905, "IBM905"}, /* IBM EBCDIC Turkish */
Packit fc043f
    {20924, "IBM00924"}, /* IBM EBCDIC Latin 1/Open System (1047 + Euro symbol) */
Packit fc043f
    {20932, "EUC-JP"}, /* Japanese (JIS 0208-1990 and 0121-1990) */
Packit fc043f
    {20936, "x-cp20936"}, /* Simplified Chinese (GB2312); Chinese Simplified (GB2312-80) */
Packit fc043f
    {20949, "x-cp20949"}, /* Korean Wansung */
Packit fc043f
    {21025, "cp1025"}, /* IBM EBCDIC Cyrillic Serbian-Bulgarian */
Packit fc043f
    /* 21027 		(deprecated) */
Packit fc043f
    {21866, "koi8-u"}, /* Ukrainian (KOI8-U); Cyrillic (KOI8-U) */
Packit fc043f
    {28591, "iso-8859-1"}, /* ISO 8859-1 Latin 1; Western European (ISO) */
Packit fc043f
    {28591, "iso8859-1"}, /* ISO 8859-1 Latin 1; Western European (ISO) */
Packit fc043f
    {28591, "iso_8859-1"},
Packit fc043f
    {28591, "iso_8859_1"},
Packit fc043f
    {28592, "iso-8859-2"}, /* ISO 8859-2 Central European; Central European (ISO) */
Packit fc043f
    {28592, "iso8859-2"}, /* ISO 8859-2 Central European; Central European (ISO) */
Packit fc043f
    {28592, "iso_8859-2"},
Packit fc043f
    {28592, "iso_8859_2"},
Packit fc043f
    {28593, "iso-8859-3"}, /* ISO 8859-3 Latin 3 */
Packit fc043f
    {28593, "iso8859-3"}, /* ISO 8859-3 Latin 3 */
Packit fc043f
    {28593, "iso_8859-3"},
Packit fc043f
    {28593, "iso_8859_3"},
Packit fc043f
    {28594, "iso-8859-4"}, /* ISO 8859-4 Baltic */
Packit fc043f
    {28594, "iso8859-4"}, /* ISO 8859-4 Baltic */
Packit fc043f
    {28594, "iso_8859-4"},
Packit fc043f
    {28594, "iso_8859_4"},
Packit fc043f
    {28595, "iso-8859-5"}, /* ISO 8859-5 Cyrillic */
Packit fc043f
    {28595, "iso8859-5"}, /* ISO 8859-5 Cyrillic */
Packit fc043f
    {28595, "iso_8859-5"},
Packit fc043f
    {28595, "iso_8859_5"},
Packit fc043f
    {28596, "iso-8859-6"}, /* ISO 8859-6 Arabic */
Packit fc043f
    {28596, "iso8859-6"}, /* ISO 8859-6 Arabic */
Packit fc043f
    {28596, "iso_8859-6"},
Packit fc043f
    {28596, "iso_8859_6"},
Packit fc043f
    {28597, "iso-8859-7"}, /* ISO 8859-7 Greek */
Packit fc043f
    {28597, "iso8859-7"}, /* ISO 8859-7 Greek */
Packit fc043f
    {28597, "iso_8859-7"},
Packit fc043f
    {28597, "iso_8859_7"},
Packit fc043f
    {28598, "iso-8859-8"}, /* ISO 8859-8 Hebrew; Hebrew (ISO-Visual) */
Packit fc043f
    {28598, "iso8859-8"}, /* ISO 8859-8 Hebrew; Hebrew (ISO-Visual) */
Packit fc043f
    {28598, "iso_8859-8"},
Packit fc043f
    {28598, "iso_8859_8"},
Packit fc043f
    {28599, "iso-8859-9"}, /* ISO 8859-9 Turkish */
Packit fc043f
    {28599, "iso8859-9"}, /* ISO 8859-9 Turkish */
Packit fc043f
    {28599, "iso_8859-9"},
Packit fc043f
    {28599, "iso_8859_9"},
Packit fc043f
    {28603, "iso-8859-13"}, /* ISO 8859-13 Estonian */
Packit fc043f
    {28603, "iso8859-13"}, /* ISO 8859-13 Estonian */
Packit fc043f
    {28603, "iso_8859-13"},
Packit fc043f
    {28603, "iso_8859_13"},
Packit fc043f
    {28605, "iso-8859-15"}, /* ISO 8859-15 Latin 9 */
Packit fc043f
    {28605, "iso8859-15"}, /* ISO 8859-15 Latin 9 */
Packit fc043f
    {28605, "iso_8859-15"},
Packit fc043f
    {28605, "iso_8859_15"},
Packit fc043f
    {29001, "x-Europa"}, /* Europa 3 */
Packit fc043f
    {38598, "iso-8859-8-i"}, /* ISO 8859-8 Hebrew; Hebrew (ISO-Logical) */
Packit fc043f
    {38598, "iso8859-8-i"}, /* ISO 8859-8 Hebrew; Hebrew (ISO-Logical) */
Packit fc043f
    {38598, "iso_8859-8-i"},
Packit fc043f
    {38598, "iso_8859_8-i"},
Packit fc043f
    {50220, "iso-2022-jp"}, /* ISO 2022 Japanese with no halfwidth Katakana; Japanese (JIS) */
Packit fc043f
    {50221, "csISO2022JP"}, /* ISO 2022 Japanese with halfwidth Katakana; Japanese (JIS-Allow 1 byte Kana) */
Packit fc043f
    {50222, "iso-2022-jp"}, /* ISO 2022 Japanese JIS X 0201-1989; Japanese (JIS-Allow 1 byte Kana - SO/SI) */
Packit fc043f
    {50225, "iso-2022-kr"}, /* ISO 2022 Korean */
Packit fc043f
    {50225, "iso2022-kr"}, /* ISO 2022 Korean */
Packit fc043f
    {50227, "x-cp50227"}, /* ISO 2022 Simplified Chinese; Chinese Simplified (ISO 2022) */
Packit fc043f
    /* 50229 		ISO 2022 Traditional Chinese */
Packit fc043f
    /* 50930 		EBCDIC Japanese (Katakana) Extended */
Packit fc043f
    /* 50931 		EBCDIC US-Canada and Japanese */
Packit fc043f
    /* 50933 		EBCDIC Korean Extended and Korean */
Packit fc043f
    /* 50935 		EBCDIC Simplified Chinese Extended and Simplified Chinese */
Packit fc043f
    /* 50936 		EBCDIC Simplified Chinese */
Packit fc043f
    /* 50937 		EBCDIC US-Canada and Traditional Chinese */
Packit fc043f
    /* 50939 		EBCDIC Japanese (Latin) Extended and Japanese */
Packit fc043f
    {51932, "euc-jp"}, /* EUC Japanese */
Packit fc043f
    {51936, "EUC-CN"}, /* EUC Simplified Chinese; Chinese Simplified (EUC) */
Packit fc043f
    {51949, "euc-kr"}, /* EUC Korean */
Packit fc043f
    /* 51950 		EUC Traditional Chinese */
Packit fc043f
    {52936, "hz-gb-2312"}, /* HZ-GB2312 Simplified Chinese; Chinese Simplified (HZ) */
Packit fc043f
    {54936, "GB18030"}, /* Windows XP and later: GB18030 Simplified Chinese (4 byte); Chinese Simplified (GB18030) */
Packit fc043f
    {57002, "x-iscii-de"}, /* ISCII Devanagari */
Packit fc043f
    {57003, "x-iscii-be"}, /* ISCII Bengali */
Packit fc043f
    {57004, "x-iscii-ta"}, /* ISCII Tamil */
Packit fc043f
    {57005, "x-iscii-te"}, /* ISCII Telugu */
Packit fc043f
    {57006, "x-iscii-as"}, /* ISCII Assamese */
Packit fc043f
    {57007, "x-iscii-or"}, /* ISCII Oriya */
Packit fc043f
    {57008, "x-iscii-ka"}, /* ISCII Kannada */
Packit fc043f
    {57009, "x-iscii-ma"}, /* ISCII Malayalam */
Packit fc043f
    {57010, "x-iscii-gu"}, /* ISCII Gujarati */
Packit fc043f
    {57011, "x-iscii-pa"}, /* ISCII Punjabi */
Packit fc043f
Packit fc043f
    {0, NULL}
Packit fc043f
};
Packit fc043f
Packit fc043f
/*
Packit fc043f
 * SJIS SHIFTJIS table              CP932 table
Packit fc043f
 * ---- --------------------------- --------------------------------
Packit fc043f
 *   5C U+00A5 YEN SIGN             U+005C REVERSE SOLIDUS
Packit fc043f
 *   7E U+203E OVERLINE             U+007E TILDE
Packit fc043f
 * 815C U+2014 EM DASH              U+2015 HORIZONTAL BAR
Packit fc043f
 * 815F U+005C REVERSE SOLIDUS      U+FF3C FULLWIDTH REVERSE SOLIDUS
Packit fc043f
 * 8160 U+301C WAVE DASH            U+FF5E FULLWIDTH TILDE
Packit fc043f
 * 8161 U+2016 DOUBLE VERTICAL LINE U+2225 PARALLEL TO
Packit fc043f
 * 817C U+2212 MINUS SIGN           U+FF0D FULLWIDTH HYPHEN-MINUS
Packit fc043f
 * 8191 U+00A2 CENT SIGN            U+FFE0 FULLWIDTH CENT SIGN
Packit fc043f
 * 8192 U+00A3 POUND SIGN           U+FFE1 FULLWIDTH POUND SIGN
Packit fc043f
 * 81CA U+00AC NOT SIGN             U+FFE2 FULLWIDTH NOT SIGN
Packit fc043f
 *
Packit fc043f
 * EUC-JP and ISO-2022-JP should be compatible with CP932.
Packit fc043f
 *
Packit fc043f
 * Kernel and MLang have different Unicode mapping table.  Make sure
Packit fc043f
 * which API is used.
Packit fc043f
 */
Packit fc043f
static compat_t cp932_compat[] = {
Packit fc043f
    {0x00A5, 0x005C, COMPAT_OUT},
Packit fc043f
    {0x203E, 0x007E, COMPAT_OUT},
Packit fc043f
    {0x2014, 0x2015, COMPAT_OUT},
Packit fc043f
    {0x301C, 0xFF5E, COMPAT_OUT},
Packit fc043f
    {0x2016, 0x2225, COMPAT_OUT},
Packit fc043f
    {0x2212, 0xFF0D, COMPAT_OUT},
Packit fc043f
    {0x00A2, 0xFFE0, COMPAT_OUT},
Packit fc043f
    {0x00A3, 0xFFE1, COMPAT_OUT},
Packit fc043f
    {0x00AC, 0xFFE2, COMPAT_OUT},
Packit fc043f
    {0, 0, 0}
Packit fc043f
};
Packit fc043f
Packit fc043f
static compat_t cp20932_compat[] = {
Packit fc043f
    {0x00A5, 0x005C, COMPAT_OUT},
Packit fc043f
    {0x203E, 0x007E, COMPAT_OUT},
Packit fc043f
    {0x2014, 0x2015, COMPAT_OUT},
Packit fc043f
    {0xFF5E, 0x301C, COMPAT_OUT|COMPAT_IN},
Packit fc043f
    {0x2225, 0x2016, COMPAT_OUT|COMPAT_IN},
Packit fc043f
    {0xFF0D, 0x2212, COMPAT_OUT|COMPAT_IN},
Packit fc043f
    {0xFFE0, 0x00A2, COMPAT_OUT|COMPAT_IN},
Packit fc043f
    {0xFFE1, 0x00A3, COMPAT_OUT|COMPAT_IN},
Packit fc043f
    {0xFFE2, 0x00AC, COMPAT_OUT|COMPAT_IN},
Packit fc043f
    {0, 0, 0}
Packit fc043f
};
Packit fc043f
Packit fc043f
static compat_t *cp51932_compat = cp932_compat;
Packit fc043f
Packit fc043f
/* cp20932_compat for kernel.  cp932_compat for mlang. */
Packit fc043f
static compat_t *cp5022x_compat = cp932_compat;
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
typedef HRESULT (WINAPI *CONVERTINETSTRING)(
Packit fc043f
    LPDWORD lpdwMode,
Packit fc043f
    DWORD dwSrcEncoding,
Packit fc043f
    DWORD dwDstEncoding,
Packit fc043f
    LPCSTR lpSrcStr,
Packit fc043f
    LPINT lpnSrcSize,
Packit fc043f
    LPBYTE lpDstStr,
Packit fc043f
    LPINT lpnDstSize
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *CONVERTINETMULTIBYTETOUNICODE)(
Packit fc043f
    LPDWORD lpdwMode,
Packit fc043f
    DWORD dwSrcEncoding,
Packit fc043f
    LPCSTR lpSrcStr,
Packit fc043f
    LPINT lpnMultiCharCount,
Packit fc043f
    LPWSTR lpDstStr,
Packit fc043f
    LPINT lpnWideCharCount
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *CONVERTINETUNICODETOMULTIBYTE)(
Packit fc043f
    LPDWORD lpdwMode,
Packit fc043f
    DWORD dwEncoding,
Packit fc043f
    LPCWSTR lpSrcStr,
Packit fc043f
    LPINT lpnWideCharCount,
Packit fc043f
    LPSTR lpDstStr,
Packit fc043f
    LPINT lpnMultiCharCount
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *ISCONVERTINETSTRINGAVAILABLE)(
Packit fc043f
    DWORD dwSrcEncoding,
Packit fc043f
    DWORD dwDstEncoding
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *LCIDTORFC1766A)(
Packit fc043f
    LCID Locale,
Packit fc043f
    LPSTR pszRfc1766,
Packit fc043f
    int nChar
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *LCIDTORFC1766W)(
Packit fc043f
    LCID Locale,
Packit fc043f
    LPWSTR pszRfc1766,
Packit fc043f
    int nChar
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *RFC1766TOLCIDA)(
Packit fc043f
    LCID *pLocale,
Packit fc043f
    LPSTR pszRfc1766
Packit fc043f
);
Packit fc043f
typedef HRESULT (WINAPI *RFC1766TOLCIDW)(
Packit fc043f
    LCID *pLocale,
Packit fc043f
    LPWSTR pszRfc1766
Packit fc043f
);
Packit fc043f
static CONVERTINETSTRING ConvertINetString;
Packit fc043f
static CONVERTINETMULTIBYTETOUNICODE ConvertINetMultiByteToUnicode;
Packit fc043f
static CONVERTINETUNICODETOMULTIBYTE ConvertINetUnicodeToMultiByte;
Packit fc043f
static ISCONVERTINETSTRINGAVAILABLE IsConvertINetStringAvailable;
Packit fc043f
static LCIDTORFC1766A LcidToRfc1766A;
Packit fc043f
static RFC1766TOLCIDA Rfc1766ToLcidA;
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int
Packit fc043f
load_mlang(void)
Packit fc043f
{
Packit fc043f
    HMODULE h;
Packit fc043f
    if (ConvertINetString != NULL)
Packit fc043f
        return TRUE;
Packit fc043f
    h = LoadLibrary(TEXT("mlang.dll"));
Packit fc043f
    if (!h)
Packit fc043f
        return FALSE;
Packit fc043f
    ConvertINetString = (CONVERTINETSTRING)GetProcAddressA(h, "ConvertINetString");
Packit fc043f
    ConvertINetMultiByteToUnicode = (CONVERTINETMULTIBYTETOUNICODE)GetProcAddressA(h, "ConvertINetMultiByteToUnicode");
Packit fc043f
    ConvertINetUnicodeToMultiByte = (CONVERTINETUNICODETOMULTIBYTE)GetProcAddressA(h, "ConvertINetUnicodeToMultiByte");
Packit fc043f
    IsConvertINetStringAvailable = (ISCONVERTINETSTRINGAVAILABLE)GetProcAddressA(h, "IsConvertINetStringAvailable");
Packit fc043f
    LcidToRfc1766A = (LCIDTORFC1766A)GetProcAddressA(h, "LcidToRfc1766A");
Packit fc043f
    Rfc1766ToLcidA = (RFC1766TOLCIDA)GetProcAddressA(h, "Rfc1766ToLcidA");
Packit fc043f
    return TRUE;
Packit fc043f
}
Packit fc043f
#endif
Packit fc043f
Packit fc043f
Packit fc043f
static int
Packit fc043f
win_iconv_open(gpgrt_w32_iconv_t cd, const char *tocode, const char *fromcode)
Packit fc043f
{
Packit fc043f
    if (!make_csconv(fromcode, &cd->from) || !make_csconv(tocode, &cd->to))
Packit fc043f
        return FALSE;
Packit fc043f
    cd->_errno = _errno;
Packit fc043f
    return TRUE;
Packit fc043f
}
Packit fc043f
Packit fc043f
static size_t
Packit fc043f
win_iconv (gpgrt_w32_iconv_t cd,
Packit fc043f
           const char **inbuf, size_t *inbytesleft,
Packit fc043f
           char **outbuf, size_t *outbytesleft)
Packit fc043f
{
Packit fc043f
    ushort wbuf[MB_CHAR_MAX]; /* enough room for one character */
Packit fc043f
    int insize;
Packit fc043f
    int outsize;
Packit fc043f
    int wsize;
Packit fc043f
    DWORD frommode;
Packit fc043f
    DWORD tomode;
Packit fc043f
    uint wc;
Packit fc043f
    compat_t *cp;
Packit fc043f
    int i;
Packit fc043f
Packit fc043f
    if (inbuf == NULL || *inbuf == NULL)
Packit fc043f
    {
Packit fc043f
        if (outbuf != NULL && *outbuf != NULL && cd->to.flush != NULL)
Packit fc043f
        {
Packit fc043f
            tomode = cd->to.mode;
Packit fc043f
            outsize = cd->to.flush(&cd->to, (uchar *)*outbuf, *outbytesleft);
Packit fc043f
            if (outsize == -1)
Packit fc043f
            {
Packit fc043f
                if ((cd->to.flags & FLAG_IGNORE) && errno != E2BIG)
Packit fc043f
                {
Packit fc043f
                    outsize = 0;
Packit fc043f
                }
Packit fc043f
                else
Packit fc043f
                {
Packit fc043f
                    cd->to.mode = tomode;
Packit fc043f
                    return (size_t)(-1);
Packit fc043f
                }
Packit fc043f
            }
Packit fc043f
            *outbuf += outsize;
Packit fc043f
            *outbytesleft -= outsize;
Packit fc043f
        }
Packit fc043f
        cd->from.mode = 0;
Packit fc043f
        cd->to.mode = 0;
Packit fc043f
        return 0;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    while (*inbytesleft != 0)
Packit fc043f
    {
Packit fc043f
        frommode = cd->from.mode;
Packit fc043f
        tomode = cd->to.mode;
Packit fc043f
        wsize = MB_CHAR_MAX;
Packit fc043f
Packit fc043f
        insize = cd->from.mbtowc(&cd->from, (const uchar *)*inbuf, *inbytesleft, wbuf, &wsize);
Packit fc043f
        if (insize == -1)
Packit fc043f
        {
Packit fc043f
            if (cd->to.flags & FLAG_IGNORE)
Packit fc043f
            {
Packit fc043f
                cd->from.mode = frommode;
Packit fc043f
                insize = 1;
Packit fc043f
                wsize = 0;
Packit fc043f
            }
Packit fc043f
            else
Packit fc043f
            {
Packit fc043f
                cd->from.mode = frommode;
Packit fc043f
                return (size_t)(-1);
Packit fc043f
            }
Packit fc043f
        }
Packit fc043f
Packit fc043f
        if (wsize == 0)
Packit fc043f
        {
Packit fc043f
            *inbuf += insize;
Packit fc043f
            *inbytesleft -= insize;
Packit fc043f
            continue;
Packit fc043f
        }
Packit fc043f
Packit fc043f
        if (cd->from.compat != NULL)
Packit fc043f
        {
Packit fc043f
            wc = utf16_to_ucs4(wbuf);
Packit fc043f
            cp = cd->from.compat;
Packit fc043f
            for (i = 0; cp[i].in != 0; ++i)
Packit fc043f
            {
Packit fc043f
                if ((cp[i].flag & COMPAT_IN) && cp[i].out == wc)
Packit fc043f
                {
Packit fc043f
                    ucs4_to_utf16(cp[i].in, wbuf, &wsize);
Packit fc043f
                    break;
Packit fc043f
                }
Packit fc043f
            }
Packit fc043f
        }
Packit fc043f
Packit fc043f
        if (cd->to.compat != NULL)
Packit fc043f
        {
Packit fc043f
            wc = utf16_to_ucs4(wbuf);
Packit fc043f
            cp = cd->to.compat;
Packit fc043f
            for (i = 0; cp[i].in != 0; ++i)
Packit fc043f
            {
Packit fc043f
                if ((cp[i].flag & COMPAT_OUT) && cp[i].in == wc)
Packit fc043f
                {
Packit fc043f
                    ucs4_to_utf16(cp[i].out, wbuf, &wsize);
Packit fc043f
                    break;
Packit fc043f
                }
Packit fc043f
            }
Packit fc043f
        }
Packit fc043f
Packit fc043f
        outsize = cd->to.wctomb(&cd->to, wbuf, wsize, (uchar *)*outbuf, *outbytesleft);
Packit fc043f
        if (outsize == -1)
Packit fc043f
        {
Packit fc043f
            if ((cd->to.flags & FLAG_IGNORE) && errno != E2BIG)
Packit fc043f
            {
Packit fc043f
                cd->to.mode = tomode;
Packit fc043f
                outsize = 0;
Packit fc043f
            }
Packit fc043f
            else
Packit fc043f
            {
Packit fc043f
                cd->from.mode = frommode;
Packit fc043f
                cd->to.mode = tomode;
Packit fc043f
                return (size_t)(-1);
Packit fc043f
            }
Packit fc043f
        }
Packit fc043f
Packit fc043f
        *inbuf += insize;
Packit fc043f
        *outbuf += outsize;
Packit fc043f
        *inbytesleft -= insize;
Packit fc043f
        *outbytesleft -= outsize;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    return 0;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
make_csconv(const char *_name, csconv_t *cv)
Packit fc043f
{
Packit fc043f
    CPINFO cpinfo;
Packit fc043f
    int use_compat = TRUE;
Packit fc043f
    int flag = 0;
Packit fc043f
    char *name;
Packit fc043f
    char *p;
Packit fc043f
Packit fc043f
    name = xstrndup(_name, strlen(_name));
Packit fc043f
    if (name == NULL)
Packit fc043f
        return FALSE;
Packit fc043f
Packit fc043f
    /* check for option "enc_name//opt1//opt2" */
Packit fc043f
    while ((p = strrstr(name, "//")) != NULL)
Packit fc043f
    {
Packit fc043f
        if (_stricmp(p + 2, "nocompat") == 0)
Packit fc043f
            use_compat = FALSE;
Packit fc043f
        else if (_stricmp(p + 2, "translit") == 0)
Packit fc043f
            flag |= FLAG_TRANSLIT;
Packit fc043f
        else if (_stricmp(p + 2, "ignore") == 0)
Packit fc043f
            flag |= FLAG_IGNORE;
Packit fc043f
        *p = 0;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    cv->mode = 0;
Packit fc043f
    cv->flags = flag;
Packit fc043f
    cv->mblen = NULL;
Packit fc043f
    cv->flush = NULL;
Packit fc043f
    cv->compat = NULL;
Packit fc043f
    cv->codepage = name_to_codepage(name);
Packit fc043f
    if (cv->codepage == 1200 || cv->codepage == 1201)
Packit fc043f
    {
Packit fc043f
        cv->mbtowc = utf16_mbtowc;
Packit fc043f
        cv->wctomb = utf16_wctomb;
Packit fc043f
        if (_stricmp(name, "UTF-16") == 0 || _stricmp(name, "UTF16") == 0 ||
Packit fc043f
          _stricmp(name, "UCS-2") == 0 || _stricmp(name, "UCS2") == 0 ||
Packit fc043f
	  _stricmp(name,"UCS-2-INTERNAL") == 0)
Packit fc043f
            cv->flags |= FLAG_USE_BOM;
Packit fc043f
    }
Packit fc043f
    else if (cv->codepage == 12000 || cv->codepage == 12001)
Packit fc043f
    {
Packit fc043f
        cv->mbtowc = utf32_mbtowc;
Packit fc043f
        cv->wctomb = utf32_wctomb;
Packit fc043f
        if (_stricmp(name, "UTF-32") == 0 || _stricmp(name, "UTF32") == 0 ||
Packit fc043f
          _stricmp(name, "UCS-4") == 0 || _stricmp(name, "UCS4") == 0)
Packit fc043f
            cv->flags |= FLAG_USE_BOM;
Packit fc043f
    }
Packit fc043f
    else if (cv->codepage == 65001)
Packit fc043f
    {
Packit fc043f
        cv->mbtowc = kernel_mbtowc;
Packit fc043f
        cv->wctomb = kernel_wctomb;
Packit fc043f
        cv->mblen = utf8_mblen;
Packit fc043f
    }
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
    else if ((cv->codepage == 50220 || cv->codepage == 50221 || cv->codepage == 50222) && load_mlang())
Packit fc043f
    {
Packit fc043f
        cv->mbtowc = iso2022jp_mbtowc;
Packit fc043f
        cv->wctomb = iso2022jp_wctomb;
Packit fc043f
        cv->flush = iso2022jp_flush;
Packit fc043f
    }
Packit fc043f
    else if (cv->codepage == 51932 && load_mlang())
Packit fc043f
    {
Packit fc043f
        cv->mbtowc = mlang_mbtowc;
Packit fc043f
        cv->wctomb = mlang_wctomb;
Packit fc043f
        cv->mblen = eucjp_mblen;
Packit fc043f
    }
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
    else if (IsValidCodePage(cv->codepage)
Packit fc043f
	     && GetCPInfo(cv->codepage, &cpinfo) != 0)
Packit fc043f
    {
Packit fc043f
        cv->mbtowc = kernel_mbtowc;
Packit fc043f
        cv->wctomb = kernel_wctomb;
Packit fc043f
        if (cpinfo.MaxCharSize == 1)
Packit fc043f
            cv->mblen = sbcs_mblen;
Packit fc043f
        else if (cpinfo.MaxCharSize == 2)
Packit fc043f
            cv->mblen = dbcs_mblen;
Packit fc043f
        else
Packit fc043f
	    cv->mblen = mbcs_mblen;
Packit fc043f
    }
Packit fc043f
    else
Packit fc043f
    {
Packit fc043f
        /* not supported */
Packit fc043f
        free(name);
Packit fc043f
        errno = EINVAL;
Packit fc043f
        return FALSE;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    if (use_compat)
Packit fc043f
    {
Packit fc043f
        switch (cv->codepage)
Packit fc043f
        {
Packit fc043f
        case 932: cv->compat = cp932_compat; break;
Packit fc043f
        case 20932: cv->compat = cp20932_compat; break;
Packit fc043f
        case 51932: cv->compat = cp51932_compat; break;
Packit fc043f
        case 50220: case 50221: case 50222: cv->compat = cp5022x_compat; break;
Packit fc043f
        }
Packit fc043f
    }
Packit fc043f
Packit fc043f
    free(name);
Packit fc043f
Packit fc043f
    return TRUE;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
name_to_codepage(const char *name)
Packit fc043f
{
Packit fc043f
    int i;
Packit fc043f
Packit fc043f
    if (*name == '\0' ||
Packit fc043f
	strcmp(name, "char") == 0)
Packit fc043f
        return GetACP();
Packit fc043f
    else if (strcmp(name, "wchar_t") == 0)
Packit fc043f
        return 1200;
Packit fc043f
    else if (_strnicmp(name, "cp", 2) == 0)
Packit fc043f
        return atoi(name + 2); /* CP123 */
Packit fc043f
    else if ('0' <= name[0] && name[0] <= '9')
Packit fc043f
        return atoi(name);     /* 123 */
Packit fc043f
    else if (_strnicmp(name, "xx", 2) == 0)
Packit fc043f
        return atoi(name + 2); /* XX123 for debug */
Packit fc043f
Packit fc043f
    for (i = 0; codepage_alias[i].name != NULL; ++i)
Packit fc043f
        if (_stricmp(name, codepage_alias[i].name) == 0)
Packit fc043f
            return codepage_alias[i].codepage;
Packit fc043f
    return -1;
Packit fc043f
}
Packit fc043f
Packit fc043f
/*
Packit fc043f
 * https://tools.ietf.org/html/rfc2781
Packit fc043f
 */
Packit fc043f
static uint
Packit fc043f
utf16_to_ucs4(const ushort *wbuf)
Packit fc043f
{
Packit fc043f
    uint wc = wbuf[0];
Packit fc043f
    if (0xD800 <= wbuf[0] && wbuf[0] <= 0xDBFF)
Packit fc043f
        wc = ((wbuf[0] & 0x3FF) << 10) + (wbuf[1] & 0x3FF) + 0x10000;
Packit fc043f
    return wc;
Packit fc043f
}
Packit fc043f
Packit fc043f
static void
Packit fc043f
ucs4_to_utf16(uint wc, ushort *wbuf, int *wbufsize)
Packit fc043f
{
Packit fc043f
    if (wc < 0x10000)
Packit fc043f
    {
Packit fc043f
        wbuf[0] = wc;
Packit fc043f
        *wbufsize = 1;
Packit fc043f
    }
Packit fc043f
    else
Packit fc043f
    {
Packit fc043f
        wc -= 0x10000;
Packit fc043f
        wbuf[0] = 0xD800 | ((wc >> 10) & 0x3FF);
Packit fc043f
        wbuf[1] = 0xDC00 | (wc & 0x3FF);
Packit fc043f
        *wbufsize = 2;
Packit fc043f
    }
Packit fc043f
}
Packit fc043f
Packit fc043f
/*
Packit fc043f
 * Check if codepage is one of those for which the dwFlags parameter
Packit fc043f
 * to MultiByteToWideChar() must be zero. Return zero or
Packit fc043f
 * MB_ERR_INVALID_CHARS.  The docs in Platform SDK for Windows
Packit fc043f
 * Server 2003 R2 claims that also codepage 65001 is one of these, but
Packit fc043f
 * that doesn't seem to be the case. The MSDN docs for MSVS2008 leave
Packit fc043f
 * out 65001 (UTF-8), and that indeed seems to be the case on XP, it
Packit fc043f
 * works fine to pass MB_ERR_INVALID_CHARS in dwFlags when converting
Packit fc043f
 * from UTF-8.
Packit fc043f
 */
Packit fc043f
static int
Packit fc043f
mbtowc_flags(int codepage)
Packit fc043f
{
Packit fc043f
    return (codepage == 50220 || codepage == 50221 ||
Packit fc043f
	    codepage == 50222 || codepage == 50225 ||
Packit fc043f
	    codepage == 50227 || codepage == 50229 ||
Packit fc043f
	    codepage == 52936 || codepage == 54936 ||
Packit fc043f
	    (codepage >= 57002 && codepage <= 57011) ||
Packit fc043f
	    codepage == 65000 || codepage == 42) ? 0 : MB_ERR_INVALID_CHARS;
Packit fc043f
}
Packit fc043f
Packit fc043f
/*
Packit fc043f
 * Check if codepage is one those for which the lpUsedDefaultChar
Packit fc043f
 * parameter to WideCharToMultiByte() must be NULL.  The docs in
Packit fc043f
 * Platform SDK for Windows Server 2003 R2 claims that this is the
Packit fc043f
 * list below, while the MSDN docs for MSVS2008 claim that it is only
Packit fc043f
 * for 65000 (UTF-7) and 65001 (UTF-8). This time the earlier Platform
Packit fc043f
 * SDK seems to be correct, at least for XP.
Packit fc043f
 */
Packit fc043f
static int
Packit fc043f
must_use_null_useddefaultchar(int codepage)
Packit fc043f
{
Packit fc043f
    return (codepage == 65000 || codepage == 65001 ||
Packit fc043f
            codepage == 50220 || codepage == 50221 ||
Packit fc043f
            codepage == 50222 || codepage == 50225 ||
Packit fc043f
            codepage == 50227 || codepage == 50229 ||
Packit fc043f
            codepage == 52936 || codepage == 54936 ||
Packit fc043f
            (codepage >= 57002 && codepage <= 57011) ||
Packit fc043f
            codepage == 42);
Packit fc043f
}
Packit fc043f
Packit fc043f
static char *
Packit fc043f
strrstr(const char *str, const char *token)
Packit fc043f
{
Packit fc043f
    int len = strlen(token);
Packit fc043f
    const char *p = str + strlen(str);
Packit fc043f
Packit fc043f
    while (str <= --p)
Packit fc043f
        if (p[0] == token[0] && strncmp(p, token, len) == 0)
Packit fc043f
            return (char *)p;
Packit fc043f
    return NULL;
Packit fc043f
}
Packit fc043f
Packit fc043f
static char *
Packit fc043f
xstrndup(const char *s, size_t n)
Packit fc043f
{
Packit fc043f
    char *p;
Packit fc043f
Packit fc043f
    p = (char *)malloc(n + 1);
Packit fc043f
    if (p == NULL)
Packit fc043f
        return NULL;
Packit fc043f
    memcpy(p, s, n);
Packit fc043f
    p[n] = '\0';
Packit fc043f
    return p;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
seterror(int err)
Packit fc043f
{
Packit fc043f
    _gpg_err_set_errno (err);
Packit fc043f
    return -1;
Packit fc043f
}
Packit fc043f
Packit fc043f
Packit fc043f
static int
Packit fc043f
sbcs_mblen(csconv_t *cv, const uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
  (void)cv;
Packit fc043f
  (void)buf;
Packit fc043f
  (void)bufsize;
Packit fc043f
  return 1;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
dbcs_mblen(csconv_t *cv, const uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    int len = IsDBCSLeadByteEx(cv->codepage, buf[0]) ? 2 : 1;
Packit fc043f
    if (bufsize < len)
Packit fc043f
        return seterror(EINVAL);
Packit fc043f
    return len;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
mbcs_mblen(csconv_t *cv, const uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    int len = 0;
Packit fc043f
Packit fc043f
    if (cv->codepage == 54936) {
Packit fc043f
	if (buf[0] <= 0x7F) len = 1;
Packit fc043f
	else if (buf[0] >= 0x81 && buf[0] <= 0xFE &&
Packit fc043f
		 bufsize >= 2 &&
Packit fc043f
		 ((buf[1] >= 0x40 && buf[1] <= 0x7E) ||
Packit fc043f
		  (buf[1] >= 0x80 && buf[1] <= 0xFE))) len = 2;
Packit fc043f
	else if (buf[0] >= 0x81 && buf[0] <= 0xFE &&
Packit fc043f
		 bufsize >= 4 &&
Packit fc043f
		 buf[1] >= 0x30 && buf[1] <= 0x39) len = 4;
Packit fc043f
	else
Packit fc043f
	    return seterror(EINVAL);
Packit fc043f
	return len;
Packit fc043f
    }
Packit fc043f
    else
Packit fc043f
	return seterror(EINVAL);
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
utf8_mblen(csconv_t *cv, const uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    int len = 0;
Packit fc043f
Packit fc043f
    (void) cv;
Packit fc043f
Packit fc043f
    if (buf[0] < 0x80) len = 1;
Packit fc043f
    else if ((buf[0] & 0xE0) == 0xC0) len = 2;
Packit fc043f
    else if ((buf[0] & 0xF0) == 0xE0) len = 3;
Packit fc043f
    else if ((buf[0] & 0xF8) == 0xF0) len = 4;
Packit fc043f
    else if ((buf[0] & 0xFC) == 0xF8) len = 5;
Packit fc043f
    else if ((buf[0] & 0xFE) == 0xFC) len = 6;
Packit fc043f
Packit fc043f
    if (len == 0)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    else if (bufsize < len)
Packit fc043f
        return seterror(EINVAL);
Packit fc043f
    return len;
Packit fc043f
}
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int
Packit fc043f
eucjp_mblen(csconv_t *cv, const uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    (void) cv;
Packit fc043f
Packit fc043f
    if (buf[0] < 0x80) /* ASCII */
Packit fc043f
        return 1;
Packit fc043f
    else if (buf[0] == 0x8E) /* JIS X 0201 */
Packit fc043f
    {
Packit fc043f
        if (bufsize < 2)
Packit fc043f
            return seterror(EINVAL);
Packit fc043f
        else if (!(0xA1 <= buf[1] && buf[1] <= 0xDF))
Packit fc043f
            return seterror(EILSEQ);
Packit fc043f
        return 2;
Packit fc043f
    }
Packit fc043f
    else if (buf[0] == 0x8F) /* JIS X 0212 */
Packit fc043f
    {
Packit fc043f
        if (bufsize < 3)
Packit fc043f
            return seterror(EINVAL);
Packit fc043f
        else if (!(0xA1 <= buf[1] && buf[1] <= 0xFE)
Packit fc043f
                || !(0xA1 <= buf[2] && buf[2] <= 0xFE))
Packit fc043f
            return seterror(EILSEQ);
Packit fc043f
        return 3;
Packit fc043f
    }
Packit fc043f
    else /* JIS X 0208 */
Packit fc043f
    {
Packit fc043f
        if (bufsize < 2)
Packit fc043f
            return seterror(EINVAL);
Packit fc043f
        else if (!(0xA1 <= buf[0] && buf[0] <= 0xFE)
Packit fc043f
                || !(0xA1 <= buf[1] && buf[1] <= 0xFE))
Packit fc043f
            return seterror(EILSEQ);
Packit fc043f
        return 2;
Packit fc043f
    }
Packit fc043f
}
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
static int
Packit fc043f
kernel_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize)
Packit fc043f
{
Packit fc043f
    int len;
Packit fc043f
Packit fc043f
    len = cv->mblen(cv, buf, bufsize);
Packit fc043f
    if (len == -1)
Packit fc043f
        return -1;
Packit fc043f
    /* If converting from ASCII, reject 8bit
Packit fc043f
     * chars. MultiByteToWideChar() doesn't. Note that for ASCII we
Packit fc043f
     * know that the mblen function is sbcs_mblen() so len is 1.
Packit fc043f
     */
Packit fc043f
    if (cv->codepage == 20127 && buf[0] >= 0x80)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    *wbufsize = MultiByteToWideChar(cv->codepage, mbtowc_flags (cv->codepage),
Packit fc043f
            (const char *)buf, len, (wchar_t *)wbuf, *wbufsize);
Packit fc043f
    if (*wbufsize == 0)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    return len;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
kernel_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    BOOL usedDefaultChar = 0;
Packit fc043f
    BOOL *p = NULL;
Packit fc043f
    int flags = 0;
Packit fc043f
    int len;
Packit fc043f
Packit fc043f
    if (bufsize == 0)
Packit fc043f
        return seterror(E2BIG);
Packit fc043f
    if (!must_use_null_useddefaultchar(cv->codepage))
Packit fc043f
    {
Packit fc043f
        p = &usedDefaultChar;
Packit fc043f
#ifdef WC_NO_BEST_FIT_CHARS
Packit fc043f
        if (!(cv->flags & FLAG_TRANSLIT))
Packit fc043f
            flags |= WC_NO_BEST_FIT_CHARS;
Packit fc043f
#endif
Packit fc043f
    }
Packit fc043f
    len = WideCharToMultiByte(cv->codepage, flags,
Packit fc043f
            (const wchar_t *)wbuf, wbufsize, (char *)buf, bufsize, NULL, p);
Packit fc043f
    if (len == 0)
Packit fc043f
    {
Packit fc043f
        if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
Packit fc043f
            return seterror(E2BIG);
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    }
Packit fc043f
    else if (usedDefaultChar && !(cv->flags & FLAG_TRANSLIT))
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    else if (cv->mblen(cv, buf, len) != len) /* validate result */
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    return len;
Packit fc043f
}
Packit fc043f
Packit fc043f
/*
Packit fc043f
 * It seems that the mode (cv->mode) is fixnum.
Packit fc043f
 * For example, when converting iso-2022-jp(cp50221) to unicode:
Packit fc043f
 *      in ascii sequence: mode=0xC42C0000
Packit fc043f
 *   in jisx0208 sequence: mode=0xC42C0001
Packit fc043f
 * "C42C" is same for each convert session.
Packit fc043f
 * It should be: ((codepage-1)<<16)|state
Packit fc043f
 */
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int
Packit fc043f
mlang_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize)
Packit fc043f
{
Packit fc043f
    int len;
Packit fc043f
    int insize;
Packit fc043f
    HRESULT hr;
Packit fc043f
Packit fc043f
    len = cv->mblen(cv, buf, bufsize);
Packit fc043f
    if (len == -1)
Packit fc043f
        return -1;
Packit fc043f
    insize = len;
Packit fc043f
    hr = ConvertINetMultiByteToUnicode(&cv->mode, cv->codepage,
Packit fc043f
            (const char *)buf, &insize, (wchar_t *)wbuf, wbufsize);
Packit fc043f
    if (hr != S_OK || insize != len)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    return len;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
mlang_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    char tmpbuf[MB_CHAR_MAX]; /* enough room for one character */
Packit fc043f
    int tmpsize = MB_CHAR_MAX;
Packit fc043f
    int insize = wbufsize;
Packit fc043f
    HRESULT hr;
Packit fc043f
Packit fc043f
    hr = ConvertINetUnicodeToMultiByte(&cv->mode, cv->codepage,
Packit fc043f
            (const wchar_t *)wbuf, &wbufsize, tmpbuf, &tmpsize);
Packit fc043f
    if (hr != S_OK || insize != wbufsize)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    else if (bufsize < tmpsize)
Packit fc043f
        return seterror(E2BIG);
Packit fc043f
    else if (cv->mblen(cv, (uchar *)tmpbuf, tmpsize) != tmpsize)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    memcpy(buf, tmpbuf, tmpsize);
Packit fc043f
    return tmpsize;
Packit fc043f
}
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
static int
Packit fc043f
utf16_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize)
Packit fc043f
{
Packit fc043f
    int codepage = cv->codepage;
Packit fc043f
Packit fc043f
    /* swap endian: 1200 <-> 1201 */
Packit fc043f
    if (cv->mode & UNICODE_MODE_SWAPPED)
Packit fc043f
        codepage ^= 1;
Packit fc043f
Packit fc043f
    if (bufsize < 2)
Packit fc043f
        return seterror(EINVAL);
Packit fc043f
    if (codepage == 1200) /* little endian */
Packit fc043f
        wbuf[0] = (buf[1] << 8) | buf[0];
Packit fc043f
    else if (codepage == 1201) /* big endian */
Packit fc043f
        wbuf[0] = (buf[0] << 8) | buf[1];
Packit fc043f
Packit fc043f
    if ((cv->flags & FLAG_USE_BOM) && !(cv->mode & UNICODE_MODE_BOM_DONE))
Packit fc043f
    {
Packit fc043f
        cv->mode |= UNICODE_MODE_BOM_DONE;
Packit fc043f
        if (wbuf[0] == 0xFFFE)
Packit fc043f
        {
Packit fc043f
            cv->mode |= UNICODE_MODE_SWAPPED;
Packit fc043f
            *wbufsize = 0;
Packit fc043f
            return 2;
Packit fc043f
        }
Packit fc043f
        else if (wbuf[0] == 0xFEFF)
Packit fc043f
        {
Packit fc043f
            *wbufsize = 0;
Packit fc043f
            return 2;
Packit fc043f
        }
Packit fc043f
    }
Packit fc043f
Packit fc043f
    if (0xDC00 <= wbuf[0] && wbuf[0] <= 0xDFFF)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    if (0xD800 <= wbuf[0] && wbuf[0] <= 0xDBFF)
Packit fc043f
    {
Packit fc043f
        if (bufsize < 4)
Packit fc043f
            return seterror(EINVAL);
Packit fc043f
        if (codepage == 1200) /* little endian */
Packit fc043f
            wbuf[1] = (buf[3] << 8) | buf[2];
Packit fc043f
        else if (codepage == 1201) /* big endian */
Packit fc043f
            wbuf[1] = (buf[2] << 8) | buf[3];
Packit fc043f
        if (!(0xDC00 <= wbuf[1] && wbuf[1] <= 0xDFFF))
Packit fc043f
            return seterror(EILSEQ);
Packit fc043f
        *wbufsize = 2;
Packit fc043f
        return 4;
Packit fc043f
    }
Packit fc043f
    *wbufsize = 1;
Packit fc043f
    return 2;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
utf16_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    if ((cv->flags & FLAG_USE_BOM) && !(cv->mode & UNICODE_MODE_BOM_DONE))
Packit fc043f
    {
Packit fc043f
        int r;
Packit fc043f
Packit fc043f
        cv->mode |= UNICODE_MODE_BOM_DONE;
Packit fc043f
        if (bufsize < 2)
Packit fc043f
            return seterror(E2BIG);
Packit fc043f
        if (cv->codepage == 1200) /* little endian */
Packit fc043f
            memcpy(buf, "\xFF\xFE", 2);
Packit fc043f
        else if (cv->codepage == 1201) /* big endian */
Packit fc043f
            memcpy(buf, "\xFE\xFF", 2);
Packit fc043f
Packit fc043f
        r = utf16_wctomb(cv, wbuf, wbufsize, buf + 2, bufsize - 2);
Packit fc043f
        if (r == -1)
Packit fc043f
            return -1;
Packit fc043f
        return r + 2;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    if (bufsize < 2)
Packit fc043f
        return seterror(E2BIG);
Packit fc043f
    if (cv->codepage == 1200) /* little endian */
Packit fc043f
    {
Packit fc043f
        buf[0] = (wbuf[0] & 0x00FF);
Packit fc043f
        buf[1] = (wbuf[0] & 0xFF00) >> 8;
Packit fc043f
    }
Packit fc043f
    else if (cv->codepage == 1201) /* big endian */
Packit fc043f
    {
Packit fc043f
        buf[0] = (wbuf[0] & 0xFF00) >> 8;
Packit fc043f
        buf[1] = (wbuf[0] & 0x00FF);
Packit fc043f
    }
Packit fc043f
    if (0xD800 <= wbuf[0] && wbuf[0] <= 0xDBFF)
Packit fc043f
    {
Packit fc043f
        if (bufsize < 4)
Packit fc043f
            return seterror(E2BIG);
Packit fc043f
        if (cv->codepage == 1200) /* little endian */
Packit fc043f
        {
Packit fc043f
            buf[2] = (wbuf[1] & 0x00FF);
Packit fc043f
            buf[3] = (wbuf[1] & 0xFF00) >> 8;
Packit fc043f
        }
Packit fc043f
        else if (cv->codepage == 1201) /* big endian */
Packit fc043f
        {
Packit fc043f
            buf[2] = (wbuf[1] & 0xFF00) >> 8;
Packit fc043f
            buf[3] = (wbuf[1] & 0x00FF);
Packit fc043f
        }
Packit fc043f
        return 4;
Packit fc043f
    }
Packit fc043f
    return 2;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
utf32_mbtowc(csconv_t *cv, const uchar *buf, int bufsize, ushort *wbuf, int *wbufsize)
Packit fc043f
{
Packit fc043f
    int codepage = cv->codepage;
Packit fc043f
    uint wc = 0xD800;
Packit fc043f
Packit fc043f
    /* swap endian: 12000 <-> 12001 */
Packit fc043f
    if (cv->mode & UNICODE_MODE_SWAPPED)
Packit fc043f
        codepage ^= 1;
Packit fc043f
Packit fc043f
    if (bufsize < 4)
Packit fc043f
        return seterror(EINVAL);
Packit fc043f
    if (codepage == 12000) /* little endian */
Packit fc043f
        wc = (buf[3] << 24) | (buf[2] << 16) | (buf[1] << 8) | buf[0];
Packit fc043f
    else if (codepage == 12001) /* big endian */
Packit fc043f
        wc = (buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3];
Packit fc043f
Packit fc043f
    if ((cv->flags & FLAG_USE_BOM) && !(cv->mode & UNICODE_MODE_BOM_DONE))
Packit fc043f
    {
Packit fc043f
        cv->mode |= UNICODE_MODE_BOM_DONE;
Packit fc043f
        if (wc == 0xFFFE0000)
Packit fc043f
        {
Packit fc043f
            cv->mode |= UNICODE_MODE_SWAPPED;
Packit fc043f
            *wbufsize = 0;
Packit fc043f
            return 4;
Packit fc043f
        }
Packit fc043f
        else if (wc == 0x0000FEFF)
Packit fc043f
        {
Packit fc043f
            *wbufsize = 0;
Packit fc043f
            return 4;
Packit fc043f
        }
Packit fc043f
    }
Packit fc043f
Packit fc043f
    if ((0xD800 <= wc && wc <= 0xDFFF) || 0x10FFFF < wc)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    ucs4_to_utf16(wc, wbuf, wbufsize);
Packit fc043f
    return 4;
Packit fc043f
}
Packit fc043f
Packit fc043f
static int
Packit fc043f
utf32_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    uint wc;
Packit fc043f
Packit fc043f
    if ((cv->flags & FLAG_USE_BOM) && !(cv->mode & UNICODE_MODE_BOM_DONE))
Packit fc043f
    {
Packit fc043f
        int r;
Packit fc043f
Packit fc043f
        cv->mode |= UNICODE_MODE_BOM_DONE;
Packit fc043f
        if (bufsize < 4)
Packit fc043f
            return seterror(E2BIG);
Packit fc043f
        if (cv->codepage == 12000) /* little endian */
Packit fc043f
            memcpy(buf, "\xFF\xFE\x00\x00", 4);
Packit fc043f
        else if (cv->codepage == 12001) /* big endian */
Packit fc043f
            memcpy(buf, "\x00\x00\xFE\xFF", 4);
Packit fc043f
Packit fc043f
        r = utf32_wctomb(cv, wbuf, wbufsize, buf + 4, bufsize - 4);
Packit fc043f
        if (r == -1)
Packit fc043f
            return -1;
Packit fc043f
        return r + 4;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    if (bufsize < 4)
Packit fc043f
        return seterror(E2BIG);
Packit fc043f
    wc = utf16_to_ucs4(wbuf);
Packit fc043f
    if (cv->codepage == 12000) /* little endian */
Packit fc043f
    {
Packit fc043f
        buf[0] = wc & 0x000000FF;
Packit fc043f
        buf[1] = (wc & 0x0000FF00) >> 8;
Packit fc043f
        buf[2] = (wc & 0x00FF0000) >> 16;
Packit fc043f
        buf[3] = (wc & 0xFF000000) >> 24;
Packit fc043f
    }
Packit fc043f
    else if (cv->codepage == 12001) /* big endian */
Packit fc043f
    {
Packit fc043f
        buf[0] = (wc & 0xFF000000) >> 24;
Packit fc043f
        buf[1] = (wc & 0x00FF0000) >> 16;
Packit fc043f
        buf[2] = (wc & 0x0000FF00) >> 8;
Packit fc043f
        buf[3] = wc & 0x000000FF;
Packit fc043f
    }
Packit fc043f
    return 4;
Packit fc043f
}
Packit fc043f
Packit fc043f
/*
Packit fc043f
 * 50220: ISO 2022 Japanese with no halfwidth Katakana; Japanese (JIS)
Packit fc043f
 * 50221: ISO 2022 Japanese with halfwidth Katakana; Japanese (JIS-Allow
Packit fc043f
 *        1 byte Kana)
Packit fc043f
 * 50222: ISO 2022 Japanese JIS X 0201-1989; Japanese (JIS-Allow 1 byte
Packit fc043f
 *        Kana - SO/SI)
Packit fc043f
 *
Packit fc043f
 * MultiByteToWideChar() and WideCharToMultiByte() behave differently
Packit fc043f
 * depending on Windows version.  On XP, WideCharToMultiByte() doesn't
Packit fc043f
 * terminate result sequence with ascii escape.  But Vista does.
Packit fc043f
 * Use MLang instead.
Packit fc043f
 */
Packit fc043f
Packit fc043f
#define ISO2022_MODE(cs, shift) (((cs) << 8) | (shift))
Packit fc043f
#define ISO2022_MODE_CS(mode) (((mode) >> 8) & 0xFF)
Packit fc043f
#define ISO2022_MODE_SHIFT(mode) ((mode) & 0xFF)
Packit fc043f
Packit fc043f
#define ISO2022_SI  0
Packit fc043f
#define ISO2022_SO  1
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
/* shift in */
Packit fc043f
static const char iso2022_SI_seq[] = "\x0F";
Packit fc043f
/* shift out */
Packit fc043f
static const char iso2022_SO_seq[] = "\x0E";
Packit fc043f
Packit fc043f
typedef struct iso2022_esc_t iso2022_esc_t;
Packit fc043f
struct iso2022_esc_t {
Packit fc043f
    const char *esc;
Packit fc043f
    int esc_len;
Packit fc043f
    int len;
Packit fc043f
    int cs;
Packit fc043f
};
Packit fc043f
#endif
Packit fc043f
Packit fc043f
#define ISO2022JP_CS_ASCII            0
Packit fc043f
#define ISO2022JP_CS_JISX0201_ROMAN   1
Packit fc043f
#define ISO2022JP_CS_JISX0201_KANA    2
Packit fc043f
#define ISO2022JP_CS_JISX0208_1978    3
Packit fc043f
#define ISO2022JP_CS_JISX0208_1983    4
Packit fc043f
#define ISO2022JP_CS_JISX0212         5
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static iso2022_esc_t iso2022jp_esc[] = {
Packit fc043f
    {"\x1B\x28\x42", 3, 1, ISO2022JP_CS_ASCII},
Packit fc043f
    {"\x1B\x28\x4A", 3, 1, ISO2022JP_CS_JISX0201_ROMAN},
Packit fc043f
    {"\x1B\x28\x49", 3, 1, ISO2022JP_CS_JISX0201_KANA},
Packit fc043f
    {"\x1B\x24\x40", 3, 2, ISO2022JP_CS_JISX0208_1983}, /* unify 1978 with 1983 */
Packit fc043f
    {"\x1B\x24\x42", 3, 2, ISO2022JP_CS_JISX0208_1983},
Packit fc043f
    {"\x1B\x24\x28\x44", 4, 2, ISO2022JP_CS_JISX0212},
Packit fc043f
    {NULL, 0, 0, 0}
Packit fc043f
};
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int
Packit fc043f
iso2022jp_mbtowc(csconv_t *cv, const uchar *buf, int bufsize,
Packit fc043f
                 ushort *wbuf, int *wbufsize)
Packit fc043f
{
Packit fc043f
    iso2022_esc_t *iesc = iso2022jp_esc;
Packit fc043f
    char tmp[MB_CHAR_MAX];
Packit fc043f
    int insize;
Packit fc043f
    HRESULT hr;
Packit fc043f
    DWORD dummy = 0;
Packit fc043f
    int len;
Packit fc043f
    int esc_len;
Packit fc043f
    int cs;
Packit fc043f
    int shift;
Packit fc043f
    int i;
Packit fc043f
Packit fc043f
    if (buf[0] == 0x1B)
Packit fc043f
    {
Packit fc043f
        for (i = 0; iesc[i].esc != NULL; ++i)
Packit fc043f
        {
Packit fc043f
            esc_len = iesc[i].esc_len;
Packit fc043f
            if (bufsize < esc_len)
Packit fc043f
            {
Packit fc043f
                if (strncmp((char *)buf, iesc[i].esc, bufsize) == 0)
Packit fc043f
                    return seterror(EINVAL);
Packit fc043f
            }
Packit fc043f
            else
Packit fc043f
            {
Packit fc043f
                if (strncmp((char *)buf, iesc[i].esc, esc_len) == 0)
Packit fc043f
                {
Packit fc043f
                    cv->mode = ISO2022_MODE(iesc[i].cs, ISO2022_SI);
Packit fc043f
                    *wbufsize = 0;
Packit fc043f
                    return esc_len;
Packit fc043f
                }
Packit fc043f
            }
Packit fc043f
        }
Packit fc043f
        /* not supported escape sequence */
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    }
Packit fc043f
    else if (buf[0] == iso2022_SO_seq[0])
Packit fc043f
    {
Packit fc043f
        cv->mode = ISO2022_MODE(ISO2022_MODE_CS(cv->mode), ISO2022_SO);
Packit fc043f
        *wbufsize = 0;
Packit fc043f
        return 1;
Packit fc043f
    }
Packit fc043f
    else if (buf[0] == iso2022_SI_seq[0])
Packit fc043f
    {
Packit fc043f
        cv->mode = ISO2022_MODE(ISO2022_MODE_CS(cv->mode), ISO2022_SI);
Packit fc043f
        *wbufsize = 0;
Packit fc043f
        return 1;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    cs = ISO2022_MODE_CS(cv->mode);
Packit fc043f
    shift = ISO2022_MODE_SHIFT(cv->mode);
Packit fc043f
Packit fc043f
    /* reset the mode for informal sequence */
Packit fc043f
    if (buf[0] < 0x20)
Packit fc043f
    {
Packit fc043f
        cs = ISO2022JP_CS_ASCII;
Packit fc043f
        shift = ISO2022_SI;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    len = iesc[cs].len;
Packit fc043f
    if (bufsize < len)
Packit fc043f
        return seterror(EINVAL);
Packit fc043f
    for (i = 0; i < len; ++i)
Packit fc043f
        if (!(buf[i] < 0x80))
Packit fc043f
            return seterror(EILSEQ);
Packit fc043f
    esc_len = iesc[cs].esc_len;
Packit fc043f
    memcpy(tmp, iesc[cs].esc, esc_len);
Packit fc043f
    if (shift == ISO2022_SO)
Packit fc043f
    {
Packit fc043f
        memcpy(tmp + esc_len, iso2022_SO_seq, 1);
Packit fc043f
        esc_len += 1;
Packit fc043f
    }
Packit fc043f
    memcpy(tmp + esc_len, buf, len);
Packit fc043f
Packit fc043f
    if ((cv->codepage == 50220 || cv->codepage == 50221
Packit fc043f
                || cv->codepage == 50222) && shift == ISO2022_SO)
Packit fc043f
    {
Packit fc043f
        /* XXX: shift-out cannot be used for mbtowc (both kernel and
Packit fc043f
         * mlang) */
Packit fc043f
        esc_len = iesc[ISO2022JP_CS_JISX0201_KANA].esc_len;
Packit fc043f
        memcpy(tmp, iesc[ISO2022JP_CS_JISX0201_KANA].esc, esc_len);
Packit fc043f
        memcpy(tmp + esc_len, buf, len);
Packit fc043f
    }
Packit fc043f
Packit fc043f
    insize = len + esc_len;
Packit fc043f
    hr = ConvertINetMultiByteToUnicode(&dummy, cv->codepage,
Packit fc043f
            (const char *)tmp, &insize, (wchar_t *)wbuf, wbufsize);
Packit fc043f
    if (hr != S_OK || insize != len + esc_len)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
Packit fc043f
    /* Check for conversion error.  Assuming defaultChar is 0x3F. */
Packit fc043f
    /* ascii should be converted from ascii */
Packit fc043f
    if (wbuf[0] == buf[0]
Packit fc043f
            && cv->mode != ISO2022_MODE(ISO2022JP_CS_ASCII, ISO2022_SI))
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
Packit fc043f
    /* reset the mode for informal sequence */
Packit fc043f
    if (cv->mode != ISO2022_MODE(cs, shift))
Packit fc043f
        cv->mode = ISO2022_MODE(cs, shift);
Packit fc043f
Packit fc043f
    return len;
Packit fc043f
}
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int
Packit fc043f
iso2022jp_wctomb(csconv_t *cv, ushort *wbuf, int wbufsize, uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    iso2022_esc_t *iesc = iso2022jp_esc;
Packit fc043f
    char tmp[MB_CHAR_MAX];
Packit fc043f
    int tmpsize = MB_CHAR_MAX;
Packit fc043f
    int insize = wbufsize;
Packit fc043f
    HRESULT hr;
Packit fc043f
    DWORD dummy = 0;
Packit fc043f
    int len;
Packit fc043f
    int esc_len;
Packit fc043f
    int cs;
Packit fc043f
    int shift;
Packit fc043f
    int i;
Packit fc043f
Packit fc043f
    /*
Packit fc043f
     * MultiByte = [escape sequence] + character + [escape sequence]
Packit fc043f
     *
Packit fc043f
     * Whether trailing escape sequence is added depends on which API is
Packit fc043f
     * used (kernel or MLang, and its version).
Packit fc043f
     */
Packit fc043f
    hr = ConvertINetUnicodeToMultiByte(&dummy, cv->codepage,
Packit fc043f
            (const wchar_t *)wbuf, &wbufsize, tmp, &tmpsize);
Packit fc043f
    if (hr != S_OK || insize != wbufsize)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    else if (bufsize < tmpsize)
Packit fc043f
        return seterror(E2BIG);
Packit fc043f
Packit fc043f
    if (tmpsize == 1)
Packit fc043f
    {
Packit fc043f
        cs = ISO2022JP_CS_ASCII;
Packit fc043f
        esc_len = 0;
Packit fc043f
    }
Packit fc043f
    else
Packit fc043f
    {
Packit fc043f
        for (i = 1; iesc[i].esc != NULL; ++i)
Packit fc043f
        {
Packit fc043f
            esc_len = iesc[i].esc_len;
Packit fc043f
            if (strncmp(tmp, iesc[i].esc, esc_len) == 0)
Packit fc043f
            {
Packit fc043f
                cs = iesc[i].cs;
Packit fc043f
                break;
Packit fc043f
            }
Packit fc043f
        }
Packit fc043f
        if (iesc[i].esc == NULL)
Packit fc043f
            /* not supported escape sequence */
Packit fc043f
            return seterror(EILSEQ);
Packit fc043f
    }
Packit fc043f
Packit fc043f
    shift = ISO2022_SI;
Packit fc043f
    if (tmp[esc_len] == iso2022_SO_seq[0])
Packit fc043f
    {
Packit fc043f
        shift = ISO2022_SO;
Packit fc043f
        esc_len += 1;
Packit fc043f
    }
Packit fc043f
Packit fc043f
    len = iesc[cs].len;
Packit fc043f
Packit fc043f
    /* Check for converting error.  Assuming defaultChar is 0x3F. */
Packit fc043f
    /* ascii should be converted from ascii */
Packit fc043f
    if (cs == ISO2022JP_CS_ASCII && !(wbuf[0] < 0x80))
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
    else if (tmpsize < esc_len + len)
Packit fc043f
        return seterror(EILSEQ);
Packit fc043f
Packit fc043f
    if (cv->mode == ISO2022_MODE(cs, shift))
Packit fc043f
    {
Packit fc043f
        /* remove escape sequence */
Packit fc043f
        if (esc_len != 0)
Packit fc043f
            memmove(tmp, tmp + esc_len, len);
Packit fc043f
        esc_len = 0;
Packit fc043f
    }
Packit fc043f
    else
Packit fc043f
    {
Packit fc043f
        if (cs == ISO2022JP_CS_ASCII)
Packit fc043f
        {
Packit fc043f
            esc_len = iesc[ISO2022JP_CS_ASCII].esc_len;
Packit fc043f
            memmove(tmp + esc_len, tmp, len);
Packit fc043f
            memcpy(tmp, iesc[ISO2022JP_CS_ASCII].esc, esc_len);
Packit fc043f
        }
Packit fc043f
        if (ISO2022_MODE_SHIFT(cv->mode) == ISO2022_SO)
Packit fc043f
        {
Packit fc043f
            /* shift-in before changing to other mode */
Packit fc043f
            memmove(tmp + 1, tmp, len + esc_len);
Packit fc043f
            memcpy(tmp, iso2022_SI_seq, 1);
Packit fc043f
            esc_len += 1;
Packit fc043f
        }
Packit fc043f
    }
Packit fc043f
Packit fc043f
    if (bufsize < len + esc_len)
Packit fc043f
        return seterror(E2BIG);
Packit fc043f
    memcpy(buf, tmp, len + esc_len);
Packit fc043f
    cv->mode = ISO2022_MODE(cs, shift);
Packit fc043f
    return len + esc_len;
Packit fc043f
}
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
#if USE_MLANG_DLL
Packit fc043f
static int
Packit fc043f
iso2022jp_flush(csconv_t *cv, uchar *buf, int bufsize)
Packit fc043f
{
Packit fc043f
    iso2022_esc_t *iesc = iso2022jp_esc;
Packit fc043f
    int esc_len;
Packit fc043f
Packit fc043f
    if (cv->mode != ISO2022_MODE(ISO2022JP_CS_ASCII, ISO2022_SI))
Packit fc043f
    {
Packit fc043f
        esc_len = 0;
Packit fc043f
        if (ISO2022_MODE_SHIFT(cv->mode) != ISO2022_SI)
Packit fc043f
            esc_len += 1;
Packit fc043f
        if (ISO2022_MODE_CS(cv->mode) != ISO2022JP_CS_ASCII)
Packit fc043f
            esc_len += iesc[ISO2022JP_CS_ASCII].esc_len;
Packit fc043f
        if (bufsize < esc_len)
Packit fc043f
            return seterror(E2BIG);
Packit fc043f
Packit fc043f
        esc_len = 0;
Packit fc043f
        if (ISO2022_MODE_SHIFT(cv->mode) != ISO2022_SI)
Packit fc043f
        {
Packit fc043f
            memcpy(buf, iso2022_SI_seq, 1);
Packit fc043f
            esc_len += 1;
Packit fc043f
        }
Packit fc043f
        if (ISO2022_MODE_CS(cv->mode) != ISO2022JP_CS_ASCII)
Packit fc043f
        {
Packit fc043f
            memcpy(buf + esc_len, iesc[ISO2022JP_CS_ASCII].esc,
Packit fc043f
                    iesc[ISO2022JP_CS_ASCII].esc_len);
Packit fc043f
            esc_len += iesc[ISO2022JP_CS_ASCII].esc_len;
Packit fc043f
        }
Packit fc043f
        return esc_len;
Packit fc043f
    }
Packit fc043f
    return 0;
Packit fc043f
}
Packit fc043f
#endif /*USE_MLANG_DLL*/
Packit fc043f
Packit fc043f
Packit fc043f
gpgrt_w32_iconv_t
Packit fc043f
gpgrt_w32_iconv_open (const char *tocode, const char *fromcode)
Packit fc043f
{
Packit fc043f
    gpgrt_w32_iconv_t cd;
Packit fc043f
Packit fc043f
    cd = calloc(1, sizeof *cd);
Packit fc043f
    if (!cd)
Packit fc043f
         return (gpgrt_w32_iconv_t)(-1);
Packit fc043f
Packit fc043f
    /* reset the errno to prevent reporting wrong error code.
Packit fc043f
     * 0 for unsorted error. */
Packit fc043f
    _gpg_err_set_errno (0);
Packit fc043f
    if (win_iconv_open(cd, tocode, fromcode))
Packit fc043f
        return cd;
Packit fc043f
Packit fc043f
    free(cd);
Packit fc043f
Packit fc043f
    return (gpgrt_w32_iconv_t)(-1);
Packit fc043f
}
Packit fc043f
Packit fc043f
int
Packit fc043f
gpgrt_w32_iconv_close (gpgrt_w32_iconv_t cd)
Packit fc043f
{
Packit fc043f
    if (cd)
Packit fc043f
    {
Packit fc043f
         free (cd);
Packit fc043f
    }
Packit fc043f
Packit fc043f
    return 0;
Packit fc043f
}
Packit fc043f
Packit fc043f
size_t
Packit fc043f
gpgrt_w32_iconv (gpgrt_w32_iconv_t cd,
Packit fc043f
                  const char **inbuf, size_t *inbytesleft,
Packit fc043f
                  char **outbuf, size_t *outbytesleft)
Packit fc043f
{
Packit fc043f
     size_t r;
Packit fc043f
Packit fc043f
     r = win_iconv (cd, inbuf, inbytesleft, outbuf, outbytesleft);
Packit fc043f
     _gpg_err_set_errno (*(cd->_errno()));
Packit fc043f
     return r;
Packit fc043f
}