|  | /* sort.c -- Sort without allocating memory | 
|  | Copyright (C) 2012-2024 Free Software Foundation, Inc. | 
|  | Written by Ian Lance Taylor, Google. | 
|  |  | 
|  | Redistribution and use in source and binary forms, with or without | 
|  | modification, are permitted provided that the following conditions are | 
|  | met: | 
|  |  | 
|  | (1) Redistributions of source code must retain the above copyright | 
|  | notice, this list of conditions and the following disclaimer. | 
|  |  | 
|  | (2) Redistributions in binary form must reproduce the above copyright | 
|  | notice, this list of conditions and the following disclaimer in | 
|  | the documentation and/or other materials provided with the | 
|  | distribution. | 
|  |  | 
|  | (3) The name of the author may not be used to | 
|  | endorse or promote products derived from this software without | 
|  | specific prior written permission. | 
|  |  | 
|  | THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR | 
|  | IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | 
|  | WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
|  | DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, | 
|  | INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 
|  | (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
|  | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | 
|  | HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, | 
|  | STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING | 
|  | IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | 
|  | POSSIBILITY OF SUCH DAMAGE.  */ | 
|  |  | 
|  | #include "config.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  | #include <sys/types.h> | 
|  |  | 
|  | #include "backtrace.h" | 
|  | #include "internal.h" | 
|  |  | 
|  | /* The GNU glibc version of qsort allocates memory, which we must not | 
|  | do if we are invoked by a signal handler.  So provide our own | 
|  | sort.  */ | 
|  |  | 
|  | static void | 
|  | swap (char *a, char *b, size_t size) | 
|  | { | 
|  | size_t i; | 
|  |  | 
|  | for (i = 0; i < size; i++, a++, b++) | 
|  | { | 
|  | char t; | 
|  |  | 
|  | t = *a; | 
|  | *a = *b; | 
|  | *b = t; | 
|  | } | 
|  | } | 
|  |  | 
|  | void | 
|  | backtrace_qsort (void *basearg, size_t count, size_t size, | 
|  | int (*compar) (const void *, const void *)) | 
|  | { | 
|  | char *base = (char *) basearg; | 
|  | size_t i; | 
|  | size_t mid; | 
|  |  | 
|  | tail_recurse: | 
|  | if (count < 2) | 
|  | return; | 
|  |  | 
|  | /* The symbol table and DWARF tables, which is all we use this | 
|  | routine for, tend to be roughly sorted.  Pick the middle element | 
|  | in the array as our pivot point, so that we are more likely to | 
|  | cut the array in half for each recursion step.  */ | 
|  | swap (base, base + (count / 2) * size, size); | 
|  |  | 
|  | mid = 0; | 
|  | for (i = 1; i < count; i++) | 
|  | { | 
|  | if ((*compar) (base, base + i * size) > 0) | 
|  | { | 
|  | ++mid; | 
|  | if (i != mid) | 
|  | swap (base + mid * size, base + i * size, size); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (mid > 0) | 
|  | swap (base, base + mid * size, size); | 
|  |  | 
|  | /* Recurse with the smaller array, loop with the larger one.  That | 
|  | ensures that our maximum stack depth is log count.  */ | 
|  | if (2 * mid < count) | 
|  | { | 
|  | backtrace_qsort (base, mid, size, compar); | 
|  | base += (mid + 1) * size; | 
|  | count -= mid + 1; | 
|  | goto tail_recurse; | 
|  | } | 
|  | else | 
|  | { | 
|  | backtrace_qsort (base + (mid + 1) * size, count - (mid + 1), | 
|  | size, compar); | 
|  | count = mid; | 
|  | goto tail_recurse; | 
|  | } | 
|  | } |