d3dvideosink: Properly copy frames to D3D with the right strides and everything

And only support color formats that are actually supported by the driver,
this allows proper zero-copy handling later and simplifies the code a lot.

Also simplify some other places, like the format mapping code.
This commit is contained in:
Sebastian Dröge 2012-12-22 17:55:08 +01:00
parent c6763c2bbb
commit 827655ffb4
4 changed files with 282 additions and 792 deletions

View file

@ -2,6 +2,7 @@
* Copyright (C) 2012 Roland Krikava <info@bluedigits.com> * Copyright (C) 2012 Roland Krikava <info@bluedigits.com>
* Copyright (C) 2010-2011 David Hoyt <dhoyt@hoytsoft.org> * Copyright (C) 2010-2011 David Hoyt <dhoyt@hoytsoft.org>
* Copyright (C) 2010 Andoni Morales <ylatuya@gmail.com> * Copyright (C) 2010 Andoni Morales <ylatuya@gmail.com>
* Copyright (C) 2012 Sebastian Dröge <sebastian.droege@collabora.co.uk>
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public * modify it under the terms of the GNU Library General Public
@ -108,20 +109,11 @@ static gint WM_D3DVIDEO_NOTIFY_DEVICE_LOST = 0;
#define D3DFMT_NV12 MAKEFOURCC ('N', 'V', '1', '2') #define D3DFMT_NV12 MAKEFOURCC ('N', 'V', '1', '2')
#endif #endif
#define BIT_MAX(bits) ((1<<bits)-1)
#define _BIT_CONV(val, from_bit, to_bit) ((gdouble)val*(((gdouble)BIT_MAX(to_bit))/((gdouble)BIT_MAX(from_bit))))
#define BIT_CONV(val,from_bit,to_bit) ((from_bit==to_bit)?val:_BIT_CONV(val, from_bit, to_bit))
#define ALPHA 3
#define RED 2
#define GREEN 1
#define BLUE 0
/** FORMATS **/ /** FORMATS **/
#define CASE(x) case x: return #x; #define CASE(x) case x: return #x;
static const gchar * static const gchar *
d3d_format2string (D3DFORMAT format) d3d_format_to_string (D3DFORMAT format)
{ {
/* Self defined up above */ /* Self defined up above */
if (format == D3DFMT_YV12) if (format == D3DFMT_YV12)
@ -137,7 +129,6 @@ d3d_format2string (D3DFORMAT format)
CASE (D3DFMT_A8R8G8B8); CASE (D3DFMT_A8R8G8B8);
CASE (D3DFMT_UYVY); CASE (D3DFMT_UYVY);
CASE (D3DFMT_R8G8B8); CASE (D3DFMT_R8G8B8);
CASE (D3DFMT_R5G6B5); CASE (D3DFMT_R5G6B5);
CASE (D3DFMT_X1R5G5B5); CASE (D3DFMT_X1R5G5B5);
CASE (D3DFMT_A1R5G5B5); CASE (D3DFMT_A1R5G5B5);
@ -195,175 +186,40 @@ d3d_format2string (D3DFORMAT format)
CASE (D3DFMT_FORCE_DWORD); CASE (D3DFMT_FORCE_DWORD);
} }
return ""; return "UNKNOWN";
}
static const gchar *
gst_video_format2string (GstVideoFormat format)
{
switch (format) {
CASE (GST_VIDEO_FORMAT_UNKNOWN);
CASE (GST_VIDEO_FORMAT_ENCODED);
CASE (GST_VIDEO_FORMAT_I420);
CASE (GST_VIDEO_FORMAT_YV12);
CASE (GST_VIDEO_FORMAT_YUY2);
CASE (GST_VIDEO_FORMAT_UYVY);
CASE (GST_VIDEO_FORMAT_AYUV);
CASE (GST_VIDEO_FORMAT_RGBx);
CASE (GST_VIDEO_FORMAT_BGRx);
CASE (GST_VIDEO_FORMAT_xRGB);
CASE (GST_VIDEO_FORMAT_xBGR);
CASE (GST_VIDEO_FORMAT_RGBA);
CASE (GST_VIDEO_FORMAT_BGRA);
CASE (GST_VIDEO_FORMAT_ARGB);
CASE (GST_VIDEO_FORMAT_ABGR);
CASE (GST_VIDEO_FORMAT_RGB);
CASE (GST_VIDEO_FORMAT_BGR);
CASE (GST_VIDEO_FORMAT_Y41B);
CASE (GST_VIDEO_FORMAT_Y42B);
CASE (GST_VIDEO_FORMAT_YVYU);
CASE (GST_VIDEO_FORMAT_Y444);
CASE (GST_VIDEO_FORMAT_v210);
CASE (GST_VIDEO_FORMAT_v216);
CASE (GST_VIDEO_FORMAT_NV12);
CASE (GST_VIDEO_FORMAT_NV21);
CASE (GST_VIDEO_FORMAT_GRAY8);
CASE (GST_VIDEO_FORMAT_GRAY16_BE);
CASE (GST_VIDEO_FORMAT_GRAY16_LE);
CASE (GST_VIDEO_FORMAT_v308);
CASE (GST_VIDEO_FORMAT_RGB16);
CASE (GST_VIDEO_FORMAT_BGR16);
CASE (GST_VIDEO_FORMAT_RGB15);
CASE (GST_VIDEO_FORMAT_BGR15);
CASE (GST_VIDEO_FORMAT_UYVP);
CASE (GST_VIDEO_FORMAT_A420);
CASE (GST_VIDEO_FORMAT_RGB8P);
CASE (GST_VIDEO_FORMAT_YUV9);
CASE (GST_VIDEO_FORMAT_YVU9);
CASE (GST_VIDEO_FORMAT_IYU1);
CASE (GST_VIDEO_FORMAT_ARGB64);
CASE (GST_VIDEO_FORMAT_AYUV64);
CASE (GST_VIDEO_FORMAT_r210);
CASE (GST_VIDEO_FORMAT_I420_10BE);
CASE (GST_VIDEO_FORMAT_I420_10LE);
CASE (GST_VIDEO_FORMAT_I422_10BE);
CASE (GST_VIDEO_FORMAT_I422_10LE);
CASE (GST_VIDEO_FORMAT_Y444_10BE);
CASE (GST_VIDEO_FORMAT_Y444_10LE);
CASE (GST_VIDEO_FORMAT_GBR);
CASE (GST_VIDEO_FORMAT_GBR_10BE);
CASE (GST_VIDEO_FORMAT_GBR_10LE);
}
return "";
} }
#undef CASE #undef CASE
static gboolean static const struct
gst_video_can_handle_d3d_rgb (D3DFORMAT fmt)
{ {
switch (fmt) { GstVideoFormat gst_format;
case D3DFMT_A8R8G8B8: D3DFORMAT d3d_format;
case D3DFMT_X8R8G8B8: } gst_d3d_format_map[] = {
case D3DFMT_A8B8G8R8: {
case D3DFMT_X8B8G8R8: GST_VIDEO_FORMAT_BGRx, D3DFMT_X8R8G8B8}, {
case D3DFMT_R8G8B8: GST_VIDEO_FORMAT_RGBx, D3DFMT_X8B8G8R8}, {
case D3DFMT_R5G6B5: GST_VIDEO_FORMAT_BGRA, D3DFMT_A8R8G8B8}, {
return TRUE; GST_VIDEO_FORMAT_RGBA, D3DFMT_A8B8G8R8}, {
/* TODO: */ GST_VIDEO_FORMAT_BGR, D3DFMT_R8G8B8}, {
case D3DFMT_X1R5G5B5: GST_VIDEO_FORMAT_RGB16, D3DFMT_R5G6B5}, {
case D3DFMT_A1R5G5B5: GST_VIDEO_FORMAT_RGB15, D3DFMT_X1R5G5B5}, {
case D3DFMT_X4R4G4B4: GST_VIDEO_FORMAT_I420, D3DFMT_YV12}, {
case D3DFMT_A4R4G4B4: GST_VIDEO_FORMAT_YV12, D3DFMT_YV12}, {
case D3DFMT_A8R3G3B2: GST_VIDEO_FORMAT_NV12, D3DFMT_NV12}, {
case D3DFMT_R3G3B2: GST_VIDEO_FORMAT_YUY2, D3DFMT_YUY2}, {
default:; GST_VIDEO_FORMAT_UYVY, D3DFMT_UYVY}
} };
return FALSE;
}
static gboolean
gst_video_can_handle_gst_rgb (GstVideoFormat format)
{
switch (format) {
case GST_VIDEO_FORMAT_RGBx:
case GST_VIDEO_FORMAT_BGRx:
case GST_VIDEO_FORMAT_xRGB:
case GST_VIDEO_FORMAT_xBGR:
case GST_VIDEO_FORMAT_RGBA:
case GST_VIDEO_FORMAT_BGRA:
case GST_VIDEO_FORMAT_ARGB:
case GST_VIDEO_FORMAT_ABGR:
case GST_VIDEO_FORMAT_RGB:
case GST_VIDEO_FORMAT_BGR:
case GST_VIDEO_FORMAT_RGB16:
case GST_VIDEO_FORMAT_BGR16:
case GST_VIDEO_FORMAT_RGB15:
case GST_VIDEO_FORMAT_BGR15:
return TRUE;
default:;
}
return FALSE;
}
static D3DFORMAT static D3DFORMAT
gst_rgb_video_format_exact_d3d (GstVideoFormat format, D3DFORMAT * next_choice) gst_video_format_to_d3d_format (GstVideoFormat format)
{ {
D3DFORMAT ret; gint i;
g_return_val_if_fail (next_choice != NULL, D3DFMT_UNKNOWN); for (i = 0; i < G_N_ELEMENTS (gst_d3d_format_map); i++)
if (gst_d3d_format_map[i].gst_format == format)
/* return gst_d3d_format_map[i].d3d_format;
* Note: Only 1st choice is an exact match return D3DFMT_UNKNOWN;
*/
*next_choice = D3DFMT_UNKNOWN;
ret = D3DFMT_UNKNOWN;
switch (format) {
case GST_VIDEO_FORMAT_BGRx:
*next_choice = D3DFMT_X8B8G8R8;
ret = D3DFMT_X8R8G8B8;
break;
case GST_VIDEO_FORMAT_RGBx:
*next_choice = D3DFMT_X8R8G8B8;
ret = D3DFMT_X8B8G8R8;
break;
case GST_VIDEO_FORMAT_xRGB:
*next_choice = D3DFMT_X8R8G8B8;
break;
case GST_VIDEO_FORMAT_xBGR:
*next_choice = D3DFMT_X8R8G8B8;
break;
case GST_VIDEO_FORMAT_RGBA:
*next_choice = D3DFMT_A8R8G8B8;
ret = D3DFMT_A8B8G8R8;
break;
case GST_VIDEO_FORMAT_BGRA:
*next_choice = D3DFMT_A8B8G8R8;
ret = D3DFMT_A8R8G8B8;
break;
case GST_VIDEO_FORMAT_ARGB:
*next_choice = D3DFMT_A8R8G8B8;
break;
case GST_VIDEO_FORMAT_ABGR:
*next_choice = D3DFMT_A8R8G8B8;
break;
case GST_VIDEO_FORMAT_RGB:
*next_choice = D3DFMT_R8G8B8;
break;
case GST_VIDEO_FORMAT_BGR:
ret = D3DFMT_R8G8B8;
break;
case GST_VIDEO_FORMAT_RGB16:
ret = D3DFMT_R5G6B5;
break;
case GST_VIDEO_FORMAT_BGR16:
*next_choice = D3DFMT_R5G6B5;
break;
default:;
}
return ret;
} }
static gboolean static gboolean
@ -378,7 +234,7 @@ gst_video_d3d_format_check (GstD3DVideoSink * sink, D3DFORMAT fmt)
D3DDEVTYPE_HAL, class->d3d.device.format, 0, D3DRTYPE_SURFACE, fmt); D3DDEVTYPE_HAL, class->d3d.device.format, 0, D3DRTYPE_SURFACE, fmt);
if (hr == D3D_OK) { if (hr == D3D_OK) {
/* test whether device can perform color-conversion /* test whether device can perform color-conversion
** from that format to target format * from that format to target format
*/ */
hr = IDirect3D9_CheckDeviceFormatConversion (class->d3d.d3d, hr = IDirect3D9_CheckDeviceFormatConversion (class->d3d.d3d,
class->d3d.device.adapter, class->d3d.device.adapter,
@ -386,98 +242,49 @@ gst_video_d3d_format_check (GstD3DVideoSink * sink, D3DFORMAT fmt)
if (hr == D3D_OK) if (hr == D3D_OK)
ret = TRUE; ret = TRUE;
} }
//GST_DEBUG_OBJECT(sink, "Checking: %s - %s", d3d_format2string(fmt), ret?"TRUE":"FALSE"); GST_DEBUG_OBJECT (sink, "Checking: %s - %s", d3d_format_to_string (fmt),
ret ? "TRUE" : "FALSE");
return ret; return ret;
} }
static D3DFORMAT static gboolean
gst_video_query_d3d_format (GstD3DVideoSink * sink, GstVideoFormat format, gst_video_query_d3d_format (GstD3DVideoSink * sink, D3DFORMAT d3dformat)
gboolean * exact)
{ {
D3DFORMAT ret = D3DFMT_UNKNOWN;
const GstVideoFormatInfo *info;
g_return_val_if_fail (exact != NULL, D3DFMT_UNKNOWN);
*exact = FALSE;
if (!(info = gst_video_format_get_info (format))) {
GST_ERROR_OBJECT (sink, "Failed to get GstVideoFormatInfo for format: %u",
format);
return D3DFMT_UNKNOWN;
}
if (GST_VIDEO_FORMAT_INFO_IS_RGB (info)) {
GstD3DVideoSinkClass *class = GST_D3DVIDEOSINK_GET_CLASS (sink); GstD3DVideoSinkClass *class = GST_D3DVIDEOSINK_GET_CLASS (sink);
D3DFORMAT try1 = D3DFMT_UNKNOWN, try2 = D3DFMT_UNKNOWN;
if (gst_video_can_handle_gst_rgb (format)) {
try1 = gst_rgb_video_format_exact_d3d (format, &try2);
if (try1 != D3DFMT_UNKNOWN && gst_video_d3d_format_check (sink, try1)) {
ret = try1;
*exact = TRUE;
} else if (try2 != D3DFMT_UNKNOWN
&& gst_video_d3d_format_check (sink, try2)) {
ret = try2;
}
/* Fall back to display adapter format */
if (ret == D3DFMT_UNKNOWN) {
if (gst_video_can_handle_d3d_rgb (class->d3d.device.format)) {
ret = class->d3d.device.format;
}
}
}
} else if (GST_VIDEO_FORMAT_INFO_IS_YUV (info)) {
switch (format) {
case GST_VIDEO_FORMAT_YV12:
case GST_VIDEO_FORMAT_I420:
if (gst_video_d3d_format_check (sink, D3DFMT_YV12))
ret = D3DFMT_YV12;
break;
case GST_VIDEO_FORMAT_YUY2:
if (gst_video_d3d_format_check (sink, D3DFMT_YUY2)) {
ret = D3DFMT_YUY2;
}
break;
case GST_VIDEO_FORMAT_UYVY:
if (gst_video_d3d_format_check (sink, D3DFMT_UYVY)) {
ret = D3DFMT_UYVY;
}
break;
case GST_VIDEO_FORMAT_NV12:
if (gst_video_d3d_format_check (sink, D3DFMT_NV12)) {
ret = D3DFMT_NV12;
}
break;
default:;
}
}
return ret; /* If it's the display adapter format we don't need to probe */
if (d3dformat == class->d3d.device.format)
return TRUE;
if (gst_video_d3d_format_check (sink, d3dformat))
return TRUE;
return FALSE;
} }
typedef struct typedef struct
{ {
GstVideoFormat fmt; GstVideoFormat fmt;
D3DFORMAT d3d_fmt; D3DFORMAT d3d_fmt;
gboolean exact;
gboolean display; gboolean display;
} GstFormatComp; } D3DFormatComp;
static void
d3d_format_comp_free (D3DFormatComp * comp)
{
g_slice_free (D3DFormatComp, comp);
}
static gint static gint
format_points (GstFormatComp * dat) d3d_format_comp_rate (const D3DFormatComp * comp)
{ {
gint points = 0; gint points = 0;
const GstVideoFormatInfo *info; const GstVideoFormatInfo *info;
if (!(info = gst_video_format_get_info (dat->fmt))) info = gst_video_format_get_info (comp->fmt);
return 0;
if (dat->display) if (comp->display)
points += 1;
else if (dat->exact)
points += 2;
if (dat->exact && dat->display)
points += 10; points += 10;
if (GST_VIDEO_FORMAT_INFO_IS_YUV (info)) if (GST_VIDEO_FORMAT_INFO_IS_YUV (info))
points += 5; points += 5;
@ -488,15 +295,18 @@ format_points (GstFormatComp * dat)
if (bit_depth >= 24) if (bit_depth >= 24)
points += 1; points += 1;
} }
return points; return points;
} }
static gint static gint
format_compare_cb (gconstpointer a, gconstpointer b) d3d_format_comp_compare (gconstpointer a, gconstpointer b)
{ {
gint ptsa = 0, ptsb = 0; gint ptsa = 0, ptsb = 0;
ptsa = format_points ((GstFormatComp *) a);
ptsb = format_points ((GstFormatComp *) b); ptsa = d3d_format_comp_rate ((const D3DFormatComp *) a);
ptsb = d3d_format_comp_rate ((const D3DFormatComp *) b);
if (ptsa < ptsb) if (ptsa < ptsb)
return -1; return -1;
else if (ptsa == ptsb) else if (ptsa == ptsb)
@ -505,30 +315,17 @@ format_compare_cb (gconstpointer a, gconstpointer b)
return 1; return 1;
} }
static GstCaps *
d3dvideosink_format_new_template_caps (GstVideoFormat fmt)
{
gchar *tmp = g_strdup_printf ("video/x-raw, "
"format=(string)%s, "
"width=(int)[ 1, 2147483647 ], "
"height=(int)[ 1, 2147483647 ], "
"framerate=(fraction)[ 0/1, 2147483647/1 ]",
gst_video_format_to_string (fmt));
//GST_DEBUG("CAPS: %s", tmp);
return gst_caps_from_string (tmp);
}
GstCaps * GstCaps *
d3d_supported_caps (GstD3DVideoSink * sink) d3d_supported_caps (GstD3DVideoSink * sink)
{ {
GstD3DVideoSinkClass *class = GST_D3DVIDEOSINK_GET_CLASS (sink); GstD3DVideoSinkClass *class = GST_D3DVIDEOSINK_GET_CLASS (sink);
int i; int i;
gboolean exact; GList *fmts = NULL, *l;
GList *lst = NULL, *lsto = NULL;
GstCaps *caps = NULL; GstCaps *caps = NULL;
GstVideoFormat gst_format;
D3DFORMAT d3d_format; D3DFORMAT d3d_format;
gchar *tmp = NULL; GValue va = { 0, };
GValue v = { 0, };
LOCK_SINK (sink); LOCK_SINK (sink);
@ -537,43 +334,54 @@ d3d_supported_caps (GstD3DVideoSink * sink)
goto unlock; goto unlock;
} }
for (i = 0; i <= GST_VIDEO_FORMAT_GBR_10LE; i++) { for (i = 0; i < G_N_ELEMENTS (gst_d3d_format_map); i++) {
GstFormatComp *dat; D3DFormatComp *comp;
d3d_format = gst_video_query_d3d_format (sink, (GstVideoFormat) i, &exact); gst_format = gst_d3d_format_map[i].gst_format;
d3d_format = gst_d3d_format_map[i].d3d_format;
if (d3d_format == D3DFMT_UNKNOWN) if (!gst_video_query_d3d_format (sink, d3d_format))
continue; continue;
dat = g_new0 (GstFormatComp, 1); comp = g_slice_new0 (D3DFormatComp);
dat->fmt = (GstVideoFormat) i; comp->fmt = (GstVideoFormat) gst_format;
dat->d3d_fmt = d3d_format; comp->d3d_fmt = d3d_format;
dat->exact = exact; comp->display = (d3d_format == class->d3d.device.format);
dat->display = (d3d_format == class->d3d.device.format); fmts = g_list_insert_sorted (fmts, comp, d3d_format_comp_compare);
lst = g_list_insert_sorted (lst, dat, format_compare_cb);
} }
GST_DEBUG_OBJECT (sink, "Supported Caps:"); GST_DEBUG_OBJECT (sink, "Supported Caps:");
lsto = lst; g_value_init (&va, GST_TYPE_LIST);
for (lst = g_list_last (lst); lst != NULL; lst = g_list_previous (lst)) { g_value_init (&v, G_TYPE_STRING);
GstFormatComp *dat = (GstFormatComp *) lst->data;
GST_DEBUG_OBJECT (sink, "%s -> %s %s%s", for (l = fmts; l; l = g_list_next (l)) {
gst_video_format2string (dat->fmt), d3d_format2string (dat->d3d_fmt), D3DFormatComp *comp = (D3DFormatComp *) l->data;
dat->exact ? "[exact]" : "", dat->display ? "[display]" : "");
if (!caps) GST_DEBUG_OBJECT (sink, "%s -> %s %s",
caps = d3dvideosink_format_new_template_caps (dat->fmt); gst_video_format_to_string (comp->fmt),
else d3d_format_to_string (comp->d3d_fmt), comp->display ? "[display]" : "");
gst_caps_append (caps, d3dvideosink_format_new_template_caps (dat->fmt)); g_value_set_string (&v, gst_video_format_to_string (comp->fmt));
gst_value_list_append_value (&va, &v);
} }
g_list_foreach (lsto, (GFunc) g_free, NULL);
g_list_free (lsto); caps = gst_caps_new_simple ("video/x-raw",
"width", GST_TYPE_INT_RANGE, 1, G_MAXINT,
"height", GST_TYPE_INT_RANGE, 1, G_MAXINT,
"framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1, NULL);
gst_caps_set_value (caps, "format", &va);
g_value_unset (&v);
g_value_unset (&va);
g_list_free_full (fmts, (GDestroyNotify) d3d_format_comp_free);
sink->supported_caps = gst_caps_ref (caps); sink->supported_caps = gst_caps_ref (caps);
GST_DEBUG_OBJECT (sink, "Supported caps: %s", (tmp = #ifndef GST_DISABLE_GST_DEBUG
gst_caps_to_string (caps))); {
gchar *tmp = gst_caps_to_string (caps);
GST_DEBUG_OBJECT (sink, "Supported caps: %s", tmp);
g_free (tmp); g_free (tmp);
}
#endif
unlock: unlock:
UNLOCK_SINK (sink); UNLOCK_SINK (sink);
@ -585,24 +393,27 @@ gboolean
d3d_set_render_format (GstD3DVideoSink * sink) d3d_set_render_format (GstD3DVideoSink * sink)
{ {
D3DFORMAT fmt; D3DFORMAT fmt;
gboolean exact = FALSE;
gboolean ret = FALSE; gboolean ret = FALSE;
LOCK_SINK (sink); LOCK_SINK (sink);
fmt = gst_video_query_d3d_format (sink, sink->format, &exact); fmt = gst_video_format_to_d3d_format (sink->format);
if (fmt == D3DFMT_UNKNOWN) { if (fmt == D3DFMT_UNKNOWN) {
GST_ERROR_OBJECT (sink, "Failed to query a D3D render format for %s", GST_ERROR_OBJECT (sink, "Unsupported video format %s",
gst_video_format2string (sink->format)); gst_video_format_to_string (sink->format));
goto end; goto end;
} }
GST_DEBUG_OBJECT (sink, "Selected %s -> %s %s", if (!gst_video_query_d3d_format (sink, fmt)) {
gst_video_format2string (sink->format), d3d_format2string (fmt), GST_ERROR_OBJECT (sink, "Failed to query a D3D render format for %s",
exact ? "(exact)" : ""); gst_video_format_to_string (sink->format));
goto end;
}
GST_DEBUG_OBJECT (sink, "Selected %s -> %s",
gst_video_format_to_string (sink->format), d3d_format_to_string (fmt));
sink->d3d.format = fmt; sink->d3d.format = fmt;
sink->d3d.exact_copy = exact;
ret = TRUE; ret = TRUE;
@ -688,9 +499,9 @@ d3d_get_render_coordinates (GstD3DVideoSink * sink, gint in_x, gint in_y,
/* Convert window coordinates to source frame pixel coordinates */ /* Convert window coordinates to source frame pixel coordinates */
if (sink->force_aspect_ratio) { if (sink->force_aspect_ratio) {
GstVideoRectangle tmp = { 0, 0, 0, 0 } GstVideoRectangle tmp = { 0, 0, 0, 0 };
, dst = { GstVideoRectangle dst = { 0, 0, 0, 0 };
0, 0, 0, 0};
tmp.w = GST_VIDEO_SINK_WIDTH (sink); tmp.w = GST_VIDEO_SINK_WIDTH (sink);
tmp.h = GST_VIDEO_SINK_HEIGHT (sink); tmp.h = GST_VIDEO_SINK_HEIGHT (sink);
gst_video_sink_center_rect (tmp, r_area, &dst, TRUE); gst_video_sink_center_rect (tmp, r_area, &dst, TRUE);
@ -1235,17 +1046,18 @@ static gboolean
d3d_copy_buffer_to_surface (GstD3DVideoSink * sink, GstBuffer * buffer) d3d_copy_buffer_to_surface (GstD3DVideoSink * sink, GstBuffer * buffer)
{ {
D3DLOCKED_RECT lr; D3DLOCKED_RECT lr;
guint8 *dest, *source; guint8 *dest;
int i; int deststride;
gboolean ret = FALSE; gboolean ret = FALSE;
gint unhdl_line = 0; gint unhdl_line = 0;
GstMapInfo map; GstVideoFrame frame;
LOCK_SINK (sink); LOCK_SINK (sink);
if (!sink->d3d.renderable || sink->d3d.device_lost) if (!sink->d3d.renderable || sink->d3d.device_lost)
goto end; goto end;
if (!buffer || !gst_buffer_map (buffer, &map, GST_MAP_READ)) { if (!buffer
|| !gst_video_frame_map (&frame, &sink->info, buffer, GST_MAP_READ)) {
GST_ERROR_OBJECT (sink, "NULL GstBuffer"); GST_ERROR_OBJECT (sink, "NULL GstBuffer");
goto end; goto end;
} }
@ -1254,210 +1066,183 @@ d3d_copy_buffer_to_surface (GstD3DVideoSink * sink, GstBuffer * buffer)
IDirect3DSurface9_LockRect (sink->d3d.surface, &lr, NULL, 0); IDirect3DSurface9_LockRect (sink->d3d.surface, &lr, NULL, 0);
dest = (guint8 *) lr.pBits; dest = (guint8 *) lr.pBits;
source = map.data;
if (!dest) { if (!dest) {
GST_ERROR_OBJECT (sink, "No D3D surface dest buffer"); GST_ERROR_OBJECT (sink, "No D3D surface dest buffer");
goto unlock_surface; goto unlock_surface;
} }
if (sink->d3d.exact_copy) { deststride = lr.Pitch;
memcpy (dest, source, gst_buffer_get_size (buffer));
goto done;
}
if (GST_VIDEO_INFO_IS_YUV (&sink->info)) {
switch (sink->format) { switch (sink->format) {
case GST_VIDEO_FORMAT_YUY2: case GST_VIDEO_FORMAT_YUY2:
case GST_VIDEO_FORMAT_UYVY:{ case GST_VIDEO_FORMAT_UYVY:{
int srcstride, dststride; const guint8 *src;
dststride = lr.Pitch; gint srcstride;
srcstride = gst_buffer_get_size (buffer) / GST_VIDEO_SINK_HEIGHT (sink); gint i, h, w;
for (i = 0; i < GST_VIDEO_SINK_HEIGHT (sink); ++i)
memcpy (dest + dststride * i, source + srcstride * i, srcstride); src = GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
srcstride = GST_VIDEO_FRAME_PLANE_STRIDE (&frame, 0);
h = GST_VIDEO_FRAME_HEIGHT (&frame);
w = GST_ROUND_UP_4 (GST_VIDEO_FRAME_WIDTH (&frame) * 2);
for (i = 0; i < h; i++) {
memcpy (dest, src, w);
dest += deststride;
src += srcstride;
}
break; break;
} }
case GST_VIDEO_FORMAT_YV12: case GST_VIDEO_FORMAT_I420:
case GST_VIDEO_FORMAT_I420:{ case GST_VIDEO_FORMAT_YV12:{
int srcystride, srcvstride, srcustride; const guint8 *src;
int dstystride, dstvstride, dstustride; gint srcstride, deststride_;
guint8 *srcv, *srcu, *dstv, *dstu; guint8 *dest_;
int rows; gint i, j, h, h_, w_;
rows = sink->height; h = GST_VIDEO_FRAME_HEIGHT (&frame);
/* Source y, u and v strides */ for (i = 0; i < 3; i++) {
srcystride = GST_ROUND_UP_4 (sink->width); src = GST_VIDEO_FRAME_COMP_DATA (&frame, i);
srcustride = GST_ROUND_UP_8 (sink->width) / 2; srcstride = GST_VIDEO_FRAME_COMP_STRIDE (&frame, i);
srcvstride = GST_ROUND_UP_8 (srcystride) / 2; h_ = GST_VIDEO_FRAME_COMP_HEIGHT (&frame, i);
w_ = GST_VIDEO_FRAME_COMP_WIDTH (&frame, i);
/* Destination y, u and v strides */ switch (i) {
dstystride = lr.Pitch; case 0:
dstustride = dstystride / 2; deststride_ = deststride;
dstvstride = dstustride; dest_ = dest;
break;
srcu = source + srcystride * GST_ROUND_UP_2 (rows); case 2:
srcv = srcu + srcustride * GST_ROUND_UP_2 (rows) / 2; deststride_ = deststride / 2;
dest_ = dest + h * deststride;
if (sink->format == GST_VIDEO_FORMAT_I420) { break;
/* swap u and v planes */ case 1:
dstv = dest + dstystride * rows; deststride_ = deststride / 2;
dstu = dstv + dstustride * rows / 2; dest_ = dest + h * deststride + h_ * deststride_;
} else { break;
dstu = dest + dstystride * rows;
dstv = dstu + dstustride * rows / 2;
} }
for (i = 0; i < rows; ++i) { for (j = 0; j < h_; j++) {
/* Copy the y plane */ memcpy (dest_, src, w_);
memcpy (dest + (dstystride * i), source + (srcystride * i), dest_ += deststride_;
srcystride); src += srcstride;
}
} }
for (i = 0; i < (rows / 2); ++i) { break;
/* Copy the u plane */
memcpy (dstu + (dstustride * i), srcu + (srcustride * i), srcustride);
/* Copy the v plane */
memcpy (dstv + (dstvstride * i), srcv + (srcvstride * i), srcvstride);
}
goto done;
} }
case GST_VIDEO_FORMAT_NV12:{ case GST_VIDEO_FORMAT_NV12:{
int srcstride, dststride; const guint8 *src;
guint8 *dst = dest; gint srcstride;
int component; guint8 *dest_;
dststride = lr.Pitch; gint i, j, h, h_, w_;
for (component = 0; component < 2; component++) {
const int compHeight =
GST_VIDEO_INFO_COMP_HEIGHT (&sink->info, component);
guint8 *src = source + GST_VIDEO_INFO_COMP_OFFSET (&sink->info,
component);
srcstride = GST_VIDEO_INFO_COMP_STRIDE (&sink->info, component);
for (i = 0; i < compHeight; i++) {
memcpy (dst + dststride * i, src + srcstride * i, srcstride);
}
dst += dststride * compHeight;
}
break;
}
default:;
unhdl_line = __LINE__;
goto unhandled_format;
} /* end switch */
} else if (GST_VIDEO_INFO_IS_RGB (&sink->info)) {
for (i = 0; i < sink->height * sink->width; i++) {
guint8 p[4] = { 0, 0, 0, 0 };
/* Input */ h = GST_VIDEO_FRAME_HEIGHT (&frame);
switch (sink->fmt_details.bpp) {
case 32: for (i = 0; i < 2; i++) {
case 24:{ src = GST_VIDEO_FRAME_PLANE_DATA (&frame, i);
guint32 *p32 = srcstride = GST_VIDEO_FRAME_PLANE_STRIDE (&frame, i);
(guint32 *) (source + (i * sink->fmt_details.pixel_width)); h_ = GST_VIDEO_FRAME_COMP_HEIGHT (&frame, i);
if (sink->fmt_details.a_shift) w_ = GST_VIDEO_FRAME_COMP_WIDTH (&frame, i);
p[ALPHA] =
(*p32 & sink->fmt_details.a_mask) >> sink->fmt_details.a_shift; switch (i) {
p[RED] = case 0:
(*p32 & sink->fmt_details.r_mask) >> sink->fmt_details.r_shift; dest_ = dest;
p[GREEN] =
(*p32 & sink->fmt_details.g_mask) >> sink->fmt_details.g_shift;
p[BLUE] =
(*p32 & sink->fmt_details.b_mask) >> sink->fmt_details.b_shift;
break; break;
} case 1:
case 16:{ dest_ = dest + h * deststride;
guint16 *p16 =
(guint16 *) (source + (i * sink->fmt_details.pixel_width));
if (sink->fmt_details.a_shift)
p[ALPHA] =
(*p16 & sink->fmt_details.a_mask16) >> sink->fmt_details.
a_shift;
p[RED] =
(*p16 & sink->fmt_details.r_mask16) >> sink->fmt_details.r_shift;
p[GREEN] =
(*p16 & sink->fmt_details.g_mask16) >> sink->fmt_details.g_shift;
p[BLUE] =
(*p16 & sink->fmt_details.b_mask16) >> sink->fmt_details.b_shift;
break; break;
} }
default:;
unhdl_line = __LINE__;
goto unhandled_format;
}
/* Output */ for (j = 0; j < h_; j++) {
switch (sink->d3d.format) { memcpy (dest_, src, w_ * 2);
case D3DFMT_A8R8G8B8: dest_ += deststride;
p[RED] = BIT_CONV (p[RED], sink->fmt_details.r_bits, 8); src += srcstride;
p[GREEN] = BIT_CONV (p[GREEN], sink->fmt_details.g_bits, 8); }
p[BLUE] = BIT_CONV (p[BLUE], sink->fmt_details.b_bits, 8); }
if (sink->fmt_details.a_shift)
p[ALPHA] = BIT_CONV (p[ALPHA], sink->fmt_details.a_bits, 8); break;
else }
p[ALPHA] = 255; case GST_VIDEO_FORMAT_BGRA:
memcpy ((dest + (i * 4)), &p, 4); case GST_VIDEO_FORMAT_RGBA:
break; case GST_VIDEO_FORMAT_BGRx:
case D3DFMT_X8R8G8B8: case GST_VIDEO_FORMAT_RGBx:{
p[RED] = BIT_CONV (p[RED], sink->fmt_details.r_bits, 8); const guint8 *src;
p[GREEN] = BIT_CONV (p[GREEN], sink->fmt_details.g_bits, 8); gint srcstride;
p[BLUE] = BIT_CONV (p[BLUE], sink->fmt_details.b_bits, 8); gint i, h, w;
memcpy ((dest + (i * 4)), &p, 4);
break; src = GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
case D3DFMT_A8B8G8R8: srcstride = GST_VIDEO_FRAME_PLANE_STRIDE (&frame, 0);
*(dest + (i * 4) + 0) = p[RED]; h = GST_VIDEO_FRAME_HEIGHT (&frame);
*(dest + (i * 4) + 1) = p[GREEN]; w = GST_VIDEO_FRAME_WIDTH (&frame) * 4;
*(dest + (i * 4) + 2) = p[BLUE];
if (sink->fmt_details.a_shift) for (i = 0; i < h; i++) {
*(dest + (i * 4) + 3) = p[ALPHA]; memcpy (dest, src, w);
else dest += deststride;
*(dest + (i * 4) + 3) = 255; src += srcstride;
break; }
case D3DFMT_X8B8G8R8:
*(dest + (i * 4) + 0) = p[RED]; break;
*(dest + (i * 4) + 1) = p[GREEN]; }
*(dest + (i * 4) + 2) = p[BLUE]; case GST_VIDEO_FORMAT_BGR:{
break; const guint8 *src;
case D3DFMT_R8G8B8: gint srcstride;
p[RED] = BIT_CONV (p[RED], sink->fmt_details.r_bits, 8); gint i, h, w;
p[GREEN] = BIT_CONV (p[GREEN], sink->fmt_details.g_bits, 8);
p[BLUE] = BIT_CONV (p[BLUE], sink->fmt_details.b_bits, 8); src = GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
memcpy ((dest + (i * 3)), &p, 3); srcstride = GST_VIDEO_FRAME_PLANE_STRIDE (&frame, 0);
break; h = GST_VIDEO_FRAME_HEIGHT (&frame);
case D3DFMT_R5G6B5:{ w = GST_VIDEO_FRAME_WIDTH (&frame) * 3;
p[RED] = BIT_CONV (p[RED], sink->fmt_details.r_bits, 5);
p[GREEN] = BIT_CONV (p[GREEN], sink->fmt_details.g_bits, 6); for (i = 0; i < h; i++) {
p[BLUE] = BIT_CONV (p[BLUE], sink->fmt_details.b_bits, 5); memcpy (dest, src, w);
*((guint16 *) (dest + (i * 2))) = dest += deststride;
(p[RED] << 11 | p[GREEN] << 5 | p[BLUE]); src += srcstride;
}
break;
}
case GST_VIDEO_FORMAT_RGB16:
case GST_VIDEO_FORMAT_RGB15:{
const guint8 *src;
gint srcstride;
gint i, h, w;
src = GST_VIDEO_FRAME_PLANE_DATA (&frame, 0);
srcstride = GST_VIDEO_FRAME_PLANE_STRIDE (&frame, 0);
h = GST_VIDEO_FRAME_HEIGHT (&frame);
w = GST_VIDEO_FRAME_WIDTH (&frame) * 2;
for (i = 0; i < h; i++) {
memcpy (dest, src, w);
dest += deststride;
src += srcstride;
}
break; break;
} }
case D3DFMT_X1R5G5B5:
case D3DFMT_A1R5G5B5:
case D3DFMT_X4R4G4B4:
case D3DFMT_A4R4G4B4:
case D3DFMT_A8R3G3B2:
case D3DFMT_R3G3B2:
//break;
default: default:
unhdl_line = __LINE__; unhdl_line = __LINE__;
goto unhandled_format; goto unhandled_format;
} }
} /* end for */
}
/* end if rgb */
goto done; goto done;
unhandled_format: unhandled_format:
GST_ERROR_OBJECT (sink, GST_ERROR_OBJECT (sink,
"Unhandled format [LN:%d] '%s' -> '%s' (should not get here)", unhdl_line, "Unhandled format [LN:%d] '%s' -> '%s' (should not get here)", unhdl_line,
gst_video_format2string (sink->format), gst_video_format_to_string (sink->format),
d3d_format2string (sink->d3d.format)); d3d_format_to_string (sink->d3d.format));
goto unlock_surface; goto unlock_surface;
done: done:
ret = TRUE; ret = TRUE;
unlock_surface: unlock_surface:
IDirect3DSurface9_UnlockRect (sink->d3d.surface); IDirect3DSurface9_UnlockRect (sink->d3d.surface);
gst_video_frame_unmap (&frame);
end: end:
UNLOCK_SINK (sink); UNLOCK_SINK (sink);
return ret; return ret;
@ -1608,6 +1393,7 @@ d3d_stretch_and_copy (GstD3DVideoSink * sink, LPDIRECT3DSURFACE9 back_buffer)
back_buffer, /* Dest Surface */ back_buffer, /* Dest Surface */
r_ptr, /* Dest Surface Rect (NULL: Whole) */ r_ptr, /* Dest Surface Rect (NULL: Whole) */
class->d3d.device.filter_type); class->d3d.device.filter_type);
if (hr == D3D_OK) { if (hr == D3D_OK) {
ret = TRUE; ret = TRUE;
} else { } else {
@ -2070,8 +1856,6 @@ d3d_class_destroy (GstD3DVideoSink * sink)
UnregisterClass (class->d3d.wnd_class.lpszClassName, UnregisterClass (class->d3d.wnd_class.lpszClassName,
class->d3d.wnd_class.hInstance); class->d3d.wnd_class.hInstance);
g_list_free (class->d3d.sink_list);
memset (&class->d3d, 0, sizeof (GstD3DDataClass)); memset (&class->d3d, 0, sizeof (GstD3DDataClass));
end: end:
@ -2167,7 +1951,8 @@ d3d_class_display_device_create (GstD3DVideoSinkClass * class, UINT adapter)
goto error; goto error;
} }
GST_DEBUG ("Display Device format: %s", d3d_format2string (disp_mode.Format)); GST_DEBUG ("Display Device format: %s",
d3d_format_to_string (disp_mode.Format));
ret = TRUE; ret = TRUE;
goto end; goto end;

View file

@ -20,7 +20,10 @@
*/ */
#ifndef _D3DHELPERS_H_ #ifndef _D3DHELPERS_H_
#define _D3DHELPERS_H_ #define _D3DHELPERS_H_
#include <glib.h>
#include <gst/gst.h>
#include <gst/video/video.h>
#include <windows.h> #include <windows.h>
#include <d3d9.h> #include <d3d9.h>
#include <d3dx9tex.h> #include <d3dx9tex.h>
@ -66,7 +69,6 @@ typedef struct _GstD3DData {
LPDIRECT3DSURFACE9 surface; LPDIRECT3DSURFACE9 surface;
D3DTEXTUREFILTERTYPE filtertype; D3DTEXTUREFILTERTYPE filtertype;
D3DFORMAT format; D3DFORMAT format;
gboolean exact_copy;
GstVideoRectangle * render_rect; GstVideoRectangle * render_rect;
gboolean renderable; gboolean renderable;
gboolean device_lost; gboolean device_lost;

View file

@ -2,6 +2,7 @@
* Copyright (C) 2012 Roland Krikava <info@bluedigits.com> * Copyright (C) 2012 Roland Krikava <info@bluedigits.com>
* Copyright (C) 2010-2011 David Hoyt <dhoyt@hoytsoft.org> * Copyright (C) 2010-2011 David Hoyt <dhoyt@hoytsoft.org>
* Copyright (C) 2010 Andoni Morales <ylatuya@gmail.com> * Copyright (C) 2010 Andoni Morales <ylatuya@gmail.com>
* Copyright (C) 2012 Sebastian Dröge <sebastian.droege@collabora.co.uk>
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public * modify it under the terms of the GNU Library General Public
@ -45,7 +46,7 @@ static GstStaticPadTemplate sink_template = GST_STATIC_PAD_TEMPLATE ("sink",
GST_PAD_SINK, GST_PAD_SINK,
GST_PAD_ALWAYS, GST_PAD_ALWAYS,
GST_STATIC_CAPS ("video/x-raw, " GST_STATIC_CAPS ("video/x-raw, "
"format = (string) { I420, YV12, UYVY, YUY2, NV12, BGRx, RGBx, xRGB, xBGR, RGBA, BGRA, ARGB, ABGR, RGB, BGR, RGB16, BGR16, RGB15, BGR15 }, " "format = (string) { I420, YV12, UYVY, YUY2, NV12, BGRx, RGBx, RGBA, BGRA, BGR, RGB16, RGB15 }, "
"framerate = (fraction) [ 0, MAX ], " "framerate = (fraction) [ 0, MAX ], "
"width = (int) [ 1, MAX ], " "height = (int) [ 1, MAX ]") "width = (int) [ 1, MAX ], " "height = (int) [ 1, MAX ]")
); );
@ -260,271 +261,6 @@ gst_d3dvideosink_get_caps (GstBaseSink * basesink, GstCaps * filter)
return caps; return caps;
} }
static guint16
flip_b16 (guint16 b)
{
guint16 ret = 0, tmp = 0x8000;
gint i;
for (i = 0, tmp = 1; tmp != 0; i++) {
//printf("%x\n", tmp);
if (b & tmp)
ret |= (0x8000 >> i);
tmp <<= 1;
}
return ret;
}
static guint32
flip_b32 (guint32 b, gboolean bitshift)
{
guint32 ret = 0, tmp = 0x80000000;
gint i;
for (i = 0, tmp = 1; tmp != 0; i++) {
//printf("%x\n", tmp);
if (b & tmp)
ret |= (0x80000000 >> i);
tmp <<= 1;
}
if (bitshift && G_BYTE_ORDER == G_LITTLE_ENDIAN)
ret >>= 8;
return ret;
}
typedef enum
{
VFMT_RGBx = 0,
VFMT_BGRx,
VFMT_xRGB,
VFMT_xBGR,
VFMT_RGBA,
VFMT_BGRA,
VFMT_ARGB,
VFMT_ABGR,
VFMT_RGB,
VFMT_BGR,
VFMT_RGB16,
VFMT_BGR16,
VFMT_RGB15,
VFMT_BGR15,
} VFmtMap;
static GstVideoFormatDetails vfmt_details[] = {
{ // GST_VIDEO_FORMAT_RGBx
0xff000000, 0x00ff0000, 0x0000ff00, 0,
0, 0, 0, 0,
0, 8, 16, 0,
8, 8, 8, 0,
32, 24, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_BGRx
0x0000ff00, 0x00ff0000, 0xff000000, 0,
0, 0, 0, 0,
16, 8, 0, 0,
8, 8, 8, 0,
32, 24, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_xRGB
0x00ff0000, 0x0000ff00, 0x000000ff, 0,
0, 0, 0, 0,
8, 16, 24, 0,
8, 8, 8, 0,
32, 24, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_xBGR
0x000000ff, 0x0000ff00, 0x00ff0000, 0,
0, 0, 0, 0,
24, 16, 8, 0,
8, 8, 8, 0,
32, 24, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_RGBA
0xff000000, 0x00ff0000, 0x0000ff00, 0x000000ff,
0, 0, 0, 0,
0, 8, 16, 24,
8, 8, 8, 8,
32, 32, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_BGRA
0x0000ff00, 0x00ff0000, 0xff000000, 0x000000ff,
0, 0, 0, 0,
16, 8, 0, 24,
8, 8, 8, 8,
32, 32, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_ARGB
0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000,
0, 0, 0, 0,
8, 16, 24, 0,
8, 8, 8, 8,
32, 32, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_ABGR
0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000,
0, 0, 0, 0,
24, 16, 8, 0,
8, 8, 8, 8,
32, 32, G_BIG_ENDIAN, 4},
{ // GST_VIDEO_FORMAT_RGB
0x00ff0000, 0x0000ff00, 0x000000ff, 0,
0, 0, 0, 0,
0, 8, 16, 0,
8, 8, 8, 0,
24, 24, G_BIG_ENDIAN, 3},
{ // GST_VIDEO_FORMAT_BGR
0x000000ff, 0x0000ff00, 0x00ff0000, 0,
0, 0, 0, 0,
16, 8, 0, 0,
8, 8, 8, 0,
24, 24, G_BIG_ENDIAN, 3},
{ // GST_VIDEO_FORMAT_RGB16
0, 0, 0, 0,
0xf800, 0x07e0, 0x001f, 0,
11, 5, 0, 0,
5, 6, 5, 0,
16, 16, G_LITTLE_ENDIAN, 2},
{ // GST_VIDEO_FORMAT_BGR16
0, 0, 0, 0,
0x001f, 0x07e0, 0xf800, 0,
0, 5, 11, 0,
5, 6, 5, 0,
16, 16, G_LITTLE_ENDIAN, 2},
{ // GST_VIDEO_FORMAT_RGB15
0, 0, 0, 0,
0x7c00, 0x03e0, 0x001f, 0,
10, 5, 0, 0,
5, 5, 5, 0,
16, 15, G_LITTLE_ENDIAN, 2},
{ // GST_VIDEO_FORMAT_BGR15
0, 0, 0, 0,
0x001f, 0x03e0, 0x7c00, 0,
0, 5, 10, 0,
5, 5, 5, 0,
16, 15, G_LITTLE_ENDIAN, 2}
};
static gboolean
gst_video_format_get_rgb_masks (GstD3DVideoSink * sink, GstVideoFormat fmt,
GstVideoFormatDetails * details)
{
gboolean ret = FALSE;
g_return_val_if_fail (details != NULL, FALSE);
switch (fmt) {
case GST_VIDEO_FORMAT_RGBx:
memcpy (details, &vfmt_details[VFMT_RGBx],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_BGRx:
memcpy (details, &vfmt_details[VFMT_BGRx],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_xRGB:
memcpy (details, &vfmt_details[VFMT_xRGB],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_xBGR:
memcpy (details, &vfmt_details[VFMT_xBGR],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_RGBA:
memcpy (details, &vfmt_details[VFMT_RGBA],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_BGRA:
memcpy (details, &vfmt_details[VFMT_BGRA],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_ARGB:
memcpy (details, &vfmt_details[VFMT_ARGB],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_ABGR:
memcpy (details, &vfmt_details[VFMT_ABGR],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_RGB:
memcpy (details, &vfmt_details[VFMT_RGB], sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_BGR:
memcpy (details, &vfmt_details[VFMT_BGR], sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_RGB16:
memcpy (details, &vfmt_details[VFMT_RGB16],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_BGR16:
memcpy (details, &vfmt_details[VFMT_BGR16],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_RGB15:
memcpy (details, &vfmt_details[VFMT_RGB15],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
case GST_VIDEO_FORMAT_BGR15:
memcpy (details, &vfmt_details[VFMT_BGR15],
sizeof (GstVideoFormatDetails));
ret = TRUE;
break;
default:;
}
if (ret) {
if (details->endianness != G_BYTE_ORDER) {
gboolean bitshift = (details->bpp == 24 && details->depth == 24);
GST_DEBUG_OBJECT (sink, "Flipping masks%s, byte order missmatch",
bitshift ? " (w/ bitshift)" : "");
details->r_mask = flip_b32 (details->r_mask, bitshift);
details->g_mask = flip_b32 (details->g_mask, bitshift);
details->b_mask = flip_b32 (details->b_mask, bitshift);
details->a_mask = flip_b32 (details->a_mask, bitshift);
details->r_mask16 = flip_b16 (details->r_mask16);
details->g_mask16 = flip_b16 (details->g_mask16);
details->b_mask16 = flip_b16 (details->b_mask16);
details->a_mask16 = flip_b16 (details->a_mask16);
}
if (details->bpp == 16) {
GST_DEBUG_OBJECT (sink, "RED MASK: 0x%04x SHIFT: %2u BITS: %u (%u)",
details->r_mask16, details->r_shift, details->r_bits,
details->r_mask16);
GST_DEBUG_OBJECT (sink, "GREEN MASK: 0x%04x SHIFT: %2u BITS: %u (%u)",
details->g_mask16, details->g_shift, details->g_bits,
details->g_mask16);
GST_DEBUG_OBJECT (sink, "BLUE MASK: 0x%04x SHIFT: %2u BITS: %u (%u)",
details->b_mask16, details->b_shift, details->b_bits,
details->b_mask16);
GST_DEBUG_OBJECT (sink, "ALPHA MASK: 0x%04x SHIFT: %2u BITS: %u (%u)",
details->a_mask16, details->a_shift, details->a_bits,
details->a_mask16);
} else {
GST_DEBUG_OBJECT (sink, "RED MASK: 0x%08x SHIFT: %2u BITS: %u (%u)",
details->r_mask, details->r_shift, details->r_bits, details->r_mask);
GST_DEBUG_OBJECT (sink, "GREEN MASK: 0x%08x SHIFT: %2u BITS: %u (%u)",
details->g_mask, details->g_shift, details->g_bits, details->g_mask);
GST_DEBUG_OBJECT (sink, "BLUE MASK: 0x%08x SHIFT: %2u BITS: %u (%u)",
details->b_mask, details->b_shift, details->b_bits, details->b_mask);
GST_DEBUG_OBJECT (sink, "ALPHA MASK: 0x%08x SHIFT: %2u BITS: %u (%u)",
details->a_mask, details->a_shift, details->a_bits, details->a_mask);
}
GST_DEBUG_OBJECT (sink, "ENDIANESS: %s",
(details->endianness ==
G_BIG_ENDIAN) ? "G_BIG_ENDIAN" : "G_LITTLE_ENDIAN");
GST_DEBUG_OBJECT (sink, "PIXEL WIDTH: %d", details->pixel_width);
}
return ret;
}
static gboolean static gboolean
gst_d3dvideosink_set_caps (GstBaseSink * bsink, GstCaps * caps) gst_d3dvideosink_set_caps (GstBaseSink * bsink, GstCaps * caps)
{ {
@ -560,15 +296,6 @@ gst_d3dvideosink_set_caps (GstBaseSink * bsink, GstCaps * caps)
GST_DEBUG_OBJECT (bsink, "Set Caps Format: %s", GST_DEBUG_OBJECT (bsink, "Set Caps Format: %s",
gst_video_format_to_string (sink->format)); gst_video_format_to_string (sink->format));
if (GST_VIDEO_INFO_IS_RGB (&sink->info)) {
if (!gst_video_format_get_rgb_masks (sink, sink->format,
&sink->fmt_details)) {
GST_ERROR_OBJECT (sink, "No RGB mapping found for format: %s",
gst_video_format_to_string (sink->format));
goto incompatible_caps;
}
}
/* get aspect ratio from caps if it's present, and /* get aspect ratio from caps if it's present, and
* convert video width and height to a display width and height * convert video width and height to a display width and height
* using wd / hd = wv / hv * PARv / PARd */ * using wd / hd = wv / hv * PARv / PARd */

View file

@ -41,29 +41,6 @@ G_BEGIN_DECLS
typedef struct _GstD3DVideoSink GstD3DVideoSink; typedef struct _GstD3DVideoSink GstD3DVideoSink;
typedef struct _GstD3DVideoSinkClass GstD3DVideoSinkClass; typedef struct _GstD3DVideoSinkClass GstD3DVideoSinkClass;
typedef struct _GstVideoFormatDetails {
guint32 r_mask;
guint32 g_mask;
guint32 b_mask;
guint32 a_mask;
guint32 r_mask16;
guint32 g_mask16;
guint32 b_mask16;
guint32 a_mask16;
guint r_shift;
guint g_shift;
guint b_shift;
guint a_shift;
guint r_bits;
guint g_bits;
guint b_bits;
guint a_bits;
gint bpp;
gint depth;
gint endianness;
gint pixel_width;
} GstVideoFormatDetails;
struct _GstD3DVideoSink struct _GstD3DVideoSink
{ {
GstVideoSink sink; GstVideoSink sink;
@ -73,7 +50,6 @@ struct _GstD3DVideoSink
GstVideoFormat format; GstVideoFormat format;
GstVideoInfo info; GstVideoInfo info;
GstVideoFormatDetails fmt_details;
gint width; gint width;
gint height; gint height;