Blame lib/mbsrtowcs-impl.h

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