| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | // https://github.com/astraw/vpx-encode
 | 
					
						
							|  |  |  | // https://github.com/astraw/env-libvpx-sys
 | 
					
						
							|  |  |  | // https://github.com/rust-av/vpx-rs/blob/master/src/decoder.rs
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use hbb_common::anyhow::{anyhow, Context};
 | 
					
						
							| 
									
										
										
										
											2023-05-20 16:23:00 +08:00
										 |  |  | use hbb_common::log;
 | 
					
						
							| 
									
										
										
										
											2022-07-06 10:39:00 +08:00
										 |  |  | use hbb_common::message_proto::{EncodedVideoFrame, EncodedVideoFrames, Message, VideoFrame};
 | 
					
						
							| 
									
										
										
										
											2023-03-23 14:31:50 +08:00
										 |  |  | use hbb_common::ResultType;
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-20 21:16:38 +08:00
										 |  |  | use crate::codec::{base_bitrate, codec_thread_num, EncoderApi, Quality};
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  | use crate::{GoogleImage, STRIDE_ALIGN};
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  | use super::vpx::{vp8e_enc_control_id::*, vpx_codec_err_t::*, *};
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  | use crate::{generate_call_macro, generate_call_ptr_macro, Error, Result};
 | 
					
						
							| 
									
										
										
										
											2022-09-11 21:46:53 +08:00
										 |  |  | use hbb_common::bytes::Bytes;
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  | use std::os::raw::{c_int, c_uint};
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | use std::{ptr, slice};
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 16:23:00 +08:00
										 |  |  | generate_call_macro!(call_vpx, false);
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  | generate_call_ptr_macro!(call_vpx_ptr);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  | const DEFAULT_QP_MAX: u32 = 56; // no more than 63
 | 
					
						
							|  |  |  | const DEFAULT_QP_MIN: u32 = 12; // no more than 63
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
 | 
					
						
							|  |  |  | pub enum VpxVideoCodecId {
 | 
					
						
							|  |  |  |     VP8,
 | 
					
						
							|  |  |  |     VP9,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl Default for VpxVideoCodecId {
 | 
					
						
							|  |  |  |     fn default() -> VpxVideoCodecId {
 | 
					
						
							|  |  |  |         VpxVideoCodecId::VP9
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct VpxEncoder {
 | 
					
						
							|  |  |  |     ctx: vpx_codec_ctx_t,
 | 
					
						
							|  |  |  |     width: usize,
 | 
					
						
							|  |  |  |     height: usize,
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |     id: VpxVideoCodecId,
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct VpxDecoder {
 | 
					
						
							|  |  |  |     ctx: vpx_codec_ctx_t,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl EncoderApi for VpxEncoder {
 | 
					
						
							| 
									
										
										
										
											2022-05-29 19:22:09 +08:00
										 |  |  |     fn new(cfg: crate::codec::EncoderCfg) -> ResultType<Self>
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |     where
 | 
					
						
							|  |  |  |         Self: Sized,
 | 
					
						
							|  |  |  |     {
 | 
					
						
							|  |  |  |         match cfg {
 | 
					
						
							| 
									
										
										
										
											2022-05-29 19:22:09 +08:00
										 |  |  |             crate::codec::EncoderCfg::VPX(config) => {
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |                 let i = match config.codec {
 | 
					
						
							|  |  |  |                     VpxVideoCodecId::VP8 => call_vpx_ptr!(vpx_codec_vp8_cx()),
 | 
					
						
							|  |  |  |                     VpxVideoCodecId::VP9 => call_vpx_ptr!(vpx_codec_vp9_cx()),
 | 
					
						
							|  |  |  |                 };
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  |                 let mut c = unsafe { std::mem::MaybeUninit::zeroed().assume_init() };
 | 
					
						
							|  |  |  |                 call_vpx!(vpx_codec_enc_config_default(i, &mut c, 0));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // https://www.webmproject.org/docs/encoder-parameters/
 | 
					
						
							|  |  |  |                 // default: c.rc_min_quantizer = 0, c.rc_max_quantizer = 63
 | 
					
						
							|  |  |  |                 // try rc_resize_allowed later
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 c.g_w = config.width;
 | 
					
						
							|  |  |  |                 c.g_h = config.height;
 | 
					
						
							| 
									
										
										
										
											2023-08-07 21:32:36 +08:00
										 |  |  |                 c.g_timebase.num = 1;
 | 
					
						
							|  |  |  |                 c.g_timebase.den = 1000; // Output timestamp precision
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  |                 c.rc_undershoot_pct = 95;
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  |                 // When the data buffer falls below this percentage of fullness, a dropped frame is indicated. Set the threshold to zero (0) to disable this feature.
 | 
					
						
							|  |  |  |                 // In dynamic scenes, low bitrate gets low fps while high bitrate gets high fps.
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  |                 c.rc_dropframe_thresh = 25;
 | 
					
						
							| 
									
										
										
										
											2023-07-20 21:16:38 +08:00
										 |  |  |                 c.g_threads = codec_thread_num() as _;
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  |                 c.g_error_resilient = VPX_ERROR_RESILIENT_DEFAULT;
 | 
					
						
							|  |  |  |                 // https://developers.google.com/media/vp9/bitrate-modes/
 | 
					
						
							|  |  |  |                 // Constant Bitrate mode (CBR) is recommended for live streaming with VP9.
 | 
					
						
							|  |  |  |                 c.rc_end_usage = vpx_rc_mode::VPX_CBR;
 | 
					
						
							| 
									
										
										
										
											2023-08-07 21:32:36 +08:00
										 |  |  |                 if let Some(keyframe_interval) = config.keyframe_interval {
 | 
					
						
							|  |  |  |                     c.kf_min_dist = 0;
 | 
					
						
							|  |  |  |                     c.kf_max_dist = keyframe_interval as _;
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     c.kf_mode = vpx_kf_mode::VPX_KF_DISABLED; // reduce bandwidth a lot
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  |                 let (q_min, q_max, b) = Self::convert_quality(config.quality);
 | 
					
						
							|  |  |  |                 if q_min > 0 && q_min < q_max && q_max < 64 {
 | 
					
						
							|  |  |  |                     c.rc_min_quantizer = q_min;
 | 
					
						
							|  |  |  |                     c.rc_max_quantizer = q_max;
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     c.rc_min_quantizer = DEFAULT_QP_MIN;
 | 
					
						
							|  |  |  |                     c.rc_max_quantizer = DEFAULT_QP_MAX;
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 let base_bitrate = base_bitrate(config.width as _, config.height as _);
 | 
					
						
							|  |  |  |                 let bitrate = base_bitrate * b / 100;
 | 
					
						
							|  |  |  |                 if bitrate > 0 {
 | 
					
						
							|  |  |  |                     c.rc_target_bitrate = bitrate;
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     c.rc_target_bitrate = base_bitrate;
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 /*
 | 
					
						
							|  |  |  |                 The VPX encoder supports two-pass encoding for rate control purposes.
 | 
					
						
							|  |  |  |                 In two-pass encoding, the entire encoding process is performed twice.
 | 
					
						
							|  |  |  |                 The first pass generates new control parameters for the second pass.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 This approach enables the best PSNR at the same bit rate.
 | 
					
						
							|  |  |  |                 */
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 let mut ctx = Default::default();
 | 
					
						
							|  |  |  |                 call_vpx!(vpx_codec_enc_init_ver(
 | 
					
						
							|  |  |  |                     &mut ctx,
 | 
					
						
							|  |  |  |                     i,
 | 
					
						
							|  |  |  |                     &c,
 | 
					
						
							|  |  |  |                     0,
 | 
					
						
							|  |  |  |                     VPX_ENCODER_ABI_VERSION as _
 | 
					
						
							|  |  |  |                 ));
 | 
					
						
							| 
									
										
										
										
											2023-07-18 13:49:52 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if config.codec == VpxVideoCodecId::VP9 {
 | 
					
						
							|  |  |  |                     // set encoder internal speed settings
 | 
					
						
							|  |  |  |                     // in ffmpeg, it is --speed option
 | 
					
						
							|  |  |  |                     /*
 | 
					
						
							|  |  |  |                     set to 0 or a positive value 1-16, the codec will try to adapt its
 | 
					
						
							|  |  |  |                     complexity depending on the time it spends encoding. Increasing this
 | 
					
						
							|  |  |  |                     number will make the speed go up and the quality go down.
 | 
					
						
							|  |  |  |                     Negative values mean strict enforcement of this
 | 
					
						
							|  |  |  |                     while positive values are adaptive
 | 
					
						
							|  |  |  |                     */
 | 
					
						
							|  |  |  |                     /* https://developers.google.com/media/vp9/live-encoding
 | 
					
						
							|  |  |  |                     Speed 5 to 8 should be used for live / real-time encoding.
 | 
					
						
							|  |  |  |                     Lower numbers (5 or 6) are higher quality but require more CPU power.
 | 
					
						
							|  |  |  |                     Higher numbers (7 or 8) will be lower quality but more manageable for lower latency
 | 
					
						
							|  |  |  |                     use cases and also for lower CPU power devices such as mobile.
 | 
					
						
							|  |  |  |                     */
 | 
					
						
							|  |  |  |                     call_vpx!(vpx_codec_control_(&mut ctx, VP8E_SET_CPUUSED as _, 7,));
 | 
					
						
							|  |  |  |                     // set row level multi-threading
 | 
					
						
							|  |  |  |                     /*
 | 
					
						
							|  |  |  |                     as some people in comments and below have already commented,
 | 
					
						
							|  |  |  |                     more recent versions of libvpx support -row-mt 1 to enable tile row
 | 
					
						
							|  |  |  |                     multi-threading. This can increase the number of tiles by up to 4x in VP9
 | 
					
						
							|  |  |  |                     (since the max number of tile rows is 4, regardless of video height).
 | 
					
						
							|  |  |  |                     To enable this, use -tile-rows N where N is the number of tile rows in
 | 
					
						
							|  |  |  |                     log2 units (so -tile-rows 1 means 2 tile rows and -tile-rows 2 means 4 tile
 | 
					
						
							|  |  |  |                     rows). The total number of active threads will then be equal to
 | 
					
						
							|  |  |  |                     $tile_rows * $tile_columns
 | 
					
						
							|  |  |  |                     */
 | 
					
						
							|  |  |  |                     call_vpx!(vpx_codec_control_(
 | 
					
						
							|  |  |  |                         &mut ctx,
 | 
					
						
							|  |  |  |                         VP9E_SET_ROW_MT as _,
 | 
					
						
							|  |  |  |                         1 as c_int
 | 
					
						
							|  |  |  |                     ));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     call_vpx!(vpx_codec_control_(
 | 
					
						
							|  |  |  |                         &mut ctx,
 | 
					
						
							|  |  |  |                         VP9E_SET_TILE_COLUMNS as _,
 | 
					
						
							|  |  |  |                         4 as c_int
 | 
					
						
							|  |  |  |                     ));
 | 
					
						
							|  |  |  |                 } else if config.codec == VpxVideoCodecId::VP8 {
 | 
					
						
							|  |  |  |                     // https://github.com/webmproject/libvpx/blob/972149cafeb71d6f08df89e91a0130d6a38c4b15/vpx/vp8cx.h#L172
 | 
					
						
							|  |  |  |                     // https://groups.google.com/a/webmproject.org/g/webm-discuss/c/DJhSrmfQ61M
 | 
					
						
							|  |  |  |                     call_vpx!(vpx_codec_control_(&mut ctx, VP8E_SET_CPUUSED as _, 12,));
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |                 }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 Ok(Self {
 | 
					
						
							|  |  |  |                     ctx,
 | 
					
						
							|  |  |  |                     width: config.width as _,
 | 
					
						
							|  |  |  |                     height: config.height as _,
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |                     id: config.codec,
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |                 })
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             _ => Err(anyhow!("encoder type mismatch")),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn encode_to_message(&mut self, frame: &[u8], ms: i64) -> ResultType<Message> {
 | 
					
						
							|  |  |  |         let mut frames = Vec::new();
 | 
					
						
							|  |  |  |         for ref frame in self
 | 
					
						
							| 
									
										
										
										
											2023-03-03 13:18:34 +08:00
										 |  |  |             .encode(ms, frame, STRIDE_ALIGN)
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |             .with_context(|| "Failed to encode")?
 | 
					
						
							|  |  |  |         {
 | 
					
						
							|  |  |  |             frames.push(VpxEncoder::create_frame(frame));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         for ref frame in self.flush().with_context(|| "Failed to flush")? {
 | 
					
						
							|  |  |  |             frames.push(VpxEncoder::create_frame(frame));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // to-do: flush periodically, e.g. 1 second
 | 
					
						
							|  |  |  |         if frames.len() > 0 {
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |             Ok(VpxEncoder::create_msg(self.id, frames))
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |         } else {
 | 
					
						
							|  |  |  |             Err(anyhow!("no valid frame"))
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn use_yuv(&self) -> bool {
 | 
					
						
							|  |  |  |         true
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2022-06-27 15:21:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  |     fn set_quality(&mut self, quality: Quality) -> ResultType<()> {
 | 
					
						
							|  |  |  |         let mut c = unsafe { *self.ctx.config.enc.to_owned() };
 | 
					
						
							|  |  |  |         let (q_min, q_max, b) = Self::convert_quality(quality);
 | 
					
						
							|  |  |  |         if q_min > 0 && q_min < q_max && q_max < 64 {
 | 
					
						
							|  |  |  |             c.rc_min_quantizer = q_min;
 | 
					
						
							|  |  |  |             c.rc_max_quantizer = q_max;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         let bitrate = base_bitrate(self.width as _, self.height as _) * b / 100;
 | 
					
						
							|  |  |  |         if bitrate > 0 {
 | 
					
						
							|  |  |  |             c.rc_target_bitrate = bitrate;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         call_vpx!(vpx_codec_enc_config_set(&mut self.ctx, &c));
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn bitrate(&self) -> u32 {
 | 
					
						
							|  |  |  |         let c = unsafe { *self.ctx.config.enc.to_owned() };
 | 
					
						
							|  |  |  |         c.rc_target_bitrate
 | 
					
						
							| 
									
										
										
										
											2022-06-27 15:21:31 +08:00
										 |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl VpxEncoder {
 | 
					
						
							| 
									
										
										
										
											2023-03-03 13:18:34 +08:00
										 |  |  |     pub fn encode(&mut self, pts: i64, data: &[u8], stride_align: usize) -> Result<EncodeFrames> {
 | 
					
						
							| 
									
										
										
										
											2022-11-10 10:27:13 +08:00
										 |  |  |         if 2 * data.len() < 3 * self.width * self.height {
 | 
					
						
							|  |  |  |             return Err(Error::FailedCall("len not enough".to_string()));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         let mut image = Default::default();
 | 
					
						
							|  |  |  |         call_vpx_ptr!(vpx_img_wrap(
 | 
					
						
							|  |  |  |             &mut image,
 | 
					
						
							|  |  |  |             vpx_img_fmt::VPX_IMG_FMT_I420,
 | 
					
						
							|  |  |  |             self.width as _,
 | 
					
						
							|  |  |  |             self.height as _,
 | 
					
						
							| 
									
										
										
										
											2023-03-03 13:18:34 +08:00
										 |  |  |             stride_align as _,
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |             data.as_ptr() as _,
 | 
					
						
							|  |  |  |         ));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         call_vpx!(vpx_codec_encode(
 | 
					
						
							|  |  |  |             &mut self.ctx,
 | 
					
						
							|  |  |  |             &image,
 | 
					
						
							|  |  |  |             pts as _,
 | 
					
						
							|  |  |  |             1, // Duration
 | 
					
						
							|  |  |  |             0, // Flags
 | 
					
						
							|  |  |  |             VPX_DL_REALTIME as _,
 | 
					
						
							|  |  |  |         ));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Ok(EncodeFrames {
 | 
					
						
							|  |  |  |             ctx: &mut self.ctx,
 | 
					
						
							|  |  |  |             iter: ptr::null(),
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Notify the encoder to return any pending packets
 | 
					
						
							|  |  |  |     pub fn flush(&mut self) -> Result<EncodeFrames> {
 | 
					
						
							|  |  |  |         call_vpx!(vpx_codec_encode(
 | 
					
						
							|  |  |  |             &mut self.ctx,
 | 
					
						
							|  |  |  |             ptr::null(),
 | 
					
						
							|  |  |  |             -1, // PTS
 | 
					
						
							|  |  |  |             1,  // Duration
 | 
					
						
							|  |  |  |             0,  // Flags
 | 
					
						
							|  |  |  |             VPX_DL_REALTIME as _,
 | 
					
						
							|  |  |  |         ));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Ok(EncodeFrames {
 | 
					
						
							|  |  |  |             ctx: &mut self.ctx,
 | 
					
						
							|  |  |  |             iter: ptr::null(),
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |     pub fn create_msg(codec_id: VpxVideoCodecId, frames: Vec<EncodedVideoFrame>) -> Message {
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |         let mut msg_out = Message::new();
 | 
					
						
							|  |  |  |         let mut vf = VideoFrame::new();
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |         let vpxs = EncodedVideoFrames {
 | 
					
						
							|  |  |  |             frames: frames.into(),
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |             ..Default::default()
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |         };
 | 
					
						
							|  |  |  |         match codec_id {
 | 
					
						
							|  |  |  |             VpxVideoCodecId::VP8 => vf.set_vp8s(vpxs),
 | 
					
						
							|  |  |  |             VpxVideoCodecId::VP9 => vf.set_vp9s(vpxs),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |         msg_out.set_video_frame(vf);
 | 
					
						
							|  |  |  |         msg_out
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							| 
									
										
										
										
											2022-07-05 16:16:08 +08:00
										 |  |  |     fn create_frame(frame: &EncodeFrame) -> EncodedVideoFrame {
 | 
					
						
							|  |  |  |         EncodedVideoFrame {
 | 
					
						
							| 
									
										
										
										
											2022-07-16 00:45:23 +08:00
										 |  |  |             data: Bytes::from(frame.data.to_vec()),
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |             key: frame.key,
 | 
					
						
							|  |  |  |             pts: frame.pts,
 | 
					
						
							|  |  |  |             ..Default::default()
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fn convert_quality(quality: Quality) -> (u32, u32, u32) {
 | 
					
						
							|  |  |  |         match quality {
 | 
					
						
							|  |  |  |             Quality::Best => (6, 45, 150),
 | 
					
						
							|  |  |  |             Quality::Balanced => (12, 56, 100 * 2 / 3),
 | 
					
						
							|  |  |  |             Quality::Low => (18, 56, 50),
 | 
					
						
							|  |  |  |             Quality::Custom(b) => {
 | 
					
						
							|  |  |  |                 let (q_min, q_max) = Self::calc_q_values(b);
 | 
					
						
							|  |  |  |                 (q_min, q_max, b)
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     fn calc_q_values(b: u32) -> (u32, u32) {
 | 
					
						
							|  |  |  |         let b = std::cmp::min(b, 200);
 | 
					
						
							|  |  |  |         let q_min1: i32 = 36;
 | 
					
						
							| 
									
										
										
										
											2023-07-27 18:29:11 +08:00
										 |  |  |         let q_min2 = 0;
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  |         let q_max1 = 56;
 | 
					
						
							|  |  |  |         let q_max2 = 37;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let t = b as f32 / 200.0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let q_min: u32 = ((1.0 - t) * q_min1 as f32 + t * q_min2 as f32).round() as u32;
 | 
					
						
							|  |  |  |         let q_max = ((1.0 - t) * q_max1 as f32 + t * q_max2 as f32).round() as u32;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         (q_min, q_max)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl Drop for VpxEncoder {
 | 
					
						
							|  |  |  |     fn drop(&mut self) {
 | 
					
						
							|  |  |  |         unsafe {
 | 
					
						
							|  |  |  |             let result = vpx_codec_destroy(&mut self.ctx);
 | 
					
						
							|  |  |  |             if result != VPX_CODEC_OK {
 | 
					
						
							|  |  |  |                 panic!("failed to destroy vpx codec");
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Clone, Copy, Debug)]
 | 
					
						
							|  |  |  | pub struct EncodeFrame<'a> {
 | 
					
						
							|  |  |  |     /// Compressed data.
 | 
					
						
							|  |  |  |     pub data: &'a [u8],
 | 
					
						
							|  |  |  |     /// Whether the frame is a keyframe.
 | 
					
						
							|  |  |  |     pub key: bool,
 | 
					
						
							|  |  |  |     /// Presentation timestamp (in timebase units).
 | 
					
						
							|  |  |  |     pub pts: i64,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Clone, Copy, Debug)]
 | 
					
						
							|  |  |  | pub struct VpxEncoderConfig {
 | 
					
						
							|  |  |  |     /// The width (in pixels).
 | 
					
						
							|  |  |  |     pub width: c_uint,
 | 
					
						
							|  |  |  |     /// The height (in pixels).
 | 
					
						
							|  |  |  |     pub height: c_uint,
 | 
					
						
							| 
									
										
										
										
											2023-07-19 13:11:24 +08:00
										 |  |  |     /// The image quality
 | 
					
						
							|  |  |  |     pub quality: Quality,
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |     /// The codec
 | 
					
						
							|  |  |  |     pub codec: VpxVideoCodecId,
 | 
					
						
							| 
									
										
										
										
											2023-08-07 21:32:36 +08:00
										 |  |  |     /// keyframe interval
 | 
					
						
							|  |  |  |     pub keyframe_interval: Option<usize>,
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Clone, Copy, Debug)]
 | 
					
						
							|  |  |  | pub struct VpxDecoderConfig {
 | 
					
						
							|  |  |  |     pub codec: VpxVideoCodecId,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct EncodeFrames<'a> {
 | 
					
						
							|  |  |  |     ctx: &'a mut vpx_codec_ctx_t,
 | 
					
						
							|  |  |  |     iter: vpx_codec_iter_t,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<'a> Iterator for EncodeFrames<'a> {
 | 
					
						
							|  |  |  |     type Item = EncodeFrame<'a>;
 | 
					
						
							|  |  |  |     fn next(&mut self) -> Option<Self::Item> {
 | 
					
						
							|  |  |  |         loop {
 | 
					
						
							|  |  |  |             unsafe {
 | 
					
						
							|  |  |  |                 let pkt = vpx_codec_get_cx_data(self.ctx, &mut self.iter);
 | 
					
						
							|  |  |  |                 if pkt.is_null() {
 | 
					
						
							|  |  |  |                     return None;
 | 
					
						
							|  |  |  |                 } else if (*pkt).kind == vpx_codec_cx_pkt_kind::VPX_CODEC_CX_FRAME_PKT {
 | 
					
						
							|  |  |  |                     let f = &(*pkt).data.frame;
 | 
					
						
							|  |  |  |                     return Some(Self::Item {
 | 
					
						
							|  |  |  |                         data: slice::from_raw_parts(f.buf as _, f.sz as _),
 | 
					
						
							|  |  |  |                         key: (f.flags & VPX_FRAME_IS_KEY) != 0,
 | 
					
						
							|  |  |  |                         pts: f.pts,
 | 
					
						
							|  |  |  |                     });
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     // Ignore the packet.
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl VpxDecoder {
 | 
					
						
							|  |  |  |     /// Create a new decoder
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// # Errors
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// The function may fail if the underlying libvpx does not provide
 | 
					
						
							|  |  |  |     /// the VP9 decoder.
 | 
					
						
							|  |  |  |     pub fn new(config: VpxDecoderConfig) -> Result<Self> {
 | 
					
						
							|  |  |  |         // This is sound because `vpx_codec_ctx` is a repr(C) struct without any field that can
 | 
					
						
							|  |  |  |         // cause UB if uninitialized.
 | 
					
						
							| 
									
										
										
										
											2023-03-31 16:10:52 +08:00
										 |  |  |         let i = match config.codec {
 | 
					
						
							|  |  |  |             VpxVideoCodecId::VP8 => call_vpx_ptr!(vpx_codec_vp8_dx()),
 | 
					
						
							|  |  |  |             VpxVideoCodecId::VP9 => call_vpx_ptr!(vpx_codec_vp9_dx()),
 | 
					
						
							|  |  |  |         };
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |         let mut ctx = Default::default();
 | 
					
						
							|  |  |  |         let cfg = vpx_codec_dec_cfg_t {
 | 
					
						
							| 
									
										
										
										
											2023-07-20 21:16:38 +08:00
										 |  |  |             threads: codec_thread_num() as _,
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |             w: 0,
 | 
					
						
							|  |  |  |             h: 0,
 | 
					
						
							|  |  |  |         };
 | 
					
						
							|  |  |  |         /*
 | 
					
						
							|  |  |  |         unsafe {
 | 
					
						
							|  |  |  |             println!("{}", vpx_codec_get_caps(i));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         */
 | 
					
						
							|  |  |  |         call_vpx!(vpx_codec_dec_init_ver(
 | 
					
						
							|  |  |  |             &mut ctx,
 | 
					
						
							|  |  |  |             i,
 | 
					
						
							|  |  |  |             &cfg,
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             VPX_DECODER_ABI_VERSION as _,
 | 
					
						
							|  |  |  |         ));
 | 
					
						
							|  |  |  |         Ok(Self { ctx })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Feed some compressed data to the encoder
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// The `data` slice is sent to the decoder
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// It matches a call to `vpx_codec_decode`.
 | 
					
						
							|  |  |  |     pub fn decode(&mut self, data: &[u8]) -> Result<DecodeFrames> {
 | 
					
						
							|  |  |  |         call_vpx!(vpx_codec_decode(
 | 
					
						
							|  |  |  |             &mut self.ctx,
 | 
					
						
							|  |  |  |             data.as_ptr(),
 | 
					
						
							|  |  |  |             data.len() as _,
 | 
					
						
							|  |  |  |             ptr::null_mut(),
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |         ));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Ok(DecodeFrames {
 | 
					
						
							|  |  |  |             ctx: &mut self.ctx,
 | 
					
						
							|  |  |  |             iter: ptr::null(),
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Notify the decoder to return any pending frame
 | 
					
						
							|  |  |  |     pub fn flush(&mut self) -> Result<DecodeFrames> {
 | 
					
						
							|  |  |  |         call_vpx!(vpx_codec_decode(
 | 
					
						
							|  |  |  |             &mut self.ctx,
 | 
					
						
							|  |  |  |             ptr::null(),
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             ptr::null_mut(),
 | 
					
						
							|  |  |  |             0
 | 
					
						
							|  |  |  |         ));
 | 
					
						
							|  |  |  |         Ok(DecodeFrames {
 | 
					
						
							|  |  |  |             ctx: &mut self.ctx,
 | 
					
						
							|  |  |  |             iter: ptr::null(),
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl Drop for VpxDecoder {
 | 
					
						
							|  |  |  |     fn drop(&mut self) {
 | 
					
						
							|  |  |  |         unsafe {
 | 
					
						
							|  |  |  |             let result = vpx_codec_destroy(&mut self.ctx);
 | 
					
						
							|  |  |  |             if result != VPX_CODEC_OK {
 | 
					
						
							|  |  |  |                 panic!("failed to destroy vpx codec");
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct DecodeFrames<'a> {
 | 
					
						
							|  |  |  |     ctx: &'a mut vpx_codec_ctx_t,
 | 
					
						
							|  |  |  |     iter: vpx_codec_iter_t,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<'a> Iterator for DecodeFrames<'a> {
 | 
					
						
							|  |  |  |     type Item = Image;
 | 
					
						
							|  |  |  |     fn next(&mut self) -> Option<Self::Item> {
 | 
					
						
							|  |  |  |         let img = unsafe { vpx_codec_get_frame(self.ctx, &mut self.iter) };
 | 
					
						
							|  |  |  |         if img.is_null() {
 | 
					
						
							|  |  |  |             return None;
 | 
					
						
							|  |  |  |         } else {
 | 
					
						
							|  |  |  |             return Some(Image(img));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://chromium.googlesource.com/webm/libvpx/+/bali/vpx/src/vpx_image.c
 | 
					
						
							|  |  |  | pub struct Image(*mut vpx_image_t);
 | 
					
						
							|  |  |  | impl Image {
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     pub fn new() -> Self {
 | 
					
						
							|  |  |  |         Self(std::ptr::null_mut())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     pub fn is_null(&self) -> bool {
 | 
					
						
							|  |  |  |         self.0.is_null()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     pub fn format(&self) -> vpx_img_fmt_t {
 | 
					
						
							|  |  |  |         // VPX_IMG_FMT_I420
 | 
					
						
							|  |  |  |         self.inner().fmt
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     pub fn inner(&self) -> &vpx_image_t {
 | 
					
						
							|  |  |  |         unsafe { &*self.0 }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  | }
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  | impl GoogleImage for Image {
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |     #[inline]
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  |     fn width(&self) -> usize {
 | 
					
						
							|  |  |  |         self.inner().d_w as _
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-28 11:44:52 +08:00
										 |  |  |     #[inline]
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  |     fn height(&self) -> usize {
 | 
					
						
							|  |  |  |         self.inner().d_h as _
 | 
					
						
							| 
									
										
										
										
											2023-04-28 11:44:52 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  |     #[inline]
 | 
					
						
							|  |  |  |     fn stride(&self) -> Vec<i32> {
 | 
					
						
							|  |  |  |         self.inner().stride.iter().map(|x| *x as i32).collect()
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							| 
									
										
										
										
											2023-05-10 09:43:27 +08:00
										 |  |  |     fn planes(&self) -> Vec<*mut u8> {
 | 
					
						
							|  |  |  |         self.inner().planes.iter().map(|p| *p as *mut u8).collect()
 | 
					
						
							| 
									
										
										
										
											2022-05-29 18:16:35 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl Drop for Image {
 | 
					
						
							|  |  |  |     fn drop(&mut self) {
 | 
					
						
							|  |  |  |         if !self.0.is_null() {
 | 
					
						
							|  |  |  |             unsafe { vpx_img_free(self.0) };
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsafe impl Send for vpx_codec_ctx_t {}
 |