// Given an input file, transcode all video streams into H.264 (using libx264) // while copying audio and subtitle streams. // // Invocation: // // transcode-x264 [] // // is a comma-delimited list of key=val. default is "preset=medium". // See https://ffmpeg.org/ffmpeg-codecs.html#libx264_002c-libx264rgb and // https://trac.ffmpeg.org/wiki/Encode/H.264 for available and commonly used // options. // // Examples: // // transcode-x264 input.flv output.mp4 // transcode-x264 input.mkv output.mkv 'preset=veryslow,crf=18' extern crate ffmpeg_next as ffmpeg; use std::collections::HashMap; use std::env; use std::time::Instant; use ffmpeg::{ codec, decoder, encoder, format, frame, log, media, picture, Dictionary, Packet, Rational, }; const DEFAULT_X264_OPTS: &str = "preset=medium"; struct Transcoder { ost_index: usize, decoder: decoder::Video, encoder: encoder::video::Video, logging_enabled: bool, frame_count: usize, last_log_frame_count: usize, starting_time: Instant, last_log_time: Instant, } impl Transcoder { fn new( ist: &format::stream::Stream, octx: &mut format::context::Output, ost_index: usize, x264_opts: Dictionary, enable_logging: bool, ) -> Result { let global_header = octx.format().flags().contains(format::Flags::GLOBAL_HEADER); let decoder = ist.codec().decoder().video()?; let mut ost = octx.add_stream(encoder::find(codec::Id::H264))?; let mut encoder = ost.codec().encoder().video()?; encoder.set_height(decoder.height()); encoder.set_width(decoder.width()); encoder.set_aspect_ratio(decoder.aspect_ratio()); encoder.set_format(decoder.format()); encoder.set_frame_rate(decoder.frame_rate()); encoder.set_time_base(decoder.frame_rate().unwrap().invert()); if global_header { encoder.set_flags(codec::Flags::GLOBAL_HEADER); } encoder .open_with(x264_opts) .expect("error opening libx264 encoder with supplied settings"); encoder = ost.codec().encoder().video()?; ost.set_parameters(encoder); Ok(Self { ost_index, decoder, encoder: ost.codec().encoder().video()?, logging_enabled: enable_logging, frame_count: 0, last_log_frame_count: 0, starting_time: Instant::now(), last_log_time: Instant::now(), }) } fn send_packet_to_decoder(&mut self, packet: &Packet) { self.decoder.send_packet(packet).unwrap(); } fn send_eof_to_decoder(&mut self) { self.decoder.send_eof().unwrap(); } fn receive_and_process_decoded_frames( &mut self, octx: &mut format::context::Output, ost_time_base: Rational, ) { let mut frame = frame::Video::empty(); while self.decoder.receive_frame(&mut frame).is_ok() { self.frame_count += 1; let timestamp = frame.timestamp(); self.log_progress(f64::from( Rational(timestamp.unwrap_or(0) as i32, 1) * self.decoder.time_base(), )); frame.set_pts(timestamp); frame.set_kind(picture::Type::None); self.send_frame_to_encoder(&frame); self.receive_and_process_encoded_packets(octx, ost_time_base); } } fn send_frame_to_encoder(&mut self, frame: &frame::Video) { self.encoder.send_frame(frame).unwrap(); } fn send_eof_to_encoder(&mut self) { self.encoder.send_eof().unwrap(); } fn receive_and_process_encoded_packets( &mut self, octx: &mut format::context::Output, ost_time_base: Rational, ) { let mut encoded = Packet::empty(); while self.encoder.receive_packet(&mut encoded).is_ok() { encoded.set_stream(self.ost_index); encoded.rescale_ts(self.decoder.time_base(), ost_time_base); encoded.write_interleaved(octx).unwrap(); } } fn log_progress(&mut self, timestamp: f64) { if !self.logging_enabled || (self.frame_count - self.last_log_frame_count < 100 && self.last_log_time.elapsed().as_secs_f64() < 1.0) { return; } eprintln!( "time elpased: \t{:8.2}\tframe count: {:8}\ttimestamp: {:8.2}", self.starting_time.elapsed().as_secs_f64(), self.frame_count, timestamp ); self.last_log_frame_count = self.frame_count; self.last_log_time = Instant::now(); } } fn parse_opts<'a>(s: String) -> Option> { let mut dict = Dictionary::new(); for keyval in s.split_terminator(',') { let tokens: Vec<&str> = keyval.split('=').collect(); match tokens[..] { [key, val] => dict.set(key, val), _ => return None, } } Some(dict) } fn main() { let input_file = env::args().nth(1).expect("missing input file"); let output_file = env::args().nth(2).expect("missing output file"); let x264_opts = parse_opts( env::args() .nth(3) .unwrap_or_else(|| DEFAULT_X264_OPTS.to_string()), ) .expect("invalid x264 options string"); eprintln!("x264 options: {:?}", x264_opts); ffmpeg::init().unwrap(); log::set_level(log::Level::Info); let mut ictx = format::input(&input_file).unwrap(); let mut octx = format::output(&output_file).unwrap(); format::context::input::dump(&ictx, 0, Some(&input_file)); let best_video_stream_index = ictx .streams() .best(media::Type::Video) .map(|stream| stream.index()); let mut stream_mapping: Vec = vec![0; ictx.nb_streams() as _]; let mut ist_time_bases = vec![Rational(0, 0); ictx.nb_streams() as _]; let mut ost_time_bases = vec![Rational(0, 0); ictx.nb_streams() as _]; let mut transcoders = HashMap::new(); let mut ost_index = 0; for (ist_index, ist) in ictx.streams().enumerate() { let ist_medium = ist.codec().medium(); if ist_medium != media::Type::Audio && ist_medium != media::Type::Video && ist_medium != media::Type::Subtitle { stream_mapping[ist_index] = -1; continue; } stream_mapping[ist_index] = ost_index; ist_time_bases[ist_index] = ist.time_base(); if ist_medium == media::Type::Video { // Initialize transcoder for video stream. transcoders.insert( ist_index, Transcoder::new( &ist, &mut octx, ost_index as _, x264_opts.to_owned(), Some(ist_index) == best_video_stream_index, ) .unwrap(), ); } else { // Set up for stream copy for non-video stream. let mut ost = octx.add_stream(encoder::find(codec::Id::None)).unwrap(); ost.set_parameters(ist.parameters()); // We need to set codec_tag to 0 lest we run into incompatible codec tag // issues when muxing into a different container format. Unfortunately // there's no high level API to do this (yet). unsafe { (*ost.parameters().as_mut_ptr()).codec_tag = 0; } } ost_index += 1; } octx.set_metadata(ictx.metadata().to_owned()); format::context::output::dump(&octx, 0, Some(&output_file)); octx.write_header().unwrap(); for (ost_index, _) in octx.streams().enumerate() { ost_time_bases[ost_index] = octx.stream(ost_index as _).unwrap().time_base(); } for (stream, mut packet) in ictx.packets() { let ist_index = stream.index(); let ost_index = stream_mapping[ist_index]; if ost_index < 0 { continue; } let ost_time_base = ost_time_bases[ost_index as usize]; match transcoders.get_mut(&ist_index) { Some(transcoder) => { packet.rescale_ts(stream.time_base(), transcoder.decoder.time_base()); transcoder.send_packet_to_decoder(&packet); transcoder.receive_and_process_decoded_frames(&mut octx, ost_time_base); } None => { // Do stream copy on non-video streams. packet.rescale_ts(ist_time_bases[ist_index], ost_time_base); packet.set_position(-1); packet.set_stream(ost_index as _); packet.write_interleaved(&mut octx).unwrap(); } } } // Flush encoders and decoders. for (ost_index, transcoder) in transcoders.iter_mut() { let ost_time_base = ost_time_bases[*ost_index]; transcoder.send_eof_to_decoder(); transcoder.receive_and_process_decoded_frames(&mut octx, ost_time_base); transcoder.send_eof_to_encoder(); transcoder.receive_and_process_encoded_packets(&mut octx, ost_time_base); } octx.write_trailer().unwrap(); }