Blame lib/mbiter.h

Packit Service fdd496
/* Iterating through multibyte strings: macros for multi-byte encodings.
Packit Service fdd496
   Copyright (C) 2001, 2005, 2007, 2009-2017 Free Software Foundation, Inc.
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
/* Written by Bruno Haible <bruno@clisp.org>.  */
Packit Service fdd496
Packit Service fdd496
/* The macros in this file implement forward iteration through a
Packit Service fdd496
   multi-byte string.
Packit Service fdd496
Packit Service fdd496
   With these macros, an iteration loop that looks like
Packit Service fdd496
Packit Service fdd496
      char *iter;
Packit Service fdd496
      for (iter = buf; iter < buf + buflen; iter++)
Packit Service fdd496
        {
Packit Service fdd496
          do_something (*iter);
Packit Service fdd496
        }
Packit Service fdd496
Packit Service fdd496
   becomes
Packit Service fdd496
Packit Service fdd496
      mbi_iterator_t iter;
Packit Service fdd496
      for (mbi_init (iter, buf, buflen); mbi_avail (iter); mbi_advance (iter))
Packit Service fdd496
        {
Packit Service fdd496
          do_something (mbi_cur_ptr (iter), mb_len (mbi_cur (iter)));
Packit Service fdd496
        }
Packit Service fdd496
Packit Service fdd496
   The benefit of these macros over plain use of mbrtowc is:
Packit Service fdd496
   - Handling of invalid multibyte sequences is possible without
Packit Service fdd496
     making the code more complicated, while still preserving the
Packit Service fdd496
     invalid multibyte sequences.
Packit Service fdd496
Packit Service fdd496
   mbi_iterator_t
Packit Service fdd496
     is a type usable for variable declarations.
Packit Service fdd496
Packit Service fdd496
   mbi_init (iter, startptr, length)
Packit Service fdd496
     initializes the iterator, starting at startptr and crossing length bytes.
Packit Service fdd496
Packit Service fdd496
   mbi_avail (iter)
Packit Service fdd496
     returns true if there are more multibyte characters available before
Packit Service fdd496
     the end of string is reached. In this case, mbi_cur (iter) is
Packit Service fdd496
     initialized to the next multibyte character.
Packit Service fdd496
Packit Service fdd496
   mbi_advance (iter)
Packit Service fdd496
     advances the iterator by one multibyte character.
Packit Service fdd496
Packit Service fdd496
   mbi_cur (iter)
Packit Service fdd496
     returns the current multibyte character, of type mbchar_t.  All the
Packit Service fdd496
     macros defined in mbchar.h can be used on it.
Packit Service fdd496
Packit Service fdd496
   mbi_cur_ptr (iter)
Packit Service fdd496
     return a pointer to the beginning of the current multibyte character.
Packit Service fdd496
Packit Service fdd496
   mbi_reloc (iter, ptrdiff)
Packit Service fdd496
     relocates iterator when the string is moved by ptrdiff bytes.
Packit Service fdd496
Packit Service fdd496
   mbi_copy (&destiter, &srciter)
Packit Service fdd496
     copies srciter to destiter.
Packit Service fdd496
Packit Service fdd496
   Here are the function prototypes of the macros.
Packit Service fdd496
Packit Service fdd496
   extern void          mbi_init (mbi_iterator_t iter,
Packit Service fdd496
                                  const char *startptr, size_t length);
Packit Service fdd496
   extern bool          mbi_avail (mbi_iterator_t iter);
Packit Service fdd496
   extern void          mbi_advance (mbi_iterator_t iter);
Packit Service fdd496
   extern mbchar_t      mbi_cur (mbi_iterator_t iter);
Packit Service fdd496
   extern const char *  mbi_cur_ptr (mbi_iterator_t iter);
Packit Service fdd496
   extern void          mbi_reloc (mbi_iterator_t iter, ptrdiff_t ptrdiff);
Packit Service fdd496
   extern void          mbi_copy (mbi_iterator_t *new, const mbi_iterator_t *old);
Packit Service fdd496
 */
