/*
This file is part of Ingen.
Copyright 2007-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 "DisconnectAll.hpp"
#include "ArcImpl.hpp"
#include "BlockImpl.hpp"
#include "Broadcaster.hpp"
#include "CompiledGraph.hpp"
#include "Disconnect.hpp"
#include "Engine.hpp"
#include "GraphImpl.hpp"
#include "InputPort.hpp"
#include "NodeImpl.hpp"
#include "PortImpl.hpp"
#include "PreProcessContext.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
namespace ingen::server::events {
DisconnectAll::DisconnectAll(Engine& engine,
const std::shared_ptr& client,
SampleCount timestamp,
const ingen::DisconnectAll& msg)
: Event(engine, client, msg.seq, timestamp)
, _msg(msg)
, _parent(nullptr)
, _block(nullptr)
, _port(nullptr)
, _deleting(false)
{}
/** Internal version for use by other events.
*/
DisconnectAll::DisconnectAll(Engine& engine,
GraphImpl* parent,
Node* object)
: Event(engine)
, _msg{0, parent->path(), object->path()}
, _parent(parent)
, _block(dynamic_cast(object))
, _port(dynamic_cast(object))
, _deleting(true)
{}
DisconnectAll::~DisconnectAll()
{
for (auto* i : _impls) {
delete i;
}
}
bool
DisconnectAll::pre_process(PreProcessContext& ctx)
{
std::unique_lock lock(_engine.store()->mutex(),
std::defer_lock);
if (!_deleting) {
lock.lock();
_parent = dynamic_cast(_engine.store()->get(_msg.graph));
if (!_parent) {
return Event::pre_process_done(Status::PARENT_NOT_FOUND,
_msg.graph);
}
NodeImpl* const object = dynamic_cast(
_engine.store()->get(_msg.path));
if (!object) {
return Event::pre_process_done(Status::NOT_FOUND, _msg.path);
}
if (object->parent_graph() != _parent
&& object->parent()->parent_graph() != _parent) {
return Event::pre_process_done(Status::INVALID_PARENT, _msg.graph);
}
// Only one of these will succeed
_block = dynamic_cast(object);
_port = dynamic_cast(object);
if (!_block && !_port) {
return Event::pre_process_done(Status::INTERNAL_ERROR, _msg.path);
}
}
// Create disconnect events to erase adjacent arcs in parent
const auto& arcs = adjacent_arcs(_parent);
std::transform(arcs.begin(),
arcs.end(),
std::back_inserter(_impls),
[this](const auto& a) {
return new Disconnect::Impl(_engine,
_parent,
a->tail(),
dynamic_cast(a->head()));
});
// Create disconnect events to erase adjacent arcs in parent's parent
if (_port && _parent->parent()) {
auto* const grandparent = dynamic_cast(_parent->parent());
const auto& parent_arcs = adjacent_arcs(grandparent);
std::transform(parent_arcs.begin(),
parent_arcs.end(),
std::back_inserter(_impls),
[this, grandparent](const auto& a) {
return new Disconnect::Impl(_engine,
grandparent,
a->tail(),
dynamic_cast(a->head()));
});
}
if (!_deleting && ctx.must_compile(*_parent)) {
if (!(_compiled_graph = compile(*_parent))) {
return Event::pre_process_done(Status::COMPILATION_FAILED);
}
}
return Event::pre_process_done(Status::SUCCESS);
}
void
DisconnectAll::execute(RunContext& ctx)
{
if (_status == Status::SUCCESS) {
for (auto& i : _impls) {
i->execute(ctx,
!_deleting || (i->head()->parent_block() != _block));
}
}
if (_compiled_graph) {
_compiled_graph = _parent->swap_compiled_graph(std::move(_compiled_graph));
}
}
void
DisconnectAll::post_process()
{
const Broadcaster::Transfer t{*_engine.broadcaster()};
if (respond() == Status::SUCCESS) {
_engine.broadcaster()->message(_msg);
}
}
void
DisconnectAll::undo(Interface& target)
{
for (auto& i : _impls) {
target.connect(i->tail()->path(), i->head()->path());
}
}
std::set
DisconnectAll::adjacent_arcs(GraphImpl* const graph)
{
std::set arcs;
for (const auto& a : graph->arcs()) {
auto* const arc = static_cast(a.second.get());
if (_block) {
if (arc->tail()->parent_block() == _block
|| arc->head()->parent_block() == _block) {
arcs.insert(arc);
}
} else if (_port) {
if (arc->tail() == _port || arc->head() == _port) {
arcs.insert(arc);
}
}
}
return arcs;
}
} // namespace ingen::server::events