Skip to content

Commit

Permalink
clap: entry point
Browse files Browse the repository at this point in the history
  • Loading branch information
micahrj committed Oct 16, 2023
1 parent e702449 commit 41b51ab
Show file tree
Hide file tree
Showing 4 changed files with 184 additions and 0 deletions.
9 changes: 9 additions & 0 deletions examples/gain/src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use std::io::{self, Read, Write};

use coupler::format::clap::*;
use coupler::format::vst3::*;
use coupler::{buffers::*, bus::*, events::*, param::*, parent::*, *};

Expand Down Expand Up @@ -92,6 +93,14 @@ impl Vst3Plugin for Gain {
}
}

impl ClapPlugin for Gain {
fn clap_info() -> ClapInfo {
ClapInfo {
id: "rs.coupler.gain".to_string(),
}
}
}

pub struct GainProcessor {
gain: f32,
}
Expand Down
106 changes: 106 additions & 0 deletions src/format/clap/factory.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,106 @@
use std::cell::UnsafeCell;
use std::ffi::{c_char, c_void, CStr, CString};
use std::marker::PhantomData;
use std::ptr;

use clap_sys::{host::*, plugin::*, plugin_factory::*, version::*};

use super::ClapPlugin;
use crate::Plugin;

#[doc(hidden)]
#[repr(C)]
pub struct Factory<P> {
#[allow(unused)]
factory: clap_plugin_factory,
descriptor: UnsafeCell<Option<clap_plugin_descriptor>>,
_marker: PhantomData<P>,
}

unsafe impl<P> Sync for Factory<P> {}

impl<P: Plugin + ClapPlugin> Factory<P> {
pub const fn new() -> Self {
Factory {
factory: clap_plugin_factory {
get_plugin_count: Some(Self::get_plugin_count),
get_plugin_descriptor: Some(Self::get_plugin_descriptor),
create_plugin: Some(Self::create_plugin),
},
descriptor: UnsafeCell::new(None),
_marker: PhantomData,
}
}

pub unsafe fn init(&self) -> bool {
let clap_info = P::clap_info();
let id = CString::new(&*clap_info.id).unwrap().into_raw();

let info = P::info();
let name = CString::new(&*info.name).unwrap().into_raw();
let vendor = CString::new(&*info.vendor).unwrap().into_raw();
let url = CString::new(&*info.url).unwrap().into_raw();

const EMPTY: &'static CStr = unsafe { CStr::from_bytes_with_nul_unchecked(b"\0") };
const FEATURES: &'static [*const c_char] = &[ptr::null()];

*self.descriptor.get() = Some(clap_plugin_descriptor {
clap_version: CLAP_VERSION,
id,
name,
vendor,
url,
manual_url: EMPTY.as_ptr(),
support_url: EMPTY.as_ptr(),
version: EMPTY.as_ptr(),
description: EMPTY.as_ptr(),
features: FEATURES.as_ptr(),
});

true
}

pub unsafe fn deinit(&self) {
if let Some(descriptor) = (*self.descriptor.get()).take() {
drop(CString::from_raw(descriptor.id as *mut c_char));
drop(CString::from_raw(descriptor.name as *mut c_char));
drop(CString::from_raw(descriptor.vendor as *mut c_char));
drop(CString::from_raw(descriptor.url as *mut c_char));
}
}

pub unsafe fn get(&self, factory_id: *const c_char) -> *const c_void {
if CStr::from_ptr(factory_id) == CLAP_PLUGIN_FACTORY_ID {
return self as *const Self as *const c_void;
}

ptr::null()
}

unsafe extern "C" fn get_plugin_count(_factory: *const clap_plugin_factory) -> u32 {
1
}

unsafe extern "C" fn get_plugin_descriptor(
factory: *const clap_plugin_factory,
index: u32,
) -> *const clap_plugin_descriptor {
let factory = &*(factory as *const Self);

if index == 0 {
if let Some(descriptor) = &*factory.descriptor.get() {
return descriptor;
}
}

ptr::null()
}

unsafe extern "C" fn create_plugin(
_factory: *const clap_plugin_factory,
_host: *const clap_host,
_plugin_id: *const c_char,
) -> *const clap_plugin {
ptr::null()
}
}
68 changes: 68 additions & 0 deletions src/format/clap/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
use std::ffi::{c_char, c_void};

use clap_sys::{entry::*, version::*};

mod factory;

#[doc(hidden)]
pub use factory::Factory;

pub struct ClapInfo {
pub id: String,
}

pub trait ClapPlugin {
fn clap_info() -> ClapInfo;
}

#[doc(hidden)]
#[repr(transparent)]
pub struct EntryPoint {
#[allow(unused)]
entry: clap_plugin_entry,
}

impl EntryPoint {
pub const fn new(
init: unsafe extern "C" fn(_plugin_path: *const c_char) -> bool,
deinit: unsafe extern "C" fn(),
get_factory: unsafe extern "C" fn(factory_id: *const c_char) -> *const c_void,
) -> EntryPoint {
EntryPoint {
entry: clap_plugin_entry {
clap_version: CLAP_VERSION,
init: Some(init),
deinit: Some(deinit),
get_factory: Some(get_factory),
},
}
}
}

#[macro_export]
macro_rules! clap {
($plugin:ty) => {
#[allow(non_upper_case_globals)]
#[no_mangle]
static clap_entry: ::coupler::format::clap::EntryPoint = {
static FACTORY: ::coupler::format::clap::Factory<$plugin> =
::coupler::format::clap::Factory::new();

unsafe extern "C" fn init(_plugin_path: *const ::std::ffi::c_char) -> bool {
FACTORY.init()
}

unsafe extern "C" fn deinit() {
FACTORY.deinit();
}

unsafe extern "C" fn get_factory(
factory_id: *const ::std::ffi::c_char,
) -> *const ::std::ffi::c_void {
FACTORY.get(factory_id)
}

::coupler::format::clap::EntryPoint::new(init, deinit, get_factory)
};
};
}
1 change: 1 addition & 0 deletions src/format/mod.rs
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
pub mod clap;
pub mod vst3;

0 comments on commit 41b51ab

Please sign in to comment.