2 * gstvaapiimage.c - VA image abstraction
4 * gstreamer-vaapi (C) 2010 Splitted-Desktop Systems
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 #include "vaapi_utils.h"
24 #include "gstvaapiimage.h"
25 #include <va/va_backend.h>
28 #include "vaapi_debug.h"
30 G_DEFINE_TYPE(GstVaapiImage, gst_vaapi_image, G_TYPE_OBJECT);
32 #define GST_VAAPI_IMAGE_GET_PRIVATE(obj) \
33 (G_TYPE_INSTANCE_GET_PRIVATE((obj), \
34 GST_VAAPI_TYPE_IMAGE, \
35 GstVaapiImagePrivate))
37 struct _GstVaapiImagePrivate {
38 GstVaapiDisplay *display;
43 GstVaapiImageFormat format;
57 gst_vaapi_image_destroy(GstVaapiImage *image)
59 GstVaapiImagePrivate * const priv = image->priv;
60 VADisplay dpy = gst_vaapi_display_get_display(priv->display);
63 gst_vaapi_image_unmap(image);
65 if (priv->image.image_id != VA_INVALID_ID) {
66 status = vaDestroyImage(dpy, priv->image.image_id);
67 if (!vaapi_check_status(status, "vaDestroyImage()"))
68 g_warning("failed to destroy image 0x%08x\n", priv->image.image_id);
69 priv->image.image_id = VA_INVALID_ID;
73 g_object_unref(priv->display);
79 gst_vaapi_image_create(GstVaapiImage *image)
81 GstVaapiImagePrivate * const priv = image->priv;
82 const VAImageFormat *format;
85 if (!gst_vaapi_display_has_image_format(priv->display, priv->format))
88 format = gst_vaapi_image_format_get_va_format(priv->format);
90 g_return_val_if_fail(format, FALSE);
92 status = vaCreateImage(
93 gst_vaapi_display_get_display(priv->display),
94 (VAImageFormat *)format,
99 if (!vaapi_check_status(status, "vaCreateImage()"))
106 gst_vaapi_image_finalize(GObject *object)
108 gst_vaapi_image_destroy(GST_VAAPI_IMAGE(object));
110 G_OBJECT_CLASS(gst_vaapi_image_parent_class)->finalize(object);
114 gst_vaapi_image_set_property(
121 GstVaapiImage * const image = GST_VAAPI_IMAGE(object);
122 GstVaapiImagePrivate * const priv = image->priv;
126 priv->display = g_object_ref(g_value_get_pointer(value));
129 priv->width = g_value_get_uint(value);
132 priv->height = g_value_get_uint(value);
135 priv->format = g_value_get_uint(value);
138 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
144 gst_vaapi_image_get_property(
151 GstVaapiImage * const image = GST_VAAPI_IMAGE(object);
152 GstVaapiImagePrivate * const priv = image->priv;
156 g_value_set_pointer(value, g_object_ref(priv->display));
159 g_value_set_uint(value, gst_vaapi_image_get_id(image));
162 g_value_set_uint(value, gst_vaapi_image_get_width(image));
165 g_value_set_uint(value, gst_vaapi_image_get_height(image));
168 g_value_set_uint(value, gst_vaapi_image_get_format(image));
171 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
177 gst_vaapi_image_constructed(GObject *object)
179 GstVaapiImage * const image = GST_VAAPI_IMAGE(object);
180 GObjectClass *parent_class;
182 D(bug("gst_vaapi_image_constructed()\n"));
184 gst_vaapi_image_create(image);
186 parent_class = G_OBJECT_CLASS(gst_vaapi_image_parent_class);
187 if (parent_class->constructed)
188 parent_class->constructed(object);
192 gst_vaapi_image_class_init(GstVaapiImageClass *klass)
194 GObjectClass * const object_class = G_OBJECT_CLASS(klass);
196 g_type_class_add_private(klass, sizeof(GstVaapiImagePrivate));
198 object_class->finalize = gst_vaapi_image_finalize;
199 object_class->set_property = gst_vaapi_image_set_property;
200 object_class->get_property = gst_vaapi_image_get_property;
201 object_class->constructed = gst_vaapi_image_constructed;
203 g_object_class_install_property
206 g_param_spec_object("display",
208 "GStreamer Va display",
209 GST_VAAPI_TYPE_DISPLAY,
210 G_PARAM_READWRITE|G_PARAM_CONSTRUCT_ONLY));
212 g_object_class_install_property
215 g_param_spec_uint("id",
218 0, G_MAXUINT32, VA_INVALID_ID,
221 g_object_class_install_property
224 g_param_spec_uint("width",
228 G_PARAM_READWRITE|G_PARAM_CONSTRUCT_ONLY));
230 g_object_class_install_property
233 g_param_spec_uint("height",
237 G_PARAM_READWRITE|G_PARAM_CONSTRUCT_ONLY));
239 g_object_class_install_property
242 g_param_spec_uint("format",
246 G_PARAM_READWRITE|G_PARAM_CONSTRUCT_ONLY));
250 gst_vaapi_image_init(GstVaapiImage *image)
252 GstVaapiImagePrivate *priv = GST_VAAPI_IMAGE_GET_PRIVATE(image);
254 D(bug("gst_vaapi_image_init()\n"));
257 priv->display = NULL;
258 priv->image_data = NULL;
263 memset(&priv->image, 0, sizeof(priv->image));
264 priv->image.image_id = VA_INVALID_ID;
265 priv->image.buf = VA_INVALID_ID;
270 GstVaapiDisplay *display,
273 GstVaapiImageFormat format
276 D(bug("gst_vaapi_image_new(): size %ux%u, format 0x%x\n",
277 width, height, format));
279 return g_object_new(GST_VAAPI_TYPE_IMAGE,
288 gst_vaapi_image_get_id(GstVaapiImage *image)
290 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), VA_INVALID_ID);
292 return image->priv->image.image_id;
296 gst_vaapi_image_get_display(GstVaapiImage *image)
298 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), NULL);
300 return g_object_ref(image->priv->display);
304 gst_vaapi_image_get_width(GstVaapiImage *image)
306 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), 0);
308 return image->priv->width;
312 gst_vaapi_image_get_height(GstVaapiImage *image)
314 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), 0);
316 return image->priv->height;
320 gst_vaapi_image_get_format(GstVaapiImage *image)
322 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), 0);
324 return image->priv->format;
327 static inline gboolean
328 _gst_vaapi_image_is_mapped(GstVaapiImage *image)
330 return image->priv->image_data != NULL;
334 gst_vaapi_image_is_mapped(GstVaapiImage *image)
336 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), FALSE);
338 return _gst_vaapi_image_is_mapped(image);
342 gst_vaapi_image_map(GstVaapiImage *image)
347 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), FALSE);
349 if (_gst_vaapi_image_is_mapped(image))
352 status = vaMapBuffer(
353 gst_vaapi_display_get_display(image->priv->display),
354 image->priv->image.buf,
357 if (!vaapi_check_status(status, "vaMapBuffer()"))
360 image->priv->image_data = image_data;
365 gst_vaapi_image_unmap(GstVaapiImage *image)
369 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), FALSE);
371 if (!_gst_vaapi_image_is_mapped(image))
374 status = vaUnmapBuffer(
375 gst_vaapi_display_get_display(image->priv->display),
376 image->priv->image.buf
378 if (!vaapi_check_status(status, "vaUnmapBuffer()"))
381 image->priv->image_data = NULL;
386 gst_vaapi_image_get_plane_count(GstVaapiImage *image)
388 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), 0);
389 g_return_val_if_fail(_gst_vaapi_image_is_mapped(image), 0);
391 return image->priv->image.num_planes;
395 gst_vaapi_image_get_plane(GstVaapiImage *image, guint plane)
397 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), NULL);
398 g_return_val_if_fail(_gst_vaapi_image_is_mapped(image), NULL);
399 g_return_val_if_fail(plane < image->priv->image.num_planes, NULL);
401 return image->priv->image_data + image->priv->image.offsets[plane];
405 gst_vaapi_image_get_pitch(GstVaapiImage *image, guint plane)
407 g_return_val_if_fail(GST_VAAPI_IS_IMAGE(image), 0);
408 g_return_val_if_fail(_gst_vaapi_image_is_mapped(image), 0);
409 g_return_val_if_fail(plane < image->priv->image.num_planes, 0);
411 return image->priv->image.pitches[plane];