xref: /freebsd/lib/libc/locale/wcsrtombs.c (revision 6155c34adf50d3d025ec0e862bb34746ea8c0a21)
1e92a3d83STim J. Robbins /*-
274f90defSTim J. Robbins  * Copyright (c) 2002-2004 Tim J. Robbins.
3e92a3d83STim J. Robbins  * All rights reserved.
4e92a3d83STim J. Robbins  *
5e92a3d83STim J. Robbins  * Redistribution and use in source and binary forms, with or without
6e92a3d83STim J. Robbins  * modification, are permitted provided that the following conditions
7e92a3d83STim J. Robbins  * are met:
8e92a3d83STim J. Robbins  * 1. Redistributions of source code must retain the above copyright
9e92a3d83STim J. Robbins  *    notice, this list of conditions and the following disclaimer.
10e92a3d83STim J. Robbins  * 2. Redistributions in binary form must reproduce the above copyright
11e92a3d83STim J. Robbins  *    notice, this list of conditions and the following disclaimer in the
12e92a3d83STim J. Robbins  *    documentation and/or other materials provided with the distribution.
13e92a3d83STim J. Robbins  *
14e92a3d83STim J. Robbins  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15e92a3d83STim J. Robbins  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16e92a3d83STim J. Robbins  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17e92a3d83STim J. Robbins  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18e92a3d83STim J. Robbins  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19e92a3d83STim J. Robbins  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20e92a3d83STim J. Robbins  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21e92a3d83STim J. Robbins  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22e92a3d83STim J. Robbins  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23e92a3d83STim J. Robbins  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24e92a3d83STim J. Robbins  * SUCH DAMAGE.
25e92a3d83STim J. Robbins  */
26e92a3d83STim J. Robbins 
27e92a3d83STim J. Robbins #include <sys/cdefs.h>
28e92a3d83STim J. Robbins __FBSDID("$FreeBSD$");
29e92a3d83STim J. Robbins 
30e92a3d83STim J. Robbins #include <limits.h>
31e92a3d83STim J. Robbins #include <stdlib.h>
32e92a3d83STim J. Robbins #include <string.h>
33e92a3d83STim J. Robbins #include <wchar.h>
346155c34aSTim J. Robbins #include "mblocal.h"
35e92a3d83STim J. Robbins 
36e92a3d83STim J. Robbins size_t
37e92a3d83STim J. Robbins wcsrtombs(char * __restrict dst, const wchar_t ** __restrict src, size_t len,
3874f90defSTim J. Robbins     mbstate_t * __restrict ps)
39e92a3d83STim J. Robbins {
4074f90defSTim J. Robbins 	static mbstate_t mbs;
4174f90defSTim J. Robbins 	mbstate_t mbsbak;
42e92a3d83STim J. Robbins 	char buf[MB_LEN_MAX];
43e92a3d83STim J. Robbins 	const wchar_t *s;
44e92a3d83STim J. Robbins 	size_t nbytes;
45e92a3d83STim J. Robbins 	int nb;
46e92a3d83STim J. Robbins 
47e92a3d83STim J. Robbins 	s = *src;
48e92a3d83STim J. Robbins 	nbytes = 0;
49e92a3d83STim J. Robbins 
5074f90defSTim J. Robbins 	if (ps == NULL)
5174f90defSTim J. Robbins 		ps = &mbs;
52e92a3d83STim J. Robbins 	if (dst == NULL) {
53e92a3d83STim J. Robbins 		for (;;) {
546155c34aSTim J. Robbins 			if ((nb = (int)__wcrtomb(buf, *s, ps)) < 0)
55e92a3d83STim J. Robbins 				/* Invalid character - wcrtomb() sets errno. */
56e92a3d83STim J. Robbins 				return ((size_t)-1);
57e92a3d83STim J. Robbins 			else if (*s == L'\0')
58e92a3d83STim J. Robbins 				return (nbytes + nb - 1);
59e92a3d83STim J. Robbins 			s++;
60e92a3d83STim J. Robbins 			nbytes += nb;
61e92a3d83STim J. Robbins 		}
62e92a3d83STim J. Robbins 		/*NOTREACHED*/
63e92a3d83STim J. Robbins 	}
64e92a3d83STim J. Robbins 
65e92a3d83STim J. Robbins 	while (len > 0) {
66e92a3d83STim J. Robbins 		if (len > (size_t)MB_CUR_MAX) {
67e92a3d83STim J. Robbins 			/* Enough space to translate in-place. */
686155c34aSTim J. Robbins 			if ((nb = (int)__wcrtomb(dst, *s, ps)) < 0) {
69e92a3d83STim J. Robbins 				*src = s;
70e92a3d83STim J. Robbins 				return ((size_t)-1);
71e92a3d83STim J. Robbins 			}
72e92a3d83STim J. Robbins 		} else {
7374f90defSTim J. Robbins 			/*
7474f90defSTim J. Robbins 			 * May not be enough space; use temp. buffer.
7574f90defSTim J. Robbins 			 *
7674f90defSTim J. Robbins 			 * We need to save a copy of the conversion state
7774f90defSTim J. Robbins 			 * here so we can restore it if the multibyte
7874f90defSTim J. Robbins 			 * character is too long for the buffer.
7974f90defSTim J. Robbins 			 */
8074f90defSTim J. Robbins 			mbsbak = *ps;
816155c34aSTim J. Robbins 			if ((nb = (int)__wcrtomb(buf, *s, ps)) < 0) {
82e92a3d83STim J. Robbins 				*src = s;
83e92a3d83STim J. Robbins 				return ((size_t)-1);
84e92a3d83STim J. Robbins 			}
8574f90defSTim J. Robbins 			if (nb > (int)len) {
86e92a3d83STim J. Robbins 				/* MB sequence for character won't fit. */
8774f90defSTim J. Robbins 				*ps = mbsbak;
88e92a3d83STim J. Robbins 				break;
8974f90defSTim J. Robbins 			}
90e92a3d83STim J. Robbins 			memcpy(dst, buf, nb);
91e92a3d83STim J. Robbins 		}
92e92a3d83STim J. Robbins 		if (*s == L'\0') {
93e92a3d83STim J. Robbins 			*src = NULL;
94e92a3d83STim J. Robbins 			return (nbytes + nb - 1);
95e92a3d83STim J. Robbins 		}
96e92a3d83STim J. Robbins 		s++;
97e92a3d83STim J. Robbins 		dst += nb;
98e92a3d83STim J. Robbins 		len -= nb;
99e92a3d83STim J. Robbins 		nbytes += nb;
100e92a3d83STim J. Robbins 	}
101e92a3d83STim J. Robbins 	*src = s;
102e92a3d83STim J. Robbins 	return (nbytes);
103e92a3d83STim J. Robbins }
104