1 /*****************************************************************************
2 * yuv.c : yuv video output
3 *****************************************************************************
4 * Copyright (C) 2008, M2X BV
7 * Authors: Jean-Paul Saman <jpsaman@videolan.org>
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.
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.
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 *****************************************************************************/
24 /*****************************************************************************
26 *****************************************************************************/
32 #include <vlc_common.h>
33 #include <vlc_plugin.h>
34 #include <vlc_vout_display.h>
35 #include <vlc_picture_pool.h>
38 /*****************************************************************************
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.")
44 #define CHROMA_TEXT N_("Chroma used")
45 #define CHROMA_LONGTEXT N_(\
46 "Force use of a specific chroma for output.")
48 #define YUV4MPEG2_TEXT N_("Add a YUV4MPEG2 header")
49 #define YUV4MPEG2_LONGTEXT N_("The YUV4MPEG2 header is compatible " \
50 "with mplayer yuv video output and requires YV12/I420 fourcc.")
52 #define CFG_PREFIX "yuv-"
54 static int Open (vlc_object_t
*);
55 static void Close(vlc_object_t
*);
58 set_shortname(N_("YUV output"))
59 set_description(N_("YUV video output"))
60 set_category(CAT_VIDEO
)
61 set_subcategory(SUBCAT_VIDEO_VOUT
)
62 set_capability("vout display", 0)
64 add_string(CFG_PREFIX
"file", "stream.yuv",
65 YUV_FILE_TEXT
, YUV_FILE_LONGTEXT
, false)
66 add_string(CFG_PREFIX
"chroma", NULL
,
67 CHROMA_TEXT
, CHROMA_LONGTEXT
, true)
68 add_bool (CFG_PREFIX
"yuv4mpeg2", false,
69 YUV4MPEG2_TEXT
, YUV4MPEG2_LONGTEXT
, true)
71 set_callbacks(Open
, Close
)
74 /*****************************************************************************
76 *****************************************************************************/
79 static picture_pool_t
*Pool (vout_display_t
*, unsigned);
80 static void Display(vout_display_t
*, picture_t
*, subpicture_t
*subpicture
);
81 static int Control(vout_display_t
*, int, va_list);
83 /*****************************************************************************
84 * vout_display_sys_t: video output descriptor
85 *****************************************************************************/
86 struct vout_display_sys_t
{
95 static int Open(vlc_object_t
*object
)
97 vout_display_t
*vd
= (vout_display_t
*)object
;
98 vout_display_sys_t
*sys
;
100 /* Allocate instance and initialize some members */
101 vd
->sys
= sys
= malloc(sizeof(*sys
));
105 sys
->is_first
= false;
106 sys
->is_yuv4mpeg2
= var_InheritBool(vd
, CFG_PREFIX
"yuv4mpeg2");
110 char *psz_fcc
= var_InheritString(vd
, CFG_PREFIX
"chroma");
111 const vlc_fourcc_t requested_chroma
= vlc_fourcc_GetCodecFromString(VIDEO_ES
,
115 const vlc_fourcc_t chroma
= requested_chroma
? requested_chroma
:
117 if (sys
->is_yuv4mpeg2
) {
124 msg_Err(vd
, "YUV4MPEG2 mode needs chroma YV12 not %4.4s as requested",
130 msg_Dbg(vd
, "Using chroma %4.4s", (char *)&chroma
);
133 char *name
= var_InheritString(vd
, CFG_PREFIX
"file");
135 msg_Err(vd
, "Empty file name");
139 sys
->f
= vlc_fopen(name
, "wb");
142 msg_Err(vd
, "Failed to open %s", name
);
147 msg_Dbg(vd
, "Writing data to %s", name
);
152 video_format_ApplyRotation(&fmt
, &vd
->fmt
);
153 fmt
.i_chroma
= chroma
;
154 video_format_FixRgb(&fmt
);
160 vd
->display
= Display
;
161 vd
->control
= Control
;
163 vout_display_DeleteWindow(vd
, NULL
);
168 static void Close(vlc_object_t
*object
)
170 vout_display_t
*vd
= (vout_display_t
*)object
;
171 vout_display_sys_t
*sys
= vd
->sys
;
174 picture_pool_Release(sys
->pool
);
179 /*****************************************************************************
181 *****************************************************************************/
182 static picture_pool_t
*Pool(vout_display_t
*vd
, unsigned count
)
184 vout_display_sys_t
*sys
= vd
->sys
;
186 sys
->pool
= picture_pool_NewFromFormat(&vd
->fmt
, count
);
190 static void Display(vout_display_t
*vd
, picture_t
*picture
, subpicture_t
*subpicture
)
192 vout_display_sys_t
*sys
= vd
->sys
;
195 video_format_t fmt
= vd
->fmt
;
197 if (ORIENT_IS_SWAP(vd
->source
.orientation
))
199 fmt
.i_sar_num
= vd
->source
.i_sar_den
;
200 fmt
.i_sar_den
= vd
->source
.i_sar_num
;
204 fmt
.i_sar_num
= vd
->source
.i_sar_num
;
205 fmt
.i_sar_den
= vd
->source
.i_sar_den
;
210 if (picture
->b_progressive
)
212 else if (picture
->b_top_field_first
)
218 msg_Warn(vd
, "Received a non progressive frame, "
219 "it will be written as progressive.");
224 if (!sys
->is_first
) {
227 if (sys
->is_yuv4mpeg2
) {
228 /* MPlayer compatible header, unfortunately it doesn't tell you
229 * the exact fourcc used. */
230 header
= "YUV4MPEG2";
232 snprintf(buffer
, sizeof(buffer
), "%4.4s",
233 (const char*)&fmt
.i_chroma
);
237 fprintf(sys
->f
, "%s W%d H%d F%d:%d I%c A%d:%d\n",
239 fmt
.i_visible_width
, fmt
.i_visible_height
,
240 fmt
.i_frame_rate
, fmt
.i_frame_rate_base
,
242 fmt
.i_sar_num
, fmt
.i_sar_den
);
243 sys
->is_first
= true;
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 const uint8_t *pixels
= plane
->p_pixels
;
252 pixels
+= (vd
->fmt
.i_x_offset
* plane
->i_visible_pitch
)
253 / vd
->fmt
.i_visible_height
;
255 for( int y
= 0; y
< plane
->i_visible_lines
; y
++) {
256 const size_t written
= fwrite(pixels
, 1, plane
->i_visible_pitch
,
258 if (written
!= (size_t)plane
->i_visible_pitch
)
259 msg_Warn(vd
, "only %zd of %d bytes written",
260 written
, plane
->i_visible_pitch
);
262 pixels
+= plane
->i_pitch
;
268 picture_Release(picture
);
269 VLC_UNUSED(subpicture
);
272 static int Control(vout_display_t
*vd
, int query
, va_list args
)
274 (void) vd
; (void) query
; (void) args
;