/* 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_PORTIMPL_HPP #define INGEN_ENGINE_PORTIMPL_HPP #include "BufferFactory.hpp" #include "BufferRef.hpp" #include "NodeImpl.hpp" #include "RunContext.hpp" #include "server.h" #include "types.hpp" #include #include #include #include #include #include #include #include #include namespace raul { class Symbol; } // namespace raul namespace ingen { enum class PortType; class Properties; namespace server { class BlockImpl; /** A port (input or output) on a Block. * * The base implementation here is general and/or for output ports (which are * simplest), InputPort and DuplexPort override functions to provide * specialized behaviour where necessary. * * \ingroup engine */ class INGEN_SERVER_API PortImpl : public NodeImpl { public: struct SetState { enum class State { /// Partially set, first cycle: AAAAA => AAABB. HALF_SET_CYCLE_1, /// Partially set, second cycle: AAABB => BBBBB. HALF_SET_CYCLE_2, /// Fully set, first cycle (clear events if necessary). SET_CYCLE_1, /// Fully set, second cycle and onwards (done). SET }; SetState() = default; void set(const RunContext& ctx, FrameTime t, Sample v) { time = t; value = v; state = (time == ctx.start() ? State::SET : State::HALF_SET_CYCLE_1); } State state = State::SET; ///< State for setting control value Sample value = 0; ///< Value currently being set FrameTime time = 0; ///< Time value was set }; struct Voice { SetState set_state; BufferRef buffer{nullptr}; }; using Voices = raul::Array; PortImpl(BufferFactory& bufs, BlockImpl* block, const raul::Symbol& name, uint32_t index, uint32_t poly, PortType type, LV2_URID buffer_type, const Atom& value, size_t buffer_size = 0, bool is_output = true); GraphType graph_type() const override { return GraphType::PORT; } /** A port's parent is always a block, so static cast should be safe */ BlockImpl* parent_block() const { return reinterpret_cast(_parent); } /** Set the the voices (buffers) for this port in the audio thread. */ void set_voices(RunContext& ctx, raul::managed_ptr&& voices); /** Prepare for a new (external) polyphony value. * * Preprocessor thread, poly is actually applied by apply_poly. */ bool prepare_poly(BufferFactory& bufs, uint32_t poly) override; /** Apply a new polyphony value. * * Audio thread. * \a poly Must be < the most recent value passed to prepare_poly. */ bool apply_poly(RunContext& ctx, uint32_t poly) override; /** Return the number of arcs (pre-process thraed). */ virtual size_t num_arcs() const { return 0; } const Atom& value() const { return _value; } void set_value(const Atom& v) { _value = v; } const Atom& minimum() const { return _min; } const Atom& maximum() const { return _max; } /* The following two methods store the range in variables so it can be accessed in the process thread, which is required for applying control bindings from incoming MIDI data. */ void set_minimum(const Atom& min) { _min.set_rt(min); } void set_maximum(const Atom& max) { _max.set_rt(max); } BufferRef buffer(uint32_t voice) const { return _voices->at((_poly == 1) ? 0 : voice).buffer; } BufferRef prepared_buffer(uint32_t voice) const { return _prepared_voices->at(voice).buffer; } void update_set_state(const RunContext& ctx, uint32_t v); void set_voice_value(const RunContext& ctx, uint32_t voice, FrameTime time, Sample value); void set_control_value(const RunContext& ctx, FrameTime time, Sample value); /** Prepare this port to use an external driver-provided buffer. * * This will avoid allocating a buffer for the port, instead the driver * buffer is used directly. This only makes sense for ports on the * top-level graph, which are monophonic. Non-real-time, must be called * before using the port, followed by a call to set_driver_buffer() in the * processing thread. */ virtual void set_is_driver_port(BufferFactory& bufs); bool is_driver_port() const { return _is_driver_port; } /** Called once per process cycle */ virtual void pre_process(RunContext& ctx); virtual void pre_run(RunContext& ctx); virtual void post_process(RunContext& ctx); /** Clear/silence all buffers */ virtual void clear_buffers(const RunContext& ctx); /** Claim and apply buffers in the real-time thread. */ virtual bool setup_buffers(RunContext& ctx, BufferFactory& bufs, uint32_t poly); void activate(BufferFactory& bufs); void deactivate(); /** Inherit any properties from a connected neighbour. This is used for Graph ports, so e.g. a control input has the range of all the ports it is connected to. */ virtual void inherit_neighbour(const PortImpl* port, Properties& remove, Properties& add) {} virtual void connect_buffers(SampleCount offset=0); virtual void recycle_buffers(); uint32_t index() const { return _index; } void set_index(RunContext&, uint32_t index) { _index = index; } bool is_a(PortType type) const { return _type == type; } bool has_value() const; PortType type() const { return _type; } LV2_URID value_type() const { return _value.is_valid() ? _value.type() : 0; } LV2_URID buffer_type() const { return _buffer_type; } bool supports(const URIs::Quark& value_type) const; size_t buffer_size() const { return _buffer_size; } uint32_t poly() const { return _poly; } uint32_t prepared_poly() const { return (_prepared_voices) ? _prepared_voices->size() : 1; } void set_buffer_size(RunContext& ctx, BufferFactory& bufs, size_t size); /** Return true iff this port is explicitly monitored. * * This is used for plugin UIs which require monitoring for particular * ports, even if the Ingen client has not requested broadcasting in * general (e.g. for canvas animation). */ bool is_monitored() const { return _monitored; } /** Explicitly turn on monitoring for this port. */ void enable_monitoring(bool monitored) { _monitored = monitored; } /** Monitor port value and broadcast to clients periodically. */ void monitor(RunContext& ctx, bool send_now=false); BufferFactory& bufs() const { return _bufs; } BufferRef value_buffer(uint32_t voice) const; BufferRef user_buffer(RunContext&) const { return _user_buffer; } void set_user_buffer(RunContext&, BufferRef b) { _user_buffer = std::move(b); } /** Return offset of the first value change after `offset`. */ virtual SampleCount next_value_offset(SampleCount offset, SampleCount end) const; /** Update value buffer for `voice` to be current as of `offset`. */ void update_values(SampleCount offset, uint32_t voice) const; void force_monitor_update() { _force_monitor_update = true; } void set_morphable(bool is_morph, bool is_auto_morph) { _is_morph = is_morph; _is_auto_morph = is_auto_morph; } void set_type(PortType port_type, LV2_URID buffer_type); void cache_properties(); bool is_input() const { return !_is_output; } bool is_output() const { return _is_output; } bool is_morph() const { return _is_morph; } bool is_auto_morph() const { return _is_auto_morph; } bool is_logarithmic() const { return _is_logarithmic; } bool is_sample_rate() const { return _is_sample_rate; } bool is_toggled() const { return _is_toggled; } protected: using GetFn = BufferRef (BufferFactory::*)(LV2_URID, LV2_URID, uint32_t); /** Set `voices` as the buffers to be used for this port. * * This is real-time safe only if `get` is as well, use in the real-time * thread should pass &BufferFactory::claim_buffer. * * @return true iff buffers are locally owned by the port */ virtual bool get_buffers(BufferFactory& bufs, GetFn get, const raul::managed_ptr& voices, uint32_t poly, size_t num_in_arcs) const; BufferFactory& _bufs; uint32_t _index; uint32_t _poly; uint32_t _buffer_size; uint32_t _frames_since_monitor{0}; float _monitor_value{0.0f}; float _peak{0.0f}; PortType _type; LV2_URID _buffer_type; Atom _value; Atom _min; Atom _max; raul::managed_ptr _voices; raul::managed_ptr _prepared_voices; BufferRef _user_buffer; std::atomic_flag _connected_flag{false}; bool _monitored{false}; bool _force_monitor_update{false}; bool _is_morph{false}; bool _is_auto_morph{false}; bool _is_logarithmic{false}; bool _is_sample_rate{false}; bool _is_toggled{false}; bool _is_driver_port{false}; bool _is_output; }; } // namespace server } // namespace ingen #endif // INGEN_ENGINE_PORTIMPL_HPP