# HG changeset patch # User Taylor R Campbell # Date 1734197238 0 # Sat Dec 14 17:27:18 2024 +0000 # Branch trunk # Node ID d2cfe16f944c79bf27b4bbced8978f6471ede6a0 # Parent b45077e9d43a44f447dc69ccea01027bfe07bd30 # EXP-Topic riastradh-pr58898-videolocking video(4): KNF Also nix #if NVIDEO > 0. This is already conditional on the video attribute in files.dev, so this condition is always true. No functional change intended. diff -r b45077e9d43a -r d2cfe16f944c sys/dev/video.c --- a/sys/dev/video.c Wed Dec 11 12:56:31 2024 +0000 +++ b/sys/dev/video.c Sat Dec 14 17:27:18 2024 +0000 @@ -39,22 +39,23 @@ __KERNEL_RCSID(0, "$NetBSD: video.c,v 1.45 2022/03/03 06:23:25 riastradh Exp $"); #include "video.h" -#if NVIDEO > 0 #include -#include +#include + +#include +#include +#include #include -#include +#include +#include #include -#include -#include #include -#include +#include +#include #include -#include -#include -#include #include +#include #include @@ -64,11 +65,11 @@ #ifdef VIDEO_DEBUG #define DPRINTF(x) do { if (videodebug) printf x; } while (0) -#define DPRINTFN(n,x) do { if (videodebug>(n)) printf x; } while (0) +#define DPRINTFN(n,x) do { if (videodebug > (n)) printf x; } while (0) int videodebug = VIDEO_DEBUG; #else -#define DPRINTF(x) -#define DPRINTFN(n,x) +#define DPRINTF(x) __nothing +#define DPRINTFN(n,x) __nothing #endif #define PAGE_ALIGN(a) (((a) + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1)) @@ -86,9 +87,11 @@ int videodebug = VIDEO_DEBUG; #define VIDEO_MAX_BUFS 32 #define VIDEO_NUM_BUFS 4 -/* Scatter Buffer - an array of fixed size (PAGE_SIZE) chunks +/* + * Scatter Buffer - an array of fixed size (PAGE_SIZE) chunks * allocated non-contiguously and functions to get data into and out - * of the scatter buffer. */ + * of the scatter buffer. + */ struct scatter_buf { pool_cache_t sb_pool; size_t sb_size; /* size in bytes */ @@ -107,14 +110,16 @@ static void scatter_buf_destroy(struct s static int scatter_buf_set_size(struct scatter_buf *, size_t); static paddr_t scatter_buf_map(struct scatter_buf *, off_t); -static bool scatter_io_init(struct scatter_buf *, off_t, size_t, struct scatter_io *); +static bool scatter_io_init(struct scatter_buf *, off_t, size_t, + struct scatter_io *); static bool scatter_io_next(struct scatter_io *, void **, size_t *); static void scatter_io_undo(struct scatter_io *, size_t); static void scatter_io_copyin(struct scatter_io *, const void *); -/* static void scatter_io_copyout(struct scatter_io *, void *); */ +#if 0 +static void scatter_io_copyout(struct scatter_io *, void *); +#endif static int scatter_io_uiomove(struct scatter_io *, struct uio *); - enum video_stream_method { VIDEO_STREAM_METHOD_NONE, VIDEO_STREAM_METHOD_READ, @@ -148,7 +153,8 @@ struct video_stream { struct scatter_buf vs_data; /* stores video data for MMAP * and READ */ - /* Video samples may exist in different locations. Initially, + /* + * Video samples may exist in different locations. Initially, * samples are queued into the ingress queue. The driver * grabs these in turn and fills them with video data. Once * filled, they are moved to the egress queue. Samples are @@ -157,7 +163,8 @@ struct video_stream { * ingress queue without dequeing. In the first case, the * user re-queues the buffer when finished, and videoread() * does the same when all data has been read. The sample now - * returns to the ingress queue. */ + * returns to the ingress queue. + */ struct sample_queue vs_ingress; /* samples under driver control */ struct sample_queue vs_egress; /* samples headed for userspace */ @@ -224,50 +231,49 @@ const struct cdevsw video_cdevsw = { #define VIDEOUNIT(n) (minor(n)) CFATTACH_DECL_NEW(video, sizeof(struct video_softc), - video_match, video_attach, video_detach, video_activate); + video_match, video_attach, video_detach, video_activate); static const char * video_pixel_format_str(enum video_pixel_format); -/* convert various values from V4L2 to native values of this driver */ +/* + * convert various values from V4L2 to native values of this driver + */ static uint16_t v4l2id_to_control_id(uint32_t); static uint32_t control_flags_to_v4l2flags(uint32_t); static enum v4l2_ctrl_type control_type_to_v4l2type(enum video_control_type); static void v4l2_format_to_video_format(const struct v4l2_format *, - struct video_format *); + struct video_format *); static void video_format_to_v4l2_format(const struct video_format *, - struct v4l2_format *); + struct v4l2_format *); static void v4l2_standard_to_video_standard(v4l2_std_id, - enum video_standard *); + enum video_standard *); static void video_standard_to_v4l2_standard(enum video_standard, - struct v4l2_standard *); + struct v4l2_standard *); static void v4l2_input_to_video_input(const struct v4l2_input *, - struct video_input *); + struct video_input *); static void video_input_to_v4l2_input(const struct video_input *, - struct v4l2_input *); + struct v4l2_input *); static void v4l2_audio_to_video_audio(const struct v4l2_audio *, - struct video_audio *); + struct video_audio *); static void video_audio_to_v4l2_audio(const struct video_audio *, - struct v4l2_audio *); + struct v4l2_audio *); static void v4l2_tuner_to_video_tuner(const struct v4l2_tuner *, - struct video_tuner *); + struct video_tuner *); static void video_tuner_to_v4l2_tuner(const struct video_tuner *, - struct v4l2_tuner *); - -/* V4L2 api functions, typically called from videoioctl() */ + struct v4l2_tuner *); + +/* + * V4L2 api functions, typically called from videoioctl() + */ static int video_enum_format(struct video_softc *, struct v4l2_fmtdesc *); -static int video_get_format(struct video_softc *, - struct v4l2_format *); -static int video_set_format(struct video_softc *, - struct v4l2_format *); -static int video_try_format(struct video_softc *, - struct v4l2_format *); -static int video_get_parm(struct video_softc *, - struct v4l2_streamparm *); -static int video_set_parm(struct video_softc *, - struct v4l2_streamparm *); +static int video_get_format(struct video_softc *, struct v4l2_format *); +static int video_set_format(struct video_softc *, struct v4l2_format *); +static int video_try_format(struct video_softc *, struct v4l2_format *); +static int video_get_parm(struct video_softc *, struct v4l2_streamparm *); +static int video_set_parm(struct video_softc *, struct v4l2_streamparm *); static int video_enum_standard(struct video_softc *, - struct v4l2_standard *); + struct v4l2_standard *); static int video_get_standard(struct video_softc *, v4l2_std_id *); static int video_set_standard(struct video_softc *, v4l2_std_id); static int video_enum_input(struct video_softc *, struct v4l2_input *); @@ -279,17 +285,16 @@ static int video_set_audio(struct video_ static int video_get_tuner(struct video_softc *, struct v4l2_tuner *); static int video_set_tuner(struct video_softc *, struct v4l2_tuner *); static int video_get_frequency(struct video_softc *, - struct v4l2_frequency *); + struct v4l2_frequency *); static int video_set_frequency(struct video_softc *, - struct v4l2_frequency *); + struct v4l2_frequency *); static int video_query_control(struct video_softc *, - struct v4l2_queryctrl *); -static int video_get_control(struct video_softc *, - struct v4l2_control *); + struct v4l2_queryctrl *); +static int video_get_control(struct video_softc *, struct v4l2_control *); static int video_set_control(struct video_softc *, - const struct v4l2_control *); + const struct v4l2_control *); static int video_request_bufs(struct video_softc *, - struct v4l2_requestbuffers *); + struct v4l2_requestbuffers *); static int video_query_buf(struct video_softc *, struct v4l2_buffer *); static int video_queue_buf(struct video_softc *, struct v4l2_buffer *); static int video_dequeue_buf(struct video_softc *, struct v4l2_buffer *); @@ -299,14 +304,15 @@ static int video_stream_off(struct video static struct video_buffer * video_buffer_alloc(void); static void video_buffer_free(struct video_buffer *); - -/* functions for video_stream */ +/* + * functions for video_stream + */ static void video_stream_init(struct video_stream *); static void video_stream_fini(struct video_stream *); static int video_stream_setup_bufs(struct video_stream *, - enum video_stream_method, - uint8_t); + enum video_stream_method, + uint8_t); static void video_stream_teardown_bufs(struct video_stream *); static int video_stream_realloc_bufs(struct video_stream *, uint8_t); @@ -314,10 +320,10 @@ static int video_stream_realloc_bufs(str video_stream_realloc_bufs((vs), 0) static void video_stream_enqueue(struct video_stream *, - struct video_buffer *); + struct video_buffer *); static struct video_buffer * video_stream_dequeue(struct video_stream *); static void video_stream_write(struct video_stream *, - const struct video_payload *); + const struct video_payload *); static void video_stream_sample_done(struct video_stream *); #ifdef VIDEO_DEBUG @@ -325,7 +331,6 @@ static void video_stream_sample_done(str static const char * video_ioctl_str(u_long); #endif - static int video_match(device_t parent, cfdata_t match, void *aux) { @@ -338,7 +343,6 @@ video_match(device_t parent, cfdata_t ma return 1; } - static void video_attach(device_t parent, device_t self, void *aux) { @@ -369,7 +373,6 @@ video_attach(device_t parent, device_t s aprint_error_dev(self, "couldn't establish power handler\n"); } - static int video_activate(device_t self, enum devact act) { @@ -385,7 +388,6 @@ video_activate(device_t self, enum devac } } - static int video_detach(device_t self, int flags) { @@ -409,10 +411,10 @@ video_detach(device_t self, int flags) return 0; } - static int video_print(void *aux, const char *pnp) { + if (pnp != NULL) { DPRINTF(("video_print: have pnp\n")); aprint_normal("%s at %s\n", "video", pnp); @@ -422,7 +424,6 @@ video_print(void *aux, const char *pnp) return UNCONF; } - /* * Called from hardware driver. This is where the MI audio driver * gets probed/attached to the hardware driver. @@ -438,7 +439,9 @@ video_attach_mi(const struct video_hw_if CFARGS(.iattr = "videobus")); } -/* video_submit_payload - called by hardware driver to submit payload data */ +/* + * video_submit_payload - called by hardware driver to submit payload data + */ void video_submit_payload(device_t self, const struct video_payload *payload) { @@ -455,6 +458,7 @@ video_submit_payload(device_t self, cons static const char * video_pixel_format_str(enum video_pixel_format px) { + switch (px) { case VIDEO_FORMAT_UYVY: return "UYVY"; case VIDEO_FORMAT_YUV420: return "YUV420"; @@ -471,11 +475,15 @@ video_pixel_format_str(enum video_pixel_ } } -/* Takes a V4L2 id and returns a "native" video driver control id. - * TODO: is there a better way to do this? some kind of array? */ +/* + * Takes a V4L2 id and returns a "native" video driver control id. + * + * TODO: is there a better way to do this? some kind of array? + */ static uint16_t v4l2id_to_control_id(uint32_t v4l2id) { + /* mask includes class bits and control id bits */ switch (v4l2id & 0xffffff) { case V4L2_CID_BRIGHTNESS: return VIDEO_CONTROL_BRIGHTNESS; @@ -486,9 +494,11 @@ v4l2id_to_control_id(uint32_t v4l2id) case V4L2_CID_SHARPNESS: return VIDEO_CONTROL_SHARPNESS; case V4L2_CID_GAMMA: return VIDEO_CONTROL_GAMMA; - /* "black level" means the same as "brightness", but V4L2 + /* + * "black level" means the same as "brightness", but V4L2 * defines two separate controls that are not identical. - * V4L2_CID_BLACK_LEVEL is deprecated however in V4L2. */ + * V4L2_CID_BLACK_LEVEL is deprecated however in V4L2. + */ case V4L2_CID_BLACK_LEVEL: return VIDEO_CONTROL_BRIGHTNESS; case V4L2_CID_AUDIO_VOLUME: return VIDEO_CONTROL_UNDEFINED; @@ -525,7 +535,6 @@ v4l2id_to_control_id(uint32_t v4l2id) } } - static uint32_t control_flags_to_v4l2flags(uint32_t flags) { @@ -543,22 +552,22 @@ control_flags_to_v4l2flags(uint32_t flag return v4l2flags; } - static enum v4l2_ctrl_type -control_type_to_v4l2type(enum video_control_type type) { +control_type_to_v4l2type(enum video_control_type type) +{ + switch (type) { case VIDEO_CONTROL_TYPE_INT: return V4L2_CTRL_TYPE_INTEGER; case VIDEO_CONTROL_TYPE_BOOL: return V4L2_CTRL_TYPE_BOOLEAN; case VIDEO_CONTROL_TYPE_LIST: return V4L2_CTRL_TYPE_MENU; case VIDEO_CONTROL_TYPE_ACTION: return V4L2_CTRL_TYPE_BUTTON; - default: return V4L2_CTRL_TYPE_INTEGER; /* err? */ + /* err? */ + default: return V4L2_CTRL_TYPE_INTEGER; } } - static int -video_query_control(struct video_softc *sc, - struct v4l2_queryctrl *query) +video_query_control(struct video_softc *sc, struct v4l2_queryctrl *query) { const struct video_hw_if *hw; struct video_control_desc_group desc_group; @@ -592,12 +601,12 @@ video_query_control(struct video_softc * } } - -/* Takes a single Video4Linux2 control and queries the driver for the - * current value. */ +/* + * Takes a single Video4Linux2 control and queries the driver for the + * current value. + */ static int -video_get_control(struct video_softc *sc, - struct v4l2_control *vcontrol) +video_get_control(struct video_softc *sc, struct v4l2_control *vcontrol) { const struct video_hw_if *hw; struct video_control_group group; @@ -608,9 +617,11 @@ video_get_control(struct video_softc *sc if (hw->get_control_group) { control.group_id = control.control_id = v4l2id_to_control_id(vcontrol->id); - /* ?? if "control_id" is arbitrarily defined by the + /* + * ?? if "control_id" is arbitrarily defined by the * driver, then we need some way to store it... Maybe - * it doesn't matter for single value controls. */ + * it doesn't matter for single value controls. + */ control.value = 0; group.group_id = control.group_id; @@ -630,8 +641,9 @@ video_get_control(struct video_softc *sc static void video_format_to_v4l2_format(const struct video_format *src, - struct v4l2_format *dest) + struct v4l2_format *dest) { + /* TODO: what about win and vbi formats? */ dest->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; dest->fmt.pix.width = src->width; @@ -692,19 +704,18 @@ video_format_to_v4l2_format(const struct case VIDEO_FORMAT_UNDEFINED: default: DPRINTF(("video_get_format: unknown pixel format %d\n", - src->pixel_format)); - dest->fmt.pix.pixelformat = 0; /* V4L2 doesn't define - * and "undefined" - * format? */ + src->pixel_format)); + /* V4L2 doesn't define an "undefined" format? */ + dest->fmt.pix.pixelformat = 0; break; } - } static void v4l2_format_to_video_format(const struct v4l2_format *src, - struct video_format *dest) + struct video_format *dest) { + switch (src->type) { case V4L2_BUF_TYPE_VIDEO_CAPTURE: dest->width = src->fmt.pix.width; @@ -766,7 +777,7 @@ v4l2_format_to_video_format(const struct break; default: DPRINTF(("video: unknown v4l2 pixel format %d\n", - src->fmt.pix.pixelformat)); + src->fmt.pix.pixelformat)); dest->pixel_format = VIDEO_FORMAT_UNDEFINED; break; } @@ -774,7 +785,7 @@ v4l2_format_to_video_format(const struct default: /* TODO: other v4l2 format types */ DPRINTF(("video: unsupported v4l2 format type %d\n", - src->type)); + src->type)); break; } } @@ -797,13 +808,14 @@ video_enum_format(struct video_softc *sc video_format_to_v4l2_format(&vfmt, &fmt); - fmtdesc->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; /* TODO: only one type for now */ + /* TODO: only one type for now */ + fmtdesc->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; fmtdesc->flags = 0; if (vfmt.pixel_format >= VIDEO_FORMAT_MJPEG) fmtdesc->flags = V4L2_FMT_FLAG_COMPRESSED; strlcpy(fmtdesc->description, - video_pixel_format_str(vfmt.pixel_format), - sizeof(fmtdesc->description)); + video_pixel_format_str(vfmt.pixel_format), + sizeof(fmtdesc->description)); fmtdesc->pixelformat = fmt.fmt.pix.pixelformat; return 0; @@ -831,7 +843,8 @@ video_enum_framesizes(struct video_softc fmt.fmt.pix.pixelformat, frmdesc->pixel_format); } - frmdesc->type = V4L2_FRMSIZE_TYPE_DISCRETE; /* TODO: only one type for now */ + /* TODO: only one type for now */ + frmdesc->type = V4L2_FRMSIZE_TYPE_DISCRETE; frmdesc->discrete.width = vfmt.width; frmdesc->discrete.height = vfmt.height; return 0; @@ -853,8 +866,7 @@ video_enum_frameival(struct video_softc } static int -video_get_format(struct video_softc *sc, - struct v4l2_format *format) +video_get_format(struct video_softc *sc, struct v4l2_format *format) { const struct video_hw_if *hw; struct video_format vfmt; @@ -896,10 +908,8 @@ video_set_format(struct video_softc *sc, return 0; } - static int -video_try_format(struct video_softc *sc, - struct v4l2_format *format) +video_try_format(struct video_softc *sc, struct v4l2_format *format) { const struct video_hw_if *hw; struct video_format vfmt; @@ -941,7 +951,8 @@ video_get_parm(struct video_softc *sc, s return error; parm->parm.capture.capability = V4L2_CAP_TIMEPERFRAME; parm->parm.capture.timeperframe.numerator = fract.numerator; - parm->parm.capture.timeperframe.denominator = fract.denominator; + parm->parm.capture.timeperframe.denominator = + fract.denominator; } return 0; @@ -972,8 +983,7 @@ video_set_parm(struct video_softc *sc, s } static void -v4l2_standard_to_video_standard(v4l2_std_id stdid, - enum video_standard *vstd) +v4l2_standard_to_video_standard(v4l2_std_id stdid, enum video_standard *vstd) { #define VSTD(id, vid) case (id): *vstd = (vid); break; switch (stdid) { @@ -989,6 +999,7 @@ static void video_standard_to_v4l2_standard(enum video_standard vstd, struct v4l2_standard *std) { + switch (vstd) { case VIDEO_STANDARD_NTSC_M: std->id = V4L2_STD_NTSC_M; @@ -1077,6 +1088,7 @@ static void v4l2_input_to_video_input(const struct v4l2_input *input, struct video_input *vi) { + vi->index = input->index; strlcpy(vi->name, input->name, sizeof(vi->name)); switch (input->type) { @@ -1107,6 +1119,7 @@ static void video_input_to_v4l2_input(const struct video_input *vi, struct v4l2_input *input) { + input->index = vi->index; strlcpy(input->name, vi->name, sizeof(input->name)); switch (vi->type) { @@ -1213,6 +1226,7 @@ static void v4l2_audio_to_video_audio(const struct v4l2_audio *audio, struct video_audio *va) { + va->index = audio->index; strlcpy(va->name, audio->name, sizeof(va->name)); va->caps = va->mode = 0; @@ -1228,6 +1242,7 @@ static void video_audio_to_v4l2_audio(const struct video_audio *va, struct v4l2_audio *audio) { + audio->index = va->index; strlcpy(audio->name, va->name, sizeof(audio->name)); audio->capability = audio->mode = 0; @@ -1299,6 +1314,7 @@ static void v4l2_tuner_to_video_tuner(const struct v4l2_tuner *tuner, struct video_tuner *vt) { + vt->index = tuner->index; strlcpy(vt->name, tuner->name, sizeof(vt->name)); vt->freq_lo = tuner->rangelow; @@ -1335,6 +1351,7 @@ static void video_tuner_to_v4l2_tuner(const struct video_tuner *vt, struct v4l2_tuner *tuner) { + tuner->index = vt->index; strlcpy(tuner->name, vt->name, sizeof(tuner->name)); tuner->rangelow = vt->freq_lo; @@ -1452,11 +1469,12 @@ video_set_frequency(struct video_softc * return hw->set_frequency(sc->hw_softc, &vfreq); } -/* Takes a single Video4Linux2 control, converts it to a struct - * video_control, and calls the hardware driver. */ +/* + * Takes a single Video4Linux2 control, converts it to a struct + * video_control, and calls the hardware driver. + */ static int -video_set_control(struct video_softc *sc, - const struct v4l2_control *vcontrol) +video_set_control(struct video_softc *sc, const struct v4l2_control *vcontrol) { const struct video_hw_if *hw; struct video_control_group group; @@ -1466,24 +1484,25 @@ video_set_control(struct video_softc *sc if (hw->set_control_group) { control.group_id = control.control_id = v4l2id_to_control_id(vcontrol->id); - /* ?? if "control_id" is arbitrarily defined by the + /* + * ?? if "control_id" is arbitrarily defined by the * driver, then we need some way to store it... Maybe - * it doesn't matter for single value controls. */ + * it doesn't matter for single value controls. + */ control.value = vcontrol->value; group.group_id = control.group_id; group.length = 1; group.control = &control; - return (hw->set_control_group(sc->hw_softc, &group)); + return hw->set_control_group(sc->hw_softc, &group); } else { return EINVAL; } } static int -video_request_bufs(struct video_softc *sc, - struct v4l2_requestbuffers *req) +video_request_bufs(struct video_softc *sc, struct v4l2_requestbuffers *req) { struct video_stream *vs = &sc->sc_stream_in; struct v4l2_buffer *buf; @@ -1502,8 +1521,8 @@ video_request_bufs(struct video_softc *s req->count = VIDEO_MAX_BUFS; err = video_stream_setup_bufs(vs, - VIDEO_STREAM_METHOD_MMAP, - req->count); + VIDEO_STREAM_METHOD_MMAP, + req->count); if (err != 0) return err; @@ -1522,8 +1541,7 @@ video_request_bufs(struct video_softc *s } static int -video_query_buf(struct video_softc *sc, - struct v4l2_buffer *buf) +video_query_buf(struct video_softc *sc, struct v4l2_buffer *buf) { struct video_stream *vs = &sc->sc_stream_in; @@ -1537,8 +1555,10 @@ video_query_buf(struct video_softc *sc, return 0; } -/* Accept a buffer descriptor from userspace and return the indicated - * buffer to the driver's queue. */ +/* + * Accept a buffer descriptor from userspace and return the indicated + * buffer to the driver's queue. + */ static int video_queue_buf(struct video_softc *sc, struct v4l2_buffer *userbuf) { @@ -1548,12 +1568,12 @@ video_queue_buf(struct video_softc *sc, if (userbuf->type != vs->vs_type) { DPRINTF(("video_queue_buf: expected type=%d got type=%d\n", - userbuf->type, vs->vs_type)); + userbuf->type, vs->vs_type)); return EINVAL; } if (userbuf->index >= vs->vs_nbufs) { DPRINTF(("video_queue_buf: invalid index %d >= %d\n", - userbuf->index, vs->vs_nbufs)); + userbuf->index, vs->vs_nbufs)); return EINVAL; } @@ -1561,7 +1581,7 @@ video_queue_buf(struct video_softc *sc, case VIDEO_STREAM_METHOD_MMAP: if (userbuf->memory != V4L2_MEMORY_MMAP) { DPRINTF(("video_queue_buf: invalid memory=%d\n", - userbuf->memory)); + userbuf->memory)); return EINVAL; } @@ -1571,7 +1591,7 @@ video_queue_buf(struct video_softc *sc, driverbuf = vb->vb_buf; if (driverbuf->flags & V4L2_BUF_FLAG_QUEUED) { DPRINTF(("video_queue_buf: buf already queued; " - "flags=0x%x\n", driverbuf->flags)); + "flags=0x%x\n", driverbuf->flags)); mutex_exit(&vs->vs_lock); return EINVAL; } @@ -1587,8 +1607,10 @@ video_queue_buf(struct video_softc *sc, return 0; } -/* Dequeue the described buffer from the driver queue, making it - * available for reading via mmap. */ +/* + * Dequeue the described buffer from the driver queue, making it + * available for reading via mmap. + */ static int video_dequeue_buf(struct video_softc *sc, struct v4l2_buffer *buf) { @@ -1628,7 +1650,7 @@ video_dequeue_buf(struct video_softc *sc return EINVAL; } err = cv_wait_sig(&vs->vs_sample_cv, - &vs->vs_lock); + &vs->vs_lock); if (err != 0) { mutex_exit(&vs->vs_lock); return EINTR; @@ -1665,7 +1687,6 @@ video_stream_on(struct video_softc *sc, if (hw == NULL) return ENXIO; - err = hw->start_transfer(sc->hw_softc); if (err != 0) return err; @@ -1726,7 +1747,7 @@ videoopen(dev_t dev, int flags, int ifmt sc->sc_stream_in.vs_flags = flags; DPRINTF(("videoopen: flags=0x%x sc=%p parent=%p\n", - flags, sc, sc->hw_dev)); + flags, sc, sc->hw_dev)); hw = sc->hw_if; if (hw == NULL) @@ -1742,8 +1763,11 @@ videoopen(dev_t dev, int flags, int ifmt return err; } - /* set up input stream. TODO: check flags to determine if - * "read" is desired? */ + /* + * set up input stream. + * + * TODO: check flags to determine if "read" is desired? + */ vs = &sc->sc_stream_in; if (hw->get_format != NULL) { @@ -1754,7 +1778,6 @@ videoopen(dev_t dev, int flags, int ifmt return 0; } - int videoclose(dev_t dev, int flags, int ifmt, struct lwp *l) { @@ -1787,7 +1810,6 @@ videoclose(dev_t dev, int flags, int ifm return 0; } - int videoread(dev_t dev, struct uio *uio, int ioflag) { @@ -1811,8 +1833,8 @@ videoread(dev_t dev, struct uio *uio, in /* userspace has chosen read() method */ if (vs->vs_method == VIDEO_STREAM_METHOD_NONE) { err = video_stream_setup_bufs(vs, - VIDEO_STREAM_METHOD_READ, - VIDEO_NUM_BUFS); + VIDEO_STREAM_METHOD_READ, + VIDEO_NUM_BUFS); if (err != 0) return err; @@ -1835,7 +1857,7 @@ retry: /* Block until we have a sample */ while (SIMPLEQ_EMPTY(&vs->vs_egress)) { err = cv_wait_sig(&vs->vs_sample_cv, - &vs->vs_lock); + &vs->vs_lock); if (err != 0) { mutex_exit(&vs->vs_lock); return EINTR; @@ -1869,8 +1891,10 @@ retry: DPRINTF(("video: invalid read\n")); } - /* Move the sample to the ingress queue if everything has - * been read */ + /* + * Move the sample to the ingress queue if everything has + * been read + */ if (vs->vs_bytesread >= vb->vb_buf->bytesused) { mutex_enter(&vs->vs_lock); vb = video_stream_dequeue(vs); @@ -1883,14 +1907,13 @@ retry: return 0; } - int videowrite(dev_t dev, struct uio *uio, int ioflag) { + return ENXIO; } - /* * Before 64-bit time_t, timeval's tv_sec was 'long'. Thus on LP64 ports * v4l2_buffer is the same size and layout as before. However it did change @@ -1983,12 +2006,12 @@ videoioctl(dev_t dev, u_long cmd, void * cap = data; memset(cap, 0, sizeof(*cap)); strlcpy(cap->driver, - device_cfdriver(sc->hw_dev)->cd_name, - sizeof(cap->driver)); + device_cfdriver(sc->hw_dev)->cd_name, + sizeof(cap->driver)); strlcpy(cap->card, hw->get_devname(sc->hw_softc), - sizeof(cap->card)); + sizeof(cap->card)); strlcpy(cap->bus_info, hw->get_businfo(sc->hw_softc), - sizeof(cap->bus_info)); + sizeof(cap->bus_info)); cap->version = VIDEO_DRIVER_VERSION; cap->capabilities = 0; if (hw->start_transfer != NULL && hw->stop_transfer != NULL) @@ -2076,18 +2099,18 @@ videoioctl(dev_t dev, u_long cmd, void * return video_set_frequency(sc, freq); case VIDIOC_QUERYCTRL: query = data; - return (video_query_control(sc, query)); + return video_query_control(sc, query); case VIDIOC_G_CTRL: control = data; - return (video_get_control(sc, control)); + return video_get_control(sc, control); case VIDIOC_S_CTRL: control = data; if ((flag & FWRITE) == 0) return EPERM; - return (video_set_control(sc, control)); + return video_set_control(sc, control); case VIDIOC_REQBUFS: reqbufs = data; - return (video_request_bufs(sc, reqbufs)); + return video_request_bufs(sc, reqbufs); case VIDIOC_QUERYBUF: buf = data; return video_query_buf(sc, buf); @@ -2132,7 +2155,7 @@ videoioctl(dev_t dev, u_long cmd, void * return video_enum_frameival(sc, ival); default: DPRINTF(("videoioctl: invalid cmd %s (%lx)\n", - video_ioctl_str(cmd), cmd)); + video_ioctl_str(cmd), cmd)); return EINVAL; } } @@ -2159,7 +2182,7 @@ video_ioctl_str(u_long cmd) case VIDIOC_S_FMT: str = "VIDIOC_S_FMT"; break; -/* 6 and 7 are VIDIOC_[SG]_COMP, which are unsupported */ + /* 6 and 7 are VIDIOC_[SG]_COMP, which are unsupported */ case VIDIOC_REQBUFS: str = "VIDIOC_REQBUFS"; break; @@ -2324,7 +2347,6 @@ video_ioctl_str(u_long cmd) } #endif - int videopoll(dev_t dev, int events, struct lwp *l) { @@ -2336,13 +2358,13 @@ videopoll(dev_t dev, int events, struct vs = &sc->sc_stream_in; if (sc->sc_dying) - return (POLLHUP); + return POLLHUP; /* userspace has chosen read() method */ if (vs->vs_method == VIDEO_STREAM_METHOD_NONE) { err = video_stream_setup_bufs(vs, - VIDEO_STREAM_METHOD_READ, - VIDEO_NUM_BUFS); + VIDEO_STREAM_METHOD_READ, + VIDEO_NUM_BUFS); if (err != 0) return POLLERR; @@ -2358,16 +2380,14 @@ videopoll(dev_t dev, int events, struct selrecord(l, &vs->vs_sel); mutex_exit(&vs->vs_lock); - return (revents); + return revents; } - paddr_t videommap(dev_t dev, off_t off, int prot) { struct video_softc *sc; struct video_stream *vs; - /* paddr_t pa; */ sc = device_lookup_private(&video_cd, VIDEOUNIT(dev)); if (sc->sc_dying) @@ -2378,12 +2398,14 @@ videommap(dev_t dev, off_t off, int prot return scatter_buf_map(&vs->vs_data, off); } - -/* Allocates buffers and initializes some fields. The format field - * must already have been initialized. */ +/* + * Allocates buffers and initializes some fields. The format field + * must already have been initialized. + */ void video_stream_init(struct video_stream *vs) { + vs->vs_method = VIDEO_STREAM_METHOD_NONE; vs->vs_flags = 0; vs->vs_frameno = -1; @@ -2408,11 +2430,14 @@ video_stream_init(struct video_stream *v void video_stream_fini(struct video_stream *vs) { + /* Sample data in queues has already been freed */ - /* while (SIMPLEQ_FIRST(&vs->vs_ingress) != NULL) +#if 0 + while (SIMPLEQ_FIRST(&vs->vs_ingress) != NULL) SIMPLEQ_REMOVE_HEAD(&vs->vs_ingress, entries); while (SIMPLEQ_FIRST(&vs->vs_egress) != NULL) - SIMPLEQ_REMOVE_HEAD(&vs->vs_egress, entries); */ + SIMPLEQ_REMOVE_HEAD(&vs->vs_egress, entries); +#endif mutex_destroy(&vs->vs_lock); cv_destroy(&vs->vs_sample_cv); @@ -2423,15 +2448,17 @@ video_stream_fini(struct video_stream *v static int video_stream_setup_bufs(struct video_stream *vs, - enum video_stream_method method, - uint8_t nbufs) + enum video_stream_method method, + uint8_t nbufs) { int i, err; mutex_enter(&vs->vs_lock); - /* Ensure that all allocated buffers are queued and not under - * userspace control. */ + /* + * Ensure that all allocated buffers are queued and not under + * userspace control. + */ for (i = 0; i < vs->vs_nbufs; ++i) { if (!(vs->vs_buf[i]->vb_buf->flags & V4L2_BUF_FLAG_QUEUED)) { mutex_exit(&vs->vs_lock); @@ -2446,12 +2473,16 @@ video_stream_setup_bufs(struct video_str return err; } - /* Queue up buffers for read method. Other methods are queued - * by VIDIOC_QBUF ioctl. */ + /* + * Queue up buffers for read method. Other methods are queued + * by VIDIOC_QBUF ioctl. + */ if (method == VIDEO_STREAM_METHOD_READ) { - for (i = 0; i < nbufs; ++i) - if (!(vs->vs_buf[i]->vb_buf->flags & V4L2_BUF_FLAG_QUEUED)) + for (i = 0; i < nbufs; ++i) { + if (!(vs->vs_buf[i]->vb_buf->flags & + V4L2_BUF_FLAG_QUEUED)) video_stream_enqueue(vs, vs->vs_buf[i]); + } } vs->vs_method = method; @@ -2460,10 +2491,12 @@ video_stream_setup_bufs(struct video_str return 0; } -/* Free all buffer memory in preparation for close(). This should +/* + * Free all buffer memory in preparation for close(). This should * free buffers regardless of errors. Use video_stream_setup_bufs if * you need to check for errors. Streaming should be off before - * calling this function. */ + * calling this function. + */ static void video_stream_teardown_bufs(struct video_stream *vs) { @@ -2473,7 +2506,7 @@ video_stream_teardown_bufs(struct video_ if (vs->vs_streaming) { DPRINTF(("video_stream_teardown_bufs: " - "tearing down bufs while streaming\n")); + "tearing down bufs while streaming\n")); } /* dequeue all buffers */ @@ -2485,8 +2518,8 @@ video_stream_teardown_bufs(struct video_ err = video_stream_free_bufs(vs); if (err != 0) { DPRINTF(("video_stream_teardown_bufs: " - "error releasing buffers: %d\n", - err)); + "error releasing buffers: %d\n", + err)); } vs->vs_method = VIDEO_STREAM_METHOD_NONE; @@ -2506,12 +2539,14 @@ video_buffer_alloc(void) static void video_buffer_free(struct video_buffer *vb) { + kmem_free(vb->vb_buf, sizeof(*vb->vb_buf)); vb->vb_buf = NULL; kmem_free(vb, sizeof(*vb)); } -/* TODO: for userptr method +/* TODO: for userptr method */ +#if 0 struct video_buffer * video_buf_alloc_with_ubuf(struct v4l2_buffer *buf) { @@ -2521,7 +2556,7 @@ void video_buffer_free_with_ubuf(struct video_buffer *vb) { } -*/ +#endif static int video_stream_realloc_bufs(struct video_stream *vs, uint8_t nbufs) @@ -2543,8 +2578,7 @@ video_stream_realloc_bufs(struct video_s vs->vs_nbufs = nbufs; if (nbufs > 0) { - vs->vs_buf = - kmem_alloc(sizeof(struct video_buffer *) * nbufs, KM_SLEEP); + vs->vs_buf = kmem_alloc(sizeof(*vs->vs_buf) * nbufs, KM_SLEEP); } else { vs->vs_buf = NULL; } @@ -2564,7 +2598,7 @@ video_stream_realloc_bufs(struct video_s /* Free old buffer metadata */ if (oldbuf != NULL) - kmem_free(oldbuf, sizeof(struct video_buffer *) * oldnbufs); + kmem_free(oldbuf, sizeof(*oldbuf) * oldnbufs); /* initialize bufs */ offset = 0; @@ -2588,11 +2622,14 @@ video_stream_realloc_bufs(struct video_s return 0; } -/* Accepts a video_sample into the ingress queue. Caller must hold - * the stream lock. */ +/* + * Accepts a video_sample into the ingress queue. Caller must hold + * the stream lock. + */ void video_stream_enqueue(struct video_stream *vs, struct video_buffer *vb) { + if (vb->vb_buf->flags & V4L2_BUF_FLAG_QUEUED) { DPRINTF(("video_stream_enqueue: sample already queued\n")); return; @@ -2606,9 +2643,10 @@ video_stream_enqueue(struct video_stream SIMPLEQ_INSERT_TAIL(&vs->vs_ingress, vb, entries); } - -/* Removes the head of the egress queue for use by userspace. Caller - * must hold the stream lock. */ +/* + * Removes the head of the egress queue for use by userspace. Caller + * must hold the stream lock. + */ struct video_buffer * video_stream_dequeue(struct video_stream *vs) { @@ -2638,7 +2676,7 @@ v4l2buf_set_timestamp(struct v4l2_buffer */ void video_stream_write(struct video_stream *vs, - const struct video_payload *payload) + const struct video_payload *payload) { struct video_buffer *vb; struct v4l2_buffer *buf; @@ -2653,8 +2691,10 @@ video_stream_write(struct video_stream * vs->vs_frameno = payload->frameno; if (vs->vs_drop || SIMPLEQ_EMPTY(&vs->vs_ingress)) { - /* DPRINTF(("video_stream_write: dropping sample %d\n", - vs->vs_sequence)); */ +#if 0 + DPRINTF(("video_stream_write: dropping sample %d\n", + vs->vs_sequence)); +#endif vs->vs_drop = true; } else if (payload->size > 0) { vb = SIMPLEQ_FIRST(&vs->vs_ingress); @@ -2663,21 +2703,21 @@ video_stream_write(struct video_stream * v4l2buf_set_timestamp(buf); if (payload->size > buf->length - buf->bytesused) { DPRINTF(("video_stream_write: " - "payload would overflow\n")); + "payload would overflow\n")); } else if (scatter_io_init(&vs->vs_data, - buf->m.offset + buf->bytesused, - payload->size, - &sio)) - { + buf->m.offset + buf->bytesused, + payload->size, + &sio)) { scatter_io_copyin(&sio, payload->data); buf->bytesused += (payload->size - sio.sio_resid); } else { - DPRINTF(("video_stream_write: failed to init scatter io " - "vb=%p buf=%p " - "buf->m.offset=%d buf->bytesused=%u " - "payload->size=%zu\n", - vb, buf, - buf->m.offset, buf->bytesused, payload->size)); + DPRINTF(("video_stream_write:" + " failed to init scatter io " + "vb=%p buf=%p " + "buf->m.offset=%d buf->bytesused=%u " + "payload->size=%zu\n", + vb, buf, + buf->m.offset, buf->bytesused, payload->size)); } } @@ -2688,10 +2728,11 @@ video_stream_write(struct video_stream * mutex_exit(&vs->vs_lock); } - -/* Moves the head of the ingress queue to the tail of the egress +/* + * Moves the head of the ingress queue to the tail of the egress * queue, or resets drop status if we were dropping this sample. - * Caller should hold the stream queue lock. */ + * Caller should hold the stream queue lock. + */ void video_stream_sample_done(struct video_stream *vs) { @@ -2715,22 +2756,24 @@ video_stream_sample_done(struct video_st vs->vs_sequence++; } -/* Check if all buffers are queued, i.e. none are under control of - * userspace. */ /* + * Check if all buffers are queued, i.e. none are under control of + * userspace. + */ +#if 0 static bool video_stream_all_queued(struct video_stream *vs) { } -*/ - +#endif static void scatter_buf_init(struct scatter_buf *sb) { + sb->sb_pool = pool_cache_init(PAGE_SIZE, 0, 0, 0, - "video", NULL, IPL_VIDEO, - NULL, NULL, NULL); + "video", NULL, IPL_VIDEO, + NULL, NULL, NULL); sb->sb_size = 0; sb->sb_npages = 0; sb->sb_page_ary = NULL; @@ -2739,6 +2782,7 @@ scatter_buf_init(struct scatter_buf *sb) static void scatter_buf_destroy(struct scatter_buf *sb) { + /* Do we need to return everything to the pool first? */ scatter_buf_set_size(sb, 0); pool_cache_destroy(sb->sb_pool); @@ -2747,7 +2791,9 @@ scatter_buf_destroy(struct scatter_buf * sb->sb_page_ary = NULL; } -/* Increase or decrease the size of the buffer */ +/* + * Increase or decrease the size of the buffer + */ static int scatter_buf_set_size(struct scatter_buf *sb, size_t sz) { @@ -2791,7 +2837,6 @@ scatter_buf_set_size(struct scatter_buf return 0; } - static paddr_t scatter_buf_map(struct scatter_buf *sb, off_t off) { @@ -2802,23 +2847,26 @@ scatter_buf_map(struct scatter_buf *sb, if (pg >= sb->sb_npages) return -1; - else if (!pmap_extract(pmap_kernel(), (vaddr_t)sb->sb_page_ary[pg], &pa)) + else if (!pmap_extract(pmap_kernel(), (vaddr_t)sb->sb_page_ary[pg], + &pa)) return -1; return atop(pa); } -/* Initialize data for an io operation on a scatter buffer. Returns - * true if the transfer is valid, or false if out of range. */ +/* + * Initialize data for an io operation on a scatter buffer. Returns + * true if the transfer is valid, or false if out of range. + */ static bool -scatter_io_init(struct scatter_buf *sb, - off_t off, size_t len, - struct scatter_io *sio) +scatter_io_init(struct scatter_buf *sb, off_t off, size_t len, + struct scatter_io *sio) { + if ((off + len) > sb->sb_size) { DPRINTF(("video: scatter_io_init failed: off=%" PRId64 - " len=%zu sb->sb_size=%zu\n", - off, len, sb->sb_size)); + " len=%zu sb->sb_size=%zu\n", + off, len, sb->sb_size)); return false; } @@ -2829,9 +2877,11 @@ scatter_io_init(struct scatter_buf *sb, return true; } -/* Store the pointer and size of the next contiguous segment. Returns +/* + * Store the pointer and size of the next contiguous segment. Returns * true if the segment is valid, or false if all has been transferred. - * Does not check for overflow. */ + * Does not check for overflow. + */ static bool scatter_io_next(struct scatter_io *sio, void **p, size_t *sz) { @@ -2852,16 +2902,21 @@ scatter_io_next(struct scatter_io *sio, return true; } -/* Semi-undo of a failed segment copy. Updates the scatter_io - * struct to the previous values prior to a failed segment copy. */ +/* + * Semi-undo of a failed segment copy. Updates the scatter_io + * struct to the previous values prior to a failed segment copy. + */ static void scatter_io_undo(struct scatter_io *sio, size_t sz) { + sio->sio_offset -= sz; sio->sio_resid += sz; } -/* Copy data from src into the scatter_buf as described by io. */ +/* + * Copy data from src into the scatter_buf as described by io. + */ static void scatter_io_copyin(struct scatter_io *sio, const void *p) { @@ -2869,13 +2924,13 @@ scatter_io_copyin(struct scatter_io *sio const uint8_t *src = p; size_t sz; - while(scatter_io_next(sio, &dst, &sz)) { + while (scatter_io_next(sio, &dst, &sz)) { memcpy(dst, src, sz); src += sz; } } -/* --not used; commented to avoid compiler warnings-- +#if 0 static void scatter_io_copyout(struct scatter_io *sio, void *p) { @@ -2883,16 +2938,18 @@ scatter_io_copyout(struct scatter_io *si uint8_t *dst = p; size_t sz; - while(scatter_io_next(sio, &src, &sz)) { + while (scatter_io_next(sio, &src, &sz)) { memcpy(dst, src, sz); dst += sz; } } -*/ - -/* Performat a series of uiomove calls on a scatter buf. Returns +#endif + +/* + * Perform a series of uiomove calls on a scatter buf. Returns * EFAULT if uiomove EFAULTs on the first segment. Otherwise, returns - * an incomplete transfer but with no error. */ + * an incomplete transfer but with no error. + */ static int scatter_io_uiomove(struct scatter_io *sio, struct uio *uio) { @@ -2901,7 +2958,7 @@ scatter_io_uiomove(struct scatter_io *si bool first = true; int err; - while(scatter_io_next(sio, &p, &sz)) { + while (scatter_io_next(sio, &p, &sz)) { err = uiomove(p, sz, uio); if (err == EFAULT) { scatter_io_undo(sio, sz); @@ -2915,5 +2972,3 @@ scatter_io_uiomove(struct scatter_io *si return 0; } - -#endif /* NVIDEO > 0 */ # HG changeset patch # User Taylor R Campbell # Date 1734197427 0 # Sat Dec 14 17:30:27 2024 +0000 # Branch trunk # Node ID 17b6bf7fa8ecfdb7caa7b1352d6e13809850b742 # Parent d2cfe16f944c79bf27b4bbced8978f6471ede6a0 # EXP-Topic riastradh-pr58898-videolocking video(4): Query current I/O flags for nonblocking status. Don't record the flags at open time; they might change with fcntl. diff -r d2cfe16f944c -r 17b6bf7fa8ec sys/dev/video.c --- a/sys/dev/video.c Sat Dec 14 17:27:18 2024 +0000 +++ b/sys/dev/video.c Sat Dec 14 17:30:27 2024 +0000 @@ -135,8 +135,6 @@ struct video_buffer { SIMPLEQ_HEAD(sample_queue, video_buffer); struct video_stream { - int vs_flags; /* flags given to open() */ - struct video_format vs_format; int vs_frameno; /* toggles between 0 and 1, @@ -297,7 +295,8 @@ static int video_request_bufs(struct vid struct v4l2_requestbuffers *); static int video_query_buf(struct video_softc *, struct v4l2_buffer *); static int video_queue_buf(struct video_softc *, struct v4l2_buffer *); -static int video_dequeue_buf(struct video_softc *, struct v4l2_buffer *); +static int video_dequeue_buf(struct video_softc *, struct v4l2_buffer *, + int); static int video_stream_on(struct video_softc *, enum v4l2_buf_type); static int video_stream_off(struct video_softc *, enum v4l2_buf_type); @@ -1612,7 +1611,7 @@ video_queue_buf(struct video_softc *sc, * available for reading via mmap. */ static int -video_dequeue_buf(struct video_softc *sc, struct v4l2_buffer *buf) +video_dequeue_buf(struct video_softc *sc, struct v4l2_buffer *buf, int ioflag) { struct video_stream *vs = &sc->sc_stream_in; struct video_buffer *vb; @@ -1636,7 +1635,7 @@ video_dequeue_buf(struct video_softc *sc mutex_enter(&vs->vs_lock); - if (vs->vs_flags & O_NONBLOCK) { + if (ioflag & IO_NDELAY) { vb = video_stream_dequeue(vs); if (vb == NULL) { mutex_exit(&vs->vs_lock); @@ -1744,8 +1743,6 @@ videoopen(dev_t dev, int flags, int ifmt return EIO; } - sc->sc_stream_in.vs_flags = flags; - DPRINTF(("videoopen: flags=0x%x sc=%p parent=%p\n", flags, sc, sc->hw_dev)); @@ -1849,7 +1846,7 @@ videoread(dev_t dev, struct uio *uio, in retry: if (SIMPLEQ_EMPTY(&vs->vs_egress)) { - if (vs->vs_flags & O_NONBLOCK) { + if (ioflag & IO_NDELAY) { mutex_exit(&vs->vs_lock); return EAGAIN; } @@ -2132,11 +2129,11 @@ videoioctl(dev_t dev, u_long cmd, void * #endif case VIDIOC_DQBUF: buf = data; - return video_dequeue_buf(sc, buf); + return video_dequeue_buf(sc, buf, flag); #ifndef _LP64 case VIDIOC_DQBUF50: buf50tobuf(data, buf = &bufspace); - if ((error = video_dequeue_buf(sc, buf)) != 0) + if ((error = video_dequeue_buf(sc, buf, flag)) != 0) return error; buftobuf50(data, buf); return 0; @@ -2407,7 +2404,6 @@ video_stream_init(struct video_stream *v { vs->vs_method = VIDEO_STREAM_METHOD_NONE; - vs->vs_flags = 0; vs->vs_frameno = -1; vs->vs_sequence = 0; vs->vs_type = V4L2_BUF_TYPE_VIDEO_CAPTURE; # HG changeset patch # User Taylor R Campbell # Date 1734202031 0 # Sat Dec 14 18:47:11 2024 +0000 # Branch trunk # Node ID b7099c50c3ced4e46ee490b8502fe868039a4670 # Parent 17b6bf7fa8ecfdb7caa7b1352d6e13809850b742 # EXP-Topic riastradh-pr58898-videolocking video_if.h: KNF No functional change intended. diff -r 17b6bf7fa8ec -r b7099c50c3ce sys/dev/video_if.h --- a/sys/dev/video_if.h Sat Dec 14 17:30:27 2024 +0000 +++ b/sys/dev/video_if.h Sat Dec 14 18:47:11 2024 +0000 @@ -52,7 +52,8 @@ struct video_softc; -/* Controls provide a way to query and set controls in the camera +/* + * Controls provide a way to query and set controls in the camera * hardware. The control structure is the primitive unit. Control * groups are arrays of controls that must be set together (e.g. pan * direction and pan speed). Control descriptors describe a control @@ -103,9 +104,11 @@ enum video_control_id { VIDEO_CONTROL_SATURATION, VIDEO_CONTROL_SHARPNESS, VIDEO_CONTROL_GAMMA, - /* Generic WHITE_BALANCE controls applies to whichever type of + /* + * Generic WHITE_BALANCE controls applies to whichever type of * white balance the hardware implements to either perform one - * white balance action or enable auto white balance. */ + * white balance action or enable auto white balance. + */ VIDEO_CONTROL_WHITE_BALANCE_ACTION, VIDEO_CONTROL_WHITE_BALANCE_AUTO, VIDEO_CONTROL_WHITE_BALANCE_TEMPERATURE, @@ -123,8 +126,10 @@ enum video_control_id { /* misc, not in UVC */ VIDEO_CONTROL_HFLIP, VIDEO_CONTROL_VFLIP, - /* Custom controls start here; any controls beyond this are - * valid and considered "extended". */ + /* + * Custom controls start here; any controls beyond this are + * valid and considered "extended". + */ VIDEO_CONTROL_EXTENDED }; @@ -180,7 +185,7 @@ struct video_control_iter { /* format of video data in a video sample */ enum video_pixel_format { VIDEO_FORMAT_UNDEFINED, - + /* uncompressed frame-based formats */ VIDEO_FORMAT_YUY2, /* packed 4:2:2 */ VIDEO_FORMAT_NV12, /* planar 4:2:0 */ @@ -228,13 +233,14 @@ enum video_standard { VIDEO_STANDARD_UNKNOWN = 0x00000000 }; -/* interlace_flags bits are allocated like this: - 7 6 5 4 3 2 1 0 - \_/ | | |interlaced or progressive - | | |packing style of fields (interlaced or planar) - | |fields per sample (1 or 2) - |pattern (F1 only, F2 only, F12, RND) -*/ +/* + * interlace_flags bits are allocated like this: + * 7 6 5 4 3 2 1 0 + * \_/ | | |interlaced or progressive + * | | |packing style of fields (interlaced or planar) + * | |fields per sample (1 or 2) + * |pattern (F1 only, F2 only, F12, RND) + */ /* two bits */ #define VIDEO_INTERLACED(iflags) (iflags & 1) @@ -251,18 +257,20 @@ enum video_interlace_packing { VIDEO_INTERLACE_PLANAR = 1 /* entire F1 is followed by F2 */ }; -/* one bit, not in V4L2; Is this not redundant with PATTERN below? +/* + * one bit, not in V4L2; Is this not redundant with PATTERN below? * For now, I'm assuming it describes where the "end-of-frame" markers * appear in the stream data: after every field or after every two - * fields. */ -#define VIDEO_INTERLACE_FIELDS_PER_SAMPLE(iflags) ((iflags >> 3) & 1) + * fields. + */ +#define VIDEO_INTERLACE_FIELDS_PER_SAMPLE(iflags) (((iflags) >> 3) & 1) enum video_interlace_fields_per_sample { VIDEO_INTERLACE_TWO_FIELDS_PER_SAMPLE = 0, VIDEO_INTERLACE_ONE_FIELD_PER_SAMPLE = 1 }; /* two bits */ -#define VIDEO_INTERLACE_PATTERN(iflags) ((iflags >> 4) & 3) +#define VIDEO_INTERLACE_PATTERN(iflags) (((iflags) >> 4) & 3) enum video_interlace_pattern { VIDEO_INTERLACE_PATTERN_F1 = 0, VIDEO_INTERLACE_PATTERN_F2 = 1, @@ -310,11 +318,13 @@ struct video_colorspace { enum video_matrix_coeff matrix_coeff; }; -#ifdef undef -/* Structs for future split into format/frame/interval. All functions +#if 0 +/* + * Structs for future split into format/frame/interval. All functions * interacting with the hardware layer will deal with these structs. * This video layer will handle translating them to V4L2 structs as - * necessary. */ + * necessary. + */ struct video_format { enum video_pixel_format vfo_pixel_format; @@ -338,9 +348,11 @@ enum video_frame_interval_type { VIDEO_FRAME_INTERVAL_TYPE_DISCRETE }; -/* UVC spec frame interval units are 100s of nanoseconds. V4L2 spec +/* + * UVC spec frame interval units are 100s of nanoseconds. V4L2 spec * uses a {32/32} bit struct fraction in seconds. We use 100ns units - * here. */ + * here. + */ #define VIDEO_FRAME_INTERVAL_UNITS_PER_US (10) #define VIDEO_FRAME_INTERVAL_UNITS_PER_MS (10 * 1000) #define VIDEO_FRAME_INTERVAL_UNITS_PER_S (10 * 1000 * 1000) @@ -358,7 +370,8 @@ struct video_frame_interval { }; #endif /* undef */ -/* Describes a video format. For frame based formats, one sample is +/* + * Describes a video format. For frame based formats, one sample is * equivalent to one frame. For stream based formats such as MPEG, a * sample is logical unit of that streaming format. */ @@ -384,9 +397,11 @@ struct video_fract { uint32_t denominator; }; -/* A payload is the smallest unit transferred from the hardware driver +/* + * A payload is the smallest unit transferred from the hardware driver * to the video layer. Multiple video payloads make up one video - * sample. */ + * sample. + */ struct video_payload { const uint8_t *data; size_t size; /* size in bytes of this payload */ @@ -477,9 +492,10 @@ struct video_hw_if { int (*control_iter_init)(void *, struct video_control_iter *); int (*control_iter_next)(void *, struct video_control_iter *); int (*get_control_desc_group)(void *, - struct video_control_desc_group *); + struct video_control_desc_group *); int (*get_control_group)(void *, struct video_control_group *); - int (*set_control_group)(void *, const struct video_control_group *); + int (*set_control_group)(void *, + const struct video_control_group *); int (*enum_input)(void *, uint32_t, struct video_input *); int (*get_input)(void *, struct video_input *); # HG changeset patch # User Taylor R Campbell # Date 1734561708 0 # Wed Dec 18 22:41:48 2024 +0000 # Branch trunk # Node ID cae37e5dd049b3de9f05a6dffc39ca10d712538f # Parent b7099c50c3ced4e46ee490b8502fe868039a4670 # EXP-Topic riastradh-pr58898-videolocking WIP: video(4): Allow only one videoread at a time. diff -r b7099c50c3ce -r cae37e5dd049 sys/dev/video.c --- a/sys/dev/video.c Sat Dec 14 18:47:11 2024 +0000 +++ b/sys/dev/video.c Wed Dec 18 22:41:48 2024 +0000 @@ -1845,6 +1845,14 @@ videoread(dev_t dev, struct uio *uio, in mutex_enter(&vs->vs_lock); retry: + while (vs->vs_busy) { + err = cv_wait_sig(&vs->vs_sample_cv, &vs->vs_lock); + if (err != 0) { + mutex_exit(&vs->vs_lock); + return err; + } + } + if (SIMPLEQ_EMPTY(&vs->vs_egress)) { if (ioflag & IO_NDELAY) { mutex_exit(&vs->vs_lock); @@ -1874,6 +1882,7 @@ retry: goto retry; } + vs->vs_busy = true; mutex_exit(&vs->vs_lock); len = uimin(uio->uio_resid, vb->vb_buf->bytesused - vs->vs_bytesread); @@ -1882,7 +1891,7 @@ retry: if (scatter_io_init(&vs->vs_data, offset, len, &sio)) { err = scatter_io_uiomove(&sio, uio); if (err == EFAULT) - return EFAULT; + goto out; vs->vs_bytesread += (len - sio.sio_resid); } else { DPRINTF(("video: invalid read\n")); @@ -1901,7 +1910,14 @@ retry: vs->vs_bytesread = 0; } - return 0; + err = 0; + +out: + mutex_enter(&vs->vs_lock); + vs->vs_busy = false; + cv_signal(&vs->vs_busy_cv); + mutex_exit(&vs->vs_lock); + return err; } int