xref: /aosp_15_r20/external/libcxx/src/support/solaris/wcsnrtombs.inc (revision 58b9f456b02922dfdb1fad8a988d5fd8765ecb80)
1*58b9f456SAndroid Build Coastguard Worker/*-
2*58b9f456SAndroid Build Coastguard Worker * Copyright (c) 2002-2004 Tim J. Robbins.
3*58b9f456SAndroid Build Coastguard Worker *
4*58b9f456SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
5*58b9f456SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
6*58b9f456SAndroid Build Coastguard Worker * are met:
7*58b9f456SAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
8*58b9f456SAndroid Build Coastguard Worker *    notice, this list of conditions and the following disclaimer.
9*58b9f456SAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
10*58b9f456SAndroid Build Coastguard Worker *    notice, this list of conditions and the following disclaimer in the
11*58b9f456SAndroid Build Coastguard Worker *    documentation and/or other materials provided with the distribution.
12*58b9f456SAndroid Build Coastguard Worker *
13*58b9f456SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14*58b9f456SAndroid Build Coastguard Worker * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15*58b9f456SAndroid Build Coastguard Worker * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16*58b9f456SAndroid Build Coastguard Worker * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17*58b9f456SAndroid Build Coastguard Worker * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18*58b9f456SAndroid Build Coastguard Worker * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19*58b9f456SAndroid Build Coastguard Worker * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20*58b9f456SAndroid Build Coastguard Worker * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21*58b9f456SAndroid Build Coastguard Worker * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22*58b9f456SAndroid Build Coastguard Worker * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23*58b9f456SAndroid Build Coastguard Worker * SUCH DAMAGE.
24*58b9f456SAndroid Build Coastguard Worker */
25*58b9f456SAndroid Build Coastguard Worker
26*58b9f456SAndroid Build Coastguard Worker
27*58b9f456SAndroid Build Coastguard Workersize_t
28*58b9f456SAndroid Build Coastguard Workerwcsnrtombs_l(char * __restrict dst, const wchar_t ** __restrict src,
29*58b9f456SAndroid Build Coastguard Worker    size_t nwc, size_t len, mbstate_t * __restrict ps, locale_t loc)
30*58b9f456SAndroid Build Coastguard Worker{
31*58b9f456SAndroid Build Coastguard Worker  FIX_LOCALE(loc);
32*58b9f456SAndroid Build Coastguard Worker  mbstate_t mbsbak;
33*58b9f456SAndroid Build Coastguard Worker  char buf[MB_CUR_MAX_L(loc)];
34*58b9f456SAndroid Build Coastguard Worker  const wchar_t *s;
35*58b9f456SAndroid Build Coastguard Worker  size_t nbytes;
36*58b9f456SAndroid Build Coastguard Worker  size_t nb;
37*58b9f456SAndroid Build Coastguard Worker
38*58b9f456SAndroid Build Coastguard Worker  s = *src;
39*58b9f456SAndroid Build Coastguard Worker  nbytes = 0;
40*58b9f456SAndroid Build Coastguard Worker
41*58b9f456SAndroid Build Coastguard Worker  if (dst == NULL) {
42*58b9f456SAndroid Build Coastguard Worker    while (nwc-- > 0) {
43*58b9f456SAndroid Build Coastguard Worker      if ((nb = wcrtomb_l(buf, *s, ps, loc)) == (size_t)-1)
44*58b9f456SAndroid Build Coastguard Worker        /* Invalid character - wcrtomb() sets errno. */
45*58b9f456SAndroid Build Coastguard Worker        return ((size_t)-1);
46*58b9f456SAndroid Build Coastguard Worker      else if (*s == L'\0')
47*58b9f456SAndroid Build Coastguard Worker        return (nbytes + nb - 1);
48*58b9f456SAndroid Build Coastguard Worker      s++;
49*58b9f456SAndroid Build Coastguard Worker      nbytes += nb;
50*58b9f456SAndroid Build Coastguard Worker    }
51*58b9f456SAndroid Build Coastguard Worker    return (nbytes);
52*58b9f456SAndroid Build Coastguard Worker  }
53*58b9f456SAndroid Build Coastguard Worker
54*58b9f456SAndroid Build Coastguard Worker  while (len > 0 && nwc-- > 0) {
55*58b9f456SAndroid Build Coastguard Worker    if (len > (size_t)MB_CUR_MAX_L(loc)) {
56*58b9f456SAndroid Build Coastguard Worker      /* Enough space to translate in-place. */
57*58b9f456SAndroid Build Coastguard Worker      if ((nb = wcrtomb_l(dst, *s, ps, loc)) == (size_t)-1) {
58*58b9f456SAndroid Build Coastguard Worker        *src = s;
59*58b9f456SAndroid Build Coastguard Worker        return ((size_t)-1);
60*58b9f456SAndroid Build Coastguard Worker      }
61*58b9f456SAndroid Build Coastguard Worker    } else {
62*58b9f456SAndroid Build Coastguard Worker      /*
63*58b9f456SAndroid Build Coastguard Worker       * May not be enough space; use temp. buffer.
64*58b9f456SAndroid Build Coastguard Worker       *
65*58b9f456SAndroid Build Coastguard Worker       * We need to save a copy of the conversion state
66*58b9f456SAndroid Build Coastguard Worker       * here so we can restore it if the multibyte
67*58b9f456SAndroid Build Coastguard Worker       * character is too long for the buffer.
68*58b9f456SAndroid Build Coastguard Worker       */
69*58b9f456SAndroid Build Coastguard Worker      mbsbak = *ps;
70*58b9f456SAndroid Build Coastguard Worker      if ((nb = wcrtomb_l(buf, *s, ps, loc)) == (size_t)-1) {
71*58b9f456SAndroid Build Coastguard Worker        *src = s;
72*58b9f456SAndroid Build Coastguard Worker        return ((size_t)-1);
73*58b9f456SAndroid Build Coastguard Worker      }
74*58b9f456SAndroid Build Coastguard Worker      if (nb > (int)len) {
75*58b9f456SAndroid Build Coastguard Worker        /* MB sequence for character won't fit. */
76*58b9f456SAndroid Build Coastguard Worker        *ps = mbsbak;
77*58b9f456SAndroid Build Coastguard Worker        break;
78*58b9f456SAndroid Build Coastguard Worker      }
79*58b9f456SAndroid Build Coastguard Worker      memcpy(dst, buf, nb);
80*58b9f456SAndroid Build Coastguard Worker    }
81*58b9f456SAndroid Build Coastguard Worker    if (*s == L'\0') {
82*58b9f456SAndroid Build Coastguard Worker      *src = NULL;
83*58b9f456SAndroid Build Coastguard Worker      return (nbytes + nb - 1);
84*58b9f456SAndroid Build Coastguard Worker    }
85*58b9f456SAndroid Build Coastguard Worker    s++;
86*58b9f456SAndroid Build Coastguard Worker    dst += nb;
87*58b9f456SAndroid Build Coastguard Worker    len -= nb;
88*58b9f456SAndroid Build Coastguard Worker    nbytes += nb;
89*58b9f456SAndroid Build Coastguard Worker  }
90*58b9f456SAndroid Build Coastguard Worker  *src = s;
91*58b9f456SAndroid Build Coastguard Worker  return (nbytes);
92*58b9f456SAndroid Build Coastguard Worker}
93*58b9f456SAndroid Build Coastguard Worker
94