/* This file is part of Ingen. Copyright 2007-2016 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 . */ #ifndef INGEN_ENGINE_BLOCKIMPL_HPP #define INGEN_ENGINE_BLOCKIMPL_HPP #include #include #include #include "ingen/Node.hpp" #include "lilv/lilv.h" #include "raul/Array.hpp" #include "BufferRef.hpp" #include "PluginImpl.hpp" #include "PortType.hpp" #include "RunContext.hpp" #include "types.hpp" namespace Raul { class Maid; } namespace Ingen { namespace Server { class Buffer; class BufferFactory; class Engine; class GraphImpl; class PluginImpl; class PortImpl; class RunContext; class Worker; /** A Block in a Graph (which is also a Block). * * This is what is often called a "Module" in modular synthesizers. A Block is * a unit with input/output ports, a process() method, and some other things. * * \ingroup engine */ class BlockImpl : public Node , public boost::intrusive::slist_base_hook<> // In GraphImpl { public: typedef Raul::Array Ports; BlockImpl(PluginImpl* plugin, const Raul::Symbol& symbol, bool poly, GraphImpl* parent, SampleRate rate); virtual ~BlockImpl(); virtual GraphType graph_type() const { return GraphType::BLOCK; } BlockImpl* parent() const { return _parent; } GraphImpl* graph_parent() const { return (GraphImpl*)_parent; } const Raul::Path& path() const { return _path; } const Raul::Symbol& symbol() const { return _symbol; } void set_uri(const Raul::URI& uri) override { _path = uri_to_path(uri); _symbol = Raul::Symbol(_path.is_root() ? "main" : _path.symbol()); Node::set_uri(uri); } /** Return true iff this is main (the top level graph). * * This is sometimes called "the root graph", but the term "main" is used * to avoid ambiguity with the root path, since main does not have the path * "/", but usually "/main" to leave namespace for non-node things. */ bool is_main() const { return !_parent; } /** Activate this Block. * * This function must be called in a non-realtime thread before it is * inserted in to a graph. Any non-realtime actions that need to be * done before the Block is ready for use should be done here. */ virtual void activate(BufferFactory& bufs); /** Deactivate this Block. * * This function must be called in a non-realtime thread after the * block has been removed from its graph (i.e. processing is finished). */ virtual void deactivate(); /** Duplicate this Node. */ virtual BlockImpl* duplicate(Engine& engine, const Raul::Symbol& symbol, GraphImpl* parent) { return NULL; } /** Return true iff this block is activated */ bool activated() const { return _activated; } /** Return true iff this block is enabled (not bypassed). */ bool enabled() const { return _enabled; } /** Enable or disable (bypass) this block. */ void set_enabled(bool e) { _enabled = e; } /** Load a preset from the world for this block. */ virtual LilvState* load_preset(const Raul::URI& uri) { return NULL; } /** Restore `state`. */ virtual void apply_state(Worker* worker, const LilvState* state) {} /** Save current state as preset. */ virtual boost::optional save_preset(const Raul::URI& bundle, const Properties& props) { return boost::optional(); } /** Learn the next incoming MIDI event (for internals) */ virtual void learn() {} /** Do whatever needs doing in the process thread before process() is called */ virtual void pre_process(RunContext& context); /** Run block for an entire process cycle (calls run()). */ virtual void process(RunContext& context); /** Bypass block for an entire process cycle (called from process()). */ virtual void bypass(RunContext& context); /** Run block for a portion of process cycle (called from process()). */ virtual void run(RunContext& context) = 0; /** Do whatever needs doing in the process thread after process() is called */ virtual void post_process(RunContext& context); /** Set the buffer of a port to a given buffer (e.g. connect plugin to buffer) */ virtual void set_port_buffer(uint32_t voice, uint32_t port_num, BufferRef buf, SampleCount offset); virtual Node* port(uint32_t index) const; virtual PortImpl* port_impl(uint32_t index) const { return (*_ports)[index]; } /** Get a port by symbol. */ virtual PortImpl* port_by_symbol(const char* symbol); /** Blocks that are connected to this Block's inputs. */ std::set& providers() { return _providers; } /** Blocks that are connected to this Block's outputs. */ std::set& dependants() { return _dependants; } /** Flag block as polyphonic. * * Note this will not actually allocate voices etc., prepare_poly * and apply_poly must be called after this function to truly make * a block polyphonic. */ virtual void set_polyphonic(bool p) { _polyphonic = p; } virtual bool prepare_poly(BufferFactory& bufs, uint32_t poly); virtual bool apply_poly(RunContext& context, uint32_t poly); /** Information about the Plugin this Block is an instance of. * Not the best name - not all blocks come from plugins (ie Graph) */ virtual const Resource* plugin() const; /** Information about the Plugin this Block is an instance of. * Not the best name - not all blocks come from plugins (ie Graph) */ virtual const PluginImpl* plugin_impl() const; virtual void plugin(PluginImpl* pi) { _plugin = pi; } virtual void set_buffer_size(RunContext& context, BufferFactory& bufs, LV2_URID type, uint32_t size); /** The Graph this Block belongs to. */ inline GraphImpl* parent_graph() const { return (GraphImpl*)_parent; } uint32_t num_ports() const { return _ports ? _ports->size() : 0; } virtual uint32_t polyphony() const { return _polyphony; } virtual LilvInstance* instance() { return nullptr; } /** Mark used during graph compilation */ enum class Mark { UNVISITED, VISITING, VISITED }; Mark get_mark() const { return _mark; } void set_mark(Mark m) { _mark = m; } protected: PortImpl* nth_port_by_type(uint32_t n, bool input, PortType type); BlockImpl* _parent; Raul::Path _path; Raul::Symbol _symbol; PluginImpl* _plugin; MPtr _ports; ///< Access in audio thread only uint32_t _polyphony; std::set _providers; ///< Blocks connected to this one's input ports std::set _dependants; ///< Blocks this one's output ports are connected to Mark _mark; ///< Mark for graph compilation algorithm bool _polyphonic; bool _activated; bool _enabled; }; } // namespace Server } // namespace Ingen #endif // INGEN_ENGINE_BLOCKIMPL_HPP