/*
This file is part of Ingen.
Copyright 2007-2024 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 .
*/
#include "BlockFactory.hpp"
#include "InternalPlugin.hpp"
#include "LV2Plugin.hpp"
#include "PluginImpl.hpp"
#include "PortType.hpp"
#include "ThreadManager.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
namespace ingen::server {
BlockFactory::BlockFactory(ingen::World& world)
: _world(world)
{
load_internal_plugins();
}
const BlockFactory::Plugins&
BlockFactory::plugins()
{
ThreadManager::assert_thread(THREAD_PRE_PROCESS);
if (!_has_loaded) {
load_lv2_plugins();
_has_loaded = true;
}
return _plugins;
}
std::set>
BlockFactory::refresh()
{
// Record current plugins, and those that are currently zombies
const Plugins old_plugins(_plugins);
std::set> zombies;
for (const auto& p : _plugins) {
if (p.second->is_zombie()) {
zombies.insert(p.second);
}
}
// Re-load plugins
load_lv2_plugins();
// Add any new plugins to response
std::set> new_plugins;
for (const auto& p : _plugins) {
auto o = old_plugins.find(p.first);
if (o == old_plugins.end()) {
new_plugins.insert(p.second);
}
}
// Add any resurrected plugins to response
std::copy_if(zombies.begin(),
zombies.end(),
std::inserter(new_plugins, new_plugins.end()),
[](const auto& z) { return !z->is_zombie(); });
return new_plugins;
}
PluginImpl*
BlockFactory::plugin(const URI& uri)
{
load_plugin(uri);
const auto i = _plugins.find(uri);
return ((i != _plugins.end()) ? i->second.get() : nullptr);
}
void
BlockFactory::load_internal_plugins()
{
ingen::URIs& uris = _world.uris();
InternalPlugin* block_delay =
internals::BlockDelayNode::internal_plugin(uris);
_plugins.emplace(block_delay->uri(), block_delay);
InternalPlugin* controller =
internals::ControllerNode::internal_plugin(uris);
_plugins.emplace(controller->uri(), controller);
InternalPlugin* note = internals::NoteNode::internal_plugin(uris);
_plugins.emplace(note->uri(), note);
InternalPlugin* time = internals::TimeNode::internal_plugin(uris);
_plugins.emplace(time->uri(), time);
InternalPlugin* trigger = internals::TriggerNode::internal_plugin(uris);
_plugins.emplace(trigger->uri(), trigger);
}
void
BlockFactory::load_plugin(const URI& uri)
{
if (_has_loaded || _plugins.find(uri) != _plugins.end()) {
return;
}
LilvNode* node = lilv_new_uri(_world.lilv_world(), uri.c_str());
const LilvPlugins* plugs = lilv_world_get_all_plugins(_world.lilv_world());
const LilvPlugin* plug = lilv_plugins_get_by_uri(plugs, node);
if (plug) {
auto* const ingen_plugin = new LV2Plugin(_world, plug);
_plugins.emplace(uri, ingen_plugin);
}
lilv_node_free(node);
}
/** Loads information about all LV2 plugins into internal plugin database.
*/
void
BlockFactory::load_lv2_plugins()
{
// Build an array of port type nodes for checking compatibility
using Types = std::vector>;
Types types;
for (auto t = static_cast(PortType::AUDIO);
t <= static_cast(PortType::ATOM);
++t) {
const URI uri = port_type_uri(static_cast(t));
types.push_back(std::shared_ptr(
lilv_new_uri(_world.lilv_world(), uri.c_str()), lilv_node_free));
}
const LilvPlugins* plugins = lilv_world_get_all_plugins(_world.lilv_world());
LILV_FOREACH (plugins, i, plugins) {
const LilvPlugin* lv2_plug = lilv_plugins_get(plugins, i);
const URI uri(lilv_node_as_uri(lilv_plugin_get_uri(lv2_plug)));
// Ignore plugins that require features Ingen doesn't support
LilvNodes* features = lilv_plugin_get_required_features(lv2_plug);
bool supported = true;
LILV_FOREACH (nodes, f, features) {
const char* feature = lilv_node_as_uri(lilv_nodes_get(features, f));
if (!_world.lv2_features().is_supported(feature)) {
supported = false;
_world.log().warn("Ignoring <%1%>; required feature <%2%>\n",
uri, feature);
break;
}
}
lilv_nodes_free(features);
if (!supported) {
continue;
}
// Ignore plugins that are missing ports
if (!lilv_plugin_get_port_by_index(lv2_plug, 0)) {
_world.log().warn("Ignoring <%1%>; missing or corrupt ports\n",
uri);
continue;
}
const uint32_t n_ports = lilv_plugin_get_num_ports(lv2_plug);
for (uint32_t p = 0; p < n_ports; ++p) {
const LilvPort* port = lilv_plugin_get_port_by_index(lv2_plug, p);
supported =
std::any_of(types.begin(),
types.end(),
[&lv2_plug, &port](const auto& t) {
return lilv_port_is_a(lv2_plug, port, t.get());
});
if (!supported &&
!lilv_port_has_property(lv2_plug,
port,
_world.uris().lv2_connectionOptional)) {
_world.log().warn("Ignoring <%1%>; unsupported port <%2%>\n",
uri,
lilv_node_as_string(
lilv_port_get_symbol(lv2_plug, port)));
break;
}
}
if (!supported) {
continue;
}
auto p = _plugins.find(uri);
if (p == _plugins.end()) {
auto* const plugin = new LV2Plugin(_world, lv2_plug);
_plugins.emplace(uri, plugin);
} else if (lilv_plugin_verify(lv2_plug)) {
p->second->set_is_zombie(false);
}
}
_world.log().info("Loaded %1% plugins\n", _plugins.size());
}
} // namespace ingen::server