use crate::audio::AudioPacket; use crate::config::AudioFormat; use std::io; pub trait Open { fn open(_: Option, format: AudioFormat) -> Self; } pub trait Sink { fn start(&mut self) -> io::Result<()>; fn stop(&mut self) -> io::Result<()>; fn write(&mut self, packet: &AudioPacket) -> io::Result<()>; } pub trait SinkAsBytes { fn write_bytes(&mut self, data: &[u8]) -> io::Result<()>; } fn mk_sink(device: Option, format: AudioFormat) -> Box { Box::new(S::open(device, format)) } // reuse code for various backends macro_rules! sink_as_bytes { () => { fn write(&mut self, packet: &AudioPacket) -> io::Result<()> { use zerocopy::AsBytes; match packet { AudioPacket::Samples(samples) => match self.format { AudioFormat::F32 => self.write_bytes(samples.as_bytes()), AudioFormat::S32 => { let samples_s32 = AudioPacket::f32_to_s32(samples); self.write_bytes(samples_s32.as_bytes()) } AudioFormat::S16 => { let samples_s16 = AudioPacket::f32_to_s16(samples); self.write_bytes(samples_s16.as_bytes()) } }, AudioPacket::OggData(samples) => self.write_bytes(samples), } } }; } macro_rules! start_stop_noop { () => { fn start(&mut self) -> io::Result<()> { Ok(()) } fn stop(&mut self) -> io::Result<()> { Ok(()) } }; } #[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; #[cfg(feature = "jackaudio-backend")] mod jackaudio; #[cfg(feature = "jackaudio-backend")] use self::jackaudio::JackSink; #[cfg(all( feature = "rodiojack-backend", not(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd")) ))] compile_error!("Rodio JACK backend is currently only supported on linux."); #[cfg(all( feature = "rodiojack-backend", any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd") ))] use self::rodio::JackRodioSink; #[cfg(feature = "gstreamer-backend")] mod gstreamer; #[cfg(feature = "gstreamer-backend")] use self::gstreamer::GstreamerSink; #[cfg(any(feature = "rodio-backend", feature = "rodiojack-backend"))] mod rodio; #[cfg(feature = "rodio-backend")] use self::rodio::RodioSink; #[cfg(feature = "sdl-backend")] mod sdl; #[cfg(feature = "sdl-backend")] use self::sdl::SdlSink; mod pipe; use self::pipe::StdoutSink; mod subprocess; use self::subprocess::SubprocessSink; pub const BACKENDS: &'static [( &'static str, fn(Option, AudioFormat) -> Box, )] = &[ #[cfg(feature = "alsa-backend")] ("alsa", mk_sink::), #[cfg(feature = "portaudio-backend")] ("portaudio", mk_sink::), #[cfg(feature = "pulseaudio-backend")] ("pulseaudio", mk_sink::), #[cfg(feature = "jackaudio-backend")] ("jackaudio", mk_sink::), #[cfg(all( feature = "rodiojack-backend", any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd") ))] ("rodiojack", mk_sink::), #[cfg(feature = "gstreamer-backend")] ("gstreamer", mk_sink::), #[cfg(feature = "rodio-backend")] ("rodio", mk_sink::), #[cfg(feature = "sdl-backend")] ("sdl", mk_sink::), ("pipe", mk_sink::), ("subprocess", mk_sink::), ]; pub fn find(name: Option) -> Option, AudioFormat) -> Box> { if let Some(name) = name { BACKENDS .iter() .find(|backend| name == backend.0) .map(|backend| backend.1) } else { Some( BACKENDS .first() .expect("No backends were enabled at build time") .1, ) } }