switch to Rust2018
[nihav-tool.git] / src / main.rs
1 extern crate nihav;
2
3 use std::io::SeekFrom;
4 use std::fs::File;
5 use std::path::Path;
6 use nihav::io::byteio::{FileReader, ByteReader};
7 use nihav::frame::*;
8 use nihav::codecs::*;
9 use nihav::demuxers::*;
10 use nihav::detect;
11 use std::env;
12
13 mod frmwriter;
14 use crate::frmwriter::*;
15
16 mod wavwriter;
17 use crate::wavwriter::WavWriter;
18
19 #[derive(Clone,Copy,PartialEq)]
20 enum NumberMode {
21 Counter,
22 PktPTS,
23 FrmPTS,
24 }
25
26 struct FrameOutput {
27 prefix: &'static str,
28 streamno: usize,
29 frameno: u64,
30 nmode: NumberMode,
31 }
32
33 impl FrameOutput {
34 fn output_frame(&mut self, pkt: &NAPacket, frmref: NAFrameRef) {
35 let frm = frmref.borrow();
36 if frm.get_frame_type() != FrameType::Skip {
37 let pts = match self.nmode {
38 NumberMode::Counter => { self.frameno },
39 NumberMode::PktPTS => { pkt.get_pts().unwrap() },
40 NumberMode::FrmPTS => { if let Some(pt) = frm.get_pts() { pt } else { pkt.get_pts().unwrap() } },
41 };
42 let frmbuf = frm.get_buffer().get_vbuf().unwrap();
43 if frmbuf.get_info().get_format().is_paletted() {
44 write_palppm(self.prefix, self.streamno, pts, frm);
45 } else {
46 write_pgmyuv(self.prefix, self.streamno, pts, frm);
47 }
48 }
49 self.frameno += 1;
50 }
51 }
52
53 struct AudioOutput {
54 wwr: WavWriter<'static>,
55 wrote_header: bool,
56 }
57
58 impl AudioOutput {
59 fn new(name: &String) -> Self { Self { wwr: WavWriter::new(name), wrote_header: false } }
60 fn output_frame(&mut self, _pkt: &NAPacket, frm: NAFrameRef) {
61 if !self.wrote_header {
62 self.wwr.write_header(frm.borrow().get_info().as_ref().get_properties().get_audio_info().unwrap()).unwrap();
63 self.wrote_header = true;
64 }
65 self.wwr.write_frame(frm.borrow().get_buffer()).unwrap();
66 }
67 }
68
69 enum Outputter {
70 Video(FrameOutput),
71 Audio(AudioOutput),
72 None,
73 }
74
75 fn main() {
76 let args: Vec<_> = env::args().collect();
77
78 if args.len() == 1 {
79 println!("usage: nihav-tool [-noout] [-vn] [-an] input [lastpts]");
80 return;
81 }
82 let mut lastpts: Option<u64> = None;
83 let mut cur_arg: usize = 1;
84 let mut noout = false;
85 let mut decode_video = true;
86 let mut decode_audio = true;
87 let mut nmode = NumberMode::FrmPTS;
88
89 while (cur_arg < args.len()) && (args[cur_arg].chars().next().unwrap() == '-') {
90 match args[cur_arg].as_str() {
91 "--" => { break; },
92 "-noout" => { noout = true; },
93 "-an" => { decode_audio = false; },
94 "-vn" => { decode_video = false; },
95 "-nm=count" => { nmode = NumberMode::Counter; },
96 "-nm=pktpts" => { nmode = NumberMode::PktPTS; },
97 "-nm=frmpts" => { nmode = NumberMode::FrmPTS; },
98 _ => { println!("unknown option {}", args[cur_arg]); return; },
99 }
100 cur_arg += 1;
101 }
102 let name = args[cur_arg].as_str();
103 cur_arg += 1;
104 if cur_arg < args.len() {
105 lastpts = Some(u64::from_str_radix(args[cur_arg].as_str(), 10).unwrap());
106 }
107
108 let path = Path::new(name);
109 let mut file = File::open(path).unwrap();
110 let dmx_fact;
111 let mut fr = FileReader::new_read(&mut file);
112 let mut br = ByteReader::new(&mut fr);
113 let res = detect::detect_format(name, &mut br);
114 if res.is_none() {
115 println!("cannot detect format for {}", name);
116 return;
117 }
118 let (dmx_name, _) = res.unwrap();
119 println!("trying demuxer {} on {}", dmx_name, name);
120 dmx_fact = find_demuxer(dmx_name).unwrap();
121 br.seek(SeekFrom::Start(0)).unwrap();
122 let mut dmx = create_demuxer(dmx_fact, &mut br).unwrap();
123
124 let mut decs: Vec<Option<Box<NADecoder>>> = Vec::new();
125 let mut sids: Vec<u32> = Vec::new();
126 let mut writers: Vec<Outputter> = Vec::new();
127 for i in 0..dmx.get_num_streams() {
128 let s = dmx.get_stream(i).unwrap();
129 let info = s.get_info();
130 let decfunc = find_decoder(info.get_name());
131 println!("stream {} - {} {}", i, s, info.get_name());
132 let str_id = s.get_id();
133 let mut has_out = false;
134 sids.push(str_id);
135 if info.is_video() {
136 if decode_video {
137 if decfunc.is_none() {
138 println!("no video decoder found!");
139 return;
140 }
141 let mut dec = (decfunc.unwrap())();
142 dec.init(info).unwrap();
143 decs.push(Some(dec));
144 if !noout {
145 writers.push(Outputter::Video(FrameOutput{prefix: "", streamno: i, frameno: 1, nmode: nmode}));
146 has_out = true;
147 }
148 } else {
149 decs.push(None);
150 }
151 } else if info.is_audio() {
152 if decode_audio {
153 if decfunc.is_none() {
154 println!("no audio decoder found!");
155 return;
156 }
157 let mut dec = (decfunc.unwrap())();
158 dec.init(info).unwrap();
159 decs.push(Some(dec));
160 if !noout {
161 let name = format!("out{:02}.wav", i);
162 writers.push(Outputter::Audio(AudioOutput::new(&name)));
163 has_out = true;
164 }
165 } else {
166 decs.push(None);
167 }
168 } else {
169 decs.push(None);
170 panic!("decoder {} not found", info.get_name());
171 }
172 if !has_out {
173 writers.push(Outputter::None);
174 }
175 }
176
177 loop {
178 let pktres = dmx.get_frame();
179 if let Err(e) = pktres {
180 if e == DemuxerError::EOF { break; }
181 }
182 let pkt = pktres.unwrap();
183 let streamno = pkt.get_stream().get_id();
184 let sr = sids.iter().position(|x| *x == streamno);
185 let idx = sr.unwrap();
186 if let Some(ref mut dec) = decs[idx] {
187 let frm = dec.decode(&pkt).unwrap();
188 if !noout {
189 match writers[idx] {
190 Outputter::Video(ref mut wr) => { wr.output_frame(&pkt, frm); },
191 Outputter::Audio(ref mut wr) => { wr.output_frame(&pkt, frm); },
192 _ => {},
193 };
194 }
195 }
196 if pkt.get_pts() != None && lastpts.is_some() && pkt.get_pts() >= lastpts { break; }
197 }
198 //panic!("end");
199 }