Blame gnulib-tests/test-strstr.c

Packit 709fb3
/*
Packit 709fb3
 * Copyright (C) 2004, 2007-2017 Free Software Foundation, Inc.
Packit 709fb3
 * Written by Bruno Haible and Eric Blake
Packit 709fb3
 *
Packit 709fb3
 * This program is free software: you can redistribute it and/or modify
Packit 709fb3
 * it under the terms of the GNU General Public License as published by
Packit 709fb3
 * the Free Software Foundation; either version 3 of the License, or
Packit 709fb3
 * (at your option) any later version.
Packit 709fb3
 *
Packit 709fb3
 * This program is distributed in the hope that it will be useful,
Packit 709fb3
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 709fb3
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 709fb3
 * GNU General Public License for more details.
Packit 709fb3
 *
Packit 709fb3
 * You should have received a copy of the GNU General Public License
Packit 709fb3
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 709fb3
Packit 709fb3
#include <config.h>
Packit 709fb3
Packit 709fb3
#include <string.h>
Packit 709fb3
Packit 709fb3
#include "signature.h"
Packit 709fb3
SIGNATURE_CHECK (strstr, char *, (char const *, char const *));
Packit 709fb3
Packit 709fb3
#include <signal.h>
Packit 709fb3
#include <stdlib.h>
Packit 709fb3
#include <unistd.h>
Packit 709fb3
Packit 709fb3
#include "zerosize-ptr.h"
Packit 709fb3
#include "macros.h"
Packit 709fb3
Packit 709fb3
int
Packit 709fb3
main (int argc, char *argv[])
Packit 709fb3
{
Packit 709fb3
#if HAVE_DECL_ALARM
Packit 709fb3
  /* Declare failure if test takes too long, by using default abort
Packit 709fb3
     caused by SIGALRM.  All known platforms that lack alarm also have
Packit 709fb3
     a quadratic strstr, and the replacement strstr is known to not
Packit 709fb3
     take too long.  */
Packit 709fb3
  int alarm_value = 50;
Packit 709fb3
  signal (SIGALRM, SIG_DFL);
Packit 709fb3
  alarm (alarm_value);
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "foo";
Packit 709fb3
    const char *result = strstr (input, "");
Packit 709fb3
    ASSERT (result == input);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "foo";
Packit 709fb3
    const char *result = strstr (input, "o");
Packit 709fb3
    ASSERT (result == input + 1);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    /* On some platforms, the memchr() functions reads past the first
Packit 709fb3
       occurrence of the byte to be searched, leading to an out-of-bounds
Packit 709fb3
       read access for strstr().
Packit 709fb3
       See <http://bugs.debian.org/cgi-bin/bugreport.cgi?bug=521737>.
Packit 709fb3
       This is a bug in memchr(), see the Austin Group's clarification
Packit 709fb3
       <http://www.opengroup.org/austin/docs/austin_454.txt>.  */
Packit 709fb3
    const char *fix = "aBaaaaaaaaaaax";
Packit 709fb3
    char *page_boundary = (char *) zerosize_ptr ();
Packit 709fb3
    size_t len = strlen (fix) + 1;
Packit 709fb3
    char *input = page_boundary ? page_boundary - len : malloc (len);
Packit 709fb3
    const char *result;
Packit 709fb3
Packit 709fb3
    strcpy (input, fix);
Packit 709fb3
    result = strstr (input, "B1x");
Packit 709fb3
    ASSERT (result == NULL);
Packit 709fb3
    if (!page_boundary)
Packit 709fb3
      free (input);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "ABC ABCDAB ABCDABCDABDE";
Packit 709fb3
    const char *result = strstr (input, "ABCDABD");
Packit 709fb3
    ASSERT (result == input + 15);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "ABC ABCDAB ABCDABCDABDE";
Packit 709fb3
    const char *result = strstr (input, "ABCDABE");
Packit 709fb3
    ASSERT (result == NULL);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "ABC ABCDAB ABCDABCDABDE";
Packit 709fb3
    const char *result = strstr (input, "ABCDABCD");
Packit 709fb3
    ASSERT (result == input + 11);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  /* Check that a long periodic needle does not cause false positives.  */
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "F_BD_CE_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD"
Packit 709fb3
                         "_C3_88_20_EF_BF_BD_EF_BF_BD_EF_BF_BD"
Packit 709fb3
                         "_C3_A7_20_EF_BF_BD";
Packit 709fb3
    const char need[] = "_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD";
