/* 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