summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--ChangeLog7
-rw-r--r--sys/v4l2/gstv4l2src.c55
-rw-r--r--sys/v4l2/v4l2src_calls.c58
-rw-r--r--sys/v4l2/v4l2src_calls.h7
4 files changed, 113 insertions, 14 deletions
diff --git a/ChangeLog b/ChangeLog
index 5bcdcb8a..3ce7674b 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,10 @@
+2006-08-29 Edgard Lima <edgard.lima@indt.org.br>
+
+ * sys/v4l2/gstv4l2src.c:
+ * sys/v4l2/v4l2src_calls.c:
+ * sys/v4l2/v4l2src_calls.h:
+ Fixed framerate negotiation.
+
2006-08-29 Tim-Philipp Müller <tim at centricular dot net>
Patch by: Andrew Andkjar <enki at goodship net>
diff --git a/sys/v4l2/gstv4l2src.c b/sys/v4l2/gstv4l2src.c
index d5cb1940..3ad39a43 100644
--- a/sys/v4l2/gstv4l2src.c
+++ b/sys/v4l2/gstv4l2src.c
@@ -755,13 +755,9 @@ gst_v4l2src_get_caps (GstBaseSrc * src)
gst_structure_set (structure,
"width", GST_TYPE_INT_RANGE, min_w, max_w,
"height", GST_TYPE_INT_RANGE, min_h, max_h, NULL);
- if (fps_n > 0) {
- gst_structure_set (structure, "framerate", GST_TYPE_FRACTION,
- fps_n, fps_d, NULL);
- } else {
- gst_structure_set (structure, "framerate", GST_TYPE_FRACTION_RANGE,
- 1, 1, 100, 1, NULL);
- }
+
+ gst_structure_set (structure, "framerate", GST_TYPE_FRACTION_RANGE,
+ 1, 1, 100, 1, NULL);
gst_caps_append_structure (caps, structure);
@@ -778,6 +774,8 @@ gst_v4l2src_set_caps (GstBaseSrc * src, GstCaps * caps)
gint w, h;
GstStructure *structure;
struct v4l2_fmtdesc *format;
+ const GValue *framerate;
+ guint fps_n, fps_d;
v4l2src = GST_V4L2SRC (src);
@@ -803,18 +801,37 @@ gst_v4l2src_set_caps (GstBaseSrc * src, GstCaps * caps)
gst_structure_get_int (structure, "width", &w);
gst_structure_get_int (structure, "height", &h);
+ framerate = gst_structure_get_value (structure, "framerate");
GST_DEBUG_OBJECT (v4l2src, "trying to set_capture %dx%d, format %s",
w, h, format->description);
- /* this only fills in v4l2src->mmap values */
- if (!gst_v4l2src_set_capture (v4l2src, format, &w, &h)) {
+
+ if (framerate) {
+ fps_n = gst_value_get_fraction_numerator (framerate);
+ fps_d = gst_value_get_fraction_denominator (framerate);
+ } else {
+ fps_n = 0;
+ fps_d = 1;
+ }
+
+ if (!gst_v4l2src_set_capture (v4l2src, format, &w, &h, &fps_n, &fps_d)) {
GST_WARNING_OBJECT (v4l2src, "could not set_capture %dx%d, format %s",
w, h, format->description);
return FALSE;
}
- gst_structure_set (structure, "width", G_TYPE_INT, w, "height", G_TYPE_INT, h,
- NULL);
+ if (fps_n) {
+ gst_structure_set (structure,
+ "width", G_TYPE_INT, w,
+ "height", G_TYPE_INT, h,
+ "framerate", GST_TYPE_FRACTION,
+ gst_value_get_fraction_numerator (framerate),
+ gst_value_get_fraction_denominator (framerate), NULL);
+ } else {
+ gst_structure_set (structure,
+ "width", G_TYPE_INT, w,
+ "height", G_TYPE_INT, h, "framerate", GST_TYPE_FRACTION, NULL);
+ }
if (!gst_v4l2src_capture_init (v4l2src))
return FALSE;
@@ -822,6 +839,20 @@ gst_v4l2src_set_caps (GstBaseSrc * src, GstCaps * caps)
if (!gst_v4l2src_capture_start (v4l2src))
return FALSE;
+ if (v4l2src->fps_n != fps_n || v4l2src->fps_d != fps_d) {
+ GST_WARNING_OBJECT (v4l2src,
+ "framerate changed after start capturing from %u/%u to %u/%u", fps_n,
+ fps_d, v4l2src->fps_n, v4l2src->fps_d);
+ if (fps_n) {
+ gst_structure_set (structure,
+ "width", G_TYPE_INT, w,
+ "height", G_TYPE_INT, h,
+ "framerate", GST_TYPE_FRACTION,
+ gst_value_get_fraction_numerator (framerate),
+ gst_value_get_fraction_denominator (framerate), NULL);
+ }
+ }
+
return TRUE;
}
@@ -948,7 +979,7 @@ gst_v4l2src_create (GstPushSrc * src, GstBuffer ** buf)
if (v4l2src->use_fixed_fps && v4l2src->fps_n == 0) {
GST_ELEMENT_ERROR (v4l2src, RESOURCE, SETTINGS, (NULL),
- ("could not get frame rate for element"));
+ ("could not get frame rate for element, try to set use-fixed-fps property to false"));
return GST_FLOW_ERROR;
}
diff --git a/sys/v4l2/v4l2src_calls.c b/sys/v4l2/v4l2src_calls.c
index 801120ff..3cfd0b08 100644
--- a/sys/v4l2/v4l2src_calls.c
+++ b/sys/v4l2/v4l2src_calls.c
@@ -240,8 +240,13 @@ gst_v4l2src_get_capture (GstV4l2Src * v4l2src)
gboolean
gst_v4l2src_set_capture (GstV4l2Src * v4l2src,
- struct v4l2_fmtdesc * fmt, gint * width, gint * height)
+ struct v4l2_fmtdesc * fmt, gint * width, gint * height,
+ guint * fps_n, guint * fps_d)
{
+
+ guint new_fps_n = *fps_n;
+ guint new_fps_d = *fps_d;
+
DEBUG ("Setting capture format to %dx%d, format %s",
*width, *height, fmt->description);
@@ -279,9 +284,31 @@ gst_v4l2src_set_capture (GstV4l2Src * v4l2src,
}
if (fmt->pixelformat != v4l2src->format.fmt.pix.pixelformat) {
+ GST_ELEMENT_ERROR (v4l2src, RESOURCE, SETTINGS, (NULL),
+ ("failed to set pixelformat to %s @ %dx%d for device %s: %s",
+ fmt->description, *width, *height,
+ v4l2src->v4l2object->videodev, g_strerror (errno)));
goto fail;
}
+ if (*fps_n) {
+ if (gst_v4l2src_set_fps (v4l2src, &new_fps_n, &new_fps_d)) {
+ if (new_fps_n != *fps_n || new_fps_d != *fps_d) {
+ DEBUG ("Updating framerate from %u/%u to %u%u",
+ *fps_n, *fps_d, new_fps_n, new_fps_d);
+ *fps_n = new_fps_n;
+ *fps_d = new_fps_d;
+ }
+ }
+ } else {
+ if (gst_v4l2src_get_fps (v4l2src, &new_fps_n, &new_fps_d)) {
+ DEBUG ("framerate is %u/%u", new_fps_n, new_fps_d);
+ *fps_n = new_fps_n;
+ *fps_d = new_fps_d;
+ }
+ }
+
+
*width = v4l2src->format.fmt.pix.width;
*height = v4l2src->format.fmt.pix.height;
@@ -644,6 +671,34 @@ gst_v4l2src_update_fps (GstV4l2Object * v4l2object)
return gst_v4l2src_get_fps (v4l2src, &v4l2src->fps_n, &v4l2src->fps_d);
}
+
+gboolean
+gst_v4l2src_set_fps (GstV4l2Src * v4l2src, guint * fps_n, guint * fps_d)
+{
+
+ GstV4l2Object *v4l2object = v4l2src->v4l2object;
+ struct v4l2_streamparm stream;
+
+ memset (&stream, 0x00, sizeof (struct v4l2_streamparm));
+ stream.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
+ if (ioctl (v4l2object->video_fd, VIDIOC_G_PARM, &stream) == 0 &&
+ stream.parm.capture.capability & V4L2_CAP_TIMEPERFRAME) {
+
+ stream.parm.capture.timeperframe.denominator = *fps_n;
+ stream.parm.capture.timeperframe.numerator = *fps_d;
+
+ if (ioctl (v4l2object->video_fd, VIDIOC_S_PARM, &stream) == 0) {
+ *fps_n = stream.parm.capture.timeperframe.denominator;
+ *fps_d = stream.parm.capture.timeperframe.numerator;
+ return TRUE;
+ }
+ }
+
+ return FALSE;
+
+}
+
+
gboolean
gst_v4l2src_get_fps (GstV4l2Src * v4l2src, guint * fps_n, guint * fps_d)
{
@@ -656,6 +711,7 @@ gst_v4l2src_get_fps (GstV4l2Src * v4l2src, guint * fps_n, guint * fps_d)
return FALSE;
/* Try to get the frame rate directly from the device using VIDIOC_G_PARM */
+ memset (&stream, 0x00, sizeof (struct v4l2_streamparm));
stream.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (ioctl (v4l2object->video_fd, VIDIOC_G_PARM, &stream) == 0 &&
stream.parm.capture.capability & V4L2_CAP_TIMEPERFRAME) {
diff --git a/sys/v4l2/v4l2src_calls.h b/sys/v4l2/v4l2src_calls.h
index 4b5a3871..6cb637b3 100644
--- a/sys/v4l2/v4l2src_calls.h
+++ b/sys/v4l2/v4l2src_calls.h
@@ -28,7 +28,8 @@
gboolean gst_v4l2src_get_capture (GstV4l2Src * v4l2src);
gboolean gst_v4l2src_set_capture (GstV4l2Src * v4l2src,
struct v4l2_fmtdesc *fmt,
- gint * width, gint * height);
+ gint * width, gint * height,
+ guint *fps_n, guint * fps_d);
gboolean gst_v4l2src_capture_init (GstV4l2Src * v4l2src);
gboolean gst_v4l2src_capture_start (GstV4l2Src * v4l2src);
gint gst_v4l2src_grab_frame (GstV4l2Src * v4l2src);
@@ -54,6 +55,10 @@ gst_v4l2src_update_fps (GstV4l2Object * v4l2object);
extern gboolean
gst_v4l2src_get_fps (GstV4l2Src * v4l2src, guint * fps_n, guint * fps_d);
+
+extern gboolean
+gst_v4l2src_set_fps (GstV4l2Src * v4l2src, guint * fps_n, guint * fps_d);
+
GValue *gst_v4l2src_get_fps_list (GstV4l2Src * v4l2src);
GstBuffer *gst_v4l2src_buffer_new (GstV4l2Src * v4l2src,