/*
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_LV2BLOCK_HPP
#define INGEN_ENGINE_LV2BLOCK_HPP
#include "BlockImpl.hpp"
#include "BufferRef.hpp"
#include "State.hpp"
#include "types.hpp"
#include "ingen/LV2Features.hpp"
#include "ingen/Properties.hpp"
#include "ingen/URI.hpp"
#include "lilv/lilv.h"
#include "lv2/worker/worker.h"
#include "raul/Array.hpp"
#include "raul/Maid.hpp"
#include "raul/Noncopyable.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
namespace raul {
class Symbol;
} // namespace raul
namespace ingen {
class Resource;
class URIs;
class World;
namespace server {
class BufferFactory;
class Engine;
class GraphImpl;
class LV2Plugin;
class RunContext;
class Worker;
/** An instance of a LV2 plugin.
*
* \ingroup engine
*/
class LV2Block final : public BlockImpl
{
public:
LV2Block(LV2Plugin* plugin,
const raul::Symbol& symbol,
bool polyphonic,
GraphImpl* parent,
SampleRate srate);
~LV2Block() override;
bool instantiate(BufferFactory& bufs, const LilvState* state);
LilvInstance* instance() override { return instance(0); }
bool save_state(const std::filesystem::path& dir) const override;
BlockImpl* duplicate(Engine& engine,
const raul::Symbol& symbol,
GraphImpl* parent) override;
bool prepare_poly(BufferFactory& bufs, uint32_t poly) override;
bool apply_poly(RunContext& ctx, uint32_t poly) override;
void activate(BufferFactory& bufs) override;
void deactivate() override;
LV2_Worker_Status work(uint32_t size, const void* data);
void run(RunContext& ctx) override;
void post_process(RunContext& ctx) override;
StatePtr load_preset(const URI& uri) override;
void apply_state(const std::unique_ptr& worker,
const LilvState* state) override;
std::optional save_preset(const URI& uri,
const Properties& props) override;
void set_port_buffer(uint32_t voice,
uint32_t port_num,
const BufferRef& buf,
SampleCount offset) override;
static StatePtr load_state(World& world, const std::filesystem::path& path);
protected:
struct Instance : public raul::Noncopyable {
explicit Instance(LilvInstance* i) noexcept : instance(i) {}
~Instance() { lilv_instance_free(instance); }
LilvInstance* const instance;
};
std::shared_ptr
make_instance(URIs& uris, SampleRate rate, uint32_t voice, bool preparing);
LilvInstance* instance(uint32_t voice) {
return static_cast((*_instances)[voice]->instance);
}
using Instances = raul::Array>;
static void drop_instances(const raul::managed_ptr& instances) {
if (instances) {
for (size_t i = 0; i < instances->size(); ++i) {
(*instances)[i].reset();
}
}
}
struct Response : public raul::Maid::Disposable
, public raul::Noncopyable
, public boost::intrusive::slist_base_hook<> {
Response(uint32_t s, const void* d)
: size(s)
, data(malloc(s))
{
memcpy(data, d, s);
}
~Response() override {
free(data);
}
const uint32_t size;
void* const data;
};
using Responses = boost::intrusive::slist<
Response,
boost::intrusive::cache_last,
boost::intrusive::constant_time_size>;
static LV2_Worker_Status work_respond(
LV2_Worker_Respond_Handle handle, uint32_t size, const void* data);
LV2Plugin* _lv2_plugin;
raul::managed_ptr _instances;
raul::managed_ptr _prepared_instances;
const LV2_Worker_Interface* _worker_iface{nullptr};
std::mutex _work_mutex;
Responses _responses;
std::shared_ptr _features;
};
} // namespace server
} // namespace ingen
#endif // INGEN_ENGINE_LV2BLOCK_HPP