From c20978fc015acfb0367cd460c6d0ac661576bc1b Mon Sep 17 00:00:00 2001 From: mitchmindtree Date: Sat, 17 Feb 2018 15:16:03 +1100 Subject: [PATCH] Add an example that demonstrates feeding input directly to output with some latency --- examples/feedback.rs | 90 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 90 insertions(+) create mode 100644 examples/feedback.rs diff --git a/examples/feedback.rs b/examples/feedback.rs new file mode 100644 index 0000000..f75b0d2 --- /dev/null +++ b/examples/feedback.rs @@ -0,0 +1,90 @@ +//! Feeds back the input stream directly into the output stream. +//! +//! Assumes that the input and output devices can use the same stream format and that they support +//! the f32 sample format. +//! +//! Uses a delay of `LATENCY_MS` milliseconds in case the default input and output streams are not +//! precisely synchronised. + +extern crate cpal; + +const LATENCY_MS: f32 = 150.0; + +fn main() { + let event_loop = cpal::EventLoop::new(); + + // Default devices. + let input_device = cpal::default_input_device().expect("Failed to get default input device"); + let output_device = cpal::default_output_device().expect("Failed to get default output device"); + println!("Using default input device: \"{}\"", input_device.name()); + println!("Using default output device: \"{}\"", output_device.name()); + + // We'll try and use the same format between streams to keep it simple + let mut format = input_device.default_output_format().expect("Failed to get default format"); + format.data_type = cpal::SampleFormat::F32; + + // Build streams. + println!("Attempting to build both streams with `{:?}`.", format); + let input_stream_id = event_loop.build_input_stream(&input_device, &format).unwrap(); + let output_stream_id = event_loop.build_output_stream(&output_device, &format).unwrap(); + println!("Successfully built streams."); + + // Create a delay in case the input and output devices aren't synced. + let latency_frames = (LATENCY_MS / 1_000.0) * format.sample_rate.0 as f32; + let latency_samples = latency_frames as usize * format.channels as usize; + + // The channel to share samples. + let (tx, rx) = std::sync::mpsc::sync_channel(latency_samples * 2); + + // Fill the samples with 0.0 equal to the length of the delay. + for _ in 0..latency_samples { + tx.send(0.0).unwrap(); + } + + // Play the streams. + println!("Starting the input and output streams with `{}` milliseconds of latency.", LATENCY_MS); + event_loop.play_stream(input_stream_id.clone()); + event_loop.play_stream(output_stream_id.clone()); + + // Run the event loop on a separate thread. + std::thread::spawn(move || { + event_loop.run(move |id, data| { + match data { + cpal::StreamData::Input { buffer: cpal::UnknownTypeInputBuffer::F32(buffer) } => { + assert_eq!(id, input_stream_id); + let mut output_fell_behind = false; + for &sample in buffer.iter() { + if tx.try_send(sample).is_err() { + output_fell_behind = true; + } + } + if output_fell_behind { + eprintln!("output stream fell behind: try increasing latency"); + } + }, + cpal::StreamData::Output { buffer: cpal::UnknownTypeOutputBuffer::F32(mut buffer) } => { + assert_eq!(id, output_stream_id); + let mut input_fell_behind = None; + for sample in buffer.iter_mut() { + *sample = match rx.try_recv() { + Ok(s) => s, + Err(err) => { + input_fell_behind = Some(err); + 0.0 + }, + }; + } + if let Some(err) = input_fell_behind { + eprintln!("input stream fell behind: {}: try increasing latency", err); + } + }, + _ => panic!("we're expecting f32 data"), + } + }); + }); + + // Run for 3 seconds before closing. + println!("Playing for 3 seconds... "); + std::thread::sleep(std::time::Duration::from_secs(3)); + println!("Done!"); +}