From 0bf92c1f25a854566212e42deafe72ecd2a5f1a1 Mon Sep 17 00:00:00 2001 From: David Robillard Date: Mon, 21 Jan 2008 15:14:53 +0000 Subject: Work on generic LV2 events. git-svn-id: http://svn.drobilla.net/lad/slv2@1090 a436a847-0d15-0410-975c-d299462d15a1 --- hosts/lv2_jack_host.c | 117 ++++++++++++++++++++++++++----------------- hosts/lv2_simple_jack_host.c | 66 +++++++++++++++--------- 2 files changed, 114 insertions(+), 69 deletions(-) (limited to 'hosts') diff --git a/hosts/lv2_jack_host.c b/hosts/lv2_jack_host.c index a92cb62..96d18cb 100644 --- a/hosts/lv2_jack_host.c +++ b/hosts/lv2_jack_host.c @@ -30,23 +30,39 @@ #define MIDI_BUFFER_SIZE 1024 +enum PortDirection { + INPUT, + OUTPUT +}; + +enum PortType { + CONTROL, + AUDIO, + MIDI +}; + struct Port { - SLV2PortDirection direction; - SLV2PortDataType type; - SLV2Port slv2_port; - jack_port_t* jack_port; /**< For audio and MIDI ports, otherwise NULL */ - float control; /**< For control ports, otherwise 0.0f */ - LV2_MIDI* midi_buffer; /**< For midi ports, otherwise NULL */ + SLV2Port slv2_port; + enum PortDirection direction; + enum PortType type; + jack_port_t* jack_port; /**< For audio and MIDI ports, otherwise NULL */ + float control; /**< For control ports, otherwise 0.0f */ + LV2_MIDI* midi_buffer; /**< For midi ports, otherwise NULL */ }; /** This program's data */ struct JackHost { - jack_client_t* jack_client; /**< Jack client */ - SLV2Plugin plugin; /**< Plugin "class" (actually just a few strings) */ - SLV2Instance instance; /**< Plugin "instance" (loaded shared lib) */ - uint32_t num_ports; /**< Size of the two following arrays: */ - struct Port* ports; /** Port array of size num_ports */ + jack_client_t* jack_client; /**< Jack client */ + SLV2Plugin plugin; /**< Plugin "class" (actually just a few strings) */ + SLV2Instance instance; /**< Plugin "instance" (loaded shared lib) */ + uint32_t num_ports; /**< Size of the two following arrays: */ + struct Port* ports; /** Port array of size num_ports */ + SLV2Value input_class; /**< Input port class (URI) */ + SLV2Value output_class; /**< Output port class (URI) */ + SLV2Value control_class; /**< Control port class (URI) */ + SLV2Value audio_class; /**< Audio port class (URI) */ + SLV2Value midi_class; /**< MIDI port class (URI) */ }; @@ -68,6 +84,13 @@ main(int argc, char** argv) SLV2World world = slv2_world_new(); slv2_world_load_all(world); SLV2Plugins plugins = slv2_world_get_all_plugins(world); + + /* Set up the port classes this app supports */ + host.input_class = slv2_value_new_uri(world, SLV2_PORT_CLASS_INPUT); + host.output_class = slv2_value_new_uri(world, SLV2_PORT_CLASS_OUTPUT); + host.control_class = slv2_value_new_uri(world, SLV2_PORT_CLASS_CONTROL); + host.audio_class = slv2_value_new_uri(world, SLV2_PORT_CLASS_AUDIO); + host.midi_class = slv2_value_new_uri(world, SLV2_PORT_CLASS_MIDI); /* Find the plugin to run */ const char* plugin_uri = (argc == 2) ? argv[1] : NULL; @@ -112,7 +135,7 @@ main(int argc, char** argv) jack_set_process_callback(host.jack_client, &jack_process_cb, (void*)(&host)); /* Create ports */ - host.num_ports = slv2_plugin_get_num_ports(host.plugin); + host.num_ports = slv2_plugin_get_num_ports(host.plugin); host.ports = calloc((size_t)host.num_ports, sizeof(struct Port)); for (uint32_t i=0; i < host.num_ports; ++i) @@ -127,9 +150,8 @@ main(int argc, char** argv) getc(stdin); printf("\n"); - /* Deactivate plugin and JACK */ - slv2_instance_free(host.instance); - slv2_plugins_free(world, plugins); + /* Deactivate JACK */ + jack_deactivate(host.jack_client); printf("Shutting down JACK.\n"); for (unsigned long i=0; i < host.num_ports; ++i) { @@ -143,6 +165,16 @@ main(int argc, char** argv) } jack_client_close(host.jack_client); + /* Deactivate plugin */ + slv2_instance_deactivate(host.instance); + slv2_instance_free(host.instance); + + /* Clean up */ + slv2_value_free(host.input_class); + slv2_value_free(host.output_class); + slv2_value_free(host.control_class); + slv2_value_free(host.audio_class); + slv2_value_free(host.midi_class); slv2_plugins_free(world, plugins); slv2_world_free(world); @@ -172,8 +204,6 @@ create_port(struct JackHost* host, { struct Port* const port = &host->ports[port_index]; - port->direction = SLV2_PORT_DIRECTION_UNKNOWN; - port->type = SLV2_PORT_DATA_TYPE_UNKNOWN; port->slv2_port = slv2_plugin_get_port_by_index(host->plugin, port_index); port->jack_port = NULL; port->control = 0.0f; @@ -184,43 +214,40 @@ create_port(struct JackHost* host, /* Get the port symbol (label) for console printing */ char* symbol = slv2_port_get_symbol(host->plugin, port->slv2_port); - /* Get the direction of the port (input, output) */ - port->direction = slv2_port_get_direction(host->plugin, port->slv2_port); - - /* Get the (data) type of the port (control, audio, MIDI, OSC) */ - port->type = slv2_port_get_data_type(host->plugin, port->slv2_port); - - if (port->type == SLV2_PORT_DATA_TYPE_CONTROL) - port->control = slv2_port_get_default_value(host->plugin, port->slv2_port); - enum JackPortFlags jack_flags = 0; - switch (port->direction) { - case SLV2_PORT_DIRECTION_INPUT: - jack_flags = JackPortIsInput; break; - case SLV2_PORT_DIRECTION_OUTPUT: - jack_flags = JackPortIsOutput; break; - default: - // FIXME: check if port connection is is optional and die if not - slv2_instance_connect_port(host->instance, port_index, NULL); - return; + if (slv2_port_is_a(host->plugin, port->slv2_port, host->input_class)) { + jack_flags = JackPortIsInput; + port->direction = INPUT; + } else if (slv2_port_is_a(host->plugin, port->slv2_port, host->output_class)) { + jack_flags = JackPortIsOutput; + port->direction = OUTPUT; + } else if (slv2_port_has_property(host->plugin, port->slv2_port, SLV2_NAMESPACE_LV2 "connectionOptional")) { + slv2_instance_connect_port(host->instance, port_index, NULL); + } else { + die("Mandatory port has unknown type (neither input or output)"); } - + /* Set control values */ - if (port->direction == SLV2_PORT_DIRECTION_INPUT && port->type == SLV2_PORT_DATA_TYPE_CONTROL) { + if (slv2_port_is_a(host->plugin, port->slv2_port, host->control_class)) { + port->type = CONTROL; port->control = slv2_port_get_default_value(host->plugin, port->slv2_port); printf("Set %s to %f\n", symbol, host->ports[port_index].control); + } else if (slv2_port_is_a(host->plugin, port->slv2_port, host->audio_class)) { + port->type = AUDIO; + } else if (slv2_port_is_a(host->plugin, port->slv2_port, host->midi_class)) { + port->type = MIDI; } /* Connect the port based on it's type */ switch (port->type) { - case SLV2_PORT_DATA_TYPE_CONTROL: + case CONTROL: slv2_instance_connect_port(host->instance, port_index, &port->control); break; - case SLV2_PORT_DATA_TYPE_AUDIO: + case AUDIO: port->jack_port = jack_port_register(host->jack_client, symbol, JACK_DEFAULT_AUDIO_TYPE, jack_flags, 0); break; - case SLV2_PORT_DATA_TYPE_MIDI: + case MIDI: port->jack_port = jack_port_register(host->jack_client, symbol, JACK_DEFAULT_MIDI_TYPE, JackPortIsInput, 0); port->midi_buffer = lv2midi_new(MIDI_BUFFER_SIZE); @@ -247,16 +274,16 @@ jack_process_cb(jack_nframes_t nframes, void* data) if (!host->ports[p].jack_port) continue; - if (host->ports[p].type == SLV2_PORT_DATA_TYPE_AUDIO) { + if (host->ports[p].type == AUDIO) { slv2_instance_connect_port(host->instance, p, jack_port_get_buffer(host->ports[p].jack_port, nframes)); - } else if (host->ports[p].type == SLV2_PORT_DATA_TYPE_MIDI) { + } else if (host->ports[p].type == MIDI) { lv2midi_reset_buffer(host->ports[p].midi_buffer); - if (host->ports[p].direction == SLV2_PORT_DIRECTION_INPUT) { + if (host->ports[p].direction == INPUT) { void* jack_buffer = jack_port_get_buffer(host->ports[p].jack_port, nframes); lv2midi_reset_buffer(host->ports[p].midi_buffer); @@ -286,8 +313,8 @@ jack_process_cb(jack_nframes_t nframes, void* data) /* Deliver output */ for (uint32_t p=0; p < host->num_ports; ++p) { if (host->ports[p].jack_port - && host->ports[p].direction == SLV2_PORT_DIRECTION_OUTPUT - && host->ports[p].type == SLV2_PORT_DATA_TYPE_MIDI) { + && host->ports[p].direction == INPUT + && host->ports[p].type == MIDI) { void* jack_buffer = jack_port_get_buffer(host->ports[p].jack_port, nframes); diff --git a/hosts/lv2_simple_jack_host.c b/hosts/lv2_simple_jack_host.c index 5543074..89bc904 100644 --- a/hosts/lv2_simple_jack_host.c +++ b/hosts/lv2_simple_jack_host.c @@ -25,12 +25,16 @@ /** This program's data */ struct JackHost { - jack_client_t* jack_client; /**< Jack client */ - SLV2Plugin plugin; /**< Plugin "class" (actually just a few strings) */ - SLV2Instance instance; /**< Plugin "instance" (loaded shared lib) */ - uint32_t num_ports; /**< Size of the two following arrays: */ - jack_port_t** jack_ports; /**< For audio ports, otherwise NULL */ - float* controls; /**< For control ports, otherwise 0.0f */ + jack_client_t* jack_client; /**< Jack client */ + SLV2World world; /**< SLV2 "world" object */ + SLV2Plugin plugin; /**< Plugin "class" (actually just a few strings) */ + SLV2Instance instance; /**< Plugin "instance" (loaded shared lib) */ + uint32_t num_ports; /**< Size of the two following arrays: */ + jack_port_t** jack_ports; /**< For audio ports, otherwise NULL */ + float* controls; /**< For control ports, otherwise 0.0f */ + SLV2Value input_class; /**< Input port class (URI) */ + SLV2Value control_class; /**< Control port class (URI) */ + SLV2Value audio_class; /**< Audio port class (URI) */ }; @@ -48,11 +52,20 @@ main(int argc, char** argv) host.num_ports = 0; host.jack_ports = NULL; host.controls = NULL; + + /* Set up the port classes this app supports */ + host.input_class = slv2_value_new_uri(host.world, SLV2_PORT_CLASS_INPUT); + host.audio_class = slv2_value_new_uri(host.world, SLV2_PORT_CLASS_OUTPUT); + /* Note that SLV2_PORT_CLASS_* are simply strings defined for convenience. + * host.control_class = slv2_value_new(host.world, SLV2_PORT_CLASS_CONTROL); + * is the same as: */ + host.control_class = slv2_value_new_uri(host.world, + "http://lv2plug.in/ns/lv2core#ControlPort"); /* Find all installed plugins */ - SLV2World world = slv2_world_new(); - slv2_world_load_all(world); - SLV2Plugins plugins = slv2_world_get_all_plugins(world); + host.world = slv2_world_new(); + slv2_world_load_all(host.world); + SLV2Plugins plugins = slv2_world_get_all_plugins(host.world); /* Find the plugin to run */ const char* plugin_uri = (argc == 2) ? argv[1] : NULL; @@ -61,7 +74,7 @@ main(int argc, char** argv) fprintf(stderr, "\nYou must specify a plugin URI to load.\n"); fprintf(stderr, "\nKnown plugins:\n\n"); list_plugins(plugins); - slv2_world_free(world); + slv2_world_free(host.world); return EXIT_FAILURE; } @@ -70,7 +83,7 @@ main(int argc, char** argv) if (!host.plugin) { fprintf(stderr, "Failed to find plugin %s.\n", plugin_uri); - slv2_world_free(world); + slv2_world_free(host.world); return EXIT_FAILURE; } @@ -113,8 +126,8 @@ main(int argc, char** argv) getc(stdin); printf("\n"); - /* Deactivate plugin and JACK */ - slv2_instance_free(host.instance); + /* Deactivate JACK */ + jack_deactivate(host.jack_client); printf("Shutting down JACK.\n"); for (unsigned long i=0; i < host.num_ports; ++i) { @@ -125,7 +138,16 @@ main(int argc, char** argv) } jack_client_close(host.jack_client); - slv2_world_free(world); + /* Deactivate plugin */ + slv2_instance_deactivate(host.instance); + slv2_instance_free(host.instance); + + /* Clean up */ + slv2_value_free(host.input_class); + slv2_value_free(host.audio_class); + slv2_value_free(host.control_class); + slv2_plugins_free(host.world, plugins); + slv2_world_free(host.world); return 0; } @@ -160,28 +182,24 @@ create_port(struct JackHost* host, host->jack_ports[index] = NULL; host->controls[index] = 0.0f; - /* Get the direction of the port (input, output) */ - SLV2PortDirection direction = slv2_port_get_direction(host->plugin, port); - - /* Get the (data) type of the port (control, audio, MIDI, OSC) */ - SLV2PortDataType type = slv2_port_get_data_type(host->plugin, port); - /* Connect control ports to controls array */ - if (type == SLV2_PORT_DATA_TYPE_CONTROL) { + if (slv2_port_is_a(host->plugin, port, host->control_class)) { /* Set default control values for inputs */ - if (direction == SLV2_PORT_DIRECTION_INPUT) { + if (slv2_port_is_a(host->plugin, port, host->input_class)) { host->controls[index] = slv2_port_get_default_value(host->plugin, port); printf("Set %s to %f\n", symbol, host->controls[index]); } slv2_instance_connect_port(host->instance, index, &host->controls[index]); - } else if (type == SLV2_PORT_DATA_TYPE_AUDIO) { + } else if (slv2_port_is_a(host->plugin, port, host->audio_class)) { host->jack_ports[index] = jack_port_register(host->jack_client, symbol, JACK_DEFAULT_AUDIO_TYPE, - (direction == SLV2_PORT_DIRECTION_INPUT) ? JackPortIsInput : JackPortIsOutput, 0); + slv2_port_is_a(host->plugin, port, host->input_class) + ? JackPortIsInput : JackPortIsOutput, + 0); } else { // Simple examples don't have to be robust :) -- cgit v1.2.1