]>
git.saurik.com Git - apple/libc.git/blob - locale/FreeBSD/euc.c
2 * Copyright (c) 2002-2004 Tim J. Robbins. All rights reserved.
4 * The Regents of the University of California. All rights reserved.
6 * This code is derived from software contributed to Berkeley by
7 * Paul Borman at Krystal Technologies.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3. All advertising materials mentioning features or use of this software
18 * must display the following acknowledgement:
19 * This product includes software developed by the University of
20 * California, Berkeley and its contributors.
21 * 4. Neither the name of the University nor the names of its contributors
22 * may be used to endorse or promote products derived from this software
23 * without specific prior written permission.
25 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
26 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
27 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
28 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
29 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
30 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
31 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
32 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
33 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
34 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
38 #if defined(LIBC_SCCS) && !defined(lint)
39 static char sccsid
[] = "@(#)euc.c 8.1 (Berkeley) 6/4/93";
40 #endif /* LIBC_SCCS and not lint */
41 #include <sys/param.h>
42 __FBSDID("$FreeBSD: src/lib/libc/locale/euc.c,v 1.20 2004/06/23 07:01:43 tjr Exp $");
52 int _EUC_init(_RuneLocale
*);
53 size_t _EUC_mbrtowc(wchar_t * __restrict
, const char * __restrict
, size_t,
54 mbstate_t * __restrict
);
55 int _EUC_mbsinit(const mbstate_t *);
56 size_t _EUC_wcrtomb(char * __restrict
, wchar_t, mbstate_t * __restrict
);
71 _EUC_init(_RuneLocale
*rl
)
74 int x
, new__mb_cur_max
;
77 if (rl
->__variable
== NULL
)
80 v
= (char *)rl
->__variable
;
82 while (*v
== ' ' || *v
== '\t')
85 if ((ei
= malloc(sizeof(_EucInfo
))) == NULL
)
86 return (errno
== 0 ? ENOMEM
: errno
);
89 for (x
= 0; x
< 4; ++x
) {
90 ei
->count
[x
] = (int)strtol(v
, &e
, 0);
91 if (v
== e
|| !(v
= e
)) {
95 if (new__mb_cur_max
< ei
->count
[x
])
96 new__mb_cur_max
= ei
->count
[x
];
97 while (*v
== ' ' || *v
== '\t')
99 ei
->bits
[x
] = (int)strtol(v
, &e
, 0);
100 if (v
== e
|| !(v
= e
)) {
104 while (*v
== ' ' || *v
== '\t')
107 ei
->mask
= (int)strtol(v
, &e
, 0);
108 if (v
== e
|| !(v
= e
)) {
113 rl
->__variable_len
= sizeof(_EucInfo
);
114 _CurrentRuneLocale
= rl
;
115 __mb_cur_max
= new__mb_cur_max
;
116 __mbrtowc
= _EUC_mbrtowc
;
117 __wcrtomb
= _EUC_wcrtomb
;
118 __mbsinit
= _EUC_mbsinit
;
123 _EUC_mbsinit(const mbstate_t *ps
)
126 return (ps
== NULL
|| ((const _EucState
*)ps
)->want
== 0);
129 #define CEI ((_EucInfo *)(_CurrentRuneLocale->__variable))
134 #define GR_BITS 0x80808080 /* XXX: to be fixed */
140 return ((c
& 0x80) ? c
== _SS3
? 3 : c
== _SS2
? 2 : 1 : 0);
144 _EUC_mbrtowc(wchar_t * __restrict pwc
, const char * __restrict s
, size_t n
,
145 mbstate_t * __restrict ps
)
152 es
= (_EucState
*)ps
;
154 if (es
->want
< 0 || es
->want
> MB_CUR_MAX
|| es
->set
< 0 ||
167 /* Incomplete multibyte sequence */
173 want
= CEI
->count
[set
= _euc_set(*s
)];
174 if (set
== 2 || set
== 3) {
177 /* Incomplete multibyte sequence */
189 wc
= (unsigned char)*s
++;
195 for (i
= (es
->want
== 0) ? 1 : 0; i
< MIN(want
, n
); i
++) {
200 wc
= (wc
<< 8) | (unsigned char)*s
++;
203 /* Incomplete multibyte sequence */
209 wc
= (wc
& ~CEI
->mask
) | CEI
->bits
[set
];
213 return (wc
== L
'\0' ? 0 : s
- os
);
217 _EUC_wcrtomb(char * __restrict s
, wchar_t wc
, mbstate_t * __restrict ps
)
223 es
= (_EucState
*)ps
;
231 /* Reset to initial shift state (no-op) */
237 if (m
== CEI
->bits
[1]) {
239 /* Codeset 1: The first byte must have 0x80 in it. */
240 i
= len
= CEI
->count
[1];
242 *s
++ = (nm
>> (i
<< 3)) | 0x80;
244 if (m
== CEI
->bits
[0])
245 i
= len
= CEI
->count
[0];
246 else if (m
== CEI
->bits
[2]) {
247 i
= len
= CEI
->count
[2];
250 /* SS2 designates G2 into GR */
252 } else if (m
== CEI
->bits
[3]) {
253 i
= len
= CEI
->count
[3];
256 /* SS3 designates G3 into GR */
259 goto CodeSet1
; /* Bletch */
261 *s
++ = (nm
>> (i
<< 3)) & 0xff;