mpv/libmpdemux/demux_ty.c

1057 lines
31 KiB
C
Raw Normal View History

/*
* tivo@wingert.org, February 2003
*
* Copyright (C) 2003 Christopher R. Wingert
*
* The license covers the portions of this file regarding TiVo additions.
*
* Olaf Beck and Tridge (indirectly) were essential at providing
* information regarding the format of the TiVo streams.
*
* However, no code in the following subsection is directly copied from
* either author.
*
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <time.h>
#include <stdarg.h>
#include "config.h"
#include "mp_msg.h"
#include "help_mp.h"
#include "stream.h"
#include "demuxer.h"
#include "parse_es.h"
#include "stheader.h"
//#include "mp3_hdr.h"
//#include "../subreader.h"
#include "../sub_cc.h"
//#include "../libvo/sub.h"
//#include "dvdauth.h"
extern void resync_audio_stream( sh_audio_t *sh_audio );
extern void skip_audio_frame( sh_audio_t *sh_audio );
extern int sub_justify;
// 2/c0: audio data
// 3/c0: audio packet header (PES header)
// 4/c0: audio data (S/A only?)
// 9/c0: audio packet header, AC-3 audio
// 2/e0: video data
// 6/e0: video packet header (PES header)
// 7/e0: video sequence header start
// 8/e0: video I-frame header start
// a/e0: video P-frame header start
// b/e0: video B-frame header start
// c/e0: video GOP header start
// e/01: closed-caption data
// e/02: Extended data services data
#define TIVO_PES_FILEID ( 0xf5467abd )
#define TIVO_PART_LENGTH ( 0x20000000 )
#define CHUNKSIZE ( 128 * 1024 )
#define MAX_AUDIO_BUFFER ( 16 * 1024 )
#define PTS_MHZ ( 90 )
#define PTS_KHZ ( PTS_MHZ * 1000 )
#define TY_V ( 1 )
#define TY_A ( 1 )
typedef struct sTivoInfo
{
unsigned char lastAudio[ MAX_AUDIO_BUFFER ];
int lastAudioEnd;
int tivoType; // 1 = SA, 2 = DTiVo
float firstAudioPTS;
float firstVideoPTS;
float lastAudioPTS;
float lastVideoPTS;
int headerOk;
unsigned int pesFileId; // Should be 0xf5467abd
int streamType; // Should be 0x02
int chunkSize; // Should always be 128k
off_t size;
int readHeader;
} TiVoInfo;
off_t vstream_streamsize( );
void ty_ClearOSD( int start );
// DTiVo MPEG 336, 480, 576, 768
// SA TiVo 864
// DTiVo AC-3 1550
//
#define SERIES1_PTS_LENGTH ( 11 )
#define SERIES1_PTS_OFFSET ( 6 )
#define SERIES2_PTS_LENGTH ( 16 )
#define SERIES2_PTS_OFFSET ( 9 )
#define AC3_PTS_LENGTH ( 16 )
#define AC3_PTS_OFFSET ( 9 )
#define NUMBER_DIFFERENT_AUDIO_SIZES ( 6 )
static int Series1AudioWithPTS[ NUMBER_DIFFERENT_AUDIO_SIZES ] =
{
336 + SERIES1_PTS_LENGTH,
480 + SERIES1_PTS_LENGTH,
576 + SERIES1_PTS_LENGTH,
768 + SERIES1_PTS_LENGTH,
864 + SERIES1_PTS_LENGTH
};
static int Series2AudioWithPTS[ NUMBER_DIFFERENT_AUDIO_SIZES ] =
{
336 + SERIES2_PTS_LENGTH,
480 + SERIES2_PTS_LENGTH,
576 + SERIES2_PTS_LENGTH,
768 + SERIES2_PTS_LENGTH,
864 + SERIES2_PTS_LENGTH
};
static int IsValidAudioPacket( int size, int *ptsOffset, int *ptsLen )
{
int count;
*ptsOffset = 0;
*ptsLen = 0;
// AC-3
if ( ( size == 1550 ) || ( size == 1552 ) )
{
*ptsOffset = AC3_PTS_OFFSET;
*ptsLen = AC3_PTS_LENGTH;
return( 1 );
}
// MPEG
for( count = 0 ; count < NUMBER_DIFFERENT_AUDIO_SIZES ; count++ )
{
if ( size == Series1AudioWithPTS[ count ] )
{
*ptsOffset = SERIES1_PTS_OFFSET;
*ptsLen = SERIES1_PTS_LENGTH;
break;
}
}
if ( *ptsOffset == 0 )
{
for( count = 0 ; count < NUMBER_DIFFERENT_AUDIO_SIZES ; count++ )
{
if ( size == Series2AudioWithPTS[ count ] )
{
*ptsOffset = SERIES2_PTS_OFFSET;
*ptsLen = SERIES2_PTS_LENGTH;
break;
}
}
}
if ( *ptsOffset == 0 )
{
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Tossing Audio Packet Size %d\n",
size );
return( 0 );
}
else
{
return( 1 );
}
}
static float get_ty_pts( unsigned char *buf )
{
float result = 0;
unsigned char temp;
temp = ( buf[ 0 ] & 0xE ) >> 1;
result = ( (float) temp ) * ( (float) ( 1L << 30 ) ) / ( (float)PTS_KHZ );
temp = buf[ 1 ];
result += ( (float) temp ) * ( (float) ( 1L << 22 ) ) / ( (float)PTS_KHZ );
temp = ( buf[ 2 ] & 0xFE ) >> 1;
result += ( (float) temp ) * ( (float) ( 1L << 15 ) ) / ( (float)PTS_KHZ );
temp = buf[ 3 ];
result += ( (float) temp ) * ( (float) ( 1L << 7 ) ) / ( (float)PTS_KHZ );
temp = ( buf[ 4 ] & 0xFE ) >> 1;
result += ( (float) temp ) / ( (float)PTS_MHZ );
return result;
}
static void demux_ty_AddToAudioBuffer( TiVoInfo *tivo, unsigned char *buffer,
int size )
{
if ( ( tivo->lastAudioEnd + size ) < MAX_AUDIO_BUFFER )
{
memcpy( &( tivo->lastAudio[ tivo->lastAudioEnd ] ),
buffer, size );
tivo->lastAudioEnd += size;
}
else
{
mp_msg( MSGT_DEMUX, MSGL_ERR,
"ty:WARNING - Would have blown my audio buffer\n" );
}
}
static void demux_ty_CopyToDemuxPacket( int type, TiVoInfo *tivo, demux_stream_t *ds,
unsigned char *buffer, int size, off_t pos, float pts )
{
demux_packet_t *dp;
// mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Calling ds_add_packet() %7.1f\n", pts );
// printf( "%x %x %x %x\n",
// buffer[ 0 ], buffer[ 1 ], buffer[ 2 ], buffer[ 3 ] );
dp = new_demux_packet( size );
memcpy( dp->buffer, buffer, size );
dp->pts = pts;
dp->pos = pos;
dp->flags = 0;
ds_add_packet( ds, dp );
ds->pts = pts;
if ( type == TY_V )
{
if ( tivo->firstVideoPTS == -1 )
{
tivo->firstVideoPTS = pts;
}
}
if ( type == TY_A )
{
if ( tivo->firstAudioPTS == -1 )
{
tivo->firstAudioPTS = pts;
}
}
}
static int demux_ty_FindESHeader( unsigned char *header, int headerSize,
unsigned char *buffer, int bufferSize, int *esOffset1 )
{
int count;
*esOffset1 = -1;
for( count = 0 ; count < bufferSize ; count++ )
{
if ( ( buffer[ count + 0 ] == header[ 0 ] ) &&
( buffer[ count + 1 ] == header[ 1 ] ) &&
( buffer[ count + 2 ] == header[ 2 ] ) &&
( buffer[ count + 3 ] == header[ 3 ] ) )
{
*esOffset1 = count;
return( 1 );
}
}
return( -1 );
}
static void demux_ty_FindESPacket( unsigned char *header, int headerSize,
unsigned char *buffer, int bufferSize, int *esOffset1, int *esOffset2 )
{
int count;
*esOffset1 = -1;
*esOffset2 = -1;
for( count = 0 ; count < bufferSize ; count++ )
{
if ( ( buffer[ count + 0 ] == header[ 0 ] ) &&
( buffer[ count + 1 ] == header[ 1 ] ) &&
( buffer[ count + 2 ] == header[ 2 ] ) &&
( buffer[ count + 3 ] == header[ 3 ] ) )
{
*esOffset1 = count;
break;
}
}
if ( *esOffset1 != -1 )
{
for( count = *esOffset1 + 1 ;
count < bufferSize ; count++ )
{
if ( ( buffer[ count + 0 ] == header[ 0 ] ) &&
( buffer[ count + 1 ] == header[ 1 ] ) &&
( buffer[ count + 2 ] == header[ 2 ] ) &&
( buffer[ count + 3 ] == header[ 3 ] ) )
{
*esOffset2 = count;
break;
}
}
}
}
static int tivobuffer2hostlong( unsigned char *buffer )
{
return
(
buffer[ 0 ] << 24 | buffer[ 1 ] << 16 | buffer[ 2 ] << 8 | buffer[ 3 ]
);
}
static unsigned char tivo_reversebyte( unsigned char val )
{
int count;
unsigned char ret;
ret = 0;
for ( count = 0 ; count < 8 ; count++ )
{
ret = ret << 1;
ret |= ( ( val >> count ) & 0x01 );
}
return( ret );
}
static unsigned char ty_VideoPacket[] = { 0x00, 0x00, 0x01, 0xe0 };
static unsigned char ty_MPEGAudioPacket[] = { 0x00, 0x00, 0x01, 0xc0 };
static unsigned char ty_AC3AudioPacket[] = { 0x00, 0x00, 0x01, 0xbd };
int demux_ty_fill_buffer( demuxer_t *demux )
{
int invalidType = 0;
int errorHeader = 0;
int recordsDecoded = 0;
off_t filePos = 0;
unsigned char chunk[ CHUNKSIZE ];
int whichChunk;
int readSize;
unsigned int pesFileId = 0;
int numberRecs;
unsigned char *recPtr;
int offset;
int size;
int type;
int nybbleType;
int counter;
int aid;
demux_stream_t *ds = NULL;
int esOffset1;
int esOffset2;
TiVoInfo *tivo = 0;
if ( demux->stream->type == STREAMTYPE_DVD )
{
return( 0 );
}
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Parsing a chunk\n" );
if ( ( demux->a_streams[ MAX_A_STREAMS - 1 ] ) == 0 )
{
demux->a_streams[ MAX_A_STREAMS - 1 ] = malloc( sizeof( TiVoInfo ) );
tivo = demux->a_streams[ MAX_A_STREAMS - 1 ];
memset( tivo, 0, sizeof( TiVoInfo ) );
tivo->firstAudioPTS = -1;
tivo->firstVideoPTS = -1;
}
else
{
tivo = demux->a_streams[ MAX_A_STREAMS - 1 ];
}
if( demux->stream->eof ) return 0;
// ======================================================================
// If we haven't figured out the size of the stream, let's do so
// ======================================================================
#ifdef STREAMTYPE_STREAM_TY
if ( demux->stream->type == STREAMTYPE_STREAM_TY )
{
// The vstream code figures out the exact size of the stream
demux->movi_start = 0;
demux->movi_end = vstream_streamsize();
tivo->size = vstream_streamsize();
}
else
#endif
{
// If its a local file, try to find the Part Headers, so we can
// calculate the ACTUAL stream size
// If we can't find it, go off with the file size and hope the
// extract program did the "right thing"
if ( tivo->readHeader == 0 )
{
tivo->readHeader = 1;
filePos = demux->filepos;
stream_seek( demux->stream, 0 );
// mp_msg( MSGT_DEMUX, MSGL_DBG3,
// "ty:Reading a chunk %d\n", __LINE__ );
readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
if ( readSize == CHUNKSIZE )
{
tivo->pesFileId = tivobuffer2hostlong( &chunk[ 0x00 ] );
tivo->streamType = tivobuffer2hostlong( &chunk[ 0x04 ] );
tivo->chunkSize = tivobuffer2hostlong( &chunk[ 0x08 ] );
tivo->size = tivobuffer2hostlong( &chunk[ 0x0c ] );
if ( tivo->pesFileId == TIVO_PES_FILEID )
{
off_t numberParts;
off_t size;
if ( demux->stream->end_pos > TIVO_PART_LENGTH )
{
numberParts = demux->stream->end_pos / TIVO_PART_LENGTH;
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Number Parts %d\n",
numberParts );
stream_seek( demux->stream, numberParts * TIVO_PART_LENGTH );
// mp_msg( MSGT_DEMUX, MSGL_DBG3,
// "ty:Reading a chunk %d\n", __LINE__ );
readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
pesFileId = tivobuffer2hostlong( &chunk[ 0x00 ] );
if ( pesFileId == TIVO_PES_FILEID )
{
size = tivobuffer2hostlong( &chunk[ 0x0c ] );
size /= 256;
size -= 4;
size *= CHUNKSIZE;
tivo->size = numberParts * TIVO_PART_LENGTH;
tivo->size += size;
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Header Calc Stream Size %lld\n", tivo->size );
}
else
{
tivo->size = demux->stream->end_pos;
}
}
else
{
tivo->size = demux->stream->end_pos;
}
}
else
{
tivo->size = demux->stream->end_pos;
}
}
if ( tivo->size > demux->stream->end_pos )
{
tivo->size = demux->stream->end_pos;
}
if ( demux->stream->start_pos > 0 )
{
filePos = demux->stream->start_pos;
}
stream_seek( demux->stream, filePos );
demux->filepos = stream_tell( demux->stream );
}
demux->movi_start = 0;
demux->movi_end = tivo->size;
}
// ======================================================================
// Give a clue as to where we are in the stream
// ======================================================================
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:ty header size %llx\n", tivo->size );
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:file end_pos %llx\n", demux->stream->end_pos );
// mp_msg( MSGT_DEMUX, MSGL_DBG3,
// "ty:vstream size %llx\n", vstream_streamsize() );
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"\nty:wanted current offset %llx\n", stream_tell( demux->stream ) );
if ( tivo->size > 0 )
{
if ( stream_tell( demux->stream ) > tivo->size )
{
demux->stream->eof = 1;
return( 0 );
}
}
// Make sure we are on a 128k boundary
if ( ( demux->filepos % CHUNKSIZE ) != 0 )
{
whichChunk = demux->filepos / CHUNKSIZE;
if ( ( demux->filepos % CHUNKSIZE ) > ( CHUNKSIZE / 2 ) )
{
whichChunk++;
}
stream_seek( demux->stream, ( whichChunk * CHUNKSIZE ) );
}
demux->filepos = stream_tell( demux->stream );
readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
if ( readSize != CHUNKSIZE )
{
return( 0 );
}
// We found a part header, skip it
pesFileId = tivobuffer2hostlong( &chunk[ 0x00 ] );
if( pesFileId == TIVO_PES_FILEID )
{
demux->filepos = stream_tell( demux->stream );
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Skipping PART Header\n" );
readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
if ( readSize != CHUNKSIZE )
{
return( 0 );
}
}
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"\nty:actual current offset %llx\n", ( stream_tell( demux->stream ) -
0x20000 ) );
// Let's make a Video Demux Stream for Mplayer
aid = 0x0;
if( !demux->v_streams[ aid ] ) new_sh_video( demux, aid );
if( demux->video->id == -1 ) demux->video->id = aid;
if( demux->video->id == aid )
{
ds = demux->video;
if( !ds->sh ) ds->sh = demux->v_streams[ aid ];
}
// ======================================================================
// Finally, we get to actually parse the chunk
// ======================================================================
numberRecs = chunk[ 0 ];
recPtr = &chunk[ 4 ];
offset = ( numberRecs * 16 ) + 4;
for ( counter = 0 ; counter < numberRecs ; counter++ )
{
size = ( recPtr[ 0 ] << 8 | recPtr[ 1 ] ) << 4 | ( recPtr[ 2 ] >> 4 );
type = recPtr[ 3 ];
nybbleType = recPtr[ 2 ] & 0x0f;
recordsDecoded++;
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Record Type %x/%x %d\n", nybbleType, type, size );
// ================================================================
// Video Parsing
// ================================================================
if ( type == 0xe0 )
{
if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
{
#if 0
printf( "Video Chunk Header " );
for( count = 0 ; count < 24 ; count++ )
{
printf( "%2.2x ", chunk[ offset + count ] );
}
printf( "\n" );
#endif
demux_ty_FindESHeader( ty_VideoPacket, 4, &chunk[ offset ],
size, &esOffset1 );
if ( esOffset1 != -1 )
{
tivo->lastVideoPTS = get_ty_pts(
&chunk[ offset + esOffset1 + 9 ] );
mp_msg( MSGT_DEMUX, MSGL_DBG3, "Video PTS %7.1f\n",
tivo->lastVideoPTS );
}
// Do NOT Pass the PES Header onto the MPEG2 Decode
if( nybbleType != 0x06 )
{
demux_ty_CopyToDemuxPacket( TY_V, tivo, demux->video,
&chunk[ offset ], size, ( demux->filepos + offset ),
tivo->lastVideoPTS );
}
offset += size;
}
else
{
errorHeader++;
}
}
// ================================================================
// Audio Parsing
// ================================================================
else if ( type == 0xc0 )
{
if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
{
#if 0
printf( "Audio Chunk Header " );
for( count = 0 ; count < 24 ; count++ )
{
printf( "%2.2x ", chunk[ offset + count ] );
}
printf( "\n" );
#endif
if( demux->audio->id == -1 )
{
if ( nybbleType == 0x02 )
{
continue; // DTiVo inconclusive, wait for more
}
else if ( nybbleType == 0x09 )
{
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Setting AC-3 Audio\n" );
aid = 0x80; // AC-3
}
else
{
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Setting MPEG Audio\n" );
aid = 0x0; // MPEG Audio
}
demux->audio->id = aid;
if( !demux->a_streams[ aid ] ) new_sh_audio( demux, aid );
if( demux->audio->id == aid )
{
ds = demux->audio;
if( !ds->sh ) ds->sh = demux->a_streams[ aid ];
}
}
aid = demux->audio->id;
// SA DTiVo Audio Data, no PES
// ================================================
if ( nybbleType == 0x02 )
{
if ( tivo->tivoType == 2 )
{
demux_ty_AddToAudioBuffer( tivo, &chunk[ offset ], size );
}
else
{
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Adding Audio Packet Size %d\n", size );
demux_ty_CopyToDemuxPacket( TY_A, tivo, demux->audio,
&chunk[ offset ], size, ( demux->filepos + offset ),
tivo->lastAudioPTS );
}
}
// MPEG Audio with PES Header, either SA or DTiVo
// ================================================
if ( nybbleType == 0x03 )
{
demux_ty_FindESHeader( ty_MPEGAudioPacket, 4, &chunk[ offset ],
size, &esOffset1 );
// SA PES Header, No Audio Data
// ================================================
if ( ( esOffset1 == 0 ) && ( size == 16 ) )
{
tivo->tivoType = 1;
tivo->lastAudioPTS = get_ty_pts( &chunk[ offset +
SERIES2_PTS_OFFSET ] );
mp_msg( MSGT_DEMUX, MSGL_DBG3, "SA Audio PTS %7.1f\n",
tivo->lastAudioPTS );
}
else
// DTiVo Audio with PES Header
// ================================================
{
tivo->tivoType = 2;
demux_ty_AddToAudioBuffer( tivo, &chunk[ offset ], size );
demux_ty_FindESPacket( ty_MPEGAudioPacket, 4,
tivo->lastAudio, tivo->lastAudioEnd, &esOffset1,
&esOffset2 );
if ( ( esOffset1 != -1 ) && ( esOffset2 != -1 ) )
{
int packetSize = esOffset2 - esOffset1;
int headerSize;
int ptsOffset;
if ( IsValidAudioPacket( packetSize, &ptsOffset,
&headerSize ) )
{
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Adding DTiVo Audio Packet Size %d\n",
packetSize );
tivo->lastAudioPTS = get_ty_pts(
&tivo->lastAudio[ esOffset1 + ptsOffset ] );
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"MPEG Audio PTS %7.1f\n", tivo->lastAudioPTS );
demux_ty_CopyToDemuxPacket
(
TY_A,
tivo,
demux->audio,
&( tivo->lastAudio[ esOffset1 + headerSize ] ),
( packetSize - headerSize ),
( demux->filepos + offset ),
tivo->lastAudioPTS
);
}
// Collapse the Audio Buffer
memmove( &(tivo->lastAudio[ 0 ] ),
&( tivo->lastAudio[ esOffset2 ] ),
( tivo->lastAudioEnd - esOffset2 ) );
tivo->lastAudioEnd -= esOffset2;
}
}
}
// SA Audio with no PES Header
// ================================================
if ( nybbleType == 0x04 )
{
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Adding Audio Packet Size %d\n", size );
demux_ty_CopyToDemuxPacket( TY_A, tivo, demux->audio,
&chunk[ offset ], size, ( demux->filepos + offset ),
tivo->lastAudioPTS );
}
// DTiVo AC3 Audio Data with PES Header
// ================================================
if ( nybbleType == 0x09 )
{
tivo->tivoType = 2;
demux_ty_AddToAudioBuffer( tivo, &chunk[ offset ], size );
demux_ty_FindESPacket( ty_AC3AudioPacket, 4,
tivo->lastAudio, tivo->lastAudioEnd, &esOffset1,
&esOffset2 );
if ( ( esOffset1 != -1 ) && ( esOffset2 != -1 ) )
{
int packetSize = esOffset2 - esOffset1;
int headerSize;
int ptsOffset;
if ( IsValidAudioPacket( packetSize, &ptsOffset,
&headerSize ) )
{
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Adding DTiVo Audio Packet Size %d\n",
packetSize );
tivo->lastAudioPTS = get_ty_pts(
&tivo->lastAudio[ esOffset1 + ptsOffset ] );
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"AC3 Audio PTS %7.1f\n", tivo->lastAudioPTS );
// AC3 Decoder WANTS the PTS
demux_ty_CopyToDemuxPacket
(
TY_A,
tivo,
demux->audio,
&( tivo->lastAudio[ esOffset1 ] ),
( packetSize ),
( demux->filepos + offset ),
tivo->lastAudioPTS
);
}
// Collapse the Audio Buffer
memmove( &(tivo->lastAudio[ 0 ] ),
&( tivo->lastAudio[ esOffset2 ] ),
( tivo->lastAudioEnd - esOffset2 ) );
tivo->lastAudioEnd -= esOffset2;
}
}
offset += size;
}
else
{
errorHeader++;
}
}
// ================================================================
// Closed Caption
// ================================================================
else if ( type == 0x01 )
{
unsigned char b1;
unsigned char b2;
unsigned char buffer[ 16 ];
b1 = ( ( ( recPtr[ 0 ] & 0x0f ) << 4 ) |
( ( recPtr[ 1 ] & 0xf0 ) >> 4 ) );
b1 &= 0x7f;
b2 = ( ( ( recPtr[ 1 ] & 0x0f ) << 4 ) |
( ( recPtr[ 2 ] & 0xf0 ) >> 4 ) );
b2 &= 0x7f;
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:CC %x %x\n", b1, b2 );
buffer[ 0x00 ] = 0x00;
buffer[ 0x01 ] = 0x00;
buffer[ 0x02 ] = 0x01;
buffer[ 0x03 ] = 0xb2;
buffer[ 0x04 ] = 'T';
buffer[ 0x05 ] = 'Y';
buffer[ 0x06 ] = 0x01;
buffer[ 0x07 ] = b1;
buffer[ 0x08 ] = b2;
demux_ty_CopyToDemuxPacket( TY_V, tivo, demux->video, buffer, 0x09,
( demux->filepos + offset ), tivo->lastVideoPTS );
}
// ================================================================
// Extended Data Services
// ================================================================
else if ( type == 0x02 )
{
unsigned char b1;
unsigned char b2;
unsigned char buffer[ 16 ];
b1 = ( ( ( recPtr[ 0 ] & 0x0f ) << 4 ) |
( ( recPtr[ 1 ] & 0xf0 ) >> 4 ) );
b1 &= 0x7f;
b2 = ( ( ( recPtr[ 1 ] & 0x0f ) << 4 ) |
( ( recPtr[ 2 ] & 0xf0 ) >> 4 ) );
b2 &= 0x7f;
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:XDS %x %x\n", b1, b2 );
buffer[ 0x00 ] = 0x00;
buffer[ 0x01 ] = 0x00;
buffer[ 0x02 ] = 0x01;
buffer[ 0x03 ] = 0xb2;
buffer[ 0x04 ] = 'T';
buffer[ 0x05 ] = 'Y';
buffer[ 0x06 ] = 0x02;
buffer[ 0x07 ] = b1;
buffer[ 0x08 ] = b2;
demux_ty_CopyToDemuxPacket( TY_V, tivo, demux->video, buffer, 0x09,
( demux->filepos + offset ), tivo->lastVideoPTS );
}
// ================================================================
// Found a 0x03 on Droid's TiVo, I have no idea what it is
// ================================================================
else if ( type == 0x03 )
{
if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
{
offset += size;
}
}
// ================================================================
// Unknown
// ================================================================
else if ( type == 0x05 )
{
if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
{
offset += size;
}
}
else
{
if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
{
offset += size;
}
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Invalid Type %x\n", type );
invalidType++;
}
recPtr += 16;
}
if ( errorHeader > 0 )
{
mp_msg( MSGT_DEMUX, MSGL_DBG3,
"ty:Error Check - Records %d, Parsed %d, Errors %d\n",
numberRecs, recordsDecoded, errorHeader );
// Invalid MPEG ES Size Check
if ( errorHeader > ( numberRecs / 2 ) )
{
return( 0 );
}
// Invalid MPEG Stream Type Check
if ( invalidType > ( numberRecs / 2 ) )
{
return( 0 );
}
}
demux->filepos = stream_tell( demux->stream );
return( 1 );
}
void demux_seek_ty( demuxer_t *demuxer, float rel_seek_secs, int flags )
{
demux_stream_t *d_audio = demuxer->audio;
demux_stream_t *d_video = demuxer->video;
sh_audio_t *sh_audio = d_audio->sh;
sh_video_t *sh_video = d_video->sh;
off_t newpos;
off_t res;
TiVoInfo *tivo = 0;
mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Seeking to %7.1f\n", rel_seek_secs );
if ( ( demuxer->a_streams[ MAX_A_STREAMS - 1 ] ) != 0 )
{
tivo = demuxer->a_streams[ MAX_A_STREAMS - 1 ];
tivo->lastAudioEnd = 0;
tivo->lastAudioPTS = 0;
tivo->lastVideoPTS = 0;
}
//
//================= seek in MPEG ==========================
demuxer->filepos = stream_tell( demuxer->stream );
newpos = ( flags & 1 ) ? demuxer->movi_start : demuxer->filepos;
if( flags & 2 )
{
// float seek 0..1
newpos += ( demuxer->movi_end - demuxer->movi_start ) * rel_seek_secs;
}
else
{
// time seek (secs)
if( ! sh_video->i_bps ) // unspecified or VBR
{
newpos += 2324 * 75 * rel_seek_secs; // 174.3 kbyte/sec
}
else
{
newpos += sh_video->i_bps * rel_seek_secs;
}
}
if ( newpos < demuxer->movi_start )
{
if( demuxer->stream->type != STREAMTYPE_VCD ) demuxer->movi_start = 0;
if( newpos < demuxer->movi_start ) newpos = demuxer->movi_start;
}
res = newpos / CHUNKSIZE;
if ( rel_seek_secs >= 0 )
{
newpos = ( res + 1 ) * CHUNKSIZE;
}
else
{
newpos = res * CHUNKSIZE;
}
if ( newpos < 0 )
{
newpos = 0;
}
stream_seek( demuxer->stream, newpos );
// re-sync video:
videobuf_code_len = 0; // reset ES stream buffer
ds_fill_buffer( d_video );
if( sh_audio )
{
ds_fill_buffer( d_audio );
resync_audio_stream( sh_audio );
}
while( 1 )
{
int i;
if( sh_audio && !d_audio->eof && d_video->pts && d_audio->pts )
{
float a_pts = d_audio->pts;
a_pts += ( ds_tell_pts( d_audio ) - sh_audio->a_in_buffer_len ) /
(float)sh_audio->i_bps;
if( d_video->pts > a_pts )
{
skip_audio_frame( sh_audio ); // sync audio
continue;
}
}
i = sync_video_packet( d_video );
if( i == 0x1B3 || i == 0x1B8 ) break; // found it!
if( !i || !skip_video_packet( d_video ) ) break; // EOF?
}
if ( subcc_enabled )
{
ty_ClearOSD( 0 );
}
}
int demux_ty_control( demuxer_t *demuxer,int cmd, void *arg )
{
demux_stream_t *d_video = demuxer->video;
sh_video_t *sh_video = d_video->sh;
switch(cmd)
{
case DEMUXER_CTRL_GET_TIME_LENGTH:
if(!sh_video->i_bps) // unspecified or VBR
return DEMUXER_CTRL_DONTKNOW;
*((unsigned long *)arg)=
(demuxer->movi_end-demuxer->movi_start)/sh_video->i_bps;
return DEMUXER_CTRL_GUESS;
case DEMUXER_CTRL_GET_PERCENT_POS:
if (demuxer->movi_end==demuxer->movi_start)
return DEMUXER_CTRL_DONTKNOW;
*((int *)arg)=
(int)((demuxer->filepos-demuxer->movi_start)/
((demuxer->movi_end-demuxer->movi_start)/100));
return DEMUXER_CTRL_OK;
default:
return DEMUXER_CTRL_NOTIMPL;
}
}
int demux_close_ty( demuxer_t *demux )
{
TiVoInfo *tivo = 0;
if ( ( demux->a_streams[ MAX_A_STREAMS - 1 ] ) != 0 )
{
tivo = demux->a_streams[ MAX_A_STREAMS - 1 ];
free( tivo );
demux->a_streams[ MAX_A_STREAMS - 1 ] = 0;
sub_justify = 0;
}
return( 0 );
}