svg.c
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:15k
源码类别:

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * svg.c : Put SVG on the video
  3.  *****************************************************************************
  4.  * Copyright (C) 2002, 2003 VideoLAN
  5.  * $Id: svg.c,v 1.2 2003/07/23 17:26:56 oaubert Exp $
  6.  *
  7.  * Authors: Olivier Aubert <oaubert@lisi.univ-lyon1.fr>
  8.  *
  9.  * This program is free software; you can redistribute it and/or modify
  10.  * it under the terms of the GNU General Public License as published by
  11.  * the Free Software Foundation; either version 2 of the License, or
  12.  * (at your option ) any later version.
  13.  *
  14.  * This program is distributed in the hope that it will be useful,
  15.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  17.  * GNU General Public License for more details.
  18.  *
  19.  * You should have received a copy of the GNU General Public License
  20.  * along with this program; if not, write to the Free Software
  21.  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #include <stdlib.h>                                      /* malloc( ), free( ) */
  27. #include <string.h>
  28. #include <sys/types.h>
  29. #include <sys/stat.h>
  30. #include <unistd.h>
  31. #include <vlc/vlc.h>
  32. #include <vlc/vout.h>
  33. #include <librsvg-2/librsvg/rsvg.h>
  34. /*****************************************************************************
  35.  * Local prototypes
  36.  *****************************************************************************/
  37. static int  Create    ( vlc_object_t * );
  38. static void Destroy   ( vlc_object_t * );
  39. static void Render    ( vout_thread_t *, picture_t *,
  40.                         const subpicture_t * );
  41. static subpicture_t *AddText ( vout_thread_t *p_vout, int i_channel,
  42.                                char *psz_string, text_style_t *p_style, int i_flags,
  43.                                int i_hmargin, int i_vmargin, mtime_t i_start,
  44.                                mtime_t i_stop );
  45. static byte_t *svg_GetTemplate ();
  46. static void svg_SizeCallback  (int *width, int *height, gpointer data );
  47. static void svg_RenderPicture (vout_thread_t *p_vout,
  48.                                subpicture_sys_t *p_string );
  49. static void FreeString( subpicture_t * );
  50. /*****************************************************************************
  51.  * Module descriptor
  52.  *****************************************************************************/
  53. #define TEMPLATE_TEXT N_( "SVG template file" )
  54. #define TEMPLATE_LONGTEXT N_( "Location of a file holding a SVG template for automatic string conversion" )
  55. vlc_module_begin();
  56.  set_capability( "text renderer", 100 );
  57.  add_shortcut( "svg" );
  58.  add_string( "svg-template-file", "", NULL, TEMPLATE_TEXT, TEMPLATE_LONGTEXT, VLC_TRUE );
  59.  set_callbacks( Create, Destroy );
  60. vlc_module_end();
  61. /**
  62.    Describes a SVG string to be displayed on the video
  63. */
  64. struct subpicture_sys_t
  65. {
  66.     int            i_width;
  67.     int            i_height;
  68.     int            i_chroma;
  69.     /** The SVG source associated with this subpicture */
  70.     byte_t        *psz_text;
  71.     /* The rendered SVG, as a GdkPixbuf */
  72.     GdkPixbuf      *p_rendition;
  73. };
  74. /*****************************************************************************
  75.  * vout_sys_t: svg local data
  76.  *****************************************************************************
  77.  * This structure is part of the video output thread descriptor.
  78.  * It describes the svg specific properties of an output thread.
  79.  *****************************************************************************/
  80. struct text_renderer_sys_t
  81. {
  82.     /* The SVG template used to convert strings */
  83.     byte_t        *psz_template;
  84.     vlc_mutex_t   *lock;
  85. };
  86. /*****************************************************************************
  87.  * Create: allocates svg video thread output method
  88.  *****************************************************************************
  89.  * This function allocates and initializes a  vout method.
  90.  *****************************************************************************/
  91. static int Create( vlc_object_t *p_this )
  92. {
  93.     vout_thread_t *p_vout = ( vout_thread_t * )p_this;
  94.     /* Allocate structure */
  95.     p_vout->p_text_renderer_data = malloc( sizeof( text_renderer_sys_t ) );
  96.     if( p_vout->p_text_renderer_data == NULL )
  97.     {
  98.         msg_Err( p_vout, "Out of memory" );
  99.         return VLC_ENOMEM;
  100.     }
  101.     p_vout->pf_add_string = AddText;
  102.     /* Initialize psz_template */
  103.     p_vout->p_text_renderer_data->psz_template = svg_GetTemplate( p_this );
  104.     if( !p_vout->p_text_renderer_data->psz_template )
  105.     {
  106.         msg_Err( p_vout, "Out of memory" );
  107.         return VLC_ENOMEM;
  108.     }
  109.     return VLC_SUCCESS;
  110. }
  111. static byte_t *svg_GetTemplate( vlc_object_t *p_this )
  112. {
  113.     vout_thread_t *p_vout = ( vout_thread_t * )p_this;
  114.     char *psz_filename;
  115.     char *psz_template;
  116.     FILE *file;
  117.     psz_filename = config_GetPsz( p_vout, "svg-template-file" );
  118.     if( !psz_filename || psz_filename[0] == 0 )
  119.     {
  120.         /* No filename. Use a default value. */
  121.         psz_template = NULL;
  122.     }
  123.     else
  124.     {
  125.         /* Read the template */
  126.         file = fopen( psz_filename, "rt" );
  127.         if( !file )
  128.         {
  129.             msg_Warn( p_this, "SVG template file %s does not exist.", psz_filename );
  130.             psz_template = NULL;
  131.         }
  132.         else
  133.         {
  134.             struct stat s;
  135.             int i_ret;
  136.             i_ret = lstat( psz_filename, &s );
  137.             if( i_ret )
  138.             {
  139.                 /* Problem accessing file information. Should not
  140.                    happen as we could open it. */
  141.                 psz_template = NULL;
  142.             }
  143.             else
  144.             {
  145.                 fprintf( stderr, "Reading %ld bytes from %sn", (long)s.st_size, psz_filename );
  146.                 psz_template = malloc( s.st_size + 42 );
  147.                 if( !psz_template )
  148.                 {
  149.                     msg_Err( p_vout, "Out of memory" );
  150.                     return NULL;
  151.                 }
  152.                 fread( psz_template, s.st_size, 1, file );
  153.                 fclose( file );
  154.             }
  155.         }
  156.     }
  157.     if( !psz_template )
  158.     {
  159.         /* Either there was no file, or there was an error.
  160.            Use the default value */
  161.         psz_template = strdup( "<?xml version='1.0' encoding='UTF-8' standalone='no'?> 
  162. <svg version='1' preserveAspectRatio='xMinYMin meet' viewBox='0 0 800 600'> 
  163.   <text x='10' y='560' fill='white' font-size='32'  
  164.         font-family='sans-serif'>%s</text></svg>" );
  165.     }
  166.     return psz_template;
  167. }
  168. /*****************************************************************************
  169.  * Destroy: destroy Clone video thread output method
  170.  *****************************************************************************
  171.  * Clean up all data and library connections
  172.  *****************************************************************************/
  173. static void Destroy( vlc_object_t *p_this )
  174. {
  175.     vout_thread_t *p_vout = ( vout_thread_t * )p_this;
  176.     free( p_vout->p_text_renderer_data->psz_template );
  177.     free( p_vout->p_text_renderer_data );
  178. }
  179. /*****************************************************************************
  180.  * Render: render SVG in picture
  181.  *****************************************************************************
  182.  * This function merges the previously rendered SVG subpicture into a picture
  183.  *****************************************************************************/
  184. static void Render( vout_thread_t *p_vout, picture_t *p_pic,
  185.                     const subpicture_t *p_subpic )
  186. {
  187.     subpicture_sys_t *p_string = p_subpic->p_sys;
  188.     guchar *pixels_in = NULL;
  189.     guchar *pixels_out = NULL;
  190.     int rowstride_in, rowstride_out;
  191.     int channels_in, channels_out;
  192.     int x, y;
  193.     int i_width, i_height;
  194.     int alpha;
  195.     if( p_string->p_rendition == NULL ) {
  196.         /* Something changed ( presumably the dimensions ). Get the new
  197.            dimensions and update the pixbuf */
  198.         p_string->i_width = p_vout->output.i_width;
  199.         p_string->i_height = p_vout->output.i_height;
  200.         svg_RenderPicture( p_vout, p_string );
  201.     }
  202.     /* This rendering code is in no way optimized. If someone has some
  203.        time to lose to make it work faster, please do.
  204.     */
  205.     /* FIXME: The alpha value is not taken into account. */
  206.     /*
  207.       p_pixbuf->get_rowstride() is the number of bytes in a line.
  208.       p_pixbuf->get_height() is the number of lines.
  209.       The number of bytes of p_pixbuf->p_pixels is get_rowstride * get_height
  210.       if( has_alpha() ) {
  211.       alpha = pixels [ n_channels * ( y*rowstride + x ) + 3 ];
  212.       }
  213.       red   = pixels [ n_channels * ( y*rowstride ) + x ) ];
  214.       green = pixels [ n_channels * ( y*rowstride ) + x ) + 1 ];
  215.       blue  = pixels [ n_channels * ( y*rowstride ) + x ) + 2 ];
  216.     */
  217.     pixels_in = gdk_pixbuf_get_pixels( p_string->p_rendition );
  218.     pixels_out = p_pic->p->p_pixels;
  219.     rowstride_in = gdk_pixbuf_get_rowstride( p_string->p_rendition );
  220.     rowstride_out = p_pic->p->i_pitch;
  221.     channels_in = gdk_pixbuf_get_n_channels( p_string->p_rendition );
  222.     channels_out = p_pic->p->i_pixel_pitch;
  223.     alpha = gdk_pixbuf_get_has_alpha( p_string->p_rendition );
  224. #define INDEX_IN( x, y ) ( y * rowstride_in + x * channels_in )
  225. #define INDEX_OUT( x, y ) ( y * rowstride_out + x * channels_out )
  226. #define UV_INDEX_OUT( x, y ) ( y * p_pic->p[U_PLANE].i_pitch / 2 + x * p_pic->p[U_PLANE].i_pixel_pitch / 2 )
  227.     i_width = gdk_pixbuf_get_width( p_string->p_rendition );
  228.     i_height = gdk_pixbuf_get_height( p_string->p_rendition );
  229.     switch( p_vout->output.i_chroma )
  230.     {
  231.         /* I420 target, no scaling */
  232.     case VLC_FOURCC( 'I','4','2','0' ):
  233.     case VLC_FOURCC( 'I','Y','U','V' ):
  234.     case VLC_FOURCC( 'Y','V','1','2' ):
  235.         for( y = 0; y < i_height; y++ )
  236.         {
  237.             for( x = 0; x < i_width; x++ )
  238.             {
  239.                 guchar *p_in;
  240.                 int i_out;
  241.                 int i_uv_out;
  242.                 p_in = &pixels_in[INDEX_IN( x, y )];
  243. #define R( pixel ) *pixel
  244. #define G( pixel ) *( pixel+1 )
  245. #define B( pixel ) *( pixel+2 )
  246. #define ALPHA( pixel ) *( pixel+3 )
  247.                 /* From http://www.geocrawler.com/archives/3/8263/2001/6/0/6020594/ :
  248.                    Y = 0.29900 * R + 0.58700 * G + 0.11400 * B
  249.                    U = -0.1687 * r  - 0.3313 * g + 0.5 * b + 128
  250.                    V = 0.5   * r - 0.4187 * g - 0.0813 * b + 128
  251.                 */
  252.                 if( (alpha && ALPHA( p_in ) > 10 ) || ( ! alpha )) {
  253.                     i_out = INDEX_OUT( x, y );
  254.                     p_pic->p[Y_PLANE].p_pixels[i_out] = .299 * R( p_in ) + .587 * G( p_in ) + .114 * B( p_in );
  255.                     if( ( x % 2 == 0 ) && ( y % 2 == 0 ) ) {
  256.                         i_uv_out = UV_INDEX_OUT( x, y );
  257.                         p_pic->p[U_PLANE].p_pixels[i_uv_out] = -.1687 * R( p_in ) - .3313 * G( p_in ) + .5 * B( p_in ) + 128;
  258.                         p_pic->p[V_PLANE].p_pixels[i_uv_out] = .5 * R( p_in ) - .4187 * G( p_in ) - .0813 * B( p_in ) + 128;
  259.                     }
  260.                 }
  261.             }
  262.         }
  263.         break;
  264.         /* RV32 target, scaling */
  265.     case VLC_FOURCC( 'R','V','2','4' ):
  266.     case VLC_FOURCC( 'R','V','3','2' ):
  267.         for( y = 0; y < i_height; y++ )
  268.         {
  269.             for( x = 0; x < i_width; x++ )
  270.             {
  271.                 guchar *p_in;
  272.                 guchar *p_out;
  273.                 p_in = &pixels_in[INDEX_IN( x, y )];
  274.                 p_out = &pixels_out[INDEX_OUT( x, y )];
  275.                 *p_out = *p_in;
  276.                 *( p_out+1 ) = *( p_in+1 );
  277.                 *( p_out+2 ) = *( p_in+2 );
  278.             }
  279.         }
  280.         break;
  281.     default:
  282.         msg_Err( p_vout, "unknown chroma, can't render SVG" );
  283.         break;
  284.     }
  285. }
  286. static void svg_SizeCallback( int *width, int *height, gpointer data )
  287. {
  288.     subpicture_sys_t *p_string = data;
  289.     *width = p_string->i_width;
  290.     *height = p_string->i_height;
  291.     return;
  292. }
  293. static void svg_RenderPicture( vout_thread_t *p_vout,
  294.                                subpicture_sys_t *p_string )
  295. {
  296.     /* Render the SVG string p_string->psz_text into a new picture_t
  297.        p_string->p_rendition with dimensions ( ->i_width, ->i_height ) */
  298.     RsvgHandle *p_handle;
  299.     GError *error;
  300.     p_handle = rsvg_handle_new();
  301.     rsvg_handle_set_size_callback( p_handle, svg_SizeCallback, p_string, NULL );
  302.     rsvg_handle_write( p_handle,
  303.                        p_string->psz_text, strlen( p_string->psz_text ) + 1,
  304.                        &error );
  305.     rsvg_handle_close( p_handle, &error );
  306.     p_string->p_rendition = rsvg_handle_get_pixbuf( p_handle );
  307.     rsvg_handle_free( p_handle );
  308. }
  309. /**
  310.  * This function receives a SVG string and creates a subpicture for it.
  311.  * It is used as pf_add_string callback in the vout method by this module.
  312.  */
  313. static subpicture_t *AddText ( vout_thread_t *p_vout, int i_channel,
  314.                                char *psz_string, text_style_t *p_style, int i_flags,
  315.                                int i_hmargin, int i_vmargin, mtime_t i_start,
  316.                                mtime_t i_stop )
  317. {
  318.     subpicture_sys_t *p_string;
  319.     subpicture_t *p_subpic;
  320.     msg_Dbg( p_vout, "adding string "%s" start_date "I64Fd
  321.              " end_date" I64Fd, psz_string, i_start, i_stop );
  322.     /* Create and initialize a subpicture */
  323.     p_subpic = vout_CreateSubPicture( p_vout, i_channel, GRAPH_CONTENT,
  324.                                       MEMORY_SUBPICTURE );
  325.     if( p_subpic == NULL )
  326.     {
  327.         return NULL;
  328.     }
  329.     p_subpic->pf_render = Render;
  330.     p_subpic->pf_destroy = FreeString;
  331.     p_subpic->i_start = i_start;
  332.     p_subpic->i_stop = i_stop;
  333.     if( i_stop == 0 )
  334.     {
  335.         p_subpic->b_ephemer = VLC_TRUE;
  336.     }
  337.     else
  338.     {
  339.         p_subpic->b_ephemer = VLC_FALSE;
  340.     }
  341.     /* Create and initialize private data for the subpicture */
  342.     p_string = malloc( sizeof( subpicture_sys_t ) );
  343.     if( p_string == NULL )
  344.     {
  345.         vout_DestroySubPicture( p_vout, p_subpic );
  346.         return NULL;
  347.     }
  348.     p_subpic->p_sys = p_string;
  349.     /* Check if the data is SVG or pure text. In the latter case,
  350.        convert the text to SVG. FIXME: find a better test */
  351.     if( strstr( psz_string, "<svg" ))
  352.     {
  353.         /* Data is SVG: duplicate */
  354.         p_string->psz_text = strdup( psz_string );
  355.     }
  356.     else
  357.     {
  358.         /* Data is text. Convert to SVG */
  359.         int length;
  360.         byte_t* psz_template = p_vout->p_text_renderer_data->psz_template;
  361.         length = strlen( psz_string ) + strlen( psz_template ) + 42;
  362.         p_string->psz_text = malloc( length + 1 );
  363.         if( p_string->psz_text == NULL )
  364.         {
  365.             return NULL;
  366.         }
  367.         snprintf( p_string->psz_text, length, psz_template, psz_string );
  368.     }
  369.     p_string->i_width = p_vout->output.i_width;
  370.     p_string->i_height = p_vout->output.i_height;
  371.     p_string->i_chroma = p_vout->output.i_chroma;
  372.     /* Render the SVG.
  373.        The input data is stored in the p_string structure,
  374.        and the function updates the p_rendition attribute. */
  375.     svg_RenderPicture( p_vout, p_string );
  376.     vout_DisplaySubPicture( p_vout, p_subpic );
  377.     return p_subpic;
  378. }
  379. static void FreeString( subpicture_t *p_subpic )
  380. {
  381.     subpicture_sys_t *p_string = p_subpic->p_sys;
  382.     free( p_string->psz_text );
  383.     free( p_string->p_rendition );
  384.     free( p_string );
  385. }