/*
This file is part of Ingen.
Copyright 2016 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 "events/Mark.hpp"
#include "Engine.hpp"
#include "PreProcessContext.hpp"
#include "UndoStack.hpp"
#include
namespace ingen {
namespace server {
namespace events {
Mark::Mark(Engine& engine,
const SPtr& client,
SampleCount timestamp,
const ingen::BundleBegin& msg)
: Event(engine, client, msg.seq, timestamp)
, _type(Type::BUNDLE_BEGIN)
, _depth(-1)
{}
Mark::Mark(Engine& engine,
const SPtr& client,
SampleCount timestamp,
const ingen::BundleEnd& msg)
: Event(engine, client, msg.seq, timestamp)
, _type(Type::BUNDLE_END)
, _depth(-1)
{}
void
Mark::mark(PreProcessContext&)
{
const UPtr& stack = ((_mode == Mode::UNDO)
? _engine.redo_stack()
: _engine.undo_stack());
switch (_type) {
case Type::BUNDLE_BEGIN:
_depth = stack->start_entry();
break;
case Type::BUNDLE_END:
_depth = stack->finish_entry();
break;
}
}
bool
Mark::pre_process(PreProcessContext& ctx)
{
if (_depth < 0) {
mark(ctx);
}
switch (_type) {
case Type::BUNDLE_BEGIN:
ctx.set_in_bundle(true);
break;
case Type::BUNDLE_END:
ctx.set_in_bundle(false);
if (!ctx.dirty_graphs().empty()) {
for (GraphImpl* g : ctx.dirty_graphs()) {
MPtr cg = compile(*_engine.maid(), *g);
if (cg) {
_compiled_graphs.emplace(g, std::move(cg));
}
}
ctx.dirty_graphs().clear();
}
break;
}
return Event::pre_process_done(Status::SUCCESS);
}
void
Mark::execute(RunContext&)
{
for (auto& g : _compiled_graphs) {
g.first->set_compiled_graph(std::move(g.second));
}
}
void
Mark::post_process()
{
respond();
}
Event::Execution
Mark::get_execution() const
{
assert(_depth >= 0);
if (!_engine.atomic_bundles()) {
return Execution::NORMAL;
}
switch (_type) {
case Type::BUNDLE_BEGIN:
if (_depth == 1) {
return Execution::BLOCK;
}
break;
case Type::BUNDLE_END:
if (_depth == 0) {
return Execution::UNBLOCK;
}
break;
}
return Execution::NORMAL;
}
} // namespace events
} // namespace server
} // namespace ingen