mirror of
https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs.git
synced 2025-01-09 18:55:27 +00:00
fmp4mux: Handle GOPs ending after the desired fragment end correctly
Either create further chunks if enough data is queued or simply start the new fragment at a later time if the keyframe is later. Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs/-/merge_requests/1086>
This commit is contained in:
parent
dae1d8b5ef
commit
da7743d2e7
1 changed files with 80 additions and 33 deletions
|
@ -991,27 +991,6 @@ impl FMP4Mux {
|
||||||
// Check if this stream is filled enough now.
|
// Check if this stream is filled enough now.
|
||||||
if let Some(chunk_duration) = settings.chunk_duration {
|
if let Some(chunk_duration) = settings.chunk_duration {
|
||||||
// In chunk mode
|
// In chunk mode
|
||||||
let (gop_idx, gop) = match stream
|
|
||||||
.queued_gops
|
|
||||||
.iter()
|
|
||||||
.enumerate()
|
|
||||||
.find(|(_idx, gop)| gop.final_earliest_pts || all_eos || stream.sinkpad.is_eos())
|
|
||||||
{
|
|
||||||
Some(res) => res,
|
|
||||||
None => {
|
|
||||||
gst::trace!(CAT, obj: stream.sinkpad, "Chunked mode but no GOP with final earliest PTS known yet");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
gst::trace!(
|
|
||||||
CAT,
|
|
||||||
obj: stream.sinkpad,
|
|
||||||
"GOP {gop_idx} start PTS {}, GOP end PTS {} (final {})",
|
|
||||||
gop.start_pts,
|
|
||||||
gop.end_pts,
|
|
||||||
gop.final_end_pts || all_eos || stream.sinkpad.is_eos(),
|
|
||||||
);
|
|
||||||
gst::trace!(
|
gst::trace!(
|
||||||
CAT,
|
CAT,
|
||||||
obj: stream.sinkpad,
|
obj: stream.sinkpad,
|
||||||
|
@ -1033,10 +1012,53 @@ impl FMP4Mux {
|
||||||
|
|
||||||
// First check if the next split should be the end of a fragment or the end of a chunk.
|
// First check if the next split should be the end of a fragment or the end of a chunk.
|
||||||
// If both are the same then a fragment split has preference.
|
// If both are the same then a fragment split has preference.
|
||||||
if fragment_end_pts <= chunk_end_pts && gop.start_pts >= fragment_end_pts {
|
if fragment_end_pts <= chunk_end_pts {
|
||||||
gst::debug!(CAT, obj: stream.sinkpad, "Stream queued enough data for finishing this fragment");
|
// We can only finish a fragment if a full GOP with final end PTS is queued and it
|
||||||
stream.fragment_filled = true;
|
// ends at or after the fragment end PTS.
|
||||||
} else if chunk_end_pts < fragment_end_pts {
|
if let Some((gop_idx, gop)) = stream
|
||||||
|
.queued_gops
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.find(|(_idx, gop)| gop.final_end_pts || all_eos || stream.sinkpad.is_eos())
|
||||||
|
{
|
||||||
|
gst::trace!(
|
||||||
|
CAT,
|
||||||
|
obj: stream.sinkpad,
|
||||||
|
"GOP {gop_idx} start PTS {}, GOP end PTS {}",
|
||||||
|
gop.start_pts,
|
||||||
|
gop.end_pts,
|
||||||
|
);
|
||||||
|
if gop.end_pts >= fragment_end_pts {
|
||||||
|
gst::debug!(CAT, obj: stream.sinkpad, "Stream queued enough data for finishing this fragment");
|
||||||
|
stream.fragment_filled = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !stream.fragment_filled {
|
||||||
|
let (gop_idx, gop) = match stream.queued_gops.iter().enumerate().find(
|
||||||
|
|(_idx, gop)| gop.final_earliest_pts || all_eos || stream.sinkpad.is_eos(),
|
||||||
|
) {
|
||||||
|
Some(res) => res,
|
||||||
|
None => {
|
||||||
|
gst::trace!(
|
||||||
|
CAT,
|
||||||
|
obj: stream.sinkpad,
|
||||||
|
"Chunked mode and want to finish fragment but no GOP with final end PTS known yet",
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
gst::trace!(
|
||||||
|
CAT,
|
||||||
|
obj: stream.sinkpad,
|
||||||
|
"GOP {gop_idx} start PTS {}, GOP end PTS {} (final {})",
|
||||||
|
gop.start_pts,
|
||||||
|
gop.end_pts,
|
||||||
|
gop.final_end_pts || all_eos || stream.sinkpad.is_eos(),
|
||||||
|
);
|
||||||
let last_pts = gop.buffers.last().map(|b| b.pts);
|
let last_pts = gop.buffers.last().map(|b| b.pts);
|
||||||
|
|
||||||
if gop.end_pts >= chunk_end_pts
|
if gop.end_pts >= chunk_end_pts
|
||||||
|
@ -1294,8 +1316,15 @@ impl FMP4Mux {
|
||||||
gop.final_end_pts || all_eos || stream.sinkpad.is_eos()
|
gop.final_end_pts || all_eos || stream.sinkpad.is_eos()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// If we have a final GOP then include it as long as it's either
|
||||||
|
// - ending before the dequeue end PTS
|
||||||
|
// - no GOPs were dequeued yet and this is the first stream
|
||||||
|
//
|
||||||
|
// The second case would happen if no GOP ends between the last chunk of the
|
||||||
|
// fragment and the fragment duration.
|
||||||
if (gop.final_end_pts || all_eos || stream.sinkpad.is_eos())
|
if (gop.final_end_pts || all_eos || stream.sinkpad.is_eos())
|
||||||
&& gop.end_pts <= dequeue_end_pts
|
&& (gop.end_pts <= dequeue_end_pts
|
||||||
|
|| (gops.is_empty() && chunk_end_pts.is_none()))
|
||||||
{
|
{
|
||||||
gst::trace!(
|
gst::trace!(
|
||||||
CAT,
|
CAT,
|
||||||
|
@ -1309,7 +1338,16 @@ impl FMP4Mux {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
gst::error!(CAT, obj: stream.sinkpad, "Don't have a full GOP at the end of a fragment");
|
// Otherwise if this is the first stream and no full GOP is queued then we need
|
||||||
|
// to wait for more data.
|
||||||
|
//
|
||||||
|
// If this is not the first stream then take an incomplete GOP.
|
||||||
|
if chunk_end_pts.is_none() {
|
||||||
|
gst::info!(CAT, obj: stream.sinkpad, "Don't have a full GOP at the end of a fragment");
|
||||||
|
return Err(gst_base::AGGREGATOR_FLOW_NEED_DATA);
|
||||||
|
} else {
|
||||||
|
gst::info!(CAT, obj: stream.sinkpad, "Including incomplete GOP");
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
gst::trace!(
|
gst::trace!(
|
||||||
CAT,
|
CAT,
|
||||||
|
@ -1772,7 +1810,7 @@ impl FMP4Mux {
|
||||||
// This is handled below generally if nothing was dequeued
|
// This is handled below generally if nothing was dequeued
|
||||||
} else {
|
} else {
|
||||||
if settings.chunk_duration.is_some() {
|
if settings.chunk_duration.is_some() {
|
||||||
gst::warning!(
|
gst::debug!(
|
||||||
CAT,
|
CAT,
|
||||||
obj: stream.sinkpad,
|
obj: stream.sinkpad,
|
||||||
"Don't have anything to drain for the first stream on timeout in a live pipeline",
|
"Don't have anything to drain for the first stream on timeout in a live pipeline",
|
||||||
|
@ -2146,6 +2184,19 @@ impl FMP4Mux {
|
||||||
let min_earliest_pts = min_earliest_pts.unwrap();
|
let min_earliest_pts = min_earliest_pts.unwrap();
|
||||||
let chunk_end_pts = chunk_end_pts.unwrap();
|
let chunk_end_pts = chunk_end_pts.unwrap();
|
||||||
|
|
||||||
|
gst::debug!(
|
||||||
|
CAT,
|
||||||
|
imp: self,
|
||||||
|
concat!(
|
||||||
|
"Draining chunk (fragment start: {} fragment end: {}) ",
|
||||||
|
"from PTS {} to {}"
|
||||||
|
),
|
||||||
|
fragment_start,
|
||||||
|
fragment_filled,
|
||||||
|
min_earliest_pts,
|
||||||
|
chunk_end_pts,
|
||||||
|
);
|
||||||
|
|
||||||
let mut fmp4_header = None;
|
let mut fmp4_header = None;
|
||||||
if !state.sent_headers {
|
if !state.sent_headers {
|
||||||
let mut buffer = state.stream_header.as_ref().unwrap().copy();
|
let mut buffer = state.stream_header.as_ref().unwrap().copy();
|
||||||
|
@ -2317,11 +2368,7 @@ impl FMP4Mux {
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
if err == gst_base::AGGREGATOR_FLOW_NEED_DATA {
|
if err == gst_base::AGGREGATOR_FLOW_NEED_DATA {
|
||||||
assert!(!all_eos);
|
assert!(!all_eos);
|
||||||
gst::element_imp_warning!(
|
gst::debug!(CAT, imp: self, "Need more data");
|
||||||
self,
|
|
||||||
gst::StreamError::Format,
|
|
||||||
["Longer GOPs than fragment duration"]
|
|
||||||
);
|
|
||||||
state.timeout_delay += 1.seconds();
|
state.timeout_delay += 1.seconds();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue