summaryrefslogtreecommitdiffstats
path: root/ext/directfb/dfbvideosink.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/directfb/dfbvideosink.c')
-rw-r--r--ext/directfb/dfbvideosink.c169
1 files changed, 114 insertions, 55 deletions
diff --git a/ext/directfb/dfbvideosink.c b/ext/directfb/dfbvideosink.c
index b425de61..3a637608 100644
--- a/ext/directfb/dfbvideosink.c
+++ b/ext/directfb/dfbvideosink.c
@@ -140,6 +140,8 @@ static DFBSurfacePixelFormat gst_dfbvideosink_get_format_from_caps (GstCaps *
caps);
static void gst_dfbvideosink_update_colorbalance (GstDfbVideoSink *
dfbvideosink);
+static void gst_dfbvideosink_surface_destroy (GstDfbVideoSink * dfbvideosink,
+ GstDfbSurface * surface);
static GstVideoSinkClass *parent_class = NULL;
@@ -203,11 +205,15 @@ gst_dfbvideosink_surface_create (GstDfbVideoSink * dfbvideosink, GstCaps * caps,
DFBSurfaceDescription s_dsc;
gpointer data;
gint pitch;
+ gboolean succeeded = FALSE;
g_return_val_if_fail (GST_IS_DFBVIDEOSINK (dfbvideosink), NULL);
surface = (GstDfbSurface *) gst_mini_object_new (GST_TYPE_DFBSURFACE);
+ /* Keep a ref to our sink */
+ surface->dfbvideosink = gst_object_ref (dfbvideosink);
+ /* Surface is not locked yet */
surface->locked = FALSE;
structure = gst_caps_get_structure (caps, 0);
@@ -216,55 +222,91 @@ gst_dfbvideosink_surface_create (GstDfbVideoSink * dfbvideosink, GstCaps * caps,
!gst_structure_get_int (structure, "height", &surface->height)) {
GST_WARNING_OBJECT (dfbvideosink, "failed getting geometry from caps %"
GST_PTR_FORMAT, caps);
+ goto fallback;
}
+ /* Pixel format from caps */
surface->pixel_format = gst_dfbvideosink_get_format_from_caps (caps);
+ if (surface->pixel_format == DSPF_UNKNOWN) {
+ goto fallback;
+ }
- if (dfbvideosink->dfb) {
- /* Creating an internal surface which will be used as GstBuffer, we used
- the detected pixel format and video dimensions */
-
- s_dsc.flags =
- DSDESC_PIXELFORMAT | DSDESC_WIDTH | DSDESC_HEIGHT | DSDESC_CAPS;
-
- s_dsc.pixelformat = surface->pixel_format;
- s_dsc.width = surface->width;
- s_dsc.height = surface->height;
- s_dsc.caps = DSCAPS_VIDEOONLY;
-
- ret = dfbvideosink->dfb->CreateSurface (dfbvideosink->dfb, &s_dsc,
- &surface->surface);
- if (ret != DFB_OK) {
- GST_WARNING_OBJECT (dfbvideosink, "failed creating a DirectFB surface");
- gst_object_unref (surface);
- surface = NULL;
- goto beach;
- }
+ if (!dfbvideosink->dfb) {
+ GST_DEBUG_OBJECT (dfbvideosink, "no DirectFB context to create a surface");
+ goto fallback;
+ }
- /* Clearing surface */
- surface->surface->Clear (surface->surface, 0x00, 0x00, 0x00, 0xFF);
+ /* Creating an internal surface which will be used as GstBuffer, we used
+ the detected pixel format and video dimensions */
- /* Locking the surface to acquire the memory pointer */
- surface->surface->Lock (surface->surface, DSLF_WRITE, &data, &pitch);
- surface->locked = TRUE;
- GST_BUFFER_DATA (surface) = data;
- GST_BUFFER_SIZE (surface) = pitch * surface->height;
+ s_dsc.flags =
+ DSDESC_PIXELFORMAT | DSDESC_WIDTH | DSDESC_HEIGHT /*| DSDESC_CAPS */ ;
- GST_DEBUG_OBJECT (dfbvideosink, "creating a %dx%d surface with %s pixel "
- "format, line pitch %d", surface->width, surface->height,
- gst_dfbvideosink_get_format_name (surface->pixel_format), pitch);
- } else {
- GST_BUFFER (surface)->malloc_data = g_malloc (size);
- GST_BUFFER_DATA (surface) = GST_BUFFER (surface)->malloc_data;
- GST_BUFFER_SIZE (surface) = size;
+ s_dsc.pixelformat = surface->pixel_format;
+ s_dsc.width = surface->width;
+ s_dsc.height = surface->height;
+ /*s_dsc.caps = DSCAPS_VIDEOONLY; */
+
+ ret = dfbvideosink->dfb->CreateSurface (dfbvideosink->dfb, &s_dsc,
+ &surface->surface);
+ if (ret != DFB_OK) {
+ GST_WARNING_OBJECT (dfbvideosink, "failed creating a DirectFB surface");
surface->surface = NULL;
- GST_DEBUG_OBJECT (dfbvideosink, "allocating a buffer of %d bytes", size);
+ goto fallback;
}
- /* Keep a ref to our sink */
- surface->dfbvideosink = gst_object_ref (dfbvideosink);
+ /* Clearing surface */
+ surface->surface->Clear (surface->surface, 0x00, 0x00, 0x00, 0xFF);
+
+ /* Locking the surface to acquire the memory pointer */
+ surface->surface->Lock (surface->surface, DSLF_WRITE, &data, &pitch);
+ surface->locked = TRUE;
+ GST_BUFFER_DATA (surface) = data;
+ GST_BUFFER_SIZE (surface) = pitch * surface->height;
+
+ /* Be carefull here. If size is different from the surface size
+ (pitch * height), we can't use that surface through buffer alloc system
+ or we are going to run into serious stride issues */
+ if (GST_BUFFER_SIZE (surface) != size) {
+ GST_WARNING_OBJECT (dfbvideosink, "DirectFB surface size (%dx%d=%d) "
+ "differs from GStreamer requested size %d", pitch, surface->height,
+ GST_BUFFER_SIZE (surface), size);
+ goto fallback;
+ }
+
+ GST_DEBUG_OBJECT (dfbvideosink, "creating a %dx%d surface (%p) with %s "
+ "pixel format, line pitch %d", surface->width, surface->height, surface,
+ gst_dfbvideosink_get_format_name (surface->pixel_format), pitch);
+
+ succeeded = TRUE;
+
+ goto beach;
+
+fallback:
+
+ /* We allocate a standard buffer ourselves to store it in our buffer pool,
+ this is an optimisation for memory allocation */
+ GST_BUFFER (surface)->malloc_data = g_malloc (size);
+ GST_BUFFER_DATA (surface) = GST_BUFFER (surface)->malloc_data;
+ GST_BUFFER_SIZE (surface) = size;
+ if (surface->surface) {
+ if (surface->locked) {
+ surface->surface->Unlock (surface->surface);
+ surface->locked = FALSE;
+ }
+ surface->surface->Release (surface->surface);
+ surface->surface = NULL;
+ }
+ GST_DEBUG_OBJECT (dfbvideosink, "allocating a buffer (%p) of %d bytes",
+ surface, size);
+
+ succeeded = TRUE;
beach:
+ if (!succeeded) {
+ gst_dfbvideosink_surface_destroy (dfbvideosink, surface);
+ surface = NULL;
+ }
return surface;
}
@@ -560,7 +602,7 @@ gst_dfbvideosink_setup (GstDfbVideoSink * dfbvideosink)
goto beach;
}
- /* Get Hardwared capabilities */
+ /* Get Hardware capabilities */
ret = dfbvideosink->dfb->GetDeviceDescription (dfbvideosink->dfb,
&hw_caps);
@@ -1461,6 +1503,7 @@ gst_dfbvideosink_show_frame (GstBaseSink * bsink, GstBuffer * buf)
DFBResult res;
GstVideoRectangle dst, src, result;
GstFlowReturn ret = GST_FLOW_OK;
+ gboolean mem_cpy = TRUE;
dfbvideosink = GST_DFBVIDEOSINK (bsink);
@@ -1469,9 +1512,27 @@ gst_dfbvideosink_show_frame (GstBaseSink * bsink, GstBuffer * buf)
goto beach;
}
- /* If we are rendering from a buffer we did not allocate or to an external
- * surface, we will memcpy data */
- if (!GST_IS_DFBSURFACE (buf) || dfbvideosink->ext_surface) {
+ /* Is that a buffer we allocated ourselves ? */
+ if (GST_IS_DFBSURFACE (buf)) {
+ GstDfbSurface *tmp_surface = GST_DFBSURFACE (buf);
+
+ /* Does it have a surface ? */
+ if (tmp_surface->surface) {
+ mem_cpy = FALSE;
+ GST_DEBUG_OBJECT (dfbvideosink, "we have a buffer (%p) we allocated "
+ "ourselves and it has a surface, no memcpy then", buf);
+ } else {
+ /* No surface, that's a malloc */
+ GST_DEBUG_OBJECT (dfbvideosink, "we have a buffer (%p) we allocated "
+ "ourselves but it does not hold a surface", buf);
+ }
+ } else {
+ /* Not our baby */
+ GST_DEBUG_OBJECT (dfbvideosink, "we have a buffer (%p) we did not allocate",
+ buf);
+ }
+
+ if (mem_cpy) {
IDirectFBSurface *dest = NULL, *surface = NULL;
gpointer data;
gint dest_pitch, src_pitch, line;
@@ -1492,6 +1553,7 @@ gst_dfbvideosink_show_frame (GstBaseSink * bsink, GstBuffer * buf)
"(vsync %d)", dfbvideosink->vsync);
}
+ /* Get the video frame geometry from the buffer caps */
structure = gst_caps_get_structure (GST_BUFFER_CAPS (buf), 0);
if (structure) {
gst_structure_get_int (structure, "width", &src.w);
@@ -1547,7 +1609,7 @@ gst_dfbvideosink_show_frame (GstBaseSink * bsink, GstBuffer * buf)
res = surface->Flip (surface, NULL, DSFLIP_NONE);
}
}
- } else if (dfbvideosink->primary) {
+ } else {
/* Else we will [Stretch]Blit to our primary */
GstDfbSurface *surface = GST_DFBSURFACE (buf);
@@ -1560,8 +1622,10 @@ gst_dfbvideosink_show_frame (GstBaseSink * bsink, GstBuffer * buf)
dfbvideosink->primary->GetSize (dfbvideosink->primary, &dst.w, &dst.h);
/* Unlocking surface before blit */
- surface->surface->Unlock (surface->surface);
- surface->locked = FALSE;
+ if (surface->locked) {
+ surface->surface->Unlock (surface->surface);
+ surface->locked = FALSE;
+ }
gst_video_sink_center_rect (src, dst, &result, dfbvideosink->hw_scaling);
@@ -1591,11 +1655,6 @@ gst_dfbvideosink_show_frame (GstBaseSink * bsink, GstBuffer * buf)
dfbvideosink->primary->Flip (dfbvideosink->primary, NULL, DSFLIP_NONE);
}
}
- } else {
- GST_WARNING_OBJECT (dfbvideosink, "no primary, no external surface "
- "what's going on ?");
- ret = GST_FLOW_UNEXPECTED;
- goto beach;
}
beach:
@@ -1740,6 +1799,7 @@ alloc:
}
}
}
+ g_mutex_unlock (dfbvideosink->pool_lock);
/* We haven't found anything, creating a new one */
if (!surface) {
@@ -1758,12 +1818,11 @@ alloc:
gst_buffer_set_caps (GST_BUFFER (surface), caps);
}
}
- g_mutex_unlock (dfbvideosink->pool_lock);
-
- gst_caps_unref (desired_caps);
*buf = GST_BUFFER (surface);
+ gst_caps_unref (desired_caps);
+
return ret;
}
@@ -1786,13 +1845,13 @@ gst_dfbsurface_finalize (GstDfbSurface * surface)
if ((surface->width != dfbvideosink->video_width) ||
(surface->height != dfbvideosink->video_height) ||
(surface->pixel_format != dfbvideosink->pixel_format)) {
- GST_DEBUG_OBJECT (dfbvideosink, "destroy image as its size changed %dx%d "
- "vs current %dx%d", surface->width, surface->height,
+ GST_DEBUG_OBJECT (dfbvideosink, "destroy surface %p as its size changed "
+ "%dx%d vs current %dx%d", surface, surface->width, surface->height,
dfbvideosink->video_width, dfbvideosink->video_height);
gst_dfbvideosink_surface_destroy (dfbvideosink, surface);
} else {
/* In that case we can reuse the image and add it to our image pool. */
- GST_DEBUG_OBJECT (dfbvideosink, "recycling image in pool");
+ GST_DEBUG_OBJECT (dfbvideosink, "recycling surface %p in pool", surface);
/* need to increment the refcount again to recycle */
gst_buffer_ref (GST_BUFFER (surface));
g_mutex_lock (dfbvideosink->pool_lock);