Skip to content

Commit f19a27a

Browse files
committed
chore: refactor thread support sources
1 parent d1a37ee commit f19a27a

11 files changed

+471
-347
lines changed
+165
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,165 @@
1+
//
2+
// Licensed under the Apache License v2.0 with LLVM Exceptions.
3+
// See https://llvm.org/LICENSE.txt for license information.
4+
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
5+
//
6+
// Copyright (c) 2023 Vinnie Falco ([email protected])
7+
//
8+
// Official repository: https://github.com/cppalliance/mrdox
9+
//
10+
11+
#ifndef MRDOX_SUPPORT_EXECUTORGROUP_HPP
12+
#define MRDOX_SUPPORT_EXECUTORGROUP_HPP
13+
14+
#include <mrdox/Platform.hpp>
15+
#include <mrdox/Support/ThreadPool.hpp>
16+
#include <mrdox/Support/unlock_guard.hpp>
17+
#include <condition_variable>
18+
#include <deque>
19+
#include <mutex>
20+
#include <vector>
21+
22+
namespace clang {
23+
namespace mrdox {
24+
25+
/** A set of execution agents for performing concurrent work.
26+
*/
27+
template<class Agent>
28+
class ExecutorGroup
29+
{
30+
struct Impl
31+
{
32+
std::mutex mutex_;
33+
std::condition_variable cv_;
34+
};
35+
36+
ThreadPool& threadPool_;
37+
std::unique_ptr<Impl> impl_;
38+
std::vector<std::unique_ptr<Agent>> agents_;
39+
std::deque<any_callable<void(Agent&)>> work_;
40+
std::size_t busy_ = 0;
41+
42+
public:
43+
template<class T>
44+
using arg_t = ThreadPool::arg_t<T>;
45+
46+
ExecutorGroup(ExecutorGroup const&) = delete;
47+
ExecutorGroup& operator=(ExecutorGroup&&) = delete;
48+
ExecutorGroup& operator=(ExecutorGroup const&) = delete;
49+
ExecutorGroup(ExecutorGroup&&) = default;
50+
51+
explicit
52+
ExecutorGroup(
53+
ThreadPool& threadPool) noexcept
54+
: threadPool_(threadPool)
55+
, impl_(std::make_unique<Impl>())
56+
{
57+
}
58+
59+
/** Construct a new agent in the group.
60+
61+
The behavior is undefined if there is
62+
any outstanding work or busy threads.
63+
*/
64+
template<class... Args>
65+
void
66+
emplace(Args&&... args)
67+
{
68+
agents_.emplace_back(std::make_unique<Agent>(
69+
std::forward<Args>(args)...));
70+
}
71+
72+
/** Submit work to be executed.
73+
74+
The function object must have this
75+
equivalent signature:
76+
@code
77+
void( Agent&, Args... );
78+
@endcode
79+
*/
80+
template<class F, class... Args>
81+
void
82+
async(F&& f, Args&&... args)
83+
{
84+
static_assert(std::is_invocable_v<F, Agent&, arg_t<Args>...>);
85+
std::unique_lock<std::mutex> lock(impl_->mutex_);
86+
work_.emplace_back(
87+
[
88+
f = std::forward<F>(f),
89+
args = std::tuple<arg_t<Args>...>(args...)
90+
](Agent& agent)
91+
{
92+
std::apply(f, std::tuple_cat(
93+
std::tuple<Agent&>(agent),
94+
std::move(args)));
95+
});
96+
if(agents_.empty())
97+
return;
98+
run(std::move(lock));
99+
}
100+
101+
/** Block until all work has completed.
102+
*/
103+
void
104+
wait()
105+
{
106+
std::unique_lock<std::mutex> lock(impl_->mutex_);
107+
impl_->cv_.wait(lock,
108+
[&]
109+
{
110+
return work_.empty() && busy_ == 0;
111+
});
112+
}
113+
114+
private:
115+
class scoped_agent
116+
{
117+
ExecutorGroup& group_;
118+
std::unique_ptr<Agent> agent_;
119+
120+
public:
121+
scoped_agent(
122+
ExecutorGroup& group,
123+
std::unique_ptr<Agent> agent) noexcept
124+
: group_(group)
125+
, agent_(std::move(agent))
126+
{
127+
}
128+
129+
~scoped_agent()
130+
{
131+
--group_.busy_;
132+
group_.agents_.emplace_back(std::move(agent_));
133+
group_.impl_->cv_.notify_all();
134+
}
135+
};
136+
137+
void
138+
run(std::unique_lock<std::mutex> lock)
139+
{
140+
std::unique_ptr<Agent> agent(std::move(agents_.back()));
141+
agents_.pop_back();
142+
++busy_;
143+
144+
threadPool_.async(
145+
[this, agent = std::move(agent)]() mutable
146+
{
147+
scoped_agent scope(*this, std::move(agent));
148+
std::unique_lock<std::mutex> lock(impl_->mutex_);
149+
for(;;)
150+
{
151+
if(work_.empty())
152+
break;
153+
any_callable<void(Agent&)> work(std::move(work_.front()));
154+
work_.pop_front();
155+
unlock_guard unlock(impl_->mutex_);
156+
work(*agent);
157+
}
158+
});
159+
}
160+
};
161+
162+
} // mrdox
163+
} // clang
164+
165+
#endif

include/mrdox/Support/Handlebars.hpp

-135
This file was deleted.

0 commit comments

Comments
 (0)