Add the missing pieces to enable full streams support:
- Add set_routing
- Drop the explicit uses of a single stream, and instead use the streams
mask.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen+renesas@ideasonboard.com>
---
drivers/media/platform/renesas/rcar-csi2.c | 85 ++++++++++++++++++++++--------
1 file changed, 63 insertions(+), 22 deletions(-)
diff --git a/drivers/media/platform/renesas/rcar-csi2.c b/drivers/media/platform/renesas/rcar-csi2.c
index 8f708196ef49..4a73d223229c 100644
--- a/drivers/media/platform/renesas/rcar-csi2.c
+++ b/drivers/media/platform/renesas/rcar-csi2.c
@@ -694,6 +694,17 @@ static const struct rcar_csi2_format rcar_csi2_formats[] = {
},
};
+static const struct v4l2_mbus_framefmt rcar_csi2_default_fmt = {
+ .width = 1920,
+ .height = 1080,
+ .code = MEDIA_BUS_FMT_RGB888_1X24,
+ .colorspace = V4L2_COLORSPACE_SRGB,
+ .field = V4L2_FIELD_NONE,
+ .ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT,
+ .quantization = V4L2_QUANTIZATION_DEFAULT,
+ .xfer_func = V4L2_XFER_FUNC_DEFAULT,
+};
+
static const struct rcar_csi2_format *rcsi2_code_to_fmt(unsigned int code)
{
unsigned int i;
@@ -1641,10 +1652,8 @@ static int rcsi2_enable_streams(struct v4l2_subdev *sd,
u64 source_streams_mask)
{
struct rcar_csi2 *priv = sd_to_csi2(sd);
- int ret = 0;
-
- if (source_streams_mask != 1)
- return -EINVAL;
+ u64 sink_streams;
+ int ret;
if (!priv->remote)
return -ENODEV;
@@ -1655,8 +1664,13 @@ static int rcsi2_enable_streams(struct v4l2_subdev *sd,
return ret;
}
+ sink_streams = v4l2_subdev_state_xlate_streams(state,
+ RCAR_CSI2_SOURCE_VC0,
+ RCAR_CSI2_SINK,
+ &source_streams_mask);
+
ret = v4l2_subdev_enable_streams(priv->remote, priv->remote_pad,
- BIT_ULL(0));
+ sink_streams);
if (ret) {
rcsi2_stop(priv);
return ret;
@@ -1672,10 +1686,7 @@ static int rcsi2_disable_streams(struct v4l2_subdev *sd,
u32 source_pad, u64 source_streams_mask)
{
struct rcar_csi2 *priv = sd_to_csi2(sd);
- int ret = 0;
-
- if (source_streams_mask != 1)
- return -EINVAL;
+ u64 sink_streams;
if (!priv->remote)
return -ENODEV;
@@ -1683,11 +1694,17 @@ static int rcsi2_disable_streams(struct v4l2_subdev *sd,
if (priv->stream_count == 1)
rcsi2_stop(priv);
- v4l2_subdev_disable_streams(priv->remote, priv->remote_pad, BIT_ULL(0));
+ sink_streams = v4l2_subdev_state_xlate_streams(state,
+ RCAR_CSI2_SOURCE_VC0,
+ RCAR_CSI2_SINK,
+ &source_streams_mask);
+
+ v4l2_subdev_disable_streams(priv->remote, priv->remote_pad,
+ sink_streams);
priv->stream_count -= 1;
- return ret;
+ return 0;
}
static int rcsi2_set_pad_format(struct v4l2_subdev *sd,
@@ -1720,6 +1737,40 @@ static int rcsi2_set_pad_format(struct v4l2_subdev *sd,
return 0;
}
+static int rcsi2_set_routing(struct v4l2_subdev *sd,
+ struct v4l2_subdev_state *state,
+ enum v4l2_subdev_format_whence which,
+ struct v4l2_subdev_krouting *routing)
+{
+ struct rcar_csi2 *priv = sd_to_csi2(sd);
+ int ret;
+
+ if (!priv->info->use_isp)
+ return -ENOTTY;
+
+ if (routing->num_routes > V4L2_FRAME_DESC_ENTRY_MAX)
+ return -EINVAL;
+
+ if (priv->info->use_isp) {
+ ret = v4l2_subdev_routing_validate(sd, routing,
+ V4L2_SUBDEV_ROUTING_ONLY_1_TO_1);
+ } else {
+ ret = v4l2_subdev_routing_validate(sd, routing,
+ V4L2_SUBDEV_ROUTING_ONLY_1_TO_1 |
+ V4L2_SUBDEV_ROUTING_NO_SOURCE_MULTIPLEXING);
+ }
+
+ if (ret)
+ return ret;
+
+ ret = v4l2_subdev_set_routing_with_fmt(sd, state, routing,
+ &rcar_csi2_default_fmt);
+ if (ret)
+ return ret;
+
+ return 0;
+}
+
static int rcsi2_get_frame_desc_fallback(struct v4l2_subdev *sd,
unsigned int pad,
struct v4l2_mbus_frame_desc *fd)
@@ -1781,6 +1832,7 @@ static const struct v4l2_subdev_pad_ops rcar_csi2_pad_ops = {
.set_fmt = rcsi2_set_pad_format,
.get_fmt = v4l2_subdev_get_fmt,
+ .set_routing = rcsi2_set_routing,
.get_frame_desc = rcsi2_get_frame_desc,
};
@@ -1801,17 +1853,6 @@ static int rcsi2_init_state(struct v4l2_subdev *sd,
},
};
- static const struct v4l2_mbus_framefmt rcar_csi2_default_fmt = {
- .width = 1920,
- .height = 1080,
- .code = MEDIA_BUS_FMT_RGB888_1X24,
- .colorspace = V4L2_COLORSPACE_SRGB,
- .field = V4L2_FIELD_NONE,
- .ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT,
- .quantization = V4L2_QUANTIZATION_DEFAULT,
- .xfer_func = V4L2_XFER_FUNC_DEFAULT,
- };
-
static const struct v4l2_subdev_krouting routing = {
.num_routes = ARRAY_SIZE(routes),
.routes = routes,
--
2.43.0
Hi Tomi,
Thank you for the patch.
On Fri, May 30, 2025 at 04:50:43PM +0300, Tomi Valkeinen wrote:
> Add the missing pieces to enable full streams support:
>
> - Add set_routing
> - Drop the explicit uses of a single stream, and instead use the streams
> mask.
>
> Signed-off-by: Tomi Valkeinen <tomi.valkeinen+renesas@ideasonboard.com>
> ---
> drivers/media/platform/renesas/rcar-csi2.c | 85 ++++++++++++++++++++++--------
> 1 file changed, 63 insertions(+), 22 deletions(-)
>
> diff --git a/drivers/media/platform/renesas/rcar-csi2.c b/drivers/media/platform/renesas/rcar-csi2.c
> index 8f708196ef49..4a73d223229c 100644
> --- a/drivers/media/platform/renesas/rcar-csi2.c
> +++ b/drivers/media/platform/renesas/rcar-csi2.c
> @@ -694,6 +694,17 @@ static const struct rcar_csi2_format rcar_csi2_formats[] = {
> },
> };
>
> +static const struct v4l2_mbus_framefmt rcar_csi2_default_fmt = {
> + .width = 1920,
> + .height = 1080,
> + .code = MEDIA_BUS_FMT_RGB888_1X24,
> + .colorspace = V4L2_COLORSPACE_SRGB,
> + .field = V4L2_FIELD_NONE,
> + .ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT,
> + .quantization = V4L2_QUANTIZATION_DEFAULT,
> + .xfer_func = V4L2_XFER_FUNC_DEFAULT,
> +};
> +
> static const struct rcar_csi2_format *rcsi2_code_to_fmt(unsigned int code)
> {
> unsigned int i;
> @@ -1641,10 +1652,8 @@ static int rcsi2_enable_streams(struct v4l2_subdev *sd,
> u64 source_streams_mask)
> {
> struct rcar_csi2 *priv = sd_to_csi2(sd);
> - int ret = 0;
> -
> - if (source_streams_mask != 1)
> - return -EINVAL;
> + u64 sink_streams;
> + int ret;
>
> if (!priv->remote)
> return -ENODEV;
> @@ -1655,8 +1664,13 @@ static int rcsi2_enable_streams(struct v4l2_subdev *sd,
> return ret;
> }
>
> + sink_streams = v4l2_subdev_state_xlate_streams(state,
> + RCAR_CSI2_SOURCE_VC0,
> + RCAR_CSI2_SINK,
> + &source_streams_mask);
> +
> ret = v4l2_subdev_enable_streams(priv->remote, priv->remote_pad,
> - BIT_ULL(0));
> + sink_streams);
> if (ret) {
> rcsi2_stop(priv);
> return ret;
> @@ -1672,10 +1686,7 @@ static int rcsi2_disable_streams(struct v4l2_subdev *sd,
> u32 source_pad, u64 source_streams_mask)
> {
> struct rcar_csi2 *priv = sd_to_csi2(sd);
> - int ret = 0;
> -
> - if (source_streams_mask != 1)
> - return -EINVAL;
> + u64 sink_streams;
>
> if (!priv->remote)
> return -ENODEV;
> @@ -1683,11 +1694,17 @@ static int rcsi2_disable_streams(struct v4l2_subdev *sd,
> if (priv->stream_count == 1)
> rcsi2_stop(priv);
>
> - v4l2_subdev_disable_streams(priv->remote, priv->remote_pad, BIT_ULL(0));
> + sink_streams = v4l2_subdev_state_xlate_streams(state,
> + RCAR_CSI2_SOURCE_VC0,
> + RCAR_CSI2_SINK,
> + &source_streams_mask);
> +
> + v4l2_subdev_disable_streams(priv->remote, priv->remote_pad,
> + sink_streams);
>
> priv->stream_count -= 1;
>
> - return ret;
> + return 0;
This seems to belong to a previous patch.
> }
>
> static int rcsi2_set_pad_format(struct v4l2_subdev *sd,
> @@ -1720,6 +1737,40 @@ static int rcsi2_set_pad_format(struct v4l2_subdev *sd,
> return 0;
> }
>
> +static int rcsi2_set_routing(struct v4l2_subdev *sd,
> + struct v4l2_subdev_state *state,
> + enum v4l2_subdev_format_whence which,
> + struct v4l2_subdev_krouting *routing)
> +{
> + struct rcar_csi2 *priv = sd_to_csi2(sd);
> + int ret;
> +
> + if (!priv->info->use_isp)
> + return -ENOTTY;
> +
> + if (routing->num_routes > V4L2_FRAME_DESC_ENTRY_MAX)
> + return -EINVAL;
A comment to explain this check would be nice.
> +
> + if (priv->info->use_isp) {
You return an error above when this condition is false.
> + ret = v4l2_subdev_routing_validate(sd, routing,
> + V4L2_SUBDEV_ROUTING_ONLY_1_TO_1);
> + } else {
> + ret = v4l2_subdev_routing_validate(sd, routing,
> + V4L2_SUBDEV_ROUTING_ONLY_1_TO_1 |
> + V4L2_SUBDEV_ROUTING_NO_SOURCE_MULTIPLEXING);
> + }
> +
> + if (ret)
> + return ret;
> +
> + ret = v4l2_subdev_set_routing_with_fmt(sd, state, routing,
> + &rcar_csi2_default_fmt);
> + if (ret)
> + return ret;
> +
> + return 0;
> +}
> +
> static int rcsi2_get_frame_desc_fallback(struct v4l2_subdev *sd,
> unsigned int pad,
> struct v4l2_mbus_frame_desc *fd)
> @@ -1781,6 +1832,7 @@ static const struct v4l2_subdev_pad_ops rcar_csi2_pad_ops = {
> .set_fmt = rcsi2_set_pad_format,
> .get_fmt = v4l2_subdev_get_fmt,
>
> + .set_routing = rcsi2_set_routing,
> .get_frame_desc = rcsi2_get_frame_desc,
> };
>
> @@ -1801,17 +1853,6 @@ static int rcsi2_init_state(struct v4l2_subdev *sd,
> },
> };
>
> - static const struct v4l2_mbus_framefmt rcar_csi2_default_fmt = {
> - .width = 1920,
> - .height = 1080,
> - .code = MEDIA_BUS_FMT_RGB888_1X24,
> - .colorspace = V4L2_COLORSPACE_SRGB,
> - .field = V4L2_FIELD_NONE,
> - .ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT,
> - .quantization = V4L2_QUANTIZATION_DEFAULT,
> - .xfer_func = V4L2_XFER_FUNC_DEFAULT,
> - };
> -
> static const struct v4l2_subdev_krouting routing = {
> .num_routes = ARRAY_SIZE(routes),
> .routes = routes,
--
Regards,
Laurent Pinchart
Hi,
On 02/06/2025 16:54, Laurent Pinchart wrote:
> Hi Tomi,
>
> Thank you for the patch.
>
> On Fri, May 30, 2025 at 04:50:43PM +0300, Tomi Valkeinen wrote:
>> Add the missing pieces to enable full streams support:
>>
>> - Add set_routing
>> - Drop the explicit uses of a single stream, and instead use the streams
>> mask.
>>
>> Signed-off-by: Tomi Valkeinen <tomi.valkeinen+renesas@ideasonboard.com>
>> ---
>> drivers/media/platform/renesas/rcar-csi2.c | 85 ++++++++++++++++++++++--------
>> 1 file changed, 63 insertions(+), 22 deletions(-)
>>
>> diff --git a/drivers/media/platform/renesas/rcar-csi2.c b/drivers/media/platform/renesas/rcar-csi2.c
>> index 8f708196ef49..4a73d223229c 100644
>> --- a/drivers/media/platform/renesas/rcar-csi2.c
>> +++ b/drivers/media/platform/renesas/rcar-csi2.c
>> @@ -694,6 +694,17 @@ static const struct rcar_csi2_format rcar_csi2_formats[] = {
>> },
>> };
>>
>> +static const struct v4l2_mbus_framefmt rcar_csi2_default_fmt = {
>> + .width = 1920,
>> + .height = 1080,
>> + .code = MEDIA_BUS_FMT_RGB888_1X24,
>> + .colorspace = V4L2_COLORSPACE_SRGB,
>> + .field = V4L2_FIELD_NONE,
>> + .ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT,
>> + .quantization = V4L2_QUANTIZATION_DEFAULT,
>> + .xfer_func = V4L2_XFER_FUNC_DEFAULT,
>> +};
>> +
>> static const struct rcar_csi2_format *rcsi2_code_to_fmt(unsigned int code)
>> {
>> unsigned int i;
>> @@ -1641,10 +1652,8 @@ static int rcsi2_enable_streams(struct v4l2_subdev *sd,
>> u64 source_streams_mask)
>> {
>> struct rcar_csi2 *priv = sd_to_csi2(sd);
>> - int ret = 0;
>> -
>> - if (source_streams_mask != 1)
>> - return -EINVAL;
>> + u64 sink_streams;
>> + int ret;
>>
>> if (!priv->remote)
>> return -ENODEV;
>> @@ -1655,8 +1664,13 @@ static int rcsi2_enable_streams(struct v4l2_subdev *sd,
>> return ret;
>> }
>>
>> + sink_streams = v4l2_subdev_state_xlate_streams(state,
>> + RCAR_CSI2_SOURCE_VC0,
>> + RCAR_CSI2_SINK,
>> + &source_streams_mask);
>> +
>> ret = v4l2_subdev_enable_streams(priv->remote, priv->remote_pad,
>> - BIT_ULL(0));
>> + sink_streams);
>> if (ret) {
>> rcsi2_stop(priv);
>> return ret;
>> @@ -1672,10 +1686,7 @@ static int rcsi2_disable_streams(struct v4l2_subdev *sd,
>> u32 source_pad, u64 source_streams_mask)
>> {
>> struct rcar_csi2 *priv = sd_to_csi2(sd);
>> - int ret = 0;
>> -
>> - if (source_streams_mask != 1)
>> - return -EINVAL;
>> + u64 sink_streams;
>>
>> if (!priv->remote)
>> return -ENODEV;
>> @@ -1683,11 +1694,17 @@ static int rcsi2_disable_streams(struct v4l2_subdev *sd,
>> if (priv->stream_count == 1)
>> rcsi2_stop(priv);
>>
>> - v4l2_subdev_disable_streams(priv->remote, priv->remote_pad, BIT_ULL(0));
>> + sink_streams = v4l2_subdev_state_xlate_streams(state,
>> + RCAR_CSI2_SOURCE_VC0,
>> + RCAR_CSI2_SINK,
>> + &source_streams_mask);
>> +
>> + v4l2_subdev_disable_streams(priv->remote, priv->remote_pad,
>> + sink_streams);
>>
>> priv->stream_count -= 1;
>>
>> - return ret;
>> + return 0;
>
> This seems to belong to a previous patch.
Yep. Although that patch would be before this series =). I'll clean that
up in an earlier one in this series.
>> }
>>
>> static int rcsi2_set_pad_format(struct v4l2_subdev *sd,
>> @@ -1720,6 +1737,40 @@ static int rcsi2_set_pad_format(struct v4l2_subdev *sd,
>> return 0;
>> }
>>
>> +static int rcsi2_set_routing(struct v4l2_subdev *sd,
>> + struct v4l2_subdev_state *state,
>> + enum v4l2_subdev_format_whence which,
>> + struct v4l2_subdev_krouting *routing)
>> +{
>> + struct rcar_csi2 *priv = sd_to_csi2(sd);
>> + int ret;
>> +
>> + if (!priv->info->use_isp)
>> + return -ENOTTY;
>> +
>> + if (routing->num_routes > V4L2_FRAME_DESC_ENTRY_MAX)
>> + return -EINVAL;
>
> A comment to explain this check would be nice.
This is no longer needed, as the framework checks it.
>> +
>> + if (priv->info->use_isp) {
>
> You return an error above when this condition is false.
Yes, no idea why (probably some early gen4-only code). I'll drop the
earlier check.
Tomi
>> + ret = v4l2_subdev_routing_validate(sd, routing,
>> + V4L2_SUBDEV_ROUTING_ONLY_1_TO_1);
>> + } else {
>> + ret = v4l2_subdev_routing_validate(sd, routing,
>> + V4L2_SUBDEV_ROUTING_ONLY_1_TO_1 |
>> + V4L2_SUBDEV_ROUTING_NO_SOURCE_MULTIPLEXING);
>> + }
>> +
>> + if (ret)
>> + return ret;
>> +
>> + ret = v4l2_subdev_set_routing_with_fmt(sd, state, routing,
>> + &rcar_csi2_default_fmt);
>> + if (ret)
>> + return ret;
>> +
>> + return 0;
>> +}
>> +
>> static int rcsi2_get_frame_desc_fallback(struct v4l2_subdev *sd,
>> unsigned int pad,
>> struct v4l2_mbus_frame_desc *fd)
>> @@ -1781,6 +1832,7 @@ static const struct v4l2_subdev_pad_ops rcar_csi2_pad_ops = {
>> .set_fmt = rcsi2_set_pad_format,
>> .get_fmt = v4l2_subdev_get_fmt,
>>
>> + .set_routing = rcsi2_set_routing,
>> .get_frame_desc = rcsi2_get_frame_desc,
>> };
>>
>> @@ -1801,17 +1853,6 @@ static int rcsi2_init_state(struct v4l2_subdev *sd,
>> },
>> };
>>
>> - static const struct v4l2_mbus_framefmt rcar_csi2_default_fmt = {
>> - .width = 1920,
>> - .height = 1080,
>> - .code = MEDIA_BUS_FMT_RGB888_1X24,
>> - .colorspace = V4L2_COLORSPACE_SRGB,
>> - .field = V4L2_FIELD_NONE,
>> - .ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT,
>> - .quantization = V4L2_QUANTIZATION_DEFAULT,
>> - .xfer_func = V4L2_XFER_FUNC_DEFAULT,
>> - };
>> -
>> static const struct v4l2_subdev_krouting routing = {
>> .num_routes = ARRAY_SIZE(routes),
>> .routes = routes,
>
© 2016 - 2025 Red Hat, Inc.