]> sourceware.org Git - glibc.git/commitdiff
x86: Add support for compiling {raw|w}memchr with high ISA level
authorNoah Goldstein <goldstein.w.n@gmail.com>
Wed, 22 Jun 2022 23:51:20 +0000 (16:51 -0700)
committerNoah Goldstein <goldstein.w.n@gmail.com>
Thu, 23 Jun 2022 02:41:35 +0000 (19:41 -0700)
1. Refactor files so that all implementations for in the multiarch
   directory.
    - Essentially moved sse2 {raw|w}memchr.S implementation to
      multiarch/{raw|w}memchr-sse2.S

    - The non-multiarch {raw|w}memchr.S file now only includes one of
      the implementations in the multiarch directory based on the
      compiled ISA level (only used for non-multiarch builds.
      Otherwise we go through the ifunc selector).

2. Add ISA level build guards to different implementations.
    - I.e memchr-avx2.S which is ISA level 3 will only build if
      compiled ISA level <= 3. Otherwise there is no reason to include
      it as we will always use one of the ISA level 4
      implementations (memchr-evex{-rtm}.S).

3. Add new multiarch/rtld-{raw}memchr.S that just include the
   non-multiarch {raw}memchr.S which will in turn select the best
   implementation based on the compiled ISA level.

4. Refactor the ifunc selector and ifunc implementation list to use
   the ISA level aware wrapper macros that allow functions below the
   compiled ISA level (with a guranteed replacement) to be skipped.
    - Guranteed replacement essentially means that for any ISA level
      build there must be a function that the baseline of the ISA
      supports. So for {raw|w}memchr.S since there is not ISA level 2
      function, the ISA level 2 build still includes the ISA level
      1 (sse2) function. Once we reach the ISA level 3 build, however,
      {raw|w}memchr-avx2{-rtm}.S will always be sufficient so the ISA
      level 1 implementation ({raw|w}memchr-sse2.S) will not be built.

Tested with and without multiarch on x86_64 for ISA levels:
{generic, x86-64-v2, x86-64-v3, x86-64-v4}

And m32 with and without multiarch.

17 files changed:
sysdeps/x86_64/isa-default-impl.h
sysdeps/x86_64/memchr.S
sysdeps/x86_64/multiarch/ifunc-evex.h
sysdeps/x86_64/multiarch/ifunc-impl-list.c
sysdeps/x86_64/multiarch/memchr-avx2.S
sysdeps/x86_64/multiarch/memchr-evex.S
sysdeps/x86_64/multiarch/memchr-sse2.S
sysdeps/x86_64/multiarch/rawmemchr-avx2.S
sysdeps/x86_64/multiarch/rawmemchr-evex.S
sysdeps/x86_64/multiarch/rawmemchr-sse2.S
sysdeps/x86_64/multiarch/rtld-memchr.S [new file with mode: 0644]
sysdeps/x86_64/multiarch/rtld-rawmemchr.S [new file with mode: 0644]
sysdeps/x86_64/multiarch/wmemchr-avx2.S
sysdeps/x86_64/multiarch/wmemchr-evex.S
sysdeps/x86_64/multiarch/wmemchr-sse2.S
sysdeps/x86_64/rawmemchr.S
sysdeps/x86_64/wmemchr.S [new file with mode: 0644]

index 34634668e56669c90d311537881cbb4b61cbc723..7d7832b1f53125fcf9df5a19bf21342a8f29886e 100644 (file)
 # error "Unsupported ISA Level!"
 #endif
 
+#if IS_IN(rtld)
+# if !defined USE_MULTIARCH
+#  error "RTLD version should only exist in multiarch build"
+# endif
+#else
+# if defined USE_MULTIARCH
+#  error "Multiarch build should not use ISA_DEFAULT_IMPL without RTLD"
+# endif
+#endif
+
 #include ISA_DEFAULT_IMPL
index a160fd9b00ccc5c206e16daf6b09ce20f7c9cf22..20b43508c4559b51abb90ca1c9e1acb16056aada 100644 (file)
    License along with the GNU C Library; if not, see
    <https://www.gnu.org/licenses/>.  */
 
-#include <sysdep.h>
+#define MEMCHR __memchr
 
-#ifdef USE_AS_WMEMCHR
-# define MEMCHR                wmemchr
-# define PCMPEQ                pcmpeqd
-# define CHAR_PER_VEC  4
-#else
-# define MEMCHR                memchr
-# define PCMPEQ                pcmpeqb
-# define CHAR_PER_VEC  16
-#endif
+#define DEFAULT_IMPL_V1        "multiarch/memchr-sse2.S"
+#define DEFAULT_IMPL_V3        "multiarch/memchr-avx2.S"
+#define DEFAULT_IMPL_V4        "multiarch/memchr-evex.S"
 
-/* fast SSE2 version with using pmaxub and 64 byte loop */
+#include "isa-default-impl.h"
 
