update for HEAD-2003091401
[reactos.git] / lib / msvcrt / stdlib / wcstombs.c
1 /* Copyright (C) 1991, 1992, 1995, 1996, 1997 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Library General Public License as
6    published by the Free Software Foundation; either version 2 of the
7    License, or (at your option) any later version.
8
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Library General Public License for more details.
13
14    You should have received a copy of the GNU Library General Public
15    License along with the GNU C Library; see the file COPYING.LIB.  If not,
16    write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17    Boston, MA 02111-1307, USA.  */
18
19 #include <msvcrt/stdlib.h>
20 #include <msvcrt/wchar.h>
21
22 #include <msvcrt/errno.h>
23 #include <msvcrt/wchar.h>
24
25 #ifndef EILSEQ
26 #define EILSEQ EINVAL
27 #endif
28
29
30 static const wchar_t encoding_mask[] =
31 {
32   ~0x7ff, ~0xffff, ~0x1fffff, ~0x3ffffff
33 };
34
35 static const unsigned char encoding_byte[] =
36 {
37   0xc0, 0xe0, 0xf0, 0xf8, 0xfc
38 };
39
40 /* We don't need the state really because we don't have shift states
41    to maintain between calls to this function.  */
42 typedef int mbstate_t;
43 static mbstate_t mbstate_internal;
44
45
46 mbstate_t __no_r_state;  /* Now defined in wcstombs.c.  */
47 //extern mbstate_t __no_r_state;  /* Defined in mbtowc.c.  */
48
49 size_t
50 __wcsrtombs (char *dst, const wchar_t **src, size_t len, mbstate_t *ps);
51
52 /*
53  * Convert the `wchar_t' string in PWCS to a multibyte character string
54  * in S, writing no more than N characters.  Return the number of bytes
55  * written, or (size_t) -1 if an invalid `wchar_t' was found.
56  *
57  * Attention: this function should NEVER be intentionally used.
58  * The interface is completely stupid.  The state is shared between
59  * all conversion functions.  You should use instead the restartable
60  * version `wcsrtombs'.
61  *
62  * @implemented
63  */
64 size_t
65 wcstombs (char *s, const wchar_t *pwcs, size_t n)
66 {
67   mbstate_t save_shift = __no_r_state;
68   size_t written;
69
70   written = __wcsrtombs (s, &pwcs, n, &__no_r_state);
71
72   /* Restore the old shift state.  */
73   __no_r_state = save_shift;
74
75   /* Return how many we wrote (or maybe an error).  */
76   return written;
77 }
78
79 size_t
80 __wcsrtombs (char *dst, const wchar_t **src, size_t len, mbstate_t *ps)
81 {
82   size_t written = 0;
83   const wchar_t *run = *src;
84
85   if (ps == NULL)
86     ps = &mbstate_internal;
87
88   if (dst == NULL)
89     /* The LEN parameter has to be ignored if we don't actually write
90        anything.  */
91     len = ~0;
92
93   while (written < len)
94     {
95       wchar_t wc = *run++;
96
97       if (wc < 0 || wc > 0x7fffffff)
98         {
99           /* This is no correct ISO 10646 character.  */
100           __set_errno (EILSEQ);
101           return (size_t) -1;
102         }
103
104       if (wc == L'\0')
105         {
106           /* Found the end.  */
107           if (dst != NULL)
108             *dst = '\0';
109           *src = NULL;
110           return written;
111         }
112       else if (wc < 0x80)
113         {
114           /* It's an one byte sequence.  */
115           if (dst != NULL)
116             *dst++ = (char) wc;
117           ++written;
118         }
119       else
120         {
121           size_t step;
122
123           for (step = 2; step < 6; ++step)
124             if ((wc & encoding_mask[step - 2]) == 0)
125               break;
126
127           if (written + step >= len)
128             /* Too long.  */
129             break;
130
131           if (dst != NULL)
132             {
133               size_t cnt = step;
134
135               dst[0] = encoding_byte[cnt - 2];
136
137               --cnt;
138               do
139                 {
140                   dst[cnt] = 0x80 | (wc & 0x3f);
141                   wc >>= 6;
142                 }
143               while (--cnt > 0);
144               dst[0] |= wc;
145
146               dst += step;
147             }
148
149           written += step;
150         }
151     }
152
153   /* Store position of first unprocessed word.  */
154   *src = run;
155
156   return written;
157 }
158 //weak_alias (__wcsrtombs, wcsrtombs)