/*
This file is part of Ingen.
Copyright 2015-2017 David Robillard
Ingen is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or any later version.
Ingen is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU Affero General Public License for details.
You should have received a copy of the GNU Affero General Public License
along with Ingen. If not, see .
*/
#include "Task.hpp"
#include "BlockImpl.hpp"
#include "RunContext.hpp"
#include
#include
#include
namespace ingen::server {
void
Task::run(RunContext& ctx)
{
switch (_mode) {
case Mode::SINGLE:
// fprintf(stderr, "%u run %s\n", context.id(), _block->path().c_str());
_block->process(ctx);
break;
case Mode::SEQUENTIAL:
for (const auto& task : _children) {
task->run(ctx);
}
break;
case Mode::PARALLEL:
// Initialize (not) done state of sub-tasks
for (const auto& task : _children) {
task->set_done(false);
}
// Grab the first sub-task
_next = 0;
_done_end = 0;
Task* t = steal(ctx);
// Allow other threads to steal sub-tasks
ctx.claim_task(this);
// Run available tasks until this task is finished
for (; t; t = get_task(ctx)) {
t->run(ctx);
}
ctx.claim_task(nullptr);
break;
}
set_done(true);
}
Task*
Task::steal(RunContext&)
{
if (_mode == Mode::PARALLEL) {
const unsigned i = _next++;
if (i < _children.size()) {
return _children[i].get();
}
}
return nullptr;
}
Task*
Task::get_task(RunContext& ctx)
{
// Attempt to "steal" a task from ourselves
Task* t = steal(ctx);
if (t) {
return t;
}
while (true) {
// Push done end index as forward as possible
while (_done_end < _children.size() && _children[_done_end]->done()) {
++_done_end;
}
if (_done_end >= _children.size()) {
return nullptr; // All child tasks are finished
}
// All child tasks claimed, but some are unfinished, steal a task
if ((t = ctx.steal_task())) {
return t;
}
/* All child tasks are claimed, and we failed to steal any tasks. Spin
to prevent blocking, though it would probably be wiser to wait for a
signal in non-main threads, and maybe even in the main thread
depending on your real-time safe philosophy... more experimentation
here is needed. */
}
}
std::unique_ptr
Task::simplify(std::unique_ptr&& task)
{
if (task->mode() == Mode::SINGLE) {
return std::move(task);
}
std::unique_ptr ret = std::make_unique(task->mode());
for (auto&& c : task->_children) {
auto child = simplify(std::move(c));
if (!child->empty()) {
if (child->mode() == task->mode()) {
// Merge child into parent
for (auto&& grandchild : child->_children) {
ret->append(std::move(grandchild));
}
} else {
// Add child task
ret->append(std::move(child));
}
}
}
if (ret->_children.size() == 1) {
return std::move(ret->_children.front());
}
return ret;
}
void
Task::dump(const std::function& sink,
unsigned indent,
bool first) const
{
if (!first) {
sink("\n");
for (unsigned i = 0; i < indent; ++i) {
sink(" ");
}
}
if (_mode == Mode::SINGLE) {
sink(_block->path());
} else {
sink(((_mode == Mode::SEQUENTIAL) ? "(seq " : "(par "));
for (size_t i = 0; i < _children.size(); ++i) {
_children[i]->dump(sink, indent + 5, i == 0);
}
sink(")");
}
}
} // namespace ingen::server