-       .text
-ENTRY(MEMCHR)
-       movd    %esi, %xmm1
-       mov     %edi, %ecx
-
-#ifdef __ILP32__
-       /* Clear the upper 32 bits.  */
-       movl    %edx, %edx
-#endif
-#ifdef USE_AS_WMEMCHR
-       test    %RDX_LP, %RDX_LP
-       jz      L(return_null)
-#else
-       punpcklbw %xmm1, %xmm1
-       test    %RDX_LP, %RDX_LP
-       jz      L(return_null)
-       punpcklbw %xmm1, %xmm1
-#endif
-
-       and     $63, %ecx
-       pshufd  $0, %xmm1, %xmm1
-
-       cmp     $48, %ecx
-       ja      L(crosscache)
-
-       movdqu  (%rdi), %xmm0
-       PCMPEQ  %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-
-       jnz     L(matches_1)
-       sub     $CHAR_PER_VEC, %rdx
-       jbe     L(return_null)
-       add     $16, %rdi
-       and     $15, %ecx
-       and     $-16, %rdi
-#ifdef USE_AS_WMEMCHR
-       shr     $2, %ecx
-#endif
-       add     %rcx, %rdx
-       sub     $(CHAR_PER_VEC * 4), %rdx
-       jbe     L(exit_loop)
-       jmp     L(loop_prolog)
-
-       .p2align 4
-L(crosscache):
-       and     $15, %ecx
-       and     $-16, %rdi
-       movdqa  (%rdi), %xmm0
-
-       PCMPEQ  %xmm1, %xmm0
-       /* Check if there is a match.  */
-       pmovmskb %xmm0, %eax
-       /* Remove the leading bytes.  */
-       sar     %cl, %eax
-       test    %eax, %eax
-       je      L(unaligned_no_match)
-       /* Check which byte is a match.  */
-       bsf     %eax, %eax
-#ifdef USE_AS_WMEMCHR
-       mov     %eax, %esi
-       shr     $2, %esi
-       sub     %rsi, %rdx
-#else
-       sub     %rax, %rdx
-#endif
-       jbe     L(return_null)
-       add     %rdi, %rax
-       add     %rcx, %rax
-       ret
-
-       .p2align 4
-L(unaligned_no_match):
-       /* "rcx" is less than 16.  Calculate "rdx + rcx - 16" by using
-          "rdx - (16 - rcx)" instead of "(rdx + rcx) - 16" to void
-          possible addition overflow.  */
-       neg     %rcx
-       add     $16, %rcx
-#ifdef USE_AS_WMEMCHR
-       shr     $2, %ecx
-#endif
-       sub     %rcx, %rdx
-       jbe     L(return_null)
-       add     $16, %rdi
-       sub     $(CHAR_PER_VEC * 4), %rdx
-       jbe     L(exit_loop)
-
-       .p2align 4
-L(loop_prolog):
-       movdqa  (%rdi), %xmm0
-       PCMPEQ  %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       movdqa  16(%rdi), %xmm2
-       PCMPEQ  %xmm1, %xmm2
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       PCMPEQ  %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32)
-
-       movdqa  48(%rdi), %xmm4
-       PCMPEQ  %xmm1, %xmm4
-       add     $64, %rdi
-       pmovmskb %xmm4, %eax
-       test    %eax, %eax
-       jnz     L(matches0)
-
-       test    $0x3f, %rdi
-       jz      L(align64_loop)
-
-       sub     $(CHAR_PER_VEC * 4), %rdx
-       jbe     L(exit_loop)
-
-       movdqa  (%rdi), %xmm0
-       PCMPEQ  %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       movdqa  16(%rdi), %xmm2
-       PCMPEQ  %xmm1, %xmm2
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       PCMPEQ  %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32)
-
-       movdqa  48(%rdi), %xmm3
-       PCMPEQ  %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-
-       add     $64, %rdi
-       test    %eax, %eax
-       jnz     L(matches0)
-
-       mov     %rdi, %rcx
-       and     $-64, %rdi
-       and     $63, %ecx
-#ifdef USE_AS_WMEMCHR
-       shr     $2, %ecx
-#endif
-       add     %rcx, %rdx
-
-       .p2align 4
-L(align64_loop):
-       sub     $(CHAR_PER_VEC * 4), %rdx
-       jbe     L(exit_loop)
-       movdqa  (%rdi), %xmm0
-       movdqa  16(%rdi), %xmm2
-       movdqa  32(%rdi), %xmm3
-       movdqa  48(%rdi), %xmm4
-
-       PCMPEQ  %xmm1, %xmm0
-       PCMPEQ  %xmm1, %xmm2
-       PCMPEQ  %xmm1, %xmm3
-       PCMPEQ  %xmm1, %xmm4
-
-       pmaxub  %xmm0, %xmm3
-       pmaxub  %xmm2, %xmm4
-       pmaxub  %xmm3, %xmm4
-       pmovmskb %xmm4, %eax
-
-       add     $64, %rdi
-
-       test    %eax, %eax
-       jz      L(align64_loop)
-
-       sub     $64, %rdi
-
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       PCMPEQ  %xmm1, %xmm3
-
-       PCMPEQ  48(%rdi), %xmm1
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32)
-
-       pmovmskb %xmm1, %eax
-       bsf     %eax, %eax
-       lea     48(%rdi, %rax), %rax
-       ret
-
-       .p2align 4
-L(exit_loop):
-       add     $(CHAR_PER_VEC * 2), %edx
-       jle     L(exit_loop_32)
-
-       movdqa  (%rdi), %xmm0
-       PCMPEQ  %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       movdqa  16(%rdi), %xmm2
-       PCMPEQ  %xmm1, %xmm2
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       PCMPEQ  %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32_1)
-       sub     $CHAR_PER_VEC, %edx
-       jle     L(return_null)
-
-       PCMPEQ  48(%rdi), %xmm1
-       pmovmskb %xmm1, %eax
-       test    %eax, %eax
-       jnz     L(matches48_1)
-       xor     %eax, %eax
-       ret
-
-       .p2align 4
-L(exit_loop_32):
-       add     $(CHAR_PER_VEC * 2), %edx
-       movdqa  (%rdi), %xmm0
-       PCMPEQ  %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches_1)
-       sub     $CHAR_PER_VEC, %edx
-       jbe     L(return_null)
-
-       PCMPEQ  16(%rdi), %xmm1
-       pmovmskb %xmm1, %eax
-       test    %eax, %eax
-       jnz     L(matches16_1)
-       xor     %eax, %eax
-       ret
-
-       .p2align 4
-L(matches0):
-       bsf     %eax, %eax
-       lea     -16(%rax, %rdi), %rax
-       ret
-
-       .p2align 4
-L(matches):
-       bsf     %eax, %eax
-       add     %rdi, %rax
-       ret
-
-       .p2align 4
-L(matches16):
-       bsf     %eax, %eax
-       lea     16(%rax, %rdi), %rax
-       ret
-
-       .p2align 4
-L(matches32):
-       bsf     %eax, %eax
-       lea     32(%rax, %rdi), %rax
-       ret
-
-       .p2align 4
-L(matches_1):
-       bsf     %eax, %eax
-#ifdef USE_AS_WMEMCHR
-       mov     %eax, %esi
-       shr     $2, %esi
-       sub     %rsi, %rdx
-#else
-       sub     %rax, %rdx
-#endif
-       jbe     L(return_null)
-       add     %rdi, %rax
-       ret
-
-       .p2align 4
-L(matches16_1):
-       bsf     %eax, %eax
-#ifdef USE_AS_WMEMCHR
-       mov     %eax, %esi
-       shr     $2, %esi
-       sub     %rsi, %rdx
-#else
-       sub     %rax, %rdx
-#endif
-       jbe     L(return_null)
-       lea     16(%rdi, %rax), %rax
-       ret
-
-       .p2align 4
-L(matches32_1):
-       bsf     %eax, %eax
-#ifdef USE_AS_WMEMCHR
-       mov     %eax, %esi
-       shr     $2, %esi
-       sub     %rsi, %rdx
-#else
-       sub     %rax, %rdx
-#endif
-       jbe     L(return_null)
-       lea     32(%rdi, %rax), %rax
-       ret
-
-       .p2align 4
-L(matches48_1):
-       bsf     %eax, %eax
-#ifdef USE_AS_WMEMCHR
-       mov     %eax, %esi
-       shr     $2, %esi
-       sub     %rsi, %rdx
-#else
-       sub     %rax, %rdx
-#endif
-       jbe     L(return_null)
-       lea     48(%rdi, %rax), %rax
-       ret
-
-       .p2align 4
-L(return_null):
-       xor     %eax, %eax
-       ret
-END(MEMCHR)
-
-#ifndef USE_AS_WMEMCHR
-strong_alias (memchr, __memchr)
+weak_alias (__memchr, memchr)
 libc_hidden_builtin_def(memchr)
