/*
This file is part of Ingen.
Copyright 2007-2015 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 "ingen/Store.hpp"
#include "raul/Maid.hpp"
#include "raul/Path.hpp"
#include "ArcImpl.hpp"
#include "Broadcaster.hpp"
#include "Buffer.hpp"
#include "DuplexPort.hpp"
#include "Engine.hpp"
#include "GraphImpl.hpp"
#include "InputPort.hpp"
#include "OutputPort.hpp"
#include "PortImpl.hpp"
#include "ProcessContext.hpp"
#include "ThreadManager.hpp"
#include "events/Disconnect.hpp"
namespace Ingen {
namespace Server {
namespace Events {
Disconnect::Disconnect(Engine& engine,
SPtr client,
int32_t id,
SampleCount timestamp,
const Raul::Path& tail_path,
const Raul::Path& head_path)
: Event(engine, client, id, timestamp)
, _tail_path(tail_path)
, _head_path(head_path)
, _graph(NULL)
, _impl(NULL)
, _compiled_graph(NULL)
{
}
Disconnect::~Disconnect()
{
delete _impl;
}
Disconnect::Impl::Impl(Engine& e,
GraphImpl* graph,
OutputPort* s,
InputPort* d)
: _engine(e)
, _src_output_port(s)
, _dst_input_port(d)
, _arc(graph->remove_arc(_src_output_port, _dst_input_port))
, _voices(NULL)
{
ThreadManager::assert_thread(THREAD_PRE_PROCESS);
BlockImpl* const src_block = _src_output_port->parent_block();
BlockImpl* const dst_block = _dst_input_port->parent_block();
for (std::list::iterator i = dst_block->providers().begin();
i != dst_block->providers().end(); ++i) {
if ((*i) == src_block) {
dst_block->providers().erase(i);
break;
}
}
for (std::list::iterator i = src_block->dependants().begin();
i != src_block->dependants().end(); ++i) {
if ((*i) == dst_block) {
src_block->dependants().erase(i);
break;
}
}
_dst_input_port->decrement_num_arcs();
if (_dst_input_port->num_arcs() == 0) {
_voices = new Raul::Array(_dst_input_port->poly());
_dst_input_port->get_buffers(*_engine.buffer_factory(),
_voices,
_dst_input_port->poly(),
false);
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() : 0;
for (uint32_t i = 0; i < _voices->size(); ++i) {
if (is_control) {
Buffer* buf = _voices->at(i).buffer.get();
buf->set_block(value, 0, buf->nframes());
} else {
_voices->at(i).buffer->clear();
}
}
}
}
bool
Disconnect::pre_process()
{
std::unique_lock lock(_engine.store()->mutex());
if (_tail_path.parent().parent() != _head_path.parent().parent()
&& _tail_path.parent() != _head_path.parent().parent()
&& _tail_path.parent().parent() != _head_path.parent()) {
return Event::pre_process_done(Status::PARENT_DIFFERS, _head_path);
}
PortImpl* tail = dynamic_cast(_engine.store()->get(_tail_path));
if (!tail) {
return Event::pre_process_done(Status::PORT_NOT_FOUND, _tail_path);
}
PortImpl* head = dynamic_cast(_engine.store()->get(_head_path));
if (!head) {
return Event::pre_process_done(Status::PORT_NOT_FOUND, _head_path);
}
BlockImpl* const src_block = tail->parent_block();
BlockImpl* const dst_block = head->parent_block();
if (src_block->parent_graph() != dst_block->parent_graph()) {
// Arc to a graph port from inside the graph
assert(src_block->parent() == dst_block || dst_block->parent() == src_block);
if (src_block->parent() == dst_block) {
_graph = dynamic_cast(dst_block);
} else {
_graph = dynamic_cast(src_block);
}
} else if (src_block == dst_block && dynamic_cast(src_block)) {
// Arc from a graph input to a graph output (pass through)
_graph = dynamic_cast(src_block);
} else {
// Normal arc between blocks with the same parent
_graph = src_block->parent_graph();
}
if (!_graph) {
return Event::pre_process_done(Status::INTERNAL_ERROR, _head_path);
} else if (!_graph->has_arc(tail, head)) {
return Event::pre_process_done(Status::NOT_FOUND, _head_path);
}
if (src_block == NULL || dst_block == NULL) {
return Event::pre_process_done(Status::PARENT_NOT_FOUND, _head_path);
}
_impl = new Impl(_engine,
_graph,
dynamic_cast(tail),
dynamic_cast(head));
if (_graph->enabled())
_compiled_graph = _graph->compile();
return Event::pre_process_done(Status::SUCCESS);
}
bool
Disconnect::Impl::execute(ProcessContext& context, bool set_dst_buffers)
{
ArcImpl* const port_arc =
_dst_input_port->remove_arc(context, _src_output_port);
if (!port_arc) {
return false;
}
if (set_dst_buffers) {
if (_voices) {
_engine.maid()->dispose(_dst_input_port->set_voices(context, _voices));
} else {
_dst_input_port->setup_buffers(*_engine.buffer_factory(),
_dst_input_port->poly(),
true);
}
_dst_input_port->connect_buffers();
} else {
_dst_input_port->recycle_buffers();
}
return true;
}
void
Disconnect::execute(ProcessContext& context)
{
if (_status == Status::SUCCESS) {
if (_impl->execute(context, true)) {
_graph->set_compiled_graph(_compiled_graph);
} else {
_status = Status::NOT_FOUND;
}
}
}
void
Disconnect::post_process()
{
Broadcaster::Transfer t(*_engine.broadcaster());
if (respond() == Status::SUCCESS) {
_engine.broadcaster()->disconnect(_tail_path, _head_path);
}
}
} // namespace Events
} // namespace Server
} // namespace Ingen