mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-22 21:10:07 +00:00
721314c980
As indicated in a recent thread, this it is a simple brute-force algorithm that checks the whole needle at a matching character pair (and does so 1 byte at a time after the first 64 bytes of a needle). Also it never skips ahead and thus can match at every haystack position after trying to match all of the needle, which generic implementation avoids. As indicated by Wilco, a 4x larger needle and 16x larger haystack gives a clear 65x slowdown both basic_strstr and __strstr_avx512: "ifuncs": ["basic_strstr", "twoway_strstr", "__strstr_avx512", "__strstr_sse2_unaligned", "__strstr_generic"], { "len_haystack": 65536, "len_needle": 1024, "align_haystack": 0, "align_needle": 0, "fail": 1, "desc": "Difficult bruteforce needle", "timings": [4.0948e+07, 15094.5, 3.20818e+07, 108558, 10839.2] }, { "len_haystack": 1048576, "len_needle": 4096, "align_haystack": 0, "align_needle": 0, "fail": 1, "desc": "Difficult bruteforce needle", "timings": [2.69767e+09, 100797, 2.08535e+09, 495706, 82666.9] } PS: I don't have an AVX512 capable machine to verify this issues, but skimming through the code it does seems to follow what Wilco has described. Reviewed-by: Noah Goldstein <goldstein.w.n@gmail.com>
50 lines
1.8 KiB
C
50 lines
1.8 KiB
C
/* Multiple versions of strstr.
|
|
All versions must be listed in ifunc-impl-list.c.
|
|
Copyright (C) 2012-2024 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
The GNU C Library 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
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with the GNU C Library; if not, see
|
|
<https://www.gnu.org/licenses/>. */
|
|
|
|
/* Redefine strstr so that the compiler won't complain about the type
|
|
mismatch with the IFUNC selector in strong_alias, below. */
|
|
#undef strstr
|
|
#define strstr __redirect_strstr
|
|
#include <string.h>
|
|
#undef strstr
|
|
|
|
#define STRSTR __strstr_generic
|
|
#ifdef SHARED
|
|
# undef libc_hidden_builtin_def
|
|
# define libc_hidden_builtin_def(name) \
|
|
__hidden_ver1 (__strstr_generic, __GI_strstr, __strstr_generic);
|
|
#endif
|
|
|
|
#include "string/strstr.c"
|
|
|
|
extern __typeof (__redirect_strstr) __strstr_sse2_unaligned attribute_hidden;
|
|
extern __typeof (__redirect_strstr) __strstr_generic attribute_hidden;
|
|
|
|
#include "init-arch.h"
|
|
|
|
/* Avoid DWARF definition DIE on ifunc symbol so that GDB can handle
|
|
ifunc symbol properly. */
|
|
extern __typeof (__redirect_strstr) __libc_strstr;
|
|
libc_ifunc (__libc_strstr,
|
|
HAS_ARCH_FEATURE (Fast_Unaligned_Load)
|
|
? __strstr_sse2_unaligned
|
|
: __strstr_generic)
|
|
#undef strstr
|
|
strong_alias (__libc_strstr, strstr)
|