essej 8bbddf66e9 git subrepo clone (merge) https://github.com/free-audio/clap deps/clap-juce-extensions/clap-libs/clap
subrepo:
  subdir:   "deps/clap-juce-extensions/clap-libs/clap"
  merged:   "3189bdfaf"
upstream:
  origin:   "https://github.com/free-audio/clap"
  branch:   "main"
  commit:   "3189bdfaf"
git-subrepo:
  version:  "0.4.3"
  origin:   "https://github.com/ingydotnet/git-subrepo.git"
  commit:   "2f68596"
2022-06-14 21:34:50 -04:00

349 lines
11 KiB
C

// This file is here to demonstrate how to wire a CLAP plugin
// You can use it as a starting point, however if you are implementing a C++
// plugin, I'd encourage you to use the C++ glue layer instead:
// https://github.com/free-audio/clap-helpers/blob/main/include/clap/helpers/plugin.hh
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <clap/clap.h>
static const clap_plugin_descriptor_t s_my_plug_desc = {
.clap_version = CLAP_VERSION_INIT,
.id = "com.your-company.YourPlugin",
.name = "Plugin Name",
.vendor = "Vendor",
.url = "https://your-domain.com/your-plugin",
.manual_url = "https://your-domain.com/your-plugin/manual",
.support_url = "https://your-domain.com/support",
.version = "1.4.2",
.description = "The plugin description.",
.features = (const char *[]){
CLAP_PLUGIN_FEATURE_INSTRUMENT,
CLAP_PLUGIN_FEATURE_STEREO
},
};
typedef struct {
clap_plugin_t plugin;
const clap_host_t *host;
const clap_host_latency_t *hostLatency;
const clap_host_log_t *hostLog;
const clap_host_thread_check_t *hostThreadCheck;
uint32_t latency;
} my_plug_t;
/////////////////////////////
// clap_plugin_audio_ports //
/////////////////////////////
static uint32_t my_plug_audio_ports_count(const clap_plugin_t *plugin, bool is_input) { return 1; }
static bool my_plug_audio_ports_get(const clap_plugin_t *plugin,
uint32_t index,
bool is_input,
clap_audio_port_info_t *info) {
if (index > 0)
return false;
info->id = 0;
snprintf(info->name, sizeof(info->name), "%s", "My Port Name");
info->channel_count = 2;
info->flags = CLAP_AUDIO_PORT_IS_MAIN;
info->port_type = CLAP_PORT_STEREO;
info->in_place_pair = CLAP_INVALID_ID;
return true;
}
static const clap_plugin_audio_ports_t s_my_plug_audio_ports = {
.count = my_plug_audio_ports_count,
.get = my_plug_audio_ports_get,
};
////////////////////////////
// clap_plugin_note_ports //
////////////////////////////
static uint32_t my_plug_note_ports_count(const clap_plugin_t *plugin, bool is_input) { return 1; }
static bool my_plug_note_ports_get(const clap_plugin_t *plugin,
uint32_t index,
bool is_input,
clap_note_port_info_t *info) {
if (index > 0)
return false;
info->id = 0;
snprintf(info->name, sizeof(info->name), "%s", "My Port Name");
info->supported_dialects =
CLAP_NOTE_DIALECT_CLAP | CLAP_NOTE_DIALECT_MIDI_MPE | CLAP_NOTE_DIALECT_MIDI2;
info->preferred_dialect = CLAP_NOTE_DIALECT_CLAP;
return true;
}
static const clap_plugin_note_ports_t s_my_plug_note_ports = {
.count = my_plug_note_ports_count,
.get = my_plug_note_ports_get,
};
//////////////////
// clap_latency //
//////////////////
uint32_t my_plug_latency_get(const clap_plugin_t *plugin) {
my_plug_t *plug = plugin->plugin_data;
return plug->latency;
}
static const clap_plugin_latency_t s_my_plug_latency = {
.get = my_plug_latency_get,
};
/////////////////
// clap_plugin //
/////////////////
static bool my_plug_init(const struct clap_plugin *plugin) {
my_plug_t *plug = plugin->plugin_data;
// Fetch host's extensions here
plug->hostLog = plug->host->get_extension(plug->host, CLAP_EXT_LOG);
plug->hostThreadCheck = plug->host->get_extension(plug->host, CLAP_EXT_THREAD_CHECK);
plug->hostLatency = plug->host->get_extension(plug->host, CLAP_EXT_LATENCY);
return true;
}
static void my_plug_destroy(const struct clap_plugin *plugin) {}
static bool my_plug_activate(const struct clap_plugin *plugin,
double sample_rate,
uint32_t min_frames_count,
uint32_t max_frames_count) {
return true;
}
static void my_plug_deactivate(const struct clap_plugin *plugin) {}
static bool my_plug_start_processing(const struct clap_plugin *plugin) { return true; }
static void my_plug_stop_processing(const struct clap_plugin *plugin) {}
static void my_plug_reset(const struct clap_plugin *plugin) {}
static void my_plug_process_event(my_plug_t *plug, const clap_event_header_t *hdr) {
if (hdr->space_id == CLAP_CORE_EVENT_SPACE_ID) {
switch (hdr->type) {
case CLAP_EVENT_NOTE_ON: {
const clap_event_note_t *ev = (const clap_event_note_t *)hdr;
// TODO: handle note on
break;
}
case CLAP_EVENT_NOTE_OFF: {
const clap_event_note_t *ev = (const clap_event_note_t *)hdr;
// TODO: handle note on
break;
}
case CLAP_EVENT_NOTE_CHOKE: {
const clap_event_note_t *ev = (const clap_event_note_t *)hdr;
// TODO: handle note choke
break;
}
case CLAP_EVENT_NOTE_EXPRESSION: {
const clap_event_note_expression_t *ev = (const clap_event_note_expression_t *)hdr;
// TODO: handle note expression
break;
}
case CLAP_EVENT_PARAM_VALUE: {
const clap_event_param_value_t *ev = (const clap_event_param_value_t *)hdr;
// TODO: handle parameter change
break;
}
case CLAP_EVENT_PARAM_MOD: {
const clap_event_param_mod_t *ev = (const clap_event_param_mod_t *)hdr;
// TODO: handle parameter modulation
break;
}
case CLAP_EVENT_TRANSPORT: {
const clap_event_transport_t *ev = (const clap_event_transport_t *)hdr;
// TODO: handle transport event
break;
}
case CLAP_EVENT_MIDI: {
const clap_event_midi_t *ev = (const clap_event_midi_t *)hdr;
// TODO: handle MIDI event
break;
}
case CLAP_EVENT_MIDI_SYSEX: {
const clap_event_midi_sysex_t *ev = (const clap_event_midi_sysex_t *)hdr;
// TODO: handle MIDI Sysex event
break;
}
case CLAP_EVENT_MIDI2: {
const clap_event_midi2_t *ev = (const clap_event_midi2_t *)hdr;
// TODO: handle MIDI2 event
break;
}
}
}
}
static clap_process_status my_plug_process(const struct clap_plugin *plugin,
const clap_process_t *process) {
my_plug_t *plug = plugin->plugin_data;
const uint32_t nframes = process->frames_count;
const uint32_t nev = process->in_events->size(process->in_events);
uint32_t ev_index = 0;
uint32_t next_ev_frame = nev > 0 ? 0 : nframes;
for (uint32_t i = 0; i < nframes;) {
/* handle every events that happrens at the frame "i" */
while (ev_index < nev && next_ev_frame == i) {
const clap_event_header_t *hdr = process->in_events->get(process->in_events, ev_index);
if (hdr->time != i) {
next_ev_frame = hdr->time;
break;
}
my_plug_process_event(plug, hdr);
++ev_index;
if (ev_index == nev) {
// we reached the end of the event list
next_ev_frame = nframes;
break;
}
}
/* process every samples until the next event */
for (; i < next_ev_frame; ++i) {
// fetch input samples
const float in_l = process->audio_inputs[0].data32[0][i];
const float in_r = process->audio_inputs[0].data32[1][i];
/* TODO: process samples, here we simply swap left and right channels */
const float out_l = in_r;
const float out_r = in_l;
// store output samples
process->audio_outputs[0].data32[0][i] = out_l;
process->audio_outputs[0].data32[1][i] = out_r;
}
}
return CLAP_PROCESS_CONTINUE;
}
static const void *my_plug_get_extension(const struct clap_plugin *plugin, const char *id) {
if (!strcmp(id, CLAP_EXT_LATENCY))
return &s_my_plug_latency;
if (!strcmp(id, CLAP_EXT_AUDIO_PORTS))
return &s_my_plug_audio_ports;
if (!strcmp(id, CLAP_EXT_NOTE_PORTS))
return &s_my_plug_note_ports;
// TODO: add support to CLAP_EXT_PARAMS
// TODO: add support to CLAP_EXT_STATE
return NULL;
}
static void my_plug_on_main_thread(const struct clap_plugin *plugin) {}
clap_plugin_t *my_plug_create(const clap_host_t *host) {
my_plug_t *p = calloc(1, sizeof(*p));
p->host = host;
p->plugin.desc = &s_my_plug_desc;
p->plugin.plugin_data = p;
p->plugin.init = my_plug_init;
p->plugin.destroy = my_plug_destroy;
p->plugin.activate = my_plug_activate;
p->plugin.deactivate = my_plug_deactivate;
p->plugin.start_processing = my_plug_start_processing;
p->plugin.stop_processing = my_plug_stop_processing;
p->plugin.reset = my_plug_reset;
p->plugin.process = my_plug_process;
p->plugin.get_extension = my_plug_get_extension;
p->plugin.on_main_thread = my_plug_on_main_thread;
// Don't call into the host here
return &p->plugin;
}
/////////////////////////
// clap_plugin_factory //
/////////////////////////
static struct {
const clap_plugin_descriptor_t *desc;
clap_plugin_t *(*create)(const clap_host_t *host);
} s_plugins[] = {
{
.desc = &s_my_plug_desc,
.create = my_plug_create,
},
};
static uint32_t plugin_factory_get_plugin_count(const struct clap_plugin_factory *factory) {
return sizeof(s_plugins) / sizeof(s_plugins[0]);
}
static const clap_plugin_descriptor_t *
plugin_factory_get_plugin_descriptor(const struct clap_plugin_factory *factory, uint32_t index) {
return s_plugins[index].desc;
}
static const clap_plugin_t *plugin_factory_create_plugin(const struct clap_plugin_factory *factory,
const clap_host_t *host,
const char *plugin_id) {
if (!clap_version_is_compatible(host->clap_version)) {
return NULL;
}
const int N = sizeof(s_plugins) / sizeof(s_plugins[0]);
for (int i = 0; i < N; ++i)
if (!strcmp(plugin_id, s_plugins[i].desc->id))
return s_plugins[i].create(host);
return NULL;
}
static const clap_plugin_factory_t s_plugin_factory = {
.get_plugin_count = plugin_factory_get_plugin_count,
.get_plugin_descriptor = plugin_factory_get_plugin_descriptor,
.create_plugin = plugin_factory_create_plugin,
};
////////////////
// clap_entry //
////////////////
static bool entry_init(const char *plugin_path) {
// called only once, and very first
return true;
}
static void entry_deinit(void) {
// called before unloading the DSO
}
static const void *entry_get_factory(const char *factory_id) {
if (!strcmp(factory_id, CLAP_PLUGIN_FACTORY_ID))
return &s_plugin_factory;
return NULL;
}
CLAP_EXPORT const clap_plugin_entry_t clap_entry = {
.clap_version = CLAP_VERSION_INIT,
.init = entry_init,
.deinit = entry_deinit,
.get_factory = entry_get_factory,
};