/* 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 "BlockImpl.hpp" #include "Buffer.hpp" #include "GraphImpl.hpp" #include "PluginImpl.hpp" #include "PortImpl.hpp" #include "RunContext.hpp" #include "ThreadManager.hpp" #include "lv2/urid/urid.h" #include "raul/Array.hpp" #include "raul/Symbol.hpp" #include #include #include #include namespace ingen { namespace server { BlockImpl::BlockImpl(PluginImpl* plugin, const Raul::Symbol& symbol, bool polyphonic, GraphImpl* parent, SampleRate) : NodeImpl(plugin->uris(), parent, symbol) , _plugin(plugin) , _polyphony((polyphonic && parent) ? parent->internal_poly() : 1) , _mark(Mark::UNVISITED) , _polyphonic(polyphonic) , _activated(false) , _enabled(true) { assert(_plugin); assert(_polyphony > 0); } BlockImpl::~BlockImpl() { assert(!_activated); if (is_linked()) { reinterpret_cast(_parent)->remove_block(*this); } } Node* BlockImpl::port(uint32_t index) const { return (*_ports)[index]; } const Resource* BlockImpl::plugin() const { return _plugin; } const PluginImpl* BlockImpl::plugin_impl() const { return _plugin; } void BlockImpl::activate(BufferFactory& bufs) { ThreadManager::assert_thread(THREAD_PRE_PROCESS); _activated = true; for (uint32_t p = 0; p < num_ports(); ++p) { PortImpl* const port = _ports->at(p); port->activate(bufs); } } void BlockImpl::deactivate() { _activated = false; for (uint32_t p = 0; p < num_ports(); ++p) { PortImpl* const port = _ports->at(p); port->deactivate(); } } bool BlockImpl::prepare_poly(BufferFactory& bufs, uint32_t poly) { ThreadManager::assert_thread(THREAD_PRE_PROCESS); if (!_polyphonic) { poly = 1; } if (_ports) { for (uint32_t i = 0; i < _ports->size(); ++i) { _ports->at(i)->prepare_poly(bufs, poly); } } return true; } bool BlockImpl::apply_poly(RunContext& ctx, uint32_t poly) { if (!_polyphonic) { poly = 1; } _polyphony = poly; if (_ports) { for (uint32_t i = 0; i < num_ports(); ++i) { _ports->at(i)->apply_poly(ctx, poly); } } return true; } void BlockImpl::set_buffer_size(RunContext& ctx, BufferFactory& bufs, LV2_URID type, uint32_t size) { if (_ports) { for (uint32_t i = 0; i < _ports->size(); ++i) { PortImpl* const p = _ports->at(i); if (p->buffer_type() == type) { p->set_buffer_size(ctx, bufs, size); } } } } PortImpl* BlockImpl::nth_port_by_type(uint32_t n, bool input, PortType type) { uint32_t count = 0; for (uint32_t i = 0; _ports && i < _ports->size(); ++i) { PortImpl* const port = _ports->at(i); if (port->is_input() == input && port->type() == type) { if (count++ == n) { return port; } } } return nullptr; } PortImpl* BlockImpl::port_by_symbol(const char* symbol) { for (uint32_t p = 0; _ports && p < _ports->size(); ++p) { if (_ports->at(p)->symbol() == symbol) { return _ports->at(p); } } return nullptr; } void BlockImpl::pre_process(RunContext& ctx) { // Mix down input ports for (uint32_t i = 0; i < num_ports(); ++i) { PortImpl* const port = _ports->at(i); port->pre_process(ctx); port->connect_buffers(); } } void BlockImpl::bypass(RunContext& ctx) { if (!_ports) { return; } // Prepare port buffers for reading, converting/mixing if necessary for (uint32_t i = 0; i < _ports->size(); ++i) { _ports->at(i)->connect_buffers(); _ports->at(i)->pre_run(ctx); } // Dumb bypass for (PortType t : { PortType::AUDIO, PortType::CV, PortType::ATOM }) { for (uint32_t i = 0;; ++i) { PortImpl* in = nth_port_by_type(i, true, t); PortImpl* out = nth_port_by_type(i, false, t); if (!out) { break; // Finished writing all outputs } else if (in) { // Copy corresponding input to output for (uint32_t v = 0; v < _polyphony; ++v) { out->buffer(v)->copy(ctx, in->buffer(v).get()); } } else { // Output but no corresponding input, clear for (uint32_t v = 0; v < _polyphony; ++v) { out->buffer(v)->clear(); } } } } post_process(ctx); } void BlockImpl::process(RunContext& ctx) { pre_process(ctx); if (!_enabled) { bypass(ctx); post_process(ctx); return; } RunContext subcontext(ctx); for (SampleCount offset = 0; offset < ctx.nframes();) { // Find earliest offset of a value change SampleCount chunk_end = ctx.nframes(); for (uint32_t i = 0; _ports && i < _ports->size(); ++i) { PortImpl* const port = _ports->at(i); if (port->type() == PortType::CONTROL && port->is_input()) { const SampleCount o = port->next_value_offset( offset, ctx.nframes()); if (o < chunk_end) { chunk_end = o; } } } // Slice context into a chunk from now until the next change subcontext.slice(offset, chunk_end - offset); // Prepare port buffers for reading, converting/mixing if necessary for (uint32_t i = 0; _ports && i < _ports->size(); ++i) { _ports->at(i)->connect_buffers(offset); _ports->at(i)->pre_run(subcontext); } // Run the chunk run(subcontext); // Emit control port outputs as events for (uint32_t i = 0; _ports && i < _ports->size(); ++i) { PortImpl* const port = _ports->at(i); if (port->type() == PortType::CONTROL && port->is_output()) { // TODO: Only emit events when value has actually changed? for (uint32_t v = 0; v < _polyphony; ++v) { port->buffer(v)->append_event(offset, port->buffer(v)->value()); } } } offset = chunk_end; subcontext.slice(offset, chunk_end - offset); } post_process(ctx); } void BlockImpl::post_process(RunContext& ctx) { // Write output ports for (uint32_t i = 0; _ports && i < _ports->size(); ++i) { _ports->at(i)->post_process(ctx); } } void BlockImpl::set_port_buffer(uint32_t, uint32_t, const BufferRef&, SampleCount) { /*std::cout << path() << " set port " << port_num << " voice " << voice << " buffer " << buf << " offset " << offset << std::endl;*/ } } // namespace server } // namespace ingen