EnTT 3.14.0
Loading...
Searching...
No Matches
algorithm.hpp
1#ifndef ENTT_CORE_ALGORITHM_HPP
2#define ENTT_CORE_ALGORITHM_HPP
3
4#include <algorithm>
5#include <functional>
6#include <iterator>
7#include <utility>
8#include <vector>
9#include "utility.hpp"
10
11namespace entt {
12
21struct std_sort {
35 template<typename It, typename Compare = std::less<>, typename... Args>
36 void operator()(It first, It last, Compare compare = Compare{}, Args &&...args) const {
37 std::sort(std::forward<Args>(args)..., std::move(first), std::move(last), std::move(compare));
38 }
39};
40
54 template<typename It, typename Compare = std::less<>>
55 void operator()(It first, It last, Compare compare = Compare{}) const {
56 if(first < last) {
57 for(auto it = first + 1; it < last; ++it) {
58 auto value = std::move(*it);
59 auto pre = it;
60
61 // NOLINTBEGIN(cppcoreguidelines-pro-bounds-pointer-arithmetic)
62 for(; pre > first && compare(value, *(pre - 1)); --pre) {
63 *pre = std::move(*(pre - 1));
64 }
65 // NOLINTEND(cppcoreguidelines-pro-bounds-pointer-arithmetic)
66
67 *pre = std::move(value);
68 }
69 }
70 }
71};
72
78template<std::size_t Bit, std::size_t N>
79struct radix_sort {
80 static_assert((N % Bit) == 0, "The maximum number of bits to sort must be a multiple of the number of bits processed per pass");
81
97 template<typename It, typename Getter = identity>
98 void operator()(It first, It last, Getter getter = Getter{}) const {
99 if(first < last) {
100 constexpr auto passes = N / Bit;
101
102 using value_type = typename std::iterator_traits<It>::value_type;
103 std::vector<value_type> aux(std::distance(first, last));
104
105 auto part = [getter = std::move(getter)](auto from, auto to, auto out, auto start) {
106 constexpr auto mask = (1 << Bit) - 1;
107 constexpr auto buckets = 1 << Bit;
108
109 // NOLINTNEXTLINE(cppcoreguidelines-avoid-c-arrays, modernize-avoid-c-arrays)
110 std::size_t count[buckets]{};
111
112 for(auto it = from; it != to; ++it) {
113 ++count[(getter(*it) >> start) & mask];
114 }
115
116 // NOLINTNEXTLINE(cppcoreguidelines-avoid-c-arrays, modernize-avoid-c-arrays)
117 std::size_t index[buckets]{};
118
119 for(std::size_t pos{}, end = buckets - 1u; pos < end; ++pos) {
120 index[pos + 1u] = index[pos] + count[pos];
121 }
122
123 for(auto it = from; it != to; ++it) {
124 out[index[(getter(*it) >> start) & mask]++] = std::move(*it);
125 }
126 };
127
128 for(std::size_t pass = 0; pass < (passes & ~1); pass += 2) {
129 part(first, last, aux.begin(), pass * Bit);
130 part(aux.begin(), aux.end(), first, (pass + 1) * Bit);
131 }
132
133 if constexpr(passes & 1) {
134 part(first, last, aux.begin(), (passes - 1) * Bit);
135 std::move(aux.begin(), aux.end(), first);
136 }
137 }
138 }
139};
140
141} // namespace entt
142
143#endif
EnTT default namespace.
Definition dense_map.hpp:22
constexpr Type make_obj_using_allocator(const Allocator &allocator, Args &&...args)
Uses-allocator construction utility (waiting for C++20).
Definition memory.hpp:219
Function object for performing insertion sort.
Definition algorithm.hpp:42
void operator()(It first, It last, Compare compare=Compare{}) const
Sorts the elements in a range.
Definition algorithm.hpp:55
Function object for performing LSD radix sort.
Definition algorithm.hpp:79
void operator()(It first, It last, Getter getter=Getter{}) const
Sorts the elements in a range.
Definition algorithm.hpp:98
Function object to wrap std::sort in a class type.
Definition algorithm.hpp:21
void operator()(It first, It last, Compare compare=Compare{}, Args &&...args) const
Sorts the elements in a range.
Definition algorithm.hpp:36