blob: b9699776df89c14cdccd4e88e5cdfa53f5e2c2aa [file] [log] [blame]
//===-- Implementation of heap sort -----------------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_LIBC_SRC_STDLIB_HEAP_SORT_H
#define LLVM_LIBC_SRC_STDLIB_HEAP_SORT_H
#include "src/__support/CPP/cstddef.h"
#include "src/stdlib/qsort_data.h"
namespace LIBC_NAMESPACE_DECL {
namespace internal {
// A simple in-place heapsort implementation.
// Follow the implementation in https://en.wikipedia.org/wiki/Heapsort.
template <typename A, typename F>
LIBC_INLINE void heap_sort(const A &array, const F &is_less) {
size_t end = array.len();
size_t start = end / 2;
const auto left_child = [](size_t i) -> size_t { return 2 * i + 1; };
while (end > 1) {
if (start > 0) {
// Select the next unheapified element to sift down.
--start;
} else {
// Extract the max element of the heap, moving a leaf to root to be sifted
// down.
--end;
array.swap(0, end);
}
// Sift start down the heap.
size_t root = start;
while (left_child(root) < end) {
size_t child = left_child(root);
// If there are two children, set child to the greater.
if ((child + 1 < end) && is_less(array.get(child), array.get(child + 1)))
++child;
// If the root is less than the greater child
if (!is_less(array.get(root), array.get(child)))
break;
// Swap the root with the greater child and continue sifting down.
array.swap(root, child);
root = child;
}
}
}
} // namespace internal
} // namespace LIBC_NAMESPACE_DECL
#endif // LLVM_LIBC_SRC_STDLIB_HEAP_SORT_H