mirror of
https://github.com/alfg/mp4-rust.git
synced 2025-01-29 20:58:13 +00:00
4df9fa1bd4
* Add ReadBox trait * Add boxtype macro * Remove offset in BoxHeader * Fix parsing error when box has largesize * Remove duplicated codes reading version and flags * Add test code for all leaves in box tree * Remove entry_count that is duplacated with entries.len() * Change volume type to Ratio<u16> Co-authored-by: Byungwan Jun <unipro.kr@gmail.com>
118 lines
4.2 KiB
Rust
118 lines
4.2 KiB
Rust
use mp4;
|
|
use std::env;
|
|
use std::fs::File;
|
|
|
|
fn main() {
|
|
let args: Vec<String> = env::args().collect();
|
|
|
|
match args.len() {
|
|
2 => {
|
|
let filename = &args[1];
|
|
let f = File::open(filename).unwrap();
|
|
|
|
let bmff = mp4::read_mp4(f).unwrap();
|
|
let moov = bmff.moov.unwrap();
|
|
|
|
// Print results.
|
|
println!("File:");
|
|
println!(" file size: {}", bmff.size);
|
|
println!(
|
|
" brands: {:?} {:?}\n",
|
|
bmff.ftyp.major_brand, bmff.ftyp.compatible_brands
|
|
);
|
|
|
|
println!("Movie:");
|
|
println!(" version: {:?}", moov.mvhd.version);
|
|
println!(
|
|
" creation time: {}",
|
|
creation_time(moov.mvhd.creation_time)
|
|
);
|
|
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();
|
|
println!("Track: {:?}", tkhd.track_id);
|
|
println!(" flags: {:?}", tkhd.flags);
|
|
println!(" id: {:?}", tkhd.track_id);
|
|
println!(" duration: {:?}", tkhd.duration);
|
|
if tkhd.width != 0 && tkhd.height != 0 {
|
|
println!(" width: {:?}", tkhd.width);
|
|
println!(" height: {:?}", tkhd.height);
|
|
}
|
|
if let Some(ref mdia) = trak.mdia {
|
|
let hdlr = mdia.hdlr.as_ref().unwrap();
|
|
let mdhd = mdia.mdhd.as_ref().unwrap();
|
|
let stts = mdia
|
|
.minf
|
|
.as_ref()
|
|
.map(|m| m.stbl.as_ref().map(|s| s.stts.as_ref()).flatten())
|
|
.flatten();
|
|
|
|
println!(
|
|
" type: {:?}",
|
|
get_handler_type(hdlr.handler_type.value.as_ref())
|
|
);
|
|
println!(" language: {:?}", mdhd.language);
|
|
|
|
println!(" media:");
|
|
if let Some(ref s) = stts {
|
|
println!(" sample count: {:?}", s.entries[0].sample_count);
|
|
}
|
|
println!(" timescale: {:?}", mdhd.timescale);
|
|
println!(
|
|
" duration: {:?} (media timescale units)",
|
|
mdhd.duration
|
|
);
|
|
println!(
|
|
" duration: {:?} (ms)",
|
|
get_duration_ms(mdhd.duration, mdhd.timescale)
|
|
);
|
|
if get_handler_type(hdlr.handler_type.value.as_ref()) == mp4::TrackType::Video {
|
|
if let Some(ref s) = stts {
|
|
println!(
|
|
" frame rate: (computed): {:?}",
|
|
get_framerate(s.entries[0].sample_count, mdhd.duration, mdhd.timescale)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {
|
|
println!("Usage: mp4info <filename>");
|
|
}
|
|
}
|
|
}
|
|
|
|
fn get_handler_type(handler: &str) -> mp4::TrackType {
|
|
let mut typ: mp4::TrackType = mp4::TrackType::Unknown;
|
|
match handler {
|
|
"vide" => typ = mp4::TrackType::Video,
|
|
"soun" => typ = mp4::TrackType::Audio,
|
|
"meta" => typ = mp4::TrackType::Unknown,
|
|
_ => (),
|
|
}
|
|
return typ;
|
|
}
|
|
|
|
fn get_duration_ms(duration: u64, timescale: u32) -> String {
|
|
let ms = (duration as f64 / timescale as f64) * 1000.0;
|
|
return format!("{:.2}", ms.floor());
|
|
}
|
|
|
|
fn get_framerate(sample_count: u32, duration: u64, timescale: u32) -> String {
|
|
let sc = (sample_count as f64) * 1000.0;
|
|
let ms = (duration as f64 / timescale as f64) * 1000.0;
|
|
return format!("{:.2}", sc / ms.floor());
|
|
}
|
|
|
|
fn creation_time(creation_time: u64) -> u64 {
|
|
// convert from MP4 epoch (1904-01-01) to Unix epoch (1970-01-01)
|
|
if creation_time >= 2082844800 {
|
|
creation_time - 2082844800
|
|
} else {
|
|
creation_time
|
|
}
|
|
}
|