LibC: Optimize qsort

Apply same optimizations as to BAN quick sort!
This commit is contained in:
2026-05-13 04:36:13 +03:00
parent d181f9e553
commit d345f96387

View File

@@ -780,46 +780,73 @@ static void qsort_swap(void* lhs, void* rhs, size_t width)
uint8_t* ulhs = static_cast<uint8_t*>(lhs); uint8_t* ulhs = static_cast<uint8_t*>(lhs);
uint8_t* urhs = static_cast<uint8_t*>(rhs); uint8_t* urhs = static_cast<uint8_t*>(rhs);
uint8_t buffer[64]; uint8_t temp[64];
while (width > 0) while (width > 0)
{ {
const size_t to_swap = BAN::Math::min(width, sizeof(buffer)); const size_t to_swap = BAN::Math::min(width, sizeof(temp));
memcpy(buffer, ulhs, to_swap); memcpy(temp, ulhs, to_swap);
memcpy(ulhs, urhs, to_swap); memcpy(ulhs, urhs, to_swap);
memcpy(urhs, buffer, to_swap); memcpy(urhs, temp, to_swap);
width -= to_swap; width -= to_swap;
ulhs += to_swap; ulhs += to_swap;
urhs += to_swap; urhs += to_swap;
} }
} }
static uint8_t* qsort_partition(uint8_t* pbegin, uint8_t* pend, size_t width, int (*compar)(const void*, const void*)) struct qsort_pair
{ {
uint8_t* pivot = pend - width; uint8_t* lt;
uint8_t* p1 = pbegin; uint8_t* gt;
for (uint8_t* p2 = pbegin; p2 < pivot; p2 += width) };
static qsort_pair qsort_partition(uint8_t* pbegin, uint8_t* pend, size_t width, int (*compar)(const void*, const void*))
{
uint8_t* pivot = pbegin + (pend - pbegin) / width / 2 * width;
uint8_t* lt = pbegin;
uint8_t* eq = pbegin;
uint8_t* gt = pend;
while (eq < gt)
{ {
if (compar(p2, pivot) >= 0) const int comp = compar(eq, pivot);
continue;
qsort_swap(p1, p2, width); if (comp < 0)
p1 += width; {
qsort_swap(eq, lt, width);
if (pivot == lt)
pivot = eq;
lt += width;
eq += width;
}
else if (comp > 0)
{
gt -= width;
qsort_swap(eq, gt, width);
if (pivot == gt)
pivot = eq;
}
else
{
eq += width;
}
} }
qsort_swap(p1, pivot, width);
return p1; return { lt, gt };
} }
static void qsort_impl(uint8_t* pbegin, uint8_t* pend, size_t width, int (*compar)(const void*, const void*)) static void qsort_impl(uint8_t* pbegin, uint8_t* pend, size_t width, int (*compar)(const void*, const void*))
{ {
if ((pend - pbegin) / width <= 1) if (pbegin + width >= pend)
return; return;
uint8_t* mid = qsort_partition(pbegin, pend, width, compar); auto [lt, gt] = qsort_partition(pbegin, pend, width, compar);
qsort_impl(pbegin, mid, width, compar); qsort_impl(pbegin, lt, width, compar);
qsort_impl(mid + width, pend, width, compar); qsort_impl(gt, pend, width, compar);
} }
void qsort(void* base, size_t nel, size_t width, int (*compar)(const void*, const void*)) void qsort(void* base, size_t nel, size_t width, int (*compar)(const void*, const void*))
{ {
if (width == 0) if (width == 0 || nel <= 1)
return; return;
uint8_t* pbegin = static_cast<uint8_t*>(base); uint8_t* pbegin = static_cast<uint8_t*>(base);
qsort_impl(pbegin, pbegin + nel * width, width, compar); qsort_impl(pbegin, pbegin + nel * width, width, compar);