diff options
author | Jacob Meuser <jakemsr@cvs.openbsd.org> | 2011-03-26 08:13:06 +0000 |
---|---|---|
committer | Jacob Meuser <jakemsr@cvs.openbsd.org> | 2011-03-26 08:13:06 +0000 |
commit | 0ce7af11d44b572c932838786f93d41e47de1d4c (patch) | |
tree | b7b1713512ff913adc0dd478b0f17abcb8776d2e /sys/dev/usb/uvideo.c | |
parent | 14fecd0c78405a2d0d4cefc5045fb02d9a6ac2ff (diff) |
support VIDIOC_S_PARM and VIDIOC_G_PARM. these ioctls are used for
setting and and getting the frame interval.
Diffstat (limited to 'sys/dev/usb/uvideo.c')
-rw-r--r-- | sys/dev/usb/uvideo.c | 120 |
1 files changed, 114 insertions, 6 deletions
diff --git a/sys/dev/usb/uvideo.c b/sys/dev/usb/uvideo.c index 518eb39d064..e901cc6537b 100644 --- a/sys/dev/usb/uvideo.c +++ b/sys/dev/usb/uvideo.c @@ -1,4 +1,4 @@ -/* $OpenBSD: uvideo.c,v 1.153 2011/03/26 07:57:42 jakemsr Exp $ */ +/* $OpenBSD: uvideo.c,v 1.154 2011/03/26 08:13:05 jakemsr Exp $ */ /* * Copyright (c) 2008 Robert Nagy <robert@openbsd.org> @@ -174,6 +174,8 @@ 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_s_parm(void *, struct v4l2_streamparm *); +int uvideo_g_parm(void *, struct v4l2_streamparm *); int uvideo_enum_input(void *, struct v4l2_input *); int uvideo_s_input(void *, int); int uvideo_reqbufs(void *, struct v4l2_requestbuffers *); @@ -221,6 +223,8 @@ struct video_hw_if uvideo_hw_if = { uvideo_enum_fivals, /* VIDIOC_ENUM_FRAMEINTERVALS */ uvideo_s_fmt, /* VIDIOC_S_FMT */ uvideo_g_fmt, /* VIDIOC_G_FMT */ + uvideo_s_parm, /* VIDIOC_S_PARM */ + uvideo_g_parm, /* VIDIOC_G_PARM */ uvideo_enum_input, /* VIDIOC_ENUMINPUT */ uvideo_s_input, /* VIDIOC_S_INPUT */ uvideo_reqbufs, /* VIDIOC_REQBUFS */ @@ -1309,13 +1313,19 @@ usbd_status uvideo_vs_negotiation(struct uvideo_softc *sc, int commit) { struct usb_video_probe_commit *pc; + struct uvideo_format_group *fmtgrp; + uint8_t *p, *cur; uint8_t probe_data[34]; + uint32_t frame_ival, nivals, min, max, step, diff; usbd_status error; + int i, ival_bytes; pc = (struct usb_video_probe_commit *)probe_data; + fmtgrp = sc->sc_fmtgrp_cur; + /* check if the format descriptor contains frame descriptors */ - if (sc->sc_fmtgrp_cur->frame_num == 0) { + if (fmtgrp->frame_num == 0) { printf("%s: %s: no frame descriptors found!\n", __func__, DEVNAME(sc)); return (USBD_INVAL); @@ -1323,12 +1333,63 @@ uvideo_vs_negotiation(struct uvideo_softc *sc, int commit) /* set probe */ bzero(probe_data, sizeof(probe_data)); + /* hint that dwFrameInterval should be favored over other parameters */ USETW(pc->bmHint, 0x1); - pc->bFormatIndex = sc->sc_fmtgrp_cur->format->bFormatIndex; - pc->bFrameIndex = sc->sc_fmtgrp_cur->frame_cur->bFrameIndex; + pc->bFormatIndex = fmtgrp->format->bFormatIndex; + pc->bFrameIndex = fmtgrp->frame_cur->bFrameIndex; /* dwFrameInterval: 30fps=333333, 15fps=666666, 10fps=1000000 */ - USETDW(pc->dwFrameInterval, - UGETDW(sc->sc_fmtgrp_cur->frame_cur->dwDefaultFrameInterval)); + frame_ival = UGETDW(fmtgrp->frame_cur->dwDefaultFrameInterval); + if (sc->sc_frame_rate != 0) { + frame_ival = 10000000 / sc->sc_frame_rate; + /* find closest matching interval the device supports */ + p = (uint8_t *)fmtgrp->frame_cur; + p += sizeof(struct usb_video_frame_desc); + nivals = fmtgrp->frame_cur->bFrameIntervalType; + ival_bytes = fmtgrp->frame_cur->bLength - + sizeof(struct usb_video_frame_desc); + if (!nivals && (ival_bytes >= sizeof(uDWord) * 3)) { + /* continuous */ + min = UGETDW(p); + p += sizeof(uDWord); + max = UGETDW(p); + p += sizeof(uDWord); + step = UGETDW(p); + p += sizeof(uDWord); + if (frame_ival <= min) + frame_ival = min; + else if (frame_ival >= max) + frame_ival = max; + else { + for (i = min; i + step/2 < frame_ival; i+= step) + ; /* nothing */ + frame_ival = i; + } + } else if (nivals > 0 && ival_bytes >= sizeof(uDWord)) { + /* discrete */ + cur = p; + min = UINT_MAX; + for (i = 0; i < nivals; i++) { + if (ival_bytes < sizeof(uDWord)) { + /* short descriptor ? */ + break; + } + diff = abs(UGETDW(p) - frame_ival); + if (diff < min) { + min = diff; + cur = p; + if (diff == 0) + break; + } + p += sizeof(uDWord); + ival_bytes -= sizeof(uDWord); + } + frame_ival = UGETDW(cur); + } else { + DPRINTF(1, "%s: %s: bad frame ival descriptor\n", + DEVNAME(sc), __func__); + } + } + USETDW(pc->dwFrameInterval, frame_ival); error = uvideo_vs_set_probe(sc, probe_data); if (error != USBD_NORMAL_COMPLETION) return (error); @@ -2860,6 +2921,53 @@ uvideo_g_fmt(void *v, struct v4l2_format *fmt) } int +uvideo_s_parm(void *v, struct v4l2_streamparm *parm) +{ + struct uvideo_softc *sc = v; + usbd_status error; + + if (parm->type == V4L2_BUF_TYPE_VIDEO_CAPTURE) { + if (parm->parm.capture.timeperframe.numerator == 0 || + parm->parm.capture.timeperframe.denominator == 0) + return (EINVAL); + + /* only whole number frame rates for now */ + sc->sc_frame_rate = + parm->parm.capture.timeperframe.numerator / + parm->parm.capture.timeperframe.denominator; + } else + return (EINVAL); + + /* renegotiate if necessary */ + if (sc->sc_negotiated_flag) { + error = uvideo_vs_negotiation(sc, 1); + if (error != USBD_NORMAL_COMPLETION) + return (error); + } + + return (0); +} + +int +uvideo_g_parm(void *v, struct v4l2_streamparm *parm) +{ + struct uvideo_softc *sc = v; + int ns; + + ns = UGETDW(sc->sc_desc_probe.dwFrameInterval); + if (parm->type == V4L2_BUF_TYPE_VIDEO_CAPTURE) { + parm->parm.capture.capability = V4L2_CAP_TIMEPERFRAME; + parm->parm.capture.capturemode = 0; + parm->parm.capture.timeperframe.numerator = + (ns == 0) ? 0 : 10000000 / ns; + parm->parm.capture.timeperframe.denominator = 1; + } else + return (EINVAL); + + return (0); +} + +int uvideo_enum_input(void *v, struct v4l2_input *input) { if (input->index != 0) |