Commit c20bdd8d authored by Francois Cartegnie's avatar Francois Cartegnie

demux: subtitle: dont break non-seekable streams

parent e35efa48
...@@ -56,7 +56,7 @@ demux_LTLIBRARIES += libaiff_plugin.la ...@@ -56,7 +56,7 @@ demux_LTLIBRARIES += libaiff_plugin.la
libmjpeg_plugin_la_SOURCES = demux/mjpeg.c demux/mxpeg_helper.h libmjpeg_plugin_la_SOURCES = demux/mjpeg.c demux/mxpeg_helper.h
demux_LTLIBRARIES += libmjpeg_plugin.la demux_LTLIBRARIES += libmjpeg_plugin.la
libsubtitle_plugin_la_SOURCES = demux/subtitle.c libsubtitle_plugin_la_SOURCES = demux/subtitle.c demux/subtitle_helper.h
libsubtitle_plugin_la_LIBADD = $(LIBM) libsubtitle_plugin_la_LIBADD = $(LIBM)
demux_LTLIBRARIES += libsubtitle_plugin.la demux_LTLIBRARIES += libsubtitle_plugin.la
...@@ -64,7 +64,8 @@ libty_plugin_la_SOURCES = demux/ty.c codec/cc.h ...@@ -64,7 +64,8 @@ libty_plugin_la_SOURCES = demux/ty.c codec/cc.h
demux_LTLIBRARIES += libty_plugin.la demux_LTLIBRARIES += libty_plugin.la
libvobsub_plugin_la_SOURCES = demux/vobsub.c demux/vobsub.h \ libvobsub_plugin_la_SOURCES = demux/vobsub.c demux/vobsub.h \
demux/mpeg/ps.h demux/mpeg/pes.h demux/mpeg/ps.h demux/mpeg/pes.h \
demux/subtitle_helper.h
demux_LTLIBRARIES += libvobsub_plugin.la demux_LTLIBRARIES += libvobsub_plugin.la
libvoc_plugin_la_SOURCES = demux/voc.c libvoc_plugin_la_SOURCES = demux/voc.c
......
...@@ -38,10 +38,13 @@ ...@@ -38,10 +38,13 @@
#include <ctype.h> #include <ctype.h>
#include <math.h> #include <math.h>
#include <assert.h>
#include <vlc_demux.h> #include <vlc_demux.h>
#include <vlc_charset.h> #include <vlc_charset.h>
#include "subtitle_helper.h"
/***************************************************************************** /*****************************************************************************
* Module descriptor * Module descriptor
*****************************************************************************/ *****************************************************************************/
...@@ -299,6 +302,11 @@ static int Open ( vlc_object_t *p_this ) ...@@ -299,6 +302,11 @@ static int Open ( vlc_object_t *p_this )
} }
free( psz_type ); free( psz_type );
#ifndef NDEBUG
const uint64_t i_start_pos = stream_Tell( p_demux->s );
#endif
uint64_t i_read_offset = 0;
/* Detect Unicode while skipping the UTF-8 Byte Order Mark */ /* Detect Unicode while skipping the UTF-8 Byte Order Mark */
bool unicode = false; bool unicode = false;
const uint8_t *p_data; const uint8_t *p_data;
...@@ -306,7 +314,7 @@ static int Open ( vlc_object_t *p_this ) ...@@ -306,7 +314,7 @@ static int Open ( vlc_object_t *p_this )
&& !memcmp( p_data, "\xEF\xBB\xBF", 3 ) ) && !memcmp( p_data, "\xEF\xBB\xBF", 3 ) )
{ {
unicode = true; unicode = true;
stream_Seek( p_demux->s, 3 ); /* skip BOM */ i_read_offset = 3; /* skip BOM */
msg_Dbg( p_demux, "detected Unicode Byte Order Mark" ); msg_Dbg( p_demux, "detected Unicode Byte Order Mark" );
} }
...@@ -322,7 +330,7 @@ static int Open ( vlc_object_t *p_this ) ...@@ -322,7 +330,7 @@ static int Open ( vlc_object_t *p_this )
int i_dummy; int i_dummy;
char p_dummy; char p_dummy;
if( ( s = stream_ReadLine( p_demux->s ) ) == NULL ) if( (s = peek_Readline( p_demux->s, &i_read_offset )) == NULL )
break; break;
if( strcasestr( s, "<SAMI>" ) ) if( strcasestr( s, "<SAMI>" ) )
...@@ -474,17 +482,15 @@ static int Open ( vlc_object_t *p_this ) ...@@ -474,17 +482,15 @@ static int Open ( vlc_object_t *p_this )
} }
free( s ); free( s );
/* It will nearly always work even for non seekable stream thanks the
* caching system, and if it fails we lose just a few sub */
if( stream_Seek( p_demux->s, unicode ? 3 : 0 ) )
msg_Warn( p_demux, "failed to rewind" );
} }
/* Quit on unknown subtitles */ /* Quit on unknown subtitles */
if( p_sys->i_type == SUB_TYPE_UNKNOWN ) if( p_sys->i_type == SUB_TYPE_UNKNOWN )
{ {
stream_Seek( p_demux->s, 0 ); #ifndef NDEBUG
/* Ensure it will work with non seekable streams */
assert( i_start_pos == stream_Tell( p_demux->s ) );
#endif
msg_Warn( p_demux, "failed to recognize subtitle type" ); msg_Warn( p_demux, "failed to recognize subtitle type" );
free( p_sys ); free( p_sys );
return VLC_EGENERIC; return VLC_EGENERIC;
...@@ -503,6 +509,9 @@ static int Open ( vlc_object_t *p_this ) ...@@ -503,6 +509,9 @@ static int Open ( vlc_object_t *p_this )
msg_Dbg( p_demux, "loading all subtitles..." ); msg_Dbg( p_demux, "loading all subtitles..." );
if( unicode ) /* skip BOM */
stream_Seek( p_demux->s, 3 );
/* Load the whole file */ /* Load the whole file */
TextLoad( &p_sys->txt, p_demux->s ); TextLoad( &p_sys->txt, p_demux->s );
......
/*****************************************************************************
* subtitle.h: subtitle helper functions
*****************************************************************************
* Copyright (C) 2015 VLC authors and VideoLAN
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This program 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
inline static char * peek_Readline( stream_t *p_demuxstream, uint64_t *pi_offset )
{
uint8_t *p_peek;
ssize_t i_peek = stream_Peek( p_demuxstream, (const uint8_t **) &p_peek,
*pi_offset + 2048 );
if( i_peek <= 0 )
return NULL;
const uint64_t i_bufsize = (uint64_t) i_peek - *pi_offset;
char *psz_line = NULL;
/* Create a stream memory from that offset */
stream_t *p_memorystream = stream_MemoryNew( p_demuxstream, &p_peek[*pi_offset],
i_bufsize, true );
if( p_memorystream )
{
psz_line = stream_ReadLine( p_memorystream );
*pi_offset += stream_Tell( p_memorystream );
stream_Delete( p_memorystream );
}
return psz_line;
}
...@@ -39,6 +39,7 @@ ...@@ -39,6 +39,7 @@
#include "mpeg/pes.h" #include "mpeg/pes.h"
#include "mpeg/ps.h" #include "mpeg/ps.h"
#include "vobsub.h" #include "vobsub.h"
#include "subtitle_helper.h"
/***************************************************************************** /*****************************************************************************
* Module descriptor * Module descriptor
...@@ -122,17 +123,14 @@ static int Open ( vlc_object_t *p_this ) ...@@ -122,17 +123,14 @@ static int Open ( vlc_object_t *p_this )
demux_sys_t *p_sys; demux_sys_t *p_sys;
char *psz_vobname, *s; char *psz_vobname, *s;
int i_len; int i_len;
uint64_t i_read_offset = 0;
if( ( s = stream_ReadLine( p_demux->s ) ) != NULL ) if( ( s = peek_Readline( p_demux->s, &i_read_offset ) ) != NULL )
{ {
if( !strcasestr( s, "# VobSub index file" ) ) if( !strcasestr( s, "# VobSub index file" ) )
{ {
msg_Dbg( p_demux, "this doesn't seem to be a vobsub file" ); msg_Dbg( p_demux, "this doesn't seem to be a vobsub file" );
free( s ); free( s );
if( stream_Seek( p_demux->s, 0 ) )
{
msg_Warn( p_demux, "failed to rewind" );
}
return VLC_EGENERIC; return VLC_EGENERIC;
} }
free( s ); free( s );
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment