Blame gnulib/lib/strstr.c

Packit 06dd63
/* Copyright (C) 1991-1994, 1996-1998, 2000, 2004, 2007-2019 Free Software
Packit 06dd63
   Foundation, Inc.
Packit 06dd63
   This file is part of the GNU C Library.
Packit 06dd63
Packit 06dd63
   This program is free software; you can redistribute it and/or modify
Packit 06dd63
   it under the terms of the GNU Lesser General Public License as published by
Packit 06dd63
   the Free Software Foundation; either version 2.1, or (at your option)
Packit 06dd63
   any later version.
Packit 06dd63
Packit 06dd63
   This program is distributed in the hope that it will be useful,
Packit 06dd63
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 06dd63
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 06dd63
   GNU Lesser General Public License for more details.
Packit 06dd63
Packit 06dd63
   You should have received a copy of the GNU Lesser General Public License along
Packit 06dd63
   with this program; if not, see <https://www.gnu.org/licenses/>.  */
Packit 06dd63
Packit 06dd63
/* This particular implementation was written by Eric Blake, 2008.  */
Packit 06dd63
Packit 06dd63
#ifndef _LIBC
Packit 06dd63
# include <config.h>
Packit 06dd63
#endif
Packit 06dd63
Packit 06dd63
/* Specification of strstr.  */
Packit 06dd63
#include <string.h>
Packit 06dd63
Packit 06dd63
#include <stdbool.h>
Packit 06dd63
Packit 06dd63
#define RETURN_TYPE char *
Packit 06dd63
#define AVAILABLE(h, h_l, j, n_l)                       \
Packit 06dd63
  (!memchr ((h) + (h_l), '\0', (j) + (n_l) - (h_l))     \
Packit 06dd63
   && ((h_l) = (j) + (n_l)))
Packit 06dd63
#include "str-two-way.h"
Packit 06dd63
Packit 06dd63
/* Return the first occurrence of NEEDLE in HAYSTACK.  Return HAYSTACK
Packit 06dd63
   if NEEDLE is empty, otherwise NULL if NEEDLE is not found in
Packit 06dd63
   HAYSTACK.  */
Packit 06dd63
char *
Packit 06dd63
strstr (const char *haystack_start, const char *needle_start)
Packit 06dd63
{
Packit 06dd63
  const char *haystack = haystack_start;
Packit 06dd63
  const char *needle = needle_start;
Packit 06dd63
  size_t needle_len; /* Length of NEEDLE.  */
Packit 06dd63
  size_t haystack_len; /* Known minimum length of HAYSTACK.  */
Packit 06dd63
  bool ok = true; /* True if NEEDLE is prefix of HAYSTACK.  */
Packit 06dd63
Packit 06dd63
  /* Determine length of NEEDLE, and in the process, make sure
Packit 06dd63
     HAYSTACK is at least as long (no point processing all of a long
Packit 06dd63
     NEEDLE if HAYSTACK is too short).  */
Packit 06dd63
  while (*haystack && *needle)
Packit 06dd63
    ok &= *haystack++ == *needle++;
Packit 06dd63
  if (*needle)
Packit 06dd63
    return NULL;
Packit 06dd63
  if (ok)
Packit 06dd63
    return (char *) haystack_start;
Packit 06dd63
Packit 06dd63
  /* Reduce the size of haystack using strchr, since it has a smaller
Packit 06dd63
     linear coefficient than the Two-Way algorithm.  */
Packit 06dd63
  needle_len = needle - needle_start;
Packit 06dd63
  haystack = strchr (haystack_start + 1, *needle_start);
Packit 06dd63
  if (!haystack || __builtin_expect (needle_len == 1, 0))
Packit 06dd63
    return (char *) haystack;
Packit 06dd63
  needle -= needle_len;
Packit 06dd63
  haystack_len = (haystack > haystack_start + needle_len ? 1
Packit 06dd63
                  : needle_len + haystack_start - haystack);
Packit 06dd63
Packit 06dd63
  /* Perform the search.  Abstract memory is considered to be an array
Packit 06dd63
     of 'unsigned char' values, not an array of 'char' values.  See
Packit 06dd63
     ISO C 99 section 6.2.6.1.  */
Packit 06dd63
  if (needle_len < LONG_NEEDLE_THRESHOLD)
Packit 06dd63
    return two_way_short_needle ((const unsigned char *) haystack,
Packit 06dd63
                                 haystack_len,
Packit 06dd63
                                 (const unsigned char *) needle, needle_len);
Packit 06dd63
  return two_way_long_needle ((const unsigned char *) haystack, haystack_len,
Packit 06dd63
                              (const unsigned char *) needle, needle_len);
Packit 06dd63
}
Packit 06dd63
Packit 06dd63
#undef LONG_NEEDLE_THRESHOLD