/*****************************************************************************
* mp4.c : MP4 file input module for vlc
*****************************************************************************
- * Copyright (C) 2001-2004 the VideoLAN team
+ * Copyright (C) 2001-2004, 2010 the VideoLAN team
* $Id$
* Authors: Laurent Aimar <fenrir@via.ecp.fr>
*
#include "libmp4.h"
#include "drms.h"
+#include "../../meta_engine/id3genres.h"
/*****************************************************************************
* Module descriptor
p_demux->pf_control = Control;
/* create our structure that will contains all data */
- p_demux->p_sys = p_sys = malloc( sizeof( demux_sys_t ) );
- memset( p_sys, 0, sizeof( demux_sys_t ) );
+ p_demux->p_sys = p_sys = calloc( 1, sizeof( demux_sys_t ) );
/* Now load all boxes ( except raw data ) */
if( ( p_sys->p_root = MP4_BoxGetRoot( p_demux->s ) ) == NULL )
input_thread_t *p_input = demux_GetParentInput( p_demux );
input_item_t *p_current = input_GetItem( p_input );
+ input_item_node_t *p_subitems = input_item_node_Create( p_current );
+
for( i = 0; i < i_count; i++ )
{
MP4_Box_t *p_rdrf = MP4_BoxGet( p_rmra, "rmda[%d]/rdrf", i );
msg_Dbg( p_demux, "adding ref = `%s'", psz_ref );
input_item_t *p_input = input_item_New( p_demux, psz_ref, NULL );
input_item_CopyOptions( p_current, p_input );
- input_item_AddSubItem( p_current, p_input );
+ input_item_node_AppendItem( p_subitems, p_input );
vlc_gc_decref( p_input );
}
else
}
free( psz_ref );
}
+ input_item_node_PostAndDelete( p_subitems );
vlc_object_release( p_input );
}
SET( vlc_meta_SetGenre );
break;
+ case FOURCC_gnre:
+ if( p_0xa9xxx->data.p_gnre->i_genre <= NUM_GENRES )
+ vlc_meta_SetGenre( p_meta, ppsz_genres[p_0xa9xxx->data.p_gnre->i_genre - 1] );
+ break;
+
case FOURCC_0xa9alb: /* Album */
SET( vlc_meta_SetAlbum );
break;
case FOURCC_0xa9trk: /* Track */
SET( vlc_meta_SetTrackNum );
break;
-
+ case FOURCC_trkn:
+ {
+ char psz_trck[11];
+ snprintf( psz_trck, sizeof( psz_trck ), "%i",
+ p_0xa9xxx->data.p_trkn->i_track_number );
+ vlc_meta_SetTrackNum( p_meta, psz_trck );
+ break;
+ }
case FOURCC_0xa9cmt: /* Commment */
SET( vlc_meta_SetDescription );
break;
SET( vlc_meta_SetURL );
break;
+ case FOURCC_0xa9too: /* Encoder Tool */
case FOURCC_0xa9enc: /* Encoded By */
SET( vlc_meta_SetEncodedBy );
break;
-
- case FOURCC_0xa9swr:
- case FOURCC_0xa9inf: /* Information */
- case FOURCC_0xa9dir: /* Director */
- case FOURCC_0xa9dis: /* Disclaimer */
- case FOURCC_0xa9req: /* Requirements */
- case FOURCC_0xa9fmt: /* Original Format */
- case FOURCC_0xa9dsa: /* Display Source As */
- case FOURCC_0xa9hst: /* Host Computer */
- case FOURCC_0xa9prd: /* Producer */
- case FOURCC_0xa9prf: /* Performers */
- case FOURCC_0xa9ope: /* Original Performer */
- case FOURCC_0xa9src: /* Providers Source Content */
- case FOURCC_0xa9wrt: /* Writer */
- case FOURCC_0xa9com: /* Composer */
- case FOURCC_WLOC: /* Window Location */
- /* TODO one day, but they aren't really meaningfull */
- break;
-#undef SET
default:
break;
}
+#undef SET
+ static const struct { uint32_t xa9_type; char metadata[25]; } xa9typetoextrameta[] =
+ {
+ { FOURCC_0xa9wrt, N_("Writer") },
+ { FOURCC_0xa9com, N_("Composr") },
+ { FOURCC_0xa9prd, N_("Producer") },
+ { FOURCC_0xa9inf, N_("Information") },
+ { FOURCC_0xa9dir, N_("Director") },
+ { FOURCC_0xa9dis, N_("Disclaimer") },
+ { FOURCC_0xa9req, N_("Requirements") },
+ { FOURCC_0xa9fmt, N_("Original Format") },
+ { FOURCC_0xa9dsa, N_("Display Source As") },
+ { FOURCC_0xa9hst, N_("Host Computer") },
+ { FOURCC_0xa9prf, N_("Performers") },
+ { FOURCC_0xa9ope, N_("Original Performer") },
+ { FOURCC_0xa9src, N_("Providers Source Content") },
+ { FOURCC_0xa9wrn, N_("Warning") },
+ { FOURCC_0xa9swr, N_("Software") },
+ { FOURCC_0xa9lyr, N_("Lyrics") },
+ { FOURCC_0xa9mak, N_("Make") },
+ { FOURCC_0xa9mod, N_("Model") },
+ { FOURCC_0xa9PRD, N_("Product") },
+ { FOURCC_0xa9grp, N_("Grouping") },
+ { 0, "" },
+ };
+ for( unsigned i = 0; xa9typetoextrameta[i].xa9_type; i++ )
+ {
+ if( p_0xa9xxx->i_type == xa9typetoextrameta[i].xa9_type )
+ {
+ char *psz_utf = strdup( p_0xa9xxx->data.p_0xa9xxx->psz_text ? p_0xa9xxx->data.p_0xa9xxx->psz_text : "" );
+ if( psz_utf )
+ {
+ EnsureUTF8( psz_utf );
+ vlc_meta_AddExtra( p_meta, _(xa9typetoextrameta[i].metadata), psz_utf );
+ free( psz_utf );
+ }
+ break;
+ }
+ }
}
return VLC_SUCCESS;
}
MP4_Box_t *p_sample;
MP4_Box_t *p_esds;
MP4_Box_t *p_frma;
+ MP4_Box_t *p_enda;
if( pp_es )
*pp_es = NULL;
p_sample->i_type = p_frma->data.p_frma->i_type;
}
+ p_enda = MP4_BoxGet( p_sample, "wave/enda" );
+ if( !p_enda )
+ p_enda = MP4_BoxGet( p_sample, "enda" );
+
if( p_track->fmt.i_cat == AUDIO_ES && ( p_track->i_sample_size == 1 || p_track->i_sample_size == 2 ) )
{
MP4_Box_data_sample_soun_t *p_soun;
case VLC_FOURCC( 'a', 'l', 'a', 'w' ):
case VLC_FOURCC( 'u', 'l', 'a', 'w' ):
p_soun->i_samplesize = 8;
+ p_track->i_sample_size = p_soun->i_channelcount;
break;
case VLC_FOURCC( 'N', 'O', 'N', 'E' ):
case VLC_FOURCC( 'r', 'a', 'w', ' ' ):
p_soun->i_qt_version = 0;
}
}
+ else if( p_track->fmt.i_cat == AUDIO_ES && p_sample->data.p_sample_soun->i_qt_version == 1 )
+ {
+ MP4_Box_data_sample_soun_t *p_soun = p_sample->data.p_sample_soun;
+
+ switch( p_sample->i_type )
+ {
+ case( VLC_FOURCC( '.', 'm', 'p', '3' ) ):
+ case( VLC_FOURCC( 'm', 's', 0x00, 0x55 ) ):
+ {
+ if( p_track->i_sample_size > 1 )
+ p_soun->i_qt_version = 0;
+ break;
+ }
+ case( VLC_FOURCC( 'a', 'c', '-', '3' ) ):
+ case( VLC_FOURCC( 'e', 'c', '-', '3' ) ):
+ case( VLC_FOURCC( 'm', 's', 0x20, 0x00 ) ):
+ p_soun->i_qt_version = 0;
+ break;
+ default:
+ break;
+ }
+ }
/* */
switch( p_track->fmt.i_cat )
/* Work-around buggy muxed files */
p_sample->data.p_sample_vide->i_width != p_track->i_width )
{
- p_track->fmt.video.i_sar_num = p_track->i_width / p_track->fmt.video.i_height;
+ p_track->fmt.video.i_sar_num = p_track->i_width * p_track->fmt.video.i_height;
p_track->fmt.video.i_sar_den = p_track->i_height * p_track->fmt.video.i_width;
}
case( VLC_FOURCC( '.', 'm', 'p', '3' ) ):
case( VLC_FOURCC( 'm', 's', 0x00, 0x55 ) ):
{
- MP4_Box_data_sample_soun_t *p_soun = p_sample->data.p_sample_soun;
p_track->fmt.i_codec = VLC_FOURCC( 'm', 'p', 'g', 'a' );
- if( p_track->i_sample_size > 1 )
- p_soun->i_qt_version = 0;
break;
}
case( VLC_FOURCC( 'a', 'c', '-', '3' ) ):
break;
case VLC_FOURCC('i','n','2','4'):
- /* in in24/in32 there's enda-atom to tell it's little-endian (if present) */
- if( ( MP4_BoxGet( p_sample, "wave/enda" ) ) ||
- ( MP4_BoxGet( p_sample, "enda" ) ) )
- {
- p_track->fmt.i_codec = VLC_FOURCC('4','2','n','i');
- } else {
- p_track->fmt.i_codec = p_sample->i_type;
- }
+ p_track->fmt.i_codec = p_enda && p_enda->data.p_enda->i_little_endian == 1 ?
+ VLC_FOURCC('4','2','n','i') : VLC_FOURCC('i','n','2','4');
+ break;
+ case VLC_FOURCC('f','l','3','2'):
+ p_track->fmt.i_codec = p_enda && p_enda->data.p_enda->i_little_endian == 1 ?
+ VLC_CODEC_F32L : VLC_CODEC_F32B;
+ break;
+ case VLC_FOURCC('f','l','6','4'):
+ p_track->fmt.i_codec = p_enda && p_enda->data.p_enda->i_little_endian == 1 ?
+ VLC_CODEC_F64L : VLC_CODEC_F64B;
break;
default:
break;
case( 0x40):
p_track->fmt.i_codec = VLC_FOURCC( 'm','p','4','a' );
+ if( p_decconfig->i_decoder_specific_info_len >= 2 &&
+ p_decconfig->p_decoder_specific_info[0] == 0xF8 &&
+ (p_decconfig->p_decoder_specific_info[1]&0xE0) == 0x80 )
+ {
+ p_track->fmt.i_codec = VLC_CODEC_ALS;
+ }
break;
case( 0x60):
case( 0x61):
uint32_t *pi_sample )
{
demux_sys_t *p_sys = p_demux->p_sys;
- MP4_Box_t *p_stss;
+ MP4_Box_t *p_box_stss;
uint64_t i_dts;
unsigned int i_sample;
unsigned int i_chunk;
/* *** Try to find nearest sync points *** */
- if( ( p_stss = MP4_BoxGet( p_track->p_stbl, "stss" ) ) )
+ if( ( p_box_stss = MP4_BoxGet( p_track->p_stbl, "stss" ) ) )
{
- unsigned int i_index;
- msg_Dbg( p_demux,
- "track[Id 0x%x] using Sync Sample Box (stss)",
- p_track->i_track_ID );
- for( i_index = 0; i_index < p_stss->data.p_stss->i_entry_count; i_index++ )
+ MP4_Box_data_stss_t *p_stss = p_box_stss->data.p_stss;
+ msg_Dbg( p_demux, "track[Id 0x%x] using Sync Sample Box (stss)",
+ p_track->i_track_ID );
+ for( unsigned i_index = 0; i_index < p_stss->i_entry_count; i_index++ )
{
- if( p_stss->data.p_stss->i_sample_number[i_index] >= i_sample )
+ if( i_index >= p_stss->i_entry_count - 1 ||
+ i_sample < p_stss->i_sample_number[i_index+1] )
{
- if( i_index > 0 )
+ unsigned i_sync_sample = p_stss->i_sample_number[i_index];
+ msg_Dbg( p_demux, "stts gives %d --> %d (sample number)",
+ i_sample, i_sync_sample );
+
+ if( i_sync_sample <= i_sample )
{
- msg_Dbg( p_demux, "stts gives %d --> %d (sample number)",
- i_sample,
- p_stss->data.p_stss->i_sample_number[i_index-1] );
- i_sample = p_stss->data.p_stss->i_sample_number[i_index-1];
- /* new i_sample is less than old so i_chunk can only decreased */
while( i_chunk > 0 &&
- i_sample < p_track->chunk[i_chunk].i_sample_first )
- {
+ i_sync_sample < p_track->chunk[i_chunk].i_sample_first )
i_chunk--;
- }
}
else
{
- msg_Dbg( p_demux, "stts gives %d --> %d (sample number)",
- i_sample,
- p_stss->data.p_stss->i_sample_number[i_index] );
- i_sample = p_stss->data.p_stss->i_sample_number[i_index];
- /* new i_sample is more than old so i_chunk can only increased */
while( i_chunk < p_track->i_chunk_count - 1 &&
- i_sample >= p_track->chunk[i_chunk].i_sample_first +
- p_track->chunk[i_chunk].i_sample_count )
- {
+ i_sync_sample >= p_track->chunk[i_chunk].i_sample_first +
+ p_track->chunk[i_chunk].i_sample_count )
i_chunk++;
- }
}
+ i_sample = i_sync_sample;
break;
}
}