Blame lib/mbuiter.h

Packit 33f14e
/* Iterating through multibyte strings: macros for multi-byte encodings.
Packit 33f14e
   Copyright (C) 2001, 2005, 2007, 2009-2017 Free Software Foundation, Inc.
Packit 33f14e
Packit 33f14e
   This program is free software: you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3 of the License, or
Packit 33f14e
   (at your option) any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
/* Written by Bruno Haible <bruno@clisp.org>.  */
Packit 33f14e
Packit 33f14e
/* The macros in this file implement forward iteration through a
Packit 33f14e
   multi-byte string, without knowing its length a-priori.
Packit 33f14e
Packit 33f14e
   With these macros, an iteration loop that looks like
Packit 33f14e
Packit 33f14e
      char *iter;
Packit 33f14e
      for (iter = buf; *iter != '\0'; iter++)
Packit 33f14e
        {
Packit 33f14e
          do_something (*iter);
Packit 33f14e
        }
Packit 33f14e
Packit 33f14e
   becomes
Packit 33f14e
Packit 33f14e
      mbui_iterator_t iter;
Packit 33f14e
      for (mbui_init (iter, buf); mbui_avail (iter); mbui_advance (iter))
Packit 33f14e
        {
Packit 33f14e
          do_something (mbui_cur_ptr (iter), mb_len (mbui_cur (iter)));
Packit 33f14e
        }
Packit 33f14e
Packit 33f14e
   The benefit of these macros over plain use of mbrtowc is:
Packit 33f14e
   - Handling of invalid multibyte sequences is possible without
Packit 33f14e
     making the code more complicated, while still preserving the
Packit 33f14e
     invalid multibyte sequences.
Packit 33f14e
Packit 33f14e
   Compared to mbiter.h, the macros here don't need to know the string's
Packit 33f14e
   length a-priori.  The downside is that at each step, the look-ahead
Packit 33f14e
   that guards against overrunning the terminating '\0' is more expensive.
Packit 33f14e
   The mbui_* macros are therefore suitable when there is a high probability
Packit 33f14e
   that only the first few multibyte characters need to be inspected.
Packit 33f14e
   Whereas the mbi_* macros are better if usually the iteration runs
Packit 33f14e
   through the entire string.
Packit 33f14e
Packit 33f14e
   mbui_iterator_t
Packit 33f14e
     is a type usable for variable declarations.
Packit 33f14e
Packit 33f14e
   mbui_init (iter, startptr)
Packit 33f14e
     initializes the iterator, starting at startptr.
Packit 33f14e
Packit 33f14e
   mbui_avail (iter)
Packit 33f14e
     returns true if there are more multibyte characters available before
Packit 33f14e
     the end of string is reached. In this case, mbui_cur (iter) is
Packit 33f14e
     initialized to the next multibyte character.
Packit 33f14e
Packit 33f14e
   mbui_advance (iter)
Packit 33f14e
     advances the iterator by one multibyte character.
Packit 33f14e
Packit 33f14e
   mbui_cur (iter)
Packit 33f14e
     returns the current multibyte character, of type mbchar_t.  All the
Packit 33f14e
     macros defined in mbchar.h can be used on it.
Packit 33f14e
Packit 33f14e
   mbui_cur_ptr (iter)
Packit 33f14e
     return a pointer to the beginning of the current multibyte character.
Packit 33f14e
Packit 33f14e
   mbui_reloc (iter, ptrdiff)
Packit 33f14e
     relocates iterator when the string is moved by ptrdiff bytes.
Packit 33f14e
Packit 33f14e
   mbui_copy (&destiter, &srciter)
Packit 33f14e
     copies srciter to destiter.
Packit 33f14e
Packit 33f14e
   Here are the function prototypes of the macros.
Packit 33f14e
Packit 33f14e
   extern void          mbui_init (mbui_iterator_t iter, const char *startptr);
Packit 33f14e
   extern bool          mbui_avail (mbui_iterator_t iter);
Packit 33f14e
   extern void          mbui_advance (mbui_iterator_t iter);
Packit 33f14e
   extern mbchar_t      mbui_cur (mbui_iterator_t iter);
Packit 33f14e
   extern const char *  mbui_cur_ptr (mbui_iterator_t iter);
Packit 33f14e
   extern void          mbui_reloc (mbui_iterator_t iter, ptrdiff_t ptrdiff);
Packit 33f14e
   extern void          mbui_copy (mbui_iterator_t *new, const mbui_iterator_t *old);
Packit 33f14e
 */
