/* 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; | |

} | |

} |