/* This file is part of Ingen. Copyright 2007-2015 David Robillard <http://drobilla.net/> 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 <http://www.gnu.org/licenses/>. */ #include <stdlib.h> #include "lilv/lilv.h" #include "ingen/LV2Features.hpp" #include "ingen/Log.hpp" #include "ingen/World.hpp" #include "internals/Controller.hpp" #include "internals/Delay.hpp" #include "internals/Note.hpp" #include "internals/Time.hpp" #include "internals/Trigger.hpp" #include "InternalPlugin.hpp" #include "LV2Plugin.hpp" #include "BlockFactory.hpp" #include "ThreadManager.hpp" using namespace std; namespace Ingen { namespace Server { using namespace Internals; BlockFactory::BlockFactory(Ingen::World* world) : _world(world) , _lv2_info(new LV2Info(world)) , _has_loaded(false) { load_internal_plugins(); } BlockFactory::~BlockFactory() { for (auto& p : _plugins) delete p.second; _plugins.clear(); } const BlockFactory::Plugins& BlockFactory::plugins() { ThreadManager::assert_thread(THREAD_PRE_PROCESS); if (!_has_loaded) { _has_loaded = true; // TODO: Plugin list refreshing load_lv2_plugins(); _has_loaded = true; } return _plugins; } PluginImpl* BlockFactory::plugin(const Raul::URI& uri) { load_plugin(uri); const Plugins::const_iterator i = _plugins.find(uri); return ((i != _plugins.end()) ? i->second : NULL); } void BlockFactory::load_internal_plugins() { Ingen::URIs& uris = _world->uris(); InternalPlugin* controller_plug = ControllerNode::internal_plugin(uris); _plugins.insert(make_pair(controller_plug->uri(), controller_plug)); InternalPlugin* delay_plug = DelayNode::internal_plugin(uris); _plugins.insert(make_pair(delay_plug->uri(), delay_plug)); InternalPlugin* note_plug = NoteNode::internal_plugin(uris); _plugins.insert(make_pair(note_plug->uri(), note_plug)); InternalPlugin* time_plug = TimeNode::internal_plugin(uris); _plugins.insert(make_pair(time_plug->uri(), time_plug)); InternalPlugin* trigger_plug = TriggerNode::internal_plugin(uris); _plugins.insert(make_pair(trigger_plug->uri(), trigger_plug)); } void BlockFactory::load_plugin(const Raul::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) { LV2Plugin* const ingen_plugin = new LV2Plugin(_lv2_info, uri); ingen_plugin->lilv_plugin(plug); _plugins.insert(make_pair(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 typedef std::vector< SPtr<LilvNode> > Types; Types types; for (unsigned t = PortType::ID::AUDIO; t <= PortType::ID::ATOM; ++t) { const Raul::URI& uri(PortType((PortType::ID)t).uri()); types.push_back( SPtr<LilvNode>(lilv_new_uri(_world->lilv_world(), uri.c_str()), lilv_node_free)); } LilvNode* lv2_connectionOptional = lilv_new_uri( _world->lilv_world(), LV2_CORE__connectionOptional); 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 Raul::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( fmt("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( fmt("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 = false; for (const auto& t : types) { if (lilv_port_is_a(lv2_plug, port, t.get())) { supported = true; break; } } if (!supported && !lilv_port_has_property(lv2_plug, port, lv2_connectionOptional)) { _world->log().warn( fmt("Ignoring <%1%>; unsupported port <%2%>\n") % uri % lilv_node_as_string( lilv_port_get_symbol(lv2_plug, port))); break; } } if (!supported) { continue; } if (_plugins.find(uri) == _plugins.end()) { LV2Plugin* const plugin = new LV2Plugin(_lv2_info, uri); plugin->lilv_plugin(lv2_plug); _plugins.insert(make_pair(uri, plugin)); } } _world->log().info(fmt("Loaded %1% plugins\n") % _plugins.size()); lilv_node_free(lv2_connectionOptional); } } // namespace Server } // namespace Ingen