From: Satish Nagireddy <satish.nagireddy@getcruise.com>
To: Tomi Valkeinen <tomi.valkeinen@ideasonboard.com>
Cc: linux-media@vger.kernel.org, sakari.ailus@linux.intel.com,
Jacopo Mondi <jacopo+renesas@jmondi.org>,
Laurent Pinchart <laurent.pinchart@ideasonboard.com>,
niklas.soderlund+renesas@ragnatech.se,
Mauro Carvalho Chehab <mchehab@kernel.org>,
Hans Verkuil <hverkuil-cisco@xs4all.nl>,
Pratyush Yadav <p.yadav@ti.com>,
Kishon Vijay Abraham <kishon@ti.com>,
Tomasz Figa <tfiga@chromium.org>
Subject: Re: [EXT] [PATCH v12 18/30] media: subdev: use streams in v4l2_subdev_link_validate()
Date: Fri, 29 Jul 2022 02:12:49 -0700 [thread overview]
Message-ID: <CAG0LG96XkmpP2JmMG+Nxkd0ViCdKbHOhPPqE0JxUKBgK-xrRkw@mail.gmail.com> (raw)
In-Reply-To: <20220727103639.581567-19-tomi.valkeinen@ideasonboard.com>
Hi Tomi,
Thanks for the patch.
On Wed, Jul 27, 2022 at 3:37 AM Tomi Valkeinen
<tomi.valkeinen@ideasonboard.com> wrote:
>
> Update v4l2_subdev_link_validate() to use routing and streams for
> validation.
>
> Instead of just looking at the format on the pad on both ends of the
> link, the routing tables are used to collect all the streams going from
> the source to the sink over the link, and the streams' formats on both
> ends of the link are verified.
>
> Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ideasonboard.com>
> ---
> drivers/media/v4l2-core/v4l2-subdev.c | 255 ++++++++++++++++++++++++--
> 1 file changed, 235 insertions(+), 20 deletions(-)
>
> diff --git a/drivers/media/v4l2-core/v4l2-subdev.c b/drivers/media/v4l2-core/v4l2-subdev.c
> index d26715cbd955..7ca2337ca8d3 100644
> --- a/drivers/media/v4l2-core/v4l2-subdev.c
> +++ b/drivers/media/v4l2-core/v4l2-subdev.c
> @@ -16,6 +16,7 @@
> #include <linux/videodev2.h>
> #include <linux/export.h>
> #include <linux/version.h>
> +#include <linux/sort.h>
>
> #include <media/v4l2-ctrls.h>
> #include <media/v4l2-device.h>
> @@ -1069,7 +1070,7 @@ int v4l2_subdev_link_validate_default(struct v4l2_subdev *sd,
> EXPORT_SYMBOL_GPL(v4l2_subdev_link_validate_default);
>
> static int
> -v4l2_subdev_link_validate_get_format(struct media_pad *pad,
> +v4l2_subdev_link_validate_get_format(struct media_pad *pad, u32 stream,
> struct v4l2_subdev_format *fmt)
> {
> if (is_media_entity_v4l2_subdev(pad->entity)) {
> @@ -1078,7 +1079,11 @@ v4l2_subdev_link_validate_get_format(struct media_pad *pad,
>
> fmt->which = V4L2_SUBDEV_FORMAT_ACTIVE;
> fmt->pad = pad->index;
> - return v4l2_subdev_call_state_active(sd, pad, get_fmt, fmt);
> + fmt->stream = stream;
> +
> + return v4l2_subdev_call(sd, pad, get_fmt,
> + v4l2_subdev_get_locked_active_state(sd),
> + fmt);
> }
>
> WARN(pad->entity->function != MEDIA_ENT_F_IO_V4L,
> @@ -1088,31 +1093,241 @@ v4l2_subdev_link_validate_get_format(struct media_pad *pad,
> return -EINVAL;
> }
>
> -int v4l2_subdev_link_validate(struct media_link *link)
> +static int cmp_u32(const void *a, const void *b)
> {
> - struct v4l2_subdev *sink;
> - struct v4l2_subdev_format sink_fmt, source_fmt;
> - int rval;
> + u32 a32 = *(u32 *)a;
> + u32 b32 = *(u32 *)b;
>
> - rval = v4l2_subdev_link_validate_get_format(
> - link->source, &source_fmt);
> - if (rval < 0)
> - return 0;
> + return a32 > b32 ? 1 : (a32 < b32 ? -1 : 0);
> +}
> +
> +static int v4l2_link_validate_get_streams(struct media_link *link,
> + bool is_source, u32 *out_num_streams,
> + const u32 **out_streams,
> + bool *allocated)
I have a suggestion to refactor this function a bit to avoid the usage
of is_source boolean, hopefully that makes this implementation look
better.
Here is my idea: Pass the media_pad (source or sink) to the function
directly and drop the parameters link and is_source. Then...
> +{
> + static const u32 default_streams[] = { 0 };
> + struct v4l2_subdev_krouting *routing;
> + struct v4l2_subdev *subdev;
> + u32 num_streams;
> + u32 *streams;
> + unsigned int i;
> + struct v4l2_subdev_state *state;
> + int ret;
>
> - rval = v4l2_subdev_link_validate_get_format(
> - link->sink, &sink_fmt);
> - if (rval < 0)
> + if (is_source)
> + subdev = media_entity_to_v4l2_subdev(link->source->entity);
> + else
> + subdev = media_entity_to_v4l2_subdev(link->sink->entity);
... we can avoid the if check here and directly assign subdev as below.
subdev = media_entity_to_v4l2_subdev(pad->entity);
Then...
> +
> + if (!(subdev->flags & V4L2_SUBDEV_FL_STREAMS)) {
> + *out_num_streams = 1;
> + *out_streams = default_streams;
> + *allocated = false;
> return 0;
> + }
>
> - sink = media_entity_to_v4l2_subdev(link->sink->entity);
> + state = v4l2_subdev_get_locked_active_state(subdev);
>
> - rval = v4l2_subdev_call(sink, pad, link_validate, link,
> - &source_fmt, &sink_fmt);
> - if (rval != -ENOIOCTLCMD)
> - return rval;
> + routing = &state->routing;
> +
> + streams = kmalloc_array(routing->num_routes, sizeof(u32), GFP_KERNEL);
> + if (!streams)
> + return -ENOMEM;
> +
> + num_streams = 0;
> +
> + for (i = 0; i < routing->num_routes; ++i) {
> + struct v4l2_subdev_route *route = &routing->routes[i];
> + int j;
> + u32 route_pad;
> + u32 route_stream;
> + u32 link_pad;
> +
> + if (!(route->flags & V4L2_SUBDEV_ROUTE_FL_ACTIVE))
> + continue;
> +
> + if (is_source) {
> + route_pad = route->source_pad;
> + route_stream = route->source_stream;
> + link_pad = link->source->index;
> + } else {
> + route_pad = route->sink_pad;
> + route_stream = route->sink_stream;
> + link_pad = link->sink->index;
> + }
This can be achieved by checking pad->flags against
MEDIA_PAD_FL_SOURCE or MEDIA_PAD_FL_SINK.
Please let me know your opinion.
> +
> + if (route_pad != link_pad)
> + continue;
> +
> + /* look for duplicates */
> + for (j = 0; j < num_streams; ++j) {
> + if (streams[j] == route_stream) {
> + ret = -EINVAL;
> + goto free_streams;
> + }
> + }
This is good logic, but seems to be a bit inefficient as it is
repeatedly checking the array from the start.
I do not have a better idea :)
> +
> + streams[num_streams++] = route_stream;
> + }
> +
> + sort(streams, num_streams, sizeof(u32), &cmp_u32, NULL);
> +
> + *out_num_streams = num_streams;
> + *out_streams = streams;
> + *allocated = true;
> +
> + return 0;
> +
> +free_streams:
> + kfree(streams);
> +
> + return ret;
> +}
> +
> +static int v4l2_subdev_link_validate_locked(struct media_link *link)
> +{
> + struct v4l2_subdev *sink_subdev =
> + media_entity_to_v4l2_subdev(link->sink->entity);
> + struct device *dev = sink_subdev->entity.graph_obj.mdev->dev;
> + u32 num_source_streams;
> + const u32 *source_streams;
> + bool source_allocated;
> + u32 num_sink_streams;
> + const u32 *sink_streams;
> + bool sink_allocated;
> + unsigned int sink_idx;
> + unsigned int source_idx;
> + int ret;
> +
> + dev_dbg(dev, "validating link \"%s\":%u -> \"%s\":%u\n",
> + link->source->entity->name, link->source->index,
> + link->sink->entity->name, link->sink->index);
> +
> + ret = v4l2_link_validate_get_streams(link, true, &num_source_streams,
This function call can be modified as below
v4l2_link_validate_get_streams(link->source, &num_source_streams,
> + &source_streams,
> + &source_allocated);
> + if (ret)
> + return ret;
> +
> + ret = v4l2_link_validate_get_streams(link, false, &num_sink_streams,
This function call can be modified as below
v4l2_link_validate_get_streams(link->sink, &num_source_streams,
> + &sink_streams, &sink_allocated);
> + if (ret)
> + goto free_source;
> +
> + /*
> + * It is ok to have more source streams than sink streams as extra
> + * source streams can just be ignored by the receiver, but having extra
> + * sink streams is an error as streams must have a source.
> + */
> + if (num_source_streams < num_sink_streams) {
> + dev_err(dev,
> + "Not enough source streams: %d < %d\n",
> + num_source_streams, num_sink_streams);
> + ret = -EINVAL;
> + goto free_sink;
> + }
> +
> + /* Validate source and sink stream formats */
> +
> + source_idx = 0;
Nit, This can be assigned at the declaration.
Regards,
Satish
> +
> + for (sink_idx = 0; sink_idx < num_sink_streams; ++sink_idx) {
> + struct v4l2_subdev_format sink_fmt, source_fmt;
> + u32 stream;
> +
> + stream = sink_streams[sink_idx];
> +
> + for (; source_idx < num_source_streams; ++source_idx) {
> + if (source_streams[source_idx] == stream)
> + break;
> + }
> +
> + if (source_idx == num_source_streams) {
> + dev_err(dev, "No source stream for sink stream %u\n",
> + stream);
> + ret = -EINVAL;
> + goto free_sink;
> + }
> +
> + dev_dbg(dev, "validating stream \"%s\":%u:%u -> \"%s\":%u:%u\n",
> + link->source->entity->name, link->source->index, stream,
> + link->sink->entity->name, link->sink->index, stream);
> +
> + ret = v4l2_subdev_link_validate_get_format(link->source, stream,
> + &source_fmt);
> + if (ret < 0) {
> + dev_dbg(dev, "Failed to get format for \"%s\":%u:%u (but that's ok)\n",
> + link->source->entity->name, link->source->index,
> + stream);
> + ret = 0;
> + continue;
> + }
> +
> + ret = v4l2_subdev_link_validate_get_format(link->sink, stream,
> + &sink_fmt);
> + if (ret < 0) {
> + dev_dbg(dev, "Failed to get format for \"%s\":%u:%u (but that's ok)\n",
> + link->sink->entity->name, link->sink->index,
> + stream);
> + ret = 0;
> + continue;
> + }
>
> - return v4l2_subdev_link_validate_default(
> - sink, link, &source_fmt, &sink_fmt);
> + /* TODO: add stream number to link_validate() */
> + ret = v4l2_subdev_call(sink_subdev, pad, link_validate, link,
> + &source_fmt, &sink_fmt);
> + if (!ret)
> + continue;
> +
> + if (ret != -ENOIOCTLCMD)
> + goto free_sink;
> +
> + ret = v4l2_subdev_link_validate_default(sink_subdev, link,
> + &source_fmt, &sink_fmt);
> +
> + if (ret)
> + goto free_sink;
> + }
> +
> +free_sink:
> + if (sink_allocated)
> + kfree(sink_streams);
> +
> +free_source:
> + if (source_allocated)
> + kfree(source_streams);
> +
> + return ret;
> +}
> +
> +int v4l2_subdev_link_validate(struct media_link *link)
> +{
> + struct v4l2_subdev *source_sd, *sink_sd;
> + struct v4l2_subdev_state *source_state, *sink_state;
> + int ret;
> +
> + sink_sd = media_entity_to_v4l2_subdev(link->sink->entity);
> + source_sd = media_entity_to_v4l2_subdev(link->source->entity);
> +
> + sink_state = v4l2_subdev_get_unlocked_active_state(sink_sd);
> + source_state = v4l2_subdev_get_unlocked_active_state(source_sd);
> +
> + if (sink_state)
> + v4l2_subdev_lock_state(sink_state);
> +
> + if (source_state)
> + v4l2_subdev_lock_state(source_state);
> +
> + ret = v4l2_subdev_link_validate_locked(link);
> +
> + if (sink_state)
> + v4l2_subdev_unlock_state(sink_state);
> +
> + if (source_state)
> + v4l2_subdev_unlock_state(source_state);
> +
> + return ret;
> }
> EXPORT_SYMBOL_GPL(v4l2_subdev_link_validate);
>
> --
> 2.34.1
>
next prev parent reply other threads:[~2022-07-29 9:13 UTC|newest]
Thread overview: 65+ messages / expand[flat|nested] mbox.gz Atom feed top
2022-07-27 10:36 [PATCH v12 00/30] v4l: routing and streams support Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 01/30] media: Documentation: mc: add definitions for stream and pipeline Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 02/30] media: mc: entity: Add iterator helper for entity pads Tomi Valkeinen
2022-07-30 11:11 ` Sakari Ailus
2022-08-01 5:58 ` Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 03/30] media: mc: entity: Merge media_entity_enum_init and __media_entity_enum_init Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 04/30] media: mc: entity: Move media_entity_get_fwnode_pad() out of graph walk section Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 05/30] media: mc: entity: Add media_entity_pipeline() to access the media pipeline Tomi Valkeinen
2022-07-30 11:21 ` Sakari Ailus
2022-08-01 6:01 ` Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 06/30] media: mc: entity: Add has_route entity operation and media_entity_has_route() helper Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 07/30] media: mc: entity: Rename streaming_count -> start_count Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 08/30] media: mc: entity: add media_pipeline_alloc_start & media_pipeline_stop_free Tomi Valkeinen
2022-07-29 8:30 ` [EXT] " Satish Nagireddy
2022-07-29 8:40 ` Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 09/30] media: mc: entity: Rewrite media_pipeline_start() to support routes Tomi Valkeinen
2022-07-29 8:45 ` [EXT] " Satish Nagireddy
2022-07-29 8:53 ` Tomi Valkeinen
2022-07-29 9:19 ` [EXT] " Satish Nagireddy
2022-07-29 10:27 ` Tomi Valkeinen
2022-07-29 17:00 ` [EXT] " Satish Nagireddy
2022-07-29 17:07 ` Tomi Valkeinen
2022-07-29 18:20 ` [EXT] " Satish Nagireddy
2022-07-30 11:56 ` Sakari Ailus
2022-08-01 9:33 ` Tomi Valkeinen
2022-08-01 11:06 ` Sakari Ailus
2022-07-27 10:36 ` [PATCH v12 10/30] media: add V4L2_SUBDEV_FL_STREAMS Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 11/30] media: add V4L2_SUBDEV_CAP_MPLEXED Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 12/30] media: Documentation: Add GS_ROUTING documentation Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 13/30] media: subdev: Add [GS]_ROUTING subdev ioctls and operations Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 14/30] media: subdev: add v4l2_subdev_has_route() Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 15/30] media: subdev: add v4l2_subdev_set_routing helper() Tomi Valkeinen
2022-08-01 6:59 ` Sakari Ailus
2022-08-01 7:38 ` Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 16/30] media: Documentation: add multiplexed streams documentation Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 17/30] media: subdev: add stream based configuration Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 18/30] media: subdev: use streams in v4l2_subdev_link_validate() Tomi Valkeinen
2022-07-29 9:12 ` Satish Nagireddy [this message]
2022-07-29 11:00 ` [EXT] " Tomi Valkeinen
2022-07-29 17:33 ` [EXT] " Satish Nagireddy
2022-07-27 10:36 ` [PATCH v12 19/30] media: subdev: add "opposite" stream helper funcs Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 20/30] media: subdev: add streams to v4l2_subdev_get_fmt() helper function Tomi Valkeinen
2022-07-29 9:16 ` [EXT] " Satish Nagireddy
2022-07-29 10:30 ` Tomi Valkeinen
2022-07-29 17:01 ` [EXT] " Satish Nagireddy
2022-07-27 10:36 ` [PATCH v12 21/30] media: subdev: add v4l2_subdev_set_routing_with_fmt() helper Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 22/30] media: subdev: add v4l2_subdev_routing_validate() helper Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 23/30] media: subdev: Add for_each_active_route() macro Tomi Valkeinen
2022-08-01 8:40 ` Tomi Valkeinen
2022-08-01 11:09 ` Sakari Ailus
2022-07-27 10:36 ` [PATCH v12 24/30] media: subdev: use for_each_active_route() in v4l2_subdev_init_stream_configs() Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 25/30] media: subdev: use for_each_active_route() in v4l2_link_validate_get_streams() Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 26/30] media: v4l2-subdev: Add v4l2_subdev_state_xlate_streams() helper Tomi Valkeinen
2022-08-01 12:37 ` Sakari Ailus
2022-08-01 14:25 ` Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 27/30] media: v4l2-subdev: Add subdev .(enable|disable)_streams() operations Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 28/30] media: v4l2-subdev: Add v4l2_subdev_s_stream_helper() function Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 29/30] media: Add stream to frame descriptor Tomi Valkeinen
2022-07-27 10:36 ` [PATCH v12 30/30] media: subdev: increase V4L2_FRAME_DESC_ENTRY_MAX to 8 Tomi Valkeinen
2022-07-27 10:42 ` [PATCH v12 00/30] v4l: routing and streams support Tomi Valkeinen
2022-07-31 20:47 ` Sakari Ailus
2022-08-01 6:28 ` Tomi Valkeinen
[not found] ` <MW4PR02MB737849AF15E8004B2CB39C3BB09C9@MW4PR02MB7378.namprd02.prod.outlook.com>
2022-08-03 11:37 ` Tomi Valkeinen
[not found] ` <MW4PR02MB73781CDA5C792C28390BAF29B09E9@MW4PR02MB7378.namprd02.prod.outlook.com>
2022-08-08 6:45 ` Tomi Valkeinen
2022-08-19 2:31 ` Laurent Pinchart
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=CAG0LG96XkmpP2JmMG+Nxkd0ViCdKbHOhPPqE0JxUKBgK-xrRkw@mail.gmail.com \
--to=satish.nagireddy@getcruise.com \
--cc=hverkuil-cisco@xs4all.nl \
--cc=jacopo+renesas@jmondi.org \
--cc=kishon@ti.com \
--cc=laurent.pinchart@ideasonboard.com \
--cc=linux-media@vger.kernel.org \
--cc=mchehab@kernel.org \
--cc=niklas.soderlund+renesas@ragnatech.se \
--cc=p.yadav@ti.com \
--cc=sakari.ailus@linux.intel.com \
--cc=tfiga@chromium.org \
--cc=tomi.valkeinen@ideasonboard.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).