]> git.sesse.net Git - vlc/blob - plugins/filter/invert.c
* ALL: the first libvlc commit.
[vlc] / plugins / filter / invert.c
1 /*****************************************************************************
2  * invert.c : Invert video plugin for vlc
3  *****************************************************************************
4  * Copyright (C) 2000, 2001 VideoLAN
5  * $Id: invert.c,v 1.12 2002/06/01 12:31:59 sam Exp $
6  *
7  * Authors: Samuel Hocevar <sam@zoy.org>
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 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27 #include <errno.h>
28 #include <stdlib.h>                                      /* malloc(), free() */
29 #include <string.h>
30
31 #include <vlc/vlc.h>
32 #include <vlc/vout.h>
33
34 #include "filter_common.h"
35
36 /*****************************************************************************
37  * Capabilities defined in the other files.
38  *****************************************************************************/
39 static void vout_getfunctions( function_list_t * p_function_list );
40
41 /*****************************************************************************
42  * Build configuration tree.
43  *****************************************************************************/
44 MODULE_CONFIG_START
45 MODULE_CONFIG_STOP
46
47 MODULE_INIT_START
48     SET_DESCRIPTION( _("invert video module") )
49     /* Capability score set to 0 because we don't want to be spawned
50      * as a video output unless explicitly requested to */
51     ADD_CAPABILITY( VOUT, 0 )
52     ADD_SHORTCUT( "invert" )
53 MODULE_INIT_STOP
54
55 MODULE_ACTIVATE_START
56     vout_getfunctions( &p_module->p_functions->vout );
57 MODULE_ACTIVATE_STOP
58
59 MODULE_DEACTIVATE_START
60 MODULE_DEACTIVATE_STOP
61
62 /*****************************************************************************
63  * vout_sys_t: Invert video output method descriptor
64  *****************************************************************************
65  * This structure is part of the video output thread descriptor.
66  * It describes the Invert specific properties of an output thread.
67  *****************************************************************************/
68 struct vout_sys_s
69 {
70     vout_thread_t *p_vout;
71 };
72
73 /*****************************************************************************
74  * Local prototypes
75  *****************************************************************************/
76 static int  vout_Create    ( vout_thread_t * );
77 static int  vout_Init      ( vout_thread_t * );
78 static void vout_End       ( vout_thread_t * );
79 static void vout_Destroy   ( vout_thread_t * );
80 static int  vout_Manage    ( vout_thread_t * );
81 static void vout_Render    ( vout_thread_t *, picture_t * );
82 static void vout_Display   ( vout_thread_t *, picture_t * );
83
84 /*****************************************************************************
85  * Functions exported as capabilities. They are declared as static so that
86  * we don't pollute the namespace too much.
87  *****************************************************************************/
88 static void vout_getfunctions( function_list_t * p_function_list )
89 {
90     p_function_list->functions.vout.pf_create     = vout_Create;
91     p_function_list->functions.vout.pf_init       = vout_Init;
92     p_function_list->functions.vout.pf_end        = vout_End;
93     p_function_list->functions.vout.pf_destroy    = vout_Destroy;
94     p_function_list->functions.vout.pf_manage     = vout_Manage;
95     p_function_list->functions.vout.pf_render     = vout_Render;
96     p_function_list->functions.vout.pf_display    = vout_Display;
97 }
98
99 /*****************************************************************************
100  * vout_Create: allocates Invert video thread output method
101  *****************************************************************************
102  * This function allocates and initializes a Invert vout method.
103  *****************************************************************************/
104 static int vout_Create( vout_thread_t *p_vout )
105 {
106     /* Allocate structure */
107     p_vout->p_sys = malloc( sizeof( vout_sys_t ) );
108     if( p_vout->p_sys == NULL )
109     {
110         msg_Err( p_vout, "out of memory" );
111         return( 1 );
112     }
113
114     return( 0 );
115 }
116
117 /*****************************************************************************
118  * vout_Init: initialize Invert video thread output method
119  *****************************************************************************/
120 static int vout_Init( vout_thread_t *p_vout )
121 {
122     int i_index;
123     char *psz_filter;
124     picture_t *p_pic;
125
126     I_OUTPUTPICTURES = 0;
127
128     /* Initialize the output structure */
129     p_vout->output.i_chroma = p_vout->render.i_chroma;
130     p_vout->output.i_width  = p_vout->render.i_width;
131     p_vout->output.i_height = p_vout->render.i_height;
132     p_vout->output.i_aspect = p_vout->render.i_aspect;
133
134     /* Try to open the real video output */
135     psz_filter = config_GetPsz( p_vout, "filter" );
136     config_PutPsz( p_vout, "filter", NULL );
137
138     msg_Dbg( p_vout, "spawning the real video output" );
139
140     p_vout->p_sys->p_vout =
141         vout_CreateThread( p_vout->p_this,
142                            p_vout->render.i_width, p_vout->render.i_height,
143                            p_vout->render.i_chroma, p_vout->render.i_aspect );
144
145     config_PutPsz( p_vout, "filter", psz_filter );
146     if( psz_filter ) free( psz_filter );
147
148     /* Everything failed */
149     if( p_vout->p_sys->p_vout == NULL )
150     {
151         msg_Err( p_vout, "can't open vout, aborting" );
152
153         return( 0 );
154     }
155  
156     ALLOCATE_DIRECTBUFFERS( VOUT_MAX_PICTURES );
157
158     return( 0 );
159 }
160
161 /*****************************************************************************
162  * vout_End: terminate Invert video thread output method
163  *****************************************************************************/
164 static void vout_End( vout_thread_t *p_vout )
165 {
166     int i_index;
167
168     /* Free the fake output buffers we allocated */
169     for( i_index = I_OUTPUTPICTURES ; i_index ; )
170     {
171         i_index--;
172         free( PP_OUTPUTPICTURE[ i_index ]->p_data_orig );
173     }
174 }
175
176 /*****************************************************************************
177  * vout_Destroy: destroy Invert video thread output method
178  *****************************************************************************
179  * Terminate an output method created by InvertCreateOutputMethod
180  *****************************************************************************/
181 static void vout_Destroy( vout_thread_t *p_vout )
182 {
183     vout_DestroyThread( p_vout->p_sys->p_vout );
184
185     free( p_vout->p_sys );
186 }
187
188 /*****************************************************************************
189  * vout_Manage: handle Invert events
190  *****************************************************************************
191  * This function should be called regularly by video output thread. It manages
192  * console events. It returns a non null value on error.
193  *****************************************************************************/
194 static int vout_Manage( vout_thread_t *p_vout )
195 {
196     return( 0 );
197 }
198
199 /*****************************************************************************
200  * vout_Render: displays previously rendered output
201  *****************************************************************************
202  * This function send the currently rendered image to Invert image, waits
203  * until it is displayed and switch the two rendering buffers, preparing next
204  * frame.
205  *****************************************************************************/
206 static void vout_Render( vout_thread_t *p_vout, picture_t *p_pic )
207 {
208     picture_t *p_outpic;
209     int i_index;
210
211     /* This is a new frame. Get a structure from the video_output. */
212     while( ( p_outpic = vout_CreatePicture( p_vout->p_sys->p_vout, 0, 0, 0 ) )
213               == NULL )
214     {
215         if( p_vout->b_die || p_vout->b_error )
216         {
217             return;
218         }
219         msleep( VOUT_OUTMEM_SLEEP );
220     }   
221
222     vout_DatePicture( p_vout->p_sys->p_vout, p_outpic, p_pic->date );
223     vout_LinkPicture( p_vout->p_sys->p_vout, p_outpic );
224
225     for( i_index = 0 ; i_index < p_pic->i_planes ; i_index++ )
226     {
227         u8 *p_in, *p_in_end, *p_out;
228
229         p_in = p_pic->p[i_index].p_pixels;
230         p_in_end = p_in - 64 + p_pic->p[i_index].i_lines
231                                 * p_pic->p[i_index].i_pitch;
232
233         p_out = p_outpic->p[i_index].p_pixels;
234
235         for( ; p_in < p_in_end ; )
236         {
237             /* Do 64 pixels at a time */
238             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
239             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
240             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
241             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
242             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
243             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
244             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
245             *((u64*)p_out)++ = ~( *((u64*)p_in)++ );
246         }
247
248         p_in_end += 64;
249
250         for( ; p_in < p_in_end ; )
251         {
252             /* Do 1 pixel at a time */
253             *p_out++ = ~( *p_in++ );
254         }
255     }
256
257     vout_UnlinkPicture( p_vout->p_sys->p_vout, p_outpic );
258
259     vout_DisplayPicture( p_vout->p_sys->p_vout, p_outpic );
260 }
261
262 /*****************************************************************************
263  * vout_Display: displays previously rendered output
264  *****************************************************************************
265  * This function send the currently rendered image to Invert image, waits
266  * until it is displayed and switch the two rendering buffers, preparing next
267  * frame.
268  *****************************************************************************/
269 static void vout_Display( vout_thread_t *p_vout, picture_t *p_pic )
270 {
271     ;
272 }
273