Packit 709fb3
    const char *result = strstr (input, need);
Packit 709fb3
    ASSERT (result == NULL);
Packit 709fb3
  }
Packit 709fb3
  {
Packit 709fb3
    const char input[] = "F_BD_CE_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD"
Packit 709fb3
                         "_C3_88_20_EF_BF_BD_EF_BF_BD_EF_BF_BD"
Packit 709fb3
                         "_C3_A7_20_EF_BF_BD_DA_B5_C2_A6_20"
Packit 709fb3
                         "_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD";
Packit 709fb3
    const char need[] = "_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD_EF_BF_BD";
Packit 709fb3
    const char *result = strstr (input, need);
Packit 709fb3
    ASSERT (result == input + 115);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  /* Check that a very long haystack is handled quickly if the needle is
Packit 709fb3
     short and occurs near the beginning.  */
Packit 709fb3
  {
Packit 709fb3
    size_t repeat = 10000;
Packit 709fb3
    size_t m = 1000000;
Packit 709fb3
    const char *needle =
Packit 709fb3
      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
Packit 709fb3
      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA";
Packit 709fb3
    char *haystack = (char *) malloc (m + 1);
Packit 709fb3
    if (haystack != NULL)
Packit 709fb3
      {
Packit 709fb3
        memset (haystack, 'A', m);
Packit 709fb3
        haystack[0] = 'B';
Packit 709fb3
        haystack[m] = '\0';
Packit 709fb3
Packit 709fb3
        for (; repeat > 0; repeat--)
Packit 709fb3
          {
Packit 709fb3
            ASSERT (strstr (haystack, needle) == haystack + 1);
Packit 709fb3
          }
Packit 709fb3
Packit 709fb3
        free (haystack);
Packit 709fb3
      }
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  /* Check that a very long needle is discarded quickly if the haystack is
Packit 709fb3
     short.  */
Packit 709fb3
  {
Packit 709fb3
    size_t repeat = 10000;
Packit 709fb3
    size_t m = 1000000;
Packit 709fb3
    const char *haystack =
Packit 709fb3
      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
Packit 709fb3
      "ABABABABABABABABABABABABABABABABABABABABABABABABABABABABABABABABABAB";
Packit 709fb3
    char *needle = (char *) malloc (m + 1);
Packit 709fb3
    if (needle != NULL)
Packit 709fb3
      {
Packit 709fb3
        memset (needle, 'A', m);
Packit 709fb3
        needle[m] = '\0';
Packit 709fb3
Packit 709fb3
        for (; repeat > 0; repeat--)
Packit 709fb3
          {
Packit 709fb3
            ASSERT (strstr (haystack, needle) == NULL);
Packit 709fb3
          }
Packit 709fb3
Packit 709fb3
        free (needle);
Packit 709fb3
      }
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  /* Check that the asymptotic worst-case complexity is not quadratic.  */
Packit 709fb3
  {
Packit 709fb3
    size_t m = 1000000;
Packit 709fb3
    char *haystack = (char *) malloc (2 * m + 2);
Packit 709fb3
    char *needle = (char *) malloc (m + 2);
Packit 709fb3
    if (haystack != NULL && needle != NULL)
Packit 709fb3
      {
Packit 709fb3
        const char *result;
Packit 709fb3
Packit 709fb3
        memset (haystack, 'A', 2 * m);
Packit 709fb3
        haystack[2 * m] = 'B';
Packit 709fb3
        haystack[2 * m + 1] = '\0';
Packit 709fb3
Packit 709fb3
        memset (needle, 'A', m);
Packit 709fb3
        needle[m] = 'B';
Packit 709fb3
        needle[m + 1] = '\0';
Packit 709fb3
Packit 709fb3
        result = strstr (haystack, needle);
Packit 709fb3
        ASSERT (result == haystack + m);
Packit 709fb3
      }
Packit 709fb3
    free (needle);
Packit 709fb3
    free (haystack);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  /* Sublinear speed is only possible in memmem; strstr must examine
Packit 709fb3
     every character of haystack to find its length.  */
Packit 709fb3
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    /* Ensure that with a barely periodic "short" needle, strstr's
Packit 709fb3
       search does not mistakenly skip just past the match point.
Packit 709fb3
       This use of strstr would mistakenly return NULL before
Packit 709fb3
       gnulib v0.0-4927.  */
Packit 709fb3
    const char *haystack =
Packit 709fb3
      "\n"
Packit 709fb3
      "with_build_libsubdir\n"
Packit 709fb3
      "with_local_prefix\n"
Packit 709fb3
      "with_gxx_include_dir\n"
Packit 709fb3
      "with_cpp_install_dir\n"
Packit 709fb3
      "enable_generated_files_in_srcdir\n"
Packit 709fb3
      "with_gnu_ld\n"
Packit 709fb3
      "with_ld\n"
Packit 709fb3
      "with_demangler_in_ld\n"
Packit 709fb3
      "with_gnu_as\n"
Packit 709fb3
      "with_as\n"
Packit 709fb3
      "enable_largefile\n"
Packit 709fb3
      "enable_werror_always\n"
Packit 709fb3
      "enable_checking\n"
Packit 709fb3
      "enable_coverage\n"
Packit 709fb3
      "enable_gather_detailed_mem_stats\n"
Packit 709fb3
      "enable_build_with_cxx\n"
Packit 709fb3
      "with_stabs\n"
Packit 709fb3
      "enable_multilib\n"
Packit 709fb3
      "enable___cxa_atexit\n"
Packit 709fb3
      "enable_decimal_float\n"
Packit 709fb3
      "enable_fixed_point\n"
Packit 709fb3
      "enable_threads\n"
Packit 709fb3
      "enable_tls\n"
Packit 709fb3
      "enable_objc_gc\n"
Packit 709fb3
      "with_dwarf2\n"
Packit 709fb3
      "enable_shared\n"
Packit 709fb3
      "with_build_sysroot\n"
Packit 709fb3
      "with_sysroot\n"
Packit 709fb3
      "with_specs\n"
Packit 709fb3
      "with_pkgversion\n"
Packit 709fb3
      "with_bugurl\n"
Packit 709fb3
      "enable_languages\n"
Packit 709fb3
      "with_multilib_list\n";
Packit 709fb3
    const char *needle = "\n"
Packit 709fb3
      "with_gnu_ld\n";
Packit 709fb3
    const char* p = strstr (haystack, needle);
Packit 709fb3
    ASSERT (p - haystack == 114);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    /* Same bug, shorter trigger.  */
Packit 709fb3
    const char *haystack = "..wi.d.";
Packit 709fb3
    const char *needle = ".d.";
Packit 709fb3
    const char* p = strstr (haystack, needle);
Packit 709fb3
    ASSERT (p - haystack == 4);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  {
Packit 709fb3
    /* Like the above, but trigger the flaw in two_way_long_needle
Packit 709fb3
       by using a needle of length LONG_NEEDLE_THRESHOLD (32) or greater.
Packit 709fb3
       Rather than trying to find the right alignment manually, I've
Packit 709fb3
       arbitrarily chosen the following needle and template for the
Packit 709fb3
       haystack, and ensure that for each placement of the needle in
Packit 709fb3
       that haystack, strstr finds it.  */
Packit 709fb3
    const char *needle = "\nwith_gnu_ld-extend-to-len-32-b\n";
Packit 709fb3
    const char *h =
Packit 709fb3
      "\n"
Packit 709fb3
      "with_build_libsubdir\n"
Packit 709fb3
      "with_local_prefix\n"
Packit 709fb3
      "with_gxx_include_dir\n"
Packit 709fb3
      "with_cpp_install_dir\n"
Packit 709fb3
      "with_e_\n"
Packit 709fb3
      "..............................\n"
Packit 709fb3
      "with_FGHIJKLMNOPQRSTUVWXYZ\n"
Packit 709fb3
      "with_567890123456789\n"
Packit 709fb3
      "with_multilib_list\n";
Packit 709fb3
    size_t h_len = strlen (h);
Packit 709fb3
    char *haystack = malloc (h_len + 1);
Packit 709fb3
    size_t i;
Packit 709fb3
    ASSERT (haystack);
Packit 709fb3
    for (i = 0; i < h_len - strlen (needle); i++)
Packit 709fb3
      {
Packit 709fb3
        const char *p;
Packit 709fb3
        memcpy (haystack, h, h_len + 1);
Packit 709fb3
        memcpy (haystack + i, needle, strlen (needle) + 1);
Packit 709fb3
        p = strstr (haystack, needle);
Packit 709fb3
        ASSERT (p);
Packit 709fb3
        ASSERT (p - haystack == i);
Packit 709fb3
      }
Packit 709fb3
    free (haystack);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  return 0;
Packit 709fb3
}