Blame lib/mbsrtowcs-impl.h

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