2006-07-25 16:30:14 +02:00
/*
* MXF demuxer .
* Copyright ( c ) 2006 SmartJog S . A . , Baptiste Coudurier < baptiste dot coudurier at smartjog dot com > .
*
2006-10-07 17:30:46 +02:00
* This file is part of FFmpeg .
*
* FFmpeg is free software ; you can redistribute it and / or
2006-07-25 16:30:14 +02:00
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
2006-10-07 17:30:46 +02:00
* version 2.1 of the License , or ( at your option ) any later version .
2006-07-25 16:30:14 +02:00
*
2006-10-07 17:30:46 +02:00
* FFmpeg is distributed in the hope that it will be useful ,
2006-07-25 16:30:14 +02:00
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
2006-10-07 17:30:46 +02:00
* License along with FFmpeg ; if not , write to the Free Software
2006-07-25 16:30:14 +02:00
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
/*
* References
* SMPTE 336 M KLV Data Encoding Protocol Using Key - Length - Value
* SMPTE 377 M MXF File Format Specifications
* SMPTE 378 M Operational Pattern 1 a
* SMPTE 379 M MXF Generic Container
* SMPTE 381 M Mapping MPEG Streams into the MXF Generic Container
* SMPTE 382 M Mapping AES3 and Broadcast Wave Audio into the MXF Generic Container
* SMPTE 383 M Mapping DV - DIF Data to the MXF Generic Container
*
* Principle
* Search for Track numbers which will identify essence element KLV packets .
* Search for SourcePackage which define tracks which contains Track numbers .
2006-07-29 00:42:31 +02:00
* Material Package contains tracks with reference to SourcePackage tracks .
2006-07-25 16:30:14 +02:00
* Search for Descriptors ( Picture , Sound ) which contains codec info and parameters .
* Assign Descriptors to correct Tracks .
*
2006-08-02 17:02:42 +02:00
* Metadata reading functions read Local Tags , get InstanceUID ( 0x3C0A ) then add MetaDataSet to MXFContext .
* Metadata parsing resolves Strong References to objects .
2006-07-29 00:42:31 +02:00
*
* Simple demuxer , only OP1A supported and some files might not work at all .
* Only tracks with associated descriptors will be decoded . " Highly Desirable " SMPTE 377 M D .1
2006-07-25 16:30:14 +02:00
*/
2006-08-03 13:59:38 +02:00
//#define DEBUG
2006-07-25 16:30:14 +02:00
2008-05-09 13:56:36 +02:00
# include "libavutil/aes.h"
# include "libavcodec/bytestream.h"
2006-07-25 16:30:14 +02:00
# include "avformat.h"
2006-07-29 15:13:08 +02:00
typedef uint8_t UID [ 16 ] ;
2006-07-28 20:18:12 +02:00
2006-08-02 17:02:42 +02:00
enum MXFMetadataSetType {
2007-01-14 01:32:25 +01:00
AnyType ,
2006-07-29 00:42:31 +02:00
MaterialPackage ,
SourcePackage ,
SourceClip ,
2006-08-02 17:02:42 +02:00
TimecodeComponent ,
Sequence ,
MultipleDescriptor ,
Descriptor ,
Track ,
2007-02-11 13:50:33 +01:00
CryptoContext ,
2006-07-29 00:42:31 +02:00
} ;
2007-06-04 00:56:11 +02:00
typedef struct {
2007-02-11 13:50:33 +01:00
UID uid ;
enum MXFMetadataSetType type ;
UID source_container_ul ;
} MXFCryptoContext ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID uid ;
2006-08-02 17:02:42 +02:00
enum MXFMetadataSetType type ;
2006-07-29 00:42:31 +02:00
UID source_package_uid ;
UID data_definition_ul ;
int64_t duration ;
int64_t start_position ;
int source_track_id ;
} MXFStructuralComponent ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID uid ;
2006-08-02 17:02:42 +02:00
enum MXFMetadataSetType type ;
2006-07-29 00:42:31 +02:00
UID data_definition_ul ;
UID * structural_components_refs ;
int structural_components_count ;
int64_t duration ;
} MXFSequence ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID uid ;
2006-08-02 17:02:42 +02:00
enum MXFMetadataSetType type ;
2006-07-29 00:42:31 +02:00
MXFSequence * sequence ; /* mandatory, and only one */
UID sequence_ref ;
2006-07-25 16:30:14 +02:00
int track_id ;
2006-07-29 15:13:08 +02:00
uint8_t track_number [ 4 ] ;
2006-07-29 00:42:31 +02:00
AVRational edit_rate ;
2006-07-25 16:30:14 +02:00
} MXFTrack ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID uid ;
2006-08-02 17:02:42 +02:00
enum MXFMetadataSetType type ;
2006-07-29 01:00:53 +02:00
UID essence_container_ul ;
UID essence_codec_ul ;
2006-07-25 16:30:14 +02:00
AVRational sample_rate ;
AVRational aspect_ratio ;
int width ;
int height ;
int channels ;
int bits_per_sample ;
2006-07-29 00:42:31 +02:00
UID * sub_descriptors_refs ;
int sub_descriptors_count ;
2006-07-25 16:30:14 +02:00
int linked_track_id ;
2006-08-03 14:31:15 +02:00
uint8_t * extradata ;
int extradata_size ;
2006-07-25 16:30:14 +02:00
} MXFDescriptor ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID uid ;
2006-08-02 17:02:42 +02:00
enum MXFMetadataSetType type ;
2006-07-29 00:42:31 +02:00
UID package_uid ;
UID * tracks_refs ;
2006-07-25 16:30:14 +02:00
int tracks_count ;
2006-07-29 00:42:31 +02:00
MXFDescriptor * descriptor ; /* only one */
UID descriptor_ref ;
} MXFPackage ;
2006-08-02 17:02:42 +02:00
typedef struct {
UID uid ;
enum MXFMetadataSetType type ;
} MXFMetadataSet ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID * packages_refs ;
int packages_count ;
2006-08-02 17:02:42 +02:00
MXFMetadataSet * * metadata_sets ;
int metadata_sets_count ;
2006-07-29 00:42:31 +02:00
AVFormatContext * fc ;
2007-02-11 13:50:33 +01:00
struct AVAES * aesc ;
2008-01-19 17:17:39 +01:00
uint8_t * local_tags ;
int local_tags_count ;
2006-07-25 16:30:14 +02:00
} MXFContext ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-28 20:18:12 +02:00
UID key ;
2006-07-25 16:30:14 +02:00
offset_t offset ;
uint64_t length ;
} KLVPacket ;
2006-08-03 13:21:54 +02:00
enum MXFWrappingScheme {
Frame ,
Clip ,
} ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-29 00:42:31 +02:00
UID uid ;
2008-01-19 14:16:00 +01:00
unsigned matching_len ;
2006-07-29 00:42:31 +02:00
enum CodecID id ;
} MXFCodecUL ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-09-29 14:23:38 +02:00
UID uid ;
enum CodecType type ;
} MXFDataDefinitionUL ;
2007-06-04 00:56:11 +02:00
typedef struct {
2006-07-31 17:26:33 +02:00
const UID key ;
2007-01-15 00:14:36 +01:00
int ( * read ) ( ) ;
int ctx_size ;
enum MXFMetadataSetType type ;
2006-07-31 17:26:33 +02:00
} MXFMetadataReadTableEntry ;
2006-07-28 20:18:12 +02:00
/* partial keys to match */
2006-07-29 01:10:03 +02:00
static const uint8_t mxf_header_partition_pack_key [ ] = { 0x06 , 0x0e , 0x2b , 0x34 , 0x02 , 0x05 , 0x01 , 0x01 , 0x0d , 0x01 , 0x02 , 0x01 , 0x01 , 0x02 } ;
2006-07-29 15:13:08 +02:00
static const uint8_t mxf_essence_element_key [ ] = { 0x06 , 0x0e , 0x2b , 0x34 , 0x01 , 0x02 , 0x01 , 0x01 , 0x0d , 0x01 , 0x03 , 0x01 } ;
2007-06-03 19:10:55 +02:00
static const uint8_t mxf_klv_key [ ] = { 0x06 , 0x0e , 0x2b , 0x34 } ;
2007-01-14 22:23:32 +01:00
/* complete keys to match */
2008-01-19 17:17:39 +01:00
static const uint8_t mxf_crypto_source_container_ul [ ] = { 0x06 , 0x0e , 0x2b , 0x34 , 0x01 , 0x01 , 0x01 , 0x09 , 0x06 , 0x01 , 0x01 , 0x02 , 0x02 , 0x00 , 0x00 , 0x00 } ;
2007-01-14 22:23:32 +01:00
static const uint8_t mxf_encrypted_triplet_key [ ] = { 0x06 , 0x0e , 0x2b , 0x34 , 0x02 , 0x04 , 0x01 , 0x07 , 0x0d , 0x01 , 0x03 , 0x01 , 0x02 , 0x7e , 0x01 , 0x00 } ;
2007-02-11 13:50:33 +01:00
static const uint8_t mxf_encrypted_essence_container [ ] = { 0x06 , 0x0e , 0x2b , 0x34 , 0x04 , 0x01 , 0x01 , 0x07 , 0x0d , 0x01 , 0x03 , 0x01 , 0x02 , 0x0b , 0x01 , 0x00 } ;
2008-01-19 20:58:48 +01:00
static const uint8_t mxf_sony_mpeg4_extradata [ ] = { 0x06 , 0x0e , 0x2b , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x0e , 0x06 , 0x06 , 0x02 , 0x02 , 0x01 , 0x00 , 0x00 } ;
2006-07-25 16:30:14 +02:00
# define IS_KLV_KEY(x, y) (!memcmp(x, y, sizeof(y)))
2007-03-12 13:36:41 +01:00
# define PRINT_KEY(pc, s, x) dprintf(pc, "%s %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X\n", s, \
2006-07-30 17:30:00 +02:00
( x ) [ 0 ] , ( x ) [ 1 ] , ( x ) [ 2 ] , ( x ) [ 3 ] , ( x ) [ 4 ] , ( x ) [ 5 ] , ( x ) [ 6 ] , ( x ) [ 7 ] , ( x ) [ 8 ] , ( x ) [ 9 ] , ( x ) [ 10 ] , ( x ) [ 11 ] , ( x ) [ 12 ] , ( x ) [ 13 ] , ( x ) [ 14 ] , ( x ) [ 15 ] )
2006-07-25 16:30:14 +02:00
static int64_t klv_decode_ber_length ( ByteIOContext * pb )
{
2007-01-14 17:02:22 +01:00
uint64_t size = get_byte ( pb ) ;
if ( size & 0x80 ) { /* long form */
int bytes_num = size & 0x7f ;
2006-07-25 16:30:14 +02:00
/* SMPTE 379M 5.3.4 guarantee that bytes_num must not exceed 8 bytes */
if ( bytes_num > 8 )
return - 1 ;
2007-01-14 17:02:22 +01:00
size = 0 ;
2006-07-25 16:30:14 +02:00
while ( bytes_num - - )
size = size < < 8 | get_byte ( pb ) ;
}
return size ;
}
2007-06-03 19:10:55 +02:00
static int mxf_read_sync ( ByteIOContext * pb , const uint8_t * key , unsigned size )
{
int i , b ;
for ( i = 0 ; i < size & & ! url_feof ( pb ) ; i + + ) {
b = get_byte ( pb ) ;
if ( b = = key [ 0 ] )
i = 0 ;
else if ( b ! = key [ i ] )
i = - 1 ;
}
return i = = size ;
}
2006-07-25 16:30:14 +02:00
static int klv_read_packet ( KLVPacket * klv , ByteIOContext * pb )
{
2007-06-03 19:10:55 +02:00
if ( ! mxf_read_sync ( pb , mxf_klv_key , 4 ) )
return - 1 ;
klv - > offset = url_ftell ( pb ) - 4 ;
memcpy ( klv - > key , mxf_klv_key , 4 ) ;
get_buffer ( pb , klv - > key + 4 , 12 ) ;
2006-07-25 16:30:14 +02:00
klv - > length = klv_decode_ber_length ( pb ) ;
2006-07-29 01:07:03 +02:00
return klv - > length = = - 1 ? - 1 : 0 ;
2006-07-25 16:30:14 +02:00
}
static int mxf_get_stream_index ( AVFormatContext * s , KLVPacket * klv )
{
int i ;
for ( i = 0 ; i < s - > nb_streams ; i + + ) {
2006-07-29 15:13:08 +02:00
MXFTrack * track = s - > streams [ i ] - > priv_data ;
2007-01-15 00:16:20 +01:00
/* SMPTE 379M 7.3 */
2006-07-29 15:13:08 +02:00
if ( ! memcmp ( klv - > key + sizeof ( mxf_essence_element_key ) , track - > track_number , sizeof ( track - > track_number ) ) )
2006-07-25 16:30:14 +02:00
return i ;
}
2006-10-02 15:22:51 +02:00
/* return 0 if only one stream, for OP Atom files with 0 as track number */
return s - > nb_streams = = 1 ? 0 : - 1 ;
2006-07-25 16:30:14 +02:00
}
2006-10-05 11:01:04 +02:00
/* XXX: use AVBitStreamFilter */
static int mxf_get_d10_aes3_packet ( ByteIOContext * pb , AVStream * st , AVPacket * pkt , int64_t length )
{
uint8_t buffer [ 61444 ] ;
2008-02-01 15:58:35 +01:00
const uint8_t * buf_ptr , * end_ptr ;
uint8_t * data_ptr ;
2007-09-21 18:28:17 +02:00
int i ;
2006-10-05 11:01:04 +02:00
if ( length > 61444 ) /* worst case PAL 1920 samples 8 channels */
return - 1 ;
get_buffer ( pb , buffer , length ) ;
av_new_packet ( pkt , length ) ;
data_ptr = pkt - > data ;
end_ptr = buffer + length ;
buf_ptr = buffer + 4 ; /* skip SMPTE 331M header */
2007-09-21 18:28:17 +02:00
for ( ; buf_ptr < end_ptr ; ) {
for ( i = 0 ; i < st - > codec - > channels ; i + + ) {
uint32_t sample = bytestream_get_le32 ( & buf_ptr ) ;
if ( st - > codec - > bits_per_sample = = 24 )
bytestream_put_le24 ( & data_ptr , ( sample > > 4 ) & 0xffffff ) ;
else
bytestream_put_le16 ( & data_ptr , ( sample > > 12 ) & 0xffff ) ;
2006-10-05 11:01:04 +02:00
}
2007-09-21 18:28:17 +02:00
buf_ptr + = 32 - st - > codec - > channels * 4 ; // always 8 channels stored SMPTE 331M
2006-10-05 11:01:04 +02:00
}
pkt - > size = data_ptr - pkt - > data ;
return 0 ;
}
2007-02-11 13:50:33 +01:00
static int mxf_decrypt_triplet ( AVFormatContext * s , AVPacket * pkt , KLVPacket * klv )
{
static const uint8_t checkv [ 16 ] = { 0x43 , 0x48 , 0x55 , 0x4b , 0x43 , 0x48 , 0x55 , 0x4b , 0x43 , 0x48 , 0x55 , 0x4b , 0x43 , 0x48 , 0x55 , 0x4b } ;
MXFContext * mxf = s - > priv_data ;
2007-11-21 08:41:00 +01:00
ByteIOContext * pb = s - > pb ;
2007-02-11 13:50:33 +01:00
offset_t end = url_ftell ( pb ) + klv - > length ;
uint64_t size ;
uint64_t orig_size ;
uint64_t plaintext_size ;
uint8_t ivec [ 16 ] ;
uint8_t tmpbuf [ 16 ] ;
int index ;
if ( ! mxf - > aesc & & s - > key & & s - > keylen = = 16 ) {
mxf - > aesc = av_malloc ( av_aes_size ) ;
2008-01-19 16:21:30 +01:00
if ( ! mxf - > aesc )
return - 1 ;
2007-02-11 13:50:33 +01:00
av_aes_init ( mxf - > aesc , s - > key , 128 , 1 ) ;
}
// crypto context
url_fskip ( pb , klv_decode_ber_length ( pb ) ) ;
// plaintext offset
klv_decode_ber_length ( pb ) ;
plaintext_size = get_be64 ( pb ) ;
// source klv key
klv_decode_ber_length ( pb ) ;
get_buffer ( pb , klv - > key , 16 ) ;
2007-06-03 20:04:33 +02:00
if ( ! IS_KLV_KEY ( klv , mxf_essence_element_key ) )
return - 1 ;
2007-02-11 13:50:33 +01:00
index = mxf_get_stream_index ( s , klv ) ;
2007-06-03 20:04:33 +02:00
if ( index < 0 )
return - 1 ;
2007-02-11 13:50:33 +01:00
// source size
klv_decode_ber_length ( pb ) ;
orig_size = get_be64 ( pb ) ;
2007-06-03 20:04:33 +02:00
if ( orig_size < plaintext_size )
return - 1 ;
2007-02-11 13:50:33 +01:00
// enc. code
size = klv_decode_ber_length ( pb ) ;
2007-06-03 20:04:33 +02:00
if ( size < 32 | | size - 32 < orig_size )
return - 1 ;
2007-02-11 13:50:33 +01:00
get_buffer ( pb , ivec , 16 ) ;
get_buffer ( pb , tmpbuf , 16 ) ;
if ( mxf - > aesc )
av_aes_crypt ( mxf - > aesc , tmpbuf , tmpbuf , 1 , ivec , 1 ) ;
if ( memcmp ( tmpbuf , checkv , 16 ) )
av_log ( s , AV_LOG_ERROR , " probably incorrect decryption key \n " ) ;
size - = 32 ;
av_get_packet ( pb , pkt , size ) ;
size - = plaintext_size ;
if ( mxf - > aesc )
av_aes_crypt ( mxf - > aesc , & pkt - > data [ plaintext_size ] ,
& pkt - > data [ plaintext_size ] , size > > 4 , ivec , 1 ) ;
pkt - > size = orig_size ;
pkt - > stream_index = index ;
url_fskip ( pb , end - url_ftell ( pb ) ) ;
return 0 ;
}
2006-07-25 16:30:14 +02:00
static int mxf_read_packet ( AVFormatContext * s , AVPacket * pkt )
{
KLVPacket klv ;
2007-11-21 08:41:00 +01:00
while ( ! url_feof ( s - > pb ) ) {
if ( klv_read_packet ( & klv , s - > pb ) < 0 )
2006-07-25 16:30:14 +02:00
return - 1 ;
2006-07-29 15:18:24 +02:00
# ifdef DEBUG
2007-03-12 13:36:41 +01:00
PRINT_KEY ( s , " read packet " , klv . key ) ;
2006-07-29 15:18:24 +02:00
# endif
2007-01-14 22:23:32 +01:00
if ( IS_KLV_KEY ( klv . key , mxf_encrypted_triplet_key ) ) {
2007-02-11 13:50:33 +01:00
int res = mxf_decrypt_triplet ( s , pkt , & klv ) ;
if ( res < 0 ) {
av_log ( s , AV_LOG_ERROR , " invalid encoded triplet \n " ) ;
return - 1 ;
}
return 0 ;
2007-01-14 22:23:32 +01:00
}
2006-07-25 16:30:14 +02:00
if ( IS_KLV_KEY ( klv . key , mxf_essence_element_key ) ) {
2006-10-05 11:01:04 +02:00
int index = mxf_get_stream_index ( s , & klv ) ;
if ( index < 0 ) {
av_log ( s , AV_LOG_ERROR , " error getting stream index \n " ) ;
2008-03-16 18:54:36 +01:00
goto skip ;
2006-10-05 11:01:04 +02:00
}
2008-03-06 14:40:29 +01:00
if ( s - > streams [ index ] - > discard = = AVDISCARD_ALL )
goto skip ;
2006-10-05 11:01:04 +02:00
/* check for 8 channels AES3 element */
if ( klv . key [ 12 ] = = 0x06 & & klv . key [ 13 ] = = 0x01 & & klv . key [ 14 ] = = 0x10 ) {
2007-11-21 08:41:00 +01:00
if ( mxf_get_d10_aes3_packet ( s - > pb , s - > streams [ index ] , pkt , klv . length ) < 0 ) {
2006-10-05 11:01:04 +02:00
av_log ( s , AV_LOG_ERROR , " error reading D-10 aes3 frame \n " ) ;
return - 1 ;
}
} else
2007-11-21 08:41:00 +01:00
av_get_packet ( s - > pb , pkt , klv . length ) ;
2006-10-05 11:01:04 +02:00
pkt - > stream_index = index ;
2007-06-03 19:29:49 +02:00
pkt - > pos = klv . offset ;
2006-10-05 11:01:04 +02:00
return 0 ;
2006-07-25 16:30:14 +02:00
} else
2008-03-06 14:40:29 +01:00
skip :
2007-11-21 08:41:00 +01:00
url_fskip ( s - > pb , klv . length ) ;
2006-07-25 16:30:14 +02:00
}
2007-07-19 17:23:32 +02:00
return AVERROR ( EIO ) ;
2006-07-25 16:30:14 +02:00
}
2008-01-19 17:17:39 +01:00
static int mxf_read_primer_pack ( MXFContext * mxf )
{
ByteIOContext * pb = mxf - > fc - > pb ;
int item_num = get_be32 ( pb ) ;
int item_len = get_be32 ( pb ) ;
if ( item_len ! = 18 ) {
av_log ( mxf - > fc , AV_LOG_ERROR , " unsupported primer pack item length \n " ) ;
return - 1 ;
}
if ( item_num > UINT_MAX / item_len )
return - 1 ;
mxf - > local_tags_count = item_num ;
mxf - > local_tags = av_malloc ( item_num * item_len ) ;
if ( ! mxf - > local_tags )
return - 1 ;
get_buffer ( pb , mxf - > local_tags , item_num * item_len ) ;
return 0 ;
}
2006-08-02 17:02:42 +02:00
static int mxf_add_metadata_set ( MXFContext * mxf , void * metadata_set )
{
mxf - > metadata_sets = av_realloc ( mxf - > metadata_sets , ( mxf - > metadata_sets_count + 1 ) * sizeof ( * mxf - > metadata_sets ) ) ;
2008-01-19 16:21:30 +01:00
if ( ! mxf - > metadata_sets )
return - 1 ;
2006-08-02 17:02:42 +02:00
mxf - > metadata_sets [ mxf - > metadata_sets_count ] = metadata_set ;
mxf - > metadata_sets_count + + ;
return 0 ;
}
2008-01-19 17:20:06 +01:00
static int mxf_read_cryptographic_context ( MXFCryptoContext * cryptocontext , ByteIOContext * pb , int tag , int size , UID uid )
2007-02-11 13:50:33 +01:00
{
2008-01-19 17:17:39 +01:00
if ( size ! = 16 )
return - 1 ;
2008-01-19 17:18:32 +01:00
if ( IS_KLV_KEY ( uid , mxf_crypto_source_container_ul ) )
2007-02-11 13:50:33 +01:00
get_buffer ( pb , cryptocontext - > source_container_ul , 16 ) ;
return 0 ;
}
2008-01-19 17:20:06 +01:00
static int mxf_read_content_storage ( MXFContext * mxf , ByteIOContext * pb , int tag )
2006-07-29 00:42:31 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
case 0x1901 :
mxf - > packages_count = get_be32 ( pb ) ;
if ( mxf - > packages_count > = UINT_MAX / sizeof ( UID ) )
return - 1 ;
mxf - > packages_refs = av_malloc ( mxf - > packages_count * sizeof ( UID ) ) ;
2008-01-19 16:21:30 +01:00
if ( ! mxf - > packages_refs )
return - 1 ;
2007-01-15 00:16:20 +01:00
url_fskip ( pb , 4 ) ; /* useless size of objects, always 16 according to specs */
get_buffer ( pb , ( uint8_t * ) mxf - > packages_refs , mxf - > packages_count * sizeof ( UID ) ) ;
break ;
}
2006-07-29 00:42:31 +02:00
return 0 ;
}
2008-01-19 17:20:06 +01:00
static int mxf_read_source_clip ( MXFStructuralComponent * source_clip , ByteIOContext * pb , int tag )
2006-07-29 00:42:31 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
case 0x0202 :
source_clip - > duration = get_be64 ( pb ) ;
break ;
case 0x1201 :
source_clip - > start_position = get_be64 ( pb ) ;
break ;
case 0x1101 :
/* UMID, only get last 16 bytes */
url_fskip ( pb , 16 ) ;
get_buffer ( pb , source_clip - > source_package_uid , 16 ) ;
break ;
case 0x1102 :
source_clip - > source_track_id = get_be32 ( pb ) ;
break ;
}
return 0 ;
2006-07-29 00:42:31 +02:00
}
2008-01-19 17:20:06 +01:00
static int mxf_read_material_package ( MXFPackage * package , ByteIOContext * pb , int tag )
2006-07-29 00:42:31 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
case 0x4403 :
package - > tracks_count = get_be32 ( pb ) ;
if ( package - > tracks_count > = UINT_MAX / sizeof ( UID ) )
return - 1 ;
package - > tracks_refs = av_malloc ( package - > tracks_count * sizeof ( UID ) ) ;
2008-01-19 16:21:30 +01:00
if ( ! package - > tracks_refs )
return - 1 ;
2007-01-15 00:16:20 +01:00
url_fskip ( pb , 4 ) ; /* useless size of objects, always 16 according to specs */
get_buffer ( pb , ( uint8_t * ) package - > tracks_refs , package - > tracks_count * sizeof ( UID ) ) ;
break ;
}
return 0 ;
2006-07-29 00:42:31 +02:00
}
2008-01-19 17:20:06 +01:00
static int mxf_read_track ( MXFTrack * track , ByteIOContext * pb , int tag )
2006-07-29 00:42:31 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
case 0x4801 :
track - > track_id = get_be32 ( pb ) ;
break ;
case 0x4804 :
get_buffer ( pb , track - > track_number , 4 ) ;
break ;
case 0x4B01 :
track - > edit_rate . den = get_be32 ( pb ) ;
track - > edit_rate . num = get_be32 ( pb ) ;
break ;
case 0x4803 :
get_buffer ( pb , track - > sequence_ref , 16 ) ;
break ;
}
return 0 ;
2006-07-25 16:30:14 +02:00
}
2008-01-19 17:20:06 +01:00
static int mxf_read_sequence ( MXFSequence * sequence , ByteIOContext * pb , int tag )
2006-07-25 16:30:14 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
case 0x0202 :
sequence - > duration = get_be64 ( pb ) ;
break ;
case 0x0201 :
get_buffer ( pb , sequence - > data_definition_ul , 16 ) ;
break ;
case 0x1001 :
sequence - > structural_components_count = get_be32 ( pb ) ;
if ( sequence - > structural_components_count > = UINT_MAX / sizeof ( UID ) )
return - 1 ;
sequence - > structural_components_refs = av_malloc ( sequence - > structural_components_count * sizeof ( UID ) ) ;
2008-01-19 16:21:30 +01:00
if ( ! sequence - > structural_components_refs )
return - 1 ;
2007-01-15 00:16:20 +01:00
url_fskip ( pb , 4 ) ; /* useless size of objects, always 16 according to specs */
get_buffer ( pb , ( uint8_t * ) sequence - > structural_components_refs , sequence - > structural_components_count * sizeof ( UID ) ) ;
break ;
}
return 0 ;
2006-07-25 16:30:14 +02:00
}
2008-01-19 17:20:06 +01:00
static int mxf_read_source_package ( MXFPackage * package , ByteIOContext * pb , int tag )
2006-07-25 16:30:14 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
case 0x4403 :
package - > tracks_count = get_be32 ( pb ) ;
if ( package - > tracks_count > = UINT_MAX / sizeof ( UID ) )
return - 1 ;
package - > tracks_refs = av_malloc ( package - > tracks_count * sizeof ( UID ) ) ;
2008-01-19 16:21:30 +01:00
if ( ! package - > tracks_refs )
return - 1 ;
2007-01-15 00:16:20 +01:00
url_fskip ( pb , 4 ) ; /* useless size of objects, always 16 according to specs */
get_buffer ( pb , ( uint8_t * ) package - > tracks_refs , package - > tracks_count * sizeof ( UID ) ) ;
break ;
case 0x4401 :
/* UMID, only get last 16 bytes */
url_fskip ( pb , 16 ) ;
get_buffer ( pb , package - > package_uid , 16 ) ;
break ;
case 0x4701 :
get_buffer ( pb , package - > descriptor_ref , 16 ) ;
break ;
}
return 0 ;
2006-07-25 16:30:14 +02:00
}
2008-01-19 17:20:06 +01:00
static void mxf_read_pixel_layout ( ByteIOContext * pb , MXFDescriptor * descriptor )
2006-07-29 17:19:31 +02:00
{
int code ;
do {
code = get_byte ( pb ) ;
2007-03-12 13:36:41 +01:00
dprintf ( NULL , " pixel layout: code 0x%x \n " , code ) ;
2006-07-29 17:19:31 +02:00
switch ( code ) {
case 0x52 : /* R */
descriptor - > bits_per_sample + = get_byte ( pb ) ;
break ;
case 0x47 : /* G */
descriptor - > bits_per_sample + = get_byte ( pb ) ;
break ;
case 0x42 : /* B */
descriptor - > bits_per_sample + = get_byte ( pb ) ;
break ;
default :
get_byte ( pb ) ;
}
} while ( code ! = 0 ) ; /* SMPTE 377M E.2.46 */
}
2008-01-19 20:58:48 +01:00
static int mxf_read_generic_descriptor ( MXFDescriptor * descriptor , ByteIOContext * pb , int tag , int size , UID uid )
2006-07-29 00:42:31 +02:00
{
2007-01-15 00:16:20 +01:00
switch ( tag ) {
2007-01-15 00:23:06 +01:00
case 0x3F01 :
descriptor - > sub_descriptors_count = get_be32 ( pb ) ;
if ( descriptor - > sub_descriptors_count > = UINT_MAX / sizeof ( UID ) )
return - 1 ;
descriptor - > sub_descriptors_refs = av_malloc ( descriptor - > sub_descriptors_count * sizeof ( UID ) ) ;
2008-01-19 16:21:30 +01:00
if ( ! descriptor - > sub_descriptors_refs )
return - 1 ;
2007-01-15 00:23:06 +01:00
url_fskip ( pb , 4 ) ; /* useless size of objects, always 16 according to specs */
get_buffer ( pb , ( uint8_t * ) descriptor - > sub_descriptors_refs , descriptor - > sub_descriptors_count * sizeof ( UID ) ) ;
break ;
2007-01-15 00:16:20 +01:00
case 0x3004 :
get_buffer ( pb , descriptor - > essence_container_ul , 16 ) ;
break ;
case 0x3006 :
descriptor - > linked_track_id = get_be32 ( pb ) ;
break ;
case 0x3201 : /* PictureEssenceCoding */
get_buffer ( pb , descriptor - > essence_codec_ul , 16 ) ;
break ;
case 0x3203 :
descriptor - > width = get_be32 ( pb ) ;
break ;
case 0x3202 :
descriptor - > height = get_be32 ( pb ) ;
break ;
case 0x320E :
descriptor - > aspect_ratio . num = get_be32 ( pb ) ;
descriptor - > aspect_ratio . den = get_be32 ( pb ) ;
break ;
case 0x3D03 :
descriptor - > sample_rate . num = get_be32 ( pb ) ;
descriptor - > sample_rate . den = get_be32 ( pb ) ;
break ;
case 0x3D06 : /* SoundEssenceCompression */
get_buffer ( pb , descriptor - > essence_codec_ul , 16 ) ;
break ;
case 0x3D07 :
descriptor - > channels = get_be32 ( pb ) ;
break ;
case 0x3D01 :
descriptor - > bits_per_sample = get_be32 ( pb ) ;
break ;
case 0x3401 :
2008-01-19 17:20:06 +01:00
mxf_read_pixel_layout ( pb , descriptor ) ;
2007-01-15 00:16:20 +01:00
break ;
2008-01-19 20:58:48 +01:00
default :
/* Private uid used by SONY C0023S01.mxf */
if ( IS_KLV_KEY ( uid , mxf_sony_mpeg4_extradata ) ) {
2008-01-19 20:59:04 +01:00
descriptor - > extradata = av_malloc ( size ) ;
if ( ! descriptor - > extradata )
return - 1 ;
descriptor - > extradata_size = size ;
get_buffer ( pb , descriptor - > extradata , size ) ;
2008-01-19 20:58:48 +01:00
}
2007-01-15 00:16:20 +01:00
break ;
}
return 0 ;
2006-07-25 16:30:14 +02:00
}
/* SMPTE RP224 http://www.smpte-ra.org/mdd/index.html */
2006-09-29 14:23:38 +02:00
static const MXFDataDefinitionUL mxf_data_definition_uls [ ] = {
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x01 , 0x03 , 0x02 , 0x02 , 0x01 , 0x00 , 0x00 , 0x00 } , CODEC_TYPE_VIDEO } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x01 , 0x03 , 0x02 , 0x02 , 0x02 , 0x00 , 0x00 , 0x00 } , CODEC_TYPE_AUDIO } ,
2006-09-29 14:25:44 +02:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x05 , 0x01 , 0x03 , 0x02 , 0x02 , 0x02 , 0x02 , 0x00 , 0x00 } , CODEC_TYPE_AUDIO } ,
2006-09-29 14:23:38 +02:00
{ { 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 } , CODEC_TYPE_DATA } ,
} ;
2006-07-29 00:42:31 +02:00
static const MXFCodecUL mxf_codec_uls [ ] = {
/* PictureEssenceCoding */
2008-01-19 18:25:12 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x03 , 0x04 , 0x01 , 0x02 , 0x02 , 0x01 , 0x01 , 0x11 , 0x00 } , 14 , CODEC_ID_MPEG2VIDEO } , /* MP@ML Long GoP */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x01 , 0x02 , 0x02 , 0x01 , 0x02 , 0x01 , 0x01 } , 14 , CODEC_ID_MPEG2VIDEO } , /* D-10 50Mbps PAL */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x03 , 0x04 , 0x01 , 0x02 , 0x02 , 0x01 , 0x03 , 0x03 , 0x00 } , 14 , CODEC_ID_MPEG2VIDEO } , /* MP@HL Long GoP */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x03 , 0x04 , 0x01 , 0x02 , 0x02 , 0x01 , 0x04 , 0x02 , 0x00 } , 14 , CODEC_ID_MPEG2VIDEO } , /* 422P@HL I-Frame */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x03 , 0x04 , 0x01 , 0x02 , 0x02 , 0x01 , 0x20 , 0x02 , 0x03 } , 14 , CODEC_ID_MPEG4 } , /* XDCAM proxy_pal030926.mxf */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x01 , 0x02 , 0x02 , 0x02 , 0x01 , 0x02 , 0x00 } , 13 , CODEC_ID_DVVIDEO } , /* DV25 IEC PAL */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x07 , 0x04 , 0x01 , 0x02 , 0x02 , 0x03 , 0x01 , 0x01 , 0x00 } , 14 , CODEC_ID_JPEG2000 } , /* JPEG2000 Codestream */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x01 , 0x02 , 0x01 , 0x7F , 0x00 , 0x00 , 0x00 } , 13 , CODEC_ID_RAWVIDEO } , /* Uncompressed */
2006-07-29 00:42:31 +02:00
/* SoundEssenceCompression */
2008-01-19 18:25:12 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x02 , 0x02 , 0x01 , 0x00 , 0x00 , 0x00 , 0x00 } , 13 , CODEC_ID_PCM_S16LE } , /* Uncompressed */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x02 , 0x02 , 0x01 , 0x7F , 0x00 , 0x00 , 0x00 } , 13 , CODEC_ID_PCM_S16LE } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x07 , 0x04 , 0x02 , 0x02 , 0x01 , 0x7E , 0x00 , 0x00 , 0x00 } , 13 , CODEC_ID_PCM_S16BE } , /* From Omneon MXF file */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x04 , 0x04 , 0x02 , 0x02 , 0x02 , 0x03 , 0x01 , 0x01 , 0x00 } , 15 , CODEC_ID_PCM_ALAW } , /* XDCAM Proxy C0023S01.mxf */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x02 , 0x02 , 0x02 , 0x03 , 0x02 , 0x01 , 0x00 } , 15 , CODEC_ID_AC3 } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x04 , 0x02 , 0x02 , 0x02 , 0x03 , 0x02 , 0x05 , 0x00 } , 15 , CODEC_ID_MP2 } , /* MP2 or MP3 */
//{ { 0x06,0x0E,0x2B,0x34,0x04,0x01,0x01,0x01,0x04,0x02,0x02,0x02,0x03,0x02,0x1C,0x00 }, 15, CODEC_ID_DOLBY_E }, /* Dolby-E */
{ { 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 } , 0 , CODEC_ID_NONE } ,
2006-07-25 16:30:14 +02:00
} ;
2006-08-03 11:14:56 +02:00
static const MXFCodecUL mxf_picture_essence_container_uls [ ] = {
2008-01-19 18:28:49 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x02 , 0x0D , 0x01 , 0x03 , 0x01 , 0x02 , 0x04 , 0x60 , 0x01 } , 14 , CODEC_ID_MPEG2VIDEO } , /* MPEG-ES Frame wrapped */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x0D , 0x01 , 0x03 , 0x01 , 0x02 , 0x02 , 0x41 , 0x01 } , 14 , CODEC_ID_DVVIDEO } , /* DV 625 25mbps */
{ { 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 } , 0 , CODEC_ID_NONE } ,
2006-08-03 11:14:56 +02:00
} ;
static const MXFCodecUL mxf_sound_essence_container_uls [ ] = {
2008-01-19 18:28:49 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x0D , 0x01 , 0x03 , 0x01 , 0x02 , 0x06 , 0x01 , 0x00 } , 14 , CODEC_ID_PCM_S16LE } , /* BWF Frame wrapped */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x02 , 0x0D , 0x01 , 0x03 , 0x01 , 0x02 , 0x04 , 0x40 , 0x01 } , 14 , CODEC_ID_MP2 } , /* MPEG-ES Frame wrapped, 0x40 ??? stream id */
2008-01-19 18:25:12 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x04 , 0x01 , 0x01 , 0x01 , 0x0D , 0x01 , 0x03 , 0x01 , 0x02 , 0x01 , 0x01 , 0x01 } , 14 , CODEC_ID_PCM_S16LE } , /* D-10 Mapping 50Mbps PAL Extended Template */
{ { 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 } , 0 , CODEC_ID_NONE } ,
2006-08-03 11:14:56 +02:00
} ;
2007-06-03 16:52:55 +02:00
/*
* Match an uid independently of the version byte and up to len common bytes
* Returns : boolean
*/
static int mxf_match_uid ( const UID key , const UID uid , int len )
{
int i ;
for ( i = 0 ; i < len ; i + + ) {
if ( i ! = 7 & & key [ i ] ! = uid [ i ] )
return 0 ;
}
return 1 ;
}
2006-08-03 13:21:54 +02:00
static const MXFCodecUL * mxf_get_codec_ul ( const MXFCodecUL * uls , UID * uid )
2006-07-29 00:42:31 +02:00
{
while ( uls - > id ! = CODEC_ID_NONE ) {
2008-01-19 14:16:00 +01:00
if ( mxf_match_uid ( uls - > uid , * uid , uls - > matching_len ) )
2006-08-03 13:21:54 +02:00
break ;
2006-07-29 00:42:31 +02:00
uls + + ;
}
2006-08-03 13:21:54 +02:00
return uls ;
2006-07-29 00:42:31 +02:00
}
2006-07-25 16:30:14 +02:00
2006-09-29 14:23:38 +02:00
static enum CodecType mxf_get_codec_type ( const MXFDataDefinitionUL * uls , UID * uid )
{
while ( uls - > type ! = CODEC_TYPE_DATA ) {
2007-06-03 16:52:55 +02:00
if ( mxf_match_uid ( uls - > uid , * uid , 16 ) )
2006-09-29 14:23:38 +02:00
break ;
uls + + ;
}
return uls - > type ;
}
2007-01-13 19:54:48 +01:00
static void * mxf_resolve_strong_ref ( MXFContext * mxf , UID * strong_ref , enum MXFMetadataSetType type )
2006-08-02 17:02:42 +02:00
{
int i ;
if ( ! strong_ref )
return NULL ;
for ( i = 0 ; i < mxf - > metadata_sets_count ; i + + ) {
2007-01-13 19:54:48 +01:00
if ( ! memcmp ( * strong_ref , mxf - > metadata_sets [ i ] - > uid , 16 ) & &
2007-01-14 01:32:25 +01:00
( type = = AnyType | | mxf - > metadata_sets [ i ] - > type = = type ) ) {
2006-08-02 17:02:42 +02:00
return mxf - > metadata_sets [ i ] ;
}
}
return NULL ;
}
2006-07-29 00:42:31 +02:00
static int mxf_parse_structural_metadata ( MXFContext * mxf )
2006-07-25 16:30:14 +02:00
{
2006-07-29 00:42:31 +02:00
MXFPackage * material_package = NULL ;
2006-08-02 17:02:42 +02:00
MXFPackage * temp_package = NULL ;
2006-07-29 00:42:31 +02:00
int i , j , k ;
2007-03-12 13:36:41 +01:00
dprintf ( mxf - > fc , " metadata sets count %d \n " , mxf - > metadata_sets_count ) ;
2006-07-29 00:42:31 +02:00
/* TODO: handle multiple material packages (OP3x) */
for ( i = 0 ; i < mxf - > packages_count ; i + + ) {
2007-01-13 19:54:48 +01:00
material_package = mxf_resolve_strong_ref ( mxf , & mxf - > packages_refs [ i ] , MaterialPackage ) ;
if ( material_package ) break ;
2006-07-29 00:42:31 +02:00
}
if ( ! material_package ) {
av_log ( mxf - > fc , AV_LOG_ERROR , " no material package found \n " ) ;
return - 1 ;
}
2006-07-25 16:30:14 +02:00
2006-07-29 00:42:31 +02:00
for ( i = 0 ; i < material_package - > tracks_count ; i + + ) {
2006-11-28 18:17:36 +01:00
MXFPackage * source_package = NULL ;
2006-08-02 17:02:42 +02:00
MXFTrack * material_track = NULL ;
2006-07-29 00:42:31 +02:00
MXFTrack * source_track = NULL ;
2006-08-02 17:02:42 +02:00
MXFTrack * temp_track = NULL ;
2006-07-29 00:42:31 +02:00
MXFDescriptor * descriptor = NULL ;
MXFStructuralComponent * component = NULL ;
2007-02-11 13:50:33 +01:00
UID * essence_container_ul = NULL ;
2006-08-03 13:21:54 +02:00
const MXFCodecUL * codec_ul = NULL ;
const MXFCodecUL * container_ul = NULL ;
2006-07-29 00:42:31 +02:00
AVStream * st ;
2007-01-13 19:54:48 +01:00
if ( ! ( material_track = mxf_resolve_strong_ref ( mxf , & material_package - > tracks_refs [ i ] , Track ) ) ) {
2006-08-02 17:02:42 +02:00
av_log ( mxf - > fc , AV_LOG_ERROR , " could not resolve material track strong ref \n " ) ;
continue ;
}
2007-01-13 19:54:48 +01:00
if ( ! ( material_track - > sequence = mxf_resolve_strong_ref ( mxf , & material_track - > sequence_ref , Sequence ) ) ) {
2006-08-02 17:02:42 +02:00
av_log ( mxf - > fc , AV_LOG_ERROR , " could not resolve material track sequence strong ref \n " ) ;
return - 1 ;
}
2006-07-29 00:42:31 +02:00
/* TODO: handle multiple source clips */
for ( j = 0 ; j < material_track - > sequence - > structural_components_count ; j + + ) {
/* TODO: handle timecode component */
2007-01-13 19:54:48 +01:00
component = mxf_resolve_strong_ref ( mxf , & material_track - > sequence - > structural_components_refs [ j ] , SourceClip ) ;
if ( ! component )
2006-07-29 00:42:31 +02:00
continue ;
for ( k = 0 ; k < mxf - > packages_count ; k + + ) {
2007-01-13 19:54:48 +01:00
temp_package = mxf_resolve_strong_ref ( mxf , & mxf - > packages_refs [ k ] , SourcePackage ) ;
if ( ! temp_package )
continue ;
2006-08-02 17:02:42 +02:00
if ( ! memcmp ( temp_package - > package_uid , component - > source_package_uid , 16 ) ) {
source_package = temp_package ;
2006-07-29 00:42:31 +02:00
break ;
}
}
if ( ! source_package ) {
av_log ( mxf - > fc , AV_LOG_ERROR , " material track %d: no corresponding source package found \n " , material_track - > track_id ) ;
break ;
}
for ( k = 0 ; k < source_package - > tracks_count ; k + + ) {
2007-01-13 19:54:48 +01:00
if ( ! ( temp_track = mxf_resolve_strong_ref ( mxf , & source_package - > tracks_refs [ k ] , Track ) ) ) {
2006-08-02 17:02:42 +02:00
av_log ( mxf - > fc , AV_LOG_ERROR , " could not resolve source track strong ref \n " ) ;
return - 1 ;
}
if ( temp_track - > track_id = = component - > source_track_id ) {
source_track = temp_track ;
2006-07-29 00:42:31 +02:00
break ;
}
}
if ( ! source_track ) {
av_log ( mxf - > fc , AV_LOG_ERROR , " material track %d: no corresponding source track found \n " , material_track - > track_id ) ;
break ;
}
}
if ( ! source_track )
continue ;
2006-07-29 15:13:08 +02:00
st = av_new_stream ( mxf - > fc , source_track - > track_id ) ;
2008-01-08 14:48:52 +01:00
if ( ! st ) {
av_log ( mxf - > fc , AV_LOG_ERROR , " could not allocate stream \n " ) ;
return - 1 ;
}
2006-07-29 15:13:08 +02:00
st - > priv_data = source_track ;
2006-07-29 00:42:31 +02:00
st - > duration = component - > duration ;
if ( st - > duration = = - 1 )
st - > duration = AV_NOPTS_VALUE ;
st - > start_time = component - > start_position ;
av_set_pts_info ( st , 64 , material_track - > edit_rate . num , material_track - > edit_rate . den ) ;
2006-08-02 17:02:42 +02:00
2007-01-13 19:54:48 +01:00
if ( ! ( source_track - > sequence = mxf_resolve_strong_ref ( mxf , & source_track - > sequence_ref , Sequence ) ) ) {
2006-08-02 17:02:42 +02:00
av_log ( mxf - > fc , AV_LOG_ERROR , " could not resolve source track sequence strong ref \n " ) ;
return - 1 ;
}
2006-07-29 00:42:31 +02:00
# ifdef DEBUG
2007-03-12 13:36:41 +01:00
PRINT_KEY ( mxf - > fc , " data definition ul " , source_track - > sequence - > data_definition_ul ) ;
2006-07-29 00:42:31 +02:00
# endif
2006-09-29 14:23:38 +02:00
st - > codec - > codec_type = mxf_get_codec_type ( mxf_data_definition_uls , & source_track - > sequence - > data_definition_ul ) ;
2006-07-29 00:42:31 +02:00
2007-01-14 01:32:25 +01:00
source_package - > descriptor = mxf_resolve_strong_ref ( mxf , & source_package - > descriptor_ref , AnyType ) ;
2006-07-29 00:42:31 +02:00
if ( source_package - > descriptor ) {
2006-08-02 17:02:42 +02:00
if ( source_package - > descriptor - > type = = MultipleDescriptor ) {
2006-07-29 00:42:31 +02:00
for ( j = 0 ; j < source_package - > descriptor - > sub_descriptors_count ; j + + ) {
2007-01-13 19:54:48 +01:00
MXFDescriptor * sub_descriptor = mxf_resolve_strong_ref ( mxf , & source_package - > descriptor - > sub_descriptors_refs [ j ] , Descriptor ) ;
2006-08-02 17:02:42 +02:00
if ( ! sub_descriptor ) {
av_log ( mxf - > fc , AV_LOG_ERROR , " could not resolve sub descriptor strong ref \n " ) ;
continue ;
}
if ( sub_descriptor - > linked_track_id = = source_track - > track_id ) {
descriptor = sub_descriptor ;
break ;
2006-07-25 16:30:14 +02:00
}
}
2007-01-14 01:32:25 +01:00
} else if ( source_package - > descriptor - > type = = Descriptor )
2006-07-29 00:42:31 +02:00
descriptor = source_package - > descriptor ;
}
if ( ! descriptor ) {
av_log ( mxf - > fc , AV_LOG_INFO , " source track %d: stream %d, no descriptor found \n " , source_track - > track_id , st - > index ) ;
continue ;
}
# ifdef DEBUG
2007-03-12 13:36:41 +01:00
PRINT_KEY ( mxf - > fc , " essence codec ul " , descriptor - > essence_codec_ul ) ;
PRINT_KEY ( mxf - > fc , " essence container ul " , descriptor - > essence_container_ul ) ;
2006-07-29 00:42:31 +02:00
# endif
2007-02-11 13:50:33 +01:00
essence_container_ul = & descriptor - > essence_container_ul ;
/* HACK: replacing the original key with mxf_encrypted_essence_container
* is not allowed according to s429 - 6 , try to find correct information anyway */
if ( IS_KLV_KEY ( essence_container_ul , mxf_encrypted_essence_container ) ) {
av_log ( mxf - > fc , AV_LOG_INFO , " broken encrypted mxf file \n " ) ;
for ( k = 0 ; k < mxf - > metadata_sets_count ; k + + ) {
MXFMetadataSet * metadata = mxf - > metadata_sets [ k ] ;
if ( metadata - > type = = CryptoContext ) {
essence_container_ul = & ( ( MXFCryptoContext * ) metadata ) - > source_container_ul ;
break ;
}
}
}
2006-08-03 13:21:54 +02:00
/* TODO: drop PictureEssenceCoding and SoundEssenceCompression, only check EssenceContainer */
codec_ul = mxf_get_codec_ul ( mxf_codec_uls , & descriptor - > essence_codec_ul ) ;
2006-08-03 13:28:08 +02:00
st - > codec - > codec_id = codec_ul - > id ;
2006-08-03 14:31:15 +02:00
if ( descriptor - > extradata ) {
st - > codec - > extradata = descriptor - > extradata ;
st - > codec - > extradata_size = descriptor - > extradata_size ;
}
2006-07-29 00:42:31 +02:00
if ( st - > codec - > codec_type = = CODEC_TYPE_VIDEO ) {
2007-02-11 13:50:33 +01:00
container_ul = mxf_get_codec_ul ( mxf_picture_essence_container_uls , essence_container_ul ) ;
2006-08-03 11:14:56 +02:00
if ( st - > codec - > codec_id = = CODEC_ID_NONE )
2006-08-03 13:21:54 +02:00
st - > codec - > codec_id = container_ul - > id ;
2006-07-29 00:42:31 +02:00
st - > codec - > width = descriptor - > width ;
st - > codec - > height = descriptor - > height ;
2006-07-29 17:19:31 +02:00
st - > codec - > bits_per_sample = descriptor - > bits_per_sample ; /* Uncompressed */
2007-04-15 15:51:57 +02:00
st - > need_parsing = AVSTREAM_PARSE_HEADERS ;
2006-07-29 00:42:31 +02:00
} else if ( st - > codec - > codec_type = = CODEC_TYPE_AUDIO ) {
2007-02-11 13:50:33 +01:00
container_ul = mxf_get_codec_ul ( mxf_sound_essence_container_uls , essence_container_ul ) ;
2006-08-03 11:14:56 +02:00
if ( st - > codec - > codec_id = = CODEC_ID_NONE )
2006-08-03 13:21:54 +02:00
st - > codec - > codec_id = container_ul - > id ;
2006-07-29 00:42:31 +02:00
st - > codec - > channels = descriptor - > channels ;
st - > codec - > bits_per_sample = descriptor - > bits_per_sample ;
st - > codec - > sample_rate = descriptor - > sample_rate . num / descriptor - > sample_rate . den ;
/* TODO: implement CODEC_ID_RAWAUDIO */
if ( st - > codec - > codec_id = = CODEC_ID_PCM_S16LE ) {
if ( descriptor - > bits_per_sample = = 24 )
st - > codec - > codec_id = CODEC_ID_PCM_S24LE ;
else if ( descriptor - > bits_per_sample = = 32 )
st - > codec - > codec_id = CODEC_ID_PCM_S32LE ;
} else if ( st - > codec - > codec_id = = CODEC_ID_PCM_S16BE ) {
if ( descriptor - > bits_per_sample = = 24 )
st - > codec - > codec_id = CODEC_ID_PCM_S24BE ;
else if ( descriptor - > bits_per_sample = = 32 )
st - > codec - > codec_id = CODEC_ID_PCM_S32BE ;
2006-10-02 16:12:50 +02:00
} else if ( st - > codec - > codec_id = = CODEC_ID_MP2 ) {
2007-04-15 15:51:57 +02:00
st - > need_parsing = AVSTREAM_PARSE_FULL ;
2006-07-25 16:30:14 +02:00
}
}
2008-01-19 18:25:12 +01:00
if ( st - > codec - > codec_type ! = CODEC_TYPE_DATA & & ( * essence_container_ul ) [ 15 ] > 0x01 ) {
av_log ( mxf - > fc , AV_LOG_WARNING , " only frame wrapped mappings are correctly supported \n " ) ;
2007-04-15 15:51:57 +02:00
st - > need_parsing = AVSTREAM_PARSE_FULL ;
2006-08-03 13:28:08 +02:00
}
2006-07-25 16:30:14 +02:00
}
2006-07-29 00:42:31 +02:00
return 0 ;
2006-07-25 16:30:14 +02:00
}
2006-07-31 17:26:33 +02:00
static const MXFMetadataReadTableEntry mxf_metadata_read_table [ ] = {
2008-01-19 17:17:39 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x05 , 0x01 , 0x01 , 0x0d , 0x01 , 0x02 , 0x01 , 0x01 , 0x05 , 0x01 , 0x00 } , mxf_read_primer_pack } ,
2008-01-19 17:20:06 +01:00
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x18 , 0x00 } , mxf_read_content_storage , 0 , AnyType } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x37 , 0x00 } , mxf_read_source_package , sizeof ( MXFPackage ) , SourcePackage } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x36 , 0x00 } , mxf_read_material_package , sizeof ( MXFPackage ) , MaterialPackage } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x0F , 0x00 } , mxf_read_sequence , sizeof ( MXFSequence ) , Sequence } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x11 , 0x00 } , mxf_read_source_clip , sizeof ( MXFStructuralComponent ) , SourceClip } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x44 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , MultipleDescriptor } ,
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x42 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , Descriptor } , /* Generic Sound */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x28 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , Descriptor } , /* CDCI */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x29 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , Descriptor } , /* RGBA */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x51 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , Descriptor } , /* MPEG 2 Video */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x48 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , Descriptor } , /* Wave */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x47 , 0x00 } , mxf_read_generic_descriptor , sizeof ( MXFDescriptor ) , Descriptor } , /* AES3 */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x3A , 0x00 } , mxf_read_track , sizeof ( MXFTrack ) , Track } , /* Static Track */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x3B , 0x00 } , mxf_read_track , sizeof ( MXFTrack ) , Track } , /* Generic Track */
{ { 0x06 , 0x0E , 0x2B , 0x34 , 0x02 , 0x53 , 0x01 , 0x01 , 0x0d , 0x01 , 0x04 , 0x01 , 0x02 , 0x02 , 0x00 , 0x00 } , mxf_read_cryptographic_context , sizeof ( MXFCryptoContext ) , CryptoContext } ,
2007-01-15 00:14:36 +01:00
{ { 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 } , NULL , 0 , AnyType } ,
2006-07-31 17:26:33 +02:00
} ;
2007-01-15 00:14:36 +01:00
static int mxf_read_local_tags ( MXFContext * mxf , KLVPacket * klv , int ( * read_child ) ( ) , int ctx_size , enum MXFMetadataSetType type )
{
2007-11-21 08:41:00 +01:00
ByteIOContext * pb = mxf - > fc - > pb ;
2007-01-15 00:14:36 +01:00
MXFMetadataSet * ctx = ctx_size ? av_mallocz ( ctx_size ) : mxf ;
2008-01-19 17:19:06 +01:00
uint64_t klv_end = url_ftell ( pb ) + klv - > length ;
2007-01-15 00:14:36 +01:00
2008-01-19 16:21:30 +01:00
if ( ! ctx )
return - 1 ;
2007-01-15 00:14:36 +01:00
while ( url_ftell ( pb ) + 4 < klv_end ) {
int tag = get_be16 ( pb ) ;
int size = get_be16 ( pb ) ; /* KLV specified by 0x53 */
2008-01-19 17:19:06 +01:00
uint64_t next = url_ftell ( pb ) + size ;
2008-01-19 21:00:09 +01:00
UID uid = { 0 } ;
2007-01-15 00:14:36 +01:00
if ( ! size ) { /* ignore empty tag, needed for some files with empty UMID tag */
av_log ( mxf - > fc , AV_LOG_ERROR , " local tag 0x%04X with 0 size \n " , tag ) ;
continue ;
}
2008-01-19 17:17:39 +01:00
if ( tag > 0x7FFF ) { /* dynamic tag */
int i ;
for ( i = 0 ; i < mxf - > local_tags_count ; i + + ) {
int local_tag = AV_RB16 ( mxf - > local_tags + i * 18 ) ;
if ( local_tag = = tag ) {
memcpy ( uid , mxf - > local_tags + i * 18 + 2 , 16 ) ;
dprintf ( mxf - > fc , " local tag 0x%04X \n " , local_tag ) ;
# ifdef DEBUG
PRINT_KEY ( mxf - > fc , " uid " , uid ) ;
# endif
}
}
}
2008-01-19 17:19:06 +01:00
if ( ctx_size & & tag = = 0x3C0A )
2007-01-15 00:14:36 +01:00
get_buffer ( pb , ctx - > uid , 16 ) ;
2008-01-19 20:51:55 +01:00
else if ( read_child ( ctx , pb , tag , size , uid ) < 0 )
return - 1 ;
2007-01-15 00:14:36 +01:00
url_fseek ( pb , next , SEEK_SET ) ;
}
if ( ctx_size ) ctx - > type = type ;
return ctx_size ? mxf_add_metadata_set ( mxf , ctx ) : 0 ;
}
2006-07-25 16:30:14 +02:00
static int mxf_read_header ( AVFormatContext * s , AVFormatParameters * ap )
{
MXFContext * mxf = s - > priv_data ;
KLVPacket klv ;
2007-11-21 08:41:00 +01:00
if ( ! mxf_read_sync ( s - > pb , mxf_header_partition_pack_key , 14 ) ) {
2006-10-17 13:23:04 +02:00
av_log ( s , AV_LOG_ERROR , " could not find header partition pack key \n " ) ;
return - 1 ;
}
2007-11-21 08:41:00 +01:00
url_fseek ( s - > pb , - 14 , SEEK_CUR ) ;
2006-07-25 16:30:14 +02:00
mxf - > fc = s ;
2007-11-21 08:41:00 +01:00
while ( ! url_feof ( s - > pb ) ) {
2007-01-15 02:04:43 +01:00
const MXFMetadataReadTableEntry * metadata ;
2006-07-31 17:26:33 +02:00
2007-11-21 08:41:00 +01:00
if ( klv_read_packet ( & klv , s - > pb ) < 0 )
2006-07-25 16:30:14 +02:00
return - 1 ;
2006-07-29 15:18:24 +02:00
# ifdef DEBUG
2007-03-12 13:36:41 +01:00
PRINT_KEY ( s , " read header " , klv . key ) ;
2006-07-29 15:18:24 +02:00
# endif
2007-02-11 13:50:33 +01:00
if ( IS_KLV_KEY ( klv . key , mxf_encrypted_triplet_key ) | |
IS_KLV_KEY ( klv . key , mxf_essence_element_key ) ) {
2006-07-25 16:30:14 +02:00
/* FIXME avoid seek */
2007-11-21 08:41:00 +01:00
url_fseek ( s - > pb , klv . offset , SEEK_SET ) ;
2006-07-25 16:30:14 +02:00
break ;
2006-07-29 00:53:15 +02:00
}
2006-07-31 17:26:33 +02:00
2007-01-15 02:04:43 +01:00
for ( metadata = mxf_metadata_read_table ; metadata - > read ; metadata + + ) {
if ( IS_KLV_KEY ( klv . key , metadata - > key ) ) {
2008-01-19 17:17:39 +01:00
int ( * read ) ( ) = klv . key [ 5 ] = = 0x53 ? mxf_read_local_tags : metadata - > read ;
if ( read ( mxf , & klv , metadata - > read , metadata - > ctx_size , metadata - > type ) < 0 ) {
2006-07-31 17:26:33 +02:00
av_log ( s , AV_LOG_ERROR , " error reading header metadata \n " ) ;
return - 1 ;
}
break ;
}
}
2007-01-15 02:04:43 +01:00
if ( ! metadata - > read )
2007-11-21 08:41:00 +01:00
url_fskip ( s - > pb , klv . length ) ;
2006-07-25 16:30:14 +02:00
}
2006-07-29 00:42:31 +02:00
return mxf_parse_structural_metadata ( mxf ) ;
2006-07-25 16:30:14 +02:00
}
static int mxf_read_close ( AVFormatContext * s )
{
MXFContext * mxf = s - > priv_data ;
2006-08-02 17:02:42 +02:00
int i ;
2006-07-29 00:42:31 +02:00
av_freep ( & mxf - > packages_refs ) ;
2006-08-02 17:02:42 +02:00
for ( i = 0 ; i < mxf - > metadata_sets_count ; i + + ) {
switch ( mxf - > metadata_sets [ i ] - > type ) {
case MultipleDescriptor :
av_freep ( & ( ( MXFDescriptor * ) mxf - > metadata_sets [ i ] ) - > sub_descriptors_refs ) ;
break ;
case Sequence :
av_freep ( & ( ( MXFSequence * ) mxf - > metadata_sets [ i ] ) - > structural_components_refs ) ;
break ;
case SourcePackage :
case MaterialPackage :
av_freep ( & ( ( MXFPackage * ) mxf - > metadata_sets [ i ] ) - > tracks_refs ) ;
break ;
default :
break ;
}
av_freep ( & mxf - > metadata_sets [ i ] ) ;
}
av_freep ( & mxf - > metadata_sets ) ;
2007-02-11 13:50:33 +01:00
av_freep ( & mxf - > aesc ) ;
2008-01-19 17:17:39 +01:00
av_freep ( & mxf - > local_tags ) ;
2006-07-25 16:30:14 +02:00
return 0 ;
}
static int mxf_probe ( AVProbeData * p ) {
2006-08-01 02:32:47 +02:00
uint8_t * bufp = p - > buf ;
uint8_t * end = p - > buf + p - > buf_size ;
2006-07-25 16:30:14 +02:00
if ( p - > buf_size < sizeof ( mxf_header_partition_pack_key ) )
return 0 ;
2006-08-01 02:32:47 +02:00
/* Must skip Run-In Sequence and search for MXF header partition pack key SMPTE 377M 5.5 */
end - = sizeof ( mxf_header_partition_pack_key ) ;
for ( ; bufp < end ; bufp + + ) {
2006-08-03 17:33:48 +02:00
if ( IS_KLV_KEY ( bufp , mxf_header_partition_pack_key ) )
2006-08-01 02:32:47 +02:00
return AVPROBE_SCORE_MAX ;
}
return 0 ;
2006-07-25 16:30:14 +02:00
}
2007-04-24 12:51:26 +02:00
/* rudimentary byte seek */
2006-09-29 16:28:55 +02:00
/* XXX: use MXF Index */
static int mxf_read_seek ( AVFormatContext * s , int stream_index , int64_t sample_time , int flags )
{
AVStream * st = s - > streams [ stream_index ] ;
int64_t seconds ;
2006-10-10 15:00:50 +02:00
if ( ! s - > bit_rate )
2006-09-29 16:28:55 +02:00
return - 1 ;
2006-10-10 15:00:50 +02:00
if ( sample_time < 0 )
sample_time = 0 ;
2006-09-29 16:28:55 +02:00
seconds = av_rescale ( sample_time , st - > time_base . num , st - > time_base . den ) ;
2007-11-21 08:41:00 +01:00
url_fseek ( s - > pb , ( s - > bit_rate * seconds ) > > 3 , SEEK_SET ) ;
2006-10-17 13:23:04 +02:00
av_update_cur_dts ( s , st , sample_time ) ;
return 0 ;
2006-09-29 16:28:55 +02:00
}
2006-07-25 16:30:14 +02:00
AVInputFormat mxf_demuxer = {
" mxf " ,
" MXF format " ,
sizeof ( MXFContext ) ,
mxf_probe ,
mxf_read_header ,
mxf_read_packet ,
mxf_read_close ,
2006-09-29 16:28:55 +02:00
mxf_read_seek ,
2006-07-25 16:30:14 +02:00
} ;