Blame lib/mbsrtowcs-impl.h

Packit 33f14e
/* Convert string to wide string.
Packit 33f14e
   Copyright (C) 2008-2017 Free Software Foundation, Inc.
Packit 33f14e
   Written by Bruno Haible <bruno@clisp.org>, 2008.
Packit 33f14e
Packit 33f14e
   This program is free software: you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3 of the License, or
Packit 33f14e
   (at your option) any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
size_t
Packit 33f14e
mbsrtowcs (wchar_t *dest, const char **srcp, size_t len, mbstate_t *ps)
Packit 33f14e
{
Packit 33f14e
  if (ps == NULL)
Packit 33f14e
    ps = &_gl_mbsrtowcs_state;
Packit 33f14e
  {
Packit 33f14e
    const char *src = *srcp;
Packit 33f14e
Packit 33f14e
    if (dest != NULL)
Packit 33f14e
      {
Packit 33f14e
        wchar_t *destptr = dest;
Packit 33f14e
Packit 33f14e
        for (; len > 0; destptr++, len--)
Packit 33f14e
          {
Packit 33f14e
            size_t src_avail;
Packit 33f14e
            size_t ret;
Packit 33f14e
Packit 33f14e
            /* An optimized variant of
Packit 33f14e
               src_avail = strnlen1 (src, MB_LEN_MAX);  */
Packit 33f14e
            if (src[0] == '\0')
Packit 33f14e
              src_avail = 1;
Packit 33f14e
            else if (src[1] == '\0')
Packit 33f14e
              src_avail = 2;
Packit 33f14e
            else if (src[2] == '\0')
Packit 33f14e
              src_avail = 3;
Packit 33f14e
            else if (MB_LEN_MAX <= 4 || src[3] == '\0')
Packit 33f14e
              src_avail = 4;
Packit 33f14e
            else
Packit 33f14e
              src_avail = 4 + strnlen1 (src + 4, MB_LEN_MAX - 4);
Packit 33f14e
Packit 33f14e
            /* Parse the next multibyte character.  */
Packit 33f14e
            ret = mbrtowc (destptr, src, src_avail, ps);
Packit 33f14e
Packit 33f14e
            if (ret == (size_t)(-2))
Packit 33f14e
              /* Encountered a multibyte character that extends past a '\0' byte
Packit 33f14e
                 or that is longer than MB_LEN_MAX bytes.  Cannot happen.  */
Packit 33f14e
              abort ();
Packit 33f14e
Packit 33f14e
            if (ret == (size_t)(-1))
Packit 33f14e
              goto bad_input;
Packit 33f14e
            if (ret == 0)
Packit 33f14e
              {
Packit 33f14e
                src = NULL;
Packit 33f14e
                /* Here mbsinit (ps).  */
Packit 33f14e
                break;
Packit 33f14e
              }
Packit 33f14e
            src += ret;
Packit 33f14e
          }
Packit 33f14e
Packit 33f14e
        *srcp = src;
Packit 33f14e
        return destptr - dest;
Packit 33f14e
      }
Packit 33f14e
    else
Packit 33f14e
      {
Packit 33f14e
        /* Ignore dest and len, don't store *srcp at the end, and
Packit 33f14e
           don't clobber *ps.  */
Packit 33f14e
        mbstate_t state = *ps;
Packit 33f14e
        size_t totalcount = 0;
Packit 33f14e
Packit 33f14e
        for (;; totalcount++)
Packit 33f14e
          {
Packit 33f14e
            size_t src_avail;
Packit 33f14e
            size_t ret;
Packit 33f14e
Packit 33f14e
            /* An optimized variant of
Packit 33f14e
               src_avail = strnlen1 (src, MB_LEN_MAX);  */
Packit 33f14e
            if (src[0] == '\0')
Packit 33f14e
              src_avail = 1;
Packit 33f14e
            else if (src[1] == '\0')
Packit 33f14e
              src_avail = 2;
Packit 33f14e
            else if (src[2] == '\0')
Packit 33f14e
              src_avail = 3;
Packit 33f14e
            else if (MB_LEN_MAX <= 4 || src[3] == '\0')
Packit 33f14e
              src_avail = 4;
Packit 33f14e
            else
Packit 33f14e
              src_avail = 4 + strnlen1 (src + 4, MB_LEN_MAX - 4);
Packit 33f14e
Packit 33f14e
            /* Parse the next multibyte character.  */
Packit 33f14e
            ret = mbrtowc (NULL, src, src_avail, &state);
Packit 33f14e
Packit 33f14e
            if (ret == (size_t)(-2))
Packit 33f14e
              /* Encountered a multibyte character that extends past a '\0' byte
Packit 33f14e
                 or that is longer than MB_LEN_MAX bytes.  Cannot happen.  */
Packit 33f14e
              abort ();
Packit 33f14e
Packit 33f14e
            if (ret == (size_t)(-1))
Packit 33f14e
              goto bad_input2;
Packit 33f14e
            if (ret == 0)
Packit 33f14e
              {
Packit 33f14e
                /* Here mbsinit (&state).  */
Packit 33f14e
                break;
Packit 33f14e
              }
Packit 33f14e
            src += ret;
Packit 33f14e
          }
Packit 33f14e
Packit 33f14e
        return totalcount;
Packit 33f14e
      }
Packit 33f14e
Packit 33f14e
   bad_input:
Packit 33f14e
    *srcp = src;
Packit 33f14e
   bad_input2:
Packit 33f14e
    errno = EILSEQ;
Packit 33f14e
    return (size_t)(-1);
Packit 33f14e
  }
Packit 33f14e
}