1 // Most of this was written by Mike Baker <mbm@linux.com>
2 // and released under the GPL v2+ license.
4 // Modifications and SEVERE cleanup of the code was done by
8 // Released under GPL2 License.
20 //#include "stream/stream.h"
21 //#include "demuxer.h"
22 //#include "parse_es.h"
23 //#include "stheader.h"
24 //#include "mp3_hdr.h"
25 //#include "subreader.h"
26 #include "sub/sub_cc.h"
28 #include "demux_ty_osd.h"
30 //#include "dvdauth.h"
32 extern int sub_justify
;
34 #define TY_TEXT_MODE ( 1 << 0 )
35 #define TY_OSD_MODE ( 1 << 1 )
37 static int TY_OSD_flags
= TY_TEXT_MODE
| TY_OSD_MODE
;
38 static int TY_OSD_debug
= 0;
40 // ===========================================================================
41 // Closed Caption Decoding and OSD Presentation
42 // ===========================================================================
43 #define TY_CCNONE ( -3 )
44 #define TY_CCTEXTMODE ( -2 )
45 #define TY_CCPOPUPNB ( -1 )
46 #define TY_CCPOPUP ( 0 )
47 #define TY_CCPAINTON ( 1 )
49 #define TY_CC_MAX_X ( 45 )
51 static int TY_CC_CUR_X
;
52 static int TY_CC_CUR_Y
;
53 static int TY_CC_stat
= TY_CCNONE
;
54 static char TY_CC_buf
[ 255 ];
55 static char *TY_CC_ptr
= TY_CC_buf
;
56 static unsigned TY_CC_lastcap
= 0;
57 static int TY_CC_TextItalic
;
58 static int TY_CC_Y_Offset
;
60 static subtitle ty_OSD1
;
61 static subtitle ty_OSD2
;
62 static subtitle
*ty_pOSD1
;
63 static subtitle
*ty_pOSD2
;
64 static int tyOSDInitialized
= 0;
65 static int tyOSDUpdate
= 0;
67 static void ty_DrawOSD(void)
69 // printf( "Calling ty_DrawOSD()\n" );
73 void ty_ClearOSD( int start
)
76 // printf( "Calling ty_ClearOSD()\n" );
77 for ( index
= start
; index
< SUB_MAX_TEXT
; index
++ )
79 memset( ty_OSD1
.text
[ index
], ' ', TY_CC_MAX_X
- 1 );
80 ty_OSD1
.text
[ index
][ TY_CC_MAX_X
- 1 ] = 0;
81 memset( ty_OSD2
.text
[ index
], ' ', TY_CC_MAX_X
- 1 );
82 ty_OSD2
.text
[ index
][ TY_CC_MAX_X
- 1 ] = 0;
86 static void ty_DrawChar( int *x
, int *y
, char disChar
, int fgColor
, int bgColor
)
94 if ( *x
>= ( TY_CC_MAX_X
- 1 ) )
98 if ( ( *y
+ TY_CC_Y_Offset
) > SUB_MAX_TEXT
)
100 cy
= SUB_MAX_TEXT
- TY_CC_Y_Offset
- 1;
103 // printf( "Calling ty_DrawChar() x:%d y:%d %c fg:%d bg:%d\n",
104 // cx, cy, disChar, fgColor, bgColor );
106 ty_OSD1
.text
[ TY_CC_Y_Offset
+ cy
][ cx
] = disChar
;
107 memset( &( ty_OSD1
.text
[ TY_CC_Y_Offset
+ cy
][ cx
+ 1 ] ), ' ',
108 TY_CC_MAX_X
- cx
- 2 );
112 static void ty_RollupBuf( int dest
, int source
, int numLines
)
116 // printf( "Calling ty_RollupBuf() dest:%d source %d, numLines %d\n",
117 // dest, source, numLines );
119 if ( ( source
+ TY_CC_Y_Offset
+ numLines
) > SUB_MAX_TEXT
)
125 if ( ( source
+ TY_CC_Y_Offset
+ numLines
) < 0 )
131 if ( numLines
> SUB_MAX_TEXT
)
137 for ( index
= 0 ; index
< numLines
; index
++ )
139 strcpy( ty_OSD1
.text
[ TY_CC_Y_Offset
+ dest
],
140 ty_OSD1
.text
[ TY_CC_Y_Offset
+ source
] );
144 memset( ty_OSD1
.text
[ TY_CC_Y_Offset
+ source
- 1 ], ' ', TY_CC_MAX_X
- 1 );
145 ty_OSD1
.text
[ TY_CC_Y_Offset
+ source
- 1 ][ TY_CC_MAX_X
- 1 ] = 0;
148 static void ty_drawchar( char c
)
152 if ( TY_OSD_flags
& TY_OSD_MODE
&& TY_CC_stat
!= TY_CCNONE
&&
154 ty_DrawChar( &TY_CC_CUR_X
, &TY_CC_CUR_Y
, c
, 4, 13 );
156 if ( TY_CC_ptr
- TY_CC_buf
> sizeof( TY_CC_buf
) - 1 )
158 TY_CC_ptr
= TY_CC_buf
;
159 memset( TY_CC_buf
, 0, sizeof( TY_CC_buf
) );
161 *( TY_CC_ptr
++ ) = ( c
== 14 ) ? '/' : c
; // swap a '/' for musical note
164 static void ty_draw(void)
166 if ( TY_CC_ptr
!= TY_CC_buf
&& TY_OSD_flags
& TY_TEXT_MODE
)
168 if ( *( TY_CC_ptr
- 1 ) == '\n' ) *( TY_CC_ptr
- 1 ) = 0;
170 mp_msg( MSGT_DEMUX
, MSGL_V
, "CC: %s\n", TY_CC_buf
);
172 TY_CC_lastcap
= time( NULL
);
174 TY_CC_ptr
= TY_CC_buf
;
175 memset( TY_CC_buf
, 0, sizeof( TY_CC_buf
) );
177 if ( TY_OSD_flags
& TY_OSD_MODE
) ty_DrawOSD();
178 if ( TY_CC_TextItalic
) TY_CC_TextItalic
= 0;
182 static int CC_last
= 0;
183 static char CC_mode
= 0;
184 static int CC_row
[] =
186 11, -1, 1, 2, 3, 4, 12, 13, 14, 15, 5, 6, 7, 8, 9, 10
189 // char specialchar[] = { '®', '°', '½', '¿', '*', '¢', '£', 14, 'à', ' ', 'è', 'â', 'ê', 'î', 'ô', 'û' };
191 static int ty_CCdecode( char b1
, char b2
)
194 int data
= ( b2
<< 8 ) + b1
;
196 if ( b1
& 0x60 ) // text
198 if ( !TY_OSD_debug
&& TY_CC_stat
== TY_CCNONE
) return 0;
199 if ( TY_OSD_debug
> 3 )
201 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "%c %c", b1
, b2
);
206 if ( TY_CC_stat
> 0 && TY_OSD_flags
& TY_OSD_MODE
) ty_DrawOSD();
208 else if ( ( b1
& 0x10 ) && ( b2
> 0x1F ) && ( data
!= CC_last
) )
210 #define CURRENT ( ( b1 & 0x08 ) >> 3 )
212 if ( CC_mode
!= CURRENT
&& TY_CC_stat
!= TY_CCNONE
)
214 if ( TY_OSD_debug
&& TY_CC_ptr
!= TY_CC_buf
) ty_draw();
215 TY_CC_stat
= TY_CCNONE
;
219 if ( TY_CC_stat
== TY_CCNONE
|| TY_CC_CUR_Y
== -1 )
221 if ( TY_CC_ptr
!= TY_CC_buf
)
224 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "(TY_OSD_debug) %s\n",
226 TY_CC_ptr
= TY_CC_buf
;
227 memset(TY_CC_buf
, 0, sizeof(TY_CC_buf
));
230 if ( CC_mode
!= CURRENT
) return 0;
233 // preamble address code (row & indent)
236 TY_CC_CUR_Y
= CC_row
[ ( ( b1
<< 1 ) & 14 ) | ( ( b2
>> 5 ) & 1 ) ];
238 // Offset into MPlayer's Buffer
239 if ( ( TY_CC_CUR_Y
>= 1 ) && ( TY_CC_CUR_Y
<= 4 ) )
241 TY_CC_Y_Offset
= SUB_MAX_TEXT
- 5 - 1;
243 if ( ( TY_CC_CUR_Y
>= 5 ) && ( TY_CC_CUR_Y
<= 10 ) )
245 TY_CC_Y_Offset
= SUB_MAX_TEXT
- 5 - 5;
247 if ( ( TY_CC_CUR_Y
>= 12 ) && ( TY_CC_CUR_Y
<= 15 ) )
249 TY_CC_Y_Offset
= SUB_MAX_TEXT
- 5 - 12;
252 if ( TY_OSD_debug
> 3 )
253 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "<< preamble %d >>\n", TY_CC_CUR_Y
);
255 // we still have something in the text buffer
256 if (TY_CC_ptr
!= TY_CC_buf
)
258 *(TY_CC_ptr
++) = '\n';
259 if ( TY_CC_TextItalic
)
261 TY_CC_TextItalic
= 0;
266 // row contains indent flag
269 for ( x
= 0 ; x
< ( ( b2
& 0x0F ) << 1 ) ; x
++ )
272 *(TY_CC_ptr
++) = ' ';
279 if ( TY_OSD_debug
> 3 )
280 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "<< %02x >>\n", b1
& 0x7 );
283 case 0x00: // attribute
285 if ( TY_OSD_debug
> 1 )
286 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "<<A: %d>>\n", b2
);
289 case 0x01: // midrow or char
293 case 0x20: // midrow attribute change
297 case 0x00: // italics off
299 TY_CC_TextItalic
= 0;
300 *(TY_CC_ptr
++) = ' ';
303 case 0x0e: // italics on
306 TY_CC_TextItalic
= 1;
311 if ( TY_OSD_debug
> 1 )
312 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "<<D: %d>>\n",
318 // TextUnderline = 1;
322 // TextUnderline = 0;
326 case 0x30: // special character..
329 if ( ( b2
& 0x0f ) == 9 )
332 *(TY_CC_ptr
++) = ' ';
336 // ty_drawchar(specialchar[ b2 & 0x0f ] );
346 case 0x05: // misc + F
348 if ( TY_OSD_debug
> 3 )
349 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "<< misc %02x >>\n", b2
);
352 case 0x20: // resume caption (new caption)
354 if ( TY_OSD_flags
& TY_OSD_MODE
&&
355 TY_CC_stat
!= TY_CCPOPUP
)
357 TY_CC_stat
= TY_CCPOPUP
;
361 case 0x21: // backspace
367 case 0x25: // 2-4 row captions
371 if ( TY_CC_stat
== TY_CCPOPUP
) ty_ClearOSD( 1 );
372 TY_CC_stat
= b2
- 0x23;
373 if ( TY_CC_CUR_Y
< TY_CC_stat
) TY_CC_CUR_Y
= TY_CC_stat
;
377 case 0x29: // resume direct caption
379 TY_CC_stat
= TY_CCPAINTON
;
383 case 0x2A: // text restart
389 case 0x2B: // resume text display
391 TY_CC_stat
= TY_CCTEXTMODE
;
395 case 0x2C: // erase displayed memory
398 if ( TY_OSD_flags
& TY_OSD_MODE
)
400 if ( TY_CC_stat
> TY_CCPOPUP
|| TY_CC_ptr
== TY_CC_buf
)
411 // Used to be a buffer swap here, dunno why
417 case 0x2D: // carriage return
421 if ( TY_OSD_flags
& TY_OSD_MODE
)
423 if ( TY_CC_stat
> TY_CCPAINTON
)
426 TY_CC_CUR_Y
- TY_CC_stat
+ 1 ,
427 TY_CC_CUR_Y
- TY_CC_stat
+ 2,
436 case 0x2F: // end caption + swap memory
439 /* FALL THROUGH TO 0x2E */
442 case 0x2E: // erase non-displayed memory
444 if ( TY_OSD_debug
&& TY_CC_ptr
!= TY_CC_buf
)
445 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "(TY_OSD_debug) %s\n",
447 if ( TY_OSD_flags
& TY_OSD_MODE
) ty_ClearOSD( 1 );
452 TY_CC_ptr
= TY_CC_buf
;
453 memset( TY_CC_buf
, 0, sizeof( TY_CC_buf
) );
458 case 0x07: // misc (TAB)
460 for ( x
= 0 ; x
< ( b2
- 0x20 ) ; x
++ )
471 // ===========================================================================
472 // Extended Data Service Decoding and OSD Presentation
473 // ===========================================================================
474 #define XDS_BUFFER_LENGTH ( 16 )
475 #define XDS_DISPLAY_FRAMES ( 120 )
476 static char *ty_XDS_Display
[ XDS_BUFFER_LENGTH
];
477 static int ty_XDSAddLine
= -1;
478 static int ty_XDSDisplayCount
= -1;
481 static void ty_AddXDSToDisplay( const char *format
, ... )
487 if ( ty_XDSAddLine
== -1 )
489 for( index
= 0 ; index
< XDS_BUFFER_LENGTH
; index
++ )
491 ty_XDS_Display
[ index
] = 0;
496 va_start( ap
, format
);
497 vsnprintf( line
, 80, format
, ap
);
499 mp_msg( MSGT_DEMUX
, MSGL_V
, "XDS: %s\n", line
);
501 if ( ty_XDSAddLine
== XDS_BUFFER_LENGTH
)
503 mp_msg( MSGT_DEMUX
, MSGL_ERR
, "XDS Buffer would have been blown\n" );
506 if ( ty_XDS_Display
[ ty_XDSAddLine
] != 0 )
508 free( ty_XDS_Display
[ ty_XDSAddLine
] );
509 ty_XDS_Display
[ ty_XDSAddLine
] = 0;
512 ty_XDS_Display
[ ty_XDSAddLine
] = malloc( strlen( line
) + 1 );
513 strcpy( ty_XDS_Display
[ ty_XDSAddLine
], line
);
518 static void ty_DisplayXDSInfo(void)
523 if ( ty_XDSDisplayCount
== -1 )
525 for( index
= 0 ; index
< XDS_BUFFER_LENGTH
; index
++ )
527 if ( ty_XDS_Display
[ index
] != 0 )
532 if ( index
!= XDS_BUFFER_LENGTH
)
534 size
= strlen( ty_XDS_Display
[ index
] );
536 // Right Justify the XDS Stuff
537 memcpy( &( ty_OSD1
.text
[ 0 ][ TY_CC_MAX_X
- size
- 1 ] ),
538 ty_XDS_Display
[ index
], size
);
539 free( ty_XDS_Display
[ index
] );
540 ty_XDS_Display
[ index
] = 0;
541 ty_XDSDisplayCount
= 0;
547 // We cleaned out all the XDS stuff to be displayed
553 // We displayed that piece of XDS information long enough
555 ty_XDSDisplayCount
++;
556 if ( ty_XDSDisplayCount
>= XDS_DISPLAY_FRAMES
)
558 memset( ty_OSD1
.text
[ 0 ], ' ', TY_CC_MAX_X
- 1 );
559 ty_OSD1
.text
[ 0 ][ TY_CC_MAX_X
- 1 ] = 0;
560 ty_XDSDisplayCount
= -1;
567 static int TY_XDS_mode
= 0;
568 static int TY_XDS_type
= 0;
569 static int TY_XDS_length
= 0;
570 static char TY_XDS_checksum
= 0;
572 // Array of [ Mode ][ Type ][ Length ]
573 static char TY_XDS
[ 8 ][ 25 ][ 34 ];
574 static char TY_XDS_new
[ 8 ][ 25 ][ 34 ];
576 // Array of [ MPAARating|TVRating ][ NumberRatings ]
577 static const char * const TY_XDS_CHIP
[ 2 ][ 8 ] =
579 { "(NOT APPLICABLE)", "G", "PG", "PG-13", "R", "NC-17", "X", "(NOT RATED)" },
580 { "(NOT RATED)", "TV-Y", "TV-Y7", "TV-G", "TV-PG", "TV-14", "TV-MA",
584 static const char * const TY_XDS_modes
[] =
586 "CURRENT", // 01h-02h current program
587 "FUTURE ", // 03h-04h future program
588 "CHANNEL", // 05h-06h channel
589 "MISC. ", // 07h-08h miscellaneous
590 "PUBLIC ", // 09h-0Ah public service
591 "RESERV.", // 0Bh-0Ch reserved
598 static int ty_XDSdecode( char b1
, char b2
)
605 TY_XDS_mode
= b1
>> 1; // every other mode is a resume
607 TY_XDS_checksum
= b1
+ b2
;
611 TY_XDS_checksum
+= b1
+ b2
;
613 // eof (next byte is checksum)
617 if ( !TY_XDS_length
|| TY_XDS_checksum
& 0x7F )
619 if ( TY_OSD_debug
> 3 && !TY_XDS_length
)
621 mp_msg( MSGT_DEMUX
, MSGL_DBG3
,
622 "%% TY_XDS CHECKSUM ERROR (ignoring)\n" );
632 // check to see if the data has changed.
633 if ( strncmp( TY_XDS
[ TY_XDS_mode
][ TY_XDS_type
],
634 TY_XDS_new
[ TY_XDS_mode
][ TY_XDS_type
], TY_XDS_length
- 1 ) )
636 char *TY_XDS_ptr
= TY_XDS
[ TY_XDS_mode
][ TY_XDS_type
];
638 TY_XDS_ptr
[ TY_XDS_length
] = 0;
639 memcpy( TY_XDS
[ TY_XDS_mode
][ TY_XDS_type
],
640 TY_XDS_new
[ TY_XDS_mode
][ TY_XDS_type
], TY_XDS_length
);
642 // nasty hack: only print time codes if seconds are 0
643 if ( TY_XDS_mode
== 3 && TY_XDS_type
== 1 &&
644 !( TY_XDS_new
[ 3 ][ 1 ][ 3 ] & 0x20 ) )
648 if ( TY_XDS_mode
== 0 && TY_XDS_type
== 2 &&
649 ( TY_XDS_new
[ 0 ][ 2 ][ 4 ] & 0x3f ) > 1 )
654 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "%% %s ", TY_XDS_modes
[ TY_XDS_mode
] );
657 // printf( "XDS Code %x\n",
658 // ( TY_XDS_mode << 9 ) + TY_XDS_type + 0x100 );
659 switch ( ( TY_XDS_mode
<< 9 ) + TY_XDS_type
+ 0x100 )
661 // cases are specified in 2 bytes hex representing mode, type.
662 // TY_XDS_ptr will point to the current class buffer
663 case 0x0101: // current
664 case 0x0301: // future
668 "0", "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul",
669 "Aug", "Sep", "Oct", "Nov", "Dec", "13", "14", "15"
671 ty_AddXDSToDisplay( "AIR DATE: %s %2d %d:%02d:00",
672 mon
[ TY_XDS_ptr
[ 3 ] & 0x0f ],
673 TY_XDS_ptr
[ 2 ] & 0x1f,
674 TY_XDS_ptr
[ 1 ] & 0x1f,
675 TY_XDS_ptr
[ 0 ] & 0x3f
678 // Program is tape delayed
679 if ( TY_XDS_ptr
[ 3 ] & 0x10 ) ty_AddXDSToDisplay( " TAPE" );
683 case 0x0102: // current program length
684 case 0x0302: // future
687 "DURATION: %d:%02d:%02d of %d:%02d:%02d",
688 TY_XDS_ptr
[ 3 ] & 0x3f,
689 TY_XDS_ptr
[ 2 ] & 0x3f,
690 TY_XDS_ptr
[ 4 ] & 0x3f,
691 TY_XDS_ptr
[ 1 ] & 0x3f,
692 TY_XDS_ptr
[ 0 ] & 0x3f, 0);
696 case 0x0103: // current program name
697 case 0x0303: // future
699 ty_AddXDSToDisplay( "TITLE: %s", TY_XDS_ptr
);
703 case 0x0104: // current program type
704 case 0x0304: // future
706 // for now just print out the raw data
707 // requires a 127 string array to parse
708 // properly and isn't worth it.
709 sprintf ( line
, "%sGENRE:", line
);
712 for ( x
= 0 ; x
< TY_XDS_length
; x
++ )
713 sprintf( line
, "%s %02x", line
, TY_XDS_ptr
[ x
] );
715 ty_AddXDSToDisplay( line
);
719 case 0x0105: // current program rating
720 case 0x0305: // future
722 sprintf( line
, "%sRATING: %s", line
,
723 TY_XDS_CHIP
[ ( TY_XDS_ptr
[ 0 ] & 0x08 ) >> 3 ]
724 [ TY_XDS_ptr
[ 1 ] & 0x07 ] );
725 if ( TY_XDS_ptr
[ 0 ] & 0x20 )
726 sprintf( line
, "%s DIALOGUE", line
);
727 if ( TY_XDS_ptr
[ 1 ] & 0x08 )
728 sprintf( line
, "%s LANGUAGE", line
);
729 if ( TY_XDS_ptr
[ 1 ] & 0x10 )
730 sprintf( line
, "%s SEXUAL", line
);
731 if ( TY_XDS_ptr
[ 1 ] & 0x20 )
732 sprintf( line
, "%s VIOLENCE", line
);
733 ty_AddXDSToDisplay( line
);
735 // raw output for verification.
736 if ( TY_OSD_debug
> 1 )
737 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, " (%02x %02x)",
738 TY_XDS_ptr
[ 0 ], TY_XDS_ptr
[ 1 ] );
742 case 0x0106: // current program audio services
743 case 0x0306: // future
745 // requires table, never actually seen it used either
746 ty_AddXDSToDisplay( "AUDIO: %02x %02x", TY_XDS_ptr
[ 0 ],
751 case 0x0109: // current program aspect ratio
752 case 0x0309: // future
754 // requires table, rare
755 ty_AddXDSToDisplay( "ASPECT: %02x %02x",
756 TY_XDS_ptr
[ 0 ], TY_XDS_ptr
[ 1 ] );
760 case 0x0110: // program description
769 ty_AddXDSToDisplay( "DESCRIP: %s", TY_XDS_ptr
);
773 case 0x0501: // channel network name
775 ty_AddXDSToDisplay( "NETWORK: %s", TY_XDS_ptr
);
779 case 0x0502: // channel network call letters
781 ty_AddXDSToDisplay( "CALLSIGN: %s", TY_XDS_ptr
);
785 case 0x0701: // misc. time of day
787 #define TIMEZONE ( TY_XDS[ 3 ][ 4 ][ 0 ] & 0x1f )
788 #define DST ( ( TY_XDS[ 3 ][ 4 ][ 0 ] & 0x20 ) >> 5 )
792 .tm_min
= ( TY_XDS_ptr
[ 0 ] & 0x3F ), // min
793 .tm_hour
= ( TY_XDS_ptr
[ 1 ] & 0x1F ), // hour
794 .tm_mday
= ( TY_XDS_ptr
[ 2 ] & 0x1F ), // day
795 .tm_mon
= ( TY_XDS_ptr
[ 3 ] & 0x1f ) - 1, // month
796 .tm_year
= ( TY_XDS_ptr
[ 5 ] & 0x3f ) + 90, // year
797 .tm_wday
= 0, // day of week
798 .tm_yday
= 0, // day of year
799 .tm_isdst
= 0, // DST
802 time_t time_t = mktime( &tm
);
805 time_t -= ( ( TIMEZONE
- DST
) * 60 * 60 );
806 timestr
= ctime( &time_t );
807 timestr
[ strlen( timestr
) - 1 ] = 0;
809 sprintf( line
, "%sCUR.TIME: %s ", line
, timestr
);
810 if ( TY_XDS
[ 3 ][ 4 ][ 0 ] )
812 sprintf( line
, "%sUTC-%d", line
, TIMEZONE
);
813 if (DST
) sprintf( line
, "%s DST", line
);
816 sprintf( line
, "%sUTC", line
);
818 ty_AddXDSToDisplay( line
);
823 case 0x0704: //misc. local time zone
825 sprintf( line
, "%sTIMEZONE: UTC-%d",
826 line
, TY_XDS_ptr
[ 0 ] & 0x1f );
827 if ( TY_XDS_ptr
[ 0 ] & 0x20 ) sprintf( line
, "%s DST", line
);
828 ty_AddXDSToDisplay( line
);
834 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "UNKNOWN CLASS %d TYPE %d",
835 ( TY_XDS_mode
<< 1 ) + 1, TY_XDS_type
);
836 if ( TY_OSD_debug
> 1 )
839 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "\nDUMP:\n" );
840 for ( x
= 0 ; x
< TY_XDS_length
; x
++ )
841 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, " %02x %c",
842 TY_XDS_ptr
[ x
], TY_XDS_ptr
[ x
] );
843 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, "\n" );
847 if ( TY_OSD_debug
> 1 )
848 mp_msg( MSGT_DEMUX
, MSGL_DBG3
, " (%d)", TY_XDS_length
);
853 else if ( TY_XDS_length
< 34 )
855 TY_XDS_new
[ TY_XDS_mode
][ TY_XDS_type
][ TY_XDS_length
++ ] = b1
;
856 TY_XDS_new
[ TY_XDS_mode
][ TY_XDS_type
][ TY_XDS_length
++ ] = b2
;
862 // ===========================================================================
863 // Callback from Video Display Processing to put up the OSD
864 // ===========================================================================
865 void ty_processuserdata( const unsigned char* buf
, int len
)
873 if ( tyOSDInitialized
== 0 )
875 for ( index
= 0; index
< SUB_MAX_TEXT
; index
++ )
877 ty_OSD1
.text
[ index
] = malloc( TY_CC_MAX_X
);
878 ty_OSD2
.text
[ index
] = malloc( TY_CC_MAX_X
);
881 ty_OSD1
.lines
= SUB_MAX_TEXT
;
882 ty_OSD2
.lines
= SUB_MAX_TEXT
;
886 tyOSDInitialized
= 1;
889 if ( buf
[ 0 ] == 0x01 )
891 ty_CCdecode( buf
[ 1 ], buf
[ 2 ] );
893 if ( buf
[ 0 ] == 0x02 )
895 ty_XDSdecode( buf
[ 1 ], buf
[ 2 ] );
902 // for ( index = 0; index < SUB_MAX_TEXT ; index++ )
904 // printf( "OSD:%d:%s\n", index, ty_OSD1.text[ index ] );
907 vo_osd_changed( OSDTYPE_SUBTITLE
);