Blame lib/mbscasecmp.c

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