Blame lib/mbsrtowcs-impl.h

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