Packit Service fdd496
Packit Service fdd496
#ifndef _MBITER_H
Packit Service fdd496
#define _MBITER_H 1
Packit Service fdd496
Packit Service fdd496
#include <assert.h>
Packit Service fdd496
#include <stdbool.h>
Packit Service fdd496
#include <stddef.h>
Packit Service fdd496
#include <string.h>
Packit Service fdd496
Packit Service fdd496
/* Tru64 with Desktop Toolkit C has a bug: <stdio.h> must be included before
Packit Service fdd496
   <wchar.h>.
Packit Service fdd496
   BSD/OS 4.1 has a bug: <stdio.h> and <time.h> must be included before
Packit Service fdd496
   <wchar.h>.  */
Packit Service fdd496
#include <stdio.h>
Packit Service fdd496
#include <time.h>
Packit Service fdd496
#include <wchar.h>
Packit Service fdd496
Packit Service fdd496
#include "mbchar.h"
Packit Service fdd496
Packit Service fdd496
#ifndef _GL_INLINE_HEADER_BEGIN
Packit Service fdd496
 #error "Please include config.h first."
Packit Service fdd496
#endif
Packit Service fdd496
_GL_INLINE_HEADER_BEGIN
Packit Service fdd496
#ifndef MBITER_INLINE
Packit Service fdd496
# define MBITER_INLINE _GL_INLINE
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
struct mbiter_multi
Packit Service fdd496
{
Packit Service fdd496
  const char *limit;    /* pointer to end of string */
Packit Service fdd496
  bool in_shift;        /* true if next byte may not be interpreted as ASCII */
Packit Service fdd496
  mbstate_t state;      /* if in_shift: current shift state */
Packit Service fdd496
  bool next_done;       /* true if mbi_avail has already filled the following */
Packit Service fdd496
  struct mbchar cur;    /* the current character:
Packit Service fdd496
        const char *cur.ptr             pointer to current character
Packit Service fdd496
        The following are only valid after mbi_avail.
Packit Service fdd496
        size_t cur.bytes                number of bytes of current character
Packit Service fdd496
        bool cur.wc_valid               true if wc is a valid wide character
Packit Service fdd496
        wchar_t cur.wc                  if wc_valid: the current character
Packit Service fdd496
        */
Packit Service fdd496
};
Packit Service fdd496
Packit Service fdd496
MBITER_INLINE void
Packit Service fdd496
mbiter_multi_next (struct mbiter_multi *iter)
Packit Service fdd496
{
Packit Service fdd496
  if (iter->next_done)
Packit Service fdd496
    return;
Packit Service fdd496
  if (iter->in_shift)
Packit Service fdd496
    goto with_shift;
Packit Service fdd496
  /* Handle most ASCII characters quickly, without calling mbrtowc().  */
Packit Service fdd496
  if (is_basic (*iter->cur.ptr))
Packit Service fdd496
    {
Packit Service fdd496
      /* These characters are part of the basic character set.  ISO C 99
Packit Service fdd496
         guarantees that their wide character code is identical to their
Packit Service fdd496
         char code.  */
Packit Service fdd496
      iter->cur.bytes = 1;
Packit Service fdd496
      iter->cur.wc = *iter->cur.ptr;
Packit Service fdd496
      iter->cur.wc_valid = true;
Packit Service fdd496
    }
Packit Service fdd496
  else
Packit Service fdd496
    {
Packit Service fdd496
      assert (mbsinit (&iter->state));
Packit Service fdd496
      iter->in_shift = true;
Packit Service fdd496
    with_shift:
Packit Service fdd496
      iter->cur.bytes = mbrtowc (&iter->cur.wc, iter->cur.ptr,
Packit Service fdd496
                                 iter->limit - iter->cur.ptr, &iter->state);
Packit Service fdd496
      if (iter->cur.bytes == (size_t) -1)
Packit Service fdd496
        {
Packit Service fdd496
          /* An invalid multibyte sequence was encountered.  */
Packit Service fdd496
          iter->cur.bytes = 1;
Packit Service fdd496
          iter->cur.wc_valid = false;
Packit Service fdd496
          /* Whether to set iter->in_shift = false and reset iter->state
Packit Service fdd496
             or not is not very important; the string is bogus anyway.  */
Packit Service fdd496
        }
Packit Service fdd496
      else if (iter->cur.bytes == (size_t) -2)
Packit Service fdd496
        {
Packit Service fdd496
          /* An incomplete multibyte character at the end.  */
Packit Service fdd496
          iter->cur.bytes = iter->limit - iter->cur.ptr;
Packit Service fdd496
          iter->cur.wc_valid = false;
Packit Service fdd496
          /* Whether to set iter->in_shift = false and reset iter->state
Packit Service fdd496
             or not is not important; the string end is reached anyway.  */
Packit Service fdd496
        }
Packit Service fdd496
      else
Packit Service fdd496
        {
Packit Service fdd496
          if (iter->cur.bytes == 0)
Packit Service fdd496
            {
Packit Service fdd496
              /* A null wide character was encountered.  */
Packit Service fdd496
              iter->cur.bytes = 1;
Packit Service fdd496
              assert (*iter->cur.ptr == '\0');
Packit Service fdd496
              assert (iter->cur.wc == 0);
Packit Service fdd496
            }
Packit Service fdd496
          iter->cur.wc_valid = true;
Packit Service fdd496
Packit Service fdd496
          /* When in the initial state, we can go back treating ASCII
Packit Service fdd496
             characters more quickly.  */
Packit Service fdd496
          if (mbsinit (&iter->state))
Packit Service fdd496
            iter->in_shift = false;
Packit Service fdd496
        }
Packit Service fdd496
    }
Packit Service fdd496
  iter->next_done = true;
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
MBITER_INLINE void
Packit Service fdd496
mbiter_multi_reloc (struct mbiter_multi *iter, ptrdiff_t ptrdiff)
Packit Service fdd496
{
Packit Service fdd496
  iter->cur.ptr += ptrdiff;
Packit Service fdd496
  iter->limit += ptrdiff;
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
MBITER_INLINE void
Packit Service fdd496
mbiter_multi_copy (struct mbiter_multi *new_iter, const struct mbiter_multi *old_iter)
Packit Service fdd496
{
Packit Service fdd496
  new_iter->limit = old_iter->limit;
Packit Service fdd496
  if ((new_iter->in_shift = old_iter->in_shift))
Packit Service fdd496
    memcpy (&new_iter->state, &old_iter->state, sizeof (mbstate_t));
Packit Service fdd496
  else
Packit Service fdd496
    memset (&new_iter->state, 0, sizeof (mbstate_t));
Packit Service fdd496
  new_iter->next_done = old_iter->next_done;
Packit Service fdd496
  mb_copy (&new_iter->cur, &old_iter->cur);
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
/* Iteration macros.  */
Packit Service fdd496
typedef struct mbiter_multi mbi_iterator_t;
Packit Service fdd496
#define mbi_init(iter, startptr, length) \
Packit Service fdd496
  ((iter).cur.ptr = (startptr), (iter).limit = (iter).cur.ptr + (length), \
Packit Service fdd496
   (iter).in_shift = false, memset (&(iter).state, '\0', sizeof (mbstate_t)), \
Packit Service fdd496
   (iter).next_done = false)
Packit Service fdd496
#define mbi_avail(iter) \
Packit Service fdd496
  ((iter).cur.ptr < (iter).limit && (mbiter_multi_next (&(iter)), true))
Packit Service fdd496
#define mbi_advance(iter) \
Packit Service fdd496
  ((iter).cur.ptr += (iter).cur.bytes, (iter).next_done = false)
Packit Service fdd496
Packit Service fdd496
/* Access to the current character.  */
Packit Service fdd496
#define mbi_cur(iter) (iter).cur
Packit Service fdd496
#define mbi_cur_ptr(iter) (iter).cur.ptr
Packit Service fdd496
Packit Service fdd496
/* Relocation.  */
Packit Service fdd496
#define mbi_reloc(iter, ptrdiff) mbiter_multi_reloc (&iter, ptrdiff)
Packit Service fdd496
Packit Service fdd496
/* Copying an iterator.  */
Packit Service fdd496
#define mbi_copy mbiter_multi_copy
Packit Service fdd496
Packit Service fdd496
_GL_INLINE_HEADER_END
Packit Service fdd496
Packit Service fdd496
#endif /* _MBITER_H */