summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ChangeLog6
-rw-r--r--configure.in1
-rw-r--r--include/xine.h.tmpl.in5
-rw-r--r--src/Makefile.am2
-rw-r--r--src/demuxers/demux_avi.c28
-rw-r--r--src/demuxers/demux_mpeg_block.c4
-rw-r--r--src/input/input_file.c53
-rw-r--r--src/input/input_plugin.h7
-rwxr-xr-xsrc/libmad/layer3.c4
-rw-r--r--src/libspudec/xine_decoder.c4
-rw-r--r--src/libsputext/Makefile.am47
-rw-r--r--src/libsputext/xine_decoder.c914
-rw-r--r--src/xine-engine/buffer.h3
-rw-r--r--src/xine-engine/configfile.c27
-rw-r--r--src/xine-engine/configfile.h8
-rw-r--r--src/xine-engine/video_decoder.c3
16 files changed, 1063 insertions, 53 deletions
diff --git a/ChangeLog b/ChangeLog
index 96f471ba8..016822980 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,10 +1,14 @@
+xine (0.9.7) unstable; urgency=low
* fix some win32 dll segfaults
* seamless branching on input_dvd
* fix no audio deadlock
* OSD (On Screen Display) for rendering text and graphics into overlays
* reworked spu and overlay manager (multiple overlays supported)
-
+ * support for avi text subtitles (use something like xine stream.avi:foo.sub)
+
+ -- Guenter Bartsch <guenter@users.sourceforge.net> Tue Nov 27 01:20:06 CET 2001
+
xine (0.9.6) unstable; urgency=low
* demux_asf big fragments handling
diff --git a/configure.in b/configure.in
index 734801867..49dfac1cb 100644
--- a/configure.in
+++ b/configure.in
@@ -757,6 +757,7 @@ src/libw32dll/Makefile
src/libw32dll/wine/Makefile
src/libw32dll/DirectShow/Makefile
src/libspudec/Makefile
+src/libsputext/Makefile
src/libvfill/Makefile
src/libvorbis/Makefile
src/libdivx4/Makefile
diff --git a/include/xine.h.tmpl.in b/include/xine.h.tmpl.in
index 22c8600c8..d558cb562 100644
--- a/include/xine.h.tmpl.in
+++ b/include/xine.h.tmpl.in
@@ -28,7 +28,7 @@
\endverbatim
*/
/*
- * $Id: xine.h.tmpl.in,v 1.58 2001/11/30 21:55:05 f1rmb Exp $
+ * $Id: xine.h.tmpl.in,v 1.59 2001/12/01 22:38:31 guenter Exp $
*
*/
@@ -546,9 +546,6 @@ struct config_values_s {
* from the config file otherwise
*/
- void (*register_empty) (config_values_t *self,
- char *key);
-
char* (*register_string) (config_values_t *self,
char *key,
char *def_value,
diff --git a/src/Makefile.am b/src/Makefile.am
index c1860a489..b8bec5984 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -1,7 +1,7 @@
SUBDIRS = xine-utils xine-engine audio_out video_out dxr3 input libmpeg2 libspudec demuxers \
liba52 libffmpeg liblpcm libw32dll libmad libdts libvfill \
- libvorbis libdivx4
+ libvorbis libdivx4 libsputext
debug:
@list='$(SUBDIRS)'; for subdir in $$list; do \
diff --git a/src/demuxers/demux_avi.c b/src/demuxers/demux_avi.c
index 0013fef61..781a45df7 100644
--- a/src/demuxers/demux_avi.c
+++ b/src/demuxers/demux_avi.c
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: demux_avi.c,v 1.54 2001/11/30 00:53:51 f1rmb Exp $
+ * $Id: demux_avi.c,v 1.55 2001/12/01 22:38:31 guenter Exp $
*
* demultiplexer for avi streams
*
@@ -866,10 +866,11 @@ static void demux_avi_start (demux_plugin_t *this_gen,
fifo_buffer_t *video_fifo,
fifo_buffer_t *audio_fifo,
off_t start_pos, int start_time) {
- buf_element_t *buf;
- demux_avi_t *this = (demux_avi_t *) this_gen;
- uint32_t video_pts = 0;
- int err;
+ buf_element_t *buf;
+ demux_avi_t *this = (demux_avi_t *) this_gen;
+ uint32_t video_pts = 0;
+ int err;
+ unsigned char *sub;
this->audio_fifo = audio_fifo;
this->video_fifo = video_fifo;
@@ -993,6 +994,23 @@ static void demux_avi_start (demux_plugin_t *this_gen,
this->audio_fifo->put (this->audio_fifo, buf);
}
+ /*
+ * send external spu file pointer, if present
+ */
+
+ if (this->input->get_optional_data (this->input, &sub, INPUT_OPTIONAL_DATA_TEXTSPU0)) {
+
+ buf = this->video_fifo->buffer_pool_alloc (this->video_fifo);
+ buf->content = sub;
+
+ buf->type = BUF_SPU_TEXT;
+
+ this->video_fifo->put (this->video_fifo, buf);
+
+ printf ("demux_avi: text subtitle file available\n");
+
+ }
+
if ((err = pthread_create (&this->thread, NULL, demux_avi_loop, this)) != 0) {
fprintf (stderr, "demux_avi: can't create new thread (%s)\n",
strerror(err));
diff --git a/src/demuxers/demux_mpeg_block.c b/src/demuxers/demux_mpeg_block.c
index 154c33464..81b17d775 100644
--- a/src/demuxers/demux_mpeg_block.c
+++ b/src/demuxers/demux_mpeg_block.c
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: demux_mpeg_block.c,v 1.66 2001/11/30 21:55:05 f1rmb Exp $
+ * $Id: demux_mpeg_block.c,v 1.67 2001/12/01 22:38:31 guenter Exp $
*
* demultiplexer for mpeg 1/2 program streams
*
@@ -104,7 +104,7 @@ static void demux_mpeg_block_parse_pack (demux_mpeg_block_t *this, int preview_m
printf ("demux_mpeg_block: checking if we can branch to %s\n", next_mrl);
- if (this->input->is_branch_possible
+ if (next_mrl && this->input->is_branch_possible
&& this->input->is_branch_possible (this->input, next_mrl)) {
printf ("demux_mpeg_block: branching\n");
diff --git a/src/input/input_file.c b/src/input/input_file.c
index 771108769..a05412ea1 100644
--- a/src/input/input_file.c
+++ b/src/input/input_file.c
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: input_file.c,v 1.30 2001/11/30 00:53:51 f1rmb Exp $
+ * $Id: input_file.c,v 1.31 2001/12/01 22:38:31 guenter Exp $
*/
#ifdef HAVE_CONFIG_H
@@ -77,6 +77,7 @@ typedef struct {
input_plugin_t input_plugin;
int fh;
+ FILE *sub;
char *mrl;
config_values_t *config;
@@ -234,6 +235,7 @@ static off_t get_file_size(char *filepathname, char *origin) {
*
*/
static uint32_t file_plugin_get_capabilities (input_plugin_t *this_gen) {
+
return INPUT_CAP_SEEKABLE | INPUT_CAP_GET_DIR;
}
@@ -242,15 +244,29 @@ static uint32_t file_plugin_get_capabilities (input_plugin_t *this_gen) {
*/
static int file_plugin_open (input_plugin_t *this_gen, char *mrl) {
- char *filename;
+ char *filename, *subtitle;
file_input_plugin_t *this = (file_input_plugin_t *) this_gen;
- this->mrl = mrl;
+ this->mrl = strdup(mrl); /* FIXME: small memory leak */
- if (!strncasecmp (mrl, "file:",5))
- filename = &mrl[5];
+ if (!strncasecmp (this->mrl, "file:",5))
+ filename = &this->mrl[5];
else
- filename = mrl;
+ filename = this->mrl;
+
+ subtitle = strrchr (filename, ':');
+ if (subtitle) {
+ *subtitle = 0;
+ subtitle++;
+
+ printf ("input_file: trying to open subtitle file '%s'\n",
+ subtitle);
+
+ this->sub = fopen (subtitle, "r");
+
+ } else
+ this->sub = NULL;
+
this->fh = open (filename, O_RDONLY);
@@ -689,6 +705,11 @@ static void file_plugin_close (input_plugin_t *this_gen) {
close(this->fh);
this->fh = -1;
+
+ if (this->sub) {
+ fclose (this->sub);
+ this->sub = NULL;
+ }
}
/*
@@ -719,6 +740,25 @@ static char *file_plugin_get_identifier (input_plugin_t *this_gen) {
static int file_plugin_get_optional_data (input_plugin_t *this_gen,
void *data, int data_type) {
+ file_input_plugin_t *this = (file_input_plugin_t *) this_gen;
+
+ printf ("input_file: get optional data, type %08x, sub %d\n",
+ data_type, this->sub);
+
+
+ if ( (data_type == INPUT_OPTIONAL_DATA_TEXTSPU0)
+ && this->sub ) {
+
+ FILE **tmp;
+
+ /* dirty hacks... */
+
+ tmp = data;
+ *tmp = this->sub;
+
+ return INPUT_OPTIONAL_SUCCESS;
+ }
+
return INPUT_OPTIONAL_UNSUPPORTED;
}
@@ -763,6 +803,7 @@ input_plugin_t *init_input_plugin (int iface, xine_t *xine) {
this->input_plugin.is_branch_possible = NULL;
this->fh = -1;
+ this->sub = NULL;
this->mrl = NULL;
this->config = config;
diff --git a/src/input/input_plugin.h b/src/input/input_plugin.h
index f14edf5ab..a820a6186 100644
--- a/src/input/input_plugin.h
+++ b/src/input/input_plugin.h
@@ -1,5 +1,5 @@
/*
- * Copyright (C) 2000 the xine project
+ * Copyright (C) 2000, 2001 the xine project
*
* This file is part of xine, a unix video player.
*
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: input_plugin.h,v 1.17 2001/11/30 21:55:05 f1rmb Exp $
+ * $Id: input_plugin.h,v 1.18 2001/12/01 22:38:31 guenter Exp $
*/
#ifndef HAVE_INPUT_PLUGIN_H
@@ -334,6 +334,9 @@ struct input_plugin_s
#define INPUT_OPTIONAL_DATA_CLUT 1
#define INPUT_OPTIONAL_DATA_AUDIOLANG 2
#define INPUT_OPTIONAL_DATA_SPULANG 3
+#define INPUT_OPTIONAL_DATA_TEXTSPU0 4
+#define INPUT_OPTIONAL_DATA_TEXTSPU1 5
+#define INPUT_OPTIONAL_DATA_TEXTSPU2 6
/*
* each input plugin _must_ implement this function:
diff --git a/src/libmad/layer3.c b/src/libmad/layer3.c
index d19a6f8d3..386fd1c68 100755
--- a/src/libmad/layer3.c
+++ b/src/libmad/layer3.c
@@ -16,7 +16,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
- * $Id: layer3.c,v 1.1 2001/08/12 02:57:55 guenter Exp $
+ * $Id: layer3.c,v 1.2 2001/12/01 22:38:31 guenter Exp $
*/
# ifdef HAVE_CONFIG_H
@@ -891,7 +891,7 @@ mad_fixed_t III_requantize(unsigned int value, signed int exp)
if (exp >= 5) {
/* overflow */
# if defined(DEBUG)
- fprintf(stderr, "requantize overflow (%f * 2^%d)\n",
+ printf ("mad: requantize overflow (%f * 2^%d)\n",
mad_f_todouble(requantized), exp);
# endif
requantized = MAD_F_MAX;
diff --git a/src/libspudec/xine_decoder.c b/src/libspudec/xine_decoder.c
index 25d80f7b3..3f7e24f16 100644
--- a/src/libspudec/xine_decoder.c
+++ b/src/libspudec/xine_decoder.c
@@ -19,7 +19,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: xine_decoder.c,v 1.42 2001/11/30 20:35:48 jcdutton Exp $
+ * $Id: xine_decoder.c,v 1.43 2001/12/01 22:38:31 guenter Exp $
*
* stuff needed to turn libspu into a xine decoder plugin
*/
@@ -39,7 +39,9 @@
#include "xineutils.h"
#include "spu.h"
+/*
#define LOG_DEBUG 1
+*/
static clut_t __default_clut[] = {
CLUT_Y_CR_CB_INIT(0x00, 0x80, 0x80),
diff --git a/src/libsputext/Makefile.am b/src/libsputext/Makefile.am
new file mode 100644
index 000000000..ed8c29e9b
--- /dev/null
+++ b/src/libsputext/Makefile.am
@@ -0,0 +1,47 @@
+CFLAGS = @GLOBAL_CFLAGS@
+
+LIBTOOL = $(SHELL) $(top_builddir)/libtool-nofpic
+
+libdir = $(XINE_PLUGINDIR)
+
+lib_LTLIBRARIES = xineplug_decode_sputext.la
+
+xineplug_decode_sputext_la_SOURCES = xine_decoder.c
+xineplug_decode_sputext_la_LDFLAGS = -avoid-version -module
+
+##
+## Install header files (default=$includedir/xine)
+##
+install-includeHEADERS: $(include_HEADERS)
+ @$(NORMAL_INSTALL)
+ $(mkinstalldirs) $(DESTDIR)$(includedir)/xine
+ @list='$(include_HEADERS)'; for p in $$list; do \
+ if test -f "$$p"; then d= ; else d="$(srcdir)/"; fi; \
+ echo " $(INSTALL_DATA) $$d$$p $(DESTDIR)$(includedir)/xine/$$p"; \
+ $(INSTALL_DATA) $$d$$p $(DESTDIR)$(includedir)/xine/$$p; \
+ done
+
+
+##
+## Remove them
+##
+uninstall-includeHEADERS:
+ @$(NORMAL_UNINSTALL)
+ list='$(include_HEADERS)'; for p in $$list; do \
+ rm -f $(DESTDIR)$(includedir)/xine/$$p; \
+ done
+
+
+debug:
+ @$(MAKE) CFLAGS="$(DEBUG_CFLAGS)"
+
+install-debug: debug
+ @$(MAKE) $(AM_MAKEFLAGS) install-exec-am install-data-am
+
+mostlyclean-generic:
+ -rm -f *~ \#* .*~ .\#*
+
+maintainer-clean-generic:
+ -@echo "This command is intended for maintainers to use;"
+ -@echo "it deletes files that may require special tools to rebuild."
+ -rm -f Makefile.in
diff --git a/src/libsputext/xine_decoder.c b/src/libsputext/xine_decoder.c
new file mode 100644
index 000000000..73e8b7770
--- /dev/null
+++ b/src/libsputext/xine_decoder.c
@@ -0,0 +1,914 @@
+/*
+ * Copyright (C) 2000-2001 the xine project
+ *
+ * This file is part of xine, a unix video player.
+ *
+ * xine is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * xine is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
+ *
+ * $Id: xine_decoder.c,v 1.1 2001/12/01 22:38:32 guenter Exp $
+ *
+ * code based on mplayer module:
+ *
+ * Subtitle reader with format autodetection
+ *
+ * Written by laaz
+ * Some code cleanup & realloc() by A'rpi/ESP-team
+ * dunnowhat sub format by szabi
+ */
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <unistd.h>
+#include <string.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <ctype.h>
+
+#include "buffer.h"
+#include "events.h"
+#include "xine_internal.h"
+#include "xineutils.h"
+#include "osd.h"
+
+/*
+#define LOG_DEBUG 1
+*/
+
+#define ERR (void *)-1
+
+#define SUB_MAX_TEXT 5
+
+typedef struct {
+
+ int lines;
+
+ unsigned long start;
+ unsigned long end;
+
+ char *text[SUB_MAX_TEXT];
+
+ osd_object_t *osd;
+
+} subtitle_t;
+
+
+typedef struct sputext_decoder_s {
+ spu_decoder_t spu_decoder;
+
+ xine_t *xine;
+
+ vo_instance_t *vo_out;
+ int output_open;
+
+ FILE *fd;
+
+ float mpsub_position;
+
+ int uses_time;
+ int errs;
+ subtitle_t *subtitles;
+ int num; /* number of subtitle structs */
+ int format; /* constants see below */
+ subtitle_t *previous_aqt_sub ;
+
+ osd_object_t *osd;
+
+ /* thread */
+ int running;
+ pthread_t spu_thread;
+
+} sputext_decoder_t;
+
+#define FORMAT_MICRODVD 0
+#define FORMAT_SUBRIP 1
+#define FORMAT_SUBVIEWER 2
+#define FORMAT_SAMI 3
+#define FORMAT_VPLAYER 4
+#define FORMAT_RT 5
+#define FORMAT_SSA 6 /* Sub Station Alpha */
+#define FORMAT_DUNNO 7 /*... erm ... dunnowhat. tell me if you know */
+#define FORMAT_MPSUB 8
+#define FORMAT_AQTITLE 9
+
+static int eol(char p) {
+ return (p=='\r' || p=='\n' || p=='\0');
+}
+
+static inline void trail_space(char *s) {
+ int i;
+ while (isspace(*s))
+ strcpy(s, s + 1);
+ i = strlen(s) - 1;
+ while (i > 0 && isspace(s[i]))
+ s[i--] = '\0';
+}
+
+
+static subtitle_t *sub_read_line_sami(sputext_decoder_t *this, subtitle_t *current) {
+
+ static char line[1001];
+ static char *s = NULL;
+ char text[1000], *p, *q;
+ int state;
+
+ p = NULL;
+ current->lines = current->start = current->end = 0;
+ state = 0;
+
+ /* read the first line */
+ if (!s)
+ if (!(s = fgets(line, 1000, this->fd))) return 0;
+
+ do {
+ switch (state) {
+
+ case 0: /* find "START=" */
+ s = strstr (s, "Start=");
+ if (s) {
+ current->start = strtol (s + 6, &s, 0) / 10;
+ state = 1; continue;
+ }
+ break;
+
+ case 1: /* find "<P" */
+ if ((s = strstr (s, "<P"))) { s += 2; state = 2; continue; }
+ break;
+
+ case 2: /* find ">" */
+ if ((s = strchr (s, '>'))) { s++; state = 3; p = text; continue; }
+ break;
+
+ case 3: /* get all text until '<' appears */
+ if (*s == '\0') { break; }
+ else if (*s == '<') { state = 4; }
+ else if (!strncasecmp (s, "&nbsp;", 6)) { *p++ = ' '; s += 6; }
+ else if (*s == '\r') { s++; }
+ else if (!strncasecmp (s, "<br>", 4) || *s == '\n') {
+ *p = '\0'; p = text; trail_space (text);
+ if (text[0] != '\0')
+ current->text[current->lines++] = strdup (text);
+ if (*s == '\n') s++; else s += 4;
+ }
+ else *p++ = *s++;
+ continue;
+
+ case 4: /* get current->end or skip <TAG> */
+ q = strstr (s, "Start=");
+ if (q) {
+ current->end = strtol (q + 6, &q, 0) / 10 - 1;
+ *p = '\0'; trail_space (text);
+ if (text[0] != '\0')
+ current->text[current->lines++] = strdup (text);
+ if (current->lines > 0) { state = 99; break; }
+ state = 0; continue;
+ }
+ s = strchr (s, '>');
+ if (s) { s++; state = 3; continue; }
+ break;
+ }
+
+ /* read next line */
+ if (state != 99 && !(s = fgets (line, 1000, this->fd)))
+ return 0;
+
+ } while (state != 99);
+
+ return current;
+}
+
+
+static char *sub_readtext(char *source, char **dest) {
+ int len=0;
+ char *p=source;
+
+ while ( !eol(*p) && *p!= '|' ) {
+ p++,len++;
+ }
+
+ *dest= (char *)xine_xmalloc (len+1);
+ if (!dest)
+ return ERR;
+
+ strncpy(*dest, source, len);
+ (*dest)[len]=0;
+
+ while (*p=='\r' || *p=='\n' || *p=='|')
+ p++;
+
+ if (*p) return p; /* not-last text field */
+ else return NULL; /* last text field */
+}
+
+static subtitle_t *sub_read_line_microdvd(sputext_decoder_t *this, subtitle_t *current) {
+
+ char line[1001];
+ char line2[1001];
+ char *p, *next;
+ int i;
+
+ bzero (current, sizeof(subtitle_t));
+
+ do {
+ if (!fgets (line, 1000, this->fd)) return NULL;
+ } while (sscanf (line, "{%ld}{%ld}%[^\r\n]", &(current->start), &(current->end),line2) <3);
+
+ p=line2;
+
+ next=p, i=0;
+ while ((next =sub_readtext (next, &(current->text[i])))) {
+ if (current->text[i]==ERR) return ERR;
+ i++;
+ if (i>=SUB_MAX_TEXT) {
+ printf ("Too many lines in a subtitle\n");
+ current->lines=i;
+ return current;
+ }
+ }
+ current->lines= ++i;
+
+ return current;
+}
+
+static subtitle_t *sub_read_line_subrip(sputext_decoder_t *this, subtitle_t *current) {
+
+ char line[1001];
+ int a1,a2,a3,a4,b1,b2,b3,b4;
+ char *p=NULL, *q=NULL;
+ int len;
+
+ bzero (current, sizeof(subtitle_t));
+
+ while (1) {
+ if (!fgets (line, 1000, this->fd)) return NULL;
+ if (sscanf (line, "%d:%d:%d.%d,%d:%d:%d.%d",&a1,&a2,&a3,&a4,&b1,&b2,&b3,&b4) < 8) continue;
+ current->start = a1*360000+a2*6000+a3*100+a4;
+ current->end = b1*360000+b2*6000+b3*100+b4;
+
+ if (!fgets (line, 1000, this->fd)) return NULL;
+
+ p=q=line;
+ for (current->lines=1; current->lines < SUB_MAX_TEXT; current->lines++) {
+ for (q=p,len=0; *p && *p!='\r' && *p!='\n' && strncmp(p,"[br]",4); p++,len++);
+ current->text[current->lines-1]=(char *)xine_xmalloc (len+1);
+ if (!current->text[current->lines-1]) return ERR;
+ strncpy (current->text[current->lines-1], q, len);
+ current->text[current->lines-1][len]='\0';
+ if (!*p || *p=='\r' || *p=='\n') break;
+ while (*p++!=']');
+ }
+ break;
+ }
+ return current;
+}
+
+static subtitle_t *sub_read_line_third(sputext_decoder_t *this,subtitle_t *current) {
+ char line[1001];
+ int a1,a2,a3,a4,b1,b2,b3,b4;
+ char *p=NULL;
+ int i,len;
+
+ bzero (current, sizeof(subtitle_t));
+
+ while (!current->text[0]) {
+ if (!fgets (line, 1000, this->fd)) return NULL;
+ if ((len=sscanf (line, "%d:%d:%d,%d --> %d:%d:%d,%d",&a1,&a2,&a3,&a4,&b1,&b2,&b3,&b4)) < 8)
+ continue;
+ current->start = a1*360000+a2*6000+a3*100+a4/10;
+ current->end = b1*360000+b2*6000+b3*100+b4/10;
+ for (i=0; i<SUB_MAX_TEXT;) {
+ if (!fgets (line, 1000, this->fd)) break;
+ len=0;
+ for (p=line; *p!='\n' && *p!='\r' && *p; p++,len++);
+ if (len) {
+ current->text[i]=(char *)xine_xmalloc (len+1);
+ if (!current->text[i]) return ERR;
+ strncpy (current->text[i], line, len); current->text[i][len]='\0';
+ i++;
+ } else {
+ break;
+ }
+ }
+ current->lines=i;
+ }
+ return current;
+}
+
+static subtitle_t *sub_read_line_vplayer(sputext_decoder_t *this,subtitle_t *current) {
+ char line[1001];
+ char line2[1001];
+ int a1,a2,a3,b1,b2,b3;
+ char *p=NULL, *next;
+ int i,len,len2,plen;
+
+ bzero (current, sizeof(subtitle_t));
+
+ while (!current->text[0]) {
+ if (!fgets (line, 1000, this->fd)) return NULL;
+ if ((len=sscanf (line, "%d:%d:%d:%n",&a1,&a2,&a3,&plen)) < 3)
+ continue;
+ if (!fgets (line2, 1000, this->fd)) return NULL;
+ if ((len2=sscanf (line2, "%d:%d:%d:",&b1,&b2,&b3)) < 3)
+ continue;
+ /* przewiń o linijkę do tyłu: */
+ fseek(this->fd,-strlen(line2),SEEK_CUR);
+
+ current->start = a1*360000+a2*6000+a3*100;
+ current->end = b1*360000+b2*6000+b3*100;
+ if ((current->end - current->start) > 1000)
+ current->end = current->start + 1000; /* not too long though. */
+ /* teraz czas na wkopiowanie stringu */
+ p=line;
+ /* finds the body of the subtitle_t */
+ for (i=0; i<3; i++){
+ p=strchr(p,':')+1;
+ }
+ i=0;
+
+ if (*p!='|') {
+ next = p,i=0;
+ while ((next =sub_readtext (next, &(current->text[i])))) {
+ if (current->text[i]==ERR)
+ return ERR;
+ i++;
+ if (i>=SUB_MAX_TEXT) {
+ printf ("Too many lines in a subtitle\n");
+ current->lines=i;
+ return current;
+ }
+ }
+ current->lines=i+1;
+ }
+ }
+ return current;
+}
+
+static subtitle_t *sub_read_line_rt(sputext_decoder_t *this,subtitle_t *current) {
+ /*
+ * TODO: This format uses quite rich (sub/super)set of xhtml
+ * I couldn't check it since DTD is not included.
+ * WARNING: full XML parses can be required for proper parsing
+ */
+ char line[1001];
+ int a1,a2,a3,a4,b1,b2,b3,b4;
+ char *p=NULL,*next=NULL;
+ int i,len,plen;
+
+ bzero (current, sizeof(subtitle_t));
+
+ while (!current->text[0]) {
+ if (!fgets (line, 1000, this->fd)) return NULL;
+ /*
+ * TODO: it seems that format of time is not easily determined, it may be 1:12, 1:12.0 or 0:1:12.0
+ * to describe the same moment in time. Maybe there are even more formats in use.
+ */
+ if ((len=sscanf (line, "<Time Begin=\"%d:%d:%d.%d\" End=\"%d:%d:%d.%d\"",&a1,&a2,&a3,&a4,&b1,&b2,&b3,&b4)) < 8)
+
+ plen=a1=a2=a3=a4=b1=b2=b3=b4=0;
+ if (
+ ((len=sscanf (line, "<%*[tT]ime %*[bB]egin=\"%d:%d\" %*[Ee]nd=\"%d:%d\"%*[^<]<clear/>%n",&a2,&a3,&b2,&b3,&plen)) < 4) &&
+ ((len=sscanf (line, "<%*[tT]ime %*[bB]egin=\"%d:%d\" %*[Ee]nd=\"%d:%d.%d\"%*[^<]<clear/>%n",&a2,&a3,&b2,&b3,&b4,&plen)) < 5) &&
+ /* ((len=sscanf (line, "<%*[tT]ime %*[bB]egin=\"%d:%d.%d\" %*[Ee]nd=\"%d:%d\"%*[^<]<clear/>%n",&a2,&a3,&a4,&b2,&b3,&plen)) < 5) && */
+ ((len=sscanf (line, "<%*[tT]ime %*[bB]egin=\"%d:%d.%d\" %*[Ee]nd=\"%d:%d.%d\"%*[^<]<clear/>%n",&a2,&a3,&a4,&b2,&b3,&b4,&plen)) < 6) &&
+ ((len=sscanf (line, "<%*[tT]ime %*[bB]egin=\"%d:%d:%d.%d\" %*[Ee]nd=\"%d:%d:%d.%d\"%*[^<]<clear/>%n",&a1,&a2,&a3,&a4,&b1,&b2,&b3,&b4,&plen)) < 8)
+ )
+ continue;
+ current->start = a1*360000+a2*6000+a3*100+a4/10;
+ current->end = b1*360000+b2*6000+b3*100+b4/10;
+ p=line; p+=plen;i=0;
+ /* TODO: I don't know what kind of convention is here for marking multiline subs, maybe <br/> like in xml? */
+ next = strstr(line,"<clear/>")+8;i=0;
+ while ((next =sub_readtext (next, &(current->text[i])))) {
+ if (current->text[i]==ERR)
+ return ERR;
+ i++;
+ if (i>=SUB_MAX_TEXT) {
+ printf ("Too many lines in a subtitle\n");
+ current->lines=i;
+ return current;
+ }
+ }
+ current->lines=i+1;
+ }
+ return current;
+}
+
+static subtitle_t *sub_read_line_ssa(sputext_decoder_t *this,subtitle_t *current) {
+
+ int hour1, min1, sec1, hunsec1,
+ hour2, min2, sec2, hunsec2, nothing;
+ int num;
+
+ char line[1000],
+ line3[1000],
+ *line2;
+ char *tmp;
+
+ do {
+ if (!fgets (line, 1000, this->fd))
+ return NULL;
+ } while (sscanf (line, "Dialogue: Marked=%d,%d:%d:%d.%d,%d:%d:%d.%d,"
+ "%[^\n\r]", &nothing,
+ &hour1, &min1, &sec1, &hunsec1,
+ &hour2, &min2, &sec2, &hunsec2,
+ line3) < 9);
+ line2=strstr(line3,",,");
+ if (!line2) return NULL;
+ line2 ++;
+ line2 ++;
+
+ current->lines=1;num=0;
+ current->start = 360000*hour1 + 6000*min1 + 100*sec1 + hunsec1;
+ current->end = 360000*hour2 + 6000*min2 + 100*sec2 + hunsec2;
+
+ while ( (tmp=strstr(line2, "\\n")) ) {
+ current->text[num]=(char *)xine_xmalloc(tmp-line2+1);
+ strncpy (current->text[num], line2, tmp-line2);
+ current->text[num][tmp-line2]='\0';
+ line2=tmp+2;
+ num++;
+ current->lines++;
+ if (current->lines >= SUB_MAX_TEXT) return current;
+ }
+
+
+ current->text[num]=(char *) xine_xmalloc(strlen(line2)+1);
+ strcpy(current->text[num],line2);
+
+ return current;
+}
+
+static subtitle_t *sub_read_line_dunnowhat (sputext_decoder_t *this, subtitle_t *current) {
+ char line[1001];
+ char text[1001];
+
+ bzero (current, sizeof(subtitle_t));
+
+ if (!fgets (line, 1000, this->fd))
+ return NULL;
+ if (sscanf (line, "%ld,%ld,\"%[^\"]", &(current->start),
+ &(current->end), text) <3)
+ return ERR;
+ current->text[0] = strdup(text);
+ current->lines = 1;
+
+ return current;
+}
+
+static subtitle_t *sub_read_line_mpsub (sputext_decoder_t *this, subtitle_t *current) {
+ char line[1000];
+ float a,b;
+ int num=0;
+ char *p, *q;
+
+ do {
+ if (!fgets(line, 1000, this->fd))
+ return NULL;
+ } while (sscanf (line, "%f %f", &a, &b) !=2);
+
+ this->mpsub_position += (a*100.0);
+ current->start = (int) this->mpsub_position;
+ this->mpsub_position += (b*100.0);
+ current->end = (int) this->mpsub_position;
+
+ while (num < SUB_MAX_TEXT) {
+ if (!fgets (line, 1000, this->fd))
+ return NULL;
+
+ p=line;
+ while (isspace(*p))
+ p++;
+
+ if (eol(*p) && num > 0)
+ return current;
+
+ if (eol(*p))
+ return NULL;
+
+ for (q=p; !eol(*q); q++);
+ *q='\0';
+ if (strlen(p)) {
+ current->text[num]=strdup(p);
+ printf (">%s<\n",p);
+ current->lines = ++num;
+ } else {
+ if (num)
+ return current;
+ else
+ return NULL;
+ }
+ }
+
+ return NULL;
+}
+
+static subtitle_t *sub_read_line_aqt (sputext_decoder_t *this, subtitle_t *current) {
+ char line[1001];
+
+ bzero (current, sizeof(subtitle_t));
+
+ while (1) {
+ /* try to locate next subtitle_t */
+ if (!fgets (line, 1000, this->fd))
+ return NULL;
+ if (!(sscanf (line, "-->> %ld", &(current->start)) <1))
+ break;
+ }
+
+ if (this->previous_aqt_sub != NULL)
+ this->previous_aqt_sub->end = current->start-1;
+
+ this->previous_aqt_sub = current;
+
+ if (!fgets (line, 1000, this->fd))
+ return NULL;
+
+ sub_readtext((char *) &line,&current->text[0]);
+ current->lines = 1;
+ current->end = current->start; /* will be corrected by next subtitle_t */
+
+ if (!fgets (line, 1000, this->fd))
+ return current;;
+
+ sub_readtext((char *) &line,&current->text[1]);
+ current->lines = 2;
+
+ if ((current->text[0]=="") && (current->text[1]=="")) {
+ /* void subtitle -> end of previous marked and exit */
+ this->previous_aqt_sub = NULL;
+ return NULL;
+ }
+
+ return current;
+}
+
+static int sub_autodetect (sputext_decoder_t *this) {
+
+ char line[1001];
+ int i,j=0;
+ char p;
+
+ while (j < 100) {
+ j++;
+ if (!fgets (line, 1000, this->fd))
+ return -1;
+
+ if (sscanf (line, "{%d}{%d}", &i, &i)==2) {
+ this->uses_time=0;
+ printf ("sputext: microdvd subtitle format detected\n");
+ return FORMAT_MICRODVD;
+ }
+
+ if (sscanf (line, "%d:%d:%d.%d,%d:%d:%d.%d", &i, &i, &i, &i, &i, &i, &i, &i)==8){
+ this->uses_time=1;
+ printf ("sputext: subrip subtitle format detected\n");
+ return FORMAT_SUBRIP;
+ }
+
+ if (sscanf (line, "%d:%d:%d,%d --> %d:%d:%d,%d", &i, &i, &i, &i, &i, &i, &i, &i)==8) {
+ this->uses_time=1;
+ printf ("sputext: subviewer subtitle format detected\n");
+ return FORMAT_SUBVIEWER;
+ }
+ if (strstr (line, "<SAMI>")) {
+ this->uses_time=1;
+ printf ("sputext: sami subtitle format detected\n");
+ return FORMAT_SAMI;
+ }
+ if (sscanf (line, "%d:%d:%d:", &i, &i, &i )==3) {
+ this->uses_time=1;
+ printf ("sputext: vplayer subtitle format detected\n");
+ return FORMAT_VPLAYER;
+ }
+ /*
+ * TODO: just checking if first line of sub starts with "<" is WAY
+ * too weak test for RT
+ * Please someone who knows the format of RT... FIX IT!!!
+ * It may conflict with other sub formats in the future (actually it doesn't)
+ */
+ if ( *line == '<' ) {
+ this->uses_time=1;
+ printf ("sputext: rt subtitle format detected\n");
+ return FORMAT_RT;
+ }
+ if (!memcmp(line, "Dialogue: Marked", 16)){
+ this->uses_time=1;
+ printf ("sputext: ssa subtitle format detected\n");
+ return FORMAT_SSA;
+ }
+ if (sscanf (line, "%d,%d,\"%c", &i, &i, (char *) &i) == 3) {
+ this->uses_time=0;
+ printf ("sputext: (dunno) subtitle format detected\n");
+ return FORMAT_DUNNO;
+ }
+ if (sscanf (line, "FORMAT=%d", &i) == 1) {
+ this->uses_time=0;
+ printf ("sputext: mpsub subtitle format detected\n");
+ return FORMAT_MPSUB;
+ }
+ if (sscanf (line, "FORMAT=TIM%c", &p)==1 && p=='E') {
+ this->uses_time=1;
+ printf ("sputext: mpsub subtitle format detected\n");
+ return FORMAT_MPSUB;
+ }
+ if (strstr (line, "-->>")) {
+ this->uses_time=0;
+ printf ("sputext: aqtitle subtitle format detected\n");
+ return FORMAT_AQTITLE;
+ }
+ }
+
+ return -1; /* too many bad lines */
+}
+
+static subtitle_t *sub_read_file (sputext_decoder_t *this) {
+
+ int n_max;
+ subtitle_t *first;
+ subtitle_t * (*func[])(sputext_decoder_t *this,subtitle_t *dest)=
+ {
+ sub_read_line_microdvd,
+ sub_read_line_subrip,
+ sub_read_line_third,
+ sub_read_line_sami,
+ sub_read_line_vplayer,
+ sub_read_line_rt,
+ sub_read_line_ssa,
+ sub_read_line_dunnowhat,
+ sub_read_line_mpsub,
+ sub_read_line_aqt
+
+ };
+
+ this->format=sub_autodetect (this);
+ if (this->format==-1) {
+ printf ("sputext: Could not determine file format\n");
+ return NULL;
+ }
+
+ printf ("sputext: Detected subtitle file format: %d\n",this->format);
+
+ rewind (this->fd);
+
+ this->num=0;n_max=32;
+ first = (subtitle_t *) xine_xmalloc(n_max*sizeof(subtitle_t));
+ if(!first) return NULL;
+
+ while(1){
+ subtitle_t *sub;
+ if(this->num>=n_max){
+ n_max+=16;
+ first=realloc(first,n_max*sizeof(subtitle_t));
+ }
+ sub = func[this->format] (this, &first[this->num]);
+ if (!sub)
+ break; /* EOF */
+
+ if (sub==ERR)
+ ++this->errs;
+ else {
+ ++this->num; /* Error vs. Valid */
+
+ }
+ }
+
+ printf ("sputext: Read %i subtitles", this->num);
+ if (this->errs)
+ printf (", %i bad line(s).\n", this->errs);
+ else
+ printf (".\n");
+
+ return first;
+}
+
+
+static void list_sub_file (sputext_decoder_t *this, subtitle_t* subs) {
+
+ int i,j;
+
+ for(j=0;j<this->num;j++){
+ subtitle_t* egysub=&subs[j];
+ printf ("%i line%c (%li-%li) ",
+ egysub->lines,
+ (1==egysub->lines)?' ':'s',
+ egysub->start,
+ egysub->end);
+ for (i=0; i<egysub->lines; i++) {
+ printf ("%s%s",egysub->text[i], i==egysub->lines-1?"":" <BREAK> ");
+ }
+ printf ("\n");
+ }
+
+ printf ("Subtitle format %s time.\n", this->uses_time?"uses":"doesn't use");
+ printf ("Read %i subtitles, %i errors.\n", this->num, this->errs);
+
+}
+
+static int spudec_can_handle (spu_decoder_t *this_gen, int buf_type) {
+ int type = buf_type & 0xFFFF0000;
+ return (type == BUF_SPU_TEXT);
+}
+
+
+
+static void spudec_init (spu_decoder_t *this_gen, vo_instance_t *vo_out) {
+
+ sputext_decoder_t *this = (sputext_decoder_t *) this_gen;
+
+ this->vo_out = vo_out;
+ this->output_open = 0;
+
+ this->mpsub_position = 0;
+ this->uses_time = 0;
+ this->errs = 0;
+ this->num = 0;
+ this->format = 0;
+ this->previous_aqt_sub = NULL;
+
+ this->osd = osd_open (this->xine->osd_renderer, 500, 100);
+
+ osd_set_font (this->osd,"vga");
+ osd_render_text (this->osd, 0, 0, "sputext decoder");
+ osd_set_position (this->osd, 10, 30);
+
+ osd_show (this->osd, 0);
+ osd_hide (this->osd, 300000);
+
+}
+
+#define PTS_FACTOR 9000
+
+static void *spudec_loop (void *this_gen) {
+
+ sputext_decoder_t *this = (sputext_decoder_t *) this_gen;
+
+ struct timespec tenth_second;
+ subtitle_t *current;
+ int count;
+ int32_t pts_factor;
+
+ tenth_second.tv_sec = 0;
+ tenth_second.tv_nsec = 100000000;
+
+ /* wait two seconds so metronom can determine the correct frame rate */
+
+ for (count = 0; count<20; count++)
+ nanosleep (&tenth_second, NULL);
+
+ current = this->subtitles;
+
+ count = 0;
+
+ while (current && this->running) {
+
+ int32_t diff, sub_pts, wrap_offset, pts;
+
+ pts = this->xine->metronom->get_current_time (this->xine->metronom);
+
+ if (this->uses_time)
+ pts_factor = 9000;
+ else
+ pts_factor = this->xine->metronom->get_video_rate (this->xine->metronom);
+
+#ifdef LOG
+ printf ("pts_factor : %d\n", pts_factor);
+#endif
+
+ if (!pts_factor) {
+ nanosleep (&tenth_second, NULL);
+ continue;
+ }
+
+ sub_pts = (current->start * pts_factor) + this->xine->metronom->video_wrap_offset;
+
+ diff = sub_pts - pts ;
+
+ if (diff < 0) {
+
+#ifdef LOG
+ printf ("sputext: current is '%s' - too old start time %d, diff %d\n",
+ current->text[0], sub_pts, diff );
+#endif
+
+ current++; count++;
+ if (count >= this->num)
+ current = NULL;
+
+ continue;
+
+ }
+
+#ifdef LOG
+ printf ("sputext: current is '%s' (actually %d lines), start time %d, diff %d\n",
+ current->text[0], current->lines, sub_pts, diff);
+#endif
+
+ if (diff < 30000) {
+
+ int line;
+
+ osd_filled_rect (this->osd, 0, 0, 499, 99, 0);
+
+ for (line=0; line<current->lines; line++)
+ osd_render_text (this->osd, 0, line*20, current->text[line]);
+
+ osd_show (this->osd, sub_pts);
+ osd_hide (this->osd, current->end * pts_factor + this->xine->metronom->video_wrap_offset);
+
+ current++; count++;
+ if (count >= this->num)
+ current = NULL;
+
+ }
+
+ nanosleep (&tenth_second, NULL);
+ }
+
+ printf ("sputext: thread finished\n");
+
+ pthread_exit (NULL);
+
+ return NULL;
+}
+
+static void spudec_decode_data (spu_decoder_t *this_gen, buf_element_t *buf) {
+
+ sputext_decoder_t *this = (sputext_decoder_t *) this_gen;
+ int err;
+
+ this->fd = (FILE *) buf->content;
+
+ this->subtitles = sub_read_file (this);
+
+ printf ("sputext: subtitle format %s time.\n", this->uses_time?"uses":"doesn't use");
+ printf ("sputext: read %i subtitles, %i errors.\n", this->num, this->errs);
+
+ /* start thread */
+
+ this->running = 1;
+
+ if ((err = pthread_create (&this->spu_thread,
+ NULL, spudec_loop, this)) != 0) {
+ printf ("sputext: can't create new thread (%s)\n",
+ strerror(err));
+ exit (1);
+ }
+
+}
+
+
+static void spudec_close (spu_decoder_t *this_gen) {
+ sputext_decoder_t *this = (sputext_decoder_t *) this_gen;
+ void *p;
+
+ printf ("sputext: stopping thread...\n");
+
+ this->running = 0;
+
+ pthread_join (this->spu_thread, &p);
+
+ printf ("sputext: ...thread stopped\n");
+}
+
+static char *spudec_get_id(void) {
+ return "sputext";
+}
+
+spu_decoder_t *init_spu_decoder_plugin (int iface_version, xine_t *xine) {
+
+ sputext_decoder_t *this ;
+
+ if (iface_version != 4) {
+ printf("libsputext: doesn't support plugin api version %d.\n"
+ "libsputext: This means there is a version mismatch between xine and\n"
+ "libsputext: this plugin.\n", iface_version);
+ return NULL;
+ }
+
+ this = (sputext_decoder_t *) xine_xmalloc (sizeof (sputext_decoder_t));
+
+ this->spu_decoder.interface_version = 4;
+ this->spu_decoder.can_handle = spudec_can_handle;
+ this->spu_decoder.init = spudec_init;
+ this->spu_decoder.decode_data = spudec_decode_data;
+ this->spu_decoder.close = spudec_close;
+ this->spu_decoder.get_identifier = spudec_get_id;
+ this->spu_decoder.priority = 1;
+
+ this->xine = xine;
+
+ return (spu_decoder_t *) this;
+}
+
diff --git a/src/xine-engine/buffer.h b/src/xine-engine/buffer.h
index 22288ecbd..480f1031e 100644
--- a/src/xine-engine/buffer.h
+++ b/src/xine-engine/buffer.h
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: buffer.h,v 1.25 2001/11/30 19:31:55 jcdutton Exp $
+ * $Id: buffer.h,v 1.26 2001/12/01 22:38:32 guenter Exp $
*
*
* contents:
@@ -123,6 +123,7 @@ extern "C" {
#define BUF_SPU_PACKAGE 0x04010000
#define BUF_SPU_SUBP_CONTROL 0x04020000
#define BUF_SPU_NAV 0x04030000
+#define BUF_SPU_TEXT 0x04040000
/* demuxer block types: */
diff --git a/src/xine-engine/configfile.c b/src/xine-engine/configfile.c
index abde13c37..02e220733 100644
--- a/src/xine-engine/configfile.c
+++ b/src/xine-engine/configfile.c
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: configfile.c,v 1.12 2001/11/30 21:55:06 f1rmb Exp $
+ * $Id: configfile.c,v 1.13 2001/12/01 22:38:32 guenter Exp $
*
* config file management - implementation
*
@@ -101,27 +101,6 @@ cfg_entry_t *config_file_lookup_entry (config_values_t *this, char *key) {
}
-static void config_file_register_empty (config_values_t *this, char *key) {
- cfg_entry_t *entry;
-
- assert(key);
-
-#ifdef CONFIG_LOG
- printf ("configfile: register empty %s\n", key);
-#endif
-
- entry = config_file_lookup_entry (this, key);
-
- /*
- * Don't register as empty if entry already exist.
- */
- if(entry)
- return;
-
- entry = config_file_add (this, key);
-}
-
-
static char *config_file_register_string (config_values_t *this,
char *key, char *def_value,
char *description,
@@ -594,7 +573,6 @@ config_values_t *config_file_init (char *filename) {
exit (1);
}
- this->register_empty = config_file_register_empty;
this->register_string = config_file_register_string;
this->register_range = config_file_register_range;
this->register_enum = config_file_register_enum;
@@ -613,6 +591,9 @@ config_values_t *config_file_init (char *filename) {
/*
* $Log: configfile.c,v $
+ * Revision 1.13 2001/12/01 22:38:32 guenter
+ * add avi subtitle decoder (based on mplayer code), minor cleanups, removed register_empty function from configfile (undocumented and doesn't make sense)
+ *
* Revision 1.12 2001/11/30 21:55:06 f1rmb
* Add an automatic way for input plugin to add extra valid mrls:
* add at bottom of init_input_plugin() a line like this:
diff --git a/src/xine-engine/configfile.h b/src/xine-engine/configfile.h
index 4085098b8..74831d526 100644
--- a/src/xine-engine/configfile.h
+++ b/src/xine-engine/configfile.h
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: configfile.h,v 1.6 2001/11/30 21:55:06 f1rmb Exp $
+ * $Id: configfile.h,v 1.7 2001/12/01 22:38:32 guenter Exp $
*
* config file management
*
@@ -95,9 +95,6 @@ struct config_values_s {
* from the config file otherwise
*/
- void (*register_empty) (config_values_t *this,
- char *key);
-
char* (*register_string) (config_values_t *this,
char *key,
char *def_value,
@@ -191,6 +188,9 @@ config_values_t *config_file_init (char *filename);
/*
* $Log: configfile.h,v $
+ * Revision 1.7 2001/12/01 22:38:32 guenter
+ * add avi subtitle decoder (based on mplayer code), minor cleanups, removed register_empty function from configfile (undocumented and doesn't make sense)
+ *
* Revision 1.6 2001/11/30 21:55:06 f1rmb
* Add an automatic way for input plugin to add extra valid mrls:
* add at bottom of init_input_plugin() a line like this:
diff --git a/src/xine-engine/video_decoder.c b/src/xine-engine/video_decoder.c
index d03d8d7ba..4d4c10b6c 100644
--- a/src/xine-engine/video_decoder.c
+++ b/src/xine-engine/video_decoder.c
@@ -17,7 +17,7 @@
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*
- * $Id: video_decoder.c,v 1.65 2001/11/17 14:26:39 f1rmb Exp $
+ * $Id: video_decoder.c,v 1.66 2001/12/01 22:38:32 guenter Exp $
*
*/
@@ -134,6 +134,7 @@ void *video_decoder_loop (void *this_gen) {
case BUF_SPU_SUBP_CONTROL:
case BUF_SPU_CLUT:
case BUF_SPU_PACKAGE:
+ case BUF_SPU_TEXT:
xine_profiler_start_count (prof_spu_decode);
spu_decoder = update_spu_decoder(this, buf->type);