]> git.sesse.net Git - vlc/commitdiff
Subtitle (subpicture) aspect ratio had been ignored. Use it now.
authorRocky Bernstein <rocky@videolan.org>
Mon, 4 Apr 2005 01:40:11 +0000 (01:40 +0000)
committerRocky Bernstein <rocky@videolan.org>
Mon, 4 Apr 2005 01:40:11 +0000 (01:40 +0000)
Aspect ration in subtitle means use underlying video aspect ratio.

WARNING: Since this parameter was ignored before probably there are a
number of image subtitle that have set this incorrectly. DVD's seemed
to be a case in point and that'd been fixed. Fixing other subtitles
should just be a matter of setting the i_aspect parameter correctly.

modules/codec/spudec/parse.c
modules/codec/svcdsub.c
src/video_output/vout_subpictures.c

index ca35396551294f60bcbc5fc2a90d6c014f20a0c2..2ac813054c66da16bc73790c1bf682ea619caf5a 100644 (file)
@@ -1,7 +1,7 @@
 /*****************************************************************************
  * parse.c: SPU parser
  *****************************************************************************
- * Copyright (C) 2000-2001 VideoLAN
+ * Copyright (C) 2000-2001, 2005 VideoLAN
  * $Id$
  *
  * Authors: Samuel Hocevar <sam@zoy.org>
@@ -589,7 +589,7 @@ static void Render( decoder_t *p_dec, subpicture_t *p_spu,
     /* Create a new subpicture region */
     memset( &fmt, 0, sizeof(video_format_t) );
     fmt.i_chroma = VLC_FOURCC('Y','U','V','P');
-    fmt.i_aspect = VOUT_ASPECT_FACTOR;
+    fmt.i_aspect = 0; /* 0 means use aspect ratio of background video */
     fmt.i_width = fmt.i_visible_width = p_spu->i_width;
     fmt.i_height = fmt.i_visible_height = p_spu->i_height -
         p_spu_data->i_y_top_offset - p_spu_data->i_y_bottom_offset;
index a4f8efc32f14701360548002dc22175e21fc1d2c..40186b1e5bbe78fb4df42b1ee5052257438091b4 100644 (file)
@@ -24,8 +24,6 @@
  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  *****************************************************************************/
 
-#define DEBUG_SVCD 1
-
 /*****************************************************************************
  * Preamble
  *****************************************************************************/
@@ -41,6 +39,15 @@ static int  DecoderOpen   ( vlc_object_t * );
 static int  PacketizerOpen( vlc_object_t * );
 static void DecoderClose  ( vlc_object_t * );
 
+#define DEBUG_TEXT N_( \
+     "If nonzero, this gives additional debug information." \
+     )
+
+#define DEBUG_LONGTEXT N_( \
+    "This integer when viewed in binary is a debugging mask\n" \
+    "calls                 1\n" \
+    "packet assembly info  2\n" )
+
 vlc_module_begin();
     set_description( _("Philips OGT (SVCD subtitle) decoder") );
     set_shortname( N_("SVCD subtitles"));
@@ -49,6 +56,9 @@ vlc_module_begin();
     set_capability( "decoder", 50 );
     set_callbacks( DecoderOpen, DecoderClose );
 
+    add_integer ( MODULE_STRING "-debug", 0, NULL,
+                  DEBUG_TEXT, DEBUG_LONGTEXT, VLC_TRUE );
+
     add_submodule();
     set_description( _("Philips OGT (SVCD subtitle) packetizer") );
     set_capability( "packetizer", 50 );
@@ -63,27 +73,43 @@ static block_t *Packetize  ( decoder_t *, block_t ** );
 static block_t *Reassemble ( decoder_t *, block_t * );
 static void ParseHeader( decoder_t *, block_t * );
 static subpicture_t *DecodePacket( decoder_t *, block_t * );
-static void RenderImage( decoder_t *, block_t *, subpicture_region_t * );
+static void SVCDSubRenderImage( decoder_t *, block_t *, subpicture_region_t * );
+
+#define DECODE_DBG_CALL        1 /* calls */
+#define DECODE_DBG_PACKET      2 /* packet assembly info */
 
 #define GETINT16(p) ( (p[0] <<  8) +   p[1] )  ; p +=2;
 
 #define GETINT32(p) ( (p[0] << 24) +  (p[1] << 16) +    \
                       (p[2] <<  8) +  (p[3]) ) ; p += 4;
 
