options: Remove unnecessary commas from EBNF syntax
[media-ctl.git] / src / v4l2subdev.c
index cf9c347..a2ab0c4 100644 (file)
@@ -1,20 +1,22 @@
 /*
  * V4L2 subdev interface library
  *
- * Copyright (C) 2010 Ideas on board SPRL <laurent.pinchart@ideasonboard.com>
+ * Copyright (C) 2010-2011 Ideas on board SPRL
+ *
+ * Contact: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
  *
  * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
+ * it under the terms of the GNU Lesser General Public License as published
+ * by the Free Software Foundation; either version 2.1 of the License, or
  * (at your option) any later version.
  *
  * This program 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 General Public License for more details.
+ * GNU Lesser General Public License for more details.
  *
- * You should have received a copy of the GNU General Public License along
- * with this program; if not, write to the Free Software Foundation, Inc.,
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
  */
 
 #include <sys/ioctl.h>
@@ -102,48 +104,83 @@ int v4l2_subdev_set_format(struct media_entity *entity,
        return 0;
 }
 
-int v4l2_subdev_get_crop(struct media_entity *entity, struct v4l2_rect *rect,
-                        unsigned int pad, enum v4l2_subdev_format_whence which)
+int v4l2_subdev_get_selection(struct media_entity *entity,
+       struct v4l2_rect *rect, unsigned int pad, unsigned int target,
+       enum v4l2_subdev_format_whence which)
 {
-       struct v4l2_subdev_crop crop;
+       union {
+               struct v4l2_subdev_selection sel;
+               struct v4l2_subdev_crop crop;
+       } u;
        int ret;
 
        ret = v4l2_subdev_open(entity);
        if (ret < 0)
                return ret;
 
-       memset(&crop, 0, sizeof(crop));
-       crop.pad = pad;
-       crop.which = which;
+       memset(&u.sel, 0, sizeof(u.sel));
+       u.sel.pad = pad;
+       u.sel.target = target;
+       u.sel.which = which;
+
+       ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_SELECTION, &u.sel);
+       if (ret >= 0) {
+               *rect = u.sel.r;
+               return 0;
+       }
+       if (errno != ENOTTY || target != V4L2_SUBDEV_SEL_TGT_CROP_ACTUAL)
+               return -errno;
+
+       memset(&u.crop, 0, sizeof(u.crop));
+       u.crop.pad = pad;
+       u.crop.which = which;
 
-       ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_CROP, &crop);
+       ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_CROP, &u.crop);
        if (ret < 0)
                return -errno;
 
-       *rect = crop.rect;
+       *rect = u.crop.rect;
        return 0;
 }
 
-int v4l2_subdev_set_crop(struct media_entity *entity, struct v4l2_rect *rect,
-                        unsigned int pad, enum v4l2_subdev_format_whence which)
+int v4l2_subdev_set_selection(struct media_entity *entity,
+       struct v4l2_rect *rect, unsigned int pad, unsigned int target,
+       enum v4l2_subdev_format_whence which)
 {
-       struct v4l2_subdev_crop crop;
+       union {
+               struct v4l2_subdev_selection sel;
+               struct v4l2_subdev_crop crop;
+       } u;
        int ret;
 
        ret = v4l2_subdev_open(entity);
        if (ret < 0)
                return ret;
 
-       memset(&crop, 0, sizeof(crop));
-       crop.pad = pad;
-       crop.which = which;
-       crop.rect = *rect;
+       memset(&u.sel, 0, sizeof(u.sel));
+       u.sel.pad = pad;
+       u.sel.target = target;
+       u.sel.which = which;
+       u.sel.r = *rect;
+
+       ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_SELECTION, &u.sel);
+       if (ret >= 0) {
+               *rect = u.sel.r;
+               return 0;
+       }
+       if (errno != ENOTTY || target != V4L2_SUBDEV_SEL_TGT_CROP_ACTUAL)
+               return -errno;
+
+       memset(&u.crop, 0, sizeof(u.crop));
+       u.crop.pad = pad;
+       u.crop.which = which;
+       u.crop.rect = *rect;
 
