6 #ifndef MATF_RG_PROJECT_UTILS_HPP
7 #define MATF_RG_PROJECT_UTILS_HPP
10 #include <source_location>
15 #include <unordered_set>
16 #include <type_traits>
20 struct overloaded : Ts... {
21 using Ts::operator()...;
25 overloaded(Ts...) -> overloaded<Ts...>;
27 template<
typename Func>
29 DeferImpl(Func f) : f(f) {
39 struct MakeDeferImpl {
42 template<
typename Func>
43 DeferImpl<Func> operator<<(MakeDeferImpl, Func f) {
47 #define CONCAT_IMPL(a, b) a##b
48 #define CONCAT(a, b) CONCAT_IMPL(a, b)
63 #define defer auto CONCAT(defer_stmt_, __LINE__) = MakeDeferImpl() << [&]
68 #define range(container) std::begin(container), std::end(container)
73 #define crange(container) std::cbegin(container), std::cend(container)
96 void trace(std::source_location location = std::source_location::current());
116 template<
typename Action>
118 static std::once_flag
once;
119 std::call_once(
once, action);
132 template<
typename Container,
typename T>
133 bool contains(
const Container &container,
const T &value) {
134 if constexpr (requires { container.contains(value); }) {
135 return container.contains(value);
137 return std::find(std::cbegin(container), std::cend(container), value) != std::cend(container);
148 template<
typename It,
typename Adjacent>
150 using ElementType = std::remove_reference_t<decltype(*first)>;
151 std::unordered_set<ElementType> visited;
152 std::vector<ElementType> stack;
154 auto visit = [&](
auto&
self, ElementType ¤t)
mutable ->
void {
155 visited.emplace(current);
156 for (
auto next: adjacent(current)) {
157 if (!visited.contains(next)) {
161 stack.push_back(current);
167 if (!visited.contains(*it)) {
172 std::move(stack.rbegin(), stack.rend(), first);
183 template<
typename It,
typename Adjacent,
typename OutputIt = std::
nullptr_t>
184 bool has_cycle(It first, It last, Adjacent adjacent, OutputIt cycle_output =
nullptr) {
185 using ElementType = std::remove_reference_t<decltype(*first)>;
186 std::unordered_set<ElementType> visited;
187 std::unordered_set<ElementType> path;
188 auto visit = [&](
auto&
self, ElementType ¤t)
mutable ->
bool {
189 visited.emplace(current);
190 path.emplace(current);
191 for (ElementType &next: adjacent(current)) {
192 if (!visited.contains(next) &&
self(
self, next)) {
195 if (path.contains(next)) {
203 for (
auto root = first; root != last; ++root) {
204 if (!visited.contains(*root) && visit(visit, *root)) {
205 if constexpr (!std::is_same_v<OutputIt, std::nullptr_t>) {
206 std::move(path.begin(), path.end(), cycle_output);
void topological_sort(It first, It last, Adjacent adjacent)
Topologically sorts a directed acyclic graph represented by a range and an adjacent function....
Definition: Utils.hpp:149
bool has_cycle(It first, It last, Adjacent adjacent, OutputIt cycle_output=nullptr)
Checks if a graph represented by a range and an adjacent function has a cycle.
Definition: Utils.hpp:184
bool contains(const Container &container, const T &value)
Checks if a container contains a value. Shorthand for std::find.
Definition: Utils.hpp:133
void tracing_on()
Turns on tracing.
Definition: Utils.cpp:12
void tracing_off()
Turns off tracing.
Definition: Utils.cpp:16
std::string read_text_file(const std::filesystem::path &path)
Reads a text file.
Definition: Utils.cpp:101
void once(Action action)
Calls an action once.
Definition: Utils.hpp:117
void trace(std::source_location location=std::source_location::current())
Traces a function call.
Definition: Utils.cpp:20