summaryrefslogtreecommitdiff
path: root/per.rs
blob: f4f2a1e817535c88edaa230e866a687aa421a5b3 (plain)
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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
extern mod extra;
use extra::getopts::*;
use extra::time::*;
use std::os;
use std::path::Path;
use std::rt::io::io_error;
use std::task::{SingleThreaded, spawn_sched};

use oss::*;
use mp3lame::*;
mod oss;
mod mp3lame;


#[fixed_stack_segment]
fn main() {
  let args = os::args();
  let opts = ~[
    groups::optmulti("f", "file", "OSS device file", "/dev/dsp"),
    groups::optmulti("r", "rate", "Sample rate in Hz", "44100"),
    groups::optopt("s", "split",
      "Number of minutes at which to split MP3 files", "60"),
    groups::optflag("a",
      "Align splits as if the first one happened midnight Jan. 1, 1970", "")
  ];
  let DSP_FILES = ~[~"/dev/dsp", ~"/dev/dsp1"];
  let DSP_SPEEDS = ~[44100i, 48000i];
  let matches = match groups::getopts(args.tail(), opts) {
    Ok(m) => m,
    Err(f) => {
      println(f.to_err_msg());
      print(groups::usage("Usage: per", opts));
      return;
    }
  };
  let dsp_files = match matches.opt_strs("f") {
    [] => DSP_FILES,
    f => f
  };
  let lame = LameContext::new();
  let mut foo = None;
  for file_name in dsp_files.iter() {
    match OssDevice::new(&Path(file_name.as_slice())) {
      Some(x) => { foo = Some(x); break }
      None => {}
    }
  };
  let dsp = match foo {
    Some(x) => x,
    None => fail!("Unable to open dsp device")
  };
  dsp.reset();
  dsp.set_format();
  dsp.set_stereo();
  lame.set_num_channels(2);
  let dsp_speeds = match matches.opt_strs("r") {
    [] => DSP_SPEEDS,
    r => r.map(|x| { from_str::<int>(*x).unwrap() })
  };
  let mut speed: int = 0;
  for dsp_speed in dsp_speeds.iter() {
    do io_error::cond.trap(|_| {debug!("speed %d is a no go", *dsp_speed); speed = 0}).inside {
      dsp.set_speed(*dsp_speed);
      speed = *dsp_speed;
    }
    if speed != 0 {
      break;
    }
  }
  lame.set_in_samplerate(speed);
  lame.set_quality(2);
  lame.set_bitrate(128);
  lame.set_disable_reservoir(true);
  lame.init_params();
  println(fmt!("Recording sample rate: %d Hz", speed));
  println(fmt!("Encoding sample rate:  %d Hz", lame.get_out_samplerate()));
  let split = match matches.opt_str("s") {
    Some(s) => from_str::<int>(s).unwrap() * 60,
    None => 3600
  };
  let mut next_split = if matches.opt_present("a") {
    let Timespec { sec, _ } = get_time();
    Timespec::new(sec - sec % split as i64, 0)
  } else {
    get_time()
  };
  let (port, chan) = stream::<~[u8]>();
  do spawn_sched(SingleThreaded) {
    dsp.read_all(&chan);
  }
  loop {
    if next_split <= get_time() {
      debug!("split!");
      let Timespec { sec, nsec } = next_split;
      next_split = Timespec::new(sec + split as i64, nsec);
    }
    let buffer = port.recv();
    debug!("Read buffer of length %u", buffer.len());
    let mp3buf = lame.encode_buffer_interleaved(buffer);
    debug!("Encoded buffer of length %u", mp3buf.len());
  }
}