-       ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_CROP, &crop);
+       ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_CROP, &u.crop);
        if (ret < 0)
                return -errno;
 
-       *rect = crop.rect;
+       *rect = u.crop.rect;
        return 0;
 }
 
@@ -353,30 +390,31 @@ static int set_format(struct media_pad *pad,
        return 0;
 }
 
-static int set_crop(struct media_pad *pad, struct v4l2_rect *crop)
+static int set_selection(struct media_pad *pad, unsigned int target,
+                        struct v4l2_rect *rect)
 {
        int ret;
 
-       if (crop->left == -1 || crop->top == -1)
+       if (rect->left == -1 || rect->top == -1)
                return 0;
 
        media_dbg(pad->entity->media,
-                 "Setting up crop rectangle (%u,%u)/%ux%u on pad %s/%u\n",
-                 crop->left, crop->top, crop->width, crop->height,
+                 "Setting up selection target %u rectangle (%u,%u)/%ux%u on pad %s/%u\n",
+                 target, rect->left, rect->top, rect->width, rect->height,
                  pad->entity->info.name, pad->index);
 
-       ret = v4l2_subdev_set_crop(pad->entity, crop, pad->index,
-                                  V4L2_SUBDEV_FORMAT_ACTIVE);
+       ret = v4l2_subdev_set_selection(pad->entity, rect, pad->index,
+                                       target, V4L2_SUBDEV_FORMAT_ACTIVE);
        if (ret < 0) {
                media_dbg(pad->entity->media,
-                         "Unable to set crop rectangle: %s (%d)\n",
+                         "Unable to set selection rectangle: %s (%d)\n",
                          strerror(-ret), ret);
                return ret;
        }
 
        media_dbg(pad->entity->media,
-                 "Crop rectangle set: (%u,%u)/%ux%u\n",
-                 crop->left, crop->top, crop->width, crop->height);
+                 "Selection rectangle set: (%u,%u)/%ux%u\n",
+                 rect->left, rect->top, rect->width, rect->height);
 
        return 0;
 }
@@ -428,7 +466,7 @@ static int v4l2_subdev_parse_setup_format(struct media_device *media,
        }
 
        if (pad->flags & MEDIA_PAD_FL_SOURCE) {
-               ret = set_crop(pad, &crop);
+               ret = set_selection(pad, V4L2_SUBDEV_SEL_TGT_CROP_ACTUAL, &crop);
                if (ret < 0)
                        return ret;
        }
@@ -438,7 +476,7 @@ static int v4l2_subdev_parse_setup_format(struct media_device *media,
                return ret;
 
        if (pad->flags & MEDIA_PAD_FL_SINK) {
-               ret = set_crop(pad, &crop);
+               ret = set_selection(pad, V4L2_SUBDEV_SEL_TGT_CROP_ACTUAL, &crop);
                if (ret < 0)
                        return ret;
        }
@@ -495,8 +533,10 @@ static struct {
        { "Y10", V4L2_MBUS_FMT_Y10_1X10 },
        { "Y12", V4L2_MBUS_FMT_Y12_1X12 },
        { "YUYV", V4L2_MBUS_FMT_YUYV8_1X16 },
+       { "YUYV1_5X8", V4L2_MBUS_FMT_YUYV8_1_5X8 },
        { "YUYV2X8", V4L2_MBUS_FMT_YUYV8_2X8 },
        { "UYVY", V4L2_MBUS_FMT_UYVY8_1X16 },
+       { "UYVY1_5X8", V4L2_MBUS_FMT_UYVY8_1_5X8 },
        { "UYVY2X8", V4L2_MBUS_FMT_UYVY8_2X8 },
        { "SBGGR8", V4L2_MBUS_FMT_SBGGR8_1X8 },
        { "SGBRG8", V4L2_MBUS_FMT_SGBRG8_1X8 },