diff options
-rw-r--r-- | src/libsputext/Makefile.am | 14 | ||||
-rw-r--r-- | src/libsputext/demux_sputext.c | 977 |
2 files changed, 987 insertions, 4 deletions
diff --git a/src/libsputext/Makefile.am b/src/libsputext/Makefile.am index fd001f9e8..642586265 100644 --- a/src/libsputext/Makefile.am +++ b/src/libsputext/Makefile.am @@ -3,15 +3,21 @@ LIBTOOL = $(SHELL) $(top_builddir)/libtool-nofpic libdir = $(XINE_PLUGINDIR) if HAVE_GICONV -sputext_decoder = xineplug_decode_sputext.la +#sputext_decoder = xineplug_decode_sputext.la xineplug_dmx_sputext.la +sputext_decoder = xineplug_dmx_sputext.la endif lib_LTLIBRARIES = $(sputext_decoder) -xineplug_decode_sputext_la_SOURCES = xine_decoder.c -xineplug_decode_sputext_la_LDFLAGS = -avoid-version -module @XINE_PLUGIN_MIN_SYMS@ +xineplug_dmx_sputext_la_SOURCES = demux_sputext.c ## libgiconv is only needed with FreeBSD -xineplug_decode_sputext_la_LIBADD = $(GICONV_BSD_LIBS) +xineplug_dmx_sputext_la_LIBADD = $(XINELIB) $(GICONV_BSD_LIBS) +xineplug_dmx_sputext_la_LDFLAGS = -avoid-version -module @XINE_PLUGIN_MIN_SYMS@ + +#xineplug_decode_sputext_la_SOURCES = xine_decoder.c +## libgiconv is only needed with FreeBSD +#xineplug_decode_sputext_la_LIBADD = $(XINELIB) +#xineplug_decode_sputext_la_LDFLAGS = -avoid-version -module @XINE_PLUGIN_MIN_SYMS@ ## ## Install header files (default=$includedir/xine) diff --git a/src/libsputext/demux_sputext.c b/src/libsputext/demux_sputext.c new file mode 100644 index 000000000..d316e2bf3 --- /dev/null +++ b/src/libsputext/demux_sputext.c @@ -0,0 +1,977 @@ +/* + * Copyright (C) 2000-2002 the xine project + * + * This file is part of xine, a free 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: demux_sputext.c,v 1.1 2002/12/29 16:39:38 mroi Exp $ + * + * code based on old libsputext/xine_decoder.c + * + * 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 + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#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 <iconv.h> + +#include "xine_internal.h" +#include "xineutils.h" +#include "../demuxers/demux.h" +#include "osd.h" + +#define LOG 1 + +#define ERR (void *)-1 + +#define SUB_MAX_TEXT 5 + +#define SUB_BUFSIZE 1024 + +/* + * Demuxer typedefs + */ + + +typedef struct { + + int lines; + + unsigned long start; + unsigned long end; + + char *text[SUB_MAX_TEXT]; + +} subtitle_t; + + +typedef struct { + + demux_plugin_t demux_plugin; + xine_stream_t *stream; + input_plugin_t *input; + + int status; + + char buf[SUB_BUFSIZE]; + off_t buflen; + + float mpsub_position; + + int uses_time; + int errs; + subtitle_t *subtitles; + int num; /* number of subtitle structs */ + int cur; /* current subtitle */ + int format; /* constants see below */ + subtitle_t *previous_aqt_sub ; + +} demux_sputext_t; + +typedef struct demux_sputext_class_s { + + demux_class_t demux_class; + + xine_t *xine; + config_values_t *config; + + char *src_encoding; + char *dst_encoding; + +} demux_sputext_class_t; + +/* + * Demuxer code start + */ + +#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'; +} + +/* + * Reimplementation of fgets() using the input->read() method. + */ +static char *read_line_from_input(demux_sputext_t *this, char *line, off_t len) { + off_t nread = 0; + char *s; + int linelen; + + if ((len - this->buflen) >512) + nread = this->input->read(this->input, &this->buf[this->buflen], len - this->buflen); + + this->buflen += nread; + this->buf[this->buflen] = '\0'; + + s = strstr(this->buf, "\n"); + if (line && (s || this->buflen)) { + + linelen = s ? (s - this->buf) + 1 : this->buflen; + + memcpy(line, this->buf, linelen); + line[linelen] = '\0'; + + memmove(this->buf, &this->buf[linelen], SUB_BUFSIZE - linelen); + this->buflen -= linelen; + + return line; + } + + return NULL; +} + + +static subtitle_t *sub_read_line_sami(demux_sputext_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 = read_line_from_input(this, line, 1000))) 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, " ", 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 = read_line_from_input (this, line, 1000))) + 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(demux_sputext_t *this, subtitle_t *current) { + + char line[1001]; + char line2[1001]; + char *p, *next; + int i; + + bzero (current, sizeof(subtitle_t)); + + current->end=-1; + do { + if (!read_line_from_input (this, line, 1000)) return NULL; + } while ((sscanf (line, "{%ld}{}%[^\r\n]", &(current->start), line2) !=2) && + (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(demux_sputext_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 (!read_line_from_input(this, line, 1000)) 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 (!read_line_from_input(this, line, 1000)) 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(demux_sputext_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 (!read_line_from_input(this, line, 1000)) 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 (!read_line_from_input(this, line, 1000)) 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(demux_sputext_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 (!read_line_from_input(this, line, 1000)) return NULL; + if ((len=sscanf (line, "%d:%d:%d:%n",&a1,&a2,&a3,&plen)) < 3) + continue; + if (!read_line_from_input(this, line2, 1000)) return NULL; + if ((len2=sscanf (line2, "%d:%d:%d:",&b1,&b2,&b3)) < 3) + continue; + /* przewiń o linijkę do tyłu: */ + this->input->seek(this->input, -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(demux_sputext_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 (!read_line_from_input(this, line, 1000)) 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(demux_sputext_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 (!read_line_from_input(this, line, 1000)) + return NULL; + } while (sscanf (line, "Dialogue: Marked=%d,%d:%d:%d.%d,%d:%d:%d.%d," + "%[^\n\r]", ¬hing, + &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 (demux_sputext_t *this, subtitle_t *current) { + char line[1001]; + char text[1001]; + + bzero (current, sizeof(subtitle_t)); + + if (!read_line_from_input(this, line, 1000)) + 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 (demux_sputext_t *this, subtitle_t *current) { + char line[1000]; + float a,b; + int num=0; + char *p, *q; + + do { + if (!read_line_from_input(this, line, 1000)) + 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 (!read_line_from_input(this, line, 1000)) + 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 (demux_sputext_t *this, subtitle_t *current) { + char line[1001]; + + bzero (current, sizeof(subtitle_t)); + + while (1) { + /* try to locate next subtitle_t */ + if (!read_line_from_input(this, line, 1000)) + 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 (!read_line_from_input(this, line, 1000)) + return NULL; + + sub_readtext((char *) &line,¤t->text[0]); + current->lines = 1; + current->end = current->start; /* will be corrected by next subtitle_t */ + + if (!read_line_from_input(this, line, 1000)) + return current;; + + sub_readtext((char *) &line,¤t->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 (demux_sputext_t *this) { + + char line[1001]; + int i,j=0; + char p; + + while (j < 100) { + j++; + if (!read_line_from_input(this, line, 1000)) + return -1; + + if ((sscanf (line, "{%d}{}", &i)==1) || + (sscanf (line, "{%d}{%d}", &i, &i)==2)) { + this->uses_time=0; + printf ("demux_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 ("demux_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 ("demux_sputext: subviewer subtitle format detected\n"); + return FORMAT_SUBVIEWER; + } + if (strstr (line, "<SAMI>")) { + this->uses_time=1; + printf ("demux_sputext: sami subtitle format detected\n"); + return FORMAT_SAMI; + } + if (sscanf (line, "%d:%d:%d:", &i, &i, &i )==3) { + this->uses_time=1; + printf ("demux_sputext: vplayer subtitle format detected\n"); + return FORMAT_VPLAYER; + } + /* + * A RealText format is a markup language, starts with <window> tag, + * options (behaviour modifiers) are possible. + */ + if ( !strcasecmp(line, "<window") ) { + this->uses_time=1; + printf ("demux_sputext: rt subtitle format detected\n"); + return FORMAT_RT; + } + if (!memcmp(line, "Dialogue: Marked", 16)){ + this->uses_time=1; + printf ("demux_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 ("demux_sputext: (dunno) subtitle format detected\n"); + return FORMAT_DUNNO; + } + if (sscanf (line, "FORMAT=%d", &i) == 1) { + this->uses_time=0; + printf ("demux_sputext: mpsub subtitle format detected\n"); + return FORMAT_MPSUB; + } + if (sscanf (line, "FORMAT=TIM%c", &p)==1 && p=='E') { + this->uses_time=1; + printf ("demux_sputext: mpsub subtitle format detected\n"); + return FORMAT_MPSUB; + } + if (strstr (line, "-->>")) { + this->uses_time=0; + printf ("demux_sputext: aqtitle subtitle format detected\n"); + return FORMAT_AQTITLE; + } + } + + return -1; /* too many bad lines */ +} + +static subtitle_t *sub_read_file (demux_sputext_t *this) { + + demux_sputext_class_t *class = (demux_sputext_class_t *)this->demux_plugin.demux_class; + int n_max; + subtitle_t *first; + subtitle_t * (*func[])(demux_sputext_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 + + }; + iconv_t iconv_descr; + + this->format=sub_autodetect (this); + if (this->format==-1) { + printf ("demux_sputext: Could not determine file format\n"); + return NULL; + } + + printf ("demux_sputext: Detected subtitle file format: %d\n",this->format); + + /* Rewind */ + this->input->seek(this->input, 0, SEEK_SET); + this->buflen = 0; + + this->num=0;n_max=32; + first = (subtitle_t *) xine_xmalloc(n_max*sizeof(subtitle_t)); + if(!first) return NULL; + + iconv_descr=iconv_open(class->dst_encoding,class->src_encoding); + + 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 { + int i; + if (this->num > 0 && first[this->num-1].end == -1) { + first[this->num-1].end = sub->start; + } + for(i=0; i<first[this->num].lines; i++) + { char *tmp; + char *in_buff, *out_buff; + int in_len, out_len; + + in_len=strlen(first[this->num].text[i])+1; + tmp=malloc(in_len); + in_buff=first[this->num].text[i]; + out_buff=tmp; + out_len=in_len; + if ((size_t)(-1)!=iconv(iconv_descr,&in_buff,&in_len,&out_buff,&out_len)) + { free(first[this->num].text[i]); + first[this->num].text[i]=tmp; + } + else + { printf("demux_sputext: Can't convert subtitle text\n"); } + } + ++this->num; /* Error vs. Valid */ + } + } + iconv_close(iconv_descr); + + printf ("demux_sputext: Read %i subtitles", this->num); + if (this->errs) + printf (", %i bad line(s).\n", this->errs); + else + printf (".\n"); + + return first; +} + +static void update_osd_src_encoding(void *this_gen, xine_cfg_entry_t *entry) +{ + demux_sputext_t *this = (demux_sputext_t *)this_gen; + demux_sputext_class_t *class = (demux_sputext_class_t *)this->demux_plugin.demux_class; + + class->src_encoding = entry->str_value; + + printf("demux_sputext: spu_src_encoding = %s\n", class->src_encoding ); +} + +static void update_osd_dst_encoding(void *this_gen, xine_cfg_entry_t *entry) +{ + demux_sputext_t *this = (demux_sputext_t *)this_gen; + demux_sputext_class_t *class = (demux_sputext_class_t *)this->demux_plugin.demux_class; + + class->dst_encoding = entry->str_value; + + printf("demux_sputext: spu_dst_encoding = %s\n", class->dst_encoding ); +} + +static int demux_sputext_next (demux_sputext_t *this) { + /* FIXME: send something out here */ + return 1; +} + +static void demux_sputext_dispose (demux_plugin_t *this_gen) { + free (this_gen); +} + +static int demux_sputext_get_status (demux_plugin_t *this_gen) { + demux_sputext_t *this = (demux_sputext_t *) this_gen; + return this->status; +} + +static int demux_sputext_get_stream_length (demux_plugin_t *this_gen) { + /* FIXME: what to return? */ + return 0; +} + +static int demux_sputext_send_chunk (demux_plugin_t *this_gen) { + demux_sputext_t *this = (demux_sputext_t *) this_gen; + + if (!demux_sputext_next (this)) { + this->status = DEMUX_FINISHED; + } + + return this->status; +} + +static int demux_sputext_seek (demux_plugin_t *this_gen, + off_t start_pos, int start_time) { + demux_sputext_t *this = (demux_sputext_t*)this_gen; + return this->status; +} + +static void demux_sputext_send_headers(demux_plugin_t *this_gen) { + demux_sputext_t *this = (demux_sputext_t*)this_gen; + xine_demux_control_start(this->stream); + this->stream->stream_info[XINE_STREAM_INFO_HAS_VIDEO] = 0; + this->stream->stream_info[XINE_STREAM_INFO_HAS_AUDIO] = 0; +} + +static uint32_t demux_sputext_get_capabilities(demux_plugin_t *this_gen) { + return DEMUX_CAP_NOCAP; +} + +static demux_plugin_t *open_demux_plugin (demux_class_t *class_gen, xine_stream_t *stream, + input_plugin_t *input_gen) { + + input_plugin_t *input = (input_plugin_t *) input_gen; + demux_sputext_t *this; + + printf("demux_sputext: open_plugin() called\n"); + + this = xine_xmalloc (sizeof (demux_sputext_t)); + this->stream = stream; + this->input = input; + + this->demux_plugin.send_headers = demux_sputext_send_headers; + this->demux_plugin.send_chunk = demux_sputext_send_chunk; + this->demux_plugin.seek = demux_sputext_seek; + this->demux_plugin.dispose = demux_sputext_dispose; + this->demux_plugin.get_status = demux_sputext_get_status; + this->demux_plugin.get_stream_length = demux_sputext_get_stream_length; + this->demux_plugin.get_video_frame = NULL; + this->demux_plugin.got_video_frame_cb= NULL; + this->demux_plugin.get_capabilities = demux_sputext_get_capabilities; + this->demux_plugin.get_optional_data = NULL; + this->demux_plugin.demux_class = class_gen; + this->status = DEMUX_OK; + + this->buflen = 0; + + switch (stream->content_detection_method) { + case METHOD_BY_EXTENSION: + { + char *mrl, *ending; + + mrl = input->get_mrl(input); + ending = strrchr(mrl, '.'); + + if (!ending || ( + (strncasecmp(ending, ".asc", 4) != 0) && + (strncasecmp(ending, ".txt", 4) != 0) && + (strncasecmp(ending, ".sub", 4) != 0) && + (strncasecmp(ending, ".srt", 4) != 0))) { + free (this); + return NULL; + } + } + /* falling through is intended */ + + case METHOD_EXPLICIT: + /* case METHOD_BY_CONTENT: */ + + /* FIXME: for now this demuxer only works when requested explicitly + * to make sure it does not interfere with others; + * If this is found too inconvenient, this may be changed after making + * sure the content detection does not produce any false positives. + */ + + if ((input->get_capabilities(input) & INPUT_CAP_SEEKABLE) != 0) { + + this->subtitles = sub_read_file (this); + + this->cur = 0; + + if (this->subtitles) { + printf ("demux_sputext: subtitle format %s time.\n", this->uses_time?"uses":"doesn't use"); + printf ("demux_sputext: read %i subtitles, %i errors.\n", this->num, this->errs); + return &this->demux_plugin; + } + } + /* falling through is intended */ + } + + free (this); + return NULL; +} + +static char *get_demux_description (demux_class_t *this_gen) { + return "sputext demuxer plugin"; +} + +static char *get_demux_identifier (demux_class_t *this_gen) { + return "sputext"; +} + +static char *get_demux_extensions (demux_class_t *this_gen) { + return "asc txt sub srt"; +} + +static char *get_demux_mimetypes (demux_class_t *this_gen) { + return "text/plain: asc txt sub srt: VIDEO subtitles;"; +} + +static void demux_class_dispose (demux_class_t *this_gen) { + demux_sputext_class_t *this = (demux_sputext_class_t *) this_gen; + + free (this); +} + +static void *init_sputext_demux_class (xine_t *xine, void *data) { + + demux_sputext_class_t *this ; + + printf("demux_sputext: initializing\n"); + + this = xine_xmalloc (sizeof (demux_sputext_class_t)); + this->config = xine->config; + + this->demux_class.open_plugin = open_demux_plugin; + this->demux_class.get_description = get_demux_description; + this->demux_class.get_identifier = get_demux_identifier; + this->demux_class.get_mimetypes = get_demux_mimetypes; + this->demux_class.get_extensions = get_demux_extensions; + this->demux_class.dispose = demux_class_dispose; + + this->src_encoding = xine->config->register_string(xine->config, + "misc.spu_src_encoding", + "windows-1250", + _("source encoding of subtitles"), + NULL, 10, update_osd_src_encoding, this); + this->dst_encoding = xine->config->register_string(xine->config, + "misc.spu_dst_encoding", + "iso-8859-2", + _("target encoding for subtitles (have to match font encoding)"), + NULL, 10, update_osd_dst_encoding, this); + + return this; +} + +plugin_info_t xine_plugin_info[] = { + /* type, API, "name", version, special_info, init_function */ + { PLUGIN_DEMUX, 19, "sputext", XINE_VERSION_CODE, NULL, &init_sputext_demux_class }, + { PLUGIN_NONE, 0, "", 0, NULL, NULL } +}; |