summaryrefslogtreecommitdiffstats
path: root/src/server/BlockFactory.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/server/BlockFactory.cpp')
-rw-r--r--src/server/BlockFactory.cpp229
1 files changed, 229 insertions, 0 deletions
diff --git a/src/server/BlockFactory.cpp b/src/server/BlockFactory.cpp
new file mode 100644
index 00000000..7dcfd6af
--- /dev/null
+++ b/src/server/BlockFactory.cpp
@@ -0,0 +1,229 @@
+/*
+ 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 <cstdlib>
+
+#include "lilv/lilv.h"
+
+#include "ingen/LV2Features.hpp"
+#include "ingen/Log.hpp"
+#include "ingen/World.hpp"
+#include "internals/BlockDelay.hpp"
+#include "internals/Controller.hpp"
+#include "internals/Note.hpp"
+#include "internals/Time.hpp"
+#include "internals/Trigger.hpp"
+
+#include "BlockFactory.hpp"
+#include "InternalPlugin.hpp"
+#include "LV2Plugin.hpp"
+#include "ThreadManager.hpp"
+
+namespace Ingen {
+namespace Server {
+
+using namespace Internals;
+
+BlockFactory::BlockFactory(Ingen::World* world)
+ : _world(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) {
+ load_lv2_plugins();
+ _has_loaded = true;
+ }
+ return _plugins;
+}
+
+std::set<PluginImpl*>
+BlockFactory::refresh()
+{
+ // Record current plugins, and those that are currently zombies
+ const Plugins old_plugins(_plugins);
+ std::set<PluginImpl*> 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<PluginImpl*> 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
+ for (const auto& z : zombies) {
+ if (!z->is_zombie()) {
+ new_plugins.insert(z);
+ }
+ }
+
+ return new_plugins;
+}
+
+PluginImpl*
+BlockFactory::plugin(const URI& uri)
+{
+ load_plugin(uri);
+ const Plugins::const_iterator i = _plugins.find(uri);
+ return ((i != _plugins.end()) ? i->second : nullptr);
+}
+
+void
+BlockFactory::load_internal_plugins()
+{
+ Ingen::URIs& uris = _world->uris();
+ InternalPlugin* block_delay_plug = BlockDelayNode::internal_plugin(uris);
+ _plugins.emplace(block_delay_plug->uri(), block_delay_plug);
+
+ InternalPlugin* controller_plug = ControllerNode::internal_plugin(uris);
+ _plugins.emplace(controller_plug->uri(), controller_plug);
+
+ InternalPlugin* note_plug = NoteNode::internal_plugin(uris);
+ _plugins.emplace(note_plug->uri(), note_plug);
+
+ InternalPlugin* time_plug = TimeNode::internal_plugin(uris);
+ _plugins.emplace(time_plug->uri(), time_plug);
+
+ InternalPlugin* trigger_plug = TriggerNode::internal_plugin(uris);
+ _plugins.emplace(trigger_plug->uri(), trigger_plug);
+}
+
+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) {
+ LV2Plugin* 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
+ typedef std::vector< SPtr<LilvNode> > Types;
+ Types types;
+ for (unsigned t = PortType::ID::AUDIO; t <= PortType::ID::ATOM; ++t) {
+ const URI& uri(PortType((PortType::ID)t).uri());
+ types.push_back(
+ SPtr<LilvNode>(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(
+ 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,
+ _world->uris().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;
+ }
+
+ auto p = _plugins.find(uri);
+ if (p == _plugins.end()) {
+ LV2Plugin* 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(fmt("Loaded %1% plugins\n") % _plugins.size());
+}
+
+} // namespace Server
+} // namespace Ingen