/* 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_BUFFER_HPP #define INGEN_ENGINE_BUFFER_HPP #include "BufferFactory.hpp" #include "BufferRef.hpp" #include "server.h" #include "types.hpp" #include "ingen/URIs.hpp" #include "lv2/atom/atom.h" #include "lv2/urid/urid.h" #include #include #include #include namespace ingen { enum class PortType; class Atom; namespace server { class RunContext; class INGEN_SERVER_API Buffer { public: Buffer(BufferFactory& bufs, LV2_URID type, LV2_URID value_type, uint32_t capacity, bool external = false, void* buf = nullptr); Buffer(const Buffer&) = delete; Buffer& operator=(const Buffer&) = delete; void clear(); void resize(uint32_t capacity); void copy(const RunContext& ctx, const Buffer* src); void prepare_write(RunContext& ctx); void* port_data(PortType port_type, SampleCount offset); const void* port_data(PortType port_type, SampleCount offset) const; LV2_URID type() const { return _type; } LV2_URID value_type() const { return _value_type; } uint32_t capacity() const { return _capacity; } uint32_t size() const { return is_audio() ? _capacity : sizeof(LV2_Atom) + get()->size; } using GetFn = BufferRef (BufferFactory::*)(LV2_URID, LV2_URID, uint32_t); /** Set the buffer type and optional value type for this buffer. * * @param get_func Called to get auxiliary buffers if necessary. * @param type Type of buffer. * @param value_type Type of values in buffer if applicable (for sequences). */ void set_type(GetFn get_func, LV2_URID type, LV2_URID value_type); bool is_audio() const { return _type == _factory.uris().atom_Sound; } bool is_control() const { return _type == _factory.uris().atom_Float; } bool is_sequence() const { return _type == _factory.uris().atom_Sequence; } /// Audio or float buffers only const Sample* samples() const { if (is_control()) { return static_cast( LV2_ATOM_BODY_CONST(get())); } if (is_audio()) { return static_cast(_buf); } return nullptr; } /// Audio buffers only Sample* samples() { if (is_control()) { return static_cast(LV2_ATOM_BODY(get())); } if (is_audio()) { return static_cast(_buf); } return nullptr; } /// Numeric buffers only Sample value_at(SampleCount offset) const { if (is_audio() || is_control()) { return samples()[offset]; } if (_value_buffer) { return reinterpret_cast(value())->body; } return 0.0f; } void set_block(const Sample val, const SampleCount start, const SampleCount end) { if (is_sequence()) { append_event(start, sizeof(val), _factory.uris().atom_Float, reinterpret_cast( static_cast(&val))); _value_buffer->get()->body = val; return; } assert(is_audio() || is_control()); assert(end <= _capacity / sizeof(Sample)); // Note: Do not change this without ensuring GCC can still vectorize it Sample* const buf = samples() + start; for (SampleCount i = 0; i < (end - start); ++i) { buf[i] = val; } } void add_block(const Sample val, const SampleCount start, const SampleCount end) { assert(is_audio() || is_control()); assert(end <= _capacity / sizeof(Sample)); // Note: Do not change this without ensuring GCC can still vectorize it Sample* const buf = samples() + start; for (SampleCount i = 0; i < (end - start); ++i) { buf[i] += val; } } void write_block(const Sample val, const SampleCount start, const SampleCount end, const bool add) { if (add) { add_block(val, start, end); } else { set_block(val, start, end); } } /// Audio buffers only float peak(const RunContext& ctx) const; /// Sequence buffers only void prepare_output_write(RunContext& ctx); /// Sequence buffers only bool append_event(int64_t frames, uint32_t size, uint32_t type, const uint8_t* data); /// Sequence buffers only bool append_event(int64_t frames, const LV2_Atom* body); /// Sequence buffers only bool append_event_buffer(const Buffer* buf); /// Value buffer for numeric sequences BufferRef value_buffer() { return _value_buffer; } /// Return the current value const LV2_Atom* value() const; /// Set/initialise current value in value buffer void set_value(const Atom& value); /// Return offset of the first value change after `offset` SampleCount next_value_offset(SampleCount offset, SampleCount end) const; /// Update value buffer to value as of offset void update_value_buffer(SampleCount offset); /// Set/add to audio buffer from the Sequence of Float in `src` void render_sequence(const RunContext& ctx, const Buffer* src, bool add); void set_capacity(uint32_t capacity) { _capacity = capacity; } void set_buffer(void* buf) { assert(_external); _buf = buf; } static void* aligned_alloc(size_t size); template const T* get() const { return reinterpret_cast(_buf); } template T* get() { return reinterpret_cast(_buf); } void ref() { ++_refs; } void deref() { if ((--_refs) == 0) { recycle(); } } private: friend class BufferFactory; ~Buffer(); void recycle(); BufferFactory& _factory; // NOLINTNEXTLINE(clang-analyzer-webkit.NoUncountedMemberChecker) Buffer* _next{nullptr}; ///< Intrusive linked list for BufferFactory void* _buf; ///< Actual buffer memory BufferRef _value_buffer; ///< Value buffer for numeric sequences int64_t _latest_event{0}; LV2_URID _type; LV2_URID _value_type; uint32_t _capacity; std::atomic _refs{0}; ///< Intrusive reference count bool _external; ///< Buffer is externally allocated }; } // namespace server } // namespace ingen #endif // INGEN_ENGINE_BUFFER_HPP