From 1ee737a48ec7a767c64bd72e67313ef67cfcc367 Mon Sep 17 00:00:00 2001 From: Kostya Shishkov Date: Sat, 3 Jun 2023 16:56:56 +0200 Subject: [PATCH] add YUV4MPEG muxer --- nihav-commonfmt/Cargo.toml | 3 +- nihav-commonfmt/src/muxers/mod.rs | 4 + nihav-commonfmt/src/muxers/y4m.rs | 120 ++++++++++++++++++++++++++++++ 3 files changed, 126 insertions(+), 1 deletion(-) create mode 100644 nihav-commonfmt/src/muxers/y4m.rs diff --git a/nihav-commonfmt/Cargo.toml b/nihav-commonfmt/Cargo.toml index 12c33fd..6e7d020 100644 --- a/nihav-commonfmt/Cargo.toml +++ b/nihav-commonfmt/Cargo.toml @@ -28,9 +28,10 @@ demuxer_avi = ["demuxers"] demuxer_mov = ["demuxers"] demuxer_wav = ["demuxers"] demuxer_y4m = ["demuxers"] -all_muxers = ["muxer_avi", "muxer_wav"] +all_muxers = ["muxer_avi", "muxer_wav", "muxer_y4m"] muxer_avi = ["muxers"] muxer_wav = ["muxers"] +muxer_y4m = ["muxers"] all_decoders = ["all_video_decoders", "all_audio_decoders"] diff --git a/nihav-commonfmt/src/muxers/mod.rs b/nihav-commonfmt/src/muxers/mod.rs index 6df929c..809283f 100644 --- a/nihav-commonfmt/src/muxers/mod.rs +++ b/nihav-commonfmt/src/muxers/mod.rs @@ -4,12 +4,16 @@ use nihav_core::muxers::*; mod avi; #[cfg(feature="muxer_wav")] mod wav; +#[cfg(feature="muxer_y4m")] +mod y4m; const MUXERS: &[&dyn MuxerCreator] = &[ #[cfg(feature="muxer_avi")] &avi::AVIMuxerCreator {}, #[cfg(feature="muxer_wav")] &wav::WAVMuxerCreator {}, +#[cfg(feature="muxer_y4m")] + &y4m::Y4MMuxerCreator {}, ]; pub fn generic_register_all_muxers(rm: &mut RegisteredMuxers) { diff --git a/nihav-commonfmt/src/muxers/y4m.rs b/nihav-commonfmt/src/muxers/y4m.rs new file mode 100644 index 0000000..600f78e --- /dev/null +++ b/nihav-commonfmt/src/muxers/y4m.rs @@ -0,0 +1,120 @@ +use nihav_core::muxers::*; + +struct Y4MMuxer<'a> { + bw: &'a mut ByteWriter<'a>, +} + +impl<'a> Y4MMuxer<'a> { + fn new(bw: &'a mut ByteWriter<'a>) -> Self { + Self { + bw, + } + } +} + +fn get_format_name(fmt: NAPixelFormaton) -> MuxerResult<&'static str> { + if fmt.model.is_yuv() && fmt.get_max_depth() == 8 { + match fmt.components { + 1 => Ok("mono"), + 3 => { + let uinfo = fmt.comp_info[1].unwrap(); + let vinfo = fmt.comp_info[2].unwrap(); + if uinfo.h_ss != vinfo.h_ss || uinfo.v_ss != vinfo.v_ss { + return Err(MuxerError::UnsupportedFormat); + } + match (uinfo.h_ss, vinfo.v_ss) { + (0, 0) => Ok("444"), + (1, 0) => Ok("422"), + (1, 1) => Ok("420"), + (2, 0) => Ok("411"), + _ => Err(MuxerError::UnsupportedFormat), + } + }, + _ => Err(MuxerError::UnsupportedFormat), + } + } else { + Err(MuxerError::UnsupportedFormat) + } +} + +impl<'a> MuxCore<'a> for Y4MMuxer<'a> { + fn create(&mut self, strmgr: &StreamManager) -> MuxerResult<()> { + if strmgr.get_num_streams() != 1 { + return Err(MuxerError::InvalidArgument); + } + let vstr = strmgr.get_stream(0).unwrap(); + if vstr.get_media_type() != StreamType::Video { + return Err(MuxerError::UnsupportedFormat); + } + let info = vstr.get_info(); + let vinfo = info.get_properties().get_video_info().unwrap(); + self.bw.write_buf(format!("YUV4MPEG2 W{} H{} F{}:{} C{}\n", vinfo.width, vinfo.height, vstr.tb_den, vstr.tb_num, get_format_name(vinfo.format)?).as_bytes())?; + + Ok(()) + } + fn mux_frame(&mut self, _strmgr: &StreamManager, pkt: NAPacket) -> MuxerResult<()> { + if self.bw.tell() == 0 { + return Err(MuxerError::NotCreated); + } + let src = pkt.get_buffer(); + self.bw.write_buf(b"FRAME\n")?; + self.bw.write_buf(&src)?; + Ok(()) + } + fn flush(&mut self) -> MuxerResult<()> { + Ok(()) + } + fn end(&mut self) -> MuxerResult<()> { + Ok(()) + } +} + +impl<'a> NAOptionHandler for Y4MMuxer<'a> { + fn get_supported_options(&self) -> &[NAOptionDefinition] { &[] } + fn set_options(&mut self, _options: &[NAOption]) { } + fn query_option_value(&self, _name: &str) -> Option { None } +} + +pub struct Y4MMuxerCreator {} + +impl MuxerCreator for Y4MMuxerCreator { + fn new_muxer<'a>(&self, bw: &'a mut ByteWriter<'a>) -> Box + 'a> { + Box::new(Y4MMuxer::new(bw)) + } + fn get_name(&self) -> &'static str { "yuv4mpeg" } + fn get_capabilities(&self) -> MuxerCapabilities { MuxerCapabilities::SingleVideo("rawvideo") } +} + +#[cfg(test)] +mod test { + use nihav_core::codecs::*; + use nihav_core::demuxers::*; + use nihav_core::muxers::*; + use nihav_codec_support::test::enc_video::*; + use crate::*; + + #[test] + fn test_y4m_muxer() { + let mut dmx_reg = RegisteredDemuxers::new(); + generic_register_all_demuxers(&mut dmx_reg); + // sample: self-created with avconv + let dec_config = DecoderTestParams { + demuxer: "yuv4mpeg", + in_name: "assets/Misc/test.y4m", + limit: None, + stream_type: StreamType::None, + dmx_reg, dec_reg: RegisteredDecoders::new(), + }; + let mut mux_reg = RegisteredMuxers::new(); + generic_register_all_muxers(&mut mux_reg); + /*let enc_config = EncoderTestParams { + muxer: "yuv4mpeg", + enc_name: "", + out_name: "muxed.y4m", + mux_reg, enc_reg: RegisteredEncoders::new(), + }; + test_remuxing(&dec_config, &enc_config);*/ + test_remuxing_md5(&dec_config, "yuv4mpeg", &mux_reg, + [0x7586c1c5, 0x388209b8, 0xe08af8f8, 0x4b6b96c7]); + } +} -- 2.39.5