1
0
Fork 0
mirror of https://github.com/alfg/mp4-rust.git synced 2025-01-05 09:58:40 +00:00

cleanup on lib and add more mp4info example output.

This commit is contained in:
Alf 2020-01-21 21:41:51 -08:00
parent 56e49bea57
commit b345450e19
2 changed files with 154 additions and 108 deletions

View file

@ -2,6 +2,8 @@ extern crate mp4;
use std::env; use std::env;
use std::fs::File; use std::fs::File;
use std::any::Any;
use mp4::{FourCC, TrackType};
fn main() { fn main() {
let args: Vec<String> = env::args().collect(); let args: Vec<String> = env::args().collect();
@ -11,14 +13,48 @@ fn main() {
let filename = &args[1]; let filename = &args[1];
let f = File::open(filename).unwrap(); let f = File::open(filename).unwrap();
let bmff = mp4::read_mp4(f); let bmff = mp4::read_mp4(f).unwrap();
let moov = bmff.moov.unwrap();
// Print results. // Print results.
println!("{:?}", bmff.unwrap()); // println!("{:?}", bmff.unwrap());
println!("File:");
println!(" file size: {}", bmff.size);
println!(" brands: {:?} {:?}\n", bmff.ftyp.major_brand, bmff.ftyp.compatible_brands);
println!("Movie:");
println!(" duration: {:?}", moov.mvhd.duration);
println!(" timescale: {:?}\n", moov.mvhd.timescale);
println!("Found {} Tracks", moov.traks.len());
for trak in moov.traks.iter() {
let tkhd = trak.tkhd.as_ref().unwrap();
let mdia = trak.mdia.as_ref().unwrap();
let hdlr = mdia.hdlr.as_ref().unwrap();
let mdhd = mdia.mdhd.as_ref().unwrap();
println!(" flags: {:?}", tkhd.flags);
println!(" id: {:?}", tkhd.track_id);
println!(" type: {:?}", get_handler_type(hdlr.handler_type.value.as_ref()));
println!(" duration: {:?}", tkhd.duration);
println!(" language: {:?}", mdhd.language_string);
println!(" width: {:?}", tkhd.width);
println!(" height: {:?}\n", tkhd.height);
}
}, },
_ => { _ => {
println!("Usage: mp4info <filename>"); println!("Usage: mp4info <filename>");
} }
} }
}
fn get_handler_type(handler: &str) -> TrackType {
let mut typ: TrackType = TrackType::Unknown;
match handler {
"vide" => typ = TrackType::Video,
"soun" => typ = TrackType::Audio,
"meta" => typ = TrackType::Unknown,
_ => (),
}
return typ;
} }

View file

