]> git.sesse.net Git - vlc/blob - modules/codec/ffmpeg/postprocess.c
36b00eec271b73d757714c3fc9afde61f2169691
[vlc] / modules / codec / ffmpeg / postprocess.c
1 /*****************************************************************************
2  * postprocess.c: video postprocessing using the ffmpeg library
3  *****************************************************************************
4  * Copyright (C) 1999-2001 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
8  *          Gildas Bazin <gbazin@netcourrier.com>
9  *
10  * This program is free software; you can redistribute it and/or modify
11  * it under the terms of the GNU General Public License as published by
12  * the Free Software Foundation; either version 2 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23  *****************************************************************************/
24
25 #ifdef HAVE_CONFIG_H
26 # include "config.h"
27 #endif
28
29 #include <vlc/vlc.h>
30 #include <vlc_vout.h>
31 #include <vlc_codec.h>
32
33 /* ffmpeg header */
34 #ifdef HAVE_FFMPEG_AVCODEC_H
35 #   include <ffmpeg/avcodec.h>
36 #else
37 #   include <avcodec.h>
38 #endif
39
40 #include "ffmpeg.h"
41
42 #ifdef HAVE_POSTPROC_POSTPROCESS_H
43 #   include <postproc/postprocess.h>
44 #else
45 #   include <libpostproc/postprocess.h>
46 #endif
47
48 #ifndef PP_CPU_CAPS_ALTIVEC
49 #   define PP_CPU_CAPS_ALTIVEC 0
50 #endif
51
52 /*****************************************************************************
53  * video_postproc_sys_t : ffmpeg video postprocessing descriptor
54  *****************************************************************************/
55 typedef struct video_postproc_sys_t
56 {
57     pp_context_t *pp_context;
58     pp_mode_t    *pp_mode;
59
60     vlc_bool_t   *pb_pp;
61
62     int i_width;
63     int i_height;
64
65 } video_postproc_sys_t;
66
67 static int PPQCallback( vlc_object_t *p_this, char const *psz_cmd,
68                         vlc_value_t oldval, vlc_value_t newval, void *p_data );
69
70 /*****************************************************************************
71  * OpenPostproc: probe and open the postproc
72  *****************************************************************************/
73 void *E_(OpenPostproc)( decoder_t *p_dec, vlc_bool_t *pb_pp )
74 {
75     video_postproc_sys_t *p_sys;
76     vlc_value_t val, val_orig, text;
77
78     p_sys = malloc( sizeof(video_postproc_sys_t) );
79     p_sys->pp_context = NULL;
80     p_sys->pp_mode = NULL;
81
82     *pb_pp = VLC_FALSE;
83     p_sys->pb_pp = pb_pp;
84
85     /* Create object variable if not already done */
86     if( var_Type( p_dec, "ffmpeg-pp-q" ) == 0 )
87     {
88         var_Create( p_dec, "ffmpeg-pp-q",
89                     VLC_VAR_INTEGER | VLC_VAR_HASCHOICE | VLC_VAR_DOINHERIT );
90         text.psz_string = _("Post processing");
91         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_SETTEXT, &text, NULL );
92
93         var_Get( p_dec, "ffmpeg-pp-q", &val_orig );
94         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_DELCHOICE, &val_orig, NULL );
95
96         val.i_int = 0; text.psz_string = _("Disable");
97         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, &text );
98         val.i_int = 1; text.psz_string = _("1 (Lowest)");
99         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, &text );
100         val.i_int = 2;
101         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, NULL );
102         val.i_int = 3;
103         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, NULL );
104         val.i_int = 4;
105         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, NULL );
106         val.i_int = 5;
107         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, NULL );
108         val.i_int = 6; text.psz_string = _("6 (Highest)");
109         var_Change( p_dec, "ffmpeg-pp-q", VLC_VAR_ADDCHOICE, &val, &text );
110         var_AddCallback( p_dec, "ffmpeg-pp-q", PPQCallback, p_sys );
111     }
112
113     /* ***** Load post processing if enabled ***** */
114     var_Get( p_dec, "ffmpeg-pp-q", &val );
115     var_Set( p_dec, "ffmpeg-pp-q", val_orig );
116     if( val_orig.i_int )
117         *pb_pp = VLC_TRUE;
118
119     return p_sys;
120 }
121
122 /*****************************************************************************
123  * InitPostproc:
124  *****************************************************************************/
125 int E_(InitPostproc)( decoder_t *p_dec, void *p_data,
126                       int i_width, int i_height, int pix_fmt )
127 {
128     video_postproc_sys_t *p_sys = (video_postproc_sys_t *)p_data;
129     unsigned i_cpu = vlc_CPU();
130     int i_flags = 0;
131
132     /* Set CPU capabilities */
133     if( i_cpu & CPU_CAPABILITY_MMX )
134     {
135         i_flags |= PP_CPU_CAPS_MMX;
136     }
137     if( i_cpu & CPU_CAPABILITY_MMXEXT )
138     {
139         i_flags |= PP_CPU_CAPS_MMX2;
140     }
141     if( i_cpu & CPU_CAPABILITY_3DNOW )
142     {
143         i_flags |= PP_CPU_CAPS_3DNOW;
144     }
145     if( i_cpu & CPU_CAPABILITY_ALTIVEC )
146     {
147         i_flags |= PP_CPU_CAPS_ALTIVEC;
148     }
149
150     switch( pix_fmt )
151     {
152     case PIX_FMT_YUV444P:
153         i_flags |= PP_FORMAT_444;
154         break;
155     case PIX_FMT_YUV422P:
156         i_flags |= PP_FORMAT_422;
157         break;
158     case PIX_FMT_YUV411P:
159         i_flags |= PP_FORMAT_411;
160         break;
161     default:
162         i_flags |= PP_FORMAT_420;
163         break;
164     }
165
166     p_sys->pp_context = pp_get_context( i_width, i_height, i_flags );
167     p_sys->i_width = i_width;
168     p_sys->i_height = i_height;
169
170     return VLC_SUCCESS;
171 }
172
173 /*****************************************************************************
174  * PostprocPict:
175  *****************************************************************************/
176 int E_(PostprocPict)( decoder_t *p_dec, void *p_data,
177                       picture_t *p_pic, AVFrame *p_ff_pic )
178 {
179     video_postproc_sys_t *p_sys = (video_postproc_sys_t *)p_data;
180
181     uint8_t *src[3], *dst[3];
182     int i_plane, i_src_stride[3], i_dst_stride[3];
183
184     for( i_plane = 0; i_plane < p_pic->i_planes; i_plane++ )
185     {
186         src[i_plane] = p_ff_pic->data[i_plane];
187         dst[i_plane] = p_pic->p[i_plane].p_pixels;
188
189         i_src_stride[i_plane] = p_ff_pic->linesize[i_plane];
190         i_dst_stride[i_plane] = p_pic->p[i_plane].i_pitch;
191     }
192
193     pp_postprocess( src, i_src_stride, dst, i_dst_stride,
194                     p_sys->i_width, p_sys->i_height,
195                     p_ff_pic->qscale_table, p_ff_pic->qstride,
196                     p_sys->pp_mode, p_sys->pp_context,
197                     p_ff_pic->pict_type );
198
199     return VLC_SUCCESS;
200 }
201
202 /*****************************************************************************
203  * ClosePostproc:
204  *****************************************************************************/
205 void E_(ClosePostproc)( decoder_t *p_dec, void *p_data )
206 {
207     video_postproc_sys_t *p_sys = (video_postproc_sys_t *)p_data;
208
209     if( p_sys && p_sys->pp_mode )
210     {
211         pp_free_mode( p_sys->pp_mode );
212         if( p_sys->pp_context ) pp_free_context( p_sys->pp_context );
213     }
214
215     var_DelCallback( p_dec, "ffmpeg-pp-q", PPQCallback, p_sys );
216
217     if( p_sys ) free( p_sys );
218 }
219
220 /*****************************************************************************
221  * object variables callbacks: a bunch of object variables are used by the
222  * interfaces to interact with the decoder.
223  *****************************************************************************/
224 static int PPQCallback( vlc_object_t *p_this, char const *psz_cmd,
225                         vlc_value_t oldval, vlc_value_t newval, void *p_data )
226 {
227     decoder_t *p_dec = (decoder_t *)p_this;
228     video_postproc_sys_t *p_sys = (video_postproc_sys_t *)p_data;
229
230     if( newval.i_int > 0 )
231     {
232         int  i_quality = newval.i_int;
233         char *psz_name = config_GetPsz( p_dec, "ffmpeg-pp-name" );
234         pp_mode_t *pp_mode;
235
236         if( !psz_name )
237         {
238             psz_name = strdup( "default" );
239         }
240         else if( *psz_name == '\0' )
241         {
242             free( psz_name );
243             psz_name = strdup( "default" );
244         }
245
246         pp_mode = pp_get_mode_by_name_and_quality( psz_name, i_quality );
247
248         if( !pp_mode )
249         {
250             msg_Err( p_dec, "failed getting mode for postproc" );
251             newval.i_int = 0;
252         }
253         else
254         {
255             msg_Dbg( p_dec, "postprocessing enabled" );
256         }
257         free( psz_name );
258
259         p_sys->pp_mode = pp_mode;
260     }
261     else
262     {
263         msg_Dbg( p_dec, "postprocessing disabled" );
264     }
265
266     *p_sys->pb_pp = newval.i_int;
267
268     return VLC_SUCCESS;
269 }