Blame string/strverscmp.c

Packit Service 82fcde
/* Compare strings while treating digits characters numerically.
Packit Service 82fcde
   Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Jean-François Bignolles <bignolle@ecoledoc.ibp.fr>, 1997.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <stdint.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
#include <ctype.h>
Packit Service 82fcde
Packit Service 82fcde
/* states: S_N: normal, S_I: comparing integral part, S_F: comparing
Packit Service 82fcde
           fractionnal parts, S_Z: idem but with leading Zeroes only */
Packit Service 82fcde
#define  S_N    0x0
Packit Service 82fcde
#define  S_I    0x3
Packit Service 82fcde
#define  S_F    0x6
Packit Service 82fcde
#define  S_Z    0x9
Packit Service 82fcde
Packit Service 82fcde
/* result_type: CMP: return diff; LEN: compare using len_diff/diff */
Packit Service 82fcde
#define  CMP    2
Packit Service 82fcde
#define  LEN    3
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* Compare S1 and S2 as strings holding indices/version numbers,
Packit Service 82fcde
   returning less than, equal to or greater than zero if S1 is less than,
Packit Service 82fcde
   equal to or greater than S2 (for more info, see the texinfo doc).
Packit Service 82fcde
*/
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
__strverscmp (const char *s1, const char *s2)
Packit Service 82fcde
{
Packit Service 82fcde
  const unsigned char *p1 = (const unsigned char *) s1;
Packit Service 82fcde
  const unsigned char *p2 = (const unsigned char *) s2;
Packit Service 82fcde
Packit Service 82fcde
  /* Symbol(s)    0       [1-9]   others
Packit Service 82fcde
     Transition   (10) 0  (01) d  (00) x   */
Packit Service 82fcde
  static const uint8_t next_state[] =
Packit Service 82fcde
  {
Packit Service 82fcde
      /* state    x    d    0  */
Packit Service 82fcde
      /* S_N */  S_N, S_I, S_Z,
Packit Service 82fcde
      /* S_I */  S_N, S_I, S_I,
Packit Service 82fcde
      /* S_F */  S_N, S_F, S_F,
Packit Service 82fcde
      /* S_Z */  S_N, S_F, S_Z
Packit Service 82fcde
  };
Packit Service 82fcde
Packit Service 82fcde
  static const int8_t result_type[] =
Packit Service 82fcde
  {
Packit Service 82fcde
      /* state   x/x  x/d  x/0  d/x  d/d  d/0  0/x  0/d  0/0  */
Packit Service 82fcde
Packit Service 82fcde
      /* S_N */  CMP, CMP, CMP, CMP, LEN, CMP, CMP, CMP, CMP,
Packit Service 82fcde
      /* S_I */  CMP, -1,  -1,  +1,  LEN, LEN, +1,  LEN, LEN,
Packit Service 82fcde
      /* S_F */  CMP, CMP, CMP, CMP, CMP, CMP, CMP, CMP, CMP,
Packit Service 82fcde
      /* S_Z */  CMP, +1,  +1,  -1,  CMP, CMP, -1,  CMP, CMP
Packit Service 82fcde
  };
Packit Service 82fcde
Packit Service 82fcde
  if (p1 == p2)
Packit Service 82fcde
    return 0;
Packit Service 82fcde
Packit Service 82fcde
  unsigned char c1 = *p1++;
Packit Service 82fcde
  unsigned char c2 = *p2++;
Packit Service 82fcde
  /* Hint: '0' is a digit too.  */
Packit Service 82fcde
  int state = S_N + ((c1 == '0') + (isdigit (c1) != 0));
Packit Service 82fcde
Packit Service 82fcde
  int diff;
Packit Service 82fcde
  while ((diff = c1 - c2) == 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (c1 == '\0')
Packit Service 82fcde
	return diff;
Packit Service 82fcde
Packit Service 82fcde
      state = next_state[state];
Packit Service 82fcde
      c1 = *p1++;
Packit Service 82fcde
      c2 = *p2++;
Packit Service 82fcde
      state += (c1 == '0') + (isdigit (c1) != 0);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  state = result_type[state * 3 + (((c2 == '0') + (isdigit (c2) != 0)))];
Packit Service 82fcde
Packit Service 82fcde
  switch (state)
Packit Service 82fcde
  {
Packit Service 82fcde
    case CMP:
Packit Service 82fcde
      return diff;
Packit Service 82fcde
Packit Service 82fcde
    case LEN:
Packit Service 82fcde
      while (isdigit (*p1++))
Packit Service 82fcde
	if (!isdigit (*p2++))
Packit Service 82fcde
	  return 1;
Packit Service 82fcde
Packit Service 82fcde
      return isdigit (*p2) ? -1 : diff;
Packit Service 82fcde
Packit Service 82fcde
    default:
Packit Service 82fcde
      return state;
Packit Service 82fcde
  }
Packit Service 82fcde
}
Packit Service 82fcde
libc_hidden_def (__strverscmp)
Packit Service 82fcde
weak_alias (__strverscmp, strverscmp)