-#define SUBTITLE_BLOCK_EMPTY 0
-#define SUBTITLE_BLOCK_PARTIAL 1
-#define SUBTITLE_BLOCK_COMPLETE 2
+typedef enum  {
+  SUBTITLE_BLOCK_EMPTY    = 0,
+  SUBTITLE_BLOCK_PARTIAL  = 1,
+  SUBTITLE_BLOCK_COMPLETE = 2
+} packet_state_t;
+
+#ifndef DECODE_DEBUG
+#define DECODE_DEBUG 1
+#endif
+#if DECODE_DEBUG
+#define dbg_print(mask, s, args...) \
+   if (p_sys && p_sys->i_debug & mask) \
+     msg_Dbg(p_dec, "%s: "s, __func__ , ##args)
+#else
+#define dbg_print(mask, s, args...)
+#endif
 
 struct decoder_sys_t
 {
-  int      b_packetizer;
+  int      i_debug;       /* debugging mask */
 
-  int      i_state;    /* data-gathering state for this subtitle */
+  packet_state_t i_state; /* data-gathering state for this subtitle */
 
-  block_t  *p_spu;   /* Bytes of the packet. */
+  block_t  *p_spu;        /* Bytes of the packet. */
 
-  uint16_t i_image;  /* image number in the subtitle stream */
-  uint8_t  i_packet; /* packet number for above image number */
+  uint16_t i_image;       /* image number in the subtitle stream */
+  uint8_t  i_packet;      /* packet number for above image number */
 
   int     i_spu_size;     /* goal for subtitle_data_pos while gathering,
                              size of used subtitle_data later */
@@ -118,9 +144,9 @@ static int DecoderOpen( vlc_object_t *p_this )
         return VLC_EGENERIC;
     }
 
-    p_dec->p_sys = p_sys = malloc( sizeof( decoder_sys_t ) );
+    p_dec->p_sys = p_sys = calloc( 1, sizeof( decoder_sys_t ) );
+    p_sys->i_debug       = config_GetInt( p_this, MODULE_STRING "-debug" );
 
-    p_sys->b_packetizer  = VLC_FALSE;
     p_sys->i_image       = -1;
 
     p_sys->i_state = SUBTITLE_BLOCK_EMPTY;
@@ -131,6 +157,7 @@ static int DecoderOpen( vlc_object_t *p_this )
     p_dec->pf_decode_sub = Decode;
     p_dec->pf_packetize  = Packetize;
 
+    dbg_print( (DECODE_DBG_CALL) , "");
     return VLC_SUCCESS;
 }
 
@@ -139,12 +166,8 @@ static int DecoderOpen( vlc_object_t *p_this )
  *****************************************************************************/
 static int PacketizerOpen( vlc_object_t *p_this )
 {
-    decoder_t *p_dec = (decoder_t*)p_this;
-
     if( DecoderOpen( p_this ) != VLC_SUCCESS ) return VLC_EGENERIC;
 
-    p_dec->p_sys->b_packetizer = VLC_TRUE;
-
     return VLC_SUCCESS;
 }
 
@@ -166,6 +189,9 @@ void DecoderClose( vlc_object_t *p_this )
 static subpicture_t *Decode( decoder_t *p_dec, block_t **pp_block )
 {
     block_t *p_block, *p_spu;
+    decoder_sys_t *p_sys = p_dec->p_sys;
+
+    dbg_print( (DECODE_DBG_CALL) , "");
 
     if( pp_block == NULL || *pp_block == NULL ) return NULL;
 
@@ -256,13 +282,14 @@ static block_t *Reassemble( decoder_t *p_dec, block_t *p_block )
         }
 
         vlc_object_release( p_input );
-        msg_Dbg( p_dec, "val.i_int %x p_buffer[i] %x", val.i_int, p_buffer[1]);
+        dbg_print( (DECODE_DBG_PACKET),
+                   "val.i_int %x p_buffer[i] %x", val.i_int, p_buffer[1]);
 
         /* The dummy ES that the menu selection uses has an 0x70 at
            the head which we need to strip off. */
         if( val.i_int == -1 || (val.i_int & 0x03) != p_buffer[1] )
         {
-           msg_Dbg( p_dec, "subtitle not for us." );
+            dbg_print( DECODE_DBG_PACKET, "subtitle not for us.\n");
             return NULL;
         }
     }
