/*
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 "raul/Maid.hpp"
#include "raul/Path.hpp"
#include "raul/log.hpp"
#include "AudioBuffer.hpp"
#include "ClientBroadcaster.hpp"
#include "ConnectionImpl.hpp"
#include "DuplexPort.hpp"
#include "Engine.hpp"
#include "EngineStore.hpp"
#include "InputPort.hpp"
#include "OutputPort.hpp"
#include "PatchImpl.hpp"
#include "PortImpl.hpp"
#include "ProcessContext.hpp"
#include "ThreadManager.hpp"
#include "events/Disconnect.hpp"
using namespace std;
using namespace Raul;
namespace Ingen {
namespace Server {
namespace Events {
Disconnect::Disconnect(Engine& engine,
Interface* client,
int32_t id,
SampleCount timestamp,
const Raul::Path& src_port_path,
const Raul::Path& dst_port_path)
: Event(engine, client, id, timestamp)
, _src_port_path(src_port_path)
, _dst_port_path(dst_port_path)
, _patch(NULL)
, _src_port(NULL)
, _dst_port(NULL)
, _impl(NULL)
, _compiled_patch(NULL)
{
}
Disconnect::Impl::Impl(Engine& e,
PatchImpl* patch,
OutputPort* s,
InputPort* d)
: _engine(e)
, _src_output_port(s)
, _dst_input_port(d)
, _patch(patch)
, _connection(patch->remove_connection(_src_output_port, _dst_input_port))
, _buffers(NULL)
{
ThreadManager::assert_thread(THREAD_PRE_PROCESS);
NodeImpl* const src_node = _src_output_port->parent_node();
NodeImpl* const dst_node = _dst_input_port->parent_node();
for (std::list::iterator i = dst_node->providers().begin();
i != dst_node->providers().end(); ++i) {
if ((*i) == src_node) {
dst_node->providers().erase(i);
break;
}
}
for (std::list::iterator i = src_node->dependants().begin();
i != src_node->dependants().end(); ++i) {
if ((*i) == dst_node) {
src_node->dependants().erase(i);
break;
}
}
_dst_input_port->decrement_num_connections();
if (_dst_input_port->num_connections() == 0) {
_buffers = new Raul::Array(_dst_input_port->poly());
_dst_input_port->get_buffers(*_engine.buffer_factory(),
_buffers, _dst_input_port->poly());
const bool is_control = _dst_input_port->is_a(PortType::CONTROL) ||
_dst_input_port->is_a(PortType::CV);
const float value = is_control ? _dst_input_port->value().get_float() : 0;
for (uint32_t i = 0; i < _buffers->size(); ++i) {
if (is_control) {
PtrCast(_buffers->at(i))->set_value(value, 0, 0);
} else {
_buffers->at(i)->clear();
}
}
}
}
void
Disconnect::pre_process()
{
Glib::RWLock::WriterLock lock(_engine.engine_store()->lock());
if (_src_port_path.parent().parent() != _dst_port_path.parent().parent()
&& _src_port_path.parent() != _dst_port_path.parent().parent()
&& _src_port_path.parent().parent() != _dst_port_path.parent()) {
_status = PARENT_DIFFERS;
Event::pre_process();
return;
}
_src_port = _engine.engine_store()->find_port(_src_port_path);
_dst_port = _engine.engine_store()->find_port(_dst_port_path);
if (_src_port == NULL || _dst_port == NULL) {
_status = PORT_NOT_FOUND;
Event::pre_process();
return;
}
NodeImpl* const src_node = _src_port->parent_node();
NodeImpl* const dst_node = _dst_port->parent_node();
// Connection to a patch port from inside the patch
if (src_node->parent_patch() != dst_node->parent_patch()) {
assert(src_node->parent() == dst_node || dst_node->parent() == src_node);
if (src_node->parent() == dst_node)
_patch = dynamic_cast(dst_node);
else
_patch = dynamic_cast(src_node);
// Connection from a patch input to a patch output (pass through)
} else if (src_node == dst_node && dynamic_cast(src_node)) {
_patch = dynamic_cast(src_node);
// Normal connection between nodes with the same parent
} else {
_patch = src_node->parent_patch();
}
assert(_patch);
if (!_patch->has_connection(_src_port, _dst_port)) {
_status = NOT_FOUND;
Event::pre_process();
return;
}
if (src_node == NULL || dst_node == NULL) {
_status = PARENT_NOT_FOUND;
Event::pre_process();
return;
}
_impl = new Impl(_engine,
_patch,
dynamic_cast(_src_port),
dynamic_cast(_dst_port));
if (_patch->enabled())
_compiled_patch = _patch->compile();
Event::pre_process();
}
bool
Disconnect::Impl::execute(ProcessContext& context, bool set_dst_buffers)
{
ThreadManager::assert_thread(THREAD_PROCESS);
ConnectionImpl* const port_connection
= _dst_input_port->remove_connection(context, _src_output_port);
if (!port_connection) {
return false;
}
if (set_dst_buffers) {
if (_buffers) {
_engine.maid()->push(_dst_input_port->set_buffers(_buffers));
} else {
_dst_input_port->setup_buffers(*_engine.buffer_factory(),
_dst_input_port->poly());
}
_dst_input_port->connect_buffers();
} else {
_dst_input_port->recycle_buffers();
}
assert(_connection);
assert(port_connection == _connection.get());
return true;
}
void
Disconnect::execute(ProcessContext& context)
{
Event::execute(context);
if (_status == SUCCESS) {
if (!_impl->execute(context, true)) {
_status = NOT_FOUND;
return;
}
_engine.maid()->push(_patch->compiled_patch());
_patch->compiled_patch(_compiled_patch);
}
}
void
Disconnect::post_process()
{
respond(_status);
if (!_status) {
_engine.broadcaster()->disconnect(_src_port->path(), _dst_port->path());
}
delete _impl;
}
} // namespace Events
} // namespace Server
} // namespace Ingen