Re: [PATCH v3 4/8] media: renesas: vsp1: Fix crop left and top clamping on RPF

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Fri, Jul 04, 2025 at 03:18:08AM +0300, Laurent Pinchart wrote:
> The RPF doesn't enforces the alignment constraint on the sink pad
> format, which could have an odd size, possibly down to 1x1. In that
> case, the upper bounds for the left and top coordinates clamping would
> become negative, cast to a very large positive value. Incorrect crop
> rectangle coordinates would then be incorrectly accepted.
>
> A second issue can occur when the requested left and top coordinates are
> negative. They are cast to a large unsigned value, clamped to the
> maximum. While the calculation will produce valid values for the
> hardware, this is not compliant with the V4L2 specification that
> requires values to be adjusted to the closest valid value.
>
> Fix both issues by switching to signed clamping, with an explicit
> minimum to adjust negative values, and adjusting the clamp bounds to
> avoid negative upper bounds.
>
> Signed-off-by: Laurent Pinchart <laurent.pinchart+renesas@xxxxxxxxxxxxxxxx>
> ---
>  .../media/platform/renesas/vsp1/vsp1_rwpf.c   | 28 ++++++++++++++++---
>  1 file changed, 24 insertions(+), 4 deletions(-)
>
> diff --git a/drivers/media/platform/renesas/vsp1/vsp1_rwpf.c b/drivers/media/platform/renesas/vsp1/vsp1_rwpf.c
> index 56464875a6c5..ffc1b3ab54e2 100644
> --- a/drivers/media/platform/renesas/vsp1/vsp1_rwpf.c
> +++ b/drivers/media/platform/renesas/vsp1/vsp1_rwpf.c
> @@ -201,6 +201,8 @@ static int vsp1_rwpf_set_selection(struct v4l2_subdev *subdev,
>  				   struct v4l2_subdev_state *sd_state,
>  				   struct v4l2_subdev_selection *sel)
>  {
> +	unsigned int min_width = RWPF_MIN_WIDTH;
> +	unsigned int min_height = RWPF_MIN_HEIGHT;
>  	struct vsp1_rwpf *rwpf = to_rwpf(subdev);
>  	struct v4l2_subdev_state *state;
>  	struct v4l2_mbus_framefmt *format;
> @@ -229,18 +231,36 @@ static int vsp1_rwpf_set_selection(struct v4l2_subdev *subdev,
>  	format = v4l2_subdev_state_get_format(state, RWPF_PAD_SINK);
>
>  	/*
> -	 * Restrict the crop rectangle coordinates to multiples of 2 to avoid
> -	 * shifting the color plane.
> +	 * For YUV formats, restrict the crop rectangle coordinates to multiples
> +	 * of 2 to avoid shifting the color plane.
>  	 */
>  	if (format->code == MEDIA_BUS_FMT_AYUV8_1X32) {
>  		sel->r.left = ALIGN(sel->r.left, 2);
>  		sel->r.top = ALIGN(sel->r.top, 2);
>  		sel->r.width = round_down(sel->r.width, 2);
>  		sel->r.height = round_down(sel->r.height, 2);
> +
> +		/*
> +		 * The RPF doesn't enforces the alignment constraint on the sink
> +		 * pad format, which could have an odd size, possibly down to
> +		 * 1x1. In that case, the minimum width and height would be
> +		 * smaller than the sink pad format, leading to a negative upper
> +		 * bound in the left and top clamping. Clamp the minimum width
> +		 * and height to the format width and height to avoid this.
> +		 *
> +		 * In such a situation, odd values for the crop rectangle size
> +		 * would be accepted when clamping the width and height below.
> +		 * While that would create an invalid hardware configuration,
> +		 * the video device enforces proper alignment of the pixel
> +		 * format, and the mismatch will then result in link validation
> +		 * failure. Incorrect operation of the hardware is not possible.
> +		 */
> +		min_width = min(ALIGN(min_width, 2), format->width);
> +		min_height = min(ALIGN(min_height, 2), format->height);

I wonder if initializing min_width (and height) to RWPF_MIN_WIDTH and
then here unconditionally make it 2 (because of ALIGN) wouldn't just
be better expressed as

		min_width = min(2, format->width);

>  	}
>
> -	sel->r.left = min_t(unsigned int, sel->r.left, format->width - 2);
> -	sel->r.top = min_t(unsigned int, sel->r.top, format->height - 2);
> +	sel->r.left = clamp_t(int, sel->r.left, 0, format->width - min_width);
> +	sel->r.top = clamp_t(int, sel->r.top, 0, format->height - min_height);

Regardless of the above
Reviewed-by: Jacopo Mondi <jacopo.mondi@xxxxxxxxxxxxxxxx>

>  	sel->r.width = min_t(unsigned int, sel->r.width,
>  			     format->width - sel->r.left);
>  	sel->r.height = min_t(unsigned int, sel->r.height,
> --
> Regards,
>
> Laurent Pinchart
>
>




[Index of Archives]     [Linux Samsung SOC]     [Linux Wireless]     [Linux Kernel]     [ATH6KL]     [Linux Bluetooth]     [Linux Netdev]     [Kernel Newbies]     [IDE]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux ATA RAID]     [Samba]     [Device Mapper]

  Powered by Linux