Blame lib/mbsrtowcs-impl.h

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