125#ifdef SUBPAR_CUSTOM_PARALLELIZE_RANGE
126 if constexpr(nothrow_) {
127#ifdef SUBPAR_CUSTOM_PARALLELIZE_RANGE_NOTHROW
128 return SUBPAR_CUSTOM_PARALLELIZE_RANGE_NOTHROW(num_workers, num_tasks, run_task_range);
130 return SUBPAR_CUSTOM_PARALLELIZE_RANGE(num_workers, num_tasks, run_task_range);
133 return SUBPAR_CUSTOM_PARALLELIZE_RANGE(num_workers, num_tasks, run_task_range);
137 if (num_tasks <= 0) {
141 if (num_workers <= 1 || num_tasks == 1) {
142 run_task_range(0, 0, num_tasks);
147 Task_ tasks_per_worker = 1;
149 if (sanisizer::is_greater_than_or_equal(num_workers, num_tasks)) {
150 num_workers = num_tasks;
152 tasks_per_worker = num_tasks / num_workers;
153 remainder = num_tasks % num_workers;
156 const auto get_start = [&tasks_per_worker,&remainder](
const int w) -> Task_ {
158 return w * tasks_per_worker + (w < remainder ? w : remainder);
161 const auto get_length = [&tasks_per_worker,&remainder](
const int w) -> Task_ {
162 return tasks_per_worker + (w < remainder);
167 if constexpr(nothrow_) {
170 return sanisizer::create<std::vector<std::exception_ptr> >(num_workers);
174#if defined(_OPENMP) && !defined(SUBPAR_NO_OPENMP_RANGE) && !defined(SUBPAR_NO_OPENMP)
175#define SUBPAR_USES_OPENMP 1
176#define SUBPAR_USES_OPENMP_RANGE 1
180 #pragma omp parallel for num_threads(num_workers)
181 for (
int w = 0; w < num_workers; ++w) {
182 const Task_ start = get_start(w);
183 const Task_ length = get_length(w);
185 if constexpr(nothrow_) {
186 run_task_range(w, start, length);
189 run_task_range(w, start, length);
191 errors[w] = std::current_exception();
198#undef SUBPAR_USES_OPENMP
199#undef SUBPAR_USES_OPENMP_RANGE
202 std::vector<std::thread> workers;
203 sanisizer::reserve(workers, num_workers - 1);
205 for (
int w = 1; w < num_workers; ++w) {
206 const Task_ start = get_start(w);
207 const Task_ length = get_length(w);
209 if constexpr(nothrow_) {
210 workers.emplace_back(run_task_range, w, start, length);
212 workers.emplace_back([&run_task_range,&errors](
int w, Task_ start, Task_ length) ->
void {
214 run_task_range(w, start, length);
216 errors[w] = std::current_exception();
218 }, w, start, length);
223 const Task_ start = get_start(0);
224 const Task_ length = get_length(0);
226 if constexpr(nothrow_) {
227 run_task_range(0, start, length);
230 run_task_range(0, start, length);
232 errors[0] = std::current_exception();
237 for (
auto& wrk : workers) {
242 if constexpr(!nothrow_) {
243 for (
const auto& e : errors) {
245 std::rethrow_exception(e);