]> git.sesse.net Git - vlc/blob - modules/video_output/yuv.c
Wav: No tabs in source code
[vlc] / modules / video_output / yuv.c
1 /*****************************************************************************
2  * yuv.c : yuv video output
3  *****************************************************************************
4  * Copyright (C) 2008, M2X BV
5  * $Id$
6  *
7  * Authors: Jean-Paul Saman <jpsaman@videolan.org>
8  *
9  * This program is free software; you can redistribute it and/or modify it
10  * under the terms of the GNU Lesser General Public License as published by
11  * the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
18  *
19  * You should have received a copy of the GNU Lesser General Public License
20  * along with this program; if not, write to the Free Software Foundation,
21  * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27
28 #ifdef HAVE_CONFIG_H
29 # include "config.h"
30 #endif
31
32 #include <vlc_common.h>
33 #include <vlc_plugin.h>
34 #include <vlc_vout_display.h>
35 #include <vlc_picture_pool.h>
36 #include <vlc_fs.h>
37
38 /*****************************************************************************
39  * Module descriptor
40  *****************************************************************************/
41 #define YUV_FILE_TEXT N_("device, fifo or filename")
42 #define YUV_FILE_LONGTEXT N_("device, fifo or filename to write yuv frames too.")
43
44 #define CHROMA_TEXT N_("Chroma used")
45 #define CHROMA_LONGTEXT N_(\
46     "Force use of a specific chroma for output. Default is I420.")
47
48 #define YUV4MPEG2_TEXT N_("YUV4MPEG2 header (default disabled)")
49 #define YUV4MPEG2_LONGTEXT N_("The YUV4MPEG2 header is compatible " \
50     "with mplayer yuv video output and requires YV12/I420 fourcc. By default "\
51     "vlc writes the fourcc of the picture frame into the output destination.")
52
53 #define CFG_PREFIX "yuv-"
54
55 static int  Open (vlc_object_t *);
56 static void Close(vlc_object_t *);
57
58 vlc_module_begin()
59     set_shortname(N_("YUV output"))
60     set_description(N_("YUV video output"))
61     set_category(CAT_VIDEO)
62     set_subcategory(SUBCAT_VIDEO_VOUT)
63     set_capability("vout display", 0)
64
65     add_string(CFG_PREFIX "file", "stream.yuv",
66                 YUV_FILE_TEXT, YUV_FILE_LONGTEXT, false)
67     add_string(CFG_PREFIX "chroma", NULL,
68                 CHROMA_TEXT, CHROMA_LONGTEXT, true)
69     add_bool  (CFG_PREFIX "yuv4mpeg2", false,
70                 YUV4MPEG2_TEXT, YUV4MPEG2_LONGTEXT, true)
71
72     set_callbacks(Open, Close)
73 vlc_module_end()
74
75 /*****************************************************************************
76  * Local prototypes
77  *****************************************************************************/
78 static const char *const ppsz_vout_options[] = {
79     "file", "chroma", "yuv4mpeg2", NULL
80 };
81
82 /* */
83 static picture_pool_t *Pool  (vout_display_t *, unsigned);
84 static void           Display(vout_display_t *, picture_t *, subpicture_t *subpicture);
85 static int            Control(vout_display_t *, int, va_list);
86
87 /*****************************************************************************
88  * vout_display_sys_t: video output descriptor
89  *****************************************************************************/
90 struct vout_display_sys_t {
91     FILE *f;
92     bool  is_first;
93     bool  is_yuv4mpeg2;
94
95     picture_pool_t *pool;
96 };
97
98 /* */
99 static int Open(vlc_object_t *object)
100 {
101     vout_display_t *vd = (vout_display_t *)object;
102     vout_display_sys_t *sys;
103
104     /* Allocate instance and initialize some members */
105     vd->sys = sys = malloc(sizeof(*sys));
106     if (!sys)
107         return VLC_ENOMEM;
108
109     sys->is_first = false;
110     sys->is_yuv4mpeg2 = var_InheritBool(vd, CFG_PREFIX "yuv4mpeg2");
111     sys->pool = NULL;
112
113     /* */
114     char *psz_fcc = var_InheritString(vd, CFG_PREFIX "chroma");
115     const vlc_fourcc_t requested_chroma = vlc_fourcc_GetCodecFromString(VIDEO_ES,
116                                                                         psz_fcc);
117     free(psz_fcc);
118
119     const vlc_fourcc_t chroma = requested_chroma ? requested_chroma :
120                                                    VLC_CODEC_I420;
121     if (sys->is_yuv4mpeg2) {
122         switch (chroma) {
123         case VLC_CODEC_YV12:
124         case VLC_CODEC_I420:
125         case VLC_CODEC_J420:
126             break;
127         default:
128             msg_Err(vd, "YUV4MPEG2 mode needs chroma YV12 not %4.4s as requested",
129                     (char *)&chroma);
130             free(sys);
131             return VLC_EGENERIC;
132         }
133     }
134     msg_Dbg(vd, "Using chroma %4.4s", (char *)&chroma);
135
136     /* */
137     char *name = var_InheritString(vd, CFG_PREFIX "file");
138     if (!name) {
139         msg_Err(vd, "Empty file name");
140         free(sys);
141         return VLC_EGENERIC;
142     }
143     sys->f = vlc_fopen(name, "wb");
144
145     if (!sys->f) {
146         msg_Err(vd, "Failed to open %s", name);
147         free(name);
148         free(sys);
149         return VLC_EGENERIC;
150     }
151     msg_Dbg(vd, "Writing data to %s", name);
152     free(name);
153
154     /* */
155     video_format_t fmt = vd->fmt;
156     fmt.i_chroma = chroma;
157     video_format_FixRgb(&fmt);
158
159     /* */
160     vout_display_info_t info = vd->info;
161     info.has_hide_mouse = true;
162
163     /* */
164     vd->fmt     = fmt;
165     vd->info    = info;
166     vd->pool    = Pool;
167     vd->prepare = NULL;
168     vd->display = Display;
169     vd->control = Control;
170     vd->manage  = NULL;
171
172     vout_display_SendEventFullscreen(vd, false);
173     return VLC_SUCCESS;
174 }
175
176 /* */
177 static void Close(vlc_object_t *object)
178 {
179     vout_display_t *vd = (vout_display_t *)object;
180     vout_display_sys_t *sys = vd->sys;
181
182     if (sys->pool)
183         picture_pool_Delete(sys->pool);
184     fclose(sys->f);
185     free(sys);
186 }
187
188 /*****************************************************************************
189  *
190  *****************************************************************************/
191 static picture_pool_t *Pool(vout_display_t *vd, unsigned count)
192 {
193     vout_display_sys_t *sys = vd->sys;
194     if (!sys->pool)
195         sys->pool = picture_pool_NewFromFormat(&vd->fmt, count);
196     return sys->pool;
197 }
198
199 static void Display(vout_display_t *vd, picture_t *picture, subpicture_t *subpicture)
200 {
201     vout_display_sys_t *sys = vd->sys;
202
203     /* */
204     video_format_t fmt = vd->fmt;
205     fmt.i_sar_num = vd->source.i_sar_num;
206     fmt.i_sar_den = vd->source.i_sar_den;
207
208     /* */
209     char type;
210     if (picture->b_progressive)
211         type = 'p';
212     else if (picture->b_top_field_first)
213         type = 't';
214     else
215         type = 'b';
216
217     if (type != 'p') {
218         msg_Warn(vd, "Received a non progressive frame, "
219                      "it will be written as progressive.");
220         type = 'p';
221     }
222
223     /* */
224     if (!sys->is_first) {
225         const char *header;
226         char buffer[5];
227         if (sys->is_yuv4mpeg2) {
228             /* MPlayer compatible header, unfortunately it doesn't tell you
229              * the exact fourcc used. */
230             header = "YUV4MPEG2";
231         } else {
232             snprintf(buffer, sizeof(buffer), "%4.4s", 
233                      (const char*)&fmt.i_chroma);
234             header = buffer;
235         }
236
237         fprintf(sys->f, "%s W%d H%d F%d:%d I%c A%d:%d\n",
238                 header,
239                 fmt.i_visible_width, fmt.i_visible_height,
240                 fmt.i_frame_rate, fmt.i_frame_rate_base,
241                 type,
242                 fmt.i_sar_num, fmt.i_sar_den);
243         sys->is_first = true;
244     }
245
246     /* */
247     fprintf(sys->f, "FRAME\n");
248     for (int i = 0; i < picture->i_planes; i++) {
249         const plane_t *plane = &picture->p[i];
250         for( int y = 0; y < plane->i_visible_lines; y++) {
251             const size_t written = fwrite(&plane->p_pixels[y*plane->i_pitch],
252                                           1, plane->i_visible_pitch, sys->f);
253             if (written != (size_t)plane->i_visible_pitch)
254                 msg_Warn(vd, "only %zd of %d bytes written",
255                          written, plane->i_visible_pitch);
256         }
257     }
258     fflush(sys->f);
259
260     /* */
261     picture_Release(picture);
262     VLC_UNUSED(subpicture);
263 }
264
265 static int Control(vout_display_t *vd, int query, va_list args)
266 {
267     VLC_UNUSED(vd);
268     switch (query) {
269     case VOUT_DISPLAY_CHANGE_FULLSCREEN: {
270         const vout_display_cfg_t *cfg = va_arg(args, const vout_display_cfg_t *);
271         if (cfg->is_fullscreen)
272             return VLC_EGENERIC;
273         return VLC_SUCCESS;
274     }
275     default:
276         return VLC_EGENERIC;
277     }
278 }
279