Blame lib/mbiter.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.
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 < buf + buflen; iter++)
Packit 33f14e
        {
Packit 33f14e
          do_something (*iter);
Packit 33f14e
        }
Packit 33f14e
Packit 33f14e
   becomes
Packit 33f14e
Packit 33f14e
      mbi_iterator_t iter;
Packit 33f14e
      for (mbi_init (iter, buf, buflen); mbi_avail (iter); mbi_advance (iter))
Packit 33f14e
        {
Packit 33f14e
          do_something (mbi_cur_ptr (iter), mb_len (mbi_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
   mbi_iterator_t
Packit 33f14e
     is a type usable for variable declarations.
Packit 33f14e
Packit 33f14e
   mbi_init (iter, startptr, length)
Packit 33f14e
     initializes the iterator, starting at startptr and crossing length bytes.
Packit 33f14e
Packit 33f14e
   mbi_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, mbi_cur (iter) is
Packit 33f14e
     initialized to the next multibyte character.
Packit 33f14e
Packit 33f14e
   mbi_advance (iter)
Packit 33f14e
     advances the iterator by one multibyte character.
Packit 33f14e
Packit 33f14e
   mbi_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
   mbi_cur_ptr (iter)
Packit 33f14e
     return a pointer to the beginning of the current multibyte character.
Packit 33f14e
Packit 33f14e
   mbi_reloc (iter, ptrdiff)
Packit 33f14e
     relocates iterator when the string is moved by ptrdiff bytes.
Packit 33f14e
Packit 33f14e
   mbi_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          mbi_init (mbi_iterator_t iter,
Packit 33f14e
                                  const char *startptr, size_t length);
Packit 33f14e
   extern bool          mbi_avail (mbi_iterator_t iter);
Packit 33f14e
   extern void          mbi_advance (mbi_iterator_t iter);
Packit 33f14e
   extern mbchar_t      mbi_cur (mbi_iterator_t iter);
Packit 33f14e
   extern const char *  mbi_cur_ptr (mbi_iterator_t iter);
Packit 33f14e
   extern void          mbi_reloc (mbi_iterator_t iter, ptrdiff_t ptrdiff);
Packit 33f14e
   extern void          mbi_copy (mbi_iterator_t *new, const mbi_iterator_t *old);
Packit 33f14e
 */
Packit 33f14e
Packit 33f14e
#ifndef _MBITER_H
Packit 33f14e
#define _MBITER_H 1
Packit 33f14e
Packit 33f14e
#include <assert.h>
Packit 33f14e
#include <stdbool.h>
Packit 33f14e
#include <stddef.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
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 MBITER_INLINE
Packit 33f14e
# define MBITER_INLINE _GL_INLINE
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
struct mbiter_multi
Packit 33f14e
{
Packit 33f14e
  const char *limit;    /* pointer to end of string */
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 mbi_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 mbi_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
MBITER_INLINE void
Packit 33f14e
mbiter_multi_next (struct mbiter_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
                                 iter->limit - iter->cur.ptr, &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 = iter->limit - 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
MBITER_INLINE void
Packit 33f14e
mbiter_multi_reloc (struct mbiter_multi *iter, ptrdiff_t ptrdiff)
Packit 33f14e
{
Packit 33f14e
  iter->cur.ptr += ptrdiff;
Packit 33f14e
  iter->limit += ptrdiff;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
MBITER_INLINE void
Packit 33f14e
mbiter_multi_copy (struct mbiter_multi *new_iter, const struct mbiter_multi *old_iter)
Packit 33f14e
{
Packit 33f14e
  new_iter->limit = old_iter->limit;
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 mbiter_multi mbi_iterator_t;
Packit 33f14e
#define mbi_init(iter, startptr, length) \
Packit 33f14e
  ((iter).cur.ptr = (startptr), (iter).limit = (iter).cur.ptr + (length), \
Packit 33f14e
   (iter).in_shift = false, memset (&(iter).state, '\0', sizeof (mbstate_t)), \
Packit 33f14e
   (iter).next_done = false)
Packit 33f14e
#define mbi_avail(iter) \
Packit 33f14e
  ((iter).cur.ptr < (iter).limit && (mbiter_multi_next (&(iter)), true))
Packit 33f14e
#define mbi_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 mbi_cur(iter) (iter).cur
Packit 33f14e
#define mbi_cur_ptr(iter) (iter).cur.ptr
Packit 33f14e
Packit 33f14e
/* Relocation.  */
Packit 33f14e
#define mbi_reloc(iter, ptrdiff) mbiter_multi_reloc (&iter, ptrdiff)
Packit 33f14e
Packit 33f14e
/* Copying an iterator.  */
Packit 33f14e
#define mbi_copy mbiter_multi_copy
Packit 33f14e
Packit 33f14e
_GL_INLINE_HEADER_END
Packit 33f14e
Packit 33f14e
#endif /* _MBITER_H */