mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-25 16:48:11 +00:00
v4l2: object: Move the GstPoll into v4l2object
Moves the GstPoll from the buffer pool into v4l2object. This will be needed to poll for events before the pool has been created. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4525>
This commit is contained in:
parent
d4a428e61f
commit
7599821c42
4 changed files with 178 additions and 108 deletions
|
@ -1077,7 +1077,7 @@ gst_v4l2_buffer_pool_flush_start (GstBufferPool * bpool)
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (pool, "start flushing");
|
GST_DEBUG_OBJECT (pool, "start flushing");
|
||||||
|
|
||||||
gst_poll_set_flushing (pool->poll, TRUE);
|
gst_poll_set_flushing (pool->obj->poll, TRUE);
|
||||||
|
|
||||||
GST_OBJECT_LOCK (pool);
|
GST_OBJECT_LOCK (pool);
|
||||||
pool->empty = FALSE;
|
pool->empty = FALSE;
|
||||||
|
@ -1098,13 +1098,12 @@ gst_v4l2_buffer_pool_flush_stop (GstBufferPool * bpool)
|
||||||
if (pool->other_pool && gst_buffer_pool_is_active (pool->other_pool))
|
if (pool->other_pool && gst_buffer_pool_is_active (pool->other_pool))
|
||||||
gst_buffer_pool_set_flushing (pool->other_pool, FALSE);
|
gst_buffer_pool_set_flushing (pool->other_pool, FALSE);
|
||||||
|
|
||||||
gst_poll_set_flushing (pool->poll, FALSE);
|
gst_poll_set_flushing (pool->obj->poll, FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstFlowReturn
|
static GstFlowReturn
|
||||||
gst_v4l2_buffer_pool_poll (GstV4l2BufferPool * pool, gboolean wait)
|
gst_v4l2_buffer_pool_poll (GstV4l2BufferPool * pool, gboolean wait)
|
||||||
{
|
{
|
||||||
gint ret;
|
|
||||||
GstClockTime timeout;
|
GstClockTime timeout;
|
||||||
|
|
||||||
if (wait)
|
if (wait)
|
||||||
|
@ -1119,7 +1118,7 @@ gst_v4l2_buffer_pool_poll (GstV4l2BufferPool * pool, gboolean wait)
|
||||||
|
|
||||||
if (!wait && pool->empty) {
|
if (!wait && pool->empty) {
|
||||||
GST_OBJECT_UNLOCK (pool);
|
GST_OBJECT_UNLOCK (pool);
|
||||||
goto no_buffers;
|
return GST_V4L2_FLOW_LAST_BUFFER;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (pool->empty)
|
while (pool->empty)
|
||||||
|
@ -1128,87 +1127,14 @@ gst_v4l2_buffer_pool_poll (GstV4l2BufferPool * pool, gboolean wait)
|
||||||
GST_OBJECT_UNLOCK (pool);
|
GST_OBJECT_UNLOCK (pool);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pool->can_poll_device) {
|
if (!pool->obj->can_poll_device) {
|
||||||
if (wait)
|
if (wait)
|
||||||
goto done;
|
return GST_FLOW_OK;
|
||||||
else
|
else
|
||||||
goto no_buffers;
|
return GST_V4L2_FLOW_LAST_BUFFER;
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_LOG_OBJECT (pool, "polling device");
|
return gst_v4l2_object_poll (pool->obj, timeout);
|
||||||
|
|
||||||
again:
|
|
||||||
ret = gst_poll_wait (pool->poll, timeout);
|
|
||||||
if (G_UNLIKELY (ret < 0)) {
|
|
||||||
switch (errno) {
|
|
||||||
case EBUSY:
|
|
||||||
goto stopped;
|
|
||||||
case EAGAIN:
|
|
||||||
case EINTR:
|
|
||||||
goto again;
|
|
||||||
case ENXIO:
|
|
||||||
GST_WARNING_OBJECT (pool,
|
|
||||||
"v4l2 device doesn't support polling. Disabling"
|
|
||||||
" using libv4l2 in this case may cause deadlocks");
|
|
||||||
pool->can_poll_device = FALSE;
|
|
||||||
goto done;
|
|
||||||
default:
|
|
||||||
goto select_error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (gst_poll_fd_has_error (pool->poll, &pool->pollfd))
|
|
||||||
goto select_error;
|
|
||||||
|
|
||||||
/* PRI is used to signal that events are available */
|
|
||||||
if (gst_poll_fd_has_pri (pool->poll, &pool->pollfd)) {
|
|
||||||
struct v4l2_event event = { 0, };
|
|
||||||
|
|
||||||
if (!gst_v4l2_dequeue_event (pool->obj, &event))
|
|
||||||
goto dqevent_failed;
|
|
||||||
|
|
||||||
if (event.type != V4L2_EVENT_SOURCE_CHANGE) {
|
|
||||||
GST_INFO_OBJECT (pool, "Received unhandled event, ignoring.");
|
|
||||||
goto again;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((event.u.src_change.changes & V4L2_EVENT_SRC_CH_RESOLUTION) == 0) {
|
|
||||||
GST_DEBUG_OBJECT (pool,
|
|
||||||
"Received non-resolution source-change, ignoring.");
|
|
||||||
goto again;
|
|
||||||
}
|
|
||||||
|
|
||||||
return GST_V4L2_FLOW_RESOLUTION_CHANGE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret == 0)
|
|
||||||
goto no_buffers;
|
|
||||||
|
|
||||||
done:
|
|
||||||
return GST_FLOW_OK;
|
|
||||||
|
|
||||||
/* ERRORS */
|
|
||||||
stopped:
|
|
||||||
{
|
|
||||||
GST_DEBUG_OBJECT (pool, "stop called");
|
|
||||||
return GST_FLOW_FLUSHING;
|
|
||||||
}
|
|
||||||
select_error:
|
|
||||||
{
|
|
||||||
GST_ELEMENT_ERROR (pool->obj->element, RESOURCE, READ, (NULL),
|
|
||||||
("poll error %d: %s (%d)", ret, g_strerror (errno), errno));
|
|
||||||
return GST_FLOW_ERROR;
|
|
||||||
}
|
|
||||||
no_buffers:
|
|
||||||
{
|
|
||||||
return GST_V4L2_FLOW_LAST_BUFFER;
|
|
||||||
}
|
|
||||||
dqevent_failed:
|
|
||||||
{
|
|
||||||
GST_ELEMENT_ERROR (pool->obj->element, RESOURCE, READ, (NULL),
|
|
||||||
("dqevent error: %s (%d)", g_strerror (errno), errno));
|
|
||||||
return GST_FLOW_ERROR;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstFlowReturn
|
static GstFlowReturn
|
||||||
|
@ -1757,8 +1683,6 @@ gst_v4l2_buffer_pool_finalize (GObject * object)
|
||||||
if (pool->video_fd >= 0)
|
if (pool->video_fd >= 0)
|
||||||
pool->obj->close (pool->video_fd);
|
pool->obj->close (pool->video_fd);
|
||||||
|
|
||||||
gst_poll_free (pool->poll);
|
|
||||||
|
|
||||||
/* This can't be done in dispose method because we must not set pointer
|
/* This can't be done in dispose method because we must not set pointer
|
||||||
* to NULL as it is part of the v4l2object and dispose could be called
|
* to NULL as it is part of the v4l2object and dispose could be called
|
||||||
* multiple times */
|
* multiple times */
|
||||||
|
@ -1774,8 +1698,6 @@ gst_v4l2_buffer_pool_finalize (GObject * object)
|
||||||
static void
|
static void
|
||||||
gst_v4l2_buffer_pool_init (GstV4l2BufferPool * pool)
|
gst_v4l2_buffer_pool_init (GstV4l2BufferPool * pool)
|
||||||
{
|
{
|
||||||
pool->poll = gst_poll_new (TRUE);
|
|
||||||
pool->can_poll_device = TRUE;
|
|
||||||
g_cond_init (&pool->empty_cond);
|
g_cond_init (&pool->empty_cond);
|
||||||
pool->empty = TRUE;
|
pool->empty = TRUE;
|
||||||
pool->orphaned = FALSE;
|
pool->orphaned = FALSE;
|
||||||
|
@ -1838,17 +1760,8 @@ gst_v4l2_buffer_pool_new (GstV4l2Object * obj, GstCaps * caps)
|
||||||
g_object_ref_sink (pool);
|
g_object_ref_sink (pool);
|
||||||
g_free (name);
|
g_free (name);
|
||||||
|
|
||||||
gst_poll_fd_init (&pool->pollfd);
|
|
||||||
pool->pollfd.fd = fd;
|
|
||||||
gst_poll_add_fd (pool->poll, &pool->pollfd);
|
|
||||||
if (V4L2_TYPE_IS_OUTPUT (obj->type))
|
|
||||||
gst_poll_fd_ctl_write (pool->poll, &pool->pollfd, TRUE);
|
|
||||||
else
|
|
||||||
gst_poll_fd_ctl_read (pool->poll, &pool->pollfd, TRUE);
|
|
||||||
|
|
||||||
pool->video_fd = fd;
|
pool->video_fd = fd;
|
||||||
pool->obj = obj;
|
pool->obj = obj;
|
||||||
pool->can_poll_device = TRUE;
|
|
||||||
|
|
||||||
pool->vallocator = gst_v4l2_allocator_new (GST_OBJECT (pool), obj);
|
pool->vallocator = gst_v4l2_allocator_new (GST_OBJECT (pool), obj);
|
||||||
if (pool->vallocator == NULL)
|
if (pool->vallocator == NULL)
|
||||||
|
@ -2331,11 +2244,5 @@ gst_v4l2_buffer_pool_flush (GstV4l2Object * v4l2object)
|
||||||
void
|
void
|
||||||
gst_v4l2_buffer_pool_enable_resolution_change (GstV4l2BufferPool * pool)
|
gst_v4l2_buffer_pool_enable_resolution_change (GstV4l2BufferPool * pool)
|
||||||
{
|
{
|
||||||
guint32 input_id = 0;
|
gst_v4l2_object_subscribe_event (pool->obj, V4L2_EVENT_SOURCE_CHANGE);
|
||||||
|
|
||||||
/* Make sure we subscribe for the current input */
|
|
||||||
gst_v4l2_get_input (pool->obj, &input_id);
|
|
||||||
|
|
||||||
if (gst_v4l2_subscribe_event (pool->obj, V4L2_EVENT_SOURCE_CHANGE, input_id))
|
|
||||||
gst_poll_fd_ctl_pri (pool->poll, &pool->pollfd, TRUE);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -64,9 +64,6 @@ struct _GstV4l2BufferPool
|
||||||
|
|
||||||
GstV4l2Object *obj; /* the v4l2 object */
|
GstV4l2Object *obj; /* the v4l2 object */
|
||||||
gint video_fd; /* a dup(2) of the v4l2object's video_fd */
|
gint video_fd; /* a dup(2) of the v4l2object's video_fd */
|
||||||
GstPoll *poll; /* a poll for video_fd */
|
|
||||||
GstPollFD pollfd;
|
|
||||||
gboolean can_poll_device;
|
|
||||||
|
|
||||||
gboolean empty;
|
gboolean empty;
|
||||||
GCond empty_cond;
|
GCond empty_cond;
|
||||||
|
|
|
@ -525,6 +525,9 @@ gst_v4l2_object_new (GstElement * element,
|
||||||
|
|
||||||
v4l2object->no_initial_format = FALSE;
|
v4l2object->no_initial_format = FALSE;
|
||||||
|
|
||||||
|
v4l2object->poll = gst_poll_new (TRUE);
|
||||||
|
v4l2object->can_poll_device = TRUE;
|
||||||
|
|
||||||
/* We now disable libv4l2 by default, but have an env to enable it. */
|
/* We now disable libv4l2 by default, but have an env to enable it. */
|
||||||
#ifdef HAVE_LIBV4L2
|
#ifdef HAVE_LIBV4L2
|
||||||
if (g_getenv ("GST_V4L2_USE_LIBV4L2")) {
|
if (g_getenv ("GST_V4L2_USE_LIBV4L2")) {
|
||||||
|
@ -572,6 +575,8 @@ gst_v4l2_object_destroy (GstV4l2Object * v4l2object)
|
||||||
g_free (v4l2object->par);
|
g_free (v4l2object->par);
|
||||||
g_free (v4l2object->channel);
|
g_free (v4l2object->channel);
|
||||||
|
|
||||||
|
gst_poll_free (v4l2object->poll);
|
||||||
|
|
||||||
if (v4l2object->formats) {
|
if (v4l2object->formats) {
|
||||||
gst_v4l2_object_clear_format_list (v4l2object);
|
gst_v4l2_object_clear_format_list (v4l2object);
|
||||||
}
|
}
|
||||||
|
@ -900,6 +905,20 @@ gst_v4l2_set_defaults (GstV4l2Object * v4l2object)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_v4l2_object_init_poll (GstV4l2Object * v4l2object)
|
||||||
|
{
|
||||||
|
gst_poll_fd_init (&v4l2object->pollfd);
|
||||||
|
v4l2object->pollfd.fd = v4l2object->video_fd;
|
||||||
|
gst_poll_add_fd (v4l2object->poll, &v4l2object->pollfd);
|
||||||
|
if (V4L2_TYPE_IS_OUTPUT (v4l2object->type))
|
||||||
|
gst_poll_fd_ctl_write (v4l2object->poll, &v4l2object->pollfd, TRUE);
|
||||||
|
else
|
||||||
|
gst_poll_fd_ctl_read (v4l2object->poll, &v4l2object->pollfd, TRUE);
|
||||||
|
|
||||||
|
v4l2object->can_poll_device = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
gst_v4l2_object_open (GstV4l2Object * v4l2object, GstV4l2Error * error)
|
gst_v4l2_object_open (GstV4l2Object * v4l2object, GstV4l2Error * error)
|
||||||
{
|
{
|
||||||
|
@ -908,17 +927,20 @@ gst_v4l2_object_open (GstV4l2Object * v4l2object, GstV4l2Error * error)
|
||||||
else
|
else
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
|
gst_v4l2_object_init_poll (v4l2object);
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
gst_v4l2_object_open_shared (GstV4l2Object * v4l2object, GstV4l2Object * other)
|
gst_v4l2_object_open_shared (GstV4l2Object * v4l2object, GstV4l2Object * other)
|
||||||
{
|
{
|
||||||
gboolean ret;
|
if (gst_v4l2_dup (v4l2object, other)) {
|
||||||
|
gst_v4l2_object_init_poll (v4l2object);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
ret = gst_v4l2_dup (v4l2object, other);
|
return FALSE;
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
|
@ -4598,6 +4620,8 @@ gst_v4l2_object_unlock (GstV4l2Object * v4l2object)
|
||||||
|
|
||||||
GST_LOG_OBJECT (v4l2object->dbg_obj, "start flushing");
|
GST_LOG_OBJECT (v4l2object->dbg_obj, "start flushing");
|
||||||
|
|
||||||
|
gst_poll_set_flushing (v4l2object->poll, TRUE);
|
||||||
|
|
||||||
if (!pool)
|
if (!pool)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -4616,6 +4640,8 @@ gst_v4l2_object_unlock_stop (GstV4l2Object * v4l2object)
|
||||||
|
|
||||||
GST_LOG_OBJECT (v4l2object->dbg_obj, "stop flushing");
|
GST_LOG_OBJECT (v4l2object->dbg_obj, "stop flushing");
|
||||||
|
|
||||||
|
gst_poll_set_flushing (v4l2object->poll, FALSE);
|
||||||
|
|
||||||
if (!pool)
|
if (!pool)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -4637,6 +4663,8 @@ gst_v4l2_object_stop (GstV4l2Object * v4l2object)
|
||||||
if (!GST_V4L2_IS_ACTIVE (v4l2object))
|
if (!GST_V4L2_IS_ACTIVE (v4l2object))
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
|
gst_poll_set_flushing (v4l2object->poll, TRUE);
|
||||||
|
|
||||||
pool = gst_v4l2_object_get_buffer_pool (v4l2object);
|
pool = gst_v4l2_object_get_buffer_pool (v4l2object);
|
||||||
if (pool) {
|
if (pool) {
|
||||||
if (!gst_v4l2_buffer_pool_orphan (v4l2object)) {
|
if (!gst_v4l2_buffer_pool_orphan (v4l2object)) {
|
||||||
|
@ -5467,3 +5495,135 @@ gst_v4l2_object_get_buffer_pool (GstV4l2Object * v4l2object)
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_v4l2_object_poll:
|
||||||
|
* @v4l2object: a #GstV4l2Object
|
||||||
|
* @timeout: timeout of type #GstClockTime
|
||||||
|
*
|
||||||
|
* Poll the video file descriptor for read when this is a capture, write when
|
||||||
|
* this is an output. It will also watch for errors and source change events.
|
||||||
|
* If a source change event is received, %GST_V4L2_FLOW_RESOLUTION_CHANGE will
|
||||||
|
* be returned. If the poll was interrupted, %GST_FLOW_FLUSHING is returned.
|
||||||
|
* If there was no read or write indicator, %GST_V4L2_FLOW_LAST_BUFFER is
|
||||||
|
* returned. It may also return %GST_FLOW_ERROR if some unexpected error
|
||||||
|
* occured.
|
||||||
|
*
|
||||||
|
* Returns: GST_FLOW_OK if buffers are ready to be queued or dequeued.
|
||||||
|
*/
|
||||||
|
GstFlowReturn
|
||||||
|
gst_v4l2_object_poll (GstV4l2Object * v4l2object, GstClockTime timeout)
|
||||||
|
{
|
||||||
|
gint ret;
|
||||||
|
|
||||||
|
if (!v4l2object->can_poll_device) {
|
||||||
|
if (timeout != 0)
|
||||||
|
goto done;
|
||||||
|
else
|
||||||
|
goto no_buffers;
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_LOG_OBJECT (v4l2object->dbg_obj, "polling device");
|
||||||
|
|
||||||
|
again:
|
||||||
|
ret = gst_poll_wait (v4l2object->poll, timeout);
|
||||||
|
if (G_UNLIKELY (ret < 0)) {
|
||||||
|
switch (errno) {
|
||||||
|
case EBUSY:
|
||||||
|
goto stopped;
|
||||||
|
case EAGAIN:
|
||||||
|
case EINTR:
|
||||||
|
goto again;
|
||||||
|
case ENXIO:
|
||||||
|
GST_WARNING_OBJECT (v4l2object->dbg_obj,
|
||||||
|
"v4l2 device doesn't support polling. Disabling"
|
||||||
|
" using libv4l2 in this case may cause deadlocks");
|
||||||
|
v4l2object->can_poll_device = FALSE;
|
||||||
|
goto done;
|
||||||
|
default:
|
||||||
|
goto select_error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gst_poll_fd_has_error (v4l2object->poll, &v4l2object->pollfd))
|
||||||
|
goto select_error;
|
||||||
|
|
||||||
|
/* PRI is used to signal that events are available */
|
||||||
|
if (gst_poll_fd_has_pri (v4l2object->poll, &v4l2object->pollfd)) {
|
||||||
|
struct v4l2_event event = { 0, };
|
||||||
|
|
||||||
|
if (!gst_v4l2_dequeue_event (v4l2object, &event))
|
||||||
|
goto dqevent_failed;
|
||||||
|
|
||||||
|
if (event.type != V4L2_EVENT_SOURCE_CHANGE) {
|
||||||
|
GST_INFO_OBJECT (v4l2object->dbg_obj,
|
||||||
|
"Received unhandled event, ignoring.");
|
||||||
|
goto again;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((event.u.src_change.changes & V4L2_EVENT_SRC_CH_RESOLUTION) == 0) {
|
||||||
|
GST_DEBUG_OBJECT (v4l2object->dbg_obj,
|
||||||
|
"Received non-resolution source-change, ignoring.");
|
||||||
|
goto again;
|
||||||
|
}
|
||||||
|
|
||||||
|
return GST_V4L2_FLOW_RESOLUTION_CHANGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ret == 0)
|
||||||
|
goto no_buffers;
|
||||||
|
|
||||||
|
done:
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
|
||||||
|
/* ERRORS */
|
||||||
|
stopped:
|
||||||
|
{
|
||||||
|
GST_DEBUG_OBJECT (v4l2object->dbg_obj, "stop called");
|
||||||
|
return GST_FLOW_FLUSHING;
|
||||||
|
}
|
||||||
|
select_error:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (v4l2object->element, RESOURCE, READ, (NULL),
|
||||||
|
("poll error %d: %s (%d)", ret, g_strerror (errno), errno));
|
||||||
|
return GST_FLOW_ERROR;
|
||||||
|
}
|
||||||
|
no_buffers:
|
||||||
|
{
|
||||||
|
return GST_V4L2_FLOW_LAST_BUFFER;
|
||||||
|
}
|
||||||
|
dqevent_failed:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (v4l2object->element, RESOURCE, READ, (NULL),
|
||||||
|
("dqevent error: %s (%d)", g_strerror (errno), errno));
|
||||||
|
return GST_FLOW_ERROR;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_v4l2_object_subscribe_event:
|
||||||
|
* @v4l2object: a #GstV4l2Object
|
||||||
|
* @event: the event ID
|
||||||
|
*
|
||||||
|
* Subscribe to an event, and enable polling for these. Note that only
|
||||||
|
* %V4L2_EVENT_SOURCE_CHANGE is currently supported by the poll helper.
|
||||||
|
*
|
||||||
|
* Returns: %TRUE if the driver supports this event
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
gst_v4l2_object_subscribe_event (GstV4l2Object * v4l2object, guint32 event)
|
||||||
|
{
|
||||||
|
guint32 id = 0;
|
||||||
|
|
||||||
|
g_return_val_if_fail (v4l2object != NULL, FALSE);
|
||||||
|
g_return_val_if_fail (GST_V4L2_IS_OPEN (v4l2object), FALSE);
|
||||||
|
|
||||||
|
v4l2object->get_in_out_func (v4l2object, &id);
|
||||||
|
|
||||||
|
if (gst_v4l2_subscribe_event (v4l2object, event, id)) {
|
||||||
|
gst_poll_fd_ctl_pri (v4l2object->poll, &v4l2object->pollfd, TRUE);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
|
@ -134,6 +134,9 @@ struct _GstV4l2Object {
|
||||||
/* the video-device's file descriptor */
|
/* the video-device's file descriptor */
|
||||||
gint video_fd;
|
gint video_fd;
|
||||||
GstV4l2IOMode mode;
|
GstV4l2IOMode mode;
|
||||||
|
GstPoll *poll;
|
||||||
|
GstPollFD pollfd;
|
||||||
|
gboolean can_poll_device;
|
||||||
|
|
||||||
gboolean active;
|
gboolean active;
|
||||||
|
|
||||||
|
@ -314,6 +317,9 @@ GstBufferPool * gst_v4l2_object_get_buffer_pool (GstV4l2Object * v4l2object);
|
||||||
|
|
||||||
GstStructure * gst_v4l2_object_v4l2fourcc_to_structure (guint32 fourcc);
|
GstStructure * gst_v4l2_object_v4l2fourcc_to_structure (guint32 fourcc);
|
||||||
|
|
||||||
|
GstFlowReturn gst_v4l2_object_poll (GstV4l2Object * v4l2object, GstClockTime timeout);
|
||||||
|
gboolean gst_v4l2_object_subscribe_event (GstV4l2Object * v4l2object, guint32 event);
|
||||||
|
|
||||||
/* crop / compose */
|
/* crop / compose */
|
||||||
gboolean gst_v4l2_object_set_crop (GstV4l2Object * obj, struct v4l2_rect *result);
|
gboolean gst_v4l2_object_set_crop (GstV4l2Object * obj, struct v4l2_rect *result);
|
||||||
gboolean gst_v4l2_object_get_crop_bounds (GstV4l2Object * obj, struct v4l2_rect *bounds);
|
gboolean gst_v4l2_object_get_crop_bounds (GstV4l2Object * obj, struct v4l2_rect *bounds);
|
||||||
|
|
Loading…
Reference in a new issue