use std::io; pub trait Open { fn open(Option<&str>) -> Self; } pub trait Sink { fn start(&mut self) -> io::Result<()>; fn stop(&mut self) -> io::Result<()>; fn write(&mut self, data: &[i16]) -> io::Result<()>; } /* * Allow #[cfg] rules around elements of a list. * Workaround until stmt_expr_attributes is stable. * * This generates 2^n declarations of the list, with every combination possible */ macro_rules! declare_backends { (pub const $name:ident : $ty:ty = & [ $($tt:tt)* ];) => ( _declare_backends!($name ; $ty ; []; []; []; $($tt)*); ); } macro_rules! _declare_backends { ($name:ident ; $ty:ty ; [ $($yes:meta,)* ] ; [ $($no:meta,)* ] ; [ $($exprs:expr,)* ] ; #[cfg($m:meta)] $e:expr, $($rest:tt)* ) => ( _declare_backends!($name ; $ty ; [ $m, $($yes,)* ] ; [ $($no,)* ] ; [ $($exprs,)* $e, ] ; $($rest)*); _declare_backends!($name ; $ty ; [ $($yes,)* ] ; [ $m, $($no,)* ] ; [ $($exprs,)* ] ; $($rest)*); ); ($name:ident ; $ty:ty ; [ $($yes:meta,)* ] ; [ $($no:meta,)* ] ; [ $($exprs:expr,)* ] ; $e:expr, $($rest:tt)*) => ( _declare_backends!($name ; $ty ; [ $($yes,)* ] ; [ $($no,)* ] ; [ $($exprs,)* $e, ] ; $($rest)*); ); ($name:ident ; $ty:ty ; [ $($yes:meta,)* ] ; [ $($no:meta,)* ] ; [ $($exprs:expr,)* ] ; #[cfg($m:meta)] $e:expr) => ( _declare_backends!($name ; $ty ; [ $m, $($yes,)* ] ; [ $($no,)* ] ; [ $($exprs,)* $e, ] ; ); _declare_backends!($name ; $ty ; [ $($yes,)* ] ; [ $m, $($no,)* ] ; [ $($exprs,)* ] ; ); ); ($name:ident ; $ty:ty ; [ $($yes:meta,)* ] ; [ $($no:meta,)* ] ; [ $($exprs:expr,)* ] ; $e:expr ) => ( _declare_backends!($name ; $ty ; [ $($yes,)* ] ; [ $($no,)* ] ; [ $($exprs,)* $e, ] ; ); ); ($name:ident ; $ty:ty ; [ $($yes:meta,)* ] ; [ $($no:meta,)* ] ; [ $($exprs:expr,)* ] ; ) => ( #[cfg(all($($yes,)* not(any($($no),*))))] pub const $name : $ty = &[ $($exprs,)* ]; ) } #[allow(dead_code)] fn mk_sink(device: Option<&str>) -> Box { Box::new(S::open(device)) } #[cfg(feature = "alsa-backend")] mod alsa; #[cfg(feature = "alsa-backend")] use self::alsa::AlsaSink; #[cfg(feature = "portaudio-backend")] mod portaudio; #[cfg(feature = "portaudio-backend")] use self::portaudio::PortAudioSink; #[cfg(feature = "pulseaudio-backend")] mod pulseaudio; #[cfg(feature = "pulseaudio-backend")] use self::pulseaudio::PulseAudioSink; mod pipe; use self::pipe::StdoutSink; declare_backends! { pub const BACKENDS : &'static [ (&'static str, &'static (Fn(Option<&str>) -> Box + Sync + Send + 'static)) ] = &[ #[cfg(feature = "alsa-backend")] ("alsa", &mk_sink::), #[cfg(feature = "portaudio-backend")] ("portaudio", &mk_sink::), #[cfg(feature = "pulseaudio-backend")] ("pulseaudio", &mk_sink::), ("pipe", &mk_sink::), ]; } pub fn find>(name: Option) -> Option<&'static (Fn(Option<&str>) -> Box + Send + Sync)> { if let Some(name) = name.as_ref().map(AsRef::as_ref) { BACKENDS.iter().find(|backend| name == backend.0).map(|backend| backend.1) } else { Some(BACKENDS.first().expect("No backends were enabled at build time").1) } }