Mercurial > mplayer.hg
comparison libmpdemux/demux_ty_osd.c @ 10263:0df8816f4665
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)
author | arpi |
---|---|
date | Mon, 09 Jun 2003 00:24:49 +0000 |
parents | |
children | 68e714ed669f |
comparison
equal
deleted
inserted
replaced
10262:e6f5487b2042 | 10263:0df8816f4665 |
---|---|
1 // Most of this was written by mbm@linux.com and released on the GPL2 License. | |
2 // | |
3 // Modifications and SEVERE cleanup of the code was done by | |
4 // Christopher Wingert | |
5 // Copyright 2003 | |
6 // | |
7 // Released under GPL2 License. | |
8 | |
9 #include <stdio.h> | |
10 #include <stdlib.h> | |
11 #include <unistd.h> | |
12 #include <time.h> | |
13 #include <stdarg.h> | |
14 | |
15 #include "config.h" | |
16 #include "mp_msg.h" | |
17 #include "help_mp.h" | |
18 | |
19 //#include "stream.h" | |
20 //#include "demuxer.h" | |
21 //#include "parse_es.h" | |
22 //#include "stheader.h" | |
23 //#include "mp3_hdr.h" | |
24 //#include "../subreader.h" | |
25 #include "../sub_cc.h" | |
26 #include "../libvo/sub.h" | |
27 | |
28 //#include "dvdauth.h" | |
29 | |
30 extern int sub_justify; | |
31 | |
32 #define TY_TEXT_MODE ( 1 << 0 ) | |
33 #define TY_OSD_MODE ( 1 << 1 ) | |
34 | |
35 static int TY_OSD_flags = TY_TEXT_MODE | TY_OSD_MODE; | |
36 static int TY_OSD_debug = 0; | |
37 | |
38 // =========================================================================== | |
39 // Closed Caption Decoding and OSD Presentation | |
40 // =========================================================================== | |
41 #define TY_CCNONE ( -3 ) | |
42 #define TY_CCTEXTMODE ( -2 ) | |
43 #define TY_CCPOPUPNB ( -1 ) | |
44 #define TY_CCPOPUP ( 0 ) | |
45 #define TY_CCPAINTON ( 1 ) | |
46 | |
47 #define TY_CC_MAX_X ( 45 ) | |
48 | |
49 static int TY_CC_CUR_X; | |
50 static int TY_CC_CUR_Y; | |
51 static int TY_CC_stat = TY_CCNONE; | |
52 static char TY_CC_buf[ 255 ]; | |
53 static char *TY_CC_ptr = TY_CC_buf; | |
54 static unsigned TY_CC_lastcap = 0; | |
55 static int TY_CC_TextItalic; | |
56 static int TY_CC_Y_Offset; | |
57 | |
58 static subtitle ty_OSD1; | |
59 static subtitle ty_OSD2; | |
60 static subtitle *ty_pOSD1; | |
61 static subtitle *ty_pOSD2; | |
62 static int tyOSDInited = 0; | |
63 static int tyOSDUpdate = 0; | |
64 | |
65 static void ty_DrawOSD() | |
66 { | |
67 // printf( "Calling ty_DrawOSD()\n" ); | |
68 tyOSDUpdate = 1; | |
69 } | |
70 | |
71 void ty_ClearOSD( int start ) | |
72 { | |
73 int index; | |
74 // printf( "Calling ty_ClearOSD()\n" ); | |
75 for ( index = start ; index < SUB_MAX_TEXT ; index++ ) | |
76 { | |
77 memset( ty_OSD1.text[ index ], ' ', TY_CC_MAX_X - 1 ); | |
78 ty_OSD1.text[ index ][ TY_CC_MAX_X - 1 ] = 0; | |
79 memset( ty_OSD2.text[ index ], ' ', TY_CC_MAX_X - 1 ); | |
80 ty_OSD2.text[ index ][ TY_CC_MAX_X - 1 ] = 0; | |
81 } | |
82 } | |
83 | |
84 static void ty_DrawChar( int *x, int *y, char disChar, int fgColor, int bgColor ) | |
85 { | |
86 int index; | |
87 int cx; | |
88 int cy; | |
89 | |
90 cx = *x; | |
91 cy = *y; | |
92 | |
93 if ( *x >= ( TY_CC_MAX_X - 1 ) ) | |
94 { | |
95 cx = 0; | |
96 } | |
97 if ( ( *y + TY_CC_Y_Offset ) > SUB_MAX_TEXT ) | |
98 { | |
99 cy = SUB_MAX_TEXT - TY_CC_Y_Offset - 1; | |
100 } | |
101 | |
102 // printf( "Calling ty_DrawChar() x:%d y:%d %c fg:%d bg:%d\n", | |
103 // cx, cy, disChar, fgColor, bgColor ); | |
104 | |
105 ty_OSD1.text[ TY_CC_Y_Offset + cy ][ cx ] = disChar; | |
106 memset( &( ty_OSD1.text[ TY_CC_Y_Offset + cy ][ cx + 1 ] ), ' ', | |
107 TY_CC_MAX_X - cx - 2 ); | |
108 ( *x )++; | |
109 } | |
110 | |
111 static void ty_RollupBuf( int dest, int source, int numLines ) | |
112 { | |
113 int index; | |
114 | |
115 // printf( "Calling ty_RollupBuf() dest:%d source %d, numLines %d\n", | |
116 // dest, source, numLines ); | |
117 // | |
118 if ( ( source + TY_CC_Y_Offset + numLines ) > SUB_MAX_TEXT ) | |
119 { | |
120 ty_ClearOSD( 1 ); | |
121 return; | |
122 } | |
123 | |
124 if ( ( source + TY_CC_Y_Offset + numLines ) < 0 ) | |
125 { | |
126 ty_ClearOSD( 1 ); | |
127 return; | |
128 } | |
129 | |
130 if ( numLines > SUB_MAX_TEXT ) | |
131 { | |
132 ty_ClearOSD( 1 ); | |
133 return; | |
134 } | |
135 | |
136 for ( index = 0 ; index < numLines ; index++ ) | |
137 { | |
138 strcpy( ty_OSD1.text[ TY_CC_Y_Offset + dest ], | |
139 ty_OSD1.text[ TY_CC_Y_Offset + source ] ); | |
140 dest++; | |
141 source++; | |
142 } | |
143 memset( ty_OSD1.text[ TY_CC_Y_Offset + source - 1 ], ' ', TY_CC_MAX_X - 1 ); | |
144 ty_OSD1.text[ TY_CC_Y_Offset + source - 1 ][ TY_CC_MAX_X - 1 ] = 0; | |
145 } | |
146 | |
147 static void ty_drawchar( char c ) | |
148 { | |
149 if ( c < 2 ) return; | |
150 | |
151 if ( TY_OSD_flags & TY_OSD_MODE && TY_CC_stat != TY_CCNONE && | |
152 TY_CC_CUR_Y != -1 ) | |
153 ty_DrawChar( &TY_CC_CUR_X, &TY_CC_CUR_Y, c, 4, 13 ); | |
154 | |
155 if ( TY_CC_ptr - TY_CC_buf > sizeof( TY_CC_buf ) - 1 ) | |
156 { // buffer overflow | |
157 TY_CC_ptr = TY_CC_buf; | |
158 memset( TY_CC_buf, 0, sizeof( TY_CC_buf ) ); | |
159 } | |
160 *( TY_CC_ptr++ ) = ( c == 14 ) ? '/' : c; // swap a '/' for musical note | |
161 } | |
162 | |
163 static void ty_draw() | |
164 { | |
165 if ( TY_CC_ptr != TY_CC_buf && TY_OSD_flags & TY_TEXT_MODE ) | |
166 { | |
167 if ( *( TY_CC_ptr - 1 ) == '\n' ) *( TY_CC_ptr - 1 ) = 0; | |
168 | |
169 mp_msg( MSGT_DEMUX, MSGL_V, "CC: %s\n", TY_CC_buf ); | |
170 } | |
171 TY_CC_lastcap = time( NULL ); | |
172 | |
173 TY_CC_ptr = TY_CC_buf; | |
174 memset( TY_CC_buf, 0, sizeof( TY_CC_buf) ); | |
175 | |
176 if ( TY_OSD_flags & TY_OSD_MODE ) ty_DrawOSD(); | |
177 if ( TY_CC_TextItalic ) TY_CC_TextItalic = 0; | |
178 } | |
179 | |
180 | |
181 static int CC_last = 0; | |
182 static char CC_mode = 0; | |
183 static int CC_row[] = | |
184 { | |
185 11, -1, 1, 2, 3, 4, 12, 13, 14, 15, 5, 6, 7, 8, 9, 10 | |
186 }; | |
187 | |
188 // char specialchar[] = { '®', '°', '½', '¿', '*', '¢', '£', 14, 'à', ' ', 'è', 'â', 'ê', 'î', 'ô', 'û' }; | |
189 | |
190 static int ty_CCdecode( char b1, char b2 ) | |
191 { | |
192 int x; | |
193 int data = ( b2 << 8 ) + b1; | |
194 | |
195 if ( b1 & 0x60 ) // text | |
196 { | |
197 if ( !TY_OSD_debug && TY_CC_stat == TY_CCNONE ) return 0; | |
198 if ( TY_OSD_debug > 3 ) | |
199 { | |
200 mp_msg( MSGT_DEMUX, MSGL_DBG3, "%c %c", b1, b2 ); | |
201 } | |
202 ty_drawchar( b1 ); | |
203 ty_drawchar( b2 ); | |
204 | |
205 if ( TY_CC_stat > 0 && TY_OSD_flags & TY_OSD_MODE ) ty_DrawOSD(); | |
206 } | |
207 else if ( ( b1 & 0x10 ) && ( b2 > 0x1F ) && ( data != CC_last ) ) | |
208 { | |
209 #define CURRENT ( ( b1 & 0x08 ) >> 3 ) | |
210 | |
211 if ( CC_mode != CURRENT && TY_CC_stat != TY_CCNONE ) | |
212 { | |
213 if ( TY_OSD_debug && TY_CC_ptr != TY_CC_buf ) ty_draw(); | |
214 TY_CC_stat = TY_CCNONE; | |
215 return 0; | |
216 } | |
217 | |
218 if ( TY_CC_stat == TY_CCNONE || TY_CC_CUR_Y == -1 ) | |
219 { | |
220 if ( TY_CC_ptr != TY_CC_buf ) | |
221 { | |
222 if ( TY_OSD_debug ) | |
223 mp_msg( MSGT_DEMUX, MSGL_DBG3, "(TY_OSD_debug) %s\n", | |
224 TY_CC_buf ); | |
225 TY_CC_ptr = TY_CC_buf; | |
226 memset(TY_CC_buf, 0, sizeof(TY_CC_buf)); | |
227 } | |
228 | |
229 if ( CC_mode != CURRENT ) return 0; | |
230 } | |
231 | |
232 // preamble address code (row & indent) | |
233 if ( b2 & 0x40 ) | |
234 { | |
235 TY_CC_CUR_Y = CC_row[ ( ( b1 << 1 ) & 14 ) | ( ( b2 >> 5 ) & 1 ) ]; | |
236 | |
237 // Offset into MPlayer's Buffer | |
238 if ( ( TY_CC_CUR_Y >= 1 ) && ( TY_CC_CUR_Y <= 4 ) ) | |
239 { | |
240 TY_CC_Y_Offset = SUB_MAX_TEXT - 5 - 1; | |
241 } | |
242 if ( ( TY_CC_CUR_Y >= 5 ) && ( TY_CC_CUR_Y <= 10 ) ) | |
243 { | |
244 TY_CC_Y_Offset = SUB_MAX_TEXT - 5 - 5; | |
245 } | |
246 if ( ( TY_CC_CUR_Y >= 12 ) && ( TY_CC_CUR_Y <= 15 ) ) | |
247 { | |
248 TY_CC_Y_Offset = SUB_MAX_TEXT - 5 - 12; | |
249 } | |
250 | |
251 if ( TY_OSD_debug > 3 ) | |
252 mp_msg( MSGT_DEMUX, MSGL_DBG3, "<< preamble %d >>\n", TY_CC_CUR_Y ); | |
253 | |
254 // we still have something in the text buffer | |
255 if (TY_CC_ptr != TY_CC_buf) | |
256 { | |
257 *(TY_CC_ptr++) = '\n'; | |
258 if ( TY_CC_TextItalic ) | |
259 { | |
260 TY_CC_TextItalic = 0; | |
261 } | |
262 } | |
263 | |
264 TY_CC_CUR_X = 1; | |
265 // row contains indent flag | |
266 if ( b2 & 0x10 ) | |
267 { | |
268 for ( x = 0 ; x < ( ( b2 & 0x0F ) << 1 ) ; x++ ) | |
269 { | |
270 TY_CC_CUR_X++; | |
271 *(TY_CC_ptr++) = ' '; | |
272 } | |
273 } | |
274 } | |
275 else | |
276 // !(b2 & 0x40) | |
277 { | |
278 if ( TY_OSD_debug > 3 ) | |
279 mp_msg( MSGT_DEMUX, MSGL_DBG3, "<< %02x >>\n", b1 & 0x7 ); | |
280 switch (b1 & 0x07) | |
281 { | |
282 case 0x00: // attribute | |
283 { | |
284 if ( TY_OSD_debug > 1 ) | |
285 mp_msg( MSGT_DEMUX, MSGL_DBG3, "<<A: %d>>\n", b2 ); | |
286 break; | |
287 } | |
288 case 0x01: // midrow or char | |
289 { | |
290 switch (b2 & 0x70) | |
291 { | |
292 case 0x20: // midrow attribute change | |
293 { | |
294 switch (b2 & 0x0e) | |
295 { | |
296 case 0x00: // italics off | |
297 { | |
298 TY_CC_TextItalic = 0; | |
299 *(TY_CC_ptr++) = ' '; | |
300 break; | |
301 } | |
302 case 0x0e: // italics on | |
303 { | |
304 ty_drawchar(' '); | |
305 TY_CC_TextItalic = 1; | |
306 break; | |
307 } | |
308 default: | |
309 { | |
310 if ( TY_OSD_debug > 1 ) | |
311 mp_msg( MSGT_DEMUX, MSGL_DBG3, "<<D: %d>>\n", | |
312 b2 & 0x0e ); | |
313 } | |
314 } | |
315 if ( b2 & 0x01 ) | |
316 { | |
317 // TextUnderline = 1; | |
318 } | |
319 else | |
320 { | |
321 // TextUnderline = 0; | |
322 } | |
323 break; | |
324 } | |
325 case 0x30: // special character.. | |
326 { | |
327 // transparent space | |
328 if ( ( b2 & 0x0f ) == 9 ) | |
329 { | |
330 TY_CC_CUR_X++; | |
331 *(TY_CC_ptr++) = ' '; | |
332 } | |
333 else | |
334 { | |
335 // ty_drawchar(specialchar[ b2 & 0x0f ] ); | |
336 ty_drawchar( ' ' ); | |
337 } | |
338 break; | |
339 } | |
340 } | |
341 break; | |
342 } | |
343 | |
344 case 0x04: // misc | |
345 case 0x05: // misc + F | |
346 { | |
347 if ( TY_OSD_debug > 3 ) | |
348 mp_msg( MSGT_DEMUX, MSGL_DBG3, "<< misc %02x >>\n", b2 ); | |
349 switch ( b2 ) | |
350 { | |
351 case 0x20: // resume caption (new caption) | |
352 { | |
353 if ( TY_OSD_flags & TY_OSD_MODE && | |
354 TY_CC_stat != TY_CCPOPUP ) | |
355 ty_ClearOSD( 1 ); | |
356 TY_CC_stat = TY_CCPOPUP; | |
357 break; | |
358 } | |
359 | |
360 case 0x21: // backspace | |
361 { | |
362 TY_CC_CUR_X--; | |
363 break; | |
364 } | |
365 | |
366 case 0x25 ... 0x27: // 2-4 row captions | |
367 { | |
368 if ( TY_CC_stat == TY_CCPOPUP ) ty_ClearOSD( 1 ); | |
369 TY_CC_stat = b2 - 0x23; | |
370 if ( TY_CC_CUR_Y < TY_CC_stat ) TY_CC_CUR_Y = TY_CC_stat; | |
371 break; | |
372 } | |
373 | |
374 case 0x29: // resume direct caption | |
375 { | |
376 TY_CC_stat = TY_CCPAINTON; | |
377 break; | |
378 } | |
379 | |
380 case 0x2A: // text restart | |
381 { | |
382 ty_draw(); | |
383 /* FALL */ | |
384 } | |
385 | |
386 case 0x2B: // resume text display | |
387 { | |
388 TY_CC_stat = TY_CCTEXTMODE; | |
389 break; | |
390 } | |
391 | |
392 case 0x2C: // erase displayed memory | |
393 { | |
394 TY_CC_lastcap = 0; | |
395 if ( TY_OSD_flags & TY_OSD_MODE ) | |
396 { | |
397 if ( TY_CC_stat > TY_CCPOPUP || TY_CC_ptr == TY_CC_buf ) | |
398 { | |
399 ty_ClearOSD( 1 ); | |
400 ty_draw(); | |
401 } | |
402 else | |
403 { | |
404 ty_ClearOSD( 1 ); | |
405 | |
406 // CRW - | |
407 // new buffer | |
408 // Used to be a buffer swap here, dunno why | |
409 } | |
410 } | |
411 break; | |
412 } | |
413 | |
414 case 0x2D: // carriage return | |
415 { | |
416 ty_draw(); | |
417 TY_CC_CUR_X = 1; | |
418 if ( TY_OSD_flags & TY_OSD_MODE ) | |
419 { | |
420 if ( TY_CC_stat > TY_CCPAINTON ) | |
421 ty_RollupBuf | |
422 ( | |
423 TY_CC_CUR_Y - TY_CC_stat + 1 , | |
424 TY_CC_CUR_Y - TY_CC_stat + 2, | |
425 TY_CC_stat - 1 | |
426 ); | |
427 else | |
428 TY_CC_CUR_Y++; | |
429 } | |
430 break; | |
431 } | |
432 | |
433 case 0x2F: // end caption + swap memory | |
434 { | |
435 ty_draw(); | |
436 /* FALL THROUGH TO 0x2E */ | |
437 } | |
438 | |
439 case 0x2E: // erase non-displayed memory | |
440 { | |
441 if ( TY_OSD_debug && TY_CC_ptr != TY_CC_buf ) | |
442 mp_msg( MSGT_DEMUX, MSGL_DBG3, "(TY_OSD_debug) %s\n", | |
443 TY_CC_buf ); | |
444 if ( TY_OSD_flags & TY_OSD_MODE ) ty_ClearOSD( 1 ); | |
445 | |
446 TY_CC_CUR_X = 1; | |
447 TY_CC_CUR_Y = -1; | |
448 | |
449 TY_CC_ptr = TY_CC_buf; | |
450 memset( TY_CC_buf, 0, sizeof( TY_CC_buf ) ); | |
451 } | |
452 } | |
453 break; | |
454 } | |
455 case 0x07: // misc (TAB) | |
456 { | |
457 for ( x = 0 ; x < ( b2 - 0x20 ) ; x++ ) | |
458 TY_CC_CUR_X++; | |
459 break; | |
460 } | |
461 } | |
462 } | |
463 } | |
464 CC_last = data; | |
465 return 0; | |
466 } | |
467 | |
468 // =========================================================================== | |
469 // Extended Data Service Decoding and OSD Presentation | |
470 // =========================================================================== | |
471 #define XDS_BUFFER_LENGTH ( 16 ) | |
472 #define XDS_DISPLAY_FRAMES ( 120 ) | |
473 static char *ty_XDS_Display[ XDS_BUFFER_LENGTH ]; | |
474 static int ty_XDSAddLine = -1; | |
475 static int ty_XDSDisplayCount = -1; | |
476 | |
477 | |
478 static void ty_AddXDSToDisplay( char *format, ... ) | |
479 { | |
480 char line[ 80 ]; | |
481 int index; | |
482 va_list ap; | |
483 | |
484 if ( ty_XDSAddLine == -1 ) | |
485 { | |
486 for( index = 0 ; index < XDS_BUFFER_LENGTH ; index++ ) | |
487 { | |
488 ty_XDS_Display[ index ] = 0; | |
489 } | |
490 ty_XDSAddLine = 0; | |
491 } | |
492 | |
493 va_start( ap, format ); | |
494 vsnprintf( line, 80, format, ap ); | |
495 va_end( ap ); | |
496 mp_msg( MSGT_DEMUX, MSGL_V, "XDS: %s\n", line ); | |
497 | |
498 if ( ty_XDSAddLine == XDS_BUFFER_LENGTH ) | |
499 { | |
500 mp_msg( MSGT_DEMUX, MSGL_ERR, "XDS Buffer would have been blown\n" ); | |
501 } | |
502 | |
503 if ( ty_XDS_Display[ ty_XDSAddLine ] != 0 ) | |
504 { | |
505 free( ty_XDS_Display[ ty_XDSAddLine ] ); | |
506 ty_XDS_Display[ ty_XDSAddLine ] = 0; | |
507 } | |
508 | |
509 ty_XDS_Display[ ty_XDSAddLine ] = malloc( strlen( line ) + 1 ); | |
510 strcpy( ty_XDS_Display[ ty_XDSAddLine ], line ); | |
511 ty_XDSAddLine++; | |
512 } | |
513 | |
514 | |
515 static void ty_DisplayXDSInfo() | |
516 { | |
517 int index; | |
518 int size; | |
519 | |
520 if ( ty_XDSDisplayCount == -1 ) | |
521 { | |
522 for( index = 0 ; index < XDS_BUFFER_LENGTH ; index++ ) | |
523 { | |
524 if ( ty_XDS_Display[ index ] != 0 ) | |
525 { | |
526 break; | |
527 } | |
528 } | |
529 if ( index != XDS_BUFFER_LENGTH ) | |
530 { | |
531 size = strlen( ty_XDS_Display[ index ] ); | |
532 | |
533 // Right Justify the XDS Stuff | |
534 memcpy( &( ty_OSD1.text[ 0 ][ TY_CC_MAX_X - size - 1 ] ), | |
535 ty_XDS_Display[ index ], size ); | |
536 free( ty_XDS_Display[ index ] ); | |
537 ty_XDS_Display[ index ] = 0; | |
538 ty_XDSDisplayCount = 0; | |
539 tyOSDUpdate = 1; | |
540 | |
541 } | |
542 else | |
543 { | |
544 // We cleaned out all the XDS stuff to be displayed | |
545 ty_XDSAddLine = 0; | |
546 } | |
547 } | |
548 else | |
549 { | |
550 // We displayed that piece of XDS information long enough | |
551 // Lets move on | |
552 ty_XDSDisplayCount++; | |
553 if ( ty_XDSDisplayCount >= XDS_DISPLAY_FRAMES ) | |
554 { | |
555 memset( ty_OSD1.text[ 0 ], ' ', TY_CC_MAX_X - 1 ); | |
556 ty_OSD1.text[ 0 ][ TY_CC_MAX_X - 1 ] = 0; | |
557 ty_XDSDisplayCount = -1; | |
558 tyOSDUpdate = 1; | |
559 } | |
560 } | |
561 } | |
562 | |
563 | |
564 static int TY_XDS_mode = 0; | |
565 static int TY_XDS_type = 0; | |
566 static int TY_XDS_length = 0; | |
567 static char TY_XDS_checksum = 0; | |
568 | |
569 // Array of [ Mode ][ Type ][ Length ] | |
570 static char TY_XDS [ 8 ][ 25 ][ 34 ]; | |
571 static char TY_XDS_new[ 8 ][ 25 ][ 34 ]; | |
572 | |
573 // Array of [ MPAARating|TVRating ][ NumberRatings ] | |
574 static char *TY_XDS_CHIP[ 2 ][ 8 ] = | |
575 { | |
576 { "(NOT APPLICABLE)", "G", "PG", "PG-13", "R", "NC-17", "X", "(NOT RATED)" }, | |
577 { "(NOT RATED)", "TV-Y", "TV-Y7", "TV-G", "TV-PG", "TV-14", "TV-MA", | |
578 "(NOT RATED)" } | |
579 }; | |
580 | |
581 static char *TY_XDS_modes[] = | |
582 { | |
583 "CURRENT", // 01h-02h current program | |
584 "FUTURE ", // 03h-04h future program | |
585 "CHANNEL", // 05h-06h channel | |
586 "MISC. ", // 07h-08h miscellaneous | |
587 "PUBLIC ", // 09h-0Ah public service | |
588 "RESERV.", // 0Bh-0Ch reserved | |
589 "UNDEF. ", | |
590 "INVALID", | |
591 "INVALID", | |
592 "INVALID" | |
593 }; | |
594 | |
595 static int ty_XDSdecode( char b1, char b2 ) | |
596 { | |
597 char line[ 80 ]; | |
598 | |
599 if ( b1 < 0x0F ) | |
600 { // start packet | |
601 TY_XDS_length = 0; | |
602 TY_XDS_mode = b1 >> 1; // every other mode is a resume | |
603 TY_XDS_type = b2; | |
604 TY_XDS_checksum = b1 + b2; | |
605 return 0; | |
606 } | |
607 | |
608 TY_XDS_checksum += b1 + b2; | |
609 | |
610 // eof (next byte is checksum) | |
611 if ( b1 == 0x0F ) | |
612 { | |
613 // validity check | |
614 if ( !TY_XDS_length || TY_XDS_checksum & 0x7F ) | |
615 { | |
616 if ( TY_OSD_debug > 3 && !TY_XDS_length ) | |
617 { | |
618 mp_msg( MSGT_DEMUX, MSGL_DBG3, | |
619 "%% TY_XDS CHECKSUM ERROR (ignoring)\n" ); | |
620 } | |
621 else | |
622 { | |
623 TY_XDS_mode = 0; | |
624 TY_XDS_type = 0; | |
625 return 1; | |
626 } | |
627 } | |
628 | |
629 // check to see if the data has changed. | |
630 if ( strncmp( TY_XDS[ TY_XDS_mode ][ TY_XDS_type ], | |
631 TY_XDS_new[ TY_XDS_mode ][ TY_XDS_type ], TY_XDS_length - 1 ) ) | |
632 { | |
633 char *TY_XDS_ptr = TY_XDS[ TY_XDS_mode ][ TY_XDS_type ]; | |
634 | |
635 TY_XDS_ptr[ TY_XDS_length ] = 0; | |
636 memcpy( TY_XDS[ TY_XDS_mode ][ TY_XDS_type ], | |
637 TY_XDS_new[ TY_XDS_mode ][ TY_XDS_type ], TY_XDS_length ); | |
638 | |
639 // nasty hack: only print time codes if seconds are 0 | |
640 if ( TY_XDS_mode == 3 && TY_XDS_type == 1 && | |
641 !( TY_XDS_new[ 3 ][ 1 ][ 3 ] & 0x20 ) ) | |
642 { | |
643 return 0; | |
644 } | |
645 if ( TY_XDS_mode == 0 && TY_XDS_type == 2 && | |
646 ( TY_XDS_new[ 0 ][ 2 ][ 4 ] & 0x3f ) > 1 ) | |
647 { | |
648 return 0; | |
649 } | |
650 | |
651 mp_msg( MSGT_DEMUX, MSGL_DBG3, "%% %s ", TY_XDS_modes[ TY_XDS_mode ] ); | |
652 | |
653 line[ 0 ] = 0; | |
654 // printf( "XDS Code %x\n", | |
655 // ( TY_XDS_mode << 9 ) + TY_XDS_type + 0x100 ); | |
656 switch ( ( TY_XDS_mode << 9 ) + TY_XDS_type + 0x100 ) | |
657 { | |
658 // cases are specified in 2 bytes hex representing mode, type. | |
659 // TY_XDS_ptr will point to the current class buffer | |
660 case 0x0101: // current | |
661 case 0x0301: // future | |
662 { | |
663 char *mon[] = | |
664 { | |
665 "0", "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", | |
666 "Aug", "Sep", "Oct", "Nov", "Dec", "13", "14", "15" | |
667 }; | |
668 ty_AddXDSToDisplay( "AIR DATE: %s %2d %d:%02d:00", | |
669 mon[ TY_XDS_ptr[ 3 ] & 0x0f ], | |
670 TY_XDS_ptr[ 2 ] & 0x1f, | |
671 TY_XDS_ptr[ 1 ] & 0x1f, | |
672 TY_XDS_ptr[ 0 ] & 0x3f | |
673 ); | |
674 | |
675 // Program is tape delayed | |
676 if ( TY_XDS_ptr[ 3 ] & 0x10 ) ty_AddXDSToDisplay( " TAPE" ); | |
677 } | |
678 break; | |
679 | |
680 case 0x0102: // current program length | |
681 case 0x0302: // future | |
682 { | |
683 ty_AddXDSToDisplay( | |
684 "DURATION: %d:%02d:%02d of %d:%02d:%02d", | |
685 TY_XDS_ptr[ 3 ] & 0x3f, | |
686 TY_XDS_ptr[ 2 ] & 0x3f, | |
687 TY_XDS_ptr[ 4 ] & 0x3f, | |
688 TY_XDS_ptr[ 1 ] & 0x3f, | |
689 TY_XDS_ptr[ 0 ] & 0x3f, 0); | |
690 break; | |
691 } | |
692 | |
693 case 0x0103: // current program name | |
694 case 0x0303: // future | |
695 { | |
696 ty_AddXDSToDisplay( "TITLE: %s", TY_XDS_ptr ); | |
697 break; | |
698 } | |
699 | |
700 case 0x0104: // current program type | |
701 case 0x0304: // future | |
702 { | |
703 // for now just print out the raw data | |
704 // requires a 127 string array to parse | |
705 // properly and isn't worth it. | |
706 sprintf ( line, "%sGENRE:", line ); | |
707 { | |
708 int x; | |
709 for ( x = 0 ; x < TY_XDS_length ; x++ ) | |
710 sprintf( line, "%s %02x", line, TY_XDS_ptr[ x ] ); | |
711 } | |
712 ty_AddXDSToDisplay( line ); | |
713 break; | |
714 } | |
715 | |
716 case 0x0105: // current program rating | |
717 case 0x0305: // future | |
718 { | |
719 sprintf( line, "%sRATING: %s", line, | |
720 TY_XDS_CHIP[ ( TY_XDS_ptr[ 0 ] & 0x08 ) >> 3 ] | |
721 [ TY_XDS_ptr[ 1 ] & 0x07 ] ); | |
722 if ( TY_XDS_ptr[ 0 ] & 0x20 ) | |
723 sprintf( line, "%s DIALOGUE", line ); | |
724 if ( TY_XDS_ptr[ 1 ] & 0x08 ) | |
725 sprintf( line, "%s LANGUAGE", line ); | |
726 if ( TY_XDS_ptr[ 1 ] & 0x10 ) | |
727 sprintf( line, "%s SEXUAL", line ); | |
728 if ( TY_XDS_ptr[ 1 ] & 0x20 ) | |
729 sprintf( line, "%s VIOLENCE", line ); | |
730 ty_AddXDSToDisplay( line ); | |
731 | |
732 // raw output for verification. | |
733 if ( TY_OSD_debug > 1 ) | |
734 mp_msg( MSGT_DEMUX, MSGL_DBG3, " (%02x %02x)", | |
735 TY_XDS_ptr[ 0 ], TY_XDS_ptr[ 1 ] ); | |
736 break; | |
737 } | |
738 | |
739 case 0x0106: // current program audio services | |
740 case 0x0306: // future | |
741 { | |
742 // requires table, never actually seen it used either | |
743 ty_AddXDSToDisplay( "AUDIO: %02x %02x", TY_XDS_ptr[ 0 ], | |
744 TY_XDS_ptr[ 1 ] ); | |
745 break; | |
746 } | |
747 | |
748 case 0x0109: // current program aspect ratio | |
749 case 0x0309: // future | |
750 { | |
751 // requires table, rare | |
752 ty_AddXDSToDisplay( "ASPECT: %02x %02x", | |
753 TY_XDS_ptr[ 0 ], TY_XDS_ptr[ 1 ] ); | |
754 break; | |
755 } | |
756 | |
757 case 0x0110 ... 0x0117: // program description | |
758 { | |
759 ty_AddXDSToDisplay( "DESCRIP: %s", TY_XDS_ptr ); | |
760 break; | |
761 } | |
762 | |
763 case 0x0501: // channel network name | |
764 { | |
765 ty_AddXDSToDisplay( "NETWORK: %s", TY_XDS_ptr ); | |
766 break; | |
767 } | |
768 | |
769 case 0x0502: // channel network call letters | |
770 { | |
771 ty_AddXDSToDisplay( "CALLSIGN: %s", TY_XDS_ptr ); | |
772 break; | |
773 } | |
774 | |
775 case 0x0701: // misc. time of day | |
776 { | |
777 #define TIMEZONE ( TY_XDS[ 3 ][ 4 ][ 0 ] & 0x1f ) | |
778 #define DST ( ( TY_XDS[ 3 ][ 4 ][ 0 ] & 0x20 ) >> 5 ) | |
779 struct tm tm = | |
780 { | |
781 0, // sec | |
782 ( TY_XDS_ptr[ 0 ] & 0x3F ), // min | |
783 ( TY_XDS_ptr[ 1 ] & 0x1F ), // hour | |
784 ( TY_XDS_ptr[ 2 ] & 0x1F ), // day | |
785 ( TY_XDS_ptr[ 3 ] & 0x1f ) - 1, // month | |
786 ( TY_XDS_ptr[ 5 ] & 0x3f ) + 90, // year | |
787 0, // day of week | |
788 0, // day of year | |
789 0, // DST | |
790 }; | |
791 | |
792 time_t time_t = mktime( &tm ); | |
793 char *timestr; | |
794 | |
795 time_t -= ( ( TIMEZONE - DST ) * 60 * 60 ); | |
796 timestr = ctime( &time_t ); | |
797 timestr[ strlen( timestr ) - 1 ] = 0; | |
798 | |
799 sprintf( line, "%sCUR.TIME: %s ", line, timestr ); | |
800 if ( TY_XDS[ 3 ][ 4 ][ 0 ] ) | |
801 { | |
802 sprintf( line, "%sUTC-%d", line, TIMEZONE ); | |
803 if (DST) sprintf( line, "%s DST", line ); | |
804 } | |
805 else | |
806 sprintf( line, "%sUTC", line ); | |
807 | |
808 ty_AddXDSToDisplay( line ); | |
809 | |
810 break; | |
811 } | |
812 | |
813 case 0x0704: //misc. local time zone | |
814 { | |
815 sprintf( line, "%sTIMEZONE: UTC-%d", | |
816 line, TY_XDS_ptr[ 0 ] & 0x1f ); | |
817 if ( TY_XDS_ptr[ 0 ] & 0x20 ) sprintf( line, "%s DST", line ); | |
818 ty_AddXDSToDisplay( line ); | |
819 break; | |
820 } | |
821 | |
822 default: | |
823 { | |
824 mp_msg( MSGT_DEMUX, MSGL_DBG3, "UNKNOWN CLASS %d TYPE %d", | |
825 ( TY_XDS_mode << 1 ) + 1, TY_XDS_type ); | |
826 if ( TY_OSD_debug > 1 ) | |
827 { | |
828 int x; | |
829 mp_msg( MSGT_DEMUX, MSGL_DBG3, "\nDUMP:\n" ); | |
830 for ( x = 0 ; x < TY_XDS_length ; x++ ) | |
831 mp_msg( MSGT_DEMUX, MSGL_DBG3, " %02x %c", | |
832 TY_XDS_ptr[ x ], TY_XDS_ptr[ x ] ); | |
833 mp_msg( MSGT_DEMUX, MSGL_DBG3, "\n" ); | |
834 } | |
835 } | |
836 } | |
837 if ( TY_OSD_debug > 1 ) | |
838 mp_msg( MSGT_DEMUX, MSGL_DBG3, " (%d)", TY_XDS_length ); | |
839 } | |
840 TY_XDS_mode = 0; | |
841 TY_XDS_type = 0; | |
842 } | |
843 else if ( TY_XDS_length < 34 ) | |
844 { | |
845 TY_XDS_new[ TY_XDS_mode ][ TY_XDS_type ][ TY_XDS_length++ ] = b1; | |
846 TY_XDS_new[ TY_XDS_mode ][ TY_XDS_type ][ TY_XDS_length++ ] = b2; | |
847 } | |
848 return 0; | |
849 } | |
850 | |
851 | |
852 // 42 x 10 | |
853 static char *testline = "0123456789012345678901234567890123456789012"; | |
854 | |
855 // =========================================================================== | |
856 // Callback from Video Display Processing to put up the OSD | |
857 // =========================================================================== | |
858 void ty_processuserdata( unsigned char* buf, int len ) | |
859 { | |
860 int index; | |
861 | |
862 sub_justify = 1; | |
863 | |
864 if ( subcc_enabled ) | |
865 { | |
866 if ( tyOSDInited == 0 ) | |
867 { | |
868 for ( index = 0; index < SUB_MAX_TEXT ; index++ ) | |
869 { | |
870 ty_OSD1.text[ index ] = malloc( TY_CC_MAX_X ); | |
871 ty_OSD2.text[ index ] = malloc( TY_CC_MAX_X ); | |
872 } | |
873 ty_ClearOSD( 0 ); | |
874 ty_OSD1.lines = SUB_MAX_TEXT; | |
875 ty_OSD2.lines = SUB_MAX_TEXT; | |
876 ty_pOSD1 = &ty_OSD1; | |
877 ty_pOSD2 = &ty_OSD2; | |
878 tyOSDUpdate = 0; | |
879 tyOSDInited = 1; | |
880 } | |
881 | |
882 if ( buf[ 0 ] == 0x01 ) | |
883 { | |
884 ty_CCdecode( buf[ 1 ], buf[ 2 ] ); | |
885 } | |
886 if ( buf[ 0 ] == 0x02 ) | |
887 { | |
888 ty_XDSdecode( buf[ 1 ], buf[ 2 ] ); | |
889 } | |
890 | |
891 ty_DisplayXDSInfo(); | |
892 | |
893 if ( tyOSDUpdate ) | |
894 { | |
895 // for ( index = 0; index < SUB_MAX_TEXT ; index++ ) | |
896 // { | |
897 // printf( "OSD:%d:%s\n", index, ty_OSD1.text[ index ] ); | |
898 // } | |
899 vo_sub = &ty_OSD1; | |
900 vo_osd_changed( OSDTYPE_SUBTITLE ); | |
901 tyOSDUpdate = 0; | |
902 } | |
903 } | |
904 } | |
905 | |
906 | |
907 |