/* This file is part of Ingen.
 * Copyright (C) 2007 Dave Robillard <http://drobilla.net>
 * 
 * Ingen is free software; you can redistribute it and/or modify it under the
 * terms of the GNU General Public License as published by the Free Software
 * Foundation; either version 2 of the License, or (at your option) 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 General Public License for details.
 * 
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
 */

#include "NodeFactory.hpp"
#include CONFIG_H_PATH
#include <cstdlib>
#include <pthread.h>
#include <dirent.h>
#include <float.h>
#include <cmath>
#include "ThreadManager.hpp"
#include "MidiNoteNode.hpp"
#include "MidiTriggerNode.hpp"
#include "MidiControlNode.hpp"
#include "TransportNode.hpp"
#include "Plugin.hpp"
#include "Patch.hpp"
#ifdef HAVE_SLV2
#include "LV2Node.hpp"
#include <slv2/slv2.h>
#endif
#ifdef HAVE_LADSPA
#include "LADSPANode.hpp"
#endif
#ifdef HAVE_DSSI
#include "DSSINode.hpp"
#endif

using namespace std;

namespace Ingen {


/* I am perfectly aware that the vast majority of this class is a 
 * vomit inducing nightmare at the moment ;)
 */



NodeFactory::NodeFactory(Ingen::Shared::World* world)
	: _world(world)
	, _has_loaded(false)
{
	// Add builtin plugin types to _internal_plugins list
	// FIXME: ewwww, definitely a better way to do this!

	Patch* parent = new Patch(*world->local_engine, "dummy", 1, NULL, 1, 1, 1);

	Node* n = NULL;
	n = new MidiNoteNode("foo", 1, parent, 1, 1);
	_internal_plugins.push_back(new Plugin(n->plugin()));
	delete n;
	n = new MidiTriggerNode("foo", 1, parent, 1, 1);
	_internal_plugins.push_back(new Plugin(n->plugin()));
	delete n;
	n = new MidiControlNode("foo", 1, parent, 1, 1);
	_internal_plugins.push_back(new Plugin(n->plugin()));
	delete n;
	n = new TransportNode("foo", 1, parent, 1, 1);
	_internal_plugins.push_back(new Plugin(n->plugin()));
	delete n;
	
	delete parent;
}


NodeFactory::~NodeFactory()
{
	for (list<Plugin*>::iterator i = _plugins.begin(); i != _plugins.end(); ++i)
		delete (*i);
	_plugins.clear();

	for (Libraries::iterator i = _libraries.begin(); i != _libraries.end(); ++i) {
		delete i->second;
	}
	_libraries.clear();
}


Glib::Module*
NodeFactory::library(const string& path)
{
	Glib::Module* plugin_library = NULL;
	Libraries::iterator library_i = _libraries.find(path);
	if (library_i != _libraries.end()) {
		plugin_library = library_i->second;
		assert(plugin_library);
	} else {
		plugin_library = new Glib::Module(path, Glib::MODULE_BIND_LOCAL | Glib::MODULE_BIND_LAZY);
		if (plugin_library && *plugin_library) {
			_libraries.insert(make_pair(path, plugin_library));
			return plugin_library;
		}
	}

	return NULL;
}


const Plugin*
NodeFactory::plugin(const string& uri)
{
	for (list<Plugin*>::iterator i = _plugins.begin(); i != _plugins.end(); ++i)
		if ((*i)->uri() == uri)
			return (*i);

	return NULL;
}


/** DEPRECATED: Find a plugin by type, lib, label.
 *
 * Do not use.
 */
const Plugin*
NodeFactory::plugin(const string& type, const string& lib, const string& label)
{
	if (type == "" || lib == "" || label == "")
		return NULL;

	for (list<Plugin*>::iterator i = _plugins.begin(); i != _plugins.end(); ++i)
		if ((*i)->type_string() == type && (*i)->lib_name() == lib && (*i)->plug_label() == label)
			return (*i);

	cerr << "ERROR: Failed to find " << type << " plugin " << lib << " / " << label << endl;

	return NULL;
}


void
NodeFactory::load_plugins()
{
	assert(ThreadManager::current_thread_id() == THREAD_PRE_PROCESS);

	// Only load if we havn't already, so every client connecting doesn't cause
	// this (expensive!) stuff to happen.  Not the best solution - would be nice
	// if clients could refresh plugins list for whatever reason :/
	if (!_has_loaded) {
		_plugins.clear();
		_plugins = _internal_plugins;
	
#ifdef HAVE_SLV2
		load_lv2_plugins();
#endif
#ifdef HAVE_DSSI
		load_dssi_plugins();
#endif
#ifdef HAVE_LADSPA
		load_ladspa_plugins();
#endif
		
		_has_loaded = true;
	}

	//cerr << "[NodeFactory] # Plugins: " << _plugins.size() << endl;
}


/** Loads a plugin.
 *
 * Calls the load_*_plugin functions to actually do things, just a wrapper.
 */
Node*
NodeFactory::load_plugin(const Plugin* a_plugin,
                         const string& name,
                         uint32_t      poly,
                         Patch*        parent)
{
	assert(parent != NULL);
	assert(poly == 1 || poly == parent->internal_poly());
	assert(a_plugin);

	Node* r = NULL;
	Plugin* plugin = NULL;

	const SampleRate srate       = parent->sample_rate();
	const size_t     buffer_size = parent->buffer_size();

	// FIXME FIXME FIXME: double lookup
	
	// Attempt to find the plugin in loaded DB
	if (a_plugin->type() != Plugin::Internal) {

		// DEPRECATED: Search by lib name / plug label
		if (a_plugin->uri().length() == 0) {
			assert(a_plugin->lib_name().length() > 0 && a_plugin->plug_label().length() > 0);
			//cerr << "Searching for: " << a_plugin->lib_name() << " : " << a_plugin->plug_label() << endl;
			for (list<Plugin*>::iterator i = _plugins.begin(); i != _plugins.end(); ++i) {
				//cerr << (*i)->lib_name() << " : " << (*i)->plug_label() << endl;
				if (a_plugin->lib_name() == (*i)->lib_name() && a_plugin->plug_label() == (*i)->plug_label()) {
					plugin = *i;
					break;
				}
			}
		} else {
			// Search by URI
			for (list<Plugin*>::iterator i = _plugins.begin(); i != _plugins.end(); ++i) {
				if (a_plugin->uri() == (*i)->uri()) {
					plugin = *i;
					break;
				}
			}
		}

		if (plugin == NULL) {
			cerr << "DID NOT FIND PLUGIN " << name << endl;
			return NULL;
		}
	}

	switch (a_plugin->type()) {
#ifdef HAVE_SLV2
	case Plugin::LV2:
		r = load_lv2_plugin(plugin->uri(), name, poly, parent, srate, buffer_size);
		break;
#endif
#ifdef HAVE_DSSI
	case Plugin::DSSI:
		r = load_dssi_plugin(plugin->uri(), name, poly, parent, srate, buffer_size);
		break;
#endif
#ifdef HAVE_LADSPA
	case Plugin::LADSPA:
		r = load_ladspa_plugin(plugin->uri(), name, poly, parent, srate, buffer_size);
		break;
#endif
	case Plugin::Internal:
		r = load_internal_plugin(a_plugin->uri(), name, poly, parent, srate, buffer_size);
		break;
	default:
		cerr << "[NodeFactory] WARNING: Unknown plugin type." << endl;
	}

	return r;
}


/** Loads an internal plugin.
 */
Node*
NodeFactory::load_internal_plugin(const string& uri,
                                  const string& name,
                                  uint32_t      poly,
                                  Patch*        parent,
                                  SampleRate    srate,
                                  size_t        buffer_size)
{
	assert(parent != NULL);
	assert(poly == 1 || poly == parent->internal_poly());
	assert(uri.length() > 6);
	assert(uri.substr(0, 6) == "ingen:");

	for (list<Plugin*>::iterator i = _internal_plugins.begin(); i != _internal_plugins.end(); ++i)
		if ((*i)->uri() == uri)
			return (*i)->instantiate(name, poly, parent, srate, buffer_size);
	
	return NULL;
}


#ifdef HAVE_SLV2

/** Loads information about all LV2 plugins into internal plugin database.
 */
void
NodeFactory::load_lv2_plugins()
{
	SLV2Plugins plugins = slv2_world_get_all_plugins(_world->slv2_world);

	//cerr << "[NodeFactory] Found " << slv2_plugins_size(plugins) << " LV2 plugins:" << endl;

	for (unsigned i=0; i < slv2_plugins_size(plugins); ++i) {

		SLV2Plugin lv2_plug = slv2_plugins_get_at(plugins, i);

		const char* uri = (const char*)slv2_plugin_get_uri(lv2_plug);
		assert(uri);
		//cerr << "\t" << uri << endl;

		Plugin* plug = NULL;

		bool found = false;
		for (list<Plugin*>::const_iterator i = _plugins.begin(); i != _plugins.end(); ++i) {
			if (!strcmp((*i)->uri().c_str(), uri)) {
				plug = (*i);
				found = true;
				break;
			}
		}

		if (!found)
			plug = new Plugin(Plugin::LV2, uri);

		plug->slv2_plugin(lv2_plug);
		plug->module(NULL); // FIXME?
		plug->lib_path(slv2_uri_to_path(slv2_plugin_get_library_uri(lv2_plug)));
		char* name = slv2_plugin_get_name(lv2_plug);
		if (!name) {
			cerr << "ERROR: LV2 Plugin " << uri << " has no name.  Ignoring." << endl;
			continue;
		}
		plug->name(name);
		free(name);
		
		if (!found)
			_plugins.push_back(plug);
	}

	slv2_plugins_free(_world->slv2_world, plugins);
}


/** Loads a LV2 plugin.
 * Returns 'poly' independant plugins as a Node*
 */
Node*
NodeFactory::load_lv2_plugin(const string& plug_uri,
                             const string& node_name,
                             uint32_t      poly,
                             Patch*        parent,
                             SampleRate    srate,
                             size_t        buffer_size)
{
	// Find (internal) Plugin
	Plugin* plugin = NULL;
	list<Plugin*>::iterator i;
	for (i = _plugins.begin(); i != _plugins.end(); ++i) {
		plugin = (*i);
		if ((*i)->uri() == plug_uri) break;
	}
	
	Node* n = NULL;

	if (plugin) {
		n = new LV2Node(plugin, node_name, poly, parent, srate, buffer_size);
		bool success = ((LV2Node*)n)->instantiate();
		if (!success) {
			delete n;
			n = NULL;
		}
	}
	
	return n;
}

#endif // HAVE_SLV2


#ifdef HAVE_DSSI

/** Loads information about all DSSI plugins into internal plugin database.
 */
void
NodeFactory::load_dssi_plugins()
{
	// FIXME: too much code duplication with load_ladspa_plugin
	
	char* env_dssi_path = getenv("DSSI_PATH");
	string dssi_path;
	if (!env_dssi_path) {
	 	cerr << "[NodeFactory] DSSI_PATH is empty.  Assuming /usr/lib/dssi:/usr/local/lib/dssi:~/.dssi" << endl;
		dssi_path = string("/usr/lib/dssi:/usr/local/lib/dssi:").append(
			getenv("HOME")).append("/.dssi");
	} else {
		dssi_path = env_dssi_path;
	}
	
	string dir;
	string full_lib_name;
	
	// Yep, this should use an sstream alright..
	while (dssi_path != "") {
		dir = dssi_path.substr(0, dssi_path.find(':'));
		if (dssi_path.find(':') != string::npos)
			dssi_path = dssi_path.substr(dssi_path.find(':')+1);
		else
			dssi_path = "";
	
		DIR* pdir = opendir(dir.c_str());
		if (pdir == NULL) {
			//cerr << "[NodeFactory] Unreadable directory in DSSI_PATH: " << dir.c_str() << endl;
			continue;
		}

		struct dirent* pfile;
		while ((pfile = readdir(pdir))) {
	
			DSSI_Descriptor_Function df         = NULL;
			DSSI_Descriptor*         descriptor = NULL;
			
			if (!strcmp(pfile->d_name, ".") || !strcmp(pfile->d_name, ".."))
				continue;
			
			full_lib_name = dir +"/"+ pfile->d_name;
			
			Glib::Module* plugin_library = library(full_lib_name);
			if (!plugin_library) {
				cerr << "WARNING: Failed to load library " << full_lib_name << endl;
				continue;
			}
	
			bool found = plugin_library->get_symbol("dssi_descriptor", (void*&)df);
			if (!found || !df) {
				// Not a DSSI plugin library
				cerr << "WARNING: Non-DSSI library found in DSSI path: " <<
					full_lib_name << endl;
				_libraries.erase(full_lib_name);
				delete plugin_library;
				continue;
			}

			const LADSPA_Descriptor* ld = NULL;
			
			for (unsigned long i=0; (descriptor = (DSSI_Descriptor*)df(i)) != NULL; ++i) {
				ld = descriptor->LADSPA_Plugin;
				assert(ld != NULL);
				string uri = string("dssi:") + pfile->d_name +":"+ ld->Label;
				Plugin* plugin = new Plugin(Plugin::DSSI, uri);
				assert(plugin_library != NULL);
				plugin->module(plugin_library);
				plugin->lib_path(dir + "/" + pfile->d_name);
				plugin->plug_label(ld->Label);
				plugin->name(ld->Name);
				plugin->type(Plugin::DSSI);
				plugin->id(ld->UniqueID);

				bool found = false;
				for (list<Plugin*>::const_iterator i = _plugins.begin(); i != _plugins.end(); ++i) {
					if ((*i)->uri() == plugin->uri()) {
						cerr << "Warning: Duplicate DSSI plugin (" << plugin->lib_name() << ":"
							<< plugin->plug_label() << ")" << " found.\nUsing " << (*i)->lib_path()
							<< " version." << endl;
						found = true;
						break;
					}
				}
				if (!found)
					_plugins.push_back(plugin);
				else
					delete plugin;
			}
		}
		closedir(pdir);
	}
}


/** Creates a Node by instancing a DSSI plugin.
 */
Node*
NodeFactory::load_dssi_plugin(const string& uri,
                              const string& name, uint32_t poly, Patch* parent, SampleRate srate, size_t buffer_size)
{
	// FIXME: awful code duplication here
	
	assert(uri != "");
	assert(name != "");
	assert(poly > 0);
	
	DSSI_Descriptor_Function df = NULL;
	const Plugin* plugin = NULL;
	Node* n = NULL;
	
	// Attempt to find the lib
	list<Plugin*>::iterator i;
	for (i = _plugins.begin(); i != _plugins.end(); ++i) {
		plugin = (*i);
		if (plugin->uri() == uri) break;
	}

	assert(plugin->id() != 0);

	if (i == _plugins.end()) {
		cerr << "Did not find DSSI plugin " << uri << " in database." << endl;
		return NULL;
	} else {
		assert(plugin);
		if (!plugin->module()->get_symbol("dssi_descriptor", (void*&)df)) {
			cerr << "Looks like this isn't a DSSI plugin." << endl;
			return NULL;
		}
	}

	// Attempt to find the plugin in lib
	DSSI_Descriptor* descriptor = NULL;
	for (unsigned long i=0; (descriptor = (DSSI_Descriptor*)df(i)) != NULL; ++i) {
		if (descriptor->LADSPA_Plugin != NULL
				&& descriptor->LADSPA_Plugin->UniqueID == plugin->id()) {
			break;
		}
	}
	
	if (descriptor == NULL) {
		cerr << "Could not find plugin \"" << plugin->id() << "\" in " << plugin->lib_name() << endl;
		return NULL;
	}

	n = new DSSINode(plugin, name, poly, parent, descriptor, srate, buffer_size);

	bool success = ((DSSINode*)n)->instantiate();
	if (!success) {
		delete n;
		n = NULL;
	}

	return n;
}
#endif // HAVE_DSSI


#ifdef HAVE_LADSPA
/** Loads information about all LADSPA plugins into internal plugin database.
 */
void
NodeFactory::load_ladspa_plugins()
{
	char* env_ladspa_path = getenv("LADSPA_PATH");
	string ladspa_path;
	if (!env_ladspa_path) {
	 	cerr << "[NodeFactory] LADSPA_PATH is empty.  Assuming /usr/lib/ladspa:/usr/local/lib/ladspa:~/.ladspa" << endl;
		ladspa_path = string("/usr/lib/ladspa:/usr/local/lib/ladspa:").append(
			getenv("HOME")).append("/.ladspa");
	} else {
		ladspa_path = env_ladspa_path;
	}

	string dir;
	string full_lib_name;
	
	// Yep, this should use an sstream alright..
	while (ladspa_path != "") {
		dir = ladspa_path.substr(0, ladspa_path.find(':'));
		if (ladspa_path.find(':') != string::npos)
			ladspa_path = ladspa_path.substr(ladspa_path.find(':')+1);
		else
			ladspa_path = "";
	
		DIR* pdir = opendir(dir.c_str());
		if (pdir == NULL) {
			//cerr << "[NodeFactory] Unreadable directory in LADSPA_PATH: " << dir.c_str() << endl;
			continue;
		}

		struct dirent* pfile;
		while ((pfile = readdir(pdir))) {
	
			LADSPA_Descriptor_Function df         = NULL;
			LADSPA_Descriptor*         descriptor = NULL;
			
			if (!strcmp(pfile->d_name, ".") || !strcmp(pfile->d_name, ".."))
				continue;
			
			full_lib_name = dir +"/"+ pfile->d_name;
			
			// Ignore stupid libtool files.  Kludge alert.
			if (full_lib_name.substr(full_lib_name.length()-3) == ".la") {
				cerr << "WARNING: Skipping stupid libtool file " << pfile->d_name << endl;
				continue;
			}
			
			Glib::Module* plugin_library = library(full_lib_name);
			if (!plugin_library) {
				cerr << "WARNING: Failed to load library " << full_lib_name << endl;
				continue;
			}
			
			bool found = plugin_library->get_symbol("ladspa_descriptor", (void*&)df);
			if (!found || !df) {
				cerr << "WARNING: Non-LADSPA library found in LADSPA path: " <<
					full_lib_name << endl;
				// Not a LADSPA plugin library
				_libraries.erase(full_lib_name);
				delete plugin_library;
				continue;
			}

			for (unsigned long i=0; (descriptor = (LADSPA_Descriptor*)df(i)) != NULL; ++i) {
				char id_str[11];
				snprintf(id_str, 11, "%lu", descriptor->UniqueID);
				string uri = string("ladspa:").append(id_str);
				Plugin* plugin = new Plugin(Plugin::LADSPA, uri);
				
				assert(plugin_library != NULL);
				plugin->module(plugin_library);
				plugin->lib_path(dir + "/" + pfile->d_name);
				plugin->plug_label(descriptor->Label);
				plugin->name(descriptor->Name);
				plugin->type(Plugin::LADSPA);
				plugin->id(descriptor->UniqueID);
				
				bool found = false;
				for (list<Plugin*>::const_iterator i = _plugins.begin(); i != _plugins.end(); ++i) {
					if ((*i)->uri() == plugin->uri()) {
						cerr << "Warning: Duplicate LADSPA plugin " << plugin->uri()
							<< " found.\nChoosing " << (*i)->lib_path()
							<< " over " << plugin->lib_path() << endl;
						found = true;
						break;
					}
				}
				if (!found)
					_plugins.push_back(plugin);
				else
					delete plugin;
			}
		}
		closedir(pdir);
	}
}


/** Loads a LADSPA plugin.
 * Returns 'poly' independant plugins as a Node*
 */
Node*
NodeFactory::load_ladspa_plugin(const string& uri,
                                const string& name,
                                uint32_t      poly,
                                Patch*        parent,
                                SampleRate    srate,
                                size_t        buffer_size)
{
	assert(uri != "");
	assert(name != "");
	assert(poly > 0);
	
	LADSPA_Descriptor_Function df = NULL;
	Plugin* plugin = NULL;
	Node* n = NULL;
	
	// Attempt to find the lib
	list<Plugin*>::iterator i;
	for (i = _plugins.begin(); i != _plugins.end(); ++i) {
		plugin = (*i);
		if (plugin->uri() == uri) break;
	}

	assert(plugin->id() != 0);

	if (i == _plugins.end()) {
		cerr << "Did not find LADSPA plugin " << uri << " in database." << endl;
		return NULL;
	} else {
		if (!plugin->module()->get_symbol("ladspa_descriptor", (void*&)df)) {
			cerr << "Looks like this isn't a LADSPA plugin." << endl;
			return NULL;
		}
	}

	// Attempt to find the plugin in lib
	LADSPA_Descriptor* descriptor = NULL;
	for (unsigned long i=0; (descriptor = (LADSPA_Descriptor*)df(i)) != NULL; ++i) {
		if (descriptor->UniqueID == plugin->id()) {
			break;
		}
	}
	
	if (descriptor == NULL) {
		cerr << "Could not find plugin \"" << plugin->id() << "\" in " << plugin->lib_path() << endl;
		return NULL;
	}

	n = new LADSPANode(plugin, name, poly, parent, descriptor, srate, buffer_size);

	bool success = ((LADSPANode*)n)->instantiate();
	if (!success) {
		delete n;
		n = NULL;
	}
	
	return n;
}


#endif // HAVE_LADSPA


} // namespace Ingen