blob: 676932b5dcaa0d93da7fabded9c006dd57e19d90 [file] [log] [blame]
#include <wchar.h>
size_t wcsnrtombs(char *restrict dst, const wchar_t **restrict wcs, size_t wn, size_t n, mbstate_t *restrict st)
{
size_t l, cnt=0, n2;
char *s, buf[256];
const wchar_t *ws = *wcs;
const wchar_t *tmp_ws;
if (!dst) s = buf, n = sizeof buf;
else s = dst;
while ( ws && n && ( (n2=wn)>=n || n2>32 ) ) {
if (n2>=n) n2=n;
tmp_ws = ws;
l = wcsrtombs(s, &ws, n2, 0);
if (!(l+1)) {
cnt = l;
n = 0;
break;
}
if (s != buf) {
s += l;
n -= l;
}
wn = ws ? wn - (ws - tmp_ws) : 0;
cnt += l;
}
if (ws) while (n && wn) {
l = wcrtomb(s, *ws, 0);
if ((l+1)<=1) {
if (!l) ws = 0;
else cnt = l;
break;
}
ws++; wn--;
/* safe - this loop runs fewer than sizeof(buf) times */
s+=l; n-=l;
cnt += l;
}
if (dst) *wcs = ws;
return cnt;
}