Blame lib/mbuiter.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, without knowing its length a-priori.
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 != '\0'; 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
      mbui_iterator_t iter;
Packit Service fdd496
      for (mbui_init (iter, buf); mbui_avail (iter); mbui_advance (iter))
Packit Service fdd496
        {
Packit Service fdd496
          do_something (mbui_cur_ptr (iter), mb_len (mbui_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
   Compared to mbiter.h, the macros here don't need to know the string's
Packit Service fdd496
   length a-priori.  The downside is that at each step, the look-ahead
Packit Service fdd496
   that guards against overrunning the terminating '\0' is more expensive.
Packit Service fdd496
   The mbui_* macros are therefore suitable when there is a high probability
Packit Service fdd496
   that only the first few multibyte characters need to be inspected.
Packit Service fdd496
   Whereas the mbi_* macros are better if usually the iteration runs
Packit Service fdd496
   through the entire string.
Packit Service fdd496
Packit Service fdd496
   mbui_iterator_t
Packit Service fdd496
     is a type usable for variable declarations.
Packit Service fdd496
Packit Service fdd496
   mbui_init (iter, startptr)
Packit Service fdd496
     initializes the iterator, starting at startptr.
Packit Service fdd496
Packit Service fdd496
   mbui_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, mbui_cur (iter) is
Packit Service fdd496
     initialized to the next multibyte character.
Packit Service fdd496
Packit Service fdd496
   mbui_advance (iter)
Packit Service fdd496
     advances the iterator by one multibyte character.
Packit Service fdd496
Packit Service fdd496
   mbui_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
   mbui_cur_ptr (iter)
Packit Service fdd496
     return a pointer to the beginning of the current multibyte character.
Packit Service fdd496
Packit Service fdd496
   mbui_reloc (iter, ptrdiff)
Packit Service fdd496
     relocates iterator when the string is moved by ptrdiff bytes.
Packit Service fdd496
Packit Service fdd496
   mbui_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          mbui_init (mbui_iterator_t iter, const char *startptr);
Packit Service fdd496
   extern bool          mbui_avail (mbui_iterator_t iter);
Packit Service fdd496
   extern void          mbui_advance (mbui_iterator_t iter);
Packit Service fdd496
   extern mbchar_t      mbui_cur (mbui_iterator_t iter);
Packit Service fdd496
   extern const char *  mbui_cur_ptr (mbui_iterator_t iter);
Packit Service fdd496
   extern void          mbui_reloc (mbui_iterator_t iter, ptrdiff_t ptrdiff);
Packit Service fdd496
   extern void          mbui_copy (mbui_iterator_t *new, const mbui_iterator_t *old);
Packit Service fdd496
 */
Packit Service fdd496
Packit Service fdd496
#ifndef _MBUITER_H
Packit Service fdd496
#define _MBUITER_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 <stdlib.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
#include "strnlen1.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 MBUITER_INLINE
Packit Service fdd496
# define MBUITER_INLINE _GL_INLINE
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
struct mbuiter_multi
Packit Service fdd496
{
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 mbui_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 mbui_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
MBUITER_INLINE void
Packit Service fdd496
mbuiter_multi_next (struct mbuiter_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
                                 strnlen1 (iter->cur.ptr, MB_CUR_MAX),
Packit Service fdd496
                                 &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 = strlen (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
MBUITER_INLINE void
Packit Service fdd496
mbuiter_multi_reloc (struct mbuiter_multi *iter, ptrdiff_t ptrdiff)
Packit Service fdd496
{
Packit Service fdd496
  iter->cur.ptr += ptrdiff;
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
MBUITER_INLINE void
Packit Service fdd496
mbuiter_multi_copy (struct mbuiter_multi *new_iter, const struct mbuiter_multi *old_iter)
Packit Service fdd496
{
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 mbuiter_multi mbui_iterator_t;
Packit Service fdd496
#define mbui_init(iter, startptr) \
Packit Service fdd496
  ((iter).cur.ptr = (startptr), \
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 mbui_avail(iter) \
Packit Service fdd496
  (mbuiter_multi_next (&(iter)), !mb_isnul ((iter).cur))
Packit Service fdd496
#define mbui_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 mbui_cur(iter) (iter).cur
Packit Service fdd496
#define mbui_cur_ptr(iter) (iter).cur.ptr
Packit Service fdd496
Packit Service fdd496
/* Relocation.  */
Packit Service fdd496
#define mbui_reloc(iter, ptrdiff) mbuiter_multi_reloc (&iter, ptrdiff)
Packit Service fdd496
Packit Service fdd496
/* Copying an iterator.  */
Packit Service fdd496
#define mbui_copy mbuiter_multi_copy
Packit Service fdd496
Packit Service fdd496
_GL_INLINE_HEADER_END
Packit Service fdd496
Packit Service fdd496
#endif /* _MBUITER_H */