* option `allow-d3d-render`, default false Add this option because it fails on some machines Signed-off-by: 21pages <sunboeasy@gmail.com> * only add nokhwa to windows and linux dependencies Signed-off-by: 21pages <sunboeasy@gmail.com> --------- Signed-off-by: 21pages <sunboeasy@gmail.com>
424 lines
12 KiB
Rust
424 lines
12 KiB
Rust
use crate::CodecFormat;
|
|
#[cfg(feature = "hwcodec")]
|
|
use hbb_common::anyhow::anyhow;
|
|
use hbb_common::{
|
|
bail, chrono, log,
|
|
message_proto::{message, video_frame, EncodedVideoFrame, Message},
|
|
ResultType,
|
|
};
|
|
#[cfg(feature = "hwcodec")]
|
|
use hwcodec::mux::{MuxContext, Muxer};
|
|
use std::{
|
|
fs::{File, OpenOptions},
|
|
io,
|
|
ops::{Deref, DerefMut},
|
|
path::PathBuf,
|
|
sync::mpsc::Sender,
|
|
time::Instant,
|
|
};
|
|
use webm::mux::{self, Segment, Track, VideoTrack, Writer};
|
|
|
|
const MIN_SECS: u64 = 1;
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct RecorderContext {
|
|
pub server: bool,
|
|
pub id: String,
|
|
pub dir: String,
|
|
pub display_idx: usize,
|
|
pub camera: bool,
|
|
pub tx: Option<Sender<RecordState>>,
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct RecorderContext2 {
|
|
pub filename: String,
|
|
pub width: usize,
|
|
pub height: usize,
|
|
pub format: CodecFormat,
|
|
}
|
|
|
|
impl RecorderContext2 {
|
|
pub fn set_filename(&mut self, ctx: &RecorderContext) -> ResultType<()> {
|
|
if !PathBuf::from(&ctx.dir).exists() {
|
|
std::fs::create_dir_all(&ctx.dir)?;
|
|
}
|
|
let file = if ctx.server { "incoming" } else { "outgoing" }.to_string()
|
|
+ "_"
|
|
+ &ctx.id.clone()
|
|
+ &chrono::Local::now().format("_%Y%m%d%H%M%S%3f_").to_string()
|
|
+ &format!(
|
|
"{}{}_",
|
|
if ctx.camera { "camera" } else { "display" },
|
|
ctx.display_idx
|
|
)
|
|
+ &self.format.to_string().to_lowercase()
|
|
+ if self.format == CodecFormat::VP9
|
|
|| self.format == CodecFormat::VP8
|
|
|| self.format == CodecFormat::AV1
|
|
{
|
|
".webm"
|
|
} else {
|
|
".mp4"
|
|
};
|
|
self.filename = PathBuf::from(&ctx.dir)
|
|
.join(file)
|
|
.to_string_lossy()
|
|
.to_string();
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
unsafe impl Send for Recorder {}
|
|
unsafe impl Sync for Recorder {}
|
|
|
|
pub trait RecorderApi {
|
|
fn new(ctx: RecorderContext, ctx2: RecorderContext2) -> ResultType<Self>
|
|
where
|
|
Self: Sized;
|
|
fn write_video(&mut self, frame: &EncodedVideoFrame) -> bool;
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub enum RecordState {
|
|
NewFile(String),
|
|
NewFrame,
|
|
WriteTail,
|
|
RemoveFile,
|
|
}
|
|
|
|
pub struct Recorder {
|
|
pub inner: Option<Box<dyn RecorderApi>>,
|
|
ctx: RecorderContext,
|
|
ctx2: Option<RecorderContext2>,
|
|
pts: Option<i64>,
|
|
check_failed: bool,
|
|
}
|
|
|
|
impl Deref for Recorder {
|
|
type Target = Option<Box<dyn RecorderApi>>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.inner
|
|
}
|
|
}
|
|
|
|
impl DerefMut for Recorder {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.inner
|
|
}
|
|
}
|
|
|
|
impl Recorder {
|
|
pub fn new(ctx: RecorderContext) -> ResultType<Self> {
|
|
Ok(Self {
|
|
inner: None,
|
|
ctx,
|
|
ctx2: None,
|
|
pts: None,
|
|
check_failed: false,
|
|
})
|
|
}
|
|
|
|
fn check(&mut self, w: usize, h: usize, format: CodecFormat) -> ResultType<()> {
|
|
match self.ctx2 {
|
|
Some(ref ctx2) => {
|
|
if ctx2.width != w || ctx2.height != h || ctx2.format != format {
|
|
let mut ctx2 = RecorderContext2 {
|
|
width: w,
|
|
height: h,
|
|
format,
|
|
filename: Default::default(),
|
|
};
|
|
ctx2.set_filename(&self.ctx)?;
|
|
self.ctx2 = Some(ctx2);
|
|
self.inner = None;
|
|
}
|
|
}
|
|
None => {
|
|
let mut ctx2 = RecorderContext2 {
|
|
width: w,
|
|
height: h,
|
|
format,
|
|
filename: Default::default(),
|
|
};
|
|
ctx2.set_filename(&self.ctx)?;
|
|
self.ctx2 = Some(ctx2);
|
|
self.inner = None;
|
|
}
|
|
}
|
|
let Some(ctx2) = &self.ctx2 else {
|
|
bail!("ctx2 is None");
|
|
};
|
|
if self.inner.is_none() {
|
|
self.inner = match format {
|
|
CodecFormat::VP8 | CodecFormat::VP9 | CodecFormat::AV1 => Some(Box::new(
|
|
WebmRecorder::new(self.ctx.clone(), (*ctx2).clone())?,
|
|
)),
|
|
#[cfg(feature = "hwcodec")]
|
|
_ => Some(Box::new(HwRecorder::new(
|
|
self.ctx.clone(),
|
|
(*ctx2).clone(),
|
|
)?)),
|
|
#[cfg(not(feature = "hwcodec"))]
|
|
_ => bail!("unsupported codec type"),
|
|
};
|
|
// pts is None when new inner is created
|
|
self.pts = None;
|
|
self.send_state(RecordState::NewFile(ctx2.filename.clone()));
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
pub fn write_message(&mut self, msg: &Message, w: usize, h: usize) {
|
|
if let Some(message::Union::VideoFrame(vf)) = &msg.union {
|
|
if let Some(frame) = &vf.union {
|
|
self.write_frame(frame, w, h).ok();
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn write_frame(
|
|
&mut self,
|
|
frame: &video_frame::Union,
|
|
w: usize,
|
|
h: usize,
|
|
) -> ResultType<()> {
|
|
if self.check_failed {
|
|
bail!("check failed");
|
|
}
|
|
let format = CodecFormat::from(frame);
|
|
if format == CodecFormat::Unknown {
|
|
bail!("unsupported frame type");
|
|
}
|
|
let res = self.check(w, h, format);
|
|
if res.is_err() {
|
|
self.check_failed = true;
|
|
log::error!("check failed: {:?}", res);
|
|
res?;
|
|
}
|
|
match frame {
|
|
video_frame::Union::Vp8s(vp8s) => {
|
|
for f in vp8s.frames.iter() {
|
|
self.check_pts(f.pts, f.key, w, h, format)?;
|
|
self.as_mut().map(|x| x.write_video(f));
|
|
}
|
|
}
|
|
video_frame::Union::Vp9s(vp9s) => {
|
|
for f in vp9s.frames.iter() {
|
|
self.check_pts(f.pts, f.key, w, h, format)?;
|
|
self.as_mut().map(|x| x.write_video(f));
|
|
}
|
|
}
|
|
video_frame::Union::Av1s(av1s) => {
|
|
for f in av1s.frames.iter() {
|
|
self.check_pts(f.pts, f.key, w, h, format)?;
|
|
self.as_mut().map(|x| x.write_video(f));
|
|
}
|
|
}
|
|
#[cfg(feature = "hwcodec")]
|
|
video_frame::Union::H264s(h264s) => {
|
|
for f in h264s.frames.iter() {
|
|
self.check_pts(f.pts, f.key, w, h, format)?;
|
|
self.as_mut().map(|x| x.write_video(f));
|
|
}
|
|
}
|
|
#[cfg(feature = "hwcodec")]
|
|
video_frame::Union::H265s(h265s) => {
|
|
for f in h265s.frames.iter() {
|
|
self.check_pts(f.pts, f.key, w, h, format)?;
|
|
self.as_mut().map(|x| x.write_video(f));
|
|
}
|
|
}
|
|
_ => bail!("unsupported frame type"),
|
|
}
|
|
self.send_state(RecordState::NewFrame);
|
|
Ok(())
|
|
}
|
|
|
|
fn check_pts(
|
|
&mut self,
|
|
pts: i64,
|
|
key: bool,
|
|
w: usize,
|
|
h: usize,
|
|
format: CodecFormat,
|
|
) -> ResultType<()> {
|
|
// https://stackoverflow.com/questions/76379101/how-to-create-one-playable-webm-file-from-two-different-video-tracks-with-same-c
|
|
if self.pts.is_none() && !key {
|
|
bail!("first frame is not key frame");
|
|
}
|
|
let old_pts = self.pts;
|
|
self.pts = Some(pts);
|
|
if old_pts.clone().unwrap_or_default() > pts {
|
|
log::info!("pts {:?} -> {}, change record filename", old_pts, pts);
|
|
self.inner = None;
|
|
self.ctx2 = None;
|
|
let res = self.check(w, h, format);
|
|
if res.is_err() {
|
|
self.check_failed = true;
|
|
log::error!("check failed: {:?}", res);
|
|
res?;
|
|
}
|
|
self.pts = Some(pts);
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn send_state(&self, state: RecordState) {
|
|
self.ctx.tx.as_ref().map(|tx| tx.send(state));
|
|
}
|
|
}
|
|
|
|
struct WebmRecorder {
|
|
vt: VideoTrack,
|
|
webm: Option<Segment<Writer<File>>>,
|
|
ctx: RecorderContext,
|
|
ctx2: RecorderContext2,
|
|
key: bool,
|
|
written: bool,
|
|
start: Instant,
|
|
}
|
|
|
|
impl RecorderApi for WebmRecorder {
|
|
fn new(ctx: RecorderContext, ctx2: RecorderContext2) -> ResultType<Self> {
|
|
let out = match {
|
|
OpenOptions::new()
|
|
.write(true)
|
|
.create_new(true)
|
|
.open(&ctx2.filename)
|
|
} {
|
|
Ok(file) => file,
|
|
Err(ref e) if e.kind() == io::ErrorKind::AlreadyExists => File::create(&ctx2.filename)?,
|
|
Err(e) => return Err(e.into()),
|
|
};
|
|
let mut webm = match mux::Segment::new(mux::Writer::new(out)) {
|
|
Some(v) => v,
|
|
None => bail!("Failed to create webm mux"),
|
|
};
|
|
let vt = webm.add_video_track(
|
|
ctx2.width as _,
|
|
ctx2.height as _,
|
|
None,
|
|
if ctx2.format == CodecFormat::VP9 {
|
|
mux::VideoCodecId::VP9
|
|
} else if ctx2.format == CodecFormat::VP8 {
|
|
mux::VideoCodecId::VP8
|
|
} else {
|
|
mux::VideoCodecId::AV1
|
|
},
|
|
);
|
|
if ctx2.format == CodecFormat::AV1 {
|
|
// [129, 8, 12, 0] in 3.6.0, but zero works
|
|
let codec_private = vec![0, 0, 0, 0];
|
|
if !webm.set_codec_private(vt.track_number(), &codec_private) {
|
|
bail!("Failed to set codec private");
|
|
}
|
|
}
|
|
Ok(WebmRecorder {
|
|
vt,
|
|
webm: Some(webm),
|
|
ctx,
|
|
ctx2,
|
|
key: false,
|
|
written: false,
|
|
start: Instant::now(),
|
|
})
|
|
}
|
|
|
|
fn write_video(&mut self, frame: &EncodedVideoFrame) -> bool {
|
|
if frame.key {
|
|
self.key = true;
|
|
}
|
|
if self.key {
|
|
let ok = self
|
|
.vt
|
|
.add_frame(&frame.data, frame.pts as u64 * 1_000_000, frame.key);
|
|
if ok {
|
|
self.written = true;
|
|
}
|
|
ok
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Drop for WebmRecorder {
|
|
fn drop(&mut self) {
|
|
let _ = std::mem::replace(&mut self.webm, None).map_or(false, |webm| webm.finalize(None));
|
|
let mut state = RecordState::WriteTail;
|
|
if !self.written || self.start.elapsed().as_secs() < MIN_SECS {
|
|
std::fs::remove_file(&self.ctx2.filename).ok();
|
|
state = RecordState::RemoveFile;
|
|
}
|
|
self.ctx.tx.as_ref().map(|tx| tx.send(state));
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "hwcodec")]
|
|
struct HwRecorder {
|
|
muxer: Option<Muxer>,
|
|
ctx: RecorderContext,
|
|
ctx2: RecorderContext2,
|
|
written: bool,
|
|
key: bool,
|
|
start: Instant,
|
|
}
|
|
|
|
#[cfg(feature = "hwcodec")]
|
|
impl RecorderApi for HwRecorder {
|
|
fn new(ctx: RecorderContext, ctx2: RecorderContext2) -> ResultType<Self> {
|
|
let muxer = Muxer::new(MuxContext {
|
|
filename: ctx2.filename.clone(),
|
|
width: ctx2.width,
|
|
height: ctx2.height,
|
|
is265: ctx2.format == CodecFormat::H265,
|
|
framerate: crate::hwcodec::DEFAULT_FPS as _,
|
|
})
|
|
.map_err(|_| anyhow!("Failed to create hardware muxer"))?;
|
|
Ok(HwRecorder {
|
|
muxer: Some(muxer),
|
|
ctx,
|
|
ctx2,
|
|
written: false,
|
|
key: false,
|
|
start: Instant::now(),
|
|
})
|
|
}
|
|
|
|
fn write_video(&mut self, frame: &EncodedVideoFrame) -> bool {
|
|
if frame.key {
|
|
self.key = true;
|
|
}
|
|
if self.key {
|
|
let ok = self
|
|
.muxer
|
|
.as_mut()
|
|
.map(|m| m.write_video(&frame.data, frame.key).is_ok())
|
|
.unwrap_or_default();
|
|
if ok {
|
|
self.written = true;
|
|
}
|
|
ok
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "hwcodec")]
|
|
impl Drop for HwRecorder {
|
|
fn drop(&mut self) {
|
|
self.muxer.as_mut().map(|m| m.write_tail().ok());
|
|
let mut state = RecordState::WriteTail;
|
|
if !self.written || self.start.elapsed().as_secs() < MIN_SECS {
|
|
// The process cannot access the file because it is being used by another process
|
|
self.muxer = None;
|
|
std::fs::remove_file(&self.ctx2.filename).ok();
|
|
state = RecordState::RemoveFile;
|
|
}
|
|
self.ctx.tx.as_ref().map(|tx| tx.send(state));
|
|
}
|
|
}
|