Blame lib/mbscasecmp.c

Packit Service fdd496
/* Case-insensitive string comparison function.
Packit Service fdd496
   Copyright (C) 1998-1999, 2005-2017 Free Software Foundation, Inc.
Packit Service fdd496
   Written by Bruno Haible <bruno@clisp.org>, 2005,
Packit Service fdd496
   based on earlier glibc code.
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
#include <config.h>
Packit Service fdd496
Packit Service fdd496
/* Specification.  */
Packit Service fdd496
#include <string.h>
Packit Service fdd496
Packit Service fdd496
#include <ctype.h>
Packit Service fdd496
#include <limits.h>
Packit Service fdd496
Packit Service fdd496
#include "mbuiter.h"
Packit Service fdd496
Packit Service fdd496
#define TOLOWER(Ch) (isupper (Ch) ? tolower (Ch) : (Ch))
Packit Service fdd496
Packit Service fdd496
/* Compare the character strings S1 and S2, ignoring case, returning less than,
Packit Service fdd496
   equal to or greater than zero if S1 is lexicographically less than, equal to
Packit Service fdd496
   or greater than S2.
Packit Service fdd496
   Note: This function may, in multibyte locales, return 0 for strings of
Packit Service fdd496
   different lengths!  */
Packit Service fdd496
int
Packit Service fdd496
mbscasecmp (const char *s1, const char *s2)
Packit Service fdd496
{
Packit Service fdd496
  if (s1 == s2)
Packit Service fdd496
    return 0;
Packit Service fdd496
Packit Service fdd496
  /* Be careful not to look at the entire extent of s1 or s2 until needed.
Packit Service fdd496
     This is useful because when two strings differ, the difference is
Packit Service fdd496
     most often already in the very few first characters.  */
Packit Service fdd496
  if (MB_CUR_MAX > 1)
Packit Service fdd496
    {
Packit Service fdd496
      mbui_iterator_t iter1;
Packit Service fdd496
      mbui_iterator_t iter2;
Packit Service fdd496
Packit Service fdd496
      mbui_init (iter1, s1);
Packit Service fdd496
      mbui_init (iter2, s2);
Packit Service fdd496
Packit Service fdd496
      while (mbui_avail (iter1) && mbui_avail (iter2))
Packit Service fdd496
        {
Packit Service fdd496
          int cmp = mb_casecmp (mbui_cur (iter1), mbui_cur (iter2));
Packit Service fdd496
Packit Service fdd496
          if (cmp != 0)
Packit Service fdd496
            return cmp;
Packit Service fdd496
Packit Service fdd496
          mbui_advance (iter1);
Packit Service fdd496
          mbui_advance (iter2);
Packit Service fdd496
        }
Packit Service fdd496
      if (mbui_avail (iter1))
Packit Service fdd496
        /* s2 terminated before s1.  */
Packit Service fdd496
        return 1;
Packit Service fdd496
      if (mbui_avail (iter2))
Packit Service fdd496
        /* s1 terminated before s2.  */
Packit Service fdd496
        return -1;
Packit Service fdd496
      return 0;
Packit Service fdd496
    }
Packit Service fdd496
  else
Packit Service fdd496
    {
Packit Service fdd496
      const unsigned char *p1 = (const unsigned char *) s1;
Packit Service fdd496
      const unsigned char *p2 = (const unsigned char *) s2;
Packit Service fdd496
      unsigned char c1, c2;
Packit Service fdd496
Packit Service fdd496
      do
Packit Service fdd496
        {
Packit Service fdd496
          c1 = TOLOWER (*p1);
Packit Service fdd496
          c2 = TOLOWER (*p2);
Packit Service fdd496
Packit Service fdd496
          if (c1 == '\0')
Packit Service fdd496
            break;
Packit Service fdd496
Packit Service fdd496
          ++p1;
Packit Service fdd496
          ++p2;
Packit Service fdd496
        }
Packit Service fdd496
      while (c1 == c2);
Packit Service fdd496
Packit Service fdd496
      if (UCHAR_MAX <= INT_MAX)
Packit Service fdd496
        return c1 - c2;
Packit Service fdd496
      else
Packit Service fdd496
        /* On machines where 'char' and 'int' are types of the same size, the
Packit Service fdd496
           difference of two 'unsigned char' values - including the sign bit -
Packit Service fdd496
           doesn't fit in an 'int'.  */
Packit Service fdd496
        return (c1 > c2 ? 1 : c1 < c2 ? -1 : 0);
Packit Service fdd496
    }
Packit Service fdd496
}