mirror of
https://github.com/Serial-ATA/lofty-rs
synced 2025-01-19 07:33:53 +00:00
MPEG: Improve duration estimation
This commit is contained in:
parent
4eae6e12a7
commit
24da5bac31
4 changed files with 108 additions and 52 deletions
|
@ -1,4 +1,5 @@
|
||||||
use super::constants::{BITRATES, PADDING_SIZES, SAMPLES, SAMPLE_RATES, SIDE_INFORMATION_SIZES};
|
use super::constants::{BITRATES, PADDING_SIZES, SAMPLES, SAMPLE_RATES, SIDE_INFORMATION_SIZES};
|
||||||
|
use crate::config::ParsingMode;
|
||||||
use crate::error::Result;
|
use crate::error::Result;
|
||||||
use crate::macros::decode_err;
|
use crate::macros::decode_err;
|
||||||
|
|
||||||
|
@ -49,10 +50,11 @@ where
|
||||||
// Unlike `search_for_frame_sync`, since this has the `Seek` bound, it will seek the reader
|
// Unlike `search_for_frame_sync`, since this has the `Seek` bound, it will seek the reader
|
||||||
// back to the start of the header.
|
// back to the start of the header.
|
||||||
const REV_FRAME_SEARCH_BOUNDS: u64 = 1024;
|
const REV_FRAME_SEARCH_BOUNDS: u64 = 1024;
|
||||||
pub(super) fn rev_search_for_frame_sync<R>(
|
pub(super) fn rev_search_for_frame_header<R>(
|
||||||
input: &mut R,
|
input: &mut R,
|
||||||
pos: &mut u64,
|
pos: &mut u64,
|
||||||
) -> std::io::Result<Option<u64>>
|
parse_mode: ParsingMode,
|
||||||
|
) -> Result<Option<Header>>
|
||||||
where
|
where
|
||||||
R: Read + Seek,
|
R: Read + Seek,
|
||||||
{
|
{
|
||||||
|
@ -61,15 +63,49 @@ where
|
||||||
*pos -= search_bounds;
|
*pos -= search_bounds;
|
||||||
input.seek(SeekFrom::Start(*pos))?;
|
input.seek(SeekFrom::Start(*pos))?;
|
||||||
|
|
||||||
let ret = search_for_frame_sync(&mut input.take(search_bounds));
|
let mut buf = Vec::with_capacity(search_bounds as usize);
|
||||||
if let Ok(Some(_)) = ret {
|
input.take(search_bounds).read_to_end(&mut buf)?;
|
||||||
// Seek to the start of the frame sync
|
|
||||||
input.seek(SeekFrom::Current(-2))?;
|
let mut frame_sync = [0u8; 2];
|
||||||
|
for (i, byte) in buf.iter().rev().enumerate() {
|
||||||
|
frame_sync[1] = frame_sync[0];
|
||||||
|
frame_sync[0] = *byte;
|
||||||
|
if verify_frame_sync(frame_sync) {
|
||||||
|
let relative_frame_start = (search_bounds as usize) - (i + 1);
|
||||||
|
if relative_frame_start + 4 > buf.len() {
|
||||||
|
if parse_mode == ParsingMode::Strict {
|
||||||
|
decode_err!(@BAIL Mpeg, "Expected full frame header (incomplete stream?)")
|
||||||
|
}
|
||||||
|
|
||||||
|
log::warn!("MPEG: Incomplete frame header, giving up");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
let header = Header::read(u32::from_be_bytes([
|
||||||
|
frame_sync[0],
|
||||||
|
frame_sync[1],
|
||||||
|
buf[relative_frame_start + 2],
|
||||||
|
buf[relative_frame_start + 3],
|
||||||
|
]));
|
||||||
|
|
||||||
|
// We need to check if the header is actually valid. For
|
||||||
|
// all we know, we could be in some junk (ex. 0xFF_FF_FF_FF).
|
||||||
|
if header.is_none() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Seek to the start of the frame sync
|
||||||
|
*pos += relative_frame_start as u64;
|
||||||
|
input.seek(SeekFrom::Start(*pos))?;
|
||||||
|
|
||||||
|
return Ok(header);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// See [`cmp_header()`].
|
||||||
pub(crate) enum HeaderCmpResult {
|
pub(crate) enum HeaderCmpResult {
|
||||||
Equal,
|
Equal,
|
||||||
Undetermined,
|
Undetermined,
|
||||||
|
@ -80,6 +116,18 @@ pub(crate) enum HeaderCmpResult {
|
||||||
// If they aren't equal, something is broken.
|
// If they aren't equal, something is broken.
|
||||||
pub(super) const HEADER_MASK: u32 = 0xFFFE_0C00;
|
pub(super) const HEADER_MASK: u32 = 0xFFFE_0C00;
|
||||||
|
|
||||||
|
/// Compares the versions, layers, and sample rates of two frame headers.
|
||||||
|
///
|
||||||
|
/// It is safe to assume that the reader will no longer produce valid headers if [`HeaderCmpResult::Undetermined`]
|
||||||
|
/// is returned.
|
||||||
|
///
|
||||||
|
/// To compare two already constructed [`Header`]s, use [`Header::cmp()`].
|
||||||
|
///
|
||||||
|
/// ## Returns
|
||||||
|
///
|
||||||
|
/// - [`HeaderCmpResult::Equal`] if the headers are equal.
|
||||||
|
/// - [`HeaderCmpResult::NotEqual`] if the headers are not equal.
|
||||||
|
/// - [`HeaderCmpResult::Undetermined`] if the comparison could not be made (Some IO error occurred).
|
||||||
pub(crate) fn cmp_header<R>(
|
pub(crate) fn cmp_header<R>(
|
||||||
reader: &mut R,
|
reader: &mut R,
|
||||||
header_size: u32,
|
header_size: u32,
|
||||||
|
@ -162,7 +210,7 @@ pub enum Emphasis {
|
||||||
CCIT_J17,
|
CCIT_J17,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone, Debug)]
|
||||||
pub(crate) struct Header {
|
pub(crate) struct Header {
|
||||||
pub(crate) sample_rate: u32,
|
pub(crate) sample_rate: u32,
|
||||||
pub(crate) len: u32,
|
pub(crate) len: u32,
|
||||||
|
@ -269,14 +317,21 @@ impl Header {
|
||||||
|
|
||||||
Some(header)
|
Some(header)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Equivalent of [`cmp_header()`], but for an already constructed `Header`.
|
||||||
|
pub(super) fn cmp(self, other: &Self) -> bool {
|
||||||
|
self.version == other.version
|
||||||
|
&& self.layer == other.layer
|
||||||
|
&& self.sample_rate == other.sample_rate
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) struct XingHeader {
|
pub(super) struct VbrHeader {
|
||||||
pub frames: u32,
|
pub frames: u32,
|
||||||
pub size: u32,
|
pub size: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl XingHeader {
|
impl VbrHeader {
|
||||||
pub(super) fn read(reader: &mut &[u8]) -> Result<Option<Self>> {
|
pub(super) fn read(reader: &mut &[u8]) -> Result<Option<Self>> {
|
||||||
let reader_len = reader.len();
|
let reader_len = reader.len();
|
||||||
|
|
||||||
|
@ -331,7 +386,9 @@ impl XingHeader {
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
|
use crate::config::ParsingMode;
|
||||||
use crate::tag::utils::test_utils::read_path;
|
use crate::tag::utils::test_utils::read_path;
|
||||||
|
|
||||||
use std::io::{Cursor, Read, Seek, SeekFrom};
|
use std::io::{Cursor, Read, Seek, SeekFrom};
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -347,21 +404,30 @@ mod tests {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn rev_search_for_frame_sync() {
|
#[rustfmt::skip]
|
||||||
fn test<R: Read + Seek>(reader: &mut R, expected_result: Option<u64>) {
|
fn rev_search_for_frame_header() {
|
||||||
|
fn test<R: Read + Seek>(reader: &mut R, expected_reader_position: Option<u64>) {
|
||||||
// We have to start these at the end to do a reverse search, of course :)
|
// We have to start these at the end to do a reverse search, of course :)
|
||||||
let mut pos = reader.seek(SeekFrom::End(0)).unwrap();
|
let mut pos = reader.seek(SeekFrom::End(0)).unwrap();
|
||||||
|
|
||||||
let ret = super::rev_search_for_frame_sync(reader, &mut pos).unwrap();
|
let ret = super::rev_search_for_frame_header(reader, &mut pos, ParsingMode::Strict);
|
||||||
assert_eq!(ret, expected_result);
|
|
||||||
|
if expected_reader_position.is_some() {
|
||||||
|
assert!(ret.is_ok());
|
||||||
|
assert!(ret.unwrap().is_some());
|
||||||
|
assert_eq!(Some(pos), expected_reader_position);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert!(ret.unwrap().is_none());
|
||||||
}
|
}
|
||||||
|
|
||||||
test(&mut Cursor::new([0xFF, 0xFB, 0x00]), Some(0));
|
test(&mut Cursor::new([0xFF, 0xFB, 0x52, 0xC4]), Some(0));
|
||||||
test(&mut Cursor::new([0x00, 0x00, 0x01, 0xFF, 0xFB]), Some(3));
|
test(&mut Cursor::new([0x00, 0x00, 0x01, 0xFF, 0xFB, 0x52, 0xC4]), Some(3));
|
||||||
test(&mut Cursor::new([0x01, 0xFF]), None);
|
test(&mut Cursor::new([0x01, 0xFF]), None);
|
||||||
|
|
||||||
let bytes = read_path("tests/files/assets/rev_frame_sync_search.mp3");
|
let bytes = read_path("tests/files/assets/rev_frame_sync_search.mp3");
|
||||||
let mut reader = Cursor::new(bytes);
|
let mut reader = Cursor::new(bytes);
|
||||||
test(&mut reader, Some(283));
|
test(&mut reader, Some(595));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,14 +1,13 @@
|
||||||
use super::header::{ChannelMode, Emphasis, Header, Layer, MpegVersion, XingHeader};
|
use super::header::{ChannelMode, Emphasis, Header, Layer, MpegVersion, VbrHeader};
|
||||||
|
use crate::config::ParsingMode;
|
||||||
use crate::error::Result;
|
use crate::error::Result;
|
||||||
use crate::mpeg::header::{cmp_header, rev_search_for_frame_sync, HeaderCmpResult, HEADER_MASK};
|
use crate::mpeg::header::rev_search_for_frame_header;
|
||||||
use crate::properties::{ChannelMask, FileProperties};
|
use crate::properties::{ChannelMask, FileProperties};
|
||||||
use crate::util::math::RoundedDivision;
|
use crate::util::math::RoundedDivision;
|
||||||
|
|
||||||
use std::io::{Read, Seek, SeekFrom};
|
use std::io::{Read, Seek, SeekFrom};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use byteorder::{BigEndian, ReadBytesExt};
|
|
||||||
|
|
||||||
/// An MPEG file's audio properties
|
/// An MPEG file's audio properties
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
|
||||||
#[non_exhaustive]
|
#[non_exhaustive]
|
||||||
|
@ -127,8 +126,9 @@ pub(super) fn read_properties<R>(
|
||||||
reader: &mut R,
|
reader: &mut R,
|
||||||
first_frame: (Header, u64),
|
first_frame: (Header, u64),
|
||||||
mut last_frame_offset: u64,
|
mut last_frame_offset: u64,
|
||||||
xing_header: Option<XingHeader>,
|
xing_header: Option<VbrHeader>,
|
||||||
file_length: u64,
|
file_length: u64,
|
||||||
|
parse_mode: ParsingMode,
|
||||||
) -> Result<()>
|
) -> Result<()>
|
||||||
where
|
where
|
||||||
R: Read + Seek,
|
R: Read + Seek,
|
||||||
|
@ -177,29 +177,17 @@ where
|
||||||
reader.seek(SeekFrom::Start(last_frame_offset))?;
|
reader.seek(SeekFrom::Start(last_frame_offset))?;
|
||||||
|
|
||||||
let mut last_frame = None;
|
let mut last_frame = None;
|
||||||
let mut pos = reader.stream_position()?;
|
let mut pos = last_frame_offset;
|
||||||
while pos > 0 {
|
while pos > 0 {
|
||||||
match rev_search_for_frame_sync(reader, &mut pos) {
|
match rev_search_for_frame_header(reader, &mut pos, parse_mode) {
|
||||||
// Found a frame sync, attempt to read a header
|
// Found a frame header
|
||||||
Ok(Some(_)) => {
|
Ok(Some(header)) => {
|
||||||
// Move `last_frame_offset` back to the actual position
|
// Move `last_frame_offset` back to the actual position
|
||||||
last_frame_offset = reader.stream_position()?;
|
last_frame_offset = pos;
|
||||||
let last_frame_data = reader.read_u32::<BigEndian>()?;
|
|
||||||
|
|
||||||
if let Some(last_frame_header) = Header::read(last_frame_data) {
|
if header.cmp(&first_frame_header) {
|
||||||
match cmp_header(
|
last_frame = Some(header);
|
||||||
reader,
|
break;
|
||||||
4,
|
|
||||||
last_frame_header.len,
|
|
||||||
last_frame_data,
|
|
||||||
HEADER_MASK,
|
|
||||||
) {
|
|
||||||
HeaderCmpResult::Equal | HeaderCmpResult::Undetermined => {
|
|
||||||
last_frame = Some(last_frame_header);
|
|
||||||
break;
|
|
||||||
},
|
|
||||||
HeaderCmpResult::NotEqual => {},
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
// Encountered some IO error, just break
|
// Encountered some IO error, just break
|
||||||
|
@ -211,11 +199,11 @@ where
|
||||||
|
|
||||||
if let Some(last_frame_header) = last_frame {
|
if let Some(last_frame_header) = last_frame {
|
||||||
let stream_len = last_frame_offset - first_frame_offset + u64::from(last_frame_header.len);
|
let stream_len = last_frame_offset - first_frame_offset + u64::from(last_frame_header.len);
|
||||||
let length = ((stream_len as f64) * 8.0) / f64::from(properties.audio_bitrate) + 0.5;
|
let length = (stream_len * 8).div_round(u64::from(properties.audio_bitrate));
|
||||||
|
|
||||||
if length > 0.0 {
|
if length > 0 {
|
||||||
properties.overall_bitrate = (((file_length as f64) * 8.0) / length) as u32;
|
properties.overall_bitrate = ((file_length * 8) / length) as u32;
|
||||||
properties.duration = Duration::from_millis(length as u64);
|
properties.duration = Duration::from_millis(length);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use super::header::{cmp_header, search_for_frame_sync, Header, HeaderCmpResult, XingHeader};
|
use super::header::{cmp_header, search_for_frame_sync, Header, HeaderCmpResult, VbrHeader};
|
||||||
use super::{MpegFile, MpegProperties};
|
use super::{MpegFile, MpegProperties};
|
||||||
use crate::ape::header::read_ape_header;
|
use crate::ape::header::read_ape_header;
|
||||||
use crate::config::{ParseOptions, ParsingMode};
|
use crate::config::{ParseOptions, ParsingMode};
|
||||||
|
@ -6,6 +6,7 @@ use crate::error::Result;
|
||||||
use crate::id3::v2::header::Id3v2Header;
|
use crate::id3::v2::header::Id3v2Header;
|
||||||
use crate::id3::v2::read::parse_id3v2;
|
use crate::id3::v2::read::parse_id3v2;
|
||||||
use crate::id3::{find_id3v1, find_lyrics3v2, FindId3v2Config, ID3FindResults};
|
use crate::id3::{find_id3v1, find_lyrics3v2, FindId3v2Config, ID3FindResults};
|
||||||
|
use crate::io::SeekStreamLen;
|
||||||
use crate::macros::{decode_err, err};
|
use crate::macros::{decode_err, err};
|
||||||
use crate::mpeg::header::HEADER_MASK;
|
use crate::mpeg::header::HEADER_MASK;
|
||||||
|
|
||||||
|
@ -182,9 +183,9 @@ where
|
||||||
let mut xing_reader = [0; 32];
|
let mut xing_reader = [0; 32];
|
||||||
reader.read_exact(&mut xing_reader)?;
|
reader.read_exact(&mut xing_reader)?;
|
||||||
|
|
||||||
let xing_header = XingHeader::read(&mut &xing_reader[..])?;
|
let xing_header = VbrHeader::read(&mut &xing_reader[..])?;
|
||||||
|
|
||||||
let file_length = reader.seek(SeekFrom::End(0))?;
|
let file_length = reader.stream_len_hack()?;
|
||||||
|
|
||||||
super::properties::read_properties(
|
super::properties::read_properties(
|
||||||
&mut file.properties,
|
&mut file.properties,
|
||||||
|
@ -193,6 +194,7 @@ where
|
||||||
last_frame_offset,
|
last_frame_offset,
|
||||||
xing_header,
|
xing_header,
|
||||||
file_length,
|
file_length,
|
||||||
|
parse_options.parsing_mode,
|
||||||
)?;
|
)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -75,7 +75,7 @@ const MP1_PROPERTIES: MpegProperties = MpegProperties {
|
||||||
copyright: false,
|
copyright: false,
|
||||||
original: true,
|
original: true,
|
||||||
duration: Duration::from_millis(588), // FFmpeg reports 576, possibly an issue
|
duration: Duration::from_millis(588), // FFmpeg reports 576, possibly an issue
|
||||||
overall_bitrate: 383, // TODO: FFmpeg reports 392
|
overall_bitrate: 384, // TODO: FFmpeg reports 392
|
||||||
audio_bitrate: 384,
|
audio_bitrate: 384,
|
||||||
sample_rate: 32000,
|
sample_rate: 32000,
|
||||||
channels: 2,
|
channels: 2,
|
||||||
|
@ -89,8 +89,8 @@ const MP2_PROPERTIES: MpegProperties = MpegProperties {
|
||||||
mode_extension: None,
|
mode_extension: None,
|
||||||
copyright: false,
|
copyright: false,
|
||||||
original: true,
|
original: true,
|
||||||
duration: Duration::from_millis(1344), // TODO: FFmpeg reports 1440 here
|
duration: Duration::from_millis(1440),
|
||||||
overall_bitrate: 411, // FFmpeg reports 384, related to above issue
|
overall_bitrate: 384,
|
||||||
audio_bitrate: 384,
|
audio_bitrate: 384,
|
||||||
sample_rate: 48000,
|
sample_rate: 48000,
|
||||||
channels: 2,
|
channels: 2,
|
||||||
|
|
Loading…
Reference in a new issue