1 /* Copyright (c) 2009 Corinna Vinschen <corinna@vinschen.de> */
2 /*
3 FUNCTION
4 <<mbsrtowcs>>, <<mbsnrtowcs>>---convert a character string to a wide-character string
5 
6 INDEX
7 	mbsrtowcs
8 INDEX
9 	_mbsrtowcs_r
10 INDEX
11 	mbsnrtowcs
12 INDEX
13 	_mbsnrtowcs_r
14 
15 SYNOPSIS
16 	#include <wchar.h>
17 	size_t mbsrtowcs(wchar_t *__restrict <[dst]>,
18 			 const char **__restrict <[src]>,
19 			 size_t <[len]>,
20 			 mbstate_t *__restrict <[ps]>);
21 
22 	#include <wchar.h>
23 	size_t mbsnrtowcs(wchar_t *__ restrict <[dst]>,
24 			  const char **__restrict <[src]>, size_t <[nms]>,
25 			  size_t <[len]>, mbstate_t *__restrict <[ps]>);
26 
27 
28 DESCRIPTION
29 The <<mbsrtowcs>> function converts a sequence of multibyte characters
30 pointed to indirectly by <[src]> into a sequence of corresponding wide
31 characters and stores at most <[len]> of them in the wchar_t array pointed
32 to by <[dst]>, until it encounters a terminating null character ('\0').
33 
34 If <[dst]> is NULL, no characters are stored.
35 
36 If <[dst]> is not NULL, the pointer pointed to by <[src]> is updated to point
37 to the character after the one that conversion stopped at.  If conversion
38 stops because a null character is encountered, *<[src]> is set to NULL.
39 
40 The mbstate_t argument, <[ps]>, is used to keep track of the shift state.  If
41 it is NULL, <<mbsrtowcs>> uses an internal, static mbstate_t object, which
42 is initialized to the initial conversion state at program startup.
43 
44 The <<mbsnrtowcs>> function behaves identically to <<mbsrtowcs>>, except that
45 conversion stops after reading at most <[nms]> bytes from the buffer pointed
46 to by <[src]>.
47 
48 RETURNS
49 The <<mbsrtowcs>> and <<mbsnrtowcs>> functions return the number of wide
50 characters stored in the array pointed to by <[dst]> if successful, otherwise
51 it returns (size_t)-1.
52 
53 PORTABILITY
54 <<mbsrtowcs>> is defined by the C99 standard.
55 <<mbsnrtowcs>> is defined by the POSIX.1-2008 standard.
56 */
57 
58 #define _DEFAULT_SOURCE
59 #include <wchar.h>
60 #include <stdlib.h>
61 #include <stdio.h>
62 #include <errno.h>
63 
64 size_t
mbsnrtowcs(wchar_t * dst,const char ** src,size_t nms,size_t len,mbstate_t * ps)65 mbsnrtowcs (
66 	wchar_t *dst,
67 	const char **src,
68 	size_t nms,
69 	size_t len,
70 	mbstate_t *ps)
71 {
72   wchar_t *ptr = dst;
73   const char *tmp_src;
74   size_t max;
75   size_t count = 0;
76   int bytes;
77 
78 #ifdef _MB_CAPABLE
79   if (ps == NULL)
80     {
81       static NEWLIB_THREAD_LOCAL mbstate_t _mbsrtowcs_state;
82       ps = &_mbsrtowcs_state;
83     }
84 #endif
85 
86   if (dst == NULL)
87     {
88       /* Ignore original len value and do not alter src pointer if the
89          dst pointer is NULL.  */
90       len = (size_t)-1;
91       tmp_src = *src;
92       src = &tmp_src;
93     }
94 
95   max = len;
96   while (len > 0)
97     {
98       bytes = mbrtowc (ptr, *src, nms, ps);
99       if (bytes > 0)
100 	{
101 	  *src += bytes;
102 	  nms -= bytes;
103 	  ++count;
104 	  ptr = (dst == NULL) ? NULL : ptr + 1;
105 	  --len;
106 	}
107       else if (bytes == -2)
108 	{
109 	  *src += nms;
110 	  return count;
111 	}
112       else if (bytes == 0)
113 	{
114 	  *src = NULL;
115 	  return count;
116 	}
117       else
118 	{
119 	  ps->__count = 0;
120 	  _REENT_ERRNO(r) = EILSEQ;
121 	  return (size_t)-1;
122 	}
123     }
124 
125   return (size_t)max;
126 }
127