123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 |
- /* sort.c -- Sort without allocating memory
- Copyright (C) 2012-2021 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;
- }
- }
|