2019-03-20 13:24:03 +00:00
|
|
|
use super::{Open, Sink};
|
2019-03-20 13:21:50 +00:00
|
|
|
extern crate cpal;
|
2019-10-08 09:31:18 +00:00
|
|
|
extern crate rodio;
|
2021-01-07 06:42:38 +00:00
|
|
|
use crate::audio::AudioPacket;
|
2020-12-02 19:45:31 +00:00
|
|
|
use cpal::traits::{DeviceTrait, HostTrait};
|
2019-03-20 13:24:03 +00:00
|
|
|
use std::process::exit;
|
2019-10-08 09:31:18 +00:00
|
|
|
use std::{io, thread, time};
|
2019-03-20 13:24:03 +00:00
|
|
|
|
|
|
|
pub struct RodioSink {
|
|
|
|
rodio_sink: rodio::Sink,
|
2020-12-02 19:45:31 +00:00
|
|
|
// We have to keep hold of this object, or the Sink can't play...
|
|
|
|
#[allow(dead_code)]
|
|
|
|
stream: rodio::OutputStream,
|
2019-03-20 13:24:03 +00:00
|
|
|
}
|
|
|
|
|
2021-02-10 01:44:05 +00:00
|
|
|
#[cfg(all(
|
|
|
|
feature = "rodiojack-backend",
|
|
|
|
any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd")
|
|
|
|
))]
|
2021-02-05 12:59:21 +00:00
|
|
|
pub struct JackRodioSink {
|
|
|
|
jackrodio_sink: rodio::Sink,
|
|
|
|
// We have to keep hold of this object, or the Sink can't play...
|
|
|
|
#[allow(dead_code)]
|
|
|
|
stream: rodio::OutputStream,
|
|
|
|
}
|
|
|
|
|
2019-03-20 13:21:50 +00:00
|
|
|
fn list_formats(ref device: &rodio::Device) {
|
2020-12-02 19:45:31 +00:00
|
|
|
let default_fmt = match device.default_output_config() {
|
|
|
|
Ok(fmt) => cpal::SupportedStreamConfig::from(fmt),
|
2019-03-20 13:21:50 +00:00
|
|
|
Err(e) => {
|
2020-12-02 19:45:31 +00:00
|
|
|
warn!("Error getting default rodio::Sink config: {}", e);
|
2019-03-20 13:21:50 +00:00
|
|
|
return;
|
2019-10-08 09:31:18 +00:00
|
|
|
}
|
2019-03-20 13:21:50 +00:00
|
|
|
};
|
2020-12-02 19:45:31 +00:00
|
|
|
debug!(" Default config:");
|
|
|
|
debug!(" {:?}", default_fmt);
|
2019-03-20 13:21:50 +00:00
|
|
|
|
2020-12-02 19:45:31 +00:00
|
|
|
let mut output_configs = match device.supported_output_configs() {
|
2019-03-20 13:21:50 +00:00
|
|
|
Ok(f) => f.peekable(),
|
|
|
|
Err(e) => {
|
2020-12-02 19:45:31 +00:00
|
|
|
warn!("Error getting supported rodio::Sink configs: {}", e);
|
2019-03-20 13:21:50 +00:00
|
|
|
return;
|
2019-10-08 09:31:18 +00:00
|
|
|
}
|
2019-03-20 13:21:50 +00:00
|
|
|
};
|
|
|
|
|
2020-12-02 19:45:31 +00:00
|
|
|
if output_configs.peek().is_some() {
|
|
|
|
debug!(" Available configs:");
|
|
|
|
for format in output_configs {
|
|
|
|
debug!(" {:?}", format);
|
2019-03-20 13:21:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-05 12:59:21 +00:00
|
|
|
fn list_outputs(ref host: &cpal::Host) {
|
|
|
|
let default_device = get_default_device(host);
|
2020-12-02 19:45:31 +00:00
|
|
|
let default_device_name = default_device.name().expect("cannot get output name");
|
|
|
|
println!("Default Audio Device:\n {}", default_device_name);
|
2019-03-20 13:21:50 +00:00
|
|
|
list_formats(&default_device);
|
2019-03-20 13:24:03 +00:00
|
|
|
|
2019-03-20 13:21:50 +00:00
|
|
|
println!("Other Available Audio Devices:");
|
2021-02-05 12:59:21 +00:00
|
|
|
|
|
|
|
let found_devices = host.output_devices().expect(&format!(
|
|
|
|
"Cannot get list of output devices of Host: {:?}",
|
|
|
|
host.id()
|
|
|
|
));
|
|
|
|
for device in found_devices {
|
2020-12-02 19:45:31 +00:00
|
|
|
let device_name = device.name().expect("cannot get output name");
|
|
|
|
if device_name != default_device_name {
|
|
|
|
println!(" {}", device_name);
|
2019-03-20 13:21:50 +00:00
|
|
|
list_formats(&device);
|
2019-03-20 13:24:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-05 12:59:21 +00:00
|
|
|
fn get_default_device(ref host: &cpal::Host) -> rodio::Device {
|
|
|
|
host.default_output_device()
|
2020-12-02 19:45:31 +00:00
|
|
|
.expect("no default output device available")
|
|
|
|
}
|
2019-03-20 13:24:03 +00:00
|
|
|
|
2021-02-05 12:59:21 +00:00
|
|
|
fn match_device(ref host: &cpal::Host, device: Option<String>) -> rodio::Device {
|
2020-12-02 19:45:31 +00:00
|
|
|
match device {
|
|
|
|
Some(device_name) => {
|
2019-03-20 13:24:03 +00:00
|
|
|
if device_name == "?".to_string() {
|
2021-02-05 12:59:21 +00:00
|
|
|
list_outputs(host);
|
2019-03-20 13:24:03 +00:00
|
|
|
exit(0)
|
|
|
|
}
|
2021-02-05 12:59:21 +00:00
|
|
|
|
|
|
|
let found_devices = host.output_devices().expect(&format!(
|
|
|
|
"Cannot get list of output devices of Host: {:?}",
|
|
|
|
host.id()
|
|
|
|
));
|
|
|
|
for d in found_devices {
|
2020-12-02 19:45:31 +00:00
|
|
|
if d.name().expect("cannot get output name") == device_name {
|
|
|
|
return d;
|
2019-03-20 13:24:03 +00:00
|
|
|
}
|
|
|
|
}
|
2020-12-02 19:45:31 +00:00
|
|
|
println!("No output sink matching '{}' found.", device_name);
|
|
|
|
exit(0)
|
2019-03-20 13:24:03 +00:00
|
|
|
}
|
2021-02-05 12:59:21 +00:00
|
|
|
None => return get_default_device(host),
|
2020-12-02 19:45:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Open for RodioSink {
|
|
|
|
fn open(device: Option<String>) -> RodioSink {
|
2021-02-05 12:59:21 +00:00
|
|
|
let host = cpal::default_host();
|
|
|
|
debug!("Using rodio sink with cpal host: {:?}", host.id());
|
2020-12-02 19:45:31 +00:00
|
|
|
|
2021-02-05 12:59:21 +00:00
|
|
|
let rodio_device = match_device(&host, device);
|
2020-12-02 19:45:31 +00:00
|
|
|
debug!("Using cpal device");
|
|
|
|
let stream = rodio::OutputStream::try_from_device(&rodio_device)
|
|
|
|
.expect("Couldn't open output stream.");
|
|
|
|
debug!("Using rodio stream");
|
|
|
|
let sink = rodio::Sink::try_new(&stream.1).expect("Couldn't create output sink.");
|
|
|
|
debug!("Using rodio sink");
|
2019-03-20 13:24:03 +00:00
|
|
|
|
2020-12-02 19:45:31 +00:00
|
|
|
RodioSink {
|
|
|
|
rodio_sink: sink,
|
|
|
|
stream: stream.0,
|
|
|
|
}
|
2019-03-20 13:24:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-10 01:44:05 +00:00
|
|
|
#[cfg(all(
|
|
|
|
feature = "rodiojack-backend",
|
|
|
|
any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd")
|
|
|
|
))]
|
2021-02-05 12:59:21 +00:00
|
|
|
impl Open for JackRodioSink {
|
|
|
|
fn open(device: Option<String>) -> JackRodioSink {
|
|
|
|
let host = cpal::host_from_id(
|
|
|
|
cpal::available_hosts()
|
|
|
|
.into_iter()
|
|
|
|
.find(|id| *id == cpal::HostId::Jack)
|
|
|
|
.expect("Jack Host not found"),
|
|
|
|
)
|
|
|
|
.expect("Jack Host not found");
|
|
|
|
debug!("Using jack rodio sink with cpal Jack host");
|
|
|
|
|
|
|
|
let rodio_device = match_device(&host, device);
|
|
|
|
debug!("Using cpal device");
|
|
|
|
let stream = rodio::OutputStream::try_from_device(&rodio_device)
|
|
|
|
.expect("Couldn't open output stream.");
|
|
|
|
debug!("Using jack rodio stream");
|
|
|
|
let sink = rodio::Sink::try_new(&stream.1).expect("Couldn't create output sink.");
|
|
|
|
debug!("Using jack rodio sink");
|
|
|
|
|
|
|
|
JackRodioSink {
|
|
|
|
jackrodio_sink: sink,
|
|
|
|
stream: stream.0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-20 13:24:03 +00:00
|
|
|
impl Sink for RodioSink {
|
|
|
|
fn start(&mut self) -> io::Result<()> {
|
2019-03-01 15:43:16 +00:00
|
|
|
// More similar to an "unpause" than "play". Doesn't undo "stop".
|
|
|
|
// self.rodio_sink.play();
|
2019-03-20 13:24:03 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn stop(&mut self) -> io::Result<()> {
|
2019-03-01 15:43:16 +00:00
|
|
|
// This will immediately stop playback, but the sink is then unusable.
|
|
|
|
// We just have to let the current buffer play till the end.
|
|
|
|
// self.rodio_sink.stop();
|
2019-03-20 13:24:03 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-01-07 06:42:38 +00:00
|
|
|
fn write(&mut self, packet: &AudioPacket) -> io::Result<()> {
|
|
|
|
let source = rodio::buffer::SamplesBuffer::new(2, 44100, packet.samples());
|
2019-03-20 13:24:03 +00:00
|
|
|
self.rodio_sink.append(source);
|
2019-03-20 13:23:20 +00:00
|
|
|
|
|
|
|
// Chunk sizes seem to be about 256 to 3000 ish items long.
|
|
|
|
// Assuming they're on average 1628 then a half second buffer is:
|
|
|
|
// 44100 elements --> about 27 chunks
|
|
|
|
while self.rodio_sink.len() > 26 {
|
|
|
|
// sleep and wait for rodio to drain a bit
|
|
|
|
thread::sleep(time::Duration::from_millis(10));
|
|
|
|
}
|
2019-03-20 13:24:03 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
2021-02-05 12:59:21 +00:00
|
|
|
|
2021-02-10 01:44:05 +00:00
|
|
|
#[cfg(all(
|
|
|
|
feature = "rodiojack-backend",
|
|
|
|
any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd")
|
|
|
|
))]
|
2021-02-05 12:59:21 +00:00
|
|
|
impl Sink for JackRodioSink {
|
|
|
|
fn start(&mut self) -> io::Result<()> {
|
|
|
|
// More similar to an "unpause" than "play". Doesn't undo "stop".
|
|
|
|
// self.rodio_sink.play();
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn stop(&mut self) -> io::Result<()> {
|
|
|
|
// This will immediately stop playback, but the sink is then unusable.
|
|
|
|
// We just have to let the current buffer play till the end.
|
|
|
|
// self.rodio_sink.stop();
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn write(&mut self, data: &[i16]) -> io::Result<()> {
|
|
|
|
let source = rodio::buffer::SamplesBuffer::new(2, 44100, data);
|
|
|
|
self.jackrodio_sink.append(source);
|
|
|
|
|
|
|
|
// Chunk sizes seem to be about 256 to 3000 ish items long.
|
|
|
|
// Assuming they're on average 1628 then a half second buffer is:
|
|
|
|
// 44100 elements --> about 27 chunks
|
|
|
|
while self.jackrodio_sink.len() > 26 {
|
|
|
|
// sleep and wait for rodio to drain a bit
|
|
|
|
thread::sleep(time::Duration::from_millis(10));
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|