2 * Some code freely inspired from VobSub <URL:http://vobsub.edensrising.com>,
3 * with kind permission from Gabest <gabest@freemail.hu>
15 #include <sys/types.h>
21 #include "libvo/video_out.h"
28 #define MIN(a, b) ((a)<(b)?(a):(b))
29 #define MAX(a, b) ((a)>(b)?(a):(b))
33 /**********************************************************************
35 * The RAR file must have the same basename as the file to open
36 * See <URL:http://www.unrarlib.org/>
37 **********************************************************************/
46 rar_open(const char *const filename
, const char *const mode
)
49 /* unrarlib can only read */
50 if (strcmp("r", mode
) && strcmp("rb", mode
)) {
54 stream
= malloc(sizeof(rar_stream_t
));
57 /* first try normal access */
58 stream
->file
= fopen(filename
, mode
);
59 if (stream
->file
== NULL
) {
63 /* Guess the RAR archive filename */
65 p
= strrchr(filename
, '.');
67 ptrdiff_t l
= p
- filename
;
68 rar_filename
= malloc(l
+ 5);
69 if (rar_filename
== NULL
) {
73 strncpy(rar_filename
, filename
, l
);
74 strcpy(rar_filename
+ l
, ".rar");
77 rar_filename
= malloc(strlen(filename
) + 5);
78 if (rar_filename
== NULL
) {
82 strcpy(rar_filename
, filename
);
83 strcat(rar_filename
, ".rar");
85 rc
= urarlib_get(&stream
->data
, &stream
->size
, (char*) filename
, rar_filename
, "");
97 rar_close(rar_stream_t
*stream
)
100 return fclose(stream
->file
);
106 rar_eof(rar_stream_t
*stream
)
109 return feof(stream
->file
);
110 return stream
->pos
>= stream
->size
;
114 rar_tell(rar_stream_t
*stream
)
117 return ftell(stream
->file
);
122 rar_seek(rar_stream_t
*stream
, long offset
, int whence
)
125 return fseek(stream
->file
, offset
, whence
);
132 stream
->pos
= offset
;
135 if (offset
< 0 && stream
->pos
< (unsigned long) -offset
) {
139 stream
->pos
+= offset
;
142 if (offset
< 0 && stream
->size
< (unsigned long) -offset
) {
146 stream
->pos
= stream
->size
+ offset
;
156 rar_getc(rar_stream_t
*stream
)
159 return getc(stream
->file
);
162 return stream
->data
[stream
->pos
++];
166 rar_read(void *ptr
, size_t size
, size_t nmemb
, rar_stream_t
*stream
)
169 unsigned long remain
;
171 return fread(ptr
, size
, nmemb
, stream
->file
);
175 remain
= stream
->size
- stream
->pos
;
177 res
= remain
/ size
* size
;
178 memcpy(ptr
, stream
->data
+ stream
->pos
, res
);
185 typedef FILE rar_stream_t
;
186 #define rar_open fopen
187 #define rar_close fclose
189 #define rar_tell ftell
190 #define rar_seek fseek
191 #define rar_getc getc
192 #define rar_read fread
195 /**********************************************************************/
198 getline(char **lineptr
, size_t *n
, rar_stream_t
*stream
)
202 if (*lineptr
== NULL
) {
203 *lineptr
= malloc(4096);
208 char *tmp
= realloc(*lineptr
, 4096);
214 if (*lineptr
== NULL
|| *n
== 0)
217 for (c
= rar_getc(stream
); c
!= EOF
; c
= rar_getc(stream
)) {
219 char *tmp
= realloc(*lineptr
, *n
* 2);
225 (*lineptr
)[res
++] = c
;
237 /**********************************************************************
239 **********************************************************************/
242 rar_stream_t
*stream
;
245 unsigned char *packet
;
246 unsigned int packet_reserve
;
247 unsigned int packet_size
;
251 mpeg_open(const char *filename
)
253 mpeg_t
*res
= malloc(sizeof(mpeg_t
));
254 int err
= res
== NULL
;
259 res
->packet_size
= 0;
260 res
->packet_reserve
= 0;
261 res
->stream
= rar_open(filename
, "r");
262 err
= res
->stream
== NULL
;
264 perror("fopen Vobsub file failed");
268 return err
? NULL
: res
;
272 mpeg_free(mpeg_t
*mpeg
)
277 rar_close(mpeg
->stream
);
282 mpeg_eof(mpeg_t
*mpeg
)
284 return rar_eof(mpeg
->stream
);
288 mpeg_tell(mpeg_t
*mpeg
)
290 return rar_tell(mpeg
->stream
);
294 mpeg_run(mpeg_t
*mpeg
)
296 unsigned int len
, idx
, version
;
298 /* Goto start of a packet, it starts with 0x000001?? */
299 const unsigned char wanted
[] = { 0, 0, 1 };
300 unsigned char buf
[5];
303 mpeg
->packet_size
= 0;
304 if (rar_read(buf
, 4, 1, mpeg
->stream
) != 1)
306 while (memcmp(buf
, wanted
, sizeof(wanted
)) != 0) {
307 c
= rar_getc(mpeg
->stream
);
310 memmove(buf
, buf
+ 1, 3);
314 case 0xb9: /* System End Code */
316 case 0xba: /* Packet start code */
317 c
= rar_getc(mpeg
->stream
);
320 if ((c
& 0xc0) == 0x40)
322 else if ((c
& 0xf0) == 0x20)
325 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Unsupported MPEG version: 0x%02x", c
);
329 if (rar_seek(mpeg
->stream
, 9, SEEK_CUR
))
332 else if (version
== 2) {
333 if (rar_seek(mpeg
->stream
, 7, SEEK_CUR
))
339 case 0xbd: /* packet */
340 if (rar_read(buf
, 2, 1, mpeg
->stream
) != 1)
342 len
= buf
[0] << 8 | buf
[1];
343 idx
= mpeg_tell(mpeg
);
344 c
= rar_getc(mpeg
->stream
);
347 if ((c
& 0xC0) == 0x40) { /* skip STD scale & size */
348 if (rar_getc(mpeg
->stream
) < 0)
350 c
= rar_getc(mpeg
->stream
);
354 if ((c
& 0xf0) == 0x20) { /* System-1 stream timestamp */
355 /* Do we need this? */
358 else if ((c
& 0xf0) == 0x30) {
359 /* Do we need this? */
362 else if ((c
& 0xc0) == 0x80) { /* System-2 (.VOB) stream */
363 unsigned int pts_flags
, hdrlen
, dataidx
;
364 c
= rar_getc(mpeg
->stream
);
368 c
= rar_getc(mpeg
->stream
);
372 dataidx
= mpeg_tell(mpeg
) + hdrlen
;
373 if (dataidx
> idx
+ len
) {
374 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Invalid header length: %d (total length: %d, idx: %d, dataidx: %d)\n",
375 hdrlen
, len
, idx
, dataidx
);
378 if ((pts_flags
& 0xc0) == 0x80) {
379 if (rar_read(buf
, 5, 1, mpeg
->stream
) != 1)
381 if (!(((buf
[0] & 0xf0) == 0x20) && (buf
[0] & 1) && (buf
[2] & 1) && (buf
[4] & 1))) {
382 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "vobsub PTS error: 0x%02x %02x%02x %02x%02x \n",
383 buf
[0], buf
[1], buf
[2], buf
[3], buf
[4]);
387 mpeg
->pts
= ((buf
[0] & 0x0e) << 29 | buf
[1] << 22 | (buf
[2] & 0xfe) << 14
388 | buf
[3] << 7 | (buf
[4] >> 1));
390 else /* if ((pts_flags & 0xc0) == 0xc0) */ {
394 rar_seek(mpeg
->stream
, dataidx
, SEEK_SET
);
395 mpeg
->aid
= rar_getc(mpeg
->stream
);
397 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Bogus aid %d\n", mpeg
->aid
);
400 mpeg
->packet_size
= len
- ((unsigned int) mpeg_tell(mpeg
) - idx
);
401 if (mpeg
->packet_reserve
< mpeg
->packet_size
) {
404 mpeg
->packet
= malloc(mpeg
->packet_size
);
406 mpeg
->packet_reserve
= mpeg
->packet_size
;
408 if (mpeg
->packet
== NULL
) {
409 mp_msg(MSGT_VOBSUB
,MSGL_FATAL
,"malloc failure");
410 mpeg
->packet_reserve
= 0;
411 mpeg
->packet_size
= 0;
414 if (rar_read(mpeg
->packet
, mpeg
->packet_size
, 1, mpeg
->stream
) != 1) {
415 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"fread failure");
416 mpeg
->packet_size
= 0;
422 case 0xbe: /* Padding */
423 if (rar_read(buf
, 2, 1, mpeg
->stream
) != 1)
425 len
= buf
[0] << 8 | buf
[1];
426 if (len
> 0 && rar_seek(mpeg
->stream
, len
, SEEK_CUR
))
430 if (0xc0 <= buf
[3] && buf
[3] < 0xf0) {
431 /* MPEG audio or video */
432 if (rar_read(buf
, 2, 1, mpeg
->stream
) != 1)
434 len
= buf
[0] << 8 | buf
[1];
435 if (len
> 0 && rar_seek(mpeg
->stream
, len
, SEEK_CUR
))
440 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"unknown header 0x%02X%02X%02X%02X\n",
441 buf
[0], buf
[1], buf
[2], buf
[3]);
448 /**********************************************************************
450 **********************************************************************/
462 unsigned int packets_reserve
;
463 unsigned int packets_size
;
464 unsigned int current_index
;
468 packet_construct(packet_t
*pkt
)
477 packet_destroy(packet_t
*pkt
)
484 packet_queue_construct(packet_queue_t
*queue
)
487 queue
->packets
= NULL
;
488 queue
->packets_reserve
= 0;
489 queue
->packets_size
= 0;
490 queue
->current_index
= 0;
494 packet_queue_destroy(packet_queue_t
*queue
)
496 if (queue
->packets
) {
497 while (queue
->packets_size
--)
498 packet_destroy(queue
->packets
+ queue
->packets_size
);
499 free(queue
->packets
);
504 /* Make sure there is enough room for needed_size packets in the
507 packet_queue_ensure(packet_queue_t
*queue
, unsigned int needed_size
)
509 if (queue
->packets_reserve
< needed_size
) {
510 if (queue
->packets
) {
511 packet_t
*tmp
= realloc(queue
->packets
, 2 * queue
->packets_reserve
* sizeof(packet_t
));
513 mp_msg(MSGT_VOBSUB
,MSGL_FATAL
,"realloc failure");
516 queue
->packets
= tmp
;
517 queue
->packets_reserve
*= 2;
520 queue
->packets
= malloc(sizeof(packet_t
));
521 if (queue
->packets
== NULL
) {
522 mp_msg(MSGT_VOBSUB
,MSGL_FATAL
,"malloc failure");
525 queue
->packets_reserve
= 1;
531 /* add one more packet */
533 packet_queue_grow(packet_queue_t
*queue
)
535 if (packet_queue_ensure(queue
, queue
->packets_size
+ 1) < 0)
537 packet_construct(queue
->packets
+ queue
->packets_size
);
538 ++queue
->packets_size
;
542 /* insert a new packet, duplicating pts from the current one */
544 packet_queue_insert(packet_queue_t
*queue
)
547 if (packet_queue_ensure(queue
, queue
->packets_size
+ 1) < 0)
549 /* XXX packet_size does not reflect the real thing here, it will be updated a bit later */
550 memmove(queue
->packets
+ queue
->current_index
+ 2,
551 queue
->packets
+ queue
->current_index
+ 1,
552 sizeof(packet_t
) * (queue
->packets_size
- queue
->current_index
- 1));
553 pkts
= queue
->packets
+ queue
->current_index
;
554 ++queue
->packets_size
;
555 ++queue
->current_index
;
556 packet_construct(pkts
+ 1);
557 pkts
[1].pts100
= pkts
[0].pts100
;
558 pkts
[1].filepos
= pkts
[0].filepos
;
562 /**********************************************************************
564 **********************************************************************/
567 unsigned int palette
[16];
568 unsigned int cuspal
[4];
571 unsigned int have_palette
;
572 unsigned int orig_frame_width
, orig_frame_height
;
573 unsigned int origin_x
, origin_y
;
575 packet_queue_t
*spu_streams
;
576 unsigned int spu_streams_size
;
577 unsigned int spu_streams_current
;
580 /* Make sure that the spu stream idx exists. */
582 vobsub_ensure_spu_stream(vobsub_t
*vob
, unsigned int index
)
584 if (index
>= vob
->spu_streams_size
) {
585 /* This is a new stream */
586 if (vob
->spu_streams
) {
587 packet_queue_t
*tmp
= realloc(vob
->spu_streams
, (index
+ 1) * sizeof(packet_queue_t
));
589 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"vobsub_ensure_spu_stream: realloc failure");
592 vob
->spu_streams
= tmp
;
595 vob
->spu_streams
= malloc((index
+ 1) * sizeof(packet_queue_t
));
596 if (vob
->spu_streams
== NULL
) {
597 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"vobsub_ensure_spu_stream: malloc failure");
601 while (vob
->spu_streams_size
<= index
) {
602 packet_queue_construct(vob
->spu_streams
+ vob
->spu_streams_size
);
603 ++vob
->spu_streams_size
;
610 vobsub_add_id(vobsub_t
*vob
, const char *id
, size_t idlen
, const unsigned int index
)
612 if (vobsub_ensure_spu_stream(vob
, index
) < 0)
615 if (vob
->spu_streams
[index
].id
)
616 free(vob
->spu_streams
[index
].id
);
617 vob
->spu_streams
[index
].id
= malloc(idlen
+ 1);
618 if (vob
->spu_streams
[index
].id
== NULL
) {
619 mp_msg(MSGT_VOBSUB
,MSGL_FATAL
,"vobsub_add_id: malloc failure");
622 vob
->spu_streams
[index
].id
[idlen
] = 0;
623 memcpy(vob
->spu_streams
[index
].id
, id
, idlen
);
625 vob
->spu_streams_current
= index
;
626 mp_msg(MSGT_VOBSUB
,MSGL_V
,"[vobsub] subtitle (vobsubid): %d language %s\n",
627 index
, vob
->spu_streams
[index
].id
);
632 vobsub_add_timestamp(vobsub_t
*vob
, off_t filepos
, int ms
)
634 packet_queue_t
*queue
;
636 if (vob
->spu_streams
== 0) {
637 mp_msg(MSGT_VOBSUB
,MSGL_WARN
,"[vobsub] warning, binning some index entries. Check your index file\n");
640 queue
= vob
->spu_streams
+ vob
->spu_streams_current
;
641 if (packet_queue_grow(queue
) >= 0) {
642 pkt
= queue
->packets
+ (queue
->packets_size
- 1);
643 pkt
->filepos
= filepos
;
644 pkt
->pts100
= ms
< 0 ? UINT_MAX
: (unsigned int)ms
* 90;
651 vobsub_parse_id(vobsub_t
*vob
, const char *line
)
668 if (strncmp("index:", q
, 6))
675 return vobsub_add_id(vob
, p
, idlen
, atoi(q
));
679 vobsub_parse_timestamp(vobsub_t
*vob
, const char *line
)
681 // timestamp: HH:MM:SS.mmm, filepos: 0nnnnnnnnn
685 while (isspace(*line
))
720 while (isspace(*line
))
722 if (strncmp("filepos:", line
, 8))
725 while (isspace(*line
))
727 if (! isxdigit(*line
))
729 filepos
= strtol(line
, NULL
, 16);
730 return vobsub_add_timestamp(vob
, filepos
, vob
->delay
+ ms
+ 1000 * (s
+ 60 * (m
+ 60 * h
)));
734 vobsub_parse_size(vobsub_t
*vob
, const char *line
)
738 while (isspace(*line
))
742 vob
->orig_frame_width
= strtoul(line
, &p
, 10);
746 vob
->orig_frame_height
= strtoul(p
, NULL
, 10);
751 vobsub_parse_origin(vobsub_t
*vob
, const char *line
)
755 while (isspace(*line
))
759 vob
->origin_x
= strtoul(line
, &p
, 10);
763 vob
->origin_y
= strtoul(p
, NULL
, 10);
768 vobsub_parse_palette(vobsub_t
*vob
, const char *line
)
770 // palette: XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX, XXXXXX
775 int r
, g
, b
, y
, u
, v
, tmp
;
776 while (isspace(*line
))
783 tmp
= strtoul(line
, NULL
, 16);
784 r
= tmp
>> 16 & 0xff;
787 y
= MIN(MAX((int)(0.1494 * r
+ 0.6061 * g
+ 0.2445 * b
), 0), 0xff);
788 u
= MIN(MAX((int)(0.6066 * r
- 0.4322 * g
- 0.1744 * b
) + 128, 0), 0xff);
789 v
= MIN(MAX((int)(-0.08435 * r
- 0.3422 * g
+ 0.4266 * b
) + 128, 0), 0xff);
790 vob
->palette
[n
++] = y
<< 16 | u
<< 8 | v
;
797 vob
->have_palette
= 1;
802 vobsub_parse_custom(vobsub_t
*vob
, const char *line
)
804 //custom colors: OFF/ON(0/1)
805 if ((strncmp("ON", line
+ 15, 2) == 0)||strncmp("1", line
+ 15, 1) == 0)
807 else if ((strncmp("OFF", line
+ 15, 3) == 0)||strncmp("0", line
+ 15, 1) == 0)
815 vobsub_parse_cuspal(vobsub_t
*vob
, const char *line
)
817 //colors: XXXXXX, XXXXXX, XXXXXX, XXXXXX
823 while (isspace(*line
))
830 vob
->cuspal
[n
++] = strtoul(line
, NULL
,16);
840 /* don't know how to use tridx */
842 vobsub_parse_tridx(const char *line
)
846 tridx
= strtoul((line
+ 26), NULL
, 16);
847 tridx
= ((tridx
&0x1000)>>12) | ((tridx
&0x100)>>7) | ((tridx
&0x10)>>2) | ((tridx
&1)<<3);
852 vobsub_parse_delay(vobsub_t
*vob
, const char *line
)
856 if (*(line
+ 7) == '+'){
860 else if (*(line
+ 7) == '-'){
864 mp_msg(MSGT_SPUDEC
,MSGL_V
, "forward=%d", forward
);
866 mp_msg(MSGT_VOBSUB
,MSGL_V
, "h=%d," ,h
);
868 mp_msg(MSGT_VOBSUB
,MSGL_V
, "m=%d,", m
);
870 mp_msg(MSGT_VOBSUB
,MSGL_V
, "s=%d,", s
);
871 ms
= atoi(line
+ 16);
872 mp_msg(MSGT_VOBSUB
,MSGL_V
, "ms=%d", ms
);
873 vob
->delay
= (ms
+ 1000 * (s
+ 60 * (m
+ 60 * h
))) * forward
;
878 vobsub_set_lang(const char *line
)
881 vobsub_id
= atoi(line
+ 8);
886 vobsub_parse_one_line(vobsub_t
*vob
, rar_stream_t
*fd
)
891 size_t line_reserve
= 0;
893 line_size
= getline(&line
, &line_reserve
, fd
);
899 if (*line
== 0 || *line
== '\r' || *line
== '\n' || *line
== '#')
901 else if (strncmp("langidx:", line
, 8) == 0)
902 res
= vobsub_set_lang(line
);
903 else if (strncmp("delay:", line
, 6) == 0)
904 res
= vobsub_parse_delay(vob
, line
);
905 else if (strncmp("id:", line
, 3) == 0)
906 res
= vobsub_parse_id(vob
, line
+ 3);
907 else if (strncmp("palette:", line
, 8) == 0)
908 res
= vobsub_parse_palette(vob
, line
+ 8);
909 else if (strncmp("size:", line
, 5) == 0)
910 res
= vobsub_parse_size(vob
, line
+ 5);
911 else if (strncmp("org:", line
, 4) == 0)
912 res
= vobsub_parse_origin(vob
, line
+ 4);
913 else if (strncmp("timestamp:", line
, 10) == 0)
914 res
= vobsub_parse_timestamp(vob
, line
+ 10);
915 else if (strncmp("custom colors:", line
, 14) == 0)
916 //custom colors: ON/OFF, tridx: XXXX, colors: XXXXXX, XXXXXX, XXXXXX,XXXXXX
917 res
= vobsub_parse_cuspal(vob
, line
) + vobsub_parse_tridx(line
) + vobsub_parse_custom(vob
, line
);
919 mp_msg(MSGT_VOBSUB
,MSGL_V
, "vobsub: ignoring %s", line
);
923 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "ERROR in %s", line
);
930 vobsub_parse_ifo(void* this, const char *const name
, unsigned int *palette
, unsigned int *width
, unsigned int *height
, int force
,
931 int sid
, char *langid
)
933 vobsub_t
*vob
= (vobsub_t
*)this;
935 rar_stream_t
*fd
= rar_open(name
, "rb");
938 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Can't open IFO file");
941 unsigned char block
[0x800];
942 const char *const ifo_magic
= "DVDVIDEO-VTS";
943 if (rar_read(block
, sizeof(block
), 1, fd
) != 1) {
945 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Can't read IFO header");
946 } else if (memcmp(block
, ifo_magic
, strlen(ifo_magic
) + 1))
947 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Bad magic in IFO header\n");
949 unsigned long pgci_sector
= block
[0xcc] << 24 | block
[0xcd] << 16
950 | block
[0xce] << 8 | block
[0xcf];
951 int standard
= (block
[0x200] & 0x30) >> 4;
952 int resolution
= (block
[0x201] & 0x0c) >> 2;
953 *height
= standard
? 576 : 480;
955 switch (resolution
) {
970 mp_msg(MSGT_VOBSUB
,MSGL_WARN
,"Vobsub: Unknown resolution %d \n", resolution
);
972 if (langid
&& 0 <= sid
&& sid
< 32) {
973 unsigned char *tmp
= block
+ 0x256 + sid
* 6 + 2;
978 if (rar_seek(fd
, pgci_sector
* sizeof(block
), SEEK_SET
)
979 || rar_read(block
, sizeof(block
), 1, fd
) != 1)
980 mp_msg(MSGT_VOBSUB
,MSGL_ERR
, "Can't read IFO PGCI");
983 unsigned long pgc_offset
= block
[0xc] << 24 | block
[0xd] << 16
984 | block
[0xe] << 8 | block
[0xf];
985 for (idx
= 0; idx
< 16; ++idx
) {
986 unsigned char *p
= block
+ pgc_offset
+ 0xa4 + 4 * idx
;
987 palette
[idx
] = p
[0] << 24 | p
[1] << 16 | p
[2] << 8 | p
[3];
990 vob
->have_palette
= 1;
1000 vobsub_open(const char *const name
,const char *const ifo
,const int force
,void** spu
)
1002 vobsub_t
*vob
= malloc(sizeof(vobsub_t
));
1008 vob
->have_palette
= 0;
1009 vob
->orig_frame_width
= 0;
1010 vob
->orig_frame_height
= 0;
1011 vob
->spu_streams
= NULL
;
1012 vob
->spu_streams_size
= 0;
1013 vob
->spu_streams_current
= 0;
1015 buf
= malloc((strlen(name
) + 5) * sizeof(char));
1019 /* read in the info file */
1022 strcat(buf
, ".ifo");
1023 vobsub_parse_ifo(vob
,buf
, vob
->palette
, &vob
->orig_frame_width
, &vob
->orig_frame_height
, force
, -1, NULL
);
1025 vobsub_parse_ifo(vob
,ifo
, vob
->palette
, &vob
->orig_frame_width
, &vob
->orig_frame_height
, force
, -1, NULL
);
1026 /* read in the index */
1028 strcat(buf
, ".idx");
1029 fd
= rar_open(buf
, "rb");
1032 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"VobSub: Can't open IDX file");
1039 while (vobsub_parse_one_line(vob
, fd
) >= 0)
1043 /* if no palette in .idx then use custom colors */
1044 if ((vob
->custom
== 0)&&(vob
->have_palette
!=1))
1046 if (spu
&& vob
->orig_frame_width
&& vob
->orig_frame_height
)
1047 *spu
= spudec_new_scaled_vobsub(vob
->palette
, vob
->cuspal
, vob
->custom
, vob
->orig_frame_width
, vob
->orig_frame_height
);
1049 /* read the indexed mpeg_stream */
1051 strcat(buf
, ".sub");
1052 mpg
= mpeg_open(buf
);
1055 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"VobSub: Can't open SUB file");
1063 long last_pts_diff
= 0;
1064 while (!mpeg_eof(mpg
)) {
1065 off_t pos
= mpeg_tell(mpg
);
1066 if (mpeg_run(mpg
) < 0) {
1068 mp_msg(MSGT_VOBSUB
,MSGL_ERR
,"mpeg_run error");
1071 if (mpg
->packet_size
) {
1072 if ((mpg
->aid
& 0xe0) == 0x20) {
1073 unsigned int sid
= mpg
->aid
& 0x1f;
1074 if (vobsub_ensure_spu_stream(vob
, sid
) >= 0) {
1075 packet_queue_t
*queue
= vob
->spu_streams
+ sid
;
1076 /* get the packet to fill */
1077 if (queue
->packets_size
== 0 && packet_queue_grow(queue
) < 0)
1079 while (queue
->current_index
+ 1 < queue
->packets_size
1080 && queue
->packets
[queue
->current_index
+ 1].filepos
<= pos
)
1081 ++queue
->current_index
;
1082 if (queue
->current_index
< queue
->packets_size
) {
1084 if (queue
->packets
[queue
->current_index
].data
) {
1085 /* insert a new packet and fix the PTS ! */
1086 packet_queue_insert(queue
);
1087 queue
->packets
[queue
->current_index
].pts100
=
1088 mpg
->pts
+ last_pts_diff
;
1090 pkt
= queue
->packets
+ queue
->current_index
;
1091 if (pkt
->pts100
!= UINT_MAX
) {
1092 if (queue
->packets_size
> 1)
1093 last_pts_diff
= pkt
->pts100
- mpg
->pts
;
1095 pkt
->pts100
= mpg
->pts
;
1096 /* FIXME: should not use mpg_sub internal informations, make a copy */
1097 pkt
->data
= mpg
->packet
;
1098 pkt
->size
= mpg
->packet_size
;
1100 mpg
->packet_reserve
= 0;
1101 mpg
->packet_size
= 0;
1106 mp_msg(MSGT_VOBSUB
,MSGL_WARN
, "don't know what to do with subtitle #%u\n", sid
);
1110 vob
->spu_streams_current
= vob
->spu_streams_size
;
1111 while (vob
->spu_streams_current
-- > 0)
1112 vob
->spu_streams
[vob
->spu_streams_current
].current_index
= 0;
1122 vobsub_close(void *this)
1124 vobsub_t
*vob
= (vobsub_t
*)this;
1125 if (vob
->spu_streams
) {
1126 while (vob
->spu_streams_size
--)
1127 packet_queue_destroy(vob
->spu_streams
+ vob
->spu_streams_size
);
1128 free(vob
->spu_streams
);
1134 vobsub_get_indexes_count(void *vobhandle
)
1136 vobsub_t
*vob
= (vobsub_t
*) vobhandle
;
1137 return vob
->spu_streams_size
;
1141 vobsub_get_id(void *vobhandle
, unsigned int index
)
1143 vobsub_t
*vob
= (vobsub_t
*) vobhandle
;
1144 return (index
< vob
->spu_streams_size
) ? vob
->spu_streams
[index
].id
: NULL
;
1148 vobsub_get_packet(void *vobhandle
, float pts
,void** data
, int* timestamp
) {
1149 vobsub_t
*vob
= (vobsub_t
*)vobhandle
;
1150 unsigned int pts100
= 90000 * pts
;
1151 if (vob
->spu_streams
&& 0 <= vobsub_id
&& (unsigned) vobsub_id
< vob
->spu_streams_size
) {
1152 packet_queue_t
*queue
= vob
->spu_streams
+ vobsub_id
;
1153 while (queue
->current_index
< queue
->packets_size
) {
1154 packet_t
*pkt
= queue
->packets
+ queue
->current_index
;
1155 if (pkt
->pts100
!= UINT_MAX
)
1156 if (pkt
->pts100
<= pts100
) {
1157 ++queue
->current_index
;
1159 *timestamp
= pkt
->pts100
;
1163 ++queue
->current_index
;
1170 vobsub_get_next_packet(void *vobhandle
, void** data
, int* timestamp
)
1172 vobsub_t
*vob
= (vobsub_t
*)vobhandle
;
1173 if (vob
->spu_streams
&& 0 <= vobsub_id
&& (unsigned) vobsub_id
< vob
->spu_streams_size
) {
1174 packet_queue_t
*queue
= vob
->spu_streams
+ vobsub_id
;
1175 if (queue
->current_index
< queue
->packets_size
) {
1176 packet_t
*pkt
= queue
->packets
+ queue
->current_index
;
1177 ++queue
->current_index
;
1179 *timestamp
= pkt
->pts100
;
1187 vobsub_reset(void *vobhandle
)
1189 vobsub_t
*vob
= (vobsub_t
*)vobhandle
;
1190 if (vob
->spu_streams
) {
1191 unsigned int n
= vob
->spu_streams_size
;
1193 vob
->spu_streams
[n
].current_index
= 0;
1197 /**********************************************************************
1199 **********************************************************************/
1208 create_idx(vobsub_out_t
*me
, const unsigned int *palette
, unsigned int orig_width
, unsigned int orig_height
)
1212 "# VobSub index file, v7 (do not modify this line!)\n"
1214 "# Generated by MPlayer " VERSION
"\n"
1215 "# See <URL:http://www.mplayerhq.hu/> for more information about MPlayer\n"
1216 "# See <URL:http://vobsub.edensrising.com/> for more information about Vobsub\n"
1219 orig_width
, orig_height
);
1221 fputs("palette:", me
->fidx
);
1222 for (i
= 0; i
< 16; ++i
) {
1223 const double y
= palette
[i
] >> 16 & 0xff,
1224 u
= (palette
[i
] >> 8 & 0xff) - 128.0,
1225 v
= (palette
[i
] & 0xff) - 128.0;
1227 putc(',', me
->fidx
);
1228 fprintf(me
->fidx
, " %02x%02x%02x",
1229 MIN(MAX((int)(y
+ 1.4022 * u
), 0), 0xff),
1230 MIN(MAX((int)(y
- 0.3456 * u
- 0.7145 * v
), 0), 0xff),
1231 MIN(MAX((int)(y
+ 1.7710 * v
), 0), 0xff));
1233 putc('\n', me
->fidx
);
1238 vobsub_out_open(const char *basename
, const unsigned int *palette
,
1239 unsigned int orig_width
, unsigned int orig_height
,
1240 const char *id
, unsigned int index
)
1242 vobsub_out_t
*result
= NULL
;
1244 filename
= malloc(strlen(basename
) + 5);
1246 result
= malloc(sizeof(vobsub_out_t
));
1247 result
->fsub
= NULL
;
1248 result
->fidx
= NULL
;
1251 result
->aid
= index
;
1252 strcpy(filename
, basename
);
1253 strcat(filename
, ".sub");
1254 result
->fsub
= fopen(filename
, "a");
1255 if (result
->fsub
== NULL
)
1256 perror("Error: vobsub_out_open subtitle file open failed");
1257 strcpy(filename
, basename
);
1258 strcat(filename
, ".idx");
1259 result
->fidx
= fopen(filename
, "a");
1261 if (ftell(result
->fidx
) == 0){
1262 create_idx(result
, palette
, orig_width
, orig_height
);
1263 /* Make the selected language the default language */
1264 fprintf(result
->fidx
, "\n# Language index in use\nlangidx: %u\n", index
);
1266 fprintf(result
->fidx
, "\nid: %s, index: %u\n", id
? id
: "xx", index
);
1267 /* So that we can check the file now */
1268 fflush(result
->fidx
);
1271 perror("Error: vobsub_out_open index file open failed");
1279 vobsub_out_close(void *me
)
1281 vobsub_out_t
*vob
= (vobsub_out_t
*)me
;
1290 vobsub_out_output(void *me
, const unsigned char *packet
, int len
, double pts
)
1292 static double last_pts
;
1293 static int last_pts_set
= 0;
1294 vobsub_out_t
*vob
= (vobsub_out_t
*)me
;
1296 /* Windows' Vobsub require that every packet is exactly 2kB long */
1297 unsigned char buffer
[2048];
1300 /* Do not output twice a line with the same timestamp, this
1301 breaks Windows' Vobsub */
1302 if (vob
->fidx
&& (!last_pts_set
|| last_pts
!= pts
)) {
1303 static unsigned int last_h
= 9999, last_m
= 9999, last_s
= 9999, last_ms
= 9999;
1304 unsigned int h
, m
, ms
;
1311 ms
= (s
- (unsigned int) s
) * 1000;
1312 if (ms
>= 1000) /* prevent overfolws or bad float stuff */
1314 if (h
!= last_h
|| m
!= last_m
|| (unsigned int) s
!= last_s
|| ms
!= last_ms
) {
1315 fprintf(vob
->fidx
, "timestamp: %02u:%02u:%02u:%03u, filepos: %09lx\n",
1316 h
, m
, (unsigned int) s
, ms
, ftell(vob
->fsub
));
1319 last_s
= (unsigned int) s
;
1326 /* Packet start code: Windows' Vobsub needs this */
1328 *p
++ = 0; /* 0x00 */
1336 static unsigned char last_pts
[5] = { 0, 0, 0, 0, 0};
1337 unsigned char now_pts
[5];
1338 int pts_len
, pad_len
, datalen
= len
;
1340 now_pts
[0] = 0x21 | (((unsigned long)pts
>> 29) & 0x0e);
1341 now_pts
[1] = ((unsigned long)pts
>> 22) & 0xff;
1342 now_pts
[2] = 0x01 | (((unsigned long)pts
>> 14) & 0xfe);
1343 now_pts
[3] = ((unsigned long)pts
>> 7) & 0xff;
1344 now_pts
[4] = 0x01 | (((unsigned long)pts
<< 1) & 0xfe);
1345 pts_len
= memcmp(last_pts
, now_pts
, sizeof(now_pts
)) ? sizeof(now_pts
) : 0;
1346 memcpy(last_pts
, now_pts
, sizeof(now_pts
));
1348 datalen
+= 3; /* Version, PTS_flags, pts_len */
1350 datalen
+= 1; /* AID */
1351 pad_len
= 2048 - (p
- buffer
) - 4 /* MPEG ID */ - 2 /* payload len */ - datalen
;
1352 /* XXX - Go figure what should go here! In any case the
1353 packet has to be completly filled. If I can fill it
1354 with padding (0x000001be) latter I'll do that. But if
1355 there is only room for 6 bytes then I can not write a
1356 padding packet. So I add some padding in the PTS
1357 field. This looks like a dirty kludge. Oh well... */
1359 /* Packet is too big. Let's try ommiting the PTS field */
1364 else if (pad_len
> 6)
1368 *p
++ = 0; /* 0x0e */
1373 *p
++ = (datalen
>> 8) & 0xff; /* length of payload */
1374 *p
++ = datalen
& 0xff;
1375 *p
++ = 0x80; /* System-2 (.VOB) stream */
1376 *p
++ = pts_len
? 0x80 : 0x00; /* pts_flags */
1377 *p
++ = pts_len
+ pad_len
;
1378 memcpy(p
, now_pts
, pts_len
);
1380 memset(p
, 0, pad_len
);
1383 *p
++ = 0x20 | vob
->aid
; /* aid */
1384 if (fwrite(buffer
, p
- buffer
, 1, vob
->fsub
) != 1
1385 || fwrite(packet
, len
, 1, vob
->fsub
) != 1)
1386 perror("ERROR: vobsub write failed");
1388 remain
-= p
- buffer
+ len
;
1397 *p
++ = (remain
- 6) >> 8;
1398 *p
++ = (remain
- 6) & 0xff;
1399 /* for better compression, blank this */
1400 memset(buffer
+ 6, 0, remain
- (p
- buffer
));
1401 if (fwrite(buffer
, remain
, 1, vob
->fsub
) != 1)
1402 perror("ERROR: vobsub padding write failed");
1404 else if (remain
> 0) {
1405 /* I don't know what to output. But anyway the block
1406 needs to be 2KB big */
1407 memset(buffer
, 0, remain
);
1408 if (fwrite(buffer
, remain
, 1, vob
->fsub
) != 1)
1409 perror("ERROR: vobsub blank padding write failed");
1411 else if (remain
< 0)
1413 "\nERROR: wrong thing happenned...\n"
1414 " I wrote a %i data bytes spu packet and that's too long\n", len
);