@ -16,10 +16,19 @@ pub enum Error {
} }
pub type Result<T> = std::result::Result<T, Error>; pub type Result<T> = std::result::Result<T, Error>;
#[derive(Debug, PartialEq)]
pub enum TrackType {
Audio,
Video,
Metadata,
Unknown,
}
#[derive(Debug, Default)] #[derive(Debug, Default)]
pub struct BMFF { pub struct BMFF {
ftyp: FtypBox, pub ftyp: FtypBox,
moov: Option<MoovBox>, pub moov: Option<MoovBox>,
pub size: u64,
} }
impl BMFF { impl BMFF {
@ -47,16 +56,16 @@ struct BoxHeader {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct FtypBox { pub struct FtypBox {
major_brand: FourCC, pub major_brand: FourCC,
minor_version: u32, pub minor_version: u32,
compatible_brands: Vec<FourCC>, pub compatible_brands: Vec<FourCC>,
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct MoovBox { pub struct MoovBox {
mvhd: MvhdBox, pub mvhd: MvhdBox,
traks: Vec<TrakBox>, pub traks: Vec<TrakBox>,
} }
impl MoovBox { impl MoovBox {
@ -66,21 +75,21 @@ impl MoovBox {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct MvhdBox { pub struct MvhdBox {
version: u8, pub version: u8,
flags: u32, pub flags: u32,
creation_time: u32, pub creation_time: u32,
modification_time: u32, pub modification_time: u32,
timescale: u32, pub timescale: u32,
duration: u32, pub duration: u32,
rate: u32, pub rate: u32,
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct TrakBox { pub struct TrakBox {
tkhd: Option<TkhdBox>, pub tkhd: Option<TkhdBox>,
edts: Option<EdtsBox>, pub edts: Option<EdtsBox>,
mdia: Option<MdiaBox>, pub mdia: Option<MdiaBox>,
} }
impl TrakBox { impl TrakBox {
@ -90,23 +99,23 @@ impl TrakBox {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct TkhdBox { pub struct TkhdBox {
version: u8, pub version: u8,
flags: u32, pub flags: u32,
creation_time: u32, pub creation_time: u32,
modification_time: u32, pub modification_time: u32,
track_id: u32, pub track_id: u32,
duration: u64, pub duration: u64,
layer: u16, pub layer: u16,
alternate_group: u16, pub alternate_group: u16,
volume: u16, pub volume: u16,
matrix: Matrix, pub matrix: Matrix,
width: u32, pub width: u32,
height: u32, pub height: u32,
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct Matrix { pub struct Matrix {
a: i32, a: i32,
b: i32, b: i32,
u: i32, u: i32,
@ -119,8 +128,8 @@ struct Matrix {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct EdtsBox { pub struct EdtsBox {
elst: Option<ElstBox>, pub elst: Option<ElstBox>,
} }
impl EdtsBox { impl EdtsBox {
@ -130,10 +139,10 @@ impl EdtsBox {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct ElstBox { pub struct ElstBox {
version: u32, pub version: u32,
entry_count: u32, pub entry_count: u32,
entries: Vec<ElstEntry>, pub entries: Vec<ElstEntry>,
} }
impl ElstBox { impl ElstBox {
@ -143,17 +152,17 @@ impl ElstBox {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct ElstEntry { pub struct ElstEntry {
segment_duration: u32, pub segment_duration: u32,
media_time: u32, pub media_time: u32,
media_rate: u16, pub media_rate: u16,
media_rate_fraction: u16, pub media_rate_fraction: u16,
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct MdiaBox { pub struct MdiaBox {
mdhd: Option<MdhdBox>, pub mdhd: Option<MdhdBox>,
hdlr: Option<HdlrBox>, pub hdlr: Option<HdlrBox>,
} }
impl MdiaBox { impl MdiaBox {
@ -163,15 +172,15 @@ impl MdiaBox {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct MdhdBox { pub struct MdhdBox {
version: u8, pub version: u8,
flags: u32, pub flags: u32,
creation_time: u32, pub creation_time: u32,
modification_time: u32, pub modification_time: u32,
timescale: u32, pub timescale: u32,
duration: u32, pub duration: u32,
language: u16, pub language: u16,
language_string: String, pub language_string: String,
} }
impl MdhdBox { impl MdhdBox {
@ -181,11 +190,11 @@ impl MdhdBox {
} }
#[derive(Debug, Default)] #[derive(Debug, Default)]
struct HdlrBox { pub struct HdlrBox {
version: u8, pub version: u8,
flags: u32, pub flags: u32,
handler_type: FourCC, pub handler_type: FourCC,
name: String, pub name: String,
} }
impl HdlrBox { impl HdlrBox {
@ -238,6 +247,7 @@ fn read_boxes(f: File) -> Result<BMFF> {
let filesize = f.metadata().unwrap().len(); let filesize = f.metadata().unwrap().len();
let mut reader = BufReader::new(f); let mut reader = BufReader::new(f);
let mut bmff = BMFF::new(); let mut bmff = BMFF::new();
bmff.size = filesize;
let mut start = 0u64; let mut start = 0u64;
while start < filesize { while start < filesize {
@ -313,8 +323,8 @@ fn read_box_header(reader: &mut BufReader<File>, start: u64) -> Result<BoxHeader
} }
fn parse_ftyp_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<FtypBox> { fn parse_ftyp_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<FtypBox> {
let major = f.read_u32::<byteorder::BigEndian>().unwrap(); let major = f.read_u32::<BigEndian>().unwrap();
let minor = f.read_u32::<byteorder::BigEndian>().unwrap(); let minor = f.read_u32::<BigEndian>().unwrap();
if size % 4 != 0 { if size % 4 != 0 {
return Err(Error::InvalidData("invalid ftyp size")); return Err(Error::InvalidData("invalid ftyp size"));
} }
@ -322,7 +332,7 @@ fn parse_ftyp_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Ft
let mut brands = Vec::new(); let mut brands = Vec::new();
for _ in 0..brand_count { for _ in 0..brand_count {
let b = f.read_u32::<byteorder::BigEndian>().unwrap(); let b = f.read_u32::<BigEndian>().unwrap();
brands.push(From::from(b)); brands.push(From::from(b));
} }
@ -378,11 +388,11 @@ fn parse_mvhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Mv
let flags_b = f.read_u8().unwrap(); let flags_b = f.read_u8().unwrap();
let flags_c = 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 flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
let creation_time = f.read_u32::<byteorder::BigEndian>().unwrap(); let creation_time = f.read_u32::<BigEndian>().unwrap();
let modification_time = f.read_u32::<byteorder::BigEndian>().unwrap(); let modification_time = f.read_u32::<BigEndian>().unwrap();
let timescale = f.read_u32::<byteorder::BigEndian>().unwrap(); let timescale = f.read_u32::<BigEndian>().unwrap();
let duration = f.read_u32::<byteorder::BigEndian>().unwrap(); let duration = f.read_u32::<BigEndian>().unwrap();
let rate = f.read_u32::<byteorder::BigEndian>().unwrap(); let rate = f.read_u32::<BigEndian>().unwrap();
// Skip remaining bytes. // Skip remaining bytes.
let after = f.seek(SeekFrom::Current(0)).unwrap(); let after = f.seek(SeekFrom::Current(0)).unwrap();
@ -404,7 +414,7 @@ fn parse_trak_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Tr
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position. let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
let mut trak = TrakBox::new(); let mut trak = TrakBox::new();
let mut 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(f, start).unwrap();
@ -452,30 +462,30 @@ fn parse_tkhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Tk
let flags_b = f.read_u8().unwrap(); let flags_b = f.read_u8().unwrap();
let flags_c = 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 flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
let creation_time = f.read_u32::<byteorder::BigEndian>().unwrap(); let creation_time = f.read_u32::<BigEndian>().unwrap();
let modification_time = f.read_u32::<byteorder::BigEndian>().unwrap(); let modification_time = f.read_u32::<BigEndian>().unwrap();
let track_id = f.read_u32::<byteorder::BigEndian>().unwrap(); let track_id = f.read_u32::<BigEndian>().unwrap();
let duration = f.read_u64::<byteorder::BigEndian>().unwrap(); let duration = f.read_u64::<BigEndian>().unwrap();
f.read_u64::<byteorder::BigEndian>().unwrap(); // skip. f.read_u64::<BigEndian>().unwrap(); // skip.
let layer = f.read_u16::<byteorder::BigEndian>().unwrap(); let layer = f.read_u16::<BigEndian>().unwrap();
let alternate_group = f.read_u16::<byteorder::BigEndian>().unwrap(); let alternate_group = f.read_u16::<BigEndian>().unwrap();
let volume = f.read_u16::<byteorder::BigEndian>().unwrap() >> 8; let volume = f.read_u16::<BigEndian>().unwrap() >> 8;
f.read_u8().unwrap(); // skip. f.read_u8().unwrap(); // skip.
let matrix = Matrix{ let matrix = Matrix{
a: f.read_i32::<byteorder::LittleEndian>().unwrap(), a: f.read_i32::<byteorder::LittleEndian>().unwrap(),
b: f.read_i32::<byteorder::BigEndian>().unwrap(), b: f.read_i32::<BigEndian>().unwrap(),
u: f.read_i32::<byteorder::BigEndian>().unwrap(), u: f.read_i32::<BigEndian>().unwrap(),
c: f.read_i32::<byteorder::BigEndian>().unwrap(), c: f.read_i32::<BigEndian>().unwrap(),
d: f.read_i32::<byteorder::BigEndian>().unwrap(), d: f.read_i32::<BigEndian>().unwrap(),
v: f.read_i32::<byteorder::BigEndian>().unwrap(), v: f.read_i32::<BigEndian>().unwrap(),
x: f.read_i32::<byteorder::BigEndian>().unwrap(), x: f.read_i32::<BigEndian>().unwrap(),
y: f.read_i32::<byteorder::BigEndian>().unwrap(), y: f.read_i32::<BigEndian>().unwrap(),
w: f.read_i32::<byteorder::BigEndian>().unwrap(), w: f.read_i32::<BigEndian>().unwrap(),
}; };
let width = f.read_u32::<byteorder::BigEndian>().unwrap() >> 8; let width = f.read_u32::<BigEndian>().unwrap() >> 8;
let height = f.read_u32::<byteorder::BigEndian>().unwrap() >> 8; let height = f.read_u32::<BigEndian>().unwrap() >> 8;
// Skip remaining bytes. // Skip remaining bytes.
let after = f.seek(SeekFrom::Current(0)).unwrap(); let after = f.seek(SeekFrom::Current(0)).unwrap();
@ -502,7 +512,7 @@ fn parse_edts_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Ed
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position. let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
let mut edts = EdtsBox::new(); let mut edts = EdtsBox::new();
let mut 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(f, start).unwrap();
@ -534,17 +544,17 @@ fn parse_edts_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Ed
fn parse_elst_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<ElstBox> { fn parse_elst_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<ElstBox> {
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position. let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
let version = f.read_u32::<byteorder::BigEndian>().unwrap(); let version = f.read_u32::<BigEndian>().unwrap();
let entry_count = f.read_u32::<byteorder::BigEndian>().unwrap(); let entry_count = f.read_u32::<BigEndian>().unwrap();
let mut entries = Vec::new(); let mut entries = Vec::new();
for i in 0..entry_count { for _i in 0..entry_count {
let entry = ElstEntry{ let entry = ElstEntry{
segment_duration: f.read_u32::<byteorder::BigEndian>().unwrap(), segment_duration: f.read_u32::<BigEndian>().unwrap(),
media_time: f.read_u32::<byteorder::BigEndian>().unwrap(), media_time: f.read_u32::<BigEndian>().unwrap(),
media_rate: f.read_u16::<byteorder::BigEndian>().unwrap(), media_rate: f.read_u16::<BigEndian>().unwrap(),
media_rate_fraction: f.read_u16::<byteorder::BigEndian>().unwrap(), media_rate_fraction: f.read_u16::<BigEndian>().unwrap(),
}; };
entries.push(entry); entries.push(entry);
} }
@ -565,7 +575,7 @@ fn parse_mdia_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Md
let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position. let current = f.seek(SeekFrom::Current(0)).unwrap(); // Current cursor position.
let mut mdia = MdiaBox::new(); let mut mdia = MdiaBox::new();
let mut 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(f, start).unwrap();
@ -606,11 +616,11 @@ fn parse_mdhd_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Md
let flags_b = f.read_u8().unwrap(); let flags_b = f.read_u8().unwrap();
let flags_c = 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 flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
let creation_time = f.read_u32::<byteorder::BigEndian>().unwrap(); let creation_time = f.read_u32::<BigEndian>().unwrap();
let modification_time = f.read_u32::<byteorder::BigEndian>().unwrap(); let modification_time = f.read_u32::<BigEndian>().unwrap();
let timescale = f.read_u32::<byteorder::BigEndian>().unwrap(); let timescale = f.read_u32::<BigEndian>().unwrap();
let duration = f.read_u32::<byteorder::BigEndian>().unwrap(); let duration = f.read_u32::<BigEndian>().unwrap();
let language = f.read_u16::<byteorder::BigEndian>().unwrap(); let language = f.read_u16::<BigEndian>().unwrap();
let language_string = get_language_string(language); let language_string = get_language_string(language);
// Skip remaining bytes. // Skip remaining bytes.
@ -653,8 +663,8 @@ fn parse_hdlr_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<Hd
let flags_b = f.read_u8().unwrap(); let flags_b = f.read_u8().unwrap();
let flags_c = 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 flags = u32::from(flags_a) << 16 | u32::from(flags_b) << 8 | u32::from(flags_c);
f.read_u32::<byteorder::BigEndian>().unwrap(); // skip. f.read_u32::<BigEndian>().unwrap(); // skip.
let handler = f.read_u32::<byteorder::BigEndian>().unwrap(); let handler = f.read_u32::<BigEndian>().unwrap();
let n = f.seek(SeekFrom::Current(12)).unwrap(); // 12 bytes reserved. let n = f.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;