/*
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 "ingen/Log.hpp"
#include "ingen/URIs.hpp"
#include "Buffer.hpp"
#include "BufferFactory.hpp"
#include "EdgeImpl.hpp"
#include "Engine.hpp"
#include "InputPort.hpp"
#include "NodeImpl.hpp"
#include "OutputPort.hpp"
#include "PatchImpl.hpp"
#include "ProcessContext.hpp"
#include "mix.hpp"
using namespace std;
namespace Ingen {
namespace Server {
InputPort::InputPort(BufferFactory& bufs,
NodeImpl* parent,
const Raul::Symbol& symbol,
uint32_t index,
uint32_t poly,
PortType type,
LV2_URID buffer_type,
const Raul::Atom& value,
size_t buffer_size)
: PortImpl(bufs, parent, symbol, index, poly, type, buffer_type, value, buffer_size)
, _num_edges(0)
{
const Ingen::URIs& uris = bufs.uris();
if (parent->graph_type() != GraphObject::PATCH) {
add_property(uris.rdf_type, uris.lv2_InputPort);
}
// Set default control range
if (type == PortType::CONTROL || type == PortType::CV) {
set_property(uris.lv2_minimum, bufs.forge().make(0.0f));
set_property(uris.lv2_maximum, bufs.forge().make(1.0f));
}
}
bool
InputPort::apply_poly(ProcessContext& context, Raul::Maid& maid, uint32_t poly)
{
bool ret = PortImpl::apply_poly(context, maid, poly);
if (!ret)
poly = 1;
assert(_buffers->size() >= poly);
return true;
}
/** Set @a buffers to the buffers to be used for this port.
* @return true iff buffers are locally owned by the port
*/
bool
InputPort::get_buffers(BufferFactory& bufs,
Raul::Array* buffers,
uint32_t poly,
bool real_time) const
{
const size_t num_edges = real_time ? _edges.size() : _num_edges;
if (is_a(PortType::AUDIO) && num_edges == 0) {
// Audio input with no edges, use shared zero buffer
for (uint32_t v = 0; v < poly; ++v) {
buffers->at(v) = bufs.silent_buffer();
}
return false;
} else if (num_edges == 1) {
if (real_time) {
if (!_edges.front().must_mix()) {
// Single non-mixing connection, use buffers directly
for (uint32_t v = 0; v < poly; ++v) {
buffers->at(v) = _edges.front().buffer(v);
}
return false;
}
}
}
// Otherwise, allocate local buffers
for (uint32_t v = 0; v < poly; ++v) {
buffers->at(v).reset();
buffers->at(v) = bufs.get(buffer_type(), _buffer_size, real_time);
buffers->at(v)->clear();
}
return true;
}
/** Add a edge. Realtime safe.
*
* The buffer of this port will be set directly to the edge's buffer
* if there is only one edge, since no copying/mixing needs to take place.
*
* Note that setup_buffers must be called after this before the change
* will audibly take effect.
*/
void
InputPort::add_edge(ProcessContext& context, EdgeImpl* c)
{
_edges.push_front(*c);
_broadcast = true; // Broadcast value/activity of connected input
}
/** Remove a edge. Realtime safe.
*
* Note that setup_buffers must be called after this before the change
* will audibly take effect.
*/
EdgeImpl*
InputPort::remove_edge(ProcessContext& context, const OutputPort* tail)
{
EdgeImpl* edge = NULL;
for (Edges::iterator i = _edges.begin(); i != _edges.end(); ++i) {
if (i->tail() == tail) {
edge = &*i;
_edges.erase(i);
break;
}
}
if (!edge) {
context.engine().log().error("Attempt to remove non-existent edge\n");
return NULL;
}
if (_edges.empty()) {
_broadcast = false; // Turn off broadcasting if no longer connected
}
return edge;
}
uint32_t
InputPort::max_tail_poly(Context& context) const
{
return parent_node()->parent_patch()->internal_poly_process();
}
static void
get_sources(const Context& context,
const EdgeImpl& edge,
uint32_t voice,
const Buffer** srcs,
uint32_t max_num_srcs,
uint32_t& num_srcs)
{
if (edge.must_mix()) {
// Mixing down voices: all tail voices => one head voice
for (uint32_t v = 0; v < edge.tail()->poly(); ++v) {
assert(num_srcs < max_num_srcs);
srcs[num_srcs++] = edge.tail()->buffer(v).get();
}
} else {
// Matching polyphony: each tail voice => corresponding head voice
assert(edge.tail()->poly() == edge.head()->poly());
assert(num_srcs < max_num_srcs);
srcs[num_srcs++] = edge.tail()->buffer(voice).get();
}
}
/** Prepare buffer for access, mixing if necessary. Realtime safe.
*/
void
InputPort::pre_process(Context& context)
{
// If value has been set (e.g. events pushed) by the user, don't smash it
if (_set_by_user)
return;
if (_edges.empty()) {
for (uint32_t v = 0; v < _poly; ++v) {
update_set_state(context, v);
}
} else if (direct_connect()) {
for (uint32_t v = 0; v < _poly; ++v) {
_buffers->at(v) = _edges.front().buffer(v);
}
} else {
const uint32_t src_poly = max_tail_poly(context);
const uint32_t max_num_srcs = _edges.size() * src_poly;
const Buffer* srcs[max_num_srcs];
for (uint32_t v = 0; v < _poly; ++v) {
// Get all the sources for this voice
uint32_t num_srcs = 0;
for (Edges::iterator e = _edges.begin(); e != _edges.end(); ++e) {
get_sources(context, *e, v, srcs, max_num_srcs, num_srcs);
}
// Then mix them into out buffer for this voice
mix(context, buffer(v).get(), srcs, num_srcs);
}
}
if (_broadcast)
broadcast_value(context, false);
}
void
InputPort::post_process(Context& context)
{
if (_set_by_user) {
if (_buffer_type == _bufs.uris().atom_Sequence) {
// Clear events received via a SetPortValue
for (uint32_t v = 0; v < _poly; ++v) {
buffer(v)->prepare_write(context);
}
}
_set_by_user = false;
}
}
bool
InputPort::direct_connect() const
{
return _edges.size() == 1
&& !_parent->path().is_root()
&& !_edges.front().must_mix();
}
} // namespace Server
} // namespace Ingen