diff options
author | Marcus Glocker <mglocker@cvs.openbsd.org> | 2008-08-13 20:29:35 +0000 |
---|---|---|
committer | Marcus Glocker <mglocker@cvs.openbsd.org> | 2008-08-13 20:29:35 +0000 |
commit | b836f59f2bb98f50cbe6fcaf1ece40f5959a37b8 (patch) | |
tree | 0a1dabc108a0911bfd7e14d7ca8e306408547854 | |
parent | ab11b24f7dcb98e06df61cd23ffdd66ccf568f92 (diff) |
Add VIDIOC_ENUM_FRAMESIZES ioctl. This permits applications to query the
available formats and resolutions of a device (e.g. "luvcview -L").
-rw-r--r-- | sys/dev/usb/uvideo.c | 64 | ||||
-rw-r--r-- | sys/dev/video.c | 12 | ||||
-rw-r--r-- | sys/dev/video_if.h | 4 |
3 files changed, 77 insertions, 3 deletions
diff --git a/sys/dev/usb/uvideo.c b/sys/dev/usb/uvideo.c index e07a6aa2e47..4d6b12734ca 100644 --- a/sys/dev/usb/uvideo.c +++ b/sys/dev/usb/uvideo.c @@ -1,4 +1,4 @@ -/* $OpenBSD: uvideo.c,v 1.80 2008/08/12 08:26:42 mglocker Exp $ */ +/* $OpenBSD: uvideo.c,v 1.81 2008/08/13 20:29:34 mglocker Exp $ */ /* * Copyright (c) 2008 Robert Nagy <robert@openbsd.org> @@ -149,6 +149,8 @@ void uvideo_debug_file_write_frame(void *); */ int uvideo_querycap(void *, struct v4l2_capability *); int uvideo_enum_fmt(void *, struct v4l2_fmtdesc *); +int uvideo_enum_fsizes(void *, struct v4l2_frmsizeenum *); +int uvideo_enum_fivals(void *, struct v4l2_frmivalenum *); int uvideo_s_fmt(void *, struct v4l2_format *); int uvideo_g_fmt(void *, struct v4l2_format *); int uvideo_enum_input(void *, struct v4l2_input *); @@ -185,6 +187,8 @@ struct video_hw_if uvideo_hw_if = { uvideo_close, /* close */ uvideo_querycap, /* VIDIOC_QUERYCAP */ uvideo_enum_fmt, /* VIDIOC_ENUM_FMT */ + uvideo_enum_fsizes, /* VIDIOC_ENUM_FRAMESIZES */ + uvideo_enum_fivals, /* VIDIOC_ENUM_FRAMEINTERVALS */ uvideo_s_fmt, /* VIDIOC_S_FMT */ uvideo_g_fmt, /* VIDIOC_G_FMT */ uvideo_enum_input, /* VIDIOC_ENUMINPUT */ @@ -2158,6 +2162,64 @@ uvideo_enum_fmt(void *v, struct v4l2_fmtdesc *fmtdesc) } int +uvideo_enum_fsizes(void *v, struct v4l2_frmsizeenum *fsizes) +{ + struct uvideo_softc *sc = v; + int i, idx, found = 0; + + for (idx = 0; idx < sc->sc_fmtgrp_num; idx++) { + if (sc->sc_fmtgrp[idx].pixelformat == fsizes->pixel_format) { + found = 1; + break; + } + } + if (found == 0) + return (EINVAL); + + i = fsizes->index + 1; + if (i > sc->sc_fmtgrp[idx].frame_num) + /* no more frames left */ + return (EINVAL); + + if (sc->sc_fmtgrp[idx].frame[i]->bFrameIntervalType == 0) { + /* TODO */ + fsizes->type = V4L2_FRMSIZE_TYPE_CONTINUOUS; + fsizes->stepwise.min_width = 0; + fsizes->stepwise.min_height = 0; + fsizes->stepwise.max_width = 0; + fsizes->stepwise.max_height = 0; + } else { + fsizes->type = V4L2_FRMSIZE_TYPE_DISCRETE; + fsizes->discrete.width = + UGETW(sc->sc_fmtgrp[idx].frame[i]->wWidth); + fsizes->discrete.height = + UGETW(sc->sc_fmtgrp[idx].frame[i]->wHeight); + } + + return (0); +} + +int +uvideo_enum_fivals(void *v, struct v4l2_frmivalenum *fivals) +{ + struct uvideo_softc *sc = v; + int idx, found = 0; + + for (idx = 0; idx < sc->sc_fmtgrp_num; idx++) { + if (sc->sc_fmtgrp[idx].pixelformat == fivals->pixel_format) { + found = 1; + break; + } + } + if (found == 0) + return (EINVAL); + + /* TODO */ + + return (EINVAL); +} + +int uvideo_s_fmt(void *v, struct v4l2_format *fmt) { struct uvideo_softc *sc = v; diff --git a/sys/dev/video.c b/sys/dev/video.c index dd741a85ac6..0f8e3cc2414 100644 --- a/sys/dev/video.c +++ b/sys/dev/video.c @@ -1,4 +1,4 @@ -/* $OpenBSD: video.c,v 1.20 2008/07/31 15:26:25 mglocker Exp $ */ +/* $OpenBSD: video.c,v 1.21 2008/08/13 20:29:34 mglocker Exp $ */ /* * Copyright (c) 2008 Robert Nagy <robert@openbsd.org> * Copyright (c) 2008 Marcus Glocker <mglocker@openbsd.org> @@ -199,6 +199,16 @@ videoioctl(dev_t dev, u_long cmd, caddr_t data, int flags, struct proc *p) error = (sc->hw_if->enum_fmt)(sc->hw_hdl, (struct v4l2_fmtdesc *)data); break; + case VIDIOC_ENUM_FRAMESIZES: + if (sc->hw_if->enum_fsizes) + error = (sc->hw_if->enum_fsizes)(sc->hw_hdl, + (struct v4l2_frmsizeenum *)data); + break; + case VIDIOC_ENUM_FRAMEINTERVALS: + if (sc->hw_if->enum_fivals) + error = (sc->hw_if->enum_fivals)(sc->hw_hdl, + (struct v4l2_frmivalenum *)data); + break; case VIDIOC_S_FMT: if (!(flags & FWRITE)) return (EACCES); diff --git a/sys/dev/video_if.h b/sys/dev/video_if.h index 69ff3174164..ea8836aa61a 100644 --- a/sys/dev/video_if.h +++ b/sys/dev/video_if.h @@ -1,4 +1,4 @@ -/* $OpenBSD: video_if.h,v 1.14 2008/07/31 15:26:25 mglocker Exp $ */ +/* $OpenBSD: video_if.h,v 1.15 2008/08/13 20:29:34 mglocker Exp $ */ /* * Copyright (c) 2008 Robert Nagy <robert@openbsd.org> * Copyright (c) 2008 Marcus Glocker <mglocker@openbsd.org> @@ -36,6 +36,8 @@ struct video_hw_if { /* ioctl's */ int (*querycap)(void *, struct v4l2_capability *); int (*enum_fmt)(void *, struct v4l2_fmtdesc *); + int (*enum_fsizes)(void *, struct v4l2_frmsizeenum *); + int (*enum_fivals)(void *, struct v4l2_frmivalenum *); int (*s_fmt)(void *, struct v4l2_format *); int (*g_fmt)(void *, struct v4l2_format *); int (*enum_input)(void *, struct v4l2_input *); |