10263
|
1 /*
|
|
2 * tivo@wingert.org, February 2003
|
|
3 *
|
|
4 * Copyright (C) 2003 Christopher R. Wingert
|
|
5 *
|
|
6 * The license covers the portions of this file regarding TiVo additions.
|
|
7 *
|
|
8 * Olaf Beck and Tridge (indirectly) were essential at providing
|
|
9 * information regarding the format of the TiVo streams.
|
|
10 *
|
|
11 * However, no code in the following subsection is directly copied from
|
|
12 * either author.
|
|
13 *
|
|
14 *
|
|
15 * This program is free software; you can redistribute it and/or
|
|
16 * modify it under the terms of the GNU General Public License
|
|
17 * as published by the Free Software Foundation; either version 2
|
|
18 * of the License, or (at your option) any later version.
|
|
19 *
|
|
20 * This program is distributed in the hope that it will be useful,
|
|
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
23 * GNU General Public License for more details.
|
|
24 *
|
|
25 * You should have received a copy of the GNU General Public License
|
|
26 * along with this program; if not, write to the Free Software
|
|
27 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
28 *
|
|
29 */
|
|
30
|
|
31
|
|
32 #include <stdio.h>
|
|
33 #include <stdlib.h>
|
|
34 #include <unistd.h>
|
|
35 #include <time.h>
|
|
36 #include <stdarg.h>
|
|
37
|
|
38 #include "config.h"
|
|
39 #include "mp_msg.h"
|
|
40 #include "help_mp.h"
|
|
41
|
|
42 #include "stream.h"
|
|
43 #include "demuxer.h"
|
|
44 #include "parse_es.h"
|
|
45 #include "stheader.h"
|
|
46 //#include "mp3_hdr.h"
|
|
47 //#include "../subreader.h"
|
|
48 #include "../sub_cc.h"
|
|
49 //#include "../libvo/sub.h"
|
|
50
|
|
51 //#include "dvdauth.h"
|
|
52
|
|
53 extern void resync_audio_stream( sh_audio_t *sh_audio );
|
|
54 extern void skip_audio_frame( sh_audio_t *sh_audio );
|
|
55 extern int sub_justify;
|
|
56
|
|
57 // 2/c0: audio data
|
|
58 // 3/c0: audio packet header (PES header)
|
|
59 // 4/c0: audio data (S/A only?)
|
|
60 // 9/c0: audio packet header, AC-3 audio
|
|
61 // 2/e0: video data
|
|
62 // 6/e0: video packet header (PES header)
|
|
63 // 7/e0: video sequence header start
|
|
64 // 8/e0: video I-frame header start
|
|
65 // a/e0: video P-frame header start
|
|
66 // b/e0: video B-frame header start
|
|
67 // c/e0: video GOP header start
|
|
68 // e/01: closed-caption data
|
|
69 // e/02: Extended data services data
|
|
70
|
|
71
|
|
72 #define TIVO_PES_FILEID ( 0xf5467abd )
|
|
73 #define TIVO_PART_LENGTH ( 0x20000000 )
|
|
74
|
|
75 #define CHUNKSIZE ( 128 * 1024 )
|
|
76 #define MAX_AUDIO_BUFFER ( 16 * 1024 )
|
|
77
|
|
78 #define PTS_MHZ ( 90 )
|
|
79 #define PTS_KHZ ( PTS_MHZ * 1000 )
|
|
80
|
|
81 #define TY_V ( 1 )
|
|
82 #define TY_A ( 1 )
|
|
83
|
|
84 typedef struct sTivoInfo
|
|
85 {
|
|
86 unsigned char lastAudio[ MAX_AUDIO_BUFFER ];
|
|
87 int lastAudioEnd;
|
|
88
|
|
89 int tivoType; // 1 = SA, 2 = DTiVo
|
|
90
|
|
91 float firstAudioPTS;
|
|
92 float firstVideoPTS;
|
|
93
|
|
94 float lastAudioPTS;
|
|
95 float lastVideoPTS;
|
|
96
|
|
97 int headerOk;
|
|
98 unsigned int pesFileId; // Should be 0xf5467abd
|
|
99 int streamType; // Should be 0x02
|
|
100 int chunkSize; // Should always be 128k
|
|
101 off_t size;
|
|
102 int readHeader;
|
|
103 } TiVoInfo;
|
|
104
|
|
105 off_t vstream_streamsize( );
|
|
106 void ty_ClearOSD( int start );
|
|
107
|
|
108 // DTiVo MPEG 336, 480, 576, 768
|
|
109 // SA TiVo 864
|
|
110 // DTiVo AC-3 1550
|
|
111 //
|
|
112 #define SERIES1_PTS_LENGTH ( 11 )
|
|
113 #define SERIES1_PTS_OFFSET ( 6 )
|
|
114 #define SERIES2_PTS_LENGTH ( 16 )
|
|
115 #define SERIES2_PTS_OFFSET ( 9 )
|
|
116 #define AC3_PTS_LENGTH ( 16 )
|
|
117 #define AC3_PTS_OFFSET ( 9 )
|
|
118
|
|
119 #define NUMBER_DIFFERENT_AUDIO_SIZES ( 6 )
|
|
120 static int Series1AudioWithPTS[ NUMBER_DIFFERENT_AUDIO_SIZES ] =
|
|
121 {
|
|
122 336 + SERIES1_PTS_LENGTH,
|
|
123 480 + SERIES1_PTS_LENGTH,
|
|
124 576 + SERIES1_PTS_LENGTH,
|
|
125 768 + SERIES1_PTS_LENGTH,
|
|
126 864 + SERIES1_PTS_LENGTH
|
|
127 };
|
|
128 static int Series2AudioWithPTS[ NUMBER_DIFFERENT_AUDIO_SIZES ] =
|
|
129 {
|
|
130 336 + SERIES2_PTS_LENGTH,
|
|
131 480 + SERIES2_PTS_LENGTH,
|
|
132 576 + SERIES2_PTS_LENGTH,
|
|
133 768 + SERIES2_PTS_LENGTH,
|
|
134 864 + SERIES2_PTS_LENGTH
|
|
135 };
|
|
136
|
|
137 static int IsValidAudioPacket( int size, int *ptsOffset, int *ptsLen )
|
|
138 {
|
|
139 int count;
|
|
140
|
|
141 *ptsOffset = 0;
|
|
142 *ptsLen = 0;
|
|
143
|
|
144 // AC-3
|
|
145 if ( ( size == 1550 ) || ( size == 1552 ) )
|
|
146 {
|
|
147 *ptsOffset = AC3_PTS_OFFSET;
|
|
148 *ptsLen = AC3_PTS_LENGTH;
|
|
149 return( 1 );
|
|
150 }
|
|
151
|
|
152 // MPEG
|
|
153 for( count = 0 ; count < NUMBER_DIFFERENT_AUDIO_SIZES ; count++ )
|
|
154 {
|
|
155 if ( size == Series1AudioWithPTS[ count ] )
|
|
156 {
|
|
157 *ptsOffset = SERIES1_PTS_OFFSET;
|
|
158 *ptsLen = SERIES1_PTS_LENGTH;
|
|
159 break;
|
|
160 }
|
|
161 }
|
|
162 if ( *ptsOffset == 0 )
|
|
163 {
|
|
164 for( count = 0 ; count < NUMBER_DIFFERENT_AUDIO_SIZES ; count++ )
|
|
165 {
|
|
166 if ( size == Series2AudioWithPTS[ count ] )
|
|
167 {
|
|
168 *ptsOffset = SERIES2_PTS_OFFSET;
|
|
169 *ptsLen = SERIES2_PTS_LENGTH;
|
|
170 break;
|
|
171 }
|
|
172 }
|
|
173 }
|
|
174 if ( *ptsOffset == 0 )
|
|
175 {
|
|
176 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Tossing Audio Packet Size %d\n",
|
|
177 size );
|
|
178 return( 0 );
|
|
179 }
|
|
180 else
|
|
181 {
|
|
182 return( 1 );
|
|
183 }
|
|
184 }
|
|
185
|
|
186
|
|
187 static float get_ty_pts( unsigned char *buf )
|
|
188 {
|
|
189 float result = 0;
|
|
190 unsigned char temp;
|
|
191
|
|
192 temp = ( buf[ 0 ] & 0xE ) >> 1;
|
|
193 result = ( (float) temp ) * ( (float) ( 1L << 30 ) ) / ( (float)PTS_KHZ );
|
|
194 temp = buf[ 1 ];
|
|
195 result += ( (float) temp ) * ( (float) ( 1L << 22 ) ) / ( (float)PTS_KHZ );
|
|
196 temp = ( buf[ 2 ] & 0xFE ) >> 1;
|
|
197 result += ( (float) temp ) * ( (float) ( 1L << 15 ) ) / ( (float)PTS_KHZ );
|
|
198 temp = buf[ 3 ];
|
|
199 result += ( (float) temp ) * ( (float) ( 1L << 7 ) ) / ( (float)PTS_KHZ );
|
|
200 temp = ( buf[ 4 ] & 0xFE ) >> 1;
|
|
201 result += ( (float) temp ) / ( (float)PTS_MHZ );
|
|
202
|
|
203 return result;
|
|
204 }
|
|
205
|
|
206 static void demux_ty_AddToAudioBuffer( TiVoInfo *tivo, unsigned char *buffer,
|
|
207 int size )
|
|
208 {
|
|
209 if ( ( tivo->lastAudioEnd + size ) < MAX_AUDIO_BUFFER )
|
|
210 {
|
|
211 memcpy( &( tivo->lastAudio[ tivo->lastAudioEnd ] ),
|
|
212 buffer, size );
|
|
213 tivo->lastAudioEnd += size;
|
|
214 }
|
|
215 else
|
|
216 {
|
|
217 mp_msg( MSGT_DEMUX, MSGL_ERR,
|
|
218 "ty:WARNING - Would have blown my audio buffer\n" );
|
|
219 }
|
|
220 }
|
|
221
|
|
222 static void demux_ty_CopyToDemuxPacket( int type, TiVoInfo *tivo, demux_stream_t *ds,
|
|
223 unsigned char *buffer, int size, off_t pos, float pts )
|
|
224 {
|
|
225 demux_packet_t *dp;
|
|
226
|
|
227 // mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Calling ds_add_packet() %7.1f\n", pts );
|
|
228 // printf( "%x %x %x %x\n",
|
|
229 // buffer[ 0 ], buffer[ 1 ], buffer[ 2 ], buffer[ 3 ] );
|
|
230
|
|
231 dp = new_demux_packet( size );
|
|
232 memcpy( dp->buffer, buffer, size );
|
|
233 dp->pts = pts;
|
|
234 dp->pos = pos;
|
|
235 dp->flags = 0;
|
|
236 ds_add_packet( ds, dp );
|
|
237 ds->pts = pts;
|
|
238 if ( type == TY_V )
|
|
239 {
|
|
240 if ( tivo->firstVideoPTS == -1 )
|
|
241 {
|
|
242 tivo->firstVideoPTS = pts;
|
|
243 }
|
|
244 }
|
|
245 if ( type == TY_A )
|
|
246 {
|
|
247 if ( tivo->firstAudioPTS == -1 )
|
|
248 {
|
|
249 tivo->firstAudioPTS = pts;
|
|
250 }
|
|
251 }
|
|
252 }
|
|
253
|
|
254 static int demux_ty_FindESHeader( unsigned char *header, int headerSize,
|
|
255 unsigned char *buffer, int bufferSize, int *esOffset1 )
|
|
256 {
|
|
257 int count;
|
|
258
|
|
259 *esOffset1 = -1;
|
|
260 for( count = 0 ; count < bufferSize ; count++ )
|
|
261 {
|
|
262 if ( ( buffer[ count + 0 ] == header[ 0 ] ) &&
|
|
263 ( buffer[ count + 1 ] == header[ 1 ] ) &&
|
|
264 ( buffer[ count + 2 ] == header[ 2 ] ) &&
|
|
265 ( buffer[ count + 3 ] == header[ 3 ] ) )
|
|
266 {
|
|
267 *esOffset1 = count;
|
|
268 return( 1 );
|
|
269 }
|
|
270 }
|
|
271 return( -1 );
|
|
272 }
|
|
273
|
|
274 static void demux_ty_FindESPacket( unsigned char *header, int headerSize,
|
|
275 unsigned char *buffer, int bufferSize, int *esOffset1, int *esOffset2 )
|
|
276 {
|
|
277 int count;
|
|
278
|
|
279 *esOffset1 = -1;
|
|
280 *esOffset2 = -1;
|
|
281
|
|
282 for( count = 0 ; count < bufferSize ; count++ )
|
|
283 {
|
|
284 if ( ( buffer[ count + 0 ] == header[ 0 ] ) &&
|
|
285 ( buffer[ count + 1 ] == header[ 1 ] ) &&
|
|
286 ( buffer[ count + 2 ] == header[ 2 ] ) &&
|
|
287 ( buffer[ count + 3 ] == header[ 3 ] ) )
|
|
288 {
|
|
289 *esOffset1 = count;
|
|
290 break;
|
|
291 }
|
|
292 }
|
|
293
|
|
294 if ( *esOffset1 != -1 )
|
|
295 {
|
|
296 for( count = *esOffset1 + 1 ;
|
|
297 count < bufferSize ; count++ )
|
|
298 {
|
|
299 if ( ( buffer[ count + 0 ] == header[ 0 ] ) &&
|
|
300 ( buffer[ count + 1 ] == header[ 1 ] ) &&
|
|
301 ( buffer[ count + 2 ] == header[ 2 ] ) &&
|
|
302 ( buffer[ count + 3 ] == header[ 3 ] ) )
|
|
303 {
|
|
304 *esOffset2 = count;
|
|
305 break;
|
|
306 }
|
|
307 }
|
|
308 }
|
|
309 }
|
|
310
|
|
311 static int tivobuffer2hostlong( unsigned char *buffer )
|
|
312 {
|
|
313 return
|
|
314 (
|
|
315 buffer[ 0 ] << 24 | buffer[ 1 ] << 16 | buffer[ 2 ] << 8 | buffer[ 3 ]
|
|
316 );
|
|
317 }
|
|
318
|
|
319 static unsigned char tivo_reversebyte( unsigned char val )
|
|
320 {
|
|
321 int count;
|
|
322 unsigned char ret;
|
|
323
|
|
324 ret = 0;
|
|
325 for ( count = 0 ; count < 8 ; count++ )
|
|
326 {
|
|
327 ret = ret << 1;
|
|
328 ret |= ( ( val >> count ) & 0x01 );
|
|
329 }
|
|
330 return( ret );
|
|
331 }
|
|
332
|
|
333
|
|
334 static unsigned char ty_VideoPacket[] = { 0x00, 0x00, 0x01, 0xe0 };
|
|
335 static unsigned char ty_MPEGAudioPacket[] = { 0x00, 0x00, 0x01, 0xc0 };
|
|
336 static unsigned char ty_AC3AudioPacket[] = { 0x00, 0x00, 0x01, 0xbd };
|
|
337
|
|
338 int demux_ty_fill_buffer( demuxer_t *demux )
|
|
339 {
|
|
340 int invalidType = 0;
|
|
341 int errorHeader = 0;
|
|
342 int recordsDecoded = 0;
|
|
343 off_t filePos = 0;
|
|
344
|
|
345 unsigned char chunk[ CHUNKSIZE ];
|
|
346 int whichChunk;
|
|
347 int readSize;
|
|
348 unsigned int pesFileId = 0;
|
|
349
|
|
350 int numberRecs;
|
|
351 unsigned char *recPtr;
|
|
352 int offset;
|
|
353 int size;
|
|
354
|
|
355 int type;
|
|
356 int nybbleType;
|
|
357
|
|
358 int counter;
|
|
359
|
|
360 int aid;
|
|
361 demux_stream_t *ds = NULL;
|
|
362
|
|
363 int esOffset1;
|
|
364 int esOffset2;
|
|
365
|
|
366 TiVoInfo *tivo = 0;
|
|
367
|
|
368 if ( demux->stream->type == STREAMTYPE_DVD )
|
|
369 {
|
|
370 return( 0 );
|
|
371 }
|
|
372
|
|
373 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Parsing a chunk\n" );
|
|
374 if ( ( demux->a_streams[ MAX_A_STREAMS - 1 ] ) == 0 )
|
|
375 {
|
|
376 demux->a_streams[ MAX_A_STREAMS - 1 ] = malloc( sizeof( TiVoInfo ) );
|
|
377 tivo = demux->a_streams[ MAX_A_STREAMS - 1 ];
|
|
378 memset( tivo, 0, sizeof( TiVoInfo ) );
|
|
379 tivo->firstAudioPTS = -1;
|
|
380 tivo->firstVideoPTS = -1;
|
|
381 }
|
|
382 else
|
|
383 {
|
|
384 tivo = demux->a_streams[ MAX_A_STREAMS - 1 ];
|
|
385 }
|
|
386
|
|
387 if( demux->stream->eof ) return 0;
|
|
388
|
|
389 // ======================================================================
|
11000
|
390 // If we haven't figured out the size of the stream, let's do so
|
10263
|
391 // ======================================================================
|
|
392 #ifdef STREAMTYPE_STREAM_TY
|
|
393 if ( demux->stream->type == STREAMTYPE_STREAM_TY )
|
|
394 {
|
|
395 // The vstream code figures out the exact size of the stream
|
|
396 demux->movi_start = 0;
|
|
397 demux->movi_end = vstream_streamsize();
|
|
398 tivo->size = vstream_streamsize();
|
|
399 }
|
|
400 else
|
|
401 #endif
|
|
402 {
|
|
403 // If its a local file, try to find the Part Headers, so we can
|
|
404 // calculate the ACTUAL stream size
|
|
405 // If we can't find it, go off with the file size and hope the
|
|
406 // extract program did the "right thing"
|
|
407 if ( tivo->readHeader == 0 )
|
|
408 {
|
|
409 tivo->readHeader = 1;
|
|
410 filePos = demux->filepos;
|
|
411 stream_seek( demux->stream, 0 );
|
|
412 // mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
413 // "ty:Reading a chunk %d\n", __LINE__ );
|
|
414 readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
|
|
415 if ( readSize == CHUNKSIZE )
|
|
416 {
|
|
417 tivo->pesFileId = tivobuffer2hostlong( &chunk[ 0x00 ] );
|
|
418 tivo->streamType = tivobuffer2hostlong( &chunk[ 0x04 ] );
|
|
419 tivo->chunkSize = tivobuffer2hostlong( &chunk[ 0x08 ] );
|
|
420 tivo->size = tivobuffer2hostlong( &chunk[ 0x0c ] );
|
|
421 if ( tivo->pesFileId == TIVO_PES_FILEID )
|
|
422 {
|
|
423 off_t numberParts;
|
|
424 off_t size;
|
|
425
|
|
426 if ( demux->stream->end_pos > TIVO_PART_LENGTH )
|
|
427 {
|
|
428 numberParts = demux->stream->end_pos / TIVO_PART_LENGTH;
|
|
429 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Number Parts %d\n",
|
|
430 numberParts );
|
|
431 stream_seek( demux->stream, numberParts * TIVO_PART_LENGTH );
|
|
432 // mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
433 // "ty:Reading a chunk %d\n", __LINE__ );
|
|
434 readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
|
|
435 pesFileId = tivobuffer2hostlong( &chunk[ 0x00 ] );
|
|
436 if ( pesFileId == TIVO_PES_FILEID )
|
|
437 {
|
|
438 size = tivobuffer2hostlong( &chunk[ 0x0c ] );
|
|
439 size /= 256;
|
|
440 size -= 4;
|
|
441 size *= CHUNKSIZE;
|
|
442 tivo->size = numberParts * TIVO_PART_LENGTH;
|
|
443 tivo->size += size;
|
|
444 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
445 "ty:Header Calc Stream Size %lld\n", tivo->size );
|
|
446 }
|
|
447 else
|
|
448 {
|
|
449 tivo->size = demux->stream->end_pos;
|
|
450 }
|
|
451 }
|
|
452 else
|
|
453 {
|
|
454 tivo->size = demux->stream->end_pos;
|
|
455 }
|
|
456 }
|
|
457 else
|
|
458 {
|
|
459 tivo->size = demux->stream->end_pos;
|
|
460 }
|
|
461 }
|
|
462 if ( tivo->size > demux->stream->end_pos )
|
|
463 {
|
|
464 tivo->size = demux->stream->end_pos;
|
|
465 }
|
|
466
|
|
467 if ( demux->stream->start_pos > 0 )
|
|
468 {
|
|
469 filePos = demux->stream->start_pos;
|
|
470 }
|
|
471 stream_seek( demux->stream, filePos );
|
|
472 demux->filepos = stream_tell( demux->stream );
|
|
473 }
|
|
474 demux->movi_start = 0;
|
|
475 demux->movi_end = tivo->size;
|
|
476 }
|
|
477
|
|
478 // ======================================================================
|
|
479 // Give a clue as to where we are in the stream
|
|
480 // ======================================================================
|
|
481 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
482 "ty:ty header size %llx\n", tivo->size );
|
|
483 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
484 "ty:file end_pos %llx\n", demux->stream->end_pos );
|
|
485 // mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
486 // "ty:vstream size %llx\n", vstream_streamsize() );
|
|
487
|
|
488 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
489 "\nty:wanted current offset %llx\n", stream_tell( demux->stream ) );
|
|
490
|
|
491 if ( tivo->size > 0 )
|
|
492 {
|
|
493 if ( stream_tell( demux->stream ) > tivo->size )
|
|
494 {
|
|
495 demux->stream->eof = 1;
|
|
496 return( 0 );
|
|
497 }
|
|
498 }
|
|
499
|
|
500 // Make sure we are on a 128k boundary
|
|
501 if ( ( demux->filepos % CHUNKSIZE ) != 0 )
|
|
502 {
|
|
503 whichChunk = demux->filepos / CHUNKSIZE;
|
|
504 if ( ( demux->filepos % CHUNKSIZE ) > ( CHUNKSIZE / 2 ) )
|
|
505 {
|
|
506 whichChunk++;
|
|
507 }
|
|
508 stream_seek( demux->stream, ( whichChunk * CHUNKSIZE ) );
|
|
509 }
|
|
510
|
|
511 demux->filepos = stream_tell( demux->stream );
|
|
512 readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
|
|
513 if ( readSize != CHUNKSIZE )
|
|
514 {
|
|
515 return( 0 );
|
|
516 }
|
|
517
|
|
518 // We found a part header, skip it
|
|
519 pesFileId = tivobuffer2hostlong( &chunk[ 0x00 ] );
|
|
520 if( pesFileId == TIVO_PES_FILEID )
|
|
521 {
|
|
522 demux->filepos = stream_tell( demux->stream );
|
|
523 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Skipping PART Header\n" );
|
|
524 readSize = stream_read( demux->stream, chunk, CHUNKSIZE );
|
|
525 if ( readSize != CHUNKSIZE )
|
|
526 {
|
|
527 return( 0 );
|
|
528 }
|
|
529 }
|
|
530 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
531 "\nty:actual current offset %llx\n", ( stream_tell( demux->stream ) -
|
|
532 0x20000 ) );
|
|
533
|
|
534
|
11000
|
535 // Let's make a Video Demux Stream for Mplayer
|
10263
|
536 aid = 0x0;
|
|
537 if( !demux->v_streams[ aid ] ) new_sh_video( demux, aid );
|
|
538 if( demux->video->id == -1 ) demux->video->id = aid;
|
|
539 if( demux->video->id == aid )
|
|
540 {
|
|
541 ds = demux->video;
|
|
542 if( !ds->sh ) ds->sh = demux->v_streams[ aid ];
|
|
543 }
|
|
544
|
|
545 // ======================================================================
|
|
546 // Finally, we get to actually parse the chunk
|
|
547 // ======================================================================
|
|
548 numberRecs = chunk[ 0 ];
|
|
549 recPtr = &chunk[ 4 ];
|
|
550 offset = ( numberRecs * 16 ) + 4;
|
|
551 for ( counter = 0 ; counter < numberRecs ; counter++ )
|
|
552 {
|
|
553 size = ( recPtr[ 0 ] << 8 | recPtr[ 1 ] ) << 4 | ( recPtr[ 2 ] >> 4 );
|
|
554 type = recPtr[ 3 ];
|
|
555 nybbleType = recPtr[ 2 ] & 0x0f;
|
|
556 recordsDecoded++;
|
|
557
|
|
558 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
559 "ty:Record Type %x/%x %d\n", nybbleType, type, size );
|
|
560
|
|
561 // ================================================================
|
|
562 // Video Parsing
|
|
563 // ================================================================
|
|
564 if ( type == 0xe0 )
|
|
565 {
|
|
566 if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
|
|
567 {
|
|
568 #if 0
|
|
569 printf( "Video Chunk Header " );
|
|
570 for( count = 0 ; count < 24 ; count++ )
|
|
571 {
|
|
572 printf( "%2.2x ", chunk[ offset + count ] );
|
|
573 }
|
|
574 printf( "\n" );
|
|
575 #endif
|
|
576 demux_ty_FindESHeader( ty_VideoPacket, 4, &chunk[ offset ],
|
|
577 size, &esOffset1 );
|
|
578 if ( esOffset1 != -1 )
|
|
579 {
|
|
580 tivo->lastVideoPTS = get_ty_pts(
|
|
581 &chunk[ offset + esOffset1 + 9 ] );
|
|
582 mp_msg( MSGT_DEMUX, MSGL_DBG3, "Video PTS %7.1f\n",
|
|
583 tivo->lastVideoPTS );
|
|
584 }
|
|
585
|
|
586 // Do NOT Pass the PES Header onto the MPEG2 Decode
|
|
587 if( nybbleType != 0x06 )
|
|
588 {
|
|
589 demux_ty_CopyToDemuxPacket( TY_V, tivo, demux->video,
|
|
590 &chunk[ offset ], size, ( demux->filepos + offset ),
|
|
591 tivo->lastVideoPTS );
|
|
592 }
|
|
593 offset += size;
|
|
594 }
|
|
595 else
|
|
596 {
|
|
597 errorHeader++;
|
|
598 }
|
|
599 }
|
|
600 // ================================================================
|
|
601 // Audio Parsing
|
|
602 // ================================================================
|
|
603 else if ( type == 0xc0 )
|
|
604 {
|
|
605 if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
|
|
606 {
|
|
607 #if 0
|
|
608 printf( "Audio Chunk Header " );
|
|
609 for( count = 0 ; count < 24 ; count++ )
|
|
610 {
|
|
611 printf( "%2.2x ", chunk[ offset + count ] );
|
|
612 }
|
|
613 printf( "\n" );
|
|
614 #endif
|
|
615
|
|
616 if( demux->audio->id == -1 )
|
|
617 {
|
|
618 if ( nybbleType == 0x02 )
|
|
619 {
|
|
620 continue; // DTiVo inconclusive, wait for more
|
|
621 }
|
|
622 else if ( nybbleType == 0x09 )
|
|
623 {
|
|
624 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Setting AC-3 Audio\n" );
|
|
625 aid = 0x80; // AC-3
|
|
626 }
|
|
627 else
|
|
628 {
|
|
629 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Setting MPEG Audio\n" );
|
|
630 aid = 0x0; // MPEG Audio
|
|
631 }
|
|
632
|
|
633 demux->audio->id = aid;
|
|
634 if( !demux->a_streams[ aid ] ) new_sh_audio( demux, aid );
|
|
635 if( demux->audio->id == aid )
|
|
636 {
|
|
637 ds = demux->audio;
|
|
638 if( !ds->sh ) ds->sh = demux->a_streams[ aid ];
|
|
639 }
|
|
640 }
|
|
641
|
|
642 aid = demux->audio->id;
|
|
643
|
|
644
|
|
645 // SA DTiVo Audio Data, no PES
|
|
646 // ================================================
|
|
647 if ( nybbleType == 0x02 )
|
|
648 {
|
|
649 if ( tivo->tivoType == 2 )
|
|
650 {
|
|
651 demux_ty_AddToAudioBuffer( tivo, &chunk[ offset ], size );
|
|
652 }
|
|
653 else
|
|
654 {
|
|
655
|
|
656 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
657 "ty:Adding Audio Packet Size %d\n", size );
|
|
658 demux_ty_CopyToDemuxPacket( TY_A, tivo, demux->audio,
|
|
659 &chunk[ offset ], size, ( demux->filepos + offset ),
|
|
660 tivo->lastAudioPTS );
|
|
661 }
|
|
662 }
|
|
663
|
|
664 // MPEG Audio with PES Header, either SA or DTiVo
|
|
665 // ================================================
|
|
666 if ( nybbleType == 0x03 )
|
|
667 {
|
|
668 demux_ty_FindESHeader( ty_MPEGAudioPacket, 4, &chunk[ offset ],
|
|
669 size, &esOffset1 );
|
|
670
|
|
671 // SA PES Header, No Audio Data
|
|
672 // ================================================
|
|
673 if ( ( esOffset1 == 0 ) && ( size == 16 ) )
|
|
674 {
|
|
675 tivo->tivoType = 1;
|
|
676 tivo->lastAudioPTS = get_ty_pts( &chunk[ offset +
|
|
677 SERIES2_PTS_OFFSET ] );
|
|
678 mp_msg( MSGT_DEMUX, MSGL_DBG3, "SA Audio PTS %7.1f\n",
|
|
679 tivo->lastAudioPTS );
|
|
680 }
|
|
681 else
|
|
682 // DTiVo Audio with PES Header
|
|
683 // ================================================
|
|
684 {
|
|
685 tivo->tivoType = 2;
|
|
686
|
|
687 demux_ty_AddToAudioBuffer( tivo, &chunk[ offset ], size );
|
|
688 demux_ty_FindESPacket( ty_MPEGAudioPacket, 4,
|
|
689 tivo->lastAudio, tivo->lastAudioEnd, &esOffset1,
|
|
690 &esOffset2 );
|
|
691
|
|
692 if ( ( esOffset1 != -1 ) && ( esOffset2 != -1 ) )
|
|
693 {
|
|
694 int packetSize = esOffset2 - esOffset1;
|
|
695 int headerSize;
|
|
696 int ptsOffset;
|
|
697
|
|
698 if ( IsValidAudioPacket( packetSize, &ptsOffset,
|
|
699 &headerSize ) )
|
|
700 {
|
|
701 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
702 "ty:Adding DTiVo Audio Packet Size %d\n",
|
|
703 packetSize );
|
|
704
|
|
705 tivo->lastAudioPTS = get_ty_pts(
|
|
706 &tivo->lastAudio[ esOffset1 + ptsOffset ] );
|
|
707 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
708 "MPEG Audio PTS %7.1f\n", tivo->lastAudioPTS );
|
|
709
|
|
710 demux_ty_CopyToDemuxPacket
|
|
711 (
|
|
712 TY_A,
|
|
713 tivo,
|
|
714 demux->audio,
|
|
715 &( tivo->lastAudio[ esOffset1 + headerSize ] ),
|
|
716 ( packetSize - headerSize ),
|
|
717 ( demux->filepos + offset ),
|
|
718 tivo->lastAudioPTS
|
|
719 );
|
|
720
|
|
721 }
|
|
722
|
|
723 // Collapse the Audio Buffer
|
|
724 memmove( &(tivo->lastAudio[ 0 ] ),
|
|
725 &( tivo->lastAudio[ esOffset2 ] ),
|
|
726 ( tivo->lastAudioEnd - esOffset2 ) );
|
|
727 tivo->lastAudioEnd -= esOffset2;
|
|
728 }
|
|
729 }
|
|
730 }
|
|
731
|
|
732 // SA Audio with no PES Header
|
|
733 // ================================================
|
|
734 if ( nybbleType == 0x04 )
|
|
735 {
|
|
736 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
737 "ty:Adding Audio Packet Size %d\n", size );
|
|
738 demux_ty_CopyToDemuxPacket( TY_A, tivo, demux->audio,
|
|
739 &chunk[ offset ], size, ( demux->filepos + offset ),
|
|
740 tivo->lastAudioPTS );
|
|
741 }
|
|
742
|
|
743 // DTiVo AC3 Audio Data with PES Header
|
|
744 // ================================================
|
|
745 if ( nybbleType == 0x09 )
|
|
746 {
|
|
747 tivo->tivoType = 2;
|
|
748
|
|
749 demux_ty_AddToAudioBuffer( tivo, &chunk[ offset ], size );
|
|
750 demux_ty_FindESPacket( ty_AC3AudioPacket, 4,
|
|
751 tivo->lastAudio, tivo->lastAudioEnd, &esOffset1,
|
|
752 &esOffset2 );
|
|
753
|
|
754 if ( ( esOffset1 != -1 ) && ( esOffset2 != -1 ) )
|
|
755 {
|
|
756 int packetSize = esOffset2 - esOffset1;
|
|
757 int headerSize;
|
|
758 int ptsOffset;
|
|
759
|
|
760 if ( IsValidAudioPacket( packetSize, &ptsOffset,
|
|
761 &headerSize ) )
|
|
762 {
|
|
763 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
764 "ty:Adding DTiVo Audio Packet Size %d\n",
|
|
765 packetSize );
|
|
766
|
|
767 tivo->lastAudioPTS = get_ty_pts(
|
|
768 &tivo->lastAudio[ esOffset1 + ptsOffset ] );
|
|
769 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
770 "AC3 Audio PTS %7.1f\n", tivo->lastAudioPTS );
|
|
771
|
|
772 // AC3 Decoder WANTS the PTS
|
|
773 demux_ty_CopyToDemuxPacket
|
|
774 (
|
|
775 TY_A,
|
|
776 tivo,
|
|
777 demux->audio,
|
|
778 &( tivo->lastAudio[ esOffset1 ] ),
|
|
779 ( packetSize ),
|
|
780 ( demux->filepos + offset ),
|
|
781 tivo->lastAudioPTS
|
|
782 );
|
|
783
|
|
784 }
|
|
785
|
|
786 // Collapse the Audio Buffer
|
|
787 memmove( &(tivo->lastAudio[ 0 ] ),
|
|
788 &( tivo->lastAudio[ esOffset2 ] ),
|
|
789 ( tivo->lastAudioEnd - esOffset2 ) );
|
|
790 tivo->lastAudioEnd -= esOffset2;
|
|
791 }
|
|
792 }
|
|
793 offset += size;
|
|
794 }
|
|
795 else
|
|
796 {
|
|
797 errorHeader++;
|
|
798 }
|
|
799 }
|
|
800 // ================================================================
|
|
801 // Closed Caption
|
|
802 // ================================================================
|
|
803 else if ( type == 0x01 )
|
|
804 {
|
|
805 unsigned char b1;
|
|
806 unsigned char b2;
|
|
807 unsigned char buffer[ 16 ];
|
|
808
|
|
809 b1 = ( ( ( recPtr[ 0 ] & 0x0f ) << 4 ) |
|
|
810 ( ( recPtr[ 1 ] & 0xf0 ) >> 4 ) );
|
|
811 b1 &= 0x7f;
|
|
812 b2 = ( ( ( recPtr[ 1 ] & 0x0f ) << 4 ) |
|
|
813 ( ( recPtr[ 2 ] & 0xf0 ) >> 4 ) );
|
|
814 b2 &= 0x7f;
|
|
815
|
|
816 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:CC %x %x\n", b1, b2 );
|
|
817
|
|
818 buffer[ 0x00 ] = 0x00;
|
|
819 buffer[ 0x01 ] = 0x00;
|
|
820 buffer[ 0x02 ] = 0x01;
|
|
821 buffer[ 0x03 ] = 0xb2;
|
|
822 buffer[ 0x04 ] = 'T';
|
|
823 buffer[ 0x05 ] = 'Y';
|
|
824 buffer[ 0x06 ] = 0x01;
|
|
825 buffer[ 0x07 ] = b1;
|
|
826 buffer[ 0x08 ] = b2;
|
|
827 demux_ty_CopyToDemuxPacket( TY_V, tivo, demux->video, buffer, 0x09,
|
|
828 ( demux->filepos + offset ), tivo->lastVideoPTS );
|
|
829 }
|
|
830 // ================================================================
|
|
831 // Extended Data Services
|
|
832 // ================================================================
|
|
833 else if ( type == 0x02 )
|
|
834 {
|
|
835 unsigned char b1;
|
|
836 unsigned char b2;
|
|
837 unsigned char buffer[ 16 ];
|
|
838
|
|
839 b1 = ( ( ( recPtr[ 0 ] & 0x0f ) << 4 ) |
|
|
840 ( ( recPtr[ 1 ] & 0xf0 ) >> 4 ) );
|
|
841 b1 &= 0x7f;
|
|
842 b2 = ( ( ( recPtr[ 1 ] & 0x0f ) << 4 ) |
|
|
843 ( ( recPtr[ 2 ] & 0xf0 ) >> 4 ) );
|
|
844 b2 &= 0x7f;
|
|
845
|
|
846 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:XDS %x %x\n", b1, b2 );
|
|
847
|
|
848 buffer[ 0x00 ] = 0x00;
|
|
849 buffer[ 0x01 ] = 0x00;
|
|
850 buffer[ 0x02 ] = 0x01;
|
|
851 buffer[ 0x03 ] = 0xb2;
|
|
852 buffer[ 0x04 ] = 'T';
|
|
853 buffer[ 0x05 ] = 'Y';
|
|
854 buffer[ 0x06 ] = 0x02;
|
|
855 buffer[ 0x07 ] = b1;
|
|
856 buffer[ 0x08 ] = b2;
|
|
857 demux_ty_CopyToDemuxPacket( TY_V, tivo, demux->video, buffer, 0x09,
|
|
858 ( demux->filepos + offset ), tivo->lastVideoPTS );
|
|
859 }
|
|
860 // ================================================================
|
|
861 // Found a 0x03 on Droid's TiVo, I have no idea what it is
|
|
862 // ================================================================
|
|
863 else if ( type == 0x03 )
|
|
864 {
|
|
865 if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
|
|
866 {
|
|
867 offset += size;
|
|
868 }
|
|
869 }
|
|
870 // ================================================================
|
|
871 // Unknown
|
|
872 // ================================================================
|
|
873 else if ( type == 0x05 )
|
|
874 {
|
|
875 if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
|
|
876 {
|
|
877 offset += size;
|
|
878 }
|
|
879 }
|
|
880 else
|
|
881 {
|
|
882 if ( ( size > 0 ) && ( ( size + offset ) <= CHUNKSIZE ) )
|
|
883 {
|
|
884 offset += size;
|
|
885 }
|
|
886 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Invalid Type %x\n", type );
|
|
887 invalidType++;
|
|
888 }
|
|
889 recPtr += 16;
|
|
890 }
|
|
891
|
|
892 if ( errorHeader > 0 )
|
|
893 {
|
|
894 mp_msg( MSGT_DEMUX, MSGL_DBG3,
|
|
895 "ty:Error Check - Records %d, Parsed %d, Errors %d\n",
|
|
896 numberRecs, recordsDecoded, errorHeader );
|
|
897
|
|
898 // Invalid MPEG ES Size Check
|
|
899 if ( errorHeader > ( numberRecs / 2 ) )
|
|
900 {
|
|
901 return( 0 );
|
|
902 }
|
|
903
|
|
904 // Invalid MPEG Stream Type Check
|
|
905 if ( invalidType > ( numberRecs / 2 ) )
|
|
906 {
|
|
907 return( 0 );
|
|
908 }
|
|
909 }
|
|
910
|
|
911 demux->filepos = stream_tell( demux->stream );
|
|
912
|
|
913 return( 1 );
|
|
914 }
|
|
915
|
|
916 void demux_seek_ty( demuxer_t *demuxer, float rel_seek_secs, int flags )
|
|
917 {
|
|
918 demux_stream_t *d_audio = demuxer->audio;
|
|
919 demux_stream_t *d_video = demuxer->video;
|
|
920 sh_audio_t *sh_audio = d_audio->sh;
|
|
921 sh_video_t *sh_video = d_video->sh;
|
|
922 off_t newpos;
|
|
923 off_t res;
|
|
924 TiVoInfo *tivo = 0;
|
|
925
|
|
926 mp_msg( MSGT_DEMUX, MSGL_DBG3, "ty:Seeking to %7.1f\n", rel_seek_secs );
|
|
927
|
|
928 if ( ( demuxer->a_streams[ MAX_A_STREAMS - 1 ] ) != 0 )
|
|
929 {
|
|
930 tivo = demuxer->a_streams[ MAX_A_STREAMS - 1 ];
|
|
931 tivo->lastAudioEnd = 0;
|
|
932 tivo->lastAudioPTS = 0;
|
|
933 tivo->lastVideoPTS = 0;
|
|
934 }
|
|
935 //
|
|
936 //================= seek in MPEG ==========================
|
|
937 demuxer->filepos = stream_tell( demuxer->stream );
|
|
938
|
|
939 newpos = ( flags & 1 ) ? demuxer->movi_start : demuxer->filepos;
|
|
940
|
|
941 if( flags & 2 )
|
|
942 {
|
|
943 // float seek 0..1
|
|
944 newpos += ( demuxer->movi_end - demuxer->movi_start ) * rel_seek_secs;
|
|
945 }
|
|
946 else
|
|
947 {
|
|
948 // time seek (secs)
|
|
949 if( ! sh_video->i_bps ) // unspecified or VBR
|
|
950 {
|
|
951 newpos += 2324 * 75 * rel_seek_secs; // 174.3 kbyte/sec
|
|
952 }
|
|
953 else
|
|
954 {
|
|
955 newpos += sh_video->i_bps * rel_seek_secs;
|
|
956 }
|
|
957 }
|
|
958
|
|
959 if ( newpos < demuxer->movi_start )
|
|
960 {
|
|
961 if( demuxer->stream->type != STREAMTYPE_VCD ) demuxer->movi_start = 0;
|
|
962 if( newpos < demuxer->movi_start ) newpos = demuxer->movi_start;
|
|
963 }
|
|
964
|
|
965 res = newpos / CHUNKSIZE;
|
|
966 if ( rel_seek_secs >= 0 )
|
|
967 {
|
|
968 newpos = ( res + 1 ) * CHUNKSIZE;
|
|
969 }
|
|
970 else
|
|
971 {
|
|
972 newpos = res * CHUNKSIZE;
|
|
973 }
|
|
974
|
|
975 if ( newpos < 0 )
|
|
976 {
|
|
977 newpos = 0;
|
|
978 }
|
|
979 stream_seek( demuxer->stream, newpos );
|
|
980
|
|
981 // re-sync video:
|
|
982 videobuf_code_len = 0; // reset ES stream buffer
|
|
983
|
|
984 ds_fill_buffer( d_video );
|
|
985 if( sh_audio )
|
|
986 {
|
|
987 ds_fill_buffer( d_audio );
|
|
988 resync_audio_stream( sh_audio );
|
|
989 }
|
|
990
|
|
991 while( 1 )
|
|
992 {
|
|
993 int i;
|
|
994 if( sh_audio && !d_audio->eof && d_video->pts && d_audio->pts )
|
|
995 {
|
|
996 float a_pts = d_audio->pts;
|
|
997 a_pts += ( ds_tell_pts( d_audio ) - sh_audio->a_in_buffer_len ) /
|
|
998 (float)sh_audio->i_bps;
|
|
999 if( d_video->pts > a_pts )
|
|
1000 {
|
|
1001 skip_audio_frame( sh_audio ); // sync audio
|
|
1002 continue;
|
|
1003 }
|
|
1004 }
|
|
1005 i = sync_video_packet( d_video );
|
|
1006 if( i == 0x1B3 || i == 0x1B8 ) break; // found it!
|
|
1007 if( !i || !skip_video_packet( d_video ) ) break; // EOF?
|
|
1008 }
|
|
1009 if ( subcc_enabled )
|
|
1010 {
|
|
1011 ty_ClearOSD( 0 );
|
|
1012 }
|
|
1013 }
|
|
1014
|
|
1015 int demux_ty_control( demuxer_t *demuxer,int cmd, void *arg )
|
|
1016 {
|
|
1017 demux_stream_t *d_video = demuxer->video;
|
|
1018 sh_video_t *sh_video = d_video->sh;
|
|
1019
|
|
1020 switch(cmd)
|
|
1021 {
|
|
1022 case DEMUXER_CTRL_GET_TIME_LENGTH:
|
|
1023 if(!sh_video->i_bps) // unspecified or VBR
|
|
1024 return DEMUXER_CTRL_DONTKNOW;
|
|
1025 *((unsigned long *)arg)=
|
|
1026 (demuxer->movi_end-demuxer->movi_start)/sh_video->i_bps;
|
|
1027 return DEMUXER_CTRL_GUESS;
|
|
1028
|
|
1029 case DEMUXER_CTRL_GET_PERCENT_POS:
|
|
1030 if (demuxer->movi_end==demuxer->movi_start)
|
|
1031 return DEMUXER_CTRL_DONTKNOW;
|
|
1032 *((int *)arg)=
|
|
1033 (int)((demuxer->filepos-demuxer->movi_start)/
|
|
1034 ((demuxer->movi_end-demuxer->movi_start)/100));
|
|
1035 return DEMUXER_CTRL_OK;
|
|
1036 default:
|
|
1037 return DEMUXER_CTRL_NOTIMPL;
|
|
1038 }
|
|
1039 }
|
|
1040
|
|
1041
|
|
1042 int demux_close_ty( demuxer_t *demux )
|
|
1043 {
|
|
1044 TiVoInfo *tivo = 0;
|
|
1045
|
|
1046 if ( ( demux->a_streams[ MAX_A_STREAMS - 1 ] ) != 0 )
|
|
1047 {
|
|
1048 tivo = demux->a_streams[ MAX_A_STREAMS - 1 ];
|
|
1049 free( tivo );
|
|
1050 demux->a_streams[ MAX_A_STREAMS - 1 ] = 0;
|
|
1051 sub_justify = 0;
|
|
1052 }
|
|
1053 return( 0 );
|
|
1054 }
|
|
1055
|
|
1056
|