/*
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 "ingen/shared/URIs.hpp"
#include "lv2/lv2plug.in/ns/ext/worker/worker.h"
#include "raul/Array.hpp"
#include "raul/Maid.hpp"
#include "AudioBuffer.hpp"
#include "BufferFactory.hpp"
#include "Engine.hpp"
#include "NodeImpl.hpp"
#include "Notification.hpp"
#include "PortImpl.hpp"
#include "PortType.hpp"
#include "ThreadManager.hpp"
using namespace std;
using namespace Raul;
namespace Ingen {
namespace Server {
PortImpl::PortImpl(BufferFactory& bufs,
NodeImpl* const node,
const Raul::Symbol& name,
uint32_t index,
uint32_t poly,
PortType type,
LV2_URID buffer_type,
const Atom& value,
size_t buffer_size)
: GraphObjectImpl(bufs.uris(), node, name)
, _bufs(bufs)
, _index(index)
, _poly(poly)
, _buffer_size(buffer_size)
, _type(type)
, _buffer_type(buffer_type)
, _value(value)
, _min(bufs.forge().make(0.0f))
, _max(bufs.forge().make(1.0f))
, _last_broadcasted_value(value)
, _buffers(new Array(static_cast(poly)))
, _prepared_buffers(NULL)
, _broadcast(false)
, _set_by_user(false)
{
assert(node != NULL);
assert(_poly > 0);
const Ingen::Shared::URIs& uris = bufs.uris();
if (_buffer_size == 0) {
_buffer_size = bufs.default_buffer_size(buffer_type);
}
if (_buffer_type == 0) {
switch (_type.symbol()) {
case PortType::CONTROL:
_buffer_type = uris.atom_Float;
break;
case PortType::AUDIO:
case PortType::CV:
_buffer_type = uris.atom_Sound;
break;
default:
break;
}
}
add_property(uris.rdf_type, bufs.forge().alloc_uri(type.uri().str()));
set_property(uris.lv2_index, bufs.forge().make((int32_t)index));
}
PortImpl::~PortImpl()
{
delete _buffers;
}
bool
PortImpl::supports(const Raul::URI& value_type) const
{
return has_property(_bufs.uris().atom_supports,
_bufs.forge().alloc_uri(value_type.str()));
}
Raul::Array*
PortImpl::set_buffers(Raul::Array* buffers)
{
ThreadManager::assert_thread(THREAD_PROCESS);
Raul::Array* ret = NULL;
if (buffers != _buffers) {
ret = _buffers;
_buffers = buffers;
}
connect_buffers();
return ret;
}
bool
PortImpl::prepare_poly(BufferFactory& bufs, uint32_t poly)
{
ThreadManager::assert_thread(THREAD_PRE_PROCESS);
if (_type != PortType::CONTROL &&
_type != PortType::CV &&
_type != PortType::AUDIO) {
return false;
}
if (_poly == poly) {
return true;
}
if (_prepared_buffers && _prepared_buffers->size() != poly) {
delete _prepared_buffers;
_prepared_buffers = NULL;
}
if (!_prepared_buffers)
_prepared_buffers = new Array(poly, *_buffers, NULL);
return true;
}
void
PortImpl::prepare_poly_buffers(BufferFactory& bufs)
{
if (_prepared_buffers)
get_buffers(bufs, _prepared_buffers, _prepared_buffers->size());
}
bool
PortImpl::apply_poly(Maid& maid, uint32_t poly)
{
ThreadManager::assert_thread(THREAD_PROCESS);
if (_type != PortType::CONTROL &&
_type != PortType::CV &&
_type != PortType::AUDIO) {
return false;
}
if (!_prepared_buffers) {
return true;
}
assert(poly == _prepared_buffers->size());
_poly = poly;
// Apply a new set of buffers from a preceding call to prepare_poly
maid.push(set_buffers(_prepared_buffers));
assert(_buffers == _prepared_buffers);
_prepared_buffers = NULL;
if (is_a(PortType::CONTROL) || is_a(PortType::CV))
for (uint32_t v = 0; v < _poly; ++v)
if (_buffers->at(v))
boost::static_pointer_cast(_buffers->at(v))->set_value(
_value.get_float(), 0, 0);
assert(_buffers->size() >= poly);
assert(this->poly() == poly);
assert(!_prepared_buffers);
return true;
}
void
PortImpl::set_buffer_size(Context& context, BufferFactory& bufs, size_t size)
{
_buffer_size = size;
for (uint32_t v = 0; v < _poly; ++v)
_buffers->at(v)->resize(size);
connect_buffers();
}
void
PortImpl::connect_buffers(SampleCount offset)
{
for (uint32_t v = 0; v < _poly; ++v)
PortImpl::parent_node()->set_port_buffer(v, _index, buffer(v), offset);
}
void
PortImpl::recycle_buffers()
{
for (uint32_t v = 0; v < _poly; ++v)
_buffers->at(v) = NULL;
}
void
PortImpl::clear_buffers()
{
for (uint32_t v = 0; v < _poly; ++v)
buffer(v)->clear();
}
void
PortImpl::broadcast_value(Context& context, bool force)
{
Ingen::Forge& forge = context.engine().world()->forge();
Notification::Type ntype = Notification::PORT_VALUE;
Raul::Atom val;
switch (_type.symbol()) {
case PortType::UNKNOWN:
break;
case PortType::AUDIO:
val = forge.make(((AudioBuffer*)buffer(0).get())->peak(context));
ntype = Notification::PORT_ACTIVITY;
break;
case PortType::CONTROL:
case PortType::CV:
val = forge.make(((AudioBuffer*)buffer(0).get())->value_at(0));
break;
case PortType::ATOM:
if (_buffer_type == _bufs.uris().atom_Sequence) {
LV2_Atom_Sequence* seq = (LV2_Atom_Sequence*)buffer(0)->atom();
if (seq->atom.size > sizeof(LV2_Atom_Sequence_Body)) {
const Notification note = Notification::make(
Notification::PORT_ACTIVITY, context.start(), this, forge.make(true));
context.event_sink().write(sizeof(note), ¬e);
}
}
break;
}
if (val.is_valid() && (force || val != _last_broadcasted_value)) {
_last_broadcasted_value = val;
const Notification note = Notification::make(
ntype, context.start(), this, val);
context.event_sink().write(sizeof(note), ¬e);
}
}
} // namespace Server
} // namespace Ingen