/*
This file is part of Ingen.
Copyright 2007-2012 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
#include
#include
#include "raul/Array.hpp"
#include "raul/Maid.hpp"
#include "raul/Path.hpp"
#include "Broadcaster.hpp"
#include "EdgeImpl.hpp"
#include "Engine.hpp"
#include "EngineStore.hpp"
#include "InputPort.hpp"
#include "NodeImpl.hpp"
#include "OutputPort.hpp"
#include "PatchImpl.hpp"
#include "PortImpl.hpp"
#include "events/Disconnect.hpp"
#include "events/DisconnectAll.hpp"
#include "util.hpp"
namespace Ingen {
namespace Server {
namespace Events {
DisconnectAll::DisconnectAll(Engine& engine,
Interface* client,
int32_t id,
SampleCount timestamp,
const Raul::Path& parent_path,
const Raul::Path& node_path)
: Event(engine, client, id, timestamp)
, _parent_path(parent_path)
, _path(node_path)
, _parent(NULL)
, _node(NULL)
, _port(NULL)
, _compiled_patch(NULL)
, _deleting(false)
{
}
/** Internal version for use by other events.
*/
DisconnectAll::DisconnectAll(Engine& engine,
PatchImpl* parent,
GraphObjectImpl* object)
: Event(engine)
, _parent_path(parent->path())
, _path(object->path())
, _parent(parent)
, _node(dynamic_cast(object))
, _port(dynamic_cast(object))
, _compiled_patch(NULL)
, _deleting(true)
{
}
DisconnectAll::~DisconnectAll()
{
for (Impls::iterator i = _impls.begin(); i != _impls.end(); ++i)
delete (*i);
}
bool
DisconnectAll::pre_process()
{
Glib::RWLock::WriterLock lock(_engine.engine_store()->lock(), Glib::NOT_LOCK);
if (!_deleting) {
lock.acquire();
_parent = _engine.engine_store()->find_patch(_parent_path);
if (!_parent) {
return Event::pre_process_done(PARENT_NOT_FOUND);
}
GraphObjectImpl* object = _engine.engine_store()->find_object(_path);
if (!object) {
return Event::pre_process_done(NOT_FOUND);
}
if (object->parent_patch() != _parent
&& object->parent()->parent_patch() != _parent) {
return Event::pre_process_done(INVALID_PARENT_PATH);
}
// Only one of these will succeed
_node = dynamic_cast(object);
_port = dynamic_cast(object);
assert((_node || _port) && !(_node && _port));
}
// Find set of edges to remove
std::set to_remove;
for (Patch::Edges::const_iterator i = _parent->edges().begin();
i != _parent->edges().end(); ++i) {
EdgeImpl* const c = (EdgeImpl*)i->second.get();
if (_node) {
if (c->tail()->parent_node() == _node
|| c->head()->parent_node() == _node) {
to_remove.insert(c);
}
} else {
assert(_port);
if (c->tail() == _port || c->head() == _port) {
to_remove.insert(c);
}
}
}
// Create disconnect events (which erases from _parent->edges())
for (std::set::const_iterator i = to_remove.begin();
i != to_remove.end(); ++i) {
_impls.push_back(new Disconnect::Impl(
_engine, _parent,
dynamic_cast((*i)->tail()),
dynamic_cast((*i)->head())));
}
if (!_deleting && _parent->enabled())
_compiled_patch = _parent->compile();
return Event::pre_process_done(SUCCESS);
}
void
DisconnectAll::execute(ProcessContext& context)
{
if (_status == SUCCESS) {
for (Impls::iterator i = _impls.begin(); i != _impls.end(); ++i) {
(*i)->execute(context,
!_deleting || ((*i)->head()->parent_node() != _node));
}
}
_engine.maid()->push(_parent->compiled_patch());
_parent->compiled_patch(_compiled_patch);
}
void
DisconnectAll::post_process()
{
respond(_status);
if (!_status) {
_engine.broadcaster()->disconnect_all(_parent_path, _path);
}
}
} // namespace Events
} // namespace Server
} // namespace Ingen