diff --git a/compiler-rt/lib/sanitizer_common/sanitizer_common_interceptors.inc b/compiler-rt/lib/sanitizer_common/sanitizer_common_interceptors.inc --- a/compiler-rt/lib/sanitizer_common/sanitizer_common_interceptors.inc +++ b/compiler-rt/lib/sanitizer_common/sanitizer_common_interceptors.inc @@ -9855,12 +9855,25 @@ } } qsort_compar_f old_compar = qsort_compar; - qsort_compar = compar; SIZE_T old_size = qsort_size; - qsort_size = size; + // Handle qsort() implementations that recurse using an + // interposable function call: + bool already_wrapped = compar == wrapped_qsort_compar; + if (already_wrapped) { + // This case should only happen if the qsort() implementation calls itself + // using a preemptible function call (e.g. the FreeBSD libc version). + // Check that the size and comparator arguments are as expected. + CHECK_EQ(compar, qsort_compar); + CHECK_EQ(qsort_size, size); + } else { + qsort_compar = compar; + qsort_size = size; + } REAL(qsort)(base, nmemb, size, wrapped_qsort_compar); - qsort_compar = old_compar; - qsort_size = old_size; + if (!already_wrapped) { + qsort_compar = old_compar; + qsort_size = old_size; + } COMMON_INTERCEPTOR_WRITE_RANGE(ctx, base, nmemb * size); } #define INIT_QSORT COMMON_INTERCEPT_FUNCTION(qsort) @@ -9893,12 +9906,25 @@ } } qsort_r_compar_f old_compar = qsort_r_compar; - qsort_r_compar = compar; SIZE_T old_size = qsort_r_size; - qsort_r_size = size; + // Handle qsort_r() implementations that recurse using an + // interposable function call: + bool already_wrapped = compar == wrapped_qsort_r_compar; + if (already_wrapped) { + // This case should only happen if the qsort() implementation calls itself + // using a preemptible function call (e.g. the FreeBSD libc version). + // Check that the size and comparator arguments are as expected. + CHECK_EQ(compar, qsort_r_compar); + CHECK_EQ(qsort_r_size, size); + } else { + qsort_r_compar = compar; + qsort_r_size = size; + } REAL(qsort_r)(base, nmemb, size, wrapped_qsort_r_compar, arg); - qsort_r_compar = old_compar; - qsort_r_size = old_size; + if (!already_wrapped) { + qsort_r_compar = old_compar; + qsort_r_size = old_size; + } COMMON_INTERCEPTOR_WRITE_RANGE(ctx, base, nmemb * size); } #define INIT_QSORT_R COMMON_INTERCEPT_FUNCTION(qsort_r) diff --git a/compiler-rt/test/sanitizer_common/TestCases/Posix/recursion-in-qsort.cpp b/compiler-rt/test/sanitizer_common/TestCases/Posix/recursion-in-qsort.cpp new file mode 100644 --- /dev/null +++ b/compiler-rt/test/sanitizer_common/TestCases/Posix/recursion-in-qsort.cpp @@ -0,0 +1,52 @@ +// Check that a qsort() comparator that calls qsort() works as expected +// RUN: %clangxx -O2 %s -o %t +// RUN: %run %t 2>&1 | FileCheck %s + +#include +#include + +struct Foo { + int array[2]; +}; +int global_array[4] = {7, 11, 9, 10}; + +#define array_size(x) (sizeof(x) / sizeof(x[0])) + +int ascending_compare_ints(const void *a, const void *b) { + return *(const int *)a - *(const int *)b; +} + +int descending_compare_ints(const void *a, const void *b) { + // Add another qsort() call to check more than one level of recursion + qsort(global_array, array_size(global_array), sizeof(int), &ascending_compare_ints); + return *(const int *)b - *(const int *)a; +} + +int sort_and_compare(const void *a, const void *b) { + struct Foo *f1 = (struct Foo *)a; + struct Foo *f2 = (struct Foo *)b; + printf("sort_and_compare({%d, %d}, {%d, %d})\n", f1->array[0], f1->array[1], + f2->array[0], f2->array[1]); + // Call qsort from within qsort() to check that interceptors handle this case: + qsort(&f1->array, array_size(f1->array), sizeof(int), &descending_compare_ints); + qsort(&f2->array, array_size(f2->array), sizeof(int), &descending_compare_ints); + // Sort by second array element: + return f1->array[1] - f2->array[1]; +} + +int main() { + struct Foo qsortArg[6] = {{1, 99}, {2, 3}, {17, 5}, {8, 6}, {11, 4}, {7, 9}}; + // Sort the individual arrays in descending order and the over all struct + // Foo array in ascending order of the second array element. + qsort(qsortArg, array_size(qsortArg), sizeof(qsortArg[0]), &sort_and_compare); + + printf("Sorted result:"); + for (const auto &f : qsortArg) { + printf(" {%d,%d}", f.array[0], f.array[1]); + } + printf("\n"); + // CHECK: Sorted result: {99,1} {3,2} {11,4} {17,5} {8,6} {9,7} + printf("Sorted global_array: %d, %d, %d, %d\n", global_array[0], global_array[1], + global_array[2], global_array[3]); + // CHECK: Sorted global_array: 7, 9, 10, 11 +}