]> git.sesse.net Git - vlc/blob - modules/misc/svg.c
MKV USF subtitles support and other Subtitles improvements.
[vlc] / modules / misc / svg.c
1 /*****************************************************************************
2  * svg.c : Put SVG on the video
3  *****************************************************************************
4  * Copyright (C) 2002, 2003 the VideoLAN team
5  * $Id$
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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27 #include <stdlib.h>                                      /* malloc( ), free( ) */
28 #include <string.h>
29
30 #include <vlc/vlc.h>
31 #include <vlc_charset.h>
32 #include <vlc_vout.h>
33 #include <vlc_osd.h>
34 #include <vlc_block.h>
35 #include <vlc_filter.h>
36
37 #ifdef HAVE_SYS_TYPES_H
38 #   include <sys/types.h>
39 #endif
40
41 #ifdef HAVE_UNISTD_H
42 #    include <unistd.h>
43 #elif defined( WIN32 ) && !defined( UNDER_CE )
44 #   include <io.h>
45 #endif
46
47 #include <glib.h>
48 #include <glib/gstdio.h>
49 #include <glib-object.h>                                  /* g_object_unref( ) */
50 #include <librsvg-2/librsvg/rsvg.h>
51
52 typedef struct svg_rendition_t svg_rendition_t;
53
54 /*****************************************************************************
55  * Local prototypes
56  *****************************************************************************/
57 static int  Create    ( vlc_object_t * );
58 static void Destroy   ( vlc_object_t * );
59 static int  RenderText( filter_t *p_filter, subpicture_region_t *p_region_out,
60                         subpicture_region_t *p_region_in );
61 static char *svg_GetTemplate( vlc_object_t *p_this );
62
63 /*****************************************************************************
64  * Module descriptor
65  *****************************************************************************/
66
67 #define TEMPLATE_TEXT N_( "SVG template file" )
68 #define TEMPLATE_LONGTEXT N_( "Location of a file holding a SVG template "\
69         "for automatic string conversion" )
70
71 vlc_module_begin();
72  set_category( CAT_INPUT);
73  set_category( SUBCAT_INPUT_SCODEC );
74  set_capability( "text renderer", 99 );
75  add_shortcut( "svg" );
76  add_string( "svg-template-file", "", NULL, TEMPLATE_TEXT, TEMPLATE_LONGTEXT, VLC_TRUE );
77  set_callbacks( Create, Destroy );
78 vlc_module_end();
79
80 /**
81    Describes a SVG string to be displayed on the video
82 */
83 struct svg_rendition_t
84 {
85     int            i_width;
86     int            i_height;
87     int            i_chroma;
88     /** The SVG source associated with this subpicture */
89     char           *psz_text;
90     /* The rendered SVG, as a GdkPixbuf */
91     GdkPixbuf      *p_rendition;
92 };
93
94 static int Render( filter_t *, subpicture_region_t *, svg_rendition_t *, int, int);
95 static char *svg_GetTemplate ();
96 static void svg_set_size( filter_t *p_filter, int width, int height );
97 static void svg_SizeCallback  ( int *width, int *height, gpointer data );
98 static void svg_RenderPicture ( filter_t *p_filter,
99                                 svg_rendition_t *p_svg );
100 static void FreeString( svg_rendition_t * );
101
102 /*****************************************************************************
103  * filter_sys_t: svg local data
104  *****************************************************************************
105  * This structure is part of the filter thread descriptor.
106  * It describes the svg specific properties of an output thread.
107  *****************************************************************************/
108 struct filter_sys_t
109 {
110     /* The SVG template used to convert strings */
111     char          *psz_template;
112     /* Default size for rendering. Initialized to the output size. */
113     int            i_width;
114     int            i_height;
115     vlc_mutex_t   *lock;
116 };
117
118 /*****************************************************************************
119  * Create: allocates svg video thread output method
120  *****************************************************************************
121  * This function allocates and initializes a  vout method.
122  *****************************************************************************/
123 static int Create( vlc_object_t *p_this )
124 {
125     filter_t *p_filter = ( filter_t * )p_this;
126     filter_sys_t *p_sys;
127
128     /* Allocate structure */
129     p_sys = malloc( sizeof( filter_sys_t ) );
130     if( !p_sys )
131     {
132         msg_Err( p_filter, "out of memory" );
133         return VLC_ENOMEM;
134     }
135
136     /* Initialize psz_template */
137     p_sys->psz_template = svg_GetTemplate( p_this );
138     if( !p_sys->psz_template )
139     {
140         msg_Err( p_filter, "out of memory" );
141         return VLC_ENOMEM;
142     }
143
144     p_sys->i_width = p_filter->fmt_out.video.i_width;
145     p_sys->i_height = p_filter->fmt_out.video.i_height;
146
147     p_filter->pf_render_text = RenderText;
148     p_filter->pf_render_html = NULL;
149     p_filter->p_sys = p_sys;
150
151     /* MUST call this before any RSVG funcs */
152     rsvg_init( );
153
154     return VLC_SUCCESS;
155 }
156
157 static char *svg_GetTemplate( vlc_object_t *p_this )
158 {
159     filter_t *p_filter = ( filter_t * )p_this;
160     char *psz_filename;
161     char *psz_template;
162     FILE *file;
163
164     psz_filename = config_GetPsz( p_filter, "svg-template-file" );
165     if( !psz_filename || psz_filename[0] == 0 )
166     {
167         /* No filename. Use a default value. */
168         psz_template = NULL;
169     }
170     else
171     {
172         /* Read the template */
173         file = utf8_fopen( psz_filename, "rt" );
174         if( !file )
175         {
176             msg_Warn( p_this, "SVG template file %s does not exist.",
177                                          psz_filename );
178             psz_template = NULL;
179         }
180         else
181         {
182             struct stat s;
183             int i_ret;
184
185             i_ret = utf8_stat( psz_filename, &s );
186             if( i_ret )
187             {
188                 /* Problem accessing file information. Should not
189                    happen as we could open it. */
190                 psz_template = NULL;
191             }
192             else
193             if( ((signed)s.st_size) < 0 )
194             {
195                 msg_Err( p_this, "SVG template too big" );
196                 psz_template = NULL;
197             }
198             else
199             {
200                 msg_Dbg( p_this, "reading %ld bytes from template %s",
201                          (unsigned long)s.st_size, psz_filename );
202
203                 psz_template = malloc( ( s.st_size + 42 ) * sizeof( char ) );
204                 if( !psz_template )
205                 {
206                     msg_Err( p_filter, "out of memory" );
207                     return NULL;
208                 }
209                 memset( psz_template, 0, s.st_size + 1 );
210                 fread( psz_template, s.st_size, 1, file );
211                 fclose( file );
212             }
213         }
214     }
215     if( !psz_template )
216     {
217         /* Either there was no file, or there was an error.
218            Use the default value */
219         psz_template = strdup( "<?xml version='1.0' encoding='UTF-8' standalone='no'?> \
220 <svg version='1' preserveAspectRatio='xMinYMin meet' viewBox='0 0 800 600'> \
221   <text x='10' y='560' fill='white' font-size='32'  \
222         font-family='sans-serif'>%s</text></svg>" );
223     }
224
225     return psz_template;
226 }
227
228 /*****************************************************************************
229  * Destroy: destroy Clone video thread output method
230  *****************************************************************************
231  * Clean up all data and library connections
232  *****************************************************************************/
233 static void Destroy( vlc_object_t *p_this )
234 {
235     filter_t *p_filter = ( filter_t * )p_this;
236     filter_sys_t *p_sys = p_filter->p_sys;
237
238     free( p_sys->psz_template );
239     free( p_sys );
240     rsvg_term( );
241 }
242
243 /*****************************************************************************
244  * Render: render SVG in picture
245  *****************************************************************************/
246 static int Render( filter_t *p_filter, subpicture_region_t *p_region,
247                    svg_rendition_t *p_svg, int i_width, int i_height )
248 {
249     video_format_t fmt;
250     uint8_t *p_y, *p_u, *p_v, *p_a;
251     int x, y, i_pitch, i_u_pitch;
252     guchar *pixels_in = NULL;
253     int rowstride_in;
254     int channels_in;
255     int alpha;
256     picture_t *p_pic;
257     subpicture_region_t *p_region_tmp;
258
259     if ( p_filter->p_sys->i_width != i_width ||
260          p_filter->p_sys->i_height != i_height )
261     {
262         svg_set_size( p_filter, i_width, i_height );
263         p_svg->p_rendition = NULL;
264     }
265
266     if( p_svg->p_rendition == NULL ) {
267         svg_RenderPicture( p_filter, p_svg );
268         if( ! p_svg->p_rendition )
269         {
270             msg_Err( p_filter, "Cannot render SVG" );
271             return VLC_EGENERIC;
272         }
273     }
274     i_width = gdk_pixbuf_get_width( p_svg->p_rendition );
275     i_height = gdk_pixbuf_get_height( p_svg->p_rendition );
276
277     /* Create a new subpicture region */
278     memset( &fmt, 0, sizeof( video_format_t ) );
279     fmt.i_chroma = VLC_FOURCC( 'Y','U','V','A' );
280     fmt.i_aspect = VOUT_ASPECT_FACTOR;
281     fmt.i_width = fmt.i_visible_width = i_width;
282     fmt.i_height = fmt.i_visible_height = i_height;
283     fmt.i_x_offset = fmt.i_y_offset = 0;
284     p_region_tmp = spu_CreateRegion( p_filter, &fmt );
285     if( !p_region_tmp )
286     {
287         msg_Err( p_filter, "cannot allocate SPU region" );
288         return VLC_EGENERIC;
289     }
290     p_region->fmt = p_region_tmp->fmt;
291     p_region->picture = p_region_tmp->picture;
292     free( p_region_tmp );
293
294     p_region->i_x = p_region->i_y = 0;
295     p_y = p_region->picture.Y_PIXELS;
296     p_u = p_region->picture.U_PIXELS;
297     p_v = p_region->picture.V_PIXELS;
298     p_a = p_region->picture.A_PIXELS;
299
300     i_pitch = p_region->picture.Y_PITCH;
301     i_u_pitch = p_region->picture.U_PITCH;
302
303     /* Initialize the region pixels (only the alpha will be changed later) */
304     memset( p_y, 0x00, i_pitch * p_region->fmt.i_height );
305     memset( p_u, 0x80, i_u_pitch * p_region->fmt.i_height );
306     memset( p_v, 0x80, i_u_pitch * p_region->fmt.i_height );
307
308     p_pic = &p_region->picture;
309
310     /* Copy the data */
311
312     /* This rendering code is in no way optimized. If someone has some time to
313        make it work faster or better, please do.
314     */
315
316     /*
317       p_pixbuf->get_rowstride() is the number of bytes in a line.
318       p_pixbuf->get_height() is the number of lines.
319
320       The number of bytes of p_pixbuf->p_pixels is get_rowstride * get_height
321
322       if( has_alpha() ) {
323       alpha = pixels [ n_channels * ( y*rowstride + x ) + 3 ];
324       }
325       red   = pixels [ n_channels * ( y*rowstride ) + x ) ];
326       green = pixels [ n_channels * ( y*rowstride ) + x ) + 1 ];
327       blue  = pixels [ n_channels * ( y*rowstride ) + x ) + 2 ];
328     */
329
330     pixels_in = gdk_pixbuf_get_pixels( p_svg->p_rendition );
331     rowstride_in = gdk_pixbuf_get_rowstride( p_svg->p_rendition );
332     channels_in = gdk_pixbuf_get_n_channels( p_svg->p_rendition );
333     alpha = gdk_pixbuf_get_has_alpha( p_svg->p_rendition );
334
335     /*
336       This crashes the plugin (if !alpha). As there is always an alpha value,
337       it does not matter for the moment :
338
339     if( !alpha )
340       memset( p_a, 0xFF, i_pitch * p_region->fmt.i_height );
341     */
342
343 #define INDEX_IN( x, y ) ( y * rowstride_in + x * channels_in )
344 #define INDEX_OUT( x, y ) ( y * i_pitch + x * p_pic->p[Y_PLANE].i_pixel_pitch )
345
346     for( y = 0; y < i_height; y++ )
347     {
348         for( x = 0; x < i_width; x++ )
349         {
350             guchar *p_in;
351             int i_out;
352
353             p_in = &pixels_in[INDEX_IN( x, y )];
354
355 #define R( pixel ) *pixel
356 #define G( pixel ) *( pixel+1 )
357 #define B( pixel ) *( pixel+2 )
358 #define ALPHA( pixel ) *( pixel+3 )
359
360             /* From http://www.geocrawler.com/archives/3/8263/2001/6/0/6020594/ :
361                Y = 0.29900 * R + 0.58700 * G + 0.11400 * B
362                U = -0.1687 * r  - 0.3313 * g + 0.5 * b + 128
363                V = 0.5   * r - 0.4187 * g - 0.0813 * b + 128
364             */
365             if ( alpha ) {
366                 i_out = INDEX_OUT( x, y );
367
368                 p_pic->Y_PIXELS[i_out] = .299 * R( p_in ) + .587 * G( p_in ) + .114 * B( p_in );
369
370                 p_pic->U_PIXELS[i_out] = -.1687 * R( p_in ) - .3313 * G( p_in ) + .5 * B( p_in ) + 128;
371                 p_pic->V_PIXELS[i_out] = .5 * R( p_in ) - .4187 * G( p_in ) - .0813 * B( p_in ) + 128;
372
373                 p_pic->A_PIXELS[i_out] = ALPHA( p_in );
374             }
375         }
376     }
377
378     return VLC_SUCCESS;
379 }
380
381 static void svg_set_size( filter_t *p_filter, int width, int height )
382 {
383   p_filter->p_sys->i_width = width;
384   p_filter->p_sys->i_height = height;
385 }
386
387 static void svg_SizeCallback( int *width, int *height, gpointer data )
388 {
389     filter_t *p_filter = data;
390
391     *width = p_filter->p_sys->i_width;
392     *height = p_filter->p_sys->i_height;
393     return;
394 }
395
396 static void svg_RenderPicture( filter_t *p_filter,
397                                svg_rendition_t *p_svg )
398 {
399     /* Render the SVG string p_string->psz_text into a new picture_t
400        p_string->p_rendition with dimensions ( ->i_width, ->i_height ) */
401     RsvgHandle *p_handle;
402     GError *error = NULL;
403
404     p_svg->p_rendition = NULL;
405
406     p_handle = rsvg_handle_new();
407
408     if( !p_handle )
409     {
410         msg_Err( p_filter, "Error creating SVG reader: %s", error->message );
411         return;
412     }
413
414     rsvg_handle_set_size_callback( p_handle, svg_SizeCallback, p_filter, NULL );
415
416     if( ! rsvg_handle_write( p_handle,
417                              ( guchar* )p_svg->psz_text, strlen( p_svg->psz_text ),
418                              &error ) ) 
419     {
420         msg_Err( p_filter, "error while rendering SVG: %s\n", error->message );
421         g_object_unref( G_OBJECT( p_handle ) );
422         return;
423     }
424
425     if( ! rsvg_handle_close( p_handle, &error ) )
426     {
427         msg_Err( p_filter, "error while rendering SVG (close): %s\n", error->message );
428         g_object_unref( G_OBJECT( p_handle ) );
429         return;
430     }
431
432     p_svg->p_rendition = rsvg_handle_get_pixbuf( p_handle );
433
434     g_object_unref( G_OBJECT( p_handle ) );
435 }
436
437
438 static int RenderText( filter_t *p_filter, subpicture_region_t *p_region_out,
439                        subpicture_region_t *p_region_in )
440 {
441     filter_sys_t *p_sys = p_filter->p_sys;
442     svg_rendition_t *p_svg = NULL;
443     char *psz_string;
444
445     /* Sanity check */
446     if( !p_region_in || !p_region_out ) return VLC_EGENERIC;
447     psz_string = p_region_in->psz_text;
448     if( !psz_string || !*psz_string ) return VLC_EGENERIC;
449
450     p_svg = ( svg_rendition_t * )malloc( sizeof( svg_rendition_t ) );
451     if( !p_svg )
452     {
453         msg_Err( p_filter, "out of memory" );
454         return VLC_ENOMEM;
455     }
456
457     p_region_out->i_x = p_region_in->i_x;
458     p_region_out->i_y = p_region_in->i_y;
459
460     /* Check if the data is SVG or pure text. In the latter case,
461        convert the text to SVG. FIXME: find a better test */
462     if( strstr( psz_string, "<svg" ))
463     {
464         /* Data is SVG: duplicate */
465         p_svg->psz_text = strdup( psz_string );
466         if( !p_svg->psz_text )
467         {
468             msg_Err( p_filter, "out of memory" );
469             free( p_svg );
470             return VLC_ENOMEM;
471         }
472     }
473     else
474     {
475         /* Data is text. Convert to SVG */
476         /* FIXME: handle p_style attributes */
477         int length;
478         char* psz_template = p_sys->psz_template;
479         length = strlen( psz_string ) + strlen( psz_template ) + 42;
480         p_svg->psz_text = malloc( ( length + 1 ) * sizeof( char ) );
481         if( !p_svg->psz_text )
482         {
483             msg_Err( p_filter, "out of memory" );
484             free( p_svg );
485             return VLC_ENOMEM;
486         }
487         memset( p_svg->psz_text, 0, length + 1 );
488         snprintf( p_svg->psz_text, length, psz_template, psz_string );
489     }
490     p_svg->i_width = p_sys->i_width;
491     p_svg->i_height = p_sys->i_height;
492     p_svg->i_chroma = VLC_FOURCC( 'Y','U','V','A' );
493
494     /* Render the SVG.
495        The input data is stored in the p_string structure,
496        and the function updates the p_rendition attribute. */
497     svg_RenderPicture( p_filter, p_svg );
498
499     Render( p_filter, p_region_out, p_svg, p_svg->i_width, p_svg->i_height );
500     FreeString( p_svg );
501
502     return VLC_SUCCESS;
503 }
504
505 static void FreeString( svg_rendition_t *p_svg )
506 {
507     if( p_svg->psz_text )
508         free( p_svg->psz_text );
509     /* p_svg->p_rendition is a GdkPixbuf, and its allocation is
510        managed through ref. counting */
511     if( p_svg->p_rendition )
512         g_object_unref( p_svg->p_rendition );
513     free( p_svg );
514 }