summaryrefslogtreecommitdiffstats
path: root/libmpdemux/demux_ty.c
diff options
context:
space:
mode:
authorarpi <arpi@b3059339-0415-0410-9bf9-f77b7e298cf2>2003-06-09 00:24:49 +0000
committerarpi <arpi@b3059339-0415-0410-9bf9-f77b7e298cf2>2003-06-09 00:24:49 +0000
commit080311d08f70042db8db88a6696a8a6c61c337df (patch)
tree82e106e42caafb41b5cfa63492b010be810e3b91 /libmpdemux/demux_ty.c
parent14328b41fedf57134a759b5cd4e3d8af330e7e0f (diff)
downloadmpv-080311d08f70042db8db88a6696a8a6c61c337df.tar.bz2
mpv-080311d08f70042db8db88a6696a8a6c61c337df.tar.xz
TiVo demuxer and sub-cc/osd decoder
patch by usenet@wingert.org (http://tivo-mplayer.sourceforge.net/releases/MPlayer-20030501-tivo-patch.gz) changes by me: - spit demux_ty to demux_ty and demux_ty_osd (later handles mpeg user-data decoding, ie sub-cc and osd) - removed some cosmetics changes - some compile fixes (gcc3 specific variable decl etc) git-svn-id: svn://svn.mplayerhq.hu/mplayer/trunk@10264 b3059339-0415-0410-9bf9-f77b7e298cf2
Diffstat (limited to 'libmpdemux/demux_ty.c')
-rw-r--r--libmpdemux/demux_ty.c1056
1 files changed, 1056 insertions, 0 deletions
diff --git a/libmpdemux/demux_ty.c b/libmpdemux/demux_ty.c
new file mode 100644
index 0000000000..a0f96e2b3b
--- /dev/null
+++ b/libmpdemux/demux_ty.c
@@ -0,0 +1,1056 @@
+/*
+ * 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, lets 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 ) );
+
+
+ // Lets 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(