Filter out any format that is not supported by the library (libgstvaapi).

Also sort the formats by HW preference.
This commit is contained in:
gb 2010-03-11 12:11:36 +00:00
parent 224d06a03d
commit acbab7a1c1
3 changed files with 103 additions and 7 deletions

View file

@ -55,6 +55,66 @@ enum {
static void
gst_vaapi_display_set_display(GstVaapiDisplay *display, VADisplay va_display);
static void
filter_formats(VAImageFormat *va_formats, unsigned int *pnum_va_formats)
{
unsigned int i = 0;
while (i < *pnum_va_formats) {
VAImageFormat * const va_format = &va_formats[i];
const GstVaapiImageFormat format = gst_vaapi_image_format(va_format);
if (format)
++i;
else {
/* Remove any format that is not supported by libgstvaapi */
GST_DEBUG("unsupported format %c%c%c%c",
va_format->fourcc & 0xff,
(va_format->fourcc >> 8) & 0xff,
(va_format->fourcc >> 16) & 0xff,
(va_format->fourcc >> 24) & 0xff);
*va_format = va_formats[--(*pnum_va_formats)];
}
}
}
/* Sort image formats. Prefer YUV formats first */
static int
compare_yuv_formats(const void *a, const void *b)
{
const GstVaapiImageFormat fmt1 = gst_vaapi_image_format((VAImageFormat *)a);
const GstVaapiImageFormat fmt2 = gst_vaapi_image_format((VAImageFormat *)b);
g_assert(fmt1 && fmt2);
const gboolean is_fmt1_yuv = gst_vaapi_image_format_is_yuv(fmt1);
const gboolean is_fmt2_yuv = gst_vaapi_image_format_is_yuv(fmt2);
if (is_fmt1_yuv != is_fmt2_yuv)
return is_fmt1_yuv ? -1 : 1;
return ((int)gst_vaapi_image_format_get_score(fmt1) -
(int)gst_vaapi_image_format_get_score(fmt2));
}
/* Sort subpicture formats. Prefer RGB formats first */
static int
compare_rgb_formats(const void *a, const void *b)
{
const GstVaapiImageFormat fmt1 = gst_vaapi_image_format((VAImageFormat *)a);
const GstVaapiImageFormat fmt2 = gst_vaapi_image_format((VAImageFormat *)b);
g_assert(fmt1 && fmt2);
const gboolean is_fmt1_rgb = gst_vaapi_image_format_is_rgb(fmt1);
const gboolean is_fmt2_rgb = gst_vaapi_image_format_is_rgb(fmt2);
if (is_fmt1_rgb != is_fmt2_rgb)
return is_fmt1_rgb ? -1 : 1;
return ((int)gst_vaapi_image_format_get_score(fmt1) -
(int)gst_vaapi_image_format_get_score(fmt2));
}
static void
gst_vaapi_display_destroy(GstVaapiDisplay *display)
{
@ -121,9 +181,11 @@ gst_vaapi_display_create(GstVaapiDisplay *display)
priv->image_formats = g_new(VAImageFormat, priv->num_image_formats);
if (!priv->image_formats)
return FALSE;
status = vaQueryImageFormats(priv->display,
priv->image_formats,
&priv->num_image_formats);
status = vaQueryImageFormats(
priv->display,
priv->image_formats,
&priv->num_image_formats
);
if (!vaapi_check_status(status, "vaQueryImageFormats()"))
return FALSE;
@ -131,6 +193,14 @@ gst_vaapi_display_create(GstVaapiDisplay *display)
for (i = 0; i < priv->num_image_formats; i++)
GST_DEBUG(" %s", string_of_FOURCC(priv->image_formats[i].fourcc));
filter_formats(priv->image_formats, &priv->num_image_formats);
qsort(
priv->image_formats,
priv->num_image_formats,
sizeof(priv->image_formats[0]),
compare_yuv_formats
);
/* VA subpicture formats */
priv->num_subpicture_formats = vaMaxNumSubpictureFormats(priv->display);
priv->subpicture_formats = g_new(VAImageFormat, priv->num_subpicture_formats);
@ -139,13 +209,23 @@ gst_vaapi_display_create(GstVaapiDisplay *display)
priv->subpicture_flags = g_new(unsigned int, priv->num_subpicture_formats);
if (!priv->subpicture_flags)
return FALSE;
status = vaQuerySubpictureFormats(priv->display,
priv->subpicture_formats,
priv->subpicture_flags,
&priv->num_subpicture_formats);
status = vaQuerySubpictureFormats(
priv->display,
priv->subpicture_formats,
priv->subpicture_flags,
&priv->num_subpicture_formats
);
if (!vaapi_check_status(status, "vaQuerySubpictureFormats()"))
return FALSE;
filter_formats(priv->subpicture_formats, &priv->num_subpicture_formats);
qsort(
priv->subpicture_formats,
priv->num_subpicture_formats,
sizeof(priv->subpicture_formats[0]),
compare_rgb_formats
);
GST_DEBUG("%d subpicture formats", priv->num_subpicture_formats);
for (i = 0; i < priv->num_subpicture_formats; i++)
GST_DEBUG(" %s", string_of_FOURCC(priv->subpicture_formats[i].fourcc));

View file

@ -50,6 +50,7 @@ struct _GstVaapiImageFormatMap {
{ DEF(RGB, FORMAT, GST_VIDEO_CAPS_##FORMAT), \
{ VA_FOURCC FOURCC, VA_##ENDIAN##_FIRST, BPP, DEPTH, R,G,B,A }, }
/* Image formats, listed in HW order preference */
static const GstVaapiImageFormatMap gst_vaapi_image_formats[] = {
DEF_YUV(NV12, ('N','V','1','2'), LSB, 12),
DEF_YUV(YV12, ('Y','V','1','2'), LSB, 12),
@ -168,3 +169,15 @@ gst_vaapi_image_format_get_caps(GstVaapiImageFormat format)
return gst_caps_make_writable(caps);
}
guint
gst_vaapi_image_format_get_score(GstVaapiImageFormat format)
{
const GstVaapiImageFormatMap *m;
m = get_map_from_gst_vaapi_image_format(format);
g_return_val_if_fail(m, G_MAXUINT);
return m - &gst_vaapi_image_formats[0];
}

View file

@ -63,6 +63,9 @@ gst_vaapi_image_format_get_va_format(GstVaapiImageFormat format);
GstCaps *
gst_vaapi_image_format_get_caps(GstVaapiImageFormat format);
guint
gst_vaapi_image_format_get_score(GstVaapiImageFormat format);
G_END_DECLS
#endif /* GST_GST_VAAPI_IMAGE_H */