forked from RustAudio/dasp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsynth.rs
82 lines (73 loc) · 2.42 KB
/
synth.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
use cpal;
use cpal::traits::{DeviceTrait, HostTrait, StreamTrait};
use dasp::{signal, Sample, Signal};
use std::sync::mpsc;
fn main() -> Result<(), anyhow::Error> {
let host = cpal::default_host();
let device = host
.default_output_device()
.expect("failed to find a default output device");
let config = device.default_output_config()?;
match config.sample_format() {
cpal::SampleFormat::F32 => run::<f32>(&device, &config.into())?,
cpal::SampleFormat::I16 => run::<i16>(&device, &config.into())?,
cpal::SampleFormat::U16 => run::<u16>(&device, &config.into())?,
}
Ok(())
}
fn run<T>(device: &cpal::Device, config: &cpal::StreamConfig) -> Result<(), anyhow::Error>
where
T: cpal::Sample,
{
// Create a signal chain to play back 1 second of each oscillator at A4.
let hz = signal::rate(config.sample_rate.0 as f64).const_hz(440.0);
let one_sec = config.sample_rate.0 as usize;
let mut synth = hz
.clone()
.sine()
.take(one_sec)
.chain(hz.clone().saw().take(one_sec))
.chain(hz.clone().square().take(one_sec))
.chain(hz.clone().noise_simplex().take(one_sec))
.chain(signal::noise(0).take(one_sec))
.map(|s| s.to_sample::<f32>() * 0.2);
// A channel for indicating when playback has completed.
let (complete_tx, complete_rx) = mpsc::sync_channel(1);
// Create and run the stream.
let err_fn = |err| eprintln!("an error occurred on stream: {}", err);
let channels = config.channels as usize;
let stream = device.build_output_stream(
config,
move |data: &mut [T], _: &cpal::OutputCallbackInfo| {
write_data(data, channels, &complete_tx, &mut synth)
},
err_fn,
)?;
stream.play()?;
// Wait for playback to complete.
complete_rx.recv().unwrap();
stream.pause()?;
Ok(())
}
fn write_data<T>(
output: &mut [T],
channels: usize,
complete_tx: &mpsc::SyncSender<()>,
signal: &mut dyn Iterator<Item = f32>,
) where
T: cpal::Sample,
{
for frame in output.chunks_mut(channels) {
let sample = match signal.next() {
None => {
complete_tx.try_send(()).ok();
0.0
}
Some(sample) => sample,
};
let value: T = cpal::Sample::from::<f32>(&sample);
for sample in frame.iter_mut() {
*sample = value;
}
}
}