16
16
17
17
#define TBB_PREVIEW_PARALLEL_PHASE 1
18
18
19
+ #include < chrono>
20
+
19
21
#include " common/test.h"
20
22
#include " common/utils.h"
21
23
#include " common/utils_concurrency_limit.h"
24
26
#include " tbb/task_arena.h"
25
27
#include " tbb/parallel_for.h"
26
28
29
+ void active_wait_for (std::chrono::microseconds duration) {
30
+ for (auto t1 = std::chrono::steady_clock::now (), t2 = t1;
31
+ std::chrono::duration_cast<std::chrono::microseconds>(t2 - t1) < duration;
32
+ t2 = std::chrono::steady_clock::now ())
33
+ {
34
+ utils::doDummyWork (100 );
35
+ }
36
+ }
37
+
27
38
struct dummy_func {
28
39
void operator ()() const {
29
40
}
@@ -66,7 +77,7 @@ std::size_t measure_median_start_time(tbb::task_arena* ta, const F1& start = F1{
66
77
} else {
67
78
work ();
68
79
}
69
- utils::doDummyWork (i* 250 );
80
+ active_wait_for ( std::chrono::microseconds (i) );
70
81
}
71
82
return utils::median (longest_start_times.begin (), longest_start_times.end ());
72
83
}
@@ -101,7 +112,7 @@ class start_time_collection : public start_time_collection_base<start_time_colle
101
112
102
113
std::size_t measure_impl () {
103
114
return measure_median_start_time (arena);
104
- };
115
+ }
105
116
};
106
117
107
118
class start_time_collection_phase_wrapped
@@ -116,7 +127,7 @@ class start_time_collection_phase_wrapped
116
127
auto median_start_time = measure_median_start_time (arena);
117
128
arena->end_parallel_phase (/* with_fast_leave*/ true );
118
129
return median_start_time;
119
- };
130
+ }
120
131
};
121
132
122
133
class start_time_collection_scoped_phase_wrapped
@@ -130,7 +141,7 @@ class start_time_collection_scoped_phase_wrapped
130
141
tbb::task_arena::scoped_parallel_phase phase{*arena};
131
142
auto median_start_time = measure_median_start_time (arena);
132
143
return median_start_time;
133
- };
144
+ }
134
145
};
135
146
136
147
class start_time_collection_sequenced_phases
@@ -143,17 +154,35 @@ class start_time_collection_sequenced_phases
143
154
144
155
std::size_t measure_impl () {
145
156
std::size_t median_start_time;
157
+ utils::SpinBarrier barrier;
158
+ auto body = [&] (std::size_t ) {
159
+ barrier.wait ();
160
+ };
146
161
if (arena) {
162
+ barrier.initialize (arena->max_concurrency ());
147
163
median_start_time = measure_median_start_time (arena,
148
- [this ] { arena->start_parallel_phase (); },
149
- [this ] { arena->end_parallel_phase (with_fast_leave); });
164
+ [&] {
165
+ std::size_t num_threads = arena->max_concurrency ();
166
+ arena->start_parallel_phase ();
167
+ arena->execute ([&] {
168
+ tbb::parallel_for (std::size_t (0 ), num_threads, body, tbb::static_partitioner{});
169
+ });
170
+ arena->end_parallel_phase (with_fast_leave);
171
+ }
172
+ );
150
173
} else {
174
+ barrier.initialize (tbb::this_task_arena::max_concurrency ());
151
175
median_start_time = measure_median_start_time (arena,
152
- [] { tbb::this_task_arena::start_parallel_phase (); },
153
- [this ] { tbb::this_task_arena::end_parallel_phase (with_fast_leave); });
176
+ [&] {
177
+ std::size_t num_threads = tbb::this_task_arena::max_concurrency ();
178
+ tbb::this_task_arena::start_parallel_phase ();
179
+ tbb::parallel_for (std::size_t (0 ), num_threads, body, tbb::static_partitioner{});
180
+ tbb::this_task_arena::end_parallel_phase (with_fast_leave);
181
+ }
182
+ );
154
183
}
155
184
return median_start_time;
156
- };
185
+ }
157
186
158
187
public:
159
188
start_time_collection_sequenced_phases (tbb::task_arena& ta, std::size_t ntrials, bool fast_leave = false ) :
@@ -168,31 +197,38 @@ class start_time_collection_sequenced_phases
168
197
class start_time_collection_sequenced_scoped_phases
169
198
: public start_time_collection_base<start_time_collection_sequenced_scoped_phases>
170
199
{
171
- using base = start_time_collection_base<start_time_collection_sequenced_scoped_phases>;
172
- friend base;
173
-
174
- bool with_fast_leave;
175
-
176
- std::size_t measure_impl () {
177
- tbb::task_arena::scoped_parallel_phase* phase = nullptr ;
178
- auto median_start_time = measure_median_start_time (arena,
179
- [this , &phase] {
180
- phase = new tbb::task_arena::scoped_parallel_phase{*arena, with_fast_leave};
181
- },
182
- [&phase] {
183
- delete phase;
184
- });
185
- return median_start_time;
186
- };
200
+ using base = start_time_collection_base<start_time_collection_sequenced_scoped_phases>;
201
+ friend base;
202
+
203
+ bool with_fast_leave;
204
+
205
+ std::size_t measure_impl () {
206
+ utils::SpinBarrier barrier{static_cast <std::size_t >(arena->max_concurrency ())};
207
+ auto body = [&] (std::size_t ) {
208
+ barrier.wait ();
209
+ };
210
+ auto median_start_time = measure_median_start_time (arena,
211
+ [&] {
212
+ std::size_t num_threads = arena->max_concurrency ();
213
+ {
214
+ tbb::task_arena::scoped_parallel_phase phase{*arena, with_fast_leave};
215
+ arena->execute ([&] {
216
+ tbb::parallel_for (std::size_t (0 ), num_threads, body, tbb::static_partitioner{});
217
+ });
218
+ }
219
+ }
220
+ );
221
+ return median_start_time;
222
+ }
187
223
188
224
public:
189
- start_time_collection_sequenced_scoped_phases (tbb::task_arena& ta, std::size_t ntrials, bool fast_leave = false ) :
190
- base (ta, ntrials), with_fast_leave(fast_leave)
191
- {}
225
+ start_time_collection_sequenced_scoped_phases (tbb::task_arena& ta, std::size_t ntrials, bool fast_leave = false ) :
226
+ base (ta, ntrials), with_fast_leave(fast_leave)
227
+ {}
192
228
193
- explicit start_time_collection_sequenced_scoped_phases (std::size_t ntrials, bool fast_leave = false ) :
194
- base(ntrials), with_fast_leave(fast_leave)
195
- {}
229
+ explicit start_time_collection_sequenced_scoped_phases (std::size_t ntrials, bool fast_leave = false ) :
230
+ base(ntrials), with_fast_leave(fast_leave)
231
+ {}
196
232
};
197
233
198
234
// ! \brief \ref interface \ref requirement
0 commit comments