mirror of
https://github.com/alfg/mp4-rust.git
synced 2024-12-31 15:48:39 +00:00
Add ReadBox trait (#7)
This commit is contained in:
parent
e637fb761e
commit
85a65ea596
2 changed files with 411 additions and 379 deletions
784
src/atoms.rs
784
src/atoms.rs
|
@ -1,6 +1,5 @@
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::io::{BufReader, SeekFrom, Seek, Read};
|
use std::io::{BufReader, SeekFrom, Seek, Read};
|
||||||
use std::fs::File;
|
|
||||||
use byteorder::{BigEndian, ReadBytesExt};
|
use byteorder::{BigEndian, ReadBytesExt};
|
||||||
use std::char::{decode_utf16, REPLACEMENT_CHARACTER};
|
use std::char::{decode_utf16, REPLACEMENT_CHARACTER};
|
||||||
use crate::{Error, read_box_header, BoxHeader, HEADER_SIZE};
|
use crate::{Error, read_box_header, BoxHeader, HEADER_SIZE};
|
||||||
|
@ -151,6 +150,10 @@ impl fmt::Display for FourCC {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait ReadBox<T>: Sized {
|
||||||
|
fn read_box(_: T, offset: u64, size: u32) -> Result<Self>;
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug, Default)]
|
||||||
pub struct FtypBox {
|
pub struct FtypBox {
|
||||||
pub major_brand: FourCC,
|
pub major_brand: FourCC,
|
||||||
|
@ -329,268 +332,285 @@ pub struct StsdBox {
|
||||||
pub flags: u32,
|
pub flags: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn parse_ftyp_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<FtypBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for FtypBox {
|
||||||
let major = f.read_u32::<BigEndian>().unwrap();
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
let minor = f.read_u32::<BigEndian>().unwrap();
|
let major = reader.read_u32::<BigEndian>().unwrap();
|
||||||
if size % 4 != 0 {
|
let minor = reader.read_u32::<BigEndian>().unwrap();
|
||||||
return Err(Error::InvalidData("invalid ftyp size"));
|
if size % 4 != 0 {
|
||||||
}
|
return Err(Error::InvalidData("invalid ftyp size"));
|
||||||
let brand_count = (size - 16) / 4; // header + major + minor
|
|
||||||
|
|
||||||
let mut brands = Vec::new();
|
|
||||||
for _ in 0..brand_count {
|
|
||||||
let b = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
brands.push(From::from(b));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(FtypBox {
|
|
||||||
major_brand: From::from(major),
|
|
||||||
minor_version: minor,
|
|
||||||
compatible_brands: brands,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn parse_moov_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<MoovBox> {
|
|
||||||
let mut moov = MoovBox::new();
|
|
||||||
|
|
||||||
let mut start = 0u64;
|
|
||||||
while start < size as u64 {
|
|
||||||
|
|
||||||
// Get box header.
|
|
||||||
let header = read_box_header(f, start).unwrap();
|
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
|
||||||
|
|
||||||
match name {
|
|
||||||
BoxType::MvhdBox => {
|
|
||||||
moov.mvhd = parse_mvhd_box(f, 0, s as u32).unwrap();
|
|
||||||
}
|
|
||||||
BoxType::TrakBox => {
|
|
||||||
let trak = parse_trak_box(f, 0, s as u32).unwrap();
|
|
||||||
moov.traks.push(trak);
|
|
||||||
}
|
|
||||||
BoxType::UdtaBox => {
|
|
||||||
start = (s as u32 - HEADER_SIZE) as u64;
|
|
||||||
}
|
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
}
|
let brand_count = (size - 16) / 4; // header + major + minor
|
||||||
Ok(moov)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_mvhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<MvhdBox> {
|
let mut brands = Vec::new();
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
for _ in 0..brand_count {
|
||||||
|
let b = reader.read_u32::<BigEndian>().unwrap();
|
||||||
let version = f.read_u8().unwrap();
|
brands.push(From::from(b));
|
||||||
let flags_a = f.read_u8().unwrap();
|
|
||||||
let flags_b = f.read_u8().unwrap();
|
|
||||||
let flags_c = f.read_u8().unwrap();
|
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
|
||||||
let creation_time = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let modification_time = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let timescale = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let duration = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let rate = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(MvhdBox{
|
|
||||||
version,
|
|
||||||
flags,
|
|
||||||
creation_time,
|
|
||||||
modification_time,
|
|
||||||
timescale,
|
|
||||||
duration,
|
|
||||||
rate,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_trak_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<TrakBox> {
|
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
|
||||||
let mut trak = TrakBox::new();
|
|
||||||
|
|
||||||
let start = 0u64;
|
|
||||||
while start < size as u64 {
|
|
||||||
// Get box header.
|
|
||||||
let header = read_box_header(f, start).unwrap();
|
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
|
||||||
|
|
||||||
match name {
|
|
||||||
BoxType::TkhdBox => {
|
|
||||||
let tkhd = parse_tkhd_box(f, 0, s as u32).unwrap();
|
|
||||||
trak.tkhd = Some(tkhd);
|
|
||||||
}
|
|
||||||
BoxType::EdtsBox => {
|
|
||||||
let edts = parse_edts_box(f, 0, s as u32).unwrap();
|
|
||||||
trak.edts = Some(edts);
|
|
||||||
}
|
|
||||||
BoxType::MdiaBox => {
|
|
||||||
let mdia = parse_mdia_box(f, 0, s as u32).unwrap();
|
|
||||||
trak.mdia = Some(mdia);
|
|
||||||
}
|
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Ok(FtypBox {
|
||||||
|
major_brand: From::from(major),
|
||||||
|
minor_version: minor,
|
||||||
|
compatible_brands: brands,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(trak)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for MoovBox {
|
||||||
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let mut moov = MoovBox::new();
|
||||||
|
|
||||||
fn parse_tkhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<TkhdBox> {
|
let mut start = 0u64;
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
while start < size as u64 {
|
||||||
|
|
||||||
let version = f.read_u8().unwrap();
|
// Get box header.
|
||||||
let flags_a = f.read_u8().unwrap();
|
let header = read_box_header(reader, start).unwrap();
|
||||||
let flags_b = f.read_u8().unwrap();
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
let flags_c = f.read_u8().unwrap();
|
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
|
||||||
let creation_time = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let modification_time = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let track_id = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let duration = f.read_u64::<BigEndian>().unwrap();
|
|
||||||
f.read_u64::<BigEndian>().unwrap(); // skip.
|
|
||||||
let layer = f.read_u16::<BigEndian>().unwrap();
|
|
||||||
let alternate_group = f.read_u16::<BigEndian>().unwrap();
|
|
||||||
let volume = f.read_u16::<BigEndian>().unwrap() >> 8;
|
|
||||||
|
|
||||||
f.read_u8().unwrap(); // skip.
|
match name {
|
||||||
let matrix = Matrix{
|
BoxType::MvhdBox => {
|
||||||
a: f.read_i32::<byteorder::LittleEndian>().unwrap(),
|
moov.mvhd = MvhdBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
b: f.read_i32::<BigEndian>().unwrap(),
|
}
|
||||||
u: f.read_i32::<BigEndian>().unwrap(),
|
BoxType::TrakBox => {
|
||||||
c: f.read_i32::<BigEndian>().unwrap(),
|
let trak = TrakBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
d: f.read_i32::<BigEndian>().unwrap(),
|
moov.traks.push(trak);
|
||||||
v: f.read_i32::<BigEndian>().unwrap(),
|
}
|
||||||
x: f.read_i32::<BigEndian>().unwrap(),
|
BoxType::UdtaBox => {
|
||||||
y: f.read_i32::<BigEndian>().unwrap(),
|
start = (s as u32 - HEADER_SIZE) as u64;
|
||||||
w: f.read_i32::<BigEndian>().unwrap(),
|
}
|
||||||
};
|
_ => break
|
||||||
|
|
||||||
let width = f.read_u32::<BigEndian>().unwrap() >> 8;
|
|
||||||
let height = f.read_u32::<BigEndian>().unwrap() >> 8;
|
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(TkhdBox {
|
|
||||||
version,
|
|
||||||
flags,
|
|
||||||
creation_time,
|
|
||||||
modification_time,
|
|
||||||
track_id,
|
|
||||||
duration,
|
|
||||||
layer,
|
|
||||||
alternate_group,
|
|
||||||
volume,
|
|
||||||
matrix,
|
|
||||||
width,
|
|
||||||
height,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_edts_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<EdtsBox> {
|
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
|
||||||
let mut edts = EdtsBox::new();
|
|
||||||
|
|
||||||
let start = 0u64;
|
|
||||||
while start < size as u64 {
|
|
||||||
// Get box header.
|
|
||||||
let header = read_box_header(f, start).unwrap();
|
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
|
||||||
|
|
||||||
match name {
|
|
||||||
BoxType::ElstBox => {
|
|
||||||
let elst = parse_elst_box(f, 0, s as u32).unwrap();
|
|
||||||
edts.elst = Some(elst);
|
|
||||||
}
|
}
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
|
Ok(moov)
|
||||||
}
|
}
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(edts)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_elst_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<ElstBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for MvhdBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
let version = f.read_u32::<BigEndian>().unwrap();
|
let version = reader.read_u8().unwrap();
|
||||||
let entry_count = f.read_u32::<BigEndian>().unwrap();
|
let flags_a = reader.read_u8().unwrap();
|
||||||
|
let flags_b = reader.read_u8().unwrap();
|
||||||
|
let flags_c = reader.read_u8().unwrap();
|
||||||
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
|
let creation_time = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let modification_time = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let timescale = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let duration = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let rate = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
let mut entries = Vec::new();
|
Ok(MvhdBox{
|
||||||
|
version,
|
||||||
|
flags,
|
||||||
|
creation_time,
|
||||||
|
modification_time,
|
||||||
|
timescale,
|
||||||
|
duration,
|
||||||
|
rate,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for _i in 0..entry_count {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for TrakBox {
|
||||||
let entry = ElstEntry{
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
segment_duration: f.read_u32::<BigEndian>().unwrap(),
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
media_time: f.read_u32::<BigEndian>().unwrap(),
|
let mut trak = TrakBox::new();
|
||||||
media_rate: f.read_u16::<BigEndian>().unwrap(),
|
|
||||||
media_rate_fraction: f.read_u16::<BigEndian>().unwrap(),
|
let start = 0u64;
|
||||||
|
while start < size as u64 {
|
||||||
|
// Get box header.
|
||||||
|
let header = read_box_header(reader, start).unwrap();
|
||||||
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
|
|
||||||
|
match name {
|
||||||
|
BoxType::TkhdBox => {
|
||||||
|
let tkhd = TkhdBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
trak.tkhd = Some(tkhd);
|
||||||
|
}
|
||||||
|
BoxType::EdtsBox => {
|
||||||
|
let edts = EdtsBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
trak.edts = Some(edts);
|
||||||
|
}
|
||||||
|
BoxType::MdiaBox => {
|
||||||
|
let mdia = MdiaBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
trak.mdia = Some(mdia);
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
|
Ok(trak)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for TkhdBox {
|
||||||
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
|
let version = reader.read_u8().unwrap();
|
||||||
|
let flags_a = reader.read_u8().unwrap();
|
||||||
|
let flags_b = reader.read_u8().unwrap();
|
||||||
|
let flags_c = reader.read_u8().unwrap();
|
||||||
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
|
let creation_time = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let modification_time = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let track_id = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let duration = reader.read_u64::<BigEndian>().unwrap();
|
||||||
|
reader.read_u64::<BigEndian>().unwrap(); // skip.
|
||||||
|
let layer = reader.read_u16::<BigEndian>().unwrap();
|
||||||
|
let alternate_group = reader.read_u16::<BigEndian>().unwrap();
|
||||||
|
let volume = reader.read_u16::<BigEndian>().unwrap() >> 8;
|
||||||
|
|
||||||
|
reader.read_u8().unwrap(); // skip.
|
||||||
|
let matrix = Matrix{
|
||||||
|
a: reader.read_i32::<byteorder::LittleEndian>().unwrap(),
|
||||||
|
b: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
u: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
c: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
d: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
v: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
x: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
y: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
|
w: reader.read_i32::<BigEndian>().unwrap(),
|
||||||
};
|
};
|
||||||
entries.push(entry);
|
|
||||||
}
|
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(ElstBox {
|
let width = reader.read_u32::<BigEndian>().unwrap() >> 8;
|
||||||
version,
|
let height = reader.read_u32::<BigEndian>().unwrap() >> 8;
|
||||||
entry_count,
|
skip(reader, current, size);
|
||||||
entries,
|
|
||||||
})
|
Ok(TkhdBox {
|
||||||
|
version,
|
||||||
|
flags,
|
||||||
|
creation_time,
|
||||||
|
modification_time,
|
||||||
|
track_id,
|
||||||
|
duration,
|
||||||
|
layer,
|
||||||
|
alternate_group,
|
||||||
|
volume,
|
||||||
|
matrix,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_mdia_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<MdiaBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for EdtsBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
let mut mdia = MdiaBox::new();
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
let mut edts = EdtsBox::new();
|
||||||
|
|
||||||
let start = 0u64;
|
let start = 0u64;
|
||||||
while start < size as u64 {
|
while start < size as u64 {
|
||||||
// Get box header.
|
// Get box header.
|
||||||
let header = read_box_header(f, start).unwrap();
|
let header = read_box_header(reader, start).unwrap();
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
|
|
||||||
match name {
|
match name {
|
||||||
BoxType::MdhdBox => {
|
BoxType::ElstBox => {
|
||||||
let mdhd = parse_mdhd_box(f, 0, s as u32).unwrap();
|
let elst = ElstBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
mdia.mdhd = Some(mdhd);
|
edts.elst = Some(elst);
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
}
|
}
|
||||||
BoxType::HdlrBox => {
|
|
||||||
let hdlr = parse_hdlr_box(f, 0, s as u32).unwrap();
|
|
||||||
mdia.hdlr = Some(hdlr);
|
|
||||||
}
|
|
||||||
BoxType::MinfBox => {
|
|
||||||
let minf = parse_minf_box(f, 0, s as u32).unwrap();
|
|
||||||
mdia.minf = Some(minf);
|
|
||||||
}
|
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
}
|
skip(reader, current, size);
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(mdia)
|
Ok(edts)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_mdhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<MdhdBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for ElstBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
let version = f.read_u8().unwrap();
|
let version = reader.read_u32::<BigEndian>().unwrap();
|
||||||
let flags_a = f.read_u8().unwrap();
|
let entry_count = reader.read_u32::<BigEndian>().unwrap();
|
||||||
let flags_b = f.read_u8().unwrap();
|
|
||||||
let flags_c = f.read_u8().unwrap();
|
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
|
||||||
let creation_time = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let modification_time = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let timescale = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let duration = f.read_u32::<BigEndian>().unwrap();
|
|
||||||
let language = f.read_u16::<BigEndian>().unwrap();
|
|
||||||
let language_string = get_language_string(language);
|
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(MdhdBox {
|
let mut entries = Vec::new();
|
||||||
version,
|
|
||||||
flags,
|
for _i in 0..entry_count {
|
||||||
creation_time,
|
let entry = ElstEntry{
|
||||||
modification_time,
|
segment_duration: reader.read_u32::<BigEndian>().unwrap(),
|
||||||
timescale,
|
media_time: reader.read_u32::<BigEndian>().unwrap(),
|
||||||
duration,
|
media_rate: reader.read_u16::<BigEndian>().unwrap(),
|
||||||
language,
|
media_rate_fraction: reader.read_u16::<BigEndian>().unwrap(),
|
||||||
language_string,
|
};
|
||||||
})
|
entries.push(entry);
|
||||||
|
}
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
|
Ok(ElstBox {
|
||||||
|
version,
|
||||||
|
entry_count,
|
||||||
|
entries,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for MdiaBox {
|
||||||
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
let mut mdia = MdiaBox::new();
|
||||||
|
|
||||||
|
let start = 0u64;
|
||||||
|
while start < size as u64 {
|
||||||
|
// Get box header.
|
||||||
|
let header = read_box_header(reader, start).unwrap();
|
||||||
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
|
|
||||||
|
match name {
|
||||||
|
BoxType::MdhdBox => {
|
||||||
|
let mdhd = MdhdBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
mdia.mdhd = Some(mdhd);
|
||||||
|
}
|
||||||
|
BoxType::HdlrBox => {
|
||||||
|
let hdlr = HdlrBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
mdia.hdlr = Some(hdlr);
|
||||||
|
}
|
||||||
|
BoxType::MinfBox => {
|
||||||
|
let minf = MinfBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
mdia.minf = Some(minf);
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
|
Ok(mdia)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for MdhdBox {
|
||||||
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
|
let version = reader.read_u8().unwrap();
|
||||||
|
let flags_a = reader.read_u8().unwrap();
|
||||||
|
let flags_b = reader.read_u8().unwrap();
|
||||||
|
let flags_c = reader.read_u8().unwrap();
|
||||||
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
|
let creation_time = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let modification_time = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let timescale = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let duration = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
let language = reader.read_u16::<BigEndian>().unwrap();
|
||||||
|
let language_string = get_language_string(language);
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
|
Ok(MdhdBox {
|
||||||
|
version,
|
||||||
|
flags,
|
||||||
|
creation_time,
|
||||||
|
modification_time,
|
||||||
|
timescale,
|
||||||
|
duration,
|
||||||
|
language,
|
||||||
|
language_string,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_language_string(language: u16) -> String {
|
fn get_language_string(language: u16) -> String {
|
||||||
|
@ -608,178 +628,190 @@ fn get_language_string(language: u16) -> String {
|
||||||
return lang_str;
|
return lang_str;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_hdlr_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<HdlrBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for HdlrBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
let version = f.read_u8().unwrap();
|
let version = reader.read_u8().unwrap();
|
||||||
let flags_a = f.read_u8().unwrap();
|
let flags_a = reader.read_u8().unwrap();
|
||||||
let flags_b = f.read_u8().unwrap();
|
let flags_b = reader.read_u8().unwrap();
|
||||||
let flags_c = f.read_u8().unwrap();
|
let flags_c = reader.read_u8().unwrap();
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
f.read_u32::<BigEndian>().unwrap(); // skip.
|
reader.read_u32::<BigEndian>().unwrap(); // skip.
|
||||||
let handler = f.read_u32::<BigEndian>().unwrap();
|
let handler = reader.read_u32::<BigEndian>().unwrap();
|
||||||
|
|
||||||
let n = f.seek(SeekFrom::Current(12)).unwrap(); // 12 bytes reserved.
|
let n = reader.seek(SeekFrom::Current(12)).unwrap(); // 12 bytes reserved.
|
||||||
let buf_size = (size as u64 - (n - current)) - HEADER_SIZE as u64;
|
let buf_size = (size as u64 - (n - current)) - HEADER_SIZE as u64;
|
||||||
let mut buf = vec![0u8; buf_size as usize];
|
let mut buf = vec![0u8; buf_size as usize];
|
||||||
f.read_exact(&mut buf).unwrap();
|
reader.read_exact(&mut buf).unwrap();
|
||||||
|
|
||||||
let handler_string = match String::from_utf8(buf) {
|
let handler_string = match String::from_utf8(buf) {
|
||||||
Ok(t) => t,
|
Ok(t) => t,
|
||||||
_ => String::from("null"),
|
_ => String::from("null"),
|
||||||
};
|
};
|
||||||
skip(f, current, size);
|
skip(reader, current, size);
|
||||||
|
|
||||||
Ok(HdlrBox {
|
Ok(HdlrBox {
|
||||||
version,
|
version,
|
||||||
flags,
|
flags,
|
||||||
handler_type: From::from(handler),
|
handler_type: From::from(handler),
|
||||||
name: handler_string,
|
name: handler_string,
|
||||||
})
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_minf_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<MinfBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for MinfBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
let mut minf = MinfBox::new();
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
let mut minf = MinfBox::new();
|
||||||
|
|
||||||
let mut start = 0u64;
|
let mut start = 0u64;
|
||||||
while start < size as u64 {
|
while start < size as u64 {
|
||||||
// Get box header.
|
// Get box header.
|
||||||
let header = read_box_header(f, start).unwrap();
|
let header = read_box_header(reader, start).unwrap();
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
|
|
||||||
match name {
|
match name {
|
||||||
BoxType::VmhdBox => {
|
BoxType::VmhdBox => {
|
||||||
let vmhd = parse_vmhd_box(f, 0, s as u32).unwrap();
|
let vmhd = VmhdBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
minf.vmhd = Some(vmhd);
|
minf.vmhd = Some(vmhd);
|
||||||
|
}
|
||||||
|
BoxType::SmhdBox => {
|
||||||
|
start = (s as u32 - HEADER_SIZE) as u64;
|
||||||
|
}
|
||||||
|
BoxType::DinfBox => {
|
||||||
|
start = (s as u32 - HEADER_SIZE) as u64;
|
||||||
|
}
|
||||||
|
BoxType::StblBox => {
|
||||||
|
let stbl = StblBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
minf.stbl = Some(stbl);
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
}
|
}
|
||||||
BoxType::SmhdBox => {
|
|
||||||
start = (s as u32 - HEADER_SIZE) as u64;
|
|
||||||
}
|
|
||||||
BoxType::DinfBox => {
|
|
||||||
start = (s as u32 - HEADER_SIZE) as u64;
|
|
||||||
}
|
|
||||||
BoxType::StblBox => {
|
|
||||||
let stbl = parse_stbl_box(f, 0, s as u32).unwrap();
|
|
||||||
minf.stbl = Some(stbl);
|
|
||||||
}
|
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
|
Ok(minf)
|
||||||
}
|
}
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(minf)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_vmhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<VmhdBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for VmhdBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
let version = f.read_u8().unwrap();
|
let version = reader.read_u8().unwrap();
|
||||||
let flags_a = f.read_u8().unwrap();
|
let flags_a = reader.read_u8().unwrap();
|
||||||
let flags_b = f.read_u8().unwrap();
|
let flags_b = reader.read_u8().unwrap();
|
||||||
let flags_c = f.read_u8().unwrap();
|
let flags_c = reader.read_u8().unwrap();
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
let graphics_mode = f.read_u16::<BigEndian>().unwrap();
|
let graphics_mode = reader.read_u16::<BigEndian>().unwrap();
|
||||||
let op_color = f.read_u16::<BigEndian>().unwrap();
|
let op_color = reader.read_u16::<BigEndian>().unwrap();
|
||||||
skip(f, current, size);
|
skip(reader, current, size);
|
||||||
|
|
||||||
Ok(VmhdBox {
|
Ok(VmhdBox {
|
||||||
version,
|
version,
|
||||||
flags,
|
flags,
|
||||||
graphics_mode,
|
graphics_mode,
|
||||||
op_color,
|
op_color,
|
||||||
})
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_stbl_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<StblBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for StblBox {
|
||||||
let mut stbl = StblBox::new();
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let mut stbl = StblBox::new();
|
||||||
|
|
||||||
let start = 0u64;
|
let start = 0u64;
|
||||||
while start < size as u64 {
|
while start < size as u64 {
|
||||||
// Get box header.
|
// Get box header.
|
||||||
let header = read_box_header(f, start).unwrap();
|
let header = read_box_header(reader, start).unwrap();
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
|
|
||||||
match name {
|
match name {
|
||||||
BoxType::StsdBox => {
|
BoxType::StsdBox => {
|
||||||
let stsd = parse_stsd_box(f, 0, s as u32).unwrap();
|
let stsd = StsdBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
stbl.stsd = Some(stsd);
|
stbl.stsd = Some(stsd);
|
||||||
|
}
|
||||||
|
BoxType::SttsBox => {
|
||||||
|
let stts = SttsBox::read_box(reader, 0, s as u32).unwrap();
|
||||||
|
stbl.stts = Some(stts);
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
}
|
}
|
||||||
BoxType::SttsBox => {
|
|
||||||
let stts = parse_stts_box(f, 0, s as u32).unwrap();
|
|
||||||
stbl.stts = Some(stts);
|
|
||||||
}
|
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
|
Ok(stbl)
|
||||||
}
|
}
|
||||||
Ok(stbl)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_stts_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<SttsBox> {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for SttsBox {
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
let version = f.read_u8().unwrap();
|
let version = reader.read_u8().unwrap();
|
||||||
let flags_a = f.read_u8().unwrap();
|
let flags_a = reader.read_u8().unwrap();
|
||||||
let flags_b = f.read_u8().unwrap();
|
let flags_b = reader.read_u8().unwrap();
|
||||||
let flags_c = f.read_u8().unwrap();
|
let flags_c = reader.read_u8().unwrap();
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
let entry_count = f.read_u32::<BigEndian>().unwrap();
|
let entry_count = reader.read_u32::<BigEndian>().unwrap();
|
||||||
let mut sample_counts = Vec::new();
|
let mut sample_counts = Vec::new();
|
||||||
let mut sample_deltas = Vec::new();
|
let mut sample_deltas = Vec::new();
|
||||||
|
|
||||||
for _i in 0..entry_count {
|
for _i in 0..entry_count {
|
||||||
let sc = f.read_u32::<BigEndian>().unwrap();
|
let sc = reader.read_u32::<BigEndian>().unwrap();
|
||||||
let sd = f.read_u32::<BigEndian>().unwrap();
|
let sd = reader.read_u32::<BigEndian>().unwrap();
|
||||||
sample_counts.push(sc);
|
sample_counts.push(sc);
|
||||||
sample_deltas.push(sd);
|
sample_deltas.push(sd);
|
||||||
}
|
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(SttsBox {
|
|
||||||
version,
|
|
||||||
flags,
|
|
||||||
entry_count,
|
|
||||||
sample_counts,
|
|
||||||
sample_deltas,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_stsd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<StsdBox> {
|
|
||||||
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
|
||||||
|
|
||||||
let version = f.read_u8().unwrap();
|
|
||||||
let flags_a = f.read_u8().unwrap();
|
|
||||||
let flags_b = f.read_u8().unwrap();
|
|
||||||
let flags_c = f.read_u8().unwrap();
|
|
||||||
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
|
||||||
f.read_u32::<BigEndian>().unwrap(); // skip.
|
|
||||||
|
|
||||||
let mut start = 0u64;
|
|
||||||
while start < size as u64 {
|
|
||||||
// Get box header.
|
|
||||||
let header = read_box_header(f, start).unwrap();
|
|
||||||
let BoxHeader{ name, size: s, offset: _ } = header;
|
|
||||||
|
|
||||||
match name {
|
|
||||||
BoxType::Avc1Box => {
|
|
||||||
start = (s as u32 - HEADER_SIZE) as u64;
|
|
||||||
}
|
|
||||||
BoxType::Mp4aBox => {
|
|
||||||
start = (s as u32 - HEADER_SIZE) as u64;
|
|
||||||
}
|
|
||||||
_ => break
|
|
||||||
}
|
}
|
||||||
}
|
skip(reader, current, size);
|
||||||
skip(f, current, size);
|
|
||||||
|
|
||||||
Ok(StsdBox {
|
Ok(SttsBox {
|
||||||
version,
|
version,
|
||||||
flags,
|
flags,
|
||||||
})
|
entry_count,
|
||||||
|
sample_counts,
|
||||||
|
sample_deltas,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn skip(f: &mut BufReader<File>, current: u64, size: u32) {
|
impl<R: Read + Seek> ReadBox<&mut BufReader<R>> for StsdBox {
|
||||||
let after = f.seek(SeekFrom::Current(0)).unwrap();
|
fn read_box(reader: &mut BufReader<R>, _offset: u64, size: u32) -> Result<Self> {
|
||||||
|
let current = reader.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
|
||||||
|
|
||||||
|
let version = reader.read_u8().unwrap();
|
||||||
|
let flags_a = reader.read_u8().unwrap();
|
||||||
|
let flags_b = reader.read_u8().unwrap();
|
||||||
|
let flags_c = reader.read_u8().unwrap();
|
||||||
|
let flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
|
||||||
|
reader.read_u32::<BigEndian>().unwrap(); // skip.
|
||||||
|
|
||||||
|
let mut start = 0u64;
|
||||||
|
while start < size as u64 {
|
||||||
|
// Get box header.
|
||||||
|
let header = read_box_header(reader, start).unwrap();
|
||||||
|
let BoxHeader{ name, size: s, offset: _ } = header;
|
||||||
|
|
||||||
|
match name {
|
||||||
|
BoxType::Avc1Box => {
|
||||||
|
start = (s as u32 - HEADER_SIZE) as u64;
|
||||||
|
}
|
||||||
|
BoxType::Mp4aBox => {
|
||||||
|
start = (s as u32 - HEADER_SIZE) as u64;
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
skip(reader, current, size);
|
||||||
|
|
||||||
|
Ok(StsdBox {
|
||||||
|
version,
|
||||||
|
flags,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn skip<R: Read + Seek>(reader: &mut BufReader<R>, current: u64, size: u32) {
|
||||||
|
let after = reader.seek(SeekFrom::Current(0)).unwrap();
|
||||||
let remaining_bytes = (size as u64 - (after - current)) as i64;
|
let remaining_bytes = (size as u64 - (after - current)) as i64;
|
||||||
f.seek(SeekFrom::Current(remaining_bytes - HEADER_SIZE as i64)).unwrap();
|
reader.seek(SeekFrom::Current(remaining_bytes - HEADER_SIZE as i64)).unwrap();
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,7 +67,7 @@ fn read_boxes(f: File) -> Result<BMFF> {
|
||||||
// Match and parse the atom boxes.
|
// Match and parse the atom boxes.
|
||||||
match name {
|
match name {
|
||||||
BoxType::FtypBox => {
|
BoxType::FtypBox => {
|
||||||
let ftyp = parse_ftyp_box(&mut reader, 0, size as u32).unwrap();
|
let ftyp = FtypBox::read_box(&mut reader, 0, size as u32).unwrap();
|
||||||
bmff.ftyp = ftyp;
|
bmff.ftyp = ftyp;
|
||||||
}
|
}
|
||||||
BoxType::FreeBox => {
|
BoxType::FreeBox => {
|
||||||
|
@ -77,7 +77,7 @@ fn read_boxes(f: File) -> Result<BMFF> {
|
||||||
start = (size as u32 - HEADER_SIZE) as u64;
|
start = (size as u32 - HEADER_SIZE) as u64;
|
||||||
}
|
}
|
||||||
BoxType::MoovBox => {
|
BoxType::MoovBox => {
|
||||||
let moov = parse_moov_box(&mut reader, 0, size as u32).unwrap();
|
let moov = MoovBox::read_box(&mut reader, 0, size as u32).unwrap();
|
||||||
bmff.moov = Some(moov);
|
bmff.moov = Some(moov);
|
||||||
}
|
}
|
||||||
BoxType::MoofBox => {
|
BoxType::MoofBox => {
|
||||||
|
@ -97,7 +97,7 @@ fn read_boxes(f: File) -> Result<BMFF> {
|
||||||
Ok(bmff)
|
Ok(bmff)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_box_header(reader: &mut BufReader<File>, start: u64) -> Result<BoxHeader> {
|
fn read_box_header<R: Read + Seek>(reader: &mut BufReader<R>, start: u64) -> Result<BoxHeader> {
|
||||||
// Seek to offset.
|
// Seek to offset.
|
||||||
let _r = reader.seek(SeekFrom::Current(start as i64));
|
let _r = reader.seek(SeekFrom::Current(start as i64));
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue