1
0
Fork 0
mirror of https://github.com/alfg/mp4-rust.git synced 2025-01-23 01:58:07 +00:00
mp4-rust/src/main.rs

194 lines
4.4 KiB
Rust
Raw Normal View History

use std::io::prelude::*;
use std::io::{BufReader, Read, SeekFrom};
use std::fs::File;
2020-01-10 06:26:08 +00:00
use std::fmt;
use std::convert::TryInto;
2020-01-12 03:34:29 +00:00
use byteorder::{ReadBytesExt};
const HEADER_SIZE: u32 = 8;
#[derive(Debug)]
pub enum Error {
InvalidData(&'static str),
}
pub type Result<T> = std::result::Result<T, Error>;
2020-01-12 08:00:30 +00:00
#[derive(Debug, Default)]
struct BMFF {
ftyp: FtypBox,
}
impl BMFF {
fn new() -> BMFF {
Default::default()
}
}
#[derive(Debug, Default)]
struct BMFFBox {
head: BoxHeader,
}
impl BMFFBox {
fn new() -> BMFFBox {
Default::default()
}
}
#[derive(Debug, Default)]
2020-01-12 03:34:29 +00:00
struct BoxHeader {
2020-01-08 06:51:21 +00:00
name: String,
2020-01-12 03:34:29 +00:00
size: u64,
offset: u64,
2020-01-08 06:51:21 +00:00
}
2020-01-12 08:00:30 +00:00
#[derive(Debug, Default)]
2020-01-10 06:26:08 +00:00
struct FtypBox {
major_brand: FourCC,
minor_version: u32,
compatible_brands: Vec<FourCC>,
}
#[derive(Default, PartialEq, Clone)]
pub struct FourCC {
pub value: String
}
impl From<u32> for FourCC {
fn from(number: u32) -> FourCC {
let mut box_chars = Vec::new();
for x in 0..4 {
let c = (number >> (x * 8) & 0x0000_00FF) as u8;
box_chars.push(c);
}
box_chars.reverse();
let box_string = match String::from_utf8(box_chars) {
Ok(t) => t,
_ => String::from("null"), // error to retrieve fourcc
};
FourCC {
value: box_string
}
}
}
impl fmt::Debug for FourCC {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.value)
}
}
fn main() -> std::io::Result<()> {
2020-01-12 08:00:30 +00:00
// Open file and read boxes.
2020-01-08 06:51:21 +00:00
let f = File::open("tears-of-steel-2s.mp4")?;
2020-01-12 08:00:30 +00:00
// let boxes = read_boxes(f);
let bmff = read_boxes(f);
// Print results.
println!("{:?}", bmff.unwrap());
// Done.
println!("done");
Ok(())
}
fn read_boxes(f: File) -> Result<BMFF> {
let filesize = f.metadata().unwrap().len();
let mut reader = BufReader::new(f);
2020-01-12 08:00:30 +00:00
let mut bmff = BMFF::new();
2020-01-12 03:34:29 +00:00
let mut start = 0u64;
while start < filesize {
2020-01-12 08:00:30 +00:00
// Get box header.
let header = read_box_header(&mut reader, start).unwrap();
let BoxHeader{ name, size, offset } = header;
2020-01-12 08:00:30 +00:00
let mut b = BMFFBox::new();
b.head = BoxHeader{
name: name.try_into().unwrap(),
size: size as u64,
offset: offset as u64,
2020-01-12 03:34:29 +00:00
};
// Match and parse the filetype.
2020-01-12 08:00:30 +00:00
match b.head.name.as_ref() {
2020-01-12 03:34:29 +00:00
"ftyp" => {
2020-01-12 08:00:30 +00:00
let ftyp = parse_ftyp_box(&mut reader, 0, size as u32).unwrap();
bmff.ftyp = ftyp;
2020-01-12 03:34:29 +00:00
}
"free" => {
start = 0;
}
"mdat" => {
2020-01-12 08:00:30 +00:00
start = (size as u32 - HEADER_SIZE) as u64;
2020-01-12 03:34:29 +00:00
}
"moov" => {
2020-01-12 08:00:30 +00:00
start = (size as u32 - HEADER_SIZE) as u64;
2020-01-12 03:34:29 +00:00
}
"moof" => {
2020-01-12 08:00:30 +00:00
start = (size as u32 - HEADER_SIZE) as u64;
2020-01-12 03:34:29 +00:00
}
_ => break
2020-01-10 06:26:08 +00:00
};
2020-01-08 06:51:21 +00:00
}
2020-01-12 08:00:30 +00:00
Ok(bmff)
}
2020-01-08 06:51:21 +00:00
2020-01-12 08:00:30 +00:00
fn read_box_header(reader: &mut BufReader<File>, start: u64) -> Result<BoxHeader> {
// Seek to offset.
let _r = reader.seek(SeekFrom::Current(start as i64));
// Create and read to buf.
let mut buf = [0u8;8]; // 8 bytes for box header.
let _n = reader.read(&mut buf);
// Get size.
let s = buf[0..4].try_into().unwrap();
let size = u32::from_be_bytes(s);
// TODO: Err if size is 0.
// if size == 0 { break; }
// Get box type string.
let t = buf[4..8].try_into().unwrap();
let typ = match std::str::from_utf8(t) {
Ok(v) => v,
Err(e) => panic!("Invalid UTF-8 sequence: {}", e),
};
let offset = match size {
1 => 4 + 4 + 8,
_ => 4 + 4,
};
Ok(BoxHeader {
name: typ.try_into().unwrap(),
size: size as u64,
offset: offset as u64,
})
2020-01-10 06:26:08 +00:00
}
2020-01-12 03:34:29 +00:00
fn parse_ftyp_box(f: &mut BufReader<File>, _offset: u64, size: u32) -> Result<FtypBox> {
2020-01-10 06:26:08 +00:00
let major = f.read_u32::<byteorder::BigEndian>().unwrap();
let minor = f.read_u32::<byteorder::BigEndian>().unwrap();
2020-01-12 03:34:29 +00:00
if size % 4 != 0 {
return Err(Error::InvalidData("invalid ftyp size"));
}
2020-01-10 06:26:08 +00:00
let brand_count = (size - 16) / 4; // header + major + minor
let mut brands = Vec::new();
for _ in 0..brand_count {
let b = f.read_u32::<byteorder::BigEndian>().unwrap();
brands.push(From::from(b));
}
2020-01-12 03:34:29 +00:00
Ok(FtypBox {
2020-01-10 06:26:08 +00:00
major_brand: From::from(major),
minor_version: minor,
compatible_brands: brands,
2020-01-12 03:34:29 +00:00
})
2020-01-10 06:26:08 +00:00
}