diff options
Diffstat (limited to 'sys/v4l2/v4l2_calls.c')
-rw-r--r-- | sys/v4l2/v4l2_calls.c | 863 |
1 files changed, 863 insertions, 0 deletions
diff --git a/sys/v4l2/v4l2_calls.c b/sys/v4l2/v4l2_calls.c new file mode 100644 index 00000000..c24d0e4d --- /dev/null +++ b/sys/v4l2/v4l2_calls.c @@ -0,0 +1,863 @@ +/* G-Streamer generic V4L2 element - generic V4L2 calls handling + * Copyright (C) 2002 Ronald Bultje <rbultje@ronald.bitfreak.net> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <sys/types.h> +#include <sys/stat.h> +#include <fcntl.h> +#include <sys/ioctl.h> +#include <sys/mman.h> +#include <string.h> +#include <errno.h> +#include "v4l2_calls.h" + +#define DEBUG(format, args...) \ + GST_DEBUG_ELEMENT(GST_CAT_PLUGIN_INFO, \ + GST_ELEMENT(v4l2element), \ + "V4L2: " format "\n", ##args) + + +/****************************************************** + * gst_v4l2_get_capabilities(): + * get the device's capturing capabilities + * return value: TRUE on success, FALSE on error + ******************************************************/ + +static gboolean +gst_v4l2_get_capabilities (GstV4l2Element *v4l2element) +{ + DEBUG("getting capabilities"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (ioctl(v4l2element->video_fd, VIDIOC_QUERYCAP, &(v4l2element->vcap)) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Error getting %s capabilities: %s", + v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_empty_lists() and gst_v4l2_fill_lists(): + * fill/empty the lists of enumerations + * return value: TRUE on success, FALSE on error + ******************************************************/ + +static gboolean +gst_v4l2_fill_lists (GstV4l2Element *v4l2element) +{ + gint n; + + DEBUG("getting enumerations"); + GST_V4L2_CHECK_OPEN(v4l2element); + + /* create enumeration lists - let's start with format enumeration */ + for (n=0;;n++) { + struct v4l2_fmtdesc format, *fmtptr; + format.index = n; + if (ioctl(v4l2element->video_fd, VIDIOC_ENUM_PIXFMT, &format) < 0) { + if (errno == EINVAL) + break; /* end of enumeration */ + else { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get no. %d in pixelformat enumeration for %s: %s", + n, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + } + fmtptr = g_malloc(sizeof(format)); + memcpy(fmtptr, &format, sizeof(format)); + v4l2element->formats = g_list_append(v4l2element->formats, fmtptr); + } + + /* and now, the inputs */ + for (n=0;;n++) { + struct v4l2_input input, *inpptr; + input.index = n; + if (ioctl(v4l2element->video_fd, VIDIOC_ENUMINPUT, &input) < 0) { + if (errno == EINVAL) + break; /* end of enumeration */ + else { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get no. %d in input enumeration for %s: %s", + n, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + } + inpptr = g_malloc(sizeof(input)); + memcpy(inpptr, &input, sizeof(input)); + v4l2element->inputs = g_list_append(v4l2element->inputs, inpptr); + } + + /* outputs */ + for (n=0;;n++) { + struct v4l2_output output, *outptr; + output.index = n; + if (ioctl(v4l2element->video_fd, VIDIOC_ENUMOUTPUT, &output) < 0) { + if (errno == EINVAL) + break; /* end of enumeration */ + else { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get no. %d in output enumeration for %s: %s", + n, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + } + outptr = g_malloc(sizeof(output)); + memcpy(outptr, &output, sizeof(output)); + v4l2element->outputs = g_list_append(v4l2element->outputs, outptr); + } + + /* norms... */ + for (n=0;;n++) { + struct v4l2_enumstd standard, *stdptr; + standard.index = n; + if (ioctl(v4l2element->video_fd, VIDIOC_ENUMSTD, &standard) < 0) { + if (errno == EINVAL) + break; /* end of enumeration */ + else { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get no. %d in norm enumeration for %s: %s", + n, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + } + stdptr = g_malloc(sizeof(standard)); + memcpy(stdptr, &standard, sizeof(standard)); + v4l2element->norms = g_list_append(v4l2element->norms, stdptr); + } + + /* and lastly, controls+menus (if appropriate) */ + for (n=0;;n++) { + struct v4l2_queryctrl control, *ctrlptr; + GList *menus = NULL; + control.id = n; + if (ioctl(v4l2element->video_fd, VIDIOC_QUERYCTRL, &control) < 0) { + if (errno == EINVAL) + break; /* end of enumeration */ + else { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get no. %d in control enumeration for %s: %s", + n, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + } + ctrlptr = g_malloc(sizeof(control)); + memcpy(ctrlptr, &control, sizeof(control)); + v4l2element->controls = g_list_append(v4l2element->controls, ctrlptr); + if (control.type == V4L2_CTRL_TYPE_MENU) { + struct v4l2_querymenu menu, *mptr; + int i; + menu.id = n; + for (i=0;;i++) { + menu.index = i; + if (ioctl(v4l2element->video_fd, VIDIOC_QUERYMENU, &menu) < 0) { + if (errno == EINVAL) + break; /* end of enumeration */ + else { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get no. %d in menu %d enumeration for %s: %s", + i, n, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + } + mptr = g_malloc(sizeof(menu)); + memcpy(mptr, &menu, sizeof(menu)); + menus = g_list_append(menus, mptr); + } + } + v4l2element->menus = g_list_append(v4l2element->menus, menus); + } + + return TRUE; +} + + +static void +gst_v4l2_empty_lists (GstV4l2Element *v4l2element) +{ + DEBUG("deleting enumerations"); + + /* empty lists */ + while (g_list_length(v4l2element->inputs) > 0) { + gpointer data = g_list_nth_data(v4l2element->inputs, 0); + v4l2element->inputs = g_list_remove(v4l2element->inputs, data); + g_free(data); + } + while (g_list_length(v4l2element->outputs) > 0) { + gpointer data = g_list_nth_data(v4l2element->outputs, 0); + v4l2element->outputs = g_list_remove(v4l2element->outputs, data); + g_free(data); + } + while (g_list_length(v4l2element->norms) > 0) { + gpointer data = g_list_nth_data(v4l2element->norms, 0); + v4l2element->norms = g_list_remove(v4l2element->norms, data); + g_free(data); + } + while (g_list_length(v4l2element->formats) > 0) { + gpointer data = g_list_nth_data(v4l2element->formats, 0); + v4l2element->formats = g_list_remove(v4l2element->formats, data); + g_free(data); + } + while (g_list_length(v4l2element->controls) > 0) { + gpointer data = g_list_nth_data(v4l2element->controls, 0); + v4l2element->controls = g_list_remove(v4l2element->controls, data); + g_free(data); + } + v4l2element->menus = g_list_remove_all(v4l2element->menus, NULL); + while (g_list_length(v4l2element->menus) > 0) { + GList *items = (GList *) g_list_nth_data(v4l2element->menus, 0); + v4l2element->inputs = g_list_remove(v4l2element->inputs, items); + while (g_list_length(items) > 0) { + gpointer data = g_list_nth_data(v4l2element->menus, 0); + items = g_list_remove(items, data); + g_free(data); + } + } +} + + +/****************************************************** + * gst_v4l2_open(): + * open the video device (v4l2element->device) + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_open (GstV4l2Element *v4l2element) +{ + DEBUG("Trying to open device %s", v4l2element->device); + GST_V4L2_CHECK_NOT_OPEN(v4l2element); + GST_V4L2_CHECK_NOT_ACTIVE(v4l2element); + + /* be sure we have a device */ + if (!v4l2element->device) + v4l2element->device = g_strdup("/dev/video"); + + /* open the device */ + v4l2element->video_fd = open(v4l2element->device, O_RDWR); + if (!GST_V4L2_IS_OPEN(v4l2element)) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to open device %s: %s", + v4l2element->device, sys_errlist[errno]); + goto error; + } + + /* get capabilities */ + if (!gst_v4l2_get_capabilities(v4l2element)) { + goto error; + } + + /* and get the video window */ + if (GST_V4L2_IS_OVERLAY(v4l2element)) { + if (ioctl(v4l2element->video_fd, VIDIOC_G_WIN, &(v4l2element->vwin)) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get video window properties of %s: %s", + v4l2element->device, sys_errlist[errno]); + goto error; + } + } + + /* create enumerations */ + if (!gst_v4l2_fill_lists(v4l2element)) + goto error; + + gst_info("Opened device '%s' (%s) successfully\n", + v4l2element->vcap.name, v4l2element->device); + + return TRUE; + +error: + if (GST_V4L2_IS_OPEN(v4l2element)) { + /* close device */ + close(v4l2element->video_fd); + v4l2element->video_fd = -1; + } + /* empty lists */ + gst_v4l2_empty_lists(v4l2element); + + return FALSE; +} + + +/****************************************************** + * gst_v4l2_close(): + * close the video device (v4l2element->video_fd) + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_close (GstV4l2Element *v4l2element) +{ + DEBUG("Trying to close %s", v4l2element->device); + GST_V4L2_CHECK_OPEN(v4l2element); + GST_V4L2_CHECK_NOT_ACTIVE(v4l2element); + + /* close device */ + close(v4l2element->video_fd); + v4l2element->video_fd = -1; + + /* empty lists */ + gst_v4l2_empty_lists(v4l2element); + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_get_norm() + * Get the norm of the current device + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_get_norm (GstV4l2Element *v4l2element, + gint *norm) +{ + struct v4l2_standard standard; + gint n; + + DEBUG("getting norm"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (ioctl(v4l2element->video_fd, VIDIOC_G_STD, &standard) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get the current norm for device %s: %s", + v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + /* try to find out what norm number this actually is */ + for (n=0;n<g_list_length(v4l2element->norms);n++) { + struct v4l2_enumstd *stdptr = (struct v4l2_enumstd *) g_list_nth_data(v4l2element->norms, n); + if (!strcmp(stdptr->std.name, standard.name)) { + *norm = n; + return TRUE; + } + } + + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to find norm '%s' in our list of available norms for device %s", + standard.name, v4l2element->device); + return FALSE; +} + + +/****************************************************** + * gst_v4l2_set_norm() + * Set the norm of the current device + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_set_norm (GstV4l2Element *v4l2element, + gint norm) +{ + struct v4l2_enumstd *standard; + + DEBUG("trying to set norm to %d", norm); + GST_V4L2_CHECK_OPEN(v4l2element); + GST_V4L2_CHECK_NOT_ACTIVE(v4l2element); + + if (norm < 0 || norm >= g_list_length(v4l2element->norms)) { + gst_element_error(GST_ELEMENT(v4l2element), + "Invalid norm number %d (%d-%d)", + norm, 0, g_list_length(v4l2element->norms)); + return FALSE; + } + + standard = (struct v4l2_enumstd *) g_list_nth_data(v4l2element->norms, norm); + + if (ioctl(v4l2element->video_fd, VIDIOC_S_STD, &standard->std) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to set norm '%s' (%d) for device %s: %s", + standard->std.name, norm, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_get_norm_names() + * Get the list of available norms + * return value: the list + ******************************************************/ + +GList * +gst_v4l2_get_norm_names (GstV4l2Element *v4l2element) +{ + GList *names = NULL; + gint n; + + DEBUG("getting a list of norm names"); + + for (n=0;n<g_list_length(v4l2element->norms);n++) { + struct v4l2_enumstd *standard = (struct v4l2_enumstd *) g_list_nth_data(v4l2element->norms, n); + names = g_list_append(names, g_strdup(standard->std.name)); + } + + return names; +} + + +/****************************************************** + * gst_v4l2_get_input() + * Get the input of the current device + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_get_input (GstV4l2Element *v4l2element, + gint *input) +{ + gint n; + + DEBUG("trying to get input"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (ioctl(v4l2element->video_fd, VIDIOC_G_INPUT, &n) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get current input on device %s: %s", + v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + *input = n; + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_set_input() + * Set the input of the current device + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_set_input (GstV4l2Element *v4l2element, + gint input) +{ + DEBUG("trying to set input to %d", input); + GST_V4L2_CHECK_OPEN(v4l2element); + GST_V4L2_CHECK_NOT_ACTIVE(v4l2element); + + if (input < 0 || input >= g_list_length(v4l2element->inputs)) { + gst_element_error(GST_ELEMENT(v4l2element), + "Invalid input number %d (%d-%d)", + input, 0, g_list_length(v4l2element->inputs)); + return FALSE; + } + + if (ioctl(v4l2element->video_fd, VIDIOC_S_INPUT, &input) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to set input %d on device %s: %s", + input, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_get_input_names() + * Get the list of available input channels + * return value: the list + ******************************************************/ + +GList * +gst_v4l2_get_input_names (GstV4l2Element *v4l2element) +{ + GList *names = NULL; + gint n; + + DEBUG("getting a list of input names"); + + for (n=0;n<g_list_length(v4l2element->inputs);n++) { + struct v4l2_input *input = (struct v4l2_input *) g_list_nth_data(v4l2element->inputs, n); + names = g_list_append(names, g_strdup(input->name)); + } + + return names; +} + + +/****************************************************** + * gst_v4l2_get_output() + * Get the output of the current device + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_get_output (GstV4l2Element *v4l2element, + gint *output) +{ + gint n; + + DEBUG("trying to get output"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (ioctl(v4l2element->video_fd, VIDIOC_G_OUTPUT, &n) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get current output on device %s: %s", + v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + *output = n; + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_set_output() + * Set the output of the current device + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_set_output (GstV4l2Element *v4l2element, + gint output) +{ + DEBUG("trying to set output to %d", output); + GST_V4L2_CHECK_OPEN(v4l2element); + GST_V4L2_CHECK_NOT_ACTIVE(v4l2element); + + if (output < 0 || output >= g_list_length(v4l2element->outputs)) { + gst_element_error(GST_ELEMENT(v4l2element), + "Invalid output number %d (%d-%d)", + output, 0, g_list_length(v4l2element->outputs)); + return FALSE; + } + + if (ioctl(v4l2element->video_fd, VIDIOC_S_OUTPUT, &output) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to set output %d on device %s: %s", + output, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + return TRUE; +} + + +/****************************************************** + * gst_v4l2_get_output_names() + * Get the list of available output channels + * return value: the list, or NULL on error + ******************************************************/ + +GList * +gst_v4l2_get_output_names (GstV4l2Element *v4l2element) +{ + GList *names = NULL; + gint n; + + DEBUG("getting a list of output names"); + + for (n=0;n<g_list_length(v4l2element->outputs);n++) { + struct v4l2_output *output = (struct v4l2_output *) g_list_nth_data(v4l2element->outputs, n); + names = g_list_append(names, g_strdup(output->name)); + } + + return names; +} + + +/****************************************************** + * gst_v4l_has_tuner(): + * Check whether the device has a tuner + * return value: TRUE if it has a tuner, else FALSE + ******************************************************/ + +gboolean +gst_v4l2_has_tuner (GstV4l2Element *v4l2element) +{ + gint input_num; + struct v4l2_input *input; + + DEBUG("detecting whether device has a tuner"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (!gst_v4l2_get_input(v4l2element, &input_num)) + return FALSE; + + input = (struct v4l2_input *) g_list_nth_data(v4l2element->inputs, input_num); + + return (input->type == V4L2_INPUT_TYPE_TUNER && + v4l2element->vcap.flags & V4L2_FLAG_TUNER); +} + + +/****************************************************** + * gst_v4l_get_frequency(): + * get the current frequency + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_get_frequency (GstV4l2Element *v4l2element, + gulong *frequency) +{ + gint n; + + DEBUG("getting current tuner frequency"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (!gst_v4l2_has_tuner(v4l2element)) + return FALSE; + + if (ioctl(v4l2element->video_fd, VIDIOC_G_FREQ, &n) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get current tuner frequency for device %s: %s", + v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + *frequency = n; + + return TRUE; +} + + +/****************************************************** + * gst_v4l_set_frequency(): + * set frequency + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_set_frequency (GstV4l2Element *v4l2element, + gulong frequency) +{ + gint n = frequency; + + DEBUG("setting current tuner frequency to %lu", frequency); + GST_V4L2_CHECK_OPEN(v4l2element); + GST_V4L2_CHECK_NOT_ACTIVE(v4l2element); + + if (!gst_v4l2_has_tuner(v4l2element)) + return FALSE; + + if (ioctl(v4l2element->video_fd, VIDIOC_G_FREQ, &n) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to set tuner frequency to %lu for device %s: %s", + frequency, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + return TRUE; +} + + +/****************************************************** + * gst_v4l_signal_strength(): + * get the strength of the signal on the current input + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_signal_strength (GstV4l2Element *v4l2element, + gulong *signal_strength) +{ + struct v4l2_tuner tuner; + + DEBUG("trying to get signal strength"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (ioctl(v4l2element->video_fd, VIDIOC_G_TUNER, &tuner) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to set signal strength for device %s: %s", + v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + *signal_strength = tuner.signal; + + return TRUE; +} + + +/****************************************************** + * gst_v4l_has_audio(): + * Check whether the device has audio capabilities + * return value: TRUE if it has a tuner, else FALSE + ******************************************************/ + +gboolean +gst_v4l2_has_audio (GstV4l2Element *v4l2element) +{ + gint input_num; + struct v4l2_input *input; + + DEBUG("detecting whether device has audio"); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (!gst_v4l2_get_input(v4l2element, &input_num)) + return FALSE; + + input = (struct v4l2_input *) g_list_nth_data(v4l2element->inputs, input_num); + + return (input->capability & V4L2_INPUT_CAP_AUDIO); +} + + +/****************************************************** + * gst_v4l_get_attributes(): + * get a list of attributes available on this device + * return value: the list + ******************************************************/ + +GList * +gst_v4l2_get_attributes (GstV4l2Element *v4l2element) +{ + gint i; + GList *list = NULL; + + DEBUG("getting a list of available attributes"); + + for (i=0;i<g_list_length(v4l2element->controls);i++) { + struct v4l2_queryctrl *control = (struct v4l2_queryctrl *) g_list_nth_data(v4l2element->controls, i); + GstV4l2Attribute* attribute = g_malloc(sizeof(GstV4l2Attribute)); + attribute->name = g_strdup(control->name); + attribute->index = i; + attribute->list_items = NULL; + switch (control->type) { + case V4L2_CTRL_TYPE_INTEGER: + attribute->val_type = GST_V4L2_ATTRIBUTE_VALUE_TYPE_INT; + break; + case V4L2_CTRL_TYPE_BOOLEAN: + attribute->val_type = GST_V4L2_ATTRIBUTE_VALUE_TYPE_BOOLEAN; + break; + case V4L2_CTRL_TYPE_MENU: { + /* list items */ + gint n; + GList *menus = (GList *) g_list_nth_data(v4l2element->menus, i); + for (n=0;n<g_list_length(menus);n++) { + struct v4l2_querymenu *menu = g_list_nth_data(menus, n); + attribute->list_items = g_list_append(attribute->list_items, g_strdup(menu->name)); + } + attribute->val_type = GST_V4L2_ATTRIBUTE_VALUE_TYPE_LIST; + break; } + case V4L2_CTRL_TYPE_BUTTON: + attribute->val_type = GST_V4L2_ATTRIBUTE_VALUE_TYPE_BUTTON; + break; + } + switch (control->category) { + case V4L2_CTRL_CAT_VIDEO: + attribute->type = GST_V4L2_ATTRIBUTE_TYPE_VIDEO; + break; + case V4L2_CTRL_CAT_AUDIO: + attribute->type = GST_V4L2_ATTRIBUTE_TYPE_AUDIO; + break; + case V4L2_CTRL_CAT_EFFECT: + attribute->type = GST_V4L2_ATTRIBUTE_TYPE_EFFECT; + break; + } + gst_v4l2_get_attribute(v4l2element, i, &attribute->value); + attribute->min = control->minimum; + attribute->max = control->maximum; + } + + return list; +} + + +/****************************************************** + * gst_v4l_get_attribute(): + * try to get the value of one specific attribute + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_get_attribute (GstV4l2Element *v4l2element, + gint attribute_num, + gint *value) +{ + struct v4l2_control control; + + DEBUG("getting value of attribute %d", attribute_num); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (attribute_num < 0 || attribute_num >= g_list_length(v4l2element->controls)) { + gst_element_error(GST_ELEMENT(v4l2element), + "Invalid control ID %d", attribute_num); + return FALSE; + } + + control.id = attribute_num; + + if (ioctl(v4l2element->video_fd, VIDIOC_G_CTRL, &control) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to get value for control %d on device %s: %s", + attribute_num, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + *value = control.value; + + return TRUE; +} + + +/****************************************************** + * gst_v4l_set_attribute(): + * try to set the value of one specific attribute + * return value: TRUE on success, FALSE on error + ******************************************************/ + +gboolean +gst_v4l2_set_attribute (GstV4l2Element *v4l2element, + gint attribute_num, + gint value) +{ + struct v4l2_control control; + + DEBUG("setting value of attribute %d to %d", attribute_num, value); + GST_V4L2_CHECK_OPEN(v4l2element); + + if (attribute_num < 0 || attribute_num >= g_list_length(v4l2element->controls)) { + gst_element_error(GST_ELEMENT(v4l2element), + "Invalid control ID %d", attribute_num); + return FALSE; + } + + control.id = attribute_num; + control.value = value; + + if (ioctl(v4l2element->video_fd, VIDIOC_S_CTRL, &control) < 0) { + gst_element_error(GST_ELEMENT(v4l2element), + "Failed to set value %d for control %d on device %s: %s", + value, attribute_num, v4l2element->device, sys_errlist[errno]); + return FALSE; + } + + return TRUE; +} + |