]> git.sesse.net Git - vlc/blob - modules/video_output/yuv.c
Bump OpenGL capability on Windows
[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;
156     video_format_ApplyRotation(&fmt, &vd->fmt);
157     fmt.i_chroma = chroma;
158     video_format_FixRgb(&fmt);
159
160     /* */
161     vout_display_info_t info = vd->info;
162     info.has_hide_mouse = true;
163
164     /* */
165     vd->fmt     = fmt;
166     vd->info    = info;
167     vd->pool    = Pool;
168     vd->prepare = NULL;
169     vd->display = Display;
170     vd->control = Control;
171     vd->manage  = NULL;
172
173     vout_display_SendEventFullscreen(vd, false);
174     vout_display_DeleteWindow(vd, NULL);
175     return VLC_SUCCESS;
176 }
177
178 /* */
179 static void Close(vlc_object_t *object)
180 {
181     vout_display_t *vd = (vout_display_t *)object;
182     vout_display_sys_t *sys = vd->sys;
183
184     if (sys->pool)
185         picture_pool_Release(sys->pool);
186     fclose(sys->f);
187     free(sys);
188 }
189
190 /*****************************************************************************
191  *
192  *****************************************************************************/
193 static picture_pool_t *Pool(vout_display_t *vd, unsigned count)
194 {
195     vout_display_sys_t *sys = vd->sys;
196     if (!sys->pool)
197         sys->pool = picture_pool_NewFromFormat(&vd->fmt, count);
198     return sys->pool;
199 }
200
201 static void Display(vout_display_t *vd, picture_t *picture, subpicture_t *subpicture)
202 {
203     vout_display_sys_t *sys = vd->sys;
204
205     /* */
206     video_format_t fmt = vd->fmt;
207
208     if (ORIENT_IS_SWAP(vd->source.orientation))
209     {
210         fmt.i_sar_num = vd->source.i_sar_den;
211         fmt.i_sar_den = vd->source.i_sar_num;
212     }
213     else
214     {
215         fmt.i_sar_num = vd->source.i_sar_num;
216         fmt.i_sar_den = vd->source.i_sar_den;
217     }
218
219     /* */
220     char type;
221     if (picture->b_progressive)
222         type = 'p';
223     else if (picture->b_top_field_first)
224         type = 't';
225     else
226         type = 'b';
227
228     if (type != 'p') {
229         msg_Warn(vd, "Received a non progressive frame, "
230                      "it will be written as progressive.");
231         type = 'p';
232     }
233
234     /* */
235     if (!sys->is_first) {
236         const char *header;
237         char buffer[5];
238         if (sys->is_yuv4mpeg2) {
239             /* MPlayer compatible header, unfortunately it doesn't tell you
240              * the exact fourcc used. */
241             header = "YUV4MPEG2";
242         } else {
243             snprintf(buffer, sizeof(buffer), "%4.4s", 
244                      (const char*)&fmt.i_chroma);
245             header = buffer;
246         }
247
248         fprintf(sys->f, "%s W%d H%d F%d:%d I%c A%d:%d\n",
249                 header,
250                 fmt.i_visible_width, fmt.i_visible_height,
251                 fmt.i_frame_rate, fmt.i_frame_rate_base,
252                 type,
253                 fmt.i_sar_num, fmt.i_sar_den);
254         sys->is_first = true;
255     }
256
257     /* */
258     fprintf(sys->f, "FRAME\n");
259     for (int i = 0; i < picture->i_planes; i++) {
260         const plane_t *plane = &picture->p[i];
261         const uint8_t *pixels = plane->p_pixels;
262
263         pixels += (vd->fmt.i_x_offset * plane->i_visible_pitch)
264                   / vd->fmt.i_visible_height;
265
266         for( int y = 0; y < plane->i_visible_lines; y++) {
267             const size_t written = fwrite(pixels, 1, plane->i_visible_pitch,
268                                           sys->f);
269             if (written != (size_t)plane->i_visible_pitch)
270                 msg_Warn(vd, "only %zd of %d bytes written",
271                          written, plane->i_visible_pitch);
272
273             pixels += plane->i_pitch;
274         }
275     }
276     fflush(sys->f);
277
278     /* */
279     picture_Release(picture);
280     VLC_UNUSED(subpicture);
281 }
282
283 static int Control(vout_display_t *vd, int query, va_list args)
284 {
285     (void) vd; (void) query; (void) args;
286     return VLC_EGENERIC;
287 }