From 21212be18a44416d948f19e7e8d966ce5de92bfc Mon Sep 17 00:00:00 2001 From: Scott Lamb Date: Sat, 25 Feb 2017 18:54:52 -0800 Subject: [PATCH] use associated types for Slice This is more readable; in particular, it avoids the need for awkward the PhantomData in the Slices struct. --- src/mp4.rs | 8 +++++--- src/slices.rs | 26 +++++++++++++------------- 2 files changed, 18 insertions(+), 16 deletions(-) diff --git a/src/mp4.rs b/src/mp4.rs index 7f58c5c..ab5960c 100644 --- a/src/mp4.rs +++ b/src/mp4.rs @@ -421,7 +421,7 @@ pub struct FileBuilder { /// This is separated out from the rest so that it can be borrowed in a loop over /// `FileBuilder::segments`; otherwise this would cause a double-self-borrow. struct BodyState { - slices: Slices, + slices: Slices, /// `self.buf[unflushed_buf_pos .. self.buf.len()]` holds bytes that should be /// appended to `slices` before any other slice. See `flush_buf()`. @@ -477,7 +477,9 @@ impl Slice { fn p(&self) -> usize { (self.0 >> 44) as usize } } -impl slices::Slice for Slice { +impl slices::Slice for Slice { + type Ctx = File; + fn end(&self) -> u64 { return self.0 & 0xFF_FF_FF_FF_FF } fn write_to(&self, f: &File, r: Range, l: u64, out: &mut io::Write) -> Result<(), Error> { @@ -1114,7 +1116,7 @@ pub struct File { db: Arc, dir: Arc, segments: Vec, - slices: Slices, + slices: Slices, buf: Vec, video_sample_entries: SmallVec<[Arc; 1]>, initial_sample_byte_pos: u64, diff --git a/src/slices.rs b/src/slices.rs index 2db566a..3ced687 100644 --- a/src/slices.rs +++ b/src/slices.rs @@ -33,12 +33,13 @@ use error::{Error, Result}; use std::fmt; use std::io; -use std::marker::PhantomData; use std::ops::Range; /// Writes a byte range to the given `io::Write` given a context argument; meant for use with /// `Slices`. -pub trait Slice { +pub trait Slice { + type Ctx; + /// The byte position (relative to the start of the `Slices`) beyond the end of this slice. /// Note the starting position (and thus length) are inferred from the previous slice. fn end(&self) -> u64; @@ -46,7 +47,7 @@ pub trait Slice { /// Writes `r` to `out`, as in `http_entity::Entity::write_to`. /// The additional argument `ctx` is as supplied to the `Slices`. /// The additional argument `l` is the length of this slice, as determined by the `Slices`. - fn write_to(&self, ctx: &Ctx, r: Range, l: u64, out: &mut io::Write) -> Result<()>; + fn write_to(&self, ctx: &Self::Ctx, r: Range, l: u64, out: &mut io::Write) -> Result<()>; } /// Calls `f` with an `io::Write` which delegates to `inner` only for the section defined by `r`. @@ -66,7 +67,7 @@ where F: FnMut(&mut Vec) -> Result<()> { /// Helper to serve byte ranges from a body which is broken down into many "slices". /// This is used to implement `.mp4` serving in `mp4::Mp4File` from `mp4::Slice` enums. -pub struct Slices where S: Slice { +pub struct Slices where S: Slice { /// The total byte length of the `Slices`. /// Equivalent to `self.slices.back().map(|s| s.end()).unwrap_or(0)`; kept for convenience and /// to avoid a branch. @@ -74,12 +75,9 @@ pub struct Slices where S: Slice { /// 0 or more slices of this file. slices: Vec, - - /// Marker so that `C` is part of the type. - phantom: PhantomData, } -impl fmt::Debug for Slices where S: fmt::Debug + Slice { +impl fmt::Debug for Slices where S: fmt::Debug + Slice { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{} slices with overall length {}:", self.slices.len(), self.len)?; let mut start = 0; @@ -93,8 +91,8 @@ impl fmt::Debug for Slices where S: fmt::Debug + Slice { } } -impl Slices where S: Slice { - pub fn new() -> Self { Slices{len: 0, slices: Vec::new(), phantom: PhantomData} } +impl Slices where S: Slice { + pub fn new() -> Self { Slices{len: 0, slices: Vec::new()} } /// Reserves space for at least `additional` more slices to be appended. pub fn reserve(&mut self, additional: usize) { @@ -116,7 +114,7 @@ impl Slices where S: Slice { /// Writes `range` to `out`. /// This interface mirrors `http_entity::Entity::write_to`, with the additional `ctx` argument. - pub fn write_to(&self, ctx: &C, range: Range, out: &mut io::Write) -> Result<()> { + pub fn write_to(&self, ctx: &S::Ctx, range: Range, out: &mut io::Write) -> Result<()> { if range.start > range.end || range.end > self.len { return Err(Error{ description: format!("Bad range {:?} for slice of length {}", range, self.len), @@ -174,7 +172,9 @@ mod tests { name: &'static str, } - impl Slice>> for FakeSlice { + impl Slice for FakeSlice { + type Ctx = RefCell>; + fn end(&self) -> u64 { self.end } fn write_to(&self, ctx: &RefCell>, r: Range, _l: u64, _out: &mut Write) @@ -184,7 +184,7 @@ mod tests { } } - pub fn new_slices() -> Slices>> { + pub fn new_slices() -> Slices { let mut s = Slices::new(); s.append(FakeSlice{end: 5, name: "a"}); s.append(FakeSlice{end: 5+13, name: "b"});