/* Test of uN_strstr() functions. Copyright (C) 2004, 2007-2023 Free Software Foundation, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ static void test_u_strstr (void) { { const UNIT input[] = { 'f', 'o', 'o', 0 }; const UNIT needle[] = { 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == input); } { const UNIT input[] = { 'f', 'o', 'o', 0 }; const UNIT needle[] = { 'o', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == input + 1); } { const UNIT input[] = { 'A', 'B', 'C', ' ', 'A', 'B', 'C', 'D', 'A', 'B', ' ', 'A', 'B', 'C', 'D', 'A', 'B', 'C', 'D', 'A', 'B', 'D', 'E', 0 }; const UNIT needle[] = { 'A', 'B', 'C', 'D', 'A', 'B', 'D', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == input + 15); } { const UNIT input[] = { 'A', 'B', 'C', ' ', 'A', 'B', 'C', 'D', 'A', 'B', ' ', 'A', 'B', 'C', 'D', 'A', 'B', 'C', 'D', 'A', 'B', 'D', 'E', 0 }; const UNIT needle[] = { 'A', 'B', 'C', 'D', 'A', 'B', 'E', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == NULL); } { const UNIT input[] = { 'A', 'B', 'C', ' ', 'A', 'B', 'C', 'D', 'A', 'B', ' ', 'A', 'B', 'C', 'D', 'A', 'B', 'C', 'D', 'A', 'B', 'D', 'E', 0 }; const UNIT needle[] = { 'A', 'B', 'C', 'D', 'A', 'B', 'C', 'D', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == input + 11); } /* Check that a long periodic needle does not cause false positives. */ { const UNIT input[] = { 'F', '_', 'B', 'D', '_', 'C', 'E', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'C', '3', '_', '8', '8', '_', '2', '0', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'C', '3', '_', 'A', '7', '_', '2', '0', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', 0 }; const UNIT needle[] = { '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == NULL); } { const UNIT input[] = { 'F', '_', 'B', 'D', '_', 'C', 'E', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'C', '3', '_', '8', '8', '_', '2', '0', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'C', '3', '_', 'A', '7', '_', '2', '0', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'D', 'A', '_', 'B', '5', '_', 'C', '2', '_', 'A', '6', '_', '2', '0', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', 0 }; const UNIT needle[] = { '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', '_', 'E', 'F', '_', 'B', 'F', '_', 'B', 'D', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == input + 115); } /* Check that a very long haystack is handled quickly if the needle is short and occurs near the beginning. */ { size_t repeat = 10000; size_t m = 1000000; const UNIT needle[] = { 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 0 }; UNIT *haystack = (UNIT *) malloc ((m + 1) * sizeof (UNIT)); if (haystack != NULL) { size_t i; haystack[0] = 'B'; for (i = 1; i < m; i++) haystack[i] = 'A'; haystack[m] = '\0'; for (; repeat > 0; repeat--) { ASSERT (U_STRSTR (haystack, needle) == haystack + 1); } free (haystack); } } /* Check that a very long needle is discarded quickly if the haystack is short. */ { size_t repeat = 10000; size_t m = 1000000; const UNIT haystack[] = { 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 'A', 'B', 0 }; UNIT *needle = (UNIT *) malloc ((m + 1) * sizeof (UNIT)); if (needle != NULL) { size_t i; for (i = 0; i < m; i++) needle[i] = 'A'; needle[m] = '\0'; for (; repeat > 0; repeat--) { ASSERT (U_STRSTR (haystack, needle) == NULL); } free (needle); } } /* Check that the asymptotic worst-case complexity is not quadratic. */ { size_t m = 1000000; UNIT *haystack = (UNIT *) malloc ((2 * m + 2) * sizeof (UNIT)); UNIT *needle = (UNIT *) malloc ((m + 2) * sizeof (UNIT)); if (haystack != NULL && needle != NULL) { size_t i; const UNIT *result; for (i = 0; i < 2 * m; i++) haystack[i] = 'A'; haystack[2 * m] = 'B'; haystack[2 * m + 1] = 0; for (i = 0; i < m; i++) needle[i] = 'A'; needle[m] = 'B'; needle[m + 1] = 0; result = U_STRSTR (haystack, needle); ASSERT (result == haystack + m); } free (needle); free (haystack); } /* Test case from Yves Bastide. */ { const UNIT input[] = { 'p', 'l', 'a', 'y', 'i', 'n', 'g', ' ', 'p', 'l', 'a', 'y', ' ', 'p', 'l', 'a', 'y', ' ', 'p', 'l', 'a', 'y', ' ', 'a', 'l', 'w', 'a', 'y', 's', 0 }; const UNIT needle[] = { 'p', 'l', 'a', 'y', ' ', 'p', 'l', 'a', 'y', ' ', 'p', 'l', 'a', 'y', 0 }; const UNIT *result = U_STRSTR (input, needle); ASSERT (result == input + 8); } /* Test long needles. */ { size_t m = 1024; UNIT *haystack = (UNIT *) malloc ((2 * m + 1) * sizeof (UNIT)); UNIT *needle = (UNIT *) malloc ((m + 1) * sizeof (UNIT)); if (haystack != NULL && needle != NULL) { const UNIT *p; haystack[0] = 'x'; U_SET (haystack + 1, ' ', m - 1); U_SET (haystack + m, 'x', m); haystack[2 * m] = '\0'; U_SET (needle, 'x', m); needle[m] = '\0'; p = U_STRSTR (haystack, needle); ASSERT (p); ASSERT (p - haystack == m); } free (needle); free (haystack); } }