Packit 33f14e
Packit 33f14e
#ifndef _MBUITER_H
Packit 33f14e
#define _MBUITER_H 1
Packit 33f14e
Packit 33f14e
#include <assert.h>
Packit 33f14e
#include <stdbool.h>
Packit 33f14e
#include <stddef.h>
Packit 33f14e
#include <stdlib.h>
Packit 33f14e
#include <string.h>
Packit 33f14e
Packit 33f14e
/* Tru64 with Desktop Toolkit C has a bug: <stdio.h> must be included before
Packit 33f14e
   <wchar.h>.
Packit 33f14e
   BSD/OS 4.1 has a bug: <stdio.h> and <time.h> must be included before
Packit 33f14e
   <wchar.h>.  */
Packit 33f14e
#include <stdio.h>
Packit 33f14e
#include <time.h>
Packit 33f14e
#include <wchar.h>
Packit 33f14e
Packit 33f14e
#include "mbchar.h"
Packit 33f14e
#include "strnlen1.h"
Packit 33f14e
Packit 33f14e
#ifndef _GL_INLINE_HEADER_BEGIN
Packit 33f14e
 #error "Please include config.h first."
Packit 33f14e
#endif
Packit 33f14e
_GL_INLINE_HEADER_BEGIN
Packit 33f14e
#ifndef MBUITER_INLINE
Packit 33f14e
# define MBUITER_INLINE _GL_INLINE
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
struct mbuiter_multi
Packit 33f14e
{
Packit 33f14e
  bool in_shift;        /* true if next byte may not be interpreted as ASCII */
Packit 33f14e
  mbstate_t state;      /* if in_shift: current shift state */
Packit 33f14e
  bool next_done;       /* true if mbui_avail has already filled the following */
Packit 33f14e
  struct mbchar cur;    /* the current character:
Packit 33f14e
        const char *cur.ptr             pointer to current character
Packit 33f14e
        The following are only valid after mbui_avail.
Packit 33f14e
        size_t cur.bytes                number of bytes of current character
Packit 33f14e
        bool cur.wc_valid               true if wc is a valid wide character
Packit 33f14e
        wchar_t cur.wc                  if wc_valid: the current character
Packit 33f14e
        */
Packit 33f14e
};
Packit 33f14e
Packit 33f14e
MBUITER_INLINE void
Packit 33f14e
mbuiter_multi_next (struct mbuiter_multi *iter)
Packit 33f14e
{
Packit 33f14e
  if (iter->next_done)
Packit 33f14e
    return;
Packit 33f14e
  if (iter->in_shift)
Packit 33f14e
    goto with_shift;
Packit 33f14e
  /* Handle most ASCII characters quickly, without calling mbrtowc().  */
Packit 33f14e
  if (is_basic (*iter->cur.ptr))
Packit 33f14e
    {
Packit 33f14e
      /* These characters are part of the basic character set.  ISO C 99
Packit 33f14e
         guarantees that their wide character code is identical to their
Packit 33f14e
         char code.  */
Packit 33f14e
      iter->cur.bytes = 1;
Packit 33f14e
      iter->cur.wc = *iter->cur.ptr;
Packit 33f14e
      iter->cur.wc_valid = true;
Packit 33f14e
    }
Packit 33f14e
  else
Packit 33f14e
    {
Packit 33f14e
      assert (mbsinit (&iter->state));
Packit 33f14e
      iter->in_shift = true;
Packit 33f14e
    with_shift:
Packit 33f14e
      iter->cur.bytes = mbrtowc (&iter->cur.wc, iter->cur.ptr,
Packit 33f14e
                                 strnlen1 (iter->cur.ptr, MB_CUR_MAX),
Packit 33f14e
                                 &iter->state);
Packit 33f14e
      if (iter->cur.bytes == (size_t) -1)
Packit 33f14e
        {
Packit 33f14e
          /* An invalid multibyte sequence was encountered.  */
Packit 33f14e
          iter->cur.bytes = 1;
Packit 33f14e
          iter->cur.wc_valid = false;
Packit 33f14e
          /* Whether to set iter->in_shift = false and reset iter->state
Packit 33f14e
             or not is not very important; the string is bogus anyway.  */
Packit 33f14e
        }
Packit 33f14e
      else if (iter->cur.bytes == (size_t) -2)
Packit 33f14e
        {
Packit 33f14e
          /* An incomplete multibyte character at the end.  */
Packit 33f14e
          iter->cur.bytes = strlen (iter->cur.ptr);
Packit 33f14e
          iter->cur.wc_valid = false;
Packit 33f14e
          /* Whether to set iter->in_shift = false and reset iter->state
Packit 33f14e
             or not is not important; the string end is reached anyway.  */
Packit 33f14e
        }
Packit 33f14e
      else
Packit 33f14e
        {
Packit 33f14e
          if (iter->cur.bytes == 0)
Packit 33f14e
            {
Packit 33f14e
              /* A null wide character was encountered.  */
Packit 33f14e
              iter->cur.bytes = 1;
Packit 33f14e
              assert (*iter->cur.ptr == '\0');
Packit 33f14e
              assert (iter->cur.wc == 0);
Packit 33f14e
            }
Packit 33f14e
          iter->cur.wc_valid = true;
Packit 33f14e
Packit 33f14e
          /* When in the initial state, we can go back treating ASCII
Packit 33f14e
             characters more quickly.  */
Packit 33f14e
          if (mbsinit (&iter->state))
Packit 33f14e
            iter->in_shift = false;
Packit 33f14e
        }
Packit 33f14e
    }
Packit 33f14e
  iter->next_done = true;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
MBUITER_INLINE void
Packit 33f14e
mbuiter_multi_reloc (struct mbuiter_multi *iter, ptrdiff_t ptrdiff)
Packit 33f14e
{
Packit 33f14e
  iter->cur.ptr += ptrdiff;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
MBUITER_INLINE void
Packit 33f14e
mbuiter_multi_copy (struct mbuiter_multi *new_iter, const struct mbuiter_multi *old_iter)
Packit 33f14e
{
Packit 33f14e
  if ((new_iter->in_shift = old_iter->in_shift))
Packit 33f14e
    memcpy (&new_iter->state, &old_iter->state, sizeof (mbstate_t));
Packit 33f14e
  else
Packit 33f14e
    memset (&new_iter->state, 0, sizeof (mbstate_t));
Packit 33f14e
  new_iter->next_done = old_iter->next_done;
Packit 33f14e
  mb_copy (&new_iter->cur, &old_iter->cur);
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
/* Iteration macros.  */
Packit 33f14e
typedef struct mbuiter_multi mbui_iterator_t;
Packit 33f14e
#define mbui_init(iter, startptr) \
Packit 33f14e
  ((iter).cur.ptr = (startptr), \
Packit 33f14e
   (iter).in_shift = false, memset (&(iter).state, '\0', sizeof (mbstate_t)), \
Packit 33f14e
   (iter).next_done = false)
Packit 33f14e
#define mbui_avail(iter) \
Packit 33f14e
  (mbuiter_multi_next (&(iter)), !mb_isnul ((iter).cur))
Packit 33f14e
#define mbui_advance(iter) \
Packit 33f14e
  ((iter).cur.ptr += (iter).cur.bytes, (iter).next_done = false)
Packit 33f14e
Packit 33f14e
/* Access to the current character.  */
Packit 33f14e
#define mbui_cur(iter) (iter).cur
Packit 33f14e
#define mbui_cur_ptr(iter) (iter).cur.ptr
Packit 33f14e
Packit 33f14e
/* Relocation.  */
Packit 33f14e
#define mbui_reloc(iter, ptrdiff) mbuiter_multi_reloc (&iter, ptrdiff)
Packit 33f14e
Packit 33f14e
/* Copying an iterator.  */
Packit 33f14e
#define mbui_copy mbuiter_multi_copy
Packit 33f14e
Packit 33f14e
_GL_INLINE_HEADER_END
Packit 33f14e
Packit 33f14e
#endif /* _MBUITER_H */