-#endif
index b8f7a12ea2ea529b45ae96a9ceca6f3362c2deda..856c6261f88503da4fdac651a2babb12ce75d5fd 100644 (file)
 
 #include <init-arch.h>
 
-extern __typeof (REDIRECT_NAME) OPTIMIZE (sse2) attribute_hidden;
-extern __typeof (REDIRECT_NAME) OPTIMIZE (avx2) attribute_hidden;
-extern __typeof (REDIRECT_NAME) OPTIMIZE (avx2_rtm) attribute_hidden;
 extern __typeof (REDIRECT_NAME) OPTIMIZE (evex) attribute_hidden;
 extern __typeof (REDIRECT_NAME) OPTIMIZE (evex_rtm) attribute_hidden;
 
+extern __typeof (REDIRECT_NAME) OPTIMIZE (avx2) attribute_hidden;
+extern __typeof (REDIRECT_NAME) OPTIMIZE (avx2_rtm) attribute_hidden;
+
+extern __typeof (REDIRECT_NAME) OPTIMIZE (sse2) attribute_hidden;
 
 static inline void *
 IFUNC_SELECTOR (void)
 {
-  const struct cpu_features* cpu_features = __get_cpu_features ();
-
-  if (CPU_FEATURE_USABLE_P (cpu_features, AVX2)
-      && CPU_FEATURE_USABLE_P (cpu_features, BMI2)
-      && CPU_FEATURES_ARCH_P (cpu_features, AVX_Fast_Unaligned_Load))
+  const struct cpu_features *cpu_features = __get_cpu_features ();
+
+  /* NB: The X86_ISA_* feature check macros are evaluated at
+     compile time.  */
+  if (X86_ISA_CPU_FEATURE_USABLE_P (cpu_features, AVX2)
+      && X86_ISA_CPU_FEATURE_USABLE_P (cpu_features, BMI2)
+      && X86_ISA_CPU_FEATURES_ARCH_P (cpu_features,
+                                     AVX_Fast_Unaligned_Load))
     {
-      if (CPU_FEATURE_USABLE_P (cpu_features, AVX512VL)
-         && CPU_FEATURE_USABLE_P (cpu_features, AVX512BW))
+      if (X86_ISA_CPU_FEATURE_USABLE_P (cpu_features, AVX512VL)
+         && X86_ISA_CPU_FEATURE_USABLE_P (cpu_features, AVX512BW))
        {
          if (CPU_FEATURE_USABLE_P (cpu_features, RTM))
            return OPTIMIZE (evex_rtm);
@@ -47,9 +51,12 @@ IFUNC_SELECTOR (void)
       if (CPU_FEATURE_USABLE_P (cpu_features, RTM))
        return OPTIMIZE (avx2_rtm);
 
-      if (!CPU_FEATURES_ARCH_P (cpu_features, Prefer_No_VZEROUPPER))
+      if (X86_ISA_CPU_FEATURES_ARCH_P (cpu_features,
+                                      Prefer_No_VZEROUPPER))
        return OPTIMIZE (avx2);
     }
 
+  /* This is unreachable (compile time checked) if ISA level >= 3
+     so no need for a robust fallback here.  */
   return OPTIMIZE (sse2);
 }
index 883362f63d8f1cfe55a576cd6ece01a17b5b3071..bf52cf96d05856145e44df71cdaf0b9c5e6af2f6 100644 (file)
@@ -25,7 +25,8 @@
 
 /* Fill ARRAY of MAX elements with IFUNC implementations for function
    NAME supported on target machine and return the number of valid
-   entries.  */
+   entries.  Each set of implementations for a given function is sorted in
+   descending order by ISA level.  */
 
 size_t
 __libc_ifunc_impl_list (const char *name, struct libc_ifunc_impl *array,
@@ -53,24 +54,27 @@ __libc_ifunc_impl_list (const char *name, struct libc_ifunc_impl *array,
 
   /* Support sysdeps/x86_64/multiarch/memchr.c.  */
   IFUNC_IMPL (i, name, memchr,
-             IFUNC_IMPL_ADD (array, i, memchr,
-                             CPU_FEATURE_USABLE (AVX2),
-                             __memchr_avx2)
-             IFUNC_IMPL_ADD (array, i, memchr,
-                             (CPU_FEATURE_USABLE (AVX2)
-                              && CPU_FEATURE_USABLE (RTM)),
-                             __memchr_avx2_rtm)
-             IFUNC_IMPL_ADD (array, i, memchr,
+             X86_IFUNC_IMPL_ADD_V4 (array, i, memchr,
                              (CPU_FEATURE_USABLE (AVX512VL)
                               && CPU_FEATURE_USABLE (AVX512BW)
                               && CPU_FEATURE_USABLE (BMI2)),
                              __memchr_evex)
-             IFUNC_IMPL_ADD (array, i, memchr,
+             X86_IFUNC_IMPL_ADD_V4 (array, i, memchr,
                              (CPU_FEATURE_USABLE (AVX512VL)
                               && CPU_FEATURE_USABLE (AVX512BW)
                               && CPU_FEATURE_USABLE (BMI2)),
                              __memchr_evex_rtm)
-             IFUNC_IMPL_ADD (array, i, memchr, 1, __memchr_sse2))
+             X86_IFUNC_IMPL_ADD_V3 (array, i, memchr,
+                             CPU_FEATURE_USABLE (AVX2),
+                             __memchr_avx2)
+             X86_IFUNC_IMPL_ADD_V3 (array, i, memchr,
+                             (CPU_FEATURE_USABLE (AVX2)
+                              && CPU_FEATURE_USABLE (RTM)),
+                             __memchr_avx2_rtm)
+             /* Can be lowered to V1 if a V2 implementation is added.  */
+             X86_IFUNC_IMPL_ADD_V2 (array, i, memchr,
+                             1,
+                             __memchr_sse2))
 
   /* Support sysdeps/x86_64/multiarch/memcmp.c.  */
   IFUNC_IMPL (i, name, memcmp,
@@ -288,24 +292,27 @@ __libc_ifunc_impl_list (const char *name, struct libc_ifunc_impl *array,
 
   /* Support sysdeps/x86_64/multiarch/rawmemchr.c.  */
   IFUNC_IMPL (i, name, rawmemchr,
-             IFUNC_IMPL_ADD (array, i, rawmemchr,
-                             CPU_FEATURE_USABLE (AVX2),
-                             __rawmemchr_avx2)
-             IFUNC_IMPL_ADD (array, i, rawmemchr,
-                             (CPU_FEATURE_USABLE (AVX2)
-                              && CPU_FEATURE_USABLE (RTM)),
-                             __rawmemchr_avx2_rtm)
-             IFUNC_IMPL_ADD (array, i, rawmemchr,
+             X86_IFUNC_IMPL_ADD_V4 (array, i, rawmemchr,
                              (CPU_FEATURE_USABLE (AVX512VL)
                               && CPU_FEATURE_USABLE (AVX512BW)
                               && CPU_FEATURE_USABLE (BMI2)),
                              __rawmemchr_evex)
-             IFUNC_IMPL_ADD (array, i, rawmemchr,
+             X86_IFUNC_IMPL_ADD_V4 (array, i, rawmemchr,
                              (CPU_FEATURE_USABLE (AVX512VL)
                               && CPU_FEATURE_USABLE (AVX512BW)
                               && CPU_FEATURE_USABLE (BMI2)),
                              __rawmemchr_evex_rtm)
-             IFUNC_IMPL_ADD (array, i, rawmemchr, 1, __rawmemchr_sse2))
+             X86_IFUNC_IMPL_ADD_V3 (array, i, rawmemchr,
+                             CPU_FEATURE_USABLE (AVX2),
+                             __rawmemchr_avx2)
+             X86_IFUNC_IMPL_ADD_V3 (array, i, rawmemchr,
+                             (CPU_FEATURE_USABLE (AVX2)
+                              && CPU_FEATURE_USABLE (RTM)),
+                             __rawmemchr_avx2_rtm)
+             /* Can be lowered to V1 if a V2 implementation is added.  */
+             X86_IFUNC_IMPL_ADD_V2 (array, i, rawmemchr,
+                             1,
+                             __rawmemchr_sse2))
 
   /* Support sysdeps/x86_64/multiarch/strlen.c.  */
   IFUNC_IMPL (i, name, strlen,
@@ -748,24 +755,27 @@ __libc_ifunc_impl_list (const char *name, struct libc_ifunc_impl *array,
 
   /* Support sysdeps/x86_64/multiarch/wmemchr.c.  */
   IFUNC_IMPL (i, name, wmemchr,
-             IFUNC_IMPL_ADD (array, i, wmemchr,
-                             CPU_FEATURE_USABLE (AVX2),
-                             __wmemchr_avx2)
-             IFUNC_IMPL_ADD (array, i, wmemchr,
-                             (CPU_FEATURE_USABLE (AVX2)
-                              && CPU_FEATURE_USABLE (RTM)),
-                             __wmemchr_avx2_rtm)
-             IFUNC_IMPL_ADD (array, i, wmemchr,
+             X86_IFUNC_IMPL_ADD_V4 (array, i, wmemchr,
                              (CPU_FEATURE_USABLE (AVX512VL)
                               && CPU_FEATURE_USABLE (AVX512BW)
                               && CPU_FEATURE_USABLE (BMI2)),
                              __wmemchr_evex)
-             IFUNC_IMPL_ADD (array, i, wmemchr,
+             X86_IFUNC_IMPL_ADD_V4 (array, i, wmemchr,
                              (CPU_FEATURE_USABLE (AVX512VL)
                               && CPU_FEATURE_USABLE (AVX512BW)
                               && CPU_FEATURE_USABLE (BMI2)),
                              __wmemchr_evex_rtm)
-             IFUNC_IMPL_ADD (array, i, wmemchr, 1, __wmemchr_sse2))
+             X86_IFUNC_IMPL_ADD_V3 (array, i, wmemchr,
+                             CPU_FEATURE_USABLE (AVX2),
+                             __wmemchr_avx2)
+             X86_IFUNC_IMPL_ADD_V3 (array, i, wmemchr,
+                             (CPU_FEATURE_USABLE (AVX2)
+                              && CPU_FEATURE_USABLE (RTM)),
+                             __wmemchr_avx2_rtm)
+             /* Can be lowered to V1 if a V2 implementation is added.  */
+             X86_IFUNC_IMPL_ADD_V2 (array, i, wmemchr,
+                             1,
+                             __wmemchr_sse2))
 
   /* Support sysdeps/x86_64/multiarch/wmemcmp.c.  */
   IFUNC_IMPL (i, name, wmemcmp,
index c5a256eb37b3db7ad25ce4c63f5ccfe1d9361e35..39be5f7083f6fb95d788ddde050698d091d41e33 100644 (file)
    License along with the GNU C Library; if not, see
    <https://www.gnu.org/licenses/>.  */
 
-#if IS_IN (libc)
+#include <isa-level.h>
+#include <sysdep.h>
 
-# include <sysdep.h>
+#if ISA_SHOULD_BUILD (3)
 
 # ifndef MEMCHR
 #  define MEMCHR       __memchr_avx2
index 0fd11b7632b55fe6f033d827a7d0d3f07a8f3ec0..0dd4f1dcced85f3f5ed3ba14cb5924104bddca48 100644 (file)
    License along with the GNU C Library; if not, see
    <https://www.gnu.org/licenses/>.  */
 
-#if IS_IN (libc)
+#include <isa-level.h>
+#include <sysdep.h>
 
-# include <sysdep.h>
+#if ISA_SHOULD_BUILD (4)
 
 # ifndef MEMCHR
 #  define MEMCHR       __memchr_evex
index 2c6fdd41d6770a8666f3502ab25952d13cfbe4d3..8c561cd687e99698579868e2613c9663ea126d39 100644 (file)
    License along with the GNU C Library; if not, see
    <https://www.gnu.org/licenses/>.  */
 
-#if IS_IN (libc)
-# define memchr __memchr_sse2
+#include <isa-level.h>
+#include <sysdep.h>
 
-# undef strong_alias
-# define strong_alias(memchr, __memchr)
-# undef libc_hidden_builtin_def
-# define libc_hidden_builtin_def(memchr)
-#endif
+/* MINIMUM_X86_ISA_LEVEL <= 2 because there is no V2 implementation
+   so we need this to build for ISA V2 builds. */
+#if ISA_SHOULD_BUILD (2)
+
+# ifndef MEMCHR
+#  define MEMCHR       __memchr_sse2
+# endif
+# ifdef USE_AS_WMEMCHR
+#  define PCMPEQ               pcmpeqd
+#  define CHAR_PER_VEC 4
+# else
+#  define PCMPEQ               pcmpeqb
+#  define CHAR_PER_VEC 16
+# endif
+
+/* fast SSE2 version with using pmaxub and 64 byte loop */
+
+       .text
+ENTRY(MEMCHR)
+       movd    %esi, %xmm1
+       mov     %edi, %ecx
+
+# ifdef __ILP32__
+       /* Clear the upper 32 bits.  */
+       movl    %edx, %edx
+# endif
+# ifdef USE_AS_WMEMCHR
+       test    %RDX_LP, %RDX_LP
+       jz      L(return_null)
+# else
+       punpcklbw %xmm1, %xmm1
+       test    %RDX_LP, %RDX_LP
+       jz      L(return_null)
+       punpcklbw %xmm1, %xmm1
+# endif
+
+       and     $63, %ecx
+       pshufd  $0, %xmm1, %xmm1
+
+       cmp     $48, %ecx
+       ja      L(crosscache)
+
+       movdqu  (%rdi), %xmm0
+       PCMPEQ  %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+
+       jnz     L(matches_1)
+       sub     $CHAR_PER_VEC, %rdx
+       jbe     L(return_null)
+       add     $16, %rdi
+       and     $15, %ecx
+       and     $-16, %rdi
+# ifdef USE_AS_WMEMCHR
+       shr     $2, %ecx
+# endif
+       add     %rcx, %rdx
+       sub     $(CHAR_PER_VEC * 4), %rdx
+       jbe     L(exit_loop)
+       jmp     L(loop_prolog)
+
+       .p2align 4
+L(crosscache):
+       and     $15, %ecx
+       and     $-16, %rdi
+       movdqa  (%rdi), %xmm0
+
+       PCMPEQ  %xmm1, %xmm0
+       /* Check if there is a match.  */
+       pmovmskb %xmm0, %eax
+       /* Remove the leading bytes.  */
+       sar     %cl, %eax
+       test    %eax, %eax
+       je      L(unaligned_no_match)
+       /* Check which byte is a match.  */
+       bsf     %eax, %eax
+# ifdef USE_AS_WMEMCHR
+       mov     %eax, %esi
+       shr     $2, %esi
+       sub     %rsi, %rdx
+# else
+       sub     %rax, %rdx
+# endif
+       jbe     L(return_null)
+       add     %rdi, %rax
+       add     %rcx, %rax
+       ret
+
+       .p2align 4
+L(unaligned_no_match):
+       /* "rcx" is less than 16.  Calculate "rdx + rcx - 16" by using
+          "rdx - (16 - rcx)" instead of "(rdx + rcx) - 16" to void
+          possible addition overflow.  */
+       neg     %rcx
+       add     $16, %rcx
+# ifdef USE_AS_WMEMCHR
+       shr     $2, %ecx
+# endif
+       sub     %rcx, %rdx
+       jbe     L(return_null)
+       add     $16, %rdi
+       sub     $(CHAR_PER_VEC * 4), %rdx
+       jbe     L(exit_loop)
+
+       .p2align 4
+L(loop_prolog):
+       movdqa  (%rdi), %xmm0
+       PCMPEQ  %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       movdqa  16(%rdi), %xmm2
+       PCMPEQ  %xmm1, %xmm2
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
+
+       movdqa  32(%rdi), %xmm3
+       PCMPEQ  %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32)
+
+       movdqa  48(%rdi), %xmm4
+       PCMPEQ  %xmm1, %xmm4
+       add     $64, %rdi
+       pmovmskb %xmm4, %eax
+       test    %eax, %eax
+       jnz     L(matches0)
+
+       test    $0x3f, %rdi
+       jz      L(align64_loop)
+
+       sub     $(CHAR_PER_VEC * 4), %rdx
+       jbe     L(exit_loop)
+
+       movdqa  (%rdi), %xmm0
+       PCMPEQ  %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       movdqa  16(%rdi), %xmm2
+       PCMPEQ  %xmm1, %xmm2
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
+
+       movdqa  32(%rdi), %xmm3
+       PCMPEQ  %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32)
+
+       movdqa  48(%rdi), %xmm3
+       PCMPEQ  %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+
+       add     $64, %rdi
+       test    %eax, %eax
+       jnz     L(matches0)
+
+       mov     %rdi, %rcx
+       and     $-64, %rdi
+       and     $63, %ecx
+# ifdef USE_AS_WMEMCHR
+       shr     $2, %ecx
+# endif
+       add     %rcx, %rdx
+
+       .p2align 4
+L(align64_loop):
+       sub     $(CHAR_PER_VEC * 4), %rdx
+       jbe     L(exit_loop)
+       movdqa  (%rdi), %xmm0
+       movdqa  16(%rdi), %xmm2
+       movdqa  32(%rdi), %xmm3
+       movdqa  48(%rdi), %xmm4
+
+       PCMPEQ  %xmm1, %xmm0
+       PCMPEQ  %xmm1, %xmm2
+       PCMPEQ  %xmm1, %xmm3
+       PCMPEQ  %xmm1, %xmm4
 
-#include "../memchr.S"
+       pmaxub  %xmm0, %xmm3
+       pmaxub  %xmm2, %xmm4
+       pmaxub  %xmm3, %xmm4
+       pmovmskb %xmm4, %eax
+
+       add     $64, %rdi
+
+       test    %eax, %eax
+       jz      L(align64_loop)
+
+       sub     $64, %rdi
+
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
+
+       movdqa  32(%rdi), %xmm3
+       PCMPEQ  %xmm1, %xmm3
+
+       PCMPEQ  48(%rdi), %xmm1
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32)
+
+       pmovmskb %xmm1, %eax
+       bsf     %eax, %eax
+       lea     48(%rdi, %rax), %rax
+       ret
+
+       .p2align 4
+L(exit_loop):
+       add     $(CHAR_PER_VEC * 2), %edx
+       jle     L(exit_loop_32)
+
+       movdqa  (%rdi), %xmm0
+       PCMPEQ  %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       movdqa  16(%rdi), %xmm2
+       PCMPEQ  %xmm1, %xmm2
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
+
+       movdqa  32(%rdi), %xmm3
+       PCMPEQ  %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32_1)
+       sub     $CHAR_PER_VEC, %edx
+       jle     L(return_null)
+
+       PCMPEQ  48(%rdi), %xmm1
+       pmovmskb %xmm1, %eax
+       test    %eax, %eax
+       jnz     L(matches48_1)
+       xor     %eax, %eax
+       ret
+
+       .p2align 4
+L(exit_loop_32):
+       add     $(CHAR_PER_VEC * 2), %edx
+       movdqa  (%rdi), %xmm0
+       PCMPEQ  %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches_1)
+       sub     $CHAR_PER_VEC, %edx
+       jbe     L(return_null)
+
+       PCMPEQ  16(%rdi), %xmm1
+       pmovmskb %xmm1, %eax
+       test    %eax, %eax
+       jnz     L(matches16_1)
+       xor     %eax, %eax
+       ret
+
+       .p2align 4
+L(matches0):
+       bsf     %eax, %eax
+       lea     -16(%rax, %rdi), %rax
+       ret
+
+       .p2align 4
+L(matches):
+       bsf     %eax, %eax
+       add     %rdi, %rax
+       ret
+
+       .p2align 4
+L(matches16):
+       bsf     %eax, %eax
+       lea     16(%rax, %rdi), %rax
+       ret
+
+       .p2align 4
+L(matches32):
+       bsf     %eax, %eax
+       lea     32(%rax, %rdi), %rax
+       ret
+
+       .p2align 4
+L(matches_1):
+       bsf     %eax, %eax
+# ifdef USE_AS_WMEMCHR
+       mov     %eax, %esi
+       shr     $2, %esi
+       sub     %rsi, %rdx
+# else
+       sub     %rax, %rdx
+# endif
+       jbe     L(return_null)
+       add     %rdi, %rax
+       ret
+
+       .p2align 4
+L(matches16_1):
+       bsf     %eax, %eax
+# ifdef USE_AS_WMEMCHR
+       mov     %eax, %esi
+       shr     $2, %esi
+       sub     %rsi, %rdx
+# else
+       sub     %rax, %rdx
+# endif
+       jbe     L(return_null)
+       lea     16(%rdi, %rax), %rax
+       ret
+
+       .p2align 4
+L(matches32_1):
+       bsf     %eax, %eax
+# ifdef USE_AS_WMEMCHR
+       mov     %eax, %esi
+       shr     $2, %esi
+       sub     %rsi, %rdx
+# else
+       sub     %rax, %rdx
+# endif
+       jbe     L(return_null)
+       lea     32(%rdi, %rax), %rax
+       ret
+
+       .p2align 4
+L(matches48_1):
+       bsf     %eax, %eax
+# ifdef USE_AS_WMEMCHR
+       mov     %eax, %esi
+       shr     $2, %esi
+       sub     %rsi, %rdx
+# else
+       sub     %rax, %rdx
+# endif
+       jbe     L(return_null)
+       lea     48(%rdi, %rax), %rax
+       ret
+
+       .p2align 4
+L(return_null):
+       xor     %eax, %eax
+       ret
+END(MEMCHR)
+#endif
index 128f9ea6378c5303603fb7d865e91c950aeb4996..d6bff28757541917a261d6f59c64a41f39621d0b 100644 (file)
@@ -1,4 +1,7 @@
-#define MEMCHR __rawmemchr_avx2
-#define USE_AS_RAWMEMCHR 1
+#ifndef RAWMEMCHR
+# define RAWMEMCHR     __rawmemchr_avx2
+#endif
+#define USE_AS_RAWMEMCHR       1
+#define MEMCHR RAWMEMCHR
 
 #include "memchr-avx2.S"
index ec942b77ba5a3598b14f6030af0e7da890e0ef15..dc1c450699e46c02c17af475520dfab4a3ed8400 100644 (file)
@@ -1,4 +1,7 @@
-#define MEMCHR __rawmemchr_evex
-#define USE_AS_RAWMEMCHR 1
+#ifndef RAWMEMCHR
+# define RAWMEMCHR     __rawmemchr_evex
+#endif
+#define USE_AS_RAWMEMCHR       1
+#define MEMCHR RAWMEMCHR
 
 #include "memchr-evex.S"
index 3841c14c34af74d422dd2f77e49767dfaece593e..e2c2e20d85ae684417d110951d27f2edbdd129c3 100644 (file)
    License along with the GNU C Library; if not, see
    <https://www.gnu.org/licenses/>.  */
 
-/* Define multiple versions only for the definition in libc. */
-#if IS_IN (libc)
-# define __rawmemchr __rawmemchr_sse2
-
-# undef weak_alias
-# define weak_alias(__rawmemchr, rawmemchr)
-# undef libc_hidden_def
-# define libc_hidden_def(__rawmemchr)
-#endif
+#include <isa-level.h>
+#include <sysdep.h>
+
+/* MINIMUM_X86_ISA_LEVEL <= 2 because there is no V2 implementation
+   so we need this to build for ISA V2 builds. */
+#if ISA_SHOULD_BUILD (2)
+
+# ifndef RAWMEMCHR
+#  define RAWMEMCHR    __rawmemchr_sse2
+# endif
+
+       .text
+ENTRY (RAWMEMCHR)
+       movd    %rsi, %xmm1
+       mov     %rdi, %rcx
+
+       punpcklbw %xmm1, %xmm1
+       punpcklbw %xmm1, %xmm1
+
+       and     $63, %rcx
+       pshufd  $0, %xmm1, %xmm1
+
+       cmp     $48, %rcx
+       ja      L(crosscache)
+
+       movdqu  (%rdi), %xmm0
+       pcmpeqb %xmm1, %xmm0
+/* Check if there is a match.  */
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+
+       jnz     L(matches)
+       add     $16, %rdi
+       and     $-16, %rdi
+       jmp     L(loop_prolog)
+
+       .p2align 4
+L(crosscache):
+       and     $15, %rcx
+       and     $-16, %rdi
+       movdqa  (%rdi), %xmm0
+
+       pcmpeqb %xmm1, %xmm0
+/* Check if there is a match.  */
+       pmovmskb %xmm0, %eax
+/* Remove the leading bytes.  */
+       sar     %cl, %eax
+       test    %eax, %eax
+       je      L(unaligned_no_match)
+/* Check which byte is a match.  */
+       bsf     %eax, %eax
+
+       add     %rdi, %rax
+       add     %rcx, %rax
+       ret
+
+       .p2align 4
+L(unaligned_no_match):
+       add     $16, %rdi
+
+       .p2align 4
+L(loop_prolog):
+       movdqa  (%rdi), %xmm0
+       pcmpeqb %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       movdqa  16(%rdi), %xmm2
+       pcmpeqb %xmm1, %xmm2
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
+
+       movdqa  32(%rdi), %xmm3
+       pcmpeqb %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32)
+
+       movdqa  48(%rdi), %xmm4
+       pcmpeqb %xmm1, %xmm4
+       add     $64, %rdi
+       pmovmskb %xmm4, %eax
+       test    %eax, %eax
+       jnz     L(matches0)
+
+       test    $0x3f, %rdi
+       jz      L(align64_loop)
+
+       movdqa  (%rdi), %xmm0
+       pcmpeqb %xmm1, %xmm0
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       movdqa  16(%rdi), %xmm2
+       pcmpeqb %xmm1, %xmm2
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
 
-#include "../rawmemchr.S"
+       movdqa  32(%rdi), %xmm3
+       pcmpeqb %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32)
+
+       movdqa  48(%rdi), %xmm3
+       pcmpeqb %xmm1, %xmm3
+       pmovmskb %xmm3, %eax
+
+       add     $64, %rdi
+       test    %eax, %eax
+       jnz     L(matches0)
+
+       and     $-64, %rdi
+
+       .p2align 4
+L(align64_loop):
+       movdqa  (%rdi), %xmm0
+       movdqa  16(%rdi), %xmm2
+       movdqa  32(%rdi), %xmm3
+       movdqa  48(%rdi), %xmm4
+
+       pcmpeqb %xmm1, %xmm0
+       pcmpeqb %xmm1, %xmm2
+       pcmpeqb %xmm1, %xmm3
+       pcmpeqb %xmm1, %xmm4
+
+       pmaxub  %xmm0, %xmm3
+       pmaxub  %xmm2, %xmm4
+       pmaxub  %xmm3, %xmm4
+       pmovmskb %xmm4, %eax
+
+       add     $64, %rdi
+
+       test    %eax, %eax
+       jz      L(align64_loop)
+
+       sub     $64, %rdi
+
+       pmovmskb %xmm0, %eax
+       test    %eax, %eax
+       jnz     L(matches)
+
+       pmovmskb %xmm2, %eax
+       test    %eax, %eax
+       jnz     L(matches16)
+
+       movdqa  32(%rdi), %xmm3
+       pcmpeqb %xmm1, %xmm3
+
+       pcmpeqb 48(%rdi), %xmm1
+       pmovmskb %xmm3, %eax
+       test    %eax, %eax
+       jnz     L(matches32)
+
+       pmovmskb %xmm1, %eax
+       bsf     %eax, %eax
+       lea     48(%rdi, %rax), %rax
+       ret
+
+       .p2align 4
+L(matches0):
+       bsf     %eax, %eax
+       lea     -16(%rax, %rdi), %rax
+       ret
+
+       .p2align 4
+L(matches):
+       bsf     %eax, %eax
+       add     %rdi, %rax
+       ret
+
+       .p2align 4
+L(matches16):
+       bsf     %eax, %eax
+       lea     16(%rax, %rdi), %rax
+       ret
+
+       .p2align 4
+L(matches32):
+       bsf     %eax, %eax
+       lea     32(%rax, %rdi), %rax
+       ret
+
+END (RAWMEMCHR)
+#endif
diff --git a/sysdeps/x86_64/multiarch/rtld-memchr.S b/sysdeps/x86_64/multiarch/rtld-memchr.S
new file mode 100644 (file)
index 0000000..a14b192
--- /dev/null
@@ -0,0 +1,18 @@
+/* Copyright (C) 2022 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/>.  */
+
+#include "../memchr.S"
diff --git a/sysdeps/x86_64/multiarch/rtld-rawmemchr.S b/sysdeps/x86_64/multiarch/rtld-rawmemchr.S
new file mode 100644 (file)
index 0000000..5d4110a
--- /dev/null
@@ -0,0 +1,18 @@
+/* Copyright (C) 2022 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/>.  */
+
+#include "../rawmemchr.S"
index 282854f1a198cb3caf679526d6932868e23b366c..2bf93fd84bddede47d130360f451a534e6caadc4 100644 (file)
@@ -1,4 +1,7 @@
-#define MEMCHR __wmemchr_avx2
-#define USE_AS_WMEMCHR 1
+#ifndef WMEMCHR
+# define WMEMCHR       __wmemchr_avx2
+#endif
+#define USE_AS_WMEMCHR 1
+#define MEMCHR WMEMCHR
 
 #include "memchr-avx2.S"
index 06cd0f9f5a7575e72b070337bfc5f98a0efaf11a..5512d5cdc36e29d49ff603e600de39c633bca336 100644 (file)
@@ -1,4 +1,7 @@
-#define MEMCHR __wmemchr_evex
-#define USE_AS_WMEMCHR 1
+#ifndef WMEMCHR
+# define WMEMCHR       __wmemchr_evex
+#endif
+#define USE_AS_WMEMCHR 1
+#define MEMCHR WMEMCHR
 
 #include "memchr-evex.S"
index 70a965d552b20ed880d6ed29b83e400feb03de83..b675a070d432a656b870cf928442d9bdd631b5e2 100644 (file)
@@ -1,4 +1,7 @@
-#define USE_AS_WMEMCHR 1
-#define wmemchr __wmemchr_sse2
+#ifndef WMEMCHR
+# define WMEMCHR       __wmemchr_sse2
+#endif
+#define USE_AS_WMEMCHR 1
+#define MEMCHR WMEMCHR
 
-#include "../memchr.S"
+#include "memchr-sse2.S"
index 4c1a3383b940d5e7007b25367011ceee4509a04e..ba7e5202e6249d3480f046b18dd20f44f5b45840 100644 (file)
    License along with the GNU C Library; if not, see
    <https://www.gnu.org/licenses/>.  */
 
-#include <sysdep.h>
+#define RAWMEMCHR      __rawmemchr
 
-       .text
-ENTRY (__rawmemchr)
-       movd    %rsi, %xmm1
-       mov     %rdi, %rcx
+#define DEFAULT_IMPL_V1        "multiarch/rawmemchr-sse2.S"
+#define DEFAULT_IMPL_V3        "multiarch/rawmemchr-avx2.S"
+#define DEFAULT_IMPL_V4        "multiarch/rawmemchr-evex.S"
 
-       punpcklbw %xmm1, %xmm1
-       punpcklbw %xmm1, %xmm1
-
-       and     $63, %rcx
-       pshufd  $0, %xmm1, %xmm1
-
-       cmp     $48, %rcx
-       ja      L(crosscache)
-
-       movdqu  (%rdi), %xmm0
-       pcmpeqb %xmm1, %xmm0
-/* Check if there is a match.  */
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-
-       jnz     L(matches)
-       add     $16, %rdi
-       and     $-16, %rdi
-       jmp     L(loop_prolog)
-
-       .p2align 4
-L(crosscache):
-       and     $15, %rcx
-       and     $-16, %rdi
-       movdqa  (%rdi), %xmm0
-
-       pcmpeqb %xmm1, %xmm0
-/* Check if there is a match.  */
-       pmovmskb %xmm0, %eax
-/* Remove the leading bytes.  */
-       sar     %cl, %eax
-       test    %eax, %eax
-       je      L(unaligned_no_match)
-/* Check which byte is a match.  */
-       bsf     %eax, %eax
-
-       add     %rdi, %rax
-       add     %rcx, %rax
-       ret
-
-       .p2align 4
-L(unaligned_no_match):
-       add     $16, %rdi
-
-       .p2align 4
-L(loop_prolog):
-       movdqa  (%rdi), %xmm0
-       pcmpeqb %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       movdqa  16(%rdi), %xmm2
-       pcmpeqb %xmm1, %xmm2
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       pcmpeqb %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32)
-
-       movdqa  48(%rdi), %xmm4
-       pcmpeqb %xmm1, %xmm4
-       add     $64, %rdi
-       pmovmskb %xmm4, %eax
-       test    %eax, %eax
-       jnz     L(matches0)
-
-       test    $0x3f, %rdi
-       jz      L(align64_loop)
-
-       movdqa  (%rdi), %xmm0
-       pcmpeqb %xmm1, %xmm0
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       movdqa  16(%rdi), %xmm2
-       pcmpeqb %xmm1, %xmm2
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       pcmpeqb %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32)
-
-       movdqa  48(%rdi), %xmm3
-       pcmpeqb %xmm1, %xmm3
-       pmovmskb %xmm3, %eax
-
-       add     $64, %rdi
-       test    %eax, %eax
-       jnz     L(matches0)
-
-       and     $-64, %rdi
-
-       .p2align 4
-L(align64_loop):
-       movdqa  (%rdi), %xmm0
-       movdqa  16(%rdi), %xmm2
-       movdqa  32(%rdi), %xmm3
-       movdqa  48(%rdi), %xmm4
-
-       pcmpeqb %xmm1, %xmm0
-       pcmpeqb %xmm1, %xmm2
-       pcmpeqb %xmm1, %xmm3
-       pcmpeqb %xmm1, %xmm4
-
-       pmaxub  %xmm0, %xmm3
-       pmaxub  %xmm2, %xmm4
-       pmaxub  %xmm3, %xmm4
-       pmovmskb %xmm4, %eax
-
-       add     $64, %rdi
-
-       test    %eax, %eax
-       jz      L(align64_loop)
-
-       sub     $64, %rdi
-
-       pmovmskb %xmm0, %eax
-       test    %eax, %eax
-       jnz     L(matches)
-
-       pmovmskb %xmm2, %eax
-       test    %eax, %eax
-       jnz     L(matches16)
-
-       movdqa  32(%rdi), %xmm3
-       pcmpeqb %xmm1, %xmm3
-
-       pcmpeqb 48(%rdi), %xmm1
-       pmovmskb %xmm3, %eax
-       test    %eax, %eax
-       jnz     L(matches32)
-
-       pmovmskb %xmm1, %eax
-       bsf     %eax, %eax
-       lea     48(%rdi, %rax), %rax
-       ret
-
-       .p2align 4
-L(matches0):
-       bsf     %eax, %eax
-       lea     -16(%rax, %rdi), %rax
-       ret
-
-       .p2align 4
-L(matches):
-       bsf     %eax, %eax
-       add     %rdi, %rax
-       ret
-
-       .p2align 4
-L(matches16):
-       bsf     %eax, %eax
-       lea     16(%rax, %rdi), %rax
-       ret
-
-       .p2align 4
-L(matches32):
-       bsf     %eax, %eax
-       lea     32(%rax, %rdi), %rax
-       ret
-
-END (__rawmemchr)
+#include "isa-default-impl.h"
 
 weak_alias (__rawmemchr, rawmemchr)
-libc_hidden_builtin_def (__rawmemchr)
+libc_hidden_def (__rawmemchr)
diff --git a/sysdeps/x86_64/wmemchr.S b/sysdeps/x86_64/wmemchr.S
new file mode 100644 (file)
index 0000000..30565b2
--- /dev/null
@@ -0,0 +1,28 @@
+/* Copyright (C) 2022 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/>.  */
+
+#define WMEMCHR        __wmemchr
+
+#define DEFAULT_IMPL_V1        "multiarch/wmemchr-sse2.S"
+#define DEFAULT_IMPL_V3        "multiarch/wmemchr-avx2.S"
+#define DEFAULT_IMPL_V4        "multiarch/wmemchr-evex.S"
+
+#include "isa-default-impl.h"
+
+libc_hidden_def (__wmemchr)
+weak_alias (__wmemchr, wmemchr)
+libc_hidden_weak (wmemchr)
This page took 2.723324 seconds and 5 git commands to generate.