1
0
mirror of https://github.com/cookiengineer/audacity synced 2025-05-01 16:19:43 +02:00
audacity/lib-src/slv2/utils/ladspa2lv2.c
2010-01-24 09:19:39 +00:00

379 lines
11 KiB
C

/* ladspa2lv2
* Copyright (C) 2007 Dave Robillard <http://drobilla.net>
*
* This program 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.
*
* This program 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 more 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 <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <inttypes.h>
#include <stdbool.h>
#include <float.h>
#include <math.h>
#include <dlfcn.h>
#include <librdf.h>
#include "ladspa.h"
#define U(x) ((const unsigned char*)(x))
#define NS_RDF(x) "http://www.w3.org/1999/02/22-rdf-syntax-ns#" x
#define NS_LV2(x) "http://lv2plug.in/ns/lv2core#" x
#define NS_DOAP(x) "http://usefulinc.com/ns/doap#" x
librdf_world* world = NULL;
void
add_resource(librdf_model* model,
librdf_node* subject,
const char* predicate_uri,
const char* object_uri)
{
librdf_node* predicate = librdf_new_node_from_uri_string(world, U(predicate_uri));
librdf_node* object = librdf_new_node_from_uri_string(world, U(object_uri));
librdf_statement* triple = librdf_new_statement_from_nodes(world, subject, predicate, object);
librdf_model_add_statement(model, triple);
//librdf_free_statement(triple);
}
void
add_node(librdf_model* model,
librdf_node* subject,
const char* predicate_uri,
librdf_node* object)
{
librdf_node* predicate = librdf_new_node_from_uri_string(world, U(predicate_uri));
librdf_statement* triple = librdf_new_statement_from_nodes(world, subject, predicate, object);
librdf_model_add_statement(model, triple);
//librdf_free_statement(triple);
}
void
add_string(librdf_model* model,
librdf_node* subject,
const char* predicate_uri,
const char* object_string)
{
librdf_node* predicate = librdf_new_node_from_uri_string(world, U(predicate_uri));
librdf_node* object = librdf_new_node_from_literal(world, U(object_string), NULL, 0);
librdf_statement* triple = librdf_new_statement_from_nodes(world, subject, predicate, object);
librdf_model_add_statement(model, triple);
//librdf_free_statement(triple);
}
void
add_int(librdf_model* model,
librdf_node* subject,
const char* predicate_uri,
int object_int)
{
static const size_t MAX_LEN = 21; // strlen(2^64) + 1
char object_str[MAX_LEN];
snprintf(object_str, MAX_LEN, "%d", object_int);
librdf_uri* type = librdf_new_uri(world, U("http://www.w3.org/2001/XMLSchema#integer"));
librdf_node* predicate = librdf_new_node_from_uri_string(world, U(predicate_uri));
librdf_node* object = librdf_new_node_from_typed_literal(world, U(object_str), NULL, type);
librdf_statement* triple = librdf_new_statement_from_nodes(world, subject, predicate, object);
librdf_model_add_statement(model, triple);
//librdf_free_statement(triple);
librdf_free_uri(type);
}
void
add_float(librdf_model* model,
librdf_node* subject,
const char* predicate_uri,
float object_float)
{
static const size_t MAX_LEN = 64; // ?
char object_str[MAX_LEN];
snprintf(object_str, MAX_LEN, "%f", object_float);
librdf_uri* type = librdf_new_uri(world, U("http://www.w3.org/2001/XMLSchema#decimal"));
librdf_node* predicate = librdf_new_node_from_uri_string(world, U(predicate_uri));
librdf_node* object = librdf_new_node_from_typed_literal(world, U(object_str), NULL, type);
librdf_statement* triple = librdf_new_statement_from_nodes(world, subject, predicate, object);
librdf_model_add_statement(model, triple);
//librdf_free_statement(triple);
librdf_free_uri(type);
}
LADSPA_Descriptor*
load_ladspa_plugin(const char* lib_path, unsigned long index)
{
void* const handle = dlopen(lib_path, RTLD_LAZY);
if (handle == NULL)
return NULL;
LADSPA_Descriptor_Function df
= (LADSPA_Descriptor_Function)dlsym(handle, "ladspa_descriptor");
if (df == NULL) {
dlclose(handle);
return NULL;
}
LADSPA_Descriptor* const descriptor = (LADSPA_Descriptor*)df(index);
return descriptor;
}
void
add_port_range(LADSPA_Descriptor* plugin,
unsigned long port_index,
librdf_model* model,
librdf_node* port)
{
LADSPA_PortRangeHintDescriptor hint_descriptor
= plugin->PortRangeHints[port_index].HintDescriptor;
bool range_valid = false;
float upper=1.0f, lower=0.0f, normal=0.0f;
/* Convert hints */
if (LADSPA_IS_HINT_SAMPLE_RATE(hint_descriptor)) {
add_resource(model, port, NS_LV2("portHint"), NS_LV2("sampleRate"));
upper = plugin->PortRangeHints[port_index].UpperBound;
lower = plugin->PortRangeHints[port_index].LowerBound;
range_valid = true;
}
if (LADSPA_IS_HINT_INTEGER(hint_descriptor)) {
add_resource(model, port, NS_LV2("portHint"), NS_LV2("integer"));
upper = plugin->PortRangeHints[port_index].UpperBound;
lower = plugin->PortRangeHints[port_index].LowerBound;
range_valid = true;
}
if (LADSPA_IS_HINT_TOGGLED(hint_descriptor)) {
add_resource(model, port, NS_LV2("portHint"), NS_LV2("toggled"));
upper = 1.0;
lower = 0.0;
range_valid = true;
}
if (LADSPA_IS_HINT_LOGARITHMIC(hint_descriptor)) {
/* FLT_EPSILON is defined as the different between 1.0 and the minimum
* float greater than 1.0. So, if lower is < FLT_EPSILON, it will be 1.0
* and the logarithmic control will have a base of 1 and thus not change
*/
if (range_valid && lower < FLT_EPSILON)
lower = FLT_EPSILON;
}
if (LADSPA_IS_HINT_HAS_DEFAULT(hint_descriptor)) {
bool valid = true;
if (range_valid && LADSPA_IS_HINT_DEFAULT_MINIMUM(hint_descriptor)) {
normal = lower;
} else if (range_valid && LADSPA_IS_HINT_DEFAULT_LOW(hint_descriptor)) {
if (LADSPA_IS_HINT_LOGARITHMIC(hint_descriptor)) {
normal = exp(log(lower) * 0.75 + log(upper) * 0.25);
} else {
normal = lower * 0.75 + upper * 0.25;
}
} else if (range_valid && LADSPA_IS_HINT_DEFAULT_MIDDLE(hint_descriptor)) {
if (LADSPA_IS_HINT_LOGARITHMIC(hint_descriptor)) {
normal = exp(log(lower) * 0.5 + log(upper) * 0.5);
} else {
normal = lower * 0.5 + upper * 0.5;
}
} else if (range_valid && LADSPA_IS_HINT_DEFAULT_HIGH(hint_descriptor)) {
if (LADSPA_IS_HINT_LOGARITHMIC(hint_descriptor)) {
normal = exp(log(lower) * 0.25 + log(upper) * 0.75);
} else {
normal = lower * 0.25 + upper * 0.75;
}
} else if (range_valid && LADSPA_IS_HINT_DEFAULT_MAXIMUM(hint_descriptor)) {
normal = upper;
} else if (LADSPA_IS_HINT_DEFAULT_0(hint_descriptor)) {
normal = 0.0;
} else if (LADSPA_IS_HINT_DEFAULT_1(hint_descriptor)) {
normal = 1.0;
} else if (LADSPA_IS_HINT_DEFAULT_100(hint_descriptor)) {
normal = 100.0;
} else if (LADSPA_IS_HINT_DEFAULT_440(hint_descriptor)) {
normal = 440.0;
} else {
valid = false;
}
if (valid)
add_float(model, port, NS_LV2("default"), normal);
} else { // No default hint
if (range_valid && LADSPA_IS_HINT_BOUNDED_BELOW(hint_descriptor)) {
normal = lower;
add_float(model, port, NS_LV2("default"), normal);
} else if (range_valid && LADSPA_IS_HINT_BOUNDED_ABOVE(hint_descriptor)) {
normal = upper;
add_float(model, port, NS_LV2("default"), normal);
}
}
}
void
write_lv2_turtle(LADSPA_Descriptor* descriptor, const char* plugin_uri, const char* filename)
{
//librdf_storage* storage = librdf_new_storage(world,
// "hashes", NULL, "hash-type='memory'");
librdf_storage* storage = librdf_new_storage(world, "memory", NULL, NULL);
librdf_model* model = librdf_new_model(world, storage, NULL);
librdf_serializer* serializer = librdf_new_serializer(world, "turtle", NULL, NULL);
librdf_node* plugin = librdf_new_node_from_uri_string(world, U(plugin_uri));
// Set up namespaces
librdf_serializer_set_namespace(serializer, librdf_new_uri(world,
U("http://www.w3.org/1999/02/22-rdf-syntax-ns#")), "rdf");
librdf_serializer_set_namespace(serializer, librdf_new_uri(world,
U("http://www.w3.org/2000/01/rdf-schema#")), "rdfs");
librdf_serializer_set_namespace(serializer, librdf_new_uri(world,
U("http://www.w3.org/2001/XMLSchema")), "xsd");
librdf_serializer_set_namespace(serializer, librdf_new_uri(world,
U("http://usefulinc.com/ns/doap#")), "doap");
librdf_serializer_set_namespace(serializer, librdf_new_uri(world,
U("http://xmlns.com/foaf/0.1/")), "foaf");
librdf_serializer_set_namespace(serializer, librdf_new_uri(world,
U("http://lv2plug.in/ns/lv2core#")), "lv2");
add_resource(model, plugin,
NS_RDF("type"),
NS_LV2("Plugin"));
add_string(model, plugin,
NS_DOAP("name"),
descriptor->Name);
if (LADSPA_IS_HARD_RT_CAPABLE(descriptor->Properties))
add_resource(model, plugin,
NS_LV2("optionalFeature"),
NS_LV2("hardRTCapable"));
for (uint32_t i=0; i < descriptor->PortCount; ++i) {
char index_str[32];
snprintf(index_str, (size_t)32, "%u", i);
const LADSPA_PortDescriptor port_descriptor
= descriptor->PortDescriptors[i];
librdf_node* port_node = librdf_new_node(world);
add_node(model, plugin,
NS_LV2("port"),
port_node);
add_int(model, port_node,
NS_LV2("index"),
(int)i);
if (LADSPA_IS_PORT_INPUT(port_descriptor))
add_resource(model, port_node,
NS_RDF("type"),
NS_LV2("InputPort"));
else
add_resource(model, port_node,
NS_RDF("type"),
NS_LV2("OutputPort"));
if (LADSPA_IS_PORT_AUDIO(port_descriptor))
add_resource(model, port_node,
NS_RDF("type"),
NS_LV2("AudioPort"));
else
add_resource(model, port_node,
NS_RDF("type"),
NS_LV2("ControlPort"));
add_string(model, port_node,
NS_LV2("name"),
descriptor->PortNames[i]);
add_port_range(descriptor, i, model, port_node);
}
librdf_serializer_serialize_model_to_file(serializer, filename, NULL, model);
}
void
print_usage()
{
printf("Usage: ladspa2lv2 /path/to/ladspalib.so ladspa_index lv2_uri output_data_file\n");
printf("Partially convert a LADSPA plugin to an LV2 plugin.\n");
printf("This utility is for developers, it will not generate a usable\n");
printf("LV2 plugin directly.\n\n");
}
int
main(int argc, char** argv)
{
if (argc != 5) {
print_usage();
return 1;
}
const char* const lib_path = argv[1];
const unsigned long index = atol(argv[2]);
const char* const uri = argv[3];
world = librdf_new_world();
librdf_world_open(world);
LADSPA_Descriptor* descriptor = load_ladspa_plugin(lib_path, index);
if (descriptor) {
printf("Loaded %s : %lu\n", lib_path, index);
write_lv2_turtle(descriptor, uri, argv[4]);
} else {
printf("Failed to load %s : %lu\n", lib_path, index);
}
librdf_free_world(world);
return 0;
}