branch update for HEAD-2003021201
[reactos.git] / lib / crtdll / 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 /* Convert the `wchar_t' string in PWCS to a multibyte character string
53    in S, writing no more than N characters.  Return the number of bytes
54    written, or (size_t) -1 if an invalid `wchar_t' was found.
55
56    Attention: this function should NEVER be intentionally used.
57    The interface is completely stupid.  The state is shared between
58    all conversion functions.  You should use instead the restartable
59    version `wcsrtombs'.  */
60 size_t
61 wcstombs (char *s, const wchar_t *pwcs, size_t n)
62 {
63   mbstate_t save_shift = __no_r_state;
64   size_t written;
65
66   written = __wcsrtombs (s, &pwcs, n, &__no_r_state);
67
68   /* Restore the old shift state.  */
69   __no_r_state = save_shift;
70
71   /* Return how many we wrote (or maybe an error).  */
72   return written;
73 }
74
75 size_t
76 __wcsrtombs (char *dst, const wchar_t **src, size_t len, mbstate_t *ps)
77 {
78   size_t written = 0;
79   const wchar_t *run = *src;
80
81   if (ps == NULL)
82     ps = &mbstate_internal;
83
84   if (dst == NULL)
85     /* The LEN parameter has to be ignored if we don't actually write
86        anything.  */
87     len = ~0;
88
89   while (written < len)
90     {
91       wchar_t wc = *run++;
92
93       if (wc < 0 || wc > 0x7fffffff)
94         {
95           /* This is no correct ISO 10646 character.  */
96           __set_errno (EILSEQ);
97           return (size_t) -1;
98         }
99
100       if (wc == L'\0')
101         {
102           /* Found the end.  */
103           if (dst != NULL)
104             *dst = '\0';
105           *src = NULL;
106           return written;
107         }
108       else if (wc < 0x80)
109         {
110           /* It's an one byte sequence.  */
111           if (dst != NULL)
112             *dst++ = (char) wc;
113           ++written;
114         }
115       else
116         {
117           size_t step;
118
119           for (step = 2; step < 6; ++step)
120             if ((wc & encoding_mask[step - 2]) == 0)
121               break;
122
123           if (written + step >= len)
124             /* Too long.  */
125             break;
126
127           if (dst != NULL)
128             {
129               size_t cnt = step;
130
131               dst[0] = encoding_byte[cnt - 2];
132
133               --cnt;
134               do
135                 {
136                   dst[cnt] = 0x80 | (wc & 0x3f);
137                   wc >>= 6;
138                 }
139               while (--cnt > 0);
140               dst[0] |= wc;
141
142               dst += step;
143             }
144
145           written += step;
146         }
147     }
148
149   /* Store position of first unprocessed word.  */
150   *src = run;
151
152   return written;
153 }
154 //weak_alias (__wcsrtombs, wcsrtombs)