/*
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 "types.hpp"
#include "ingen/LV2Features.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
namespace ingen {
class FilePath;
namespace server {
class LV2Plugin;
/** 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();
bool instantiate(BufferFactory& bufs, const LilvState* state);
LilvInstance* instance() override { return instance(0); }
bool save_state(const FilePath& 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;
LilvState* load_preset(const URI& uri) override;
void apply_state(const UPtr& worker, const LilvState* state) override;
boost::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 LilvState* load_state(World& world, const FilePath& path);
protected:
struct Instance : public Raul::Noncopyable {
explicit Instance(LilvInstance* i) : instance(i) {}
~Instance() { lilv_instance_free(instance); }
LilvInstance* const instance;
};
SPtr make_instance(URIs& uris,
SampleRate rate,
uint32_t voice,
bool preparing);
inline LilvInstance* instance(uint32_t voice) {
return static_cast((*_instances)[voice]->instance);
}
using Instances = Raul::Array>;
static void drop_instances(const MPtr& 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<>
{
inline Response(uint32_t s, const void* d)
: size(s)
, data(malloc(s))
{
memcpy(data, d, s);
}
~Response() {
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;
MPtr _instances;
MPtr _prepared_instances;
const LV2_Worker_Interface* _worker_iface;
std::mutex _work_mutex;
Responses _responses;
SPtr _features;
};
} // namespace server
} // namespace ingen
#endif // INGEN_ENGINE_LV2BLOCK_HPP