EnTT 3.13.0
Loading...
Searching...
No Matches
flow.hpp
1#ifndef ENTT_GRAPH_FLOW_HPP
2#define ENTT_GRAPH_FLOW_HPP
3
4#include <algorithm>
5#include <cstddef>
6#include <functional>
7#include <iterator>
8#include <memory>
9#include <type_traits>
10#include <utility>
11#include <vector>
12#include "../config/config.h"
13#include "../container/dense_map.hpp"
14#include "../container/dense_set.hpp"
15#include "../core/compressed_pair.hpp"
16#include "../core/fwd.hpp"
17#include "../core/iterator.hpp"
18#include "../core/utility.hpp"
19#include "adjacency_matrix.hpp"
20#include "fwd.hpp"
21
22namespace entt {
23
28template<typename Allocator>
30 using alloc_traits = std::allocator_traits<Allocator>;
31 static_assert(std::is_same_v<typename alloc_traits::value_type, id_type>, "Invalid value type");
32 using task_container_type = dense_set<id_type, identity, std::equal_to<id_type>, typename alloc_traits::template rebind_alloc<id_type>>;
33 using ro_rw_container_type = std::vector<std::pair<std::size_t, bool>, typename alloc_traits::template rebind_alloc<std::pair<std::size_t, bool>>>;
34 using deps_container_type = dense_map<id_type, ro_rw_container_type, identity, std::equal_to<id_type>, typename alloc_traits::template rebind_alloc<std::pair<const id_type, ro_rw_container_type>>>;
36
37 void emplace(const id_type res, const bool is_rw) {
38 ENTT_ASSERT(index.first() < vertices.size(), "Invalid node");
39
40 if(!deps.contains(res) && sync_on != vertices.size()) {
41 deps[res].emplace_back(sync_on, true);
42 }
43
44 deps[res].emplace_back(index.first(), is_rw);
45 }
46
47 void setup_graph(adjacency_matrix_type &matrix) const {
48 for(const auto &elem: deps) {
49 const auto last = elem.second.cend();
50 auto it = elem.second.cbegin();
51
52 while(it != last) {
53 if(it->second) {
54 // rw item
55 if(auto curr = it++; it != last) {
56 if(it->second) {
57 matrix.insert(curr->first, it->first);
58 } else if(const auto next = std::find_if(it, last, [](const auto &value) { return value.second; }); next != last) {
59 for(; it != next; ++it) {
60 matrix.insert(curr->first, it->first);
61 matrix.insert(it->first, next->first);
62 }
63 } else {
64 for(; it != next; ++it) {
65 matrix.insert(curr->first, it->first);
66 }
67 }
68 }
69 } else {
70 // ro item (first iteration only)
71 if(const auto next = std::find_if(it, last, [](const auto &value) { return value.second; }); next != last) {
72 for(; it != next; ++it) {
73 matrix.insert(it->first, next->first);
74 }
75 } else {
76 it = last;
77 }
78 }
79 }
80 }
81 }
82
83 void transitive_closure(adjacency_matrix_type &matrix) const {
84 const auto length = matrix.size();
85
86 for(std::size_t vk{}; vk < length; ++vk) {
87 for(std::size_t vi{}; vi < length; ++vi) {
88 for(std::size_t vj{}; vj < length; ++vj) {
89 if(matrix.contains(vi, vk) && matrix.contains(vk, vj)) {
90 matrix.insert(vi, vj);
91 }
92 }
93 }
94 }
95 }
96
97 void transitive_reduction(adjacency_matrix_type &matrix) const {
98 const auto length = matrix.size();
99
100 for(std::size_t vert{}; vert < length; ++vert) {
101 matrix.erase(vert, vert);
102 }
103
104 for(std::size_t vj{}; vj < length; ++vj) {
105 for(std::size_t vi{}; vi < length; ++vi) {
106 if(matrix.contains(vi, vj)) {
107 for(std::size_t vk{}; vk < length; ++vk) {
108 if(matrix.contains(vj, vk)) {
109 matrix.erase(vi, vk);
110 }
111 }
112 }
113 }
114 }
115 }
116
117public:
119 using allocator_type = Allocator;
121 using size_type = std::size_t;
126
130
135 explicit basic_flow(const allocator_type &allocator)
136 : index{0u, allocator},
137 vertices{allocator},
138 deps{allocator},
139 sync_on{} {}
140
142 basic_flow(const basic_flow &) = default;
143
149 basic_flow(const basic_flow &other, const allocator_type &allocator)
150 : index{other.index.first(), allocator},
151 vertices{other.vertices, allocator},
152 deps{other.deps, allocator},
153 sync_on{other.sync_on} {}
154
156 basic_flow(basic_flow &&) noexcept = default;
157
163 basic_flow(basic_flow &&other, const allocator_type &allocator)
164 : index{other.index.first(), allocator},
165 vertices{std::move(other.vertices), allocator},
166 deps{std::move(other.deps), allocator},
167 sync_on{other.sync_on} {}
168
173 basic_flow &operator=(const basic_flow &) = default;
174
179 basic_flow &operator=(basic_flow &&) noexcept = default;
180
185 [[nodiscard]] constexpr allocator_type get_allocator() const noexcept {
186 return allocator_type{index.second()};
187 }
188
194 [[nodiscard]] id_type operator[](const size_type pos) const {
195 return vertices.cbegin()[pos];
196 }
197
199 void clear() noexcept {
200 index.first() = {};
201 vertices.clear();
202 deps.clear();
203 sync_on = {};
204 }
205
210 void swap(basic_flow &other) {
211 using std::swap;
212 std::swap(index, other.index);
213 std::swap(vertices, other.vertices);
214 std::swap(deps, other.deps);
215 std::swap(sync_on, other.sync_on);
216 }
217
222 [[nodiscard]] size_type size() const noexcept {
223 return vertices.size();
224 }
225
231 basic_flow &bind(const id_type value) {
232 sync_on += (sync_on == vertices.size());
233 const auto it = vertices.emplace(value).first;
234 index.first() = size_type(it - vertices.begin());
235 return *this;
236 }
237
243 ENTT_ASSERT(index.first() < vertices.size(), "Invalid node");
244 sync_on = index.first();
245
246 for(const auto &elem: deps) {
247 elem.second.emplace_back(sync_on, true);
248 }
249
250 return *this;
251 }
252
259 basic_flow &set(const id_type res, bool is_rw = false) {
260 emplace(res, is_rw);
261 return *this;
262 }
263
269 basic_flow &ro(const id_type res) {
270 emplace(res, false);
271 return *this;
272 }
273
281 template<typename It>
282 std::enable_if_t<std::is_same_v<std::remove_const_t<typename std::iterator_traits<It>::value_type>, id_type>, basic_flow &>
283 ro(It first, It last) {
284 for(; first != last; ++first) {
285 emplace(*first, false);
286 }
287
288 return *this;
289 }
290
296 basic_flow &rw(const id_type res) {
297 emplace(res, true);
298 return *this;
299 }
300
308 template<typename It>
309 std::enable_if_t<std::is_same_v<std::remove_const_t<typename std::iterator_traits<It>::value_type>, id_type>, basic_flow &>
310 rw(It first, It last) {
311 for(; first != last; ++first) {
312 emplace(*first, true);
313 }
314
315 return *this;
316 }
317
322 [[nodiscard]] graph_type graph() const {
323 graph_type matrix{vertices.size(), get_allocator()};
324
325 setup_graph(matrix);
326 transitive_closure(matrix);
327 transitive_reduction(matrix);
328
329 return matrix;
330 }
331
332private:
334 task_container_type vertices;
335 deps_container_type deps;
336 size_type sync_on;
337};
338
339} // namespace entt
340
341#endif
Basic implementation of a directed adjacency matrix.
std::pair< edge_iterator, bool > insert(const vertex_type lhs, const vertex_type rhs)
Inserts an edge into the adjacency matrix, if it does not exist.
size_type size() const noexcept
Returns the number of vertices.
bool contains(const vertex_type lhs, const vertex_type rhs) const
Checks if an adjacency matrix contains a given edge.
size_type erase(const vertex_type lhs, const vertex_type rhs)
Removes the edge associated with a pair of given vertices.
Utility class for creating task graphs.
Definition flow.hpp:29
void clear() noexcept
Clears the flow builder.
Definition flow.hpp:199
id_type operator[](const size_type pos) const
Returns the identifier at specified location.
Definition flow.hpp:194
Allocator allocator_type
Allocator type.
Definition flow.hpp:119
size_type size() const noexcept
Returns the number of tasks.
Definition flow.hpp:222
std::size_t size_type
Unsigned integer type.
Definition flow.hpp:121
basic_flow & sync()
Turns the current task into a sync point.
Definition flow.hpp:242
basic_flow & rw(const id_type res)
Assigns a writable resource to the current task.
Definition flow.hpp:296
std::enable_if_t< std::is_same_v< std::remove_const_t< typename std::iterator_traits< It >::value_type >, id_type >, basic_flow & > rw(It first, It last)
Assigns a range of writable resources to the current task.
Definition flow.hpp:310
constexpr allocator_type get_allocator() const noexcept
Returns the associated allocator.
Definition flow.hpp:185
graph_type graph() const
Generates a task graph for the current content.
Definition flow.hpp:322
basic_flow(const allocator_type &allocator)
Constructs a flow builder with a given allocator.
Definition flow.hpp:135
basic_flow & ro(const id_type res)
Assigns a read-only resource to the current task.
Definition flow.hpp:269
basic_flow(basic_flow &&) noexcept=default
Default move constructor.
basic_flow & bind(const id_type value)
Binds a task to a flow builder.
Definition flow.hpp:231
basic_flow & set(const id_type res, bool is_rw=false)
Assigns a resource to the current task with a given access mode.
Definition flow.hpp:259
basic_flow & operator=(basic_flow &&) noexcept=default
Default move assignment operator.
basic_flow()
Default constructor.
Definition flow.hpp:128
basic_flow(const basic_flow &other, const allocator_type &allocator)
Allocator-extended copy constructor.
Definition flow.hpp:149
std::enable_if_t< std::is_same_v< std::remove_const_t< typename std::iterator_traits< It >::value_type >, id_type >, basic_flow & > ro(It first, It last)
Assigns a range of read-only resources to the current task.
Definition flow.hpp:283
void swap(basic_flow &other)
Exchanges the contents with those of a given flow builder.
Definition flow.hpp:210
basic_flow & operator=(const basic_flow &)=default
Default copy assignment operator.
basic_flow(const basic_flow &)=default
Default copy constructor.
A compressed pair.
constexpr second_type & second() noexcept
Returns the second element that a pair stores.
constexpr first_type & first() noexcept
Returns the first element that a pair stores.
void clear() noexcept
Clears the container.
bool contains(const key_type &key) const
Checks if the container contains an element with a given key.
const_iterator cend() const noexcept
Returns an iterator to the end.
void clear() noexcept
Clears the container.
const_iterator begin() const noexcept
Returns an iterator to the beginning.
std::pair< iterator, bool > emplace(Args &&...args)
Constructs an element in-place, if it does not exist.
size_type size() const noexcept
Returns the number of elements in a container.
const_iterator cbegin() const noexcept
Returns an iterator to the beginning.
EnTT default namespace.
Definition dense_map.hpp:21
std::uint32_t id_type
Alias declaration for type identifiers.
Definition fwd.hpp:13
Utility class to create an iterable object from a pair of iterators.
Definition iterator.hpp:141