-
Notifications
You must be signed in to change notification settings - Fork 153
/
Copy pathffmpeg_source.rs
133 lines (119 loc) · 5.66 KB
/
ffmpeg_source.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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
use crate::source::{Fps, Source};
use crate::{BinResult, SrcPath};
use gifski::{Collector, Settings};
use imgref::*;
use rgb::*;
pub struct FfmpegDecoder {
input_context: ffmpeg::format::context::Input,
frames: u64,
rate: Fps,
settings: Settings,
}
impl Source for FfmpegDecoder {
fn total_frames(&self) -> Option<u64> {
Some(self.frames)
}
fn collect(&mut self, dest: &mut Collector) -> BinResult<()> {
self.collect_frames(dest)
}
}
impl FfmpegDecoder {
pub fn new(src: SrcPath, rate: Fps, settings: Settings) -> BinResult<Self> {
ffmpeg::init().map_err(|e| format!("Unable to initialize ffmpeg: {}", e))?;
let input_context = match src {
SrcPath::Path(path) => ffmpeg::format::input(&path)
.map_err(|e| format!("Unable to open video file {}: {}", path.display(), e))?,
SrcPath::Stdin(_) => return Err("Video files must be specified as a path on disk. Input via stdin is not supported".into()),
};
// take fps override into account
let filter_fps = rate.fps / rate.speed;
let stream = input_context.streams().best(ffmpeg::media::Type::Video).ok_or("The file has no video tracks")?;
let time_base = stream.time_base().numerator() as f64 / stream.time_base().denominator() as f64;
let frames = (stream.duration() as f64 * time_base * filter_fps as f64).ceil() as u64;
Ok(Self { input_context, frames, rate, settings })
}
#[inline(never)]
pub fn collect_frames(&mut self, dest: &mut Collector) -> BinResult<()> {
let (stream_index, mut decoder, mut filter) = {
let filter_fps = self.rate.fps / self.rate.speed;
let stream = self.input_context.streams().best(ffmpeg::media::Type::Video).ok_or("The file has no video tracks")?;
let mut codec_context = ffmpeg::codec::context::Context::new();
codec_context.set_parameters(stream.parameters())?;
let decoder = codec_context.decoder().video().map_err(|e| format!("Unable to decode the codec used in the video: {}", e))?;
let (dest_width, dest_height) = self.settings.dimensions_for_image(decoder.width() as _, decoder.height() as _);
let buffer_args = format!("width={}:height={}:video_size={}x{}:pix_fmt={}:time_base={}:sar={}",
dest_width,
dest_height,
decoder.width(),
decoder.height(),
decoder.format().descriptor().ok_or("ffmpeg format error")?.name(),
stream.time_base(),
(|sar: ffmpeg::util::rational::Rational| match sar.numerator() {
0 => "1".to_string(),
_ => format!("{}/{}", sar.numerator(), sar.denominator()),
})(decoder.aspect_ratio()),
);
let mut filter = ffmpeg::filter::Graph::new();
filter.add(&ffmpeg::filter::find("buffer").ok_or("ffmpeg format error")?, "in", &buffer_args)?;
filter.add(&ffmpeg::filter::find("buffersink").ok_or("ffmpeg format error")?, "out", "")?;
filter.output("in", 0)?.input("out", 0)?.parse(&format!("fps=fps={},format=rgba", filter_fps))?;
filter.validate()?;
(stream.index(), decoder, filter)
};
let add_frame = |rgba_frame: &ffmpeg::util::frame::Video, pts: f64, pos: i64| -> BinResult<()> {
let stride = rgba_frame.stride(0) as usize;
if stride % 4 != 0 {
Err("incompatible video")?;
}
let rgba_frame = ImgVec::new_stride(
rgba_frame.data(0).as_rgba().to_owned(),
rgba_frame.width() as usize,
rgba_frame.height() as usize,
stride / 4,
);
Ok(dest.add_frame_rgba(pos as usize, rgba_frame, pts)?)
};
let mut vid_frame = ffmpeg::util::frame::Video::empty();
let mut filt_frame = ffmpeg::util::frame::Video::empty();
let mut i = 0;
let mut pts_last_packet = 0;
let pts_frame_step = 1.0 / self.rate.fps as f64;
let packets = self.input_context.packets().filter_map(|(s, packet)| {
if s.index() != stream_index {
// ignore irrelevant streams
None
} else {
pts_last_packet = packet.pts()? + packet.duration();
Some(packet)
}
})
// extra packet to flush remaining frames
.chain(std::iter::once(ffmpeg::Packet::empty()));
for packet in packets {
decoder.send_packet(&packet)?;
loop {
match decoder.receive_frame(&mut vid_frame) {
Ok(()) => (),
Err(ffmpeg::Error::Other { errno: ffmpeg::error::EAGAIN }) | Err(ffmpeg::Error::Eof) => break,
Err(e) => return Err(Box::new(e)),
}
filter.get("in").ok_or("ffmpeg format error")?.source().add(&vid_frame)?;
let mut out = filter.get("out").ok_or("ffmpeg format error")?;
let mut out = out.sink();
while let Ok(..) = out.frame(&mut filt_frame) {
add_frame(&filt_frame, pts_frame_step * i as f64, i)?;
i += 1;
}
}
}
// now flush filter's buffer
filter.get("in").ok_or("ffmpeg format error")?.source().close(pts_last_packet)?;
let mut out = filter.get("out").ok_or("ffmpeg format error")?;
let mut out = out.sink();
while let Ok(..) = out.frame(&mut filt_frame) {
add_frame(&filt_frame, pts_frame_step * i as f64, i)?;
i += 1;
}
Ok(())
}
}