aboutsummaryrefslogtreecommitdiff
path: root/remuxer/src/lib.rs
diff options
context:
space:
mode:
authormetamuffin <metamuffin@disroot.org>2023-01-28 10:55:10 +0100
committermetamuffin <metamuffin@disroot.org>2023-01-28 10:55:10 +0100
commit0427a45ce8fa4762b087eeaf7e24f00678ceb48b (patch)
tree3f55f4bf994dbff980d930f51ac201beb22aa5e3 /remuxer/src/lib.rs
parenta742f7dbd8bda0bf23a6d5273e5dd2f83b9d4c9f (diff)
downloadjellything-0427a45ce8fa4762b087eeaf7e24f00678ceb48b.tar
jellything-0427a45ce8fa4762b087eeaf7e24f00678ceb48b.tar.bz2
jellything-0427a45ce8fa4762b087eeaf7e24f00678ceb48b.tar.zst
seeking logic
Diffstat (limited to 'remuxer/src/lib.rs')
-rw-r--r--remuxer/src/lib.rs181
1 files changed, 66 insertions, 115 deletions
diff --git a/remuxer/src/lib.rs b/remuxer/src/lib.rs
index 3ce6b78..ead0051 100644
--- a/remuxer/src/lib.rs
+++ b/remuxer/src/lib.rs
@@ -4,18 +4,29 @@
Copyright (C) 2023 metamuffin <metamuffin.org>
*/
pub mod import;
+pub mod segment_extractor;
+pub mod trim_writer;
-use anyhow::{anyhow, Context, Result};
+use crate::{
+ segment_extractor::{AbsoluteBlock, SegmentExtractIter},
+ trim_writer::TrimWriter,
+};
+use anyhow::{anyhow, Context};
use jellycommon::{BlockIndex, ItemInfo, SeekIndex, SourceTrack, SourceTrackKind};
use jellymatroska::{
- block::Block,
read::EbmlReader,
- unflatten::{IterWithPos, Unflat, Unflatten},
- write::EbmlWriter,
+ unflatten::{IterWithPos, Unflatten},
+ write::{vint_length, EbmlWriter},
Master, MatroskaTag,
};
-use log::{debug, info, trace, warn};
-use std::{collections::VecDeque, fs::File, io::Write, path::PathBuf};
+use log::{debug, info, trace};
+use std::{
+ fs::File,
+ io::{Seek, SeekFrom, Write},
+ ops::Range,
+ path::PathBuf,
+ time::Instant,
+};
#[derive(Debug, Clone)]
pub struct RemuxerContext {}
@@ -29,13 +40,14 @@ impl RemuxerContext {
pub fn generate_into(
&self,
writer: impl Write + 'static,
- _offset: usize,
+ range: Range<usize>,
path_base: PathBuf,
iteminfo: ItemInfo,
selection: Vec<usize>,
webm: bool,
) -> anyhow::Result<()> {
info!("remuxing {:?} to have tracks {selection:?}", iteminfo.title);
+ let writer = TrimWriter::new(writer, range.clone());
let mut output = EbmlWriter::new(writer, 0);
struct ReaderC {
@@ -46,6 +58,8 @@ impl RemuxerContext {
temp_index: usize,
}
+ let timing_cp = Instant::now();
+
let mut inputs = selection
.iter()
.enumerate()
@@ -82,6 +96,12 @@ impl RemuxerContext {
})
.collect::<anyhow::Result<Vec<_>>>()?;
+ info!(
+ "(perf) prepare inputs: {}ms",
+ (Instant::now() - timing_cp).as_millis()
+ );
+ let timing_cp = Instant::now();
+
output.write_tag(&MatroskaTag::Ebml(Master::Collected(vec![
MatroskaTag::EbmlVersion(1),
MatroskaTag::EbmlReadVersion(1),
@@ -106,7 +126,6 @@ impl RemuxerContext {
MatroskaTag::WritingApp("jellything".to_string()),
])))?;
output.write_tag(&MatroskaTag::Tags(Master::Collected(vec![])))?;
- // output.write_tag(&MatroskaTag::Cues(Master::Collected(vec![])))?;
let tracks_header = inputs
.iter()
@@ -143,12 +162,11 @@ impl RemuxerContext {
}
inputs[best_index].temp_index += 1;
if best_block.pts > cluster_pts + 2_000 {
- let cluster_content_size = 1 // timestamp tag
- + 1 // timestamp tag size
- + EbmlWriter::vint_length(cluster_pts) // timestamp tag value
+ let cluster_content_size = 1 + 1 // timestamp {tag, size}
+ + vint_length(cluster_pts) // timestamp tag value
+ p;
let cluster_header_size = 4 // tag length
- + EbmlWriter::vint_length(cluster_content_size as u64)// size varint
+ + vint_length(cluster_content_size as u64) // size varint
+ cluster_content_size;
clusters.push(ClusterLayout {
position: gp,
@@ -157,14 +175,14 @@ impl RemuxerContext {
});
cluster_pts = best_block.pts;
- gp += p + cluster_header_size;
+ gp += cluster_header_size;
p = 0;
}
if best_block.pts == u64::MAX {
break;
}
p += 1; // simpleblock tag
- p += EbmlWriter::vint_length(1 + 2 + 1 + best_block.size as u64); // simpleblock size vint
+ p += vint_length(1 + 2 + 1 + best_block.size as u64); // simpleblock size vint
p += 1 + 2 + 1; // block {tracknum, pts_off, flags}
// TODO does not work, if more than 127 tracks are present
p += best_block.size; // block payload
@@ -173,6 +191,11 @@ impl RemuxerContext {
info!("segment layout computed ({} clusters)", clusters.len());
clusters
};
+ info!(
+ "(perf) compute segment layout: {}ms",
+ (Instant::now() - timing_cp).as_millis()
+ );
+ let timing_cp = Instant::now();
output.write_tag(&MatroskaTag::Cues(Master::Collected(
segment_layout
@@ -202,6 +225,19 @@ impl RemuxerContext {
.collect(),
)))?;
+ let segment_start_position = output.position();
+ let mut skip = 0;
+ for cluster in &segment_layout {
+ if (cluster.position + segment_start_position) > range.start {
+ break;
+ }
+ skip += 1;
+ }
+ if skip != 0 {
+ info!("skipping {skip} clusters");
+ output.seek(SeekFrom::Start(segment_layout[skip].position as u64))?;
+ }
+
struct ReaderD<'a> {
_info: SourceTrack,
peek: Option<AbsoluteBlock>,
@@ -218,14 +254,17 @@ impl RemuxerContext {
break;
}
}
- let mut stream = SegmentExtractIter {
- segment: Unflatten::new_with_end(
- &mut i.reader,
+ i.reader
+ .seek(
+ segment_layout[skip].position,
MatroskaTag::Segment(Master::Start),
- ),
- extract: i.info.track_number,
- emission_queue: VecDeque::new(),
- };
+ )
+ .context("seeking in input")?;
+ let mut stream = SegmentExtractIter::new(
+ Unflatten::new_with_end(&mut i.reader, MatroskaTag::Segment(Master::Start)),
+ i.info.track_number,
+ );
+
ks.push(ReaderD {
mapped: i.mapped,
peek: Some(stream.next()?),
@@ -233,9 +272,12 @@ impl RemuxerContext {
_info: i.info.clone(),
});
}
+ info!(
+ "(perf) seek inputs: {}ms",
+ (Instant::now() - timing_cp).as_millis()
+ );
- let segment_start_position = output.position();
- for (cluster_index, cluster) in segment_layout.into_iter().enumerate() {
+ for (cluster_index, cluster) in segment_layout.into_iter().skip(skip).enumerate() {
info!(
"writing cluster {cluster_index} (pts_base={}) with {} blocks",
cluster.timestamp,
@@ -258,7 +300,7 @@ impl RemuxerContext {
block.inner.track = kn.mapped;
block.inner.timestamp_off = (iblock.pts - cluster.timestamp).try_into().unwrap();
- debug!("n={} tso={}", block.inner.track, block.inner.timestamp_off);
+ trace!("n={} tso={}", block.inner.track, block.inner.timestamp_off);
let buf = block.inner.dump();
cluster_blocks.push(MatroskaTag::SimpleBlock(buf))
}
@@ -269,97 +311,6 @@ impl RemuxerContext {
}
}
-struct AbsoluteBlock {
- pts_base: u64,
- inner: Block,
-}
-
-struct SegmentExtractIter<'a> {
- segment: Unflatten<'a>,
- extract: u64,
- emission_queue: VecDeque<AbsoluteBlock>,
-}
-
-impl AbsoluteBlock {
- pub fn pts(&self) -> u64 {
- self.inner.timestamp_off as u64 + self.pts_base
- }
-}
-
-impl SegmentExtractIter<'_> {
- pub fn next(&mut self) -> Result<AbsoluteBlock> {
- loop {
- if let Some(b) = self.emission_queue.pop_front() {
- break Ok(b);
- }
- self.read()?;
- }
- }
-
- pub fn read(&mut self) -> Result<()> {
- let Unflat { children, item } = self.segment.n().ok_or(anyhow!("eof"))??;
- let mut pts_base = 0;
- match item {
- MatroskaTag::SeekHead(_) => {}
- MatroskaTag::Info(_) => {}
- MatroskaTag::Cluster(_) => {
- let mut children = children.unwrap();
- while let Some(Ok(Unflat { children, item })) = children.n() {
- match item {
- MatroskaTag::Crc32(_) => (),
- MatroskaTag::Timestamp(ts) => {
- trace!("read pts={ts}");
- pts_base = ts;
- }
- MatroskaTag::BlockGroup(_) => {
- trace!("group");
- let mut children = children.unwrap();
-
- // let mut duration = None;
- let mut block = None;
-
- while let Some(Ok(Unflat { children: _, item })) = children.n() {
- match item {
- MatroskaTag::Block(buf) => block = Some(buf),
- // MatroskaTag::BlockDuration(v) => duration = Some(v),
- _ => debug!("ignored {item:?}"),
- }
- }
- // TODO duration
- let block = Block::parse(&block.unwrap())?;
- if block.track == self.extract {
- trace!("block: track={} tso={}", block.track, block.timestamp_off);
- self.emission_queue.push_back(AbsoluteBlock {
- pts_base,
- inner: block,
- });
- }
- }
- MatroskaTag::SimpleBlock(buf) => {
- let block = Block::parse(&buf)?;
- if block.track == self.extract {
- trace!("block: track={} tso={}", block.track, block.timestamp_off);
- self.emission_queue.push_back(AbsoluteBlock {
- pts_base,
- inner: block,
- });
- }
- }
- _ => warn!("(rsc) tag ignored: {item:?}"),
- }
- }
- }
- MatroskaTag::Tags(_) => {}
- MatroskaTag::Cues(_) => {}
- MatroskaTag::Chapters(_) => {}
- MatroskaTag::Tracks(_) => {}
- MatroskaTag::Void(_) => {}
- _ => debug!("(rs) tag ignored: {item:?}"),
- }
- Ok(())
- }
-}
-
pub fn track_to_ebml(number: u64, track: &SourceTrack) -> MatroskaTag {
let mut els = vec![
MatroskaTag::TrackNumber(number),