@@ -320,11 +347,12 @@ static block_t *Reassemble( decoder_t *p_dec, block_t *p_block )
 
         if( p_spu->i_buffer != p_sys->i_spu_size )
         {
-            msg_Warn( p_dec, "SPU packets size=%d should be %d",
+            msg_Warn( p_dec, "subtitle packets size=%d should be %d",
                       p_spu->i_buffer, p_sys->i_spu_size );
         }
 
-        msg_Dbg( p_dec, "subtitle packet complete, size=%d", p_spu->i_buffer);
+       dbg_print( (DECODE_DBG_PACKET),
+                 "subtitle packet complete, size=%d", p_spu->i_buffer );
 
         p_sys->i_state = SUBTITLE_BLOCK_EMPTY;
         p_sys->p_spu = 0;
@@ -403,25 +431,26 @@ static void ParseHeader( decoder_t *p_dec, block_t *p_block )
     p_sys->i_image_length  = p_sys->i_spu_size - p_sys->i_image_offset;
     p_sys->metadata_length = p_sys->i_image_offset;
 
-#ifdef DEBUG_SVCD
-    msg_Dbg( p_dec, "x-start: %d, y-start: %d, width: %d, height %d, "
-             "spu size: %d, duration: %lu (d:%d p:%d)",
-             p_sys->i_x_start, p_sys->i_y_start, 
-             p_sys->i_width, p_sys->i_height, 
-             p_sys->i_spu_size, (long unsigned int) p_sys->i_duration,
-             p_sys->i_image_length, p_sys->i_image_offset);
-
-    for( i = 0; i < 4; i++ )
-    {
-        msg_Dbg( p_dec, "palette[%d]= T: %2x, Y: %2x, u: %2x, v: %2x", i,
-                 p_sys->p_palette[i][3], p_sys->p_palette[i][0], 
-                 p_sys->p_palette[i][1], p_sys->p_palette[i][2] );
-    }
-#endif
+  if (p_sys && p_sys->i_debug & DECODE_DBG_PACKET) 
+  {
+      msg_Dbg( p_dec, "x-start: %d, y-start: %d, width: %d, height %d, "
+              "spu size: %d, duration: %lu (d:%d p:%d)",
+              p_sys->i_x_start, p_sys->i_y_start, 
+              p_sys->i_width, p_sys->i_height, 
+              p_sys->i_spu_size, (long unsigned int) p_sys->i_duration,
+              p_sys->i_image_length, p_sys->i_image_offset);
+      
+      for( i = 0; i < 4; i++ )
+      {
+          msg_Dbg( p_dec, "palette[%d]= T: %2x, Y: %2x, u: %2x, v: %2x", i,
+                  p_sys->p_palette[i][3], p_sys->p_palette[i][0], 
+                  p_sys->p_palette[i][1], p_sys->p_palette[i][2] );
+      }
+  }
 }
 
 /*****************************************************************************
- * DecodePacket: parse and decode an SPU packet
+ * DecodePacket: parse and decode an subtitle packet
  *****************************************************************************
  * This function parses and decodes an SPU packet and, if valid, returns a
  * subpicture.
@@ -439,26 +468,37 @@ static subpicture_t *DecodePacket( decoder_t *p_dec, block_t *p_data )
     if( !p_spu ) return NULL;
 
     p_spu->i_x = p_sys->i_x_start;
-    p_spu->i_x = p_spu->i_x * 3 / 4; /* FIXME: use aspect ratio for x? */
     p_spu->i_y = p_sys->i_y_start;
     p_spu->i_start = p_data->i_pts;
     p_spu->i_stop  = p_data->i_pts + p_sys->i_duration;
     p_spu->b_ephemer = VLC_TRUE;
 
-    /* Create new SPU region */
+    /* Create new subtitle region */
     memset( &fmt, 0, sizeof(video_format_t) );
     fmt.i_chroma = VLC_FOURCC('Y','U','V','P');
+
+    /**
+       The video on which the subtitle sits, is scaled, probably
+       4:3. However subtitle bitmaps assume an 1:1 aspect ratio.
+
+       FIXME: We should get the video aspect ratio from somewhere.
+       Two candidates are the video and the other possibility would be
+       the access module.
+    */
     fmt.i_aspect = VOUT_ASPECT_FACTOR;
+
     fmt.i_width = fmt.i_visible_width = p_sys->i_width;
     fmt.i_height = fmt.i_visible_height = p_sys->i_height;
     fmt.i_x_offset = fmt.i_y_offset = 0;
     p_region = p_spu->pf_create_region( VLC_OBJECT(p_dec), &fmt );
     if( !p_region )
     {
-        msg_Err( p_dec, "cannot allocate SPU region" );
+        msg_Err( p_dec, "cannot allocate SVCD subtitle region" );
         //goto error;
     }
 
