+
+pub struct OutputBuffer {
+ idling: bool,
+ // TODO: something more efficient
+ samples: VecDeque<i16>,
+ latency: Duration,
+}
+
+impl OutputBuffer {
+ pub fn new() -> Self {
+ Self {
+ idling: true,
+ samples: VecDeque::new(),
+ latency: Duration::ZERO,
+ }
+ }
+}
+
+pub trait OutputSink: Send + Sync + 'static {
+ fn start(&self, event_tx: SyncSender<SoundmodemEvent>, buffer: Arc<RwLock<OutputBuffer>>);
+ fn close(&self);
+}
+
+pub struct OutputRrcFile {
+ path: PathBuf,
+ end_tx: Mutex<Option<Sender<()>>>,
+}
+
+impl OutputRrcFile {
+ pub fn new(path: PathBuf) -> Self {
+ Self {
+ path,
+ end_tx: Mutex::new(None),
+ }
+ }
+}
+
+impl OutputSink for OutputRrcFile {
+ fn start(&self, event_tx: SyncSender<SoundmodemEvent>, buffer: Arc<RwLock<OutputBuffer>>) {
+ let (end_tx, end_rx) = channel();
+ let path = self.path.clone();
+ std::thread::spawn(move || {
+ // TODO: error handling
+ let mut file = File::create(path).unwrap();
+
+ // assuming 48 kHz for now
+ const TICK: Duration = Duration::from_millis(25);
+ const SAMPLES_PER_TICK: usize = 1200;
+
+ // flattened BE i16s for writing
+ let mut buf = [0u8; SAMPLES_PER_TICK * 2];
+ let mut next_tick = Instant::now() + TICK;
+
+ loop {
+ std::thread::sleep(next_tick.duration_since(Instant::now()));
+ next_tick = next_tick + TICK;
+ if end_rx.try_recv() != Err(TryRecvError::Empty) {
+ break;
+ }
+
+ let mut buffer = buffer.write().unwrap();
+ for out in buf.chunks_mut(2) {
+ if let Some(s) = buffer.samples.pop_front() {
+ let be = s.to_be_bytes();
+ out.copy_from_slice(&[be[0], be[1]]);
+ } else if buffer.idling {
+ out.copy_from_slice(&[0, 0]);
+ } else {
+ debug!("output rrc file had underrun");
+ let _ = event_tx.send(SoundmodemEvent::OutputUnderrun);
+ break;
+ }
+ }
+ if let Err(e) = file.write_all(&buf) {
+ debug!("failed to write to rrc file: {e:?}");
+ break;
+ }
+ }
+
+ });
+ *self.end_tx.lock().unwrap() = Some(end_tx);
+ }
+
+ fn close(&self) {
+ let _ = self.end_tx.lock().unwrap().take();
+ }
+}