]> git.ipfire.org Git - thirdparty/glibc.git/commitdiff
Optimize bsearch() implementation for performance
authorKuan-Wei Chiu <visitorckw@gmail.com>
Mon, 9 Sep 2024 16:23:27 +0000 (00:23 +0800)
committerDJ Delorie <dj@redhat.com>
Wed, 11 Dec 2024 22:04:42 +0000 (17:04 -0500)
Optimize the bsearch() function to improve binary search performance.
Although the code size grew by 8 bytes, the new implementation achieves
a 15% reduction in execution time on my x86 machine, according to the
bench-bsearch benchmark results.

Signed-off-by: Kuan-Wei Chiu <visitorckw@gmail.com>
Reviewed-by: Noah Goldstein <goldstein.w.n@gmail.com>
Reviewed-by: DJ Delorie <dj@redhat.com>
bits/stdlib-bsearch.h

index 540d7189520ba225bf7f9cb7579fc5f6039591d7..57f060b5046d1595068a09f78770bbca622b559f 100644 (file)
@@ -20,22 +20,14 @@ __extern_inline void *
 bsearch (const void *__key, const void *__base, size_t __nmemb, size_t __size,
         __compar_fn_t __compar)
 {
-  size_t __l, __u, __idx;
   const void *__p;
   int __comparison;
 
-  __l = 0;
-  __u = __nmemb;
-  while (__l < __u)
+  while (__nmemb)
     {
-      __idx = (__l + __u) / 2;
-      __p = (const void *) (((const char *) __base) + (__idx * __size));
+      __p = (const void *) (((const char *) __base) + ((__nmemb >> 1) * __size));
       __comparison = (*__compar) (__key, __p);
-      if (__comparison < 0)
-       __u = __idx;
-      else if (__comparison > 0)
-       __l = __idx + 1;
-      else
+      if (__comparison == 0)
        {
 #if __GNUC_PREREQ(4, 6)
 # pragma GCC diagnostic push
@@ -46,6 +38,12 @@ bsearch (const void *__key, const void *__base, size_t __nmemb, size_t __size,
 # pragma GCC diagnostic pop
 #endif
        }
+      if (__comparison > 0)
+       {
+         __base = ((const char *) __p) + __size;
+         --__nmemb;
+       }
+      __nmemb >>= 1;
     }
 
   return NULL;