+    p_region->fmt.i_aspect = VOUT_ASPECT_FACTOR;
+    
     p_spu->p_region = p_region;
     p_region->i_x = p_region->i_y = 0;
 
@@ -472,15 +512,14 @@ static subpicture_t *DecodePacket( decoder_t *p_dec, block_t *p_data )
         fmt.p_palette->palette[i][3] = p_sys->p_palette[i][3];
     }
 
-    RenderImage( p_dec, p_data, p_region );
+    SVCDSubRenderImage( p_dec, p_data, p_region );
 
     return p_spu;
 }
 
 /*****************************************************************************
- * ParseImage: parse the image part of the subtitle
+ * SVCDSubRenderImage: reorders bytes of image data in subpicture region.
  *****************************************************************************
- This part parses the subtitle graphical data and renders it.
 
  The image is encoded using two bits per pixel that select a palette
  entry except that value 0 starts a limited run-length encoding for
@@ -496,8 +535,8 @@ static subpicture_t *DecodePacket( decoder_t *p_dec, block_t *p_data )
  However we'll transform this so that that the RLE is expanded and
  interlacing will also be removed.
  *****************************************************************************/
-static void RenderImage( decoder_t *p_dec, block_t *p_data,
-                         subpicture_region_t *p_region )
+static void SVCDSubRenderImage( decoder_t *p_dec, block_t *p_data,
+                               subpicture_region_t *p_region )
 {
     decoder_sys_t *p_sys = p_dec->p_sys;
     uint8_t *p_dest = p_region->picture.Y_PIXELS;
index 8ed320e1c37ebcdf8903c8619aaa130d1c95bd87..137f9e9df0e242fd20bed134b326137298d47f60 100644 (file)
@@ -474,6 +474,8 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
                             int i_scale_width_orig, int i_scale_height_orig )
 {
 
+    int i_subpic_x;
+  
     /* Get lock */
     vlc_mutex_lock( &p_spu->subpicture_lock );
 
@@ -526,6 +528,12 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
                     p_fmt->i_height;
         }
 
+       if (p_subpic->p_region->fmt.i_aspect == 0)
+         /* Set subtitle to be the same aspect ratio as the background
+            source video.
+          */
+         p_subpic->p_region->fmt.i_aspect = p_pic_src->format.i_aspect;
+
         i_scale_width = i_scale_width_orig;
         i_scale_height = i_scale_height_orig;
 
@@ -539,6 +547,16 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
         }
 
         /* Take care of the aspect ratio */
+
+       if( p_subpic->p_region->fmt.i_aspect != p_pic_src->format.i_aspect)
+       {
+         i_scale_width = (p_subpic->p_region->fmt.i_aspect*1000) / 
+           p_pic_src->format.i_aspect;
+         i_subpic_x = p_subpic->i_x * i_scale_width / 1000;
+       } else 
+         i_subpic_x = p_subpic->i_x;
+
+
         if( p_region && p_region->fmt.i_sar_num * p_fmt->i_sar_den !=
             p_region->fmt.i_sar_den * p_fmt->i_sar_num )
         {
@@ -547,7 +565,6 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
                 p_region->fmt.i_sar_den / p_fmt->i_sar_num;
         }
 
-        /* Load the scaling module */
         if( !p_spu->p_scale && (i_scale_width != 1000 ||
             i_scale_height != 1000) )
         {
@@ -581,7 +598,7 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
                     p_spu->p_blend->pf_video_blend )
         {
             int i_fade_alpha = 255;
-            int i_x_offset = p_region->i_x + p_subpic->i_x;
+            int i_x_offset = p_region->i_x + i_subpic_x;
             int i_y_offset = p_region->i_y + p_subpic->i_y;
 
             if( p_region->fmt.i_chroma == VLC_FOURCC('T','E','X','T') )
@@ -679,7 +696,7 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
             if( p_subpic->i_flags & SUBPICTURE_ALIGN_RIGHT )
             {
                 i_x_offset = p_fmt->i_width - p_region->fmt.i_width -
-                    p_subpic->i_x;
+                    i_subpic_x;
             }
             else if ( !(p_subpic->i_flags & SUBPICTURE_ALIGN_LEFT) )
             {
@@ -689,7 +706,7 @@ void spu_RenderSubpictures( spu_t *p_spu, video_format_t *p_fmt,
             if( p_subpic->b_absolute )
             {
                 i_x_offset = p_region->i_x +
-                    p_subpic->i_x * i_scale_width / 1000;
+                    i_subpic_x * i_scale_width / 1000;
                 i_y_offset = p_region->i_y +
                     p_subpic->i_y * i_scale_height / 1000;