Convert the Big5, EUC, MSKanji and UTF-8 encoding methods to implement
mbrtowc() and wcrtomb() directly. GB18030, GBK and UTF2 are left unconverted; GB18030 will be done eventually, but GBK and UTF2 may just be removed, as they are subsets of GB18030 and UTF-8 respectively.
This commit is contained in:
parent
fcca8c1dde
commit
02f4f60ad5
@ -1,4 +1,5 @@
|
||||
/*-
|
||||
* Copyright (c) 2002, 2003 Tim J. Robbins. All rights reserved.
|
||||
* Copyright (c) 1993
|
||||
* The Regents of the University of California. All rights reserved.
|
||||
*
|
||||
@ -40,80 +41,76 @@ static char sccsid[] = "@(#)big5.c 8.1 (Berkeley) 6/4/93";
|
||||
#include <sys/cdefs.h>
|
||||
__FBSDID("$FreeBSD$");
|
||||
|
||||
#include <rune.h>
|
||||
#include <sys/types.h>
|
||||
#include <runetype.h>
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <sys/types.h>
|
||||
#include <wchar.h>
|
||||
|
||||
rune_t _BIG5_sgetrune(const char *, size_t, char const **);
|
||||
int _BIG5_sputrune(rune_t, char *, size_t, char **);
|
||||
extern size_t (*__mbrtowc)(wchar_t * __restrict, const char * __restrict,
|
||||
size_t, mbstate_t * __restrict);
|
||||
extern size_t (*__wcrtomb)(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
int _BIG5_init(_RuneLocale *);
|
||||
size_t _BIG5_mbrtowc(wchar_t * __restrict, const char * __restrict, size_t,
|
||||
mbstate_t * __restrict);
|
||||
size_t _BIG5_wcrtomb(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
int
|
||||
_BIG5_init(rl)
|
||||
_RuneLocale *rl;
|
||||
_BIG5_init(_RuneLocale *rl)
|
||||
{
|
||||
rl->sgetrune = _BIG5_sgetrune;
|
||||
rl->sputrune = _BIG5_sputrune;
|
||||
|
||||
__mbrtowc = _BIG5_mbrtowc;
|
||||
__wcrtomb = _BIG5_wcrtomb;
|
||||
_CurrentRuneLocale = rl;
|
||||
__mb_cur_max = 2;
|
||||
return (0);
|
||||
}
|
||||
|
||||
static inline int
|
||||
_big5_check(c)
|
||||
u_int c;
|
||||
static __inline int
|
||||
_big5_check(u_int c)
|
||||
{
|
||||
|
||||
c &= 0xff;
|
||||
return ((c >= 0xa1 && c <= 0xfe) ? 2 : 1);
|
||||
}
|
||||
|
||||
rune_t
|
||||
_BIG5_sgetrune(string, n, result)
|
||||
const char *string;
|
||||
size_t n;
|
||||
char const **result;
|
||||
size_t
|
||||
_BIG5_mbrtowc(wchar_t * __restrict pwc, const char * __restrict s, size_t n,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
rune_t rune = 0;
|
||||
int len;
|
||||
wchar_t wc;
|
||||
int i, len;
|
||||
|
||||
if (n < 1 || (len = _big5_check(*string)) > n) {
|
||||
if (result)
|
||||
*result = string;
|
||||
return (_INVALID_RUNE);
|
||||
}
|
||||
while (--len >= 0)
|
||||
rune = (rune << 8) | ((u_int)(*string++) & 0xff);
|
||||
if (result)
|
||||
*result = string;
|
||||
return rune;
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (0);
|
||||
if (n == 0 || (size_t)(len = _big5_check(*s)) > n)
|
||||
/* Incomplete multibyte sequence */
|
||||
return ((size_t)-2);
|
||||
wc = 0;
|
||||
i = len;
|
||||
while (i-- > 0)
|
||||
wc = (wc << 8) | (unsigned char)*s++;
|
||||
if (pwc != NULL)
|
||||
*pwc = wc;
|
||||
return (wc == L'\0' ? 0 : len);
|
||||
}
|
||||
|
||||
int
|
||||
_BIG5_sputrune(c, string, n, result)
|
||||
rune_t c;
|
||||
char *string, **result;
|
||||
size_t n;
|
||||
size_t
|
||||
_BIG5_wcrtomb(char * __restrict s, wchar_t wc,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
if (c & 0x8000) {
|
||||
if (n >= 2) {
|
||||
string[0] = (c >> 8) & 0xff;
|
||||
string[1] = c & 0xff;
|
||||
if (result)
|
||||
*result = string + 2;
|
||||
return (2);
|
||||
}
|
||||
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (1);
|
||||
if (wc & 0x8000) {
|
||||
*s++ = (wc >> 8) & 0xff;
|
||||
*s = wc & 0xff;
|
||||
return (2);
|
||||
}
|
||||
else {
|
||||
if (n >= 1) {
|
||||
*string = c & 0xff;
|
||||
if (result)
|
||||
*result = string + 1;
|
||||
return (1);
|
||||
}
|
||||
}
|
||||
if (result)
|
||||
*result = string;
|
||||
return (0);
|
||||
|
||||
*s = wc & 0xff;
|
||||
return (1);
|
||||
}
|
||||
|
@ -1,4 +1,5 @@
|
||||
/*-
|
||||
* Copyright (c) 2002, 2003 Tim J. Robbins. All rights reserved.
|
||||
* Copyright (c) 1993
|
||||
* The Regents of the University of California. All rights reserved.
|
||||
*
|
||||
@ -43,32 +44,35 @@ __FBSDID("$FreeBSD$");
|
||||
#include <sys/types.h>
|
||||
|
||||
#include <errno.h>
|
||||
#include <rune.h>
|
||||
#include <runetype.h>
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <wchar.h>
|
||||
|
||||
rune_t _EUC_sgetrune(const char *, size_t, char const **);
|
||||
int _EUC_sputrune(rune_t, char *, size_t, char **);
|
||||
extern size_t (*__mbrtowc)(wchar_t * __restrict, const char * __restrict,
|
||||
size_t, mbstate_t * __restrict);
|
||||
extern size_t (*__wcrtomb)(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
int _EUC_init(_RuneLocale *);
|
||||
size_t _EUC_mbrtowc(wchar_t * __restrict, const char * __restrict, size_t,
|
||||
mbstate_t * __restrict);
|
||||
size_t _EUC_wcrtomb(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
typedef struct {
|
||||
int count[4];
|
||||
rune_t bits[4];
|
||||
rune_t mask;
|
||||
wchar_t bits[4];
|
||||
wchar_t mask;
|
||||
} _EucInfo;
|
||||
|
||||
int
|
||||
_EUC_init(rl)
|
||||
_RuneLocale *rl;
|
||||
_EUC_init(_RuneLocale *rl)
|
||||
{
|
||||
_EucInfo *ei;
|
||||
int x, new__mb_cur_max;
|
||||
char *v, *e;
|
||||
|
||||
rl->sgetrune = _EUC_sgetrune;
|
||||
rl->sputrune = _EUC_sputrune;
|
||||
|
||||
if (rl->variable == NULL)
|
||||
return (EFTYPE);
|
||||
|
||||
@ -108,6 +112,8 @@ _EUC_init(rl)
|
||||
rl->variable_len = sizeof(_EucInfo);
|
||||
_CurrentRuneLocale = rl;
|
||||
__mb_cur_max = new__mb_cur_max;
|
||||
__mbrtowc = _EUC_mbrtowc;
|
||||
__wcrtomb = _EUC_wcrtomb;
|
||||
return (0);
|
||||
}
|
||||
|
||||
@ -118,105 +124,85 @@ _EUC_init(rl)
|
||||
|
||||
#define GR_BITS 0x80808080 /* XXX: to be fixed */
|
||||
|
||||
static inline int
|
||||
_euc_set(c)
|
||||
u_int c;
|
||||
static __inline int
|
||||
_euc_set(u_int c)
|
||||
{
|
||||
c &= 0xff;
|
||||
|
||||
return ((c & 0x80) ? c == _SS3 ? 3 : c == _SS2 ? 2 : 1 : 0);
|
||||
}
|
||||
rune_t
|
||||
_EUC_sgetrune(string, n, result)
|
||||
const char *string;
|
||||
size_t n;
|
||||
char const **result;
|
||||
{
|
||||
rune_t rune = 0;
|
||||
int len, set;
|
||||
|
||||
if (n < 1 || (len = CEI->count[set = _euc_set(*string)]) > n) {
|
||||
if (result)
|
||||
*result = string;
|
||||
return (_INVALID_RUNE);
|
||||
}
|
||||
size_t
|
||||
_EUC_mbrtowc(wchar_t * __restrict pwc, const char * __restrict s, size_t n,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
int len, remain, set;
|
||||
wchar_t wc;
|
||||
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (0);
|
||||
if (n == 0 || (size_t)(len = CEI->count[set = _euc_set(*s)]) > n)
|
||||
/* Incomplete multibyte sequence */
|
||||
return ((size_t)-2);
|
||||
wc = 0;
|
||||
remain = len;
|
||||
switch (set) {
|
||||
case 3:
|
||||
case 2:
|
||||
--len;
|
||||
++string;
|
||||
--remain;
|
||||
++s;
|
||||
/* FALLTHROUGH */
|
||||
case 1:
|
||||
case 0:
|
||||
while (len-- > 0)
|
||||
rune = (rune << 8) | ((u_int)(*string++) & 0xff);
|
||||
while (remain-- > 0)
|
||||
wc = (wc << 8) | (unsigned char)*s++;
|
||||
break;
|
||||
}
|
||||
if (result)
|
||||
*result = string;
|
||||
return ((rune & ~CEI->mask) | CEI->bits[set]);
|
||||
wc = (wc & ~CEI->mask) | CEI->bits[set];
|
||||
if (pwc != NULL)
|
||||
*pwc = wc;
|
||||
return (wc == L'\0' ? 0 : len);
|
||||
}
|
||||
|
||||
int
|
||||
_EUC_sputrune(c, string, n, result)
|
||||
rune_t c;
|
||||
char *string, **result;
|
||||
size_t n;
|
||||
size_t
|
||||
_EUC_wcrtomb(char * __restrict s, wchar_t wc,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
rune_t m = c & CEI->mask;
|
||||
rune_t nm = c & ~m;
|
||||
wchar_t m, nm;
|
||||
int i, len;
|
||||
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (1);
|
||||
|
||||
m = wc & CEI->mask;
|
||||
nm = wc & ~m;
|
||||
|
||||
if (m == CEI->bits[1]) {
|
||||
CodeSet1:
|
||||
/* Codeset 1: The first byte must have 0x80 in it. */
|
||||
i = len = CEI->count[1];
|
||||
if (n >= len) {
|
||||
if (result)
|
||||
*result = string + len;
|
||||
while (i-- > 0)
|
||||
*string++ = (nm >> (i << 3)) | 0x80;
|
||||
} else
|
||||
if (result)
|
||||
*result = (char *) 0;
|
||||
} else {
|
||||
if (m == CEI->bits[0]) {
|
||||
i = len = CEI->count[0];
|
||||
if (n < len) {
|
||||
if (result)
|
||||
*result = NULL;
|
||||
return (len);
|
||||
}
|
||||
} else
|
||||
if (m == CEI->bits[2]) {
|
||||
i = len = CEI->count[2];
|
||||
if (n < len) {
|
||||
if (result)
|
||||
*result = NULL;
|
||||
return (len);
|
||||
}
|
||||
*string++ = _SS2;
|
||||
--i;
|
||||
/* SS2 designates G2 into GR */
|
||||
nm |= GR_BITS;
|
||||
} else
|
||||
if (m == CEI->bits[3]) {
|
||||
i = len = CEI->count[3];
|
||||
if (n < len) {
|
||||
if (result)
|
||||
*result = NULL;
|
||||
return (len);
|
||||
}
|
||||
*string++ = _SS3;
|
||||
--i;
|
||||
/* SS3 designates G3 into GR */
|
||||
nm |= GR_BITS;
|
||||
} else
|
||||
goto CodeSet1; /* Bletch */
|
||||
while (i-- > 0)
|
||||
*string++ = (nm >> (i << 3)) & 0xff;
|
||||
if (result)
|
||||
*result = string;
|
||||
*s++ = (nm >> (i << 3)) | 0x80;
|
||||
} else {
|
||||
if (m == CEI->bits[0])
|
||||
i = len = CEI->count[0];
|
||||
else if (m == CEI->bits[2]) {
|
||||
i = len = CEI->count[2];
|
||||
*s++ = _SS2;
|
||||
--i;
|
||||
/* SS2 designates G2 into GR */
|
||||
nm |= GR_BITS;
|
||||
} else if (m == CEI->bits[3]) {
|
||||
i = len = CEI->count[3];
|
||||
*s++ = _SS3;
|
||||
--i;
|
||||
/* SS3 designates G3 into GR */
|
||||
nm |= GR_BITS;
|
||||
} else
|
||||
goto CodeSet1; /* Bletch */
|
||||
while (i-- > 0)
|
||||
*s++ = (nm >> (i << 3)) & 0xff;
|
||||
}
|
||||
return (len);
|
||||
}
|
||||
|
@ -1,4 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2002, 2003 Tim J. Robbins. All rights reserved.
|
||||
* ja_JP.SJIS locale table for BSD4.4/rune
|
||||
* version 1.0
|
||||
* (C) Sin'ichiro MIYATANI / Phase One, Inc
|
||||
@ -38,74 +39,71 @@ static char sccsid[] = "@(#)mskanji.c 1.0 (Phase One) 5/5/95";
|
||||
__FBSDID("$FreeBSD$");
|
||||
|
||||
#include <sys/types.h>
|
||||
|
||||
#include <rune.h>
|
||||
#include <runetype.h>
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <wchar.h>
|
||||
|
||||
rune_t _MSKanji_sgetrune(const char *, size_t, char const **);
|
||||
int _MSKanji_sputrune(rune_t, char *, size_t, char **);
|
||||
extern size_t (*__mbrtowc)(wchar_t * __restrict, const char * __restrict,
|
||||
size_t, mbstate_t * __restrict);
|
||||
extern size_t (*__wcrtomb)(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
int _MSKanji_init(_RuneLocale *);
|
||||
size_t _MSKanji_mbrtowc(wchar_t * __restrict, const char * __restrict, size_t,
|
||||
mbstate_t * __restrict);
|
||||
size_t _MSKanji_wcrtomb(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
int
|
||||
_MSKanji_init(rl)
|
||||
_RuneLocale *rl;
|
||||
_MSKanji_init(_RuneLocale *rl)
|
||||
{
|
||||
rl->sgetrune = _MSKanji_sgetrune;
|
||||
rl->sputrune = _MSKanji_sputrune;
|
||||
|
||||
__mbrtowc = _MSKanji_mbrtowc;
|
||||
__wcrtomb = _MSKanji_wcrtomb;
|
||||
_CurrentRuneLocale = rl;
|
||||
__mb_cur_max = 2;
|
||||
return (0);
|
||||
}
|
||||
|
||||
rune_t
|
||||
_MSKanji_sgetrune(string, n, result)
|
||||
const char *string;
|
||||
size_t n;
|
||||
char const **result;
|
||||
size_t
|
||||
_MSKanji_mbrtowc(wchar_t * __restrict pwc, const char * __restrict s, size_t n,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
rune_t rune = 0;
|
||||
wchar_t wc;
|
||||
int len;
|
||||
|
||||
if (n < 1) {
|
||||
if (result != NULL)
|
||||
*result = string;
|
||||
return (_INVALID_RUNE);
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (0);
|
||||
if (n == 0)
|
||||
/* Incomplete multibyte sequence */
|
||||
return ((size_t)-2);
|
||||
len = 1;
|
||||
wc = *s++ & 0xff;
|
||||
if ((wc > 0x80 && wc < 0xa0) || (wc >= 0xe0 && wc < 0xfd)) {
|
||||
if (n < 2)
|
||||
/* Incomplete multibyte sequence */
|
||||
return ((size_t)-2);
|
||||
wc = (wc << 8) | (*s++ & 0xff);
|
||||
len = 2;
|
||||
}
|
||||
|
||||
rune = *string++ & 0xff;
|
||||
if ((rune > 0x80 && rune < 0xa0) ||
|
||||
(rune >= 0xe0 && rune < 0xfd)) {
|
||||
if (n < 2) {
|
||||
rune = _INVALID_RUNE;
|
||||
--string;
|
||||
} else
|
||||
rune = (rune << 8) | (*string++ & 0xff);
|
||||
}
|
||||
if (result != NULL)
|
||||
*result = string;
|
||||
|
||||
return (rune);
|
||||
if (pwc != NULL)
|
||||
*pwc = wc;
|
||||
return (wc == L'\0' ? 0 : len);
|
||||
}
|
||||
|
||||
int
|
||||
_MSKanji_sputrune(c, string, n, result)
|
||||
rune_t c;
|
||||
char *string, **result;
|
||||
size_t n;
|
||||
size_t
|
||||
_MSKanji_wcrtomb(char * __restrict s, wchar_t wc,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
int len, i;
|
||||
|
||||
len = (c > 0x100) ? 2 : 1;
|
||||
if (n < len) {
|
||||
if (result != NULL)
|
||||
*result = NULL;
|
||||
} else {
|
||||
if (result != NULL)
|
||||
*result = string + len;
|
||||
for (i = len; i-- > 0; )
|
||||
*string++ = c >> (i << 3);
|
||||
}
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (1);
|
||||
|
||||
len = (wc > 0x100) ? 2 : 1;
|
||||
for (i = len; i-- > 0; )
|
||||
*s++ = wc >> (i << 3);
|
||||
return (len);
|
||||
}
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*-
|
||||
* Copyright (c) 2002 Tim J. Robbins
|
||||
* Copyright (c) 2002, 2003 Tim J. Robbins
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -27,37 +27,46 @@
|
||||
#include <sys/cdefs.h>
|
||||
__FBSDID("$FreeBSD$");
|
||||
|
||||
#include <rune.h>
|
||||
#include <errno.h>
|
||||
#include <runetype.h>
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <wchar.h>
|
||||
|
||||
rune_t _UTF8_sgetrune(const char *, size_t, char const **);
|
||||
int _UTF8_sputrune(rune_t, char *, size_t, char **);
|
||||
extern size_t (*__mbrtowc)(wchar_t * __restrict, const char * __restrict,
|
||||
size_t, mbstate_t * __restrict);
|
||||
extern size_t (*__wcrtomb)(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
size_t _UTF8_mbrtowc(wchar_t * __restrict, const char * __restrict, size_t,
|
||||
mbstate_t * __restrict);
|
||||
size_t _UTF8_wcrtomb(char * __restrict, wchar_t, mbstate_t * __restrict);
|
||||
|
||||
int
|
||||
_UTF8_init(_RuneLocale *rl)
|
||||
{
|
||||
|
||||
rl->sgetrune = _UTF8_sgetrune;
|
||||
rl->sputrune = _UTF8_sputrune;
|
||||
__mbrtowc = _UTF8_mbrtowc;
|
||||
__wcrtomb = _UTF8_wcrtomb;
|
||||
_CurrentRuneLocale = rl;
|
||||
__mb_cur_max = 6;
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
rune_t
|
||||
_UTF8_sgetrune(const char *string, size_t n, const char **result)
|
||||
size_t
|
||||
_UTF8_mbrtowc(wchar_t * __restrict pwc, const char * __restrict s, size_t n,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
int ch, len, mask;
|
||||
rune_t lbound, wch;
|
||||
int ch, i, len, mask;
|
||||
wchar_t lbound, wch;
|
||||
|
||||
if (n < 1) {
|
||||
if (result != NULL)
|
||||
*result = string;
|
||||
return (_INVALID_RUNE);
|
||||
}
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (0);
|
||||
if (n == 0)
|
||||
/* Incomplete multibyte sequence */
|
||||
return ((size_t)-2);
|
||||
|
||||
/*
|
||||
* Determine the number of octets that make up this character from
|
||||
@ -70,7 +79,7 @@ _UTF8_sgetrune(const char *string, size_t n, const char **result)
|
||||
* character. This enforces a 1-to-1 mapping between character
|
||||
* codes and their multibyte representations.
|
||||
*/
|
||||
ch = (unsigned char)*string;
|
||||
ch = (unsigned char)*s;
|
||||
if ((ch & 0x80) == 0) {
|
||||
mask = 0x7f;
|
||||
len = 1;
|
||||
@ -99,106 +108,95 @@ _UTF8_sgetrune(const char *string, size_t n, const char **result)
|
||||
/*
|
||||
* Malformed input; input is not UTF-8.
|
||||
*/
|
||||
if (result != NULL)
|
||||
*result = string + 1;
|
||||
return (_INVALID_RUNE);
|
||||
errno = EILSEQ;
|
||||
return ((size_t)-1);
|
||||
}
|
||||
|
||||
if (n < len) {
|
||||
/*
|
||||
* Truncated or partial input.
|
||||
*/
|
||||
if (result != NULL)
|
||||
*result = string;
|
||||
return (_INVALID_RUNE);
|
||||
}
|
||||
if (n < (size_t)len)
|
||||
/* Incomplete multibyte sequence */
|
||||
return ((size_t)-2);
|
||||
|
||||
/*
|
||||
* Decode the octet sequence representing the character in chunks
|
||||
* of 6 bits, most significant first.
|
||||
*/
|
||||
wch = (unsigned char)*string++ & mask;
|
||||
while (--len != 0) {
|
||||
if ((*string & 0xc0) != 0x80) {
|
||||
wch = (unsigned char)*s++ & mask;
|
||||
i = len;
|
||||
while (--i != 0) {
|
||||
if ((*s & 0xc0) != 0x80) {
|
||||
/*
|
||||
* Malformed input; bad characters in the middle
|
||||
* of a character.
|
||||
*/
|
||||
wch = _INVALID_RUNE;
|
||||
if (result != NULL)
|
||||
*result = string + 1;
|
||||
return (_INVALID_RUNE);
|
||||
errno = EILSEQ;
|
||||
return ((size_t)-1);
|
||||
}
|
||||
wch <<= 6;
|
||||
wch |= *string++ & 0x3f;
|
||||
wch |= *s++ & 0x3f;
|
||||
}
|
||||
if (wch != _INVALID_RUNE && wch < lbound)
|
||||
if (wch < lbound) {
|
||||
/*
|
||||
* Malformed input; redundant encoding.
|
||||
*/
|
||||
wch = _INVALID_RUNE;
|
||||
if (result != NULL)
|
||||
*result = string;
|
||||
return (wch);
|
||||
errno = EILSEQ;
|
||||
return ((size_t)-1);
|
||||
}
|
||||
if (pwc != NULL)
|
||||
*pwc = wch;
|
||||
return (wch == L'\0' ? 0 : i);
|
||||
}
|
||||
|
||||
int
|
||||
_UTF8_sputrune(rune_t c, char *string, size_t n, char **result)
|
||||
size_t
|
||||
_UTF8_wcrtomb(char * __restrict s, wchar_t wc,
|
||||
mbstate_t * __restrict ps __unused)
|
||||
{
|
||||
unsigned char lead;
|
||||
int i, len;
|
||||
|
||||
if (s == NULL)
|
||||
/* Reset to initial shift state (no-op) */
|
||||
return (1);
|
||||
|
||||
/*
|
||||
* Determine the number of octets needed to represent this character.
|
||||
* We always output the shortest sequence possible. Also specify the
|
||||
* first few bits of the first octet, which contains the information
|
||||
* about the sequence length.
|
||||
*/
|
||||
if ((c & ~0x7f) == 0) {
|
||||
if ((wc & ~0x7f) == 0) {
|
||||
lead = 0;
|
||||
len = 1;
|
||||
} else if ((c & ~0x7ff) == 0) {
|
||||
} else if ((wc & ~0x7ff) == 0) {
|
||||
lead = 0xc0;
|
||||
len = 2;
|
||||
} else if ((c & ~0xffff) == 0) {
|
||||
} else if ((wc & ~0xffff) == 0) {
|
||||
lead = 0xe0;
|
||||
len = 3;
|
||||
} else if ((c & ~0x1fffff) == 0) {
|
||||
} else if ((wc & ~0x1fffff) == 0) {
|
||||
lead = 0xf0;
|
||||
len = 4;
|
||||
} else if ((c & ~0x3ffffff) == 0) {
|
||||
} else if ((wc & ~0x3ffffff) == 0) {
|
||||
lead = 0xf8;
|
||||
len = 5;
|
||||
} else if ((c & ~0x7fffffff) == 0) {
|
||||
} else if ((wc & ~0x7fffffff) == 0) {
|
||||
lead = 0xfc;
|
||||
len = 6;
|
||||
} else {
|
||||
/*
|
||||
* Wide character code is out of range.
|
||||
*/
|
||||
if (result != NULL)
|
||||
*result = NULL;
|
||||
return (0);
|
||||
errno = EILSEQ;
|
||||
return ((size_t)-1);
|
||||
}
|
||||
|
||||
if (n < len) {
|
||||
if (result != NULL)
|
||||
*result = NULL;
|
||||
} else {
|
||||
/*
|
||||
* Output the octets representing the character in chunks
|
||||
* of 6 bits, least significant last. The first octet is
|
||||
* a special case because it contains the sequence length
|
||||
* information.
|
||||
*/
|
||||
for (i = len - 1; i > 0; i--) {
|
||||
string[i] = (c & 0x3f) | 0x80;
|
||||
c >>= 6;
|
||||
}
|
||||
*string = (c & 0xff) | lead;
|
||||
if (result != NULL)
|
||||
*result = string + len;
|
||||
/*
|
||||
* Output the octets representing the character in chunks
|
||||
* of 6 bits, least significant last. The first octet is
|
||||
* a special case because it contains the sequence length
|
||||
* information.
|
||||
*/
|
||||
for (i = len - 1; i > 0; i--) {
|
||||
s[i] = (wc & 0x3f) | 0x80;
|
||||
wc >>= 6;
|
||||
}
|
||||
*s = (wc & 0xff) | lead;
|
||||
|
||||
return (len);
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user