Hi Sylwester, On Tuesday 03 May 2011 20:07:43 Sylwester Nawrocki wrote: > On 05/03/2011 11:16 AM, Laurent Pinchart wrote: > > On Thursday 21 April 2011 17:21:04 Sylwester Nawrocki wrote: [snip] > >> + struct media_pad pads[CSIS_PADS_NUM]; > >> + struct v4l2_subdev sd; > >> + struct platform_device *pdev; > >> + struct resource *regs_res; > >> + void __iomem *regs; > >> + struct clk *clock[NUM_CSIS_CLOCKS]; > >> + int irq; > >> + struct regulator *supply; > >> + u32 flags; > >> + /* Common format for the source and sink pad. */ > >> + const struct csis_pix_format *csis_fmt; > >> + struct v4l2_mbus_framefmt mf[CSIS_NUM_FMTS]; > > > > As try formats are stored in the file handle, and as the formats on the > > sink and source pads are identical, a single v4l2_mbus_framefmt will do > > here. > > Ok. How about a situation when the caller never provides a file handle? > Is it not supposed to happen? Good question :-) The subdev pad-level operations have been designed with a userspace interface in mind, so they require a file handle to store try the formats (and crop rectangles). > For V4L2_SUBDEV_FORMAT_TRY, should set_fmt just abandon storing the format > and should get_fmt just return -EINVAL when passed fh == NULL ? For such a simple subdev, that should work as a workaround, yes. You can use it as a temporary solution at least. > Or should the host driver allocate the file handle just for the sake of > set_fmt/get_fmt calls (assuming that cropping ops are not supported > by the subdev) ? That's another solution. We could also pass an internal structure that contains formats and crop rectangles to the pad operations handlers, instead of passing the whole file handle. Do you think that would be better ? > It's not my intention to create a broken implementation but it would > be nice to be able to drop functionality which would never be used. > > As a note, I wanted to avoid bothering user space with setting up the MIPI > CSI receiver sub-device. There wouldn't be any gain from it, just more > things to care about for the applications. Quoting one of your comments below, x--- FIMC_0 (/dev/video1) SENSOR -> MIPI_CSIS --| x--- FIMC_1 (/dev/video3) How do you expect to configure the MIPI_CSIS block from the FIMC_0 and FIMC_1 blocks, without any help from userspace ? Conflicts will need to be handled, and the best way to handle them is to have userspace configuring the MIPI_CSIS explicitly. > Moreover I don't see a good usage for the stored TRY format (yet). So I > originally thought this subdev could be configurable by the host driver > which wouldn't provide a file handle. [snip] > >> +#define csis_pad_valid(pad) (pad == CSIS_PAD_SOURCE || pad == > >> CSIS_PAD_SINK) + > >> +static struct csis_state *sd_to_csis_state(struct v4l2_subdev *sdev) > >> +{ > >> + return container_of(sdev, struct csis_state, sd); > >> +} > >> + > >> +static const struct csis_pix_format *find_csis_format( > >> + struct v4l2_mbus_framefmt *mf) > >> +{ > >> + int i = ARRAY_SIZE(s5pcsis_formats); > >> + > >> + while (--i>= 0) > > > > I'm curious, why do you search backward instead of doing the usual > > > > for (i = 0; i< ARRAY_SIZE(s5pcsis_formats); ++i) > > > > (in that case 'i' could be unsigned) ? > > Perhaps doing it either way does not make any difference with the > toolchains we use, but the loops with test for 0 are supposed to be faster > on ARM. I didn't know that. I wonder if it makes a real difference with gcc. [snip] > >> +static void s5pcsis_try_format(struct v4l2_mbus_framefmt *mf) > >> +{ > >> + struct csis_pix_format const *csis_fmt; > >> + > >> + csis_fmt = find_csis_format(mf); > >> + if (csis_fmt == NULL) > >> + csis_fmt =&s5pcsis_formats[0]; > >> + > >> + mf->code = csis_fmt->code; > >> + v4l_bound_align_image(&mf->width, 1, CSIS_MAX_PIX_WIDTH, > >> + csis_fmt->pix_hor_align, > >> + &mf->height, 1, CSIS_MAX_PIX_HEIGHT, 1, > >> + 0); > >> +} > >> + > >> +static int s5pcsis_set_fmt(struct v4l2_subdev *sd, struct > >> v4l2_subdev_fh *fh, + struct v4l2_subdev_format *fmt) > >> +{ > >> + struct csis_state *state = sd_to_csis_state(sd); > >> + struct v4l2_mbus_framefmt *mf =&fmt->format; > >> + struct csis_pix_format const *csis_fmt = find_csis_format(mf); > >> + > >> + v4l2_dbg(1, debug, sd, "%s: %dx%d, code: %x, csis_fmt: %p\n", > >> + __func__, mf->width, mf->height, mf->code, csis_fmt); > >> + > >> + if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) { > >> + s5pcsis_try_format(mf); > > > > You need to take the pad into account here. As you mention below, source > > and sink formats are identical. When the user tries to set the source > > format, the driver should just return the sink format without performing > > any modification. > > > >> + state->mf[CSIS_FMT_TRY] = *mf; > >> + return 0; > >> + } > >> + > >> + /* Both source and sink pad have always same format. */ > >> + if (!csis_pad_valid(fmt->pad) || > >> + csis_fmt == NULL || > >> + mf->width> CSIS_MAX_PIX_WIDTH || > >> + mf->height> CSIS_MAX_PIX_HEIGHT || > >> + mf->width& (u32)(csis_fmt->pix_hor_align - 1)) > >> + return -EINVAL; > > > > Don't return an error, adjust the user supplied format instead. > > > >> + > >> + mutex_lock(&state->lock); > >> + state->mf[CSIS_FMT_ACTIVE] = *mf; > >> + state->csis_fmt = csis_fmt; > >> + mutex_unlock(&state->lock); > > > > The logic in this function is not correct. First of all, you need to > > adjust the user-supplied format in all cases, regardless of the format > > type (try/active). Then, as the formats on the sink and source pads are > > always identical, you should return the sink pad format when the user > > tries to set the source pad format. Setting the source pad format will > > have no effect. Finally, you should store try formats in the subdev file > > handle, not in the csis_state structure (see ccp2_set_format() in > > drivers/media/video/omap3isp/ispccp2.c for an example, in your case you > > don't need to propagate the format change from sink to source, as the > > source always has the same format as the sink). > > Thanks, that's all very useful. The only thing I am concerned is what > should be done when the file handle is null? Is it not allowed by design? It wasn't allowed by design. As explained above, we could replace the file handle with another structure, and have the caller allocate it. In your case you could just skip storing try formats. This needs to be thought about some more. > I've initially reworked s5pcsis_set_fmt to something like this: > > static int s5pcsis_set_fmt(struct v4l2_subdev *sd, struct v4l2_subdev_fh > *fh, struct v4l2_subdev_format *fmt) > { > struct csis_state *state = sd_to_csis_state(sd); > struct v4l2_mbus_framefmt *mf = &fmt->format; > struct csis_pix_format const *csis_fmt; > > v4l2_dbg(1, debug, sd, "%s: %dx%d, code: %x\n", > __func__, mf->width, mf->height, mf->code); > > if (!csis_pad_valid(fmt->pad)) > return -EINVAL; > > mutex_lock(&state->lock); > if (fmt->pad == CSIS_PAD_SOURCE) { > fmt->format = state->format; > goto unlock; > } > csis_fmt = s5pcsis_try_format(&fmt->format); Try try_format call doesn't require the mutex to be locked. You can lock/unlock the mutex around fmt->format = state->format; above, and lock/unlock it around state->formt = fmt->format; state->csis_fmt = csis_fmt; below. > if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) { > /* TODO: store format in *fh */ > goto unlock; > } > > /* Common format for the source and the sink pad */ > state->format = fmt->format; > state->csis_fmt = csis_fmt; > unlock: > mutex_unlock(&state->lock); > return 0; > } [snip] -- Regards, Laurent Pinchart -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html